root/tools/testing/selftests/bpf/prog_tests/file_reader.c
// SPDX-License-Identifier: GPL-2.0
/* Copyright (c) 2025 Meta Platforms, Inc. and affiliates. */

#include <test_progs.h>
#include <network_helpers.h>
#include "file_reader.skel.h"
#include "file_reader_fail.skel.h"
#include <dlfcn.h>
#include <sys/mman.h>

const char *user_ptr = "hello world";
char file_contents[256000];

void *get_executable_base_addr(void)
{
        Dl_info info;

        if (!dladdr((void *)&get_executable_base_addr, &info)) {
                fprintf(stderr, "dladdr failed\n");
                return NULL;
        }

        return info.dli_fbase;
}

static int initialize_file_contents(void)
{
        int fd, page_sz = sysconf(_SC_PAGESIZE);
        ssize_t n = 0, cur, off;
        void *addr;

        fd = open("/proc/self/exe", O_RDONLY);
        if (!ASSERT_OK_FD(fd, "Open /proc/self/exe\n"))
                return 1;

        do {
                cur = read(fd, file_contents + n, sizeof(file_contents) - n);
                if (!ASSERT_GT(cur, 0, "read success"))
                        break;
                n += cur;
        } while (n < sizeof(file_contents));

        close(fd);

        if (!ASSERT_EQ(n, sizeof(file_contents), "Read /proc/self/exe\n"))
                return 1;

        addr = get_executable_base_addr();
        if (!ASSERT_NEQ(addr, NULL, "get executable address"))
                return 1;

        /* page-align base file address */
        addr = (void *)((unsigned long)addr & ~(page_sz - 1));

        /*
         * Page out range 0..512K, use 0..256K for positive tests and
         * 256K..512K for negative tests expecting page faults
         */
        for (off = 0; off < sizeof(file_contents) * 2; off += page_sz) {
                if (!ASSERT_OK(madvise(addr + off, page_sz, MADV_PAGEOUT),
                               "madvise pageout"))
                        return errno;
        }

        return 0;
}

static void run_test(const char *prog_name)
{
        struct file_reader *skel;
        struct bpf_program *prog;
        int err, fd;

        err = initialize_file_contents();
        if (!ASSERT_OK(err, "initialize file contents"))
                return;

        skel = file_reader__open();
        if (!ASSERT_OK_PTR(skel, "file_reader__open"))
                return;

        bpf_object__for_each_program(prog, skel->obj) {
                bpf_program__set_autoload(prog, strcmp(bpf_program__name(prog), prog_name) == 0);
        }

        memcpy(skel->bss->user_buf, file_contents, sizeof(file_contents));
        skel->bss->pid = getpid();

        err = file_reader__load(skel);
        if (!ASSERT_OK(err, "file_reader__load"))
                goto cleanup;

        err = file_reader__attach(skel);
        if (!ASSERT_OK(err, "file_reader__attach"))
                goto cleanup;

        fd = open("/proc/self/exe", O_RDONLY);
        if (fd >= 0)
                close(fd);

        ASSERT_EQ(skel->bss->err, 0, "err");
        ASSERT_EQ(skel->bss->run_success, 1, "run_success");
cleanup:
        file_reader__destroy(skel);
}

void test_file_reader(void)
{
        if (test__start_subtest("on_open_expect_fault"))
                run_test("on_open_expect_fault");

        if (test__start_subtest("on_open_validate_file_read"))
                run_test("on_open_validate_file_read");

        if (test__start_subtest("negative"))
                RUN_TESTS(file_reader_fail);
}