summaryrefslogtreecommitdiffstats
path: root/lib/tests/glob_kunit.c
blob: 362b1eda8e5bc49781a20d1dc7d2d65bf2b68248 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
// SPDX-License-Identifier: MIT OR GPL-2.0
/*
 * Test cases for glob functions.
 */

#include <kunit/test.h>
#include <linux/glob.h>
#include <linux/module.h>

/**
 * struct glob_test_case - Test case for glob matching.
 * @pat: Pattern to match.
 * @str: String to match against.
 * @expected: Expected glob_match result, true if matched.
 */
struct glob_test_case {
	const char *pat;
	const char *str;
	bool expected;
};

static const struct glob_test_case glob_test_cases[] = {
	/* Some basic tests */
	{ .pat = "a", .str = "a", .expected = true },
	{ .pat = "a", .str = "b", .expected = false },
	{ .pat = "a", .str = "aa", .expected = false },
	{ .pat = "a", .str = "", .expected = false },
	{ .pat = "", .str = "", .expected = true },
	{ .pat = "", .str = "a", .expected = false },
	/* Simple character class tests */
	{ .pat = "[a]", .str = "a", .expected = true },
	{ .pat = "[a]", .str = "b", .expected = false },
	{ .pat = "[!a]", .str = "a", .expected = false },
	{ .pat = "[!a]", .str = "b", .expected = true },
	{ .pat = "[ab]", .str = "a", .expected = true },
	{ .pat = "[ab]", .str = "b", .expected = true },
	{ .pat = "[ab]", .str = "c", .expected = false },
	{ .pat = "[!ab]", .str = "c", .expected = true },
	{ .pat = "[a-c]", .str = "b", .expected = true },
	{ .pat = "[a-c]", .str = "d", .expected = false },
	/* Corner cases in character class parsing */
	{ .pat = "[a-c-e-g]", .str = "-", .expected = true },
	{ .pat = "[a-c-e-g]", .str = "d", .expected = false },
	{ .pat = "[a-c-e-g]", .str = "f", .expected = true },
	{ .pat = "[]a-ceg-ik[]", .str = "a", .expected = true },
	{ .pat = "[]a-ceg-ik[]", .str = "]", .expected = true },
	{ .pat = "[]a-ceg-ik[]", .str = "[", .expected = true },
	{ .pat = "[]a-ceg-ik[]", .str = "h", .expected = true },
	{ .pat = "[]a-ceg-ik[]", .str = "f", .expected = false },
	{ .pat = "[!]a-ceg-ik[]", .str = "h", .expected = false },
	{ .pat = "[!]a-ceg-ik[]", .str = "]", .expected = false },
	{ .pat = "[!]a-ceg-ik[]", .str = "f", .expected = true },
	/* Simple wild cards */
	{ .pat = "?", .str = "a", .expected = true },
	{ .pat = "?", .str = "aa", .expected = false },
	{ .pat = "??", .str = "a", .expected = false },
	{ .pat = "?x?", .str = "axb", .expected = true },
	{ .pat = "?x?", .str = "abx", .expected = false },
	{ .pat = "?x?", .str = "xab", .expected = false },
	/* Asterisk wild cards (backtracking) */
	{ .pat = "*??", .str = "a", .expected = false },
	{ .pat = "*??", .str = "ab", .expected = true },
	{ .pat = "*??", .str = "abc", .expected = true },
	{ .pat = "*??", .str = "abcd", .expected = true },
	{ .pat = "??*", .str = "a", .expected = false },
	{ .pat = "??*", .str = "ab", .expected = true },
	{ .pat = "??*", .str = "abc", .expected = true },
	{ .pat = "??*", .str = "abcd", .expected = true },
	{ .pat = "?*?", .str = "a", .expected = false },
	{ .pat = "?*?", .str = "ab", .expected = true },
	{ .pat = "?*?", .str = "abc", .expected = true },
	{ .pat = "?*?", .str = "abcd", .expected = true },
	{ .pat = "*b", .str = "b", .expected = true },
	{ .pat = "*b", .str = "ab", .expected = true },
	{ .pat = "*b", .str = "ba", .expected = false },
	{ .pat = "*b", .str = "bb", .expected = true },
	{ .pat = "*b", .str = "abb", .expected = true },
	{ .pat = "*b", .str = "bab", .expected = true },
	{ .pat = "*bc", .str = "abbc", .expected = true },
	{ .pat = "*bc", .str = "bc", .expected = true },
	{ .pat = "*bc", .str = "bbc", .expected = true },
	{ .pat = "*bc", .str = "bcbc", .expected = true },
	/* Multiple asterisks (complex backtracking) */
	{ .pat = "*ac*", .str = "abacadaeafag", .expected = true },
	{ .pat = "*ac*ae*ag*", .str = "abacadaeafag", .expected = true },
	{ .pat = "*a*b*[bc]*[ef]*g*", .str = "abacadaeafag", .expected = true },
	{ .pat = "*a*b*[ef]*[cd]*g*", .str = "abacadaeafag", .expected = false },
	{ .pat = "*abcd*", .str = "abcabcabcabcdefg", .expected = true },
	{ .pat = "*ab*cd*", .str = "abcabcabcabcdefg", .expected = true },
	{ .pat = "*abcd*abcdef*", .str = "abcabcdabcdeabcdefg", .expected = true },
	{ .pat = "*abcd*", .str = "abcabcabcabcefg", .expected = false },
	{ .pat = "*ab*cd*", .str = "abcabcabcabcefg", .expected = false },
};

static void glob_case_to_desc(const struct glob_test_case *t, char *desc)
{
	snprintf(desc, KUNIT_PARAM_DESC_SIZE, "pat:\"%s\" str:\"%s\"", t->pat, t->str);
}

KUNIT_ARRAY_PARAM(glob, glob_test_cases, glob_case_to_desc);

static void glob_test_match(struct kunit *test)
{
	const struct glob_test_case *params = test->param_value;

	KUNIT_EXPECT_EQ_MSG(test,
			    glob_match(params->pat, params->str),
			    params->expected,
			    "Pattern: \"%s\", String: \"%s\", Expected: %d",
			    params->pat, params->str, params->expected);
}

static struct kunit_case glob_kunit_test_cases[] = {
	KUNIT_CASE_PARAM(glob_test_match, glob_gen_params),
	{}
};

static struct kunit_suite glob_test_suite = {
	.name = "glob",
	.test_cases = glob_kunit_test_cases,
};

kunit_test_suite(glob_test_suite);
MODULE_DESCRIPTION("Test cases for glob functions");
MODULE_LICENSE("Dual MIT/GPL");