| // SPDX-License-Identifier: GPL-2.0 |
| /* Copyright (c) 2019 Facebook */ |
| #include <stdio.h> |
| #include <errno.h> |
| #include <string.h> |
| #include <unistd.h> |
| |
| #include <bpf/bpf.h> |
| #include <bpf/libbpf.h> |
| |
| #include <bpf_util.h> |
| #include <test_maps.h> |
| |
| static void map_batch_update(int map_fd, __u32 max_entries, int *keys, |
| void *values, bool is_pcpu) |
| { |
| typedef BPF_DECLARE_PERCPU(int, value); |
| value *v = NULL; |
| int i, j, err; |
| DECLARE_LIBBPF_OPTS(bpf_map_batch_opts, opts, |
| .elem_flags = 0, |
| .flags = 0, |
| ); |
| |
| if (is_pcpu) |
| v = (value *)values; |
| |
| for (i = 0; i < max_entries; i++) { |
| keys[i] = i + 1; |
| if (is_pcpu) |
| for (j = 0; j < bpf_num_possible_cpus(); j++) |
| bpf_percpu(v[i], j) = i + 2 + j; |
| else |
| ((int *)values)[i] = i + 2; |
| } |
| |
| err = bpf_map_update_batch(map_fd, keys, values, &max_entries, &opts); |
| CHECK(err, "bpf_map_update_batch()", "error:%s\n", strerror(errno)); |
| } |
| |
| static void map_batch_verify(int *visited, __u32 max_entries, |
| int *keys, void *values, bool is_pcpu) |
| { |
| typedef BPF_DECLARE_PERCPU(int, value); |
| value *v = NULL; |
| int i, j; |
| |
| if (is_pcpu) |
| v = (value *)values; |
| |
| memset(visited, 0, max_entries * sizeof(*visited)); |
| for (i = 0; i < max_entries; i++) { |
| |
| if (is_pcpu) { |
| for (j = 0; j < bpf_num_possible_cpus(); j++) { |
| CHECK(keys[i] + 1 + j != bpf_percpu(v[i], j), |
| "key/value checking", |
| "error: i %d j %d key %d value %d\n", |
| i, j, keys[i], bpf_percpu(v[i], j)); |
| } |
| } else { |
| CHECK(keys[i] + 1 != ((int *)values)[i], |
| "key/value checking", |
| "error: i %d key %d value %d\n", i, keys[i], |
| ((int *)values)[i]); |
| } |
| |
| visited[i] = 1; |
| |
| } |
| for (i = 0; i < max_entries; i++) { |
| CHECK(visited[i] != 1, "visited checking", |
| "error: keys array at index %d missing\n", i); |
| } |
| } |
| |
| void __test_map_lookup_and_delete_batch(bool is_pcpu) |
| { |
| __u32 batch, count, total, total_success; |
| typedef BPF_DECLARE_PERCPU(int, value); |
| int map_fd, *keys, *visited, key; |
| const __u32 max_entries = 10; |
| value pcpu_values[max_entries]; |
| int err, step, value_size; |
| bool nospace_err; |
| void *values; |
| DECLARE_LIBBPF_OPTS(bpf_map_batch_opts, opts, |
| .elem_flags = 0, |
| .flags = 0, |
| ); |
| |
| map_fd = bpf_map_create(is_pcpu ? BPF_MAP_TYPE_PERCPU_HASH : BPF_MAP_TYPE_HASH, |
| "hash_map", sizeof(int), sizeof(int), max_entries, NULL); |
| CHECK(map_fd == -1, |
| "bpf_map_create()", "error:%s\n", strerror(errno)); |
| |
| value_size = is_pcpu ? sizeof(value) : sizeof(int); |
| keys = malloc(max_entries * sizeof(int)); |
| if (is_pcpu) |
| values = pcpu_values; |
| else |
| values = malloc(max_entries * sizeof(int)); |
| visited = malloc(max_entries * sizeof(int)); |
| CHECK(!keys || !values || !visited, "malloc()", |
| "error:%s\n", strerror(errno)); |
| |
| /* test 1: lookup/delete an empty hash table, -ENOENT */ |
| count = max_entries; |
| err = bpf_map_lookup_and_delete_batch(map_fd, NULL, &batch, keys, |
| values, &count, &opts); |
| CHECK((err && errno != ENOENT), "empty map", |
| "error: %s\n", strerror(errno)); |
| |
| /* populate elements to the map */ |
| map_batch_update(map_fd, max_entries, keys, values, is_pcpu); |
| |
| /* test 2: lookup/delete with count = 0, success */ |
| count = 0; |
| err = bpf_map_lookup_and_delete_batch(map_fd, NULL, &batch, keys, |
| values, &count, &opts); |
| CHECK(err, "count = 0", "error: %s\n", strerror(errno)); |
| |
| /* test 3: lookup/delete with count = max_entries, success */ |
| memset(keys, 0, max_entries * sizeof(*keys)); |
| memset(values, 0, max_entries * value_size); |
| count = max_entries; |
| err = bpf_map_lookup_and_delete_batch(map_fd, NULL, &batch, keys, |
| values, &count, &opts); |
| CHECK((err && errno != ENOENT), "count = max_entries", |
| "error: %s\n", strerror(errno)); |
| CHECK(count != max_entries, "count = max_entries", |
| "count = %u, max_entries = %u\n", count, max_entries); |
| map_batch_verify(visited, max_entries, keys, values, is_pcpu); |
| |
| /* bpf_map_get_next_key() should return -ENOENT for an empty map. */ |
| err = bpf_map_get_next_key(map_fd, NULL, &key); |
| CHECK(!err, "bpf_map_get_next_key()", "error: %s\n", strerror(errno)); |
| |
| /* test 4: lookup/delete in a loop with various steps. */ |
| total_success = 0; |
| for (step = 1; step < max_entries; step++) { |
| map_batch_update(map_fd, max_entries, keys, values, is_pcpu); |
| memset(keys, 0, max_entries * sizeof(*keys)); |
| memset(values, 0, max_entries * value_size); |
| total = 0; |
| /* iteratively lookup/delete elements with 'step' |
| * elements each |
| */ |
| count = step; |
| nospace_err = false; |
| while (true) { |
| err = bpf_map_lookup_batch(map_fd, |
| total ? &batch : NULL, |
| &batch, keys + total, |
| values + |
| total * value_size, |
| &count, &opts); |
| /* It is possible that we are failing due to buffer size |
| * not big enough. In such cases, let us just exit and |
| * go with large steps. Not that a buffer size with |
| * max_entries should always work. |
| */ |
| if (err && errno == ENOSPC) { |
| nospace_err = true; |
| break; |
| } |
| |
| CHECK((err && errno != ENOENT), "lookup with steps", |
| "error: %s\n", strerror(errno)); |
| |
| total += count; |
| if (err) |
| break; |
| |
| } |
| if (nospace_err == true) |
| continue; |
| |
| CHECK(total != max_entries, "lookup with steps", |
| "total = %u, max_entries = %u\n", total, max_entries); |
| map_batch_verify(visited, max_entries, keys, values, is_pcpu); |
| |
| total = 0; |
| count = step; |
| while (total < max_entries) { |
| if (max_entries - total < step) |
| count = max_entries - total; |
| err = bpf_map_delete_batch(map_fd, |
| keys + total, |
| &count, &opts); |
| CHECK((err && errno != ENOENT), "delete batch", |
| "error: %s\n", strerror(errno)); |
| total += count; |
| if (err) |
| break; |
| } |
| CHECK(total != max_entries, "delete with steps", |
| "total = %u, max_entries = %u\n", total, max_entries); |
| |
| /* check map is empty, errno == ENOENT */ |
| err = bpf_map_get_next_key(map_fd, NULL, &key); |
| CHECK(!err || errno != ENOENT, "bpf_map_get_next_key()", |
| "error: %s\n", strerror(errno)); |
| |
| /* iteratively lookup/delete elements with 'step' |
| * elements each |
| */ |
| map_batch_update(map_fd, max_entries, keys, values, is_pcpu); |
| memset(keys, 0, max_entries * sizeof(*keys)); |
| memset(values, 0, max_entries * value_size); |
| total = 0; |
| count = step; |
| nospace_err = false; |
| while (true) { |
| err = bpf_map_lookup_and_delete_batch(map_fd, |
| total ? &batch : NULL, |
| &batch, keys + total, |
| values + |
| total * value_size, |
| &count, &opts); |
| /* It is possible that we are failing due to buffer size |
| * not big enough. In such cases, let us just exit and |
| * go with large steps. Not that a buffer size with |
| * max_entries should always work. |
| */ |
| if (err && errno == ENOSPC) { |
| nospace_err = true; |
| break; |
| } |
| |
| CHECK((err && errno != ENOENT), "lookup with steps", |
| "error: %s\n", strerror(errno)); |
| |
| total += count; |
| if (err) |
| break; |
| } |
| |
| if (nospace_err == true) |
| continue; |
| |
| CHECK(total != max_entries, "lookup/delete with steps", |
| "total = %u, max_entries = %u\n", total, max_entries); |
| |
| map_batch_verify(visited, max_entries, keys, values, is_pcpu); |
| err = bpf_map_get_next_key(map_fd, NULL, &key); |
| CHECK(!err, "bpf_map_get_next_key()", "error: %s\n", |
| strerror(errno)); |
| |
| total_success++; |
| } |
| |
| CHECK(total_success == 0, "check total_success", |
| "unexpected failure\n"); |
| free(keys); |
| free(visited); |
| if (!is_pcpu) |
| free(values); |
| close(map_fd); |
| } |
| |
| void htab_map_batch_ops(void) |
| { |
| __test_map_lookup_and_delete_batch(false); |
| printf("test_%s:PASS\n", __func__); |
| } |
| |
| void htab_percpu_map_batch_ops(void) |
| { |
| __test_map_lookup_and_delete_batch(true); |
| printf("test_%s:PASS\n", __func__); |
| } |
| |
| void test_htab_map_batch_ops(void) |
| { |
| htab_map_batch_ops(); |
| htab_percpu_map_batch_ops(); |
| } |