printk.c 5.5 KB

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