yincheng.zhong
7 天以前 b53fff11e6f0d560594834de32886239cbba90a3
STM32H743/FML/GPS.c
@@ -6,39 +6,36 @@
#include "math.h"
#include "HIDO_Util.h"
#include "HIDO_Debug.h"
#include "HIDO_Timer.h"
#include "FreeRTOS.h"
#include "task.h"
#include "GPS.h"
#include "DBG.h"
#include "GPIO.h"
#include "Uart.h"
#include "geo_utils.h"
#include "UDPClient.h"
// #include "global_param.h"
#include "TTS.h"
#define GPS_DBG(level, fmt, ...) HIDO_Debug(fmt, __VA_ARGS__)
#define GPS_UART_RX_BUF_SIZE 1024
#define GPS_UART_TX_BUF_SIZE (2048 + 512)
typedef enum
{
    GPS_RECV_STATE_IDLE = 0,
    GPS_RECV_STATE_HEAD,
    GPS_RECV_STATE_CR,
    GPS_RECV_STATE_LF,
} E_GPSRecvState;
#define IM23A_HEADER_LEN                (4U)
#define IM23A_NAV_FRAME_LEN             (100U)   /* fmin: 包含头、负载、校验和及尾部 */
#define IM23A_IMU_FRAME_LEN             (52U)    /* fmim: IMU数据帧 */
#define IM23A_GIG_FRAME_LEN             (95U)    /* fmig: GPS导航数据帧 */
#define IM23A_MAX_FRAME_LEN             IM23A_NAV_FRAME_LEN
#define IM23A_CHECKSUM_LEN              (2U)
#define IM23A_TAIL_LEN                  (2U)
#define IM23A_NAV_HEADER                "fmin"
#define IM23A_IMU_HEADER                "fmim"
#define IM23A_GIG_HEADER                "fmig"
#define IM23A_GPS_TIME_SCALE            (100.0)  /* hhmmss.ss -> centisecond */
#define IM23A_IMU_TIME_SCALE            (1000.0) /* hhmmss.ss -> millisecond */
typedef struct
{
    E_GPSRecvState m_eState;
    HIDO_CHAR m_acHeader[10], m_acHeader2[10], m_acHeader3[10], m_acHeader4[10], m_acHeader5[10];
    HIDO_UINT32 m_u32HeaderLen, m_u32Header2Len, m_u32Header3Len, m_u32Header4Len, m_u32Header5Len;
    HIDO_CHAR m_acRecvBuf[256];  // 增大缓冲区以容纳GPRMI的23个字段
    HIDO_UINT8 m_au8Buffer[IM23A_MAX_FRAME_LEN];
    HIDO_UINT32 m_u32RecvLen;
    HIDO_UINT32 m_u32ExpectedLen;
} ST_GPSRecv;
static HIDO_UINT8 l_au8GPSUartRxBuf[GPS_UART_RX_BUF_SIZE];
@@ -48,441 +45,386 @@
static ST_GPSRecv l_stGPSRecv;
static HIDO_UINT8 l_u8PosState = 0;
static HIDO_UINT8 l_u8GPS_HZ = 0;
static HIDO_BOOL l_bGPSConfig = HIDO_FALSE;
static HIDO_UINT32 l_u32QXTick = 0;
HIDO_UINT32 getRTK_Tick = 0;
/* 存储最新的GPRMI和GPIMU数据 */
/* 存储最新的GPRMI、GPIMU和GPGIG数据 */
static ST_GPRMI l_stGPRMI;
static ST_GPIMU l_stGPIMU;
static ST_GPGIG l_stGPGIG;
static HIDO_UINT32 s_gprmi_log_idx = 0U;
/* ENU坐标系原点(开机后第一个固定解时初始化) */
static ST_GeoOrigin l_stGeoOrigin = {0};
static float l_fCurrentENU[3] = {0.0f, 0.0f, 0.0f};
/* GPS���� */
static HIDO_UINT8 l_au8CmdSave[] = {0xB5, 0x62, 0x06, 0x09, 0x0D, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x17, 0x31, 0xBF};
static HIDO_UINT8 l_au8CmdRate1Hz[] = {0xB5, 0x62, 0x06, 0x08, 0x06, 0x00, 0xE8, 0x03, 0x01, 0x00, 0x01, 0x00, 0x01, 0x39};
static HIDO_UINT8 l_au8CmdRate2Hz[] = {0xB5, 0x62, 0x06, 0x08, 0x06, 0x00, 0xF4, 0x01, 0x01, 0x00, 0x01, 0x00, 0x0B, 0x77};
static HIDO_UINT8 l_au8CmdRate5Hz[] = {0xB5, 0x62, 0x06, 0x08, 0x06, 0x00, 0xC8, 0x00, 0x01, 0x00, 0x01, 0x00, 0xDE, 0x6A};
static HIDO_UINT8 l_au8CmdRate10Hz[] = {0xB5, 0x62, 0x06, 0x08, 0x06, 0x00, 0x64, 0x00, 0x01, 0x00, 0x01, 0x00, 0x7A, 0x12};
/*******************************************************************************
 * Function Name     : GPS_RateConfig
 * Description       : GPS�������� ֧��1HZ��2HZ��5HZ��10HZ
 * Input             : _u8Rate ����
 * Output            : None
 * Return            : HIDO_OK �ɹ�, HIDO_ERR ʧ��
 * Author            : www.hido-studio.com
 * Modified Date:    : 2021��5��07��
 *                          IM23A Helper Declarations                          *
 *******************************************************************************/
static HIDO_INT32 GPS_RateConfig(HIDO_UINT8 _u8Rate)
static HIDO_VOID IM23A_ResetParser(ST_GPSRecv *parser);
static HIDO_BOOL IM23A_ValidateFrame(const HIDO_UINT8 *frame, HIDO_UINT32 len);
static HIDO_VOID IM23A_HandleFrame(const HIDO_UINT8 *frame, HIDO_UINT32 len);
static HIDO_VOID IM23A_HandleNavFrame(const HIDO_UINT8 *frame);
static HIDO_VOID IM23A_HandleImuFrame(const HIDO_UINT8 *frame);
static HIDO_VOID IM23A_HandleGigFrame(const HIDO_UINT8 *frame);
static HIDO_DOUBLE IM23A_ReadDouble(const HIDO_UINT8 *ptr);
static HIDO_FLOAT IM23A_ReadFloat(const HIDO_UINT8 *ptr);
static HIDO_UINT32 IM23A_ReadU32(const HIDO_UINT8 *ptr);
static HIDO_UINT16 IM23A_ReadU16(const HIDO_UINT8 *ptr);
static HIDO_UINT32 IM23A_ConvertTime(double utc_val, double scale);
/* 旧版NMEA解析函数已移除,以下为IM23A专用解析工具函数 */
static HIDO_DOUBLE IM23A_ReadDouble(const HIDO_UINT8 *ptr)
{
    l_u8GPS_HZ = _u8Rate;
    switch (_u8Rate)
    {
    case 1:
    {
        Uart_Send(UART_ID_GPS, l_au8CmdRate1Hz, sizeof(l_au8CmdRate1Hz));
        Uart_Send(UART_ID_GPS, l_au8CmdSave, sizeof(l_au8CmdSave));
        break;
    }
    case 2:
    {
        Uart_Send(UART_ID_GPS, l_au8CmdRate2Hz, sizeof(l_au8CmdRate2Hz));
        Uart_Send(UART_ID_GPS, l_au8CmdSave, sizeof(l_au8CmdSave));
        break;
    }
    case 5:
    {
        Uart_Send(UART_ID_GPS, l_au8CmdRate5Hz, sizeof(l_au8CmdRate5Hz));
        Uart_Send(UART_ID_GPS, l_au8CmdSave, sizeof(l_au8CmdSave));
        break;
    }
    case 10:
    {
        Uart_Send(UART_ID_GPS, l_au8CmdRate10Hz, sizeof(l_au8CmdRate10Hz));
        Uart_Send(UART_ID_GPS, l_au8CmdSave, sizeof(l_au8CmdSave));
        break;
    }
    default:
    {
        break;
    }
    }
    return HIDO_OK;
    HIDO_DOUBLE val = 0.0;
    memcpy(&val, ptr, sizeof(val));
    return val;
}
/*******************************************************************************
 * Function Name     : GPS_DataCheck
 * Description       : GPS���ݸ�ʽ���
 * Input             : _pcData GPS����
 *                   : _u32Len GPS���ݳ���
 * Output            : None
 * Return            : HIDO_OK �ɹ�, HIDO_ERR ʧ��
 * Author            : www.hido-studio.com
 * Modified Date:    : 2021��5��07��
 *******************************************************************************/
