Files
RadioHead/examples/raspi/rf95/shared/gpsMT3339.cpp
Lazarewicz Julien 6c6451c92c first commit
2025-07-22 15:27:00 +02:00

220 lines
5.1 KiB
C++

// -*- mode: C++ -*-
/*
* gpsMT3339.cpp
*
* Created on: Jun 11, 2020
* Author: Tilman Glötzner
*/
#include <gpsMT3339.h>
#include <stdio.h>
#include <help_functions.h>
#include <unistd.h>
#include <stdlib.h>
gps_MT3339::gps_MT3339(const char* SerialDevice, pthread_mutex_t* mutex) {
// TODO Auto-generated constructor stub
this->mutex = mutex;
gpsDevice = (char*)malloc(strlen(SerialDevice)*sizeof(char));
if (gpsDevice == NULL)
return;
strcpy(gpsDevice,SerialDevice);
}
gps_MT3339::~gps_MT3339() {
// TODO Auto-generated destructor stub
stop_reading = true;
}
char* gps_MT3339::getLatitude() {
return latitude;
}
char* gps_MT3339::getLongitude() {
return longitude;
}
char* gps_MT3339::getTimestamp() {
return timestamp;
}
//function to read gps device of Dragino GPS hat (to be executed by a thread)
char* gps_MT3339::strnstr(const char *haystack, const char *needle, size_t len)
{
int i;
size_t needle_len;
if (0 == (needle_len = strnlen(needle, len)))
return (char *)haystack;
for (i=0; i<=(int)(len-needle_len); i++)
{
if ((haystack[0] == needle[0]) &&
(0 == strncmp(haystack, needle, needle_len)))
return (char *)haystack;
haystack++;
}
return NULL;
}
void* gps_MT3339::read_gps(void* ptr)
{
ssize_t n, gga_len,i, comma_count;
char* start_ptr;
char* end_ptr;
char gps_buffer[LEN_GPS_READ_BUFFER];
char GPGGA_sentence[LEN_GPS_SENTENCE_BUFFER];
int gps_fd = -1;
int nofchar;
unsigned int empty_read_counter = 0;
printf("\nGPS READER started\n");
print_scheduler();
print_scope();
memset(gps_buffer,0x00,LEN_GPS_READ_BUFFER);
while (!stop_reading)
{
if (gps_fd < 0)
{
// Open the device in non-blocking mode
gps_fd = open(gpsDevice, O_RDWR | O_NONBLOCK);
if(gps_fd < 0)
{
printf("\nWarning: Could not open device file of GPS\n" );
}
}
else
{
n = read(gps_fd, gps_buffer, 200);
if (n == 0)
{
// counter incremented if there was nothing read back
empty_read_counter++;
}
if (n <= 0)
{
if (((errno != EAGAIN) and (errno != EWOULDBLOCK))
|| (empty_read_counter > MAX_EMPTY_READ))
{
// on read error close file descriptor and restart
printf("GPS device being reset\n");
strcpy(longitude,"NoValue");
strcpy(latitude,"NoValue");
strcpy(timestamp,"NoValue");
close(gps_fd);
gps_fd = -1;
empty_read_counter = 0;
memset(gps_buffer,0x00,LEN_GPS_READ_BUFFER);
}
}
else
{
// sucessful read => reset counter tracking empty reads
empty_read_counter = 0;
// Scan for sentence read from GPS device containing position
start_ptr = strnstr (gps_buffer, "$GPGGA", LEN_GPS_READ_BUFFER);
if( start_ptr != NULL)
{
end_ptr = (char*)memchr(start_ptr, '*',
LEN_GPS_READ_BUFFER - (start_ptr - gps_buffer));
if (end_ptr != NULL)
{
gga_len = (end_ptr-start_ptr) < LEN_GPS_SENTENCE_BUFFER ?
(end_ptr-start_ptr) : LEN_GPS_SENTENCE_BUFFER - 1;
memcpy(GPGGA_sentence,gps_buffer,gga_len);
// make sure that string is delimited
GPGGA_sentence[gga_len+1] = '\0';
//printf(GPGGA_sentence);
// extract time and gps coordinates from GPGGA sentence
// Sample: GPGGA,193630.000,4846.8772,N,00912.2288,E,1,5,1.57,131.5,M,48.0,M,,
comma_count = 0;
//printf("%s\n", GPGGA_sentence);
start_ptr = strchr(GPGGA_sentence,',') + 1;
end_ptr=start_ptr;
pthread_mutex_lock(mutex);
while (end_ptr != NULL && start_ptr != NULL)
{
end_ptr = strchr(start_ptr,',');
if (end_ptr != NULL)
{
nofchar = (end_ptr - start_ptr);
if (nofchar < LEN_GPS_INFO_BUFFER)
end_ptr[0] = '\0';
else
break;
int temp = nofchar;
if (comma_count == 0)
{
nofchar = temp < LEN_GPS_INFO_BUFFER ?
temp : LEN_GPS_INFO_BUFFER - 1;
strncpy(timestamp,start_ptr,nofchar);
timestamp[nofchar] = '\0';
}
if (comma_count == 1)
{
nofchar = temp < LEN_GPS_INFO_BUFFER - 2 ?
temp : LEN_GPS_INFO_BUFFER - 2;
strncpy(latitude,start_ptr,nofchar);
latitude[nofchar] = '\0';
}
if (comma_count == 2)
{
nofchar = strlen(latitude);
strncpy(latitude+nofchar,start_ptr,1);
latitude[nofchar+1] = '\0';
}
if (comma_count == 3)
{
nofchar = temp < LEN_GPS_INFO_BUFFER - 2 ?
temp : LEN_GPS_INFO_BUFFER - 2;
strncpy(longitude,start_ptr,nofchar);
longitude[nofchar] = '\0';
}
if (comma_count == 4)
{
nofchar = strlen(longitude);
strncpy(longitude+nofchar,start_ptr,1);
longitude[nofchar+1] = '\0';
// all data extracted => bail out
end_ptr = NULL;
start_ptr = NULL;
}
start_ptr = end_ptr+1;
comma_count++;
}
}
pthread_mutex_unlock(mutex);
}
}
}
}
pthread_yield();
}
free(gpsDevice);
close(gps_fd);
}
void gps_MT3339::stop() {
stop_reading = true;
}