log_core.h 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931
  1. /*
  2. * Copyright (c) 2018 Nordic Semiconductor ASA
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_LOGGING_LOG_CORE_H_
  7. #define ZEPHYR_INCLUDE_LOGGING_LOG_CORE_H_
  8. #include <logging/log_msg.h>
  9. #include <logging/log_core2.h>
  10. #include <logging/log_instance.h>
  11. #include <stdbool.h>
  12. #include <stdint.h>
  13. #include <stdarg.h>
  14. #include <syscall.h>
  15. #include <sys/util.h>
  16. #define LOG_LEVEL_NONE 0U
  17. #define LOG_LEVEL_ERR 1U
  18. #define LOG_LEVEL_WRN 2U
  19. #define LOG_LEVEL_INF 3U
  20. #define LOG_LEVEL_DBG 4U
  21. #ifdef __cplusplus
  22. extern "C" {
  23. #endif
  24. #ifndef CONFIG_LOG
  25. #define CONFIG_LOG_DEFAULT_LEVEL 0U
  26. #define CONFIG_LOG_MAX_LEVEL 0U
  27. #endif
  28. #if !defined(CONFIG_LOG) || defined(CONFIG_LOG_MINIMAL)
  29. #define CONFIG_LOG_DOMAIN_ID 0U
  30. #endif
  31. #define LOG_FUNCTION_PREFIX_MASK \
  32. (((uint32_t)IS_ENABLED(CONFIG_LOG_FUNC_NAME_PREFIX_ERR) << \
  33. LOG_LEVEL_ERR) | \
  34. ((uint32_t)IS_ENABLED(CONFIG_LOG_FUNC_NAME_PREFIX_WRN) << \
  35. LOG_LEVEL_WRN) | \
  36. ((uint32_t)IS_ENABLED(CONFIG_LOG_FUNC_NAME_PREFIX_INF) << \
  37. LOG_LEVEL_INF) | \
  38. ((uint32_t)IS_ENABLED(CONFIG_LOG_FUNC_NAME_PREFIX_DBG) << LOG_LEVEL_DBG))
  39. /** @brief Macro for returning local level value if defined or default.
  40. *
  41. * Check @ref IS_ENABLED macro for detailed explanation of the trick.
  42. */
  43. #define Z_LOG_RESOLVED_LEVEL(_level, _default) \
  44. Z_LOG_RESOLVED_LEVEL1(_level, _default)
  45. #define Z_LOG_RESOLVED_LEVEL1(_level, _default) \
  46. __COND_CODE(_LOG_XXXX##_level, (_level), (_default))
  47. #define _LOG_XXXX0 _LOG_YYYY,
  48. #define _LOG_XXXX0U _LOG_YYYY,
  49. #define _LOG_XXXX1 _LOG_YYYY,
  50. #define _LOG_XXXX1U _LOG_YYYY,
  51. #define _LOG_XXXX2 _LOG_YYYY,
  52. #define _LOG_XXXX2U _LOG_YYYY,
  53. #define _LOG_XXXX3 _LOG_YYYY,
  54. #define _LOG_XXXX3U _LOG_YYYY,
  55. #define _LOG_XXXX4 _LOG_YYYY,
  56. #define _LOG_XXXX4U _LOG_YYYY,
  57. /**
  58. * @brief Macro for conditional code generation if provided log level allows.
  59. *
  60. * Macro behaves similarly to standard \#if \#else \#endif clause. The
  61. * difference is that it is evaluated when used and not when header file is
  62. * included.
  63. *
  64. * @param _eval_level Evaluated level. If level evaluates to one of existing log
  65. * log level (1-4) then macro evaluates to _iftrue.
  66. * @param _iftrue Code that should be inserted when evaluated to true. Note,
  67. * that parameter must be provided in brackets.
  68. * @param _iffalse Code that should be inserted when evaluated to false.
  69. * Note, that parameter must be provided in brackets.
  70. */
  71. #define Z_LOG_EVAL(_eval_level, _iftrue, _iffalse) \
  72. Z_LOG_EVAL1(_eval_level, _iftrue, _iffalse)
  73. #define Z_LOG_EVAL1(_eval_level, _iftrue, _iffalse) \
  74. __COND_CODE(_LOG_ZZZZ##_eval_level, _iftrue, _iffalse)
  75. #define _LOG_ZZZZ1 _LOG_YYYY,
  76. #define _LOG_ZZZZ1U _LOG_YYYY,
  77. #define _LOG_ZZZZ2 _LOG_YYYY,
  78. #define _LOG_ZZZZ2U _LOG_YYYY,
  79. #define _LOG_ZZZZ3 _LOG_YYYY,
  80. #define _LOG_ZZZZ3U _LOG_YYYY,
  81. #define _LOG_ZZZZ4 _LOG_YYYY,
  82. #define _LOG_ZZZZ4U _LOG_YYYY,
  83. /** @brief Macro for getting log level for given module.
  84. *
  85. * It is evaluated to LOG_LEVEL if defined. Otherwise CONFIG_LOG_DEFAULT_LEVEL
  86. * is used.
  87. */
  88. #define _LOG_LEVEL() Z_LOG_RESOLVED_LEVEL(LOG_LEVEL, CONFIG_LOG_DEFAULT_LEVEL)
  89. /**
  90. * @def LOG_CONST_ID_GET
  91. * @brief Macro for getting ID of the element of the section.
  92. *
  93. * @param _addr Address of the element.
  94. */
  95. #define LOG_CONST_ID_GET(_addr) \
  96. COND_CODE_1(CONFIG_LOG, ((__log_level ? log_const_source_id(_addr) : 0)), (0))
  97. /**
  98. * @def LOG_CURRENT_MODULE_ID
  99. * @brief Macro for getting ID of current module.
  100. */
  101. #define LOG_CURRENT_MODULE_ID() (__log_level != 0 ? \
  102. log_const_source_id(__log_current_const_data) : 0U)
  103. /**
  104. * @def LOG_CURRENT_DYNAMIC_DATA_ADDR
  105. * @brief Macro for getting address of dynamic structure of current module.
  106. */
  107. #define LOG_CURRENT_DYNAMIC_DATA_ADDR() (__log_level ? \
  108. __log_current_dynamic_data : (struct log_source_dynamic_data *)0U)
  109. /** @brief Macro for getting ID of the element of the section.
  110. *
  111. * @param _addr Address of the element.
  112. */
  113. #define LOG_DYNAMIC_ID_GET(_addr) \
  114. COND_CODE_1(CONFIG_LOG, ((__log_level ? log_dynamic_source_id(_addr) : 0)), (0))
  115. /* Set of defines that are set to 1 if function name prefix is enabled for given level. */
  116. #define Z_LOG_FUNC_PREFIX_0U 0
  117. #define Z_LOG_FUNC_PREFIX_1U COND_CODE_1(CONFIG_LOG_FUNC_NAME_PREFIX_ERR, (1), (0))
  118. #define Z_LOG_FUNC_PREFIX_2U COND_CODE_1(CONFIG_LOG_FUNC_NAME_PREFIX_WRN, (1), (0))
  119. #define Z_LOG_FUNC_PREFIX_3U COND_CODE_1(CONFIG_LOG_FUNC_NAME_PREFIX_INF, (1), (0))
  120. #define Z_LOG_FUNC_PREFIX_4U COND_CODE_1(CONFIG_LOG_FUNC_NAME_PREFIX_DBG, (1), (0))
  121. /**
  122. * @brief Macro for optional injection of function name as first argument of
  123. * formatted string. COND_CODE_0() macro is used to handle no arguments
  124. * case.
  125. *
  126. * The purpose of this macro is to prefix string literal with format specifier
  127. * for function name and inject function name as first argument. In order to
  128. * handle string with no arguments _LOG_Z_EVAL is used.
  129. */
  130. #define Z_LOG_STR_WITH_PREFIX2(...) \
  131. "%s: " GET_ARG_N(1, __VA_ARGS__), (const char *)__func__\
  132. COND_CODE_0(NUM_VA_ARGS_LESS_1(__VA_ARGS__),\
  133. (),\
  134. (, GET_ARGS_LESS_N(1, __VA_ARGS__))\
  135. )
  136. /* Macro handles case when no format string is provided: e.g. LOG_DBG().
  137. * Handling of format string is deferred to the next level macro.
  138. */
  139. #define Z_LOG_STR_WITH_PREFIX(...) \
  140. COND_CODE_0(NUM_VA_ARGS_LESS_1(_, ##__VA_ARGS__), \
  141. ("%s", (const char *)__func__), \
  142. (Z_LOG_STR_WITH_PREFIX2(__VA_ARGS__)))
  143. /**
  144. * @brief Handle optional injection of function name as the first argument.
  145. *
  146. * Additionally, macro is handling the empty message case.
  147. */
  148. #define Z_LOG_STR(_level, ...) \
  149. COND_CODE_1(UTIL_CAT(Z_LOG_FUNC_PREFIX_##_level), \
  150. (Z_LOG_STR_WITH_PREFIX(__VA_ARGS__)), (__VA_ARGS__))
  151. /******************************************************************************/
  152. /****************** Internal macros for log frontend **************************/
  153. /******************************************************************************/
  154. /**@brief Second stage for Z_LOG_NARGS_POSTFIX */
  155. #define Z_LOG_NARGS_POSTFIX_IMPL( \
  156. _ignored, \
  157. _0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, \
  158. _11, _12, _13, _14, N, ...) N
  159. /**@brief Macro to get the postfix for further log message processing.
  160. *
  161. * Logs with more than 3 arguments are processed in a generic way.
  162. *
  163. * param[in] ... List of arguments
  164. *
  165. * @retval Postfix, number of arguments or _LONG when more than 3 arguments.
  166. */
  167. #define Z_LOG_NARGS_POSTFIX(...) \
  168. Z_LOG_NARGS_POSTFIX_IMPL(__VA_ARGS__, LONG, LONG, LONG, LONG, LONG, \
  169. LONG, LONG, LONG, LONG, LONG, LONG, LONG, 3, 2, 1, 0, ~)
  170. #define Z_LOG_INTERNAL_X(N, ...) UTIL_CAT(_LOG_INTERNAL_, N)(__VA_ARGS__)
  171. #define Z_LOG_INTERNAL2(is_user_context, _src_level, ...) do { \
  172. if (is_user_context) { \
  173. log_from_user(_src_level, __VA_ARGS__); \
  174. } else if (IS_ENABLED(CONFIG_LOG_IMMEDIATE)) { \
  175. log_string_sync(_src_level, __VA_ARGS__); \
  176. } else { \
  177. Z_LOG_INTERNAL_X(Z_LOG_NARGS_POSTFIX(__VA_ARGS__), \
  178. _src_level, __VA_ARGS__); \
  179. } \
  180. } while (false)
  181. #define Z_LOG_INTERNAL(is_user_context, _level, _source, _dsource, ...) do { \
  182. uint16_t src_id = \
  183. IS_ENABLED(CONFIG_LOG_RUNTIME_FILTERING) ? \
  184. LOG_DYNAMIC_ID_GET(_dsource) : LOG_CONST_ID_GET(_source); \
  185. struct log_msg_ids src_level = { \
  186. .level = _level, \
  187. .domain_id = CONFIG_LOG_DOMAIN_ID, \
  188. .source_id = src_id \
  189. }; \
  190. Z_LOG_INTERNAL2(is_user_context, src_level, \
  191. Z_LOG_STR(_level, __VA_ARGS__)); \
  192. } while (0)
  193. #define _LOG_INTERNAL_0(_src_level, _str) \
  194. log_0(_str, _src_level)
  195. #define _LOG_INTERNAL_1(_src_level, _str, _arg0) \
  196. log_1(_str, (log_arg_t)(_arg0), _src_level)
  197. #define _LOG_INTERNAL_2(_src_level, _str, _arg0, _arg1) \
  198. log_2(_str, (log_arg_t)(_arg0), (log_arg_t)(_arg1), _src_level)
  199. #define _LOG_INTERNAL_3(_src_level, _str, _arg0, _arg1, _arg2) \
  200. log_3(_str, (log_arg_t)(_arg0), (log_arg_t)(_arg1), (log_arg_t)(_arg2), _src_level)
  201. #define __LOG_ARG_CAST(_x) (log_arg_t)(_x)
  202. #define __LOG_ARGUMENTS(...) FOR_EACH(__LOG_ARG_CAST, (,), __VA_ARGS__)
  203. #define _LOG_INTERNAL_LONG(_src_level, _str, ...) \
  204. do { \
  205. log_arg_t args[] = {__LOG_ARGUMENTS(__VA_ARGS__)};\
  206. log_n(_str, args, ARRAY_SIZE(args), _src_level); \
  207. } while (false)
  208. #define Z_LOG_LEVEL_CHECK(_level, _check_level, _default_level) \
  209. (_level <= Z_LOG_RESOLVED_LEVEL(_check_level, _default_level))
  210. #define Z_LOG_CONST_LEVEL_CHECK(_level) \
  211. (IS_ENABLED(CONFIG_LOG) && \
  212. (Z_LOG_LEVEL_CHECK(_level, CONFIG_LOG_OVERRIDE_LEVEL, LOG_LEVEL_NONE) \
  213. || \
  214. ((IS_ENABLED(CONFIG_LOG_OVERRIDE_LEVEL) == false) && \
  215. (_level <= __log_level) && \
  216. (_level <= CONFIG_LOG_MAX_LEVEL) \
  217. ) \
  218. ))
  219. /*****************************************************************************/
  220. /****************** Defiinitions used by minimal logging *********************/
  221. /*****************************************************************************/
  222. void z_log_minimal_hexdump_print(int level, const void *data, size_t size);
  223. void z_log_minimal_vprintk(const char *fmt, va_list ap);
  224. void z_log_minimal_printk(const char *fmt, ...);
  225. #define Z_LOG_TO_PRINTK(_level, fmt, ...) do { \
  226. z_log_minimal_printk("%c: " fmt "\n", \
  227. z_log_minimal_level_to_char(_level), \
  228. ##__VA_ARGS__); \
  229. } while (false)
  230. #define Z_LOG_TO_VPRINTK(_level, fmt, valist) do { \
  231. z_log_minimal_printk("%c: ", z_log_minimal_level_to_char(_level)); \
  232. z_log_minimal_vprintk(fmt, valist); \
  233. z_log_minimal_printk("\n"); \
  234. } while (false)
  235. static inline char z_log_minimal_level_to_char(int level)
  236. {
  237. switch (level) {
  238. case LOG_LEVEL_ERR:
  239. return 'E';
  240. case LOG_LEVEL_WRN:
  241. return 'W';
  242. case LOG_LEVEL_INF:
  243. return 'I';
  244. case LOG_LEVEL_DBG:
  245. return 'D';
  246. default:
  247. return '?';
  248. }
  249. }
  250. #define Z_LOG_INST(_inst) COND_CODE_1(CONFIG_LOG, (_inst), NULL)
  251. /*****************************************************************************/
  252. /****************** Macros for standard logging ******************************/
  253. /*****************************************************************************/
  254. #define Z_LOG2(_level, _source, _dsource, ...) do { \
  255. if (!Z_LOG_CONST_LEVEL_CHECK(_level)) { \
  256. break; \
  257. } \
  258. if (IS_ENABLED(CONFIG_LOG_MINIMAL)) { \
  259. Z_LOG_TO_PRINTK(_level, __VA_ARGS__); \
  260. break; \
  261. } \
  262. \
  263. bool is_user_context = k_is_user_context(); \
  264. uint32_t filters = IS_ENABLED(CONFIG_LOG_RUNTIME_FILTERING) ? \
  265. (_dsource)->filters : 0;\
  266. if (IS_ENABLED(CONFIG_LOG_RUNTIME_FILTERING) && !is_user_context && \
  267. _level > Z_LOG_RUNTIME_FILTER(filters)) { \
  268. break; \
  269. } \
  270. if (IS_ENABLED(CONFIG_LOG2)) { \
  271. int _mode; \
  272. void *_src = IS_ENABLED(CONFIG_LOG_RUNTIME_FILTERING) ? \
  273. (void *)_dsource : (void *)_source; \
  274. Z_LOG_MSG2_CREATE(UTIL_NOT(IS_ENABLED(CONFIG_USERSPACE)), _mode, \
  275. CONFIG_LOG_DOMAIN_ID, _src, _level, NULL,\
  276. 0, __VA_ARGS__); \
  277. } else { \
  278. Z_LOG_INTERNAL(is_user_context, _level, \
  279. _source, _dsource, __VA_ARGS__);\
  280. } \
  281. if (false) { \
  282. /* Arguments checker present but never evaluated.*/ \
  283. /* Placed here to ensure that __VA_ARGS__ are*/ \
  284. /* evaluated once when log is enabled.*/ \
  285. z_log_printf_arg_checker(__VA_ARGS__); \
  286. } \
  287. } while (false)
  288. #define Z_LOG(_level, ...) \
  289. Z_LOG2(_level, __log_current_const_data, __log_current_dynamic_data, __VA_ARGS__)
  290. #define Z_LOG_INSTANCE(_level, _inst, ...) \
  291. Z_LOG2(_level, \
  292. COND_CODE_1(CONFIG_LOG_RUNTIME_FILTERING, (NULL), (Z_LOG_INST(_inst))), \
  293. (struct log_source_dynamic_data *)COND_CODE_1( \
  294. CONFIG_LOG_RUNTIME_FILTERING, \
  295. (Z_LOG_INST(_inst)), (NULL)), \
  296. __VA_ARGS__)
  297. /*****************************************************************************/
  298. /****************** Macros for hexdump logging *******************************/
  299. /*****************************************************************************/
  300. #define Z_LOG_HEXDUMP2(_level, _source, _dsource, _data, _len, ...) do { \
  301. const char *_str = GET_ARG_N(1, __VA_ARGS__); \
  302. if (!Z_LOG_CONST_LEVEL_CHECK(_level)) { \
  303. break; \
  304. } \
  305. bool is_user_context = k_is_user_context(); \
  306. uint32_t filters = IS_ENABLED(CONFIG_LOG_RUNTIME_FILTERING) ? \
  307. (_dsource)->filters : 0;\
  308. \
  309. if (IS_ENABLED(CONFIG_LOG_MINIMAL)) { \
  310. Z_LOG_TO_PRINTK(_level, "%s", _str); \
  311. z_log_minimal_hexdump_print(_level, \
  312. (const char *)_data, _len);\
  313. break; \
  314. } \
  315. if (IS_ENABLED(CONFIG_LOG_RUNTIME_FILTERING) && !is_user_context && \
  316. _level > Z_LOG_RUNTIME_FILTER(filters)) { \
  317. break; \
  318. } \
  319. if (IS_ENABLED(CONFIG_LOG2)) { \
  320. int mode; \
  321. void *_src = IS_ENABLED(CONFIG_LOG_RUNTIME_FILTERING) ? \
  322. (void *)_dsource : (void *)_source; \
  323. Z_LOG_MSG2_CREATE(UTIL_NOT(IS_ENABLED(CONFIG_USERSPACE)), mode, \
  324. CONFIG_LOG_DOMAIN_ID, _src, _level, \
  325. _data, _len, \
  326. COND_CODE_0(NUM_VA_ARGS_LESS_1(_, ##__VA_ARGS__), \
  327. (), \
  328. (COND_CODE_0(NUM_VA_ARGS_LESS_1(__VA_ARGS__), \
  329. ("%s", __VA_ARGS__), (__VA_ARGS__)))));\
  330. break; \
  331. } \
  332. uint16_t src_id = \
  333. IS_ENABLED(CONFIG_LOG_RUNTIME_FILTERING) ? \
  334. LOG_DYNAMIC_ID_GET(_dsource) : LOG_CONST_ID_GET(_source);\
  335. struct log_msg_ids src_level = { \
  336. .level = _level, \
  337. .domain_id = CONFIG_LOG_DOMAIN_ID, \
  338. .source_id = src_id, \
  339. }; \
  340. if (is_user_context) { \
  341. log_hexdump_from_user(src_level, _str, \
  342. (const char *)_data, _len); \
  343. } else if (IS_ENABLED(CONFIG_LOG_IMMEDIATE)) { \
  344. log_hexdump_sync(src_level, _str, (const char *)_data, _len); \
  345. } else { \
  346. log_hexdump(_str, (const char *)_data, _len, src_level); \
  347. } \
  348. } while (false)
  349. #define Z_LOG_HEXDUMP(_level, _data, _length, ...) \
  350. Z_LOG_HEXDUMP2(_level, \
  351. __log_current_const_data, \
  352. __log_current_dynamic_data, \
  353. _data, _length, __VA_ARGS__)
  354. #define Z_LOG_HEXDUMP_INSTANCE(_level, _inst, _data, _length, _str) \
  355. Z_LOG_HEXDUMP2(_level, \
  356. COND_CODE_1(CONFIG_LOG_RUNTIME_FILTERING, (NULL), (Z_LOG_INST(_inst))), \
  357. (struct log_source_dynamic_data *)COND_CODE_1( \
  358. CONFIG_LOG_RUNTIME_FILTERING, \
  359. (Z_LOG_INST(_inst)), (NULL)), \
  360. _data, _length, _str)
  361. /*****************************************************************************/
  362. /****************** Filtering macros *****************************************/
  363. /*****************************************************************************/
  364. /** @brief Number of bits used to encode log level. */
  365. #define LOG_LEVEL_BITS 3U
  366. /** @brief Filter slot size. */
  367. #define LOG_FILTER_SLOT_SIZE LOG_LEVEL_BITS
  368. /** @brief Number of slots in one word. */
  369. #define LOG_FILTERS_NUM_OF_SLOTS (32 / LOG_FILTER_SLOT_SIZE)
  370. /** @brief Slot mask. */
  371. #define LOG_FILTER_SLOT_MASK (BIT(LOG_FILTER_SLOT_SIZE) - 1U)
  372. /** @brief Bit offset of a slot.
  373. *
  374. * @param _id Slot ID.
  375. */
  376. #define LOG_FILTER_SLOT_SHIFT(_id) (LOG_FILTER_SLOT_SIZE * (_id))
  377. #define LOG_FILTER_SLOT_GET(_filters, _id) \
  378. ((*(_filters) >> LOG_FILTER_SLOT_SHIFT(_id)) & LOG_FILTER_SLOT_MASK)
  379. #define LOG_FILTER_SLOT_SET(_filters, _id, _filter) \
  380. do { \
  381. *(_filters) &= ~(LOG_FILTER_SLOT_MASK << \
  382. LOG_FILTER_SLOT_SHIFT(_id)); \
  383. *(_filters) |= ((_filter) & LOG_FILTER_SLOT_MASK) << \
  384. LOG_FILTER_SLOT_SHIFT(_id); \
  385. } while (false)
  386. #define LOG_FILTER_AGGR_SLOT_IDX 0
  387. #define LOG_FILTER_AGGR_SLOT_GET(_filters) \
  388. LOG_FILTER_SLOT_GET(_filters, LOG_FILTER_AGGR_SLOT_IDX)
  389. #define LOG_FILTER_FIRST_BACKEND_SLOT_IDX 1
  390. /* Return aggregated (highest) level for all enabled backends, e.g. if there
  391. * are 3 active backends, one backend is set to get INF logs from a module and
  392. * two other backends are set for ERR, returned level is INF.
  393. */
  394. #define Z_LOG_RUNTIME_FILTER(_filter) \
  395. LOG_FILTER_SLOT_GET(&_filter, LOG_FILTER_AGGR_SLOT_IDX)
  396. /** @brief Log level value used to indicate log entry that should not be
  397. * formatted (raw string).
  398. */
  399. #define LOG_LEVEL_INTERNAL_RAW_STRING LOG_LEVEL_NONE
  400. extern struct log_source_const_data __log_const_start[];
  401. extern struct log_source_const_data __log_const_end[];
  402. /** @brief Enum with possible actions for strdup operation. */
  403. enum log_strdup_action {
  404. LOG_STRDUP_SKIP, /**< None RAM string duplication. */
  405. LOG_STRDUP_EXEC, /**< Always duplicate RAM strings. */
  406. LOG_STRDUP_CHECK_EXEC/**< Duplicate RAM strings, if not dupl. before.*/
  407. };
  408. #define Z_LOG_PRINTK(...) do { \
  409. if (IS_ENABLED(CONFIG_LOG_MINIMAL) || !IS_ENABLED(CONFIG_LOG2)) { \
  410. z_log_minimal_printk(__VA_ARGS__); \
  411. break; \
  412. } \
  413. int _mode; \
  414. if (0) {\
  415. z_log_printf_arg_checker(__VA_ARGS__); \
  416. } \
  417. Z_LOG_MSG2_CREATE(!IS_ENABLED(CONFIG_USERSPACE), _mode, \
  418. CONFIG_LOG_DOMAIN_ID, NULL, \
  419. LOG_LEVEL_INTERNAL_RAW_STRING, NULL, 0, __VA_ARGS__);\
  420. } while (0)
  421. /** @brief Get name of the log source.
  422. *
  423. * @param source_id Source ID.
  424. * @return Name.
  425. */
  426. static inline const char *log_name_get(uint32_t source_id)
  427. {
  428. return __log_const_start[source_id].name;
  429. }
  430. /** @brief Get compiled level of the log source.
  431. *
  432. * @param source_id Source ID.
  433. * @return Level.
  434. */
  435. static inline uint8_t log_compiled_level_get(uint32_t source_id)
  436. {
  437. return __log_const_start[source_id].level;
  438. }
  439. /** @brief Get index of the log source based on the address of the constant data
  440. * associated with the source.
  441. *
  442. * @param data Address of the constant data.
  443. *
  444. * @return Source ID.
  445. */
  446. static inline uint32_t log_const_source_id(
  447. const struct log_source_const_data *data)
  448. {
  449. return ((uint8_t *)data - (uint8_t *)__log_const_start)/
  450. sizeof(struct log_source_const_data);
  451. }
  452. /** @brief Get number of registered sources. */
  453. static inline uint32_t log_sources_count(void)
  454. {
  455. return log_const_source_id(__log_const_end);
  456. }
  457. extern struct log_source_dynamic_data __log_dynamic_start[];
  458. extern struct log_source_dynamic_data __log_dynamic_end[];
  459. /** @brief Creates name of variable and section for runtime log data.
  460. *
  461. * @param _name Name.
  462. */
  463. #define LOG_ITEM_DYNAMIC_DATA(_name) UTIL_CAT(log_dynamic_, _name)
  464. #define LOG_INSTANCE_DYNAMIC_DATA(_module_name, _inst) \
  465. LOG_ITEM_DYNAMIC_DATA(LOG_INSTANCE_FULL_NAME(_module_name, _inst))
  466. /** @brief Get pointer to the filter set of the log source.
  467. *
  468. * @param source_id Source ID.
  469. *
  470. * @return Pointer to the filter set.
  471. */
  472. static inline uint32_t *log_dynamic_filters_get(uint32_t source_id)
  473. {
  474. return &__log_dynamic_start[source_id].filters;
  475. }
  476. /** @brief Get index of the log source based on the address of the dynamic data
  477. * associated with the source.
  478. *
  479. * @param data Address of the dynamic data.
  480. *
  481. * @return Source ID.
  482. */
  483. static inline uint32_t log_dynamic_source_id(struct log_source_dynamic_data *data)
  484. {
  485. return ((uint8_t *)data - (uint8_t *)__log_dynamic_start)/
  486. sizeof(struct log_source_dynamic_data);
  487. }
  488. /* Initialize runtime filters */
  489. void z_log_runtime_filters_init(void);
  490. /* Notify log_core that a backend was enabled. */
  491. void z_log_notify_backend_enabled(void);
  492. /** @brief Dummy function to trigger log messages arguments type checking. */
  493. static inline __printf_like(1, 2)
  494. void z_log_printf_arg_checker(const char *fmt, ...)
  495. {
  496. ARG_UNUSED(fmt);
  497. }
  498. /** @brief Standard log with no arguments.
  499. *
  500. * @param str String.
  501. * @param src_level Log identification.
  502. */
  503. void log_0(const char *str, struct log_msg_ids src_level);
  504. /** @brief Standard log with one argument.
  505. *
  506. * @param str String.
  507. * @param arg1 First argument.
  508. * @param src_level Log identification.
  509. */
  510. void log_1(const char *str,
  511. log_arg_t arg1,
  512. struct log_msg_ids src_level);
  513. /** @brief Standard log with two arguments.
  514. *
  515. * @param str String.
  516. * @param arg1 First argument.
  517. * @param arg2 Second argument.
  518. * @param src_level Log identification.
  519. */
  520. void log_2(const char *str,
  521. log_arg_t arg1,
  522. log_arg_t arg2,
  523. struct log_msg_ids src_level);
  524. /** @brief Standard log with three arguments.
  525. *
  526. * @param str String.
  527. * @param arg1 First argument.
  528. * @param arg2 Second argument.
  529. * @param arg3 Third argument.
  530. * @param src_level Log identification.
  531. */
  532. void log_3(const char *str,
  533. log_arg_t arg1,
  534. log_arg_t arg2,
  535. log_arg_t arg3,
  536. struct log_msg_ids src_level);
  537. /** @brief Standard log with arguments list.
  538. *
  539. * @param str String.
  540. * @param args Array with arguments.
  541. * @param narg Number of arguments in the array.
  542. * @param src_level Log identification.
  543. */
  544. void log_n(const char *str,
  545. log_arg_t *args,
  546. uint32_t narg,
  547. struct log_msg_ids src_level);
  548. /** @brief Hexdump log.
  549. *
  550. * @param str String.
  551. * @param data Data.
  552. * @param length Data length.
  553. * @param src_level Log identification.
  554. */
  555. void log_hexdump(const char *str, const void *data, uint32_t length,
  556. struct log_msg_ids src_level);
  557. /** @brief Process log message synchronously.
  558. *
  559. * @param src_level Log message details.
  560. * @param fmt String to format.
  561. * @param ... Variable list of arguments.
  562. */
  563. void log_string_sync(struct log_msg_ids src_level, const char *fmt, ...);
  564. /** @brief Process log hexdump message synchronously.
  565. *
  566. * @param src_level Log message details.
  567. * @param metadata Raw string associated with the data.
  568. * @param data Data.
  569. * @param len Data length.
  570. */
  571. void log_hexdump_sync(struct log_msg_ids src_level, const char *metadata,
  572. const void *data, uint32_t len);
  573. /**
  574. * @brief Writes a generic log message to the log.
  575. *
  576. * @note This function is intended to be used when porting other log systems.
  577. *
  578. * @param src_level Log identification.
  579. * @param fmt String to format.
  580. * @param ap Poiner to arguments list.
  581. * @param strdup_action Manages strdup activity.
  582. */
  583. void log_generic(struct log_msg_ids src_level, const char *fmt, va_list ap,
  584. enum log_strdup_action strdup_action);
  585. /**
  586. * @brief Writes a generic log message to the logging v2.
  587. *
  588. * @note This function is intended to be used when porting other log systems.
  589. *
  590. * @param level Log level..
  591. * @param fmt String to format.
  592. * @param ap Poiner to arguments list.
  593. */
  594. static inline void log2_generic(uint8_t level, const char *fmt, va_list ap)
  595. {
  596. z_log_msg2_runtime_vcreate(CONFIG_LOG_DOMAIN_ID, NULL, level,
  597. NULL, 0, fmt, ap);
  598. }
  599. /**
  600. * @brief Returns number of arguments visible from format string.
  601. *
  602. * @note This function is intended to be used when porting other log systems.
  603. *
  604. * @param fmt Format string.
  605. *
  606. * @return Number of arguments.
  607. */
  608. uint32_t log_count_args(const char *fmt);
  609. /**
  610. * @brief Writes a generic log message to the log from user mode.
  611. *
  612. * @note This function is intended to be used internally
  613. * by the logging subsystem.
  614. */
  615. void log_generic_from_user(struct log_msg_ids src_level,
  616. const char *fmt, va_list ap);
  617. /** @brief Check if address belongs to the memory pool used for transient.
  618. *
  619. * @param buf Buffer.
  620. *
  621. * @return True if address within the pool, false otherwise.
  622. */
  623. bool log_is_strdup(const void *buf);
  624. /** @brief Free allocated buffer.
  625. *
  626. * @param buf Buffer.
  627. */
  628. void log_free(void *buf);
  629. /**
  630. * @brief Get current number of allocated buffers for string duplicates.
  631. */
  632. uint32_t log_get_strdup_pool_current_utilization(void);
  633. /**
  634. * @brief Get maximal number of simultaneously allocated buffers for string
  635. * duplicates.
  636. *
  637. * Value can be used to determine pool size.
  638. */
  639. uint32_t log_get_strdup_pool_utilization(void);
  640. /**
  641. * @brief Get length of the longest string duplicated.
  642. *
  643. * Value can be used to determine buffer size in the string duplicates pool.
  644. */
  645. uint32_t log_get_strdup_longest_string(void);
  646. /** @brief Indicate to the log core that one log message has been dropped.
  647. */
  648. void z_log_dropped(void);
  649. /** @brief Read and clear current drop indications counter.
  650. *
  651. * @return Dropped count.
  652. */
  653. uint32_t z_log_dropped_read_and_clear(void);
  654. /** @brief Check if there are any pending drop notifications.
  655. *
  656. * @retval true Pending unreported drop indications.
  657. * @retval false No pending unreported drop indications.
  658. */
  659. bool z_log_dropped_pending(void);
  660. /** @brief Log a message from user mode context.
  661. *
  662. * @note This function is intended to be used internally
  663. * by the logging subsystem.
  664. *
  665. * @param src_level Log identification.
  666. * @param fmt String to format.
  667. * @param ... Variable list of arguments.
  668. */
  669. void __printf_like(2, 3) log_from_user(struct log_msg_ids src_level,
  670. const char *fmt, ...);
  671. /* Internal function used by log_from_user(). */
  672. __syscall void z_log_string_from_user(uint32_t src_level_val, const char *str);
  673. /**
  674. * @brief Create mask with occurences of a string format specifiers (%s).
  675. *
  676. * Result is stored as the mask (argument n is n'th bit). Bit is set if string
  677. * format specifier was found.
  678. *
  679. * @param str String.
  680. * @param nargs Number of arguments in the string.
  681. *
  682. * @return Mask with %s format specifiers found.
  683. */
  684. uint32_t z_log_get_s_mask(const char *str, uint32_t nargs);
  685. /** @brief Log binary data (displayed as hexdump) from user mode context.
  686. *
  687. * @note This function is intended to be used internally
  688. * by the logging subsystem.
  689. *
  690. * @param src_level Log identification.
  691. * @param metadata Raw string associated with the data.
  692. * @param data Data.
  693. * @param len Data length.
  694. */
  695. void log_hexdump_from_user(struct log_msg_ids src_level, const char *metadata,
  696. const void *data, uint32_t len);
  697. /* Internal function used by log_hexdump_from_user(). */
  698. __syscall void z_log_hexdump_from_user(uint32_t src_level_val,
  699. const char *metadata,
  700. const uint8_t *data, uint32_t len);
  701. /******************************************************************************/
  702. /********** Mocros _VA operate on var-args parameters. ***************/
  703. /********* Intended to be used when porting other log systems. ***************/
  704. /********* Shall be used in the log entry interface function. ***************/
  705. /********* Speed optimized for up to three arguments number. ***************/
  706. /******************************************************************************/
  707. #define Z_LOG_VA(_level, _str, _valist, _argnum, _strdup_action)\
  708. __LOG_VA(_level, \
  709. __log_current_const_data, \
  710. __log_current_dynamic_data, \
  711. _str, _valist, _argnum, _strdup_action)
  712. #define __LOG_VA(_level, _source, _dsource, _str, _valist, _argnum, _strdup_action) do { \
  713. if (!Z_LOG_CONST_LEVEL_CHECK(_level)) { \
  714. break; \
  715. } \
  716. if (IS_ENABLED(CONFIG_LOG_MINIMAL)) { \
  717. Z_LOG_TO_VPRINTK(_level, _str, _valist); \
  718. break; \
  719. } \
  720. \
  721. bool is_user_context = k_is_user_context(); \
  722. uint32_t filters = IS_ENABLED(CONFIG_LOG_RUNTIME_FILTERING) ? \
  723. _dsource->filters : 0;\
  724. if (IS_ENABLED(CONFIG_LOG_RUNTIME_FILTERING) && !is_user_context && \
  725. _level > Z_LOG_RUNTIME_FILTER(filters)) { \
  726. break; \
  727. } \
  728. if (IS_ENABLED(CONFIG_LOG2)) { \
  729. z_log_msg2_runtime_vcreate(CONFIG_LOG_DOMAIN_ID, _source, \
  730. _level, NULL, 0, _str, _valist); \
  731. break; \
  732. } \
  733. uint16_t _id = \
  734. IS_ENABLED(CONFIG_LOG_RUNTIME_FILTERING) ? \
  735. LOG_DYNAMIC_ID_GET(_dsource) : LOG_CONST_ID_GET(_source);\
  736. struct log_msg_ids src_level = { \
  737. .level = _level, \
  738. .domain_id = CONFIG_LOG_DOMAIN_ID, \
  739. .source_id = _id \
  740. }; \
  741. __LOG_INTERNAL_VA(is_user_context, \
  742. src_level, \
  743. _str, _valist, _argnum, \
  744. _strdup_action); \
  745. } while (false)
  746. /**
  747. * @brief Inline function to perform strdup, used in __LOG_INTERNAL_VA macro
  748. *
  749. * @note This function is intended to be used when porting other log systems.
  750. *
  751. * @param msk Bitmask marking all %s arguments.
  752. * @param idx Index of actually processed argument.
  753. * @param param Value of actually processed argument.
  754. * @param action Action for strdup operation.
  755. *
  756. * @return Duplicated string or not changed param.
  757. */
  758. static inline log_arg_t z_log_do_strdup(uint32_t msk, uint32_t idx,
  759. log_arg_t param,
  760. enum log_strdup_action action)
  761. {
  762. #ifndef CONFIG_LOG_MINIMAL
  763. char *z_log_strdup(const char *str);
  764. if (msk & (1 << idx)) {
  765. const char *str = (const char *)param;
  766. /* is_rodata(str) is not checked,
  767. * because log_strdup does it.
  768. * Hence, we will do only optional check
  769. * if already not duplicated.
  770. */
  771. if (action == LOG_STRDUP_EXEC || !log_is_strdup(str)) {
  772. param = (log_arg_t)z_log_strdup(str);
  773. }
  774. }
  775. #else
  776. ARG_UNUSED(msk);
  777. ARG_UNUSED(idx);
  778. ARG_UNUSED(action);
  779. #endif
  780. return param;
  781. }
  782. #define __LOG_INTERNAL_VA(is_user_context, _src_level, _str, _valist, \
  783. _argnum, _strdup_action) \
  784. do { \
  785. if (is_user_context) { \
  786. log_generic_from_user(_src_level, _str, _valist); \
  787. } else if (IS_ENABLED(CONFIG_LOG_IMMEDIATE)) { \
  788. log_generic(_src_level, _str, _valist, _strdup_action); \
  789. } else if (_argnum == 0) { \
  790. _LOG_INTERNAL_0(_src_level, _str); \
  791. } else { \
  792. uint32_t mask = (_strdup_action != LOG_STRDUP_SKIP) ? \
  793. z_log_get_s_mask(_str, _argnum) \
  794. : 0; \
  795. \
  796. if (_argnum == 1) { \
  797. _LOG_INTERNAL_1(_src_level, _str, \
  798. z_log_do_strdup(mask, 0, \
  799. va_arg(_valist, log_arg_t), _strdup_action));\
  800. } else if (_argnum == 2) { \
  801. _LOG_INTERNAL_2(_src_level, _str, \
  802. z_log_do_strdup(mask, 0, \
  803. va_arg(_valist, log_arg_t), _strdup_action), \
  804. z_log_do_strdup(mask, 1, \
  805. va_arg(_valist, log_arg_t), _strdup_action));\
  806. } else if (_argnum == 3) { \
  807. _LOG_INTERNAL_3(_src_level, _str, \
  808. z_log_do_strdup(mask, 0, \
  809. va_arg(_valist, log_arg_t), _strdup_action), \
  810. z_log_do_strdup(mask, 1, \
  811. va_arg(_valist, log_arg_t), _strdup_action), \
  812. z_log_do_strdup(mask, 2, \
  813. va_arg(_valist, log_arg_t), _strdup_action));\
  814. } else { \
  815. log_generic(_src_level, _str, _valist, _strdup_action);\
  816. } \
  817. } \
  818. } while (false)
  819. #include <syscalls/log_core.h>
  820. #ifdef __cplusplus
  821. }
  822. #endif
  823. #endif /* ZEPHYR_INCLUDE_LOGGING_LOG_CORE_H_ */