|  | // SPDX-License-Identifier: GPL-2.0 | 
|  |  | 
|  | #include <kunit/test.h> | 
|  |  | 
|  | #define MAX_PHYS_REGIONS	16 | 
|  | #define INVALID_VALUE		(~0ull) | 
|  |  | 
|  | struct ne_phys_regions_test { | 
|  | u64           paddr; | 
|  | u64           size; | 
|  | int           expect_rc; | 
|  | unsigned long expect_num; | 
|  | u64           expect_last_paddr; | 
|  | u64           expect_last_size; | 
|  | } phys_regions_test_cases[] = { | 
|  | /* | 
|  | * Add the region from 0x1000 to (0x1000 + 0x200000 - 1): | 
|  | *   Expected result: | 
|  | *       Failed, start address is not 2M-aligned | 
|  | * | 
|  | * Now the instance of struct ne_phys_contig_mem_regions is: | 
|  | *   num = 0 | 
|  | *   regions = {} | 
|  | */ | 
|  | {0x1000, 0x200000, -EINVAL, 0, INVALID_VALUE, INVALID_VALUE}, | 
|  |  | 
|  | /* | 
|  | * Add the region from 0x200000 to (0x200000 + 0x1000 - 1): | 
|  | *   Expected result: | 
|  | *       Failed, size is not 2M-aligned | 
|  | * | 
|  | * Now the instance of struct ne_phys_contig_mem_regions is: | 
|  | *   num = 0 | 
|  | *   regions = {} | 
|  | */ | 
|  | {0x200000, 0x1000, -EINVAL, 0, INVALID_VALUE, INVALID_VALUE}, | 
|  |  | 
|  | /* | 
|  | * Add the region from 0x200000 to (0x200000 + 0x200000 - 1): | 
|  | *   Expected result: | 
|  | *       Successful | 
|  | * | 
|  | * Now the instance of struct ne_phys_contig_mem_regions is: | 
|  | *   num = 1 | 
|  | *   regions = { | 
|  | *       {start=0x200000, end=0x3fffff}, // len=0x200000 | 
|  | *   } | 
|  | */ | 
|  | {0x200000, 0x200000, 0, 1, 0x200000, 0x200000}, | 
|  |  | 
|  | /* | 
|  | * Add the region from 0x0 to (0x0 + 0x200000 - 1): | 
|  | *   Expected result: | 
|  | *       Successful | 
|  | * | 
|  | * Now the instance of struct ne_phys_contig_mem_regions is: | 
|  | *   num = 2 | 
|  | *   regions = { | 
|  | *       {start=0x200000, end=0x3fffff}, // len=0x200000 | 
|  | *       {start=0x0,      end=0x1fffff}, // len=0x200000 | 
|  | *   } | 
|  | */ | 
|  | {0x0, 0x200000, 0, 2, 0x0, 0x200000}, | 
|  |  | 
|  | /* | 
|  | * Add the region from 0x600000 to (0x600000 + 0x400000 - 1): | 
|  | *   Expected result: | 
|  | *       Successful | 
|  | * | 
|  | * Now the instance of struct ne_phys_contig_mem_regions is: | 
|  | *   num = 3 | 
|  | *   regions = { | 
|  | *       {start=0x200000, end=0x3fffff}, // len=0x200000 | 
|  | *       {start=0x0,      end=0x1fffff}, // len=0x200000 | 
|  | *       {start=0x600000, end=0x9fffff}, // len=0x400000 | 
|  | *   } | 
|  | */ | 
|  | {0x600000, 0x400000, 0, 3, 0x600000, 0x400000}, | 
|  |  | 
|  | /* | 
|  | * Add the region from 0xa00000 to (0xa00000 + 0x400000 - 1): | 
|  | *   Expected result: | 
|  | *       Successful, merging case! | 
|  | * | 
|  | * Now the instance of struct ne_phys_contig_mem_regions is: | 
|  | *   num = 3 | 
|  | *   regions = { | 
|  | *       {start=0x200000, end=0x3fffff}, // len=0x200000 | 
|  | *       {start=0x0,      end=0x1fffff}, // len=0x200000 | 
|  | *       {start=0x600000, end=0xdfffff}, // len=0x800000 | 
|  | *   } | 
|  | */ | 
|  | {0xa00000, 0x400000, 0, 3, 0x600000, 0x800000}, | 
|  |  | 
|  | /* | 
|  | * Add the region from 0x1000 to (0x1000 + 0x200000 - 1): | 
|  | *   Expected result: | 
|  | *       Failed, start address is not 2M-aligned | 
|  | * | 
|  | * Now the instance of struct ne_phys_contig_mem_regions is: | 
|  | *   num = 3 | 
|  | *   regions = { | 
|  | *       {start=0x200000, end=0x3fffff}, // len=0x200000 | 
|  | *       {start=0x0,      end=0x1fffff}, // len=0x200000 | 
|  | *       {start=0x600000, end=0xdfffff}, // len=0x800000 | 
|  | *   } | 
|  | */ | 
|  | {0x1000, 0x200000, -EINVAL, 3, 0x600000, 0x800000}, | 
|  | }; | 
|  |  | 
|  | static void ne_misc_dev_test_merge_phys_contig_memory_regions(struct kunit *test) | 
|  | { | 
|  | struct ne_phys_contig_mem_regions phys_contig_mem_regions = {}; | 
|  | int rc = 0; | 
|  | int i = 0; | 
|  |  | 
|  | phys_contig_mem_regions.regions = kunit_kcalloc(test, MAX_PHYS_REGIONS, | 
|  | sizeof(*phys_contig_mem_regions.regions), | 
|  | GFP_KERNEL); | 
|  | KUNIT_ASSERT_TRUE(test, phys_contig_mem_regions.regions); | 
|  |  | 
|  | for (i = 0; i < ARRAY_SIZE(phys_regions_test_cases); i++) { | 
|  | struct ne_phys_regions_test *test_case = &phys_regions_test_cases[i]; | 
|  | unsigned long num = 0; | 
|  |  | 
|  | rc = ne_merge_phys_contig_memory_regions(&phys_contig_mem_regions, | 
|  | test_case->paddr, test_case->size); | 
|  | KUNIT_EXPECT_EQ(test, rc, test_case->expect_rc); | 
|  | KUNIT_EXPECT_EQ(test, phys_contig_mem_regions.num, test_case->expect_num); | 
|  |  | 
|  | if (test_case->expect_last_paddr == INVALID_VALUE) | 
|  | continue; | 
|  |  | 
|  | num = phys_contig_mem_regions.num; | 
|  | KUNIT_EXPECT_EQ(test, phys_contig_mem_regions.regions[num - 1].start, | 
|  | test_case->expect_last_paddr); | 
|  | KUNIT_EXPECT_EQ(test, range_len(&phys_contig_mem_regions.regions[num - 1]), | 
|  | test_case->expect_last_size); | 
|  | } | 
|  |  | 
|  | kunit_kfree(test, phys_contig_mem_regions.regions); | 
|  | } | 
|  |  | 
|  | static struct kunit_case ne_misc_dev_test_cases[] = { | 
|  | KUNIT_CASE(ne_misc_dev_test_merge_phys_contig_memory_regions), | 
|  | {} | 
|  | }; | 
|  |  | 
|  | static struct kunit_suite ne_misc_dev_test_suite = { | 
|  | .name = "ne_misc_dev_test", | 
|  | .test_cases = ne_misc_dev_test_cases, | 
|  | }; | 
|  |  | 
|  | kunit_test_suite(ne_misc_dev_test_suite); |