summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--include/wchar.h14
-rw-r--r--tests/Makefile2
-rw-r--r--tests/test_mbsnrtowcs_dynamic.c28
-rw-r--r--tests/test_mbsnrtowcs_static.c26
4 files changed, 60 insertions, 10 deletions
diff --git a/include/wchar.h b/include/wchar.h
index 9e32720..2036245 100644
--- a/include/wchar.h
+++ b/include/wchar.h
@@ -75,16 +75,10 @@ _FORTIFY_FN(mbsnrtowcs) size_t mbsnrtowcs(wchar_t * _FORTIFY_POS0 __d,
75 size_t __b = __bos(__d, 0); 75 size_t __b = __bos(__d, 0);
76 size_t __r; 76 size_t __r;
77 77
78 if (__wn > __n / sizeof(wchar_t)) { 78 __b /= sizeof(wchar_t);
79 __b /= sizeof(wchar_t); 79 __r = __orig_mbsnrtowcs(__d, __s, __n, __wn > __b ? __b : __wn, __st);
80 __r = __orig_mbsnrtowcs(__d, __s, __n, __wn > __b ? __b : __wn, __st); 80 if (__b < __wn && __d && *__s && __r != (size_t)-1)
81 if (__b < __wn && __d && *__s && __r != (size_t)-1) 81 __builtin_trap();
82 __builtin_trap();
83 } else {
84 __r = __orig_mbsnrtowcs(__d, __s, __n > __b ? __b : __n, __wn, __st);
85 if (__b < __n && __d && *__s && __r != (size_t)-1)
86 __builtin_trap();
87 }
88 return __r; 82 return __r;
89} 83}
90#endif 84#endif
diff --git a/tests/Makefile b/tests/Makefile
index 9fc8287..6904b2d 100644
--- a/tests/Makefile
+++ b/tests/Makefile
@@ -50,6 +50,8 @@ RUNTIME_TARGETS= \
50 test_mempcpy_static_write \ 50 test_mempcpy_static_write \
51 test_memset_dynamic_write \ 51 test_memset_dynamic_write \
52 test_memset_static_write \ 52 test_memset_static_write \
53 test_mbsnrtowcs_dynamic \
54 test_mbsnrtowcs_static \
53 test_poll_dynamic \ 55 test_poll_dynamic \
54 test_poll_static \ 56 test_poll_static \
55 test_ppoll_dynamic \ 57 test_ppoll_dynamic \
diff --git a/tests/test_mbsnrtowcs_dynamic.c b/tests/test_mbsnrtowcs_dynamic.c
new file mode 100644
index 0000000..77b9082
--- /dev/null
+++ b/tests/test_mbsnrtowcs_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 wchar_t buffer[4] = {0};
8 const char *src = "ABCDEFGHIJ";
9 const char *srcp = src;
10 mbstate_t st;
11 memset(&st, 0, sizeof(st));
12
13 /* Safe: convert up to 2 source bytes into at most 2 wide chars */
14 srcp = src;
15 mbsnrtowcs(buffer, &srcp, 2, 2, &st);
16
17 /* Unsafe: ask to write argc (10) wide chars into 4-element buffer.
18 * Before the fix, the else branch clamped source bytes instead of
19 * the output wide-char count, allowing destination overflow. */
20 CHK_FAIL_START
21 srcp = src;
22 memset(&st, 0, sizeof(st));
23 mbsnrtowcs(buffer, &srcp, 10, argc, &st);
24 CHK_FAIL_END
25
26 printf("%ls\n", buffer);
27 return ret;
28}
diff --git a/tests/test_mbsnrtowcs_static.c b/tests/test_mbsnrtowcs_static.c
new file mode 100644
index 0000000..755d453
--- /dev/null
+++ b/tests/test_mbsnrtowcs_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 wchar_t buffer[4] = {0};
8 const char *src = "ABCDEFGHIJKLMNOP";
9 const char *srcp = src;
10 mbstate_t st;
11 memset(&st, 0, sizeof(st));
12
13 /* Safe: convert up to 4 source bytes into at most 2 wide chars */
14 srcp = src;
15 mbsnrtowcs(buffer, &srcp, 4, 2, &st);
16
17 /* Unsafe: ask to write 16 wide chars into 4-element buffer */
18 CHK_FAIL_START
19 srcp = src;
20 memset(&st, 0, sizeof(st));
21 mbsnrtowcs(buffer, &srcp, 16, 16, &st);
22 CHK_FAIL_END
23
24 printf("%ls\n", buffer);
25 return ret;
26}