app_at.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353
  1. #include <stdint.h>
  2. #include "string.h"
  3. #include "gap_api.h"
  4. #include "os_msg_q.h"
  5. #include "os_mem.h"
  6. #include "os_timer.h"
  7. #include "co_printf.h"
  8. #include "driver_uart.h"
  9. #include "plf.h"
  10. #include "driver_system.h"
  11. #include "driver_pmu.h"
  12. #include "driver_uart.h"
  13. #include "driver_uart_ex.h"
  14. #include "simple_gatt_service.h"
  15. #include "driver_pmu.h"
  16. #include "driver_timer.h"
  17. #define AT_RECV_MAX_LEN 32
  18. uint8_t app_at_recv_char;
  19. /*-------------------------------------------------------------------------
  20. Function : ascii_char2val ----add by chsheng, chsheng@accelsemi.com
  21. Return: -1=error
  22. Description:
  23. 'a' -> 0xa 'A' -> 0xa
  24. -------------------------------------------------------------------------*/
  25. static char ascii_char2val(const char c)
  26. {
  27. if(c>='0' && c<='9')
  28. return c-'0';
  29. if((c>='a' && c<='f') || (c>='A' && c<='F'))
  30. return (c&0x7)+9;
  31. return (char)(-1);
  32. }
  33. /*-------------------------------------------------------------------------
  34. Function : ascii_strn2val ----add by chsheng, chsheng@accelsemi.com
  35. Return: -1=error
  36. Description:
  37. str = "123" bas = 10 return 123
  38. str = "123" bas = 16 return 0x123
  39. -------------------------------------------------------------------------*/
  40. int ascii_strn2val( const char str[], char base, char n)
  41. {
  42. int val = 0;
  43. char v;
  44. while(n != 0){
  45. v = ascii_char2val(*str);
  46. #if 0
  47. if (v == -1 || v >= base)
  48. return -1;
  49. #else
  50. if (v == (char)(-1) || v >= base)
  51. {
  52. if(val == 0) //to filter abormal beginning and ending
  53. {
  54. str ++;
  55. n --;
  56. continue;
  57. }
  58. else
  59. {
  60. break;
  61. }
  62. }
  63. #endif
  64. val = val*base + v;
  65. str++;
  66. n--;
  67. }
  68. return val;
  69. }
  70. static void app_at_recv_cmd_A(uint8_t sub_cmd, uint8_t *data)
  71. {
  72. switch(sub_cmd)
  73. {
  74. case 'A':
  75. {
  76. mac_addr_t addr;
  77. addr.addr[5] = ascii_strn2val((const char *)&data[0], 16, 2);
  78. addr.addr[4] = ascii_strn2val((const char *)&data[2], 16, 2);
  79. addr.addr[3] = ascii_strn2val((const char *)&data[4], 16, 2);
  80. addr.addr[2] = ascii_strn2val((const char *)&data[6], 16, 2);
  81. addr.addr[1] = ascii_strn2val((const char *)&data[8], 16, 2);
  82. addr.addr[0] = ascii_strn2val((const char *)&data[10], 16, 2);
  83. gap_start_conn(&addr, ascii_strn2val((const char *)&data[12], 16, 2), 64, 64, 0, 500);
  84. }
  85. break;
  86. case 'B':
  87. {
  88. gap_stop_conn();
  89. }
  90. break;
  91. case 'E':
  92. co_printf("OOL VAL: 0x%02x.\r\n", ool_pd_read(ascii_strn2val((const char *)&data[0], 16, 2)));
  93. break;
  94. case 'F':
  95. ool_pd_write(ascii_strn2val((const char *)&data[0], 16, 2), ascii_strn2val((const char *)&data[3], 16, 2));
  96. co_printf("OK\r\n");
  97. break;
  98. case 'G':
  99. co_printf("hello world!\r\n");
  100. break;
  101. case 'H':
  102. co_printf("VAL: 0x%08x.\r\n", *(volatile uint32_t *)(ascii_strn2val((const char *)&data[0], 16, 8)));
  103. break;
  104. case 'I':
  105. *(volatile uint32_t *)ascii_strn2val((const char *)&data[0], 16, 8) = ascii_strn2val((const char *)&data[9], 16, 8);
  106. co_printf("OK\r\n");
  107. break;
  108. case 'J':
  109. co_printf("OOL VAL: 0x%02x.\r\n", ool_read(ascii_strn2val((const char *)&data[0], 16, 2)));
  110. break;
  111. case 'K':
  112. ool_write(ascii_strn2val((const char *)&data[0], 16, 2), ascii_strn2val((const char *)&data[3], 16, 2));
  113. co_printf("OK\r\n");
  114. break;
  115. case 'L':
  116. co_printf("VAL: 0x%02x.\r\n", *(uint8_t *)(ascii_strn2val((const char *)&data[0], 16, 8)));
  117. break;
  118. case 'M':
  119. *(uint8_t *)(ascii_strn2val((const char *)&data[0], 16, 8)) = ascii_strn2val((const char *)&data[9], 16, 2);
  120. co_printf("OK\r\n");
  121. break;
  122. case 'O':
  123. //new_efuse_value = ascii_strn2val((const char *)&cmd->data[0], 16, 8);
  124. co_printf("OK\r\n");
  125. break;
  126. case 'P':
  127. co_printf("VAL: 0x%02x.\r\n", *(uint8_t *)(MODEM_BASE + ascii_strn2val((const char *)&data[0], 16, 2)));
  128. break;
  129. case 'Q':
  130. *(uint8_t *)(MODEM_BASE + ascii_strn2val((const char *)&data[0], 16, 2)) = ascii_strn2val((const char *)&data[3], 16, 2);
  131. co_printf("OK\r\n");
  132. break;
  133. case 'U':
  134. {
  135. uint32_t *ptr = (uint32_t *)(ascii_strn2val((const char *)&data[0], 16, 8) & (~3));
  136. uint8_t count = ascii_strn2val((const char *)&data[9], 16, 2);
  137. uint32_t *start = (uint32_t *)((uint32_t)ptr & (~0x0f));
  138. for(uint8_t i=0; i<count;) {
  139. if(((uint32_t)start & 0x0c) == 0) {
  140. co_printf("0x%08x: ", start);
  141. }
  142. if(start < ptr) {
  143. co_printf(" ");
  144. }
  145. else {
  146. i++;
  147. co_printf("%08x", *start);
  148. }
  149. if(((uint32_t)start & 0x0c) == 0x0c) {
  150. co_printf("\r\n");
  151. }
  152. else {
  153. co_printf(" ");
  154. }
  155. start++;
  156. }
  157. }
  158. break;
  159. case 'V':
  160. co_printf("%d\r\n", ke_get_mem_usage(3));
  161. break;
  162. }
  163. }
  164. void app_at_cmd_recv_handler(uint8_t *data, uint16_t length)
  165. {
  166. switch(data[0])
  167. {
  168. case 'A':
  169. app_at_recv_cmd_A(data[1], &data[2]);
  170. break;
  171. default:
  172. break;
  173. }
  174. }
  175. #define __RAM_CODE __attribute__((section("ram_code")))
  176. __RAM_CODE static void app_at_recv_c(uint8_t c)
  177. {
  178. // printf("%c", c);
  179. uart_putc_noint_no_wait(UART0, c);
  180. switch(at_recv_state)
  181. {
  182. case 0:
  183. if(c == 'A')
  184. {
  185. at_recv_state++;
  186. }
  187. break;
  188. case 1:
  189. if(c == 'T')
  190. at_recv_state++;
  191. else
  192. at_recv_state = 0;
  193. break;
  194. case 2:
  195. if(c == '#')
  196. at_recv_state++;
  197. else
  198. at_recv_state = 0;
  199. break;
  200. case 3:
  201. at_recv_buffer[at_recv_index++] = c;
  202. if((c == '\n')
  203. ||(at_recv_index >= AT_RECV_MAX_LEN))
  204. {
  205. // os_event_t at_cmd_event;
  206. // at_cmd_event.event_id = USER_EVT_AT_COMMAND;
  207. // at_cmd_event.param = at_recv_buffer;
  208. // at_cmd_event.param_len = at_recv_index;
  209. // os_msg_post(user_task_id, &at_cmd_event);
  210. printf("%s\r\n", at_recv_buffer);
  211. at_recv_state = 0;
  212. at_recv_index = 0;
  213. }
  214. break;
  215. }
  216. }
  217. UART_HandleTypeDef Uart0_handle;
  218. UART_HandleTypeDef Uart1_handle;
  219. void uart_isr_int(UART_HandleTypeDef *huart)
  220. {
  221. huart->UARTx->IER_DLH.IER.ELSI = 1;
  222. huart->UARTx->IER_DLH.IER.ERBFI = 1;
  223. }
  224. void uart0_init(uint32_t baudrate)
  225. {
  226. co_printf("%s\r\n",__func__);
  227. __SYSTEM_UART0_CLK_SELECT_96M();
  228. __SYSTEM_UART0_CLK_ENABLE();
  229. system_set_port_pull(GPIO_PA0,GPIO_PULL_UP,true);
  230. system_set_port_mux(GPIO_PORT_A, GPIO_BIT_0, PORTA0_FUNC_UART0_RXD);
  231. system_set_port_mux(GPIO_PORT_A, GPIO_BIT_1, PORTA1_FUNC_UART0_TXD);
  232. Uart0_handle.UARTx = Uart0;
  233. Uart0_handle.Init.BaudRate = baudrate;
  234. Uart0_handle.Init.DataLength = UART_DATA_LENGTH_8BIT;
  235. Uart0_handle.Init.StopBits = UART_STOPBITS_1 ;
  236. Uart0_handle.Init.Parity = UART_PARITY_NONE;
  237. Uart0_handle.Init.FIFO_Mode = UART_FIFO_ENABLE;
  238. uart_init_ex(&Uart0_handle);
  239. NVIC_SetPriority(UART0_IRQn, 5);
  240. NVIC_EnableIRQ(UART0_IRQn);
  241. /*enable recv and line status interrupt*/
  242. uart_isr_int(&Uart0_handle);
  243. }
  244. void uart1_init(uint32_t baudrate)
  245. {
  246. co_printf("%s\r\n",__func__);
  247. __SYSTEM_UART1_CLK_ENABLE();
  248. system_set_port_pull(GPIO_PA2,GPIO_PULL_UP,true);
  249. /* set PA2 and PA3 for AT command interface */
  250. system_set_port_mux(GPIO_PORT_A, GPIO_BIT_2, PORTA2_FUNC_UART1_RXD);
  251. system_set_port_mux(GPIO_PORT_A, GPIO_BIT_3, PORTA3_FUNC_UART1_TXD);
  252. Uart1_handle.UARTx = Uart1;
  253. Uart1_handle.Init.BaudRate = baudrate;
  254. Uart1_handle.Init.DataLength = UART_DATA_LENGTH_8BIT;
  255. Uart1_handle.Init.StopBits = UART_STOPBITS_1;
  256. Uart1_handle.Init.Parity = UART_PARITY_NONE;
  257. Uart1_handle.Init.FIFO_Mode = UART_FIFO_ENABLE;
  258. uart_init_ex(&Uart1_handle);
  259. NVIC_EnableIRQ(UART1_IRQn);
  260. NVIC_SetPriority(UART1_IRQn, 5);
  261. /*enable recv and line status interrupt*/
  262. uart_isr_int(&Uart1_handle);
  263. }
  264. __attribute__((section("ram_code"))) void uart0_isr(void)
  265. {
  266. uint32_t isr_id;
  267. volatile struct_UART_t * const uart_reg_ram = (volatile struct_UART_t *)UART0_BASE;
  268. isr_id = uart_reg_ram->FCR_IID.IID;
  269. if(((isr_id & 0x04) == 0x04) || ((isr_id & 0x0c) == 0x0c)) //receciver data available or character timeout indication
  270. {
  271. while(uart_reg_ram->LSR.LSR_BIT.DR)
  272. {
  273. uart_putc_noint_no_wait(UART0, (uint8_t)uart_reg_ram->DATA_DLL.DATA);
  274. }
  275. }
  276. else if((isr_id & 0x06) == 0x06)//receiver line status interrupt
  277. {
  278. uint32_t tmp = uart_reg_ram->LSR.LSR_DWORD;
  279. uart_reg_ram->FCR_IID.FCR = isr_id;
  280. uart_reg_ram->IER_DLH.IER.ELSI = 0;
  281. }
  282. }
  283. __attribute__((section("ram_code"))) void uart1_isr(void)
  284. {
  285. uint32_t isr_id;
  286. volatile struct_UART_t * const uart_reg_ram = (volatile struct_UART_t *)UART1_BASE;
  287. isr_id = uart_reg_ram->FCR_IID.IID;
  288. if(((isr_id & 0x04) == 0x04) || ((isr_id & 0x0c) == 0x0c)) //receciver data available or character timeout indication
  289. {
  290. while(uart_reg_ram->LSR.LSR_BIT.DR)
  291. {
  292. uart_putc_noint_no_wait(UART1, (uint8_t)uart_reg_ram->DATA_DLL.DATA);
  293. // rx1_buf[index_uart1++ ] = (uint8_t)uart_reg_ram->DATA_DLL.DATA;
  294. }
  295. }
  296. else if((isr_id & 0x06) == 0x06)//receiver line status interrupt
  297. {
  298. uint32_t tmp = uart_reg_ram->LSR.LSR_DWORD;
  299. uart_reg_ram->FCR_IID.FCR = isr_id;
  300. uart_reg_ram->IER_DLH.IER.ELSI = 0;
  301. }
  302. }
  303. #if 0
  304. __attribute__((section("ram_code"))) void app_at_uart_recv(void*dummy, uint8_t status)
  305. {
  306. app_at_recv_c(app_at_recv_char);
  307. uart0_read_for_hci(&app_at_recv_char, 1, app_at_uart_recv, NULL);
  308. }
  309. __attribute__((section("ram_code"))) void app_at_init(void)
  310. {
  311. /* set PA0 and PA1 for AT command interface */
  312. system_set_port_mux(GPIO_PORT_A, GPIO_BIT_0, PORTA0_FUNC_UART0_RXD);
  313. system_set_port_mux(GPIO_PORT_A, GPIO_BIT_1, PORTA1_FUNC_UART0_TXD);
  314. uart_init(UART0, 1152);
  315. fr_uart_enableIrq(UART0, Uart_irq_erbfi);
  316. uart0_read_for_hci(&app_at_recv_char, 1, app_at_uart_recv, NULL);
  317. }
  318. #endif