diff options
| author | jvoisin | 2026-04-30 17:42:29 +0200 |
|---|---|---|
| committer | jvoisin | 2026-04-30 17:42:29 +0200 |
| commit | f9239e2c0f0be9856322727887a45333683940a6 (patch) | |
| tree | 714b611965666c4072fef6218e7a794dff1884cb | |
| parent | 6040b4a27409968c764353a98c45d972cfd89a8a (diff) | |
Fix a bug in wcsnrtombs
__d is a char * destination buffer, so __b is already the byte capacity.
Dividing by sizeof(wchar_t) makes no sense here, it was likely copy-pasted
from mbsnrtowcs (where the destination is wchar_t *). The first branch also
fails to limit __n (the byte write cap) to __b, so overflows are possible when
a wide character produces multi-byte output. The second branch (else) correctly
limits __n to __b.
This commit replaces the broken two-branch logic with the simple correct
pattern matching wcsrtombs, and adds two tests two prove that nothing broke.
| -rw-r--r-- | include/wchar.h | 13 | ||||
| -rw-r--r-- | tests/Makefile | 2 | ||||
| -rw-r--r-- | tests/test_wcsnrtombs_dynamic.c | 28 | ||||
| -rw-r--r-- | tests/test_wcsnrtombs_static.c | 26 |
4 files changed, 59 insertions, 10 deletions
diff --git a/include/wchar.h b/include/wchar.h index a840f1a..0842115 100644 --- a/include/wchar.h +++ b/include/wchar.h | |||
| @@ -190,16 +190,9 @@ _FORTIFY_FN(wcsnrtombs) size_t wcsnrtombs(char * _FORTIFY_POS0 __d, | |||
| 190 | size_t __b = __bos(__d, 0); | 190 | size_t __b = __bos(__d, 0); |
| 191 | size_t __r; | 191 | size_t __r; |
| 192 | 192 | ||
| 193 | if (__wn > __n / sizeof(wchar_t)) { | 193 | __r = __orig_wcsnrtombs(__d, __s, __wn, __n > __b ? __b : __n, __st); |
| 194 | __b /= sizeof(wchar_t); | 194 | if (__b < __n && __d && *__s && __r != (size_t)-1) |
| 195 | __r = __orig_wcsnrtombs(__d, __s, __wn > __b ? __b : __wn, __n, __st); | 195 | __builtin_trap(); |
| 196 | if (__b < __wn && __d && *__s && __r != (size_t)-1) | ||
| 197 | __builtin_trap(); | ||
| 198 | } else { | ||
| 199 | __r = __orig_wcsnrtombs(__d, __s, __wn, __n > __b ? __b : __n, __st); | ||
| 200 | if (__b < __n && __d && *__s && __r != (size_t)-1) | ||
| 201 | __builtin_trap(); | ||
| 202 | } | ||
| 203 | return __r; | 196 | return __r; |
| 204 | } | 197 | } |
| 205 | #endif | 198 | #endif |
diff --git a/tests/Makefile b/tests/Makefile index 71fb930..adea381 100644 --- a/tests/Makefile +++ b/tests/Makefile | |||
| @@ -94,6 +94,8 @@ RUNTIME_TARGETS= \ | |||
| 94 | test_vsnprintf_static \ | 94 | test_vsnprintf_static \ |
| 95 | test_vsprintf \ | 95 | test_vsprintf \ |
| 96 | test_wcrtomb \ | 96 | test_wcrtomb \ |
| 97 | test_wcsnrtombs_dynamic \ | ||
| 98 | test_wcsnrtombs_static \ | ||
| 97 | test_wcscat_static_write \ | 99 | test_wcscat_static_write \ |
| 98 | test_wcscpy_static_write \ | 100 | test_wcscpy_static_write \ |
| 99 | test_wcsncat_static_write \ | 101 | test_wcsncat_static_write \ |
diff --git a/tests/test_wcsnrtombs_dynamic.c b/tests/test_wcsnrtombs_dynamic.c new file mode 100644 index 0000000..808c9c8 --- /dev/null +++ b/tests/test_wcsnrtombs_dynamic.c | |||
| @@ -0,0 +1,28 @@ | |||
| 1 | #include "common.h" | ||
| 2 | |||
| 3 | #include <wchar.h> | ||
| 4 | #include <string.h> | ||
| 5 | |||
| 6 | int main(int argc, char** argv) { | ||
| 7 | char buffer[8] = {0}; | ||
| 8 | const wchar_t src[] = L"ABCD"; | ||
| 9 | const wchar_t *srcp = src; | ||
| 10 | mbstate_t st; | ||
| 11 | memset(&st, 0, sizeof(st)); | ||
| 12 | |||
| 13 | /* Safe: convert up to 4 wide chars, write at most 4 bytes */ | ||
| 14 | srcp = src; | ||
| 15 | wcsnrtombs(buffer, &srcp, 4, 4, &st); | ||
| 16 | |||
| 17 | /* Unsafe: ask to write argc (10) bytes into 8-byte buffer. | ||
| 18 | * Before the fix, the first branch incorrectly divided the byte-sized | ||
| 19 | * buffer capacity by sizeof(wchar_t), making the check too permissive. */ | ||
| 20 | CHK_FAIL_START | ||
| 21 | srcp = src; | ||
| 22 | memset(&st, 0, sizeof(st)); | ||
| 23 | wcsnrtombs(buffer, &srcp, 4, argc, &st); | ||
| 24 | CHK_FAIL_END | ||
| 25 | |||
| 26 | puts(buffer); | ||
| 27 | return ret; | ||
| 28 | } | ||
diff --git a/tests/test_wcsnrtombs_static.c b/tests/test_wcsnrtombs_static.c new file mode 100644 index 0000000..7f2883f --- /dev/null +++ b/tests/test_wcsnrtombs_static.c | |||
| @@ -0,0 +1,26 @@ | |||
| 1 | #include "common.h" | ||
| 2 | |||
| 3 | #include <wchar.h> | ||
| 4 | #include <string.h> | ||
| 5 | |||
| 6 | int main(int argc, char** argv) { | ||
| 7 | char buffer[4] = {0}; | ||
| 8 | const wchar_t src[] = L"ABCDEFGHIJ"; | ||
| 9 | const wchar_t *srcp = src; | ||
| 10 | mbstate_t st; | ||
| 11 | memset(&st, 0, sizeof(st)); | ||
| 12 | |||
| 13 | /* Safe: convert up to 2 wide chars, write at most 2 bytes */ | ||
| 14 | srcp = src; | ||
| 15 | wcsnrtombs(buffer, &srcp, 2, 2, &st); | ||
| 16 | |||
| 17 | /* Unsafe: ask to write 16 bytes into 4-byte buffer */ | ||
| 18 | CHK_FAIL_START | ||
| 19 | srcp = src; | ||
| 20 | memset(&st, 0, sizeof(st)); | ||
| 21 | wcsnrtombs(buffer, &srcp, 10, 16, &st); | ||
| 22 | CHK_FAIL_END | ||
| 23 | |||
| 24 | puts(buffer); | ||
| 25 | return ret; | ||
| 26 | } | ||
