forked from Minki/linux
36e68442d1
Add kernel module listener that will load/validate and unload module BTF. Module BTFs gets ID generated for them, which makes it possible to iterate them with existing BTF iteration API. They are given their respective module's names, which will get reported through GET_OBJ_INFO API. They are also marked as in-kernel BTFs for tooling to distinguish them from user-provided BTFs. Also, similarly to vmlinux BTF, kernel module BTFs are exposed through sysfs as /sys/kernel/btf/<module-name>. This is convenient for user-space tools to inspect module BTF contents and dump their types with existing tools: [vmuser@archvm bpf]$ ls -la /sys/kernel/btf total 0 drwxr-xr-x 2 root root 0 Nov 4 19:46 . drwxr-xr-x 13 root root 0 Nov 4 19:46 .. ... -r--r--r-- 1 root root 888 Nov 4 19:46 irqbypass -r--r--r-- 1 root root 100225 Nov 4 19:46 kvm -r--r--r-- 1 root root 35401 Nov 4 19:46 kvm_intel -r--r--r-- 1 root root 120 Nov 4 19:46 pcspkr -r--r--r-- 1 root root 399 Nov 4 19:46 serio_raw -r--r--r-- 1 root root 4094095 Nov 4 19:46 vmlinux Signed-off-by: Andrii Nakryiko <andrii@kernel.org> Signed-off-by: Alexei Starovoitov <ast@kernel.org> Reviewed-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org> Link: https://lore.kernel.org/bpf/20201110011932.3201430-5-andrii@kernel.org
46 lines
1.1 KiB
C
46 lines
1.1 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Provide kernel BTF information for introspection and use by eBPF tools.
|
|
*/
|
|
#include <linux/kernel.h>
|
|
#include <linux/module.h>
|
|
#include <linux/kobject.h>
|
|
#include <linux/init.h>
|
|
#include <linux/sysfs.h>
|
|
|
|
/* See scripts/link-vmlinux.sh, gen_btf() func for details */
|
|
extern char __weak __start_BTF[];
|
|
extern char __weak __stop_BTF[];
|
|
|
|
static ssize_t
|
|
btf_vmlinux_read(struct file *file, struct kobject *kobj,
|
|
struct bin_attribute *bin_attr,
|
|
char *buf, loff_t off, size_t len)
|
|
{
|
|
memcpy(buf, __start_BTF + off, len);
|
|
return len;
|
|
}
|
|
|
|
static struct bin_attribute bin_attr_btf_vmlinux __ro_after_init = {
|
|
.attr = { .name = "vmlinux", .mode = 0444, },
|
|
.read = btf_vmlinux_read,
|
|
};
|
|
|
|
struct kobject *btf_kobj;
|
|
|
|
static int __init btf_vmlinux_init(void)
|
|
{
|
|
bin_attr_btf_vmlinux.size = __stop_BTF - __start_BTF;
|
|
|
|
if (!__start_BTF || bin_attr_btf_vmlinux.size == 0)
|
|
return 0;
|
|
|
|
btf_kobj = kobject_create_and_add("btf", kernel_kobj);
|
|
if (!btf_kobj)
|
|
return -ENOMEM;
|
|
|
|
return sysfs_create_bin_file(btf_kobj, &bin_attr_btf_vmlinux);
|
|
}
|
|
|
|
subsys_initcall(btf_vmlinux_init);
|