uart_rxnb_txb.c 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295
  1. /*
  2. * Copyright (c) 2006-2024 RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2021-06-16 KyleChan the first version
  9. */
  10. #include <rtthread.h>
  11. #include "utest.h"
  12. #include <rtdevice.h>
  13. #include <stdlib.h>
  14. #ifdef UTEST_SERIAL_TC
  15. static struct rt_serial_device *serial;
  16. static rt_sem_t rx_sem;
  17. static rt_uint8_t uart_over_flag;
  18. static rt_bool_t uart_result = RT_TRUE;
  19. static rt_err_t uart_find(void)
  20. {
  21. serial = (struct rt_serial_device *)rt_device_find(RT_SERIAL_TC_DEVICE_NAME);
  22. if (serial == RT_NULL)
  23. {
  24. LOG_E("find %s device failed!\n", RT_SERIAL_TC_DEVICE_NAME);
  25. return -RT_ERROR;
  26. }
  27. return RT_EOK;
  28. }
  29. static rt_err_t uart_rx_indicate(rt_device_t device, rt_size_t size)
  30. {
  31. rt_sem_release(rx_sem);
  32. return RT_EOK;
  33. }
  34. static void uart_send_entry(void *parameter)
  35. {
  36. rt_uint8_t *uart_write_buffer;
  37. rt_uint16_t send_len;
  38. rt_uint32_t i = 0;
  39. send_len = *(rt_uint16_t *)parameter;
  40. /* assign send buffer */
  41. uart_write_buffer = (rt_uint8_t *)rt_malloc(send_len);
  42. if (uart_write_buffer == RT_NULL)
  43. {
  44. LOG_E("Without spare memory for uart dma!");
  45. uart_result = RT_FALSE;
  46. return;
  47. }
  48. rt_memset(uart_write_buffer, 0, send_len);
  49. for (i = 0; i < send_len; i++)
  50. {
  51. uart_write_buffer[i] = (rt_uint8_t)i;
  52. }
  53. /* send buffer */
  54. if (rt_device_write(&serial->parent, 0, uart_write_buffer, send_len) != send_len)
  55. {
  56. LOG_E("device write failed\r\n");
  57. }
  58. rt_free(uart_write_buffer);
  59. }
  60. static void uart_rec_entry(void *parameter)
  61. {
  62. rt_uint16_t rev_len;
  63. rev_len = *(rt_uint16_t *)parameter;
  64. rt_uint8_t *uart_write_buffer;
  65. uart_write_buffer = (rt_uint8_t *)rt_calloc(1, sizeof(rt_uint8_t) * (rev_len + 1));
  66. rt_int32_t cnt, i;
  67. rt_uint8_t last_old_data;
  68. rt_bool_t fisrt_flag = RT_TRUE;
  69. rt_uint32_t all_receive_length = 0;
  70. while (1)
  71. {
  72. rt_err_t result;
  73. result = rt_sem_take(rx_sem, RT_WAITING_FOREVER);
  74. if (result != RT_EOK)
  75. {
  76. LOG_E("take sem err in recv.");
  77. }
  78. cnt = rt_device_read(&serial->parent, 0, (void *)uart_write_buffer, rev_len);
  79. if (cnt == 0)
  80. {
  81. continue;
  82. }
  83. if (fisrt_flag != RT_TRUE)
  84. {
  85. if ((rt_uint8_t)(last_old_data + 1) != uart_write_buffer[0])
  86. {
  87. LOG_E("_Read Different data -> former data: %x, current data: %x.", last_old_data, uart_write_buffer[0]);
  88. uart_result = RT_FALSE;
  89. rt_free(uart_write_buffer);
  90. return;
  91. }
  92. }
  93. else
  94. {
  95. fisrt_flag = RT_FALSE;
  96. }
  97. for (i = 0; i < cnt - 1; i++)
  98. {
  99. if ((rt_uint8_t)(uart_write_buffer[i] + 1) != uart_write_buffer[i + 1])
  100. {
  101. LOG_E("Read Different data -> former data: %x, current data: %x.", uart_write_buffer[i], uart_write_buffer[i + 1]);
  102. uart_result = RT_FALSE;
  103. rt_free(uart_write_buffer);
  104. return;
  105. }
  106. }
  107. all_receive_length += cnt;
  108. if (all_receive_length >= rev_len)
  109. break;
  110. else
  111. last_old_data = uart_write_buffer[cnt - 1];
  112. }
  113. rt_free(uart_write_buffer);
  114. uart_over_flag = RT_TRUE;
  115. }
  116. static rt_err_t uart_api(rt_uint16_t test_buf)
  117. {
  118. rt_thread_t thread_send = RT_NULL;
  119. rt_thread_t thread_recv = RT_NULL;
  120. rt_err_t result = RT_EOK;
  121. result = uart_find();
  122. if (result != RT_EOK)
  123. {
  124. return -RT_ERROR;
  125. }
  126. rx_sem = rt_sem_create("rx_sem", 0, RT_IPC_FLAG_PRIO);
  127. if (rx_sem == RT_NULL)
  128. {
  129. LOG_E("Init sem failed.");
  130. uart_result = RT_FALSE;
  131. return -RT_ERROR;
  132. }
  133. /* reinitialize */
  134. struct serial_configure config = RT_SERIAL_CONFIG_DEFAULT;
  135. config.baud_rate = BAUD_RATE_115200;
  136. config.rx_bufsz = RT_SERIAL_TC_RXBUF_SIZE;
  137. config.tx_bufsz = RT_SERIAL_TC_TXBUF_SIZE;
  138. #ifdef RT_SERIAL_USING_DMA
  139. config.dma_ping_bufsz = RT_SERIAL_TC_RXBUF_SIZE / 2;
  140. #endif
  141. rt_device_control(&serial->parent, RT_DEVICE_CTRL_CONFIG, &config);
  142. result = rt_device_open(&serial->parent, RT_DEVICE_FLAG_RX_NON_BLOCKING | RT_DEVICE_FLAG_TX_BLOCKING);
  143. if (result != RT_EOK)
  144. {
  145. LOG_E("Open uart device failed.");
  146. uart_result = RT_FALSE;
  147. rt_sem_delete(rx_sem);
  148. return -RT_ERROR;
  149. }
  150. rt_int32_t timeout = 5000;
  151. rt_device_control(&serial->parent, RT_SERIAL_CTRL_SET_RX_TIMEOUT, (void *)&timeout);
  152. /* set receive callback function */
  153. result = rt_device_set_rx_indicate(&serial->parent, uart_rx_indicate);
  154. if (result != RT_EOK)
  155. {
  156. goto __exit;
  157. }
  158. thread_recv = rt_thread_create("uart_recv", uart_rec_entry, &test_buf, 1024, RT_THREAD_PRIORITY_MAX - 5, 10);
  159. thread_send = rt_thread_create("uart_send", uart_send_entry, &test_buf, 1024, RT_THREAD_PRIORITY_MAX - 4, 10);
  160. if (thread_send != RT_NULL && thread_recv != RT_NULL)
  161. {
  162. rt_thread_startup(thread_recv);
  163. rt_thread_startup(thread_send);
  164. }
  165. else
  166. {
  167. result = -RT_ERROR;
  168. goto __exit;
  169. }
  170. while (1)
  171. {
  172. if (uart_result != RT_TRUE)
  173. {
  174. LOG_E("The test for uart dma is failure.");
  175. result = -RT_ERROR;
  176. goto __exit;
  177. }
  178. if (uart_over_flag == RT_TRUE)
  179. {
  180. goto __exit;
  181. }
  182. /* waiting for test over */
  183. rt_thread_mdelay(5);
  184. }
  185. __exit:
  186. if (rx_sem)
  187. rt_sem_delete(rx_sem);
  188. rt_device_close(&serial->parent);
  189. rt_thread_mdelay(5);
  190. uart_over_flag = RT_FALSE;
  191. return result;
  192. }
  193. static void tc_uart_api(void)
  194. {
  195. rt_uint32_t count = 0;
  196. rt_uint16_t num = 0;
  197. rt_uint32_t i = 0;
  198. for (i = 1; i < 10; i++)
  199. {
  200. if (uart_api(RT_SERIAL_TC_TXBUF_SIZE * i + i % 2) == RT_EOK)
  201. LOG_I("data_lens [%4d], it is correct to read and write data. [%d] count testing.", RT_SERIAL_TC_TXBUF_SIZE * i + i % 2, ++count);
  202. else
  203. {
  204. LOG_E("uart test error");
  205. goto __exit;
  206. }
  207. }
  208. for (i = 1; i < 10; i++)
  209. {
  210. if (uart_api(RT_SERIAL_TC_RXBUF_SIZE * i + i % 2) == RT_EOK)
  211. LOG_I("data_lens [%4d], it is correct to read and write data. [%d] count testing.", RT_SERIAL_TC_RXBUF_SIZE * i + i % 2, ++count);
  212. else
  213. {
  214. LOG_E("uart test error");
  215. goto __exit;
  216. }
  217. }
  218. srand(rt_tick_get());
  219. while (RT_SERIAL_TC_SEND_ITERATIONS - count)
  220. {
  221. num = (rand() % 1000) + 1;
  222. if (uart_api(num) == RT_EOK)
  223. LOG_I("data_lens [%4d], it is correct to read and write data. [%d] count testing.", num, ++count);
  224. else
  225. {
  226. LOG_E("uart test error");
  227. break;
  228. }
  229. }
  230. __exit:
  231. uassert_true(uart_result == RT_TRUE);
  232. }
  233. static rt_err_t utest_tc_init(void)
  234. {
  235. LOG_I("UART TEST: Please connect Tx and Rx directly for self testing.");
  236. return RT_EOK;
  237. }
  238. static rt_err_t utest_tc_cleanup(void)
  239. {
  240. rx_sem = RT_NULL;
  241. uart_result = RT_TRUE;
  242. uart_over_flag = RT_FALSE;
  243. rt_device_t uart_dev = rt_device_find(RT_SERIAL_TC_DEVICE_NAME);
  244. while (rt_device_close(uart_dev) != -RT_ERROR);
  245. return RT_EOK;
  246. }
  247. static void testcase(void)
  248. {
  249. UTEST_UNIT_RUN(tc_uart_api);
  250. }
  251. UTEST_TC_EXPORT(testcase, "testcases.drivers.uart_rxnb_txb", utest_tc_init, utest_tc_cleanup, 30);
  252. #endif /* TC_UART_USING_TC */