| 
	
	|  |  |  
	
		| View previous topic :: View next topic |  
		| Author | Message |  
		| cvargcal 
 
 
 Joined: 17 Feb 2015
 Posts: 134
 
 
 
			    
 
 | 
			
				| Parsing GPS, example CCS |  
				|  Posted: Sun Apr 07, 2019 9:00 pm |   |  
				| 
 |  
				| Hi, I use this function, (example of CCS): 
  	  | Code: |  	  | #use delay(internal = 64MHz) #use rs232(UART2, baud=9600, errors, receive_buffer=300, stream=STREAM_GPS)
 
 
 char delimitador[]=": ,\r";                 // Token
 char dat_GPRMC[12][10];                     // Array GPRMC
 char string[128];         //! $GPGGA,045104.000,3014.1985,N,09749.2873,W,1,09,1.2,211.6,M,-22.5,M,,0000*62
 //! $GPGGA,045104.000,3014.1985,N,09749.2873,W,1,09,1.2,211.6,M,-22.5,M,,0000*62
 char gprmc[] = "$GPRMC,";
 char gpgga[] = "$GPGGA,";
 char tokens[] = ",";
 char *fix, *num ;
 char *ptr,*lat, *latDir=0, *lon=0, *lonDir=0, *time=0;
 //*validity=0,*speed=0,*course=0,*date=0,*variation=0,*east=0,*checksum;
 int1 GPS_OK=0;
 //******* Conversion Grados a Decimal ************
 //          dd + mm.mmmm/60 for latitude
 //          ddd + mm.mmmm/60 for longitude
 //          0610.9689,N,07535.1974,W
 //          6+(10.9689/60)=6.182815N          Lat
 //          75+(35.1974/60)=75.5866233333W    Long
 char    NumLat[3];            // Array para Latitud   ##'\0'
 char    NumLong[4];           // Array para Longitud ###'\0'
 char *  ptrLat;               // Array para extraer flot Latitud
 char *  ptrLong;              // Array para extraer flot Longitud
 float32   Latitud,Longitud=0.0;     // Latitud yLongitud
 int32 lat_ieee;                   // Values IEEE
 int32 log_ieee;                   // Values IEEE
 
 
 
 
 void GPS_GPRMC(){
 int i;
 fgets(string, STREAM_GPS);
 if (strstr(string, gprmc)){
 fprintf(debug,"\r\nString Capturado : %s",string);
 fprintf(debug,"\r\nParsing ...\r\n",);
 
 ptr = zstring_strtok(string, tokens);             //
 i=0;                                       //
 while(ptr != NULL){                        //
 if(*ptr != '\n'){                       //
 strcpy(&dat_GPRMC[i++][0],ptr);     // Va llenando array
 }
 ptr = zstring_strtok(NULL,tokens);         //
 }
 
 fprintf(debug,"Time Stamp     %s\r\n", &dat_GPRMC[0][0]);    //  1   220516     Time Stamp
 fprintf(debug,"validity       %s\r\n", &dat_GPRMC[1][0]);    //  2   A          validity - A-ok, V-invalid
 fprintf(debug,"Latitude       %s\r\n", &dat_GPRMC[2][0]);    //  3   5133.82    current Latitude
 fprintf(debug,"North/South    %s\r\n", &dat_GPRMC[3][0]);    //  4   N          North/South
 fprintf(debug,"Longitude      %s\r\n", &dat_GPRMC[4][0]);    //  5   00042.24   current Longitude
 fprintf(debug,"East/West      %s\r\n", &dat_GPRMC[5][0]);    //  6   W          East/West
 fprintf(debug,"Speed in knots %s\r\n", &dat_GPRMC[6][0]);    //  7   173.8      Speed in knots
 fprintf(debug,"True course    %s\r\n", &dat_GPRMC[7][0]);    //  8   231.8      True course
 fprintf(debug,"Date Stamp     %s\r\n", &dat_GPRMC[8][0]);    //  9   130694     Date Stamp
 fprintf(debug,"Variation      %s\r\n", &dat_GPRMC[9][0]);    //  10  004.2      Variation
 fprintf(debug,"East/West      %s\r\n", &dat_GPRMC[10][0]);   //  11  W          East/West
 fprintf(debug,"checksum       %s", &dat_GPRMC[11][0]);       //  12  A*70       checksum
 
 if(dat_GPRMC[1][0] == 'A'){                             // Validacion si hay GPS
 GPS_OK=1;
 }
 
 else{
 GPS_OK=0;
 fprintf(debug,"\r\nSin señal GPS\r\n");  //! www.google.com/maps?q=Latitud,Longitud
 }
 
 
 memcpy(NumLat,&dat_GPRMC[2][0],2);                 //  Lat  ##, Extrae Entero
 memcpy(NumLong,&dat_GPRMC[4][0],3);                //  Long ###, Extrae Entero
 NumLat[2]  = '\0';                                 //  Termina Array con un Null
 NumLong[3] = '\0';                                 //  Termina Array con un Null
 ptrLat = &dat_GPRMC[2][0] + 2;                     //  Extrae flotante
 ptrLong = &dat_GPRMC[4][0] + 3;                    //  Extrae flotante
 Latitud = atoi(NumLat) + (atof(ptrLat) / 60);      //  Latitud  = Entero + (flotante/60)
 Longitud = atoi(NumLong) + (atof(ptrLong) / 60);   //  Longitud = Entero + (flotante/60)
 
 if(dat_GPRMC[3][0] == 'S') Latitud *= -1;          //  Si es S Latitud es Negativa
 if(dat_GPRMC[5][0] == 'W') Longitud *= -1;         //  Si es W Longitud es Negativa
 
 lat_ieee = f_PICtoIEEE( Latitud);
 log_ieee = f_PICtoIEEE( Longitud);
 
 fprintf(debug,"www.google.com/maps?q=%01.4f,%01.4f\r\n\r\n",Latitud,Longitud);  //! www.google.com/maps?q=Latitud,Longitud
 
 
 
 }
 
 fprintf(debug,"NO capture  UART2\r\n\r\n");  //! www.google.com/maps?q=Latitud,Longitud
 }
 
 | 
 The function work fine while no receive gps, but when get GPS, let to into to this line
 
  	  | Code: |  	  | if (strstr(string, gprmc)){ | 
 and go to "no Capture", maybe are many data from GPS module?
 
 output:
 
 Here show that while is "V" (no gps) the function work.
 
  	  | Code: |  	  | String Capturado :
 $GPRMC,235952.800,V,,,,,0.00,0.00,050180,,,N*43
 Parsing ...
 Time Stamp     235952.800V
 validity       V
 Latitude       ,
 North/South    ,
 Longitude      ,
 East/West      ,
 Speed in knots 0.00
 True course    0.00
 Date Stamp     050180
 Variation      ,
 East/West      ,
 checksum       N*43
 Sin señal GPS
 | 
 www.google.com/maps?q=0.0000,0.0000
 
 what is the problem?
 |  |  
		|  |  
		| temtronic 
 
 
 Joined: 01 Jul 2010
 Posts: 9587
 Location: Greensville,Ontario
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Apr 08, 2019 4:54 am |   |  
				| 
 |  
				| If you're saying it doesn't work with a real GPS connected to the PIC, it could be hardware. What PIC, What GPS, What is the VDD for both? The program you post doesn't have a PIC type as the first line,though at 64MHz ,has to be relatively new. Most GPS 'devices' have 3 volt VDD, GPS ' modules' might be 3,5 or both.
 There is a HUGE problem if the PIC is running at 5 volts and the GPS unit at 3.
 We need more information about the actual hardware.
 
 Jay
 |  |  
		|  |  
		| ezflyr 
 
 
 Joined: 25 Oct 2010
 Posts: 1019
 Location: Tewksbury, MA
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Apr 08, 2019 6:07 am |   |  
				| 
 |  
				| Hi, 
 This is a major contributor to your problem:
 
 
  	  | Quote: |  	  | fgets(string, STREAM_GPS); | 
 
 fgets is never going to work in this application. You need to use the serial interrupt, and a circular receive buffer to reliably receive GPS data. Search the forum, this has been covered many, many times!
 
 John
 _________________
 John
 
 If it's worth doing, it's worth doing in real hardware!
 |  |  
		|  |  
		| dluu13 
 
 
 Joined: 28 Sep 2018
 Posts: 395
 Location: Toronto, ON
 
 
			      
 
 | 
			
				|  |  
				|  Posted: Mon Apr 08, 2019 7:39 am |   |  
				| 
 |  
				| receive_buffer option in the #use rs232 has already set up that circular buffer for you. 
 So what you need to do is use kbhit() to tell you if there are any contents in the circular buffer.
 
 But since you are still debugging, why not just print string before your strstr so you know what you are receiving? That may help your debugging process.
 |  |  
		|  |  
		| cvargcal 
 
 
 Joined: 17 Feb 2015
 Posts: 134
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Apr 08, 2019 9:24 am |   |  
				| 
 |  
				|  	  | dluu13 wrote: |  	  | receive_buffer option in the #use rs232 has already set up that circular buffer for you. 
 So what you need to do is use kbhit() to tell you if there are any contents in the circular buffer.
 
 But since you are still debugging, why not just print string before your strstr so you know what you are receiving? That may help your debugging process.
 | 
 
 GPS and PIC are to 3.3V
 
 
 Yes I know its better  interruption,  but I wanted use this example of ccs.
 
 #include <18F26k22.h>
 
 The code work fine, while the gps is "V" If is "A" not capture again.... maybe is many data?
 
 
 
 
 
 Full code
 
  	  | Code: |  	  | #include <18F26k22.h> #device PASS_STRINGS = IN_RAM
 #device adc=10
 #FUSES INTRC_IO        // INTRC_IO Internal RC Osc, no CLKOUT
 #FUSES NOFCMEN         //Fail-safe clock monitor disabled
 #FUSES NOIESO          //Internal External Switch Over mode disabled
 #FUSES NOPUT           //No Power Up Timer
 #FUSES NOBROWNOUT      //No brownout reset
 #FUSES NOWDT           //No Watch Dog Timer
 #FUSES NOPBADEN        //PORTB pins are configured as digital I/O on RESET
 #FUSES MCLR            //Master Clear pin NOT enabled
 #FUSES NOLVP           //No low voltage prgming, B3(PIC16) or B5(PIC18) used for I/O
 #FUSES NOXINST         //Extended set extension and Indexed Addressing mode disabled (Legacy mode)
 #FUSES NODEBUG         //No Debug mode for ICD
 #FUSES CCP2C1
 #fuses PROTECT        //Code protected from reads
 
 #use delay(internal =16MHz)
 //#use rs232(UART2, baud=4800, errors, receive_buffer=128, stream=STREAM_GPS)
 #use rs232(baud=4800, xmit=PIN_b6,rcv=PIN_b7,bits=8,parity=N,ERRORS,stream=STREAM_GPS)    // UART 2 GPS
 #use rs232(baud=57600,xmit=PIN_c6,rcv=PIN_c7,bits=8,parity=N,ERRORS,stream=STREAM_LoRa)   // UART 1 LORA
 #use rs232(baud=9600, xmit=PIN_c4,rcv=PIN_c3,bits=8,parity=N,ERRORS,stream=debug)         // UART 2 DEBUG
 
 
 // ~~~~~~~~~~~~~~~~~~~~~~  Debug ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 #define _debug
 #define show_config_RN2903
 // ~~~~~~~~~~~~~~~~~~~~~~  Debug ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 
 
 #use fast_io(A)
 #use standard_io(B)
 #use standard_io(c)
 
 
 #include <string.h>
 #include <stdlib.h>
 #include <stdio.h>
 #include <stdint.h>
 #include <stdbool.h>
 #include <ieeefloat.c>
 #include <NMEAParser.c>        // NMEA
 
 void main(){
 enable_interrupts(GLOBAL);
 enable_interrupts(int_rda);                    // Habilita Interrupción RDA UART1
 enable_interrupts(int_rda2);                   // Habilita Interrupción RDA UART2
 delay_ms(100);
 while(1){
 GPS_GPRMC_V2();
 //GPS_GPRMC();
 
 }
 }
 | 
 
 
 Lib
 
 
 
  	  | Code: |  	  | // :::::::::::::::: Varibles GPS_L86 :::::::::::::::::::::::::::::::::::::::::: char delimitador[]=": ,\r";                 // Token
 char dat_GPRMC[12][10];                     // Array GPRMC
 char string[128];         //! $GPGGA,045104.000,3014.1985,N,09749.2873,W,1,09,1.2,211.6,M,-22.5,M,,0000*62
 //! $GPGGA,045104.000,3014.1985,N,09749.2873,W,1,09,1.2,211.6,M,-22.5,M,,0000*62
 char gprmc[] = "$GPRMC,";
 char gpgga[] = "$GPGGA,";
 char tokens[] = ",";
 char *fix, *num ;
 char *ptr,*lat, *latDir=0, *lon=0, *lonDir=0, *time=0;
 //*validity=0,*speed=0,*course=0,*date=0,*variation=0,*east=0,*checksum;
 int1 GPS_OK=0;
 //******* Conversion Grados a Decimal ************
 //          dd + mm.mmmm/60 for latitude
 //          ddd + mm.mmmm/60 for longitude
 //          0610.9689,N,07535.1974,W
 //          6+(10.9689/60)=6.182815N          Lat
 //          75+(35.1974/60)=75.5866233333W    Long
 char    NumLat[3];            // Array para Latitud   ##'\0'
 char    NumLong[4];           // Array para Longitud ###'\0'
 char *  ptrLat;               // Array para extraer flot Latitud
 char *  ptrLong;              // Array para extraer flot Longitud
 float32   Latitud,Longitud=0.0;     // Latitud yLongitud
 int32 lat_ieee;                   // Values IEEE
 int32 log_ieee;                   // Values IEEE
 
 
 
 
 
 #DEFINE SIZE_BUFFER_UART2   200      // Serial Buffer Size
 #DEFINE SIZE_BUFFER_GPRMC   100      // Serial Buffer Size
 
 enum letras{
 PRIMER_LETRA,    // $
 LETRA_G,         // G
 LETRA_P,         // P
 LETRA_R,         // R
 LETRA_M,         // M
 LETRA_C,         // C
 RESTANTE         // .....
 };
 enum letras estado;
 
 int  counter_read2 = 0x00;                                  // Serial Buffer Counter
 char c=0x00;
 char Receive_String_uart2[SIZE_BUFFER_UART2];               // Serial Buffer
 //!
 
 
 
 char *zstring_strtok(char *str,  char *delim);
 /***************************************************************/
 //  FUNCIONES
 /***************************************************************/
 void GPS_GPRMC();
 void GPS_GPRMC_V2();
 void GPS_GPGGA();
 
 //!****************************************************************************
 //!******************** GPS  With UART2 ***************************************
 //!****************************************************************************
 #INT_RDA2
 void RDA_isr2() {
 switch(estado){
 case PRIMER_LETRA:  if(getchar(STREAM_GPS)=='$') estado = LETRA_G; break;
 case LETRA_G:       if(getchar(STREAM_GPS)=='G') estado = LETRA_P; break;
 case LETRA_P:       if(getchar(STREAM_GPS)=='P') estado = LETRA_R; break;
 case LETRA_R:       if(getchar(STREAM_GPS)=='R') estado = LETRA_M; break;
 case LETRA_M:       if(getchar(STREAM_GPS)=='M') estado = LETRA_C; break;
 case LETRA_C:       if(getchar(STREAM_GPS)=='C') estado = RESTANTE; counter_read2 = 0x00; break;
 case RESTANTE:
 c = getchar(STREAM_GPS);
 if(c == '$'){                        // Proximo "$"
 //disable_interrupts(INT_RDA2);
 strcpy(string,Receive_String_uart2);
 estado = LETRA_G;
 break;
 }
 
 if(counter_read2==SIZE_BUFFER_UART2) counter_read2=0x00;
 Receive_String_uart2[counter_read2++]=c;
 break;
 default:
 estado = PRIMER_LETRA; break;
 }
 }
 // :::::::::::::::: GPRM ::::::::::::::::::::::::::::::::::::::::::::::::::::::
 void GPS_GPRMC(){
 int i;
 fgets(string, STREAM_GPS);
 if (strstr(string, gprmc)){
 fprintf(debug,"\r\nString Capturado : %s",string);
 fprintf(debug,"\r\nParsing ...\r\n",);
 
 ptr = zstring_strtok(string, tokens);             //
 i=0;                                       //
 while(ptr != NULL){                        //
 if(*ptr != '\n'){                       //
 strcpy(&dat_GPRMC[i++][0],ptr);     // Va llenando array
 }
 ptr = zstring_strtok(NULL,tokens);         //
 }
 
 fprintf(debug,"Time Stamp     %s\r\n", &dat_GPRMC[0][0]);    //  1   220516     Time Stamp
 fprintf(debug,"validity       %s\r\n", &dat_GPRMC[1][0]);    //  2   A          validity - A-ok, V-invalid
 fprintf(debug,"Latitude       %s\r\n", &dat_GPRMC[2][0]);    //  3   5133.82    current Latitude
 fprintf(debug,"North/South    %s\r\n", &dat_GPRMC[3][0]);    //  4   N          North/South
 fprintf(debug,"Longitude      %s\r\n", &dat_GPRMC[4][0]);    //  5   00042.24   current Longitude
 fprintf(debug,"East/West      %s\r\n", &dat_GPRMC[5][0]);    //  6   W          East/West
 fprintf(debug,"Speed in knots %s\r\n", &dat_GPRMC[6][0]);    //  7   173.8      Speed in knots
 fprintf(debug,"True course    %s\r\n", &dat_GPRMC[7][0]);    //  8   231.8      True course
 fprintf(debug,"Date Stamp     %s\r\n", &dat_GPRMC[8][0]);    //  9   130694     Date Stamp
 fprintf(debug,"Variation      %s\r\n", &dat_GPRMC[9][0]);    //  10  004.2      Variation
 fprintf(debug,"East/West      %s\r\n", &dat_GPRMC[10][0]);   //  11  W          East/West
 fprintf(debug,"checksum       %s", &dat_GPRMC[11][0]);       //  12  A*70       checksum
 
 if(dat_GPRMC[1][0] == 'A'){                             // Validacion si hay GPS
 GPS_OK=1;
 }
 
 else{
 GPS_OK=0;
 fprintf(debug,"\r\nSin señal GPS\r\n");  //! www.google.com/maps?q=Latitud,Longitud
 }
 
 
 memcpy(NumLat,&dat_GPRMC[2][0],2);                 //  Lat  ##, Extrae Entero
 memcpy(NumLong,&dat_GPRMC[4][0],3);                //  Long ###, Extrae Entero
 NumLat[2]  = '\0';                                 //  Termina Array con un Null
 NumLong[3] = '\0';                                 //  Termina Array con un Null
 ptrLat = &dat_GPRMC[2][0] + 2;                     //  Extrae flotante
 ptrLong = &dat_GPRMC[4][0] + 3;                    //  Extrae flotante
 Latitud = atoi(NumLat) + (atof(ptrLat) / 60);      //  Latitud  = Entero + (flotante/60)
 Longitud = atoi(NumLong) + (atof(ptrLong) / 60);   //  Longitud = Entero + (flotante/60)
 
 if(dat_GPRMC[3][0] == 'S') Latitud *= -1;          //  Si es S Latitud es Negativa
 if(dat_GPRMC[5][0] == 'W') Longitud *= -1;         //  Si es W Longitud es Negativa
 
 lat_ieee = f_PICtoIEEE( Latitud);
 log_ieee = f_PICtoIEEE( Longitud);
 
 fprintf(debug,"www.google.com/maps?q=%01.4f,%01.4f\r\n\r\n",Latitud,Longitud);  //! www.google.com/maps?q=Latitud,Longitud
 
 
 
 }
 
 fprintf(debug,"NO se puede leer UART2\r\n\r\n");  //! www.google.com/maps?q=Latitud,Longitud
 }
 
 void GPS_GPGGA(){
 int i;
 fgets(string, STREAM_GPS);
 if (strstr(string, gpgga) != 0){
 ptr = strtok(string, tokens);
 i = 0;
 while(ptr != NULL){
 switch(i++){
 case 1:  time = ptr;      break;
 case 2:  lat = ptr;       break;
 case 3:  latDir =ptr;     break;
 case 4:  lon = ptr;       break;
 case 5:  lonDir = ptr;    break;
 case 6:  fix = ptr;       break;
 case 7:  num = ptr;       break;
 default: break;
 }
 ptr = strtok(NULL, tokens);
 }
 
 fprintf(debug, "Trama capturada >>  fix=%s, num=%s, time=%s, lat=%s %s, lon=%s %s\r\n",
 fix,
 num,
 time,
 lat,
 latDir,
 lon,
 lonDir
 );
 }
 }
 
 void GPS_GPRMC_V2(){
 int i;
 //******* Extraer GPRM **************************
 //strcpy(string,GPRMC_buff);               // +CBC: #,###,####
 ptr = strtok(string, delimitador);         //
 i=0;                                       //
 while(ptr != NULL){                        //
 if(*ptr != '\n'){                     //
 strcpy(&dat_GPRMC[i][0],ptr);       // Va llenando array
 i++;                                //
 }                                   //
 ptr = strtok(NULL,delimitador);     //
 }
 
 
 
 
 if(dat_GPRMC[1][0] == 'A'){               // Validacion si hay GPS
 GPS_OK=1;
 memcpy(NumLat,&dat_GPRMC[2][0],2);                 //  Lat  ##, Extrae Entero
 memcpy(NumLong,&dat_GPRMC[4][0],3);                //  Long ###, Extrae Entero
 NumLat[2]  = '\0';                                 //  Termina Array con un Null
 NumLong[3] = '\0';                                 //  Termina Array con un Null
 ptrLat = &dat_GPRMC[2][0] + 2;                     //  Extrae flotante
 ptrLong = &dat_GPRMC[4][0] + 3;                    //  Extrae flotante
 Latitud = atoi(NumLat) + (atof(ptrLat) / 60);      //  Latitud  = Entero + (flotante/60)
 Longitud = atoi(NumLong) + (atof(ptrLong) / 60);   //  Longitud = Entero + (flotante/60)
 
 if(dat_GPRMC[3][0] == 'S') Latitud *= -1;          //  Si es S Latitud es Negativa
 if(dat_GPRMC[5][0] == 'W') Longitud *= -1;         //  Si es W Longitud es Negativa
 
 lat_ieee = f_PICtoIEEE( Latitud);
 log_ieee = f_PICtoIEEE( Longitud);
 
 fprintf(debug,"www.google.com/maps?q=%01.4f,%01.4f\r\n",Latitud,Longitud);  //! www.google.com/maps?q=Latitud,Longitud
 }
 else{
 GPS_OK=0;
 // Latitud=0.0;         Longitud=0.0;
 // fprintf(debug,"No hay señal GPS\r\n");
 
 }
 memset(string, NULL, sizeof(string));
 //  memset(string,NULL,SIZE_BUFFER_GPRMC);                 // Set all elements to NULL
 // memset(Receive_String_uart2,NULL,SIZE_BUFFER_UART2);        // Set all elements to NULL
 //counter_read2 = 0x00;                                       // Reinicia Contador
 // enable_interrupts(INT_RDA2);                                // Habilita recepcion GPS
 
 //!    fprintf(debug,"%s\r\n", &dat_GPRMC[0][0]);    //  1   220516     Time Stamp
 //!    fprintf(debug,"%s\r\n", &dat_GPRMC[1][0]);    //  2   A          validity - A-ok, V-invalid
 //!    fprintf(debug,"%s\r\n", &dat_GPRMC[2][0]);    //  3   5133.82    current Latitude
 //!    fprintf(debug,"%s\r\n", &dat_GPRMC[3][0]);    //  4   N          North/South
 //!    fprintf(debug,"%s\r\n", &dat_GPRMC[4][0]);    //  5   00042.24   current Longitude
 //!    fprintf(debug,"%s\r\n", &dat_GPRMC[5][0]);    //  6   W          East/West
 //!    fprintf(debug,"%s\r\n", &dat_GPRMC[6][0]);    //  7   173.8      Speed in knots
 //!    fprintf(debug,"%s\r\n", &dat_GPRMC[7][0]);    //  8   231.8      True course
 //!    fprintf(debug,"%s\r\n", &dat_GPRMC[8][0]);    //  9   130694     Date Stamp
 //!    fprintf(debug,"%s\r\n", &dat_GPRMC[9][0]);    //  10  004.2      Variation
 //!    fprintf(debug,"%s\r\n", &dat_GPRMC[10][0]);   //  11  W          East/West
 //!    fprintf(debug,"%s\r\n", &dat_GPRMC[11][0]);   //  12  *70        checksum
 
 }
 
 
 
 
 char *zstring_strtok(char *str,  char *delim) {
 static char *static_str=0;       /* var to store last address */
 int index=0, strlength=0;         /* integers for indexes */
 int found = 0;                 /* check if delim is found */
 
 /* delimiter cannot be NULL
 * if no more char left, return NULL as well
 */
 if (delim==0 || (str == 0 && static_str == 0))
 return 0;
 
 if (str == 0)
 str = static_str;
 
 /* get length of string */
 while(str[strlength])
 strlength++;
 
 /* find the first occurrence of delim */
 for (index=0;index<strlength;index++)
 if (str[index]==delim[0]) {
 found=1;
 break;
 }
 
 /* if delim is not contained in str, return str */
 if (!found) {
 static_str = 0;
 return str;
 }
 
 /* check for consecutive delimiters
 *if first char is delim, return delim
 */
 if (str[0]==delim[0]) {
 static_str = (str + 1);
 return (char *)delim;
 }
 
 /* terminate the string
 * this assignment requires char[], so str has to
 * be char[] rather than *char
 */
 str[index] = '\0';
 
 /* save the rest of the string */
 if ((str + index + 1)!=0)
 static_str = (str + index + 1);
 else
 static_str = 0;
 
 return str;
 }
 | 
 |  |  
		|  |  
		| ezflyr 
 
 
 Joined: 25 Oct 2010
 Posts: 1019
 Location: Tewksbury, MA
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Apr 08, 2019 8:10 pm |   |  
				| 
 |  
				| Hi, 
 I haven’t studied your code in detail, but you definitely don’t want to enable interrupts when you don’t have a corresponding handler, such as you are doing for int_rda.....
 
 John
 _________________
 John
 
 If it's worth doing, it's worth doing in real hardware!
 |  |  
		|  |  
		| dluu13 
 
 
 Joined: 28 Sep 2018
 Posts: 395
 Location: Toronto, ON
 
 
			      
 
 | 
			
				|  |  
				|  Posted: Mon Apr 08, 2019 8:16 pm |   |  
				| 
 |  
				|  	  | ezflyr wrote: |  	  | Hi, 
 I haven’t studied your code in detail, but you definitely don’t want to enable interrupts when you don’t have a corresponding handler, such as you are doing for int_rda.....
 
 John
 | 
 
 As I understand, since op has specified receive buffer in use rs232, the compiler automatically generates an isr and enables the interrupt. So if anything, his own enable statement is redundant.
 
 But, you are right that in other circumstances, enabling the interrupt without providing an isr will cause problems like random restarts.
 |  |  
		|  |  
		| ezflyr 
 
 
 Joined: 25 Oct 2010
 Posts: 1019
 Location: Tewksbury, MA
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Apr 08, 2019 8:20 pm |   |  
				| 
 |  
				| Hi, 
 The OP has commented out the Use rs232  statement that defines the built in buffer. What he’s got now is the ‘classic’ method with an extra, extraneous interrupt enabled!
 _________________
 John
 
 If it's worth doing, it's worth doing in real hardware!
 |  |  
		|  |  
		| dluu13 
 
 
 Joined: 28 Sep 2018
 Posts: 395
 Location: Toronto, ON
 
 
			      
 
 | 
			
				|  |  
				|  Posted: Mon Apr 08, 2019 8:28 pm |   |  
				| 
 |  
				| Ah... I missed that. Now I see that op is using software uarts, which is more or less pretty terrible when it comes to receiving data, since you need to poll at ten times the baud rate to avoid missing incoming bits. 
 Another thing I noticed...
 In the original code, you have the gps stream at 9600. In the next version you have it at 4800. Have you configured your gps with a slower baud rate now?
 |  |  
		|  |  
		| cvargcal 
 
 
 Joined: 17 Feb 2015
 Posts: 134
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Tue Apr 09, 2019 8:15 am |   |  
				| 
 |  
				|  	  | dluu13 wrote: |  	  | Ah... I missed that. Now I see that op is using software uarts, which is more or less pretty terrible when it comes to receiving data, since you need to poll at ten times the baud rate to avoid missing incoming bits. 
 Another thing I noticed...
 In the original code, you have the gps stream at 9600. In the next version you have it at 4800. Have you configured your gps with a slower baud rate now?
 | 
 Yes,  the 9600 is not problem. but I don't know is the baud has be set in 4800.
 The problem is that I used the modulo quectel l80 and the baud  come to 9600
 
 
 The code work very good, in Proteus work very fine and in real work 50%, when the gps get data no capture the string with head "GPRMC", I think is by many data and can not process the -"string search" faster.
 
 
 I can do with interruption, but I liked many this example of CCS, I find easy for parsing.
 
 when I want use the way with interruption I enable the interruption, else I comment that lines. not to same time.
 |  |  
		|  |  
		| cvargcal 
 
 
 Joined: 17 Feb 2015
 Posts: 134
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Tue Apr 09, 2019 1:17 pm |   |  
				| 
 |  
				|  	  | cvargcal wrote: |  	  |  	  | dluu13 wrote: |  	  | Ah... I missed that. Now I see that op is using software uarts, which is more or less pretty terrible when it comes to receiving data, since you need to poll at ten times the baud rate to avoid missing incoming bits. 
 Another thing I noticed...
 In the original code, you have the gps stream at 9600. In the next version you have it at 4800. Have you configured your gps with a slower baud rate now?
 | 
 Yes,  the 9600 is not problem. but I don't know is the baud has be set in 4800.
 The problem is that I used the modulo quectel l80 and the baud  come to 9600
 
 
 The code work very good, in Proteus work very fine and in real work 50%, when the gps get data no capture the string with head "GPRMC", I think is by many data and can not process the -"string search" faster.
 
 
 I can do with interruption, but I liked many this example of CCS, I find easy for parsing.
 
 when I want use the way with interruption I enable the interruption, else I comment that lines. not to same time.
 | 
 
 
 
 
 
 Thank you .
 
 
 I fixed one error, Now my way to get GPRMC work find similart to CCS example,  I dont know if example CCS work fine in real or if the GPS need set to 4800 for can get data slow.
 
 
 
  	  | Code: |  	  | // CCS Example //#use rs232(UART2, baud=4800, errors, receive_buffer=128, stream=STREAM_GPS)
 
 
 // myself
 #use rs232(baud=9600, xmit=PIN_b6,rcv=PIN_b7,bits=8,parity=N,ERRORS,stream=STREAM_GPS)    // UART 2 GPS
 
 | 
 
 
 
 
 With interruption control myself
 
 
  	  | Code: |  	  | 
 // ****** With #INT_RDA2   ****
 #DEFINE SIZE_BUFFER_UART2   200      // Serial Buffer Size
 int  counter_read2 = 0x00;                                  // Serial Buffer Counter
 char Receive_String_uart2[SIZE_BUFFER_UART2];               // Serial Buffer
 int gprmc_messageReady=0;
 enum letras{
 PRIMER_LETRA,    // $
 LETRA_G,         // G
 LETRA_P,         // P
 LETRA_R,         // R
 LETRA_M,         // M
 LETRA_C,         // C
 RESTANTE         // .....
 };
 enum letras estado;
 // ****************************
 
 
 
 
 //!****************************************************************************
 //!******************** GPS  With UART2 ***************************************
 //!****************************************************************************
 #INT_RDA2
 void RDA_isr2() {
 char c=0x00;
 switch(estado){
 case PRIMER_LETRA:  if(getchar(STREAM_GPS)=='$') estado = LETRA_G; break;
 case LETRA_G:       if(getchar(STREAM_GPS)=='G') estado = LETRA_P; break;
 case LETRA_P:       if(getchar(STREAM_GPS)=='P') estado = LETRA_R; break;
 case LETRA_R:       if(getchar(STREAM_GPS)=='R') estado = LETRA_M; break;
 case LETRA_M:       if(getchar(STREAM_GPS)=='M') estado = LETRA_C; break;
 case LETRA_C:       if(getchar(STREAM_GPS)=='C') estado = RESTANTE;
 counter_read2 = 0x00; break;
 case RESTANTE:
 c = getchar(STREAM_GPS);
 if(c == '$'){  // Proximo "$"
 strcpy(string,Receive_String_uart2);
 estado = LETRA_G;
 gprmc_messageReady=1;
 break;
 }
 
 if(counter_read2==SIZE_BUFFER_UART2) counter_read2=0x00;
 Receive_String_uart2[counter_read2++]=c;
 break;
 default:   estado = PRIMER_LETRA; break;
 }
 }
 | 
 
 
 
  	  | Code: |  	  | // :::::::::::::::: GPRM ::::::::::  BY CONTROL #INT_RDA2 ::::::::::::::::::::: void GPS_GPRMC_v2(){
 int i;
 if ((gprmc_messageReady) ){
 gprmc_messageReady=0;
 //0,1 ........................,12
 // ,183504.373,V,,,,,,,090419,,,N*44
 // [0]= GPRMC......... [12]= CHKS
 fprintf(debug,"\r\nString Capturado : %s",string);
 fprintf(debug,"\r\nParsing ...\r\n",);
 
 ptr = zstring_strtok(string, tokens);             //
 i=0;                                       //
 while(ptr != NULL){                        //
 if(*ptr != '\n'){                       //
 strcpy(&dat_GPRMC[i++][0],ptr);     // Va llenando array
 }
 ptr = zstring_strtok(NULL,tokens);         //
 }
 
 fprintf(debug,"Head (GPRMC)   %s\r\n", &dat_GPRMC[0][0]);    //  0   Head
 fprintf(debug,"Time Stamp     %s\r\n", &dat_GPRMC[1][0]);    //  1   220516     Time Stamp
 fprintf(debug,"validity       %s\r\n", &dat_GPRMC[2][0]);    //  2   A          validity - A-ok, V-invalid
 fprintf(debug,"Latitude       %s\r\n", &dat_GPRMC[3][0]);    //  3   5133.82    current Latitude
 fprintf(debug,"North/South    %s\r\n", &dat_GPRMC[4][0]);    //  4   N          North/South
 fprintf(debug,"Longitude      %s\r\n", &dat_GPRMC[5][0]);    //  5   00042.24   current Longitude
 fprintf(debug,"East/West      %s\r\n", &dat_GPRMC[6][0]);    //  6   W          East/West
 fprintf(debug,"Speed in knots %s\r\n", &dat_GPRMC[7][0]);    //  7   173.8      Speed in knots
 fprintf(debug,"True course    %s\r\n", &dat_GPRMC[8][0]);    //  8   231.8      True course
 fprintf(debug,"Date Stamp     %s\r\n", &dat_GPRMC[9][0]);    //  9   130694     Date Stamp
 fprintf(debug,"Variation      %s\r\n", &dat_GPRMC[10][0]);    //  10  004.2      Variation
 fprintf(debug,"East/West      %s\r\n", &dat_GPRMC[11][0]);   //  11  W          East/West
 fprintf(debug,"checksum       %s", &dat_GPRMC[12][0]);       //  12  A*70       checksum
 
 if(dat_GPRMC[1][0] == 'A'){                                   // Validacion si hay GPS
 GPS_OK=1;
 memcpy(NumLat,&dat_GPRMC[2][0],2);                 //  Lat  ##, Extrae Entero
 memcpy(NumLong,&dat_GPRMC[4][0],3);                //  Long ###, Extrae Entero
 NumLat[2]  = '\0';                                 //  Termina Array con un Null
 NumLong[3] = '\0';                                 //  Termina Array con un Null
 ptrLat = &dat_GPRMC[2][0] + 2;                     //  Extrae flotante
 ptrLong = &dat_GPRMC[4][0] + 3;                    //  Extrae flotante
 Latitud = atoi(NumLat) + (atof(ptrLat) / 60);      //  Latitud  = Entero + (flotante/60)
 Longitud = atoi(NumLong) + (atof(ptrLong) / 60);   //  Longitud = Entero + (flotante/60)
 
 if(dat_GPRMC[3][0] == 'S') Latitud *= -1;          //  Si es S Latitud es Negativa
 if(dat_GPRMC[5][0] == 'W') Longitud *= -1;         //  Si es W Longitud es Negativa
 
 lat_ieee = f_PICtoIEEE( Latitud);
 log_ieee = f_PICtoIEEE( Longitud);
 
 fprintf(debug,"www.google.com/maps?q=%01.4f,%01.4f\r\n\r\n",Latitud,Longitud);  //! www.google.com/maps?q=Latitud,Longitud
 memset(string, 0, sizeof(string));
 memset(Receive_String_uart2, 0, sizeof(Receive_String_uart2));
 counter_read2 = 0;
 }
 }
 
 // :::::::::::::::: GPRM ::::::::::  Similar to CCS Example :::::::::::::::::::
 void GPS_GPRMC(){
 int i;
 fgets(string, STREAM_GPS);         // To see the UART2 Configuration... for work
 if (strstr(string, gprmc)){
 fprintf(debug,"\r\nString Capturado : %s",string);
 fprintf(debug,"\r\nParsing ...\r\n",);
 ptr = zstring_strtok(string, tokens);      //
 i=0;                                       //
 while(ptr != NULL){                        //
 if(*ptr != '\n'){                       //
 strcpy(&dat_GPRMC[i++][0],ptr);     // Va llenando array
 }
 ptr = zstring_strtok(NULL,tokens);         //
 }
 
 fprintf(debug,"Time Stamp     %s\r\n", &dat_GPRMC[0][0]);    //  1   220516     Time Stamp
 fprintf(debug,"validity       %s\r\n", &dat_GPRMC[1][0]);    //  2   A          validity - A-ok, V-invalid
 fprintf(debug,"Latitude       %s\r\n", &dat_GPRMC[2][0]);    //  3   5133.82    current Latitude
 fprintf(debug,"North/South    %s\r\n", &dat_GPRMC[3][0]);    //  4   N          North/South
 fprintf(debug,"Longitude      %s\r\n", &dat_GPRMC[4][0]);    //  5   00042.24   current Longitude
 fprintf(debug,"East/West      %s\r\n", &dat_GPRMC[5][0]);    //  6   W          East/West
 fprintf(debug,"Speed in knots %s\r\n", &dat_GPRMC[6][0]);    //  7   173.8      Speed in knots
 fprintf(debug,"True course    %s\r\n", &dat_GPRMC[7][0]);    //  8   231.8      True course
 fprintf(debug,"Date Stamp     %s\r\n", &dat_GPRMC[8][0]);    //  9   130694     Date Stamp
 fprintf(debug,"Variation      %s\r\n", &dat_GPRMC[9][0]);    //  10  004.2      Variation
 fprintf(debug,"East/West      %s\r\n", &dat_GPRMC[10][0]);   //  11  W          East/West
 fprintf(debug,"checksum       %s", &dat_GPRMC[11][0]);       //  12  A*70       checksum
 
 if(dat_GPRMC[1][0] == 'A'){                                   // Validacion si hay GPS
 GPS_OK=1;
 
 memcpy(NumLat,&dat_GPRMC[2][0],2);                 //  Lat  ##, Extrae Entero
 memcpy(NumLong,&dat_GPRMC[4][0],3);                //  Long ###, Extrae Entero
 NumLat[2]  = '\0';                                 //  Termina Array con un Null
 NumLong[3] = '\0';                                 //  Termina Array con un Null
 ptrLat = &dat_GPRMC[2][0] + 2;                     //  Extrae flotante
 ptrLong = &dat_GPRMC[4][0] + 3;                    //  Extrae flotante
 Latitud = atoi(NumLat) + (atof(ptrLat) / 60);      //  Latitud  = Entero + (flotante/60)
 Longitud = atoi(NumLong) + (atof(ptrLong) / 60);   //  Longitud = Entero + (flotante/60)
 
 if(dat_GPRMC[3][0] == 'S') Latitud *= -1;          //  Si es S Latitud es Negativa
 if(dat_GPRMC[5][0] == 'W') Longitud *= -1;         //  Si es W Longitud es Negativa
 
 lat_ieee = f_PICtoIEEE( Latitud);
 log_ieee = f_PICtoIEEE( Longitud);
 
 fprintf(debug,"www.google.com/maps?q=%01.4f,%01.4f\r\n\r\n",Latitud,Longitud);  //! www.google.com/maps?q=Latitud,Longitud
 }
 else {
 GPS_OK=0;
 fprintf(debug,"\r\nSin señal GPS\r\n");  //! www.google.com/maps?q=Latitud,Longitud
 }
 }
 }
 // :::::::::::::::: GPGGA :::::::::: Similar to CCS Example :::::::::::::::::::
 void GPS_GPGGA(){
 }
 
 
 
 }
 | 
 
 
  	  | Code: |  	  | // :::::::::::::::: strtok with empty field :::::::::: char *zstring_strtok(char *str,  char *delim) {
 static char *static_str=0;       /* var to store last address */
 int index=0, strlength=0;         /* integers for indexes */
 int found = 0;                 /* check if delim is found */
 
 /* delimiter cannot be NULL
 * if no more char left, return NULL as well
 */
 if (delim==0 || (str == 0 && static_str == 0))
 return 0;
 
 if (str == 0)
 str = static_str;
 
 /* get length of string */
 while(str[strlength])
 strlength++;
 
 /* find the first occurrence of delim */
 for (index=0;index<strlength;index++)
 if (str[index]==delim[0]) {
 found=1;
 break;
 }
 
 /* if delim is not contained in str, return str */
 if (!found) {
 static_str = 0;
 return str;
 }
 
 /* check for consecutive delimiters
 *if first char is delim, return delim
 */
 if (str[0]==delim[0]) {
 static_str = (str + 1);
 return (char *)delim;
 }
 
 /* terminate the string
 * this assignment requires char[], so str has to
 * be char[] rather than *char
 */
 str[index] = '\0';
 
 /* save the rest of the string */
 if ((str + index + 1)!=0)
 static_str = (str + index + 1);
 else
 static_str = 0;
 
 return str;
 }
 
 | 
 |  |  
		|  |  
		| dluu13 
 
 
 Joined: 28 Sep 2018
 Posts: 395
 Location: Toronto, ON
 
 
			      
 
 | 
			
				|  |  
				|  Posted: Tue Apr 09, 2019 1:45 pm |   |  
				| 
 |  
				| The GPS will send at whatever rate it's going to send at, which is whatever is in its datasheet. You can also measure it with an oscilloscope or logic analyzer. 
 If your PIC is receiving at the wrong rate, then you will never receive the right messages.
 
 I don't know if you started out with this, but I would plug the GPS into a breadboard with RS232 to USB module and try to interact with it directly from a serial terminal on your PC before going to a PIC. And even then, I like to have a logic analyzer hooked up to see what the incoming messages are to confirm that my PIC is receiving the same information as the analyzer is.
 |  |  
		|  |  
		| cvargcal 
 
 
 Joined: 17 Feb 2015
 Posts: 134
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Tue Apr 09, 2019 1:56 pm |   |  
				| 
 |  
				|  	  | dluu13 wrote: |  	  | The GPS will send at whatever rate it's going to send at, which is whatever is in its datasheet. You can also measure it with an oscilloscope or logic analyzer. 
 If your PIC is receiving at the wrong rate, then you will never receive the right messages.
 
 I don't know if you started out with this, but I would plug the GPS into a breadboard with RS232 to USB module and try to interact with it directly from a serial terminal on your PC before going to a PIC. And even then, I like to have a logic analyzer hooked up to see what the incoming messages are to confirm that my PIC is receiving the same information as the analyzer is.
 | 
 
 Yes of course, that is the first i did, alway i did that....
 for not work to blind.
 
 The gps work ok, I want use the ccs example for not control the ISR, but to end I had do it.
 |  |  
		|  |  
		| bkamen 
 
 
 Joined: 07 Jan 2004
 Posts: 1616
 Location: Central Illinois, USA
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Wed Apr 10, 2019 1:26 pm |   |  
				| 
 |  
				| [quote="cvargcal"]  	  | dluu13 wrote: |  	  | The gps work ok, I want use the ccs example for not control the ISR, but to end I had do it.
 | 
 
 Having written several GPS enabled applications, you really want to use the serial ISR.
 
 There's all sorts of stuff you can do that's better off in the ISR than in your main program.
 
 ...like filtering the NMEA message types where if a character doesn't fall in your sequence, just bail on that message and wait for the next.
 
 Is the first char a $? Ok.
 Is the next char a G and then P... ok..
 is the next 3 something you want? ok...
 
 If not... nevermind.
 
 Don't even bother collecting the whole string and passing it off to the main loop.
 
 Also, I tend to use the PICs that have decent amounts of RAM in them so the buffer can hold several NMEA sentences for the main loop processor.
 
 So yea... definitely want an ISR handling the serial reception.
 _________________
 Dazed and confused? I don't think so. Just "plain lost" will do.  :D
 |  |  
		|  |  
		|  |  
  
	| 
 
 | You cannot post new topics in this forum You cannot reply to topics in this forum
 You cannot edit your posts in this forum
 You cannot delete your posts in this forum
 You cannot vote in polls in this forum
 
 |  
 Powered by phpBB © 2001, 2005 phpBB Group
 
 |