AVR/Atmega128 강좌
적외선 통신( IR ) - 리모컨 NFC format을 시리얼로 출력하는 코드
webnautes
2023. 10. 13. 22:44
반응형
LG에서 사용하는 NEC format을 시리얼로 출력해주는 소스코드입니다. 리모컨에서 버튼을 누르면 해당되는 값을 16진수로 변환하여 시리얼로 출력해줍니다.
2016. 6. 5 최초작성
실행화면입니다.
아래 주소에 있는 코드를 atmega128에서 동작하도록 수정하고 시리얼 통신으로 결과값을 받아볼 수 있도록 수정하였습니다.
http://extremeelectronics.co.in/code-libraries/using-ir-remote-with-avr-mcus/
10진수를 16진수로 변환하는 부분은 아래 사이트에 있는 코드를 사용했습니다.
https://dojang.io/mod/page/view.php?id=743
usart.h
#ifndef USART_H_ #define USART_H_ #include <avr/io.h> #include "serial.h" #define UBRR_ASYNC_NORMAL(br) ((F_CPU / (br * 16UL)) - 1) #define _BV(bit) \ (1 << (bit)) void serial_init(uint32_t baudrate) { // Set Baudrate UBRR0H = (UBRR_ASYNC_NORMAL(baudrate) >> 8); UBRR0L = UBRR_ASYNC_NORMAL(baudrate); // Enable RX and TX UCSR0B |= _BV(RXEN0) | _BV(TXEN0); // Set 8N1 Framing UCSR0C |= _BV(UCSZ00) | _BV(UCSZ01); } void serial_write(char c) { while ( !(UCSR0A & (1 << UDRE0)) ); UDR0 = c; } char serial_read() { while ( !(UCSR0A & (1 << RXC0)) ); return UDR0; } #endif /* USART_H_ */ |
main.c
#define F_CPU 16000000UL #include <avr/io.h> #include <avr/interrupt.h> #include <stdio.h> #include <stdlib.h> #include <util/delay.h> #include "usart.h" //States #define IR_VALIDATE_LEAD_HIGH 0 #define IR_VALIDATE_LEAD_LOW 1 #define IR_RECEIVE_BITS 3 #define IR_WAIT_STOP_BIT 4 //Others #define TOL 0.1 //Tollerence for timming #define QMAX 8 //Size of the Remote command buffer #define RC_NONE 255 //This val is returned by GetRemoteCmd when no key is pressed //Globals volatile unsigned int Time; //Main timer, stores time in 10us, //Updated by ISR(TIMER0_COMP) volatile unsigned char BitNo; //Pos of next BIT volatile unsigned char ByteNo; //Pos of current Byte volatile unsigned char IrData[4]; //The four data Bytes of Ir Packet //2-Byte Address 2-Byte Data volatile unsigned char IrCmdQ[QMAX];//Final Command Received (Buffer) volatile unsigned char PrevCmd; //Used for repeat //Variables used for start repeating only after a key is pressed for certain time volatile unsigned char Repeat; //1=yes 0=no volatile unsigned char RCount; //Repeat count volatile char QFront=-1,QEnd=-1; volatile unsigned char State; //State of receiver volatile unsigned char Edge; //Edge of interrupt [ RISING=1 OR FALLING=0 ] volatile unsigned int stop; /**********************************************************************************************/ /* F U N C T I O N S S T A R T S */ /**********************************************************************************************/ void RemoteInit() { char i; for(i=0;i<4;i++) IrData[i]=0; stop=0; State=IR_VALIDATE_LEAD_HIGH; Edge=0; Repeat=0; //Setup Timer0 //------------ TCCR0|=((1<<CS00)|(1<<WGM01)); //Prescaler : Fcpu Mode : CTC TIMSK|=(1<<OCIE0); //Enable Output Compare Interrupt OCR0=160; //Set Compare Value //Set Up INT0 //------------ EICRA |= (1<<ISC01); //INT ON Falling Edge EIMSK |= (1<<INT0); //Enable INT0 //Enable Interrupts //------------------- sei(); } ISR(TIMER0_COMP_vect) { Time++; } ISR(INT0_vect) { EIMSK&=(~(1<<INT0)); //Disable INT1 sei(); if(stop) return; unsigned int TempTime=Time; Time=0; TCNT0=0; switch(State) { case IR_VALIDATE_LEAD_HIGH: { if(Edge) { //Rising if((TempTime>(900-(900*TOL))) && (TempTime<(900+(900*TOL)))) { //Lead High Correct State=IR_VALIDATE_LEAD_LOW; //INT ON FALLING EDGE EICRA&=(~((1<<ISC01)|(1<<ISC00))); EICRA|=(1<<ISC01); Edge=0; } else { ResetIR(); } } else { //Falling EICRA|=((1<<ISC01)|(1<<ISC00)); //Set INT on Rising Edge Edge=1; } break; } case IR_VALIDATE_LEAD_LOW: { if((TempTime>(450-(450*TOL))) && (TempTime<(450+(450*TOL)))) { //Got a valid leader State=IR_RECEIVE_BITS; BitNo=0; ByteNo=0; EICRA|=((1<<ISC01)|(1<<ISC00)); //Set INT on Rising Edge Edge=1; } else if((TempTime>200) && (TempTime<245)) { if(Repeat) { //Got a repeat pulse if((QEnd==(QMAX-1) && QFront==0)||((QEnd+1)==QFront)) { QFront++; if(QFront==(QMAX)) QFront=0; } if(QEnd==(QMAX-1)) QEnd=0; else QEnd++; IrCmdQ[QEnd]=PrevCmd; if(QFront==-1) QFront=0; } else { RCount++; if(RCount==4) Repeat=1; } ResetIR(); }else { ResetIR(); } break; } case IR_RECEIVE_BITS: { if(Edge) { //Rising if((TempTime>50) && (TempTime<69)) { //Correct Beg of BIT found //INT ON FALLING EDGE EICRA&=(~((1<<ISC01)|(1<<ISC00))); EICRA|=(1<<ISC01); Edge=0; } else { ResetIR(); } } else { //Falling if((TempTime>41) && (TempTime<58)) { //We got a '0' here BitNo++; if(BitNo==8) { BitNo=0; ByteNo++; if(ByteNo==4) { State=IR_WAIT_STOP_BIT; } } EICRA|=((1<<ISC01)|(1<<ISC00)); //Set INT on Rising Edge Edge=1; }else if((TempTime>(169-(169*TOL))) && (TempTime<(169+(169*TOL)))) { //We Have got a '1' here IrData[ByteNo]|=(1<<BitNo); BitNo++; if(BitNo==8) { BitNo=0; ByteNo++; if(ByteNo==4) { State=IR_WAIT_STOP_BIT; } } EICRA |=((1<<ISC01)|(1<<ISC00)); //Set INT on Rising Edge Edge=1; }else { ResetIR(); } } break; } case IR_WAIT_STOP_BIT: { if(Edge) { //Check for integrity if(IrData[2]==((unsigned char)~IrData[3])) { //Now We Have Got a packet //Add its Cmd to Queue //Step1:Check of Q full if((QEnd==(QMAX-1) && QFront==0)||((QEnd+1)==QFront)) { QFront++; if(QFront==(QMAX)) QFront=0; } if(QEnd==(QMAX-1)) QEnd=0; else QEnd++; IrCmdQ[QEnd]=IrData[2]; PrevCmd=IrData[2]; if(QFront==-1) QFront=0; //Prevent repeating immediatly Repeat=0;//It will be enabled after 4 repeat pulses RCount=0; ResetIR(); } } } break; } EIMSK|=(1<<INT0); //Enable INT1 } void ResetIR() { char i; for(i=0;i<4;i++) IrData[i]=0; State=IR_VALIDATE_LEAD_HIGH; //INT ON FALLING EDGE EICRA &=(~((1<<ISC01)|(1<<ISC00))); EICRA |=(1<<ISC01); Edge=0; } unsigned char GetRemoteCmd(char wait) { unsigned char cmd; if(wait) while(QFront==-1); else if(QFront==-1) return (RC_NONE); cmd=IrCmdQ[QFront]; if(QFront==QEnd) QFront=QEnd=-1; else { if(QFront==(QMAX-1)) QFront=0; else QFront++; } return cmd; } void main() { uint8_t cmd=0; serial_init(9600); RemoteInit(); DDRD |= 1<< 3; while(1) { cmd=GetRemoteCmd(1); if ( cmd == 0xa5 ) continue; int decimal = cmd; char hexadecimal[20] = { 0, }; // 16진수로 된 문자열을 저장할 배열 int position = 0; while (1) { int mod = decimal % 16; // 16으로 나누었을 때 나머지를 구함 if (mod < 10) // 나머지가 10보다 작으면 { // 숫자 0의 ASCII 코드 값 48 + 나머지 hexadecimal[position] = 48 + mod; } else // 나머지가 10보다 크거나 같으면 { // 나머지에서 10을 뺀 값과 영문 대문자 A의 ASCII 코드 값 65를 더함 hexadecimal[position] = 65 + (mod - 10); } decimal = decimal / 16; // 16으로 나눈 몫을 저장 position++; // 자릿수 변경 if (decimal == 0) // 몫이 0이되면 반복을 끝냄 break; } // 배열의 요소를 역순으로 출력 for (int i = position - 1; i >= 0; i--) { serial_write(hexadecimal[i]); } serial_write('\r'); serial_write('\n'); } } |
반응형