code:interrupt_driven_nmea
This is an old revision of the document!
This is a rather unconvensional approach to a NMEA parser - completely based on an ISR to parse the incoming bytes. It uses 2.2KB of code space and 44 bytes of RAM.
#include "global.h" #include "avrlibdefs.h" #include "avrlibtypes.h" #include <avr/io.h> #include <stdlib.h> #include <stdio.h> #include <string.h> #include <avr/interrupt.h> // Interrupt driven NMEA parser for Atmel AVR //header file starts here #define minutes (float)1.0/60.0 #define baudrate 4800 struct { volatile float longitude; volatile float latitude; volatile float altitude; volatile float speed; volatile float heading; volatile u08 packetflag; //packetflag lets us see when our packet has been updated volatile u08 status; } gps; //end of header file int main() { outb(UCSR0B, BV(RXCIE0)|BV(TXCIE0)|BV(RXEN0)|BV(TXEN0)); //mega xx8 registers - enable tx and rx, with interrupts u16 bauddiv = ((F_CPU+(baudrate*8L))/(baudrate*16L)-1); outb(UBRR0L, bauddiv); outb(UBRR0H, bauddiv>>8); } SIGNAL(SIG_USART_RECV) //UART interrupt on mega xx8 series { static char buffer[5]; static u08 GGA; static u08 RMC; static u08 stage; static u08 commacount; static u08 bufferindex; static u08 pointcount; char c=UDR0; switch(c) { case '$': //start of a packet commacount=0; bufferindex=0; pointcount=0; stage=0; memset(buffer,' ',5); GGA=FALSE; RMC=FALSE; //we dont know what sort of packet it is yet break; case ',': commacount++; bufferindex=0; pointcount=0; stage=0; break; case '.': pointcount++; //we need to be able to detect number of points in the case of altitude // break; THIS is deliberate, we want to run when we get a decimal point. default: //we have some of the CSV data buffer[bufferindex]=c; //stick the character in our buffer if(GGA) { if(commacount==2) //the latitude from the GGA { if(bufferindex<1) { bufferindex++; } else { if(!stage) { gps.latitude=(float)atoi(buffer); //integer degrees } if(stage==1) { gps.latitude+=minutes*(float)atoi(buffer); //integer minutes } if(stage==2) { gps.latitude+=0.01*minutes*(float)atoi(buffer); //decimal degrees } memset(buffer,' ',5); stage++; } } if(commacount==3 && c=='S') { gps.latitude=-gps.latitude; } if(commacount==4) { if( (stage && bufferindex<1) || bufferindex<2) { bufferindex++; } else { if(!stage) { gps.longitude=(float)atoi(buffer); //integer degrees } if(stage==1) { gps.longitude+=minutes*(float)atoi(buffer); //integer minutes } if(stage==2) { gps.longitude+=0.01*minutes*(float)atoi(buffer); //decimal degrees } memset(buffer,' ',5); stage++; } } if(commacount==5 && c=='W') { gps.longitude=-gps.longitude; } if(commacount==6) { gps.status=atoi(&c); } if(commacount==9) { if(stage) //decimal altitude { gps.altitude+=((float)atoi(&c))*0.1; memset(buffer,' ',5); } else { if(!pointcount) //wait until we get to a decimal point { bufferindex++; } else { gps.altitude=(float)atoi(buffer); memset(buffer,' ',5); stage++; } } } } else if(RMC) { if(commacount==7) //speed in knots { if(stage) { gps.speed+=((float)atoi(&c))*0.1; memset(buffer,' ',5); } else { if(!pointcount) { bufferindex++; } else { gps.speed=(float)atoi(buffer); memset(buffer,' ',5); stage++; } } } if(commacount==8) //the heading field { if(stage) { gps.heading+=((float)atoi(&c))*0.1; memset(buffer,' ',5); gps.packetflag=TRUE; //this is usually the last interesting part of the fix info to come through } else { if(!pointcount) { bufferindex++; } else { gps.heading=(float)atoi(buffer); memset(buffer,' ',5); stage++; } } } } else { if(commacount==0) //the header { if(bufferindex<4) { bufferindex++; //increase the position in the buffer } else { if(buffer=="GPGGA") //the last character will be a space { GGA=TRUE; } if(buffer=="GPRMC") { RMC=TRUE; } memset(buffer,' ',5); //wipe the buffer so it can be reused } } } } }
code/interrupt_driven_nmea.1208027242.txt.gz · Last modified: 2008/07/19 23:31 (external edit)