summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorjvoisin2026-04-30 17:42:29 +0200
committerjvoisin2026-04-30 17:42:29 +0200
commitf9239e2c0f0be9856322727887a45333683940a6 (patch)
tree714b611965666c4072fef6218e7a794dff1884cb
parent6040b4a27409968c764353a98c45d972cfd89a8a (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.h13
-rw-r--r--tests/Makefile2
-rw-r--r--tests/test_wcsnrtombs_dynamic.c28
-rw-r--r--tests/test_wcsnrtombs_static.c26
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
6int 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
6int 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}