xref: /linux-6.15/include/linux/hid_bpf.h (revision f9a11da1)
1f5c27da4SBenjamin Tissoires /* SPDX-License-Identifier: GPL-2.0+ */
2f5c27da4SBenjamin Tissoires 
3f5c27da4SBenjamin Tissoires #ifndef __HID_BPF_H
4f5c27da4SBenjamin Tissoires #define __HID_BPF_H
5f5c27da4SBenjamin Tissoires 
64b9a3f49SBenjamin Tissoires #include <linux/bpf.h>
74a86220eSBenjamin Tissoires #include <linux/mutex.h>
86cd735f0SBenjamin Tissoires #include <linux/srcu.h>
9f5c27da4SBenjamin Tissoires #include <uapi/linux/hid.h>
10f5c27da4SBenjamin Tissoires 
11f5c27da4SBenjamin Tissoires struct hid_device;
12f5c27da4SBenjamin Tissoires 
13f5c27da4SBenjamin Tissoires /*
14f5c27da4SBenjamin Tissoires  * The following is the user facing HID BPF API.
15f5c27da4SBenjamin Tissoires  *
16f5c27da4SBenjamin Tissoires  * Extra care should be taken when editing this part, as
17f5c27da4SBenjamin Tissoires  * it might break existing out of the tree bpf programs.
18f5c27da4SBenjamin Tissoires  */
19f5c27da4SBenjamin Tissoires 
20f5c27da4SBenjamin Tissoires /**
21f5c27da4SBenjamin Tissoires  * struct hid_bpf_ctx - User accessible data for all HID programs
22f5c27da4SBenjamin Tissoires  *
23f5c27da4SBenjamin Tissoires  * ``data`` is not directly accessible from the context. We need to issue
24c5958697SBenjamin Tissoires  * a call to hid_bpf_get_data() in order to get a pointer to that field.
25f5c27da4SBenjamin Tissoires  *
26c5958697SBenjamin Tissoires  * @hid: the &struct hid_device representing the device itself
27658ee5a6SBenjamin Tissoires  * @allocated_size: Allocated size of data.
28658ee5a6SBenjamin Tissoires  *
29658ee5a6SBenjamin Tissoires  *                  This is how much memory is available and can be requested
30658ee5a6SBenjamin Tissoires  *                  by the HID program.
31658ee5a6SBenjamin Tissoires  *                  Note that for ``HID_BPF_RDESC_FIXUP``, that memory is set to
32658ee5a6SBenjamin Tissoires  *                  ``4096`` (4 KB)
33f5c27da4SBenjamin Tissoires  * @size: Valid data in the data field.
34f5c27da4SBenjamin Tissoires  *
35f5c27da4SBenjamin Tissoires  *        Programs can get the available valid size in data by fetching this field.
36658ee5a6SBenjamin Tissoires  *        Programs can also change this value by returning a positive number in the
37658ee5a6SBenjamin Tissoires  *        program.
38658ee5a6SBenjamin Tissoires  *        To discard the event, return a negative error code.
39658ee5a6SBenjamin Tissoires  *
40658ee5a6SBenjamin Tissoires  *        ``size`` must always be less or equal than ``allocated_size`` (it is enforced
41658ee5a6SBenjamin Tissoires  *        once all BPF programs have been run).
42658ee5a6SBenjamin Tissoires  * @retval: Return value of the previous program.
43c5958697SBenjamin Tissoires  *
44c5958697SBenjamin Tissoires  * ``hid`` and ``allocated_size`` are read-only, ``size`` and ``retval`` are read-write.
45f5c27da4SBenjamin Tissoires  */
46f5c27da4SBenjamin Tissoires struct hid_bpf_ctx {
4733c0fb85SBenjamin Tissoires 	struct hid_device *hid;
48658ee5a6SBenjamin Tissoires 	__u32 allocated_size;
49658ee5a6SBenjamin Tissoires 	union {
50658ee5a6SBenjamin Tissoires 		__s32 retval;
51f5c27da4SBenjamin Tissoires 		__s32 size;
52f5c27da4SBenjamin Tissoires 	};
53658ee5a6SBenjamin Tissoires };
54f5c27da4SBenjamin Tissoires 
55f5c27da4SBenjamin Tissoires /*
56f5c27da4SBenjamin Tissoires  * Below is HID internal
57f5c27da4SBenjamin Tissoires  */
58f5c27da4SBenjamin Tissoires 
59f5c27da4SBenjamin Tissoires #define HID_BPF_MAX_PROGS_PER_DEV 64
60f5c27da4SBenjamin Tissoires #define HID_BPF_FLAG_MASK (((HID_BPF_FLAG_MAX - 1) << 1) - 1)
61f5c27da4SBenjamin Tissoires 
62f5c27da4SBenjamin Tissoires 
6391a7f802SBenjamin Tissoires struct hid_report_enum;
6491a7f802SBenjamin Tissoires 
65146a06a0SBenjamin Tissoires struct hid_ops {
6691a7f802SBenjamin Tissoires 	struct hid_report *(*hid_get_report)(struct hid_report_enum *report_enum, const u8 *data);
6791a7f802SBenjamin Tissoires 	int (*hid_hw_raw_request)(struct hid_device *hdev,
6891a7f802SBenjamin Tissoires 				  unsigned char reportnum, __u8 *buf,
6991a7f802SBenjamin Tissoires 				  size_t len, enum hid_report_type rtype,
7067eccf15SBenjamin Tissoires 				  enum hid_class_request reqtype,
71762ced16SBenjamin Tissoires 				  u64 source, bool from_bpf);
7267eccf15SBenjamin Tissoires 	int (*hid_hw_output_report)(struct hid_device *hdev, __u8 *buf, size_t len,
73762ced16SBenjamin Tissoires 				    u64 source, bool from_bpf);
749be50ac3SBenjamin Tissoires 	int (*hid_input_report)(struct hid_device *hid, enum hid_report_type type,
759acbb7baSBenjamin Tissoires 				u8 *data, u32 size, int interrupt, u64 source, bool from_bpf,
76fa03f398SBenjamin Tissoires 				bool lock_already_taken);
77f5c27da4SBenjamin Tissoires 	struct module *owner;
789b0a3839SGreg Kroah-Hartman 	const struct bus_type *bus_type;
79f5c27da4SBenjamin Tissoires };
80f5c27da4SBenjamin Tissoires 
81*f9a11da1SThomas Weißschuh extern const struct hid_ops *hid_ops;
82f5c27da4SBenjamin Tissoires 
83ebc0d809SBenjamin Tissoires /**
84ebc0d809SBenjamin Tissoires  * struct hid_bpf_ops - A BPF struct_ops of callbacks allowing to attach HID-BPF
85ebc0d809SBenjamin Tissoires  *			programs to a HID device
86ebc0d809SBenjamin Tissoires  * @hid_id: the HID uniq ID to attach to. This is writeable before ``load()``, and
87ebc0d809SBenjamin Tissoires  *	    cannot be changed after
88ebc0d809SBenjamin Tissoires  * @flags: flags used while attaching the struct_ops to the device. Currently only
89ebc0d809SBenjamin Tissoires  *	   available value is %0 or ``BPF_F_BEFORE``.
90ebc0d809SBenjamin Tissoires  *	   Writeable only before ``load()``
91ebc0d809SBenjamin Tissoires  */
92ebc0d809SBenjamin Tissoires struct hid_bpf_ops {
93ebc0d809SBenjamin Tissoires 	/* hid_id needs to stay first so we can easily change it
94ebc0d809SBenjamin Tissoires 	 * from userspace.
95ebc0d809SBenjamin Tissoires 	 */
96ebc0d809SBenjamin Tissoires 	int			hid_id;
97ebc0d809SBenjamin Tissoires 	u32			flags;
98ebc0d809SBenjamin Tissoires 
99ebc0d809SBenjamin Tissoires 	/* private: do not show up in the docs */
100ebc0d809SBenjamin Tissoires 	struct list_head	list;
101ebc0d809SBenjamin Tissoires 
102ebc0d809SBenjamin Tissoires 	/* public: rest should show up in the docs */
103ebc0d809SBenjamin Tissoires 
104ebc0d809SBenjamin Tissoires 	/**
105ebc0d809SBenjamin Tissoires 	 * @hid_device_event: called whenever an event is coming in from the device
106ebc0d809SBenjamin Tissoires 	 *
107ebc0d809SBenjamin Tissoires 	 * It has the following arguments:
108ebc0d809SBenjamin Tissoires 	 *
109ebc0d809SBenjamin Tissoires 	 * ``ctx``: The HID-BPF context as &struct hid_bpf_ctx
110ebc0d809SBenjamin Tissoires 	 *
111ebc0d809SBenjamin Tissoires 	 * Return: %0 on success and keep processing; a positive
112ebc0d809SBenjamin Tissoires 	 * value to change the incoming size buffer; a negative
113ebc0d809SBenjamin Tissoires 	 * error code to interrupt the processing of this event
114ebc0d809SBenjamin Tissoires 	 *
115ebc0d809SBenjamin Tissoires 	 * Context: Interrupt context.
116ebc0d809SBenjamin Tissoires 	 */
11767eccf15SBenjamin Tissoires 	int (*hid_device_event)(struct hid_bpf_ctx *ctx, enum hid_report_type report_type,
118762ced16SBenjamin Tissoires 				u64 source);
119ebc0d809SBenjamin Tissoires 
120ebc0d809SBenjamin Tissoires 	/**
121ebc0d809SBenjamin Tissoires 	 * @hid_rdesc_fixup: called when the probe function parses the report descriptor
122ebc0d809SBenjamin Tissoires 	 * of the HID device
123ebc0d809SBenjamin Tissoires 	 *
124ebc0d809SBenjamin Tissoires 	 * It has the following arguments:
125ebc0d809SBenjamin Tissoires 	 *
126ebc0d809SBenjamin Tissoires 	 * ``ctx``: The HID-BPF context as &struct hid_bpf_ctx
127ebc0d809SBenjamin Tissoires 	 *
128ebc0d809SBenjamin Tissoires 	 * Return: %0 on success and keep processing; a positive
129ebc0d809SBenjamin Tissoires 	 * value to change the incoming size buffer; a negative
130ebc0d809SBenjamin Tissoires 	 * error code to interrupt the processing of this device
131ebc0d809SBenjamin Tissoires 	 */
132ebc0d809SBenjamin Tissoires 	int (*hid_rdesc_fixup)(struct hid_bpf_ctx *ctx);
133ebc0d809SBenjamin Tissoires 
1348bd0488bSBenjamin Tissoires 	/**
1358bd0488bSBenjamin Tissoires 	 * @hid_hw_request: called whenever a hid_hw_raw_request() call is emitted
1368bd0488bSBenjamin Tissoires 	 * on the HID device
1378bd0488bSBenjamin Tissoires 	 *
1388bd0488bSBenjamin Tissoires 	 * It has the following arguments:
1398bd0488bSBenjamin Tissoires 	 *
1408bd0488bSBenjamin Tissoires 	 * ``ctx``: The HID-BPF context as &struct hid_bpf_ctx
141c79de517SBenjamin Tissoires 	 *
1428bd0488bSBenjamin Tissoires 	 * ``reportnum``: the report number, as in hid_hw_raw_request()
143260ffc96SBenjamin Tissoires 	 *
1448bd0488bSBenjamin Tissoires 	 * ``rtype``: the report type (``HID_INPUT_REPORT``, ``HID_FEATURE_REPORT``,
1458bd0488bSBenjamin Tissoires 	 * ``HID_OUTPUT_REPORT``)
146260ffc96SBenjamin Tissoires 	 *
1478bd0488bSBenjamin Tissoires 	 * ``reqtype``: the request
148260ffc96SBenjamin Tissoires 	 *
1498bd0488bSBenjamin Tissoires 	 * ``source``: a u64 referring to a uniq but identifiable source. If %0, the
1508bd0488bSBenjamin Tissoires 	 * kernel itself emitted that call. For hidraw, ``source`` is set
1518bd0488bSBenjamin Tissoires 	 * to the associated ``struct file *``.
1528bd0488bSBenjamin Tissoires 	 *
1538bd0488bSBenjamin Tissoires 	 * Return: %0 to keep processing the request by hid-core; any other value
1548bd0488bSBenjamin Tissoires 	 * stops hid-core from processing that event. A positive value should be
1558bd0488bSBenjamin Tissoires 	 * returned with the number of bytes returned in the incoming buffer; a
1568bd0488bSBenjamin Tissoires 	 * negative error code interrupts the processing of this call.
1578bd0488bSBenjamin Tissoires 	 */
1588bd0488bSBenjamin Tissoires 	int (*hid_hw_request)(struct hid_bpf_ctx *ctx, unsigned char reportnum,
1598bd0488bSBenjamin Tissoires 			       enum hid_report_type rtype, enum hid_class_request reqtype,
160260ffc96SBenjamin Tissoires 			       u64 source);
1618bd0488bSBenjamin Tissoires 
1629286675aSBenjamin Tissoires 	/**
1639286675aSBenjamin Tissoires 	 * @hid_hw_output_report: called whenever a hid_hw_output_report() call is emitted
1649286675aSBenjamin Tissoires 	 * on the HID device
1659286675aSBenjamin Tissoires 	 *
1669286675aSBenjamin Tissoires 	 * It has the following arguments:
1679286675aSBenjamin Tissoires 	 *
1689286675aSBenjamin Tissoires 	 * ``ctx``: The HID-BPF context as &struct hid_bpf_ctx
169c79de517SBenjamin Tissoires 	 *
1709286675aSBenjamin Tissoires 	 * ``source``: a u64 referring to a uniq but identifiable source. If %0, the
1719286675aSBenjamin Tissoires 	 * kernel itself emitted that call. For hidraw, ``source`` is set
1729286675aSBenjamin Tissoires 	 * to the associated ``struct file *``.
1739286675aSBenjamin Tissoires 	 *
1749286675aSBenjamin Tissoires 	 * Return: %0 to keep processing the request by hid-core; any other value
1759286675aSBenjamin Tissoires 	 * stops hid-core from processing that event. A positive value should be
1769286675aSBenjamin Tissoires 	 * returned with the number of bytes written to the device; a negative error
1779286675aSBenjamin Tissoires 	 * code interrupts the processing of this call.
1789286675aSBenjamin Tissoires 	 */
179c79de517SBenjamin Tissoires 	int (*hid_hw_output_report)(struct hid_bpf_ctx *ctx, u64 source);
1809286675aSBenjamin Tissoires 
1818bd0488bSBenjamin Tissoires 
182ebc0d809SBenjamin Tissoires 	/* private: do not show up in the docs */
183ebc0d809SBenjamin Tissoires 	struct hid_device *hdev;
184ebc0d809SBenjamin Tissoires };
185ebc0d809SBenjamin Tissoires 
186f5c27da4SBenjamin Tissoires /* stored in each device */
187f5c27da4SBenjamin Tissoires struct hid_bpf {
188658ee5a6SBenjamin Tissoires 	u8 *device_data;		/* allocated when a bpf program of type
189658ee5a6SBenjamin Tissoires 					 * SEC(f.../hid_bpf_device_event) has been attached
190658ee5a6SBenjamin Tissoires 					 * to this HID device
191658ee5a6SBenjamin Tissoires 					 */
192658ee5a6SBenjamin Tissoires 	u32 allocated_data;
193f5c27da4SBenjamin Tissoires 	bool destroyed;			/* prevents the assignment of any progs */
194f5c27da4SBenjamin Tissoires 
195ebc0d809SBenjamin Tissoires 	struct hid_bpf_ops *rdesc_ops;
196ebc0d809SBenjamin Tissoires 	struct list_head prog_list;
197ebc0d809SBenjamin Tissoires 	struct mutex prog_list_lock;	/* protects prog_list update */
1986cd735f0SBenjamin Tissoires 	struct srcu_struct srcu;	/* protects prog_list read-only access */
199f5c27da4SBenjamin Tissoires };
200f5c27da4SBenjamin Tissoires 
201f5c27da4SBenjamin Tissoires #ifdef CONFIG_HID_BPF
202658ee5a6SBenjamin Tissoires u8 *dispatch_hid_bpf_device_event(struct hid_device *hid, enum hid_report_type type, u8 *data,
2039acbb7baSBenjamin Tissoires 				  u32 *size, int interrupt, u64 source, bool from_bpf);
2048bd0488bSBenjamin Tissoires int dispatch_hid_bpf_raw_requests(struct hid_device *hdev,
2058bd0488bSBenjamin Tissoires 				  unsigned char reportnum, __u8 *buf,
2068bd0488bSBenjamin Tissoires 				  u32 size, enum hid_report_type rtype,
2078bd0488bSBenjamin Tissoires 				  enum hid_class_request reqtype,
208c79de517SBenjamin Tissoires 				  u64 source, bool from_bpf);
2099286675aSBenjamin Tissoires int dispatch_hid_bpf_output_report(struct hid_device *hdev, __u8 *buf, u32 size,
210c79de517SBenjamin Tissoires 				   u64 source, bool from_bpf);
211658ee5a6SBenjamin Tissoires int hid_bpf_connect_device(struct hid_device *hdev);
212658ee5a6SBenjamin Tissoires void hid_bpf_disconnect_device(struct hid_device *hdev);
213f5c27da4SBenjamin Tissoires void hid_bpf_destroy_device(struct hid_device *hid);
2146cd735f0SBenjamin Tissoires int hid_bpf_device_init(struct hid_device *hid);
2158b7fd6a1SBenjamin Tissoires const u8 *call_hid_bpf_rdesc_fixup(struct hid_device *hdev, const u8 *rdesc, unsigned int *size);
216f5c27da4SBenjamin Tissoires #else /* CONFIG_HID_BPF */
dispatch_hid_bpf_device_event(struct hid_device * hid,enum hid_report_type type,u8 * data,u32 * size,int interrupt,u64 source,bool from_bpf)217658ee5a6SBenjamin Tissoires static inline u8 *dispatch_hid_bpf_device_event(struct hid_device *hid, enum hid_report_type type,
21867eccf15SBenjamin Tissoires 						u8 *data, u32 *size, int interrupt,
2199acbb7baSBenjamin Tissoires 						u64 source, bool from_bpf) { return data; }
dispatch_hid_bpf_raw_requests(struct hid_device * hdev,unsigned char reportnum,u8 * buf,u32 size,enum hid_report_type rtype,enum hid_class_request reqtype,u64 source,bool from_bpf)2208bd0488bSBenjamin Tissoires static inline int dispatch_hid_bpf_raw_requests(struct hid_device *hdev,
2218bd0488bSBenjamin Tissoires 						unsigned char reportnum, u8 *buf,
2228bd0488bSBenjamin Tissoires 						u32 size, enum hid_report_type rtype,
2238bd0488bSBenjamin Tissoires 						enum hid_class_request reqtype,
22475839101SBenjamin Tissoires 						u64 source, bool from_bpf) { return 0; }
dispatch_hid_bpf_output_report(struct hid_device * hdev,__u8 * buf,u32 size,u64 source,bool from_bpf)2259286675aSBenjamin Tissoires static inline int dispatch_hid_bpf_output_report(struct hid_device *hdev, __u8 *buf, u32 size,
226c79de517SBenjamin Tissoires 						 u64 source, bool from_bpf) { return 0; }
hid_bpf_connect_device(struct hid_device * hdev)227658ee5a6SBenjamin Tissoires static inline int hid_bpf_connect_device(struct hid_device *hdev) { return 0; }
hid_bpf_disconnect_device(struct hid_device * hdev)228658ee5a6SBenjamin Tissoires static inline void hid_bpf_disconnect_device(struct hid_device *hdev) {}
hid_bpf_destroy_device(struct hid_device * hid)229f5c27da4SBenjamin Tissoires static inline void hid_bpf_destroy_device(struct hid_device *hid) {}
hid_bpf_device_init(struct hid_device * hid)2306cd735f0SBenjamin Tissoires static inline int hid_bpf_device_init(struct hid_device *hid) { return 0; }
call_hid_bpf_rdesc_fixup(struct hid_device * hdev,const u8 * rdesc,unsigned int * size)2318b7fd6a1SBenjamin Tissoires static inline const u8 *call_hid_bpf_rdesc_fixup(struct hid_device *hdev, const u8 *rdesc,
2328b7fd6a1SBenjamin Tissoires 						 unsigned int *size) { return rdesc; }
233ad190df1SBenjamin Tissoires 
234f5c27da4SBenjamin Tissoires #endif /* CONFIG_HID_BPF */
235f5c27da4SBenjamin Tissoires 
236f5c27da4SBenjamin Tissoires #endif /* __HID_BPF_H */
237