static HIDO_INT32 GPS_DataCheck(HIDO_CHAR *_pcData, HIDO_UINT32 _u32Len)
static HIDO_FLOAT IM23A_ReadFloat(const HIDO_UINT8 *ptr)
{
    HIDO_DataStruct stData;
    HIDO_DataStruct stCheckValue;
    HIDO_UINT8 u8CheckValue = 0;
    HIDO_UINT8 u8CalcValue = 0;
    HIDO_UINT32 i = 0;
    if (HIDO_UtilParseFormat((HIDO_UINT8 *)_pcData, _u32Len, "$%p*%p\r\n", &stData, &stCheckValue) != 2)
    {
        return HIDO_ERR;
    }
    u8CheckValue = HIDO_UtilHexStrBufToInt((HIDO_CHAR *)stCheckValue.m_pData, stCheckValue.m_u32Len);
    u8CalcValue = ((HIDO_UINT8 *)stData.m_pData)[0];
    for (i = 1; i < stData.m_u32Len; i++)
    {
        u8CalcValue ^= ((HIDO_UINT8 *)stData.m_pData)[i];
    }
    if (u8CalcValue != u8CheckValue)
    {
        return HIDO_ERR;
    }
    return HIDO_OK;
    HIDO_FLOAT val = 0.0f;
    memcpy(&val, ptr, sizeof(val));
    return val;
}
/*******************************************************************************
 * Function Name     : GPS_ParseGGA
 * Description       : GPS GGA���ݽ���(����������Ƿ���Ч)
 * Input             : _pcData GGA����
 *                   : _u32Len GGA���ݳ���
 * Output            : None
 * Return            : HIDO_OK �ɹ�, HIDO_ERR ʧ��
 * Author            : www.hido-studio.com
 * Modified Date:    : 2021��5��07��
 *******************************************************************************/
static HIDO_INT32 GPS_ParseGGA(HIDO_CHAR *_pcData, HIDO_UINT32 _u32Len)
static HIDO_UINT32 IM23A_ReadU32(const HIDO_UINT8 *ptr)
{
    ST_GPS stGPS;
    HIDO_DataStruct stPosState;
    memset(&stGPS, 0, sizeof(ST_GPS));
    if (GPS_DataCheck(_pcData, _u32Len) != HIDO_OK)
    {
        return HIDO_ERR;
    }
    if (HIDO_UtilParseFormat((HIDO_UINT8 *)_pcData, _u32Len, "$%*,%*,%*,%*,%*,%*,%p,%*,%*,%*,%*,%*,%*,%*,%**", &stPosState) != 15)
    {
        return HIDO_ERR;
    }
    l_u8PosState = atoi((HIDO_CHAR *)stPosState.m_pData);
    return HIDO_OK;
    HIDO_UINT32 val = 0U;
    memcpy(&val, ptr, sizeof(val));
    return val;
}
/*******************************************************************************
 * Function Name     : GPS_ParseGPRMI
 * Description       : 解析GPRMI数据包
 * Input             : _pcData GPRMI数据
 *                   : _u32Len GPRMI数据长度
 * Output            : None
 * Return            : HIDO_OK 成功, HIDO_ERR 失败
 * Author            : www.hido-studio.com
 * Modified Date:    : 2025年11月11日
 *******************************************************************************/
