| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474 |
- // 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);
- }
- #define PACKED_BUF_SIZE 8
- typedef struct __packed { u8 buf[PACKED_BUF_SIZE]; } packed_buf_t;
- struct test_data {
- u32 field3;
- u16 field2;
- u16 field4;
- u16 field6;
- u8 field1;
- u8 field5;
- };
- static const struct packed_field_u8 test_fields[] = {
- PACKED_FIELD(63, 61, struct test_data, field1),
- PACKED_FIELD(60, 52, struct test_data, field2),
- PACKED_FIELD(51, 28, struct test_data, field3),
- PACKED_FIELD(27, 14, struct test_data, field4),
- PACKED_FIELD(13, 9, struct test_data, field5),
- PACKED_FIELD(8, 0, struct test_data, field6),
- };
- static void packing_test_pack_fields(struct kunit *test)
- {
- const struct test_data data = {
- .field1 = 0x2,
- .field2 = 0x100,
- .field3 = 0xF00050,
- .field4 = 0x7D3,
- .field5 = 0x9,
- .field6 = 0x10B,
- };
- packed_buf_t expect = {
- .buf = { 0x50, 0x0F, 0x00, 0x05, 0x01, 0xF4, 0xD3, 0x0B },
- };
- packed_buf_t buf = {};
- pack_fields(&buf, sizeof(buf), &data, test_fields, 0);
- KUNIT_EXPECT_MEMEQ(test, &expect, &buf, sizeof(buf));
- }
- static void packing_test_unpack_fields(struct kunit *test)
- {
- const packed_buf_t buf = {
- .buf = { 0x17, 0x28, 0x10, 0x19, 0x3D, 0xA9, 0x07, 0x9C },
- };
- struct test_data data = {};
- unpack_fields(&buf, sizeof(buf), &data, test_fields, 0);
- KUNIT_EXPECT_EQ(test, 0, data.field1);
- KUNIT_EXPECT_EQ(test, 0x172, data.field2);
- KUNIT_EXPECT_EQ(test, 0x810193, data.field3);
- KUNIT_EXPECT_EQ(test, 0x36A4, data.field4);
- KUNIT_EXPECT_EQ(test, 0x3, data.field5);
- KUNIT_EXPECT_EQ(test, 0x19C, data.field6);
- }
- 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),
- KUNIT_CASE(packing_test_pack_fields),
- KUNIT_CASE(packing_test_unpack_fields),
- {},
- };
- 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");
|