UKHAS Wiki

UK High Altitude Society

User Tools

Site Tools


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 "main.h"
//global(s)
extern gps_type gps;
 
// Interrupt driven NMEA parser for Atmel AVR
 
ISR(USART_RX_vect)			//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;                      
		default:				//we have some of the CSV data
		  if(bufferindex<5)		//dont mess up ! Dont overflow
		  {									
		    	buffer[bufferindex]=c;		//stick the character in our buffer		
		  }
		  if(GGA)
		  {
			if(commacount==2)		//the latitude from the GGA
			{
				if( (bufferindex<1  && stage!=2) || bufferindex<3)
				{
					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 minutes
					}				
					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 minutes
					}				
					memset(buffer,' ',5);
					stage++;
				}
			}
			if(commacount==5 && c=='W')
			{
				gps.longitude=-gps.longitude;
			}
			if(commacount==6)
			{
				gps.status=atoi(&c);
			}
			if(commacount==9)
			{
				if(!pointcount)				//wait until we get to a decimal point		
				{
					bufferindex++;
				}
				else
				{
					gps.altitude=(float)atoi(buffer)*0.1;	//last char in buffer will be after dp
					memset(buffer,' ',5);
				}
			}
		  }
		  else if(RMC)
		  {
			if(commacount==7)		//speed in knots
			{		
				if(!pointcount)
				{
					bufferindex++;
				}
				else
				{
					gps.speed=(float)atoi(buffer)*0.1;
					memset(buffer,' ',5);
				}
			}	
			if(commacount==8)				//the heading field
			{
 
				if(!pointcount)
				{
					bufferindex++;
				}
				else
				{
					gps.heading=(float)atoi(buffer)*0.1;
					memset(buffer,' ',5);
					gps.packetflag=TRUE;		//this is usually the last interesting part of the fix info to come through
				}
			}
		  }
		  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.1208910154.txt.gz · Last modified: 2008/07/19 23:31 (external edit)

Donate Powered by PHP Valid HTML5 Valid CSS Driven by DokuWiki