Project refactoring
This commit is contained in:
parent
963d5c395b
commit
9b60c22f96
@ -96,10 +96,13 @@
|
||||
</ToolchainSettings>
|
||||
</PropertyGroup>
|
||||
<ItemGroup>
|
||||
<Compile Include="app.c">
|
||||
<Compile Include="calculator.c">
|
||||
<SubType>compile</SubType>
|
||||
</Compile>
|
||||
<Compile Include="app.h">
|
||||
<Compile Include="keyboard.c">
|
||||
<SubType>compile</SubType>
|
||||
</Compile>
|
||||
<Compile Include="keyboard.h">
|
||||
<SubType>compile</SubType>
|
||||
</Compile>
|
||||
<Compile Include="main.c">
|
||||
|
50
app.c
50
app.c
@ -1,50 +0,0 @@
|
||||
/*
|
||||
* core.c
|
||||
*
|
||||
* Created: 13.04.2023 14:28:40
|
||||
* Author: mrnek
|
||||
*/
|
||||
#include <avr/io.h>
|
||||
#include <stdbool.h>
|
||||
#include "app.h"
|
||||
|
||||
#define MASK_PIND 0b11110000
|
||||
|
||||
void check_physical_keys_state(struct key *keys) {
|
||||
uint8_t row1;
|
||||
PORTC &= ~(1 << 3);
|
||||
row1 = PIND & MASK_PIND;
|
||||
if (row1 & (1 << 4)) keys[0].is_physical_pressed = false; else keys[0].is_physical_pressed = true;
|
||||
if (row1 & (1 << 5)) keys[1].is_physical_pressed = false; else keys[1].is_physical_pressed = true;
|
||||
if (row1 & (1 << 6)) keys[2].is_physical_pressed = false; else keys[2].is_physical_pressed = true;
|
||||
if (row1 & (1 << 7)) keys[3].is_physical_pressed = false; else keys[3].is_physical_pressed = true;
|
||||
PORTC |= (1 << 3);
|
||||
|
||||
PORTC &= ~(1 << 2);
|
||||
row1 = PIND & MASK_PIND;
|
||||
if (row1 & (1 << 4)) keys[4].is_physical_pressed = false; else keys[4].is_physical_pressed = true;
|
||||
if (row1 & (1 << 5)) keys[5].is_physical_pressed = false; else keys[5].is_physical_pressed = true;
|
||||
if (row1 & (1 << 6)) keys[6].is_physical_pressed = false; else keys[6].is_physical_pressed = true;
|
||||
if (row1 & (1 << 7)) keys[7].is_physical_pressed = false; else keys[7].is_physical_pressed = true;
|
||||
PORTC |= (1 << 2);
|
||||
|
||||
PORTC &= ~(1 << 1);
|
||||
row1 = PIND & MASK_PIND;
|
||||
if (row1 & (1 << 4)) keys[8].is_physical_pressed = false; else keys[8].is_physical_pressed = true;
|
||||
if (row1 & (1 << 5)) keys[9].is_physical_pressed = false; else keys[9].is_physical_pressed = true;
|
||||
if (row1 & (1 << 6)) keys[10].is_physical_pressed = false; else keys[10].is_physical_pressed = true;
|
||||
if (row1 & (1 << 7)) keys[11].is_physical_pressed = false; else keys[11].is_physical_pressed = true;
|
||||
PORTC |= (1 << 1);
|
||||
|
||||
PORTC &= ~(1 << 0);
|
||||
row1 = PIND & MASK_PIND;
|
||||
if (row1 & (1 << 4)) keys[12].is_physical_pressed = false; else keys[12].is_physical_pressed = true;
|
||||
if (row1 & (1 << 5)) keys[13].is_physical_pressed = false; else keys[13].is_physical_pressed = true;
|
||||
if (row1 & (1 << 6)) keys[14].is_physical_pressed = false; else keys[14].is_physical_pressed = true;
|
||||
if (row1 & (1 << 7)) keys[15].is_physical_pressed = false; else keys[15].is_physical_pressed = true;
|
||||
PORTC |= (1 << 0);
|
||||
}
|
||||
|
||||
void debounce(struct key *undebounce_keys) {
|
||||
|
||||
}
|
17
app.h
17
app.h
@ -1,17 +0,0 @@
|
||||
#include <stdbool.h>
|
||||
/*
|
||||
* core.h
|
||||
*
|
||||
* Created: 13.04.2023 14:23:22
|
||||
* Author: mrnek
|
||||
*/
|
||||
#define KEYS_AMOUNT 16
|
||||
|
||||
|
||||
struct key {
|
||||
unsigned long pressed_time;
|
||||
bool is_physical_pressed;
|
||||
bool is_logical_pressed;
|
||||
};
|
||||
|
||||
void check_physical_keys_state(struct key *keys);
|
187
calculator.c
Normal file
187
calculator.c
Normal file
@ -0,0 +1,187 @@
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
enum calculator_state {
|
||||
NUMBER_FIRST,
|
||||
NUMBER_SECOND,
|
||||
RESULT
|
||||
};
|
||||
enum calculator_op {
|
||||
NO_OP,
|
||||
ADD,
|
||||
SUB,
|
||||
DIV,
|
||||
MUL
|
||||
};
|
||||
struct calculator {
|
||||
// 1 ÷èñëî
|
||||
int num1;
|
||||
// 2 ÷èñëî
|
||||
int num2;
|
||||
// 3 îïåðàöèÿ
|
||||
enum calculator_op operation;
|
||||
// 4 ðåçóëüòàò
|
||||
int result;
|
||||
// 5 ñîñòîÿíèå àòîìàòà
|
||||
enum calculator_state state;
|
||||
};
|
||||
|
||||
|
||||
|
||||
int main() {
|
||||
struct calculator calc;
|
||||
calc.state = NUMBER_FIRST;
|
||||
calc.operation = NO_OP;
|
||||
calc.num1 = 0;
|
||||
calc.num2 = 0;
|
||||
calc.result = 0;
|
||||
int input;
|
||||
while (1) {
|
||||
scanf("%d", &input);
|
||||
switch (calc.state) {
|
||||
case NUMBER_FIRST:
|
||||
switch (input) {
|
||||
case 0:
|
||||
calc.num1 = (calc.num1 * 10) + 7;
|
||||
break;
|
||||
case 1:
|
||||
calc.num1 = (calc.num1 * 10) + 8;
|
||||
break;
|
||||
case 2:
|
||||
calc.num1 = (calc.num1 * 10) + 9;
|
||||
break;
|
||||
case 3:
|
||||
calc.operation = DIV;
|
||||
calc.state = NUMBER_SECOND;
|
||||
break;
|
||||
case 4:
|
||||
calc.num1 = (calc.num1 * 10) + 4;
|
||||
break;
|
||||
case 5:
|
||||
calc.num1 = (calc.num1 * 10) + 5;
|
||||
break;
|
||||
case 6:
|
||||
calc.num1 = (calc.num1 * 10) + 6;
|
||||
break;
|
||||
case 7:
|
||||
calc.operation = MUL;
|
||||
calc.state = NUMBER_SECOND;
|
||||
break;
|
||||
case 8:
|
||||
calc.num1 = (calc.num1 * 10) + 1;
|
||||
break;
|
||||
case 9:
|
||||
calc.num1 = (calc.num1 * 10) + 2;
|
||||
case 10:
|
||||
calc.num1 = (calc.num1 * 10) + 3;
|
||||
break;
|
||||
case 11:
|
||||
calc.operation = SUB;
|
||||
calc.state = NUMBER_SECOND;
|
||||
break;
|
||||
case 12:
|
||||
calc.num1 = 0;
|
||||
break;
|
||||
case 13:
|
||||
calc.state = NUMBER_FIRST;
|
||||
calc.operation = NO_OP;
|
||||
calc.num1 = 0;
|
||||
calc.num2 = 0;
|
||||
calc.result = 0;
|
||||
break;
|
||||
case 14:
|
||||
break;
|
||||
case 15:
|
||||
calc.operation = ADD;
|
||||
calc.state = NUMBER_SECOND;
|
||||
break;
|
||||
}
|
||||
printf("num1: %d | num2: %d | result: %d | operation: %d | state: %d", calc.num1, calc.num2, calc.result, calc.operation, calc.state);
|
||||
break;
|
||||
|
||||
case NUMBER_SECOND:
|
||||
switch (input) {
|
||||
case 0:
|
||||
calc.num2 = (calc.num2 * 10) + 7;
|
||||
break;
|
||||
case 1:
|
||||
calc.num2 = (calc.num2 * 10) + 8;
|
||||
break;
|
||||
case 2:
|
||||
calc.num2 = (calc.num2 * 10) + 9;
|
||||
break;
|
||||
case 3:
|
||||
break;
|
||||
case 4:
|
||||
calc.num2 = (calc.num2 * 10) + 4;
|
||||
break;
|
||||
case 5:
|
||||
calc.num2 = (calc.num2 * 10) + 5;
|
||||
break;
|
||||
case 6:
|
||||
calc.num2 = (calc.num2 * 10) + 6;
|
||||
break;
|
||||
case 7:
|
||||
break;
|
||||
case 8:
|
||||
calc.num2 = (calc.num2 * 10) + 1;
|
||||
break;
|
||||
case 9:
|
||||
calc.num2 = (calc.num2 * 10) + 2;
|
||||
case 10:
|
||||
calc.num2 = (calc.num2 * 10) + 3;
|
||||
break;
|
||||
case 11:
|
||||
break;
|
||||
case 12:
|
||||
calc.num2 = 0;
|
||||
break;
|
||||
case 13:
|
||||
calc.state = NUMBER_FIRST;
|
||||
calc.operation = NO_OP;
|
||||
calc.num1 = 0;
|
||||
calc.num2 = 0;
|
||||
calc.result = 0;
|
||||
break;
|
||||
case 14:
|
||||
switch (calc.operation) {
|
||||
case ADD:
|
||||
calc.result = calc.num1 + calc.num2;
|
||||
calc.state = RESULT;
|
||||
break;
|
||||
case SUB:
|
||||
calc.result = calc.num1 - calc.num2;
|
||||
calc.state = RESULT;
|
||||
break;
|
||||
case MUL:
|
||||
calc.result = calc.num1 * calc.num2;
|
||||
calc.state = RESULT;
|
||||
break;
|
||||
case DIV:
|
||||
calc.result = calc.num1 / calc.num2;
|
||||
calc.state = RESULT;
|
||||
break;
|
||||
|
||||
}
|
||||
break;
|
||||
case 15:
|
||||
break;
|
||||
}
|
||||
printf("num1: %d | num2: %d | result: %d | operation: %d | state: %d", calc.num1, calc.num2, calc.result, calc.operation, calc.state);
|
||||
break;
|
||||
case RESULT:
|
||||
switch (input) {
|
||||
case 14:
|
||||
calc.state = NUMBER_FIRST;
|
||||
calc.operation = NO_OP;
|
||||
calc.num1 = 0;
|
||||
calc.num2 = 0;
|
||||
calc.result = 0;
|
||||
printf("num1: %d | num2: %d | result: %d | operation: %d | state: %d", calc.num1, calc.num2, calc.result, calc.operation, calc.state);
|
||||
break;
|
||||
}
|
||||
break;
|
||||
|
||||
}
|
||||
}
|
||||
}
|
68
keyboard.c
Normal file
68
keyboard.c
Normal file
@ -0,0 +1,68 @@
|
||||
/*
|
||||
* core.c
|
||||
*
|
||||
* Created: 13.04.2023 14:28:40
|
||||
* Author: mrnek
|
||||
*/
|
||||
#include <avr/io.h>
|
||||
#include <stdbool.h>
|
||||
#include <stdio.h>
|
||||
#include "keyboard.h"
|
||||
#include "timer.h"
|
||||
|
||||
#define MASK_PIND 0b11110000
|
||||
|
||||
static void set_key_event(struct key *button);
|
||||
|
||||
void get_physical_keys(struct key *keys) {
|
||||
uint8_t row;
|
||||
for (int i = 0; i < KEYS_AMOUNT / 4; i++) {
|
||||
PORTC &= ~(1 << i);
|
||||
for(volatile long i = 0; i < 1000; i++){};
|
||||
row = PIND & MASK_PIND;
|
||||
if (row & (1 << 4)) keys[i * 4].physical = false; else keys[i * 4].physical = true;
|
||||
if (row & (1 << 5)) keys[i * 4 + 1].physical = false; else keys[i * 4 + 1].physical = true;
|
||||
if (row & (1 << 6)) keys[i * 4 + 2].physical = false; else keys[i * 4 + 2].physical = true;
|
||||
if (row & (1 << 7)) keys[i * 4 + 3].physical = false; else keys[i * 4 + 3].physical = true;
|
||||
PORTC |= (1 << i);
|
||||
}
|
||||
}
|
||||
|
||||
void get_logical_keys(struct key *undc_keys) {
|
||||
unsigned long timestamp = get_millis();
|
||||
for(int i = 0; i < KEYS_AMOUNT; i++) {
|
||||
if (undc_keys[i].physical != undc_keys[i].logical) {
|
||||
if (timestamp - undc_keys[i].time >= DEBOUNCE_MS_THRESHOLD) {
|
||||
undc_keys[i].logical = undc_keys[i].physical;
|
||||
set_key_event(&undc_keys[i]);
|
||||
}
|
||||
} else {
|
||||
undc_keys[i].time = timestamp;
|
||||
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void fill_buttons_names(struct key *buttons) {
|
||||
for (int i = 0; i < KEYS_AMOUNT; i++) {
|
||||
buttons[i].button = i;
|
||||
}
|
||||
}
|
||||
|
||||
/************************************************************************/
|
||||
/* */
|
||||
/************************************************************************/
|
||||
static void set_key_event(struct key *button) {
|
||||
if (button->logical == true) {
|
||||
button->on_down = true;
|
||||
} else {
|
||||
button->on_up = true;
|
||||
}
|
||||
}
|
||||
|
||||
void get_event(struct key *button, bool *up, bool *down) {
|
||||
*up = button->on_up;
|
||||
*down = button->on_down;
|
||||
button->on_up = false;
|
||||
button->on_down = false;
|
||||
}
|
27
keyboard.h
Normal file
27
keyboard.h
Normal file
@ -0,0 +1,27 @@
|
||||
#include <stdbool.h>
|
||||
/*
|
||||
* core.h
|
||||
*
|
||||
* Created: 13.04.2023 14:23:22
|
||||
* Author: mrnek
|
||||
*/
|
||||
#define KEYS_AMOUNT 16
|
||||
#define DEBOUNCE_MS_THRESHOLD 20
|
||||
|
||||
struct key {
|
||||
char button;
|
||||
unsigned long time;
|
||||
bool physical;
|
||||
bool logical;
|
||||
bool on_down;
|
||||
bool on_up;
|
||||
};
|
||||
|
||||
|
||||
void get_physical_keys(struct key *keys);
|
||||
|
||||
void get_logical_keys(struct key *undc_keys);
|
||||
|
||||
void fill_buttons_names(struct key *buttons);
|
||||
|
||||
void get_event(struct key *button, bool *up, bool *down);
|
131
main.c
131
main.c
@ -3,68 +3,105 @@
|
||||
#include <stdbool.h>
|
||||
#include <stdio.h>
|
||||
#include "timer.h"
|
||||
#include "app.h"
|
||||
#include "keyboard.h"
|
||||
#include "uart.h"
|
||||
|
||||
void setup() {
|
||||
USART_Init(MYUBRR);
|
||||
|
||||
DDRD = 0b00001111;
|
||||
PORTD = 0b11110000;
|
||||
void setup_registers() {
|
||||
MCUCR &= ~(1 << 4); // PUD
|
||||
MCUCR &= ~(1 << 5);
|
||||
MCUCR &= ~(1 << 6);
|
||||
MCUCR &= ~(1 << 7);
|
||||
DDRD &= 0x0F;
|
||||
PORTD |= 0xF0;
|
||||
|
||||
DDRC = 0b111111;
|
||||
PORTC = 0b001111;
|
||||
}
|
||||
|
||||
|
||||
void countKeysPressedTime(char *keys) {
|
||||
|
||||
for (int i = 0; i < 16; i++) {
|
||||
keyTimeMap[i].key = i;
|
||||
if (keys[i] == 0) {
|
||||
keyTimeMap[i].time += 5;
|
||||
} else {
|
||||
keyTimeMap[i].time = 0;
|
||||
}
|
||||
}
|
||||
void setup() {
|
||||
USART_Init(MYUBRR);
|
||||
setup_timer();
|
||||
setup_registers();
|
||||
}
|
||||
|
||||
//void debounce(char *keys) {
|
||||
//if (ms % 5 != 0) return;
|
||||
//isDebounced = 0;
|
||||
//checkKeysState(newKeysState);
|
||||
//
|
||||
//for (int i = 0; i < 16; i++) {
|
||||
//if (newKeysState[i] != keys[i]) {
|
||||
//confidenceLevel = 0;
|
||||
//}
|
||||
//}
|
||||
//
|
||||
//if (confidenceLevel < 6) confidenceLevel++;
|
||||
//if (confidenceLevel == 5) {
|
||||
//for (int i = 0; i < 16; i++) {
|
||||
//debouncedKeys[i] = keys[i];
|
||||
//}
|
||||
//isDebounced = 1;
|
||||
//confidenceLevel = 0;
|
||||
//}
|
||||
//}
|
||||
|
||||
void loop() {
|
||||
|
||||
}
|
||||
int up_amoumt = 0;
|
||||
int down_amount = 0;
|
||||
int diff = 0;
|
||||
int diff_prev = 0;
|
||||
|
||||
int main(void) {
|
||||
|
||||
setup();
|
||||
puts("Hello\r\n");
|
||||
struct key keys[KEYS_AMOUNT];
|
||||
fill_buttons_names(keys);
|
||||
while(1) {
|
||||
get_physical_keys(keys);
|
||||
get_logical_keys(keys);
|
||||
|
||||
USART_Transmit(0x6F);
|
||||
//for(volatile long i = 0; i < 100000; i++){};
|
||||
loop();
|
||||
for (int i = 0; i < KEYS_AMOUNT; i++) {
|
||||
bool up, down;
|
||||
get_event(&keys[i], &up, &down);
|
||||
if (down)
|
||||
{
|
||||
printf("%d", keys[i].button);
|
||||
}
|
||||
}
|
||||
|
||||
//down_amount = 0;
|
||||
//up_amoumt = 0;
|
||||
//
|
||||
//for (int i = 0; i < KEYS_AMOUNT; i++) {
|
||||
//if (keys[i].logical == false) {
|
||||
//down_amount++;
|
||||
//} else {
|
||||
//up_amoumt++;
|
||||
//}
|
||||
////bool up;
|
||||
////bool down;
|
||||
////get_event(&keys[i], &up, &down);
|
||||
////if (down) {
|
||||
////down_amount++;
|
||||
////}
|
||||
////else {
|
||||
////up_amoumt++;
|
||||
////}
|
||||
//}
|
||||
//diff = up_amoumt;
|
||||
////if (diff != diff_prev) {
|
||||
//for (int i = 0; i < KEYS_AMOUNT; i++) {
|
||||
//bool up, down;
|
||||
//get_event(&keys[i], &up, &down);
|
||||
////if (keys[i].logical == true) {
|
||||
////printf("%d", keys[i].button);
|
||||
////}
|
||||
//if (down) {
|
||||
////printf("%s\r\n", i);
|
||||
//printf("%d", keys[i].button);
|
||||
//}
|
||||
//}
|
||||
////}
|
||||
//diff_prev = diff;
|
||||
}
|
||||
}
|
||||
|
||||
int fputc(int ch, FILE *stream){
|
||||
USART_Transmit(ch);
|
||||
int puts(const char * __str){
|
||||
int len = strlen(__str);
|
||||
for(size_t i = 0; i < len; i++){
|
||||
while (!(UCSR0A & (1<<UDREn)));
|
||||
UDR0 = __str[i];
|
||||
}
|
||||
}
|
||||
|
||||
int printf( const char * format, ... )
|
||||
{
|
||||
char buffer[16];
|
||||
va_list args;
|
||||
va_start (args, format);
|
||||
size_t s = vsprintf (buffer,format, args);
|
||||
for(size_t i = 0; i < s; i++){
|
||||
while (!(UCSR0A & (1<<UDREn)));
|
||||
UDR0 = buffer[i];
|
||||
}
|
||||
va_end (args);
|
||||
return s;
|
||||
}
|
26
timer.c
Normal file
26
timer.c
Normal file
@ -0,0 +1,26 @@
|
||||
#include <avr/interrupt.h>
|
||||
#include "timer.h"
|
||||
/*
|
||||
* timer.c
|
||||
*
|
||||
* Created: 13.04.2023 14:13:50
|
||||
* Author: mrnek
|
||||
*/
|
||||
|
||||
static unsigned long millis = 0;
|
||||
|
||||
void setup_timer() {
|
||||
TCCR1B |= (1 << WGM12); // настраиваем режим CTC (Clear Timer on Compare Match)
|
||||
OCR1A = 250; // задаем значение, при котором будет срабатывать прерывание (в данном случае 15999, так как Atmega328p работает на частоте 16 МГц и нужно получить прерывание раз в 1 мс)
|
||||
TIMSK1 |= (1 << OCIE1A); // включаем прерывание по совпадению со значением регистра OCR1A
|
||||
sei(); // разрешаем глобальные прерывания
|
||||
TCCR1B |= (1 << CS11) | (1 << CS10); // запускаем таймер с предделителем 64 (16000000/64=250000)
|
||||
}
|
||||
|
||||
unsigned long get_millis() {
|
||||
return millis;
|
||||
}
|
||||
|
||||
ISR(TIMER1_COMPA_vect) {
|
||||
++millis;
|
||||
}
|
10
timer.h
Normal file
10
timer.h
Normal file
@ -0,0 +1,10 @@
|
||||
/*
|
||||
* timer.h
|
||||
*
|
||||
* Created: 13.04.2023 14:12:16
|
||||
* Author: mrnek
|
||||
*/
|
||||
|
||||
void setup_timer();
|
||||
|
||||
unsigned long get_millis();
|
28
uart.c
Normal file
28
uart.c
Normal file
@ -0,0 +1,28 @@
|
||||
/*
|
||||
* uart.c
|
||||
*
|
||||
* Created: 13.04.2023 15:32:02
|
||||
* Author: mrnek
|
||||
*/
|
||||
#include <avr/io.h>
|
||||
#include "uart.h"
|
||||
|
||||
void USART_Init(unsigned int ubrr) {
|
||||
/*Set baud rate */
|
||||
UBRR0H = (unsigned char)(ubrr>>8);
|
||||
UBRR0L = (unsigned char)ubrr;
|
||||
UBRR0H = 0;
|
||||
UBRR0L = 8;
|
||||
/* Set frame format: 8data, 2stop bit */
|
||||
UCSR0C = (1<<USBS0)|(3<<UCSZ00);
|
||||
/* Enable receiver and transmitter */
|
||||
UCSR0B = (1<<RXEN0)|(1<<TXEN0);
|
||||
}
|
||||
|
||||
void USART_Transmit(unsigned char data) {
|
||||
/* Wait for empty transmit buffer */
|
||||
while (!(UCSR0A & (1<<UDREn)))
|
||||
;
|
||||
/* Put data into buffer, sends the data */
|
||||
UDR0 = data;
|
||||
}
|
Loading…
Reference in New Issue
Block a user