uart_rtt.c 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. /*
  2. * Copyright (c) 2019 omSquare s.r.o.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <drivers/uart.h>
  7. #include <SEGGER_RTT.h>
  8. #define DT_DRV_COMPAT segger_rtt_uart
  9. struct uart_rtt_config {
  10. void *up_buffer;
  11. size_t up_size;
  12. void *down_buffer;
  13. size_t down_size;
  14. uint8_t channel;
  15. };
  16. struct uart_rtt_data {
  17. #ifdef CONFIG_UART_ASYNC_API
  18. uart_callback_t callback;
  19. void *user_data;
  20. #endif /* CONFIG_UART_ASYNC_API */
  21. };
  22. static inline
  23. const struct uart_rtt_config *get_dev_config(const struct device *dev)
  24. {
  25. return dev->config;
  26. }
  27. static inline
  28. struct uart_rtt_data *get_dev_data(const struct device *dev)
  29. {
  30. return dev->data;
  31. }
  32. static int uart_rtt_init(const struct device *dev)
  33. {
  34. /*
  35. * Channel 0 is initialized at compile-time, Kconfig ensures that
  36. * it is configured in correct, non-blocking mode. Other channels
  37. * need to be configured at run-time.
  38. */
  39. if (get_dev_config(dev)) {
  40. const struct uart_rtt_config *cfg = get_dev_config(dev);
  41. SEGGER_RTT_ConfigUpBuffer(cfg->channel, dev->name,
  42. cfg->up_buffer, cfg->up_size,
  43. SEGGER_RTT_MODE_NO_BLOCK_SKIP);
  44. SEGGER_RTT_ConfigDownBuffer(cfg->channel, dev->name,
  45. cfg->down_buffer, cfg->down_size,
  46. SEGGER_RTT_MODE_NO_BLOCK_SKIP);
  47. }
  48. return 0;
  49. }
  50. /**
  51. * @brief Poll the device for input.
  52. *
  53. * @param dev UART device struct
  54. * @param c Pointer to character
  55. *
  56. * @return 0 if a character arrived, -1 if the input buffer if empty.
  57. */
  58. static int uart_rtt_poll_in(const struct device *dev, unsigned char *c)
  59. {
  60. unsigned int ch =
  61. get_dev_config(dev) ? get_dev_config(dev)->channel : 0;
  62. unsigned int ret = SEGGER_RTT_Read(ch, c, 1);
  63. return ret ? 0 : -1;
  64. }
  65. /**
  66. * @brief Output a character in polled mode.
  67. *
  68. * @param dev UART device struct
  69. * @param c Character to send
  70. */
  71. static void uart_rtt_poll_out(const struct device *dev, unsigned char c)
  72. {
  73. unsigned int ch =
  74. get_dev_config(dev) ? get_dev_config(dev)->channel : 0;
  75. SEGGER_RTT_Write(ch, &c, 1);
  76. }
  77. #ifdef CONFIG_UART_ASYNC_API
  78. static int uart_rtt_callback_set(const struct device *dev,
  79. uart_callback_t callback, void *user_data)
  80. {
  81. get_dev_data(dev)->callback = callback;
  82. get_dev_data(dev)->user_data = user_data;
  83. return 0;
  84. }
  85. static int uart_rtt_tx(const struct device *dev,
  86. const uint8_t *buf, size_t len, int32_t timeout)
  87. {
  88. const struct uart_rtt_config *cfg = get_dev_config(dev);
  89. struct uart_rtt_data *data = get_dev_data(dev);
  90. unsigned int ch = cfg ? cfg->channel : 0;
  91. ARG_UNUSED(timeout);
  92. /* RTT mutex cannot be claimed in ISRs */
  93. if (k_is_in_isr()) {
  94. return -ENOTSUP;
  95. }
  96. /* Claim the RTT lock */
  97. if (k_mutex_lock(&rtt_term_mutex, K_NO_WAIT) != 0) {
  98. return -EBUSY;
  99. }
  100. /* Output the buffer */
  101. SEGGER_RTT_WriteNoLock(ch, buf, len);
  102. /* Return RTT lock */
  103. SEGGER_RTT_UNLOCK();
  104. /* Send the TX complete callback */
  105. if (data->callback) {
  106. struct uart_event evt = {
  107. .type = UART_TX_DONE,
  108. .data.tx.buf = buf,
  109. .data.tx.len = len
  110. };
  111. data->callback(dev, &evt, data->user_data);
  112. }
  113. return 0;
  114. }
  115. static int uart_rtt_tx_abort(const struct device *dev)
  116. {
  117. /* RTT TX is a memcpy, there is never a transmission to abort */
  118. ARG_UNUSED(dev);
  119. return -EFAULT;
  120. }
  121. static int uart_rtt_rx_enable(const struct device *dev,
  122. uint8_t *buf, size_t len, int32_t timeout)
  123. {
  124. /* SEGGER RTT reception is implemented as a direct memory write to RAM
  125. * by a connected debugger. As such there is no hardware interrupt
  126. * or other mechanism to know when the debugger has added data to be
  127. * read. Asynchronous RX does not make sense in such a context, and is
  128. * therefore not supported.
  129. */
  130. ARG_UNUSED(dev);
  131. ARG_UNUSED(buf);
  132. ARG_UNUSED(len);
  133. ARG_UNUSED(timeout);
  134. return -ENOTSUP;
  135. }
  136. static int uart_rtt_rx_disable(const struct device *dev)
  137. {
  138. /* Asynchronous RX not supported, see uart_rtt_rx_enable */
  139. ARG_UNUSED(dev);
  140. return -EFAULT;
  141. }
  142. static int uart_rtt_rx_buf_rsp(const struct device *dev,
  143. uint8_t *buf, size_t len)
  144. {
  145. /* Asynchronous RX not supported, see uart_rtt_rx_enable */
  146. ARG_UNUSED(dev);
  147. ARG_UNUSED(buf);
  148. ARG_UNUSED(len);
  149. return -ENOTSUP;
  150. }
  151. #endif /* CONFIG_UART_ASYNC_API */
  152. static const struct uart_driver_api uart_rtt_driver_api = {
  153. .poll_in = uart_rtt_poll_in,
  154. .poll_out = uart_rtt_poll_out,
  155. #ifdef CONFIG_UART_ASYNC_API
  156. .callback_set = uart_rtt_callback_set,
  157. .tx = uart_rtt_tx,
  158. .tx_abort = uart_rtt_tx_abort,
  159. .rx_enable = uart_rtt_rx_enable,
  160. .rx_buf_rsp = uart_rtt_rx_buf_rsp,
  161. .rx_disable = uart_rtt_rx_disable,
  162. #endif /* CONFIG_UART_ASYNC_API */
  163. };
  164. #define UART_RTT(idx) DT_NODELABEL(rtt##idx)
  165. #define UART_RTT_PROP(idx, prop) DT_PROP(UART_RTT(idx), prop)
  166. #define UART_RTT_CONFIG_NAME(idx) uart_rtt##idx##_config
  167. #define UART_RTT_CONFIG(idx) \
  168. static \
  169. uint8_t uart_rtt##idx##_tx_buf[UART_RTT_PROP(idx, tx_buffer_size)]; \
  170. static \
  171. uint8_t uart_rtt##idx##_rx_buf[UART_RTT_PROP(idx, rx_buffer_size)]; \
  172. \
  173. static const struct uart_rtt_config UART_RTT_CONFIG_NAME(idx) = { \
  174. .up_buffer = uart_rtt##idx##_tx_buf, \
  175. .up_size = sizeof(uart_rtt##idx##_tx_buf), \
  176. .down_buffer = uart_rtt##idx##_rx_buf, \
  177. .down_size = sizeof(uart_rtt##idx##_rx_buf), \
  178. }
  179. #define UART_RTT_INIT(idx, config) \
  180. struct uart_rtt_data uart_rtt##idx##_data; \
  181. \
  182. DEVICE_DT_DEFINE(UART_RTT(idx), uart_rtt_init, NULL, \
  183. &uart_rtt##idx##_data, config, \
  184. PRE_KERNEL_2, CONFIG_KERNEL_INIT_PRIORITY_DEVICE, \
  185. &uart_rtt_driver_api)
  186. #ifdef CONFIG_UART_RTT_0
  187. UART_RTT_INIT(0, NULL);
  188. #endif
  189. #ifdef CONFIG_UART_RTT_1
  190. UART_RTT_CONFIG(1);
  191. UART_RTT_INIT(1, &UART_RTT_CONFIG_NAME(1));
  192. #endif
  193. #ifdef CONFIG_UART_RTT_2
  194. UART_RTT_CONFIG(2);
  195. UART_RTT_INIT(2, &UART_RTT_CONFIG_NAME(2));
  196. #endif
  197. #ifdef CONFIG_UART_RTT_3
  198. UART_RTT_CONFIG(3);
  199. UART_RTT_INIT(3, &UART_RTT_CONFIG_NAME(3));
  200. #endif