diff options
author | Jan Dabros <jsd@semihalf.com> | 2020-07-08 22:33:52 +0200 |
---|---|---|
committer | Patrick Georgi <pgeorgi@google.com> | 2021-03-15 06:10:24 +0000 |
commit | 9e16ca9decf6f61ce7aaa1f325e07ebd84923cb2 (patch) | |
tree | 8b03debf2833e28217b3573212206d5a506ec710 | |
parent | a776ebb801fd2d53fe2fbca008015ab670ef8b13 (diff) | |
download | coreboot-9e16ca9decf6f61ce7aaa1f325e07ebd84923cb2.tar.xz |
tests: Add lib/memrange-test test case
Signed-off-by: Jan Dabros <jsd@semihalf.com>
Signed-off-by: Jakub Czapiga <jacz@semihalf.com>
Change-Id: If30a238d32326ffd1d6719470deedc77f176ac72
Reviewed-on: https://review.coreboot.org/c/coreboot/+/43305
Tested-by: build bot (Jenkins) <no-reply@coreboot.org>
Reviewed-by: Paul Fagerburg <pfagerburg@chromium.org>
-rw-r--r-- | tests/lib/Makefile.inc | 6 | ||||
-rw-r--r-- | tests/lib/memrange-test.c | 638 |
2 files changed, 644 insertions, 0 deletions
diff --git a/tests/lib/Makefile.inc b/tests/lib/Makefile.inc index 921da1c9e1..613f8e5aac 100644 --- a/tests/lib/Makefile.inc +++ b/tests/lib/Makefile.inc @@ -22,6 +22,7 @@ tests-y += malloc-test tests-y += memmove-test tests-y += crc_byte-test tests-y += compute_ip_checksum-test +tests-y += memrange-test string-test-srcs += tests/lib/string-test.c string-test-srcs += src/lib/string.c @@ -106,3 +107,8 @@ crc_byte-test-srcs += src/lib/crc_byte.c compute_ip_checksum-test-srcs += tests/lib/compute_ip_checksum-test.c compute_ip_checksum-test-srcs += src/lib/compute_ip_checksum.c +memrange-test-srcs += tests/lib/memrange-test.c +memrange-test-srcs += src/lib/memrange.c +memrange-test-srcs += tests/stubs/console.c +memrange-test-srcs += src/device/device_util.c + diff --git a/tests/lib/memrange-test.c b/tests/lib/memrange-test.c new file mode 100644 index 0000000000..f9e6056d35 --- /dev/null +++ b/tests/lib/memrange-test.c @@ -0,0 +1,638 @@ +/* SPDX-License-Identifier: GPL-2.0-only */ + +#include <tests/test.h> + +#include <device/device.h> +#include <device/resource.h> +#include <commonlib/helpers.h> +#include <memrange.h> + +#define MEMRANGE_ALIGN (POWER_OF_2(12)) + +enum mem_types { + /* Avoid using 0 to verify that UUT really sets this memory, + but keep value small, as this will be an index in the table */ + CACHEABLE_TAG = 0x10, + RESERVED_TAG, + READONLY_TAG, + INSERTED_TAG, + HOLE_TAG, + END_OF_RESOURCES +}; + +/* Indices of entries matters, since it must reflect mem_types enum */ +struct resource res_mock_1[] = { + [CACHEABLE_TAG] = { .base = 0xE000, .size = 0xF2000, + .next = &res_mock_1[RESERVED_TAG], .flags = IORESOURCE_CACHEABLE | IORESOURCE_MEM }, + [RESERVED_TAG] = { .base = 4ULL * GiB, .size = 4ULL * KiB, + .next = &res_mock_1[READONLY_TAG], .flags = IORESOURCE_RESERVE | IORESOURCE_MEM }, + [READONLY_TAG] = { .base = 0xFF0000, .size = 0x10000, .next = NULL, + .flags = IORESOURCE_READONLY | IORESOURCE_MEM } +}; + +/* Boundary 1 byte below 4GiB and 1 byte above 4GiB. */ +struct resource res_mock_2[] = { + [CACHEABLE_TAG] = { .base = 0x1000000, .size = 4ULL * GiB - 0x1000001ULL, + .next = &res_mock_2[RESERVED_TAG], .flags = IORESOURCE_CACHEABLE | IORESOURCE_MEM }, + [RESERVED_TAG] = { .base = 4ULL * GiB + 1ULL, .size = 4ULL * GiB, + .next = &res_mock_2[READONLY_TAG], .flags = IORESOURCE_RESERVE | IORESOURCE_MEM }, + [READONLY_TAG] = { .base = 0, .size = 0x10000, .next = NULL, + .flags = IORESOURCE_READONLY | IORESOURCE_MEM} +}; + +/* Boundary crossing 4GiB. */ +struct resource res_mock_3[] = { + [CACHEABLE_TAG] = { .base = 0xD000, .size = 0xF3000, + .next = &res_mock_3[RESERVED_TAG], .flags = IORESOURCE_CACHEABLE | IORESOURCE_MEM }, + [RESERVED_TAG] = { .base = 1ULL * GiB, .size = 4ULL * GiB, + .next = &res_mock_3[READONLY_TAG], .flags = IORESOURCE_RESERVE | IORESOURCE_MEM }, + [READONLY_TAG] = { .base = 0xFF0000, .size = 0x10000, .next = NULL, + .flags = IORESOURCE_READONLY | IORESOURCE_MEM} +}; + + +struct device mock_device = { .enabled = 1 }; + +/* Fake memory devices handle */ +struct device *all_devices = &mock_device; + +int setup_test_1(void **state) +{ + *state = res_mock_1; + mock_device.resource_list = &res_mock_1[CACHEABLE_TAG]; + + return 0; +} + +int setup_test_2(void **state) +{ + *state = res_mock_2; + mock_device.resource_list = &res_mock_2[CACHEABLE_TAG]; + + return 0; +} + +int setup_test_3(void **state) +{ + *state = res_mock_3; + mock_device.resource_list = &res_mock_3[CACHEABLE_TAG]; + + return 0; +} + +resource_t get_aligned_base(struct resource *res, struct range_entry *entry) +{ + return ALIGN_DOWN(res[range_entry_tag(entry)].base, MEMRANGE_ALIGN); +} + +resource_t get_aligned_end(struct resource *res, struct range_entry *entry) +{ + resource_t end = res[range_entry_tag(entry)].base + + res[range_entry_tag(entry)].size + + (res[range_entry_tag(entry)].base - range_entry_base(entry)); + return ALIGN_UP(end, MEMRANGE_ALIGN); +} + +/* + * This test verifies memranges_init(), memranges_add_resources() and memranges_teardown() + * functions. It covers basic functionality of memrange library - implementation of creating + * memrange structure from resources available on the platform and method for free'ing + * allocated memory. + * + * Example memory ranges (res_mock1) for test_memrange_basic. + * Ranges marked with asterisks (***) are not added to the test_memrange. + * + * +--------CACHEABLE_TAG--------+ <-0xE000 + * | | + * | | + * | | + * +-----------------------------+ <-0x100000 + * + * + * + * +-----***READONLY_TAG***------+ <-0xFF0000 + * | | + * | | + * | | + * +-----------------------------+ <-0x1000000 + * + * + * +--------RESERVED_TAG---------+ <-0x100000000 + * | | + * +-----------------------------+ <-0x100001000 + */ +static void test_memrange_basic(void **state) +{ + int counter = 0; + const unsigned long cacheable = IORESOURCE_CACHEABLE; + const unsigned long reserved = IORESOURCE_RESERVE; + struct range_entry *ptr; + struct memranges test_memrange; + struct resource *res_mock = *state; + resource_t prev_base = 0; + + memranges_init(&test_memrange, cacheable, cacheable, CACHEABLE_TAG); + memranges_add_resources(&test_memrange, reserved, reserved, RESERVED_TAG); + + /* There should be two entries, since cacheable and + reserved regions are not neighbors */ + memranges_each_entry(ptr, &test_memrange) { + assert_in_range(range_entry_tag(ptr), CACHEABLE_TAG, RESERVED_TAG); + assert_int_equal(range_entry_base(ptr), get_aligned_base(res_mock, ptr)); + + assert_int_equal(range_entry_end(ptr), get_aligned_end(res_mock, ptr)); + + /* Ranges have to be returned in increasing order */ + assert_true(prev_base <= range_entry_base(ptr)); + + prev_base = range_entry_base(ptr); + counter++; + }; + assert_int_equal(counter, 2); + counter = 0; + + /* Remove initial memrange */ + memranges_teardown(&test_memrange); + memranges_each_entry(ptr, &test_memrange) + counter++; + assert_int_equal(counter, 0); +} + +/* + * This test verifies memranges_clone(), memranges_insert() and memranges_update_tag() + * functions. All operations are performed on cloned memrange. One of the most important thing + * to check, is that memrange_insert() should remove all ranges which are covered by the newly + * inserted one. + * + * Example memory ranges (res_mock1) for test_memrange_clone_insert. + * Ranges marked with asterisks (***) are not added to the clone_memrange. + * Ranges marked with (^) have tag value changed during test. + * + * +--------CACHEABLE_TAG--------+ <-0xE000 + * +------|----INSERTED_TAG----------+ | <-0xF000 + * | | (^READONLY_TAG^) | | + * | | | | + * | +-----------------------------+ <-0x100000 + * +---------------------------------+ <-0x101000 + * + * + * +-----***READONLY_TAG***------+ <-0xFF0000 + * | | + * | | + * | | + * +-----------------------------+ <-0x1000000 + * + * + * +------+---------RESERVED_TAG-----+--+ <-0x100000000 + * | | | | + * | +-----------------------------+ <-0x100001000 + * +-----------INSERTED_TAG----------+ <-0x100002000 + */ +static void test_memrange_clone_insert(void **state) +{ + int counter = 0; + const unsigned long cacheable = IORESOURCE_CACHEABLE; + const unsigned long reserved = IORESOURCE_RESERVE; + struct range_entry *ptr; + struct memranges test_memrange, clone_memrange; + struct resource *res_mock = *state; + const resource_t new_range_begin_offset = 1ULL << 12; + + memranges_init(&test_memrange, cacheable, cacheable, CACHEABLE_TAG); + memranges_add_resources(&test_memrange, reserved, reserved, RESERVED_TAG); + + memranges_clone(&clone_memrange, &test_memrange); + memranges_teardown(&test_memrange); + + /* Verify that new one is really a clone */ + memranges_each_entry(ptr, &clone_memrange) { + assert_in_range(range_entry_tag(ptr), CACHEABLE_TAG, END_OF_RESOURCES - 1); + assert_int_equal(range_entry_base(ptr), get_aligned_base(res_mock, ptr)); + + assert_int_equal(range_entry_end(ptr), get_aligned_end(res_mock, ptr)); + + counter++; + }; + assert_int_equal(counter, 2); + counter = 0; + + /* Insert new range, which will overlap with first region. */ + memranges_insert(&clone_memrange, res_mock[CACHEABLE_TAG].base + new_range_begin_offset, + res_mock[CACHEABLE_TAG].size, INSERTED_TAG); + + /* Three ranges should be there - CACHEABLE(shrunk), INSERTED and RESERVED */ + memranges_each_entry(ptr, &clone_memrange) { + resource_t expected_end; + + if (range_entry_tag(ptr) == CACHEABLE_TAG) { + assert_int_equal(range_entry_base(ptr), res_mock[CACHEABLE_TAG].base); + + expected_end = res_mock[CACHEABLE_TAG].base + new_range_begin_offset; + assert_int_equal(range_entry_end(ptr), expected_end); + } + if (range_entry_tag(ptr) == INSERTED_TAG) { + assert_int_equal(range_entry_base(ptr), + res_mock[CACHEABLE_TAG].base + new_range_begin_offset); + + expected_end = res_mock[CACHEABLE_TAG].base + new_range_begin_offset + + res_mock[CACHEABLE_TAG].size; + assert_int_equal(range_entry_end(ptr), + ALIGN_UP(expected_end, MEMRANGE_ALIGN)); + } + counter++; + } + assert_int_equal(counter, 3); + counter = 0; + + /* Insert new region, which will shadow readonly range. + * Additionally verify API for updating tags */ + memranges_update_tag(&clone_memrange, INSERTED_TAG, READONLY_TAG); + + memranges_each_entry(ptr, &clone_memrange) { + resource_t expected_end; + + assert_int_not_equal(range_entry_tag(ptr), INSERTED_TAG); + if (range_entry_tag(ptr) == READONLY_TAG) { + assert_int_equal(range_entry_base(ptr), + res_mock[CACHEABLE_TAG].base + new_range_begin_offset); + + expected_end = res_mock[CACHEABLE_TAG].base + new_range_begin_offset + + res_mock[CACHEABLE_TAG].size; + assert_int_equal(range_entry_end(ptr), + ALIGN_UP(expected_end, MEMRANGE_ALIGN)); + } + }; + + /* Check if alignment (4KiB) is properly applied, that is begin - DOWN and end - UP */ + memranges_insert(&clone_memrange, res_mock[RESERVED_TAG].base + 0xAD, + res_mock[RESERVED_TAG].size, INSERTED_TAG); + + memranges_each_entry(ptr, &clone_memrange) { + resource_t expected_end; + + assert_int_not_equal(range_entry_tag(ptr), RESERVED_TAG); + if (range_entry_tag(ptr) == INSERTED_TAG) { + assert_int_equal(range_entry_base(ptr), + ALIGN_DOWN(res_mock[RESERVED_TAG].base, + MEMRANGE_ALIGN)); + + expected_end = ALIGN_DOWN(res_mock[RESERVED_TAG].base, MEMRANGE_ALIGN) + + new_range_begin_offset + res_mock[RESERVED_TAG].size; + expected_end = ALIGN_UP(expected_end, MEMRANGE_ALIGN); + + assert_int_equal(range_entry_end(ptr), expected_end); + } + counter++; + } + assert_int_equal(counter, 3); + + /* Free clone */ + memranges_teardown(&clone_memrange); +} + +/* + * This test verifies memranges_fill_holes_up_to() and memranges_create_hole(). Idea of the test + * is to fill all holes, so that we end up with contiguous address space fully covered by + * entries. Then, holes are created on the border of two different regions + * + * Example memory ranges (res_mock1) for test_memrange_holes. + * Space marked with (/) is not covered by any region at the end of the test. + * + * +--------CACHEABLE_TAG--------+ <-0xE000 + * | | + * | | + * //|/////////////////////////////| <-0xFF000 + * //+-----------HOLE_TAG----------+ <-0x100000 + * //|/////////////////////////////| <-0x101000 + * | | + * | | + * | | + * | | + * +--------RESERVED_TAG---------+ <-0x100000000 + * | | + * +-----------------------------+ <-0x100001000 + */ +static void test_memrange_holes(void **state) +{ + int counter = 0; + const unsigned long cacheable = IORESOURCE_CACHEABLE; + const unsigned long reserved = IORESOURCE_RESERVE; + struct range_entry *ptr; + struct range_entry *hole_ptr = NULL; + struct memranges test_memrange; + struct resource *res_mock = *state; + int holes_found = 0; + resource_t last_range_end = 0; + const resource_t holes_fill_end = res_mock[RESERVED_TAG].base; + + memranges_init(&test_memrange, cacheable, cacheable, CACHEABLE_TAG); + memranges_add_resources(&test_memrange, reserved, reserved, RESERVED_TAG); + + /* Count holes in ranges */ + memranges_each_entry(ptr, &test_memrange) { + if (!last_range_end) { + last_range_end = range_entry_end(ptr); + continue; + } + + + if (range_entry_base(ptr) != last_range_end) { + holes_found++; + last_range_end = range_entry_end(ptr); + } + + if (range_entry_base(ptr) >= holes_fill_end) + break; + } + + /* Create range entries which covers continuous memory range + (but with different tags) */ + memranges_fill_holes_up_to(&test_memrange, holes_fill_end, HOLE_TAG); + + memranges_each_entry(ptr, &test_memrange) { + if (range_entry_tag(ptr) == HOLE_TAG) { + assert_int_equal(range_entry_base(ptr), + ALIGN_UP(res_mock[CACHEABLE_TAG].base + + res_mock[CACHEABLE_TAG].size, + MEMRANGE_ALIGN)); + assert_int_equal(range_entry_end(ptr), holes_fill_end); + /* Store pointer to HOLE_TAG region for future use */ + hole_ptr = ptr; + } + counter++; + } + assert_int_equal(counter, 2 + holes_found); + + /* If test data does not have any holes in it then terminate this test */ + if (holes_found == 0) + return; + + assert_non_null(hole_ptr); + counter = 0; + + /* Create hole crossing the border of two range entries */ + const resource_t new_cacheable_end = ALIGN_DOWN( + res_mock[CACHEABLE_TAG].base + res_mock[CACHEABLE_TAG].size - 4 * KiB, + MEMRANGE_ALIGN); + const resource_t new_hole_begin = ALIGN_UP(range_entry_base(hole_ptr) + 4 * KiB, + MEMRANGE_ALIGN); + const resource_t ranges_diff = new_hole_begin - new_cacheable_end; + + memranges_create_hole(&test_memrange, new_cacheable_end, ranges_diff); + + memranges_each_entry(ptr, &test_memrange) { + switch (range_entry_tag(ptr)) { + case CACHEABLE_TAG: + assert_int_equal(range_entry_base(ptr), res_mock[CACHEABLE_TAG].base); + assert_int_equal(range_entry_end(ptr), new_cacheable_end); + break; + case RESERVED_TAG: + assert_int_equal(range_entry_base(ptr), res_mock[RESERVED_TAG].base); + assert_int_equal(range_entry_end(ptr), res_mock[RESERVED_TAG].base + + res_mock[RESERVED_TAG].size); + break; + case HOLE_TAG: + assert_int_equal(range_entry_base(ptr), new_hole_begin); + assert_int_equal(range_entry_end(ptr), res_mock[RESERVED_TAG].base); + break; + default: + break; + } + counter++; + } + assert_int_equal(counter, 3); + + memranges_teardown(&test_memrange); +} + +/* + * This test verifies memranges_steal() function. Simple check is done by attempt so steal some + * memory from region with READONLY_TAG. + * + * Example memory ranges (res_mock1) for test_memrange_steal. + * Space marked with (/) is not covered by any region at the end of the test. + * + * +--------CACHEABLE_TAG--------+ <-0xE000 + * | | + * | | + * | | + * +-----------------------------+ <-0x100000 + * + * + * + * +--------READONLY_TAG---------+ <-0xFF0000 + * | | + * |/////////////////////////////| <-stolen_base + * |/////////////////////////////| <-stolen_base + 0x4000 + * +-----------------------------+ <-0x1000000 + * + * + * +--------RESERVED_TAG---------+ <-0x100000000 + * | | + * +-----------------------------+ <-0x100001000 + */ +static void test_memrange_steal(void **state) +{ + bool status = false; + resource_t stolen; + const unsigned long cacheable = IORESOURCE_CACHEABLE; + const unsigned long reserved = IORESOURCE_RESERVE; + const unsigned long readonly = IORESOURCE_READONLY; + const resource_t stolen_range_size = 0x4000; + struct memranges test_memrange; + struct resource *res_mock = *state; + struct range_entry *ptr; + size_t count = 0; + + memranges_init(&test_memrange, cacheable, cacheable, CACHEABLE_TAG); + memranges_add_resources(&test_memrange, reserved, reserved, RESERVED_TAG); + memranges_add_resources(&test_memrange, readonly, readonly, READONLY_TAG); + + status = memranges_steal(&test_memrange, res_mock[RESERVED_TAG].base + + res_mock[RESERVED_TAG].size, + stolen_range_size, 12, READONLY_TAG, &stolen); + assert_true(status); + assert_in_range(stolen, res_mock[READONLY_TAG].base, res_mock[READONLY_TAG].base + + res_mock[READONLY_TAG].size); + + memranges_each_entry(ptr, &test_memrange) { + if (range_entry_tag(ptr) == READONLY_TAG) { + assert_int_equal(range_entry_base(ptr), + ALIGN_DOWN(res_mock[READONLY_TAG].base, MEMRANGE_ALIGN) + + stolen_range_size); + } + count++; + } + assert_int_equal(count, 3); + count = 0; + + /* Check if inserting range in previously stolen area will merge it. */ + memranges_insert(&test_memrange, res_mock[READONLY_TAG].base + 0xCC, stolen_range_size, + READONLY_TAG); + memranges_each_entry(ptr, &test_memrange) { + if (range_entry_tag(ptr) == READONLY_TAG) { + assert_int_equal(range_entry_base(ptr), + ALIGN_DOWN(res_mock[READONLY_TAG].base, + MEMRANGE_ALIGN)); + assert_int_equal(range_entry_end(ptr), + ALIGN_UP(range_entry_base(ptr) + + res_mock[READONLY_TAG].size, + MEMRANGE_ALIGN)); + } + count++; + } + assert_int_equal(count, 3); + count = 0; + + memranges_teardown(&test_memrange); +} + +/* Utility function checking number of entries and alignment of their base and end pointers */ +static void check_range_entries_count_and_alignment(struct memranges *ranges, + size_t ranges_count, resource_t alignment) +{ + size_t count = 0; + struct range_entry *ptr; + + memranges_each_entry(ptr, ranges) { + assert_true(IS_ALIGNED(range_entry_base(ptr), alignment)); + assert_true(IS_ALIGNED(range_entry_end(ptr), alignment)); + + count++; + } + assert_int_equal(ranges_count, count); +} + +/* This test verifies memranges_init*() and memranges_teardown() functions. + Added ranges are checked correct count and alignment. */ +static void test_memrange_init_and_teardown(void **state) +{ + const unsigned long cacheable = IORESOURCE_CACHEABLE; + const unsigned long reserved = IORESOURCE_RESERVE; + const unsigned long readonly = IORESOURCE_READONLY; + struct memranges test_memrange; + struct range_entry range_entries[4] = { 0 }; + + /* Test memranges_init() correctness */ + memranges_init(&test_memrange, cacheable, cacheable, CACHEABLE_TAG); + memranges_add_resources(&test_memrange, reserved, reserved, RESERVED_TAG); + memranges_add_resources(&test_memrange, readonly, readonly, READONLY_TAG); + + /* Expect all entries to be aligned to 4KiB (2^12) */ + check_range_entries_count_and_alignment(&test_memrange, 3, MEMRANGE_ALIGN); + + /* Expect ranges list to be empty after teardown */ + memranges_teardown(&test_memrange); + assert_true(memranges_is_empty(&test_memrange)); + + + /* Test memranges_init_with_alignment() correctness with alignment of 1KiB (2^10) */ + memranges_init_with_alignment(&test_memrange, cacheable, cacheable, + CACHEABLE_TAG, 10); + memranges_add_resources(&test_memrange, reserved, reserved, RESERVED_TAG); + memranges_add_resources(&test_memrange, readonly, readonly, READONLY_TAG); + + check_range_entries_count_and_alignment(&test_memrange, 3, POWER_OF_2(10)); + + memranges_teardown(&test_memrange); + assert_true(memranges_is_empty(&test_memrange)); + + + /* Test memranges_init_empty() correctness */ + memranges_init_empty(&test_memrange, &range_entries[0], ARRAY_SIZE(range_entries)); + assert_true(memranges_is_empty(&test_memrange)); + + memranges_add_resources(&test_memrange, cacheable, cacheable, CACHEABLE_TAG); + memranges_add_resources(&test_memrange, reserved, reserved, RESERVED_TAG); + memranges_add_resources(&test_memrange, readonly, readonly, READONLY_TAG); + + check_range_entries_count_and_alignment(&test_memrange, 3, MEMRANGE_ALIGN); + + memranges_teardown(&test_memrange); + assert_true(memranges_is_empty(&test_memrange)); + + + /* Test memranges_init_with_alignment() correctness with alignment of 8KiB (2^13) */ + memranges_init_empty_with_alignment(&test_memrange, &range_entries[0], + ARRAY_SIZE(range_entries), 13); + assert_true(memranges_is_empty(&test_memrange)); + + memranges_add_resources(&test_memrange, cacheable, cacheable, CACHEABLE_TAG); + memranges_add_resources(&test_memrange, reserved, reserved, RESERVED_TAG); + memranges_add_resources(&test_memrange, readonly, readonly, READONLY_TAG); + + check_range_entries_count_and_alignment(&test_memrange, 3, POWER_OF_2(13)); + + memranges_teardown(&test_memrange); + assert_true(memranges_is_empty(&test_memrange)); +} + +/* Filter function accepting ranges having memory resource flag */ +static int memrange_filter_mem_only(struct device *dev, struct resource *res) +{ + /* Accept only memory resources */ + return res->flags & IORESOURCE_MEM; +} + +/* Filter function rejecting ranges having memory resource flag */ +static int memrange_filter_non_mem(struct device *dev, struct resource *res) +{ + /* Accept only memory resources */ + return !(res->flags & IORESOURCE_MEM); +} + +/* This test verifies memranges_add_resources_filter() function by providing filter functions + which accept or reject ranges. */ +static void test_memrange_add_resources_filter(void **state) +{ + const unsigned long cacheable = IORESOURCE_CACHEABLE; + const unsigned long reserved = IORESOURCE_RESERVE; + struct memranges test_memrange; + struct range_entry *ptr; + size_t count = 0; + size_t accepted_tags[] = {CACHEABLE_TAG, RESERVED_TAG}; + + /* Check if filter accepts range correctly */ + memranges_init(&test_memrange, reserved, reserved, RESERVED_TAG); + memranges_add_resources_filter(&test_memrange, cacheable, cacheable, CACHEABLE_TAG, + memrange_filter_mem_only); + + /* Check if filter accepted desired range. */ + memranges_each_entry(ptr, &test_memrange) { + assert_in_set(range_entry_tag(ptr), accepted_tags, ARRAY_SIZE(accepted_tags)); + assert_true(IS_ALIGNED(range_entry_base(ptr), MEMRANGE_ALIGN)); + assert_true(IS_ALIGNED(range_entry_end(ptr), MEMRANGE_ALIGN)); + count++; + } + assert_int_equal(2, count); + count = 0; + memranges_teardown(&test_memrange); + + /* Check if filter rejects range correctly */ + memranges_init(&test_memrange, reserved, reserved, RESERVED_TAG); + memranges_add_resources_filter(&test_memrange, cacheable, cacheable, CACHEABLE_TAG, + memrange_filter_non_mem); + + check_range_entries_count_and_alignment(&test_memrange, 1, MEMRANGE_ALIGN); + + memranges_teardown(&test_memrange); +} + +int main(void) +{ + const struct CMUnitTest tests[] = { + cmocka_unit_test(test_memrange_basic), + cmocka_unit_test(test_memrange_clone_insert), + cmocka_unit_test(test_memrange_holes), + cmocka_unit_test(test_memrange_steal), + cmocka_unit_test(test_memrange_init_and_teardown), + cmocka_unit_test(test_memrange_add_resources_filter), + }; + + return cmocka_run_group_tests_name("Boundary on 4GiB", + tests, setup_test_1, NULL) + + cmocka_run_group_tests_name("Boundaries 1 byte from 4GiB", + tests, setup_test_2, NULL) + + cmocka_run_group_tests_name("Range over 4GiB boundary", + tests, setup_test_3, NULL); +} |