cbprintf_internal.h 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425
  1. /*
  2. * Copyright (c) 2020 Nordic Semiconductor ASA
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_SYS_CBPRINTF_INTERNAL_H_
  7. #define ZEPHYR_INCLUDE_SYS_CBPRINTF_INTERNAL_H_
  8. #include <errno.h>
  9. #include <stdarg.h>
  10. #include <stddef.h>
  11. #include <stdint.h>
  12. #include <toolchain.h>
  13. #include <sys/util.h>
  14. #ifdef CONFIG_CBPRINTF_STATIC_PACKAGE_CHECK_ALIGNMENT
  15. #include <sys/__assert.h>
  16. #endif
  17. /*
  18. * Special alignment cases
  19. */
  20. #if defined(__i386__)
  21. /* there are no gaps on the stack */
  22. #define VA_STACK_ALIGN(type) 1
  23. #elif defined(__sparc__)
  24. /* there are no gaps on the stack */
  25. #define VA_STACK_ALIGN(type) 1
  26. #elif defined(__x86_64__)
  27. #define VA_STACK_MIN_ALIGN 8
  28. #elif defined(__aarch64__)
  29. #define VA_STACK_MIN_ALIGN 8
  30. #elif defined(__riscv)
  31. #define VA_STACK_MIN_ALIGN (__riscv_xlen / 8)
  32. #endif
  33. /*
  34. * Default alignment values if not specified by architecture config
  35. */
  36. #ifndef VA_STACK_MIN_ALIGN
  37. #define VA_STACK_MIN_ALIGN 1
  38. #endif
  39. #ifndef VA_STACK_ALIGN
  40. #define VA_STACK_ALIGN(type) MAX(VA_STACK_MIN_ALIGN, __alignof__(type))
  41. #endif
  42. static inline void z_cbprintf_wcpy(int *dst, int *src, size_t len)
  43. {
  44. for (size_t i = 0; i < len; i++) {
  45. dst[i] = src[i];
  46. }
  47. }
  48. #include <sys/cbprintf_cxx.h>
  49. #ifdef __cplusplus
  50. extern "C" {
  51. #endif
  52. #if defined(__sparc__)
  53. /* The SPARC V8 ABI guarantees that the arguments of a variable argument
  54. * list function are stored on the stack at addresses which are 32-bit
  55. * aligned. It means that variables of type unit64_t and double may not
  56. * be properly aligned on the stack.
  57. *
  58. * The compiler is aware of the ABI and takes care of this. However,
  59. * as we are directly accessing the variable argument list here, we need
  60. * to take the alignment into consideration and copy 64-bit arguments
  61. * as 32-bit words.
  62. */
  63. #define Z_CBPRINTF_VA_STACK_LL_DBL_MEMCPY 1
  64. #else
  65. #define Z_CBPRINTF_VA_STACK_LL_DBL_MEMCPY 0
  66. #endif
  67. /** @brief Return 1 if argument is a pointer to char or wchar_t
  68. *
  69. * @param x argument.
  70. *
  71. * @return 1 if char * or wchar_t *, 0 otherwise.
  72. */
  73. #ifdef __cplusplus
  74. #define Z_CBPRINTF_IS_PCHAR(x) z_cbprintf_cxx_is_pchar(x)
  75. #else
  76. #define Z_CBPRINTF_IS_PCHAR(x) \
  77. _Generic((x) + 0, \
  78. char * : 1, \
  79. const char * : 1, \
  80. volatile char * : 1, \
  81. const volatile char * : 1, \
  82. wchar_t * : 1, \
  83. const wchar_t * : 1, \
  84. volatile wchar_t * : 1, \
  85. const volatile wchar_t * : 1, \
  86. default : \
  87. 0)
  88. #endif
  89. /** @brief Calculate number of char * or wchar_t * arguments in the arguments.
  90. *
  91. * @param fmt string.
  92. *
  93. * @param ... string arguments.
  94. *
  95. * @return number of arguments which are char * or wchar_t *.
  96. */
  97. #define Z_CBPRINTF_HAS_PCHAR_ARGS(fmt, ...) \
  98. (FOR_EACH(Z_CBPRINTF_IS_PCHAR, (+), __VA_ARGS__))
  99. /**
  100. * @brief Check if formatted string must be packaged in runtime.
  101. *
  102. * @param skip number of char/wchar_t pointers in the argument list which are
  103. * accepted for static packaging.
  104. *
  105. * @param ... String with arguments (fmt, ...).
  106. *
  107. * @retval 1 if string must be packaged at runtime.
  108. * @retval 0 if string can be statically packaged.
  109. */
  110. #if Z_C_GENERIC
  111. #if 0
  112. #define Z_CBPRINTF_MUST_RUNTIME_PACKAGE(skip, ...) ({\
  113. _Pragma("GCC diagnostic push") \
  114. _Pragma("GCC diagnostic ignored \"-Wpointer-arith\"") \
  115. int _rv = COND_CODE_0(NUM_VA_ARGS_LESS_1(__VA_ARGS__), \
  116. (0), \
  117. (((Z_CBPRINTF_HAS_PCHAR_ARGS(__VA_ARGS__) - skip) > 0))); \
  118. _Pragma("GCC diagnostic pop")\
  119. _rv; \
  120. })
  121. #else
  122. #endif
  123. #define Z_CBPRINTF_MUST_RUNTIME_PACKAGE(skip, ...) 1
  124. #endif
  125. /** @brief Get storage size for given argument.
  126. *
  127. * Floats are promoted to double so they use size of double, others int storage
  128. * or it's own storage size if it is bigger than int.
  129. *
  130. * @param x argument.
  131. *
  132. * @return Number of bytes used for storing the argument.
  133. */
  134. #ifdef __cplusplus
  135. #define Z_CBPRINTF_ARG_SIZE(v) z_cbprintf_cxx_arg_size(v)
  136. #else
  137. #define Z_CBPRINTF_ARG_SIZE(v) ({\
  138. __auto_type _v = (v) + 0; \
  139. size_t _arg_size = _Generic((v), \
  140. float : sizeof(double), \
  141. default : \
  142. sizeof((_v)) \
  143. ); \
  144. _arg_size; \
  145. })
  146. #endif
  147. /** @brief Promote and store argument in the buffer.
  148. *
  149. * @param buf Buffer.
  150. *
  151. * @param arg Argument.
  152. */
  153. #ifdef __cplusplus
  154. #define Z_CBPRINTF_STORE_ARG(buf, arg) z_cbprintf_cxx_store_arg(buf, arg)
  155. #else
  156. #define Z_CBPRINTF_STORE_ARG(buf, arg) do { \
  157. if (Z_CBPRINTF_VA_STACK_LL_DBL_MEMCPY) { \
  158. /* If required, copy arguments by word to avoid unaligned access.*/ \
  159. __auto_type _v = (arg) + 0; \
  160. double _d = _Generic((arg) + 0, \
  161. float : (arg) + 0, \
  162. default : \
  163. 0.0); \
  164. size_t arg_size = Z_CBPRINTF_ARG_SIZE(arg); \
  165. size_t _wsize = arg_size / sizeof(int); \
  166. z_cbprintf_wcpy((int *)buf, \
  167. (int *) _Generic((arg) + 0, float : &_d, default : &_v), \
  168. _wsize); \
  169. } else { \
  170. *_Generic((arg) + 0, \
  171. char : (int *)buf, \
  172. unsigned char: (int *)buf, \
  173. short : (int *)buf, \
  174. unsigned short : (int *)buf, \
  175. int : (int *)buf, \
  176. unsigned int : (unsigned int *)buf, \
  177. long : (long *)buf, \
  178. unsigned long : (unsigned long *)buf, \
  179. long long : (long long *)buf, \
  180. unsigned long long : (unsigned long long *)buf, \
  181. float : (double *)buf, \
  182. double : (double *)buf, \
  183. long double : (long double *)buf, \
  184. default : \
  185. (const void **)buf) = arg; \
  186. } \
  187. } while (0)
  188. #endif
  189. /** @brief Return alignment needed for given argument.
  190. *
  191. * @param _arg Argument
  192. *
  193. * @return Alignment in bytes.
  194. */
  195. #ifdef __cplusplus
  196. #define Z_CBPRINTF_ALIGNMENT(_arg) z_cbprintf_cxx_alignment(_arg)
  197. #else
  198. #define Z_CBPRINTF_ALIGNMENT(_arg) \
  199. MAX(_Generic((_arg) + 0, \
  200. float : VA_STACK_ALIGN(double), \
  201. double : VA_STACK_ALIGN(double), \
  202. long double : VA_STACK_ALIGN(long double), \
  203. long long : VA_STACK_ALIGN(long long), \
  204. unsigned long long : VA_STACK_ALIGN(long long), \
  205. default : \
  206. __alignof__((_arg) + 0)), VA_STACK_MIN_ALIGN)
  207. #endif
  208. /** @brief Detect long double variable as a constant expression.
  209. *
  210. * Macro is used in static assertion. On some platforms C++ static inline
  211. * template function is not a constant expression and cannot be used. In that
  212. * case long double usage will not be detected.
  213. *
  214. * @param x Argument.
  215. *
  216. * @return 1 if @p x is a long double, 0 otherwise.
  217. */
  218. #ifdef __cplusplus
  219. #if defined(__x86_64__) || defined(__riscv) || defined(__aarch64__)
  220. #define Z_CBPRINTF_IS_LONGDOUBLE(x) 0
  221. #else
  222. #define Z_CBPRINTF_IS_LONGDOUBLE(x) z_cbprintf_cxx_is_longdouble(x)
  223. #endif
  224. #else
  225. #define Z_CBPRINTF_IS_LONGDOUBLE(x) \
  226. _Generic((x) + 0, long double : 1, default : 0)
  227. #endif
  228. /** @brief Safely package arguments to a buffer.
  229. *
  230. * Argument is put into the buffer if capable buffer is provided. Length is
  231. * incremented even if data is not packaged.
  232. *
  233. * @param _buf buffer.
  234. *
  235. * @param _idx index. Index is postincremented.
  236. *
  237. * @param _align_offset Current index with alignment offset.
  238. *
  239. * @param _max maximum index (buffer capacity).
  240. *
  241. * @param _arg argument.
  242. */
  243. #define Z_CBPRINTF_PACK_ARG2(_buf, _idx, _align_offset, _max, \
  244. _cfg_flags, _s_idx, _s_buf, _arg) \
  245. do { \
  246. BUILD_ASSERT(!((sizeof(double) < VA_STACK_ALIGN(long double)) && \
  247. Z_CBPRINTF_IS_LONGDOUBLE(_arg) && \
  248. !IS_ENABLED(CONFIG_CBPRINTF_PACKAGE_LONGDOUBLE)),\
  249. "Packaging of long double not enabled in Kconfig."); \
  250. while (_align_offset % Z_CBPRINTF_ALIGNMENT(_arg)) { \
  251. _idx += sizeof(int); \
  252. _align_offset += sizeof(int); \
  253. } \
  254. uint32_t _arg_size = Z_CBPRINTF_ARG_SIZE(_arg); \
  255. if (Z_CBPRINTF_IS_PCHAR(_arg)) { \
  256. _s_buf[_s_idx++] = _idx / sizeof(int); \
  257. } \
  258. if (_buf && _idx < (int)_max) { \
  259. Z_CBPRINTF_STORE_ARG(&_buf[_idx], _arg); \
  260. } \
  261. _idx += _arg_size; \
  262. _align_offset += _arg_size; \
  263. } while (0)
  264. /** @brief Package single argument.
  265. *
  266. * Macro is called in a loop for each argument in the string.
  267. *
  268. * @param arg argument.
  269. */
  270. #define Z_CBPRINTF_PACK_ARG(arg) \
  271. Z_CBPRINTF_PACK_ARG2(_pbuf, _pkg_len, _pkg_offset, _pmax, _flags, \
  272. _s_cnt, _s_buffer, arg)
  273. /** @brief Package descriptor.
  274. *
  275. * @param len Package length.
  276. *
  277. * @param str_cnt Number of strings stored in the package.
  278. */
  279. struct z_cbprintf_desc {
  280. uint8_t len;
  281. uint8_t str_cnt;
  282. uint8_t ro_str_cnt;
  283. };
  284. /** @brief Package header. */
  285. union z_cbprintf_hdr {
  286. struct z_cbprintf_desc desc;
  287. void *raw;
  288. };
  289. /* When using clang additional warning needs to be suppressed since each
  290. * argument of fmt string is used for sizeof() which results in the warning
  291. * if argument is a stirng literal. Suppression is added here instead of
  292. * the macro which generates the warning to not slow down the compiler.
  293. */
  294. #if __clang__ == 1
  295. #define Z_CBPRINTF_SUPPRESS_SIZEOF_ARRAY_DECAY \
  296. _Pragma("GCC diagnostic ignored \"-Wsizeof-array-decay\"")
  297. #else
  298. #define Z_CBPRINTF_SUPPRESS_SIZEOF_ARRAY_DECAY
  299. #endif
  300. /** @brief Statically package a formatted string with arguments.
  301. *
  302. * @param buf buffer. If null then only length is calculated.
  303. *
  304. * @param _inlen buffer capacity on input. Ignored when @p buf is null.
  305. *
  306. * @param _outlen number of bytes required to store the package.
  307. *
  308. * @param _align_offset Input buffer alignment offset in words. Where offset 0
  309. * means that buffer is aligned to CBPRINTF_PACKAGE_ALIGNMENT.
  310. *
  311. * @param _flags Option flags. See @ref CBPRINTF_PACKAGE_FLAGS.
  312. *
  313. * @param ... String with variable list of arguments.
  314. */
  315. #define Z_CBPRINTF_STATIC_PACKAGE_GENERIC(buf, _inlen, _outlen, _align_offset, \
  316. _flags, ... /* fmt, ... */) \
  317. do { \
  318. _Pragma("GCC diagnostic push") \
  319. _Pragma("GCC diagnostic ignored \"-Wpointer-arith\"") \
  320. Z_CBPRINTF_SUPPRESS_SIZEOF_ARRAY_DECAY \
  321. BUILD_ASSERT(!IS_ENABLED(CONFIG_XTENSA) || \
  322. (IS_ENABLED(CONFIG_XTENSA) && \
  323. !(_align_offset % CBPRINTF_PACKAGE_ALIGNMENT)), \
  324. "Xtensa requires aligned package."); \
  325. BUILD_ASSERT((_align_offset % sizeof(int)) == 0, \
  326. "Alignment offset must be multiply of a word."); \
  327. IF_ENABLED(CONFIG_CBPRINTF_STATIC_PACKAGE_CHECK_ALIGNMENT, \
  328. (__ASSERT(!((uintptr_t)buf & (CBPRINTF_PACKAGE_ALIGNMENT - 1)), \
  329. "Buffer must be aligned.");)) \
  330. bool str_idxs = _flags & CBPRINTF_PACKAGE_ADD_STRING_IDXS; \
  331. uint8_t *_pbuf = buf; \
  332. uint8_t _s_cnt = 0; \
  333. uint16_t _s_buffer[16]; \
  334. size_t _pmax = (buf != NULL) ? _inlen : INT32_MAX; \
  335. int _pkg_len = 0; \
  336. int _total_len = 0; \
  337. int _pkg_offset = _align_offset; \
  338. union z_cbprintf_hdr *_len_loc; \
  339. /* package starts with string address and field with length */ \
  340. if (_pmax < sizeof(union z_cbprintf_hdr)) { \
  341. _outlen = -ENOSPC; \
  342. break; \
  343. } \
  344. _len_loc = (union z_cbprintf_hdr *)_pbuf; \
  345. _pkg_len += sizeof(union z_cbprintf_hdr); \
  346. _pkg_offset += sizeof(union z_cbprintf_hdr); \
  347. /* Pack remaining arguments */\
  348. FOR_EACH(Z_CBPRINTF_PACK_ARG, (;), __VA_ARGS__);\
  349. _total_len = _pkg_len; \
  350. if (str_idxs) {\
  351. _total_len += _s_cnt; \
  352. if (_pbuf) { \
  353. for (int i = 0; i < _s_cnt; i++) { \
  354. _pbuf[_pkg_len + i] = _s_buffer[i]; \
  355. } \
  356. } \
  357. } \
  358. /* Store length */ \
  359. _outlen = (_total_len > (int)_pmax) ? -ENOSPC : _total_len; \
  360. /* Store length in the header, set number of dumped strings to 0 */ \
  361. if (_pbuf) { \
  362. union z_cbprintf_hdr hdr = { \
  363. .desc = { \
  364. .len = (uint8_t)(_pkg_len / sizeof(int)), \
  365. .str_cnt = 0, \
  366. .ro_str_cnt = str_idxs ? _s_cnt : (uint8_t)0, \
  367. } \
  368. }; \
  369. *_len_loc = hdr; \
  370. } \
  371. _Pragma("GCC diagnostic pop") \
  372. } while (0)
  373. #if Z_C_GENERIC
  374. #if 0
  375. #define Z_CBPRINTF_STATIC_PACKAGE(packaged, inlen, outlen, align_offset, flags, \
  376. ... /* fmt, ... */) \
  377. Z_CBPRINTF_STATIC_PACKAGE_GENERIC(packaged, inlen, outlen, \
  378. align_offset, flags, __VA_ARGS__)
  379. #else
  380. #endif
  381. #define Z_CBPRINTF_STATIC_PACKAGE(packaged, inlen, outlen, align_offset, flags, \
  382. ... /* fmt, ... */) \
  383. do { \
  384. /* Small trick needed to avoid warning on always true */ \
  385. if (((uintptr_t)packaged + 1) != 1) { \
  386. outlen = cbprintf_package(packaged, inlen, flags, __VA_ARGS__); \
  387. } else { \
  388. outlen = cbprintf_package(NULL, align_offset, flags, __VA_ARGS__); \
  389. } \
  390. } while (0)
  391. #endif /* Z_C_GENERIC */
  392. #ifdef __cplusplus
  393. }
  394. #endif
  395. #endif /* ZEPHYR_INCLUDE_SYS_CBPRINTF_INTERNAL_H_ */