초기화
//UART 관련 void Init_uart() { int pbClk = PBCLK; //시리얼 통신, 인터럽트 초기화 OpenUART1(UART_EN, UART_RX_ENABLE | UART_TX_ENABLE, pbClk/16/DESIRED_BAUDRATE-1); ConfigIntUART1(UART_INT_PR5 | UART_RX_INT_EN); OpenUART2(UART_EN, UART_RX_ENABLE | UART_TX_ENABLE, pbClk/16/DESIRED_BAUDRATE-1); ConfigIntUART2(UART_INT_PR3 | UART_RX_INT_EN); }
인터럽트 (UART2)
//UART2 인터럽트 void __ISR(_UART2_VECTOR, ipl3) _uart2(void) { // Is this an RX interrupt? if(INTGetFlag(INT_SOURCE_UART_RX(UART2))) { // Clear the RX interrupt Flag INTClearFlag(INT_SOURCE_UART_RX(UART2)); U2STAbits.OERR=0; g_RxData = UARTGetDataByte(UART2); //myprintf("[%c]\r\n", g_RxData ); UARTGetDataByte(UART2); } // We don't care about TX interrupt if ( INTGetFlag(INT_SOURCE_UART_TX(UART2)) ) { INTClearFlag(INT_SOURCE_UART_TX(UART2)); } }
myprintf 사용을 위한 추가 부분 (UART2)
myprintf는 vs등에서 printf와 같은 기능이 있다.
이 함수에서는 %d(decimal), %s(string), %c(char), %x, %X(hexadecimal), %u(long) 를 지원한다.
%f를 지원하지 않는게 참 아쉬울뿐이다.. 따로 만들어 쓰자.
static void printchar(char **str, int c)
함수의 else안에 c라는 char를 프린트 하는 함수를 넣어주면 된다.
PIC32MX460에서는
void PutCharacter(const char character); void PutCharacter(const char character) { while(!UARTTransmitterIsReady(UART2)); UARTSendDataByte(UART2, character); while(!UARTTransmissionHasCompleted(UART2)); }
를 넣어주고
TMS320F2808 에서는
while (SciaRegs.SCIFFTX.bit.TXFFST != 0) {} SciaRegs.SCITXBUF=c;
를 넣어주면 된다.
물론 해당 헤더를 include해야 한다.
/* Copyright 2001, 2002 Georges Menie (www.menie.org) stdarg version contributed by Christian Ettinger This program is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include <plib.h> #include "pic32_uart.h" /* putchar is the only external dependency for this file, if you have a working putchar, leave it commented out. If not, uncomment the define below and replace outbyte(c) by your own function call. #define putchar(c) outbyte(c) */ #include <stdarg.h> void delay( unsigned int i) { while(i--); } static void printchar(char **str, int c) { extern int putchar(int c); if (str) { **str = c; ++(*str); } else { // while(U2STAbits.TRMT == 0); // U2TXREG = c; PutCharacter(c); //? // delay(100); } } #define PAD_RIGHT 1 #define PAD_ZERO 2 static int prints(char **out, const char *string, int width, int pad) { register int pc = 0, padchar = ' '; if (width > 0) { register int len = 0; register const char *ptr; for (ptr = string; *ptr; ++ptr) ++len; if (len >= width) width = 0; else width -= len; if (pad & PAD_ZERO) padchar = '0'; } if (!(pad & PAD_RIGHT)) { for ( ; width > 0; --width) { printchar (out, padchar); ++pc; } } for ( ; *string ; ++string) { printchar (out, *string); ++pc; } for ( ; width > 0; --width) { printchar (out, padchar); ++pc; } return pc; } /* the following should be enough for 32 bit int */ #define PRINT_BUF_LEN 12 static int printi(char **out, int i, int b, int sg, int width, int pad, int letbase) { char print_buf[PRINT_BUF_LEN]; register char *s; register int t, neg = 0, pc = 0; register unsigned int u = i; if (i == 0) { print_buf[0] = '0'; print_buf[1] = '\0'; return prints (out, print_buf, width, pad); } if (sg && b == 10 && i < 0) { neg = 1; u = -i; } s = print_buf + PRINT_BUF_LEN-1; *s = '\0'; while (u) { t = u % b; if( t >= 10 ) t += letbase - '0' - 10; *--s = t + '0'; u /= b; } if (neg) { if( width && (pad & PAD_ZERO) ) { printchar (out, '-'); ++pc; --width; } else { *--s = '-'; } } return pc + prints (out, s, width, pad); } static int print( char **out, const char *format, va_list args ) { register int width, pad; register int pc = 0; char scr[2]; for (; *format != 0; ++format) { if (*format == '%') { ++format; width = pad = 0; if (*format == '\0') break; if (*format == '%') goto out; if (*format == '-') { ++format; pad = PAD_RIGHT; } while (*format == '0') { ++format; pad |= PAD_ZERO; } for ( ; *format >= '0' && *format <= '9'; ++format) { width *= 10; width += *format - '0'; } if( *format == 's' ) { register char *s = (char *)va_arg( args, int ); pc += prints (out, s?s:"(null)", width, pad); continue; } if( *format == 'd' ) { pc += printi (out, va_arg( args, int ), 10, 1, width, pad, 'a'); continue; } if( *format == 'x' ) { pc += printi (out, va_arg( args, int ), 16, 0, width, pad, 'a'); continue; } if( *format == 'X' ) { pc += printi (out, va_arg( args, int ), 16, 0, width, pad, 'A'); continue; } if( *format == 'u' ) { pc += printi (out, va_arg( args, int ), 10, 0, width, pad, 'a'); continue; } if( *format == 'c' ) { /* char are converted to int then pushed on the stack */ scr[0] = (char)va_arg( args, int ); scr[1] = '\0'; pc += prints (out, scr, width, pad); continue; } } else { out: printchar (out, *format); ++pc; } } if (out) **out = '\0'; va_end( args ); return pc; } int myprintf(const char *format, ...) { va_list args; va_start( args, format ); return print( 0, format, args ); } int sprintf(char *out, const char *format, ...) { va_list args; va_start( args, format ); return print( &out, format, args ); } int snprintf( char *buf, unsigned int count, const char *format, ... ) { va_list args; ( void ) count; va_start( args, format ); return print( &buf, format, args ); } #define TEST_PRINTF #ifdef TEST_PRINTF /* * if you compile this file with * gcc -Wall $(YOUR_C_OPTIONS) -DTEST_PRINTF -c printf.c * you will get a normal warning: * printf.c:214: warning: spurious trailing `%' in format * this line is testing an invalid % at the end of the format string. * * this should display (on 32bit int machine) : * * Hello world! * printf test * (null) is null pointer * 5 = 5 * -2147483647 = - max int * char a = 'a' * hex ff = ff * hex 00 = 00 * signed -3 = unsigned 4294967293 = hex fffffffd * 0 message(s) * 0 message(s) with % * justif: "left " * justif: " right" * 3: 0003 zero padded * 3: 3 left justif. * 3: 3 right justif. * -3: -003 zero padded * -3: -3 left justif. * -3: -3 right justif. */ #endif /* To keep linker happy. */ int write( int i, char* c, int n) { return 0; }
자이로 센서는 UART1을 사용한다.
Baudrate은 115200 //
각속도를 알려주는 것이 아니라 각도 값을 실시간으로 보내준다.
Data를 계속 떤지기 때문에 Rx 버퍼가 가득 차 버릴 수 있다.
PIC32에서는 이 경우 에러 플래그를 초기화해주지 않을 경우 UART모듈이 멈춰버린다.
(이 이를 알기까지 얼마나 어려웠는데..-_-)
U1STAbits.OERR=0;
를 인터럽트 루틴 안에서 꼭 호출 해 줘야 한다.
자이로 센서의 경우 UART의 Rx만 사용하기 때문에 Tx는 다른 용도로 사용할 수 있다.
자이로 센서 (UART1)
//UART1 인터럽트 -> 자이로 void __ISR(_UART1_VECTOR, ipl5) _uart1(void) { unsigned int uCheckSum; unsigned int uTmp; unsigned char arr[2] = {0, '\0'}; // Is this an RX interrupt? if(INTGetFlag(INT_SOURCE_UART_RX(UART1))) { // Clear the RX interrupt Flag INTClearFlag(INT_SOURCE_UART_RX(UART1)); U1STAbits.OERR=0; // if(U1STAbits.OERR) // { // U1STAbits.OERR=0; // } //* if(g_GyroFlag != 0) {//자이로 각도 계산 byRxChar = UARTGetDataByte(UART1); // myprintf( "_%c", byRxChar); if(g_GyroFlag == 1) { if(byRxChar == 0xff) { g_GyroFlag = 2; myprintf(" start \r\n", arr); } else { arr[0] = byRxChar; myprintf("%c", byRxChar); } } if(g_ByRxRemainder >= 2) { m_aGyroBuf[g_ByRxRemainder] = byRxChar; g_ByRxRemainder += 1; if(g_ByRxRemainder >= 8) { g_ByRxRemainder = 0; uCheckSum = (m_aGyroBuf[1] << 8) | m_aGyroBuf[0]; uCheckSum += (m_aGyroBuf[3] << 8) | m_aGyroBuf[2]; uCheckSum += (m_aGyroBuf[5] << 8) | m_aGyroBuf[4]; if ((uCheckSum & 0xFFFF) == (unsigned int)((m_aGyroBuf[7] << 8) | m_aGyroBuf[6])) { uTmp = m_aGyroBuf[5]; uTmp = ((uTmp << 8) & 0xff00) | m_aGyroBuf[4]; // agnle : -180 ~ 180 도 까지 출력. // m_aGyroBuf[5]는 상위 비트임. // // 0~180도는 1~18000 => Hex 0~4650 // -180~0도는 47536~65536 가지 나옴. => Hex B9B0 ~ FFFF // 그러므로 47536(B9B0)에서 제일 처음 비트가 1이 셋 되면 // 음수의 각도를 나타내므로 제일 처음 비트를 검사한다. // B9B0 의 B => 1011 bit임. // 4650 의 4 => 0100 bit임. // 따라서 최상위 숫자의 제일 상위 비트만으로도 구분 가능함. if (m_aGyroBuf[5] & 0x80) { g_Gyro = 360.0 - ((65536 - uTmp)/100.0); //g_Gyro = 360.0 - ((65536 - uTmp)/10.0); } else { g_Gyro = uTmp / 100.0; //g_Gyro = uTmp / 10.0; } if(g_GyroFlag < 10) g_GyroFlag++; } } } else { if (byRxChar == 0xFF) { m_aGyroBuf[g_ByRxRemainder] = byRxChar; g_ByRxRemainder += 1; } else { g_ByRxRemainder = 0; } } } else { int temp = 0; if(temp) temp = 1; g_byCommand = UARTGetDataByte(UART1); } //*/ } // We don't care about TX interrupt if ( INTGetFlag(INT_SOURCE_UART_TX(UART1)) ) { INTClearFlag(INT_SOURCE_UART_TX(UART1)); } }
'Enginius > Firmware' 카테고리의 다른 글
VUSB (AVRUSB) usb를 avr의 gpio를 이용해서 제어. (0) | 2011.01.26 |
---|---|
NEC 코드 리모컨 (0) | 2010.04.16 |
PIC32 - TIMER (0) | 2010.04.13 |
PIC32 - SPI (0) | 2010.04.13 |
PIC32 - PWM (0) | 2010.02.19 |