* [PATCH v2 0/3] Move io-pgtable-arm selftest to KUnit
@ 2025-09-17 19:11 Mostafa Saleh
2025-09-17 19:11 ` [PATCH v2 1/3] iommu/io-pgtable-arm: Move selftests to a separate file Mostafa Saleh
` (3 more replies)
0 siblings, 4 replies; 12+ messages in thread
From: Mostafa Saleh @ 2025-09-17 19:11 UTC (permalink / raw)
To: iommu, linux-arm-kernel, linux-kernel
Cc: robin.murphy, will, joro, jgg, praan, Mostafa Saleh
This is a small series to clean up the io-pgtable-arm library.
The first patch was originally part of the SMMUv3 KVM driver support[1],
which needed to factor out the kernel code from the io-pgtable-arm
library, and based on Jason’s suggestion this can be taken out as a
cleanup, and a step further to convert it to kunit.
The first patch just moves the code to a new file with no other changes,
so it can be easier to review with “--color-moved”
The second patch converts the sefltest to be modular, that is useful as
kunit can be a module and it can run anytime after boot.
The third patch registers the test using kunit, and converges some of
the APIs, some notes about that:
Granularity of tests:
---------------------
To make the series easier to review, the series changes the test to run in
kunit without making intrusive changes to the test itself.
It’s possible to refactor the tests to have smaller granularity (although
I think that would make it less efficient as we might create the same io-pgtable
config multiple times) and integrate them in kunit as multiple tests, that
change would be more intrusive, if you think that is the right approach,
I can add a couple of more patches re-writing the tests.
Other changes:
--------------
- Also, to make the test changes minimal, and the fail messages similar,
“KUNIT_FAIL()” is used to fail all tests instead of using KUNIT specific
assertions.
- Instead of using faux device, we rely on kunit_device_register()
- The WARN is removed when a test fails, as that doesn’t seem to be a pattern
used with kunit.
Testing:
--------
- With CONFIG_IOMMU_IO_PGTABLE_LPAE_SELFTEST=y
It will automatically run at boot as before this patch series.
- Otherwise with CONFIG_IOMMU_IO_PGTABLE_LPAE_SELFTEST=m:
1) on module load:
Once the module load the self test will run
# modprobe io-pgtable-arm-selftests
2) debugfs
With CONFIG_KUNIT_DEBUGFS=y You can run the test with
# echo 1 > /sys/kernel/debug/kunit/io-pgtable-arm-test/run
3) Using kunit.py
You can also use the helper script which uses Qemu in the background
# ./tools/testing/kunit/kunit.py run --build_dir build_kunit_arm64 --arch arm64 \
--make_options LLVM=1 --kunitconfig ./kunit/kunitconfig
[18:01:09] ============= io-pgtable-arm-test (1 subtest) ==============
[18:01:09] [PASSED] arm_lpae_do_selftests
[18:01:09] =============== [PASSED] io-pgtable-arm-test ===============
[18:01:09] ============================================================
A failure at the test with the new implementation look as
[ 66.624469] # arm_lpae_do_selftests: EXPECTATION FAILED at drivers/iommu/io-pgtable-arm-selftests.c:154
[ 66.624505] # arm_lpae_do_selftests: selftest: test failed for fmt idx 0
[ 66.624524] # arm_lpae_do_selftests: cfg: pgsize_bitmap 0x20010000, ias 48-bit
[ 66.624669] # arm_lpae_do_selftests: data: 3 levels, 0x200 pgd_size, 16 pg_shift, 13 bits_per_level, pgd @ 00000000f200237d
Main changes in v2:
-------------------
- Make the test modular
v1: https://lore.kernel.org/linux-iommu/20250917140216.2199055-1-smostafa@google.com/
[1] https://lore.kernel.org/all/20250819215156.2494305-5-smostafa@google.com/
Mostafa Saleh (3):
iommu/io-pgtable-arm: Move selftests to a separate file
iommu/io-pgtable-arm-selftests: Modularize the test
iommu/io-pgtable-arm-selftests: Use KUnit
drivers/iommu/Kconfig | 4 +-
drivers/iommu/Makefile | 1 +
drivers/iommu/io-pgtable-arm-selftests.c | 226 +++++++++++++++++++++
drivers/iommu/io-pgtable-arm.c | 243 -----------------------
drivers/iommu/io-pgtable-arm.h | 41 ++++
5 files changed, 270 insertions(+), 245 deletions(-)
create mode 100644 drivers/iommu/io-pgtable-arm-selftests.c
--
2.51.0.384.g4c02a37b29-goog
^ permalink raw reply [flat|nested] 12+ messages in thread* [PATCH v2 1/3] iommu/io-pgtable-arm: Move selftests to a separate file 2025-09-17 19:11 [PATCH v2 0/3] Move io-pgtable-arm selftest to KUnit Mostafa Saleh @ 2025-09-17 19:11 ` Mostafa Saleh 2025-09-17 19:38 ` Jason Gunthorpe 2025-09-17 19:11 ` [PATCH v2 2/3] iommu/io-pgtable-arm-selftests: Modularize the test Mostafa Saleh ` (2 subsequent siblings) 3 siblings, 1 reply; 12+ messages in thread From: Mostafa Saleh @ 2025-09-17 19:11 UTC (permalink / raw) To: iommu, linux-arm-kernel, linux-kernel Cc: robin.murphy, will, joro, jgg, praan, Mostafa Saleh Clean up the io-pgtable-arm library by moving the selftests out. Next the tests will be registered with kunit. This is useful also to factor out kernel specific code out, so it can compiled as part of the hypervisor object. Signed-off-by: Mostafa Saleh <smostafa@google.com> --- drivers/iommu/Makefile | 1 + drivers/iommu/io-pgtable-arm-selftests.c | 211 ++++++++++++++++++++ drivers/iommu/io-pgtable-arm.c | 243 ----------------------- drivers/iommu/io-pgtable-arm.h | 41 ++++ 4 files changed, 253 insertions(+), 243 deletions(-) create mode 100644 drivers/iommu/io-pgtable-arm-selftests.c diff --git a/drivers/iommu/Makefile b/drivers/iommu/Makefile index 355294fa9033..5250a2eea13f 100644 --- a/drivers/iommu/Makefile +++ b/drivers/iommu/Makefile @@ -12,6 +12,7 @@ obj-$(CONFIG_IOMMU_DMA) += dma-iommu.o obj-$(CONFIG_IOMMU_IO_PGTABLE) += io-pgtable.o obj-$(CONFIG_IOMMU_IO_PGTABLE_ARMV7S) += io-pgtable-arm-v7s.o obj-$(CONFIG_IOMMU_IO_PGTABLE_LPAE) += io-pgtable-arm.o +obj-$(CONFIG_IOMMU_IO_PGTABLE_LPAE_SELFTEST) += io-pgtable-arm-selftests.o obj-$(CONFIG_IOMMU_IO_PGTABLE_DART) += io-pgtable-dart.o obj-$(CONFIG_IOMMU_IOVA) += iova.o obj-$(CONFIG_OF_IOMMU) += of_iommu.o diff --git a/drivers/iommu/io-pgtable-arm-selftests.c b/drivers/iommu/io-pgtable-arm-selftests.c new file mode 100644 index 000000000000..f7746ff2c7a0 --- /dev/null +++ b/drivers/iommu/io-pgtable-arm-selftests.c @@ -0,0 +1,211 @@ +// SPDX-License-Identifier: GPL-2.0-only +/* + * CPU-agnostic ARM page table allocator. + * + * Copyright (C) 2014 ARM Limited + * + * Author: Will Deacon <will.deacon@arm.com> + */ +#include <linux/device/faux.h> +#include <linux/kernel.h> +#include <linux/slab.h> + +#include "io-pgtable-arm.h" + +static struct io_pgtable_cfg *cfg_cookie __initdata; + +static void __init dummy_tlb_flush_all(void *cookie) +{ + WARN_ON(cookie != cfg_cookie); +} + +static void __init dummy_tlb_flush(unsigned long iova, size_t size, + size_t granule, void *cookie) +{ + WARN_ON(cookie != cfg_cookie); + WARN_ON(!(size & cfg_cookie->pgsize_bitmap)); +} + +static void __init dummy_tlb_add_page(struct iommu_iotlb_gather *gather, + unsigned long iova, size_t granule, + void *cookie) +{ + dummy_tlb_flush(iova, granule, granule, cookie); +} + +static const struct iommu_flush_ops dummy_tlb_ops __initconst = { + .tlb_flush_all = dummy_tlb_flush_all, + .tlb_flush_walk = dummy_tlb_flush, + .tlb_add_page = dummy_tlb_add_page, +}; + +static void __init arm_lpae_dump_ops(struct io_pgtable_ops *ops) +{ + struct arm_lpae_io_pgtable *data = io_pgtable_ops_to_data(ops); + struct io_pgtable_cfg *cfg = &data->iop.cfg; + + pr_err("cfg: pgsize_bitmap 0x%lx, ias %u-bit\n", + cfg->pgsize_bitmap, cfg->ias); + pr_err("data: %d levels, 0x%zx pgd_size, %u pg_shift, %u bits_per_level, pgd @ %p\n", + ARM_LPAE_MAX_LEVELS - data->start_level, ARM_LPAE_PGD_SIZE(data), + ilog2(ARM_LPAE_GRANULE(data)), data->bits_per_level, data->pgd); +} + +#define __FAIL(ops, i) ({ \ + WARN(1, "selftest: test failed for fmt idx %d\n", (i)); \ + arm_lpae_dump_ops(ops); \ + -EFAULT; \ +}) + +static int __init arm_lpae_run_tests(struct io_pgtable_cfg *cfg) +{ + static const enum io_pgtable_fmt fmts[] __initconst = { + ARM_64_LPAE_S1, + ARM_64_LPAE_S2, + }; + + int i, j; + unsigned long iova; + size_t size, mapped; + struct io_pgtable_ops *ops; + + for (i = 0; i < ARRAY_SIZE(fmts); ++i) { + cfg_cookie = cfg; + ops = alloc_io_pgtable_ops(fmts[i], cfg, cfg); + if (!ops) { + pr_err("selftest: failed to allocate io pgtable ops\n"); + return -ENOMEM; + } + + /* + * Initial sanity checks. + * Empty page tables shouldn't provide any translations. + */ + if (ops->iova_to_phys(ops, 42)) + return __FAIL(ops, i); + + if (ops->iova_to_phys(ops, SZ_1G + 42)) + return __FAIL(ops, i); + + if (ops->iova_to_phys(ops, SZ_2G + 42)) + return __FAIL(ops, i); + + /* + * Distinct mappings of different granule sizes. + */ + iova = 0; + for_each_set_bit(j, &cfg->pgsize_bitmap, BITS_PER_LONG) { + size = 1UL << j; + + if (ops->map_pages(ops, iova, iova, size, 1, + IOMMU_READ | IOMMU_WRITE | + IOMMU_NOEXEC | IOMMU_CACHE, + GFP_KERNEL, &mapped)) + return __FAIL(ops, i); + + /* Overlapping mappings */ + if (!ops->map_pages(ops, iova, iova + size, size, 1, + IOMMU_READ | IOMMU_NOEXEC, + GFP_KERNEL, &mapped)) + return __FAIL(ops, i); + + if (ops->iova_to_phys(ops, iova + 42) != (iova + 42)) + return __FAIL(ops, i); + + iova += SZ_1G; + } + + /* Full unmap */ + iova = 0; + for_each_set_bit(j, &cfg->pgsize_bitmap, BITS_PER_LONG) { + size = 1UL << j; + + if (ops->unmap_pages(ops, iova, size, 1, NULL) != size) + return __FAIL(ops, i); + + if (ops->iova_to_phys(ops, iova + 42)) + return __FAIL(ops, i); + + /* Remap full block */ + if (ops->map_pages(ops, iova, iova, size, 1, + IOMMU_WRITE, GFP_KERNEL, &mapped)) + return __FAIL(ops, i); + + if (ops->iova_to_phys(ops, iova + 42) != (iova + 42)) + return __FAIL(ops, i); + + iova += SZ_1G; + } + + /* + * Map/unmap the last largest supported page of the IAS, this can + * trigger corner cases in the concatednated page tables. + */ + mapped = 0; + size = 1UL << __fls(cfg->pgsize_bitmap); + iova = (1UL << cfg->ias) - size; + if (ops->map_pages(ops, iova, iova, size, 1, + IOMMU_READ | IOMMU_WRITE | + IOMMU_NOEXEC | IOMMU_CACHE, + GFP_KERNEL, &mapped)) + return __FAIL(ops, i); + if (mapped != size) + return __FAIL(ops, i); + if (ops->unmap_pages(ops, iova, size, 1, NULL) != size) + return __FAIL(ops, i); + + free_io_pgtable_ops(ops); + } + + return 0; +} + +static int __init arm_lpae_do_selftests(void) +{ + static const unsigned long pgsize[] __initconst = { + SZ_4K | SZ_2M | SZ_1G, + SZ_16K | SZ_32M, + SZ_64K | SZ_512M, + }; + + static const unsigned int address_size[] __initconst = { + 32, 36, 40, 42, 44, 48, + }; + + int i, j, k, pass = 0, fail = 0; + struct faux_device *dev; + struct io_pgtable_cfg cfg = { + .tlb = &dummy_tlb_ops, + .coherent_walk = true, + .quirks = IO_PGTABLE_QUIRK_NO_WARN, + }; + + dev = faux_device_create("io-pgtable-test", NULL, 0); + if (!dev) + return -ENOMEM; + + cfg.iommu_dev = &dev->dev; + + for (i = 0; i < ARRAY_SIZE(pgsize); ++i) { + for (j = 0; j < ARRAY_SIZE(address_size); ++j) { + /* Don't use ias > oas as it is not valid for stage-2. */ + for (k = 0; k <= j; ++k) { + cfg.pgsize_bitmap = pgsize[i]; + cfg.ias = address_size[k]; + cfg.oas = address_size[j]; + pr_info("selftest: pgsize_bitmap 0x%08lx, IAS %u OAS %u\n", + pgsize[i], cfg.ias, cfg.oas); + if (arm_lpae_run_tests(&cfg)) + fail++; + else + pass++; + } + } + } + + pr_info("selftest: completed with %d PASS %d FAIL\n", pass, fail); + faux_device_destroy(dev); + + return fail ? -EFAULT : 0; +} +subsys_initcall(arm_lpae_do_selftests); diff --git a/drivers/iommu/io-pgtable-arm.c b/drivers/iommu/io-pgtable-arm.c index 7e8e2216c294..b1d968edccd6 100644 --- a/drivers/iommu/io-pgtable-arm.c +++ b/drivers/iommu/io-pgtable-arm.c @@ -12,10 +12,7 @@ #include <linux/atomic.h> #include <linux/bitops.h> #include <linux/io-pgtable.h> -#include <linux/kernel.h> -#include <linux/device/faux.h> #include <linux/sizes.h> -#include <linux/slab.h> #include <linux/types.h> #include <linux/dma-mapping.h> @@ -24,33 +21,6 @@ #include "io-pgtable-arm.h" #include "iommu-pages.h" -#define ARM_LPAE_MAX_ADDR_BITS 52 -#define ARM_LPAE_S2_MAX_CONCAT_PAGES 16 -#define ARM_LPAE_MAX_LEVELS 4 - -/* Struct accessors */ -#define io_pgtable_to_data(x) \ - container_of((x), struct arm_lpae_io_pgtable, iop) - -#define io_pgtable_ops_to_data(x) \ - io_pgtable_to_data(io_pgtable_ops_to_pgtable(x)) - -/* - * Calculate the right shift amount to get to the portion describing level l - * in a virtual address mapped by the pagetable in d. - */ -#define ARM_LPAE_LVL_SHIFT(l,d) \ - (((ARM_LPAE_MAX_LEVELS - (l)) * (d)->bits_per_level) + \ - ilog2(sizeof(arm_lpae_iopte))) - -#define ARM_LPAE_GRANULE(d) \ - (sizeof(arm_lpae_iopte) << (d)->bits_per_level) -#define ARM_LPAE_PGD_SIZE(d) \ - (sizeof(arm_lpae_iopte) << (d)->pgd_bits) - -#define ARM_LPAE_PTES_PER_TABLE(d) \ - (ARM_LPAE_GRANULE(d) >> ilog2(sizeof(arm_lpae_iopte))) - /* * Calculate the index at level l used to map virtual address a using the * pagetable in d. @@ -156,18 +126,6 @@ #define iopte_set_writeable_clean(ptep) \ set_bit(ARM_LPAE_PTE_AP_RDONLY_BIT, (unsigned long *)(ptep)) -struct arm_lpae_io_pgtable { - struct io_pgtable iop; - - int pgd_bits; - int start_level; - int bits_per_level; - - void *pgd; -}; - -typedef u64 arm_lpae_iopte; - static inline bool iopte_leaf(arm_lpae_iopte pte, int lvl, enum io_pgtable_fmt fmt) { @@ -1267,204 +1225,3 @@ struct io_pgtable_init_fns io_pgtable_arm_mali_lpae_init_fns = { .alloc = arm_mali_lpae_alloc_pgtable, .free = arm_lpae_free_pgtable, }; - -#ifdef CONFIG_IOMMU_IO_PGTABLE_LPAE_SELFTEST - -static struct io_pgtable_cfg *cfg_cookie __initdata; - -static void __init dummy_tlb_flush_all(void *cookie) -{ - WARN_ON(cookie != cfg_cookie); -} - -static void __init dummy_tlb_flush(unsigned long iova, size_t size, - size_t granule, void *cookie) -{ - WARN_ON(cookie != cfg_cookie); - WARN_ON(!(size & cfg_cookie->pgsize_bitmap)); -} - -static void __init dummy_tlb_add_page(struct iommu_iotlb_gather *gather, - unsigned long iova, size_t granule, - void *cookie) -{ - dummy_tlb_flush(iova, granule, granule, cookie); -} - -static const struct iommu_flush_ops dummy_tlb_ops __initconst = { - .tlb_flush_all = dummy_tlb_flush_all, - .tlb_flush_walk = dummy_tlb_flush, - .tlb_add_page = dummy_tlb_add_page, -}; - -static void __init arm_lpae_dump_ops(struct io_pgtable_ops *ops) -{ - struct arm_lpae_io_pgtable *data = io_pgtable_ops_to_data(ops); - struct io_pgtable_cfg *cfg = &data->iop.cfg; - - pr_err("cfg: pgsize_bitmap 0x%lx, ias %u-bit\n", - cfg->pgsize_bitmap, cfg->ias); - pr_err("data: %d levels, 0x%zx pgd_size, %u pg_shift, %u bits_per_level, pgd @ %p\n", - ARM_LPAE_MAX_LEVELS - data->start_level, ARM_LPAE_PGD_SIZE(data), - ilog2(ARM_LPAE_GRANULE(data)), data->bits_per_level, data->pgd); -} - -#define __FAIL(ops, i) ({ \ - WARN(1, "selftest: test failed for fmt idx %d\n", (i)); \ - arm_lpae_dump_ops(ops); \ - -EFAULT; \ -}) - -static int __init arm_lpae_run_tests(struct io_pgtable_cfg *cfg) -{ - static const enum io_pgtable_fmt fmts[] __initconst = { - ARM_64_LPAE_S1, - ARM_64_LPAE_S2, - }; - - int i, j; - unsigned long iova; - size_t size, mapped; - struct io_pgtable_ops *ops; - - for (i = 0; i < ARRAY_SIZE(fmts); ++i) { - cfg_cookie = cfg; - ops = alloc_io_pgtable_ops(fmts[i], cfg, cfg); - if (!ops) { - pr_err("selftest: failed to allocate io pgtable ops\n"); - return -ENOMEM; - } - - /* - * Initial sanity checks. - * Empty page tables shouldn't provide any translations. - */ - if (ops->iova_to_phys(ops, 42)) - return __FAIL(ops, i); - - if (ops->iova_to_phys(ops, SZ_1G + 42)) - return __FAIL(ops, i); - - if (ops->iova_to_phys(ops, SZ_2G + 42)) - return __FAIL(ops, i); - - /* - * Distinct mappings of different granule sizes. - */ - iova = 0; - for_each_set_bit(j, &cfg->pgsize_bitmap, BITS_PER_LONG) { - size = 1UL << j; - - if (ops->map_pages(ops, iova, iova, size, 1, - IOMMU_READ | IOMMU_WRITE | - IOMMU_NOEXEC | IOMMU_CACHE, - GFP_KERNEL, &mapped)) - return __FAIL(ops, i); - - /* Overlapping mappings */ - if (!ops->map_pages(ops, iova, iova + size, size, 1, - IOMMU_READ | IOMMU_NOEXEC, - GFP_KERNEL, &mapped)) - return __FAIL(ops, i); - - if (ops->iova_to_phys(ops, iova + 42) != (iova + 42)) - return __FAIL(ops, i); - - iova += SZ_1G; - } - - /* Full unmap */ - iova = 0; - for_each_set_bit(j, &cfg->pgsize_bitmap, BITS_PER_LONG) { - size = 1UL << j; - - if (ops->unmap_pages(ops, iova, size, 1, NULL) != size) - return __FAIL(ops, i); - - if (ops->iova_to_phys(ops, iova + 42)) - return __FAIL(ops, i); - - /* Remap full block */ - if (ops->map_pages(ops, iova, iova, size, 1, - IOMMU_WRITE, GFP_KERNEL, &mapped)) - return __FAIL(ops, i); - - if (ops->iova_to_phys(ops, iova + 42) != (iova + 42)) - return __FAIL(ops, i); - - iova += SZ_1G; - } - - /* - * Map/unmap the last largest supported page of the IAS, this can - * trigger corner cases in the concatednated page tables. - */ - mapped = 0; - size = 1UL << __fls(cfg->pgsize_bitmap); - iova = (1UL << cfg->ias) - size; - if (ops->map_pages(ops, iova, iova, size, 1, - IOMMU_READ | IOMMU_WRITE | - IOMMU_NOEXEC | IOMMU_CACHE, - GFP_KERNEL, &mapped)) - return __FAIL(ops, i); - if (mapped != size) - return __FAIL(ops, i); - if (ops->unmap_pages(ops, iova, size, 1, NULL) != size) - return __FAIL(ops, i); - - free_io_pgtable_ops(ops); - } - - return 0; -} - -static int __init arm_lpae_do_selftests(void) -{ - static const unsigned long pgsize[] __initconst = { - SZ_4K | SZ_2M | SZ_1G, - SZ_16K | SZ_32M, - SZ_64K | SZ_512M, - }; - - static const unsigned int address_size[] __initconst = { - 32, 36, 40, 42, 44, 48, - }; - - int i, j, k, pass = 0, fail = 0; - struct faux_device *dev; - struct io_pgtable_cfg cfg = { - .tlb = &dummy_tlb_ops, - .coherent_walk = true, - .quirks = IO_PGTABLE_QUIRK_NO_WARN, - }; - - dev = faux_device_create("io-pgtable-test", NULL, 0); - if (!dev) - return -ENOMEM; - - cfg.iommu_dev = &dev->dev; - - for (i = 0; i < ARRAY_SIZE(pgsize); ++i) { - for (j = 0; j < ARRAY_SIZE(address_size); ++j) { - /* Don't use ias > oas as it is not valid for stage-2. */ - for (k = 0; k <= j; ++k) { - cfg.pgsize_bitmap = pgsize[i]; - cfg.ias = address_size[k]; - cfg.oas = address_size[j]; - pr_info("selftest: pgsize_bitmap 0x%08lx, IAS %u OAS %u\n", - pgsize[i], cfg.ias, cfg.oas); - if (arm_lpae_run_tests(&cfg)) - fail++; - else - pass++; - } - } - } - - pr_info("selftest: completed with %d PASS %d FAIL\n", pass, fail); - faux_device_destroy(dev); - - return fail ? -EFAULT : 0; -} -subsys_initcall(arm_lpae_do_selftests); -#endif diff --git a/drivers/iommu/io-pgtable-arm.h b/drivers/iommu/io-pgtable-arm.h index ba7cfdf7afa0..a06a23543cff 100644 --- a/drivers/iommu/io-pgtable-arm.h +++ b/drivers/iommu/io-pgtable-arm.h @@ -2,6 +2,8 @@ #ifndef IO_PGTABLE_ARM_H_ #define IO_PGTABLE_ARM_H_ +#include <linux/io-pgtable.h> + #define ARM_LPAE_TCR_TG0_4K 0 #define ARM_LPAE_TCR_TG0_64K 1 #define ARM_LPAE_TCR_TG0_16K 2 @@ -27,4 +29,43 @@ #define ARM_LPAE_TCR_PS_48_BIT 0x5ULL #define ARM_LPAE_TCR_PS_52_BIT 0x6ULL +/* Struct accessors */ +#define io_pgtable_to_data(x) \ + container_of((x), struct arm_lpae_io_pgtable, iop) + +#define io_pgtable_ops_to_data(x) \ + io_pgtable_to_data(io_pgtable_ops_to_pgtable(x)) + +struct arm_lpae_io_pgtable { + struct io_pgtable iop; + + int pgd_bits; + int start_level; + int bits_per_level; + + void *pgd; +}; + +#define ARM_LPAE_MAX_ADDR_BITS 52 +#define ARM_LPAE_S2_MAX_CONCAT_PAGES 16 +#define ARM_LPAE_MAX_LEVELS 4 + +/* + * Calculate the right shift amount to get to the portion describing level l + * in a virtual address mapped by the pagetable in d. + */ +#define ARM_LPAE_LVL_SHIFT(l,d) \ + (((ARM_LPAE_MAX_LEVELS - (l)) * (d)->bits_per_level) + \ + ilog2(sizeof(arm_lpae_iopte))) + +#define ARM_LPAE_GRANULE(d) \ + (sizeof(arm_lpae_iopte) << (d)->bits_per_level) +#define ARM_LPAE_PGD_SIZE(d) \ + (sizeof(arm_lpae_iopte) << (d)->pgd_bits) + +#define ARM_LPAE_PTES_PER_TABLE(d) \ + (ARM_LPAE_GRANULE(d) >> ilog2(sizeof(arm_lpae_iopte))) + +typedef u64 arm_lpae_iopte; + #endif /* IO_PGTABLE_ARM_H_ */ -- 2.51.0.384.g4c02a37b29-goog ^ permalink raw reply related [flat|nested] 12+ messages in thread
* Re: [PATCH v2 1/3] iommu/io-pgtable-arm: Move selftests to a separate file 2025-09-17 19:11 ` [PATCH v2 1/3] iommu/io-pgtable-arm: Move selftests to a separate file Mostafa Saleh @ 2025-09-17 19:38 ` Jason Gunthorpe 2025-09-17 20:20 ` Mostafa Saleh 0 siblings, 1 reply; 12+ messages in thread From: Jason Gunthorpe @ 2025-09-17 19:38 UTC (permalink / raw) To: Mostafa Saleh Cc: iommu, linux-arm-kernel, linux-kernel, robin.murphy, will, joro, praan On Wed, Sep 17, 2025 at 07:11:38PM +0000, Mostafa Saleh wrote: > +static void __init arm_lpae_dump_ops(struct io_pgtable_ops *ops) > +{ > + struct arm_lpae_io_pgtable *data = io_pgtable_ops_to_data(ops); > + struct io_pgtable_cfg *cfg = &data->iop.cfg; Can be: struct io_pgtable_cfg *cfg = &io_pgtable_ops_to_pgtable(pgtbl_ops)->cfg; > + > + pr_err("cfg: pgsize_bitmap 0x%lx, ias %u-bit\n", > + cfg->pgsize_bitmap, cfg->ias); > + pr_err("data: %d levels, 0x%zx pgd_size, %u pg_shift, %u bits_per_level, pgd @ %p\n", > + ARM_LPAE_MAX_LEVELS - data->start_level, ARM_LPAE_PGD_SIZE(data), > + ilog2(ARM_LPAE_GRANULE(data)), data->bits_per_level, data->pgd); The entire struct arm_lpae_io_pgtable is exposed to a public header just for this one print.. Seems undesirable. Drop the print? Honestly, I prefer this, given the maturity let's not compromise modularity to print someting nobody will ever read.. Alternatively call a kunit-only exported function to do the print directly from ops so. smmuv3 has an example how to do that Either way a precursor patch to adjust it will allow this patch to avoid publishing most stuff. > +/* > + * Calculate the right shift amount to get to the portion describing level l > + * in a virtual address mapped by the pagetable in d. > + */ > +#define ARM_LPAE_LVL_SHIFT(l,d) \ > + (((ARM_LPAE_MAX_LEVELS - (l)) * (d)->bits_per_level) + \ > + ilog2(sizeof(arm_lpae_iopte))) Didn't see this being used? > +#define ARM_LPAE_GRANULE(d) \ > + (sizeof(arm_lpae_iopte) << (d)->bits_per_level) Only used by the above print > +#define ARM_LPAE_PGD_SIZE(d) \ > + (sizeof(arm_lpae_iopte) << (d)->pgd_bits) Only used by the above print > +#define ARM_LPAE_PTES_PER_TABLE(d) \ > + (ARM_LPAE_GRANULE(d) >> ilog2(sizeof(arm_lpae_iopte))) Didn't see this being used? > +typedef u64 arm_lpae_iopte; Only used by the defines and the above print. Jason ^ permalink raw reply [flat|nested] 12+ messages in thread
* Re: [PATCH v2 1/3] iommu/io-pgtable-arm: Move selftests to a separate file 2025-09-17 19:38 ` Jason Gunthorpe @ 2025-09-17 20:20 ` Mostafa Saleh 2025-09-17 21:48 ` Jason Gunthorpe 0 siblings, 1 reply; 12+ messages in thread From: Mostafa Saleh @ 2025-09-17 20:20 UTC (permalink / raw) To: Jason Gunthorpe Cc: iommu, linux-arm-kernel, linux-kernel, robin.murphy, will, joro, praan On Wed, Sep 17, 2025 at 04:38:18PM -0300, Jason Gunthorpe wrote: > On Wed, Sep 17, 2025 at 07:11:38PM +0000, Mostafa Saleh wrote: > > +static void __init arm_lpae_dump_ops(struct io_pgtable_ops *ops) > > +{ > > + struct arm_lpae_io_pgtable *data = io_pgtable_ops_to_data(ops); > > + struct io_pgtable_cfg *cfg = &data->iop.cfg; > > Can be: > > struct io_pgtable_cfg *cfg = > &io_pgtable_ops_to_pgtable(pgtbl_ops)->cfg; There are a bunch of other formatting issues here also, but I wanted to move the code as is, so with “--color-moved” you can see the exact difference, otherwise, it’s harder to review. I can add a patch before to fix those + the printing as you suggested next. > > > + > > + pr_err("cfg: pgsize_bitmap 0x%lx, ias %u-bit\n", > > + cfg->pgsize_bitmap, cfg->ias); > > + pr_err("data: %d levels, 0x%zx pgd_size, %u pg_shift, %u bits_per_level, pgd @ %p\n", > > + ARM_LPAE_MAX_LEVELS - data->start_level, ARM_LPAE_PGD_SIZE(data), > > + ilog2(ARM_LPAE_GRANULE(data)), data->bits_per_level, data->pgd); > > The entire struct arm_lpae_io_pgtable is exposed to a public header > just for this one print.. Seems undesirable. > > Drop the print? Honestly, I prefer this, given the maturity let's not > compromise modularity to print someting nobody will ever read.. > > Alternatively call a kunit-only exported function to do the print > directly from ops so. smmuv3 has an example how to do that > > Either way a precursor patch to adjust it will allow this patch to > avoid publishing most stuff. I don’t have a strong opinion about this, but I am more inclined to keep the prints considering it’s a low-level test for the page table, and such parameters would be useful to understand the failures. Moving arm_lpae_dump_ops() to the core library will leak the kunit struct, but we can drop the kunit_err and rely on pr_err(). I know this is not relevant to this series, but the KVM driver will need to expose arm_lpae_io_pgtable anyway. > > > +/* > > + * Calculate the right shift amount to get to the portion describing level l > > + * in a virtual address mapped by the pagetable in d. > > + */ > > +#define ARM_LPAE_LVL_SHIFT(l,d) \ > > + (((ARM_LPAE_MAX_LEVELS - (l)) * (d)->bits_per_level) + \ > > + ilog2(sizeof(arm_lpae_iopte))) > > Didn't see this being used? Yes, my bad, I will move it and the other one back. Thanks, Mostafa > > > +#define ARM_LPAE_GRANULE(d) \ > > + (sizeof(arm_lpae_iopte) << (d)->bits_per_level) > > Only used by the above print > > > +#define ARM_LPAE_PGD_SIZE(d) \ > > + (sizeof(arm_lpae_iopte) << (d)->pgd_bits) > > Only used by the above print > > > +#define ARM_LPAE_PTES_PER_TABLE(d) \ > > + (ARM_LPAE_GRANULE(d) >> ilog2(sizeof(arm_lpae_iopte))) > > Didn't see this being used? > > > +typedef u64 arm_lpae_iopte; > > Only used by the defines and the above print. > > Jason ^ permalink raw reply [flat|nested] 12+ messages in thread
* Re: [PATCH v2 1/3] iommu/io-pgtable-arm: Move selftests to a separate file 2025-09-17 20:20 ` Mostafa Saleh @ 2025-09-17 21:48 ` Jason Gunthorpe 2025-09-18 11:41 ` Mostafa Saleh 0 siblings, 1 reply; 12+ messages in thread From: Jason Gunthorpe @ 2025-09-17 21:48 UTC (permalink / raw) To: Mostafa Saleh Cc: iommu, linux-arm-kernel, linux-kernel, robin.murphy, will, joro, praan On Wed, Sep 17, 2025 at 08:20:12PM +0000, Mostafa Saleh wrote: > On Wed, Sep 17, 2025 at 04:38:18PM -0300, Jason Gunthorpe wrote: > > On Wed, Sep 17, 2025 at 07:11:38PM +0000, Mostafa Saleh wrote: > > > +static void __init arm_lpae_dump_ops(struct io_pgtable_ops *ops) > > > +{ > > > + struct arm_lpae_io_pgtable *data = io_pgtable_ops_to_data(ops); > > > + struct io_pgtable_cfg *cfg = &data->iop.cfg; > > > > Can be: > > > > struct io_pgtable_cfg *cfg = > > &io_pgtable_ops_to_pgtable(pgtbl_ops)->cfg; > > There are a bunch of other formatting issues here also, but I wanted to > move the code as is, so with “--color-moved” you can see the exact > difference, otherwise, it’s harder to review. > > I can add a patch before to fix those + the printing as you suggested > next. Yeah, I brought it up because with the above you don't need arm_lpae_io_pgtable > I don’t have a strong opinion about this, but I am more inclined to > keep the prints considering it’s a low-level test for the page table, > and such parameters would be useful to understand the failures. My general view has been that there is alot of debug prints I've added to kunits to debug them but once they are debugged I tend to drop them off as they may not be useful to debug the next issue. > I know this is not relevant to this series, but the KVM driver will > need to expose arm_lpae_io_pgtable anyway. Really? That sounds like wrong layering, why does it need to break the iopgtable abstraction? :( Jason ^ permalink raw reply [flat|nested] 12+ messages in thread
* Re: [PATCH v2 1/3] iommu/io-pgtable-arm: Move selftests to a separate file 2025-09-17 21:48 ` Jason Gunthorpe @ 2025-09-18 11:41 ` Mostafa Saleh 0 siblings, 0 replies; 12+ messages in thread From: Mostafa Saleh @ 2025-09-18 11:41 UTC (permalink / raw) To: Jason Gunthorpe Cc: iommu, linux-arm-kernel, linux-kernel, robin.murphy, will, joro, praan On Wed, Sep 17, 2025 at 06:48:24PM -0300, Jason Gunthorpe wrote: > On Wed, Sep 17, 2025 at 08:20:12PM +0000, Mostafa Saleh wrote: > > On Wed, Sep 17, 2025 at 04:38:18PM -0300, Jason Gunthorpe wrote: > > > On Wed, Sep 17, 2025 at 07:11:38PM +0000, Mostafa Saleh wrote: > > > > +static void __init arm_lpae_dump_ops(struct io_pgtable_ops *ops) > > > > +{ > > > > + struct arm_lpae_io_pgtable *data = io_pgtable_ops_to_data(ops); > > > > + struct io_pgtable_cfg *cfg = &data->iop.cfg; > > > > > > Can be: > > > > > > struct io_pgtable_cfg *cfg = > > > &io_pgtable_ops_to_pgtable(pgtbl_ops)->cfg; > > > > There are a bunch of other formatting issues here also, but I wanted to > > move the code as is, so with “--color-moved” you can see the exact > > difference, otherwise, it’s harder to review. > > > > I can add a patch before to fix those + the printing as you suggested > > next. > > Yeah, I brought it up because with the above you don't need > arm_lpae_io_pgtable > > > I don’t have a strong opinion about this, but I am more inclined to > > keep the prints considering it’s a low-level test for the page table, > > and such parameters would be useful to understand the failures. > > My general view has been that there is alot of debug prints I've added > to kunits to debug them but once they are debugged I tend to drop them > off as they may not be useful to debug the next issue. > > > I know this is not relevant to this series, but the KVM driver will > > need to expose arm_lpae_io_pgtable anyway. > > Really? That sounds like wrong layering, why does it need to break the > iopgtable abstraction? :( Oh, not any more actually, in my lastest version it's just exported for the selftest, sorry for the confusion. I will respin v3 with with this patch leaving the arm_lpae_dump_ops behind, it's a bit ugly with the guards + the exports, if Robin and Will are OK with removing it we can just drop it. Thanks, Mostafa > > Jason ^ permalink raw reply [flat|nested] 12+ messages in thread
* [PATCH v2 2/3] iommu/io-pgtable-arm-selftests: Modularize the test 2025-09-17 19:11 [PATCH v2 0/3] Move io-pgtable-arm selftest to KUnit Mostafa Saleh 2025-09-17 19:11 ` [PATCH v2 1/3] iommu/io-pgtable-arm: Move selftests to a separate file Mostafa Saleh @ 2025-09-17 19:11 ` Mostafa Saleh 2025-09-17 19:39 ` Jason Gunthorpe 2025-09-17 19:11 ` [PATCH v2 3/3] iommu/io-pgtable-arm-selftests: Use KUnit Mostafa Saleh 2025-09-17 19:27 ` [PATCH v2 0/3] Move io-pgtable-arm selftest to KUnit Jason Gunthorpe 3 siblings, 1 reply; 12+ messages in thread From: Mostafa Saleh @ 2025-09-17 19:11 UTC (permalink / raw) To: iommu, linux-arm-kernel, linux-kernel Cc: robin.murphy, will, joro, jgg, praan, Mostafa Saleh Remove the __init constraint, as the test will be converted to KUnit, it can run on-demand after later. Also, as KUnit can be a module, make this test modular. Signed-off-by: Mostafa Saleh <smostafa@google.com> --- drivers/iommu/Kconfig | 2 +- drivers/iommu/io-pgtable-arm-selftests.c | 36 +++++++++++++++--------- 2 files changed, 23 insertions(+), 15 deletions(-) diff --git a/drivers/iommu/Kconfig b/drivers/iommu/Kconfig index 70d29b14d851..553522ef3ca9 100644 --- a/drivers/iommu/Kconfig +++ b/drivers/iommu/Kconfig @@ -41,7 +41,7 @@ config IOMMU_IO_PGTABLE_LPAE up to 48-bits in size. config IOMMU_IO_PGTABLE_LPAE_SELFTEST - bool "LPAE selftests" + tristate "LPAE selftests" depends on IOMMU_IO_PGTABLE_LPAE help Enable self-tests for LPAE page table allocator. This performs diff --git a/drivers/iommu/io-pgtable-arm-selftests.c b/drivers/iommu/io-pgtable-arm-selftests.c index f7746ff2c7a0..5ac3e792c56d 100644 --- a/drivers/iommu/io-pgtable-arm-selftests.c +++ b/drivers/iommu/io-pgtable-arm-selftests.c @@ -12,34 +12,34 @@ #include "io-pgtable-arm.h" -static struct io_pgtable_cfg *cfg_cookie __initdata; +static struct io_pgtable_cfg *cfg_cookie; -static void __init dummy_tlb_flush_all(void *cookie) +static void dummy_tlb_flush_all(void *cookie) { WARN_ON(cookie != cfg_cookie); } -static void __init dummy_tlb_flush(unsigned long iova, size_t size, - size_t granule, void *cookie) +static void dummy_tlb_flush(unsigned long iova, size_t size, + size_t granule, void *cookie) { WARN_ON(cookie != cfg_cookie); WARN_ON(!(size & cfg_cookie->pgsize_bitmap)); } -static void __init dummy_tlb_add_page(struct iommu_iotlb_gather *gather, - unsigned long iova, size_t granule, - void *cookie) +static void dummy_tlb_add_page(struct iommu_iotlb_gather *gather, + unsigned long iova, size_t granule, + void *cookie) { dummy_tlb_flush(iova, granule, granule, cookie); } -static const struct iommu_flush_ops dummy_tlb_ops __initconst = { +static const struct iommu_flush_ops dummy_tlb_ops = { .tlb_flush_all = dummy_tlb_flush_all, .tlb_flush_walk = dummy_tlb_flush, .tlb_add_page = dummy_tlb_add_page, }; -static void __init arm_lpae_dump_ops(struct io_pgtable_ops *ops) +static void arm_lpae_dump_ops(struct io_pgtable_ops *ops) { struct arm_lpae_io_pgtable *data = io_pgtable_ops_to_data(ops); struct io_pgtable_cfg *cfg = &data->iop.cfg; @@ -57,9 +57,9 @@ static void __init arm_lpae_dump_ops(struct io_pgtable_ops *ops) -EFAULT; \ }) -static int __init arm_lpae_run_tests(struct io_pgtable_cfg *cfg) +static int arm_lpae_run_tests(struct io_pgtable_cfg *cfg) { - static const enum io_pgtable_fmt fmts[] __initconst = { + static const enum io_pgtable_fmt fmts[] = { ARM_64_LPAE_S1, ARM_64_LPAE_S2, }; @@ -160,15 +160,15 @@ static int __init arm_lpae_run_tests(struct io_pgtable_cfg *cfg) return 0; } -static int __init arm_lpae_do_selftests(void) +static int arm_lpae_do_selftests(void) { - static const unsigned long pgsize[] __initconst = { + static const unsigned long pgsize[] = { SZ_4K | SZ_2M | SZ_1G, SZ_16K | SZ_32M, SZ_64K | SZ_512M, }; - static const unsigned int address_size[] __initconst = { + static const unsigned int address_size[] = { 32, 36, 40, 42, 44, 48, }; @@ -208,4 +208,12 @@ static int __init arm_lpae_do_selftests(void) return fail ? -EFAULT : 0; } + +static void arm_lpae_exit_selftests(void) +{ +} + subsys_initcall(arm_lpae_do_selftests); +module_exit(arm_lpae_exit_selftests); +MODULE_DESCRIPTION("io-pgtable-arm library selftest"); +MODULE_LICENSE("GPL"); -- 2.51.0.384.g4c02a37b29-goog ^ permalink raw reply related [flat|nested] 12+ messages in thread
* Re: [PATCH v2 2/3] iommu/io-pgtable-arm-selftests: Modularize the test 2025-09-17 19:11 ` [PATCH v2 2/3] iommu/io-pgtable-arm-selftests: Modularize the test Mostafa Saleh @ 2025-09-17 19:39 ` Jason Gunthorpe 0 siblings, 0 replies; 12+ messages in thread From: Jason Gunthorpe @ 2025-09-17 19:39 UTC (permalink / raw) To: Mostafa Saleh Cc: iommu, linux-arm-kernel, linux-kernel, robin.murphy, will, joro, praan On Wed, Sep 17, 2025 at 07:11:39PM +0000, Mostafa Saleh wrote: > Remove the __init constraint, as the test will be converted to KUnit, > it can run on-demand after later. > > Also, as KUnit can be a module, make this test modular. > > Signed-off-by: Mostafa Saleh <smostafa@google.com> > --- > drivers/iommu/Kconfig | 2 +- > drivers/iommu/io-pgtable-arm-selftests.c | 36 +++++++++++++++--------- > 2 files changed, 23 insertions(+), 15 deletions(-) Reviewed-by: Jason Gunthorpe <jgg@nvidia.com> Jason ^ permalink raw reply [flat|nested] 12+ messages in thread
* [PATCH v2 3/3] iommu/io-pgtable-arm-selftests: Use KUnit 2025-09-17 19:11 [PATCH v2 0/3] Move io-pgtable-arm selftest to KUnit Mostafa Saleh 2025-09-17 19:11 ` [PATCH v2 1/3] iommu/io-pgtable-arm: Move selftests to a separate file Mostafa Saleh 2025-09-17 19:11 ` [PATCH v2 2/3] iommu/io-pgtable-arm-selftests: Modularize the test Mostafa Saleh @ 2025-09-17 19:11 ` Mostafa Saleh 2025-09-17 19:44 ` Jason Gunthorpe 2025-09-17 19:27 ` [PATCH v2 0/3] Move io-pgtable-arm selftest to KUnit Jason Gunthorpe 3 siblings, 1 reply; 12+ messages in thread From: Mostafa Saleh @ 2025-09-17 19:11 UTC (permalink / raw) To: iommu, linux-arm-kernel, linux-kernel Cc: robin.murphy, will, joro, jgg, praan, Mostafa Saleh Integrate the selftests as part of kunit, this makes the test available through debugfs. Suggested-by: Jason Gunthorpe <jgg@ziepe.ca> Signed-off-by: Mostafa Saleh <smostafa@google.com> --- drivers/iommu/Kconfig | 2 +- drivers/iommu/io-pgtable-arm-selftests.c | 93 +++++++++++++----------- 2 files changed, 51 insertions(+), 44 deletions(-) diff --git a/drivers/iommu/Kconfig b/drivers/iommu/Kconfig index 553522ef3ca9..f7e6d1db6ce6 100644 --- a/drivers/iommu/Kconfig +++ b/drivers/iommu/Kconfig @@ -42,7 +42,7 @@ config IOMMU_IO_PGTABLE_LPAE config IOMMU_IO_PGTABLE_LPAE_SELFTEST tristate "LPAE selftests" - depends on IOMMU_IO_PGTABLE_LPAE + depends on IOMMU_IO_PGTABLE_LPAE && KUNIT help Enable self-tests for LPAE page table allocator. This performs a series of page-table consistency checks during boot. diff --git a/drivers/iommu/io-pgtable-arm-selftests.c b/drivers/iommu/io-pgtable-arm-selftests.c index 5ac3e792c56d..cfff8eb78e04 100644 --- a/drivers/iommu/io-pgtable-arm-selftests.c +++ b/drivers/iommu/io-pgtable-arm-selftests.c @@ -6,6 +6,8 @@ * * Author: Will Deacon <will.deacon@arm.com> */ +#include <kunit/device.h> +#include <kunit/test.h> #include <linux/device/faux.h> #include <linux/kernel.h> #include <linux/slab.h> @@ -39,25 +41,26 @@ static const struct iommu_flush_ops dummy_tlb_ops = { .tlb_add_page = dummy_tlb_add_page, }; -static void arm_lpae_dump_ops(struct io_pgtable_ops *ops) +static void arm_lpae_dump_ops(struct kunit *test, struct io_pgtable_ops *ops) { struct arm_lpae_io_pgtable *data = io_pgtable_ops_to_data(ops); struct io_pgtable_cfg *cfg = &data->iop.cfg; - pr_err("cfg: pgsize_bitmap 0x%lx, ias %u-bit\n", - cfg->pgsize_bitmap, cfg->ias); - pr_err("data: %d levels, 0x%zx pgd_size, %u pg_shift, %u bits_per_level, pgd @ %p\n", - ARM_LPAE_MAX_LEVELS - data->start_level, ARM_LPAE_PGD_SIZE(data), - ilog2(ARM_LPAE_GRANULE(data)), data->bits_per_level, data->pgd); + kunit_err(test, "cfg: pgsize_bitmap 0x%lx, ias %u-bit\n", + cfg->pgsize_bitmap, cfg->ias); + kunit_err(test, "data: %d levels, 0x%zx pgd_size, %u pg_shift, %u bits_per_level, pgd @ %p\n", + ARM_LPAE_MAX_LEVELS - data->start_level, ARM_LPAE_PGD_SIZE(data), + ilog2(ARM_LPAE_GRANULE(data)), data->bits_per_level, data->pgd); } -#define __FAIL(ops, i) ({ \ - WARN(1, "selftest: test failed for fmt idx %d\n", (i)); \ - arm_lpae_dump_ops(ops); \ - -EFAULT; \ +#define __FAIL(test, ops, i) ({ \ + KUNIT_FAIL(test, ""); \ + kunit_err(test, "selftest: test failed for fmt idx %d\n", (i)); \ + arm_lpae_dump_ops(test, ops); \ + -EFAULT; \ }) -static int arm_lpae_run_tests(struct io_pgtable_cfg *cfg) +static int arm_lpae_run_tests(struct kunit *test, struct io_pgtable_cfg *cfg) { static const enum io_pgtable_fmt fmts[] = { ARM_64_LPAE_S1, @@ -73,7 +76,7 @@ static int arm_lpae_run_tests(struct io_pgtable_cfg *cfg) cfg_cookie = cfg; ops = alloc_io_pgtable_ops(fmts[i], cfg, cfg); if (!ops) { - pr_err("selftest: failed to allocate io pgtable ops\n"); + kunit_err(test, "selftest: failed to allocate io pgtable ops\n"); return -ENOMEM; } @@ -82,13 +85,13 @@ static int arm_lpae_run_tests(struct io_pgtable_cfg *cfg) * Empty page tables shouldn't provide any translations. */ if (ops->iova_to_phys(ops, 42)) - return __FAIL(ops, i); + return __FAIL(test, ops, i); if (ops->iova_to_phys(ops, SZ_1G + 42)) - return __FAIL(ops, i); + return __FAIL(test, ops, i); if (ops->iova_to_phys(ops, SZ_2G + 42)) - return __FAIL(ops, i); + return __FAIL(test, ops, i); /* * Distinct mappings of different granule sizes. @@ -101,16 +104,16 @@ static int arm_lpae_run_tests(struct io_pgtable_cfg *cfg) IOMMU_READ | IOMMU_WRITE | IOMMU_NOEXEC | IOMMU_CACHE, GFP_KERNEL, &mapped)) - return __FAIL(ops, i); + return __FAIL(test, ops, i); /* Overlapping mappings */ if (!ops->map_pages(ops, iova, iova + size, size, 1, IOMMU_READ | IOMMU_NOEXEC, GFP_KERNEL, &mapped)) - return __FAIL(ops, i); + return __FAIL(test, ops, i); if (ops->iova_to_phys(ops, iova + 42) != (iova + 42)) - return __FAIL(ops, i); + return __FAIL(test, ops, i); iova += SZ_1G; } @@ -121,18 +124,18 @@ static int arm_lpae_run_tests(struct io_pgtable_cfg *cfg) size = 1UL << j; if (ops->unmap_pages(ops, iova, size, 1, NULL) != size) - return __FAIL(ops, i); + return __FAIL(test, ops, i); if (ops->iova_to_phys(ops, iova + 42)) - return __FAIL(ops, i); + return __FAIL(test, ops, i); /* Remap full block */ if (ops->map_pages(ops, iova, iova, size, 1, IOMMU_WRITE, GFP_KERNEL, &mapped)) - return __FAIL(ops, i); + return __FAIL(test, ops, i); if (ops->iova_to_phys(ops, iova + 42) != (iova + 42)) - return __FAIL(ops, i); + return __FAIL(test, ops, i); iova += SZ_1G; } @@ -148,11 +151,11 @@ static int arm_lpae_run_tests(struct io_pgtable_cfg *cfg) IOMMU_READ | IOMMU_WRITE | IOMMU_NOEXEC | IOMMU_CACHE, GFP_KERNEL, &mapped)) - return __FAIL(ops, i); + return __FAIL(test, ops, i); if (mapped != size) - return __FAIL(ops, i); + return __FAIL(test, ops, i); if (ops->unmap_pages(ops, iova, size, 1, NULL) != size) - return __FAIL(ops, i); + return __FAIL(test, ops, i); free_io_pgtable_ops(ops); } @@ -160,7 +163,7 @@ static int arm_lpae_run_tests(struct io_pgtable_cfg *cfg) return 0; } -static int arm_lpae_do_selftests(void) +static void arm_lpae_do_selftests(struct kunit *test) { static const unsigned long pgsize[] = { SZ_4K | SZ_2M | SZ_1G, @@ -173,18 +176,19 @@ static int arm_lpae_do_selftests(void) }; int i, j, k, pass = 0, fail = 0; - struct faux_device *dev; + struct device *dev; struct io_pgtable_cfg cfg = { .tlb = &dummy_tlb_ops, .coherent_walk = true, .quirks = IO_PGTABLE_QUIRK_NO_WARN, }; - dev = faux_device_create("io-pgtable-test", NULL, 0); - if (!dev) - return -ENOMEM; + dev = kunit_device_register(test, "io-pgtable-test"); + KUNIT_EXPECT_NOT_ERR_OR_NULL(test, dev); + if (IS_ERR_OR_NULL(dev)) + return; - cfg.iommu_dev = &dev->dev; + cfg.iommu_dev = dev; for (i = 0; i < ARRAY_SIZE(pgsize); ++i) { for (j = 0; j < ARRAY_SIZE(address_size); ++j) { @@ -193,9 +197,9 @@ static int arm_lpae_do_selftests(void) cfg.pgsize_bitmap = pgsize[i]; cfg.ias = address_size[k]; cfg.oas = address_size[j]; - pr_info("selftest: pgsize_bitmap 0x%08lx, IAS %u OAS %u\n", - pgsize[i], cfg.ias, cfg.oas); - if (arm_lpae_run_tests(&cfg)) + kunit_info(test, "selftest: pgsize_bitmap 0x%08lx, IAS %u OAS %u\n", + pgsize[i], cfg.ias, cfg.oas); + if (arm_lpae_run_tests(test, &cfg)) fail++; else pass++; @@ -203,17 +207,20 @@ static int arm_lpae_do_selftests(void) } } - pr_info("selftest: completed with %d PASS %d FAIL\n", pass, fail); - faux_device_destroy(dev); - - return fail ? -EFAULT : 0; + kunit_info(test, "selftest: completed with %d PASS %d FAIL\n", pass, fail); } -static void arm_lpae_exit_selftests(void) -{ -} +static struct kunit_case io_pgtable_arm_test_cases[] = { + KUNIT_CASE(arm_lpae_do_selftests), + {}, +}; + +static struct kunit_suite io_pgtable_arm_test = { + .name = "io-pgtable-arm-test", + .test_cases = io_pgtable_arm_test_cases, +}; + +kunit_test_suite(io_pgtable_arm_test); -subsys_initcall(arm_lpae_do_selftests); -module_exit(arm_lpae_exit_selftests); MODULE_DESCRIPTION("io-pgtable-arm library selftest"); MODULE_LICENSE("GPL"); -- 2.51.0.384.g4c02a37b29-goog ^ permalink raw reply related [flat|nested] 12+ messages in thread
* Re: [PATCH v2 3/3] iommu/io-pgtable-arm-selftests: Use KUnit 2025-09-17 19:11 ` [PATCH v2 3/3] iommu/io-pgtable-arm-selftests: Use KUnit Mostafa Saleh @ 2025-09-17 19:44 ` Jason Gunthorpe 2025-09-17 20:31 ` Mostafa Saleh 0 siblings, 1 reply; 12+ messages in thread From: Jason Gunthorpe @ 2025-09-17 19:44 UTC (permalink / raw) To: Mostafa Saleh Cc: iommu, linux-arm-kernel, linux-kernel, robin.murphy, will, joro, praan On Wed, Sep 17, 2025 at 07:11:40PM +0000, Mostafa Saleh wrote: > Integrate the selftests as part of kunit, this makes the test > available through debugfs. > > Suggested-by: Jason Gunthorpe <jgg@ziepe.ca> > Signed-off-by: Mostafa Saleh <smostafa@google.com> > --- > drivers/iommu/Kconfig | 2 +- > drivers/iommu/io-pgtable-arm-selftests.c | 93 +++++++++++++----------- > 2 files changed, 51 insertions(+), 44 deletions(-) > > diff --git a/drivers/iommu/Kconfig b/drivers/iommu/Kconfig > index 553522ef3ca9..f7e6d1db6ce6 100644 > --- a/drivers/iommu/Kconfig > +++ b/drivers/iommu/Kconfig > @@ -42,7 +42,7 @@ config IOMMU_IO_PGTABLE_LPAE > > config IOMMU_IO_PGTABLE_LPAE_SELFTEST This should probably be renamed to xxx_KUNIT_TEST > tristate "LPAE selftests" > - depends on IOMMU_IO_PGTABLE_LPAE > + depends on IOMMU_IO_PGTABLE_LPAE && KUNIT I wonder if it should be: select IOMMU_IO_PGTABLE_LPAE depends on KUNIT default KUNIT_ALL_TESTS ? At least the default should be there > help > Enable self-tests for LPAE page table allocator. This performs > a series of page-table consistency checks during boot. "kunit tests" here as well > -subsys_initcall(arm_lpae_do_selftests); Oh so it does drop the initcall, the coverletter is a bit confusing then. Reviewed-by: Jason Gunthorpe <jgg@nvidia.com> Jason ^ permalink raw reply [flat|nested] 12+ messages in thread
* Re: [PATCH v2 3/3] iommu/io-pgtable-arm-selftests: Use KUnit 2025-09-17 19:44 ` Jason Gunthorpe @ 2025-09-17 20:31 ` Mostafa Saleh 0 siblings, 0 replies; 12+ messages in thread From: Mostafa Saleh @ 2025-09-17 20:31 UTC (permalink / raw) To: Jason Gunthorpe Cc: iommu, linux-arm-kernel, linux-kernel, robin.murphy, will, joro, praan On Wed, Sep 17, 2025 at 04:44:54PM -0300, Jason Gunthorpe wrote: > On Wed, Sep 17, 2025 at 07:11:40PM +0000, Mostafa Saleh wrote: > > Integrate the selftests as part of kunit, this makes the test > > available through debugfs. > > > > Suggested-by: Jason Gunthorpe <jgg@ziepe.ca> > > Signed-off-by: Mostafa Saleh <smostafa@google.com> > > --- > > drivers/iommu/Kconfig | 2 +- > > drivers/iommu/io-pgtable-arm-selftests.c | 93 +++++++++++++----------- > > 2 files changed, 51 insertions(+), 44 deletions(-) > > > > diff --git a/drivers/iommu/Kconfig b/drivers/iommu/Kconfig > > index 553522ef3ca9..f7e6d1db6ce6 100644 > > --- a/drivers/iommu/Kconfig > > +++ b/drivers/iommu/Kconfig > > @@ -42,7 +42,7 @@ config IOMMU_IO_PGTABLE_LPAE > > > > config IOMMU_IO_PGTABLE_LPAE_SELFTEST > > This should probably be renamed to xxx_KUNIT_TEST Will do. > > > tristate "LPAE selftests" > > - depends on IOMMU_IO_PGTABLE_LPAE > > + depends on IOMMU_IO_PGTABLE_LPAE && KUNIT > > I wonder if it should be: > > select IOMMU_IO_PGTABLE_LPAE > depends on KUNIT > default KUNIT_ALL_TESTS > > ? > > At least the default should be there Makes sense, will do. > > > help > > Enable self-tests for LPAE page table allocator. This performs > > a series of page-table consistency checks during boot. > > "kunit tests" here as well Will do. > > > -subsys_initcall(arm_lpae_do_selftests); > > Oh so it does drop the initcall, the coverletter is a bit confusing then. Yes, I will reword the cover letter. Thanks, Mostafa > > Reviewed-by: Jason Gunthorpe <jgg@nvidia.com> > > Jason ^ permalink raw reply [flat|nested] 12+ messages in thread
* Re: [PATCH v2 0/3] Move io-pgtable-arm selftest to KUnit 2025-09-17 19:11 [PATCH v2 0/3] Move io-pgtable-arm selftest to KUnit Mostafa Saleh ` (2 preceding siblings ...) 2025-09-17 19:11 ` [PATCH v2 3/3] iommu/io-pgtable-arm-selftests: Use KUnit Mostafa Saleh @ 2025-09-17 19:27 ` Jason Gunthorpe 3 siblings, 0 replies; 12+ messages in thread From: Jason Gunthorpe @ 2025-09-17 19:27 UTC (permalink / raw) To: Mostafa Saleh Cc: iommu, linux-arm-kernel, linux-kernel, robin.murphy, will, joro, praan On Wed, Sep 17, 2025 at 07:11:37PM +0000, Mostafa Saleh wrote: > -------- > - With CONFIG_IOMMU_IO_PGTABLE_LPAE_SELFTEST=y > It will automatically run at boot as before this patch series. I would drop this kconfig, kunit does everything required including run at boot if you want. > - Otherwise with CONFIG_IOMMU_IO_PGTABLE_LPAE_SELFTEST=m: > 1) on module load: > Once the module load the self test will run > # modprobe io-pgtable-arm-selftests > > 2) debugfs > With CONFIG_KUNIT_DEBUGFS=y You can run the test with > # echo 1 > /sys/kernel/debug/kunit/io-pgtable-arm-test/run > > 3) Using kunit.py > You can also use the helper script which uses Qemu in the background > > # ./tools/testing/kunit/kunit.py run --build_dir build_kunit_arm64 --arch arm64 \ > --make_options LLVM=1 --kunitconfig ./kunit/kunitconfig > [18:01:09] ============= io-pgtable-arm-test (1 subtest) ============== > [18:01:09] [PASSED] arm_lpae_do_selftests > [18:01:09] =============== [PASSED] io-pgtable-arm-test =============== > [18:01:09] ============================================================ Suggest to include the above instruction in the commit message adding the kunit so at least it is in the history Jason ^ permalink raw reply [flat|nested] 12+ messages in thread
end of thread, other threads:[~2025-09-18 11:42 UTC | newest] Thread overview: 12+ messages (download: mbox.gz follow: Atom feed -- links below jump to the message on this page -- 2025-09-17 19:11 [PATCH v2 0/3] Move io-pgtable-arm selftest to KUnit Mostafa Saleh 2025-09-17 19:11 ` [PATCH v2 1/3] iommu/io-pgtable-arm: Move selftests to a separate file Mostafa Saleh 2025-09-17 19:38 ` Jason Gunthorpe 2025-09-17 20:20 ` Mostafa Saleh 2025-09-17 21:48 ` Jason Gunthorpe 2025-09-18 11:41 ` Mostafa Saleh 2025-09-17 19:11 ` [PATCH v2 2/3] iommu/io-pgtable-arm-selftests: Modularize the test Mostafa Saleh 2025-09-17 19:39 ` Jason Gunthorpe 2025-09-17 19:11 ` [PATCH v2 3/3] iommu/io-pgtable-arm-selftests: Use KUnit Mostafa Saleh 2025-09-17 19:44 ` Jason Gunthorpe 2025-09-17 20:31 ` Mostafa Saleh 2025-09-17 19:27 ` [PATCH v2 0/3] Move io-pgtable-arm selftest to KUnit Jason Gunthorpe
This is an external index of several public inboxes, see mirroring instructions on how to clone and mirror all data and code used by this external index.