1*150ec68fSViacheslav Dubeyko // SPDX-License-Identifier: GPL-2.0 2*150ec68fSViacheslav Dubeyko /* 3*150ec68fSViacheslav Dubeyko * KUnit tests for HFS string operations 4*150ec68fSViacheslav Dubeyko * 5*150ec68fSViacheslav Dubeyko * Copyright (C) 2025 Viacheslav Dubeyko <slava@dubeyko.com> 6*150ec68fSViacheslav Dubeyko */ 7*150ec68fSViacheslav Dubeyko 8*150ec68fSViacheslav Dubeyko #include <kunit/test.h> 9*150ec68fSViacheslav Dubeyko #include <linux/dcache.h> 10*150ec68fSViacheslav Dubeyko #include "hfs_fs.h" 11*150ec68fSViacheslav Dubeyko 12*150ec68fSViacheslav Dubeyko /* Test hfs_strcmp function */ 13*150ec68fSViacheslav Dubeyko static void hfs_strcmp_test(struct kunit *test) 14*150ec68fSViacheslav Dubeyko { 15*150ec68fSViacheslav Dubeyko /* Test equal strings */ 16*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 0, hfs_strcmp("hello", 5, "hello", 5)); 17*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 0, hfs_strcmp("test", 4, "test", 4)); 18*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 0, hfs_strcmp("", 0, "", 0)); 19*150ec68fSViacheslav Dubeyko 20*150ec68fSViacheslav Dubeyko /* Test unequal strings */ 21*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_NE(test, 0, hfs_strcmp("hello", 5, "world", 5)); 22*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_NE(test, 0, hfs_strcmp("test", 4, "testing", 7)); 23*150ec68fSViacheslav Dubeyko 24*150ec68fSViacheslav Dubeyko /* Test different lengths */ 25*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_LT(test, hfs_strcmp("test", 4, "testing", 7), 0); 26*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_GT(test, hfs_strcmp("testing", 7, "test", 4), 0); 27*150ec68fSViacheslav Dubeyko 28*150ec68fSViacheslav Dubeyko /* Test case insensitive comparison (HFS should handle case) */ 29*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 0, hfs_strcmp("Test", 4, "TEST", 4)); 30*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 0, hfs_strcmp("hello", 5, "HELLO", 5)); 31*150ec68fSViacheslav Dubeyko 32*150ec68fSViacheslav Dubeyko /* Test with special characters */ 33*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 0, hfs_strcmp("file.txt", 8, "file.txt", 8)); 34*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_NE(test, 0, hfs_strcmp("file.txt", 8, "file.dat", 8)); 35*150ec68fSViacheslav Dubeyko 36*150ec68fSViacheslav Dubeyko /* Test boundary cases */ 37*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 0, hfs_strcmp("a", 1, "a", 1)); 38*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_NE(test, 0, hfs_strcmp("a", 1, "b", 1)); 39*150ec68fSViacheslav Dubeyko } 40*150ec68fSViacheslav Dubeyko 41*150ec68fSViacheslav Dubeyko /* Test hfs_hash_dentry function */ 42*150ec68fSViacheslav Dubeyko static void hfs_hash_dentry_test(struct kunit *test) 43*150ec68fSViacheslav Dubeyko { 44*150ec68fSViacheslav Dubeyko struct qstr test_name1, test_name2, test_name3; 45*150ec68fSViacheslav Dubeyko struct dentry dentry = {}; 46*150ec68fSViacheslav Dubeyko char name1[] = "testfile"; 47*150ec68fSViacheslav Dubeyko char name2[] = "TestFile"; 48*150ec68fSViacheslav Dubeyko char name3[] = "different"; 49*150ec68fSViacheslav Dubeyko 50*150ec68fSViacheslav Dubeyko /* Initialize test strings */ 51*150ec68fSViacheslav Dubeyko test_name1.name = name1; 52*150ec68fSViacheslav Dubeyko test_name1.len = strlen(name1); 53*150ec68fSViacheslav Dubeyko test_name1.hash = 0; 54*150ec68fSViacheslav Dubeyko 55*150ec68fSViacheslav Dubeyko test_name2.name = name2; 56*150ec68fSViacheslav Dubeyko test_name2.len = strlen(name2); 57*150ec68fSViacheslav Dubeyko test_name2.hash = 0; 58*150ec68fSViacheslav Dubeyko 59*150ec68fSViacheslav Dubeyko test_name3.name = name3; 60*150ec68fSViacheslav Dubeyko test_name3.len = strlen(name3); 61*150ec68fSViacheslav Dubeyko test_name3.hash = 0; 62*150ec68fSViacheslav Dubeyko 63*150ec68fSViacheslav Dubeyko /* Test hashing */ 64*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 0, hfs_hash_dentry(&dentry, &test_name1)); 65*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 0, hfs_hash_dentry(&dentry, &test_name2)); 66*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 0, hfs_hash_dentry(&dentry, &test_name3)); 67*150ec68fSViacheslav Dubeyko 68*150ec68fSViacheslav Dubeyko /* Case insensitive names should hash the same */ 69*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, test_name1.hash, test_name2.hash); 70*150ec68fSViacheslav Dubeyko 71*150ec68fSViacheslav Dubeyko /* Different names should have different hashes */ 72*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_NE(test, test_name1.hash, test_name3.hash); 73*150ec68fSViacheslav Dubeyko } 74*150ec68fSViacheslav Dubeyko 75*150ec68fSViacheslav Dubeyko /* Test hfs_compare_dentry function */ 76*150ec68fSViacheslav Dubeyko static void hfs_compare_dentry_test(struct kunit *test) 77*150ec68fSViacheslav Dubeyko { 78*150ec68fSViacheslav Dubeyko struct qstr test_name; 79*150ec68fSViacheslav Dubeyko struct dentry dentry = {}; 80*150ec68fSViacheslav Dubeyko char name[] = "TestFile"; 81*150ec68fSViacheslav Dubeyko 82*150ec68fSViacheslav Dubeyko test_name.name = name; 83*150ec68fSViacheslav Dubeyko test_name.len = strlen(name); 84*150ec68fSViacheslav Dubeyko 85*150ec68fSViacheslav Dubeyko /* Test exact match */ 86*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 0, hfs_compare_dentry(&dentry, 8, 87*150ec68fSViacheslav Dubeyko "TestFile", &test_name)); 88*150ec68fSViacheslav Dubeyko 89*150ec68fSViacheslav Dubeyko /* Test case insensitive match */ 90*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 0, hfs_compare_dentry(&dentry, 8, 91*150ec68fSViacheslav Dubeyko "testfile", &test_name)); 92*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 0, hfs_compare_dentry(&dentry, 8, 93*150ec68fSViacheslav Dubeyko "TESTFILE", &test_name)); 94*150ec68fSViacheslav Dubeyko 95*150ec68fSViacheslav Dubeyko /* Test different names */ 96*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 1, hfs_compare_dentry(&dentry, 8, 97*150ec68fSViacheslav Dubeyko "DiffFile", &test_name)); 98*150ec68fSViacheslav Dubeyko 99*150ec68fSViacheslav Dubeyko /* Test different lengths */ 100*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 1, hfs_compare_dentry(&dentry, 7, 101*150ec68fSViacheslav Dubeyko "TestFil", &test_name)); 102*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 1, hfs_compare_dentry(&dentry, 9, 103*150ec68fSViacheslav Dubeyko "TestFiles", &test_name)); 104*150ec68fSViacheslav Dubeyko 105*150ec68fSViacheslav Dubeyko /* Test empty string */ 106*150ec68fSViacheslav Dubeyko test_name.name = ""; 107*150ec68fSViacheslav Dubeyko test_name.len = 0; 108*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 0, hfs_compare_dentry(&dentry, 0, "", &test_name)); 109*150ec68fSViacheslav Dubeyko 110*150ec68fSViacheslav Dubeyko /* Test HFS_NAMELEN boundary */ 111*150ec68fSViacheslav Dubeyko test_name.name = "This_is_a_very_long_filename_that_exceeds_normal_limits"; 112*150ec68fSViacheslav Dubeyko test_name.len = strlen(test_name.name); 113*150ec68fSViacheslav Dubeyko KUNIT_EXPECT_EQ(test, 0, hfs_compare_dentry(&dentry, HFS_NAMELEN, 114*150ec68fSViacheslav Dubeyko "This_is_a_very_long_filename_th", &test_name)); 115*150ec68fSViacheslav Dubeyko } 116*150ec68fSViacheslav Dubeyko 117*150ec68fSViacheslav Dubeyko static struct kunit_case hfs_string_test_cases[] = { 118*150ec68fSViacheslav Dubeyko KUNIT_CASE(hfs_strcmp_test), 119*150ec68fSViacheslav Dubeyko KUNIT_CASE(hfs_hash_dentry_test), 120*150ec68fSViacheslav Dubeyko KUNIT_CASE(hfs_compare_dentry_test), 121*150ec68fSViacheslav Dubeyko {} 122*150ec68fSViacheslav Dubeyko }; 123*150ec68fSViacheslav Dubeyko 124*150ec68fSViacheslav Dubeyko static struct kunit_suite hfs_string_test_suite = { 125*150ec68fSViacheslav Dubeyko .name = "hfs_string", 126*150ec68fSViacheslav Dubeyko .test_cases = hfs_string_test_cases, 127*150ec68fSViacheslav Dubeyko }; 128*150ec68fSViacheslav Dubeyko 129*150ec68fSViacheslav Dubeyko kunit_test_suite(hfs_string_test_suite); 130*150ec68fSViacheslav Dubeyko 131*150ec68fSViacheslav Dubeyko MODULE_DESCRIPTION("KUnit tests for HFS string operations"); 132*150ec68fSViacheslav Dubeyko MODULE_LICENSE("GPL"); 133*150ec68fSViacheslav Dubeyko MODULE_IMPORT_NS("EXPORTED_FOR_KUNIT_TESTING"); 134