log_msg2.h 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643
  1. /*
  2. * Copyright (c) 2021 Nordic Semiconductor ASA
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_LOGGING_LOG_MSG2_H_
  7. #define ZEPHYR_INCLUDE_LOGGING_LOG_MSG2_H_
  8. #include <logging/log_instance.h>
  9. #include <sys/mpsc_packet.h>
  10. #include <sys/cbprintf.h>
  11. #include <sys/atomic.h>
  12. #include <sys/util.h>
  13. #include <string.h>
  14. #ifdef __GNUC__
  15. #ifndef alloca
  16. #define alloca __builtin_alloca
  17. #endif
  18. #else
  19. #include <alloca.h>
  20. #endif
  21. #ifdef __cplusplus
  22. extern "C" {
  23. #endif
  24. #define LOG_MSG2_DEBUG 0
  25. #define LOG_MSG2_DBG(...) IF_ENABLED(LOG_MSG2_DEBUG, (printk(__VA_ARGS__)))
  26. #if CONFIG_LOG_TIMESTAMP_64BIT
  27. typedef uint64_t log_timestamp_t;
  28. #else
  29. typedef uint32_t log_timestamp_t;
  30. #endif
  31. /**
  32. * @brief Log message API
  33. * @defgroup log_msg2 Log message v2 API
  34. * @ingroup logger
  35. * @{
  36. */
  37. #define Z_LOG_MSG2_LOG 0
  38. #define Z_LOG_MSG2_TRACE 1
  39. #define LOG_MSG2_GENERIC_HDR \
  40. MPSC_PBUF_HDR;\
  41. uint32_t type:1
  42. struct log_msg2_desc {
  43. LOG_MSG2_GENERIC_HDR;
  44. uint32_t domain:3;
  45. uint32_t level:3;
  46. uint32_t package_len:10;
  47. uint32_t data_len:12;
  48. uint32_t reserved:1;
  49. };
  50. struct log_msg2_trace_hdr {
  51. LOG_MSG2_GENERIC_HDR;
  52. uint32_t evt_id:5;
  53. #if CONFIG_LOG_TRACE_SHORT_TIMESTAMP
  54. uint32_t timestamp:24;
  55. #else
  56. log_timestamp_t timestamp;
  57. #endif
  58. };
  59. union log_msg2_source {
  60. const struct log_source_const_data *fixed;
  61. struct log_source_dynamic_data *dynamic;
  62. void *raw;
  63. };
  64. struct log_msg2_hdr {
  65. struct log_msg2_desc desc;
  66. /* Attempting to keep best alignment. When address is 64 bit and timestamp 32
  67. * swap the order to have 16 byte header instead of 24 byte.
  68. */
  69. #if (INTPTR_MAX > INT32_MAX) && !CONFIG_LOG_TIMESTAMP_64BIT
  70. log_timestamp_t timestamp;
  71. const void *source;
  72. #else
  73. const void *source;
  74. log_timestamp_t timestamp;
  75. #endif
  76. #if defined(__xtensa__) && !defined(CONFIG_LOG_TIMESTAMP_64BIT)
  77. /* xtensa requires that cbprintf package that follows the header is
  78. * aligned to 16 bytes. Adding padding when necessary.
  79. */
  80. uint32_t padding;
  81. #endif
  82. };
  83. struct log_msg2_trace {
  84. struct log_msg2_trace_hdr hdr;
  85. };
  86. struct log_msg2_trace_ptr {
  87. struct log_msg2_trace_hdr hdr;
  88. void *ptr;
  89. };
  90. struct log_msg2 {
  91. struct log_msg2_hdr hdr;
  92. uint8_t data[];
  93. };
  94. struct log_msg2_generic_hdr {
  95. LOG_MSG2_GENERIC_HDR;
  96. };
  97. union log_msg2_generic {
  98. union mpsc_pbuf_generic buf;
  99. struct log_msg2_generic_hdr generic;
  100. struct log_msg2_trace trace;
  101. struct log_msg2_trace_ptr trace_ptr;
  102. struct log_msg2 log;
  103. };
  104. /** @brief Method used for creating a log message.
  105. *
  106. * It is used for testing purposes to validate that expected mode was used.
  107. */
  108. enum z_log_msg2_mode {
  109. /* Runtime mode is least efficient but supports all cases thus it is
  110. * threated as a fallback method when others cannot be used.
  111. */
  112. Z_LOG_MSG2_MODE_RUNTIME,
  113. /* Mode creates statically a string package on stack and calls a
  114. * function for creating a message. It takes code size than
  115. * Z_LOG_MSG2_MODE_ZERO_COPY but is a bit slower.
  116. */
  117. Z_LOG_MSG2_MODE_FROM_STACK,
  118. /* Mode calculates size of the message and allocates it and writes
  119. * directly to the message space. It is the fastest method but requires
  120. * more code size.
  121. */
  122. Z_LOG_MSG2_MODE_ZERO_COPY,
  123. /* Mode used when synchronous logging is enabled. */
  124. Z_LOG_MSG2_MODE_SYNC
  125. };
  126. #define Z_LOG_MSG_DESC_INITIALIZER(_domain_id, _level, _plen, _dlen) \
  127. { \
  128. .valid = 0, \
  129. .busy = 0, \
  130. .type = Z_LOG_MSG2_LOG, \
  131. .domain = _domain_id, \
  132. .level = _level, \
  133. .package_len = _plen, \
  134. .data_len = _dlen, \
  135. .reserved = 0, \
  136. }
  137. /* Messages are aligned to alignment required by cbprintf package. */
  138. #define Z_LOG_MSG2_ALIGNMENT CBPRINTF_PACKAGE_ALIGNMENT
  139. #if CONFIG_LOG2_USE_VLA
  140. #define Z_LOG_MSG2_ON_STACK_ALLOC(ptr, len) \
  141. long long _ll_buf[ceiling_fraction(len, sizeof(long long))]; \
  142. long double _ld_buf[ceiling_fraction(len, sizeof(long double))]; \
  143. ptr = (sizeof(long double) == Z_LOG_MSG2_ALIGNMENT) ? \
  144. (struct log_msg2 *)_ld_buf : (struct log_msg2 *)_ll_buf; \
  145. if (IS_ENABLED(CONFIG_LOG_TEST_CLEAR_MESSAGE_SPACE)) { \
  146. /* During test fill with 0's to simplify message comparison */ \
  147. memset(ptr, 0, len); \
  148. }
  149. #else /* Z_LOG_MSG2_USE_VLA */
  150. /* When VLA cannot be used we need to trick compiler a bit and create multiple
  151. * fixed size arrays and take the smallest one that will fit the message.
  152. * Compiler will remove unused arrays and stack usage will be kept similar
  153. * to vla case, rounded to the size of the used buffer.
  154. */
  155. #define Z_LOG_MSG2_ON_STACK_ALLOC(ptr, len) \
  156. long long _ll_buf32[32 / sizeof(long long)]; \
  157. long long _ll_buf48[48 / sizeof(long long)]; \
  158. long long _ll_buf64[64 / sizeof(long long)]; \
  159. long long _ll_buf128[128 / sizeof(long long)]; \
  160. long long _ll_buf256[256 / sizeof(long long)]; \
  161. long double _ld_buf32[32 / sizeof(long double)]; \
  162. long double _ld_buf48[48 / sizeof(long double)]; \
  163. long double _ld_buf64[64 / sizeof(long double)]; \
  164. long double _ld_buf128[128 / sizeof(long double)]; \
  165. long double _ld_buf256[256 / sizeof(long double)]; \
  166. if (sizeof(long double) == Z_LOG_MSG2_ALIGNMENT) { \
  167. ptr = (len > 128) ? (struct log_msg2 *)_ld_buf256 : \
  168. ((len > 64) ? (struct log_msg2 *)_ld_buf128 : \
  169. ((len > 48) ? (struct log_msg2 *)_ld_buf64 : \
  170. ((len > 32) ? (struct log_msg2 *)_ld_buf48 : \
  171. (struct log_msg2 *)_ld_buf32)));\
  172. } else { \
  173. ptr = (len > 128) ? (struct log_msg2 *)_ll_buf256 : \
  174. ((len > 64) ? (struct log_msg2 *)_ll_buf128 : \
  175. ((len > 48) ? (struct log_msg2 *)_ll_buf64 : \
  176. ((len > 32) ? (struct log_msg2 *)_ll_buf48 : \
  177. (struct log_msg2 *)_ll_buf32)));\
  178. } \
  179. if (IS_ENABLED(CONFIG_LOG_TEST_CLEAR_MESSAGE_SPACE)) { \
  180. /* During test fill with 0's to simplify message comparison */ \
  181. memset(ptr, 0, len); \
  182. }
  183. #endif /* Z_LOG_MSG2_USE_VLA */
  184. #define Z_LOG_MSG2_ALIGN_OFFSET \
  185. sizeof(struct log_msg2_hdr)
  186. #define Z_LOG_MSG2_LEN(pkg_len, data_len) \
  187. (sizeof(struct log_msg2_hdr) + pkg_len + (data_len))
  188. #define Z_LOG_MSG2_ALIGNED_WLEN(pkg_len, data_len) \
  189. ceiling_fraction(ROUND_UP(Z_LOG_MSG2_LEN(pkg_len, data_len), \
  190. Z_LOG_MSG2_ALIGNMENT), \
  191. sizeof(uint32_t))
  192. #define Z_LOG_MSG2_SYNC(_domain_id, _source, _level, _data, _dlen, ...) do { \
  193. int _plen; \
  194. CBPRINTF_STATIC_PACKAGE(NULL, 0, _plen, Z_LOG_MSG2_ALIGN_OFFSET, \
  195. 0, __VA_ARGS__); \
  196. struct log_msg2 *_msg; \
  197. Z_LOG_MSG2_ON_STACK_ALLOC(_msg, Z_LOG_MSG2_LEN(_plen, _dlen)); \
  198. if (_plen) {\
  199. CBPRINTF_STATIC_PACKAGE(_msg->data, _plen, _plen, \
  200. Z_LOG_MSG2_ALIGN_OFFSET, \
  201. 0, __VA_ARGS__); \
  202. } \
  203. struct log_msg2_desc _desc = \
  204. Z_LOG_MSG_DESC_INITIALIZER(_domain_id, _level, \
  205. (uint32_t)_plen, _dlen); \
  206. z_log_msg2_finalize(_msg, _source, _desc, _data); \
  207. } while (0)
  208. #define Z_LOG_MSG2_STACK_CREATE(_domain_id, _source, _level, _data, _dlen, ...)\
  209. do { \
  210. int _plen; \
  211. if (GET_ARG_N(1, __VA_ARGS__) == NULL) { \
  212. _plen = 0; \
  213. } else { \
  214. CBPRINTF_STATIC_PACKAGE(NULL, 0, _plen, Z_LOG_MSG2_ALIGN_OFFSET, \
  215. 0, __VA_ARGS__); \
  216. } \
  217. struct log_msg2 *_msg; \
  218. Z_LOG_MSG2_ON_STACK_ALLOC(_msg, Z_LOG_MSG2_LEN(_plen, 0)); \
  219. if (_plen) { \
  220. CBPRINTF_STATIC_PACKAGE(_msg->data, _plen, \
  221. _plen, Z_LOG_MSG2_ALIGN_OFFSET, \
  222. 0, __VA_ARGS__);\
  223. } \
  224. struct log_msg2_desc _desc = \
  225. Z_LOG_MSG_DESC_INITIALIZER(_domain_id, _level, \
  226. (uint32_t)_plen, _dlen); \
  227. LOG_MSG2_DBG("creating message on stack: package len: %d, data len: %d\n", \
  228. _plen, (int)(_dlen)); \
  229. z_log_msg2_static_create((void *)_source, _desc, _msg->data, _data); \
  230. } while (0)
  231. #if CONFIG_LOG_SPEED
  232. #define Z_LOG_MSG2_SIMPLE_CREATE(_domain_id, _source, _level, ...) do { \
  233. int _plen; \
  234. CBPRINTF_STATIC_PACKAGE(NULL, 0, _plen, Z_LOG_MSG2_ALIGN_OFFSET, \
  235. 0, __VA_ARGS__); \
  236. size_t _msg_wlen = Z_LOG_MSG2_ALIGNED_WLEN(_plen, 0); \
  237. struct log_msg2 *_msg = z_log_msg2_alloc(_msg_wlen); \
  238. struct log_msg2_desc _desc = \
  239. Z_LOG_MSG_DESC_INITIALIZER(_domain_id, _level, (uint32_t)_plen, 0); \
  240. LOG_MSG2_DBG("creating message zero copy: package len: %d, msg: %p\n", \
  241. _plen, _msg); \
  242. if (_msg) { \
  243. CBPRINTF_STATIC_PACKAGE(_msg->data, _plen, _plen, \
  244. Z_LOG_MSG2_ALIGN_OFFSET, \
  245. 0, __VA_ARGS__); \
  246. } \
  247. z_log_msg2_finalize(_msg, (void *)_source, _desc, NULL); \
  248. } while (0)
  249. #else
  250. /* Alternative empty macro created to speed up compilation when LOG_SPEED is
  251. * disabled (default).
  252. */
  253. #define Z_LOG_MSG2_SIMPLE_CREATE(...)
  254. #endif
  255. /* Macro handles case when local variable with log message string is created.It
  256. * replaces origing string literal with that variable.
  257. */
  258. #define Z_LOG_FMT_ARGS_2(_name, ...) \
  259. COND_CODE_1(CONFIG_LOG2_FMT_SECTION, \
  260. (COND_CODE_0(NUM_VA_ARGS_LESS_1(__VA_ARGS__), \
  261. (_name), (_name, GET_ARGS_LESS_N(1, __VA_ARGS__)))), \
  262. (__VA_ARGS__))
  263. /** @brief Wrapper for log message string with arguments.
  264. *
  265. * Wrapper is replacing first argument with a variable from a dedicated memory
  266. * section if option is enabled. Macro handles the case when there is no
  267. * log message provided.
  268. *
  269. * @param _name Name of the variable with log message string. It is optionally used.
  270. * @param ... Optional log message with arguments (may be empty).
  271. */
  272. #define Z_LOG_FMT_ARGS(_name, ...) \
  273. COND_CODE_0(NUM_VA_ARGS_LESS_1(_, ##__VA_ARGS__), \
  274. (NULL), \
  275. (Z_LOG_FMT_ARGS_2(_name, ##__VA_ARGS__)))
  276. /* Macro handles case when there is no string provided, in that case variable
  277. * is not created.
  278. */
  279. #define Z_LOG_MSG2_STR_VAR_IN_SECTION(_name, ...) \
  280. COND_CODE_0(NUM_VA_ARGS_LESS_1(_, ##__VA_ARGS__), \
  281. (/* No args provided, no variable */), \
  282. (static const char _name[] \
  283. __attribute__((__section__(".log_strings"))) = \
  284. GET_ARG_N(1, __VA_ARGS__);))
  285. /** @brief Create variable in the dedicated memory section (if enabled).
  286. *
  287. * Variable is initialized with a format string from the log message.
  288. *
  289. * @param _name Variable name.
  290. * @param ... Optional log message with arguments (may be empty).
  291. */
  292. #define Z_LOG_MSG2_STR_VAR(_name, ...) \
  293. IF_ENABLED(CONFIG_LOG2_FMT_SECTION, \
  294. (Z_LOG_MSG2_STR_VAR_IN_SECTION(_name, ##__VA_ARGS__)))
  295. /** @brief Create log message and write it into the logger buffer.
  296. *
  297. * Macro handles creation of log message which includes storing log message
  298. * description, timestamp, arguments, copying string arguments into message and
  299. * copying user data into the message space. The are 3 modes of message
  300. * creation:
  301. * - at compile time message size is determined, message is allocated and
  302. * content is written directly to the message. It is the fastest but cannot
  303. * be used in user mode. Message size cannot be determined at compile time if
  304. * it contains data or string arguments which are string pointers.
  305. * - at compile time message size is determined, string package is created on
  306. * stack, message is created in function call. String package can only be
  307. * created on stack if it does not contain unexpected pointers to strings.
  308. * - string package is created at runtime. This mode has no limitations but
  309. * it is significantly slower.
  310. *
  311. * @param _try_0cpy If positive then, if possible, message content is written
  312. * directly to message. If 0 then, if possible, string package is created on
  313. * the stack and message is created in the function call.
  314. *
  315. * @param _mode Used for testing. It is set according to message creation mode
  316. * used.
  317. *
  318. * @param _cstr_cnt Number of constant strings present in the string. It is
  319. * used to help detect messages which must be runtime processed, compared to
  320. * message which can be prebuilt at compile time.
  321. *
  322. * @param _domain_id Domain ID.
  323. *
  324. * @param _source Pointer to the constant descriptor of the log message source.
  325. *
  326. * @param _level Log message level.
  327. *
  328. * @param _data Pointer to the data. Can be null.
  329. *
  330. * @param _dlen Number of data bytes. 0 if data is not provided.
  331. *
  332. * @param ... Optional string with arguments (fmt, ...). It may be empty.
  333. */
  334. #if CONFIG_LOG2_ALWAYS_RUNTIME
  335. #define Z_LOG_MSG2_CREATE2(_try_0cpy, _mode, _cstr_cnt, _domain_id, _source,\
  336. _level, _data, _dlen, ...) \
  337. do {\
  338. Z_LOG_MSG2_STR_VAR(_fmt, ##__VA_ARGS__) \
  339. z_log_msg2_runtime_create(_domain_id, (void *)_source, \
  340. _level, (uint8_t *)_data, _dlen,\
  341. Z_LOG_FMT_ARGS(_fmt, ##__VA_ARGS__));\
  342. _mode = Z_LOG_MSG2_MODE_RUNTIME; \
  343. } while (0)
  344. #elif CONFIG_LOG2_MODE_IMMEDIATE /* CONFIG_LOG2_ALWAYS_RUNTIME */
  345. #define Z_LOG_MSG2_CREATE2(_try_0cpy, _mode, _cstr_cnt, _domain_id, _source,\
  346. _level, _data, _dlen, ...) \
  347. do { \
  348. Z_LOG_MSG2_STR_VAR(_fmt, ##__VA_ARGS__); \
  349. if (CBPRINTF_MUST_RUNTIME_PACKAGE(_cstr_cnt, __VA_ARGS__)) { \
  350. LOG_MSG2_DBG("create runtime message\n");\
  351. z_log_msg2_runtime_create(_domain_id, (void *)_source, \
  352. _level, (uint8_t *)_data, _dlen,\
  353. Z_LOG_FMT_ARGS(_fmt, ##__VA_ARGS__));\
  354. _mode = Z_LOG_MSG2_MODE_RUNTIME; \
  355. } else {\
  356. Z_LOG_MSG2_SYNC(_domain_id, _source, _level, \
  357. _data, _dlen, Z_LOG_FMT_ARGS(_fmt, ##__VA_ARGS__)); \
  358. _mode = Z_LOG_MSG2_MODE_SYNC; \
  359. } \
  360. } while (0)
  361. #else /* CONFIG_LOG2_ALWAYS_RUNTIME */
  362. #define Z_LOG_MSG2_CREATE2(_try_0cpy, _mode, _cstr_cnt, _domain_id, _source,\
  363. _level, _data, _dlen, ...) \
  364. do { \
  365. Z_LOG_MSG2_STR_VAR(_fmt, ##__VA_ARGS__); \
  366. if (CBPRINTF_MUST_RUNTIME_PACKAGE(_cstr_cnt, __VA_ARGS__)) { \
  367. LOG_MSG2_DBG("create runtime message\n");\
  368. z_log_msg2_runtime_create(_domain_id, (void *)_source, \
  369. _level, (uint8_t *)_data, _dlen,\
  370. Z_LOG_FMT_ARGS(_fmt, ##__VA_ARGS__));\
  371. _mode = Z_LOG_MSG2_MODE_RUNTIME; \
  372. } else if (IS_ENABLED(CONFIG_LOG_SPEED) && _try_0cpy && ((_dlen) == 0)) {\
  373. LOG_MSG2_DBG("create zero-copy message\n");\
  374. Z_LOG_MSG2_SIMPLE_CREATE(_domain_id, _source, \
  375. _level, Z_LOG_FMT_ARGS(_fmt, ##__VA_ARGS__)); \
  376. _mode = Z_LOG_MSG2_MODE_ZERO_COPY; \
  377. } else { \
  378. LOG_MSG2_DBG("create on stack message\n");\
  379. Z_LOG_MSG2_STACK_CREATE(_domain_id, _source, _level, _data, \
  380. _dlen, Z_LOG_FMT_ARGS(_fmt, ##__VA_ARGS__)); \
  381. _mode = Z_LOG_MSG2_MODE_FROM_STACK; \
  382. } \
  383. (void)_mode; \
  384. } while (0)
  385. #endif /* CONFIG_LOG2_ALWAYS_RUNTIME */
  386. #define Z_LOG_MSG2_CREATE(_try_0cpy, _mode, _domain_id, _source,\
  387. _level, _data, _dlen, ...) \
  388. Z_LOG_MSG2_CREATE2(_try_0cpy, _mode, UTIL_CAT(Z_LOG_FUNC_PREFIX_, _level), \
  389. _domain_id, _source, _level, _data, _dlen, \
  390. Z_LOG_STR(_level, __VA_ARGS__))
  391. #define Z_TRACING_LOG_HDR_INIT(name, id) \
  392. struct log_msg2_trace name = { \
  393. .hdr = { \
  394. .type = Z_LOG_MSG2_TRACE, \
  395. .valid = 1, \
  396. .busy = 0, \
  397. .evt_id = id, \
  398. } \
  399. }
  400. /** @brief Finalize message.
  401. *
  402. * Finalization includes setting source, copying data and timestamp in the
  403. * message followed by committing the message.
  404. *
  405. * @param msg Message.
  406. *
  407. * @param source Address of the source descriptor.
  408. *
  409. * @param desc Message descriptor.
  410. *
  411. * @param data Data.
  412. */
  413. void z_log_msg2_finalize(struct log_msg2 *msg, const void *source,
  414. const struct log_msg2_desc desc, const void *data);
  415. /** @brief Create simple message from message details and string package.
  416. *
  417. * @param source Source.
  418. *
  419. * @param desc Message descriptor.
  420. *
  421. * @param package Package.
  422. *
  423. * @oaram data Data.
  424. */
  425. __syscall void z_log_msg2_static_create(const void *source,
  426. const struct log_msg2_desc desc,
  427. uint8_t *package, const void *data);
  428. /** @brief Create message at runtime.
  429. *
  430. * Function allows to build any log message based on input data. Processing
  431. * time is significantly higher than statically message creating.
  432. *
  433. * @param domain_id Domain ID.
  434. *
  435. * @param source Source.
  436. *
  437. * @param level Log level.
  438. *
  439. * @param data Data.
  440. *
  441. * @param dlen Data length.
  442. *
  443. * @param fmt String.
  444. *
  445. * @param ap Variable list of string arguments.
  446. */
  447. __syscall void z_log_msg2_runtime_vcreate(uint8_t domain_id, const void *source,
  448. uint8_t level, const void *data,
  449. size_t dlen, const char *fmt,
  450. va_list ap);
  451. /** @brief Create message at runtime.
  452. *
  453. * Function allows to build any log message based on input data. Processing
  454. * time is significantly higher than statically message creating.
  455. *
  456. * @param domain_id Domain ID.
  457. *
  458. * @param source Source.
  459. *
  460. * @param level Log level.
  461. *
  462. * @param data Data.
  463. *
  464. * @param dlen Data length.
  465. *
  466. * @param fmt String.
  467. *
  468. * @param ... String arguments.
  469. */
  470. static inline void z_log_msg2_runtime_create(uint8_t domain_id,
  471. const void *source,
  472. uint8_t level, const void *data,
  473. size_t dlen, const char *fmt, ...)
  474. {
  475. va_list ap;
  476. va_start(ap, fmt);
  477. z_log_msg2_runtime_vcreate(domain_id, source, level,
  478. data, dlen, fmt, ap);
  479. va_end(ap);
  480. }
  481. static inline bool z_log_item_is_msg(union log_msg2_generic *msg)
  482. {
  483. return msg->generic.type == Z_LOG_MSG2_LOG;
  484. }
  485. /** @brief Get total length (in 32 bit words) of a log message.
  486. *
  487. * @param desc Log message descriptor.
  488. *
  489. * @return Length.
  490. */
  491. static inline uint32_t log_msg2_get_total_wlen(const struct log_msg2_desc desc)
  492. {
  493. return Z_LOG_MSG2_ALIGNED_WLEN(desc.package_len, desc.data_len);
  494. }
  495. /** @brief Get length of the log item.
  496. *
  497. * @param item Item.
  498. *
  499. * @return Length in 32 bit words.
  500. */
  501. static inline uint32_t log_msg2_generic_get_wlen(const union mpsc_pbuf_generic *item)
  502. {
  503. union log_msg2_generic *generic_msg = (union log_msg2_generic *)item;
  504. if (z_log_item_is_msg(generic_msg)) {
  505. struct log_msg2 *msg = (struct log_msg2 *)generic_msg;
  506. return log_msg2_get_total_wlen(msg->hdr.desc);
  507. }
  508. /* trace TODO */
  509. return 0;
  510. }
  511. /** @brief Get log message domain ID.
  512. *
  513. * @param msg Log message.
  514. *
  515. * @return Domain ID
  516. */
  517. static inline uint8_t log_msg2_get_domain(struct log_msg2 *msg)
  518. {
  519. return msg->hdr.desc.domain;
  520. }
  521. /** @brief Get log message level.
  522. *
  523. * @param msg Log message.
  524. *
  525. * @return Log level.
  526. */
  527. static inline uint8_t log_msg2_get_level(struct log_msg2 *msg)
  528. {
  529. return msg->hdr.desc.level;
  530. }
  531. /** @brief Get message source data.
  532. *
  533. * @param msg Log message.
  534. *
  535. * @return Pointer to the source data.
  536. */
  537. static inline const void *log_msg2_get_source(struct log_msg2 *msg)
  538. {
  539. return msg->hdr.source;
  540. }
  541. /** @brief Get timestamp.
  542. *
  543. * @param msg Log message.
  544. *
  545. * @return Timestamp.
  546. */
  547. static inline log_timestamp_t log_msg2_get_timestamp(struct log_msg2 *msg)
  548. {
  549. return msg->hdr.timestamp;
  550. }
  551. /** @brief Get data buffer.
  552. *
  553. * @param msg log message.
  554. *
  555. * @param len location where data length is written.
  556. *
  557. * @return pointer to the data buffer.
  558. */
  559. static inline uint8_t *log_msg2_get_data(struct log_msg2 *msg, size_t *len)
  560. {
  561. *len = msg->hdr.desc.data_len;
  562. return msg->data + msg->hdr.desc.package_len;
  563. }
  564. /** @brief Get string package.
  565. *
  566. * @param msg log message.
  567. *
  568. * @param len location where string package length is written.
  569. *
  570. * @return pointer to the package.
  571. */
  572. static inline uint8_t *log_msg2_get_package(struct log_msg2 *msg, size_t *len)
  573. {
  574. *len = msg->hdr.desc.package_len;
  575. return msg->data;
  576. }
  577. /**
  578. * @}
  579. */
  580. #include <syscalls/log_msg2.h>
  581. #ifdef __cplusplus
  582. }
  583. #endif
  584. #endif /* ZEPHYR_INCLUDE_LOGGING_LOG_MSG2_H_ */