blob: 00afd57de5bc0bc9290b0dea8d7886f1b2006adb [file] [log] [blame]
b.liu68a94c92025-05-24 12:53:41 +08001#include <stdio.h>
2#include <string.h>
3#include <strings.h>
4#include <stdlib.h>
5#include <errno.h>
6#include <fcntl.h>
7#include <signal.h>
8#include <sys/types.h>
9#include <unistd.h>
10#include <pthread.h>
11#include <termios.h>
12#include <time.h>
13#include <sys/ioctl.h>
14#include <dlfcn.h>
15#include <stdint.h>
16#include <stdbool.h>
17
zw.wang581aab12025-05-28 19:43:53 +080018#include "gsw_gnss_interface.h"
19
20#define MBTK_GNSS_IND_LOCATION (1) // 1
21#define MBTK_GNSS_IND_NMEA (1 << 1) // 2
22#define QSER_GNSS_TIMEOUT 5
b.liu68a94c92025-05-24 12:53:41 +080023
24#ifndef LOG_ERR_LEVEL
25#define LOG_ERR_LEVEL 3 /* error conditions */
26#endif
27#ifndef LOG_WARN_LEVEL
28#define LOG_WARN_LEVEL 4 /* warning conditions */
29#endif
30#ifndef LOG_INFO_LEVEL
31#define LOG_INFO_LEVEL 6 /* informational */
32#endif
33#ifndef LOG_DEBUG_LEVEL
34#define LOG_DEBUG_LEVEL 7 /* debug-level messages */
35#endif
36#ifndef LOG_VERBOSE_LEVEL
37#define LOG_VERBOSE_LEVEL 8
38#endif
39
l.yang6a42e4d2025-05-28 01:04:20 -070040#define GSW_GNSS "[HAL][GSW_GNSS]"
41
42
b.liu68a94c92025-05-24 12:53:41 +080043#define LOGV(fmt, args ...) \
44 do{ \
45 char *file_ptr_1001 = __FILE__; \
46 char *ptr_1001 = file_ptr_1001 + strlen(file_ptr_1001) - 1; \
47 char line_1001[10] = {0}; \
48 sprintf(line_1001, "%d", __LINE__); \
49 while(ptr_1001 >= file_ptr_1001 && *ptr_1001){ \
50 if(*ptr_1001 == '/') \
51 break; \
52 ptr_1001--; \
53 } \
l.yang6a42e4d2025-05-28 01:04:20 -070054 fun_ptr_log(LOG_VERBOSE_LEVEL, "%s#%s: "GSW_GNSS"" fmt, ptr_1001 + 1, line_1001, ##args); \
b.liu68a94c92025-05-24 12:53:41 +080055 } while(0)
56
57#define LOGI(fmt, args...) \
58 do{ \
59 char *file_ptr_1001 = __FILE__; \
60 char *ptr_1001 = file_ptr_1001 + strlen(file_ptr_1001) - 1; \
61 char line_1001[10] = {0}; \
62 sprintf(line_1001, "%d", __LINE__); \
63 while(ptr_1001 >= file_ptr_1001 && *ptr_1001){ \
64 if(*ptr_1001 == '/') \
65 break; \
66 ptr_1001--; \
67 } \
l.yang6a42e4d2025-05-28 01:04:20 -070068 fun_ptr_log(LOG_INFO_LEVEL, "%s#%s: "GSW_GNSS"" fmt, ptr_1001 + 1, line_1001, ##args); \
b.liu68a94c92025-05-24 12:53:41 +080069 } while(0)
70
71#define LOGD(fmt, args...) \
72 do{ \
73 char *file_ptr_1001 = __FILE__; \
74 char *ptr_1001 = file_ptr_1001 + strlen(file_ptr_1001) - 1; \
75 char line_1001[10] = {0}; \
76 sprintf(line_1001, "%d", __LINE__); \
77 while(ptr_1001 >= file_ptr_1001 && *ptr_1001){ \
78 if(*ptr_1001 == '/') \
79 break; \
80 ptr_1001--; \
81 } \
l.yang6a42e4d2025-05-28 01:04:20 -070082 fun_ptr_log(LOG_DEBUG_LEVEL, "%s#%s: "GSW_GNSS"" fmt, ptr_1001 + 1, line_1001, ##args); \
b.liu68a94c92025-05-24 12:53:41 +080083 } while(0)
84
85#define LOGW(fmt, args...) \
86 do{ \
87 char *file_ptr_1001 = __FILE__; \
88 char *ptr_1001 = file_ptr_1001 + strlen(file_ptr_1001) - 1; \
89 char line_1001[10] = {0}; \
90 sprintf(line_1001, "%d", __LINE__); \
91 while(ptr_1001 >= file_ptr_1001 && *ptr_1001){ \
92 if(*ptr_1001 == '/') \
93 break; \
94 ptr_1001--; \
95 } \
l.yang6a42e4d2025-05-28 01:04:20 -070096 fun_ptr_log(LOG_WARN_LEVEL, "%s#%s: "GSW_GNSS"" fmt, ptr_1001 + 1, line_1001, ##args); \
b.liu68a94c92025-05-24 12:53:41 +080097 } while(0)
98
99#define LOGE(fmt, args...) \
100 do{ \
101 char *file_ptr_1001 = __FILE__; \
102 char *ptr_1001 = file_ptr_1001 + strlen(file_ptr_1001) - 1; \
103 char line_1001[10] = {0}; \
104 sprintf(line_1001, "%d", __LINE__); \
105 while(ptr_1001 >= file_ptr_1001 && *ptr_1001){ \
106 if(*ptr_1001 == '/') \
107 break; \
108 ptr_1001--; \
109 } \
l.yang6a42e4d2025-05-28 01:04:20 -0700110 fun_ptr_log(LOG_ERR_LEVEL, "%s#%s: "GSW_GNSS"" fmt, ptr_1001 + 1, line_1001, ##args); \
b.liu68a94c92025-05-24 12:53:41 +0800111 } while(0)
b.liu68a94c92025-05-24 12:53:41 +0800112
113#define MOPEN_GNSS_NMEA_MAX_LENGTH 255 /** NMEA string maximum length. */
114static gsw_gnss_cb *gsw_cb = NULL;
115static bool inited = false;
116static bool strated = false;
117
118typedef void (*mbtk_gnss_callback_func)(uint32_t ind_type, const void* data, uint32_t data_len);
119typedef void (*mbtk_log)(int level, const char *format,...);
120static mbtk_log fun_ptr_log = NULL;
121
122int (*mbtk_gnss_init)(mbtk_gnss_callback_func cb);
123int (*mbtk_gnss_deinit)();
124
125int (*mbtk_gnss_ind_set)(uint32_t ,int);
126int (*mbtk_gnss_open)(int, int);
127int (*mbtk_gnss_close)(int);
128int (*mbtk_gnss_setting)(const char *setting_cmd, int);
129
130int (*mbtk_gnss_eph_download)(int);
131int (*mbtk_gnss_eph_inject)(int);
132
133int gnss_freq = -1;
134GSW_GNSS_MODE_CONFIGURATION gnss_startmode = -1;
zw.wang581aab12025-05-28 19:43:53 +0800135GSW_CONF_SWITCH gnss_switch_op = -1;
b.liu68a94c92025-05-24 12:53:41 +0800136void *dlHandle_gnss;
137char *lynqLib_gnss = "/lib/libmbtk_lib.so";
138
139typedef enum
140{
141 E_MT_LOC_MSG_ID_LOCATION_INFO = 1, /**< pv_data = & mopen_location_info_t */
142 E_MT_LOC_MSG_ID_NMEA_INFO = 3, /**< pv_data = & mopen_gnss_nmea_info_t */
143} e_msg_id_t;
144
145typedef struct
146{
147 int64_t timestamp; /**< System Timestamp, marked for when got the nmea data */
148 int length; /**< NMEA string length. */
149 char nmea[MOPEN_GNSS_NMEA_MAX_LENGTH + 1]; /**< NMEA string.*/
150}mopen_gnss_nmea_info_t; /* Message */
151
152typedef struct
153{
154 uint32_t size; /**< Set to the size of mcm_gps_location_t. */
155 int flags; /**< Contains GPS location flags bits. */
156 int position_source; /**< Provider indicator for HYBRID or GPS. */ //功能暂未实现,可不用添加进结构体
157 double latitude; /**< Latitude in degrees. */
158 double longitude; /**< Longitude in degrees. */
159 double altitude; /**< Altitude in meters above the WGS 84 reference ellipsoid. */
160 float speed; /**< Speed in meters per second. */
161 float bearing; /**< Heading in degrees. */ //功能暂未实现,可不用添加进结构体
162 float accuracy; /**< Expected accuracy in meters. */ //功能暂未实现,可不用添加进结构体
163 int64_t timestamp; /**< Timestamp for the location fix in UTC million-second base. */
164 int32_t is_indoor; /**< Location is indoors. */ //功能暂未实现,可不用添加进结构体
165 float floor_number; /**< Indicates the floor number. */
166}mopen_location_info_t;//功能暂未实现,可不用添加进结构体
167
168
169typedef struct {
170 uint32_t flags;
171 double latitude; /**< Latitude in degrees. */
172 double longitude; /**< Longitude in degrees. */
173 double altitude; /**< Altitude in meters above the WGS 84 reference ellipsoid. */
174 float speed; /**< Speed in meters per second. */
175 float bearing; /**< Heading in degrees. */
176 int64_t timestamp; /**< Timestamp for the location fix in UTC million-second base. */
177} mbtk_gnss_location_info_t;
178
zw.wang581aab12025-05-28 19:43:53 +0800179typedef struct{
180 GSW_GNSS_MODE_CONFIGURATION start_mode;
181 gnss_freq_type freq;
182 gsw_gnss_cb callback;
183}gsw_gnss_init_configure_t;
184
b.liu68a94c92025-05-24 12:53:41 +0800185typedef enum{
186 MODE_GPS = 1, /**< GPS only */
187 MODE_BEIDOU, /**< BEIDOU only*/
188 MODE_GPS_BEIDOU, /**< GPS+BEIDOU */
189 MODE_GLONASS, /**< GLONASS only */
190 MODE_GPS_GLONASS, /**< GPS+GLONASS */
191 MODE_GLONASS_BEIDOU, /**< GLONASS+BEIDOU */ /* The type does not support this type */
192 MODE_GPS_GLONASS_BEIDOU, /**< GPS+GLONASS+BEIDOU */ /* The type does not support this type */
193 MODE_GALILEO, /**< GALILEO only */
194 MODE_GPS_GALILEO, /**< GPS+GALILEO */
195 MODE_BEIDOU_GALILEO, /**< BEIDOU+GALILEO */
196 MODE_GPS_BEIDOU_GALILEO, /**< GPS+BEIDOU+GALILEO */
197 MODE_GLONASS_GALILEO, /**< GLONASS+GALILEO */
198 MODE_GPS_GLONASS_GALILEO, /**< GPS+GLONASS+GALILEO */
199 MODE_BEIDOU_GLONASS_GALILEO, /**< BEIDOU+GLONASS+GALILEO */ /* The type does not support this type */
200 MODE_END = -1, /**< init value */
201}GSW_GNSS_MODE_CONFIGURATION_HD;
202
203
204GSW_GNSS_MODE_CONFIGURATION_HD map_gnss_mode(GSW_GNSS_MODE_CONFIGURATION mode)
205{
206 switch (mode)
207 {
208 case GSW_MODE_GPS_GLONASS:
209 return MODE_GPS_GLONASS;
210 case GSW_MODE_GPS_BEIDOU:
211 return MODE_GPS_BEIDOU;
212 case GSW_MODE_GPS_GLONASS_BEIDOU:
213 return MODE_END;
214 case GSW_MODE_GPS:
215 return MODE_GPS;
216 case GSW_MODE_BEIDOU:
217 return MODE_BEIDOU;
218 case GSW_MODE_GLONASS:
219 return MODE_GLONASS;
220 case GSW_MODE_GPS_GLONASS_BEIDOU_GALILEO:
221 return MODE_END;
222 case GSW_MODE_GPS_GALILEO:
223 return MODE_GPS_GALILEO;
224 case GSW_MODE_GPS_GLONASS_GALILEO:
225 return MODE_GPS_GLONASS_GALILEO;
226 case GSW_MODE_GPS_GALILEO_ONLY:
227 return MODE_GALILEO;
228 case GSW_MODE_GPS_GLONASS_BEIDOU_GALILEO_NAVIC:
229 return MODE_END;
230 case GSW_MODE_GNSS_END:
231 return MODE_END;
232 default:
233 return MODE_END;
234 }
235}
236
237int mbtk_gnss_set_VTG()
238{
239 int ret;
zw.wang57f3a9f2025-07-10 14:26:52 +0800240 ret = system("/usr/sbin/gnss_gpio.sh VTG > /dev/null 2>&1");
b.liu68a94c92025-05-24 12:53:41 +0800241 if(ret != 0)
242 {
zw.wang57f3a9f2025-07-10 14:26:52 +0800243 LOGE("[GSW_gnss] mbtk_gnss_set_VTG on fail.ret = [%d]", ret);
hong.liud2417072025-06-27 07:10:37 -0700244 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800245 }
246 return GSW_HAL_SUCCESS;
247}
248
249/**
250 * @brief SDK interface to set gnss sampling frequency, support 1Hz、2Hz、5Hz
251 * @param [in] freq
252 * @retval 0: success
253 * @retval other: fail
254 */
255int gsw_gnss_set_freq(int freq)
256{
zw.wangbc534c02025-06-26 09:31:44 +0800257 if(!inited)
hong.liud2417072025-06-27 07:10:37 -0700258 return GSW_HAL_NORMAL_FAIL;
zw.wangbc534c02025-06-26 09:31:44 +0800259
b.liu68a94c92025-05-24 12:53:41 +0800260 int ret;
261 if (!strated)
262 {
263 gnss_freq = freq;
264 return GSW_HAL_SUCCESS;
265 }
266 char param_buf[32] = {0};
267 snprintf(param_buf, 32, "$FREQCFG,%d", freq);
268 mbtk_gnss_setting=(int(*)(const char *setting_cmd, int))dlsym(dlHandle_gnss, "mbtk_gnss_setting");
269 ret = mbtk_gnss_setting(param_buf, QSER_GNSS_TIMEOUT);
270 if(ret != 0)
271 {
272 LOGE("[qser_gnss] mbtk_gnss_setting fail.ret = [%d]", ret);
hong.liud2417072025-06-27 07:10:37 -0700273 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800274 }
275 return GSW_HAL_SUCCESS;
276}
277
278/**
279 * @brief SDK interface to set gnss start mode,specific mode refreence GSW_HAL_GNSS_MODE_CONFIGURATION
280 * @param [in] start_mode
281 * @retval 0: success
282 * @retval other: fail
283 */
hong.liud2417072025-06-27 07:10:37 -0700284int32_t gsw_gnss_set_start_mode(GSW_GNSS_MODE_CONFIGURATION start_mode)
b.liu68a94c92025-05-24 12:53:41 +0800285{
zw.wangbc534c02025-06-26 09:31:44 +0800286 if(!inited)
hong.liud2417072025-06-27 07:10:37 -0700287 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800288 int ret;
289 if (!strated)
290 {
291 gnss_startmode = start_mode;
292 return GSW_HAL_SUCCESS;
293 }
294 char param_buf[32] = {0};
295 snprintf(param_buf, 32, "$SYSCFG,%d", map_gnss_mode(start_mode));
296 if(map_gnss_mode(start_mode) == -1)
297 {
298 LOGE("[qser_gnss] mbtk_gnss_start_mode con't support");
hong.liud2417072025-06-27 07:10:37 -0700299 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800300 }
301 mbtk_gnss_setting=(int(*)(const char *setting_cmd, int))dlsym(dlHandle_gnss, "mbtk_gnss_setting");
302 ret = mbtk_gnss_setting(param_buf, QSER_GNSS_TIMEOUT);
303 if(ret != 0)
304 {
305 LOGE("[qser_gnss] mbtk_gnss_setting fail.ret = [%d]", ret);
hong.liud2417072025-06-27 07:10:37 -0700306 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800307 }
308 return GSW_HAL_SUCCESS;
309}
310
311/**
312 * @brief SDK interface to set EPO switch if open or close
313 * @param [in] switch_op
314 * @retval 0: success
315 * @retval other: fail
316 */
hong.liud2417072025-06-27 07:10:37 -0700317int32_t gsw_gnss_epo_switch(GSW_CONF_SWITCH switch_op)
b.liu68a94c92025-05-24 12:53:41 +0800318{
zw.wangbc534c02025-06-26 09:31:44 +0800319 if(!inited)
hong.liud2417072025-06-27 07:10:37 -0700320 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800321 int ret;
322 if (!strated)
323 {
324 gnss_switch_op = switch_op;
325 return GSW_HAL_SUCCESS;
326 }
327
328 if(switch_op)
329 {
330 mbtk_gnss_eph_download=(int(*)(int))dlsym(dlHandle_gnss, "mbtk_gnss_eph_download");
331 ret = mbtk_gnss_eph_download(QSER_GNSS_TIMEOUT);
332 if(ret != 0)
333 {
334 LOGE("[qser_gnss] mbtk_gnss_eph_download fail.ret = [%d]", ret);
hong.liud2417072025-06-27 07:10:37 -0700335 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800336 }
337 mbtk_gnss_eph_inject=(int(*)(int))dlsym(dlHandle_gnss, "mbtk_gnss_eph_inject");
338 ret = mbtk_gnss_eph_inject(QSER_GNSS_TIMEOUT);
339 if(ret != 0)
340 {
341 LOGE("[qser_gnss] mbtk_gnss_eph_inject fail.ret = [%d]", ret);
hong.liud2417072025-06-27 07:10:37 -0700342 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800343 }
344 }
345 return GSW_HAL_SUCCESS;
346}
347
348static void gnss_callback(uint32_t ind_type, const void* data, uint32_t data_len)
349{
350 if(data == NULL || data_len <= 0)
351 {
352 LOGE("[GSW_gnss] data is NULL.");
353 return;
354 }
355
356 if(gsw_cb == NULL)
357 {
358 //LOGE("[qser_gnss] qser_gnss_callback is NULL.");
359 return;
360 }
361
zw.wanga5250d62025-06-13 16:01:34 +0800362 if(ind_type == MBTK_GNSS_IND_LOCATION && gsw_cb->gsw_location_cb != NULL) {
b.liu68a94c92025-05-24 12:53:41 +0800363 if(data_len != sizeof(mbtk_gnss_location_info_t))
364 {
365 LOGE("[GSW_gnss] data size error");
366 return;
367 }
368 GSW_GNSS_LOCATION_EXT_T gsw_location;
hong.liud2417072025-06-27 07:10:37 -0700369 GSW_GNSS_LOCATION_T gsw_location_t = {0};
b.liu68a94c92025-05-24 12:53:41 +0800370 mbtk_gnss_location_info_t *locl_info = (mbtk_gnss_location_info_t *)data;
371 gsw_location_t.altitude = locl_info->altitude;
372 gsw_location_t.latitude = locl_info->latitude;
373 gsw_location_t.longitude = locl_info->longitude;
374 gsw_location_t.speed = locl_info->speed;
375 gsw_location_t.bearing = locl_info->bearing;
376 gsw_location_t.timestamp = locl_info->timestamp;
zw.wanga5250d62025-06-13 16:01:34 +0800377 gsw_location_t.flags = locl_info->flags;
b.liu68a94c92025-05-24 12:53:41 +0800378 gsw_location.legacyLocation = gsw_location_t;
379 gsw_cb->gsw_location_cb(&gsw_location);
zw.wanga5250d62025-06-13 16:01:34 +0800380 } else if(ind_type == MBTK_GNSS_IND_NMEA && gsw_cb->gsw_nmea_cb != NULL) {
b.liu68a94c92025-05-24 12:53:41 +0800381 mopen_gnss_nmea_info_t qser_nmea = {0};
382 memset(&qser_nmea, 0x0, sizeof(mopen_gnss_nmea_info_t));
383 qser_nmea.length = data_len;
384 memcpy(qser_nmea.nmea, (char *)data, data_len);
385 //qser_nmea.timestamp = qser_get_gnss_time_sec(data, data_len);
386 gsw_cb->gsw_nmea_cb(data,data_len);
387 } else {
388 LOGD("Unknown IND : %d\n", ind_type);
389 }
390}
391
hong.liud2417072025-06-27 07:10:37 -0700392int32_t gsw_gnss_init(void)
b.liu68a94c92025-05-24 12:53:41 +0800393{
lichengzhanga7089172025-07-01 20:41:46 +0800394 int ret;
b.liu68a94c92025-05-24 12:53:41 +0800395 if(!inited)
396 {
lichengzhanga7089172025-07-01 20:41:46 +0800397 ret = system("serial_atcmd AT*IMLCONFIG=13");
398 if(ret != 0)
399 {
400 LOGE("serial_atcmd fail\n");
401 return GSW_HAL_NORMAL_FAIL;
402 }
zw.wang1907e8f2025-07-01 16:26:39 +0800403 if(dlHandle_gnss == NULL)
404 dlHandle_gnss = dlopen(lynqLib_gnss, RTLD_NOW);
405 if(fun_ptr_log == NULL)
406 fun_ptr_log = (mbtk_log)dlsym(dlHandle_gnss, "mbtk_log");
b.liu68a94c92025-05-24 12:53:41 +0800407 if(fun_ptr_log == NULL || dlHandle_gnss == NULL)
408 {
hong.liud2417072025-06-27 07:10:37 -0700409 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800410 }
411 mbtk_gnss_init=(int(*)(mbtk_gnss_callback_func))dlsym(dlHandle_gnss, "mbtk_gnss_init");
412 ret = mbtk_gnss_init(gnss_callback);
413 if(ret == 0)
414 {
415 mbtk_gnss_ind_set=(int(*)(uint32_t ,int))dlsym(dlHandle_gnss, "mbtk_gnss_ind_set");
zw.wanga5250d62025-06-13 16:01:34 +0800416 ret = mbtk_gnss_ind_set(MBTK_GNSS_IND_NMEA |MBTK_GNSS_IND_LOCATION, QSER_GNSS_TIMEOUT);
b.liu68a94c92025-05-24 12:53:41 +0800417 if(ret == 0)
418 {
419 inited = true;
420 }
421 else
422 {
423 LOGE("[GSW_gnss] init mbtk_gnss_ind_set() fail.ret = [%d]", ret);
hong.liud2417072025-06-27 07:10:37 -0700424 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800425 }
426 }
427 else
428 {
429 LOGE("[GSW_gnss] mbtk_gnss_init() fail.ret = [%d]", ret);
hong.liud2417072025-06-27 07:10:37 -0700430 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800431 }
432 }
433
434 return GSW_HAL_SUCCESS;
435}
436
437/**
438 * @brief SDK interface to registered callback function
439 * @param [in] callback
440 * @retval 0: success
441 * @retval other: fail
442 */
hong.liud2417072025-06-27 07:10:37 -0700443int32_t gsw_gnss_reg_cb_group(gsw_gnss_cb callback)
b.liu68a94c92025-05-24 12:53:41 +0800444{
zw.wangbc534c02025-06-26 09:31:44 +0800445 if(!inited)
hong.liud2417072025-06-27 07:10:37 -0700446 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800447 if(callback.gsw_location_cb == NULL && callback.gsw_nmea_cb == NULL)
448 {
449 LOGE("[GSW_gnss] handler_ptr is NULL.");
hong.liud2417072025-06-27 07:10:37 -0700450 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800451 }
452 if (gsw_cb == NULL)
453 {
454 gsw_cb = (gsw_gnss_cb *)malloc(sizeof(gsw_gnss_cb));
455 if (gsw_cb == NULL)
456 {
457 LOGE("[GSW_gnss] Memory allocation failed.");
hong.liud2417072025-06-27 07:10:37 -0700458 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800459 }
460 }
461 gsw_cb->gsw_location_cb = callback.gsw_location_cb;
462 gsw_cb->gsw_nmea_cb = callback.gsw_nmea_cb;
463 return GSW_HAL_SUCCESS;
464}
465/**
466 * @brief SDK interface to start gnss
467 * @param
468 * @retval 0: success
469 * @retval other: fail
470 */
hong.liud2417072025-06-27 07:10:37 -0700471int32_t gsw_gnss_start(void)
b.liu68a94c92025-05-24 12:53:41 +0800472{
473 int ret;
zw.wangbc534c02025-06-26 09:31:44 +0800474 if(!inited)
hong.liud2417072025-06-27 07:10:37 -0700475 return GSW_HAL_NORMAL_FAIL;
zw.wangbc534c02025-06-26 09:31:44 +0800476 ret = system("/usr/sbin/gnss_gpio.sh on > /dev/null 2>&1");
477 if(ret != 0)
478 {
479 LOGE("[GSW_gnss] gnss_gpio.sh on fail.ret = [%d]", ret);
hong.liud2417072025-06-27 07:10:37 -0700480 return GSW_HAL_NORMAL_FAIL;
zw.wangbc534c02025-06-26 09:31:44 +0800481 }
zw.wang57f3a9f2025-07-10 14:26:52 +0800482 // 记录mbtk_gnss_open前的时间戳
483 struct timespec start_time;
484 if (clock_gettime(CLOCK_MONOTONIC, &start_time) != 0) {
485 LOGE("[GSW_gnss] Failed to get start time");
486 return GSW_HAL_NORMAL_FAIL;
487 }
488
b.liu68a94c92025-05-24 12:53:41 +0800489 mbtk_gnss_open=(int(*)(int,int))dlsym(dlHandle_gnss, "mbtk_gnss_open");
490 ret = mbtk_gnss_open(255, QSER_GNSS_TIMEOUT);
491 if(ret != 0)
492 {
493 LOGE("[GSW_gnss] mbtk_gnss_open is error.ret = [%d]", ret);
hong.liud2417072025-06-27 07:10:37 -0700494 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800495 }
zw.wang89c18242025-06-24 19:07:10 +0800496
b.liu68a94c92025-05-24 12:53:41 +0800497 strated = true;
zw.wang7aa34062025-06-27 17:54:55 +0800498 /*
b.liu68a94c92025-05-24 12:53:41 +0800499 if (gnss_freq > 0)
500 gsw_gnss_set_freq(gnss_freq);
501 if (gnss_startmode >= 0)
502 gsw_gnss_set_start_mode(gnss_startmode);
503 if (gnss_switch_op > 0)
504 gsw_gnss_epo_switch(gnss_switch_op);
zw.wang7aa34062025-06-27 17:54:55 +0800505 */
zw.wang57f3a9f2025-07-10 14:26:52 +0800506
507 // 记录mbtk_gnss_set_VTG前的时间戳并计算等待时间
508 struct timespec end_time;
509 if (clock_gettime(CLOCK_MONOTONIC, &end_time) != 0) {
510 LOGE("[GSW_gnss] Failed to get end time");
511 return GSW_HAL_NORMAL_FAIL;
512 }
513 long start_ms = start_time.tv_sec * 1000 + start_time.tv_nsec / 1000000;
514 long end_ms = end_time.tv_sec * 1000 + end_time.tv_nsec / 1000000;
515 long diff_ms = end_ms - start_ms;
516 if (diff_ms < 700) {
517 long wait_ms = 700 - diff_ms;
518 usleep(wait_ms * 1000); // 转换为微秒等待
519 }
520
b.liu68a94c92025-05-24 12:53:41 +0800521 mbtk_gnss_set_VTG();
522
523 return GSW_HAL_SUCCESS;
524}
525
zw.wang57f3a9f2025-07-10 14:26:52 +0800526
b.liu68a94c92025-05-24 12:53:41 +0800527/**
528 * @brief SDK interface to stop gnss
529 * @param
530 * @retval 0: success
531 * @retval other: fail
532 */
hong.liud2417072025-06-27 07:10:37 -0700533int32_t gsw_gnss_stop(void)
b.liu68a94c92025-05-24 12:53:41 +0800534{
535 int ret;
zw.wangbc534c02025-06-26 09:31:44 +0800536 if(!inited)
hong.liud2417072025-06-27 07:10:37 -0700537 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800538 mbtk_gnss_close=(int(*)(int))dlsym(dlHandle_gnss, "mbtk_gnss_close");
539 ret = mbtk_gnss_close(QSER_GNSS_TIMEOUT);
540 if(ret != 0)
541 {
542 LOGE("[GSW_gnss] mbtk_gnss_close is error.ret = [%d]", ret);
hong.liud2417072025-06-27 07:10:37 -0700543 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800544 }
zw.wangbc534c02025-06-26 09:31:44 +0800545 ret = system("/usr/sbin/gnss_gpio.sh off > /dev/null 2>&1");
546 if(ret != 0)
547 {
548 LOGE("[GSW_gnss] gnss_gpio.sh off fail.ret = [%d]", ret);
hong.liud2417072025-06-27 07:10:37 -0700549 return GSW_HAL_NORMAL_FAIL;
zw.wangbc534c02025-06-26 09:31:44 +0800550 }
b.liu68a94c92025-05-24 12:53:41 +0800551 strated = false;
552 return GSW_HAL_SUCCESS;
553}
554
555/**
556 * @brief SDK interface to de initialization gnss
557 * @param
558 * @retval 0: success
559 * @retval other: fail
560 */
hong.liud2417072025-06-27 07:10:37 -0700561int32_t gsw_gnss_deinit(void)
b.liu68a94c92025-05-24 12:53:41 +0800562{
563 int ret;
zw.wangbc534c02025-06-26 09:31:44 +0800564 if(!inited)
hong.liud2417072025-06-27 07:10:37 -0700565 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800566 if(inited)
567 {
lichengzhanga7089172025-07-01 20:41:46 +0800568 ret = system("serial_atcmd AT*IMLCONFIG=12");
569 if(ret != 0)
570 {
571 LOGE("serial_atcmd fail\n");
572 return GSW_HAL_NORMAL_FAIL;
573 }
b.liu68a94c92025-05-24 12:53:41 +0800574 mbtk_gnss_deinit=(int(*)())dlsym(dlHandle_gnss, "mbtk_gnss_deinit");
575 ret = mbtk_gnss_deinit();
576 if(ret == 0)
577 {
578 inited = false;
579 }
580 else
581 {
zw.wang1907e8f2025-07-01 16:26:39 +0800582 LOGE("[GSW_gnss] mbtk_gnss_deinit() fail.ret = [%d]", ret);
583 /*
b.liu68a94c92025-05-24 12:53:41 +0800584 dlclose(dlHandle_gnss);
585 dlHandle_gnss = NULL;
zw.wang1907e8f2025-07-01 16:26:39 +0800586 */
hong.liud2417072025-06-27 07:10:37 -0700587 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800588 }
589 }
590
zw.wang1907e8f2025-07-01 16:26:39 +0800591 /*
b.liu68a94c92025-05-24 12:53:41 +0800592 dlclose(dlHandle_gnss);
593 dlHandle_gnss = NULL;
zw.wang1907e8f2025-07-01 16:26:39 +0800594 */
b.liu68a94c92025-05-24 12:53:41 +0800595 gnss_freq = -1;
596 gnss_startmode = -1;
597 gnss_switch_op = -1;
598 return GSW_HAL_SUCCESS;
599}
600
zw.wang75e98ea2025-05-29 17:57:38 +0800601int gsw_gnss_add_lib(void)
602{
603 return GSW_HAL_SUCCESS;
604}
b.liu68a94c92025-05-24 12:53:41 +0800605
606/**
607 * @brief SDK interface to enable XTRA switch
608 * @param [in] state
609 * @retval 0: success
610 * @retval other: fail
611 */
hong.liud2417072025-06-27 07:10:37 -0700612int32_t gsw_gnss_xtra_is_enable(gsw_xtra_state_e state)
b.liu68a94c92025-05-24 12:53:41 +0800613{
614 return GSW_HAL_SUCCESS;
615}
616
617/**
618 * @brief SDK interface to delete aiding data,delete aiding data for cold start(1-H,2-W,3-C)
619 * @param [in] switch_op
620 * @retval 0: success
621 * @retval other: fail
622 */
hong.liud2417072025-06-27 07:10:37 -0700623int32_t gsw_gnss_delete_aiding_data(unsigned int flags) /*1-don`t delete == hot_start ; 2-delete EPH == warm start ; 3-delete all == cold start*/
b.liu68a94c92025-05-24 12:53:41 +0800624{
zw.wangbc534c02025-06-26 09:31:44 +0800625 if(!inited)
hong.liud2417072025-06-27 07:10:37 -0700626 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800627 int ret;
628 char param_buf[32] = {0};
629 snprintf(param_buf, 32, "$RESET,%u", flags);
630 mbtk_gnss_setting=(int(*)(const char *setting_cmd, int))dlsym(dlHandle_gnss, "mbtk_gnss_setting");
631 ret = mbtk_gnss_setting(param_buf, QSER_GNSS_TIMEOUT);
632 if(ret != 0)
633 {
634 LOGE("[qser_gnss] mbtk_gnss_setting fail.ret = [%d]", ret);
hong.liud2417072025-06-27 07:10:37 -0700635 return GSW_HAL_NORMAL_FAIL;
b.liu68a94c92025-05-24 12:53:41 +0800636 }
637 return GSW_HAL_SUCCESS;
638}
639
640/**
641 * @brief init and configure gps
642 * @param [in] init_configure
643 * @retval 0: success
644 * @retval other: fail
645 */
hong.liud2417072025-06-27 07:10:37 -0700646int32_t gsw_gnss_init_configure_gps(gsw_gnss_init_configure_t init_configure)
b.liu68a94c92025-05-24 12:53:41 +0800647{
648 return GSW_HAL_SUCCESS;
649}
hong.liud2417072025-06-27 07:10:37 -0700650
651int gsw_gnss_get_tail_nmea_type(char *tail_type, int len)
652{
653 if(NULL == tail_type){
654 LOGE("get_tail_nmea_type fail, tail_type is NULL");
655 return GSW_HAL_ERROR_GNSS_FAIL;
656 }
657
658 if(len >= 3){
659 strcpy(tail_type, "RMC");
660 return GSW_HAL_GNSS_SUCCESS;
661 }else{
662 LOGE("get_tail_nmea_type fail, len[%d] is too short", len);
663 return GSW_HAL_ERROR_GNSS_FAIL;
664 }
665}