wip(ex04)
This commit is contained in:
parent
a87568e5f7
commit
5a954c3713
12 changed files with 403 additions and 57 deletions
105
ex04/src/main.c
Normal file
105
ex04/src/main.c
Normal file
|
|
@ -0,0 +1,105 @@
|
|||
#include <avr/interrupt.h>
|
||||
#include <avr/io.h>
|
||||
|
||||
#include "mystd.h"
|
||||
#include "timer.h"
|
||||
#include "uart.h"
|
||||
#include "utils.h"
|
||||
|
||||
#define BUFFERSIZE 20
|
||||
|
||||
enum State {
|
||||
ASK_USERNAME,
|
||||
ASK_PASSWORD,
|
||||
|
||||
ASKING_USERNAME,
|
||||
ASKING_PASSWORD,
|
||||
|
||||
VERIFY_USERNAME,
|
||||
VERIFY_PASSWORD,
|
||||
|
||||
VALID_PASSWORD,
|
||||
INVALID_PASSWORD,
|
||||
|
||||
OVERFLOW_INPUT,
|
||||
};
|
||||
|
||||
int main(void) {
|
||||
uart_init();
|
||||
// enable the S_REG
|
||||
// SREG |= _BV(SREG_I);
|
||||
char buffer[BUFFERSIZE] = {};
|
||||
uint8_t cursor = 0;
|
||||
enum State state = ASK_USERNAME;
|
||||
while (true) {
|
||||
switch (state) {
|
||||
case ASK_USERNAME: {
|
||||
uart_sendstring("\r\nUsername: ");
|
||||
state = ASKING_USERNAME;
|
||||
cursor = 0;
|
||||
ft_bzero(&buffer, sizeof(buffer));
|
||||
break;
|
||||
}
|
||||
case ASK_PASSWORD: {
|
||||
break;
|
||||
}
|
||||
|
||||
case OVERFLOW_INPUT: {
|
||||
uart_sendstring("\r\n/!\\OVERFLOW DETECTED /!\\\r\n/!\\RESETING/!\\\r\n");
|
||||
ft_bzero(&buffer, sizeof(buffer));
|
||||
cursor = 0;
|
||||
state = ASK_USERNAME;
|
||||
break;
|
||||
}
|
||||
|
||||
case ASKING_USERNAME: {
|
||||
char data = uart_rx();
|
||||
if (data == '\r') {
|
||||
state = VERIFY_USERNAME;
|
||||
uart_sendstring("\r\n");
|
||||
continue;
|
||||
}
|
||||
if (data == '\x7F') {
|
||||
if (cursor) {
|
||||
uart_sendstring("\b \b");
|
||||
cursor--;
|
||||
}
|
||||
continue;
|
||||
}
|
||||
uart_tx(data);
|
||||
|
||||
if (cursor + 1 >= BUFFERSIZE) {
|
||||
state = OVERFLOW_INPUT;
|
||||
continue;
|
||||
}
|
||||
|
||||
buffer[cursor++] = data;
|
||||
break;
|
||||
}
|
||||
|
||||
case ASKING_PASSWORD: {
|
||||
break;
|
||||
}
|
||||
|
||||
case VERIFY_USERNAME: {
|
||||
uart_sendstring("\r\n");
|
||||
uart_sendstring(buffer);
|
||||
cursor = 0;
|
||||
ft_bzero(buffer, sizeof(buffer));
|
||||
state = ASK_USERNAME;
|
||||
break;
|
||||
}
|
||||
case VERIFY_PASSWORD: {
|
||||
break;
|
||||
}
|
||||
|
||||
case VALID_PASSWORD: {
|
||||
break;
|
||||
}
|
||||
|
||||
case INVALID_PASSWORD: {
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
38
ex04/src/timer.c
Normal file
38
ex04/src/timer.c
Normal file
|
|
@ -0,0 +1,38 @@
|
|||
|
||||
#include <avr/io.h>
|
||||
#include <avr/interrupt.h>
|
||||
#include "mystd.h"
|
||||
#include "utils.h"
|
||||
|
||||
#define PRESCALER 256
|
||||
#define TIMER_FREQ (F_CPU / PRESCALER)
|
||||
|
||||
// at a high level:
|
||||
// Set the OC1B (PB2) pin as output
|
||||
// set the TIMER1 mode to COMPARE (CTC)
|
||||
// say to compare against OC1A
|
||||
// set the value to be compated at X count
|
||||
// say the presacler for the timer is 512
|
||||
//
|
||||
// all these information are on page ~140
|
||||
void timer1_init(void) {
|
||||
// Set PB1 (OC1A) as output
|
||||
DDRB |= _BV(PB1);
|
||||
|
||||
// CTC mode (WGM12 = 1)
|
||||
TCCR1B |= _BV(WGM12);
|
||||
|
||||
// Toggle OC1B on compare match (COM1B0 = 1)
|
||||
// TCCR1A |= _BV(COM1A0);
|
||||
|
||||
// Set compare values
|
||||
OCR1A = TIMER_FREQ / 2;
|
||||
|
||||
// Start timer with prescaler 256 (CS12)
|
||||
TCCR1B |= _BV(CS12);
|
||||
|
||||
// set OCR1A interrupt
|
||||
TIMSK1 = _BV(1);
|
||||
|
||||
sei();
|
||||
}
|
||||
63
ex04/src/uart.c
Normal file
63
ex04/src/uart.c
Normal file
|
|
@ -0,0 +1,63 @@
|
|||
#include "uart.h"
|
||||
#include <avr/io.h>
|
||||
#include "mystd.h"
|
||||
#include "utils.h"
|
||||
|
||||
#define BAUD_RATE 115200
|
||||
|
||||
#define UBRR_VALUE ((F_CPU / (8UL * BAUD_RATE)) - 1)
|
||||
|
||||
// uart is 115200 baud rate, 8 bits per word, no parrity and 1 stop bit
|
||||
// 115200 8N1
|
||||
void uart_init(void) {
|
||||
// Set baud rate
|
||||
UBRR0H = (uint8_t)(UBRR_VALUE >> 8);
|
||||
UBRR0L = (uint8_t)(UBRR_VALUE);
|
||||
|
||||
UCSR0A |= _BV(U2X0);
|
||||
// Enable transmitter
|
||||
UCSR0B = _BV(TXEN0) | _BV(RXEN0) | _BV(RXCIE0);
|
||||
|
||||
// Set frame format: 8 data bits, no parity, 1 stop bit
|
||||
UCSR0C = _BV(UCSZ01) | _BV(UCSZ00);
|
||||
|
||||
// Set TX (PD1) as output
|
||||
DDRD |= _BV(PD1);
|
||||
}
|
||||
|
||||
void uart_tx(char data) {
|
||||
// wait for transmit buffer to be empty
|
||||
while (!(UCSR0A & _BV(UDRE0)))
|
||||
;
|
||||
// load data into transmit register
|
||||
UDR0 = data;
|
||||
}
|
||||
|
||||
char uart_rx(void) {
|
||||
while (!(UCSR0A & _BV(RXC0)))
|
||||
;
|
||||
return UDR0;
|
||||
}
|
||||
|
||||
void uart_sendstring(const char* str) {
|
||||
if (!str)
|
||||
return;
|
||||
while (*str) {
|
||||
uart_tx(*str);
|
||||
str++;
|
||||
}
|
||||
}
|
||||
|
||||
void uart_send_u8(uint8_t val) {
|
||||
char buf[4] = {0, 0, 0, 0};
|
||||
buf[0] = '0' + val / 100;
|
||||
val %= 100;
|
||||
buf[1] = '0' + val / 10;
|
||||
buf[2] = '0' + val % 10;
|
||||
if (buf[0] != '0')
|
||||
uart_sendstring(&buf[0]);
|
||||
else if (buf[1] != '0')
|
||||
uart_sendstring(&buf[1]);
|
||||
else if (buf[2] != '0')
|
||||
uart_sendstring(&buf[2]);
|
||||
}
|
||||
28
ex04/src/utils.c
Normal file
28
ex04/src/utils.c
Normal file
|
|
@ -0,0 +1,28 @@
|
|||
#include "utils.h"
|
||||
|
||||
// this just burns cycles.
|
||||
// the volatile is important, it means that the cpu can't optimize any
|
||||
// read/writes for the value
|
||||
static inline void spin_loop(volatile uint16_t counts) {
|
||||
while (counts)
|
||||
counts--;
|
||||
}
|
||||
|
||||
void delay_ms(uint16_t ms) {
|
||||
while (ms) {
|
||||
// this value was taken using a delay of 500ms, and just recording the led
|
||||
// blinking. it seems to be high enough such that each loop of delay_loop
|
||||
// takes 1ms :D
|
||||
spin_loop((F_CPU) / 5000);
|
||||
ms--;
|
||||
}
|
||||
}
|
||||
|
||||
void ft_bzero(void* data, uint16_t size) {
|
||||
char* d = data;
|
||||
while (size) {
|
||||
*d = 0;
|
||||
d++;
|
||||
size--;
|
||||
}
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue