| // SPDX-License-Identifier: GPL-2.0 |
| /* Copyright (c) 2024, Vladimir Oltean <olteanv@gmail.com> |
| * Copyright (c) 2024, Intel Corporation. |
| */ |
| #include <kunit/test.h> |
| #include <linux/packing.h> |
| |
| struct packing_test_case { |
| const char *desc; |
| const u8 *pbuf; |
| size_t pbuf_size; |
| u64 uval; |
| size_t start_bit; |
| size_t end_bit; |
| u8 quirks; |
| }; |
| |
| #define NO_QUIRKS 0 |
| |
| /** |
| * PBUF - Initialize .pbuf and .pbuf_size |
| * @array: elements of constant physical buffer |
| * |
| * Initializes the .pbuf and .pbuf_size fields of a struct packing_test_case |
| * with a constant array of the specified elements. |
| */ |
| #define PBUF(array...) \ |
| .pbuf = (const u8[]){ array }, \ |
| .pbuf_size = sizeof((const u8 []){ array }) |
| |
| static const struct packing_test_case cases[] = { |
| /* These tests pack and unpack a magic 64-bit value |
| * (0xcafedeadbeefcafe) at a fixed logical offset (32) within an |
| * otherwise zero array of 128 bits (16 bytes). They test all possible |
| * bit layouts of the 128 bit buffer. |
| */ |
| { |
| .desc = "no quirks, 16 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0xca, 0xfe, 0xde, 0xad, |
| 0xbe, 0xef, 0xca, 0xfe, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = NO_QUIRKS, |
| }, |
| { |
| .desc = "lsw32 first, 16 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0xbe, 0xef, 0xca, 0xfe, |
| 0xca, 0xfe, 0xde, 0xad, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = QUIRK_LSW32_IS_FIRST, |
| }, |
| { |
| .desc = "little endian words, 16 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0xad, 0xde, 0xfe, 0xca, |
| 0xfe, 0xca, 0xef, 0xbe, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = QUIRK_LITTLE_ENDIAN, |
| }, |
| { |
| .desc = "lsw32 first + little endian words, 16 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0xfe, 0xca, 0xef, 0xbe, |
| 0xad, 0xde, 0xfe, 0xca, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = QUIRK_LSW32_IS_FIRST | QUIRK_LITTLE_ENDIAN, |
| }, |
| { |
| .desc = "msb right, 16 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0x53, 0x7f, 0x7b, 0xb5, |
| 0x7d, 0xf7, 0x53, 0x7f, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = QUIRK_MSB_ON_THE_RIGHT, |
| }, |
| { |
| .desc = "msb right + lsw32 first, 16 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0x7d, 0xf7, 0x53, 0x7f, |
| 0x53, 0x7f, 0x7b, 0xb5, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = QUIRK_MSB_ON_THE_RIGHT | QUIRK_LSW32_IS_FIRST, |
| }, |
| { |
| .desc = "msb right + little endian words, 16 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0xb5, 0x7b, 0x7f, 0x53, |
| 0x7f, 0x53, 0xf7, 0x7d, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = QUIRK_MSB_ON_THE_RIGHT | QUIRK_LITTLE_ENDIAN, |
| }, |
| { |
| .desc = "msb right + lsw32 first + little endian words, 16 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0x7f, 0x53, 0xf7, 0x7d, |
| 0xb5, 0x7b, 0x7f, 0x53, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = QUIRK_MSB_ON_THE_RIGHT | QUIRK_LSW32_IS_FIRST | QUIRK_LITTLE_ENDIAN, |
| }, |
| /* These tests pack and unpack a magic 64-bit value |
| * (0xcafedeadbeefcafe) at a fixed logical offset (32) within an |
| * otherwise zero array of varying size from 18 bytes to 24 bytes. |
| */ |
| { |
| .desc = "no quirks, 18 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xca, 0xfe, |
| 0xde, 0xad, 0xbe, 0xef, 0xca, 0xfe, 0x00, 0x00, |
| 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = NO_QUIRKS, |
| }, |
| { |
| .desc = "no quirks, 19 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xca, |
| 0xfe, 0xde, 0xad, 0xbe, 0xef, 0xca, 0xfe, 0x00, |
| 0x00, 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = NO_QUIRKS, |
| }, |
| { |
| .desc = "no quirks, 20 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, |
| 0xca, 0xfe, 0xde, 0xad, 0xbe, 0xef, 0xca, 0xfe, |
| 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = NO_QUIRKS, |
| }, |
| { |
| .desc = "no quirks, 22 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, |
| 0x00, 0x00, 0xca, 0xfe, 0xde, 0xad, 0xbe, 0xef, |
| 0xca, 0xfe, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = NO_QUIRKS, |
| }, |
| { |
| .desc = "no quirks, 24 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, |
| 0x00, 0x00, 0x00, 0x00, 0xca, 0xfe, 0xde, 0xad, |
| 0xbe, 0xef, 0xca, 0xfe, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = NO_QUIRKS, |
| }, |
| { |
| .desc = "lsw32 first + little endian words, 18 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0xfe, 0xca, 0xef, 0xbe, |
| 0xad, 0xde, 0xfe, 0xca, 0x00, 0x00, 0x00, 0x00, |
| 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = QUIRK_LSW32_IS_FIRST | QUIRK_LITTLE_ENDIAN, |
| }, |
| { |
| .desc = "lsw32 first + little endian words, 19 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0xfe, 0xca, 0xef, 0xbe, |
| 0xad, 0xde, 0xfe, 0xca, 0x00, 0x00, 0x00, 0x00, |
| 0x00, 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = QUIRK_LSW32_IS_FIRST | QUIRK_LITTLE_ENDIAN, |
| }, |
| { |
| .desc = "lsw32 first + little endian words, 20 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0xfe, 0xca, 0xef, 0xbe, |
| 0xad, 0xde, 0xfe, 0xca, 0x00, 0x00, 0x00, 0x00, |
| 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = QUIRK_LSW32_IS_FIRST | QUIRK_LITTLE_ENDIAN, |
| }, |
| { |
| .desc = "lsw32 first + little endian words, 22 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0xfe, 0xca, 0xef, 0xbe, |
| 0xad, 0xde, 0xfe, 0xca, 0x00, 0x00, 0x00, 0x00, |
| 0x00, 0x00, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = QUIRK_LSW32_IS_FIRST | QUIRK_LITTLE_ENDIAN, |
| }, |
| { |
| .desc = "lsw32 first + little endian words, 24 bytes", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0xfe, 0xca, 0xef, 0xbe, |
| 0xad, 0xde, 0xfe, 0xca, 0x00, 0x00, 0x00, 0x00, |
| 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xcafedeadbeefcafe, |
| .start_bit = 95, |
| .end_bit = 32, |
| .quirks = QUIRK_LSW32_IS_FIRST | QUIRK_LITTLE_ENDIAN, |
| }, |
| /* These tests pack and unpack a magic 64-bit value |
| * (0x1122334455667788) at an odd starting bit (43) within an |
| * otherwise zero array of 128 bits (16 bytes). They test all possible |
| * bit layouts of the 128 bit buffer. |
| */ |
| { |
| .desc = "no quirks, 16 bytes, non-aligned", |
| PBUF(0x00, 0x00, 0x00, 0x89, 0x11, 0x9a, 0x22, 0xab, |
| 0x33, 0xbc, 0x40, 0x00, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0x1122334455667788, |
| .start_bit = 106, |
| .end_bit = 43, |
| .quirks = NO_QUIRKS, |
| }, |
| { |
| .desc = "lsw32 first, 16 bytes, non-aligned", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0x33, 0xbc, 0x40, 0x00, |
| 0x11, 0x9a, 0x22, 0xab, 0x00, 0x00, 0x00, 0x89), |
| .uval = 0x1122334455667788, |
| .start_bit = 106, |
| .end_bit = 43, |
| .quirks = QUIRK_LSW32_IS_FIRST, |
| }, |
| { |
| .desc = "little endian words, 16 bytes, non-aligned", |
| PBUF(0x89, 0x00, 0x00, 0x00, 0xab, 0x22, 0x9a, 0x11, |
| 0x00, 0x40, 0xbc, 0x33, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0x1122334455667788, |
| .start_bit = 106, |
| .end_bit = 43, |
| .quirks = QUIRK_LITTLE_ENDIAN, |
| }, |
| { |
| .desc = "lsw32 first + little endian words, 16 bytes, non-aligned", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0x00, 0x40, 0xbc, 0x33, |
| 0xab, 0x22, 0x9a, 0x11, 0x89, 0x00, 0x00, 0x00), |
| .uval = 0x1122334455667788, |
| .start_bit = 106, |
| .end_bit = 43, |
| .quirks = QUIRK_LSW32_IS_FIRST | QUIRK_LITTLE_ENDIAN, |
| }, |
| { |
| .desc = "msb right, 16 bytes, non-aligned", |
| PBUF(0x00, 0x00, 0x00, 0x91, 0x88, 0x59, 0x44, 0xd5, |
| 0xcc, 0x3d, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0x1122334455667788, |
| .start_bit = 106, |
| .end_bit = 43, |
| .quirks = QUIRK_MSB_ON_THE_RIGHT, |
| }, |
| { |
| .desc = "msb right + lsw32 first, 16 bytes, non-aligned", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0xcc, 0x3d, 0x02, 0x00, |
| 0x88, 0x59, 0x44, 0xd5, 0x00, 0x00, 0x00, 0x91), |
| .uval = 0x1122334455667788, |
| .start_bit = 106, |
| .end_bit = 43, |
| .quirks = QUIRK_MSB_ON_THE_RIGHT | QUIRK_LSW32_IS_FIRST, |
| }, |
| { |
| .desc = "msb right + little endian words, 16 bytes, non-aligned", |
| PBUF(0x91, 0x00, 0x00, 0x00, 0xd5, 0x44, 0x59, 0x88, |
| 0x00, 0x02, 0x3d, 0xcc, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0x1122334455667788, |
| .start_bit = 106, |
| .end_bit = 43, |
| .quirks = QUIRK_MSB_ON_THE_RIGHT | QUIRK_LITTLE_ENDIAN, |
| }, |
| { |
| .desc = "msb right + lsw32 first + little endian words, 16 bytes, non-aligned", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0x3d, 0xcc, |
| 0xd5, 0x44, 0x59, 0x88, 0x91, 0x00, 0x00, 0x00), |
| .uval = 0x1122334455667788, |
| .start_bit = 106, |
| .end_bit = 43, |
| .quirks = QUIRK_MSB_ON_THE_RIGHT | QUIRK_LSW32_IS_FIRST | QUIRK_LITTLE_ENDIAN, |
| }, |
| /* These tests pack and unpack a u64 with all bits set |
| * (0xffffffffffffffff) at an odd starting bit (43) within an |
| * otherwise zero array of 128 bits (16 bytes). They test all possible |
| * bit layouts of the 128 bit buffer. |
| */ |
| { |
| .desc = "no quirks, 16 bytes, non-aligned, 0xff", |
| PBUF(0x00, 0x00, 0x07, 0xff, 0xff, 0xff, 0xff, 0xff, |
| 0xff, 0xff, 0xf8, 0x00, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xffffffffffffffff, |
| .start_bit = 106, |
| .end_bit = 43, |
| .quirks = NO_QUIRKS, |
| }, |
| { |
| .desc = "lsw32 first, 16 bytes, non-aligned, 0xff", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0xf8, 0x00, |
| 0xff, 0xff, 0xff, 0xff, 0x00, 0x00, 0x07, 0xff), |
| .uval = 0xffffffffffffffff, |
| .start_bit = 106, |
| .end_bit = 43, |
| .quirks = QUIRK_LSW32_IS_FIRST, |
| }, |
| { |
| .desc = "little endian words, 16 bytes, non-aligned, 0xff", |
| PBUF(0xff, 0x07, 0x00, 0x00, 0xff, 0xff, 0xff, 0xff, |
| 0x00, 0xf8, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xffffffffffffffff, |
| .start_bit = 106, |
| .end_bit = 43, |
| .quirks = QUIRK_LITTLE_ENDIAN, |
| }, |
| { |
| .desc = "lsw32 first + little endian words, 16 bytes, non-aligned, 0xff", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0x00, 0xf8, 0xff, 0xff, |
| 0xff, 0xff, 0xff, 0xff, 0xff, 0x07, 0x00, 0x00), |
| .uval = 0xffffffffffffffff, |
| .start_bit = 106, |
| .end_bit = 43, |
| .quirks = QUIRK_LSW32_IS_FIRST | QUIRK_LITTLE_ENDIAN, |
| }, |
| { |
| .desc = "msb right, 16 bytes, non-aligned, 0xff", |
| PBUF(0x00, 0x00, 0xe0, 0xff, 0xff, 0xff, 0xff, 0xff, |
| 0xff, 0xff, 0x1f, 0x00, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xffffffffffffffff, |
| .start_bit = 106, |
| .end_bit = 43, |
| .quirks = QUIRK_MSB_ON_THE_RIGHT, |
| }, |
| { |
| .desc = "msb right + lsw32 first, 16 bytes, non-aligned, 0xff", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0x1f, 0x00, |
| 0xff, 0xff, 0xff, 0xff, 0x00, 0x00, 0xe0, 0xff), |
| .uval = 0xffffffffffffffff, |
| .start_bit = 106, |
| .end_bit = 43, |
| .quirks = QUIRK_MSB_ON_THE_RIGHT | QUIRK_LSW32_IS_FIRST, |
| }, |
| { |
| .desc = "msb right + little endian words, 16 bytes, non-aligned, 0xff", |
| PBUF(0xff, 0xe0, 0x00, 0x00, 0xff, 0xff, 0xff, 0xff, |
| 0x00, 0x1f, 0xff, 0xff, 0x00, 0x00, 0x00, 0x00), |
| .uval = 0xffffffffffffffff, |
| .start_bit = 106, |
| .end_bit = 43, |
| .quirks = QUIRK_MSB_ON_THE_RIGHT | QUIRK_LITTLE_ENDIAN, |
| }, |
| { |
| .desc = "msb right + lsw32 first + little endian words, 16 bytes, non-aligned, 0xff", |
| PBUF(0x00, 0x00, 0x00, 0x00, 0x00, 0x1f, 0xff, 0xff, |
| 0xff, 0xff, 0xff, 0xff, 0xff, 0xe0, 0x00, 0x00), |
| .uval = 0xffffffffffffffff, |
| .start_bit = 106, |
| .end_bit = 43, |
| .quirks = QUIRK_MSB_ON_THE_RIGHT | QUIRK_LSW32_IS_FIRST | QUIRK_LITTLE_ENDIAN, |
| }, |
| }; |
| |
| KUNIT_ARRAY_PARAM_DESC(packing, cases, desc); |
| |
| static void packing_test_pack(struct kunit *test) |
| { |
| const struct packing_test_case *params = test->param_value; |
| u8 *pbuf; |
| int err; |
| |
| pbuf = kunit_kzalloc(test, params->pbuf_size, GFP_KERNEL); |
| KUNIT_ASSERT_NOT_NULL(test, pbuf); |
| |
| err = pack(pbuf, params->uval, params->start_bit, params->end_bit, |
| params->pbuf_size, params->quirks); |
| |
| KUNIT_EXPECT_EQ_MSG(test, err, 0, "pack() returned %pe\n", ERR_PTR(err)); |
| KUNIT_EXPECT_MEMEQ(test, pbuf, params->pbuf, params->pbuf_size); |
| } |
| |
| static void packing_test_unpack(struct kunit *test) |
| { |
| const struct packing_test_case *params = test->param_value; |
| u64 uval; |
| int err; |
| |
| err = unpack(params->pbuf, &uval, params->start_bit, params->end_bit, |
| params->pbuf_size, params->quirks); |
| KUNIT_EXPECT_EQ_MSG(test, err, 0, "unpack() returned %pe\n", ERR_PTR(err)); |
| KUNIT_EXPECT_EQ(test, uval, params->uval); |
| } |
| |
| static struct kunit_case packing_test_cases[] = { |
| KUNIT_CASE_PARAM(packing_test_pack, packing_gen_params), |
| KUNIT_CASE_PARAM(packing_test_unpack, packing_gen_params), |
| {}, |
| }; |
| |
| static struct kunit_suite packing_test_suite = { |
| .name = "packing", |
| .test_cases = packing_test_cases, |
| }; |
| |
| kunit_test_suite(packing_test_suite); |
| |
| MODULE_LICENSE("GPL"); |
| MODULE_DESCRIPTION("KUnit tests for packing library"); |