printk.c 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280
  1. /*
  2. * Copyright (c) 2010, 2013-2014 Wind River Systems, Inc.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief Low-level debug output
  9. *
  10. * Low-level debugging output. Platform installs a character output routine at
  11. * init time. If no routine is installed, a nop routine is called.
  12. */
  13. #include <kernel.h>
  14. #include <sys/printk.h>
  15. #include <stdarg.h>
  16. #include <toolchain.h>
  17. #include <linker/sections.h>
  18. #include <syscall_handler.h>
  19. #include <logging/log.h>
  20. #include <sys/cbprintf.h>
  21. #include <sys/types.h>
  22. #include <soc.h>
  23. #if defined(CONFIG_PRINTK_SYNC) && \
  24. !(defined(CONFIG_LOG_PRINTK) && defined(CONFIG_LOG2))
  25. static struct k_spinlock lock;
  26. #endif
  27. #ifdef CONFIG_PRINTK
  28. /**
  29. * @brief Default character output routine that does nothing
  30. * @param c Character to swallow
  31. *
  32. * Note this is defined as a weak symbol, allowing architecture code
  33. * to override it where possible to enable very early logging.
  34. *
  35. * @return 0
  36. */
  37. /* LCOV_EXCL_START */
  38. __attribute__((weak)) int arch_printk_char_out(int c)
  39. {
  40. ARG_UNUSED(c);
  41. /* do nothing */
  42. return 0;
  43. }
  44. /* LCOV_EXCL_STOP */
  45. int (*_char_out)(int) = arch_printk_char_out;
  46. /**
  47. * @brief Install the character output routine for printk
  48. *
  49. * To be called by the platform's console driver at init time. Installs a
  50. * routine that outputs one ASCII character at a time.
  51. * @param fn putc routine to install
  52. *
  53. * @return N/A
  54. */
  55. void __printk_hook_install(int (*fn)(int))
  56. {
  57. _char_out = fn;
  58. }
  59. /**
  60. * @brief Get the current character output routine for printk
  61. *
  62. * To be called by any console driver that would like to save
  63. * current hook - if any - for later re-installation.
  64. *
  65. * @return a function pointer or NULL if no hook is set
  66. */
  67. void *__printk_get_hook(void)
  68. {
  69. return _char_out;
  70. }
  71. #endif /* CONFIG_PRINTK */
  72. #if defined(CONFIG_PRINTK) && \
  73. !(defined(CONFIG_LOG_PRINTK) && defined(CONFIG_LOG2))
  74. #ifdef CONFIG_USERSPACE
  75. struct buf_out_context {
  76. int count;
  77. unsigned int buf_count;
  78. char buf[CONFIG_PRINTK_BUFFER_SIZE];
  79. };
  80. static void buf_flush(struct buf_out_context *ctx)
  81. {
  82. k_str_out(ctx->buf, ctx->buf_count);
  83. ctx->buf_count = 0U;
  84. }
  85. static int buf_char_out(int c, void *ctx_p)
  86. {
  87. struct buf_out_context *ctx = ctx_p;
  88. ctx->count++;
  89. ctx->buf[ctx->buf_count++] = c;
  90. if (ctx->buf_count == CONFIG_PRINTK_BUFFER_SIZE) {
  91. buf_flush(ctx);
  92. }
  93. return c;
  94. }
  95. #endif /* CONFIG_USERSPACE */
  96. struct out_context {
  97. int count;
  98. };
  99. static int char_out(int c, void *ctx_p)
  100. {
  101. struct out_context *ctx = ctx_p;
  102. ctx->count++;
  103. return _char_out(c);
  104. }
  105. #ifdef CONFIG_USERSPACE
  106. void vprintk(const char *fmt, va_list ap)
  107. {
  108. if (k_is_user_context()) {
  109. struct buf_out_context ctx = { 0 };
  110. cbvprintf(buf_char_out, &ctx, fmt, ap);
  111. if (ctx.buf_count) {
  112. buf_flush(&ctx);
  113. }
  114. } else {
  115. struct out_context ctx = { 0 };
  116. #ifdef CONFIG_PRINTK_SYNC
  117. k_spinlock_key_t key = k_spin_lock(&lock);
  118. #endif
  119. cbvprintf(char_out, &ctx, fmt, ap);
  120. #ifdef CONFIG_PRINTK_SYNC
  121. k_spin_unlock(&lock, key);
  122. #endif
  123. }
  124. }
  125. #else
  126. #ifdef CONFIG_ACTIONS_PRINTK_DMA
  127. void __vprintk(const char *fmt, va_list ap)
  128. #else
  129. void vprintk(const char *fmt, va_list ap)
  130. #endif
  131. {
  132. struct out_context ctx = { 0 };
  133. #ifdef CONFIG_PRINTK_SYNC
  134. k_spinlock_key_t key = k_spin_lock(&lock);
  135. #endif
  136. if (soc_in_sleep_mode()) {
  137. sl_vprintk(fmt, ap);
  138. } else {
  139. cbvprintf(char_out, &ctx, fmt, ap);
  140. }
  141. #ifdef CONFIG_PRINTK_SYNC
  142. k_spin_unlock(&lock, key);
  143. #endif
  144. }
  145. #endif /* CONFIG_USERSPACE */
  146. void z_impl_k_str_out(char *c, size_t n)
  147. {
  148. size_t i;
  149. #ifdef CONFIG_PRINTK_SYNC
  150. k_spinlock_key_t key = k_spin_lock(&lock);
  151. #endif
  152. for (i = 0; i < n; i++) {
  153. _char_out(c[i]);
  154. }
  155. #ifdef CONFIG_PRINTK_SYNC
  156. k_spin_unlock(&lock, key);
  157. #endif
  158. }
  159. #ifdef CONFIG_USERSPACE
  160. static inline void z_vrfy_k_str_out(char *c, size_t n)
  161. {
  162. Z_OOPS(Z_SYSCALL_MEMORY_READ(c, n));
  163. z_impl_k_str_out((char *)c, n);
  164. }
  165. #include <syscalls/k_str_out_mrsh.c>
  166. #endif /* CONFIG_USERSPACE */
  167. /**
  168. * @brief Output a string
  169. *
  170. * Output a string on output installed by platform at init time. Some
  171. * printf-like formatting is available.
  172. *
  173. * Available formatting:
  174. * - %x/%X: outputs a number in hexadecimal format
  175. * - %s: outputs a null-terminated string
  176. * - %p: pointer, same as %x with a 0x prefix
  177. * - %u: outputs a number in unsigned decimal format
  178. * - %d/%i: outputs a number in signed decimal format
  179. *
  180. * Field width (with or without leading zeroes) is supported.
  181. * Length attributes h, hh, l, ll and z are supported. However, integral
  182. * values with %lld and %lli are only printed if they fit in a long
  183. * otherwise 'ERR' is printed. Full 64-bit values may be printed with %llx.
  184. *
  185. * @param fmt formatted string to output
  186. *
  187. * @return N/A
  188. */
  189. void printk(const char *fmt, ...)
  190. {
  191. va_list ap;
  192. va_start(ap, fmt);
  193. if (IS_ENABLED(CONFIG_LOG_PRINTK)) {
  194. log_printk(fmt, ap);
  195. } else {
  196. vprintk(fmt, ap);
  197. }
  198. va_end(ap);
  199. }
  200. #endif /* defined(CONFIG_PRINTK) && \
  201. * !(defined(CONFIG_LOG_PRINTK) && defined(CONFIG_LOG2))
  202. */
  203. struct str_context {
  204. char *str;
  205. int max;
  206. int count;
  207. };
  208. static int str_out(int c, struct str_context *ctx)
  209. {
  210. if (ctx->str == NULL || ctx->count >= ctx->max) {
  211. ctx->count++;
  212. return c;
  213. }
  214. if (ctx->count == ctx->max - 1) {
  215. ctx->str[ctx->count++] = '\0';
  216. } else {
  217. ctx->str[ctx->count++] = c;
  218. }
  219. return c;
  220. }
  221. int snprintk(char *str, size_t size, const char *fmt, ...)
  222. {
  223. va_list ap;
  224. int ret;
  225. va_start(ap, fmt);
  226. ret = vsnprintk(str, size, fmt, ap);
  227. va_end(ap);
  228. return ret;
  229. }
  230. int vsnprintk(char *str, size_t size, const char *fmt, va_list ap)
  231. {
  232. struct str_context ctx = { str, size, 0 };
  233. cbvprintf(str_out, &ctx, fmt, ap);
  234. if (ctx.count < ctx.max) {
  235. str[ctx.count] = '\0';
  236. }
  237. return ctx.count;
  238. }