From: "Steven Rostedt (VMware)" <rostedt@xxxxxxxxxxx> Only allow the flags to have TRACEFS_FL_RESET set in tracefs_function_filter() if the previous call with the same instance did not have the TRACEFS_FL_CONTINUE set. The RESET flag means to reset the filter, and since it can only do that when the file is opened, it does not make sense to have it set. It also is dangerous to simply close (and commit previous changes) if it is set and the file is already opened. The only rational solution is to fail the call in this case. Link: https://lore.kernel.org/linux-trace-devel/20210330105250.0a7105a1@xxxxxxxxxxxxxxxxxx/ Signed-off-by: Steven Rostedt (VMware) <rostedt@xxxxxxxxxxx> --- Documentation/libtracefs-function-filter.txt | 24 ++++++++++++-------- src/tracefs-tools.c | 13 ++++++++--- 2 files changed, 25 insertions(+), 12 deletions(-) diff --git a/Documentation/libtracefs-function-filter.txt b/Documentation/libtracefs-function-filter.txt index 5631ff75d54e..a022a2196b75 100644 --- a/Documentation/libtracefs-function-filter.txt +++ b/Documentation/libtracefs-function-filter.txt @@ -45,9 +45,10 @@ The _flags_ parameter may have the following set, or be zero. *TRACEFS_FL_RESET* : If _flags_ contains *TRACEFS_FL_RESET*, then it will clear the filters that are currently set before applying _filter_. Otherwise, _filter_ is added to -the current set of filters already enabled. This flag is ignored if a +the current set of filters already enabled. If this flag is set and the previous call to tracefs_function_filter() had the same _instance_ and the -*TRACEFS_FL_CONTINUE* flag was set. +*TRACEFS_FL_CONTINUE* flag was set, then the function will fail with a +return of -1 and errno set to EBUSY. *TRACEFS_FL_CONTINUE* : If _flags_ contains *TRACEFS_FL_CONTINUE*, then _filter_ will not take @@ -58,9 +59,9 @@ commit all the filters. It can be called multiple times to add more filters. A call without this flag set will commit the changes before returning (if the _filter_ passed in successfully matched). A tracefs_function_filter() call after one that had -the *TRACEFS_FL_CONTINUE* flag set for the same instance will ignore the -*TRACEFS_FL_RESET* flag, as the reset flag is only applicable for the first -filters to be added before committing. +the *TRACEFS_FL_CONTINUE* flag set for the same instance will fail if +*TRACEFS_FL_RESET* flag is set, as the reset flag is only applicable for the +first filter to be added before committing. RETURN VALUE ------------ @@ -79,6 +80,10 @@ ERRORS *EINVAL* The filter is invalid or did not match any functions. +*EBUSY* The previous call of *tracefs_function_filter*() was called +with the same instance and *TRACEFS_FL_CONTINUE* set and the current call +had *TRACEFS_FL_RESET* set. + Other errors may also happen caused by internal system calls. EXAMPLE @@ -97,6 +102,7 @@ int main(int argc, char *argv[]) { struct tracefs_instance *inst = tracefs_instance_create(INST); int ret; + int reset = TRACEFS_FL_RESET; int i; if (!inst) { @@ -105,12 +111,11 @@ int main(int argc, char *argv[]) for (i = 0; filters[i]; i++) { /* - * Note, only the first call does something - * with TRACEFS_FL_RESET. It is ignored in the following - * calls. + * Only the first call can have TRACEFS_FL_RESET set + * while TRACEFS_FL_CONTINUE is set. */ ret = tracefs_function_filter(inst, filters[i], NULL, - TRACEFS_FL_RESET | TRACEFS_FL_CONTINUE); + reset | TRACEFS_FL_CONTINUE); if (ret) { if (errno == EINVAL) @@ -118,6 +123,7 @@ int main(int argc, char *argv[]) else printf("Failed writing %s\n", filters[i]); } + reset = 0; } ret = tracefs_function_filter(inst, "*", "ext4", 0); diff --git a/src/tracefs-tools.c b/src/tracefs-tools.c index 165a9dbd8a02..5719ddf66982 100644 --- a/src/tracefs-tools.c +++ b/src/tracefs-tools.c @@ -739,9 +739,9 @@ static int write_func_list(int fd, struct func_list *list) * * @flags: * TRACEFS_FL_RESET - will clear the functions in the filter file - * before applying the @filter. This flag is ignored - * if this function is called again when the previous - * call had TRACEFS_FL_CONTINUE set. + * before applying the @filter. This will error with -1 + * and errno of EBUSY if this flag is set and a previous + * call had the same instance and TRACEFS_FL_CONTINUE set. * TRACEFS_FL_CONTINUE - will keep the filter file open on return. * The filter is updated on closing of the filter file. * With this flag set, the file is not closed, and more filters @@ -773,6 +773,13 @@ int tracefs_function_filter(struct tracefs_instance *instance, const char *filte else fd = &ftrace_filter_fd; + /* RESET is only allowed if the file is not opened yet */ + if (reset && *fd >= 0) { + errno = EBUSY; + ret = -1; + goto out; + } + /* * Set EINVAL on no matching filter. But errno may still be modified * on another type of failure (allocation or opening a file). -- 2.30.1