gps_tg9255.c 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. /*
  2. * Copyright (c) 2022 Actions Technology Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <soc.h>
  7. #include <board.h>
  8. #include <device.h>
  9. #include <drivers/gpio.h>
  10. #include <drivers/uart.h>
  11. #include <logging/log.h>
  12. #include <gps/gps.h>
  13. LOG_MODULE_REGISTER(gps_9255, LOG_LEVEL_INF);
  14. #define GPS_UART_DEV_NAME CONFIG_UART_2_NAME
  15. #define UART_FIFO_MAX 16
  16. #define GPS_DATA_BUFFER_SIZE 100
  17. struct tg9255_data {
  18. const struct device *gpio_dev;
  19. const struct device *uart_dev;
  20. uint8_t gps_data[2][GPS_DATA_BUFFER_SIZE];
  21. uint8_t gpa_data_cur;
  22. uint8_t gpa_data_prev;
  23. gps_notify_t notify;
  24. struct k_work gps_work;
  25. };
  26. static const struct gpio_cfg power_gpio_cfg = CONFIG_GPS_POWER_GPIO;
  27. static struct tg9255_data tg9255_data;
  28. static void uart_fifo_callback(const struct device *dev, void *user_data)
  29. {
  30. uint8_t rx_buff[UART_FIFO_MAX];
  31. int read_size;
  32. static uint8_t cnt;
  33. uart_irq_update(dev);
  34. if(uart_irq_rx_ready(dev))
  35. {
  36. read_size = uart_fifo_read(dev,rx_buff,UART_FIFO_MAX);
  37. if(read_size == UART_FIFO_MAX)
  38. LOG_ERR("uart fifo buffer overflow");
  39. for(int i = 0; i < read_size; i++)
  40. {
  41. switch (rx_buff[i])
  42. {
  43. case '$':
  44. if(tg9255_data.gpa_data_cur)
  45. tg9255_data.gpa_data_cur = 0;
  46. else
  47. tg9255_data.gpa_data_cur = 1;
  48. memset(tg9255_data.gps_data[tg9255_data.gpa_data_cur], 0, sizeof(tg9255_data.gps_data[tg9255_data.gpa_data_cur]));
  49. cnt = 0;
  50. tg9255_data.gps_data[tg9255_data.gpa_data_cur][cnt] = rx_buff[i];
  51. break;
  52. case '\n':
  53. cnt ++;
  54. tg9255_data.gps_data[tg9255_data.gpa_data_cur][cnt] = rx_buff[i];
  55. tg9255_data.gpa_data_prev = tg9255_data.gpa_data_cur;
  56. k_work_submit(&tg9255_data.gps_work);
  57. break;
  58. default:
  59. cnt ++;
  60. tg9255_data.gps_data[tg9255_data.gpa_data_cur][cnt] = rx_buff[i];
  61. /* protection data cannot exceed boundary */
  62. if(cnt >= GPS_DATA_BUFFER_SIZE)
  63. cnt --;
  64. break;
  65. }
  66. }
  67. }
  68. }
  69. static void tg9255_enable(const struct device *dev)
  70. {
  71. struct tg9255_data *data = (struct tg9255_data *)dev->data;
  72. gpio_pin_set(data->gpio_dev, power_gpio_cfg.gpion, 1);
  73. uart_irq_rx_enable(data->uart_dev);
  74. }
  75. static void tg9255_disable(const struct device *dev)
  76. {
  77. struct tg9255_data *data = (struct tg9255_data *)dev->data;
  78. uart_irq_rx_disable(data->uart_dev);
  79. gpio_pin_set(data->gpio_dev, power_gpio_cfg.gpion, 0);
  80. }
  81. static void tg9255_register_notify(const struct device *dev, gps_notify_t notify)
  82. {
  83. struct tg9255_data *data = (struct tg9255_data *)dev->data;
  84. data->notify = notify;
  85. }
  86. static void tg9255_unregister_notify(const struct device *dev, gps_notify_t notify)
  87. {
  88. struct tg9255_data *data = (struct tg9255_data *)dev->data;
  89. data->notify = NULL;
  90. }
  91. static const struct gps_dev_driver_api tg9255_api = {
  92. .enable = tg9255_enable,
  93. .disable = tg9255_disable,
  94. .inquiry = NULL,
  95. .register_notify = tg9255_register_notify,
  96. .unregister_notify = tg9255_unregister_notify,
  97. };
  98. static void _gps_work_handler(struct k_work *work)
  99. {
  100. struct gps_value val = {0};
  101. // LOG_INF("%s",tg9255_data.gps_data[tg9255_data.gpa_data_prev]);
  102. if (tg9255_data.notify)
  103. {
  104. val.gps_nmea_data = tg9255_data.gps_data[tg9255_data.gpa_data_prev];
  105. tg9255_data.notify(NULL, &val);
  106. }
  107. }
  108. static int tg9255_init(const struct device *dev)
  109. {
  110. struct tg9255_data *data = (struct tg9255_data *)dev->data;
  111. LOG_INF("tg9255_init");
  112. data->gpio_dev = device_get_binding(power_gpio_cfg.gpio_dev_name);
  113. if (data->gpio_dev == NULL) {
  114. LOG_ERR("Couldn't find gpio dev");
  115. return -ENODEV;
  116. }
  117. gpio_pin_configure(data->gpio_dev, power_gpio_cfg.gpion, GPIO_OUTPUT);
  118. gpio_pin_set(data->gpio_dev, power_gpio_cfg.gpion, 0);
  119. data->uart_dev = device_get_binding(GPS_UART_DEV_NAME);
  120. if (data->uart_dev == NULL) {
  121. LOG_ERR("Couldn't find gps uart");
  122. return -ENODEV;
  123. }
  124. k_work_init(&data->gps_work, _gps_work_handler);
  125. uart_irq_callback_set(data->uart_dev, uart_fifo_callback);
  126. data->gpa_data_cur = 0;
  127. return 0;
  128. }
  129. DEVICE_DEFINE(tg9255, CONFIG_GPS_DEV_NAME, &tg9255_init,
  130. NULL, &tg9255_data, NULL, POST_KERNEL,
  131. 60, &tg9255_api);