uart2_rec.c 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371
  1. /*
  2. * Copyright (c) 2025 Wingcool 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/uart.h>
  10. #include <aem_core_if.h>
  11. #include <logging/log.h>
  12. //#include <gps/gps.h>
  13. #include "aem_device.h"
  14. #include "aem_adapter_dev.h"
  15. #include <drivers/hrtimer.h>
  16. LOG_MODULE_REGISTER(uart2_rec, LOG_LEVEL_INF);
  17. #define CONFIG_UART2_REC_DEV_NAME "uart2_rec"
  18. #define UART2_REC_DEV_NAME CONFIG_UART_2_NAME
  19. // 定义数据区大小
  20. #define UART_FIFO_MAX 64
  21. #define REC_DATA_BUFFER_SIZE 192
  22. #define DEVICE_ID_SIZE 8
  23. #define PROJECT_ID_SIZE 4
  24. #define WRITE_KEY_SIZE 160
  25. #define DEVICE_SN_SIZE 6
  26. #define FW_VERSION_SIZE 3
  27. // 定义命令
  28. #define DEVICE_ID_CMD 0xB1
  29. #define PROJECT_ID_CMD 0xB2
  30. #define WRITE_KEY_CMD 0xB3
  31. #define IS_KEY_INSIDE_CMD 0xB4
  32. #define SET_TIME_CMD 0xC1
  33. #define GET_DEVICE_SN_CMD 0xD1
  34. #define GET_FW_VERSION_CMD 0xD3
  35. // 定义帧头和帧尾
  36. #define FRAME_HEADER1 0xAA
  37. #define FRAME_HEADER2 0x55
  38. #define FRAME_TAIL1 0x55
  39. #define FRAME_TAIL2 0xAA
  40. struct uart2_rec_data {
  41. const struct device *uart_gpio_dev;
  42. uint8_t rec_data[REC_DATA_BUFFER_SIZE];
  43. //gps_notify_t notify;
  44. };
  45. static struct uart2_rec_data uart2_rec_data;
  46. static struct hrtimer g_rtc_uart_proc;
  47. uint8_t bySetTimeStep = 0;
  48. extern uint8_t bySetHour;
  49. extern uint8_t bySetMinute;
  50. extern uint8_t bySetWeekday;
  51. extern uint16_t wSetYear;
  52. extern uint8_t bySetMonth;
  53. extern uint8_t bySetDay;
  54. extern void uart2_poll_out_ch(int c); // UART2发送一个字节
  55. extern bool aem_read_aem_key_record(aem_key_w_record_t *record); // 读取AEM KEY记录
  56. extern bool aem_write_aem_key_record(uint8_t mothod, uint8_t is_pass); // 写入AEM KEY记录
  57. extern bool aem_defender_factory_write_key(uint8_t * data, uint16_t len); // 写入KEY
  58. extern uint16_t aem_factory_get_product_info(uint8_t *data, uint16_t in_len); // 获取产品信息
  59. // 计算累加和校验值
  60. //1 字节,对命令码、数据长度和数据内容的所有字节进行累加求和,取低8位作为校验值,用于检测数据在传输过程中是否出错
  61. uint8_t calculate_checksum(uint8_t cmd, uint8_t *data, uint8_t data_len)
  62. {
  63. uint8_t checksum = cmd + data_len;
  64. for (uint8_t i = 0; i < data_len; i++) {
  65. checksum += data[i];
  66. }
  67. return checksum;
  68. }
  69. // 发送一帧数据
  70. void send_frame(uint8_t cmd, uint8_t *data, uint8_t data_len)
  71. {
  72. uint8_t checksum = calculate_checksum(cmd, data, data_len);
  73. // 发送帧头
  74. uart2_poll_out_ch(FRAME_HEADER1);
  75. uart2_poll_out_ch(FRAME_HEADER2);
  76. // 发送命令码
  77. uart2_poll_out_ch(cmd);
  78. // 发送数据长度
  79. uart2_poll_out_ch(data_len);
  80. // 发送数据内容
  81. for (uint8_t i = 0; i < data_len; i++) {
  82. uart2_poll_out_ch(data[i]);
  83. }
  84. // 发送累加和校验值
  85. uart2_poll_out_ch(checksum);
  86. // 发送帧尾
  87. uart2_poll_out_ch(FRAME_TAIL1);
  88. uart2_poll_out_ch(FRAME_TAIL2);
  89. }
  90. // 检查一帧数据是否正确
  91. bool check_receive_frame(uint8_t *data)
  92. {
  93. uint8_t calculated_checksum;
  94. uint8_t data_len = data[3];
  95. // 帧头
  96. if (data[0] != FRAME_HEADER1)
  97. {
  98. return false; // 帧头错误
  99. }
  100. if (data[1] != FRAME_HEADER2)
  101. {
  102. return false; // 帧头错误
  103. }
  104. // 计算本地累加和校验值
  105. calculated_checksum = calculate_checksum(data[2], data + 4, data_len);
  106. if (data[data_len + 4] != calculated_checksum) {
  107. return false; // 累加和校验错误
  108. }
  109. // 读取帧尾
  110. if (data[data_len + 5] != FRAME_TAIL1 || data[data_len + 6] != FRAME_TAIL2) {
  111. return false; // 帧尾错误
  112. }
  113. return true; // 接收成功
  114. }
  115. static void uart_proc_timer_acts_handler(struct k_work *work)
  116. {
  117. // 定义发送缓冲区
  118. uint8_t tx_buff[2];
  119. // 定义AEM密钥记录结构体
  120. aem_key_w_record_t key_record;
  121. // 根据接收到的命令,执行不同的操作
  122. switch (uart2_rec_data.rec_data[2])
  123. {
  124. case WRITE_KEY_CMD:
  125. // 发送缓冲区初始化
  126. tx_buff[0] = 0;
  127. // 调用写入密钥函数,如果写入成功
  128. if (aem_defender_factory_write_key(&uart2_rec_data.rec_data[4], WRITE_KEY_SIZE))
  129. {
  130. aem_write_aem_key_record(0, 1); // 写入成功,写入NVRAM记录
  131. tx_buff[0] = 0xAA;
  132. }
  133. send_frame(WRITE_KEY_CMD, tx_buff, 1);
  134. break;
  135. case IS_KEY_INSIDE_CMD:
  136. tx_buff[0] = 0;
  137. if (aem_read_aem_key_record(&key_record)) // 读取NVRAM记录
  138. {
  139. //if (&key_record != NULL)
  140. {
  141. if (key_record.is_pass)
  142. tx_buff[0] = 0xAA;
  143. }
  144. }
  145. send_frame(IS_KEY_INSIDE_CMD, tx_buff, 1);
  146. break;
  147. default:
  148. break;
  149. }
  150. }
  151. K_WORK_DEFINE(uart_proc_timer_acts, uart_proc_timer_acts_handler);
  152. static void htimer_uart_proc(struct hrtimer *ttimer, void *expiry_fn_arg)
  153. {
  154. k_work_submit(&uart_proc_timer_acts); //向系统工作队列提交一个工作项,让工作队列的线程将执行该工作
  155. }
  156. static uint8_t byAACnt = 0;
  157. static uint8_t by55Cnt = 0;
  158. static bool bReceiveFrameDone = false;
  159. static void uart_fifo_callback(const struct device *dev, void *user_data)
  160. {
  161. uint8_t rx_buff[UART_FIFO_MAX];
  162. uint8_t tx_buff[UART_FIFO_MAX];
  163. char mac_str[7] = { 0 }; // MAC地址字符串
  164. int read_size;
  165. static uint8_t cnt;
  166. aem_dev_info_t info; // 定义AEM设备信息结构体
  167. //printk("uart_fifo_callback start\n");
  168. uart_irq_update(dev);
  169. if (uart_irq_rx_ready(dev)) {
  170. read_size = uart_fifo_read(dev, rx_buff, UART_FIFO_MAX);
  171. //printk("read_size: %d\n", read_size);
  172. if (read_size == UART_FIFO_MAX)
  173. printk("uart fifo buffer overflow\n");
  174. for (int i = 0; i < read_size; i++)
  175. {
  176. switch (rx_buff[i]) {
  177. case FRAME_HEADER1:
  178. if (byAACnt == 0)
  179. byAACnt++;
  180. else if (byAACnt == 1 && by55Cnt == 2)
  181. byAACnt++;
  182. if (by55Cnt == 0 && byAACnt == 1)
  183. {
  184. cnt = 0;
  185. bReceiveFrameDone = false;
  186. }
  187. else if (by55Cnt == 2 && byAACnt == 2)
  188. {
  189. bReceiveFrameDone = true;
  190. by55Cnt = 0;
  191. byAACnt = 0;
  192. }
  193. uart2_rec_data.rec_data[cnt++] = rx_buff[i];
  194. break;
  195. case FRAME_HEADER2:
  196. if (by55Cnt == 0 && cnt == 1)
  197. by55Cnt++;
  198. if (cnt > 4 && (cnt - 5 == uart2_rec_data.rec_data[3])) //长度大于4且与接收数据的长度相等
  199. by55Cnt++;
  200. uart2_rec_data.rec_data[cnt++] = rx_buff[i];
  201. break;
  202. default:
  203. uart2_rec_data.rec_data[cnt++] = rx_buff[i];
  204. /* protection data cannot exceed boundary */
  205. if (cnt >= REC_DATA_BUFFER_SIZE)
  206. cnt--;
  207. break;
  208. }
  209. }
  210. if (bReceiveFrameDone && check_receive_frame(uart2_rec_data.rec_data))
  211. {
  212. printk("receive frame success\n");
  213. switch (uart2_rec_data.rec_data[2])
  214. {
  215. case DEVICE_ID_CMD:
  216. aem_factory_get_product_info(tx_buff, 64);
  217. send_frame(DEVICE_ID_CMD, tx_buff, DEVICE_ID_SIZE);
  218. break;
  219. case PROJECT_ID_CMD:
  220. tx_buff[0] = 0x0F;
  221. tx_buff[1] = 0x6B;
  222. tx_buff[2] = 0x3B;
  223. tx_buff[3] = 0xD8;
  224. send_frame(PROJECT_ID_CMD, tx_buff, PROJECT_ID_SIZE);
  225. break;
  226. case WRITE_KEY_CMD:
  227. case IS_KEY_INSIDE_CMD:
  228. hrtimer_init(&g_rtc_uart_proc, htimer_uart_proc, NULL);
  229. hrtimer_start(&g_rtc_uart_proc, 1000*10, 0); //10ms 后执行回调函数
  230. break;
  231. case SET_TIME_CMD:
  232. if (uart2_rec_data.rec_data[3] == 6) //时间数据长度为6
  233. {
  234. bySetMinute = uart2_rec_data.rec_data[4];
  235. bySetHour = uart2_rec_data.rec_data[5];
  236. bySetDay = uart2_rec_data.rec_data[6];
  237. bySetWeekday = uart2_rec_data.rec_data[7];
  238. bySetMonth = uart2_rec_data.rec_data[8];
  239. wSetYear = uart2_rec_data.rec_data[9];
  240. bySetTimeStep = 1;
  241. }
  242. send_frame(SET_TIME_CMD, tx_buff, 0);
  243. break;
  244. case GET_DEVICE_SN_CMD:
  245. aem_get_ble_mac_addr(mac_str, 6);
  246. for (int i = 0; i < DEVICE_SN_SIZE; i++)
  247. {
  248. tx_buff[i] = 0xFF - mac_str[i];
  249. }
  250. send_frame(GET_DEVICE_SN_CMD, tx_buff, DEVICE_SN_SIZE);
  251. break;
  252. case GET_FW_VERSION_CMD:
  253. info = get_device_info();
  254. tx_buff[0] = info.major_v; //主版本号
  255. tx_buff[1] = info.minor_v; //次版本号
  256. tx_buff[2] = info.micro_v; //补丁版本号
  257. send_frame(GET_FW_VERSION_CMD, tx_buff, FW_VERSION_SIZE);
  258. break;
  259. default:
  260. break;
  261. }
  262. }
  263. }
  264. }
  265. /*
  266. static void uart2_rec_enable(const struct device *dev)
  267. {
  268. struct uart2_rec_data *data = (struct uart2_rec_data *)dev->data;
  269. printk("uart2_rec_enable\n");
  270. uart_irq_rx_enable(data->uart_gpio_dev);
  271. }
  272. static void uart2_rec_disable(const struct device *dev)
  273. {
  274. struct uart2_rec_data *data = (struct uart2_rec_data *)dev->data;
  275. printk("uart2_rec_disable\n");
  276. uart_irq_rx_disable(data->uart_gpio_dev);
  277. }
  278. static void uart2_rec_register_notify(const struct device *dev, gps_notify_t notify)
  279. {
  280. printk("uart2_rec_register_notify\n");
  281. struct uart2_rec_data *data = (struct uart2_rec_data *)dev->data;
  282. data->notify = notify;
  283. }
  284. static void uart2_rec_unregister_notify(const struct device *dev, gps_notify_t notify)
  285. {
  286. struct uart2_rec_data *data = (struct uart2_rec_data *)dev->data;
  287. data->notify = NULL;
  288. }
  289. static const struct gps_dev_driver_api uart2_rec_api = {
  290. .enable = uart2_rec_enable,
  291. .disable = uart2_rec_disable,
  292. .inquiry = NULL,
  293. .register_notify = uart2_rec_register_notify,
  294. .unregister_notify = uart2_rec_unregister_notify,
  295. };
  296. */
  297. static int uart2_rec_init(const struct device *dev)
  298. {
  299. struct uart2_rec_data *data = (struct uart2_rec_data *)dev->data;
  300. printk("uart2_rec_init\n");
  301. data->uart_gpio_dev = device_get_binding(UART2_REC_DEV_NAME);
  302. if (data->uart_gpio_dev == NULL) {
  303. printk("Couldn't find uart2\n");
  304. return -ENODEV;
  305. }
  306. uart_irq_callback_set(data->uart_gpio_dev, uart_fifo_callback);
  307. uart_irq_rx_enable(data->uart_gpio_dev);
  308. return 0;
  309. }
  310. #if IS_ENABLED(CONFIG_UART_ACTS)
  311. DEVICE_DEFINE(uart2_rec, CONFIG_UART2_REC_DEV_NAME, &uart2_rec_init, NULL, &uart2_rec_data, NULL, POST_KERNEL, 60, NULL);
  312. #endif