wip(ex04)

This commit is contained in:
Maix0 2026-04-15 17:16:45 +02:00
parent a87568e5f7
commit 5a954c3713
12 changed files with 403 additions and 57 deletions

105
ex04/src/main.c Normal file
View 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
View 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
View 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
View 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--;
}
}