rewrite of nmea.c for fixing errors and saving memory

pull/2/head
sq2ips 2025-02-25 13:43:22 +01:00
rodzic 573dadfdf2
commit 05c49e3000
Nie znaleziono w bazie danych klucza dla tego podpisu
ID klucza GPG: C383A71588BA55F5
7 zmienionych plików z 158 dodań i 79 usunięć

30
.vscode/launch.json vendored
Wyświetl plik

@ -8,13 +8,35 @@
"servertype": "openocd",
"cwd": "${workspaceFolder}",
"runToEntryPoint": "main",
"executable": "./build/m20.elf",
"executable": "./m20/build/m20.elf",
"device": "STM32L051R6T6",
"configFiles": [
"debug/stlink.cfg",
"debug/stm32l0.cfg"
"./m20/openocd/openocd_m20.cfg"
],
"serverArgs": [
"-s",
"./m20/openocd/"
],
"showDevDebugOutput": "raw"
},
},
{
"name": "C/C++ Runner: Debug Session",
"type": "cppdbg",
"request": "launch",
"args": [],
"stopAtEntry": false,
"externalConsole": false,
"cwd": "/home/pawel/coding/m20-custom-firmware",
"program": "/home/pawel/coding/m20-custom-firmware/build/Debug/outDebug",
"MIMode": "gdb",
"miDebuggerPath": "gdb",
"setupCommands": [
{
"description": "Enable pretty-printing for gdb",
"text": "-enable-pretty-printing",
"ignoreFailures": true
}
]
}
]
}

Wyświetl plik

@ -8,16 +8,16 @@
/*-----------------------------------------------------------------*/
// Sonde configuration, parameters that should be changed
#define PAYLOAD_ID 256 // Sonde payload ID 256 - for 4FSKTEST-V2
#define PAYLOAD_ID 256 // Sonde payload ID 256 - for 4FSKTEST-V2
#define TIME_PERIOD 6 // Time betwen starts of transmissions (in seconds) (must be more than 3)
#define TIME_PERIOD 6 // Time betwen starts of transmissions (in seconds) (must be more than 3)
#define GPS_TYPE 1 // Type of GPS module: 1 - u-blox | 2 - XM1110
#define GPS_TYPE 1 // Type of GPS module: 1 - u-blox | 2 - XM1110
#define QRG_FSK4 435100000 // Frequency fo horus modulation (in Hz)
#define QRG_FSK4 435100000 // Frequency fo horus modulation (in Hz)
#define PA_FSK4 10 // RF power setting for horus transmission values 0-63
#define RF_BOOST_ACTIVE 1 // RF booster enabled for transmissions about 15dB gain, but more power consumed - normally should be ON(1).
#define PA_FSK4 10 // RF power setting for horus transmission values 0-63
#define RF_BOOST_ACTIVE 0 // RF booster enabled for transmissions about 15dB gain, but more power consumed - normally should be ON(1).
#define ADF_FREQ_CORRECTION 19 // correction of frequency from crystal inaccuracy in 270Hz steps. To be individually set for each sonde.
@ -40,7 +40,10 @@
#define AscentRateTime 10 // Time of ascent rate mesure
// type 1
#define DATA_SIZE 35 // Max number of NMEA sentences in one parsing
#define DATA_SIZE 35 // Max number of NMEA sentences in one parsing
#define SENTENCE_SIZE 82+1 // Max lenght of a NMEA sentence is 82 characters
#define MAX_SENTENCE_ELEMENTS 10 // Max number of NMEA sentence elements (no element with number bigger than 9 is used)
#define SENTENCE_ELEMENT_LEN 12 // Max lenght of a sentence element
// type 2
#define FrameLen 62 // Length of XM1110 frame

Wyświetl plik