static HIDO_INT32 GPS_ParseGPRMI(HIDO_CHAR *_pcData, HIDO_UINT32 _u32Len)
static HIDO_UINT16 IM23A_ReadU16(const HIDO_UINT8 *ptr)
{
    HIDO_DataStruct astFields[23];
    HIDO_UINT16 val = 0U;
    memcpy(&val, ptr, sizeof(val));
    return val;
}
static HIDO_UINT32 IM23A_ConvertTime(double utc_val, double scale)
{
    if (!isfinite(utc_val) || utc_val < 0.0)
    {
        return 0U;
    }
    double scaled = utc_val * scale;
    if (scaled < 0.0)
    {
        scaled = 0.0;
    }
    return (HIDO_UINT32)(scaled + 0.5);
}
static HIDO_VOID IM23A_ResetParser(ST_GPSRecv *parser)
{
    memset(parser->m_au8Buffer, 0, sizeof(parser->m_au8Buffer));
    parser->m_u32RecvLen = 0U;
    parser->m_u32ExpectedLen = 0U;
}
static HIDO_BOOL IM23A_ValidateFrame(const HIDO_UINT8 *frame, HIDO_UINT32 len)
{
    if (frame == HIDO_NULL)
    {
        return HIDO_FALSE;
    }
    if ((len != IM23A_NAV_FRAME_LEN) && (len != IM23A_IMU_FRAME_LEN) && (len != IM23A_GIG_FRAME_LEN))
    {
        return HIDO_FALSE;
    }
    if (frame[len - 2U] != 'e' || frame[len - 1U] != 'd')
    {
        return HIDO_FALSE;
    }
    const HIDO_UINT32 sum_len = len - IM23A_TAIL_LEN - IM23A_CHECKSUM_LEN;
    HIDO_UINT16 calc = 0U;
    for (HIDO_UINT32 i = 0U; i < sum_len; ++i)
    {
        calc = (HIDO_UINT16)(calc + frame[i]);
    }
    const HIDO_UINT16 expect = IM23A_ReadU16(&frame[sum_len]);
    return (calc == expect);
}
static HIDO_VOID IM23A_HandleFrame(const HIDO_UINT8 *frame, HIDO_UINT32 len)
{
    if (len == IM23A_NAV_FRAME_LEN && memcmp(frame, IM23A_NAV_HEADER, IM23A_HEADER_LEN) == 0)
    {
        IM23A_HandleNavFrame(frame);
    }
    else if (len == IM23A_IMU_FRAME_LEN && memcmp(frame, IM23A_IMU_HEADER, IM23A_HEADER_LEN) == 0)
    {
        IM23A_HandleImuFrame(frame);
    }
    else if (len == IM23A_GIG_FRAME_LEN && memcmp(frame, IM23A_GIG_HEADER, IM23A_HEADER_LEN) == 0)
    {
        IM23A_HandleGigFrame(frame);
    }
}
static HIDO_VOID IM23A_HandleNavFrame(const HIDO_UINT8 *frame)
{
    if (IM23A_ValidateFrame(frame, IM23A_NAV_FRAME_LEN) == HIDO_FALSE)
    {
        return;
    }
    const HIDO_UINT8 *p = frame + IM23A_HEADER_LEN;
    double utc = IM23A_ReadDouble(p); p += 8U;
    double latitude = IM23A_ReadDouble(p); p += 8U;
    double longitude = IM23A_ReadDouble(p); p += 8U;
    double altitude = IM23A_ReadDouble(p); p += 8U;
    float vel_n = IM23A_ReadFloat(p); p += 4U;
    float vel_e = IM23A_ReadFloat(p); p += 4U;
    float vel_d = IM23A_ReadFloat(p); p += 4U;
    float roll = IM23A_ReadFloat(p); p += 4U;
    float pitch = IM23A_ReadFloat(p); p += 4U;
    float heading = IM23A_ReadFloat(p); p += 4U;
    float pos_accuracy = IM23A_ReadFloat(p); p += 4U;
    float accel_bias_x = IM23A_ReadFloat(p); p += 4U;
    float accel_bias_y = IM23A_ReadFloat(p); p += 4U;
    float accel_bias_z = IM23A_ReadFloat(p); p += 4U;
    float gyro_bias_x = IM23A_ReadFloat(p); p += 4U;
    float gyro_bias_y = IM23A_ReadFloat(p); p += 4U;
    float gyro_bias_z = IM23A_ReadFloat(p); p += 4U;
    float sensor_temp = IM23A_ReadFloat(p); p += 4U;
    HIDO_UINT32 status = IM23A_ReadU32(p);
    memset(&l_stGPRMI, 0, sizeof(ST_GPRMI));
    l_stGPRMI.m_bValid = HIDO_FALSE;
    l_stGPRMI.m_u32UTCTime = IM23A_ConvertTime(utc, IM23A_GPS_TIME_SCALE);
    l_stGPRMI.m_dLatitude = latitude;
    l_stGPRMI.m_dLongitude = longitude;
    l_stGPRMI.m_fAltitude = (HIDO_FLOAT)altitude;
    l_stGPRMI.m_fNorthVelocity = vel_n;
    l_stGPRMI.m_fEastVelocity = vel_e;
    l_stGPRMI.m_fUpVelocity = (HIDO_FLOAT)(-vel_d);
    /* IM23A协议中 roll/pitch/heading 均为弧度,需转换为角度 */
    l_stGPRMI.m_fRollAngle = roll * 57.29577951308232f;     // RAD2DEG
    l_stGPRMI.m_fPitchAngle = pitch * 57.29577951308232f;   // RAD2DEG
    l_stGPRMI.m_fHeadingAngle = heading * 57.29577951308232f;  // RAD2DEG
    
    // 检查数据格式校验
    if (GPS_DataCheck(_pcData, _u32Len) != HIDO_OK)
    /* Heading 转换为 0-360 范围 */
    while (l_stGPRMI.m_fHeadingAngle < 0.0f)
    {
        return HIDO_ERR;
        l_stGPRMI.m_fHeadingAngle += 360.0f;
    }
    // 解析23个字段: $GPRMI,<1>,<2>,<3>,<4>,<5>,<6>,<7>,<8>,<9>,<10>,<11>,<12>,<13>,<14>,<15>,<16>,<17>,<18>,<19>,<20>,<21>,<22>,<23>*<CR><LF>
    if (HIDO_UtilParseFormat((HIDO_UINT8 *)_pcData, _u32Len,
        "$%*,%p,%p,%p,%p,%p,%p,%p,%p,%p,%p,%p,%p,%p,%p,%p,%p,%p,%p,%p,%p,%p,%p,%p*%**",
        &astFields[0], &astFields[1], &astFields[2], &astFields[3], &astFields[4],
        &astFields[5], &astFields[6], &astFields[7], &astFields[8], &astFields[9],
        &astFields[10], &astFields[11], &astFields[12], &astFields[13], &astFields[14],
        &astFields[15], &astFields[16], &astFields[17], &astFields[18], &astFields[19],
        &astFields[20], &astFields[21], &astFields[22]) != 25)
    while (l_stGPRMI.m_fHeadingAngle >= 360.0f)
    {
        return HIDO_ERR;
        l_stGPRMI.m_fHeadingAngle -= 360.0f;
    }
    // 解析各字段
    l_stGPRMI.m_u32UTCTime = (HIDO_UINT32)(atof((HIDO_CHAR *)astFields[0].m_pData) * 100);  // 转换为整数
    l_stGPRMI.m_u16WeekNumber = (HIDO_UINT16)atoi((HIDO_CHAR *)astFields[1].m_pData);
    l_stGPRMI.m_u32TimeOfWeek = (HIDO_UINT32)(atof((HIDO_CHAR *)astFields[2].m_pData) * 1000);  // 转换为毫秒
    l_stGPRMI.m_dLatitude = atof((HIDO_CHAR *)astFields[3].m_pData);
    l_stGPRMI.m_dLongitude = atof((HIDO_CHAR *)astFields[4].m_pData);
    l_stGPRMI.m_fAltitude = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[5].m_pData);
    l_stGPRMI.m_fEastStdDev = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[6].m_pData);
    l_stGPRMI.m_fNorthStdDev = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[7].m_pData);
    l_stGPRMI.m_fAltStdDev = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[8].m_pData);
    l_stGPRMI.m_fEastVelocity = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[9].m_pData);
    l_stGPRMI.m_fNorthVelocity = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[10].m_pData);
    l_stGPRMI.m_fUpVelocity = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[11].m_pData);
    l_stGPRMI.m_fEastVelStdDev = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[12].m_pData);
    l_stGPRMI.m_fNorthVelStdDev = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[13].m_pData);
    l_stGPRMI.m_fUpVelStdDev = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[14].m_pData);
    l_stGPRMI.m_fRollAngle = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[15].m_pData);
    l_stGPRMI.m_fPitchAngleStdDev = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[16].m_pData);
    l_stGPRMI.m_fRollAngleStdDev = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[17].m_pData);
    l_stGPRMI.m_fHeadingAngleStdDev = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[18].m_pData);
    l_stGPRMI.m_fBaselineDistance = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[19].m_pData);
    l_stGPRMI.m_u8SatelliteCount = (HIDO_UINT8)atoi((HIDO_CHAR *)astFields[20].m_pData);
    l_stGPRMI.m_u8ReservedFlags = (HIDO_UINT8)atoi((HIDO_CHAR *)astFields[21].m_pData);
    l_stGPRMI.m_u8PositionQuality = (HIDO_UINT8)atoi((HIDO_CHAR *)astFields[22].m_pData);
    l_stGPRMI.m_fHorizontalVelStdDev = pos_accuracy;
    l_stGPRMI.m_fAccelBiasX = accel_bias_x;
    l_stGPRMI.m_fAccelBiasY = accel_bias_y;
    l_stGPRMI.m_fAccelBiasZ = accel_bias_z;
    l_stGPRMI.m_fGyroBiasX = gyro_bias_x;
    l_stGPRMI.m_fGyroBiasY = gyro_bias_y;
    l_stGPRMI.m_fGyroBiasZ = gyro_bias_z;
    l_stGPRMI.m_fImuTemperature = sensor_temp;
    l_stGPRMI.m_u32StatusFlags = status;
    l_stGPRMI.m_u8PositionQuality = (HIDO_UINT8)(status & 0xFFU);
    l_stGPRMI.m_bValid = HIDO_TRUE;
   // l_u8PosState = l_stGPRMI.m_u8PositionQuality;
    
    return HIDO_OK;
    /* 如果是第一个固定解(解状态=4),初始化ENU原点 */
    if (l_stGeoOrigin.initialized == HIDO_FALSE && l_u8PosState == 4U)
    {
        Geo_OriginInit(&l_stGeoOrigin, l_stGPRMI.m_dLatitude, l_stGPRMI.m_dLongitude, (double)l_stGPRMI.m_fAltitude);
        HIDO_Debug2("[GPS] ENU origin initialized: lat=%.6f, lon=%.6f, alt=%.2f\r\n",
                    l_stGeoOrigin.lat_deg, l_stGeoOrigin.lon_deg, l_stGeoOrigin.alt_m);
    }
    /* 计算当前ENU坐标 */
    if (l_stGeoOrigin.initialized == HIDO_TRUE)
    {
        Geo_GprmiToENU(&l_stGPRMI, &l_stGeoOrigin, l_fCurrentENU);
    }
}
/*******************************************************************************
 * Function Name     : GPS_ParseGPIMU
 * Description       : 解析GPIMU数据包
 * Input             : _pcData GPIMU数据
 *                   : _u32Len GPIMU数据长度
 * Output            : None
 * Return            : HIDO_OK 成功, HIDO_ERR 失败
 * Author            : www.hido-studio.com
 * Modified Date:    : 2025年11月11日
 *******************************************************************************/
