마이크로 프로세서 응용 수업에 개인 텀프를 발표떄 만든 작품이다.
캡스톤 때문에 코엑스에서 전시하고 와서 정신 못차리고 있다가 할룻밤만에 만든 코드와 작품....
초기에 OP-Amp의 비교기를 이용하여 인터럽트 파형을 만들어 주려고 했는데 OP-Amp가 생각처럼 동작하지 않았다.
OP-Amp는 HA17741을 사용하였는데 양전원 용이라는거 같았다...
(아니면 코딩에서 실수를 했었을 수도.....)
(지금생각해보면 OP-Amp 보다 코딩의 실수인듯 싶다.)
대략적인 구성은 BLDC 변속기에 PWM 파형을 입력하면 변속기가 모터를 돌리고
모터의 금속부분에(나머지는 검정테이프를 붙여놈) 적외선이 반사되면 SG-2BC를 통해 빛을 감지하여
타이머의 입력 캡쳐 기능을 이용하여 TCNT값을 읽어들여 시간을 계산하여 CLDC에 출력하는 구조이다.
위 사진은 SG-2BC를 이용하여 적외선 센서를 이용하여 입력을 받는 구조를 나타낸 것이다.
위 사진은 모터가 회전할때의 수광부에서의 파형이다.
난 평소에 이클립스를 이용하여 GCC로 코딩을 하는데 CLCD는 한번도 GCC로 코딩해본적이 없는걸?..... 당장 다음날 발표를 해야 하는데 말이지....ㅠㅠ
그래서 결국 코드비젼을 이용하였다.
왜냐면 수업시간에 코드비젼으로 작업을 해서 코드비젼에 맞게 하드웨어가 작성되어 있어서 그냥 입력만 하면 됐거든......
아래는 회로도
사실 별거 없다는거..... 저기에서 OC1A로 출력을 했던거 같은데.... 회로도는 안그려져 있네...ㅠㅠ 흐미 저 오른쪽에 CON16은 뭥미??......
겁네 대충그렸구먼....ㅠ.ㅠ
하여튼 이러한 구조를 가지고 있다.....
그런데 데이터를 처리하는데서 어마어마한 실수를 해버렸다.
//// temp =TCNT3L;
//// clock=(TCNT3H<<8);
//// clock=clock|temp;
clock=(ICR3H);
clock=clock<<8;
clock=clock|temp;
이것의 차이는.... 오버플로우가 일어나고 일어나지 않고의 차이다.
그러면 숫자가 정말 개념없이 튀게 된다.....
이거때문에 얼마나 고생을 했던지.....
(도데체 인터넷에 이렇게 코드올려놓은 사람은 누구야~!!!!!)
아마도 OP-Amp가 작동하지 않는다고 생각한것도 이부분 때문이지 않을까 싶다......ㅠㅠ
하여튼 이렇게 해결하여 하루밤만에 뚝딱 만들어버림...ㅋㅋㅋㅋㅋㅋ
이클립스를 이용하여 LCD 출력하는 LCD.h 파일을 하나 만들어봐야겠당.......
(그래야 이제 마음놓고 이클립스를 쓰지.....)
소스코드
#include
#include
#include
#include
#define LedInit() DDRB |= (1<
volatile unsigned int i=0,j=0,rpm;
volatile unsigned int clock,temp,f_over;
// External Interrupt 3 service routine
//interrupt [EXT_INT3] void ext_int3_isr(void)
//{
//// if(j==2){
////// ETIMSK |=0x20;
////
//// TCNT3H=0x00;
//// TCNT3L=0x00;
//// }else if(j==3){
//// temp =TCNT3L;
//// clock=(TCNT3H<<8);
//// clock=clock|temp;
//// }else if(j>6)j=0;
//
//}
interrupt [TIM3_CAPT] void timer3_capt_isr(void)
{
// Place your code here
TCNT3H=0x00;
TCNT3L=0x00;
temp =ICR3L;
clock=(ICR3H);
clock=clock<<8;
clock=clock|temp;
// clock=(ICR3H);
f_over=0;
}
// Timer3 overflow interrupt service routine
interrupt [TIM3_OVF] void timer3_ovf_isr(void)
{
// Place your code here
clock=0;
f_over=1 ;
}
interrupt [TIM1_OVF] void timer1_ovf_isr(void)
{
i++;
j++;
}
void setMoterSpeed(unsigned int moterSpeed){
moterSpeed+=16000;
OCR1CH = (moterSpeed&0xff00)>>8; //16000 == 1ms
OCR1CL = (moterSpeed&0x00ff);
}
#define ADC_VREF_TYPE 0x40
// Read the AD conversion result
unsigned int read_adc(unsigned char adc_input)
{
ADMUX=adc_input | (ADC_VREF_TYPE & 0xff);
// Delay needed for the stabilization of the ADC input voltage
delay_us(10);
// Start the AD conversion
ADCSRA|=0x40;
// Wait for the AD conversion to complete
while ((ADCSRA & 0x10)==0);
ADCSRA|=0x10;
return ADCW;
}
// Declare your global variables here
void main(void)
{
unsigned int adc=0,time;
unsigned char str[20];
// Declare your local variables here
// Input/Output Ports initialization
// Port A initialization
// Func7=In Func6=In Func5=In Func4=In Func3=In Func2=In Func1=In Func0=In
// State7=T State6=T State5=T State4=T State3=T State2=T State1=T State0=T
PORTA=0x00;
DDRA=0x00;
// Port B initialization
// Func7=Out Func6=In Func5=In Func4=In Func3=In Func2=In Func1=In Func0=In
// State7=0 State6=T State5=T State4=T State3=T State2=T State1=T State0=T
PORTB=0x00;
DDRB=0x80;
// Port C initialization
// Func7=In Func6=In Func5=In Func4=In Func3=In Func2=In Func1=In Func0=In
// State7=T State6=T State5=T State4=T State3=T State2=T State1=T State0=T
PORTC=0x00;
DDRC=0x00;
// Port D initialization
// Func7=In Func6=In Func5=In Func4=In Func3=In Func2=In Func1=In Func0=In
// State7=T State6=T State5=T State4=T State3=T State2=T State1=T State0=T
PORTD=0x00;
DDRD=0x00;
// Port E initialization
// Func7=In Func6=In Func5=In Func4=In Func3=In Func2=In Func1=In Func0=In
// State7=T State6=T State5=T State4=T State3=T State2=T State1=T State0=T
PORTE=0x00;
DDRE=0x00;
// Port F initialization
// Func7=In Func6=In Func5=In Func4=In Func3=In Func2=In Func1=In Func0=In
// State7=T State6=T State5=T State4=T State3=T State2=T State1=T State0=T
PORTF=0x00;
DDRF=0x00;
// Port G initialization
// Func4=In Func3=In Func2=In Func1=In Func0=In
// State4=T State3=T State2=T State1=T State0=T
PORTG=0x00;
DDRG=0x00;
// Timer/Counter 0 initialization
// Clock source: System Clock
// Clock value: Timer 0 Stopped
// Mode: Normal top=FFh
// OC0 output: Disconnected
ASSR=0x00;
TCCR0=0x00;
TCNT0=0x00;
OCR0=0x00;
// Timer/Counter 1 initialization
// Clock source: System Clock
// Clock value: 16000.000 kHz
// Mode: Fast PWM top=ICR1
// OC1A output: Discon.
// OC1B output: Discon.
// OC1C output: Non-Inv.
// Noise Canceler: Off
// Input Capture on Falling Edge
// Timer1 Overflow Interrupt: Off
// Input Capture Interrupt: Off
// Compare A Match Interrupt: Off
// Compare B Match Interrupt: Off
// Compare C Match Interrupt: Off
TCCR1A=0x0A;
TCCR1B=0x19;
TCNT1H=0x00;
TCNT1L=0x00;
ICR1H=0x00;
ICR1L=0x00;
OCR1AH=0x00;
OCR1AL=0x00;
OCR1BH=0x00;
OCR1BL=0x00;
OCR1CH=(16000&0xff00)>>8;
OCR1CL=16000&0x00ff;
ICR1=64000;
// Timer/Counter 2 initialization
// Clock source: System Clock
// Clock value: Timer2 Stopped
// Mode: Normal top=FFh
// OC2 output: Disconnected
TCCR2=0x00;
TCNT2=0x00;
OCR2=0x00;
// Timer/Counter 3 initialization
// Clock source: System Clock
// Clock value: 2000.000 kHz
// Mode: Normal top=FFFFh
// OC3A output: Discon.
// OC3B output: Discon.
// OC3C output: Discon.
// Noise Canceler: Off
// Input Capture on Falling Edge
// Timer3 Overflow Interrupt: On
// Input Capture Interrupt: Off
// Compare A Match Interrupt: Off
// Compare B Match Interrupt: Off
// Compare C Match Interrupt: Off
TCCR3A=0x00;
TCCR3B=0x03;
TCNT3H=0x00;
TCNT3L=0x00;
ICR3H=0x00;
ICR3L=0x00;
OCR3AH=0x00;
OCR3AL=0x00;
OCR3BH=0x00;
OCR3BL=0x00;
OCR3CH=0x00;
OCR3CL=0x00;
// External Interrupt(s) initialization
// INT0: Off
// INT1: Off
// INT2: Off
// INT3: On
// INT3 Mode: Rising Edge
// INT4: Off
// INT5: Off
// INT6: Off
// INT7: Off
//EICRA=0x80;
//EICRB=0x00;
//EIMSK=0x08;
//EIFR=0x08;
EICRA=0x00;
EICRB=0x00;
EIMSK=0x00;
// Timer(s)/Counter(s) Interrupt(s) initialization
TIMSK=0x04;
ETIMSK=0x24;
// Analog Comparator initialization
// Analog Comparator: Off
// Analog Comparator Input Capture by Timer/Counter 1: Off
ACSR=0x80;
SFIOR=0x00;
// ADC initialization
// ADC Clock frequency: 1000.000 kHz
// ADC Voltage Reference: AVCC pin
ADMUX=ADC_VREF_TYPE & 0xff;
ADCSRA=0x84;
// LCD module initialization
lcd_init(16);
// Global enable interrupts
#asm("sei")
while (1)
{
adc=read_adc(0);
adc=adc*16;
setMoterSpeed(adc);
if(i>60){
i=0;
lcd_clear();
lcd_gotoxy(0,0);
rpm= 15000000/clock;
if(rpm<100 || f_over==1|| rpm>15000)rpm=0;
sprintf(str,"RPM : %5lu RPM",rpm);
lcd_puts(str);
lcd_gotoxy(0,1);
if(adc>16000) adc=16000;
sprintf(str,"Motor : %3lu %%",adc/160);
lcd_puts(str);
lcd_gotoxy(0,2);
time=clock/250;
sprintf(str,"Period : %3lu ms",time);
lcd_puts(str);
}
}
}