/* * Copyright (c) 2021 BayLibre, SAS * * SPDX-License-Identifier: Apache-2.0 */ #include #include #include #include #include #include #include #include #include #include #include LOG_MODULE_REGISTER(cbprintf_package, CONFIG_CBPRINTF_PACKAGE_LOG_LEVEL); #if defined(CONFIG_CBPRINTF_PACKAGE_SUPPORT_TAGGED_ARGUMENTS) && \ !Z_C_GENERIC #error "CONFIG_CBPRINTF_PACKAGE_SUPPORT_TAGGED_ARGUMENTS " \ "requires toolchain to support _Generic!" #endif /** * @brief Check if address is in read only section. * * @param addr Address. * * @return True if address identified within read only section. */ static inline bool ptr_in_rodata(const char *addr) { #if defined(CBPRINTF_VIA_UNIT_TEST) /* Unit test is X86 (or other host) but not using Zephyr * linker scripts. */ return false; #else return linker_is_in_rodata(addr); #endif } /* * va_list creation */ #if defined(__CHECKER__) static int cbprintf_via_va_list(cbprintf_cb out, cbvprintf_external_formatter_func formatter, void *ctx, const char *fmt, void *buf) { return 0; } #elif defined(__aarch64__) /* * Reference: * * Procedure Call Standard for the ARM 64-bit Architecture */ struct __va_list { void *__stack; void *__gr_top; void *__vr_top; int __gr_offs; int __vr_offs; }; BUILD_ASSERT(sizeof(va_list) == sizeof(struct __va_list), "architecture specific support is wrong"); static int cbprintf_via_va_list(cbprintf_cb out, cbvprintf_external_formatter_func formatter, void *ctx, const char *fmt, void *buf) { union { va_list ap; struct __va_list __ap; } u; /* create a valid va_list with our buffer */ u.__ap.__stack = buf; u.__ap.__gr_top = NULL; u.__ap.__vr_top = NULL; u.__ap.__gr_offs = 0; u.__ap.__vr_offs = 0; return formatter(out, ctx, fmt, u.ap); } #elif defined(__x86_64__) /* * Reference: * * System V Application Binary Interface * AMD64 Architecture Processor Supplement */ struct __va_list { unsigned int gp_offset; unsigned int fp_offset; void *overflow_arg_area; void *reg_save_area; }; BUILD_ASSERT(sizeof(va_list) == sizeof(struct __va_list), "architecture specific support is wrong"); static int cbprintf_via_va_list(cbprintf_cb out, cbvprintf_external_formatter_func formatter, void *ctx, const char *fmt, void *buf) { union { va_list ap; struct __va_list __ap; } u; /* create a valid va_list with our buffer */ u.__ap.overflow_arg_area = buf; u.__ap.reg_save_area = NULL; u.__ap.gp_offset = (6 * 8); u.__ap.fp_offset = (6 * 8 + 16 * 16); return formatter(out, ctx, fmt, u.ap); } #elif defined(__xtensa__) /* * Reference: * * gcc source code (gcc/config/xtensa/xtensa.c) * xtensa_build_builtin_va_list(), xtensa_va_start(), * xtensa_gimplify_va_arg_expr() */ struct __va_list { void *__va_stk; void *__va_reg; int __va_ndx; }; BUILD_ASSERT(sizeof(va_list) == sizeof(struct __va_list), "architecture specific support is wrong"); static int cbprintf_via_va_list(cbprintf_cb out, cbvprintf_external_formatter_func formatter, void *ctx, const char *fmt, void *buf) { union { va_list ap; struct __va_list __ap; } u; /* create a valid va_list with our buffer */ u.__ap.__va_stk = (char *)buf - 32; u.__ap.__va_reg = NULL; u.__ap.__va_ndx = (6 + 2) * 4; return formatter(out, ctx, fmt, u.ap); } #else /* * Default implementation shared by many architectures like * 32-bit ARM and Intel. * * We assume va_list is a simple pointer. */ BUILD_ASSERT(sizeof(va_list) == sizeof(void *), "architecture specific support is needed"); static int cbprintf_via_va_list(cbprintf_cb out, cbvprintf_external_formatter_func formatter, void *ctx, const char *fmt, void *buf) { union { va_list ap; void *ptr; } u; u.ptr = buf; return formatter(out, ctx, fmt, u.ap); } #endif static size_t get_package_len(void *packaged) { __ASSERT_NO_MSG(packaged != NULL); uint8_t *buf = packaged; uint8_t *start = buf; unsigned int args_size, s_nbr, ros_nbr; args_size = buf[0] * sizeof(int); s_nbr = buf[1]; ros_nbr = buf[2]; /* Move beyond args. */ buf += args_size; /* Move beyond read-only string indexes array. */ buf += ros_nbr; /* Move beyond strings appended to the package. */ for (unsigned int i = 0; i < s_nbr; i++) { buf++; buf += strlen((const char *)buf) + 1; } return (size_t)(uintptr_t)(buf - start); } static int append_string(cbprintf_convert_cb cb, void *ctx, const char *str, uint16_t strl) { if (cb == NULL) { return 1 + strlen(str); } strl = strl > 0 ? strl : strlen(str) + 1; return cb(str, strl, ctx); } int cbvprintf_package(void *packaged, size_t len, uint32_t flags, const char *fmt, va_list ap) { /* * Internally, a byte is used to store location of a string argument within a * package. MSB bit is set if string is read-only so effectively 7 bits are * used for index, which should be enough. */ #define STR_POS_RO_FLAG BIT(7) #define STR_POS_MASK BIT_MASK(7) /* Buffer offset abstraction for better code clarity. */ #define BUF_OFFSET (buf - (uintptr_t)buf0) uint8_t *buf0 = packaged; /* buffer start (may be NULL) */ uintptr_t buf = (uintptr_t)buf0; /* current buffer position */ unsigned int size; /* current argument's size */ unsigned int align; /* current argument's required alignment */ uint8_t str_ptr_pos[16]; /* string pointer positions */ uint8_t str_ptr_arg[16]; /* string pointer argument index */ unsigned int s_idx = 0; /* index into str_ptr_pos[] */ unsigned int s_rw_cnt = 0; /* number of rw strings */ unsigned int s_ro_cnt = 0; /* number of ro strings */ int arg_idx = -1; /* Argument index. Preincremented thus starting from -1.*/ unsigned int i; const char *s; bool parsing = false; /* Flag indicates that rw strings are stored as array with positions, * instead of appending them to the package. */ bool rws_pos_en = !!(flags & CBPRINTF_PACKAGE_ADD_RW_STR_POS); /* Get number of first read only strings present in the string. * There is always at least 1 (fmt) but flags can indicate more, e.g * fixed prefix appended to all strings. */ int fros_cnt = 1 + Z_CBPRINTF_PACKAGE_FIRST_RO_STR_CNT_GET(flags); bool is_str_arg = false; union cbprintf_package_hdr *pkg_hdr = packaged; /* Buffer must be aligned at least to size of a pointer. */ if ((uintptr_t)packaged % sizeof(void *)) { return -EFAULT; } #if defined(__xtensa__) /* Xtensa requires package to be 16 bytes aligned. */ if ((uintptr_t)packaged % CBPRINTF_PACKAGE_ALIGNMENT) { return -EFAULT; } #endif /* * Make room to store the arg list size, the number of * appended writable strings and the number of appended * read-only strings. They both occupy 1 byte each. * Skip a byte. Then a uint32_t to store flags used to * create the package. * * Given the next value to store is the format string pointer * which is guaranteed to be at least 4 bytes, we just reserve * multiple of pointer size for the above to preserve alignment. * * Refer to union cbprintf_package_hdr for more details. */ buf += sizeof(*pkg_hdr); /* * When buf0 is NULL we don't store anything. * Instead we count the needed space to store the data. * In this case, incoming len argument indicates the anticipated * buffer "misalignment" offset. */ if (buf0 == NULL) { buf += len % CBPRINTF_PACKAGE_ALIGNMENT; /* * The space to store the data is represented by both the * buffer offset as well as the extra string data to be * appended. When only figuring out the needed space, we * don't append anything. Instead, we reuse the len variable * to sum the size of that data. * * Also, we subtract any initial misalignment offset from * the total as this won't be part of the buffer. To avoid * going negative with an unsigned variable, we add an offset * (CBPRINTF_PACKAGE_ALIGNMENT) that will be removed before * returning. */ len = CBPRINTF_PACKAGE_ALIGNMENT - (len % CBPRINTF_PACKAGE_ALIGNMENT); } /* * Otherwise we must ensure we can store at least * the pointer to the format string itself. */ if ((buf0 != NULL) && (BUF_OFFSET + sizeof(char *)) > len) { return -ENOSPC; } /* * Then process the format string itself. * Here we branch directly into the code processing strings * which is in the middle of the following while() loop. That's the * reason for the post-decrement on fmt as it will be incremented * prior to the next (actually first) round of that loop. */ s = fmt; --fmt; align = VA_STACK_ALIGN(char *); size = sizeof(char *); goto process_string; while (true) { #if defined(CONFIG_CBPRINTF_PACKAGE_SUPPORT_TAGGED_ARGUMENTS) if ((flags & CBPRINTF_PACKAGE_ARGS_ARE_TAGGED) == CBPRINTF_PACKAGE_ARGS_ARE_TAGGED) { int arg_tag = va_arg(ap, int); /* * Here we copy the tag over to the package. */ align = VA_STACK_ALIGN(int); size = sizeof(int); /* align destination buffer location */ buf = ROUND_UP(buf, align); /* make sure the data fits */ if (buf0 != NULL && BUF_OFFSET + size > len) { return -ENOSPC; } if (buf0 != NULL) { *(int *)buf = arg_tag; } buf += sizeof(int); if (arg_tag == CBPRINTF_PACKAGE_ARG_TYPE_END) { /* End of arguments */ break; } /* * There are lots of __fallthrough here since * quite a few of the data types have the same * storage size. */ switch (arg_tag) { case CBPRINTF_PACKAGE_ARG_TYPE_CHAR: __fallthrough; case CBPRINTF_PACKAGE_ARG_TYPE_UNSIGNED_CHAR: __fallthrough; case CBPRINTF_PACKAGE_ARG_TYPE_SHORT: __fallthrough; case CBPRINTF_PACKAGE_ARG_TYPE_UNSIGNED_SHORT: __fallthrough; case CBPRINTF_PACKAGE_ARG_TYPE_INT: __fallthrough; case CBPRINTF_PACKAGE_ARG_TYPE_UNSIGNED_INT: align = VA_STACK_ALIGN(int); size = sizeof(int); break; case CBPRINTF_PACKAGE_ARG_TYPE_LONG: __fallthrough; case CBPRINTF_PACKAGE_ARG_TYPE_UNSIGNED_LONG: align = VA_STACK_ALIGN(long); size = sizeof(long); break; case CBPRINTF_PACKAGE_ARG_TYPE_LONG_LONG: __fallthrough; case CBPRINTF_PACKAGE_ARG_TYPE_UNSIGNED_LONG_LONG: align = VA_STACK_ALIGN(long long); size = sizeof(long long); break; case CBPRINTF_PACKAGE_ARG_TYPE_FLOAT: __fallthrough; case CBPRINTF_PACKAGE_ARG_TYPE_DOUBLE: __fallthrough; case CBPRINTF_PACKAGE_ARG_TYPE_LONG_DOUBLE: { /* * Handle floats separately as they may be * held in a different register set. */ union { double d; long double ld; } v; if (arg_tag == CBPRINTF_PACKAGE_ARG_TYPE_LONG_DOUBLE) { v.ld = va_arg(ap, long double); align = VA_STACK_ALIGN(long double); size = sizeof(long double); } else { v.d = va_arg(ap, double); align = VA_STACK_ALIGN(double); size = sizeof(double); } /* align destination buffer location */ buf = ROUND_UP(buf, align); if (buf0 != NULL) { /* make sure it fits */ if ((BUF_OFFSET + size) > len) { return -ENOSPC; } if (Z_CBPRINTF_VA_STACK_LL_DBL_MEMCPY) { memcpy((void *)buf, (uint8_t *)&v, size); } else if (fmt[-1] == 'L') { *(long double *)buf = v.ld; } else { *(double *)buf = v.d; } } buf += size; parsing = false; continue; } case CBPRINTF_PACKAGE_ARG_TYPE_PTR_CHAR: is_str_arg = true; __fallthrough; case CBPRINTF_PACKAGE_ARG_TYPE_PTR_VOID: align = VA_STACK_ALIGN(void *); size = sizeof(void *); break; default: return -EINVAL; } } else #endif /* CONFIG_CBPRINTF_PACKAGE_SUPPORT_TAGGED_ARGUMENTS */ { /* Scan the format string */ if (*++fmt == '\0') { break; } if (!parsing) { if (*fmt == '%') { parsing = true; arg_idx++; align = VA_STACK_ALIGN(int); size = sizeof(int); } continue; } switch (*fmt) { case '%': parsing = false; arg_idx--; continue; case '#': case '-': case '+': case ' ': case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9': case '.': case 'h': case 'l': case 'L': continue; case '*': break; case 'j': align = VA_STACK_ALIGN(intmax_t); size = sizeof(intmax_t); continue; case 'z': align = VA_STACK_ALIGN(size_t); size = sizeof(size_t); continue; case 't': align = VA_STACK_ALIGN(ptrdiff_t); size = sizeof(ptrdiff_t); continue; case 'c': case 'd': case 'i': case 'o': case 'u': case 'x': case 'X': if (fmt[-1] == 'l') { if (fmt[-2] == 'l') { align = VA_STACK_ALIGN(long long); size = sizeof(long long); } else { align = VA_STACK_ALIGN(long); size = sizeof(long); } } parsing = false; break; case 's': is_str_arg = true; __fallthrough; case 'p': case 'n': align = VA_STACK_ALIGN(void *); size = sizeof(void *); parsing = false; break; case 'a': case 'A': case 'e': case 'E': case 'f': case 'F': case 'g': case 'G': { /* * Handle floats separately as they may be * held in a different register set. */ union { double d; long double ld; } v; if (fmt[-1] == 'L') { v.ld = va_arg(ap, long double); align = VA_STACK_ALIGN(long double); size = sizeof(long double); } else { v.d = va_arg(ap, double); align = VA_STACK_ALIGN(double); size = sizeof(double); } /* align destination buffer location */ buf = ROUND_UP(buf, align); if (buf0 != NULL) { /* make sure it fits */ if (BUF_OFFSET + size > len) { return -ENOSPC; } if (Z_CBPRINTF_VA_STACK_LL_DBL_MEMCPY) { memcpy((void *)buf, (uint8_t *)&v, size); } else if (fmt[-1] == 'L') { *(long double *)buf = v.ld; } else { *(double *)buf = v.d; } } buf += size; parsing = false; continue; } default: parsing = false; continue; } } /* align destination buffer location */ buf = ROUND_UP(buf, align); /* make sure the data fits */ if ((buf0 != NULL) && (BUF_OFFSET + size) > len) { return -ENOSPC; } /* copy va_list data over to our buffer */ if (is_str_arg) { s = va_arg(ap, char *); process_string: if (buf0 != NULL) { *(const char **)buf = s; } bool is_ro = (fros_cnt-- > 0) ? true : ptr_in_rodata(s); bool do_ro = !!(flags & CBPRINTF_PACKAGE_ADD_RO_STR_POS); if (is_ro && !do_ro) { /* nothing to do */ } else { uint32_t s_ptr_idx = BUF_OFFSET / sizeof(int); /* * In the do_ro case we must consider * room for possible STR_POS_RO_FLAG. * Otherwise the index range is 8 bits * and any overflow is caught later. */ if (do_ro && s_ptr_idx > STR_POS_MASK) { __ASSERT(false, "String with too many arguments"); return -EINVAL; } if (s_idx >= ARRAY_SIZE(str_ptr_pos)) { __ASSERT(false, "str_ptr_pos[] too small"); return -EINVAL; } if (buf0 != NULL) { /* * Remember string pointer location. * We will append non-ro strings later. */ str_ptr_pos[s_idx] = s_ptr_idx; str_ptr_arg[s_idx] = arg_idx; if (is_ro) { /* flag read-only string. */ str_ptr_pos[s_idx] |= STR_POS_RO_FLAG; s_ro_cnt++; } else { s_rw_cnt++; } } else if (is_ro) { /* * Add only pointer position prefix * when counting strings. */ len += 1; } else if (rws_pos_en) { /* * Add only pointer position prefix and * argument index when counting strings. */ len += 2; } else { /* * Add the string length, the final '\0' * and size of the pointer position prefix. */ len += strlen(s) + 1 + 1; } s_idx++; } buf += sizeof(char *); is_str_arg = false; } else if (size == sizeof(int)) { int v = va_arg(ap, int); if (buf0 != NULL) { *(int *)buf = v; } buf += sizeof(int); } else if (size == sizeof(long)) { long v = va_arg(ap, long); if (buf0 != NULL) { *(long *)buf = v; } buf += sizeof(long); } else if (size == sizeof(long long)) { long long v = va_arg(ap, long long); if (buf0 != NULL) { if (Z_CBPRINTF_VA_STACK_LL_DBL_MEMCPY) { memcpy((void *)buf, (uint8_t *)&v, sizeof(long long)); } else { *(long long *)buf = v; } } buf += sizeof(long long); } else { __ASSERT(false, "unexpected size %u", size); return -EINVAL; } } /* * We remember the size of the argument list as a multiple of * sizeof(int) and limit it to a 8-bit field. That means 1020 bytes * worth of va_list, or about 127 arguments on a 64-bit system * (twice that on 32-bit systems). That ought to be good enough. */ if ((BUF_OFFSET / sizeof(int)) > 255) { __ASSERT(false, "too many format args"); return -EINVAL; } /* * If all we wanted was to count required buffer size * then we have it now. */ if (buf0 == NULL) { return BUF_OFFSET + len - CBPRINTF_PACKAGE_ALIGNMENT; } /* Clear our buffer header. We made room for it initially. */ *(char **)buf0 = NULL; /* Record end of argument list. */ pkg_hdr->desc.len = BUF_OFFSET / sizeof(int); if (rws_pos_en) { /* Strings are appended, update location counter. */ pkg_hdr->desc.str_cnt = 0; pkg_hdr->desc.rw_str_cnt = s_rw_cnt; } else { /* Strings are appended, update append counter. */ pkg_hdr->desc.str_cnt = s_rw_cnt; pkg_hdr->desc.rw_str_cnt = 0; } pkg_hdr->desc.ro_str_cnt = s_ro_cnt; #ifdef CONFIG_CBPRINTF_PACKAGE_HEADER_STORE_CREATION_FLAGS pkg_hdr->desc.pkg_flags = flags; #endif /* Store strings pointer locations of read only strings. */ if (s_ro_cnt != 0U) { for (i = 0; i < s_idx; i++) { if (!(str_ptr_pos[i] & STR_POS_RO_FLAG)) { continue; } uint8_t pos = str_ptr_pos[i] & STR_POS_MASK; /* make sure it fits */ if ((BUF_OFFSET + 1) > len) { return -ENOSPC; } /* store the pointer position prefix */ *(uint8_t *)buf = pos; ++buf; } } /* Store strings prefixed by their pointer location. */ for (i = 0; i < s_idx; i++) { /* Process only RW strings. */ if (s_ro_cnt && str_ptr_pos[i] & STR_POS_RO_FLAG) { continue; } if (rws_pos_en) { size = 0; *(uint8_t *)buf = str_ptr_arg[i]; ++buf; } else { /* retrieve the string pointer */ s = *(char **)(buf0 + str_ptr_pos[i] * sizeof(int)); /* clear the in-buffer pointer (less entropy if compressed) */ *(char **)(buf0 + str_ptr_pos[i] * sizeof(int)) = NULL; /* find the string length including terminating '\0' */ size = strlen(s) + 1; } /* make sure it fits */ if ((BUF_OFFSET + 1 + size) > len) { return -ENOSPC; } /* store the pointer position prefix */ *(uint8_t *)buf = str_ptr_pos[i]; ++buf; /* copy the string with its terminating '\0' */ memcpy((void *)buf, (uint8_t *)s, size); buf += size; } /* * TODO: remove pointers for appended strings since they're useless. * TODO: explore leveraging same mechanism to remove alignment padding */ return BUF_OFFSET; #undef BUF_OFFSET #undef STR_POS_RO_FLAG #undef STR_POS_MASK } int cbprintf_package(void *packaged, size_t len, uint32_t flags, const char *format, ...) { va_list ap; int ret; va_start(ap, format); ret = cbvprintf_package(packaged, len, flags, format, ap); va_end(ap); return ret; } int cbpprintf_external(cbprintf_cb out, cbvprintf_external_formatter_func formatter, void *ctx, void *packaged) { uint8_t *buf = packaged; struct cbprintf_package_hdr_ext *hdr = packaged; char *s, **ps; unsigned int i, args_size, s_nbr, ros_nbr, rws_nbr, s_idx; if (buf == NULL) { return -EINVAL; } /* Retrieve the size of the arg list and number of strings. */ args_size = hdr->hdr.desc.len * sizeof(int); s_nbr = hdr->hdr.desc.str_cnt; ros_nbr = hdr->hdr.desc.ro_str_cnt; rws_nbr = hdr->hdr.desc.rw_str_cnt; /* Locate the string table */ s = (char *)(buf + args_size + ros_nbr + 2 * rws_nbr); /* * Patch in string pointers. */ for (i = 0; i < s_nbr; i++) { /* Locate pointer location for this string */ s_idx = *(uint8_t *)s; ++s; ps = (char **)(buf + s_idx * sizeof(int)); /* update the pointer with current string location */ *ps = s; /* move to next string */ s += strlen(s) + 1; } /* Skip past the header */ buf += sizeof(*hdr); /* Turn this into a va_list and print it */ return cbprintf_via_va_list(out, formatter, ctx, hdr->fmt, buf); } /* Function checks if character might be format specifier. Check is relaxed since * compiler ensures that correct format specifier is used so it is enough to check * that character is not one of potential modifier (e.g. number, dot, etc.). */ static bool is_fmt_spec(char c) { return (c >= 64) && (c <= 122); } /* Function checks if nth argument is a pointer (%p). Returns true is yes. Returns * false if not or if string does not have nth argument. */ bool is_ptr(const char *fmt, int n) { char c; bool mod = false; int cnt = 0; while ((c = *fmt++) != '\0') { if (mod) { if (cnt == n) { if (c == 'p') { return true; } else if (is_fmt_spec(c)) { return false; } } else if (is_fmt_spec(c)) { cnt++; mod = false; } } if (c == '%') { mod = !mod; } } return false; } int cbprintf_package_convert(void *in_packaged, size_t in_len, cbprintf_convert_cb cb, void *ctx, uint32_t flags, uint16_t *strl, size_t strl_len) { __ASSERT_NO_MSG(in_packaged != NULL); uint8_t *buf = in_packaged; uint32_t *buf32 = in_packaged; unsigned int args_size, ros_nbr, rws_nbr; bool fmt_present = flags & CBPRINTF_PACKAGE_CONVERT_PTR_CHECK ? true : false; bool rw_cpy; bool ro_cpy; struct cbprintf_package_desc *in_desc = in_packaged; in_len = in_len != 0 ? in_len : get_package_len(in_packaged); /* Get number of RO string indexes in the package and check if copying * includes appending those strings. */ ros_nbr = in_desc->ro_str_cnt; ro_cpy = ros_nbr && (flags & CBPRINTF_PACKAGE_CONVERT_RO_STR) == CBPRINTF_PACKAGE_CONVERT_RO_STR; /* Get number of RW string indexes in the package and check if copying * includes appending those strings. */ rws_nbr = in_desc->rw_str_cnt; rw_cpy = rws_nbr > 0 && (flags & CBPRINTF_PACKAGE_CONVERT_RW_STR) == CBPRINTF_PACKAGE_CONVERT_RW_STR; /* If flags are not set or appending request without rw string indexes * present is chosen, just do a simple copy (or length calculation). * Assuming that it is the most common case. */ if (!rw_cpy && !ro_cpy) { if (cb) { cb(in_packaged, in_len, ctx); } return in_len; } /* If we got here, it means that coping will be more complex and will be * done with strings appending. * Retrieve the size of the arg list. */ args_size = in_desc->len * sizeof(int); int out_len; /* Pointer to array with string locations. Array starts with read-only * string locations. */ const char *fmt = *(const char **)(buf + sizeof(void *)); uint8_t *str_pos = &buf[args_size]; size_t strl_cnt = 0; /* If null destination, just calculate output length. */ if (cb == NULL) { out_len = (int)in_len; if (ro_cpy) { for (unsigned int i = 0; i < ros_nbr; i++) { const char *str = *(const char **)&buf32[*str_pos]; int len = append_string(cb, NULL, str, 0); /* If possible store calculated string length. */ if (strl && strl_cnt < strl_len) { strl[strl_cnt++] = (uint16_t)len; } out_len += len; str_pos++; } } else { str_pos += ros_nbr; } bool drop_ro_str_pos = !(flags & (CBPRINTF_PACKAGE_CONVERT_KEEP_RO_STR | CBPRINTF_PACKAGE_CONVERT_RO_STR)); /* Handle RW strings. */ for (unsigned int i = 0; i < rws_nbr; i++) { uint8_t arg_idx = *str_pos++; uint8_t arg_pos = *str_pos++; const char *str = *(const char **)&buf32[arg_pos]; bool is_ro = ptr_in_rodata(str); int len; if (IS_ENABLED(CONFIG_CBPRINTF_CONVERT_CHECK_PTR) && fmt_present && is_ptr(fmt, arg_idx)) { LOG_WRN("(unsigned) char * used for %%p argument. " "It's recommended to cast it to void * because " "it may cause misbehavior in certain " "configurations. String:\"%s\" argument:%d", fmt, arg_idx); /* Since location is being dropped, decrement * output length by 2 (argument index + position) */ out_len -= 2; continue; } if (is_ro) { if (flags & CBPRINTF_PACKAGE_CONVERT_RO_STR) { goto calculate_string_length; } else { out_len -= drop_ro_str_pos ? 2 : 1; } } else if (flags & CBPRINTF_PACKAGE_CONVERT_RW_STR) { calculate_string_length: len = append_string(cb, NULL, str, 0); /* If possible store calculated string length. */ if (strl && strl_cnt < strl_len) { strl[strl_cnt++] = (uint16_t)len; } /* string length decremented by 1 because argument * index is dropped. */ out_len += (len - 1); } } return out_len; } struct cbprintf_package_desc out_desc; /* At least one is copied in. */ uint8_t cpy_str_pos[16]; /* Up to one will be kept since if both types are kept it returns earlier. */ uint8_t keep_str_pos[16]; uint8_t scpy_cnt; uint8_t keep_cnt; uint8_t *dst; int rv; /* If read-only strings shall be appended to the output package copy * their indexes to the local array, otherwise indicate that indexes * shall remain in the output package. */ if (ro_cpy) { scpy_cnt = ros_nbr; keep_cnt = 0; dst = cpy_str_pos; } else if (ros_nbr && flags & CBPRINTF_PACKAGE_CONVERT_KEEP_RO_STR) { scpy_cnt = 0; keep_cnt = ros_nbr; dst = keep_str_pos; } else { scpy_cnt = 0; keep_cnt = 0; dst = NULL; } if (dst) { memcpy(dst, str_pos, ros_nbr); } str_pos += ros_nbr; /* Go through read-write strings and identify which shall be appended. * Note that there may be read-only strings there. Use address evaluation * to determine if strings is read-only. */ for (unsigned int i = 0; i < rws_nbr; i++) { uint8_t arg_idx = *str_pos++; uint8_t arg_pos = *str_pos++; const char *str = *(const char **)&buf32[arg_pos]; bool is_ro = ptr_in_rodata(str); if (IS_ENABLED(CONFIG_CBPRINTF_CONVERT_CHECK_PTR) && fmt_present && is_ptr(fmt, arg_idx)) { continue; } if (is_ro) { if (flags & CBPRINTF_PACKAGE_CONVERT_RO_STR) { __ASSERT_NO_MSG(scpy_cnt < sizeof(cpy_str_pos)); cpy_str_pos[scpy_cnt++] = arg_pos; } else if (flags & CBPRINTF_PACKAGE_CONVERT_KEEP_RO_STR) { __ASSERT_NO_MSG(keep_cnt < sizeof(keep_str_pos)); keep_str_pos[keep_cnt++] = arg_pos; } else { /* Drop information about ro_str location. */ } } else { if (flags & CBPRINTF_PACKAGE_CONVERT_RW_STR) { __ASSERT_NO_MSG(scpy_cnt < sizeof(cpy_str_pos)); cpy_str_pos[scpy_cnt++] = arg_pos; } else { __ASSERT_NO_MSG(keep_cnt < sizeof(keep_str_pos)); keep_str_pos[keep_cnt++] = arg_idx; keep_str_pos[keep_cnt++] = arg_pos; } } } /* Set amount of strings appended to the package. */ out_desc.len = in_desc->len; out_desc.str_cnt = in_desc->str_cnt + scpy_cnt; out_desc.rw_str_cnt = (flags & CBPRINTF_PACKAGE_CONVERT_RW_STR) ? 0 : (keep_cnt / 2); out_desc.ro_str_cnt = (flags & CBPRINTF_PACKAGE_CONVERT_RO_STR) ? 0 : ((flags & CBPRINTF_PACKAGE_CONVERT_KEEP_RO_STR) ? keep_cnt : 0); /* Temporary overwrite input descriptor to allow bulk transfer */ struct cbprintf_package_desc in_desc_backup = *in_desc; *in_desc = out_desc; /* Copy package header and arguments. */ rv = cb(in_packaged, args_size, ctx); if (rv < 0) { return rv; } out_len = rv; /* Restore input descriptor. */ *in_desc = in_desc_backup; /* Copy string positions which are kept. */ rv = cb(keep_str_pos, keep_cnt, ctx); if (rv < 0) { return rv; } out_len += rv; /* Copy appended strings from source package to destination. */ size_t strs_len = in_len - (args_size + ros_nbr + 2 * rws_nbr); rv = cb(str_pos, strs_len, ctx); if (rv < 0) { return rv; } out_len += rv; /* Append strings */ for (unsigned int i = 0; i < scpy_cnt; i++) { uint8_t loc = cpy_str_pos[i]; const char *str = *(const char **)&buf32[loc]; uint16_t str_len = (strl && (i < strl_len)) ? strl[i] : 0; rv = cb(&loc, 1, ctx); if (rv < 0) { return rv; } out_len += rv; rv = append_string(cb, ctx, str, str_len); if (rv < 0) { return rv; } out_len += rv; } /* Empty call (can be interpreted as flushing) */ (void)cb(NULL, 0, ctx); return out_len; }