162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci#include <linux/module.h> 362306a36Sopenharmony_ci#include <linux/printk.h> 462306a36Sopenharmony_ci#include <linux/slab.h> 562306a36Sopenharmony_ci#include <linux/string.h> 662306a36Sopenharmony_ci 762306a36Sopenharmony_cistatic __init int memset16_selftest(void) 862306a36Sopenharmony_ci{ 962306a36Sopenharmony_ci unsigned i, j, k; 1062306a36Sopenharmony_ci u16 v, *p; 1162306a36Sopenharmony_ci 1262306a36Sopenharmony_ci p = kmalloc(256 * 2 * 2, GFP_KERNEL); 1362306a36Sopenharmony_ci if (!p) 1462306a36Sopenharmony_ci return -1; 1562306a36Sopenharmony_ci 1662306a36Sopenharmony_ci for (i = 0; i < 256; i++) { 1762306a36Sopenharmony_ci for (j = 0; j < 256; j++) { 1862306a36Sopenharmony_ci memset(p, 0xa1, 256 * 2 * sizeof(v)); 1962306a36Sopenharmony_ci memset16(p + i, 0xb1b2, j); 2062306a36Sopenharmony_ci for (k = 0; k < 512; k++) { 2162306a36Sopenharmony_ci v = p[k]; 2262306a36Sopenharmony_ci if (k < i) { 2362306a36Sopenharmony_ci if (v != 0xa1a1) 2462306a36Sopenharmony_ci goto fail; 2562306a36Sopenharmony_ci } else if (k < i + j) { 2662306a36Sopenharmony_ci if (v != 0xb1b2) 2762306a36Sopenharmony_ci goto fail; 2862306a36Sopenharmony_ci } else { 2962306a36Sopenharmony_ci if (v != 0xa1a1) 3062306a36Sopenharmony_ci goto fail; 3162306a36Sopenharmony_ci } 3262306a36Sopenharmony_ci } 3362306a36Sopenharmony_ci } 3462306a36Sopenharmony_ci } 3562306a36Sopenharmony_ci 3662306a36Sopenharmony_cifail: 3762306a36Sopenharmony_ci kfree(p); 3862306a36Sopenharmony_ci if (i < 256) 3962306a36Sopenharmony_ci return (i << 24) | (j << 16) | k | 0x8000; 4062306a36Sopenharmony_ci return 0; 4162306a36Sopenharmony_ci} 4262306a36Sopenharmony_ci 4362306a36Sopenharmony_cistatic __init int memset32_selftest(void) 4462306a36Sopenharmony_ci{ 4562306a36Sopenharmony_ci unsigned i, j, k; 4662306a36Sopenharmony_ci u32 v, *p; 4762306a36Sopenharmony_ci 4862306a36Sopenharmony_ci p = kmalloc(256 * 2 * 4, GFP_KERNEL); 4962306a36Sopenharmony_ci if (!p) 5062306a36Sopenharmony_ci return -1; 5162306a36Sopenharmony_ci 5262306a36Sopenharmony_ci for (i = 0; i < 256; i++) { 5362306a36Sopenharmony_ci for (j = 0; j < 256; j++) { 5462306a36Sopenharmony_ci memset(p, 0xa1, 256 * 2 * sizeof(v)); 5562306a36Sopenharmony_ci memset32(p + i, 0xb1b2b3b4, j); 5662306a36Sopenharmony_ci for (k = 0; k < 512; k++) { 5762306a36Sopenharmony_ci v = p[k]; 5862306a36Sopenharmony_ci if (k < i) { 5962306a36Sopenharmony_ci if (v != 0xa1a1a1a1) 6062306a36Sopenharmony_ci goto fail; 6162306a36Sopenharmony_ci } else if (k < i + j) { 6262306a36Sopenharmony_ci if (v != 0xb1b2b3b4) 6362306a36Sopenharmony_ci goto fail; 6462306a36Sopenharmony_ci } else { 6562306a36Sopenharmony_ci if (v != 0xa1a1a1a1) 6662306a36Sopenharmony_ci goto fail; 6762306a36Sopenharmony_ci } 6862306a36Sopenharmony_ci } 6962306a36Sopenharmony_ci } 7062306a36Sopenharmony_ci } 7162306a36Sopenharmony_ci 7262306a36Sopenharmony_cifail: 7362306a36Sopenharmony_ci kfree(p); 7462306a36Sopenharmony_ci if (i < 256) 7562306a36Sopenharmony_ci return (i << 24) | (j << 16) | k | 0x8000; 7662306a36Sopenharmony_ci return 0; 7762306a36Sopenharmony_ci} 7862306a36Sopenharmony_ci 7962306a36Sopenharmony_cistatic __init int memset64_selftest(void) 8062306a36Sopenharmony_ci{ 8162306a36Sopenharmony_ci unsigned i, j, k; 8262306a36Sopenharmony_ci u64 v, *p; 8362306a36Sopenharmony_ci 8462306a36Sopenharmony_ci p = kmalloc(256 * 2 * 8, GFP_KERNEL); 8562306a36Sopenharmony_ci if (!p) 8662306a36Sopenharmony_ci return -1; 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_ci for (i = 0; i < 256; i++) { 8962306a36Sopenharmony_ci for (j = 0; j < 256; j++) { 9062306a36Sopenharmony_ci memset(p, 0xa1, 256 * 2 * sizeof(v)); 9162306a36Sopenharmony_ci memset64(p + i, 0xb1b2b3b4b5b6b7b8ULL, j); 9262306a36Sopenharmony_ci for (k = 0; k < 512; k++) { 9362306a36Sopenharmony_ci v = p[k]; 9462306a36Sopenharmony_ci if (k < i) { 9562306a36Sopenharmony_ci if (v != 0xa1a1a1a1a1a1a1a1ULL) 9662306a36Sopenharmony_ci goto fail; 9762306a36Sopenharmony_ci } else if (k < i + j) { 9862306a36Sopenharmony_ci if (v != 0xb1b2b3b4b5b6b7b8ULL) 9962306a36Sopenharmony_ci goto fail; 10062306a36Sopenharmony_ci } else { 10162306a36Sopenharmony_ci if (v != 0xa1a1a1a1a1a1a1a1ULL) 10262306a36Sopenharmony_ci goto fail; 10362306a36Sopenharmony_ci } 10462306a36Sopenharmony_ci } 10562306a36Sopenharmony_ci } 10662306a36Sopenharmony_ci } 10762306a36Sopenharmony_ci 10862306a36Sopenharmony_cifail: 10962306a36Sopenharmony_ci kfree(p); 11062306a36Sopenharmony_ci if (i < 256) 11162306a36Sopenharmony_ci return (i << 24) | (j << 16) | k | 0x8000; 11262306a36Sopenharmony_ci return 0; 11362306a36Sopenharmony_ci} 11462306a36Sopenharmony_ci 11562306a36Sopenharmony_cistatic __init int strchr_selftest(void) 11662306a36Sopenharmony_ci{ 11762306a36Sopenharmony_ci const char *test_string = "abcdefghijkl"; 11862306a36Sopenharmony_ci const char *empty_string = ""; 11962306a36Sopenharmony_ci char *result; 12062306a36Sopenharmony_ci int i; 12162306a36Sopenharmony_ci 12262306a36Sopenharmony_ci for (i = 0; i < strlen(test_string) + 1; i++) { 12362306a36Sopenharmony_ci result = strchr(test_string, test_string[i]); 12462306a36Sopenharmony_ci if (result - test_string != i) 12562306a36Sopenharmony_ci return i + 'a'; 12662306a36Sopenharmony_ci } 12762306a36Sopenharmony_ci 12862306a36Sopenharmony_ci result = strchr(empty_string, '\0'); 12962306a36Sopenharmony_ci if (result != empty_string) 13062306a36Sopenharmony_ci return 0x101; 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_ci result = strchr(empty_string, 'a'); 13362306a36Sopenharmony_ci if (result) 13462306a36Sopenharmony_ci return 0x102; 13562306a36Sopenharmony_ci 13662306a36Sopenharmony_ci result = strchr(test_string, 'z'); 13762306a36Sopenharmony_ci if (result) 13862306a36Sopenharmony_ci return 0x103; 13962306a36Sopenharmony_ci 14062306a36Sopenharmony_ci return 0; 14162306a36Sopenharmony_ci} 14262306a36Sopenharmony_ci 14362306a36Sopenharmony_cistatic __init int strnchr_selftest(void) 14462306a36Sopenharmony_ci{ 14562306a36Sopenharmony_ci const char *test_string = "abcdefghijkl"; 14662306a36Sopenharmony_ci const char *empty_string = ""; 14762306a36Sopenharmony_ci char *result; 14862306a36Sopenharmony_ci int i, j; 14962306a36Sopenharmony_ci 15062306a36Sopenharmony_ci for (i = 0; i < strlen(test_string) + 1; i++) { 15162306a36Sopenharmony_ci for (j = 0; j < strlen(test_string) + 2; j++) { 15262306a36Sopenharmony_ci result = strnchr(test_string, j, test_string[i]); 15362306a36Sopenharmony_ci if (j <= i) { 15462306a36Sopenharmony_ci if (!result) 15562306a36Sopenharmony_ci continue; 15662306a36Sopenharmony_ci return ((i + 'a') << 8) | j; 15762306a36Sopenharmony_ci } 15862306a36Sopenharmony_ci if (result - test_string != i) 15962306a36Sopenharmony_ci return ((i + 'a') << 8) | j; 16062306a36Sopenharmony_ci } 16162306a36Sopenharmony_ci } 16262306a36Sopenharmony_ci 16362306a36Sopenharmony_ci result = strnchr(empty_string, 0, '\0'); 16462306a36Sopenharmony_ci if (result) 16562306a36Sopenharmony_ci return 0x10001; 16662306a36Sopenharmony_ci 16762306a36Sopenharmony_ci result = strnchr(empty_string, 1, '\0'); 16862306a36Sopenharmony_ci if (result != empty_string) 16962306a36Sopenharmony_ci return 0x10002; 17062306a36Sopenharmony_ci 17162306a36Sopenharmony_ci result = strnchr(empty_string, 1, 'a'); 17262306a36Sopenharmony_ci if (result) 17362306a36Sopenharmony_ci return 0x10003; 17462306a36Sopenharmony_ci 17562306a36Sopenharmony_ci result = strnchr(NULL, 0, '\0'); 17662306a36Sopenharmony_ci if (result) 17762306a36Sopenharmony_ci return 0x10004; 17862306a36Sopenharmony_ci 17962306a36Sopenharmony_ci return 0; 18062306a36Sopenharmony_ci} 18162306a36Sopenharmony_ci 18262306a36Sopenharmony_cistatic __init int strspn_selftest(void) 18362306a36Sopenharmony_ci{ 18462306a36Sopenharmony_ci static const struct strspn_test { 18562306a36Sopenharmony_ci const char str[16]; 18662306a36Sopenharmony_ci const char accept[16]; 18762306a36Sopenharmony_ci const char reject[16]; 18862306a36Sopenharmony_ci unsigned a; 18962306a36Sopenharmony_ci unsigned r; 19062306a36Sopenharmony_ci } tests[] __initconst = { 19162306a36Sopenharmony_ci { "foobar", "", "", 0, 6 }, 19262306a36Sopenharmony_ci { "abba", "abc", "ABBA", 4, 4 }, 19362306a36Sopenharmony_ci { "abba", "a", "b", 1, 1 }, 19462306a36Sopenharmony_ci { "", "abc", "abc", 0, 0}, 19562306a36Sopenharmony_ci }; 19662306a36Sopenharmony_ci const struct strspn_test *s = tests; 19762306a36Sopenharmony_ci size_t i, res; 19862306a36Sopenharmony_ci 19962306a36Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(tests); ++i, ++s) { 20062306a36Sopenharmony_ci res = strspn(s->str, s->accept); 20162306a36Sopenharmony_ci if (res != s->a) 20262306a36Sopenharmony_ci return 0x100 + 2*i; 20362306a36Sopenharmony_ci res = strcspn(s->str, s->reject); 20462306a36Sopenharmony_ci if (res != s->r) 20562306a36Sopenharmony_ci return 0x100 + 2*i + 1; 20662306a36Sopenharmony_ci } 20762306a36Sopenharmony_ci return 0; 20862306a36Sopenharmony_ci} 20962306a36Sopenharmony_ci 21062306a36Sopenharmony_cistatic __exit void string_selftest_remove(void) 21162306a36Sopenharmony_ci{ 21262306a36Sopenharmony_ci} 21362306a36Sopenharmony_ci 21462306a36Sopenharmony_cistatic __init int string_selftest_init(void) 21562306a36Sopenharmony_ci{ 21662306a36Sopenharmony_ci int test, subtest; 21762306a36Sopenharmony_ci 21862306a36Sopenharmony_ci test = 1; 21962306a36Sopenharmony_ci subtest = memset16_selftest(); 22062306a36Sopenharmony_ci if (subtest) 22162306a36Sopenharmony_ci goto fail; 22262306a36Sopenharmony_ci 22362306a36Sopenharmony_ci test = 2; 22462306a36Sopenharmony_ci subtest = memset32_selftest(); 22562306a36Sopenharmony_ci if (subtest) 22662306a36Sopenharmony_ci goto fail; 22762306a36Sopenharmony_ci 22862306a36Sopenharmony_ci test = 3; 22962306a36Sopenharmony_ci subtest = memset64_selftest(); 23062306a36Sopenharmony_ci if (subtest) 23162306a36Sopenharmony_ci goto fail; 23262306a36Sopenharmony_ci 23362306a36Sopenharmony_ci test = 4; 23462306a36Sopenharmony_ci subtest = strchr_selftest(); 23562306a36Sopenharmony_ci if (subtest) 23662306a36Sopenharmony_ci goto fail; 23762306a36Sopenharmony_ci 23862306a36Sopenharmony_ci test = 5; 23962306a36Sopenharmony_ci subtest = strnchr_selftest(); 24062306a36Sopenharmony_ci if (subtest) 24162306a36Sopenharmony_ci goto fail; 24262306a36Sopenharmony_ci 24362306a36Sopenharmony_ci test = 6; 24462306a36Sopenharmony_ci subtest = strspn_selftest(); 24562306a36Sopenharmony_ci if (subtest) 24662306a36Sopenharmony_ci goto fail; 24762306a36Sopenharmony_ci 24862306a36Sopenharmony_ci pr_info("String selftests succeeded\n"); 24962306a36Sopenharmony_ci return 0; 25062306a36Sopenharmony_cifail: 25162306a36Sopenharmony_ci pr_crit("String selftest failure %d.%08x\n", test, subtest); 25262306a36Sopenharmony_ci return 0; 25362306a36Sopenharmony_ci} 25462306a36Sopenharmony_ci 25562306a36Sopenharmony_cimodule_init(string_selftest_init); 25662306a36Sopenharmony_cimodule_exit(string_selftest_remove); 25762306a36Sopenharmony_ciMODULE_LICENSE("GPL v2"); 258