2 * Copyright (c) 2018 Cisco and/or its affiliates.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at:
7 * http://www.apache.org/licenses/LICENSE-2.0
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
15 #include <vlib/vlib.h>
16 #include <vppinfra/string.h>
19 test_memset_s (vlib_main_t * vm, unformat_input_t * input)
25 vlib_cli_output (vm, "Test memset_s...");
27 err = memset_s (dst, ARRAY_LEN (dst), 0xfe, ARRAY_LEN (dst));
32 for (i = 0; i < ARRAY_LEN (dst); i++)
36 err = memset_s (dst, ARRAY_LEN (dst), 0xfa, ARRAY_LEN (dst) + 1);
45 test_clib_memset (vlib_main_t * vm, unformat_input_t * input)
51 vlib_cli_output (vm, "Test clib_memset...");
53 err = clib_memset (dst, 0xfe, ARRAY_LEN (dst));
58 for (i = 0; i < ARRAY_LEN (dst); i++)
66 test_memcpy_s (vlib_main_t * vm, unformat_input_t * input)
68 char src[64], dst[64];
72 vlib_cli_output (vm, "Test memcpy_s...");
74 for (i = 0; i < ARRAY_LEN (src); i++)
78 err = memcpy_s (dst, sizeof (dst), src, sizeof (src));
83 /* This better not fail but check anyhow */
84 for (i = 0; i < ARRAY_LEN (dst); i++)
89 err = memcpy_s (dst + 1, sizeof (dst) - 1, src, sizeof (src));
95 err = memcpy_s (dst, sizeof (dst), dst + 1, sizeof (dst) - 1);
100 /* Zero length copy */
101 err = memcpy_s (0, sizeof (dst), src, 0);
106 /* OK, seems to work */
111 test_clib_memcpy (vlib_main_t * vm, unformat_input_t * input)
113 char src[64], dst[64];
117 vlib_cli_output (vm, "Test clib_memcpy...");
119 for (i = 0; i < ARRAY_LEN (src); i++)
123 err = clib_memcpy (dst, src, sizeof (src));
128 /* This better not fail but check anyhow */
129 for (i = 0; i < ARRAY_LEN (dst); i++)
130 if (src[i] != dst[i])
132 /* verify it against memcpy */
133 memcpy (dst, src, sizeof (src));
135 /* This better not fail but check anyhow */
136 for (i = 0; i < ARRAY_LEN (dst); i++)
137 if (src[i] != dst[i])
140 /* Zero length copy */
141 err = clib_memcpy (0, src, 0);
146 /* OK, seems to work */
151 test_memcmp_s (vlib_main_t * vm, unformat_input_t * input)
153 char src[64], dst[64];
157 vlib_cli_output (vm, "Test memcmp_s...");
159 /* Fill array with different values */
160 err = clib_memset (src, 0x1, ARRAY_LEN (src));
163 err = clib_memset (dst, 0x3, ARRAY_LEN (dst));
167 /* s1 > s2, > 0 is expected in diff */
168 err = memcmp_s (dst, ARRAY_LEN (dst), src, ARRAY_LEN (src), &diff);
174 /* s1 < s2, < 0 is expected in diff */
175 err = memcmp_s (src, ARRAY_LEN (src), dst, ARRAY_LEN (dst), &diff);
181 err = clib_memset (dst, 0x1, ARRAY_LEN (dst));
185 /* s1 == s2, 0 is expected in diff */
186 err = memcmp_s (src, ARRAY_LEN (src), dst, ARRAY_LEN (dst), &diff);
192 /* Try negative tests */
193 err = memcmp_s (0, 0, 0, 0, 0);
197 /* Try s2max > s1max */
198 err = memcmp_s (src, ARRAY_LEN (src) - 1, dst, ARRAY_LEN (dst), &diff);
202 /* OK, seems to work */
207 test_clib_memcmp (vlib_main_t * vm, unformat_input_t * input)
209 char src[64], dst[64];
213 vlib_cli_output (vm, "Test clib_memcmp...");
215 /* Fill array with different values */
216 err = clib_memset (src, 0x1, ARRAY_LEN (src));
219 err = clib_memset (dst, 0x3, ARRAY_LEN (dst));
223 /* s1 > s2, > 0 is expected in diff */
224 if (!(clib_memcmp (dst, src, ARRAY_LEN (src)) > 0))
226 /* verify it against memcmp */
227 if (!(memcmp (dst, src, ARRAY_LEN (src)) > 0))
230 /* s1 < s2, < 0 is expected in diff */
231 if (!(clib_memcmp (src, dst, ARRAY_LEN (dst)) < 0))
233 /* verify it against memcmp */
234 if (!(memcmp (src, dst, ARRAY_LEN (dst)) < 0))
237 err = clib_memset (dst, 0x1, ARRAY_LEN (dst));
241 /* s1 == s2, 0 is expected in diff */
242 if (clib_memcmp (src, dst, ARRAY_LEN (dst)) != 0)
244 /* verify it against memcmp */
245 if (memcmp (src, dst, ARRAY_LEN (dst)) != 0)
248 /* Try negative tests */
250 if (clib_memcmp (s, s, 0) != 0)
252 /* verify it against memcmp */
253 if (memcmp (s, s, 0) != 0)
256 /* OK, seems to work */
261 test_strcmp_s (vlib_main_t * vm, unformat_input_t * input)
263 char s1[] = "Simplicity is the ultimate sophistication";
264 uword s1len = sizeof (s1) - 1; // excluding null
268 vlib_cli_output (vm, "Test strcmp_s...");
270 /* s1 == s2, 0 is expected */
271 err = strcmp_s (s1, s1len, "Simplicity is the ultimate sophistication",
278 /* s1 > s2, > 0 is expected */
279 err = strcmp_s (s1, s1len, "Simplicity is the ultimate", &indicator);
282 if (!(indicator > 0))
285 /* s1 < s2, < 0 is expected */
286 err = strcmp_s (s1, s1len, "Simplicity is the ultimate sophistication!",
290 if (!(indicator < 0))
293 /* Try some negative tests */
295 /* Null pointers test */
296 err = strcmp_s (0, 0, 0, 0);
300 /* non-null terminated s1 */
302 err = strcmp_s (s1, s1len, "Simplicity is the ultimate sophistication",
307 /* OK, seems to work */
312 test_clib_strcmp (vlib_main_t * vm, unformat_input_t * input)
314 char s1[] = "Simplicity is the ultimate sophistication";
318 vlib_cli_output (vm, "Test clib_strcmp...");
320 /* s1 == s2, 0 is expected */
321 indicator = clib_strcmp (s1, "Simplicity is the ultimate sophistication");
324 /* verify it against strcmp */
325 indicator = strcmp (s1, "Simplicity is the ultimate sophistication");
329 /* s1 > s2, > 0 is expected */
330 indicator = clib_strcmp (s1, "Simplicity is the ultimate");
331 if (!(indicator > 0))
333 /* verify it against strcmp */
334 indicator = strcmp (s1, "Simplicity is the ultimate");
335 if (!(indicator > 0))
338 /* s1 < s2, < 0 is expected */
339 indicator = clib_strcmp (s1, "Simplicity is the ultimate sophistication!");
340 if (!(indicator < 0))
342 /* verify it against strcmp */
343 indicator = strcmp (s1, "Simplicity is the ultimate sophistication!");
344 if (!(indicator < 0))
347 /* Try some negative tests */
349 /* Null pointers comparison */
351 indicator = clib_strcmp (s, s);
355 /* OK, seems to work */
360 test_strncmp_s (vlib_main_t * vm, unformat_input_t * input)
362 char s1[] = "Every moment is a fresh beginning";
363 uword s1len = sizeof (s1) - 1; // excluding null
367 vlib_cli_output (vm, "Test strncmp_s...");
369 /* s1 == s2, 0 is expected */
370 err = strncmp_s (s1, s1len, "Every moment is a fresh beginning", s1len,
377 /* s1 > s2, 0 is expected since comparison is no more than n character */
378 err = strncmp_s (s1, s1len, "Every moment is a fresh begin",
379 sizeof ("Every moment is a fresh begin") - 1, &indicator);
385 /* s1 < s2, < 0 is expected */
386 err = strncmp_s (s1, s1len, "Every moment is fresh beginning",
387 sizeof ("Every moment is fresh beginning") - 1,
391 if (!(indicator < 0))
394 /* s1 > s2, > 0 is expected */
395 err = strncmp_s ("Every moment is fresh beginning. ",
396 sizeof ("Every moment is fresh beginning. ") - 1, s1,
400 if (!(indicator > 0))
403 /* Try some negative tests */
406 err = strncmp_s (0, 0, 0, 0, 0);
411 err = strncmp_s (s1, s1len, "Every moment is a fresh beginning", s1len + 1,
416 /* unterminated s1 */
418 err = strncmp_s (s1, s1len, "Every moment is a fresh beginning",
419 sizeof ("Every moment is a fresh beginning") - 1,
424 /* OK, seems to work */
429 test_clib_strncmp (vlib_main_t * vm, unformat_input_t * input)
431 char s1[] = "Every moment is a fresh beginning";
432 uword s1len = sizeof (s1) - 1; // excluding null
433 int indicator, v_indicator;
435 vlib_cli_output (vm, "Test clib_strncmp...");
437 /* s1 == s2, 0 is expected */
438 indicator = clib_strncmp (s1, "Every moment is a fresh beginning", s1len);
441 /* verify it against strncmp */
442 v_indicator = strncmp (s1, "Every moment is a fresh beginning", s1len);
443 if (v_indicator != 0)
445 if (v_indicator != indicator)
448 /* s1 > s2, 0 is expected since comparison is no more than n character */
449 indicator = clib_strncmp (s1, "Every moment is a fresh begin",
450 sizeof ("Every moment is a fresh begin") - 1);
453 /* verify it against strncmp */
454 v_indicator = strncmp (s1, "Every moment is a fresh begin",
455 sizeof ("Every moment is a fresh begin") - 1);
456 if (v_indicator != 0)
458 if (v_indicator != indicator)
461 /* s1 < s2, < 0 is expected */
462 indicator = clib_strncmp (s1, "Every moment is fresh beginning",
463 sizeof ("Every moment is fresh beginning") - 1);
464 if (!(indicator < 0))
466 /* verify it against strncmp */
467 v_indicator = strncmp (s1, "Every moment is fresh beginning",
468 sizeof ("Every moment is fresh beginning") - 1);
469 if (!(v_indicator < 0))
471 if (v_indicator != indicator)
474 /* s1 > s2, > 0 is expected */
475 indicator = clib_strncmp ("Every moment is fresh beginning. ", s1, s1len);
476 if (!(indicator > 0))
478 /* verify it against strncmp */
479 v_indicator = strncmp ("Every moment is fresh beginning. ", s1, s1len);
480 if (!(v_indicator > 0))
482 if (v_indicator != indicator)
485 /* Try some negative tests */
489 /* make sure we don't crash */
490 indicator = clib_strncmp (0, 0, 0);
491 if (indicator != EOK)
496 clib_strncmp (s1, "Every moment is a fresh beginning", s1len + 1);
499 /* verify it against strncmp */
500 v_indicator = strncmp (s1, "Every moment is a fresh beginning", s1len + 1);
501 if (v_indicator != 0)
503 if (v_indicator != indicator)
506 /* unterminated s1 */
508 indicator = clib_strncmp (s1, "Every moment is a fresh beginning",
509 sizeof ("every moment is a fresh beginning") - 1);
512 /* verify it against strncmp */
513 v_indicator = strncmp (s1, "Every moment is a fresh beginning",
514 sizeof ("Every moment is a fresh beginning") - 1);
515 if (v_indicator != 0)
518 /* OK, seems to work */
523 test_strcpy_s (vlib_main_t * vm, unformat_input_t * input)
525 char src[] = "To err is human.";
528 size_t s1size = sizeof (dst); // including null
531 vlib_cli_output (vm, "Test strcpy_s...");
533 err = strcpy_s (dst, s1size, src);
537 /* This better not fail but check anyhow */
538 if (strcmp_s (dst, clib_strnlen (dst, sizeof (dst)), src, &indicator) !=
546 err = strcpy_s (0, 0, 0);
551 err = strcpy_s (dst, 10, src);
557 /* GCC 8 flunks this one at compile time... */
558 err = strcpy_s (dst, s1size, dst);
564 err = strcpy_s (dst, s1size, dst + 1);
568 /* OK, seems to work */
573 test_clib_strcpy (vlib_main_t * vm, unformat_input_t * input)
575 char src[] = "The journey of a one thousand miles begins with one step.";
580 vlib_cli_output (vm, "Test clib_strcpy...");
582 err = clib_strcpy (dst, src);
586 /* This better not fail but check anyhow */
587 if (strcmp_s (dst, clib_strnlen (dst, sizeof (dst)), src, &indicator) !=
593 /* verify it against strcpy */
596 /* This better not fail but check anyhow */
597 if (strcmp_s (dst, clib_strnlen (dst, sizeof (dst)), src, &indicator) !=
605 err = clib_strcpy (0, 0);
611 /* GCC 8 flunks this one at compile time... */
612 err = clib_strcpy (dst, dst);
618 err = clib_strcpy (dst, dst + 1);
622 /* OK, seems to work */
627 test_strncpy_s (vlib_main_t * vm, unformat_input_t * input)
629 char src[] = "Those who dare to fail miserably can achieve greatly.";
630 char dst[100], old_dst[100];
632 size_t s1size = sizeof (dst); // including null
635 vlib_cli_output (vm, "Test strncpy_s...");
637 /* dmax includes null, n excludes null */
639 /* n == string len of src */
640 err = strncpy_s (dst, s1size, src, clib_strnlen (src, sizeof (src)));
643 if (strcmp_s (dst, clib_strnlen (dst, sizeof (dst)), src, &indicator) !=
649 /* limited copy -- strlen src > n, copy up to n */
650 err = strncpy_s (dst, s1size, "The price of greatness is responsibility.",
654 if (strcmp_s (dst, clib_strnlen (dst, sizeof (dst)), "The price ",
660 /* n > string len of src */
661 err = clib_memset (dst, 1, sizeof (dst));
665 err = strncpy_s (dst, s1size, src, clib_strnlen (src, sizeof (src)) + 10);
668 if (strcmp_s (dst, clib_strnlen (dst, sizeof (dst)), src, &indicator) !=
674 /* Make sure bytes after strlen(dst) is untouched */
675 for (i = 1 + clib_strnlen (dst, sizeof (dst)); i < sizeof (dst); i++)
679 /* truncation, n >= dmax */
680 err = strncpy_s (dst, clib_strnlen (src, sizeof (src)), src,
681 clib_strnlen (src, sizeof (src)));
682 if (err != EOVERFLOW)
685 /* Check dst content */
686 if (dst[strlen (dst)] != '\0')
688 if (strncmp_s (dst, clib_strnlen (dst, sizeof (dst)), src,
689 clib_strnlen (dst, sizeof (dst)), &indicator) != EOK)
694 /* zero length copy */
695 clib_strncpy (old_dst, dst, clib_strnlen (dst, sizeof (dst)));
696 err = strncpy_s (dst, sizeof (dst), src, 0);
699 /* verify dst is untouched */
700 if (strcmp_s (dst, clib_strnlen (dst, sizeof (dst)), old_dst, &indicator) !=
708 err = strncpy_s (0, 0, 0, 1);
713 err = strncpy_s (dst, s1size, dst + 1, s1size - 1);
719 /* GCC 8 flunks this one at compile time... */
720 err = strncpy_s (dst, s1size, dst, s1size);
725 /* OK, seems to work */
730 test_clib_strncpy (vlib_main_t * vm, unformat_input_t * input)
732 char src[] = "Those who dare to fail miserably can achieve greatly.";
733 char dst[100], old_dst[100];
735 size_t s1size = sizeof (dst); // including null
738 vlib_cli_output (vm, "Test clib_strncpy...");
740 /* n == string len of src */
741 err = clib_strncpy (dst, src, clib_strnlen (src, sizeof (src)));
745 /* This better not fail but check anyhow */
746 if (strcmp_s (dst, clib_strnlen (dst, sizeof (dst)), src, &indicator) !=
752 /* Verify it against strncpy */
754 /* GCC 8 debian flunks this one at compile time */
755 strncpy (dst, src, strlen (src));
757 /* This better not fail but check anyhow */
758 if (strcmp_s (dst, clib_strnlen (dst, sizeof (dst)), src, &indicator) !=
765 /* limited copy -- strlen src > n, copy up to n */
766 err = clib_strncpy (dst, "The price of greatness is responsibility.", 10);
769 if (strcmp_s (dst, clib_strnlen (dst, sizeof (dst)), "The price ",
774 /* verify it against strncpy */
775 memset_s (dst, sizeof (dst), 0, sizeof (dst));
778 /* GCC 8 flunks this one at compile time... */
779 strncpy (dst, "The price of greatness is responsibility.", 10);
780 if (strcmp_s (dst, clib_strnlen (dst, sizeof (dst)), "The price ",
787 /* n > string len of src */
788 err = clib_strncpy (dst, src, clib_strnlen (src, sizeof (src)) + 10);
791 if (strcmp_s (dst, clib_strnlen (dst, sizeof (dst)), src, &indicator) !=
796 /* Verify it against strncpy */
798 /* GCC 8 debian flunks this one at compile time */
799 strncpy (dst, src, strlen (src));
800 if (strcmp_s (dst, clib_strnlen (dst, sizeof (dst)), src, &indicator) !=
807 /* zero length copy */
808 clib_strncpy (old_dst, dst, clib_strnlen (dst, sizeof (dst)));
809 err = clib_strncpy (dst, src, 0);
812 /* verify dst is untouched */
813 if (strcmp_s (dst, clib_strnlen (dst, sizeof (dst)), old_dst, &indicator) !=
821 err = clib_strncpy (0, 0, 1);
826 err = clib_strncpy (dst, dst + 1, s1size);
832 /* GCC 8 flunks this one at compile time... */
833 err = clib_strncpy (dst, dst, s1size);
838 /* OK, seems to work */
843 test_strcat_s (vlib_main_t * vm, unformat_input_t * input)
845 char src[100], dst[100], old_dst[100];
846 size_t s1size = sizeof (dst); // including null
850 vlib_cli_output (vm, "Test strcat_s...");
852 strcpy_s (dst, sizeof (dst), "Tough time never last ");
853 strcpy_s (src, sizeof (src), "but tough people do");
854 err = strcat_s (dst, s1size, src);
857 if (strcmp_s (dst, s1size - 1,
858 "Tough time never last but tough people do",
864 /* empty string concatenation */
865 clib_strncpy (old_dst, dst, clib_strnlen (dst, sizeof (dst)));
866 err = strcat_s (dst, s1size, "");
869 /* verify dst is untouched */
870 if (strcmp_s (dst, s1size - 1, old_dst, &indicator) != EOK)
876 err = strcat_s (0, 0, 0);
881 err = strcat_s (dst, s1size, dst + 1);
887 /* GCC 8 flunks this one at compile time... */
888 err = strcat_s (dst, s1size, dst);
893 /* not enough space for dst */
894 err = strcat_s (dst, 10, src);
898 /* OK, seems to work */
903 test_clib_strcat (vlib_main_t * vm, unformat_input_t * input)
905 char src[100], dst[100], old_dst[100];
906 size_t s1size = sizeof (dst); // including null
910 vlib_cli_output (vm, "Test clib_strcat...");
912 strcpy_s (dst, sizeof (dst), "Tough time never last ");
913 strcpy_s (src, sizeof (src), "but tough people do");
914 err = clib_strcat (dst, src);
917 if (strcmp_s (dst, s1size - 1,
918 "Tough time never last but tough people do",
923 /* verify it against strcat */
924 strcpy_s (dst, sizeof (dst), "Tough time never last ");
925 strcpy_s (src, sizeof (src), "but tough people do");
927 if (strcmp_s (dst, s1size - 1,
928 "Tough time never last but tough people do",
934 /* empty string concatenation */
935 clib_strncpy (old_dst, dst, clib_strnlen (dst, sizeof (dst)));
936 err = clib_strcat (dst, "");
939 /* verify dst is untouched */
940 if (strcmp_s (dst, s1size - 1, old_dst, &indicator) != EOK)
946 err = clib_strcat (0, 0);
951 err = clib_strcat (dst, dst + 1);
957 /* GCC 8 flunks this one at compile time... */
958 err = clib_strcat (dst, dst);
963 /* OK, seems to work */
968 test_strncat_s (vlib_main_t * vm, unformat_input_t * input)
970 char src[100], dst[100], old_dst[100];
971 size_t s1size = sizeof (dst); // including null
973 char s1[] = "Two things are infinite: ";
974 char s2[] = "the universe and human stupidity; ";
975 char s3[] = "I am not sure about the universe.";
978 vlib_cli_output (vm, "Test strncat_s...");
980 strcpy_s (dst, sizeof (dst), s1);
981 strcpy_s (src, sizeof (src), s2);
982 err = strncat_s (dst, s1size, src, clib_strnlen (src, sizeof (src)));
985 if (strcmp_s (dst, s1size - 1,
986 "Two things are infinite: the universe and human stupidity; ",
992 /* truncation, n >= dmax - strnlen_s (dst, dmax) */
993 err = strncat_s (dst, clib_strnlen (dst, sizeof (dst)) +
994 clib_strnlen (s3, sizeof (s3)), s3,
995 clib_strnlen (s3, sizeof (s3)));
996 if (err != EOVERFLOW)
999 * resulting string is dst + strlen (s3) - 1 characters + null.
1000 * notice the "." is missing at the end of the resulting string because
1001 * the space is needed to accommodate the null
1002 * Notice strcmp_s will check s1 or dst to make sure it is null terminated
1004 if (strcmp_s (dst, s1size - 1,
1005 "Two things are infinite: the universe and human stupidity; "
1006 "I am not sure about the universe", &indicator) != EOK)
1011 /* n > strlen src */
1012 strcpy_s (dst, sizeof (dst), s1);
1013 err = strncat_s (dst, s1size, src, clib_strnlen (src, sizeof (src)) + 10);
1016 if (strcmp_s (dst, s1size - 1,
1017 "Two things are infinite: the universe and human stupidity; ",
1023 /* zero length strncat */
1024 clib_strncpy (old_dst, dst, clib_strnlen (dst, sizeof (dst)));
1025 err = strncat_s (dst, sizeof (dst), src, 0);
1028 /* verify dst is untouched */
1029 if (strcmp_s (dst, s1size - 1, old_dst, &indicator) != EOK)
1034 /* empty string, wrong n concatenation */
1035 err = strncat_s (dst, sizeof (dst), "", 10);
1038 /* verify dst is untouched */
1039 if (strcmp_s (dst, s1size - 1, old_dst, &indicator) != EOK)
1044 /* limited concatenation, string > n, copy up to n */
1045 strcpy_s (dst, sizeof (dst), s1);
1046 err = strncat_s (dst, s1size, s2, 13);
1049 if (strcmp_s (dst, s1size - 1, "Two things are infinite: the universe ",
1054 /* verify it against strncat */
1056 /* GCC 8 debian flunks this one at compile time */
1057 strcpy_s (dst, sizeof (dst), s1);
1058 strncat (dst, s2, 13);
1059 if (strcmp_s (dst, s1size - 1, "Two things are infinite: the universe ",
1066 /* negative stuff */
1067 err = strncat_s (0, 0, 0, 1);
1071 /* no room for dst -- dmax - strnlen_s (dst, dmax) == 0 */
1072 err = strncat_s (dst, clib_strnlen (dst, sizeof (dst)), s2,
1073 clib_strnlen (s2, sizeof (s2)));
1078 err = strncat_s (dst, s1size, dst + 1, clib_strnlen (dst + 1, s1size - 1));
1084 /* GCC 8 flunks this one at compile time... */
1085 err = strncat_s (dst, s1size, dst, clib_strnlen (dst, sizeof (dst)));
1090 /* OK, seems to work */
1095 test_clib_strncat (vlib_main_t * vm, unformat_input_t * input)
1097 char src[100], dst[100], old_dst[100];
1098 size_t s1size = sizeof (dst); // including null
1100 char s1[] = "Two things are infinite: ";
1101 char s2[] = "the universe and human stupidity; ";
1104 vlib_cli_output (vm, "Test clib_strncat...");
1106 /* n == strlen src */
1107 strcpy_s (dst, sizeof (dst), s1);
1108 strcpy_s (src, sizeof (src), s2);
1109 err = clib_strncat (dst, src, clib_strnlen (src, sizeof (src)));
1112 if (strcmp_s (dst, s1size - 1,
1113 "Two things are infinite: the universe and human stupidity; ",
1118 /* verify it against strncat */
1119 strcpy_s (dst, sizeof (dst), s1);
1120 strncat (dst, src, clib_strnlen (src, sizeof (src)));
1121 if (strcmp_s (dst, s1size - 1,
1122 "Two things are infinite: the universe and human stupidity; ",
1128 /* n > strlen src */
1129 strcpy_s (dst, sizeof (dst), s1);
1130 err = clib_strncat (dst, src, clib_strnlen (src, sizeof (src)) + 10);
1133 if (strcmp_s (dst, s1size - 1,
1134 "Two things are infinite: the universe and human stupidity; ",
1139 /* verify it against strncat */
1140 strcpy_s (dst, sizeof (dst), s1);
1141 strncat (dst, src, clib_strnlen (src, sizeof (src)));
1142 if (strcmp_s (dst, s1size - 1,
1143 "Two things are infinite: the universe and human stupidity; ",
1149 /* zero length strncat */
1150 clib_strncpy (old_dst, dst, clib_strnlen (dst, sizeof (dst)));
1151 err = clib_strncat (dst, src, 0);
1154 /* verify dst is untouched */
1155 if (strcmp_s (dst, s1size - 1, old_dst, &indicator) != EOK)
1160 /* empty string, wrong n concatenation */
1161 err = clib_strncat (dst, "", 10);
1164 /* verify dst is untouched */
1165 if (strcmp_s (dst, s1size - 1, old_dst, &indicator) != EOK)
1170 /* limited concatenation, string > n, copy up to n */
1171 strcpy_s (dst, sizeof (dst), s1);
1172 err = clib_strncat (dst, s2, 13);
1175 if (strcmp_s (dst, s1size - 1, "Two things are infinite: the universe ",
1180 /* verify it against strncat */
1182 /* GCC 8 debian flunks this one at compile time */
1183 strcpy_s (dst, sizeof (dst), s1);
1184 strncat (dst, s2, 13);
1185 if (strcmp_s (dst, s1size - 1, "Two things are infinite: the universe ",
1192 /* negative stuff */
1193 err = clib_strncat (0, 0, 1);
1198 err = clib_strncat (dst, dst + 1, s1size - 1);
1204 /* GCC 8 flunks this one at compile time... */
1205 err = clib_strncat (dst, dst, clib_strnlen (dst, sizeof (dst)));
1210 /* OK, seems to work */
1215 test_strtok_s (vlib_main_t * vm, unformat_input_t * input)
1223 char *tok1, *tok2, *tok3, *tok4, *tok5, *tok6, *tok7;
1225 vlib_cli_output (vm, "Test strtok_s...");
1226 strcpy_s (str1, sizeof (str1), "brevity is the soul of wit");
1227 len = strnlen_s (str1, sizeof (str1));
1228 tok1 = strtok_s (str1, &len, " ", &p2str);
1229 tok2 = strtok_s (0, &len, " ", &p2str);
1230 tok3 = strtok_s (0, &len, " ", &p2str);
1231 tok4 = strtok_s (0, &len, " ", &p2str);
1232 tok5 = strtok_s (0, &len, " ", &p2str);
1233 tok6 = strtok_s (0, &len, " ", &p2str);
1234 tok7 = strtok_s (0, &len, " ", &p2str);
1236 strcmp_s (tok1, strlen (tok1), "brevity", &indicator) != EOK)
1240 if ((tok2 == 0) || strcmp_s (tok2, strlen (tok2), "is", &indicator) != EOK)
1244 if ((tok3 == 0) || strcmp_s (tok3, strlen (tok3), "the", &indicator) != EOK)
1249 || strcmp_s (tok4, strlen (tok4), "soul", &indicator) != EOK)
1253 if ((tok5 == 0) || strcmp_s (tok5, strlen (tok5), "of", &indicator) != EOK)
1257 if ((tok6 == 0) || strcmp_s (tok6, strlen (tok6), "wit", &indicator) != EOK)
1264 /* delimiter not present in the string -- the whole string is returned */
1265 strcpy_s (str1, sizeof (str1), "brevity is the soul of wit");
1266 len = strnlen_s (str1, sizeof (str1) - 1);
1268 tok1 = strtok_s (str1, &len, ",", &p2str);
1269 if ((tok1 == 0) || strcmp_s (tok1, strlen (tok1), str1, &indicator) != EOK)
1274 /* negative stuff */
1275 tok = strtok_s (0, 0, 0, 0);
1279 /* s1 and ptr contents are null */
1281 tok = strtok_s (0, 0, 0, &ptr);
1285 /* unterminate s1 */
1287 len = strnlen_s (str1, sizeof (str1) - 1);
1288 str1[strlen (str1)] = 0x2;
1289 tok = strtok_s (str1, &len, ",", &p2str);
1294 * unterminated s2. This test case in not perfect because there is no
1295 * argument for s2max. But s2 len is limited to 16 characters. If the API
1296 * does not find the null character at s2[15], it declares the string s2
1299 memset_s (str2, sizeof (str2), 0xfa, sizeof (str2));
1300 tok = strtok_s (str1, &len, str2, &p2str);
1304 /* OK, seems to work */
1309 test_clib_strtok (vlib_main_t * vm, unformat_input_t * input)
1312 char *s1 __attribute__ ((unused));
1313 char *tok __attribute__ ((unused));
1314 char *ptr __attribute__ ((unused));
1317 char *tok1, *tok2, *tok3, *tok4, *tok5, *tok6, *tok7;
1319 vlib_cli_output (vm, "Test clib_strtok...");
1320 strcpy_s (str1, sizeof (str1), "brevity is the soul of wit");
1322 tok1 = clib_strtok (str1, " ", &p2str);
1323 tok2 = clib_strtok (0, " ", &p2str);
1324 tok3 = clib_strtok (0, " ", &p2str);
1325 tok4 = clib_strtok (0, " ", &p2str);
1326 tok5 = clib_strtok (0, " ", &p2str);
1327 tok6 = clib_strtok (0, " ", &p2str);
1328 tok7 = clib_strtok (0, " ", &p2str);
1330 strcmp_s (tok1, strlen (tok1), "brevity", &indicator) != EOK)
1334 if ((tok2 == 0) || strcmp_s (tok2, strlen (tok2), "is", &indicator) != EOK)
1338 if ((tok3 == 0) || strcmp_s (tok3, strlen (tok3), "the", &indicator) != EOK)
1343 || strcmp_s (tok4, strlen (tok4), "soul", &indicator) != EOK)
1347 if ((tok5 == 0) || strcmp_s (tok5, strlen (tok5), "of", &indicator) != EOK)
1351 if ((tok6 == 0) || strcmp_s (tok6, strlen (tok6), "wit", &indicator) != EOK)
1357 /* verify it againest strtok_r */
1358 strcpy_s (str1, sizeof (str1), "brevity is the soul of wit");
1360 tok1 = strtok_r (str1, " ", &p2str);
1361 tok2 = strtok_r (0, " ", &p2str);
1362 tok3 = strtok_r (0, " ", &p2str);
1363 tok4 = strtok_r (0, " ", &p2str);
1364 tok5 = strtok_r (0, " ", &p2str);
1365 tok6 = strtok_r (0, " ", &p2str);
1366 tok7 = strtok_r (0, " ", &p2str);
1368 strcmp_s (tok1, strlen (tok1), "brevity", &indicator) != EOK)
1372 if ((tok2 == 0) || strcmp_s (tok2, strlen (tok2), "is", &indicator) != EOK)
1376 if ((tok3 == 0) || strcmp_s (tok3, strlen (tok3), "the", &indicator) != EOK)
1381 || strcmp_s (tok4, strlen (tok4), "soul", &indicator) != EOK)
1385 if ((tok5 == 0) || strcmp_s (tok5, strlen (tok5), "of", &indicator) != EOK)
1389 if ((tok6 == 0) || strcmp_s (tok6, strlen (tok6), "wit", &indicator) != EOK)
1396 /* delimiter not present in the string -- the whole string is returned */
1397 strcpy_s (str1, sizeof (str1), "brevity is the soul of wit");
1399 tok1 = clib_strtok (str1, ",", &p2str);
1400 if ((tok1 == 0) || strcmp_s (tok1, strlen (tok1), str1, &indicator) != EOK)
1404 /* verify it against strtok_r */
1405 strcpy_s (str1, sizeof (str1), "brevity is the soul of wit");
1407 tok1 = strtok_r (str1, ",", &p2str);
1408 if ((tok1 == 0) || strcmp_s (tok1, strlen (tok1), str1, &indicator) != EOK)
1413 /* negative stuff */
1417 /* GCC 8 flunks this one at compile time... */
1418 tok = clib_strtok (s1, s1, (char **) 0);
1422 /* s1 and ptr contents are null */
1423 tok = clib_strtok (s1, s1, &ptr);
1428 /* verify it against strtok_r */
1429 /* No can do. This causes a crash in strtok_r */
1430 // tok = strtok_r (s1, " ", &ptr);
1435 * Can't test unterminated string s1 and s2 becuase clib_strtok does not
1436 * supply s1 and s2 max
1439 /* OK, seems to work */
1444 test_strnlen_s (vlib_main_t * vm, unformat_input_t * input)
1446 const char s1[] = "Truth is incontrovertible";
1449 vlib_cli_output (vm, "Test strnlen_s...");
1451 len = strnlen_s (s1, sizeof (s1));
1452 if (len != sizeof (s1) - 1)
1455 len = strnlen_s (s1, 5);
1459 /* negative stuff */
1460 len = strnlen_s (0, 0);
1464 /* OK, seems to work */
1469 test_clib_strnlen (vlib_main_t * vm, unformat_input_t * input)
1471 const char s1[] = "Truth is incontrovertible";
1474 vlib_cli_output (vm, "Test clib_strnlen...");
1476 len = clib_strnlen (s1, sizeof (s1));
1477 if (len != sizeof (s1) - 1)
1480 len = clib_strnlen (s1, 5);
1484 /* negative stuff */
1485 len = clib_strnlen (0, 0);
1489 /* OK, seems to work */
1494 test_strstr_s (vlib_main_t * vm, unformat_input_t * input)
1499 size_t s1len = sizeof (s1) - 1; // excluding null
1502 vlib_cli_output (vm, "Test strstr_s...");
1504 /* substring not present */
1505 strcpy_s (s1, s1len, "success is not final, failure is not fatal.");
1506 err = strstr_s (s1, s1len, "failures", sizeof ("failures"), &sub);;
1510 /* substring present */
1511 err = strstr_s (s1, s1len, "failure", sizeof ("failure"), &sub);
1516 strcmp_s (sub, strlen (sub), "failure is not fatal.", &indicator)
1522 /* negative stuff */
1524 /* Null pointers test */
1525 err = strstr_s (0, 0, 0, 0, 0);
1529 /* unterminated s1 and s2 */
1530 memset_s (s1, ARRAY_LEN (s1), 0xfe, ARRAY_LEN (s1));
1531 err = strstr_s (s1, s1len, s1, s1len, &sub);
1535 /* OK, seems to work */
1540 test_clib_strstr (vlib_main_t * vm, unformat_input_t * input)
1544 size_t s1len = sizeof (s1) - 1; // excluding null
1547 vlib_cli_output (vm, "Test clib_strstr...");
1549 /* substring not present */
1550 strcpy_s (s1, s1len, "success is not final, failure is not fatal.");
1551 sub = clib_strstr (s1, "failures");
1554 /* verify it against strstr */
1555 sub = strstr (s1, "failures");
1559 /* substring present */
1560 sub = clib_strstr (s1, "failure");
1563 if (strcmp_s (sub, strlen (sub), "failure is not fatal.", &indicator) !=
1568 /* verify it against strstr */
1569 sub = strstr (s1, "failure");
1572 if (strcmp_s (sub, strlen (sub), "failure is not fatal.", &indicator) !=
1578 /* negative stuff */
1580 /* Null pointers test */
1582 sub = clib_strstr (s, s);
1586 * Can't verify it against strstr for this test. Null pointers cause strstr
1587 * to crash. Go figure!
1590 /* unterminated s1 and s2 */
1591 memset_s (s1, ARRAY_LEN (s1), 0xfe, ARRAY_LEN (s1));
1592 sub = clib_strstr (s1, s1);
1596 * Can't verify it against strstr for this test. Unterminated string causes
1597 * strstr to crash. Go figure!
1600 /* OK, seems to work */
1604 #define foreach_string_test \
1605 _ (0, MEMCPY_S, "memcpy_s", memcpy_s) \
1606 _ (1, CLIB_MEMCPY, "clib_memcpy", clib_memcpy) \
1607 _ (2, MEMSET_S , "memset_s", memset_s) \
1608 _ (3, CLIB_MEMSET , "clib_memset", clib_memset) \
1609 _ (4, MEMCMP_S, "memcmp_s", memcmp_s) \
1610 _ (5, CLIB_MEMCMP, "clib_memcmp", clib_memcmp) \
1611 _ (6, STRCMP_S, "strcmp_s", strcmp_s) \
1612 _ (7, CLIB_STRCMP, "clib_strcmp", clib_strcmp) \
1613 _ (8, STRNCMP_S, "strncmp_s", strncmp_s) \
1614 _ (9, CLIB_STRNCMP, "clib_strncmp", clib_strncmp) \
1615 _ (10, STRCPY_S, "strcpy_s", strcpy_s) \
1616 _ (11, CLIB_STRCPY, "clib_strcpy", clib_strcpy) \
1617 _ (12, STRNCPY_S, "strncpy_s", strncpy_s) \
1618 _ (13, CLIB_STRNCPY, "clib_strncpy", clib_strncpy) \
1619 _ (14, STRCAT_S, "strcat_s", strcat_s) \
1620 _ (15, CLIB_STRCAT, "clib_strcat", clib_strcat) \
1621 _ (16, STRNCAT_S, "strncat_s", strncat_s) \
1622 _ (17, CLIB_STRNCAT, "clib_strncat", clib_strncat) \
1623 _ (18, STRTOK_S, "strtok_s", strtok_s) \
1624 _ (19, CLIB_STRTOK, "clib_strtok", clib_strtok) \
1625 _ (20, STRNLEN_S, "strnlen_s", strnlen_s) \
1626 _ (21, CLIB_STRNLEN, "clib_strnlen", clib_strnlen) \
1627 _ (22, STRSTR_S, "strstr_s", strstr_s) \
1628 _ (23, CLIB_STRSTR, "clib_strstr", clib_strstr)
1632 #define _(v,f,s,p) STRING_TEST_##f = v,
1638 unformat_string_test (unformat_input_t * input, va_list * args)
1640 u8 *r = va_arg (*args, u8 *);
1644 #define _(v,f,s,p) else if (unformat (input, s)) *r = STRING_TEST_##f;
1653 typedef int (*string_test_func) (vlib_main_t * vm, unformat_input_t * input);
1657 string_test_func test;
1658 } string_test_func_t;
1660 static clib_error_t *
1661 string_test_command_fn (vlib_main_t * vm,
1662 unformat_input_t * input,
1663 vlib_cli_command_t * cmd_arg)
1665 string_test_func_t string_func[] = {
1666 #define _(v,f,s,p) { test_##p },
1670 const char *string_table[] = {
1671 #define _(v,f,s,p) s,
1676 i8 specific_test = ~0;
1678 while (unformat_check_input (input) != UNFORMAT_END_OF_INPUT)
1680 if (unformat (input, "%U", unformat_string_test, &specific_test))
1683 return clib_error_return (0, "unknown input `%U'",
1684 format_unformat_error, input);
1687 if (specific_test == ~0)
1689 for (specific_test = STRING_TEST_MEMCPY_S;
1690 specific_test <= STRING_TEST_CLIB_STRSTR; specific_test++)
1692 ok = (string_func[specific_test]).test (vm, input);
1695 vlib_cli_output (vm, "test_%s failed",
1696 string_table[specific_test]);
1700 res = (string_func[specific_test]).test (vm, input);
1702 vlib_cli_output (vm, "String unit test(s) failed...");
1704 vlib_cli_output (vm, "String unit test(s) OK...");
1709 VLIB_CLI_COMMAND (string_test_command, static) =
1711 .path = "test string",
1712 .short_help = "test string [memcpy_s | clib_memcpy | memset_s | "
1713 "clib_memset | memcmp_s | clib_memcmp | strcmp_s | clib_strcmp | "
1714 "strncmp_s | clib_strncmp | strcpy_s | clib_strcpy | strncpy_s | "
1715 "clib_strncpy | strcat_s | clib_strcat | strncat_s | clib_strncat | "
1716 "strtok_s | clib_strtok | strnlen_s | clib_strnlen | strstr_s | "
1718 .function = string_test_command_fn,
1723 * fd.io coding-style-patch-verification: ON
1726 * eval: (c-set-style "gnu")