log_output.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762
  1. /*
  2. * Copyright (c) 2018 Nordic Semiconductor ASA
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <logging/log_output.h>
  7. #include <logging/log_ctrl.h>
  8. #include <logging/log.h>
  9. #include <sys/__assert.h>
  10. #include <sys/cbprintf.h>
  11. #include <ctype.h>
  12. #include <time.h>
  13. #include <stdio.h>
  14. #include <stdbool.h>
  15. #define LOG_COLOR_CODE_DEFAULT "\x1B[0m"
  16. #define LOG_COLOR_CODE_RED "\x1B[1;31m"
  17. #define LOG_COLOR_CODE_YELLOW "\x1B[1;33m"
  18. #define HEXDUMP_BYTES_IN_LINE 16
  19. #define DROPPED_COLOR_PREFIX \
  20. Z_LOG_EVAL(CONFIG_LOG_BACKEND_SHOW_COLOR, (LOG_COLOR_CODE_RED), ())
  21. #define DROPPED_COLOR_POSTFIX \
  22. Z_LOG_EVAL(CONFIG_LOG_BACKEND_SHOW_COLOR, (LOG_COLOR_CODE_DEFAULT), ())
  23. static const char *const severity[] = {
  24. NULL,
  25. "err",
  26. "wrn",
  27. "inf",
  28. "dbg"
  29. };
  30. static const char *const colors[] = {
  31. NULL,
  32. LOG_COLOR_CODE_RED, /* err */
  33. LOG_COLOR_CODE_YELLOW, /* warn */
  34. NULL, /* info */
  35. NULL /* dbg */
  36. };
  37. static uint32_t freq;
  38. static uint32_t timestamp_div;
  39. extern void log_output_msg_syst_process(const struct log_output *output,
  40. struct log_msg *msg, uint32_t flag);
  41. extern void log_output_string_syst_process(const struct log_output *output,
  42. struct log_msg_ids src_level,
  43. const char *fmt, va_list ap, uint32_t flag);
  44. extern void log_output_hexdump_syst_process(const struct log_output *output,
  45. struct log_msg_ids src_level,
  46. const uint8_t *data, uint32_t length, uint32_t flag);
  47. /* The RFC 5424 allows very flexible mapping and suggest the value 0 being the
  48. * highest severity and 7 to be the lowest (debugging level) severity.
  49. *
  50. * 0 Emergency System is unusable
  51. * 1 Alert Action must be taken immediately
  52. * 2 Critical Critical conditions
  53. * 3 Error Error conditions
  54. * 4 Warning Warning conditions
  55. * 5 Notice Normal but significant condition
  56. * 6 Informational Informational messages
  57. * 7 Debug Debug-level messages
  58. */
  59. static int level_to_rfc5424_severity(uint32_t level)
  60. {
  61. uint8_t ret;
  62. switch (level) {
  63. case LOG_LEVEL_NONE:
  64. ret = 7U;
  65. break;
  66. case LOG_LEVEL_ERR:
  67. ret = 3U;
  68. break;
  69. case LOG_LEVEL_WRN:
  70. ret = 4U;
  71. break;
  72. case LOG_LEVEL_INF:
  73. ret = 6U;
  74. break;
  75. case LOG_LEVEL_DBG:
  76. ret = 7U;
  77. break;
  78. default:
  79. ret = 7U;
  80. break;
  81. }
  82. return ret;
  83. }
  84. static int out_func(int c, void *ctx)
  85. {
  86. const struct log_output *out_ctx = (const struct log_output *)ctx;
  87. int idx;
  88. if (IS_ENABLED(CONFIG_LOG_IMMEDIATE)) {
  89. /* Backend must be thread safe in synchronous operation. */
  90. out_ctx->func((uint8_t *)&c, 1, out_ctx->control_block->ctx);
  91. return 0;
  92. }
  93. if (out_ctx->control_block->offset == out_ctx->size) {
  94. log_output_flush(out_ctx);
  95. }
  96. idx = atomic_inc(&out_ctx->control_block->offset);
  97. out_ctx->buf[idx] = (uint8_t)c;
  98. __ASSERT_NO_MSG(out_ctx->control_block->offset <= out_ctx->size);
  99. return 0;
  100. }
  101. static int cr_out_func(int c, void *ctx)
  102. {
  103. out_func(c, ctx);
  104. if (c == '\n') {
  105. out_func((int)'\r', ctx);
  106. }
  107. return 0;
  108. }
  109. static int print_formatted(const struct log_output *output,
  110. const char *fmt, ...)
  111. {
  112. va_list args;
  113. int length = 0;
  114. va_start(args, fmt);
  115. length = cbvprintf(out_func, (void *)output, fmt, args);
  116. va_end(args);
  117. return length;
  118. }
  119. static void buffer_write(log_output_func_t outf, uint8_t *buf, size_t len,
  120. void *ctx)
  121. {
  122. int processed;
  123. do {
  124. processed = outf(buf, len, ctx);
  125. len -= processed;
  126. buf += processed;
  127. } while (len != 0);
  128. }
  129. void log_output_flush(const struct log_output *output)
  130. {
  131. buffer_write(output->func, output->buf,
  132. output->control_block->offset,
  133. output->control_block->ctx);
  134. output->control_block->offset = 0;
  135. }
  136. static int timestamp_print(const struct log_output *output,
  137. uint32_t flags, uint32_t timestamp)
  138. {
  139. int length;
  140. bool format =
  141. (flags & LOG_OUTPUT_FLAG_FORMAT_TIMESTAMP) |
  142. (flags & LOG_OUTPUT_FLAG_FORMAT_SYSLOG);
  143. if (!format) {
  144. length = print_formatted(output, "[%08lu] ", timestamp);
  145. } else if (freq != 0U) {
  146. uint32_t total_seconds;
  147. uint32_t remainder;
  148. uint32_t seconds;
  149. uint32_t hours;
  150. uint32_t mins;
  151. uint32_t ms;
  152. uint32_t us;
  153. timestamp /= timestamp_div;
  154. total_seconds = timestamp / freq;
  155. seconds = total_seconds;
  156. hours = seconds / 3600U;
  157. seconds -= hours * 3600U;
  158. mins = seconds / 60U;
  159. seconds -= mins * 60U;
  160. remainder = timestamp % freq;
  161. ms = (remainder * 1000U) / freq;
  162. us = (1000 * (remainder * 1000U - (ms * freq))) / freq;
  163. if (IS_ENABLED(CONFIG_LOG_BACKEND_NET) &&
  164. flags & LOG_OUTPUT_FLAG_FORMAT_SYSLOG) {
  165. #if defined(CONFIG_NEWLIB_LIBC)
  166. char time_str[sizeof("1970-01-01T00:00:00")];
  167. struct tm *tm;
  168. time_t time;
  169. time = total_seconds;
  170. tm = gmtime(&time);
  171. strftime(time_str, sizeof(time_str), "%FT%T", tm);
  172. length = print_formatted(output, "%s.%06uZ ",
  173. time_str, ms * 1000U + us);
  174. #else
  175. length = print_formatted(output,
  176. "1970-01-01T%02u:%02u:%02u.%06uZ ",
  177. hours, mins, seconds, ms * 1000U + us);
  178. #endif
  179. } else {
  180. length = print_formatted(output,
  181. "[%02u:%02u:%02u.%03u,%03u] ",
  182. hours, mins, seconds, ms, us);
  183. }
  184. } else {
  185. length = 0;
  186. }
  187. return length;
  188. }
  189. static void color_print(const struct log_output *output,
  190. bool color, bool start, uint32_t level)
  191. {
  192. if (color) {
  193. const char *log_color = start && (colors[level] != NULL) ?
  194. colors[level] : LOG_COLOR_CODE_DEFAULT;
  195. print_formatted(output, "%s", log_color);
  196. }
  197. }
  198. static void color_prefix(const struct log_output *output,
  199. bool color, uint32_t level)
  200. {
  201. color_print(output, color, true, level);
  202. }
  203. static void color_postfix(const struct log_output *output,
  204. bool color, uint32_t level)
  205. {
  206. color_print(output, color, false, level);
  207. }
  208. static int ids_print(const struct log_output *output, bool level_on,
  209. bool func_on, uint32_t domain_id, int16_t source_id,
  210. uint32_t level)
  211. {
  212. int total = 0;
  213. if (level_on) {
  214. total += print_formatted(output, "<%s> ", severity[level]);
  215. }
  216. if (source_id >= 0) {
  217. total += print_formatted(output,
  218. (func_on &&
  219. ((1 << level) & LOG_FUNCTION_PREFIX_MASK)) ?
  220. "%s." : "%s: ",
  221. log_source_name_get(domain_id, source_id));
  222. }
  223. return total;
  224. }
  225. static void newline_print(const struct log_output *ctx, uint32_t flags)
  226. {
  227. if (IS_ENABLED(CONFIG_LOG_BACKEND_NET) &&
  228. flags & LOG_OUTPUT_FLAG_FORMAT_SYSLOG) {
  229. return;
  230. }
  231. if ((flags & LOG_OUTPUT_FLAG_CRLF_NONE) != 0U) {
  232. return;
  233. }
  234. if ((flags & LOG_OUTPUT_FLAG_CRLF_LFONLY) != 0U) {
  235. print_formatted(ctx, "\n");
  236. } else {
  237. print_formatted(ctx, "\r\n");
  238. }
  239. }
  240. static void std_print(struct log_msg *msg,
  241. const struct log_output *output)
  242. {
  243. const char *str = log_msg_str_get(msg);
  244. uint32_t nargs = log_msg_nargs_get(msg);
  245. log_arg_t *args = alloca(sizeof(log_arg_t)*nargs);
  246. int i;
  247. for (i = 0; i < nargs; i++) {
  248. args[i] = log_msg_arg_get(msg, i);
  249. }
  250. switch (log_msg_nargs_get(msg)) {
  251. case 0:
  252. print_formatted(output, str);
  253. break;
  254. case 1:
  255. print_formatted(output, str, args[0]);
  256. break;
  257. case 2:
  258. print_formatted(output, str, args[0], args[1]);
  259. break;
  260. case 3:
  261. print_formatted(output, str, args[0], args[1], args[2]);
  262. break;
  263. case 4:
  264. print_formatted(output, str, args[0], args[1], args[2],
  265. args[3]);
  266. break;
  267. case 5:
  268. print_formatted(output, str, args[0], args[1], args[2],
  269. args[3], args[4]);
  270. break;
  271. case 6:
  272. print_formatted(output, str, args[0], args[1], args[2],
  273. args[3], args[4], args[5]);
  274. break;
  275. case 7:
  276. print_formatted(output, str, args[0], args[1], args[2],
  277. args[3], args[4], args[5], args[6]);
  278. break;
  279. case 8:
  280. print_formatted(output, str, args[0], args[1], args[2],
  281. args[3], args[4], args[5], args[6], args[7]);
  282. break;
  283. case 9:
  284. print_formatted(output, str, args[0], args[1], args[2],
  285. args[3], args[4], args[5], args[6], args[7],
  286. args[8]);
  287. break;
  288. case 10:
  289. print_formatted(output, str, args[0], args[1], args[2],
  290. args[3], args[4], args[5], args[6], args[7],
  291. args[8], args[9]);
  292. break;
  293. case 11:
  294. print_formatted(output, str, args[0], args[1], args[2],
  295. args[3], args[4], args[5], args[6], args[7],
  296. args[8], args[9], args[10]);
  297. break;
  298. case 12:
  299. print_formatted(output, str, args[0], args[1], args[2],
  300. args[3], args[4], args[5], args[6], args[7],
  301. args[8], args[9], args[10], args[11]);
  302. break;
  303. case 13:
  304. print_formatted(output, str, args[0], args[1], args[2],
  305. args[3], args[4], args[5], args[6], args[7],
  306. args[8], args[9], args[10], args[11], args[12]);
  307. break;
  308. case 14:
  309. print_formatted(output, str, args[0], args[1], args[2],
  310. args[3], args[4], args[5], args[6], args[7],
  311. args[8], args[9], args[10], args[11], args[12],
  312. args[13]);
  313. break;
  314. case 15:
  315. print_formatted(output, str, args[0], args[1], args[2],
  316. args[3], args[4], args[5], args[6], args[7],
  317. args[8], args[9], args[10], args[11], args[12],
  318. args[13], args[14]);
  319. break;
  320. default:
  321. /* Unsupported number of arguments. */
  322. __ASSERT_NO_MSG(true);
  323. break;
  324. }
  325. }
  326. static void hexdump_line_print(const struct log_output *output,
  327. const uint8_t *data, uint32_t length,
  328. int prefix_offset, uint32_t flags)
  329. {
  330. newline_print(output, flags);
  331. for (int i = 0; i < prefix_offset; i++) {
  332. print_formatted(output, " ");
  333. }
  334. for (int i = 0; i < HEXDUMP_BYTES_IN_LINE; i++) {
  335. if (i > 0 && !(i % 8)) {
  336. print_formatted(output, " ");
  337. }
  338. if (i < length) {
  339. print_formatted(output, "%02x ", data[i]);
  340. } else {
  341. print_formatted(output, " ");
  342. }
  343. }
  344. print_formatted(output, "|");
  345. for (int i = 0; i < HEXDUMP_BYTES_IN_LINE; i++) {
  346. if (i > 0 && !(i % 8)) {
  347. print_formatted(output, " ");
  348. }
  349. if (i < length) {
  350. char c = (char)data[i];
  351. print_formatted(output, "%c",
  352. isprint((int)c) ? c : '.');
  353. } else {
  354. print_formatted(output, " ");
  355. }
  356. }
  357. }
  358. static void hexdump_print(struct log_msg *msg,
  359. const struct log_output *output,
  360. int prefix_offset, uint32_t flags)
  361. {
  362. uint32_t offset = 0U;
  363. uint8_t buf[HEXDUMP_BYTES_IN_LINE];
  364. size_t length;
  365. print_formatted(output, "%s", log_msg_str_get(msg));
  366. do {
  367. length = sizeof(buf);
  368. log_msg_hexdump_data_get(msg, buf, &length, offset);
  369. if (length) {
  370. hexdump_line_print(output, buf, length,
  371. prefix_offset, flags);
  372. offset += length;
  373. } else {
  374. break;
  375. }
  376. } while (true);
  377. }
  378. static void log_msg2_hexdump(const struct log_output *output,
  379. uint8_t *data, uint32_t len,
  380. int prefix_offset, uint32_t flags)
  381. {
  382. size_t length;
  383. do {
  384. length = MIN(len, HEXDUMP_BYTES_IN_LINE);
  385. hexdump_line_print(output, data, length,
  386. prefix_offset, flags);
  387. data += length;
  388. len -= length;
  389. } while (len);
  390. }
  391. static void raw_string_print(struct log_msg *msg,
  392. const struct log_output *output)
  393. {
  394. __ASSERT_NO_MSG(output->size);
  395. size_t offset = 0;
  396. size_t length;
  397. bool eol = false;
  398. do {
  399. length = output->size;
  400. /* Sting is stored in a hexdump message. */
  401. log_msg_hexdump_data_get(msg, output->buf, &length, offset);
  402. output->control_block->offset = length;
  403. if (length != 0) {
  404. eol = (output->buf[length - 1] == '\n');
  405. }
  406. log_output_flush(output);
  407. offset += length;
  408. } while (length > 0);
  409. if (eol) {
  410. print_formatted(output, "\r");
  411. }
  412. }
  413. static uint32_t prefix_print(const struct log_output *output,
  414. uint32_t flags, bool func_on, uint32_t timestamp, uint8_t level,
  415. uint8_t domain_id, int16_t source_id)
  416. {
  417. uint32_t length = 0U;
  418. bool stamp = flags & LOG_OUTPUT_FLAG_TIMESTAMP;
  419. bool colors_on = flags & LOG_OUTPUT_FLAG_COLORS;
  420. bool level_on = flags & LOG_OUTPUT_FLAG_LEVEL;
  421. if (IS_ENABLED(CONFIG_LOG_BACKEND_NET) &&
  422. flags & LOG_OUTPUT_FLAG_FORMAT_SYSLOG) {
  423. /* TODO: As there is no way to figure out the
  424. * facility at this point, use a pre-defined value.
  425. * Change this to use the real facility of the
  426. * logging call when that info is available.
  427. */
  428. static const int facility = 16; /* local0 */
  429. length += print_formatted(
  430. output,
  431. "<%d>1 ",
  432. facility * 8 +
  433. level_to_rfc5424_severity(level));
  434. }
  435. if (stamp) {
  436. length += timestamp_print(output, flags, timestamp);
  437. }
  438. if (IS_ENABLED(CONFIG_LOG_BACKEND_NET) &&
  439. flags & LOG_OUTPUT_FLAG_FORMAT_SYSLOG) {
  440. length += print_formatted(
  441. output, "%s - - - - ",
  442. output->control_block->hostname ?
  443. output->control_block->hostname :
  444. "zephyr");
  445. } else {
  446. color_prefix(output, colors_on, level);
  447. }
  448. length += ids_print(output, level_on, func_on,
  449. domain_id, source_id, level);
  450. return length;
  451. }
  452. static void postfix_print(const struct log_output *output,
  453. uint32_t flags, uint8_t level)
  454. {
  455. color_postfix(output, (flags & LOG_OUTPUT_FLAG_COLORS),
  456. level);
  457. newline_print(output, flags);
  458. }
  459. void log_output_msg_process(const struct log_output *output,
  460. struct log_msg *msg,
  461. uint32_t flags)
  462. {
  463. bool std_msg = log_msg_is_std(msg);
  464. uint32_t timestamp = log_msg_timestamp_get(msg);
  465. uint8_t level = (uint8_t)log_msg_level_get(msg);
  466. uint8_t domain_id = (uint8_t)log_msg_domain_id_get(msg);
  467. int16_t source_id = (int16_t)log_msg_source_id_get(msg);
  468. bool raw_string = (level == LOG_LEVEL_INTERNAL_RAW_STRING);
  469. int prefix_offset;
  470. if (IS_ENABLED(CONFIG_LOG_MIPI_SYST_ENABLE) &&
  471. flags & LOG_OUTPUT_FLAG_FORMAT_SYST) {
  472. log_output_msg_syst_process(output, msg, flags);
  473. return;
  474. }
  475. prefix_offset = raw_string ?
  476. 0 : prefix_print(output, flags, std_msg, timestamp,
  477. level, domain_id, source_id);
  478. if (log_msg_is_std(msg)) {
  479. std_print(msg, output);
  480. } else if (raw_string) {
  481. raw_string_print(msg, output);
  482. } else {
  483. hexdump_print(msg, output, prefix_offset, flags);
  484. }
  485. if (!raw_string) {
  486. postfix_print(output, flags, level);
  487. }
  488. log_output_flush(output);
  489. }
  490. void log_output_msg2_process(const struct log_output *output,
  491. struct log_msg2 *msg, uint32_t flags)
  492. {
  493. log_timestamp_t timestamp = log_msg2_get_timestamp(msg);
  494. uint8_t level = log_msg2_get_level(msg);
  495. bool raw_string = (level == LOG_LEVEL_INTERNAL_RAW_STRING);
  496. uint32_t prefix_offset;
  497. if (IS_ENABLED(CONFIG_LOG_MIPI_SYST_ENABLE) &&
  498. flags & LOG_OUTPUT_FLAG_FORMAT_SYST) {
  499. __ASSERT_NO_MSG(0);
  500. /* todo not supported
  501. * log_output_msg_syst_process(output, msg, flags);
  502. */
  503. return;
  504. }
  505. if (!raw_string) {
  506. void *source = (void *)log_msg2_get_source(msg);
  507. uint8_t domain_id = log_msg2_get_domain(msg);
  508. int16_t source_id = source ?
  509. (IS_ENABLED(CONFIG_LOG_RUNTIME_FILTERING) ?
  510. log_dynamic_source_id(source) :
  511. log_const_source_id(source)) :
  512. -1;
  513. prefix_offset = prefix_print(output, flags, 0, timestamp,
  514. level, domain_id, source_id);
  515. } else {
  516. prefix_offset = 0;
  517. }
  518. size_t len;
  519. uint8_t *data = log_msg2_get_package(msg, &len);
  520. if (len) {
  521. int err = cbpprintf(raw_string ? cr_out_func : out_func,
  522. (void *)output, data);
  523. (void)err;
  524. __ASSERT_NO_MSG(err >= 0);
  525. }
  526. data = log_msg2_get_data(msg, &len);
  527. if (len) {
  528. log_msg2_hexdump(output, data, len, prefix_offset, flags);
  529. }
  530. if (!raw_string) {
  531. postfix_print(output, flags, level);
  532. }
  533. log_output_flush(output);
  534. }
  535. static bool ends_with_newline(const char *fmt)
  536. {
  537. char c = '\0';
  538. while (*fmt != '\0') {
  539. c = *fmt;
  540. fmt++;
  541. }
  542. return (c == '\n');
  543. }
  544. void log_output_string(const struct log_output *output,
  545. struct log_msg_ids src_level, uint32_t timestamp,
  546. const char *fmt, va_list ap, uint32_t flags)
  547. {
  548. int length;
  549. uint8_t level = (uint8_t)src_level.level;
  550. uint8_t domain_id = (uint8_t)src_level.domain_id;
  551. int16_t source_id = (int16_t)src_level.source_id;
  552. bool raw_string = (level == LOG_LEVEL_INTERNAL_RAW_STRING);
  553. if (IS_ENABLED(CONFIG_LOG_MIPI_SYST_ENABLE) &&
  554. flags & LOG_OUTPUT_FLAG_FORMAT_SYST) {
  555. log_output_string_syst_process(output,
  556. src_level, fmt, ap, flags);
  557. return;
  558. }
  559. if (!raw_string) {
  560. prefix_print(output, flags, true, timestamp,
  561. level, domain_id, source_id);
  562. }
  563. length = cbvprintf(out_func, (void *)output, fmt, ap);
  564. (void)length;
  565. if (raw_string) {
  566. /* add \r if string ends with newline. */
  567. if (ends_with_newline(fmt)) {
  568. print_formatted(output, "\r");
  569. }
  570. } else {
  571. postfix_print(output, flags, level);
  572. }
  573. log_output_flush(output);
  574. }
  575. void log_output_hexdump(const struct log_output *output,
  576. struct log_msg_ids src_level, uint32_t timestamp,
  577. const char *metadata, const uint8_t *data,
  578. uint32_t length, uint32_t flags)
  579. {
  580. uint32_t prefix_offset;
  581. uint8_t level = (uint8_t)src_level.level;
  582. uint8_t domain_id = (uint8_t)src_level.domain_id;
  583. int16_t source_id = (int16_t)src_level.source_id;
  584. if (IS_ENABLED(CONFIG_LOG_MIPI_SYST_ENABLE) &&
  585. flags & LOG_OUTPUT_FLAG_FORMAT_SYST) {
  586. log_output_hexdump_syst_process(output,
  587. src_level, data, length, flags);
  588. return;
  589. }
  590. prefix_offset = prefix_print(output, flags, true, timestamp,
  591. level, domain_id, source_id);
  592. /* Print metadata */
  593. print_formatted(output, "%s", metadata);
  594. while (length != 0U) {
  595. uint32_t part_len = length > HEXDUMP_BYTES_IN_LINE ?
  596. HEXDUMP_BYTES_IN_LINE : length;
  597. hexdump_line_print(output, data, part_len,
  598. prefix_offset, flags);
  599. data += part_len;
  600. length -= part_len;
  601. }
  602. postfix_print(output, flags, level);
  603. log_output_flush(output);
  604. }
  605. void log_output_dropped_process(const struct log_output *output, uint32_t cnt)
  606. {
  607. char buf[5];
  608. int len;
  609. static const char prefix[] = DROPPED_COLOR_PREFIX "--- ";
  610. static const char postfix[] =
  611. " messages dropped ---\r\n" DROPPED_COLOR_POSTFIX;
  612. log_output_func_t outf = output->func;
  613. cnt = MIN(cnt, 9999);
  614. len = snprintk(buf, sizeof(buf), "%d", cnt);
  615. buffer_write(outf, (uint8_t *)prefix, sizeof(prefix) - 1,
  616. output->control_block->ctx);
  617. buffer_write(outf, buf, len, output->control_block->ctx);
  618. buffer_write(outf, (uint8_t *)postfix, sizeof(postfix) - 1,
  619. output->control_block->ctx);
  620. }
  621. void log_output_timestamp_freq_set(uint32_t frequency)
  622. {
  623. timestamp_div = 1U;
  624. /* There is no point to have frequency higher than 1MHz (ns are not
  625. * printed) and too high frequency leads to overflows in calculations.
  626. */
  627. while (frequency > 1000000) {
  628. frequency /= 2U;
  629. timestamp_div *= 2U;
  630. }
  631. freq = frequency;
  632. }
  633. uint64_t log_output_timestamp_to_us(uint32_t timestamp)
  634. {
  635. timestamp /= timestamp_div;
  636. return ((uint64_t) timestamp * 1000000U) / freq;
  637. }