cbprintf_packaged.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683
  1. /*
  2. * Copyright (c) 2021 BayLibre, SAS
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <errno.h>
  7. #include <stdarg.h>
  8. #include <stdint.h>
  9. #include <string.h>
  10. #include <sys/cbprintf.h>
  11. #include <sys/types.h>
  12. #include <sys/util.h>
  13. #include <sys/__assert.h>
  14. /**
  15. * @brief Check if address is in read only section.
  16. *
  17. * @param addr Address.
  18. *
  19. * @return True if address identified within read only section.
  20. */
  21. static inline bool ptr_in_rodata(const char *addr)
  22. {
  23. #if defined(CBPRINTF_VIA_UNIT_TEST)
  24. /* Unit test is X86 (or other host) but not using Zephyr
  25. * linker scripts.
  26. */
  27. #define RO_START 0
  28. #define RO_END 0
  29. #elif defined(CONFIG_ARC) || defined(CONFIG_ARM) || defined(CONFIG_X86) \
  30. || defined(CONFIG_RISCV) || defined(CONFIG_ARM64) \
  31. || defined(CONFIG_NIOS2)
  32. extern char __rodata_region_start[];
  33. extern char __rodata_region_end[];
  34. #define RO_START __rodata_region_start
  35. #define RO_END __rodata_region_end
  36. #elif defined(CONFIG_XTENSA)
  37. extern char _rodata_start[];
  38. extern char _rodata_end[];
  39. #define RO_START _rodata_start
  40. #define RO_END _rodata_end
  41. #else
  42. #define RO_START 0
  43. #define RO_END 0
  44. #endif
  45. return ((addr >= (const char *)RO_START) &&
  46. (addr < (const char *)RO_END));
  47. }
  48. /*
  49. * va_list creation
  50. */
  51. #if defined(__aarch64__)
  52. /*
  53. * Reference:
  54. *
  55. * Procedure Call Standard for the ARM 64-bit Architecture
  56. */
  57. struct __va_list {
  58. void *__stack;
  59. void *__gr_top;
  60. void *__vr_top;
  61. int __gr_offs;
  62. int __vr_offs;
  63. };
  64. BUILD_ASSERT(sizeof(va_list) == sizeof(struct __va_list),
  65. "architecture specific support is wrong");
  66. static int cbprintf_via_va_list(cbprintf_cb out, void *ctx,
  67. const char *fmt, void *buf)
  68. {
  69. union {
  70. va_list ap;
  71. struct __va_list __ap;
  72. } u;
  73. /* create a valid va_list with our buffer */
  74. u.__ap.__stack = buf;
  75. u.__ap.__gr_top = NULL;
  76. u.__ap.__vr_top = NULL;
  77. u.__ap.__gr_offs = 0;
  78. u.__ap.__vr_offs = 0;
  79. return cbvprintf(out, ctx, fmt, u.ap);
  80. }
  81. #elif defined(__x86_64__)
  82. /*
  83. * Reference:
  84. *
  85. * System V Application Binary Interface
  86. * AMD64 Architecture Processor Supplement
  87. */
  88. struct __va_list {
  89. unsigned int gp_offset;
  90. unsigned int fp_offset;
  91. void *overflow_arg_area;
  92. void *reg_save_area;
  93. };
  94. BUILD_ASSERT(sizeof(va_list) == sizeof(struct __va_list),
  95. "architecture specific support is wrong");
  96. static int cbprintf_via_va_list(cbprintf_cb out, void *ctx,
  97. const char *fmt, void *buf)
  98. {
  99. union {
  100. va_list ap;
  101. struct __va_list __ap;
  102. } u;
  103. /* create a valid va_list with our buffer */
  104. u.__ap.overflow_arg_area = buf;
  105. u.__ap.reg_save_area = NULL;
  106. u.__ap.gp_offset = (6 * 8);
  107. u.__ap.fp_offset = (6 * 8 + 16 * 16);
  108. return cbvprintf(out, ctx, fmt, u.ap);
  109. }
  110. #elif defined(__xtensa__)
  111. /*
  112. * Reference:
  113. *
  114. * gcc source code (gcc/config/xtensa/xtensa.c)
  115. * xtensa_build_builtin_va_list(), xtensa_va_start(),
  116. * xtensa_gimplify_va_arg_expr()
  117. */
  118. struct __va_list {
  119. void *__va_stk;
  120. void *__va_reg;
  121. int __va_ndx;
  122. };
  123. BUILD_ASSERT(sizeof(va_list) == sizeof(struct __va_list),
  124. "architecture specific support is wrong");
  125. static int cbprintf_via_va_list(cbprintf_cb out, void *ctx,
  126. const char *fmt, void *buf)
  127. {
  128. union {
  129. va_list ap;
  130. struct __va_list __ap;
  131. } u;
  132. /* create a valid va_list with our buffer */
  133. u.__ap.__va_stk = (char *)buf - 32;
  134. u.__ap.__va_reg = NULL;
  135. u.__ap.__va_ndx = (6 + 2) * 4;
  136. return cbvprintf(out, ctx, fmt, u.ap);
  137. }
  138. #else
  139. /*
  140. * Default implementation shared by many architectures like
  141. * 32-bit ARM and Intel.
  142. *
  143. * We assume va_list is a simple pointer.
  144. */
  145. BUILD_ASSERT(sizeof(va_list) == sizeof(void *),
  146. "architecture specific support is needed");
  147. static int cbprintf_via_va_list(cbprintf_cb out, void *ctx,
  148. const char *fmt, void *buf)
  149. {
  150. union {
  151. va_list ap;
  152. void *ptr;
  153. } u;
  154. u.ptr = buf;
  155. return cbvprintf(out, ctx, fmt, u.ap);
  156. }
  157. #endif
  158. int cbvprintf_package(void *packaged, size_t len, uint32_t flags,
  159. const char *fmt, va_list ap)
  160. {
  161. /* Internally, byte is used to store location of a string argument within a
  162. * package. MSB bit is set if string is read-only so effectively 7 bits are
  163. * used for index, which should be enough.
  164. */
  165. #define CBPRINTF_STR_POS_RO_FLAG BIT(7)
  166. #define CBPRINTF_STR_POS_MASK BIT_MASK(7)
  167. char *buf = packaged, *buf0 = buf;
  168. unsigned int align, size, i, s_idx = 0, s_rw_cnt = 0, s_ro_cnt = 0;
  169. uint8_t str_ptr_pos[16];
  170. const char *s;
  171. bool parsing = false;
  172. /* Buffer must be aligned at least to size of a pointer. */
  173. if ((uintptr_t)packaged & (sizeof(void *) - 1)) {
  174. return -EFAULT;
  175. }
  176. #if defined(__xtensa__)
  177. /* Xtensa requires package to be 16 bytes aligned. */
  178. if ((uintptr_t)packaged & (CBPRINTF_PACKAGE_ALIGNMENT - 1)) {
  179. return -EFAULT;
  180. }
  181. #endif
  182. /*
  183. * Make room to store the arg list size and the number of
  184. * appended strings. They both occupy 1 byte each.
  185. *
  186. * Given the next value to store is the format string pointer
  187. * which is guaranteed to be at least 4 bytes, we just reserve
  188. * a pointer size for the above to preserve alignment.
  189. */
  190. buf += sizeof(char *);
  191. /*
  192. * When buf0 is NULL we don't store anything.
  193. * Instead we count the needed space to store the data.
  194. * In this case, incoming len argument indicates the anticipated
  195. * buffer "misalignment" offset.
  196. */
  197. if (!buf0) {
  198. #if defined(__xtensa__)
  199. if (len % CBPRINTF_PACKAGE_ALIGNMENT) {
  200. return -EFAULT;
  201. }
  202. #endif
  203. buf += len % CBPRINTF_PACKAGE_ALIGNMENT;
  204. len = -(len % CBPRINTF_PACKAGE_ALIGNMENT);
  205. }
  206. /*
  207. * Otherwise we must ensure we can store at least
  208. * thepointer to the format string itself.
  209. */
  210. if (buf0 && buf - buf0 + sizeof(char *) > len) {
  211. return -ENOSPC;
  212. }
  213. /*
  214. * Then process the format string itself.
  215. * Here we branch directly into the code processing strings
  216. * which is in the middle of the following while() loop. That's the
  217. * reason for the post-decrement on fmt as it will be incremented
  218. * prior to the next (actually first) round of that loop.
  219. */
  220. s = fmt--;
  221. align = VA_STACK_ALIGN(char *);
  222. size = sizeof(char *);
  223. goto process_string;
  224. /* Scan the format string */
  225. while (*++fmt) {
  226. if (!parsing) {
  227. if (*fmt == '%') {
  228. parsing = true;
  229. align = VA_STACK_ALIGN(int);
  230. size = sizeof(int);
  231. }
  232. continue;
  233. }
  234. switch (*fmt) {
  235. case '%':
  236. parsing = false;
  237. continue;
  238. case '#':
  239. case '-':
  240. case '+':
  241. case ' ':
  242. case '0':
  243. case '1':
  244. case '2':
  245. case '3':
  246. case '4':
  247. case '5':
  248. case '6':
  249. case '7':
  250. case '8':
  251. case '9':
  252. case '.':
  253. case 'h':
  254. case 'l':
  255. case 'L':
  256. continue;
  257. case '*':
  258. break;
  259. case 'j':
  260. align = VA_STACK_ALIGN(intmax_t);
  261. size = sizeof(intmax_t);
  262. continue;
  263. case 'z':
  264. align = VA_STACK_ALIGN(size_t);
  265. size = sizeof(size_t);
  266. continue;
  267. case 't':
  268. align = VA_STACK_ALIGN(ptrdiff_t);
  269. size = sizeof(ptrdiff_t);
  270. continue;
  271. case 'c':
  272. case 'd':
  273. case 'i':
  274. case 'o':
  275. case 'u':
  276. case 'x':
  277. case 'X':
  278. if (fmt[-1] == 'l') {
  279. if (fmt[-2] == 'l') {
  280. align = VA_STACK_ALIGN(long long);
  281. size = sizeof(long long);
  282. } else {
  283. align = VA_STACK_ALIGN(long);
  284. size = sizeof(long);
  285. }
  286. }
  287. parsing = false;
  288. break;
  289. case 's':
  290. case 'p':
  291. case 'n':
  292. align = VA_STACK_ALIGN(void *);
  293. size = sizeof(void *);
  294. parsing = false;
  295. break;
  296. case 'a':
  297. case 'A':
  298. case 'e':
  299. case 'E':
  300. case 'f':
  301. case 'F':
  302. case 'g':
  303. case 'G': {
  304. /*
  305. * Handle floats separately as they may be
  306. * held in a different register set.
  307. */
  308. union { double d; long double ld; } v;
  309. if (fmt[-1] == 'L') {
  310. v.ld = va_arg(ap, long double);
  311. align = VA_STACK_ALIGN(long double);
  312. size = sizeof(long double);
  313. } else {
  314. v.d = va_arg(ap, double);
  315. align = VA_STACK_ALIGN(double);
  316. size = sizeof(double);
  317. }
  318. /* align destination buffer location */
  319. buf = (void *) ROUND_UP(buf, align);
  320. if (buf0) {
  321. /* make sure it fits */
  322. if (buf - buf0 + size > len) {
  323. return -ENOSPC;
  324. }
  325. if (Z_CBPRINTF_VA_STACK_LL_DBL_MEMCPY) {
  326. memcpy(buf, &v, size);
  327. } else if (fmt[-1] == 'L') {
  328. *(long double *)buf = v.ld;
  329. } else {
  330. *(double *)buf = v.d;
  331. }
  332. }
  333. buf += size;
  334. parsing = false;
  335. continue;
  336. }
  337. default:
  338. parsing = false;
  339. continue;
  340. }
  341. /* align destination buffer location */
  342. buf = (void *) ROUND_UP(buf, align);
  343. /* make sure the data fits */
  344. if (buf0 && buf - buf0 + size > len) {
  345. return -ENOSPC;
  346. }
  347. /* copy va_list data over to our buffer */
  348. if (*fmt == 's') {
  349. s = va_arg(ap, char *);
  350. process_string:
  351. if (buf0) {
  352. *(const char **)buf = s;
  353. }
  354. /* Bother about read only strings only if storing
  355. * string indexes is requested.
  356. */
  357. bool is_ro = ptr_in_rodata(s);
  358. bool str_idxs = flags & CBPRINTF_PACKAGE_ADD_STRING_IDXS;
  359. bool need_ro = is_ro && str_idxs;
  360. if (ptr_in_rodata(s) && !str_idxs) {
  361. /* do nothing special */
  362. } else if (buf0) {
  363. /*
  364. * Remember string pointer location.
  365. * We will append it later.
  366. */
  367. if (s_idx >= ARRAY_SIZE(str_ptr_pos)) {
  368. __ASSERT(false, "str_ptr_pos[] too small");
  369. return -EINVAL;
  370. }
  371. if ((buf - buf0) > CBPRINTF_STR_POS_MASK) {
  372. __ASSERT(false, "String with too many arguments");
  373. return -EINVAL;
  374. }
  375. /* Add marking to identify if read only string. */
  376. uint8_t ro_flag = need_ro ?
  377. CBPRINTF_STR_POS_RO_FLAG : 0;
  378. if (ro_flag) {
  379. s_ro_cnt++;
  380. } else {
  381. s_rw_cnt++;
  382. }
  383. /* Use same multiple as the arg list size. */
  384. str_ptr_pos[s_idx++] = ro_flag |
  385. (buf - buf0) / sizeof(int);
  386. } else {
  387. if (!is_ro) {
  388. /*
  389. * Add the string length, the final '\0'
  390. * and size of the pointer position prefix.
  391. */
  392. len += strlen(s) + 1 + 1;
  393. } else if (need_ro) {
  394. /*
  395. * Add only pointer position prefix for
  396. * read only string is requested.
  397. */
  398. len += 1;
  399. }
  400. }
  401. buf += sizeof(char *);
  402. } else if (size == sizeof(int)) {
  403. int v = va_arg(ap, int);
  404. if (buf0) {
  405. *(int *)buf = v;
  406. }
  407. buf += sizeof(int);
  408. } else if (size == sizeof(long)) {
  409. long v = va_arg(ap, long);
  410. if (buf0) {
  411. *(long *)buf = v;
  412. }
  413. buf += sizeof(long);
  414. } else if (size == sizeof(long long)) {
  415. long long v = va_arg(ap, long long);
  416. if (buf0) {
  417. if (Z_CBPRINTF_VA_STACK_LL_DBL_MEMCPY) {
  418. memcpy(buf, &v, sizeof(long long));
  419. } else {
  420. *(long long *)buf = v;
  421. }
  422. }
  423. buf += sizeof(long long);
  424. } else {
  425. __ASSERT(false, "unexpected size %u", size);
  426. return -EINVAL;
  427. }
  428. }
  429. /*
  430. * We remember the size of the argument list as a multiple of
  431. * sizeof(int) and limit it to a 8-bit field. That means 1020 bytes
  432. * worth of va_list, or about 127 arguments on a 64-bit system
  433. * (twice that on 32-bit systems). That ought to be good enough.
  434. */
  435. if ((buf - buf0) / sizeof(int) > 255) {
  436. __ASSERT(false, "too many format args");
  437. return -EINVAL;
  438. }
  439. /*
  440. * If all we wanted was to count required buffer size
  441. * then we have it now.
  442. */
  443. if (!buf0) {
  444. return len + buf - buf0;
  445. }
  446. /* Clear our buffer header. We made room for it initially. */
  447. *(char **)buf0 = NULL;
  448. /* Record end of argument list and number of appended strings. */
  449. buf0[0] = (buf - buf0) / sizeof(int);
  450. buf0[1] = s_rw_cnt;
  451. buf0[2] = s_ro_cnt;
  452. /* Store strings pointer locations of read only strings. */
  453. if (s_ro_cnt) {
  454. for (i = 0; i < s_idx; i++) {
  455. if (!(str_ptr_pos[i] & CBPRINTF_STR_POS_RO_FLAG)) {
  456. continue;
  457. }
  458. uint8_t pos = str_ptr_pos[i] & CBPRINTF_STR_POS_MASK;
  459. /* make sure it fits */
  460. if (buf - buf0 + 1 > len) {
  461. return -ENOSPC;
  462. }
  463. /* store the pointer position prefix */
  464. *buf++ = pos;
  465. }
  466. }
  467. /* Store strings prefixed by their pointer location. */
  468. for (i = 0; i < s_idx; i++) {
  469. /* Process only RW strings. */
  470. if (str_ptr_pos[i] & CBPRINTF_STR_POS_RO_FLAG) {
  471. continue;
  472. }
  473. /* retrieve the string pointer */
  474. s = *(char **)(buf0 + str_ptr_pos[i] * sizeof(int));
  475. /* clear the in-buffer pointer (less entropy if compressed) */
  476. *(char **)(buf0 + str_ptr_pos[i] * sizeof(int)) = NULL;
  477. /* find the string length including terminating '\0' */
  478. size = strlen(s) + 1;
  479. /* make sure it fits */
  480. if (buf - buf0 + 1 + size > len) {
  481. return -ENOSPC;
  482. }
  483. /* store the pointer position prefix */
  484. *buf++ = str_ptr_pos[i];
  485. /* copy the string with its terminating '\0' */
  486. memcpy(buf, s, size);
  487. buf += size;
  488. }
  489. /*
  490. * TODO: remove pointers for appended strings since they're useless.
  491. * TODO: explore leveraging same mechanism to remove alignment padding
  492. */
  493. return buf - buf0;
  494. #undef CBPRINTF_STR_POS_RO_FLAG
  495. #undef CBPRINTF_STR_POS_MASK
  496. }
  497. int cbprintf_package(void *packaged, size_t len, uint32_t flags,
  498. const char *format, ...)
  499. {
  500. va_list ap;
  501. int ret;
  502. va_start(ap, format);
  503. ret = cbvprintf_package(packaged, len, flags, format, ap);
  504. va_end(ap);
  505. return ret;
  506. }
  507. int cbpprintf(cbprintf_cb out, void *ctx, void *packaged)
  508. {
  509. char *buf = packaged, *fmt, *s, **ps;
  510. unsigned int i, args_size, s_nbr, ros_nbr, s_idx;
  511. if (!buf) {
  512. return -EINVAL;
  513. }
  514. /* Retrieve the size of the arg list and number of strings. */
  515. args_size = ((uint8_t *)buf)[0] * sizeof(int);
  516. s_nbr = ((uint8_t *)buf)[1];
  517. ros_nbr = ((uint8_t *)buf)[2];
  518. /* Locate the string table */
  519. s = buf + args_size + ros_nbr;
  520. /*
  521. * Patch in string pointers.
  522. */
  523. for (i = 0; i < s_nbr; i++) {
  524. /* Locate pointer location for this string */
  525. s_idx = *(uint8_t *)s++;
  526. ps = (char **)(buf + s_idx * sizeof(int));
  527. /* update the pointer with current string location */
  528. *ps = s;
  529. /* move to next string */
  530. s += strlen(s) + 1;
  531. }
  532. /* Retrieve format string */
  533. fmt = ((char **)buf)[1];
  534. /* skip past format string pointer */
  535. buf += sizeof(char *) * 2;
  536. /* Turn this into a va_list and print it */
  537. return cbprintf_via_va_list(out, ctx, fmt, buf);
  538. }
  539. int cbprintf_fsc_package(void *in_packaged,
  540. size_t in_len,
  541. void *packaged,
  542. size_t len)
  543. {
  544. uint8_t *buf = in_packaged, *out = packaged;
  545. char **ps;
  546. unsigned int args_size, s_nbr, ros_nbr, s_idx;
  547. size_t out_len;
  548. size_t slen;
  549. if (!buf) {
  550. return -EINVAL;
  551. }
  552. if (packaged && (len < in_len)) {
  553. return -ENOSPC;
  554. }
  555. /* Retrieve the size of the arg list and number of strings. */
  556. args_size = buf[0] * sizeof(int);
  557. s_nbr = buf[1];
  558. ros_nbr = buf[2];
  559. out_len = in_len;
  560. if (packaged) {
  561. unsigned int rw_strs_len = in_len - (args_size + ros_nbr);
  562. memcpy(out, buf, args_size);
  563. out[1] = s_nbr + ros_nbr;
  564. out[2] = 0;
  565. out += args_size;
  566. /* Append all strings that were already part of the package. */
  567. memcpy(out, &buf[args_size + ros_nbr], rw_strs_len);
  568. out += rw_strs_len;
  569. }
  570. for (unsigned int i = 0; i < ros_nbr; i++) {
  571. /* Get string address location */
  572. s_idx = buf[args_size + i];
  573. ps = (char **)(buf + s_idx * sizeof(int));
  574. /* Get string length */
  575. slen = strlen(*ps) + 1;
  576. out_len += slen;
  577. /* Copy string into the buffer (if provided) and enough space. */
  578. if (packaged) {
  579. if (out_len > len) {
  580. return -ENOSPC;
  581. }
  582. *out++ = s_idx;
  583. memcpy(out, *ps, slen);
  584. out += slen;
  585. }
  586. }
  587. return out_len;
  588. }