log_msg.h 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494
  1. /*
  2. * Copyright (c) 2018 Nordic Semiconductor ASA
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_LOGGING_LOG_MSG_H_
  7. #define ZEPHYR_INCLUDE_LOGGING_LOG_MSG_H_
  8. #include <sys/atomic.h>
  9. #include <sys/util.h>
  10. #include <string.h>
  11. #include <logging/log_msg2.h>
  12. #ifdef __cplusplus
  13. extern "C" {
  14. #endif
  15. /**
  16. * @brief Log message API
  17. * @defgroup log_msg Log message API
  18. * @ingroup logger
  19. * @{
  20. */
  21. /** @brief Log argument type.
  22. *
  23. * Should preferably be equivalent to a native word size.
  24. */
  25. typedef unsigned long log_arg_t;
  26. /** @brief Maximum number of arguments in the standard log entry.
  27. *
  28. * It is limited by 4 bit nargs field in the log message.
  29. */
  30. #define LOG_MAX_NARGS 15
  31. /** @brief Number of arguments in the log entry which fits in one chunk.*/
  32. #ifdef CONFIG_64BIT
  33. #define LOG_MSG_NARGS_SINGLE_CHUNK 4U
  34. #else
  35. #define LOG_MSG_NARGS_SINGLE_CHUNK 3U
  36. #endif
  37. /** @brief Number of arguments in the head of extended standard log message..*/
  38. #define LOG_MSG_NARGS_HEAD_CHUNK \
  39. (LOG_MSG_NARGS_SINGLE_CHUNK - (sizeof(void *)/sizeof(log_arg_t)))
  40. /** @brief Maximal amount of bytes in the hexdump entry which fits in one chunk.
  41. */
  42. #define LOG_MSG_HEXDUMP_BYTES_SINGLE_CHUNK \
  43. (LOG_MSG_NARGS_SINGLE_CHUNK * sizeof(log_arg_t))
  44. /** @brief Number of bytes in the first chunk of hexdump message if message
  45. * consists of more than one chunk.
  46. */
  47. #define LOG_MSG_HEXDUMP_BYTES_HEAD_CHUNK \
  48. (LOG_MSG_HEXDUMP_BYTES_SINGLE_CHUNK - sizeof(void *))
  49. /** @brief Number of bytes that can be stored in chunks following head chunk
  50. * in hexdump log message.
  51. */
  52. #define HEXDUMP_BYTES_CONT_MSG \
  53. (sizeof(struct log_msg) - sizeof(void *))
  54. #define ARGS_CONT_MSG (HEXDUMP_BYTES_CONT_MSG / sizeof(log_arg_t))
  55. /** @brief Flag indicating standard log message. */
  56. #define LOG_MSG_TYPE_STD 0U
  57. /** @brief Flag indicating hexdump log message. */
  58. #define LOG_MSG_TYPE_HEXDUMP 1
  59. /** @brief Common part of log message header. */
  60. #define COMMON_PARAM_HDR() \
  61. uint16_t type : 1; \
  62. uint16_t ext : 1
  63. /** @brief Number of bits used for storing length of hexdump log message. */
  64. #define LOG_MSG_HEXDUMP_LENGTH_BITS 14
  65. /** @brief Maximum length of log hexdump message. */
  66. #define LOG_MSG_HEXDUMP_MAX_LENGTH (BIT(LOG_MSG_HEXDUMP_LENGTH_BITS) - 1)
  67. /** @brief Part of log message header identifying source and level. */
  68. struct log_msg_ids {
  69. uint16_t level : 3; /*!< Severity. */
  70. uint16_t domain_id : 3; /*!< Originating domain. */
  71. uint16_t source_id : 10; /*!< Source ID. */
  72. };
  73. /** Part of log message header common to standard and hexdump log message. */
  74. struct log_msg_generic_hdr {
  75. COMMON_PARAM_HDR();
  76. uint16_t reserved : 14;
  77. };
  78. /** Part of log message header specific to standard log message. */
  79. struct log_msg_std_hdr {
  80. COMMON_PARAM_HDR();
  81. uint16_t reserved : 10;
  82. uint16_t nargs : 4;
  83. };
  84. /** Part of log message header specific to hexdump log message. */
  85. struct log_msg_hexdump_hdr {
  86. COMMON_PARAM_HDR();
  87. uint16_t length : LOG_MSG_HEXDUMP_LENGTH_BITS;
  88. };
  89. /** Log message header structure */
  90. struct log_msg_hdr {
  91. atomic_t ref_cnt; /*!< Reference counter for tracking message users. */
  92. union log_msg_hdr_params {
  93. struct log_msg_generic_hdr generic;
  94. struct log_msg_std_hdr std;
  95. struct log_msg_hexdump_hdr hexdump;
  96. uint16_t raw;
  97. } params;
  98. struct log_msg_ids ids; /*!< Identification part of the message.*/
  99. uint32_t timestamp; /*!< Timestamp. */
  100. };
  101. /** @brief Data part of log message. */
  102. union log_msg_head_data {
  103. log_arg_t args[LOG_MSG_NARGS_SINGLE_CHUNK];
  104. uint8_t bytes[LOG_MSG_HEXDUMP_BYTES_SINGLE_CHUNK];
  105. };
  106. /** @brief Data part of extended log message. */
  107. struct log_msg_ext_head_data {
  108. struct log_msg_cont *next;
  109. union log_msg_ext_head_data_data {
  110. log_arg_t args[LOG_MSG_NARGS_HEAD_CHUNK];
  111. uint8_t bytes[LOG_MSG_HEXDUMP_BYTES_HEAD_CHUNK];
  112. } data;
  113. };
  114. /** @brief Log message structure. */
  115. struct log_msg {
  116. struct log_msg *next; /*!< Used by logger core list.*/
  117. struct log_msg_hdr hdr; /*!< Message header. */
  118. const char *str;
  119. union log_msg_data {
  120. union log_msg_head_data single;
  121. struct log_msg_ext_head_data ext;
  122. } payload; /*!< Message data. */
  123. };
  124. /** @brief Chunks following message head when message is extended. */
  125. struct log_msg_cont {
  126. struct log_msg_cont *next; /*!< Pointer to the next chunk. */
  127. union log_msg_cont_data {
  128. log_arg_t args[ARGS_CONT_MSG];
  129. uint8_t bytes[HEXDUMP_BYTES_CONT_MSG];
  130. } payload;
  131. };
  132. /** @brief Log message */
  133. union log_msg_chunk {
  134. struct log_msg head;
  135. struct log_msg_cont cont;
  136. };
  137. /** @brief Function for initialization of the log message pool. */
  138. void log_msg_pool_init(void);
  139. /** @brief Function for indicating that message is in use.
  140. *
  141. * @details Message can be used (read) by multiple users. Internal reference
  142. * counter is atomically increased. See @ref log_msg_put.
  143. *
  144. * @param msg Message.
  145. */
  146. void log_msg_get(struct log_msg *msg);
  147. /** @brief Function for indicating that message is no longer in use.
  148. *
  149. * @details Internal reference counter is atomically decreased. If reference
  150. * counter equals 0 message is freed.
  151. *
  152. * @param msg Message.
  153. */
  154. void log_msg_put(struct log_msg *msg);
  155. /** @brief Get domain ID of the message.
  156. *
  157. * @param msg Message
  158. *
  159. * @return Domain ID.
  160. */
  161. static inline uint32_t log_msg_domain_id_get(struct log_msg *msg)
  162. {
  163. return msg->hdr.ids.domain_id;
  164. }
  165. /** @brief Get source ID (module or instance) of the message.
  166. *
  167. * @param msg Message
  168. *
  169. * @return Source ID.
  170. */
  171. static inline uint32_t log_msg_source_id_get(struct log_msg *msg)
  172. {
  173. return msg->hdr.ids.source_id;
  174. }
  175. /** @brief Get severity level of the message.
  176. *
  177. * @param msg Message
  178. *
  179. * @return Severity message.
  180. */
  181. static inline uint32_t log_msg_level_get(struct log_msg *msg)
  182. {
  183. return msg->hdr.ids.level;
  184. }
  185. /** @brief Get timestamp of the message.
  186. *
  187. * @param msg Message
  188. *
  189. * @return Timestamp value.
  190. */
  191. static inline uint32_t log_msg_timestamp_get(struct log_msg *msg)
  192. {
  193. return msg->hdr.timestamp;
  194. }
  195. /** @brief Check if message is of standard type.
  196. *
  197. * @param msg Message
  198. *
  199. * @retval true Standard message.
  200. * @retval false Hexdump message.
  201. */
  202. static inline bool log_msg_is_std(struct log_msg *msg)
  203. {
  204. return (msg->hdr.params.generic.type == LOG_MSG_TYPE_STD);
  205. }
  206. /** @brief Returns number of arguments in standard log message.
  207. *
  208. * @param msg Standard log message.
  209. *
  210. * @return Number of arguments.
  211. */
  212. uint32_t log_msg_nargs_get(struct log_msg *msg);
  213. /** @brief Gets argument from standard log message.
  214. *
  215. * @param msg Standard log message.
  216. * @param arg_idx Argument index.
  217. *
  218. * @return Argument value or 0 if arg_idx exceeds number of arguments in the
  219. * message.
  220. */
  221. log_arg_t log_msg_arg_get(struct log_msg *msg, uint32_t arg_idx);
  222. /** @brief Gets pointer to the unformatted string from standard log message.
  223. *
  224. * @param msg Standard log message.
  225. *
  226. * @return Pointer to the string.
  227. */
  228. const char *log_msg_str_get(struct log_msg *msg);
  229. /** @brief Allocates chunks for hexdump message and copies the data.
  230. *
  231. * @details Function resets header and sets following fields:
  232. * - message type
  233. * - length
  234. *
  235. * @note Allocation and partial filling is combined for performance reasons.
  236. *
  237. * @param str String.
  238. * @param data Data.
  239. * @param length Data length.
  240. *
  241. * @return Pointer to allocated head of the message or NULL
  242. */
  243. struct log_msg *log_msg_hexdump_create(const char *str,
  244. const uint8_t *data,
  245. uint32_t length);
  246. /** @brief Put data into hexdump log message.
  247. *
  248. * @param[in] msg Message.
  249. * @param[in] data Data to be copied.
  250. * @param[in, out] length Input: requested amount. Output: actual amount.
  251. * @param[in] offset Offset.
  252. */
  253. void log_msg_hexdump_data_put(struct log_msg *msg,
  254. uint8_t *data,
  255. size_t *length,
  256. size_t offset);
  257. /** @brief Get data from hexdump log message.
  258. *
  259. * @param[in] msg Message.
  260. * @param[in] data Buffer for data.
  261. * @param[in, out] length Input: requested amount. Output: actual amount.
  262. * @param[in] offset Offset.
  263. */
  264. void log_msg_hexdump_data_get(struct log_msg *msg,
  265. uint8_t *data,
  266. size_t *length,
  267. size_t offset);
  268. union log_msg_chunk *log_msg_no_space_handle(void);
  269. /** @brief Allocate single chunk from the pool.
  270. *
  271. * @return Pointer to the allocated chunk or NULL if failed to allocate.
  272. */
  273. union log_msg_chunk *log_msg_chunk_alloc(void);
  274. /** @brief Allocate chunk for standard log message.
  275. *
  276. * @return Allocated chunk of NULL.
  277. */
  278. static inline struct log_msg *z_log_msg_std_alloc(void)
  279. {
  280. struct log_msg *msg = (struct log_msg *)log_msg_chunk_alloc();
  281. if (msg != NULL) {
  282. /* all fields reset to 0, reference counter to 1 */
  283. msg->hdr.ref_cnt = 1;
  284. msg->hdr.params.raw = 0U;
  285. msg->hdr.params.std.type = LOG_MSG_TYPE_STD;
  286. if (IS_ENABLED(CONFIG_USERSPACE)) {
  287. /* it may be used in msg_free() function. */
  288. msg->hdr.ids.level = 0;
  289. msg->hdr.ids.domain_id = 0;
  290. msg->hdr.ids.source_id = 0;
  291. }
  292. }
  293. return msg;
  294. }
  295. /** @brief Create standard log message with no arguments.
  296. *
  297. * @details Function resets header and sets following fields:
  298. * - message type
  299. * - string pointer
  300. *
  301. * @return Pointer to allocated head of the message or NULL.
  302. */
  303. static inline struct log_msg *log_msg_create_0(const char *str)
  304. {
  305. struct log_msg *msg = z_log_msg_std_alloc();
  306. if (msg != NULL) {
  307. msg->str = str;
  308. }
  309. return msg;
  310. }
  311. /** @brief Create standard log message with one argument.
  312. *
  313. * @details Function resets header and sets following fields:
  314. * - message type
  315. * - string pointer
  316. * - number of arguments
  317. * - argument
  318. *
  319. * @param str String.
  320. * @param arg1 Argument.
  321. *
  322. * @return Pointer to allocated head of the message or NULL.
  323. */
  324. static inline struct log_msg *log_msg_create_1(const char *str,
  325. log_arg_t arg1)
  326. {
  327. struct log_msg *msg = z_log_msg_std_alloc();
  328. if (msg != NULL) {
  329. msg->str = str;
  330. msg->hdr.params.std.nargs = 1U;
  331. msg->payload.single.args[0] = arg1;
  332. }
  333. return msg;
  334. }
  335. /** @brief Create standard log message with two arguments.
  336. *
  337. * @details Function resets header and sets following fields:
  338. * - message type
  339. * - string pointer
  340. * - number of arguments
  341. * - arguments
  342. *
  343. * @param str String.
  344. * @param arg1 Argument 1.
  345. * @param arg2 Argument 2.
  346. *
  347. * @return Pointer to allocated head of the message or NULL.
  348. */
  349. static inline struct log_msg *log_msg_create_2(const char *str,
  350. log_arg_t arg1,
  351. log_arg_t arg2)
  352. {
  353. struct log_msg *msg = z_log_msg_std_alloc();
  354. if (msg != NULL) {
  355. msg->str = str;
  356. msg->hdr.params.std.nargs = 2U;
  357. msg->payload.single.args[0] = arg1;
  358. msg->payload.single.args[1] = arg2;
  359. }
  360. return msg;
  361. }
  362. /** @brief Create standard log message with three arguments.
  363. *
  364. * @details Function resets header and sets following fields:
  365. * - message type
  366. * - string pointer
  367. * - number of arguments
  368. * - arguments
  369. *
  370. * @param str String.
  371. * @param arg1 Argument 1.
  372. * @param arg2 Argument 2.
  373. * @param arg3 Argument 3.
  374. *
  375. * @return Pointer to allocated head of the message or NULL.
  376. */
  377. static inline struct log_msg *log_msg_create_3(const char *str,
  378. log_arg_t arg1,
  379. log_arg_t arg2,
  380. log_arg_t arg3)
  381. {
  382. struct log_msg *msg = z_log_msg_std_alloc();
  383. if (msg != NULL) {
  384. msg->str = str;
  385. msg->hdr.params.std.nargs = 3U;
  386. msg->payload.single.args[0] = arg1;
  387. msg->payload.single.args[1] = arg2;
  388. msg->payload.single.args[2] = arg3;
  389. }
  390. return msg;
  391. }
  392. /** @brief Create standard log message with variable number of arguments.
  393. *
  394. * @details Function resets header and sets following fields:
  395. * - message type
  396. * - string pointer
  397. * - number of arguments
  398. * - arguments
  399. *
  400. * @param str String.
  401. * @param args Array with arguments.
  402. * @param nargs Number of arguments.
  403. *
  404. * @return Pointer to allocated head of the message or NULL.
  405. */
  406. struct log_msg *log_msg_create_n(const char *str,
  407. log_arg_t *args,
  408. uint32_t nargs);
  409. /**
  410. * @brief Get number of free blocks from the log mem pool
  411. */
  412. uint32_t log_msg_mem_get_free(void);
  413. /**
  414. * @brief Get number of used blocks from the log mem pool
  415. */
  416. uint32_t log_msg_mem_get_used(void);
  417. /**
  418. * @brief Get max used blocks from the log mem pool
  419. */
  420. uint32_t log_msg_mem_get_max_used(void);
  421. /**
  422. * @}
  423. */
  424. #ifdef __cplusplus
  425. }
  426. #endif
  427. #endif /* ZEPHYR_INCLUDE_LOGGING_LOG_MSG_H_ */