2022-11-03 16:57:44 +01:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0+ */
|
|
|
|
|
|
|
|
#ifndef __HID_BPF_H
|
|
|
|
#define __HID_BPF_H
|
|
|
|
|
2023-01-13 10:09:32 +01:00
|
|
|
#include <linux/bpf.h>
|
2024-06-08 11:01:20 +02:00
|
|
|
#include <linux/mutex.h>
|
2022-11-03 16:57:44 +01:00
|
|
|
#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
|
2024-06-08 11:01:22 +02:00
|
|
|
* a call to hid_bpf_get_data() in order to get a pointer to that field.
|
2022-11-03 16:57:44 +01:00
|
|
|
*
|
2024-06-08 11:01:22 +02:00
|
|
|
* @hid: the &struct hid_device representing the device itself
|
2022-11-03 16:57:47 +01:00
|
|
|
* @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)
|
2022-11-03 16:57:44 +01:00
|
|
|
* @size: Valid data in the data field.
|
|
|
|
*
|
|
|
|
* Programs can get the available valid size in data by fetching this field.
|
2022-11-03 16:57:47 +01:00
|
|
|
* 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.
|
2024-06-08 11:01:22 +02:00
|
|
|
*
|
|
|
|
* ``hid`` and ``allocated_size`` are read-only, ``size`` and ``retval`` are read-write.
|
2022-11-03 16:57:44 +01:00
|
|
|
*/
|
|
|
|
struct hid_bpf_ctx {
|
2024-06-08 11:01:28 +02:00
|
|
|
struct hid_device *hid;
|
2022-11-03 16:57:47 +01:00
|
|
|
__u32 allocated_size;
|
|
|
|
union {
|
|
|
|
__s32 retval;
|
|
|
|
__s32 size;
|
|
|
|
};
|
2022-11-03 16:57:44 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Below is HID internal
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define HID_BPF_MAX_PROGS_PER_DEV 64
|
|
|
|
#define HID_BPF_FLAG_MASK (((HID_BPF_FLAG_MAX - 1) << 1) - 1)
|
|
|
|
|
|
|
|
|
2022-11-03 16:57:49 +01:00
|
|
|
struct hid_report_enum;
|
|
|
|
|
2024-06-08 11:01:13 +02:00
|
|
|
struct hid_ops {
|
2022-11-03 16:57:49 +01:00
|
|
|
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);
|
2024-03-15 15:44:39 +01:00
|
|
|
int (*hid_hw_output_report)(struct hid_device *hdev, __u8 *buf, size_t len);
|
2024-03-15 15:44:42 +01:00
|
|
|
int (*hid_input_report)(struct hid_device *hid, enum hid_report_type type,
|
|
|
|
u8 *data, u32 size, int interrupt);
|
2022-11-03 16:57:44 +01:00
|
|
|
struct module *owner;
|
2023-12-20 08:38:48 +01:00
|
|
|
const struct bus_type *bus_type;
|
2022-11-03 16:57:44 +01:00
|
|
|
};
|
|
|
|
|
2024-06-08 11:01:13 +02:00
|
|
|
extern struct hid_ops *hid_ops;
|
2022-11-03 16:57:44 +01:00
|
|
|
|
2024-06-08 11:01:15 +02:00
|
|
|
/**
|
|
|
|
* struct hid_bpf_ops - A BPF struct_ops of callbacks allowing to attach HID-BPF
|
|
|
|
* programs to a HID device
|
|
|
|
* @hid_id: the HID uniq ID to attach to. This is writeable before ``load()``, and
|
|
|
|
* cannot be changed after
|
|
|
|
* @flags: flags used while attaching the struct_ops to the device. Currently only
|
|
|
|
* available value is %0 or ``BPF_F_BEFORE``.
|
|
|
|
* Writeable only before ``load()``
|
|
|
|
*/
|
|
|
|
struct hid_bpf_ops {
|
|
|
|
/* hid_id needs to stay first so we can easily change it
|
|
|
|
* from userspace.
|
|
|
|
*/
|
|
|
|
int hid_id;
|
|
|
|
u32 flags;
|
|
|
|
|
|
|
|
/* private: do not show up in the docs */
|
|
|
|
struct list_head list;
|
|
|
|
|
|
|
|
/* public: rest should show up in the docs */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @hid_device_event: called whenever an event is coming in from the device
|
|
|
|
*
|
|
|
|
* It has the following arguments:
|
|
|
|
*
|
|
|
|
* ``ctx``: The HID-BPF context as &struct hid_bpf_ctx
|
|
|
|
*
|
|
|
|
* Return: %0 on success and keep processing; a positive
|
|
|
|
* value to change the incoming size buffer; a negative
|
|
|
|
* error code to interrupt the processing of this event
|
|
|
|
*
|
|
|
|
* Context: Interrupt context.
|
|
|
|
*/
|
|
|
|
int (*hid_device_event)(struct hid_bpf_ctx *ctx, enum hid_report_type report_type);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @hid_rdesc_fixup: called when the probe function parses the report descriptor
|
|
|
|
* of the HID device
|
|
|
|
*
|
|
|
|
* It has the following arguments:
|
|
|
|
*
|
|
|
|
* ``ctx``: The HID-BPF context as &struct hid_bpf_ctx
|
|
|
|
*
|
|
|
|
* Return: %0 on success and keep processing; a positive
|
|
|
|
* value to change the incoming size buffer; a negative
|
|
|
|
* error code to interrupt the processing of this device
|
|
|
|
*/
|
|
|
|
int (*hid_rdesc_fixup)(struct hid_bpf_ctx *ctx);
|
|
|
|
|
|
|
|
/* private: do not show up in the docs */
|
|
|
|
struct hid_device *hdev;
|
|
|
|
};
|
|
|
|
|
2022-11-03 16:57:44 +01:00
|
|
|
/* stored in each device */
|
|
|
|
struct hid_bpf {
|
2022-11-03 16:57:47 +01:00
|
|
|
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;
|
2022-11-03 16:57:44 +01:00
|
|
|
bool destroyed; /* prevents the assignment of any progs */
|
|
|
|
|
2024-06-08 11:01:15 +02:00
|
|
|
struct hid_bpf_ops *rdesc_ops;
|
|
|
|
struct list_head prog_list;
|
|
|
|
struct mutex prog_list_lock; /* protects prog_list update */
|
2022-11-03 16:57:44 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
#ifdef CONFIG_HID_BPF
|
2022-11-03 16:57:47 +01:00
|
|
|
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);
|
2022-11-03 16:57:44 +01:00
|
|
|
void hid_bpf_destroy_device(struct hid_device *hid);
|
|
|
|
void hid_bpf_device_init(struct hid_device *hid);
|
2022-11-03 16:57:51 +01:00
|
|
|
u8 *call_hid_bpf_rdesc_fixup(struct hid_device *hdev, u8 *rdesc, unsigned int *size);
|
2022-11-03 16:57:44 +01:00
|
|
|
#else /* CONFIG_HID_BPF */
|
2022-11-03 16:57:47 +01:00
|
|
|
static inline u8 *dispatch_hid_bpf_device_event(struct hid_device *hid, enum hid_report_type type,
|
2022-11-16 11:31:10 +01:00
|
|
|
u8 *data, u32 *size, int interrupt) { return data; }
|
2022-11-03 16:57:47 +01:00
|
|
|
static inline int hid_bpf_connect_device(struct hid_device *hdev) { return 0; }
|
|
|
|
static inline void hid_bpf_disconnect_device(struct hid_device *hdev) {}
|
2022-11-03 16:57:44 +01:00
|
|
|
static inline void hid_bpf_destroy_device(struct hid_device *hid) {}
|
|
|
|
static inline void hid_bpf_device_init(struct hid_device *hid) {}
|
2024-04-14 19:07:31 -07:00
|
|
|
#define call_hid_bpf_rdesc_fixup(_hdev, _rdesc, _size) \
|
|
|
|
((u8 *)kmemdup(_rdesc, *(_size), GFP_KERNEL))
|
2022-11-03 16:57:51 +01:00
|
|
|
|
2022-11-03 16:57:44 +01:00
|
|
|
#endif /* CONFIG_HID_BPF */
|
|
|
|
|
|
|
|
#endif /* __HID_BPF_H */
|