static HIDO_INT32 GPS_ParseGPIMU(HIDO_CHAR *_pcData, HIDO_UINT32 _u32Len)
static HIDO_VOID IM23A_HandleImuFrame(const HIDO_UINT8 *frame)
{
    HIDO_DataStruct astFields[9];
    HIDO_UINT8 u8CalcChecksum = 0;
    HIDO_UINT32 i = 0;
    HIDO_CHAR *pCheckStart = HIDO_NULL;
    HIDO_CHAR *pCheckEnd = HIDO_NULL;
    if (IM23A_ValidateFrame(frame, IM23A_IMU_FRAME_LEN) == HIDO_FALSE)
    {
        return;
    }
    const HIDO_UINT8 *p = frame + IM23A_HEADER_LEN;
    double utc = IM23A_ReadDouble(p); p += 8U;
    float accel_x = IM23A_ReadFloat(p); p += 4U;
    float accel_y = IM23A_ReadFloat(p); p += 4U;
    float accel_z = IM23A_ReadFloat(p); p += 4U;
    float gyro_x = IM23A_ReadFloat(p); p += 4U;
    float gyro_y = IM23A_ReadFloat(p); p += 4U;
    float gyro_z = IM23A_ReadFloat(p); p += 4U;
    (void)IM23A_ReadFloat(p); p += 4U;
    (void)IM23A_ReadFloat(p); p += 4U;
    (void)IM23A_ReadFloat(p);
    memset(&l_stGPIMU, 0, sizeof(ST_GPIMU));
    l_stGPIMU.m_bValid = HIDO_FALSE;
    // 解析9个字段: $GPIMU,<1>,<2>,<3>,<4>,<5>,<6>,<7>,<8>,<9>*<CR><LF>
    if (HIDO_UtilParseFormat((HIDO_UINT8 *)_pcData, _u32Len,
        "$GPIMU,%p,%p,%p,%p,%p,%p,%p,%p,%p*%**",
        &astFields[0], &astFields[1], &astFields[2], &astFields[3],
        &astFields[4], &astFields[5], &astFields[6], &astFields[7],
        &astFields[8]) < 10)
    {
        return HIDO_ERR;
    }
    // 计算异或校验: 从第一个字段到第9个字段的异或和
    // 示例数据: $GPIMU, 054752.002, 0.000, 0.007, -1.032, -0.003, 0.053, -0.016,26.00@59
    // 格式说明: 语句示例中的异或校验和为0@59 (十六进制59)
    pCheckStart = strchr(_pcData, ',');  // 找到第一个逗号
    pCheckEnd = strrchr(_pcData, '*');    // 找到最后一个星号
    if (pCheckStart != HIDO_NULL && pCheckEnd != HIDO_NULL && pCheckEnd > pCheckStart)
    {
        // 计算从第一个逗号后到星号前的所有字符的异或
        for (i = 0; pCheckStart + i < pCheckEnd; i++)
        {
            u8CalcChecksum ^= (HIDO_UINT8)(pCheckStart[i]);
        }
    }
    // 解析各字段
    l_stGPIMU.m_u32UTCTime = (HIDO_UINT32)(atof((HIDO_CHAR *)astFields[0].m_pData) * 1000);  // 转换为毫秒
    l_stGPIMU.m_fAccelX = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[1].m_pData);
    l_stGPIMU.m_fAccelY = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[2].m_pData);
    l_stGPIMU.m_fAccelZ = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[3].m_pData);
    l_stGPIMU.m_fGyroX = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[4].m_pData);
    l_stGPIMU.m_fGyroY = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[5].m_pData);
    l_stGPIMU.m_fGyroZ = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[6].m_pData);
    l_stGPIMU.m_fTemperature = (HIDO_FLOAT)atof((HIDO_CHAR *)astFields[7].m_pData);
    // 第9个字段是异或校验,直接解析
    l_stGPIMU.m_u8Checksum = (HIDO_UINT8)atoi((HIDO_CHAR *)astFields[8].m_pData);
    // 验证校验和 (根据文档,这是异或校验)
    // 注意: 文档中显示校验值在最后,需要根据实际协议确定是否验证
    l_stGPIMU.m_u32UTCTime = IM23A_ConvertTime(utc, IM23A_IMU_TIME_SCALE);
    l_stGPIMU.m_fAccelX = accel_x;
    l_stGPIMU.m_fAccelY = accel_y;
    l_stGPIMU.m_fAccelZ = accel_z;
    l_stGPIMU.m_fGyroX = gyro_x;
    l_stGPIMU.m_fGyroY = gyro_y;
    l_stGPIMU.m_fGyroZ = gyro_z;
    l_stGPIMU.m_fTemperature = l_stGPRMI.m_fImuTemperature;
    l_stGPIMU.m_bValid = HIDO_TRUE;
    return HIDO_OK;
}
u16 g_spsum, g_snum;
static HIDO_INT32 GPS_ParseGSV(HIDO_CHAR *_pcData, HIDO_UINT32 _u32Len)
static HIDO_VOID IM23A_HandleGigFrame(const HIDO_UINT8 *frame)
{
    ST_GPS stGPS;
    HIDO_DataStruct spower[4];
    if (IM23A_ValidateFrame(frame, IM23A_GIG_FRAME_LEN) == HIDO_FALSE)
    {
        return;
    }
    memset(&stGPS, 0, sizeof(ST_GPS));
    //    if (GPS_DataCheck(_pcData, _u32Len) != HIDO_OK)
    //    {
    //        return HIDO_ERR;
    //    }
    /* fmig协议解析:解析位置信息、解状态和差分龄期信息 */
    const HIDO_UINT8 *base = frame;
    if (HIDO_UtilParseFormat((HIDO_UINT8 *)_pcData, _u32Len, "$%*,%*,%*,%*,%*,%*,%*,%p,%*,%*,%*,%p,%*,%*,%*,%p,%*,%*,%*,%p,%**", &spower[0], &spower[1], &spower[2], &spower[3]) == 21)
    {
        g_snum += 4;
        g_spsum += atoi((HIDO_CHAR *)spower[0].m_pData) + atoi((HIDO_CHAR *)spower[1].m_pData) + atoi((HIDO_CHAR *)spower[2].m_pData) + atoi((HIDO_CHAR *)spower[3].m_pData);
    }
    else if (HIDO_UtilParseFormat((HIDO_UINT8 *)_pcData, _u32Len, "$%*,%*,%*,%*,%*,%*,%*,%p,%*,%*,%*,%p,%*,%*,%*,%p,%**", &spower[0], &spower[1], &spower[2]) == 17)
    {
        g_snum += 3;
        g_spsum += atoi((HIDO_CHAR *)spower[0].m_pData) + atoi((HIDO_CHAR *)spower[1].m_pData) + atoi((HIDO_CHAR *)spower[2].m_pData);
    }
    else if (HIDO_UtilParseFormat((HIDO_UINT8 *)_pcData, _u32Len, "$%*,%*,%*,%*,%*,%*,%*,%p,%*,%*,%*,%p,%**", &spower[0], &spower[1]) == 13)
    {
        g_snum += 2;
        g_spsum += atoi((HIDO_CHAR *)spower[0].m_pData) + atoi((HIDO_CHAR *)spower[1].m_pData);
    }
    else if (HIDO_UtilParseFormat((HIDO_UINT8 *)_pcData, _u32Len, "$%*,%*,%*,%*,%*,%*,%*,%p,%**", &spower[0]) == 9)
    {
        g_snum += 1;
        g_spsum += atoi((HIDO_CHAR *)spower[0].m_pData);
    }
    // l_u8PosState = atoi((HIDO_CHAR *)stPosState.m_pData);
    /* 字段1: 时间 hhmmss.ss (double, 位置5) */
    double utc = IM23A_ReadDouble(&base[4]);
    return HIDO_OK;
    /* 字段2: 纬度 (double, 位置13) */
    double latitude = IM23A_ReadDouble(&base[12]);
    /* 字段3: 经度 (double, 位置21) */
    double longitude = IM23A_ReadDouble(&base[20]);
    /* 字段14: HRMS (float, 位置73) */
    float hrms = IM23A_ReadFloat(&base[72]);
    /* 字段15: VRMS (float, 位置77) */
    float vrms = IM23A_ReadFloat(&base[76]);
    /* 字段16: HDOP (float, 位置81) */
    float hdop = IM23A_ReadFloat(&base[80]);
    /* 字段17: VDOP (float, 位置85) */
    float vdop = IM23A_ReadFloat(&base[84]);
    /* 字段18: 卫星数量 (uint8_t, 位置89) */
    HIDO_UINT8 sat_count = base[88];
    /* 字段19: 解状态 (uint8_t, 位置90) */
    HIDO_UINT8 sol_status = base[89];
    /* 字段20: 差分龄期 (uint8_t, 位置91) */
    HIDO_UINT8 diff_age = base[90];
    /* 更新GPGIG结构体,包含位置信息 */
    memset(&l_stGPGIG, 0, sizeof(ST_GPGIG));
    l_stGPGIG.m_u32UTCTime = IM23A_ConvertTime(utc, IM23A_GPS_TIME_SCALE);
    l_stGPGIG.m_dLatitude = latitude;
    l_stGPGIG.m_dLongitude = longitude;
    l_stGPGIG.m_u8SatelliteCount = sat_count;
    l_stGPGIG.m_u8SolutionStatus = sol_status;
    l_stGPGIG.m_u8DifferentialAge = diff_age;
    l_stGPGIG.m_fHRMS = hrms;
    l_stGPGIG.m_fVRMS = vrms;
    l_stGPGIG.m_fHDOP = hdop;
    l_stGPGIG.m_fVDOP = vdop;
    l_stGPGIG.m_bValid = HIDO_TRUE;
    /* 更新全局解状态(用于兼容旧代码) */
    l_u8PosState = sol_status;
}
/*******************************************************************************
 * Function Name     : GPS_RecvFsm
 * Description       : GPS ���ݽ���״̬��
 * Input             : _u8RecvChar һ�������ַ�
 * Output            : None
 * Return            : one
 * Author            : www.hido-studio.com
 * Modified Date:    : 2021��5��07��
 *******************************************************************************/
