log_backend.h 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297
  1. /*
  2. * Copyright (c) 2018 Nordic Semiconductor ASA
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_LOGGING_LOG_BACKEND_H_
  7. #define ZEPHYR_INCLUDE_LOGGING_LOG_BACKEND_H_
  8. #include <logging/log_msg.h>
  9. #include <logging/log_msg2.h>
  10. #include <stdarg.h>
  11. #include <sys/__assert.h>
  12. #include <sys/util.h>
  13. #ifdef __cplusplus
  14. extern "C" {
  15. #endif
  16. /**
  17. * @brief Logger backend interface
  18. * @defgroup log_backend Logger backend interface
  19. * @ingroup logger
  20. * @{
  21. */
  22. /* Forward declaration of the log_backend type. */
  23. struct log_backend;
  24. /**
  25. * @brief Logger backend API.
  26. */
  27. struct log_backend_api {
  28. void (*process)(const struct log_backend *const backend,
  29. union log_msg2_generic *msg);
  30. void (*put)(const struct log_backend *const backend,
  31. struct log_msg *msg);
  32. void (*put_sync_string)(const struct log_backend *const backend,
  33. struct log_msg_ids src_level, uint32_t timestamp,
  34. const char *fmt, va_list ap);
  35. void (*put_sync_hexdump)(const struct log_backend *const backend,
  36. struct log_msg_ids src_level, uint32_t timestamp,
  37. const char *metadata, const uint8_t *data, uint32_t len);
  38. void (*dropped)(const struct log_backend *const backend, uint32_t cnt);
  39. void (*panic)(const struct log_backend *const backend);
  40. void (*init)(const struct log_backend *const backend);
  41. };
  42. /**
  43. * @brief Logger backend control block.
  44. */
  45. struct log_backend_control_block {
  46. void *ctx;
  47. uint8_t id;
  48. bool active;
  49. };
  50. /**
  51. * @brief Logger backend structure.
  52. */
  53. struct log_backend {
  54. const struct log_backend_api *api;
  55. struct log_backend_control_block *cb;
  56. const char *name;
  57. bool autostart;
  58. };
  59. extern const struct log_backend __log_backends_start[];
  60. extern const struct log_backend __log_backends_end[];
  61. /**
  62. * @brief Macro for creating a logger backend instance.
  63. *
  64. * @param _name Name of the backend instance.
  65. * @param _api Logger backend API.
  66. * @param _autostart If true backend is initialized and activated together
  67. * with the logger subsystem.
  68. * @param ... Optional context.
  69. */
  70. #define LOG_BACKEND_DEFINE(_name, _api, _autostart, ...) \
  71. static struct log_backend_control_block UTIL_CAT(backend_cb_, _name) = \
  72. { \
  73. COND_CODE_0(NUM_VA_ARGS_LESS_1(_, ##__VA_ARGS__), \
  74. (), (.ctx = __VA_ARGS__,)) \
  75. .id = 0, \
  76. .active = false, \
  77. }; \
  78. static const STRUCT_SECTION_ITERABLE(log_backend, _name) = \
  79. { \
  80. .api = &_api, \
  81. .cb = &UTIL_CAT(backend_cb_, _name), \
  82. .name = STRINGIFY(_name), \
  83. .autostart = _autostart \
  84. }
  85. /**
  86. * @brief Put message with log entry to the backend.
  87. *
  88. * @param[in] backend Pointer to the backend instance.
  89. * @param[in] msg Pointer to message with log entry.
  90. */
  91. static inline void log_backend_put(const struct log_backend *const backend,
  92. struct log_msg *msg)
  93. {
  94. __ASSERT_NO_MSG(backend != NULL);
  95. __ASSERT_NO_MSG(msg != NULL);
  96. backend->api->put(backend, msg);
  97. }
  98. static inline void log_backend_msg2_process(
  99. const struct log_backend *const backend,
  100. union log_msg2_generic *msg)
  101. {
  102. __ASSERT_NO_MSG(backend != NULL);
  103. __ASSERT_NO_MSG(msg != NULL);
  104. backend->api->process(backend, msg);
  105. }
  106. /**
  107. * @brief Synchronously process log message.
  108. *
  109. * @param[in] backend Pointer to the backend instance.
  110. * @param[in] src_level Message details.
  111. * @param[in] timestamp Timestamp.
  112. * @param[in] fmt Log string.
  113. * @param[in] ap Log string arguments.
  114. */
  115. static inline void log_backend_put_sync_string(
  116. const struct log_backend *const backend,
  117. struct log_msg_ids src_level,
  118. uint32_t timestamp, const char *fmt,
  119. va_list ap)
  120. {
  121. __ASSERT_NO_MSG(backend != NULL);
  122. if (backend->api->put_sync_string) {
  123. backend->api->put_sync_string(backend, src_level,
  124. timestamp, fmt, ap);
  125. }
  126. }
  127. /**
  128. * @brief Synchronously process log hexdump_message.
  129. *
  130. * @param[in] backend Pointer to the backend instance.
  131. * @param[in] src_level Message details.
  132. * @param[in] timestamp Timestamp.
  133. * @param[in] metadata Raw string associated with the data.
  134. * @param[in] data Data.
  135. * @param[in] len Data length.
  136. */
  137. static inline void log_backend_put_sync_hexdump(
  138. const struct log_backend *const backend,
  139. struct log_msg_ids src_level,
  140. uint32_t timestamp, const char *metadata,
  141. const uint8_t *data, uint32_t len)
  142. {
  143. __ASSERT_NO_MSG(backend != NULL);
  144. if (backend->api->put_sync_hexdump) {
  145. backend->api->put_sync_hexdump(backend, src_level, timestamp,
  146. metadata, data, len);
  147. }
  148. }
  149. /**
  150. * @brief Notify backend about dropped log messages.
  151. *
  152. * Function is optional.
  153. *
  154. * @param[in] backend Pointer to the backend instance.
  155. * @param[in] cnt Number of dropped logs since last notification.
  156. */
  157. static inline void log_backend_dropped(const struct log_backend *const backend,
  158. uint32_t cnt)
  159. {
  160. __ASSERT_NO_MSG(backend != NULL);
  161. if (backend->api->dropped != NULL) {
  162. backend->api->dropped(backend, cnt);
  163. }
  164. }
  165. /**
  166. * @brief Reconfigure backend to panic mode.
  167. *
  168. * @param[in] backend Pointer to the backend instance.
  169. */
  170. static inline void log_backend_panic(const struct log_backend *const backend)
  171. {
  172. __ASSERT_NO_MSG(backend != NULL);
  173. backend->api->panic(backend);
  174. }
  175. /**
  176. * @brief Set backend id.
  177. *
  178. * @note It is used internally by the logger.
  179. *
  180. * @param backend Pointer to the backend instance.
  181. * @param id ID.
  182. */
  183. static inline void log_backend_id_set(const struct log_backend *const backend,
  184. uint8_t id)
  185. {
  186. __ASSERT_NO_MSG(backend != NULL);
  187. backend->cb->id = id;
  188. }
  189. /**
  190. * @brief Get backend id.
  191. *
  192. * @note It is used internally by the logger.
  193. *
  194. * @param[in] backend Pointer to the backend instance.
  195. * @return Id.
  196. */
  197. static inline uint8_t log_backend_id_get(const struct log_backend *const backend)
  198. {
  199. __ASSERT_NO_MSG(backend != NULL);
  200. return backend->cb->id;
  201. }
  202. /**
  203. * @brief Get backend.
  204. *
  205. * @param[in] idx Pointer to the backend instance.
  206. *
  207. * @return Pointer to the backend instance.
  208. */
  209. static inline const struct log_backend *log_backend_get(uint32_t idx)
  210. {
  211. return &__log_backends_start[idx];
  212. }
  213. /**
  214. * @brief Get number of backends.
  215. *
  216. * @return Number of backends.
  217. */
  218. static inline int log_backend_count_get(void)
  219. {
  220. return __log_backends_end - __log_backends_start;
  221. }
  222. /**
  223. * @brief Activate backend.
  224. *
  225. * @param[in] backend Pointer to the backend instance.
  226. * @param[in] ctx User context.
  227. */
  228. static inline void log_backend_activate(const struct log_backend *const backend,
  229. void *ctx)
  230. {
  231. __ASSERT_NO_MSG(backend != NULL);
  232. backend->cb->ctx = ctx;
  233. backend->cb->active = true;
  234. }
  235. /**
  236. * @brief Deactivate backend.
  237. *
  238. * @param[in] backend Pointer to the backend instance.
  239. */
  240. static inline void log_backend_deactivate(
  241. const struct log_backend *const backend)
  242. {
  243. __ASSERT_NO_MSG(backend != NULL);
  244. backend->cb->active = false;
  245. }
  246. /**
  247. * @brief Check state of the backend.
  248. *
  249. * @param[in] backend Pointer to the backend instance.
  250. *
  251. * @return True if backend is active, false otherwise.
  252. */
  253. static inline bool log_backend_is_active(
  254. const struct log_backend *const backend)
  255. {
  256. __ASSERT_NO_MSG(backend != NULL);
  257. return backend->cb->active;
  258. }
  259. /**
  260. * @}
  261. */
  262. #ifdef __cplusplus
  263. }
  264. #endif
  265. #endif /* ZEPHYR_INCLUDE_LOGGING_LOG_BACKEND_H_ */