1 /* Test strlen functions.
2 Copyright (C) 1999-2022 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
5 The GNU C Library is free software; you can redistribute it and/or
6 modify it under the terms of the GNU Lesser General Public
7 License as published by the Free Software Foundation; either
8 version 2.1 of the License, or (at your option) any later version.
10 The GNU C Library is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 Lesser General Public License for more details.
15 You should have received a copy of the GNU Lesser General Public
16 License along with the GNU C Library; if not, see
17 <https://www.gnu.org/licenses/>. */
21 # define TEST_NAME "strnlen"
23 # define TEST_NAME "wcsnlen"
25 #include "test-string.h"
28 # define STRNLEN strnlen
29 # define MEMSET memset
31 # define BIG_CHAR CHAR_MAX
32 # define MIDDLE_CHAR 127
33 # define SIMPLE_STRNLEN simple_strnlen
36 # define STRNLEN wcsnlen
37 # define MEMSET wmemset
39 # define BIG_CHAR WCHAR_MAX
40 # define MIDDLE_CHAR 1121
41 # define SIMPLE_STRNLEN simple_wcsnlen
44 typedef size_t (*proto_t
) (const CHAR
*, size_t);
48 /* Naive implementation to verify results. */
50 SIMPLE_STRNLEN (const CHAR
*s
, size_t maxlen
)
54 for (i
= 0; i
< maxlen
&& s
[i
]; ++i
);
59 do_one_test (impl_t
*impl
, const CHAR
*s
, size_t maxlen
, size_t exp_len
)
61 size_t len
= CALL (impl
, s
, maxlen
);
64 error (0, 0, "Wrong result in function %s %zd %zd", impl
->name
,
72 do_test (size_t align
, size_t len
, size_t maxlen
, int max_char
)
77 if ((align
+ len
) * sizeof (CHAR
) >= page_size
)
80 CHAR
*buf
= (CHAR
*) (buf1
);
82 for (i
= 0; i
< len
; ++i
)
83 buf
[align
+ i
] = 1 + 11111 * i
% max_char
;
86 FOR_EACH_IMPL (impl
, 0)
87 do_one_test (impl
, (CHAR
*) (buf
+ align
), maxlen
, MIN (len
, maxlen
));
91 do_overflow_tests (void)
95 uintptr_t buf_addr
= (uintptr_t) buf1
;
97 for (i
= 0; i
< 750; ++i
)
99 do_test (1, i
, SIZE_MAX
, BIG_CHAR
);
101 do_test (0, i
, SIZE_MAX
- i
, BIG_CHAR
);
102 do_test (0, i
, i
- buf_addr
, BIG_CHAR
);
103 do_test (0, i
, -buf_addr
- i
, BIG_CHAR
);
104 do_test (0, i
, SIZE_MAX
- buf_addr
- i
, BIG_CHAR
);
105 do_test (0, i
, SIZE_MAX
- buf_addr
+ i
, BIG_CHAR
);
108 for (j
= 8 * sizeof(size_t) - 1; j
; --j
)
111 do_test (0, i
, len
- i
, BIG_CHAR
);
112 do_test (0, i
, len
+ i
, BIG_CHAR
);
113 do_test (0, i
, len
- buf_addr
- i
, BIG_CHAR
);
114 do_test (0, i
, len
- buf_addr
+ i
, BIG_CHAR
);
116 do_test (0, i
, ~len
- i
, BIG_CHAR
);
117 do_test (0, i
, ~len
+ i
, BIG_CHAR
);
118 do_test (0, i
, ~len
- buf_addr
- i
, BIG_CHAR
);
119 do_test (0, i
, ~len
- buf_addr
+ i
, BIG_CHAR
);
121 do_test (0, i
, -buf_addr
, BIG_CHAR
);
122 do_test (0, i
, j
- buf_addr
, BIG_CHAR
);
123 do_test (0, i
, -buf_addr
- j
, BIG_CHAR
);
129 do_random_tests (void)
131 size_t i
, j
, n
, align
, len
;
132 CHAR
*p
= (CHAR
*) (buf1
+ page_size
- 512 * sizeof (CHAR
));
134 for (n
= 0; n
< ITERATIONS
; n
++)
136 align
= random () & 15;
137 len
= random () & 511;
138 if (len
+ align
> 510)
139 len
= 511 - align
- (random () & 7);
140 j
= len
+ align
+ 64;
144 for (i
= 0; i
< j
; i
++)
146 if (i
== len
+ align
)
150 p
[i
] = random () & 255;
151 if (i
>= align
&& i
< len
+ align
&& !p
[i
])
152 p
[i
] = (random () & 127) + 1;
156 FOR_EACH_IMPL (impl
, 1)
159 && CALL (impl
, (CHAR
*) (p
+ align
), len
- 1) != len
- 1)
161 error (0, 0, "Iteration %zd (limited) - wrong result in function %s (%zd) %zd != %zd, p %p",
162 n
, impl
->name
, align
,
163 CALL (impl
, (CHAR
*) (p
+ align
), len
- 1), len
- 1, p
);
166 if (CALL (impl
, (CHAR
*) (p
+ align
), len
) != len
)
168 error (0, 0, "Iteration %zd (exact) - wrong result in function %s (%zd) %zd != %zd, p %p",
169 n
, impl
->name
, align
,
170 CALL (impl
, (CHAR
*) (p
+ align
), len
), len
, p
);
173 if (CALL (impl
, (CHAR
*) (p
+ align
), len
+ 1) != len
)
175 error (0, 0, "Iteration %zd (long) - wrong result in function %s (%zd) %zd != %zd, p %p",
176 n
, impl
->name
, align
,
177 CALL (impl
, (CHAR
*) (p
+ align
), len
+ 1), len
, p
);
184 /* Tests meant to unveil fail on implementation that does not access bytes
185 around the page boundary accordingly. */
189 size_t i
, exp_len
, start_offset
, offset
;
190 /* Calculate the null character offset. */
191 size_t last_offset
= (page_size
/ sizeof (CHAR
)) - 1;
193 CHAR
*s
= (CHAR
*) buf2
;
194 MEMSET (s
, 65, (last_offset
- 1));
197 /* Place short strings ending at page boundary. */
198 offset
= last_offset
;
199 for (i
= 0; i
< 128; i
++)
201 /* Decrease offset to stress several sizes and alignments. */
203 exp_len
= last_offset
- offset
;
204 FOR_EACH_IMPL (impl
, 0)
206 /* Varies maxlen value to cover the cases where it is:
207 - larger than length;
208 - slightly greater than length;
210 - slightly less than length. */
211 do_one_test (impl
, (CHAR
*) (s
+ offset
), page_size
, exp_len
);
212 do_one_test (impl
, (CHAR
*) (s
+ offset
), exp_len
+ 1, exp_len
);
213 do_one_test (impl
, (CHAR
*) (s
+ offset
), exp_len
, exp_len
);
215 do_one_test (impl
, (CHAR
*) (s
+ offset
), exp_len
- 1, exp_len
- 1);
219 /* Place long strings ending at page boundary. */
220 start_offset
= (last_offset
+ 1) / 2;
221 for (i
= 0; i
< 64; ++i
)
223 /* Increase offset to stress several alignments. */
224 offset
= start_offset
+ i
;
225 if (offset
>= (last_offset
+ 1))
227 exp_len
= last_offset
- offset
;
228 FOR_EACH_IMPL (impl
, 0)
230 /* Checks only for maxlen much larger than length because smaller
231 values are already covered in do_random_tests function. */
232 do_one_test (impl
, (CHAR
*) (s
+ offset
), page_size
, exp_len
);
237 /* Tests meant to unveil fail on implementations that access bytes
238 beyond the maxium length. */
241 do_page_2_tests (void)
243 size_t i
, exp_len
, offset
;
244 size_t last_offset
= page_size
/ sizeof (CHAR
);
246 CHAR
*s
= (CHAR
*) buf2
;
247 MEMSET (s
, 65, last_offset
);
249 /* Place short strings ending at page boundary without the null
251 offset
= last_offset
;
252 for (i
= 0; i
< 128; i
++)
254 /* Decrease offset to stress several sizes and alignments. */
256 exp_len
= last_offset
- offset
;
257 FOR_EACH_IMPL (impl
, 0)
259 /* If an implementation goes beyond EXP_LEN, it will trigger
261 do_one_test (impl
, (CHAR
*) (s
+ offset
), exp_len
, exp_len
);
269 size_t i
, length
, char_per_page
;
274 FOR_EACH_IMPL (impl
, 0)
275 printf ("\t%s", impl
->name
);
278 for (i
= 1; i
< 8; ++i
)
280 do_test (0, i
, i
- 1, MIDDLE_CHAR
);
281 do_test (0, i
, i
, MIDDLE_CHAR
);
282 do_test (0, i
, i
+ 1, MIDDLE_CHAR
);
285 for (i
= 1; i
< 8; ++i
)
287 do_test (i
, i
, i
- 1, MIDDLE_CHAR
);
288 do_test (i
, i
, i
, MIDDLE_CHAR
);
289 do_test (i
, i
, i
+ 1, MIDDLE_CHAR
);
292 for (i
= 2; i
<= 10; ++i
)
294 do_test (0, 1 << i
, 5000, MIDDLE_CHAR
);
295 do_test (1, 1 << i
, 5000, MIDDLE_CHAR
);
298 for (i
= 1; i
< 8; ++i
)
299 do_test (0, i
, 5000, BIG_CHAR
);
301 for (i
= 1; i
< 8; ++i
)
302 do_test (i
, i
, 5000, BIG_CHAR
);
304 for (i
= 2; i
<= 10; ++i
)
306 do_test (0, 1 << i
, 5000, BIG_CHAR
);
307 do_test (1, 1 << i
, 5000, BIG_CHAR
);
310 char_per_page
= getpagesize () / sizeof (CHAR
);
312 for (i
= 0; i
<= 127; i
++)
313 for (length
= i
; length
<= 512; length
++)
315 do_test (i
, length
, 512, BIG_CHAR
);
316 do_test (char_per_page
- i
, length
, 512, BIG_CHAR
);
322 do_overflow_tests ();
326 #include <support/test-driver.c>