|  | // SPDX-License-Identifier: GPL-2.0 | 
|  | #include <test_progs.h> | 
|  | #include <network_helpers.h> | 
|  |  | 
|  | static void test_global_data_number(struct bpf_object *obj, __u32 duration) | 
|  | { | 
|  | int i, err, map_fd; | 
|  | __u64 num; | 
|  |  | 
|  | map_fd = bpf_find_map(__func__, obj, "result_number"); | 
|  | if (CHECK_FAIL(map_fd < 0)) | 
|  | return; | 
|  |  | 
|  | struct { | 
|  | char *name; | 
|  | uint32_t key; | 
|  | __u64 num; | 
|  | } tests[] = { | 
|  | { "relocate .bss reference",     0, 0 }, | 
|  | { "relocate .data reference",    1, 42 }, | 
|  | { "relocate .rodata reference",  2, 24 }, | 
|  | { "relocate .bss reference",     3, 0 }, | 
|  | { "relocate .data reference",    4, 0xffeeff }, | 
|  | { "relocate .rodata reference",  5, 0xabab }, | 
|  | { "relocate .bss reference",     6, 1234 }, | 
|  | { "relocate .bss reference",     7, 0 }, | 
|  | { "relocate .rodata reference",  8, 0xab }, | 
|  | { "relocate .rodata reference",  9, 0x1111111111111111 }, | 
|  | { "relocate .rodata reference", 10, ~0 }, | 
|  | }; | 
|  |  | 
|  | for (i = 0; i < ARRAY_SIZE(tests); i++) { | 
|  | err = bpf_map_lookup_elem(map_fd, &tests[i].key, &num); | 
|  | CHECK(err || num != tests[i].num, tests[i].name, | 
|  | "err %d result %llx expected %llx\n", | 
|  | err, num, tests[i].num); | 
|  | } | 
|  | } | 
|  |  | 
|  | static void test_global_data_string(struct bpf_object *obj, __u32 duration) | 
|  | { | 
|  | int i, err, map_fd; | 
|  | char str[32]; | 
|  |  | 
|  | map_fd = bpf_find_map(__func__, obj, "result_string"); | 
|  | if (CHECK_FAIL(map_fd < 0)) | 
|  | return; | 
|  |  | 
|  | struct { | 
|  | char *name; | 
|  | uint32_t key; | 
|  | char str[32]; | 
|  | } tests[] = { | 
|  | { "relocate .rodata reference", 0, "abcdefghijklmnopqrstuvwxyz" }, | 
|  | { "relocate .data reference",   1, "abcdefghijklmnopqrstuvwxyz" }, | 
|  | { "relocate .bss reference",    2, "" }, | 
|  | { "relocate .data reference",   3, "abcdexghijklmnopqrstuvwxyz" }, | 
|  | { "relocate .bss reference",    4, "\0\0hello" }, | 
|  | }; | 
|  |  | 
|  | for (i = 0; i < ARRAY_SIZE(tests); i++) { | 
|  | err = bpf_map_lookup_elem(map_fd, &tests[i].key, str); | 
|  | CHECK(err || memcmp(str, tests[i].str, sizeof(str)), | 
|  | tests[i].name, "err %d result \'%s\' expected \'%s\'\n", | 
|  | err, str, tests[i].str); | 
|  | } | 
|  | } | 
|  |  | 
|  | struct foo { | 
|  | __u8  a; | 
|  | __u32 b; | 
|  | __u64 c; | 
|  | }; | 
|  |  | 
|  | static void test_global_data_struct(struct bpf_object *obj, __u32 duration) | 
|  | { | 
|  | int i, err, map_fd; | 
|  | struct foo val; | 
|  |  | 
|  | map_fd = bpf_find_map(__func__, obj, "result_struct"); | 
|  | if (CHECK_FAIL(map_fd < 0)) | 
|  | return; | 
|  |  | 
|  | struct { | 
|  | char *name; | 
|  | uint32_t key; | 
|  | struct foo val; | 
|  | } tests[] = { | 
|  | { "relocate .rodata reference", 0, { 42, 0xfefeefef, 0x1111111111111111ULL, } }, | 
|  | { "relocate .bss reference",    1, { } }, | 
|  | { "relocate .rodata reference", 2, { } }, | 
|  | { "relocate .data reference",   3, { 41, 0xeeeeefef, 0x2111111111111111ULL, } }, | 
|  | }; | 
|  |  | 
|  | for (i = 0; i < ARRAY_SIZE(tests); i++) { | 
|  | err = bpf_map_lookup_elem(map_fd, &tests[i].key, &val); | 
|  | CHECK(err || memcmp(&val, &tests[i].val, sizeof(val)), | 
|  | tests[i].name, "err %d result { %u, %u, %llu } expected { %u, %u, %llu }\n", | 
|  | err, val.a, val.b, val.c, tests[i].val.a, tests[i].val.b, tests[i].val.c); | 
|  | } | 
|  | } | 
|  |  | 
|  | static void test_global_data_rdonly(struct bpf_object *obj, __u32 duration) | 
|  | { | 
|  | int err = -ENOMEM, map_fd, zero = 0; | 
|  | struct bpf_map *map, *map2; | 
|  | __u8 *buff; | 
|  |  | 
|  | map = bpf_object__find_map_by_name(obj, "test_glo.rodata"); | 
|  | if (!ASSERT_OK_PTR(map, "map")) | 
|  | return; | 
|  | if (!ASSERT_TRUE(bpf_map__is_internal(map), "is_internal")) | 
|  | return; | 
|  |  | 
|  | /* ensure we can lookup internal maps by their ELF names */ | 
|  | map2 = bpf_object__find_map_by_name(obj, ".rodata"); | 
|  | if (!ASSERT_EQ(map, map2, "same_maps")) | 
|  | return; | 
|  |  | 
|  | map_fd = bpf_map__fd(map); | 
|  | if (CHECK_FAIL(map_fd < 0)) | 
|  | return; | 
|  |  | 
|  | buff = malloc(bpf_map__value_size(map)); | 
|  | if (buff) | 
|  | err = bpf_map_update_elem(map_fd, &zero, buff, 0); | 
|  | free(buff); | 
|  | CHECK(!err || errno != EPERM, "test .rodata read-only map", | 
|  | "err %d errno %d\n", err, errno); | 
|  | } | 
|  |  | 
|  | void test_global_data(void) | 
|  | { | 
|  | const char *file = "./test_global_data.bpf.o"; | 
|  | struct bpf_object *obj; | 
|  | int err, prog_fd; | 
|  | LIBBPF_OPTS(bpf_test_run_opts, topts, | 
|  | .data_in = &pkt_v4, | 
|  | .data_size_in = sizeof(pkt_v4), | 
|  | .repeat = 1, | 
|  | ); | 
|  |  | 
|  | err = bpf_prog_test_load(file, BPF_PROG_TYPE_SCHED_CLS, &obj, &prog_fd); | 
|  | if (!ASSERT_OK(err, "load program")) | 
|  | return; | 
|  |  | 
|  | err = bpf_prog_test_run_opts(prog_fd, &topts); | 
|  | ASSERT_OK(err, "pass global data run err"); | 
|  | ASSERT_OK(topts.retval, "pass global data run retval"); | 
|  |  | 
|  | test_global_data_number(obj, topts.duration); | 
|  | test_global_data_string(obj, topts.duration); | 
|  | test_global_data_struct(obj, topts.duration); | 
|  | test_global_data_rdonly(obj, topts.duration); | 
|  |  | 
|  | bpf_object__close(obj); | 
|  | } |