opnsense-src/lib/libc/tests/string/strverscmp_test.c
Aymeric Wibo 05c9a0158f libc: Add strverscmp(3) and versionsort(3)
Add a strverscmp(3) function to libc, a GNU extension I implemented by
reading its glibc manual page. It orders strings following a much more
natural ordering (e.g. "ent1 < ent2 < ent10" as opposed to
"ent1 < ent10 < ent2" with strcmp(3)'s lexicographic ordering).

Also add versionsort(3) for use as scandir(3)'s compar argument.

Update manual page for scandir(3) and add one for strverscmp(3).

Reviewed by:	pstef, gbe, kib
MFC after:	1 week
Differential Revision: https://reviews.freebsd.org/D35807
2022-08-25 03:29:03 +03:00

93 lines
1.9 KiB
C

/*-
* SPDX-License-Identifier: BSD-2-Clause
* Copyright (c) 2022 Aymeric Wibo <obiwac@gmail.com>
*/
#include <atf-c.h>
#include <string.h>
static void
check_all(size_t len, const char *ordered[len])
{
const char *a, *b;
for (size_t i = 0; i < len; i++) {
for (size_t j = 0; j < len; j++) {
a = ordered[i];
b = ordered[j];
if (i == j)
ATF_CHECK_MSG(
strverscmp(a, b) == 0,
"strverscmp(\"%s\", \"%s\") == 0",
a, b
);
else if (i < j)
ATF_CHECK_MSG(
strverscmp(a, b) < 0,
"strverscmp(\"%s\", \"%s\") < 0",
a, b
);
else if (i > j)
ATF_CHECK_MSG(
strverscmp(a, b) > 0,
"strverscmp(\"%s\", \"%s\") > 0",
a, b
);
}
}
}
#define CHECK_ALL(...) do { \
const char *ordered[] = { __VA_ARGS__ }; \
check_all(sizeof(ordered) / sizeof(*ordered), ordered); \
} while (0)
ATF_TC_WITHOUT_HEAD(strcmp_functionality);
ATF_TC_BODY(strcmp_functionality, tc)
{
CHECK_ALL("", "a", "b");
}
/* from Linux man page strverscmp(3) */
ATF_TC_WITHOUT_HEAD(vers_ordering);
ATF_TC_BODY(vers_ordering, tc)
{
CHECK_ALL("000", "00", "01", "010", "09", "0", "1", "9", "10");
}
ATF_TC_WITHOUT_HEAD(natural_ordering);
ATF_TC_BODY(natural_ordering, tc)
{
CHECK_ALL("jan1", "jan2", "jan9", "jan10", "jan11", "jan19", "jan20");
}
/* https://sourceware.org/bugzilla/show_bug.cgi?id=9913 */
ATF_TC_WITHOUT_HEAD(glibc_bug_9913);
ATF_TC_BODY(glibc_bug_9913, tc)
{
CHECK_ALL(
"B0075022800016.gbp.corp.com",
"B007502280067.gbp.corp.com",
"B007502357019.GBP.CORP.COM"
);
}
ATF_TC_WITHOUT_HEAD(semver_ordering);
ATF_TC_BODY(semver_ordering, tc)
{
CHECK_ALL("2.6.20", "2.6.21");
}
ATF_TP_ADD_TCS(tp)
{
ATF_TP_ADD_TC(tp, strcmp_functionality);
ATF_TP_ADD_TC(tp, vers_ordering);
ATF_TP_ADD_TC(tp, natural_ordering);
ATF_TP_ADD_TC(tp, glibc_bug_9913);
ATF_TP_ADD_TC(tp, semver_ordering);
return (atf_no_error());
}