static HIDO_VOID GPS_RecvFsm(HIDO_UINT8 _u8RecvChar)
{
    static int LastRTK = 0;
    ST_GPSRecv *parser = &l_stGPSRecv;
    HIDO_CHAR GPS_1[10] = "�豸����";
    HIDO_CHAR GPS_4[10] = "�豸�̶�";
    HIDO_CHAR GPS_5[10] = "�豸����";
    switch (l_stGPSRecv.m_eState)
    if (parser->m_u32ExpectedLen == 0U)
    {
    case GPS_RECV_STATE_IDLE:
    {
        if ('$' == _u8RecvChar)
        if (parser->m_u32RecvLen == 0U)
        {
            l_stGPSRecv.m_eState = GPS_RECV_STATE_HEAD;
            l_stGPSRecv.m_u32RecvLen = 0;
            l_stGPSRecv.m_acRecvBuf[l_stGPSRecv.m_u32RecvLen++] = _u8RecvChar;
            if (_u8RecvChar == 'f')
            {
                parser->m_au8Buffer[parser->m_u32RecvLen++] = _u8RecvChar;
            }
            return;
        }
        break;
    }
    case GPS_RECV_STATE_HEAD:
    {
        l_stGPSRecv.m_acRecvBuf[l_stGPSRecv.m_u32RecvLen++] = _u8RecvChar;
        if (l_stGPSRecv.m_u32RecvLen >= l_stGPSRecv.m_u32HeaderLen)
        parser->m_au8Buffer[parser->m_u32RecvLen++] = _u8RecvChar;
        if (parser->m_u32RecvLen == 2U && _u8RecvChar != 'm')
        {
            if (memcmp(l_stGPSRecv.m_acRecvBuf, l_stGPSRecv.m_acHeader, l_stGPSRecv.m_u32HeaderLen) == 0)
            if (_u8RecvChar == 'f')
            {
                l_stGPSRecv.m_eState = GPS_RECV_STATE_CR;
            }
            else if (memcmp(l_stGPSRecv.m_acRecvBuf, l_stGPSRecv.m_acHeader2, l_stGPSRecv.m_u32Header2Len) == 0)
            {
                l_stGPSRecv.m_eState = GPS_RECV_STATE_CR;
            }
            else if (memcmp(l_stGPSRecv.m_acRecvBuf, l_stGPSRecv.m_acHeader3, l_stGPSRecv.m_u32Header3Len) == 0)
            {
                l_stGPSRecv.m_eState = GPS_RECV_STATE_CR;
            }
            else if (memcmp(l_stGPSRecv.m_acRecvBuf, l_stGPSRecv.m_acHeader4, l_stGPSRecv.m_u32Header4Len) == 0)
            {
                l_stGPSRecv.m_eState = GPS_RECV_STATE_CR;
            }
            else if (memcmp(l_stGPSRecv.m_acRecvBuf, l_stGPSRecv.m_acHeader5, l_stGPSRecv.m_u32Header5Len) == 0)
            {
                l_stGPSRecv.m_eState = GPS_RECV_STATE_CR;
                parser->m_au8Buffer[0] = 'f';
                parser->m_u32RecvLen = 1U;
            }
            else
            {
                l_stGPSRecv.m_eState = GPS_RECV_STATE_IDLE;
                IM23A_ResetParser(parser);
            }
            return;
        }
        break;
    }
    case GPS_RECV_STATE_CR:
    {
        l_stGPSRecv.m_acRecvBuf[l_stGPSRecv.m_u32RecvLen++] = _u8RecvChar;
        if (l_stGPSRecv.m_u32RecvLen >= (sizeof(l_stGPSRecv.m_acRecvBuf) - 1))
        if (parser->m_u32RecvLen == 3U && _u8RecvChar != 'i')
        {
            l_stGPSRecv.m_eState = GPS_RECV_STATE_IDLE;
            break;
            if (_u8RecvChar == 'f')
            {
                parser->m_au8Buffer[0] = 'f';
                parser->m_u32RecvLen = 1U;
            }
            else
            {
                IM23A_ResetParser(parser);
            }
            return;
        }
        if ('\r' == _u8RecvChar)
        if (parser->m_u32RecvLen == IM23A_HEADER_LEN)
        {
            l_stGPSRecv.m_eState = GPS_RECV_STATE_LF;
        }
        break;
    }
    case GPS_RECV_STATE_LF:
    {
        static u8 gpsled_state = 0;
        l_stGPSRecv.m_acRecvBuf[l_stGPSRecv.m_u32RecvLen++] = _u8RecvChar;
        if ('\n' == _u8RecvChar)
        {
            /* ����GPS HZ */
            //                g_com_map[GPS_HZ] = 10;
            //if (l_bGPSConfig || l_u8GPS_HZ != g_com_map[GPS_HZ])
            if (memcmp(parser->m_au8Buffer, IM23A_NAV_HEADER, IM23A_HEADER_LEN) == 0)
            {
                l_bGPSConfig = HIDO_FALSE;
                //GPS_RateConfig(g_com_map[GPS_HZ]);
                parser->m_u32ExpectedLen = IM23A_NAV_FRAME_LEN;
            }
            if (memcmp(l_stGPSRecv.m_acRecvBuf, l_stGPSRecv.m_acHeader3, l_stGPSRecv.m_u32Header3Len) == 0)
            else if (memcmp(parser->m_au8Buffer, IM23A_IMU_HEADER, IM23A_HEADER_LEN) == 0)
            {
                GPS_ParseGSV(l_stGPSRecv.m_acRecvBuf, l_stGPSRecv.m_u32RecvLen);
                parser->m_u32ExpectedLen = IM23A_IMU_FRAME_LEN;
            }
            else if (memcmp(l_stGPSRecv.m_acRecvBuf, l_stGPSRecv.m_acHeader4, l_stGPSRecv.m_u32Header4Len) == 0)
            else if (memcmp(parser->m_au8Buffer, IM23A_GIG_HEADER, IM23A_HEADER_LEN) == 0)
            {
                // 解析GPRMI数据
                GPS_ParseGPRMI(l_stGPSRecv.m_acRecvBuf, l_stGPSRecv.m_u32RecvLen);
                parser->m_u32ExpectedLen = IM23A_GIG_FRAME_LEN;
            }
            else if (memcmp(l_stGPSRecv.m_acRecvBuf, l_stGPSRecv.m_acHeader5, l_stGPSRecv.m_u32Header5Len) == 0)
            else
            {
                // 解析GPIMU数据
                GPS_ParseGPIMU(l_stGPSRecv.m_acRecvBuf, l_stGPSRecv.m_u32RecvLen);
            }
            if (memcmp(l_stGPSRecv.m_acRecvBuf, l_stGPSRecv.m_acHeader, l_stGPSRecv.m_u32HeaderLen) == 0)
            {
                static uint32_t gps_uploadtimer = 0;
                GPS_ParseGGA(l_stGPSRecv.m_acRecvBuf, l_stGPSRecv.m_u32RecvLen);
                l_stGPSRecv.m_acRecvBuf[l_stGPSRecv.m_u32RecvLen - 2] = '\0';
                //if (HIDO_TimerGetTick() - gps_uploadtimer > g_com_map[GPSUPLOADTIME_INDEX] * 1000)
                HIDO_UINT8 restart = parser->m_au8Buffer[IM23A_HEADER_LEN - 1U];
                IM23A_ResetParser(parser);
                if (restart == 'f')
                {
                    gps_uploadtimer = HIDO_TimerGetTick();
                    UDPClient_UploadGPS(l_stGPSRecv.m_acRecvBuf);
                    parser->m_au8Buffer[0] = 'f';
                    parser->m_u32RecvLen = 1U;
                }
            }
            if ((HIDO_TimerGetTick() - l_u32QXTick) >= 1000)
            {
                l_u32QXTick = HIDO_TimerGetTick();
#ifdef __USE_QXWZ__
                qxwz_app_upload_gga(l_stGPSRecv.m_acRecvBuf);
#endif
            }
        }
        return;
    }
        l_stGPSRecv.m_eState = GPS_RECV_STATE_IDLE;
        break;
    }
    default:
    if (parser->m_u32RecvLen >= IM23A_MAX_FRAME_LEN)
    {
        break;
        IM23A_ResetParser(parser);
        return;
    }
    parser->m_au8Buffer[parser->m_u32RecvLen++] = _u8RecvChar;
    if (parser->m_u32RecvLen == parser->m_u32ExpectedLen)
    {
        IM23A_HandleFrame(parser->m_au8Buffer, parser->m_u32ExpectedLen);
        IM23A_ResetParser(parser);
    }
}
@@ -569,61 +511,15 @@
/*******************************************************************************
 * Function Name     : GPS_Poll
 * Description       : GPS��ѯ����
 * Input             : None
 * Output            : None
 * Return            : None
 * Author            : www.hido-studio.com
 * Modified Date:    : 2021��1��8��
 * Description       : GPS轮询,读取UART数据并送入IM23A解析器
 *******************************************************************************/
