// SPDX-License-Identifier: GPL-2.0-only /* * KUnit tests for S1G TIM PVB decoding. This test suite covers * IEEE80211-2024 Annex L figures 8, 9, 10, 12, 13, 14. ADE mode * is not covered as it is an optional encoding format and is not * currently supported by mac80211. * * Copyright (C) 2025 Morse Micro */ #include #include #include #define MAX_AID 128 #define BC(enc_mode, inverse, blk_off) \ ((((blk_off) & 0x1f) << 3) | ((inverse) ? BIT(2) : 0) | \ ((enc_mode) & 0x3)) static void byte_to_bitstr(u8 v, char *out) { for (int b = 7; b >= 0; b--) *out++ = (v & BIT(b)) ? '1' : '0'; *out = '\0'; } static void dump_tim_bits(struct kunit *test, const struct ieee80211_tim_ie *tim, u8 tim_len) { const u8 *ptr = tim->virtual_map; const u8 *end = (const u8 *)tim + tim_len; unsigned int oct = 1; unsigned int blk = 0; char bits[9]; while (ptr < end) { u8 ctrl = *ptr++; u8 mode = ctrl & 0x03; bool inverse = ctrl & BIT(2); u8 blk_off = ctrl >> 3; kunit_info( test, "Block %u (ENC=%s, blk_off=%u, inverse=%u)", blk, (mode == IEEE80211_S1G_TIM_ENC_MODE_BLOCK) ? "BLOCK" : (mode == IEEE80211_S1G_TIM_ENC_MODE_SINGLE) ? "SINGLE" : "OLB", blk_off, inverse); byte_to_bitstr(ctrl, bits); kunit_info(test, " octet %2u (ctrl) : %s (0x%02x)", oct, bits, ctrl); ++oct; switch (mode) { case IEEE80211_S1G_TIM_ENC_MODE_BLOCK: { u8 blkmap = *ptr++; byte_to_bitstr(blkmap, bits); kunit_info(test, " octet %2u (blk-map) : %s (0x%02x)", oct, bits, blkmap); ++oct; for (u8 sb = 0; sb < 8; sb++) { if (!(blkmap & BIT(sb))) continue; u8 sub = *ptr++; byte_to_bitstr(sub, bits); kunit_info( test, " octet %2u (SB %2u) : %s (0x%02x)", oct, sb, bits, sub); ++oct; } break; } case IEEE80211_S1G_TIM_ENC_MODE_SINGLE: { u8 single = *ptr++; byte_to_bitstr(single, bits); kunit_info(test, " octet %2u (single) : %s (0x%02x)", oct, bits, single); ++oct; break; } case IEEE80211_S1G_TIM_ENC_MODE_OLB: { u8 len = *ptr++; byte_to_bitstr(len, bits); kunit_info(test, " octet %2u (len=%2u) : %s (0x%02x)", oct, len, bits, len); ++oct; for (u8 i = 0; i < len && ptr < end; i++) { u8 sub = *ptr++; byte_to_bitstr(sub, bits); kunit_info( test, " octet %2u (SB %2u) : %s (0x%02x)", oct, i, bits, sub); ++oct; } break; } default: kunit_info(test, " ** unknown encoding 0x%x **", mode); return; } blk++; } } static void tim_push(u8 **p, u8 v) { *(*p)++ = v; } static void tim_begin(struct ieee80211_tim_ie *tim, u8 **p) { tim->dtim_count = 0; tim->dtim_period = 1; tim->bitmap_ctrl = 0; *p = tim->virtual_map; } static u8 tim_end(struct ieee80211_tim_ie *tim, u8 *tail) { return tail - (u8 *)tim; } static void pvb_add_block_bitmap(u8 **p, u8 blk_off, bool inverse, u8 blk_bmap, const u8 *subblocks) { u8 enc = IEEE80211_S1G_TIM_ENC_MODE_BLOCK; u8 n = hweight8(blk_bmap); tim_push(p, BC(enc, inverse, blk_off)); tim_push(p, blk_bmap); for (u8 i = 0; i < n; i++) tim_push(p, subblocks[i]); } static void pvb_add_single_aid(u8 **p, u8 blk_off, bool inverse, u8 single6) { u8 enc = IEEE80211_S1G_TIM_ENC_MODE_SINGLE; tim_push(p, BC(enc, inverse, blk_off)); tim_push(p, single6 & GENMASK(5, 0)); } static void pvb_add_olb(u8 **p, u8 blk_off, bool inverse, const u8 *subblocks, u8 len) { u8 enc = IEEE80211_S1G_TIM_ENC_MODE_OLB; tim_push(p, BC(enc, inverse, blk_off)); tim_push(p, len); for (u8 i = 0; i < len; i++) tim_push(p, subblocks[i]); } static void check_all_aids(struct kunit *test, const struct ieee80211_tim_ie *tim, u8 tim_len, const unsigned long *expected) { for (u16 aid = 1; aid <= MAX_AID; aid++) { bool want = test_bit(aid, expected); bool got = ieee80211_s1g_check_tim(tim, tim_len, aid); KUNIT_ASSERT_EQ_MSG(test, got, want, "AID %u mismatch (got=%d want=%d)", aid, got, want); } } static void fill_bitmap(unsigned long *bm, const u16 *list, size_t n) { size_t i; bitmap_zero(bm, MAX_AID + 1); for (i = 0; i < n; i++) __set_bit(list[i], bm); } static void fill_bitmap_inverse(unsigned long *bm, u16 max_aid, const u16 *except, size_t n_except) { bitmap_zero(bm, MAX_AID + 1); for (u16 aid = 1; aid <= max_aid; aid++) __set_bit(aid, bm); for (size_t i = 0; i < n_except; i++) if (except[i] <= max_aid) __clear_bit(except[i], bm); } static void s1g_tim_block_test(struct kunit *test) { u8 buf[256] = {}; struct ieee80211_tim_ie *tim = (void *)buf; u8 *p, tim_len; static const u8 subblocks[] = { 0x42, /* SB m=0: AIDs 1,6 */ 0xA0, /* SB m=2: AIDs 21,23 */ }; u8 blk_bmap = 0x05; /* bits 0 and 2 set */ bool inverse = false; static const u16 set_list[] = { 1, 6, 21, 23 }; DECLARE_BITMAP(exp, MAX_AID + 1); tim_begin(tim, &p); pvb_add_block_bitmap(&p, 0, inverse, blk_bmap, subblocks); tim_len = tim_end(tim, p); fill_bitmap(exp, set_list, ARRAY_SIZE(set_list)); dump_tim_bits(test, tim, tim_len); check_all_aids(test, tim, tim_len, exp); } static void s1g_tim_single_test(struct kunit *test) { u8 buf[256] = {}; struct ieee80211_tim_ie *tim = (void *)buf; u8 *p, tim_len; bool inverse = false; u8 blk_off = 0; u8 single6 = 0x1f; /* 31 */ static const u16 set_list[] = { 31 }; DECLARE_BITMAP(exp, MAX_AID + 1); tim_begin(tim, &p); pvb_add_single_aid(&p, blk_off, inverse, single6); tim_len = tim_end(tim, p); fill_bitmap(exp, set_list, ARRAY_SIZE(set_list)); dump_tim_bits(test, tim, tim_len); check_all_aids(test, tim, tim_len, exp); } static void s1g_tim_olb_test(struct kunit *test) { u8 buf[256] = {}; struct ieee80211_tim_ie *tim = (void *)buf; u8 *p, tim_len; bool inverse = false; u8 blk_off = 0; static const u16 set_list[] = { 1, 6, 13, 15, 17, 22, 29, 31, 33, 38, 45, 47, 49, 54, 61, 63, 65, 70 }; static const u8 subblocks[] = { 0x42, 0xA0, 0x42, 0xA0, 0x42, 0xA0, 0x42, 0xA0, 0x42 }; u8 len = ARRAY_SIZE(subblocks); DECLARE_BITMAP(exp, MAX_AID + 1); tim_begin(tim, &p); pvb_add_olb(&p, blk_off, inverse, subblocks, len); tim_len = tim_end(tim, p); fill_bitmap(exp, set_list, ARRAY_SIZE(set_list)); dump_tim_bits(test, tim, tim_len); check_all_aids(test, tim, tim_len, exp); } static void s1g_tim_inverse_block_test(struct kunit *test) { u8 buf[256] = {}; struct ieee80211_tim_ie *tim = (void *)buf; u8 *p, tim_len; /* Same sub-block content as Figure L-8, but inverse = true */ static const u8 subblocks[] = { 0x42, /* SB m=0: AIDs 1,6 */ 0xA0, /* SB m=2: AIDs 21,23 */ }; u8 blk_bmap = 0x05; bool inverse = true; /* All AIDs except 1,6,21,23 are set */ static const u16 except[] = { 1, 6, 21, 23 }; DECLARE_BITMAP(exp, MAX_AID + 1); tim_begin(tim, &p); pvb_add_block_bitmap(&p, 0, inverse, blk_bmap, subblocks); tim_len = tim_end(tim, p); fill_bitmap_inverse(exp, 63, except, ARRAY_SIZE(except)); dump_tim_bits(test, tim, tim_len); check_all_aids(test, tim, tim_len, exp); } static void s1g_tim_inverse_single_test(struct kunit *test) { u8 buf[256] = {}; struct ieee80211_tim_ie *tim = (void *)buf; u8 *p, tim_len; bool inverse = true; u8 blk_off = 0; u8 single6 = 0x1f; /* 31 */ /* All AIDs except 31 are set */ static const u16 except[] = { 31 }; DECLARE_BITMAP(exp, MAX_AID + 1); tim_begin(tim, &p); pvb_add_single_aid(&p, blk_off, inverse, single6); tim_len = tim_end(tim, p); fill_bitmap_inverse(exp, 63, except, ARRAY_SIZE(except)); dump_tim_bits(test, tim, tim_len); check_all_aids(test, tim, tim_len, exp); } static void s1g_tim_inverse_olb_test(struct kunit *test) { u8 buf[256] = {}; struct ieee80211_tim_ie *tim = (void *)buf; u8 *p, tim_len; bool inverse = true; u8 blk_off = 0, len; /* All AIDs except the list below are set */ static const u16 except[] = { 1, 6, 13, 15, 17, 22, 29, 31, 33, 38, 45, 47, 49, 54, 61, 63, 65, 70 }; static const u8 subblocks[] = { 0x42, 0xA0, 0x42, 0xA0, 0x42, 0xA0, 0x42, 0xA0, 0x42 }; len = ARRAY_SIZE(subblocks); DECLARE_BITMAP(exp, MAX_AID + 1); tim_begin(tim, &p); pvb_add_olb(&p, blk_off, inverse, subblocks, len); tim_len = tim_end(tim, p); fill_bitmap_inverse(exp, 127, except, ARRAY_SIZE(except)); dump_tim_bits(test, tim, tim_len); check_all_aids(test, tim, tim_len, exp); } static struct kunit_case s1g_tim_test_cases[] = { KUNIT_CASE(s1g_tim_block_test), KUNIT_CASE(s1g_tim_single_test), KUNIT_CASE(s1g_tim_olb_test), KUNIT_CASE(s1g_tim_inverse_block_test), KUNIT_CASE(s1g_tim_inverse_single_test), KUNIT_CASE(s1g_tim_inverse_olb_test), {} }; static struct kunit_suite s1g_tim = { .name = "mac80211-s1g-tim", .test_cases = s1g_tim_test_cases, }; kunit_test_suite(s1g_tim);