mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-09 22:50:41 +00:00
4b9a3f49f0
Previously, HID-BPF was relying on a bpf tracing program to be notified when a program was released from userspace. This is error prone, as LLVM sometimes inline the function and sometimes not. So instead of messing up with the bpf prog ref count, we can use the bpf_link concept which actually matches exactly what we want: - a bpf_link represents the fact that a given program is attached to a given HID device - as long as the bpf_link has fd opened (either by the userspace program still being around or by pinning the bpf object in the bpffs), the program stays attached to the HID device - once every user has closed the fd, we get called by hid_bpf_link_release() that we no longer have any users, and we can disconnect the program to the device in 2 passes: first atomically clear the bit saying that the link is active, and then calling release_work in a scheduled work item. This solves entirely the problems of BPF tracing not showing up and is definitely cleaner. Signed-off-by: Benjamin Tissoires <benjamin.tissoires@redhat.com> Acked-by: Alexei Starovoitov <ast@kernel.org> Signed-off-by: Jiri Kosina <jkosina@suse.cz>
171 lines
5.8 KiB
C
171 lines
5.8 KiB
C
/* SPDX-License-Identifier: GPL-2.0+ */
|
|
|
|
#ifndef __HID_BPF_H
|
|
#define __HID_BPF_H
|
|
|
|
#include <linux/bpf.h>
|
|
#include <linux/spinlock.h>
|
|
#include <uapi/linux/hid.h>
|
|
|
|
struct hid_device;
|
|
|
|
/*
|
|
* The following is the user facing HID BPF API.
|
|
*
|
|
* Extra care should be taken when editing this part, as
|
|
* it might break existing out of the tree bpf programs.
|
|
*/
|
|
|
|
/**
|
|
* struct hid_bpf_ctx - User accessible data for all HID programs
|
|
*
|
|
* ``data`` is not directly accessible from the context. We need to issue
|
|
* a call to ``hid_bpf_get_data()`` in order to get a pointer to that field.
|
|
*
|
|
* All of these fields are currently read-only.
|
|
*
|
|
* @index: program index in the jump table. No special meaning (a smaller index
|
|
* doesn't mean the program will be executed before another program with
|
|
* a bigger index).
|
|
* @hid: the ``struct hid_device`` representing the device itself
|
|
* @report_type: used for ``hid_bpf_device_event()``
|
|
* @allocated_size: Allocated size of data.
|
|
*
|
|
* This is how much memory is available and can be requested
|
|
* by the HID program.
|
|
* Note that for ``HID_BPF_RDESC_FIXUP``, that memory is set to
|
|
* ``4096`` (4 KB)
|
|
* @size: Valid data in the data field.
|
|
*
|
|
* Programs can get the available valid size in data by fetching this field.
|
|
* Programs can also change this value by returning a positive number in the
|
|
* program.
|
|
* To discard the event, return a negative error code.
|
|
*
|
|
* ``size`` must always be less or equal than ``allocated_size`` (it is enforced
|
|
* once all BPF programs have been run).
|
|
* @retval: Return value of the previous program.
|
|
*/
|
|
struct hid_bpf_ctx {
|
|
__u32 index;
|
|
const struct hid_device *hid;
|
|
__u32 allocated_size;
|
|
enum hid_report_type report_type;
|
|
union {
|
|
__s32 retval;
|
|
__s32 size;
|
|
};
|
|
};
|
|
|
|
/**
|
|
* enum hid_bpf_attach_flags - flags used when attaching a HIF-BPF program
|
|
*
|
|
* @HID_BPF_FLAG_NONE: no specific flag is used, the kernel choses where to
|
|
* insert the program
|
|
* @HID_BPF_FLAG_INSERT_HEAD: insert the given program before any other program
|
|
* currently attached to the device. This doesn't
|
|
* guarantee that this program will always be first
|
|
* @HID_BPF_FLAG_MAX: sentinel value, not to be used by the callers
|
|
*/
|
|
enum hid_bpf_attach_flags {
|
|
HID_BPF_FLAG_NONE = 0,
|
|
HID_BPF_FLAG_INSERT_HEAD = _BITUL(0),
|
|
HID_BPF_FLAG_MAX,
|
|
};
|
|
|
|
/* Following functions are tracepoints that BPF programs can attach to */
|
|
int hid_bpf_device_event(struct hid_bpf_ctx *ctx);
|
|
int hid_bpf_rdesc_fixup(struct hid_bpf_ctx *ctx);
|
|
|
|
/* Following functions are kfunc that we export to BPF programs */
|
|
/* available everywhere in HID-BPF */
|
|
__u8 *hid_bpf_get_data(struct hid_bpf_ctx *ctx, unsigned int offset, const size_t __sz);
|
|
|
|
/* only available in syscall */
|
|
int hid_bpf_attach_prog(unsigned int hid_id, int prog_fd, __u32 flags);
|
|
int hid_bpf_hw_request(struct hid_bpf_ctx *ctx, __u8 *buf, size_t buf__sz,
|
|
enum hid_report_type rtype, enum hid_class_request reqtype);
|
|
struct hid_bpf_ctx *hid_bpf_allocate_context(unsigned int hid_id);
|
|
void hid_bpf_release_context(struct hid_bpf_ctx *ctx);
|
|
|
|
/*
|
|
* Below is HID internal
|
|
*/
|
|
|
|
/* internal function to call eBPF programs, not to be used by anybody */
|
|
int __hid_bpf_tail_call(struct hid_bpf_ctx *ctx);
|
|
|
|
#define HID_BPF_MAX_PROGS_PER_DEV 64
|
|
#define HID_BPF_FLAG_MASK (((HID_BPF_FLAG_MAX - 1) << 1) - 1)
|
|
|
|
/* types of HID programs to attach to */
|
|
enum hid_bpf_prog_type {
|
|
HID_BPF_PROG_TYPE_UNDEF = -1,
|
|
HID_BPF_PROG_TYPE_DEVICE_EVENT, /* an event is emitted from the device */
|
|
HID_BPF_PROG_TYPE_RDESC_FIXUP,
|
|
HID_BPF_PROG_TYPE_MAX,
|
|
};
|
|
|
|
struct hid_report_enum;
|
|
|
|
struct hid_bpf_ops {
|
|
struct hid_report *(*hid_get_report)(struct hid_report_enum *report_enum, const u8 *data);
|
|
int (*hid_hw_raw_request)(struct hid_device *hdev,
|
|
unsigned char reportnum, __u8 *buf,
|
|
size_t len, enum hid_report_type rtype,
|
|
enum hid_class_request reqtype);
|
|
struct module *owner;
|
|
struct bus_type *bus_type;
|
|
};
|
|
|
|
extern struct hid_bpf_ops *hid_bpf_ops;
|
|
|
|
struct hid_bpf_prog_list {
|
|
u16 prog_idx[HID_BPF_MAX_PROGS_PER_DEV];
|
|
u8 prog_cnt;
|
|
};
|
|
|
|
/* stored in each device */
|
|
struct hid_bpf {
|
|
u8 *device_data; /* allocated when a bpf program of type
|
|
* SEC(f.../hid_bpf_device_event) has been attached
|
|
* to this HID device
|
|
*/
|
|
u32 allocated_data;
|
|
|
|
struct hid_bpf_prog_list __rcu *progs[HID_BPF_PROG_TYPE_MAX]; /* attached BPF progs */
|
|
bool destroyed; /* prevents the assignment of any progs */
|
|
|
|
spinlock_t progs_lock; /* protects RCU update of progs */
|
|
};
|
|
|
|
/* specific HID-BPF link when a program is attached to a device */
|
|
struct hid_bpf_link {
|
|
struct bpf_link link;
|
|
int hid_table_index;
|
|
};
|
|
|
|
#ifdef CONFIG_HID_BPF
|
|
u8 *dispatch_hid_bpf_device_event(struct hid_device *hid, enum hid_report_type type, u8 *data,
|
|
u32 *size, int interrupt);
|
|
int hid_bpf_connect_device(struct hid_device *hdev);
|
|
void hid_bpf_disconnect_device(struct hid_device *hdev);
|
|
void hid_bpf_destroy_device(struct hid_device *hid);
|
|
void hid_bpf_device_init(struct hid_device *hid);
|
|
u8 *call_hid_bpf_rdesc_fixup(struct hid_device *hdev, u8 *rdesc, unsigned int *size);
|
|
#else /* CONFIG_HID_BPF */
|
|
static inline u8 *dispatch_hid_bpf_device_event(struct hid_device *hid, enum hid_report_type type,
|
|
u8 *data, u32 *size, int interrupt) { return data; }
|
|
static inline int hid_bpf_connect_device(struct hid_device *hdev) { return 0; }
|
|
static inline void hid_bpf_disconnect_device(struct hid_device *hdev) {}
|
|
static inline void hid_bpf_destroy_device(struct hid_device *hid) {}
|
|
static inline void hid_bpf_device_init(struct hid_device *hid) {}
|
|
static inline u8 *call_hid_bpf_rdesc_fixup(struct hid_device *hdev, u8 *rdesc, unsigned int *size)
|
|
{
|
|
return kmemdup(rdesc, *size, GFP_KERNEL);
|
|
}
|
|
|
|
#endif /* CONFIG_HID_BPF */
|
|
|
|
#endif /* __HID_BPF_H */
|