HIDO_VOID GPS_Poll(void)
{
    static HIDO_UINT8 l_u8GPSBuff[512];
    static HIDO_UINT32 l_u8GPSLen = 0;
    static HIDO_UINT32 l_u8GPSRecvTick = 0;
    HIDO_UINT8 u8RecvChar = 0U;
    volatile HIDO_INT32 i32Result = HIDO_OK;
    HIDO_UINT8 u8RecvChar = 0;
    UART_HandleTypeDef *pstUartHandle = HIDO_NULL;
    Uart_GetHandle(UART_ID_DBG, (HIDO_VOID **)&pstUartHandle);
    while ((i32Result = Uart_GetChar(UART_ID_GPS, &u8RecvChar)) == HIDO_OK)
    while (Uart_GetChar(UART_ID_GPS, &u8RecvChar) == HIDO_OK)
    {
#if 0
        if (DBG_GetMode() == DBG_MODE_GPS || DBG_GetMode() == DBG_MODE_CFG)
        {
            HAL_UART_Transmit(pstUartHandle, &u8RecvChar, 1, 1000);
        }
#endif
        GPS_RecvFsm(u8RecvChar);
        l_u8GPSBuff[l_u8GPSLen++] = u8RecvChar;
        if (l_u8GPSLen >= sizeof(l_u8GPSBuff))
        {
#ifdef __USE_TCP_RTCM__
            RTKClient_ReportData(l_u8GPSBuff, l_u8GPSLen);
#endif
#ifdef __USE_NTRIP__
            NTRIPApp_ReportGGA(l_u8GPSBuff, l_u8GPSLen);
#endif
            l_u8GPSLen = 0;
        }
        l_u8GPSRecvTick = HIDO_TimerGetTick();
    }
    if (l_u8GPSLen > 0)
    {
        if ((HIDO_TimerGetTick() - l_u8GPSRecvTick) > 50)
        {
#ifdef __USE_TCP_RTCM__
            RTKClient_ReportData(l_u8GPSBuff, l_u8GPSLen);
#endif
#ifdef __USE_NTRIP__
            NTRIPApp_ReportGGA(l_u8GPSBuff, l_u8GPSLen);
#endif
            l_u8GPSLen = 0;
        }
    }
}
@@ -638,7 +534,6 @@
 *******************************************************************************/
