log_backend_rtt.c 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320
  1. /*
  2. * Copyright (c) 2018 omSquare s.r.o.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <logging/log_backend.h>
  7. #include <logging/log_core.h>
  8. #include <logging/log_msg.h>
  9. #include <logging/log_output.h>
  10. #include <logging/log_backend_std.h>
  11. #include <SEGGER_RTT.h>
  12. #ifndef CONFIG_LOG_BACKEND_RTT_BUFFER_SIZE
  13. #define CONFIG_LOG_BACKEND_RTT_BUFFER_SIZE 0
  14. #endif
  15. #ifndef CONFIG_LOG_BACKEND_RTT_MESSAGE_SIZE
  16. #define CONFIG_LOG_BACKEND_RTT_MESSAGE_SIZE 0
  17. #endif
  18. #ifndef CONFIG_LOG_BACKEND_RTT_OUTPUT_BUFFER_SIZE
  19. #define CONFIG_LOG_BACKEND_RTT_OUTPUT_BUFFER_SIZE 0
  20. #endif
  21. #ifndef CONFIG_LOG_BACKEND_RTT_RETRY_DELAY_MS
  22. /* Long enough to detect host presence */
  23. #define CONFIG_LOG_BACKEND_RTT_RETRY_DELAY_MS 10
  24. #endif
  25. #ifndef CONFIG_LOG_BACKEND_RTT_RETRY_CNT
  26. /* Big enough to detect host presence */
  27. #define CONFIG_LOG_BACKEND_RTT_RETRY_CNT 10
  28. #endif
  29. #define DROP_MAX 99
  30. #define DROP_MSG "messages dropped: \r\n"
  31. #define DROP_MSG_LEN (sizeof(DROP_MSG) - 1)
  32. #define MESSAGE_SIZE CONFIG_LOG_BACKEND_RTT_MESSAGE_SIZE
  33. #define CHAR_BUF_SIZE \
  34. ((IS_ENABLED(CONFIG_LOG_BACKEND_RTT_MODE_BLOCK) && \
  35. !IS_ENABLED(CONFIG_LOG_IMMEDIATE)) ? \
  36. CONFIG_LOG_BACKEND_RTT_OUTPUT_BUFFER_SIZE : 1)
  37. #define RTT_LOCK() \
  38. COND_CODE_0(CONFIG_LOG_BACKEND_RTT_BUFFER, (SEGGER_RTT_LOCK()), ())
  39. #define RTT_UNLOCK() \
  40. COND_CODE_0(CONFIG_LOG_BACKEND_RTT_BUFFER, (SEGGER_RTT_UNLOCK()), ())
  41. #define RTT_BUFFER_SIZE \
  42. COND_CODE_0(CONFIG_LOG_BACKEND_RTT_BUFFER, \
  43. (0), (CONFIG_LOG_BACKEND_RTT_BUFFER_SIZE))
  44. static const char *drop_msg = DROP_MSG;
  45. static uint8_t rtt_buf[RTT_BUFFER_SIZE];
  46. static uint8_t line_buf[MESSAGE_SIZE + DROP_MSG_LEN];
  47. static uint8_t *line_pos;
  48. static uint8_t char_buf[CHAR_BUF_SIZE];
  49. static int drop_cnt;
  50. static int drop_warn;
  51. static bool panic_mode;
  52. static bool host_present;
  53. static int data_out_block_mode(uint8_t *data, size_t length, void *ctx);
  54. static int data_out_drop_mode(uint8_t *data, size_t length, void *ctx);
  55. static int char_out_drop_mode(uint8_t data);
  56. static int line_out_drop_mode(void);
  57. static inline bool is_sync_mode(void)
  58. {
  59. return IS_ENABLED(CONFIG_LOG_IMMEDIATE) || panic_mode;
  60. }
  61. static inline bool is_panic_mode(void)
  62. {
  63. return panic_mode;
  64. }
  65. static int data_out_drop_mode(uint8_t *data, size_t length, void *ctx)
  66. {
  67. (void) ctx;
  68. uint8_t *pos;
  69. if (is_sync_mode()) {
  70. return data_out_block_mode(data, length, ctx);
  71. }
  72. for (pos = data; pos < data + length; pos++) {
  73. if (char_out_drop_mode(*pos)) {
  74. break;
  75. }
  76. }
  77. return (int) (pos - data);
  78. }
  79. static int char_out_drop_mode(uint8_t data)
  80. {
  81. if (data == '\n') {
  82. if (line_out_drop_mode()) {
  83. return 1;
  84. }
  85. line_pos = line_buf;
  86. return 0;
  87. }
  88. if (line_pos < line_buf + MESSAGE_SIZE - 1) {
  89. *line_pos++ = data;
  90. }
  91. /* not enough space in line buffer, we have to wait for EOL */
  92. return 0;
  93. }
  94. static int line_out_drop_mode(void)
  95. {
  96. /* line cannot be empty */
  97. __ASSERT_NO_MSG(line_pos > line_buf);
  98. /* Handle the case if line contains only '\n' */
  99. if (line_pos - line_buf == 1) {
  100. line_pos++;
  101. }
  102. *(line_pos - 1) = '\r';
  103. *line_pos++ = '\n';
  104. if (drop_cnt > 0 && !drop_warn) {
  105. int cnt = MIN(drop_cnt, DROP_MAX);
  106. __ASSERT_NO_MSG(line_pos - line_buf <= MESSAGE_SIZE);
  107. memmove(line_buf + DROP_MSG_LEN, line_buf, line_pos - line_buf);
  108. (void)memcpy(line_buf, drop_msg, DROP_MSG_LEN);
  109. line_pos += DROP_MSG_LEN;
  110. drop_warn = 1;
  111. if (cnt < 10) {
  112. line_buf[DROP_MSG_LEN - 2] = ' ';
  113. line_buf[DROP_MSG_LEN - 3] = (uint8_t) ('0' + cnt);
  114. line_buf[DROP_MSG_LEN - 4] = ' ';
  115. } else {
  116. line_buf[DROP_MSG_LEN - 2] = (uint8_t) ('0' + cnt % 10);
  117. line_buf[DROP_MSG_LEN - 3] = (uint8_t) ('0' + cnt / 10);
  118. line_buf[DROP_MSG_LEN - 4] = '>';
  119. }
  120. }
  121. int ret;
  122. RTT_LOCK();
  123. ret = SEGGER_RTT_WriteSkipNoLock(CONFIG_LOG_BACKEND_RTT_BUFFER,
  124. line_buf, line_pos - line_buf);
  125. RTT_UNLOCK();
  126. if (ret == 0) {
  127. drop_cnt++;
  128. } else {
  129. drop_cnt = 0;
  130. drop_warn = 0;
  131. }
  132. return 0;
  133. }
  134. static void on_failed_write(int retry_cnt)
  135. {
  136. if (retry_cnt == 0) {
  137. host_present = false;
  138. } else if (is_sync_mode()) {
  139. k_busy_wait(USEC_PER_MSEC *
  140. CONFIG_LOG_BACKEND_RTT_RETRY_DELAY_MS);
  141. } else {
  142. k_msleep(CONFIG_LOG_BACKEND_RTT_RETRY_DELAY_MS);
  143. }
  144. }
  145. static void on_write(int retry_cnt)
  146. {
  147. host_present = true;
  148. if (is_panic_mode()) {
  149. /* In panic mode block on each write until host reads it. This
  150. * way it is ensured that if system resets all messages are read
  151. * by the host. While pending on data being read by the host we
  152. * must also detect situation where host is disconnected.
  153. */
  154. while (SEGGER_RTT_HasDataUp(CONFIG_LOG_BACKEND_RTT_BUFFER) &&
  155. host_present) {
  156. on_failed_write(retry_cnt--);
  157. }
  158. }
  159. }
  160. static int data_out_block_mode(uint8_t *data, size_t length, void *ctx)
  161. {
  162. int ret = 0;
  163. /* This function is also called in drop mode for synchronous operation
  164. * in that case retry is undesired */
  165. int retry_cnt = IS_ENABLED(CONFIG_LOG_BACKEND_RTT_MODE_BLOCK) ?
  166. CONFIG_LOG_BACKEND_RTT_RETRY_CNT : 1;
  167. do {
  168. if (!is_sync_mode()) {
  169. RTT_LOCK();
  170. ret = SEGGER_RTT_WriteSkipNoLock(CONFIG_LOG_BACKEND_RTT_BUFFER,
  171. data, length);
  172. RTT_UNLOCK();
  173. } else {
  174. ret = SEGGER_RTT_WriteSkipNoLock(CONFIG_LOG_BACKEND_RTT_BUFFER,
  175. data, length);
  176. }
  177. if (ret) {
  178. on_write(retry_cnt);
  179. } else if (host_present) {
  180. retry_cnt--;
  181. on_failed_write(retry_cnt);
  182. } else {
  183. }
  184. } while ((ret == 0) && host_present);
  185. return ((ret == 0) && host_present) ? 0 : length;
  186. }
  187. LOG_OUTPUT_DEFINE(log_output_rtt,
  188. IS_ENABLED(CONFIG_LOG_BACKEND_RTT_MODE_BLOCK) ?
  189. data_out_block_mode : data_out_drop_mode,
  190. char_buf, sizeof(char_buf));
  191. static void put(const struct log_backend *const backend,
  192. struct log_msg *msg)
  193. {
  194. uint32_t flag = IS_ENABLED(CONFIG_LOG_BACKEND_RTT_SYST_ENABLE) ?
  195. LOG_OUTPUT_FLAG_FORMAT_SYST : 0;
  196. log_backend_std_put(&log_output_rtt, flag, msg);
  197. }
  198. static void log_backend_rtt_cfg(void)
  199. {
  200. SEGGER_RTT_ConfigUpBuffer(CONFIG_LOG_BACKEND_RTT_BUFFER, "Logger",
  201. rtt_buf, sizeof(rtt_buf),
  202. SEGGER_RTT_MODE_NO_BLOCK_SKIP);
  203. }
  204. static void log_backend_rtt_init(struct log_backend const *const backend)
  205. {
  206. if (CONFIG_LOG_BACKEND_RTT_BUFFER > 0) {
  207. log_backend_rtt_cfg();
  208. }
  209. host_present = true;
  210. line_pos = line_buf;
  211. }
  212. static void panic(struct log_backend const *const backend)
  213. {
  214. panic_mode = true;
  215. log_backend_std_panic(&log_output_rtt);
  216. }
  217. static void dropped(const struct log_backend *const backend, uint32_t cnt)
  218. {
  219. ARG_UNUSED(backend);
  220. log_backend_std_dropped(&log_output_rtt, cnt);
  221. }
  222. static void sync_string(const struct log_backend *const backend,
  223. struct log_msg_ids src_level, uint32_t timestamp,
  224. const char *fmt, va_list ap)
  225. {
  226. uint32_t flag = IS_ENABLED(CONFIG_LOG_BACKEND_RTT_SYST_ENABLE) ?
  227. LOG_OUTPUT_FLAG_FORMAT_SYST : 0;
  228. log_backend_std_sync_string(&log_output_rtt, flag, src_level,
  229. timestamp, fmt, ap);
  230. }
  231. static void sync_hexdump(const struct log_backend *const backend,
  232. struct log_msg_ids src_level, uint32_t timestamp,
  233. const char *metadata, const uint8_t *data, uint32_t length)
  234. {
  235. uint32_t flag = IS_ENABLED(CONFIG_LOG_BACKEND_RTT_SYST_ENABLE) ?
  236. LOG_OUTPUT_FLAG_FORMAT_SYST : 0;
  237. log_backend_std_sync_hexdump(&log_output_rtt, flag, src_level,
  238. timestamp, metadata, data, length);
  239. }
  240. static void process(const struct log_backend *const backend,
  241. union log_msg2_generic *msg)
  242. {
  243. uint32_t flags = log_backend_std_get_flags();
  244. log_output_msg2_process(&log_output_rtt, &msg->log, flags);
  245. }
  246. const struct log_backend_api log_backend_rtt_api = {
  247. .process = IS_ENABLED(CONFIG_LOG2) ? process : NULL,
  248. .put = IS_ENABLED(CONFIG_LOG_MODE_DEFERRED) ? put : NULL,
  249. .put_sync_string = IS_ENABLED(CONFIG_LOG_MODE_IMMEDIATE) ?
  250. sync_string : NULL,
  251. .put_sync_hexdump = IS_ENABLED(CONFIG_LOG_MODE_IMMEDIATE) ?
  252. sync_hexdump : NULL,
  253. .panic = panic,
  254. .init = log_backend_rtt_init,
  255. .dropped = IS_ENABLED(CONFIG_LOG_IMMEDIATE) ? NULL : dropped,
  256. };
  257. LOG_BACKEND_DEFINE(log_backend_rtt, log_backend_rtt_api, true);