| // SPDX-License-Identifier: GPL-2.0-only |
| #define pr_fmt(fmt) "min_heap_test: " fmt |
| |
| /* |
| * Test cases for the min max heap. |
| */ |
| |
| #include <linux/log2.h> |
| #include <linux/min_heap.h> |
| #include <linux/module.h> |
| #include <linux/printk.h> |
| #include <linux/random.h> |
| |
| static __init bool less_than(const void *lhs, const void *rhs) |
| { |
| return *(int *)lhs < *(int *)rhs; |
| } |
| |
| static __init bool greater_than(const void *lhs, const void *rhs) |
| { |
| return *(int *)lhs > *(int *)rhs; |
| } |
| |
| static __init void swap_ints(void *lhs, void *rhs) |
| { |
| int temp = *(int *)lhs; |
| |
| *(int *)lhs = *(int *)rhs; |
| *(int *)rhs = temp; |
| } |
| |
| static __init int pop_verify_heap(bool min_heap, |
| struct min_heap *heap, |
| const struct min_heap_callbacks *funcs) |
| { |
| int *values = heap->data; |
| int err = 0; |
| int last; |
| |
| last = values[0]; |
| min_heap_pop(heap, funcs); |
| while (heap->nr > 0) { |
| if (min_heap) { |
| if (last > values[0]) { |
| pr_err("error: expected %d <= %d\n", last, |
| values[0]); |
| err++; |
| } |
| } else { |
| if (last < values[0]) { |
| pr_err("error: expected %d >= %d\n", last, |
| values[0]); |
| err++; |
| } |
| } |
| last = values[0]; |
| min_heap_pop(heap, funcs); |
| } |
| return err; |
| } |
| |
| static __init int test_heapify_all(bool min_heap) |
| { |
| int values[] = { 3, 1, 2, 4, 0x8000000, 0x7FFFFFF, 0, |
| -3, -1, -2, -4, 0x8000000, 0x7FFFFFF }; |
| struct min_heap heap = { |
| .data = values, |
| .nr = ARRAY_SIZE(values), |
| .size = ARRAY_SIZE(values), |
| }; |
| struct min_heap_callbacks funcs = { |
| .elem_size = sizeof(int), |
| .less = min_heap ? less_than : greater_than, |
| .swp = swap_ints, |
| }; |
| int i, err; |
| |
| /* Test with known set of values. */ |
| min_heapify_all(&heap, &funcs); |
| err = pop_verify_heap(min_heap, &heap, &funcs); |
| |
| |
| /* Test with randomly generated values. */ |
| heap.nr = ARRAY_SIZE(values); |
| for (i = 0; i < heap.nr; i++) |
| values[i] = get_random_u32(); |
| |
| min_heapify_all(&heap, &funcs); |
| err += pop_verify_heap(min_heap, &heap, &funcs); |
| |
| return err; |
| } |
| |
| static __init int test_heap_push(bool min_heap) |
| { |
| const int data[] = { 3, 1, 2, 4, 0x80000000, 0x7FFFFFFF, 0, |
| -3, -1, -2, -4, 0x80000000, 0x7FFFFFFF }; |
| int values[ARRAY_SIZE(data)]; |
| struct min_heap heap = { |
| .data = values, |
| .nr = 0, |
| .size = ARRAY_SIZE(values), |
| }; |
| struct min_heap_callbacks funcs = { |
| .elem_size = sizeof(int), |
| .less = min_heap ? less_than : greater_than, |
| .swp = swap_ints, |
| }; |
| int i, temp, err; |
| |
| /* Test with known set of values copied from data. */ |
| for (i = 0; i < ARRAY_SIZE(data); i++) |
| min_heap_push(&heap, &data[i], &funcs); |
| |
| err = pop_verify_heap(min_heap, &heap, &funcs); |
| |
| /* Test with randomly generated values. */ |
| while (heap.nr < heap.size) { |
| temp = get_random_u32(); |
| min_heap_push(&heap, &temp, &funcs); |
| } |
| err += pop_verify_heap(min_heap, &heap, &funcs); |
| |
| return err; |
| } |
| |
| static __init int test_heap_pop_push(bool min_heap) |
| { |
| const int data[] = { 3, 1, 2, 4, 0x80000000, 0x7FFFFFFF, 0, |
| -3, -1, -2, -4, 0x80000000, 0x7FFFFFFF }; |
| int values[ARRAY_SIZE(data)]; |
| struct min_heap heap = { |
| .data = values, |
| .nr = 0, |
| .size = ARRAY_SIZE(values), |
| }; |
| struct min_heap_callbacks funcs = { |
| .elem_size = sizeof(int), |
| .less = min_heap ? less_than : greater_than, |
| .swp = swap_ints, |
| }; |
| int i, temp, err; |
| |
| /* Fill values with data to pop and replace. */ |
| temp = min_heap ? 0x80000000 : 0x7FFFFFFF; |
| for (i = 0; i < ARRAY_SIZE(data); i++) |
| min_heap_push(&heap, &temp, &funcs); |
| |
| /* Test with known set of values copied from data. */ |
| for (i = 0; i < ARRAY_SIZE(data); i++) |
| min_heap_pop_push(&heap, &data[i], &funcs); |
| |
| err = pop_verify_heap(min_heap, &heap, &funcs); |
| |
| heap.nr = 0; |
| for (i = 0; i < ARRAY_SIZE(data); i++) |
| min_heap_push(&heap, &temp, &funcs); |
| |
| /* Test with randomly generated values. */ |
| for (i = 0; i < ARRAY_SIZE(data); i++) { |
| temp = get_random_u32(); |
| min_heap_pop_push(&heap, &temp, &funcs); |
| } |
| err += pop_verify_heap(min_heap, &heap, &funcs); |
| |
| return err; |
| } |
| |
| static int __init test_min_heap_init(void) |
| { |
| int err = 0; |
| |
| err += test_heapify_all(true); |
| err += test_heapify_all(false); |
| err += test_heap_push(true); |
| err += test_heap_push(false); |
| err += test_heap_pop_push(true); |
| err += test_heap_pop_push(false); |
| if (err) { |
| pr_err("test failed with %d errors\n", err); |
| return -EINVAL; |
| } |
| pr_info("test passed\n"); |
| return 0; |
| } |
| module_init(test_min_heap_init); |
| |
| static void __exit test_min_heap_exit(void) |
| { |
| /* do nothing */ |
| } |
| module_exit(test_min_heap_exit); |
| |
| MODULE_LICENSE("GPL"); |