HIDO_VOID GPS_Init(void)
{
    uint32_t gpsbaudrate;
    GPS_PowerOn();
    GPS_Rest();
@@ -653,19 +548,20 @@
    stInit.m_u32TxQueueMemberCnt = 2;
    Uart_Init(UART_ID_GPS, &stInit);
    HIDO_UtilBzero(&l_stGPSRecv, sizeof(ST_GPSRecv));
    l_stGPSRecv.m_u32HeaderLen = HIDO_UtilSnprintf(l_stGPSRecv.m_acHeader, sizeof(l_stGPSRecv.m_acHeader), "$GNGGA");
    l_stGPSRecv.m_u32Header2Len = HIDO_UtilSnprintf(l_stGPSRecv.m_acHeader2, sizeof(l_stGPSRecv.m_acHeader2), "$GPGSV");
    l_stGPSRecv.m_u32Header3Len = HIDO_UtilSnprintf(l_stGPSRecv.m_acHeader3, sizeof(l_stGPSRecv.m_acHeader3), "$GBGSV");
    l_stGPSRecv.m_u32Header4Len = HIDO_UtilSnprintf(l_stGPSRecv.m_acHeader4, sizeof(l_stGPSRecv.m_acHeader4), "$GPFMI");
    l_stGPSRecv.m_u32Header5Len = HIDO_UtilSnprintf(l_stGPSRecv.m_acHeader5, sizeof(l_stGPSRecv.m_acHeader5), "$GPIMU");
    l_bGPSConfig = HIDO_TRUE;
    //gpsbaudrate = (g_com_map[GPSBAUDRATE1_INDEX] << 8) | g_com_map[GPSBAUDRATE2_INDEX];
    if (gpsbaudrate > 921600 || gpsbaudrate < 9600)
    {
        gpsbaudrate = 115200;
    }
    Uart_ReConfigBaudRate(UART_ID_GPS, gpsbaudrate);
    IM23A_ResetParser(&l_stGPSRecv);
    Uart_ReConfigBaudRate(UART_ID_GPS, 115200);
    /* 延时等待GPS模块启动完成 */
    vTaskDelay(pdMS_TO_TICKS(1000));
    /* 发送AT命令配置GPS模块输出到UART2 */
    const HIDO_CHAR *at_cmd = "AT+GNSS_OUTPUT=UART2,ON\r\n";
    Uart_Send(UART_ID_GPS, (HIDO_UINT8 *)at_cmd, strlen(at_cmd));
    vTaskDelay(pdMS_TO_TICKS(100));
     const HIDO_CHAR *at_cmd2 = "AT+SAVE_ALL\r\n";
    Uart_Send(UART_ID_GPS, (HIDO_UINT8 *)at_cmd2, strlen(at_cmd2));
    /* 等待GPS模块响应 */
    vTaskDelay(pdMS_TO_TICKS(100));
}
/*******************************************************************************
@@ -720,3 +616,231 @@
    return HIDO_OK;
}
/*******************************************************************************
 * Function Name     : GPS_GetCurrentENU
 * Description       : 获取当前ENU坐标(相对于开机后第一个固定解)
 * Input             : _enu - 存储ENU坐标的数组指针 [东, 北, 天]
 * Output            : None
 * Return            : HIDO_OK - 成功, HIDO_ERR - 失败或原点未初始化
 * Author            : www.hido-studio.com
 * Modified Date:    : 2025年12月3日
 *******************************************************************************/
HIDO_INT32 GPS_GetCurrentENU(float _enu[3])
{
    if (_enu == HIDO_NULL)
    {
        return HIDO_ERR;
    }
    if (l_stGeoOrigin.initialized == HIDO_FALSE)
    {
        _enu[0] = 0.0f;
        _enu[1] = 0.0f;
        _enu[2] = 0.0f;
        return HIDO_ERR;
    }
    _enu[0] = l_fCurrentENU[0];
    _enu[1] = l_fCurrentENU[1];
    _enu[2] = l_fCurrentENU[2];
    return HIDO_OK;
}
/*******************************************************************************
 * Function Name     : GPS_GetGPGIG
 * Description       : 获取最新的GPGIG数据(fmig报文,解状态和差分龄期)
 * Input             : _pstGPGIG - 存储GPGIG数据的结构体指针
 * Output            : None
 * Return            : HIDO_OK - 成功, HIDO_ERR - 失败或数据无效
 * Author            : www.hido-studio.com
 * Modified Date:    : 2025年12月3日
 *******************************************************************************/
HIDO_INT32 GPS_GetGPGIG(ST_GPGIG *_pstGPGIG)
{
    if (_pstGPGIG == HIDO_NULL)
    {
        return HIDO_ERR;
    }
    if (l_stGPGIG.m_bValid == HIDO_FALSE)
    {
        return HIDO_ERR;
    }
    memcpy(_pstGPGIG, &l_stGPGIG, sizeof(ST_GPGIG));
    return HIDO_OK;
}
/*******************************************************************************
 * Function Name     : GPS_ConvertLatToDDMM
 * Description       : 将纬度从度数格式转换为度分格式
 * Input             : _dLatDeg - 纬度(度),正数为北纬,负数为南纬
 *                   : _pcSign - 输出符号字符 'N' 或 'S'
 * Output            : _pcSign
 * Return            : 度分格式值 (ddmm.mmmm)
 * Author            : www.hido-studio.com
 * Modified Date:    : 2025年12月4日
 *******************************************************************************/
