// SPDX-License-Identifier: GPL-2.0-or-later /* * KUnit test for the ACPI-WMI string conversion code. * * Copyright (C) 2025 Armin Wolf */ #include #include #include #include #include #include #include struct wmi_string_param { const char *name; const struct wmi_string *wmi_string; /* * Remember that using sizeof() on a struct wmi_string will * always return a size of two bytes due to the flexible * array member! */ size_t wmi_string_length; const u8 *utf8_string; size_t utf8_string_length; }; #define TEST_WMI_STRING_LENGTH 12 static const struct wmi_string test_wmi_string = { .length = cpu_to_le16(10), .chars = { cpu_to_le16(u'T'), cpu_to_le16(u'E'), cpu_to_le16(u'S'), cpu_to_le16(u'T'), cpu_to_le16(u'\0'), }, }; static const u8 test_utf8_string[] = "TEST"; #define SPECIAL_WMI_STRING_LENGTH 14 static const struct wmi_string special_wmi_string = { .length = cpu_to_le16(12), .chars = { cpu_to_le16(u'Ä'), cpu_to_le16(u'Ö'), cpu_to_le16(u'Ü'), cpu_to_le16(u'ß'), cpu_to_le16(u'€'), cpu_to_le16(u'\0'), }, }; static const u8 special_utf8_string[] = "ÄÖÜ߀"; #define MULTI_POINT_WMI_STRING_LENGTH 12 static const struct wmi_string multi_point_wmi_string = { .length = cpu_to_le16(10), .chars = { cpu_to_le16(u'K'), /* 🐧 */ cpu_to_le16(0xD83D), cpu_to_le16(0xDC27), cpu_to_le16(u'!'), cpu_to_le16(u'\0'), }, }; static const u8 multi_point_utf8_string[] = "K🐧!"; #define PADDED_TEST_WMI_STRING_LENGTH 14 static const struct wmi_string padded_test_wmi_string = { .length = cpu_to_le16(12), .chars = { cpu_to_le16(u'T'), cpu_to_le16(u'E'), cpu_to_le16(u'S'), cpu_to_le16(u'T'), cpu_to_le16(u'\0'), cpu_to_le16(u'\0'), }, }; static const u8 padded_test_utf8_string[] = "TEST\0"; #define OVERSIZED_TEST_WMI_STRING_LENGTH 14 static const struct wmi_string oversized_test_wmi_string = { .length = cpu_to_le16(8), .chars = { cpu_to_le16(u'T'), cpu_to_le16(u'E'), cpu_to_le16(u'S'), cpu_to_le16(u'T'), cpu_to_le16(u'!'), cpu_to_le16(u'\0'), }, }; static const u8 oversized_test_utf8_string[] = "TEST!"; #define INVALID_TEST_WMI_STRING_LENGTH 14 static const struct wmi_string invalid_test_wmi_string = { .length = cpu_to_le16(12), .chars = { cpu_to_le16(u'T'), /* 🐧, with low surrogate missing */ cpu_to_le16(0xD83D), cpu_to_le16(u'E'), cpu_to_le16(u'S'), cpu_to_le16(u'T'), cpu_to_le16(u'\0'), }, }; /* We have to split the string here to end the hex escape sequence */ static const u8 invalid_test_utf8_string[] = "T" "\xF0\x9F" "EST"; static const struct wmi_string_param wmi_string_params_array[] = { { .name = "ascii_string", .wmi_string = &test_wmi_string, .wmi_string_length = TEST_WMI_STRING_LENGTH, .utf8_string = test_utf8_string, .utf8_string_length = sizeof(test_utf8_string), }, { .name = "special_string", .wmi_string = &special_wmi_string, .wmi_string_length = SPECIAL_WMI_STRING_LENGTH, .utf8_string = special_utf8_string, .utf8_string_length = sizeof(special_utf8_string), }, { .name = "multi_point_string", .wmi_string = &multi_point_wmi_string, .wmi_string_length = MULTI_POINT_WMI_STRING_LENGTH, .utf8_string = multi_point_utf8_string, .utf8_string_length = sizeof(multi_point_utf8_string), }, }; static void wmi_string_param_get_desc(const struct wmi_string_param *param, char *desc) { strscpy(desc, param->name, KUNIT_PARAM_DESC_SIZE); } KUNIT_ARRAY_PARAM(wmi_string, wmi_string_params_array, wmi_string_param_get_desc); static void wmi_string_to_utf8s_test(struct kunit *test) { const struct wmi_string_param *param = test->param_value; ssize_t ret; u8 *result; result = kunit_kzalloc(test, param->utf8_string_length, GFP_KERNEL); KUNIT_ASSERT_NOT_ERR_OR_NULL(test, result); ret = wmi_string_to_utf8s(param->wmi_string, result, param->utf8_string_length); KUNIT_EXPECT_EQ(test, ret, param->utf8_string_length - 1); KUNIT_EXPECT_MEMEQ(test, result, param->utf8_string, param->utf8_string_length); } static void wmi_string_from_utf8s_test(struct kunit *test) { const struct wmi_string_param *param = test->param_value; struct wmi_string *result; size_t max_chars; ssize_t ret; max_chars = (param->wmi_string_length - sizeof(*result)) / 2; result = kunit_kzalloc(test, param->wmi_string_length, GFP_KERNEL); KUNIT_ASSERT_NOT_ERR_OR_NULL(test, result); ret = wmi_string_from_utf8s(result, max_chars, param->utf8_string, param->utf8_string_length); KUNIT_EXPECT_EQ(test, ret, max_chars - 1); KUNIT_EXPECT_MEMEQ(test, result, param->wmi_string, param->wmi_string_length); } static void wmi_string_to_utf8s_padded_test(struct kunit *test) { u8 result[sizeof(padded_test_utf8_string)]; ssize_t ret; ret = wmi_string_to_utf8s(&padded_test_wmi_string, result, sizeof(result)); KUNIT_EXPECT_EQ(test, ret, sizeof(test_utf8_string) - 1); KUNIT_EXPECT_MEMEQ(test, result, test_utf8_string, sizeof(test_utf8_string)); } static void wmi_string_from_utf8s_padded_test(struct kunit *test) { struct wmi_string *result; size_t max_chars; ssize_t ret; max_chars = (PADDED_TEST_WMI_STRING_LENGTH - sizeof(*result)) / 2; result = kunit_kzalloc(test, PADDED_TEST_WMI_STRING_LENGTH, GFP_KERNEL); KUNIT_ASSERT_NOT_ERR_OR_NULL(test, result); ret = wmi_string_from_utf8s(result, max_chars, padded_test_utf8_string, sizeof(padded_test_utf8_string)); KUNIT_EXPECT_EQ(test, ret, sizeof(test_utf8_string) - 1); KUNIT_EXPECT_MEMEQ(test, result, &test_wmi_string, sizeof(test_wmi_string)); } static void wmi_string_to_utf8s_oversized_test(struct kunit *test) { u8 result[sizeof(oversized_test_utf8_string)]; ssize_t ret; ret = wmi_string_to_utf8s(&oversized_test_wmi_string, result, sizeof(result)); KUNIT_EXPECT_EQ(test, ret, sizeof(test_utf8_string) - 1); KUNIT_EXPECT_MEMEQ(test, result, test_utf8_string, sizeof(test_utf8_string)); } static void wmi_string_from_utf8s_oversized_test(struct kunit *test) { struct wmi_string *result; size_t max_chars; ssize_t ret; max_chars = (TEST_WMI_STRING_LENGTH - sizeof(*result)) / 2; result = kunit_kzalloc(test, TEST_WMI_STRING_LENGTH, GFP_KERNEL); KUNIT_ASSERT_NOT_ERR_OR_NULL(test, result); ret = wmi_string_from_utf8s(result, max_chars, oversized_test_utf8_string, sizeof(oversized_test_utf8_string)); KUNIT_EXPECT_EQ(test, ret, sizeof(test_utf8_string) - 1); KUNIT_EXPECT_MEMEQ(test, result, &test_wmi_string, sizeof(test_wmi_string)); } static void wmi_string_to_utf8s_invalid_test(struct kunit *test) { u8 result[sizeof(invalid_test_utf8_string)]; ssize_t ret; ret = wmi_string_to_utf8s(&invalid_test_wmi_string, result, sizeof(result)); KUNIT_EXPECT_EQ(test, ret, sizeof(test_utf8_string) - 1); KUNIT_EXPECT_MEMEQ(test, result, test_utf8_string, sizeof(test_utf8_string)); } static void wmi_string_from_utf8s_invalid_test(struct kunit *test) { struct wmi_string *result; size_t max_chars; ssize_t ret; max_chars = (INVALID_TEST_WMI_STRING_LENGTH - sizeof(*result)) / 2; result = kunit_kzalloc(test, INVALID_TEST_WMI_STRING_LENGTH, GFP_KERNEL); KUNIT_ASSERT_NOT_ERR_OR_NULL(test, result); ret = wmi_string_from_utf8s(result, max_chars, invalid_test_utf8_string, sizeof(invalid_test_utf8_string)); KUNIT_EXPECT_EQ(test, ret, -EINVAL); } static struct kunit_case wmi_string_test_cases[] = { KUNIT_CASE_PARAM(wmi_string_to_utf8s_test, wmi_string_gen_params), KUNIT_CASE_PARAM(wmi_string_from_utf8s_test, wmi_string_gen_params), KUNIT_CASE(wmi_string_to_utf8s_padded_test), KUNIT_CASE(wmi_string_from_utf8s_padded_test), KUNIT_CASE(wmi_string_to_utf8s_oversized_test), KUNIT_CASE(wmi_string_from_utf8s_oversized_test), KUNIT_CASE(wmi_string_to_utf8s_invalid_test), KUNIT_CASE(wmi_string_from_utf8s_invalid_test), {} }; static struct kunit_suite wmi_string_test_suite = { .name = "wmi_string", .test_cases = wmi_string_test_cases, }; kunit_test_suite(wmi_string_test_suite); MODULE_AUTHOR("Armin Wolf "); MODULE_DESCRIPTION("KUnit test for the ACPI-WMI string conversion code"); MODULE_LICENSE("GPL");