@ -13,13 +13,13 @@
typedef struct NMEA_DATA {
float Lat; //latitude in degrees with decimal places + for N - for S
float Lon; //longitude in degrees with decimal places
float Alt; //altitude in meters
float Speed;
uint16_t Alt; //altitude in meters
uint16_t Speed;
float AscentRate;
uint8_t Hours;
uint8_t Minutes;
uint8_t Seconds;
float HDOP; //horizontal dilution of precision
//float HDOP; //horizontal dilution of precision
uint8_t Sats; //number of satellites used in measurement
uint8_t Fix; // 0 = no data, 1 = no fix, 2 = 2D fix, 3 = 3D fix
uint8_t Corr; // number of correct frames

Wyświetl plik

@ -109,7 +109,7 @@ void main_loop(void){
// LED
LL_GPIO_TogglePin(LED_GPIO_Port, LED_Pin);
#ifdef DEBUG
//printf("%s", GpsDataBuffer);
printf("Frame: %d\r\n", HorusPacket.PacketCount);
#endif
while(GpsBufferReady){}
@ -131,13 +131,13 @@ void main_loop(void){
HorusPacket.Seconds = NmeaData.Seconds;
HorusPacket.Lat = NmeaData.Lat;
HorusPacket.Lon = NmeaData.Lon;
HorusPacket.Speed = (uint8_t)round(NmeaData.Speed);
HorusPacket.Speed = (uint8_t)NmeaData.Speed;
HorusPacket.AscentRate = (int16_t)round(NmeaData.AscentRate*100);
HorusPacket.Alt = (uint16_t)round(NmeaData.Alt);
HorusPacket.Alt = NmeaData.Alt;
HorusPacket.Sats = NmeaData.Sats;
#ifdef DEBUG
printf("\r\nFix: %d, Lat: %d, Lon: %d, Alt: %d m, Speed: %d km/h, Ascent rate: %d m/s Satellites: %d HDOP: %d, Time: %d:%d:%d, correct frames: %d\r\n",
NmeaData.Fix, (uint32_t)(NmeaData.Lat*10e6), (uint32_t)(NmeaData.Lon*10e6), (uint16_t)(NmeaData.Alt*10), (uint16_t)(NmeaData.Speed*10e3), (int16_t)round(NmeaData.AscentRate), NmeaData.Sats, (uint16_t)round(NmeaData.HDOP*1e2), NmeaData.Hours, NmeaData.Minutes, NmeaData.Seconds, NmeaData.Corr);
printf("\r\nFix: %d, Lat: %d, Lon: %d, Alt: %d m, Speed: %d km/h, Ascent rate: %d m/s Satellites: %d, Time: %d:%d:%d, correct frames: %d\r\n",
NmeaData.Fix, (uint32_t)(NmeaData.Lat*10e6), (uint32_t)(NmeaData.Lon*10e6), NmeaData.Alt, NmeaData.Speed, (int16_t)round(NmeaData.AscentRate*100), NmeaData.Sats, NmeaData.Hours, NmeaData.Minutes, NmeaData.Seconds, NmeaData.Corr);
#endif
// GPS type 2 data
@ -255,7 +255,7 @@ int main(void)
}
GpsBufferReady = false;
}
LL_mDelay(20);
LL_mDelay(10);
}
/* USER CODE END 3 */
}
@ -409,7 +409,7 @@ static void MX_IWDG_Init(void)
/* USER CODE END IWDG_Init 1 */
LL_IWDG_Enable(IWDG);
LL_IWDG_EnableWriteAccess(IWDG);
LL_IWDG_SetPrescaler(IWDG, LL_IWDG_PRESCALER_4);
LL_IWDG_SetPrescaler(IWDG, LL_IWDG_PRESCALER_8);
LL_IWDG_SetReloadCounter(IWDG, 4095);
while (LL_IWDG_IsReady(IWDG) != 1)
{

Wyświetl plik

@ -7,14 +7,34 @@
#include "main.h"
#include "config.h"
#include <stdlib.h>
#ifdef DEBUG
#include <stdio.h>
#endif
#include <string.h>
char *data[DATA_SIZE];
char data[DATA_SIZE][SENTENCE_SIZE];
uint8_t correct = 0;
uint16_t olddTime = 0;
float olddAlt = 0;
uint16_t olddAlt = 0;
uint16_t a_strtof(char *buffer){
uint8_t d_pos = 0;
uint16_t value = 0;
while(buffer[d_pos]!='.'){
if(buffer[d_pos] == '\0') return 0;
d_pos++;
}
uint16_t e = 1;
for(int8_t pos = d_pos-1; pos >= 0; pos--){
value += (buffer[pos]-'0') * e;
e *= 10;
}
if((buffer[d_pos+1]-'0') >= 5) value++; // rounding first decimal place
return value;
}
uint8_t checksum(char *nmea_frame)
{
@ -30,7 +50,19 @@ uint8_t checksum(char *nmea_frame)
for (i = 0; i < strlen(nmea_frame) - 5; i ++) {
crc ^= nmea_frame[i];
}
int receivedHash = strtol(recv_crc, NULL, 16);
int receivedHash = 0;
for (int i = 0; i < 2; i++) {
receivedHash <<= 4; // Shift left by 4 bits (equivalent to multiplying by 16)
if (recv_crc[i] >= '0' && recv_crc[i] <= '9') {
receivedHash += recv_crc[i] - '0'; // Convert '0'-'9' to 0-9
} else if (recv_crc[i] >= 'A' && recv_crc[i] <= 'F') {
receivedHash += recv_crc[i] - 'A' + 10; // Convert 'A'-'F' to 10-15
} else {
return 0;
}
}
if (crc == receivedHash) {
return 1;
}else{
@ -38,21 +70,41 @@ uint8_t checksum(char *nmea_frame)
}
}
int getValues(char*inputString, char *values[]){
char *marker = strtok(inputString, ",");
int counter = 0;
while (marker != NULL) {
values[counter++] = malloc(strlen(marker) + 1); //free later!!!!!!
strcpy(values[counter - 1], marker);
marker = strtok(NULL, ",");
uint8_t getValues(char *inputString, char values[MAX_SENTENCE_ELEMENTS][SENTENCE_ELEMENT_LEN]) {
uint8_t pos = 0;
uint8_t d_pos = 0;
uint8_t cnt = 0;
char buffer[SENTENCE_ELEMENT_LEN];
memset(buffer, 0, SENTENCE_ELEMENT_LEN);
while (pos < strlen(inputString) && inputString[pos] != '\n' && pos < SENTENCE_SIZE && cnt < MAX_SENTENCE_ELEMENTS) {
if (inputString[pos] == ',') {
// If the length of the value is within buffer limits
if (pos - d_pos < SENTENCE_ELEMENT_LEN) {
strncpy(values[cnt], buffer, pos - d_pos); // Copy the buffer into values[cnt]
memset(values[cnt] + (pos - d_pos), 0, SENTENCE_ELEMENT_LEN - (pos - d_pos)); // Ensure null termination
//cnt++;
}
cnt++;
// Reset buffer and update d_pos to start from next character
memset(buffer, 0, SENTENCE_ELEMENT_LEN);
d_pos = pos + 1;
} else {
// Check if writing to buffer exceeds its size
if (pos - d_pos < SENTENCE_ELEMENT_LEN - 1) { // Ensure space for null terminator
buffer[pos - d_pos] = inputString[pos];
}
}
pos++;
}
return counter;
return cnt;
}
int nmea_GGA(NMEA *nmea_data, char*inputString){
char *values[25];
int nmea_GGA(NMEA *nmea_data, char *inputString){
char values[MAX_SENTENCE_ELEMENTS][SENTENCE_ELEMENT_LEN];
memset(values, 0, sizeof(values));
int counter = getValues(inputString, values);
uint8_t len = getValues(inputString, values);
if(len<9) return 0;
uint8_t h = (values[1][0]-'0')*10 + (values[1][1]-'0');
uint8_t m = (values[1][2]-'0')*10 + (values[1][3]-'0');
@ -92,9 +144,10 @@ int nmea_GGA(NMEA *nmea_data, char*inputString){
if(latSide == 'S') nmea_data->Lat *= -1;
if(lonSide == 'W') nmea_data->Lon *= -1;
float altitude = strtof(values[9], NULL);
if(altitude !=0){
uint16_t altitude = a_strtof(values[9]);
if(altitude != 0){
nmea_data->Alt = altitude;
uint16_t currentTime = h*3600+m*60+s;
if(olddTime==0 || currentTime == 0){
@ -108,47 +161,53 @@ int nmea_GGA(NMEA *nmea_data, char*inputString){
}
}
nmea_data->Sats = strtol(values[7], NULL, 10);
nmea_data->Sats = (values[7][0]-'0')*10 + (values[7][1]-'0');
//nmea_data->Fix = strtol(values[6], NULL, 10);
//nmea_data->Fix = values[6][0]-'0';
float hdop = strtof(values[8], NULL);
if(nmea_data->Fix > 1) nmea_data->HDOP = hdop!=0 ? hdop : nmea_data->HDOP;
//float hdop = strtof(values[8], NULL);
//if(nmea_data->Fix > 1) nmea_data->HDOP = hdop!=0 ? hdop : nmea_data->HDOP;
for(int i=0; i<counter; i++) free(values[i]);
return 1;
}
}
for(int i=0; i<counter; i++) free(values[i]);
return 0;
}
int nmea_GSA(NMEA *nmea_data, char*inputString){
char *values[25];
char values[MAX_SENTENCE_ELEMENTS][SENTENCE_ELEMENT_LEN];
memset(values, 0, sizeof(values));
int counter = getValues(inputString, values);
uint8_t len = getValues(inputString, values);
if(len<2) return 0;
nmea_data->Fix = strtol(values[2], NULL, 10);
int satelliteCount = 0;
uint8_t fix = (values[2][0]-'0');
if(fix<=3){
nmea_data->Fix = fix;
}else{
nmea_data->Fix = 0;
return 0;
}
/*int satelliteCount = 0;
for(int i=3; i<15; i++){
if(values[i][0] != 0){
satelliteCount++;
}
}
//nmea_data->Sats = satelliteCount;
for(int i=0; i<counter; i++) free(values[i]);
nmea_data->Sats = satelliteCount;*/
return 1;
}
int nmea_GLL(NMEA *nmea_data, char*inputString) {
char *values[25];
char values[MAX_SENTENCE_ELEMENTS][SENTENCE_ELEMENT_LEN];
memset(values, 0, sizeof(values));
int counter = getValues(inputString, values);
uint8_t len = getValues(inputString, values);
if(len<3) return 0;
uint8_t lonSide = values[4][0];
uint8_t latSide = values[2][0];
@ -177,54 +236,50 @@ int nmea_GLL(NMEA *nmea_data, char*inputString) {
nmea_data->Lon = lon;
if(latSide == 'S') nmea_data->Lat *= -1;
if(lonSide == 'W') nmea_data->Lon *= -1;
for(int i = 0; i<counter; i++) free(values[i]);
return 1;
}
}
for(int i = 0; i<counter; i++) free(values[i]);
return 0;
}
int nmea_VTG(NMEA *nmea_data, char*inputString) {
char *values[25];
char values[MAX_SENTENCE_ELEMENTS][SENTENCE_ELEMENT_LEN];
memset(values, 0, sizeof(values));
int counter = getValues(inputString, values);
uint8_t len = getValues(inputString, values);
if(len<5) return 0;
float speed = strtof(values[5], NULL); // 5 for kph, 3 for knots
if(speed != 0.0) nmea_data->Speed = speed;
for(int i=0; i<counter; i++) free(values[i]);
nmea_data->Speed = a_strtof(values[5]); // 5 for kph, 3 for knots
return 1;
}
void ParseNMEA(NMEA *nmea_data, uint8_t *buffer){
memset(data, 0, sizeof(data));
char * token = strtok(buffer, "$");
char * token = strtok((char *)buffer, "$");
uint8_t cnt = 0;
while( token != NULL && cnt < DATA_SIZE) {
data[cnt] = malloc(strlen(token)+1);
strcpy(data[cnt], token);
cnt++;
if(strlen(token)<SENTENCE_SIZE){
strncpy(data[cnt], token, SENTENCE_SIZE - 1);
data[cnt][SENTENCE_SIZE - 1] = '\0';
cnt++;
}
token = strtok(NULL, "$");
}
correct = 0;
for(uint8_t i = 0; i<cnt; i++){
if(strstr(data[i], "GN")!=NULL && strstr(data[i], "\r\n")!=NULL && checksum(data[i])){
if(strstr(data[i], "GNGLL")!=NULL){
nmea_GLL(nmea_data, data[i]);
correct++;
#ifdef DEBUG
//printf(">%s", data[i]);
#endif
if(strstr(data[i], "GNGLL")!=NULL){
if(nmea_GLL(nmea_data, data[i])) correct++;
}else if(strstr(data[i], "GNGSA")!=NULL){
nmea_GSA(nmea_data, data[i]);
correct++;
if(nmea_GSA(nmea_data, data[i])) correct++;
}else if(strstr(data[i], "GNGGA")!=NULL){
nmea_GGA(nmea_data, data[i]);
correct++;
if (nmea_GGA(nmea_data, data[i])) correct++;
}else if(strstr(data[i], "GNVTG")!=NULL){
nmea_VTG(nmea_data, data[i]);
correct++;
if (nmea_VTG(nmea_data, data[i])) correct++;
}
}
nmea_data->Corr = correct;
@ -232,6 +287,4 @@ void ParseNMEA(NMEA *nmea_data, uint8_t *buffer){
nmea_data->Fix = 0;
}
}
//printf("\r\n%d\r\n", SysTick_counter);
for(uint8_t i = 0; i<cnt; i++) free(data[i]);
}

Wyświetl plik

@ -1,5 +1,5 @@
##########################################################################################################################
# File automatically-generated by tool: [projectgenerator] version: [4.5.0-RC5] date: [Fri Feb 21 18:31:14 CET 2025]
# File automatically-generated by tool: [projectgenerator] version: [4.5.0-RC5] date: [Tue Feb 25 13:01:11 CET 2025]
##########################################################################################################################
# ------------------------------------------------
@ -137,7 +137,7 @@ C_INCLUDES = \
# compile gcc flags
ASFLAGS = $(MCU) $(AS_DEFS) $(AS_INCLUDES) $(OPT) -Wall -fdata-sections -ffunction-sections
CFLAGS += $(MCU) $(C_DEFS) $(C_INCLUDES) $(OPT) -Wall -fdata-sections -ffunction-sections -DNDEBUG -flto
CFLAGS += $(MCU) $(C_DEFS) $(C_INCLUDES) $(OPT) -Wall -fdata-sections -ffunction-sections# -DNDEBUG -flto
ifeq ($(DEBUG), 1)
CFLAGS += -g -gdwarf-2

Wyświetl plik

@ -4,7 +4,8 @@ CAD.pinconfig=Dual
CAD.provider=
File.Version=6
GPIO.groupedBy=Group By Peripherals
IWDG.IPParameters=Window,Reload
IWDG.IPParameters=Window,Reload,Prescaler
IWDG.Prescaler=IWDG_PRESCALER_8
IWDG.Reload=4095
IWDG.Window=4095
KeepUserPlacement=false