|  | /* SPDX-License-Identifier: GPL-2.0 */ | 
|  | /* | 
|  | * Common header file for generic dynamic events. | 
|  | */ | 
|  |  | 
|  | #ifndef _TRACE_DYNEVENT_H | 
|  | #define _TRACE_DYNEVENT_H | 
|  |  | 
|  | #include <linux/kernel.h> | 
|  | #include <linux/list.h> | 
|  | #include <linux/mutex.h> | 
|  | #include <linux/seq_file.h> | 
|  |  | 
|  | #include "trace.h" | 
|  |  | 
|  | struct dyn_event; | 
|  |  | 
|  | /** | 
|  | * struct dyn_event_operations - Methods for each type of dynamic events | 
|  | * | 
|  | * These methods must be set for each type, since there is no default method. | 
|  | * Before using this for dyn_event_init(), it must be registered by | 
|  | * dyn_event_register(). | 
|  | * | 
|  | * @create: Parse and create event method. This is invoked when user passes | 
|  | *  a event definition to dynamic_events interface. This must not destruct | 
|  | *  the arguments and return -ECANCELED if given arguments doesn't match its | 
|  | *  command prefix. | 
|  | * @show: Showing method. This is invoked when user reads the event definitions | 
|  | *  via dynamic_events interface. | 
|  | * @is_busy: Check whether given event is busy so that it can not be deleted. | 
|  | *  Return true if it is busy, otherwise false. | 
|  | * @free: Delete the given event. Return 0 if success, otherwise error. | 
|  | * @match: Check whether given event and system name match this event. The argc | 
|  | *  and argv is used for exact match. Return true if it matches, otherwise | 
|  | *  false. | 
|  | * | 
|  | * Except for @create, these methods are called under holding event_mutex. | 
|  | */ | 
|  | struct dyn_event_operations { | 
|  | struct list_head	list; | 
|  | int (*create)(const char *raw_command); | 
|  | int (*show)(struct seq_file *m, struct dyn_event *ev); | 
|  | bool (*is_busy)(struct dyn_event *ev); | 
|  | int (*free)(struct dyn_event *ev); | 
|  | bool (*match)(const char *system, const char *event, | 
|  | int argc, const char **argv, struct dyn_event *ev); | 
|  | }; | 
|  |  | 
|  | /* Register new dyn_event type -- must be called at first */ | 
|  | int dyn_event_register(struct dyn_event_operations *ops); | 
|  |  | 
|  | /** | 
|  | * struct dyn_event - Dynamic event list header | 
|  | * | 
|  | * The dyn_event structure encapsulates a list and a pointer to the operators | 
|  | * for making a global list of dynamic events. | 
|  | * User must includes this in each event structure, so that those events can | 
|  | * be added/removed via dynamic_events interface. | 
|  | */ | 
|  | struct dyn_event { | 
|  | struct list_head		list; | 
|  | struct dyn_event_operations	*ops; | 
|  | }; | 
|  |  | 
|  | extern struct list_head dyn_event_list; | 
|  |  | 
|  | static inline | 
|  | int dyn_event_init(struct dyn_event *ev, struct dyn_event_operations *ops) | 
|  | { | 
|  | if (!ev || !ops) | 
|  | return -EINVAL; | 
|  |  | 
|  | INIT_LIST_HEAD(&ev->list); | 
|  | ev->ops = ops; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static inline int dyn_event_add(struct dyn_event *ev, | 
|  | struct trace_event_call *call) | 
|  | { | 
|  | lockdep_assert_held(&event_mutex); | 
|  |  | 
|  | if (!ev || !ev->ops) | 
|  | return -EINVAL; | 
|  |  | 
|  | call->flags |= TRACE_EVENT_FL_DYNAMIC; | 
|  | list_add_tail(&ev->list, &dyn_event_list); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static inline void dyn_event_remove(struct dyn_event *ev) | 
|  | { | 
|  | lockdep_assert_held(&event_mutex); | 
|  | list_del_init(&ev->list); | 
|  | } | 
|  |  | 
|  | void *dyn_event_seq_start(struct seq_file *m, loff_t *pos); | 
|  | void *dyn_event_seq_next(struct seq_file *m, void *v, loff_t *pos); | 
|  | void dyn_event_seq_stop(struct seq_file *m, void *v); | 
|  | int dyn_events_release_all(struct dyn_event_operations *type); | 
|  | int dyn_event_release(const char *raw_command, struct dyn_event_operations *type); | 
|  |  | 
|  | /* | 
|  | * for_each_dyn_event	-	iterate over the dyn_event list | 
|  | * @pos:	the struct dyn_event * to use as a loop cursor | 
|  | * | 
|  | * This is just a basement of for_each macro. Wrap this for | 
|  | * each actual event structure with ops filtering. | 
|  | */ | 
|  | #define for_each_dyn_event(pos)	\ | 
|  | list_for_each_entry(pos, &dyn_event_list, list) | 
|  |  | 
|  | /* | 
|  | * for_each_dyn_event	-	iterate over the dyn_event list safely | 
|  | * @pos:	the struct dyn_event * to use as a loop cursor | 
|  | * @n:		the struct dyn_event * to use as temporary storage | 
|  | */ | 
|  | #define for_each_dyn_event_safe(pos, n)	\ | 
|  | list_for_each_entry_safe(pos, n, &dyn_event_list, list) | 
|  |  | 
|  | extern void dynevent_cmd_init(struct dynevent_cmd *cmd, char *buf, int maxlen, | 
|  | enum dynevent_type type, | 
|  | dynevent_create_fn_t run_command); | 
|  |  | 
|  | typedef int (*dynevent_check_arg_fn_t)(void *data); | 
|  |  | 
|  | struct dynevent_arg { | 
|  | const char		*str; | 
|  | char			separator; /* e.g. ';', ',', or nothing */ | 
|  | }; | 
|  |  | 
|  | extern void dynevent_arg_init(struct dynevent_arg *arg, | 
|  | char separator); | 
|  | extern int dynevent_arg_add(struct dynevent_cmd *cmd, | 
|  | struct dynevent_arg *arg, | 
|  | dynevent_check_arg_fn_t check_arg); | 
|  |  | 
|  | struct dynevent_arg_pair { | 
|  | const char		*lhs; | 
|  | const char		*rhs; | 
|  | char			operator; /* e.g. '=' or nothing */ | 
|  | char			separator; /* e.g. ';', ',', or nothing */ | 
|  | }; | 
|  |  | 
|  | extern void dynevent_arg_pair_init(struct dynevent_arg_pair *arg_pair, | 
|  | char operator, char separator); | 
|  |  | 
|  | extern int dynevent_arg_pair_add(struct dynevent_cmd *cmd, | 
|  | struct dynevent_arg_pair *arg_pair, | 
|  | dynevent_check_arg_fn_t check_arg); | 
|  | extern int dynevent_str_add(struct dynevent_cmd *cmd, const char *str); | 
|  |  | 
|  | #endif |