A CAT allocation is a CBM, and the kernel must refuse masks that can't
describe a valid allocation. Nothing in the suite check this.

Implement L3_CAT_VALIDATE, first confirm a valid full mask is accepted,
then write an empty mask and a mask with bits outside cbm_mask and
confirm each is refused. A mask below min_cbm_bits is also checked where
the platform needs more than one bit.

Signed-off-by: Richard Cheng <[email protected]>
---
 tools/testing/selftests/resctrl/cat_test.c    | 79 +++++++++++++++++++
 tools/testing/selftests/resctrl/resctrl.h     |  1 +
 .../testing/selftests/resctrl/resctrl_tests.c |  1 +
 3 files changed, 81 insertions(+)

diff --git a/tools/testing/selftests/resctrl/cat_test.c 
b/tools/testing/selftests/resctrl/cat_test.c
index 16a947f1ed16..d236988916d9 100644
--- a/tools/testing/selftests/resctrl/cat_test.c
+++ b/tools/testing/selftests/resctrl/cat_test.c
@@ -603,3 +603,82 @@ struct resctrl_test l3_cat_occup_test = {
        .run_test = cat_occup_run_test,
        .cleanup = cat_occup_test_cleanup,
 };
+
+/*
+ * L3_CAT_VALIDATE - Verify the kernel rejects invalid L3 CBM writes.
+ *
+ * CAT allocation accepts a Capacity Bit Mask (CBM). The kernel must reject
+ * masks that cannot represent a valid allocation: an empty mask, a mask with
+ * bits outside cbm_mask, or (where the platform requires it) a mask with
+ * fewer than min_cbm_bits bits. A rejected schemata write makes
+ * write_schemata() return non-zero, which is the signal this test checks.
+ */
+static int cat_validate_run_test(const struct resctrl_test *test,
+                                const struct user_params *uparams)
+{
+       unsigned long full_mask, bad_mask;
+       unsigned int min_cbm_bits, start;
+       char schemata[64];
+       int count_of_bits, ret;
+
+       ret = get_full_cbm(test->resource, &full_mask);
+       if (ret)
+               return ret;
+
+       ret = resource_info_unsigned_get(test->resource, "min_cbm_bits",
+                                        &min_cbm_bits);
+       if (ret)
+               return ret;
+
+       count_of_bits = count_bits(full_mask);
+
+       /* A valid full CBM must be accepted */
+       snprintf(schemata, sizeof(schemata), "%lx", full_mask);
+       if (write_schemata("", schemata, uparams->cpu, test->resource)) {
+               ksft_print_msg("Valid CBM 0x%lx was rejected\n", full_mask);
+               return 1;
+       }
+
+       /* An empty mask must be rejected. */
+       if (!write_schemata("", "0", uparams->cpu, test->resource)) {
+               ksft_print_msg("Empty CBM was accepted, must be rejected\n");
+               return 1;
+       }
+
+       /* A mask with a bit outside cbm_mask must be rejected. */
+       bad_mask = full_mask | (1UL << count_of_bits);
+       snprintf(schemata, sizeof(schemata), "%lx", bad_mask);
+       if (!write_schemata("", schemata, uparams->cpu, test->resource)) {
+               ksft_print_msg("Out-of-range CBM 0x%lx was accepted, must be 
rejected\n",
+                              bad_mask);
+               return 1;
+       }
+
+       /*
+        * A mask with fewer than min_cbm_bits bits must be rejected. When
+        * min_cbm_bits is 1 the only sub-minimum mask is the empty mask
+        * This case only applies when the platform requires more contiguous 
bits.
+        */
+       if (min_cbm_bits > 1) {
+               count_contiguous_bits(full_mask, &start);
+               bad_mask = create_bit_mask(start, min_cbm_bits - 1);
+               snprintf(schemata, sizeof(schemata), "%lx", bad_mask);
+               if (!write_schemata("", schemata, uparams->cpu, 
test->resource)) {
+                       ksft_print_msg("CBM 0x%lx with too few bits was 
accepted\n",
+                                      bad_mask);
+                       return 1;
+               }
+       }
+
+       ksft_print_msg("Pass: CAT rejects invalid CBM writes\n");
+
+       return 0;
+}
+
+struct resctrl_test l3_cat_validate_test = {
+       .name = "L3_CAT_VALIDATE",
+       .group = "CAT",
+       .resource = "L3",
+       .feature_check = test_resource_feature_check,
+       .run_test = cat_validate_run_test,
+};
diff --git a/tools/testing/selftests/resctrl/resctrl.h 
b/tools/testing/selftests/resctrl/resctrl.h
index ce3abf0bdac2..e2e3cf7833bc 100644
--- a/tools/testing/selftests/resctrl/resctrl.h
+++ b/tools/testing/selftests/resctrl/resctrl.h
@@ -248,6 +248,7 @@ extern struct resctrl_test mba_test;
 extern struct resctrl_test cmt_test;
 extern struct resctrl_test l3_cat_test;
 extern struct resctrl_test l3_cat_occup_test;
+extern struct resctrl_test l3_cat_validate_test;
 extern struct resctrl_test l3_noncont_cat_test;
 extern struct resctrl_test l2_noncont_cat_test;
 
diff --git a/tools/testing/selftests/resctrl/resctrl_tests.c 
b/tools/testing/selftests/resctrl/resctrl_tests.c
index 324a60818aa1..66739e96f33c 100644
--- a/tools/testing/selftests/resctrl/resctrl_tests.c
+++ b/tools/testing/selftests/resctrl/resctrl_tests.c
@@ -20,6 +20,7 @@ static struct resctrl_test *resctrl_tests[] = {
        &cmt_test,
        &l3_cat_test,
        &l3_cat_occup_test,
+       &l3_cat_validate_test,
        &l3_noncont_cat_test,
        &l2_noncont_cat_test,
 };
-- 
2.43.0


Reply via email to