static HIDO_DOUBLE GPS_ConvertLatToDDMM(HIDO_DOUBLE _dLatDeg, HIDO_CHAR *_pcSign)
{
    HIDO_DOUBLE absLat = (_dLatDeg < 0.0) ? -_dLatDeg : _dLatDeg;
    *_pcSign = (_dLatDeg >= 0.0) ? 'N' : 'S';
    HIDO_INT32 degrees = (HIDO_INT32)absLat;
    HIDO_DOUBLE minutes = (absLat - degrees) * 60.0;
    return (HIDO_DOUBLE)degrees * 100.0 + minutes;
}
/*******************************************************************************
 * Function Name     : GPS_ConvertLonToDDDMM
 * Description       : 将经度从度数格式转换为度分格式
 * Input             : _dLonDeg - 经度(度),正数为东经,负数为西经
 *                   : _pcSign - 输出符号字符 'E' 或 'W'
 * Output            : _pcSign
 * Return            : 度分格式值 (dddmm.mmmm)
 * Author            : www.hido-studio.com
 * Modified Date:    : 2025年12月4日
 *******************************************************************************/
static HIDO_DOUBLE GPS_ConvertLonToDDDMM(HIDO_DOUBLE _dLonDeg, HIDO_CHAR *_pcSign)
{
    HIDO_DOUBLE absLon = (_dLonDeg < 0.0) ? -_dLonDeg : _dLonDeg;
    *_pcSign = (_dLonDeg >= 0.0) ? 'E' : 'W';
    HIDO_INT32 degrees = (HIDO_INT32)absLon;
    HIDO_DOUBLE minutes = (absLon - degrees) * 60.0;
    return (HIDO_DOUBLE)degrees * 100.0 + minutes;
}
/*******************************************************************************
 * Function Name     : GPS_CalculateNMEAChecksum
 * Description       : 计算NMEA协议校验和(XOR)
 * Input             : _pcData - 数据字符串(不包含$和*)
 *                   : _u32Len - 数据长度
 * Output            : None
 * Return            : 校验和(0-255)
 * Author            : www.hido-studio.com
 * Modified Date:    : 2025年12月4日
 *******************************************************************************/
static HIDO_UINT8 GPS_CalculateNMEAChecksum(const HIDO_CHAR *_pcData, HIDO_UINT32 _u32Len)
{
    HIDO_UINT8 checksum = 0U;
    for (HIDO_UINT32 i = 0U; i < _u32Len; i++)
    {
        checksum ^= (HIDO_UINT8)_pcData[i];
    }
    return checksum;
}
/*******************************************************************************
 * Function Name     : GPS_FormatGGA
 * Description       : 将fmin和fmig数据格式化为标准GGA报文(经纬度优先使用fmig数据)
 * Input             : _pstGPRMI - fmin数据
 *                   : _pstGPGIG - fmig数据(包含经纬度信息)
 *                   : _pcBuffer - 输出缓冲区
 *                   : _u32BufferSize - 缓冲区大小
 * Output            : _pcBuffer
 * Return            : 实际写入的字符数,失败返回0
 * Author            : www.hido-studio.com
 * Modified Date:    : 2025年12月4日
 *******************************************************************************/
HIDO_UINT32 GPS_FormatGGA(const ST_GPRMI *_pstGPRMI, const ST_GPGIG *_pstGPGIG,
                                  HIDO_CHAR *_pcBuffer, HIDO_UINT32 _u32BufferSize)
{
    if (_pstGPRMI == HIDO_NULL || _pstGPGIG == HIDO_NULL || _pcBuffer == HIDO_NULL)
    {
        return 0U;
    }
    /* 转换UTC时间:厘秒(hhmmss.ss) → 字符串格式 */
    HIDO_UINT32 utc = _pstGPRMI->m_u32UTCTime;
    HIDO_UINT32 hh = utc / 1000000U;
    HIDO_UINT32 mm = (utc / 10000U) % 100U;
    HIDO_UINT32 ss = (utc / 100U) % 100U;
    HIDO_UINT32 cs = utc % 100U;
    /* 转换经纬度:优先使用fmig数据,度数 → 度分格式 */
    HIDO_CHAR latSign, lonSign;
    HIDO_DOUBLE latitude, longitude;
    /* 优先使用fmig中的经纬度信息 */
    if (_pstGPGIG->m_bValid && _pstGPGIG->m_dLatitude != 0.0 && _pstGPGIG->m_dLongitude != 0.0)
    {
        latitude = _pstGPGIG->m_dLatitude;
        longitude = _pstGPGIG->m_dLongitude;
    }
    else
    {
        /* fmig无效时回退到fmin数据 */
        latitude = _pstGPRMI->m_dLatitude;
        longitude = _pstGPRMI->m_dLongitude;
    }
    HIDO_DOUBLE latDDMM = GPS_ConvertLatToDDMM(latitude, &latSign);
    HIDO_DOUBLE lonDDDMM = GPS_ConvertLonToDDDMM(longitude, &lonSign);
    /* 构建GGA报文主体(不含$和校验和) */
    HIDO_CHAR ggaBody[256];
    HIDO_INT32 bodyLen = snprintf(ggaBody, sizeof(ggaBody),
        "GNGGA,%02u%02u%02u.%02u,%013.8f,%c,%014.8f,%c,%u,%02u,%.1f,%.1f,M,0.0,M,%.1f,",
        hh, mm, ss, cs,                           /* UTC时间 */
        latDDMM, latSign,                         /* 纬度 (8位小数) */
        lonDDDMM, lonSign,                        /* 经度 (8位小数) */
        (HIDO_UINT32)_pstGPGIG->m_u8SolutionStatus,  /* 定位质量 */
        (HIDO_UINT32)_pstGPGIG->m_u8SatelliteCount,  /* 卫星数量 */
        (double)_pstGPGIG->m_fHDOP,               /* HDOP */
        (double)_pstGPRMI->m_fAltitude,           /* 海拔 */
        (double)_pstGPGIG->m_u8DifferentialAge    /* 差分龄期 */
    );
    if (bodyLen <= 0 || bodyLen >= (HIDO_INT32)sizeof(ggaBody))
    {
        return 0U;
    }
    /* 计算校验和 */
    HIDO_UINT8 checksum = GPS_CalculateNMEAChecksum(ggaBody, (HIDO_UINT32)bodyLen);
    /* 组装完整GGA报文:$GPGGA,...*CS */
    HIDO_INT32 totalLen = snprintf(_pcBuffer, _u32BufferSize, "$%s*%02X", ggaBody, checksum);
    if (totalLen <= 0 || totalLen >= (HIDO_INT32)_u32BufferSize)
    {
        return 0U;
    }
    return (HIDO_UINT32)totalLen;
}
/*******************************************************************************
 * Function Name     : GPS_UploadGGA
 * Description       : 每秒上传一次GGA报文到4G(拼接fmin和fmig数据)
 * Input             : None
 * Output            : None
 * Return            : None
 * Author            : www.hido-studio.com
 * Modified Date:    : 2025年12月4日
 *******************************************************************************/
HIDO_VOID GPS_UploadGGA(HIDO_VOID)
{
    static HIDO_CHAR ggaBuffer[256];
    /* 检查fmin和fmig数据有效性 */
    if (l_stGPRMI.m_bValid == HIDO_FALSE || l_stGPGIG.m_bValid == HIDO_FALSE)
    {
        return;
    }
    /* 格式化GGA报文 */
    HIDO_UINT32 ggaLen = GPS_FormatGGA(&l_stGPRMI, &l_stGPGIG, ggaBuffer, sizeof(ggaBuffer));
    if (ggaLen > 0U)
    {
        /* 调用4G上传函数 */
        UDPClient_UploadGPS(ggaBuffer);
    }
}