cregit-Linux how code gets into the kernel

Release 4.15 kernel/trace/trace_events_filter.c

Directory: kernel/trace
/*
 * trace_events_filter - generic event filtering
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
 *
 * Copyright (C) 2009 Tom Zanussi <tzanussi@gmail.com>
 */

#include <linux/module.h>
#include <linux/ctype.h>
#include <linux/mutex.h>
#include <linux/perf_event.h>
#include <linux/slab.h>

#include "trace.h"
#include "trace_output.h"


#define DEFAULT_SYS_FILTER_MESSAGE					\
	"### global filter ###\n"                                       \
        "# Use this to set filters for multiple events.\n"              \
        "# Only events with the given fields will be affected.\n"       \
        "# If no events are modified, an error message will be displayed here"


enum filter_op_ids
{
	
OP_OR,
	
OP_AND,
	
OP_GLOB,
	
OP_NE,
	
OP_EQ,
	
OP_LT,
	
OP_LE,
	
OP_GT,
	
OP_GE,
	
OP_BAND,
	
OP_NOT,
	
OP_NONE,
	
OP_OPEN_PAREN,
};


struct filter_op {
	
int id;
	
char *string;
	
int precedence;
};

/* Order must be the same as enum filter_op_ids above */

static struct filter_op filter_ops[] = {
	{ OP_OR,	"||",		1 },
	{ OP_AND,	"&&",		2 },
	{ OP_GLOB,	"~",		4 },
	{ OP_NE,	"!=",		4 },
	{ OP_EQ,	"==",		4 },
	{ OP_LT,	"<",		5 },
	{ OP_LE,	"<=",		5 },
	{ OP_GT,	">",		5 },
	{ OP_GE,	">=",		5 },
	{ OP_BAND,	"&",		6 },
	{ OP_NOT,	"!",		6 },
	{ OP_NONE,	"OP_NONE",	0 },
	{ OP_OPEN_PAREN, "(",		0 },
};

enum {
	
FILT_ERR_NONE,
	
FILT_ERR_INVALID_OP,
	
FILT_ERR_UNBALANCED_PAREN,
	
FILT_ERR_TOO_MANY_OPERANDS,
	
FILT_ERR_OPERAND_TOO_LONG,
	
FILT_ERR_FIELD_NOT_FOUND,
	
FILT_ERR_ILLEGAL_FIELD_OP,
	
FILT_ERR_ILLEGAL_INTVAL,
	
FILT_ERR_BAD_SUBSYS_FILTER,
	
FILT_ERR_TOO_MANY_PREDS,
	
FILT_ERR_MISSING_FIELD,
	
FILT_ERR_INVALID_FILTER,
	
FILT_ERR_IP_FIELD_ONLY,
	
FILT_ERR_ILLEGAL_NOT_OP,
};


static char *err_text[] = {
	"No error",
	"Invalid operator",
	"Unbalanced parens",
	"Too many operands",
	"Operand too long",
	"Field not found",
	"Illegal operation for field type",
	"Illegal integer value",
	"Couldn't find or set field in one of a subsystem's events",
	"Too many terms in predicate expression",
	"Missing field name and/or value",
	"Meaningless filter expression",
	"Only 'ip' field is supported for function trace",
	"Illegal use of '!'",
};


struct opstack_op {
	
enum filter_op_ids op;
	
struct list_head list;
};


struct postfix_elt {
	
enum filter_op_ids op;
	
char *operand;
	
struct list_head list;
};


struct filter_parse_state {
	
struct filter_op *ops;
	
struct list_head opstack;
	
struct list_head postfix;
	
int lasterr;
	
int lasterr_pos;

	struct {
		
char *string;
		
unsigned int cnt;
		
unsigned int tail;
	
} infix;

	struct {
		
char string[MAX_FILTER_STR_VAL];
		
int pos;
		
unsigned int tail;
	
} operand;
};


struct pred_stack {
	
struct filter_pred	**preds;
	
int			index;
};

/* If not of not match is equal to not of not, then it is a match */

#define DEFINE_COMPARISON_PRED(type)					\
static int filter_pred_LT_##type(struct filter_pred *pred, void *event) \
{                                                                       \
        type *addr = (type *)(event + pred->offset);                    \
        type val = (type)pred->val;                                     \
        int match = (*addr < val);                                      \
        return !!match == !pred->not;                                   \
}                                                                       \
static int filter_pred_LE_##type(struct filter_pred *pred, void *event) \
{                                                                       \
        type *addr = (type *)(event + pred->offset);                    \
        type val = (type)pred->val;                                     \
        int match = (*addr <= val);                                     \
        return !!match == !pred->not;                                   \
}                                                                       \
static int filter_pred_GT_##type(struct filter_pred *pred, void *event) \
{                                                                       \
        type *addr = (type *)(event + pred->offset);                    \
        type val = (type)pred->val;                                     \
        int match = (*addr > val);                                      \
        return !!match == !pred->not;                                   \
}                                                                       \
static int filter_pred_GE_##type(struct filter_pred *pred, void *event) \
{                                                                       \
        type *addr = (type *)(event + pred->offset);                    \
        type val = (type)pred->val;                                     \
        int match = (*addr >= val);                                     \
        return !!match == !pred->not;                                   \
}                                                                       \
static int filter_pred_BAND_##type(struct filter_pred *pred, void *event) \
{                                                                       \
        type *addr = (type *)(event + pred->offset);                    \
        type val = (type)pred->val;                                     \
        int match = !!(*addr & val);                                    \
        return match == !pred->not;                                     \
}                                                                       \
static const filter_pred_fn_t pred_funcs_##type[] = {                   \
        filter_pred_LT_##type,                                          \
        filter_pred_LE_##type,                                          \
        filter_pred_GT_##type,                                          \
        filter_pred_GE_##type,                                          \
        filter_pred_BAND_##type,                                        \
};


#define PRED_FUNC_START			OP_LT


#define DEFINE_EQUALITY_PRED(size)					\
static int filter_pred_##size(struct filter_pred *pred, void *event)    \
{                                                                       \
        u##size *addr = (u##size *)(event + pred->offset);              \
        u##size val = (u##size)pred->val;                               \
        int match;                                                      \
                                                                        \
        match = (val == *addr) ^ pred->not;                             \
                                                                        \
        return match;                                                   \
}


DEFINE_COMPARISON_PRED(s64);

DEFINE_COMPARISON_PRED(u64);

DEFINE_COMPARISON_PRED(s32);

DEFINE_COMPARISON_PRED(u32);

DEFINE_COMPARISON_PRED(s16);

DEFINE_COMPARISON_PRED(u16);

DEFINE_COMPARISON_PRED(s8);

DEFINE_COMPARISON_PRED(u8);

DEFINE_EQUALITY_PRED(64);
DEFINE_EQUALITY_PRED(32);
DEFINE_EQUALITY_PRED(16);
DEFINE_EQUALITY_PRED(8);

/* Filter predicate for fixed sized arrays of characters */

static int filter_pred_string(struct filter_pred *pred, void *event) { char *addr = (char *)(event + pred->offset); int cmp, match; cmp = pred->regex.match(addr, &pred->regex, pred->regex.field_len); match = cmp ^ pred->not; return match; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi5985.51%150.00%
Frédéric Weisbecker1014.49%150.00%
Total69100.00%2100.00%

/* Filter predicate for char * pointers */
static int filter_pred_pchar(struct filter_pred *pred, void *event) { char **addr = (char **)(event + pred->offset); int cmp, match; int len = strlen(*addr) + 1; /* including tailing '\0' */ cmp = pred->regex.match(*addr, &pred->regex, len); match = cmp ^ pred->not; return match; }

Contributors

PersonTokensPropCommitsCommitProp
Li Zefan7391.25%266.67%
Frédéric Weisbecker78.75%133.33%
Total80100.00%3100.00%

/* * Filter predicate for dynamic sized arrays of characters. * These are implemented through a list of strings at the end * of the entry. * Also each of these strings have a field in the entry which * contains its offset from the beginning of the entry. * We have then first to get this field, dereference it * and add it to the address of the entry, and at last we have * the address of the string. */
static int filter_pred_strloc(struct filter_pred *pred, void *event) { u32 str_item = *(u32 *)(event + pred->offset); int str_loc = str_item & 0xffff; int str_len = str_item >> 16; char *addr = (char *)(event + str_loc); int cmp, match; cmp = pred->regex.match(addr, &pred->regex, str_len); match = cmp ^ pred->not; return match; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker7681.72%266.67%
Li Zefan1718.28%133.33%
Total93100.00%3100.00%

/* Filter predicate for CPUs. */
static int filter_pred_cpu(struct filter_pred *pred, void *event) { int cpu, cmp; int match = 0; cpu = raw_smp_processor_id(); cmp = pred->val; switch (pred->op) { case OP_EQ: match = cpu == cmp; break; case OP_LT: match = cpu < cmp; break; case OP_LE: match = cpu <= cmp; break; case OP_GT: match = cpu > cmp; break; case OP_GE: match = cpu >= cmp; break; default: break; } return !!match == !pred->not; }

Contributors

PersonTokensPropCommitsCommitProp
Daniel Wagner106100.00%1100.00%
Total106100.00%1100.00%

/* Filter predicate for COMM. */
static int filter_pred_comm(struct filter_pred *pred, void *event) { int cmp, match; cmp = pred->regex.match(current->comm, &pred->regex, pred->regex.field_len); match = cmp ^ pred->not; return match; }

Contributors

PersonTokensPropCommitsCommitProp
Daniel Wagner55100.00%1100.00%
Total55100.00%1100.00%


static int filter_pred_none(struct filter_pred *pred, void *event) { return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi18100.00%1100.00%
Total18100.00%1100.00%

/* * regex_match_foo - Basic regex callbacks * * @str: the string to be searched * @r: the regex structure containing the pattern string * @len: the length of the string to be searched (including '\0') * * Note: * - @str might not be NULL-terminated if it's of type DYN_STRING * or STATIC_STRING */
static int regex_match_full(char *str, struct regex *r, int len) { if (strncmp(str, r->pattern, len) == 0) return 1; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker39100.00%1100.00%
Total39100.00%1100.00%


static int regex_match_front(char *str, struct regex *r, int len) { if (strncmp(str, r->pattern, r->len) == 0) return 1; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker3995.12%150.00%
Li Zefan24.88%150.00%
Total41100.00%2100.00%


static int regex_match_middle(char *str, struct regex *r, int len) { if (strnstr(str, r->pattern, len)) return 1; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker3491.89%150.00%
Li Zefan38.11%150.00%
Total37100.00%2100.00%


static int regex_match_end(char *str, struct regex *r, int len) { int strlen = len - 1; if (strlen >= r->len && memcmp(str + strlen - r->len, r->pattern, r->len) == 0) return 1; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker3761.67%150.00%
Li Zefan2338.33%150.00%
Total60100.00%2100.00%


static int regex_match_glob(char *str, struct regex *r, int len __maybe_unused) { if (glob_match(r->pattern, str)) return 1; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Masami Hiramatsu36100.00%1100.00%
Total36100.00%1100.00%

/** * filter_parse_regex - parse a basic regex * @buff: the raw regex * @len: length of the regex * @search: will point to the beginning of the string to compare * @not: tell whether the match will have to be inverted * * This passes in a buffer containing a regex and this function will * set search to point to the search part of the buffer and * return the type of search it is (see enum above). * This does modify buff. * * Returns enum type. * search returns the pointer to use for comparison. * not returns 1 if buff started with a '!' * 0 otherwise. */
enum regex_type filter_parse_regex(char *buff, int len, char **search, int *not) { int type = MATCH_FULL; int i; if (buff[0] == '!') { *not = 1; buff++; len--; } else *not = 0; *search = buff; for (i = 0; i < len; i++) { if (buff[i] == '*') { if (!i) { *search = buff + 1; type = MATCH_END_ONLY; } else if (i == len - 1) { if (type == MATCH_END_ONLY) type = MATCH_MIDDLE_ONLY; else type = MATCH_FRONT_ONLY; buff[i] = 0; break; } else { /* pattern continues, use full glob */ type = MATCH_GLOB; break; } } else if (strchr("[?\\", buff[i])) { type = MATCH_GLOB; break; } } return type; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker13578.03%150.00%
Masami Hiramatsu3821.97%150.00%
Total173100.00%2100.00%


static void filter_build_regex(struct filter_pred *pred) { struct regex *r = &pred->regex; char *search; enum regex_type type = MATCH_FULL; int not = 0; if (pred->op == OP_GLOB) { type = filter_parse_regex(r->pattern, r->len, &search, &not); r->len = strlen(search); memmove(r->pattern, search, r->len+1); } switch (type) { case MATCH_FULL: r->match = regex_match_full; break; case MATCH_FRONT_ONLY: r->match = regex_match_front; break; case MATCH_MIDDLE_ONLY: r->match = regex_match_middle; break; case MATCH_END_ONLY: r->match = regex_match_end; break; case MATCH_GLOB: r->match = regex_match_glob; break; } pred->not ^= not; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker11777.48%133.33%
Li Zefan2415.89%133.33%
Masami Hiramatsu106.62%133.33%
Total151100.00%3100.00%

enum move_type { MOVE_DOWN, MOVE_UP_FROM_LEFT, MOVE_UP_FROM_RIGHT };
static struct filter_pred * get_pred_parent(struct filter_pred *pred, struct filter_pred *preds, int index, enum move_type *move) { if (pred->parent & FILTER_PRED_IS_RIGHT) *move = MOVE_UP_FROM_RIGHT; else *move = MOVE_UP_FROM_LEFT; pred = &preds[pred->parent & ~FILTER_PRED_IS_RIGHT]; return pred; }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt61100.00%1100.00%
Total61100.00%1100.00%

enum walk_return { WALK_PRED_ABORT, WALK_PRED_PARENT, WALK_PRED_DEFAULT, }; typedef int (*filter_pred_walkcb_t) (enum move_type move, struct filter_pred *pred, int *err, void *data);
static int walk_pred_tree(struct filter_pred *preds, struct filter_pred *root, filter_pred_walkcb_t cb, void *data) { struct filter_pred *pred = root; enum move_type move = MOVE_DOWN; int done = 0; if (!preds) return -EINVAL; do { int err = 0, ret; ret = cb(move, pred, &err, data); if (ret == WALK_PRED_ABORT) return err; if (ret == WALK_PRED_PARENT) goto get_parent; switch (move) { case MOVE_DOWN: if (pred->left != FILTER_PRED_INVALID) { pred = &preds[pred->left]; continue; } goto get_parent; case MOVE_UP_FROM_LEFT: pred = &preds[pred->right]; move = MOVE_DOWN; continue; case MOVE_UP_FROM_RIGHT: get_parent: if (pred == root) break; pred = get_pred_parent(pred, preds, pred->parent, &move); continue; } done = 1; } while (!done); /* We are fine. */ return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa186100.00%1100.00%
Total186100.00%1100.00%

/* * A series of AND or ORs where found together. Instead of * climbing up and down the tree branches, an array of the * ops were made in order of checks. We can just move across * the array and short circuit if needed. */
static int process_ops(struct filter_pred *preds, struct filter_pred *op, void *rec) { struct filter_pred *pred; int match = 0; int type; int i; /* * Micro-optimization: We set type to true if op * is an OR and false otherwise (AND). Then we * just need to test if the match is equal to * the type, and if it is, we can short circuit the * rest of the checks: * * if ((match && op->op == OP_OR) || * (!match && op->op == OP_AND)) * return match; */ type = op->op == OP_OR; for (i = 0; i < op->val; i++) { pred = &preds[op->ops[i]]; if (!WARN_ON_ONCE(!pred->fn)) match = pred->fn(pred, rec); if (!!match == type) break; } /* If not of not match is equal to not of not, then it is a match */ return !!match == !op->not; }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt10287.18%250.00%
Jiri Olsa119.40%125.00%
Ingo Molnar43.42%125.00%
Total117100.00%4100.00%

struct filter_match_preds_data { struct filter_pred *preds; int match; void *rec; };
static int filter_match_preds_cb(enum move_type move, struct filter_pred *pred, int *err, void *data) { struct filter_match_preds_data *d = data; *err = 0; switch (move) { case MOVE_DOWN: /* only AND and OR have children */ if (pred->left != FILTER_PRED_INVALID) { /* If ops is set, then it was folded. */ if (!pred->ops) return WALK_PRED_DEFAULT; /* We can treat folded ops as a leaf node */ d->match = process_ops(d->preds, pred, d->rec); } else { if (!WARN_ON_ONCE(!pred->fn)) d->match = pred->fn(pred, d->rec); } return WALK_PRED_PARENT; case MOVE_UP_FROM_LEFT: /* * Check for short circuits. * * Optimization: !!match == (pred->op == OP_OR) * is the same as: * if ((match && pred->op == OP_OR) || * (!match && pred->op == OP_AND)) */ if (!!d->match == (pred->op == OP_OR)) return WALK_PRED_PARENT; break; case MOVE_UP_FROM_RIGHT: break; } return WALK_PRED_DEFAULT; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa6544.22%116.67%
Steven Rostedt6544.22%466.67%
Tom Zanussi1711.56%116.67%
Total147100.00%6100.00%

/* return 1 if event matches, 0 otherwise (discard) */
int filter_match_preds(struct event_filter *filter, void *rec) { struct filter_pred *preds; struct filter_pred *root; struct filter_match_preds_data data = { /* match is currently meaningless */ .match = -1, .rec = rec, }; int n_preds, ret; /* no filter is considered a match */ if (!filter) return 1; n_preds = filter->n_preds; if (!n_preds) return 1; /* * n_preds, root and filter->preds are protect with preemption disabled. */ root = rcu_dereference_sched(filter->root); if (!root) return 1; data.preds = preds = rcu_dereference_sched(filter->preds); ret = walk_pred_tree(preds, root, filter_match_preds_cb, &data); WARN_ON(ret); return data.match; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa9878.40%125.00%
Steven Rostedt1713.60%250.00%
Tom Zanussi108.00%125.00%
Total125100.00%4100.00%

EXPORT_SYMBOL_GPL(filter_match_preds);
static void parse_error(struct filter_parse_state *ps, int err, int pos) { ps->lasterr = err; ps->lasterr_pos = pos; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi29100.00%1100.00%
Total29100.00%1100.00%


static void remove_filter_string(struct event_filter *filter) { if (!filter) return; kfree(filter->filter_string); filter->filter_string = NULL; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi2480.00%150.00%
Steven Rostedt620.00%150.00%
Total30100.00%2100.00%


static int replace_filter_string(struct event_filter *filter, char *filter_string) { kfree(filter->filter_string); filter->filter_string = kstrdup(filter_string, GFP_KERNEL); if (!filter->filter_string) return -ENOMEM; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi47100.00%1100.00%
Total47100.00%1100.00%


static int append_filter_string(struct event_filter *filter, char *string) { int newlen; char *new_filter_string; BUG_ON(!filter->filter_string); newlen = strlen(filter->filter_string) + strlen(string) + 1; new_filter_string = kmalloc(newlen, GFP_KERNEL); if (!new_filter_string) return -ENOMEM; strcpy(new_filter_string, filter->filter_string); strcat(new_filter_string, string); kfree(filter->filter_string); filter->filter_string = new_filter_string; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi96100.00%1100.00%
Total96100.00%1100.00%


static void append_filter_err(struct filter_parse_state *ps, struct event_filter *filter) { int pos = ps->lasterr_pos; char *buf, *pbuf; buf = (char *)__get_free_page(GFP_KERNEL); if (!buf) return; append_filter_string(filter, "\n"); memset(buf, ' ', PAGE_SIZE); if (pos > PAGE_SIZE - 128) pos = 0; buf[pos] = '^'; pbuf = &buf[pos] + 1; sprintf(pbuf, "\nparse_error: %s\n", err_text[ps->lasterr]); append_filter_string(filter, buf); free_page((unsigned long) buf); }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi12199.18%150.00%
Michal Hocko10.82%150.00%
Total122100.00%2100.00%


static inline struct event_filter *event_filter(struct trace_event_file *file) { return file->filter; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi1894.74%150.00%
Steven Rostedt15.26%150.00%
Total19100.00%2100.00%

/* caller must hold event_mutex */
void print_event_filter(struct trace_event_file *file, struct trace_seq *s) { struct event_filter *filter = event_filter(file); if (filter && filter->filter_string) trace_seq_printf(s, "%s\n", filter->filter_string); else trace_seq_puts(s, "none\n"); }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi4688.46%233.33%
Oleg Nesterov23.85%116.67%
Li Zefan23.85%116.67%
Jovi Zhangwei11.92%116.67%
Steven Rostedt11.92%116.67%
Total52100.00%6100.00%


void print_subsystem_event_filter(struct event_subsystem *system, struct trace_seq *s) { struct event_filter *filter; mutex_lock(&event_mutex); filter = system->filter; if (filter && filter->filter_string) trace_seq_printf(s, "%s\n", filter->filter_string); else trace_seq_puts(s, DEFAULT_SYS_FILTER_MESSAGE "\n"); mutex_unlock(&event_mutex); }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi5380.30%116.67%
Steven Rostedt812.12%233.33%
Li Zefan46.06%233.33%
Jovi Zhangwei11.52%116.67%
Total66100.00%6100.00%


static int __alloc_pred_stack(struct pred_stack *stack, int n_preds) { stack->preds = kcalloc(n_preds + 1, sizeof(*stack->preds), GFP_KERNEL); if (!stack->preds) return -ENOMEM; stack->index = n_preds; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt5090.91%150.00%
Thomas Meyer59.09%150.00%
Total55100.00%2100.00%


static void __free_pred_stack(struct pred_stack *stack) { kfree(stack->preds); stack->index = 0; }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt24100.00%1100.00%
Total24100.00%1100.00%


static int __push_pred_stack(struct pred_stack *stack, struct filter_pred *pred) { int index = stack->index; if (WARN_ON(index == 0)) return -ENOSPC; stack->preds[--index] = pred; stack->index = index; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt55100.00%1100.00%
Total55100.00%1100.00%


static struct filter_pred * __pop_pred_stack(struct pred_stack *stack) { struct filter_pred *pred; int index = stack->index; pred = stack->preds[index++]; if (!pred) return NULL; stack->index = index; return pred; }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt52100.00%1100.00%
Total52100.00%1100.00%


static int filter_set_pred(struct event_filter *filter, int idx, struct pred_stack *stack, struct filter_pred *src) { struct filter_pred *dest = &filter->preds[idx]; struct filter_pred *left; struct filter_pred *right; *dest = *src; dest->index = idx; if (dest->op == OP_OR || dest->op == OP_AND) { right = __pop_pred_stack(stack); left = __pop_pred_stack(stack); if (!left || !right) return -EINVAL; /* * If both children can be folded * and they are the same op as this op or a leaf, * then this op can be folded. */ if (left->index & FILTER_PRED_FOLD && ((left->op == dest->op && !left->not) || left->left == FILTER_PRED_INVALID) && right->index & FILTER_PRED_FOLD && ((right->op == dest->op && !right->not) || right->left == FILTER_PRED_INVALID)) dest->index |= FILTER_PRED_FOLD; dest->left = left->index & ~FILTER_PRED_FOLD; dest->right = right->index & ~FILTER_PRED_FOLD; left->parent = dest->index & ~FILTER_PRED_FOLD; right->parent = dest->index | FILTER_PRED_IS_RIGHT; } else { /* * Make dest->left invalid to be used as a quick * way to know this is a leaf node. */ dest->left = FILTER_PRED_INVALID; /* All leafs allow folding the parent ops. */ dest->index |= FILTER_PRED_FOLD; } return __push_pred_stack(stack, dest); }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt21591.10%375.00%
Tom Zanussi218.90%125.00%
Total236100.00%4100.00%


static void __free_preds(struct event_filter *filter) { int i; if (filter->preds) { for (i = 0; i < filter->n_preds; i++) kfree(filter->preds[i].ops); kfree(filter->preds); filter->preds = NULL; } filter->a_preds = 0; filter->n_preds = 0; }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt74100.00%2100.00%
Total74100.00%2100.00%


static void filter_disable(struct trace_event_file *file) { unsigned long old_flags = file->flags; file->flags &= ~EVENT_FILE_FL_FILTERED; if (old_flags != file->flags) trace_buffered_event_disable(); }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt2156.76%375.00%
Tom Zanussi1643.24%125.00%
Total37100.00%4100.00%


static void __free_filter(struct event_filter *filter) { if (!filter) return; __free_preds(filter); kfree(filter->filter_string); kfree(filter); }

Contributors

PersonTokensPropCommitsCommitProp
Li Zefan3294.12%480.00%
Steven Rostedt25.88%120.00%
Total34100.00%5100.00%


void free_event_filter(struct event_filter *filter) { __free_filter(filter); }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi15100.00%1100.00%
Total15100.00%1100.00%


static struct event_filter *__alloc_filter(void) { struct event_filter *filter; filter = kzalloc(sizeof(*filter), GFP_KERNEL); return filter; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi2270.97%125.00%
Steven Rostedt516.13%125.00%
Li Zefan412.90%250.00%
Total31100.00%4100.00%


static int __alloc_preds(struct event_filter *filter, int n_preds) { struct filter_pred *pred; int i; if (filter->preds) __free_preds(filter); filter->preds = kcalloc(n_preds, sizeof(*filter->preds), GFP_KERNEL); if (!filter->preds) return -ENOMEM; filter->a_preds = n_preds; filter->n_preds = 0; for (i = 0; i < n_preds; i++) { pred = &filter->preds[i]; pred->fn = filter_pred_none; } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt6256.88%350.00%
Tom Zanussi4339.45%116.67%
Thomas Meyer32.75%116.67%
Li Zefan10.92%116.67%
Total109100.00%6100.00%


static inline void __remove_filter(struct trace_event_file *file) { filter_disable(file); remove_filter_string(file->filter); }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi2395.83%150.00%
Steven Rostedt14.17%150.00%
Total24100.00%2100.00%


static void filter_free_subsystem_preds(struct trace_subsystem_dir *dir, struct trace_array *tr) { struct trace_event_file *file; list_for_each_entry(file, &tr->events, list) { if (file->system != dir) continue; __remove_filter(file); } }

Contributors

PersonTokensPropCommitsCommitProp
Li Zefan2146.67%112.50%
Tom Zanussi1635.56%225.00%
Steven Rostedt511.11%450.00%
Oleg Nesterov36.67%112.50%
Total45100.00%8100.00%


static inline void __free_subsystem_filter(struct trace_event_file *file) { __free_filter(file->filter); file->filter = NULL; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi2392.00%133.33%
Li Zefan14.00%133.33%
Steven Rostedt14.00%133.33%
Total25100.00%3100.00%


static void filter_free_subsystem_filters(struct trace_subsystem_dir *dir, struct trace_array *tr) { struct trace_event_file *file; list_for_each_entry(file, &tr->events, list) { if (file->system != dir) continue; __free_subsystem_filter(file); } }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi3475.56%225.00%
Li Zefan48.89%112.50%
Steven Rostedt48.89%450.00%
Oleg Nesterov36.67%112.50%
Total45100.00%8100.00%


static int filter_add_pred(struct filter_parse_state *ps, struct event_filter *filter, struct filter_pred *pred, struct pred_stack *stack) { int err; if (WARN_ON(filter->n_preds == filter->a_preds)) { parse_error(ps, FILT_ERR_TOO_MANY_PREDS, 0); return -ENOSPC; } err = filter_set_pred(filter, filter->n_preds, stack, pred); if (err) return err; filter->n_preds++; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi6574.71%116.67%
Steven Rostedt1314.94%233.33%
Li Zefan55.75%116.67%
Jiri Olsa44.60%233.33%
Total87100.00%6100.00%


int filter_assign_type(const char *type) { if (strstr(type, "__data_loc") && strstr(type, "char")) return FILTER_DYN_STRING; if (strchr(type, '[') && strstr(type, "char")) return FILTER_STATIC_STRING; return FILTER_OTHER; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi2549.02%125.00%
Li Zefan1733.33%250.00%
Frédéric Weisbecker917.65%125.00%
Total51100.00%4100.00%


static bool is_legal_op(struct ftrace_event_field *field, enum filter_op_ids op) { if (is_string_field(field) && (op != OP_EQ && op != OP_NE && op != OP_GLOB)) return false; if (!is_string_field(field) && op == OP_GLOB) return false; return true; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi3357.89%125.00%
Li Zefan1831.58%125.00%
Yaowei Bai47.02%125.00%
Steven Rostedt23.51%125.00%
Total57100.00%4100.00%


static filter_pred_fn_t select_comparison_fn(enum filter_op_ids op, int field_size, int field_is_signed) { filter_pred_fn_t fn = NULL; switch (field_size) { case 8: if (op == OP_EQ || op == OP_NE) fn = filter_pred_64; else if (field_is_signed) fn = pred_funcs_s64[op - PRED_FUNC_START]; else fn = pred_funcs_u64[op - PRED_FUNC_START]; break; case 4: if (op == OP_EQ || op == OP_NE) fn = filter_pred_32; else if (field_is_signed) fn = pred_funcs_s32[op - PRED_FUNC_START]; else fn = pred_funcs_u32[op - PRED_FUNC_START]; break; case 2: if (op == OP_EQ || op == OP_NE) fn = filter_pred_16; else if (field_is_signed) fn = pred_funcs_s16[op - PRED_FUNC_START]; else fn = pred_funcs_u16[op - PRED_FUNC_START]; break; case 1: if (op == OP_EQ || op == OP_NE) fn = filter_pred_8; else if (field_is_signed) fn = pred_funcs_s8[op - PRED_FUNC_START]; else fn = pred_funcs_u8[op - PRED_FUNC_START]; break; } return fn; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi14874.75%250.00%
Steven Rostedt5025.25%250.00%
Total198100.00%4100.00%


static int init_pred(struct filter_parse_state *ps, struct ftrace_event_field *field, struct filter_pred *pred) { filter_pred_fn_t fn = filter_pred_none; unsigned long long val; int ret; pred->offset = field->offset; if (!is_legal_op(field, pred->op)) { parse_error(ps, FILT_ERR_ILLEGAL_FIELD_OP, 0); return -EINVAL; } if (field->filter_type == FILTER_COMM) { filter_build_regex(pred); fn = filter_pred_comm; pred->regex.field_len = TASK_COMM_LEN; } else if (is_string_field(field)) { filter_build_regex(pred); if (field->filter_type == FILTER_STATIC_STRING) { fn = filter_pred_string; pred->regex.field_len = field->size; } else if (field->filter_type == FILTER_DYN_STRING) fn = filter_pred_strloc; else fn = filter_pred_pchar; } else if (is_function_field(field)) { if (strcmp(field->name, "ip")) { parse_error(ps, FILT_ERR_IP_FIELD_ONLY, 0); return -EINVAL; } } else { if (field->is_signed) ret = kstrtoll(pred->regex.pattern, 0, &val); else ret = kstrtoull(pred->regex.pattern, 0, &val); if (ret) { parse_error(ps, FILT_ERR_ILLEGAL_INTVAL, 0); return -EINVAL; } pred->val = val; if (field->filter_type == FILTER_CPU) fn = filter_pred_cpu; else fn = select_comparison_fn(pred->op, field->size, field->is_signed); if (!fn) { parse_error(ps, FILT_ERR_INVALID_OP, 0); return -EINVAL; } } if (pred->op == OP_NE) pred->not ^= 1; pred->fn = fn; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi15446.81%212.50%
Li Zefan5015.20%425.00%
Jiri Olsa4613.98%425.00%
Frédéric Weisbecker309.12%212.50%
Daniel Wagner267.90%16.25%
Steven Rostedt216.38%212.50%
Daniel Walter20.61%16.25%
Total329100.00%16100.00%


static void parse_init(struct filter_parse_state *ps, struct filter_op *ops, char *infix_string) { memset(ps, '\0', sizeof(*ps)); ps->infix.string = infix_string; ps->infix.cnt = strlen(infix_string); ps->ops = ops; INIT_LIST_HEAD(&ps->opstack); INIT_LIST_HEAD(&ps->postfix); }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi74100.00%2100.00%
Total74100.00%2100.00%


static char infix_next(struct filter_parse_state *ps) { if (!ps->infix.cnt) return 0; ps->infix.cnt--; return ps->infix.string[ps->infix.tail++]; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi3373.33%266.67%
Steven Rostedt1226.67%133.33%
Total45100.00%3100.00%


static char infix_peek(struct filter_parse_state *ps) { if (ps->infix.tail == strlen(ps->infix.string)) return 0; return ps->infix.string[ps->infix.tail]; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi45100.00%2100.00%
Total45100.00%2100.00%


static void infix_advance(struct filter_parse_state *ps) { if (!ps->infix.cnt) return; ps->infix.cnt--; ps->infix.tail++; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi2571.43%266.67%
Steven Rostedt1028.57%133.33%
Total35100.00%3100.00%


static inline int is_precedence_lower(struct filter_parse_state *ps, int a, int b) { return ps->ops[a].precedence < ps->ops[b].precedence; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi37100.00%2100.00%
Total37100.00%2100.00%


static inline int is_op_char(struct filter_parse_state *ps, char c) { int i; for (i = 0; strcmp(ps->ops[i].string, "OP_NONE"); i++) { if (ps->ops[i].string[0] == c) return 1; } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi65100.00%2100.00%
Total65100.00%2100.00%


static int infix_get_op(struct filter_parse_state *ps, char firstc) { char nextc = infix_peek(ps); char opstr[3]; int i; opstr[0] = firstc; opstr[1] = nextc; opstr[2] = '\0'; for (i = 0; strcmp(ps->ops[i].string, "OP_NONE"); i++) { if (!strcmp(opstr, ps->ops[i].string)) { infix_advance(ps); return ps->ops[i].id; } } opstr[1] = '\0'; for (i = 0; strcmp(ps->ops[i].string, "OP_NONE"); i++) { if (!strcmp(opstr, ps->ops[i].string)) return ps->ops[i].id; } return OP_NONE; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi173100.00%5100.00%
Total173100.00%5100.00%


static inline void clear_operand_string(struct filter_parse_state *ps) { memset(ps->operand.string, '\0', MAX_FILTER_STR_VAL); ps->operand.tail = 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi33100.00%2100.00%
Total33100.00%2100.00%


static inline int append_operand_char(struct filter_parse_state *ps, char c) { if (ps->operand.tail == MAX_FILTER_STR_VAL - 1) return -EINVAL; ps->operand.string[ps->operand.tail++] = c; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi4896.00%266.67%
Li Zefan24.00%133.33%
Total50100.00%3100.00%


static int filter_opstack_push(struct filter_parse_state *ps, enum filter_op_ids op) { struct opstack_op *opstack_op; opstack_op = kmalloc(sizeof(*opstack_op), GFP_KERNEL); if (!opstack_op) return -ENOMEM; opstack_op->op = op; list_add(&opstack_op->list, &ps->opstack); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi6296.88%375.00%
Steven Rostedt23.12%125.00%
Total64100.00%4100.00%


static int filter_opstack_empty(struct filter_parse_state *ps) { return list_empty(&ps->opstack); }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi20100.00%3100.00%
Total20100.00%3100.00%


static int filter_opstack_top(struct filter_parse_state *ps) { struct opstack_op *opstack_op; if (filter_opstack_empty(ps)) return OP_NONE; opstack_op = list_first_entry(&ps->opstack, struct opstack_op, list); return opstack_op->op; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi46100.00%3100.00%
Total46100.00%3100.00%


static int filter_opstack_pop(struct filter_parse_state *ps) { struct opstack_op *opstack_op; enum filter_op_ids op; if (filter_opstack_empty(ps)) return OP_NONE; opstack_op = list_first_entry(&ps->opstack, struct opstack_op, list); op = opstack_op->op; list_del(&opstack_op->list); kfree(opstack_op); return op; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi6597.01%583.33%
Steven Rostedt22.99%116.67%
Total67100.00%6100.00%


static void filter_opstack_clear(struct filter_parse_state *ps) { while (!filter_opstack_empty(ps)) filter_opstack_pop(ps); }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi24100.00%2100.00%
Total24100.00%2100.00%


static char *curr_operand(struct filter_parse_state *ps) { return ps->operand.string; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi19100.00%2100.00%
Total19100.00%2100.00%


static int postfix_append_operand(struct filter_parse_state *ps, char *operand) { struct postfix_elt *elt; elt = kmalloc(sizeof(*elt), GFP_KERNEL); if (!elt) return -ENOMEM; elt->op = OP_NONE; elt->operand = kstrdup(operand, GFP_KERNEL); if (!elt->operand) { kfree(elt); return -ENOMEM; } list_add_tail(&elt->list, &ps->postfix); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi93100.00%4100.00%
Total93100.00%4100.00%


static int postfix_append_op(struct filter_parse_state *ps, enum filter_op_ids op) { struct postfix_elt *elt; elt = kmalloc(sizeof(*elt), GFP_KERNEL); if (!elt) return -ENOMEM; elt->op = op; elt->operand = NULL; list_add_tail(&elt->list, &ps->postfix); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi6897.14%375.00%
Steven Rostedt22.86%125.00%
Total70100.00%4100.00%


static void postfix_clear(struct filter_parse_state *ps) { struct postfix_elt *elt; while (!list_empty(&ps->postfix)) { elt = list_first_entry(&ps->postfix, struct postfix_elt, list); list_del(&elt->list); kfree(elt->operand); kfree(elt); } }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi5585.94%375.00%
Li Zefan914.06%125.00%
Total64100.00%4100.00%


static int filter_parse(struct filter_parse_state *ps) { enum filter_op_ids op, top_op; int in_string = 0; char ch; while ((ch = infix_next(ps))) { if (ch == '"') { in_string ^= 1; continue; } if (in_string) goto parse_operand; if (isspace(ch)) continue; if (is_op_char(ps, ch)) { op = infix_get_op(ps, ch); if (op == OP_NONE) { parse_error(ps, FILT_ERR_INVALID_OP, 0); return -EINVAL; } if (strlen(curr_operand(ps))) { postfix_append_operand(ps, curr_operand(ps)); clear_operand_string(ps); } while (!filter_opstack_empty(ps)) { top_op = filter_opstack_top(ps); if (!is_precedence_lower(ps, top_op, op)) { top_op = filter_opstack_pop(ps); postfix_append_op(ps, top_op); continue; } break; } filter_opstack_push(ps, op); continue; } if (ch == '(') { filter_opstack_push(ps, OP_OPEN_PAREN); continue; } if (ch == ')') { if (strlen(curr_operand(ps))) { postfix_append_operand(ps, curr_operand(ps)); clear_operand_string(ps); } top_op = filter_opstack_pop(ps); while (top_op != OP_NONE) { if (top_op == OP_OPEN_PAREN) break; postfix_append_op(ps, top_op); top_op = filter_opstack_pop(ps); } if (top_op == OP_NONE) { parse_error(ps, FILT_ERR_UNBALANCED_PAREN, 0); return -EINVAL; } continue; } parse_operand: if (append_operand_char(ps, ch)) { parse_error(ps, FILT_ERR_OPERAND_TOO_LONG, 0); return -EINVAL; } } if (strlen(curr_operand(ps))) postfix_append_operand(ps, curr_operand(ps)); while (!filter_opstack_empty(ps)) { top_op = filter_opstack_pop(ps); if (top_op == OP_NONE) break; if (top_op == OP_OPEN_PAREN) { parse_error(ps, FILT_ERR_UNBALANCED_PAREN, 0); return -EINVAL; } postfix_append_op(ps, top_op); } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi36691.73%666.67%
Frédéric Weisbecker266.52%111.11%
Steven Rostedt71.75%222.22%
Total399100.00%9100.00%


static struct filter_pred *create_pred(struct filter_parse_state *ps, struct trace_event_call *call, enum filter_op_ids op, char *operand1, char *operand2) { struct ftrace_event_field *field; static struct filter_pred pred; memset(&pred, 0, sizeof(pred)); pred.op = op; if (op == OP_AND || op == OP_OR) return &pred; if (!operand1 || !operand2) { parse_error(ps, FILT_ERR_MISSING_FIELD, 0); return NULL; } field = trace_find_event_field(call, operand1); if (!field) { parse_error(ps, FILT_ERR_FIELD_NOT_FOUND, 0); return NULL; } strcpy(pred.regex.pattern, operand2); pred.regex.len = strlen(pred.regex.pattern); pred.field = field; return init_pred(ps, field, &pred) ? NULL : &pred; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa9354.39%428.57%
Tom Zanussi6336.84%535.71%
Frédéric Weisbecker105.85%17.14%
Steven Rostedt31.75%214.29%
Jovi Zhangwei10.58%17.14%
Li Zefan10.58%17.14%
Total171100.00%14100.00%


static int check_preds(struct filter_parse_state *ps) { int n_normal_preds = 0, n_logical_preds = 0; struct postfix_elt *elt; int cnt = 0; list_for_each_entry(elt, &ps->postfix, list) { if (elt->op == OP_NONE) { cnt++; continue; } if (elt->op == OP_AND || elt->op == OP_OR) { n_logical_preds++; cnt--; continue; } if (elt->op != OP_NOT) cnt--; n_normal_preds++; /* all ops should have operands */ if (cnt < 0) break; } if (cnt != 1 || !n_normal_preds || n_logical_preds >= n_normal_preds) { parse_error(ps, FILT_ERR_INVALID_FILTER, 0); return -EINVAL; } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi9472.31%250.00%
Steven Rostedt3627.69%250.00%
Total130100.00%4100.00%


static int count_preds(struct filter_parse_state *ps) { struct postfix_elt *elt; int n_preds = 0; list_for_each_entry(elt, &ps->postfix, list) { if (elt->op == OP_NONE) continue; n_preds++; } return n_preds; }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt46100.00%1100.00%
Total46100.00%1100.00%

struct check_pred_data { int count; int max; };
static int check_pred_tree_cb(enum move_type move, struct filter_pred *pred, int *err, void *data) { struct check_pred_data *d = data; if (WARN_ON(d->count++ > d->max)) { *err = -EINVAL; return WALK_PRED_ABORT; } return WALK_PRED_DEFAULT; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa3255.17%150.00%
Steven Rostedt2644.83%150.00%
Total58100.00%2100.00%

/* * The tree is walked at filtering of an event. If the tree is not correctly * built, it may cause an infinite loop. Check here that the tree does * indeed terminate. */
static int check_pred_tree(struct event_filter *filter, struct filter_pred *root) { struct check_pred_data data = { /* * The max that we can hit a node is three times. * Once going down, once coming up from left, and * once coming up from right. This is more than enough * since leafs are only hit a single time. */ .max = 3 * filter->n_preds, .count = 0, }; return walk_pred_tree(filter->preds, root, check_pred_tree_cb, &data); }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa3771.15%150.00%
Steven Rostedt1528.85%150.00%
Total52100.00%2100.00%


static int count_leafs_cb(enum move_type move, struct filter_pred *pred, int *err, void *data) { int *count = data; if ((move == MOVE_DOWN) && (pred->left == FILTER_PRED_INVALID)) (*count)++; return WALK_PRED_DEFAULT; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa2953.70%114.29%
Steven Rostedt1425.93%228.57%
Tom Zanussi814.81%228.57%
Li Zefan35.56%228.57%
Total54100.00%7100.00%


static int count_leafs(struct filter_pred *preds, struct filter_pred *root) { int count = 0, ret; ret = walk_pred_tree(preds, root, count_leafs_cb, &count); WARN_ON(ret); return count; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa2964.44%150.00%
Steven Rostedt1635.56%150.00%
Total45100.00%2100.00%

struct fold_pred_data { struct filter_pred *root; int count; int children; };
static int fold_pred_cb(enum move_type move, struct filter_pred *pred, int *err, void *data) { struct fold_pred_data *d = data; struct filter_pred *root = d->root; if (move != MOVE_DOWN) return WALK_PRED_DEFAULT; if (pred->left != FILTER_PRED_INVALID) return WALK_PRED_DEFAULT; if (WARN_ON(d->count == d->children)) { *err = -EINVAL; return WALK_PRED_ABORT; } pred->index &= ~FILTER_PRED_FOLD; root->ops[d->count++] = pred->index; return WALK_PRED_DEFAULT; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa107100.00%1100.00%
Total107100.00%1100.00%


static int fold_pred(struct filter_pred *preds, struct filter_pred *root) { struct fold_pred_data data = { .root = root, .count = 0, }; int children; /* No need to keep the fold flag */ root->index &= ~FILTER_PRED_FOLD; /* If the root is a leaf then do nothing */ if (root->left == FILTER_PRED_INVALID) return 0; /* count the children */ children = count_leafs(preds, &preds[root->left]); children += count_leafs(preds, &preds[root->right]); root->ops = kcalloc(children, sizeof(*root->ops), GFP_KERNEL); if (!root->ops) return -ENOMEM; root->val = children; data.children = children; return walk_pred_tree(preds, root, fold_pred_cb, &data); }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt12085.11%133.33%
Jiri Olsa1812.77%133.33%
Thomas Meyer32.13%133.33%
Total141100.00%3100.00%


static int fold_pred_tree_cb(enum move_type move, struct filter_pred *pred, int *err, void *data) { struct filter_pred *preds = data; if (move != MOVE_DOWN) return WALK_PRED_DEFAULT; if (!(pred->index & FILTER_PRED_FOLD)) return WALK_PRED_DEFAULT; *err = fold_pred(preds, pred); if (*err) return WALK_PRED_ABORT; /* eveyrhing below is folded, continue with parent */ return WALK_PRED_PARENT; }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt4458.67%150.00%
Jiri Olsa3141.33%150.00%
Total75100.00%2100.00%

/* * To optimize the processing of the ops, if we have several "ors" or * "ands" together, we can put them in an array and process them all * together speeding up the filter logic. */
static int fold_pred_tree(struct event_filter *filter, struct filter_pred *root) { return walk_pred_tree(filter->preds, root, fold_pred_tree_cb, filter->preds); }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa2165.62%150.00%
Steven Rostedt1134.38%150.00%
Total32100.00%2100.00%


static int replace_preds(struct trace_event_call *call, struct event_filter *filter, struct filter_parse_state *ps, bool dry_run) { char *operand1 = NULL, *operand2 = NULL; struct filter_pred *pred; struct filter_pred *root; struct postfix_elt *elt; struct pred_stack stack = { }; /* init to NULL */ int err; int n_preds = 0; n_preds = count_preds(ps); if (n_preds >= MAX_FILTER_PRED) { parse_error(ps, FILT_ERR_TOO_MANY_PREDS, 0); return -ENOSPC; } err = check_preds(ps); if (err) return err; if (!dry_run) { err = __alloc_pred_stack(&stack, n_preds); if (err) return err; err = __alloc_preds(filter, n_preds); if (err) goto fail; } n_preds = 0; list_for_each_entry(elt, &ps->postfix, list) { if (elt->op == OP_NONE) { if (!operand1) operand1 = elt->operand; else if (!operand2) operand2 = elt->operand; else { parse_error(ps, FILT_ERR_TOO_MANY_OPERANDS, 0); err = -EINVAL; goto fail; } continue; } if (elt->op == OP_NOT) { if (!n_preds || operand1 || operand2) { parse_error(ps, FILT_ERR_ILLEGAL_NOT_OP, 0); err = -EINVAL; goto fail; } if (!dry_run) filter->preds[n_preds - 1].not ^= 1; continue; } if (WARN_ON(n_preds++ == MAX_FILTER_PRED)) { parse_error(ps, FILT_ERR_TOO_MANY_PREDS, 0); err = -ENOSPC; goto fail; } pred = create_pred(ps, call, elt->op, operand1, operand2); if (!pred) { err = -EINVAL; goto fail; } if (!dry_run) { err = filter_add_pred(ps, filter, pred, &stack); if (err) goto fail; } operand1 = operand2 = NULL; } if (!dry_run) { /* We should have one item left on the stack */ pred = __pop_pred_stack(&stack); if (!pred) return -EINVAL; /* This item is where we start from in matching */ root = pred; /* Make sure the stack is empty */ pred = __pop_pred_stack(&stack); if (WARN_ON(pred)) { err = -EINVAL; filter->root = NULL; goto fail; } err = check_pred_tree(filter, root); if (err) goto fail; /* Optimize the tree */ err = fold_pred_tree(filter, root); if (err) goto fail; /* We don't set root until we know it works */ barrier(); filter->root = root; } err = 0; fail: __free_pred_stack(&stack); return err; }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt26253.80%631.58%
Tom Zanussi18237.37%736.84%
Li Zefan285.75%315.79%
Jiri Olsa153.08%315.79%
Total487100.00%19100.00%


static inline void event_set_filtered_flag(struct trace_event_file *file) { unsigned long old_flags = file->flags; file->flags |= EVENT_FILE_FL_FILTERED; if (old_flags != file->flags) trace_buffered_event_enable(); }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt2156.76%375.00%
Tom Zanussi1643.24%125.00%
Total37100.00%4100.00%


static inline void event_set_filter(struct trace_event_file *file, struct event_filter *filter) { rcu_assign_pointer(file->filter, filter); }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi2596.15%150.00%
Steven Rostedt13.85%150.00%
Total26100.00%2100.00%


static inline void event_clear_filter(struct trace_event_file *file) { RCU_INIT_POINTER(file->filter, NULL); }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi2095.24%150.00%
Steven Rostedt14.76%150.00%
Total21100.00%2100.00%


static inline void event_set_no_set_filter_flag(struct trace_event_file *file) { file->flags |= EVENT_FILE_FL_NO_SET_FILTER; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi1688.89%133.33%
Steven Rostedt211.11%266.67%
Total18100.00%3100.00%


static inline void event_clear_no_set_filter_flag(struct trace_event_file *file) { file->flags &= ~EVENT_FILE_FL_NO_SET_FILTER; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi1473.68%125.00%
Steven Rostedt526.32%375.00%
Total19100.00%4100.00%


static inline bool event_no_set_filter_flag(struct trace_event_file *file) { if (file->flags & EVENT_FILE_FL_NO_SET_FILTER) return true; return false; }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi1973.08%120.00%
Steven Rostedt519.23%360.00%
Li Zefan27.69%120.00%
Total26100.00%5100.00%

struct filter_list { struct list_head list; struct event_filter *filter; };
static int replace_system_preds(struct trace_subsystem_dir *dir, struct trace_array *tr, struct filter_parse_state *ps, char *filter_string) { struct trace_event_file *file; struct filter_list *filter_item; struct filter_list *tmp; LIST_HEAD(filter_list); bool fail = true; int err; list_for_each_entry(file, &tr->events, list) { if (file->system != dir) continue; /* * Try to see if the filter can be applied * (filter arg is ignored on dry_run) */ err = replace_preds(file->event_call, NULL, ps, true); if (err) event_set_no_set_filter_flag(file); else event_clear_no_set_filter_flag(file); } list_for_each_entry(file, &tr->events, list) { struct event_filter *filter; if (file->system != dir) continue; if (event_no_set_filter_flag(file)) continue; filter_item = kzalloc(sizeof(*filter_item), GFP_KERNEL); if (!filter_item) goto fail_mem; list_add_tail(&filter_item->list, &filter_list); filter_item->filter = __alloc_filter(); if (!filter_item->filter) goto fail_mem; filter = filter_item->filter; /* Can only fail on no memory */ err = replace_filter_string(filter, filter_string); if (err) goto fail_mem; err = replace_preds(file->event_call, filter, ps, false); if (err) { filter_disable(file); parse_error(ps, FILT_ERR_BAD_SUBSYS_FILTER, 0); append_filter_err(ps, filter); } else event_set_filtered_flag(file); /* * Regardless of if this returned an error, we still * replace the filter for the call. */ filter = event_filter(file); event_set_filter(file, filter_item->filter); filter_item->filter = filter; fail = false; } if (fail) goto fail; /* * The calls can still be using the old filters. * Do a synchronize_sched() to ensure all calls are * done with them before we free them. */ synchronize_sched(); list_for_each_entry_safe(filter_item, tmp, &filter_list, list) { __free_filter(filter_item->filter); list_del(&filter_item->list); kfree(filter_item); } return 0; fail: /* No call succeeded */ list_for_each_entry_safe(filter_item, tmp, &filter_list, list) { list_del(&filter_item->list); kfree(filter_item); } parse_error(ps, FILT_ERR_BAD_SUBSYS_FILTER, 0); return -EINVAL; fail_mem: /* If any call succeeded, we still need to sync */ if (!fail) synchronize_sched(); list_for_each_entry_safe(filter_item, tmp, &filter_list, list) { __free_filter(filter_item->filter); list_del(&filter_item->list); kfree(filter_item); } return -ENOMEM; }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt26262.98%529.41%
Li Zefan8420.19%317.65%
Tom Zanussi4811.54%529.41%
Oleg Nesterov112.64%15.88%
Ingo Molnar51.20%15.88%
Tejun Heo30.72%15.88%
Xiao Guangrong30.72%15.88%
Total416100.00%17100.00%


static int create_filter_start(char *filter_str, bool set_str, struct filter_parse_state **psp, struct event_filter **filterp) { struct event_filter *filter; struct filter_parse_state *ps = NULL; int err = 0; WARN_ON_ONCE(*psp || *filterp); /* allocate everything, and if any fails, free all and fail */ filter = __alloc_filter(); if (filter && set_str) err = replace_filter_string(filter, filter_str); ps = kzalloc(sizeof(*ps), GFP_KERNEL); if (!filter || !ps || err) { kfree(ps); __free_filter(filter); return -ENOMEM; } /* we're committed to creating a new filter */ *filterp = filter; *psp = ps; parse_init(ps, filter_ops, filter_str); err = filter_parse(ps); if (err && set_str) append_filter_err(ps, filter); return err; }

Contributors

PersonTokensPropCommitsCommitProp
Tejun Heo11172.08%120.00%
Tom Zanussi2415.58%240.00%
Steven Rostedt1912.34%240.00%
Total154100.00%5100.00%


static void create_filter_finish(struct filter_parse_state *ps) { if (ps) { filter_opstack_clear(ps); postfix_clear(ps); kfree(ps); } }

Contributors

PersonTokensPropCommitsCommitProp
Tejun Heo3196.88%266.67%
Steven Rostedt13.12%133.33%
Total32100.00%3100.00%

/** * create_filter - create a filter for a trace_event_call * @call: trace_event_call to create a filter for * @filter_str: filter string * @set_str: remember @filter_str and enable detailed error in filter * @filterp: out param for created filter (always updated on return) * * Creates a filter for @call with @filter_str. If @set_str is %true, * @filter_str is copied and recorded in the new filter. * * On success, returns 0 and *@filterp points to the new filter. On * failure, returns -errno and *@filterp may point to %NULL or to a new * filter. In the latter case, the returned filter contains error * information if @set_str is %true and the caller is responsible for * freeing it. */
static int create_filter(struct trace_event_call *call, char *filter_str, bool set_str, struct event_filter **filterp) { struct event_filter *filter = NULL; struct filter_parse_state *ps = NULL; int err; err = create_filter_start(filter_str, set_str, &ps, &filter); if (!err) { err = replace_preds(call, filter, ps, false); if (err && set_str) append_filter_err(ps, filter); } if (err && !set_str) { free_event_filter(filter); filter = NULL; } create_filter_finish(ps); *filterp = filter; return err; }

Contributors

PersonTokensPropCommitsCommitProp
Tejun Heo9175.83%225.00%
Steven Rostedt2420.00%450.00%
Tom Zanussi54.17%225.00%
Total120100.00%8100.00%


int create_event_filter(struct trace_event_call *call, char *filter_str, bool set_str, struct event_filter **filterp) { return create_filter(call, filter_str, set_str, filterp); }

Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi3497.14%150.00%
Steven Rostedt12.86%150.00%
Total35100.00%2100.00%

/** * create_system_filter - create a filter for an event_subsystem * @system: event_subsystem to create a filter for * @filter_str: filter string * @filterp: out param for created filter (always updated on return) * * Identical to create_filter() except that it creates a subsystem filter * and always remembers @filter_str. */
static int create_system_filter(struct trace_subsystem_dir *dir, struct trace_array *tr, char *filter_str, struct event_filter **filterp) { struct event_filter *filter = NULL; struct filter_parse_state *ps = NULL; int err; err = create_filter_start(filter_str, true, &ps, &filter); if (!err) { err = replace_system_preds(dir, tr, ps, filter_str); if (!err) { /* System filters just show a default message */ kfree(filter->filter_string); filter->filter_string = NULL; } else { append_filter_err(ps, filter); } } create_filter_finish(ps); *filterp = filter; return err; }

Contributors

PersonTokensPropCommitsCommitProp
Tejun Heo10384.43%114.29%
Tom Zanussi1310.66%342.86%
Li Zefan32.46%114.29%
Oleg Nesterov21.64%114.29%
Steven Rostedt10.82%114.29%
Total122100.00%7100.00%

/* caller must hold event_mutex */
int apply_event_filter(struct trace_event_file *file, char *filter_string) { struct trace_event_call *call = file->event_call; struct event_filter *filter; int err; if (!strcmp(strstrip(filter_string), "0")) { filter_disable(file); filter = event_filter(file); if (!filter) return 0; event_clear_filter(file); /* Make sure the filter is not being used */ synchronize_sched(); __free_filter(filter); return 0; } err = create_filter(call, filter_string, true, &filter); /* * Always swap the call filter with the new filter * even if there was an error. If there was an error * in the filter, we disable the filter and show the error * string */ if (filter) { struct event_filter *tmp; tmp = event_filter(file); if (!err) event_set_filtered_flag(file); else filter_disable(file); event_set_filter(file, filter); if (tmp) { /* Make sure the call is done with the filter */ synchronize_sched(); __free_filter(tmp); } } return err; }

Contributors

PersonTokensPropCommitsCommitProp
Tejun Heo6340.13%218.18%
Tom Zanussi5736.31%327.27%
Steven Rostedt2918.47%327.27%
Oleg Nesterov63.82%19.09%
Li Zefan21.27%218.18%
Total157100.00%11100.00%


int apply_subsystem_event_filter(struct trace_subsystem_dir *dir, char *filter_string) { struct event_subsystem *system = dir->subsystem; struct trace_array *tr = dir->tr; struct event_filter *filter; int err = 0; mutex_lock(&event_mutex); /* Make sure the system still has events */ if (!dir->nr_events) { err = -ENODEV; goto out_unlock; } if (!strcmp(strstrip(filter_string), "0")) { filter_free_subsystem_preds(dir, tr); remove_filter_string(system->filter); filter = system->filter; system->filter = NULL; /* Ensure all filters are no longer used */ synchronize_sched(); filter_free_subsystem_filters(dir, tr); __free_filter(filter); goto out_unlock; } err = create_system_filter(dir, tr, filter_string, &filter); if (filter) { /* * No event actually uses the system filter * we can free it without synchronize_sched(). */ __free_filter(system->filter); system->filter = filter; } out_unlock: mutex_unlock(&event_mutex); return err; }

Contributors

PersonTokensPropCommitsCommitProp
Steven Rostedt8247.95%428.57%
Tom Zanussi6739.18%428.57%
Tejun Heo84.68%17.14%
Li Zefan84.68%321.43%
Ingo Molnar31.75%17.14%
Oleg Nesterov31.75%17.14%
Total171100.00%14100.00%

#ifdef CONFIG_PERF_EVENTS
void ftrace_profile_free_filter(struct perf_event *event) { struct event_filter *filter = event->filter; event->filter = NULL; __free_filter(filter); }

Contributors

PersonTokensPropCommitsCommitProp
Li Zefan2996.67%150.00%
Steven Rostedt13.33%150.00%
Total30100.00%2100.00%

struct function_filter_data { struct ftrace_ops *ops; int first_filter; int first_notrace; }; #ifdef CONFIG_FUNCTION_TRACER
static char ** ftrace_function_filter_re(char *buf, int len, int *count) { char *str, **re; str = kstrndup(buf, len, GFP_KERNEL); if (!str) return NULL; /* * The argv_split function takes white space * as a separator, so convert ',' into spaces. */ strreplace(str, ',', ' '); re = argv_split(GFP_KERNEL, str, count); kfree(str); return re; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa5472.00%240.00%
Li Zefan1722.67%120.00%
Rasmus Villemoes34.00%120.00%
Ingo Molnar11.33%120.00%
Total75100.00%5100.00%


static int ftrace_function_set_regexp(struct ftrace_ops *ops, int filter, int reset, char *re, int len) { int ret; if (filter) ret = ftrace_set_filter(ops, re, len, reset); else ret = ftrace_set_notrace(ops, re, len, reset); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa5081.97%133.33%
Li Zefan1016.39%133.33%
Tejun Heo11.64%133.33%
Total61100.00%3100.00%


static int __ftrace_function_set_filter(int filter, char *buf, int len, struct function_filter_data *data) { int i, re_cnt, ret = -EINVAL; int *reset; char **re; reset = filter ? &data->first_filter : &data->first_notrace; /* * The 'ip' field could have multiple filters set, separated * either by space or comma. We first cut the filter and apply * all pieces separatelly. */ re = ftrace_function_filter_re(buf, len, &re_cnt); if (!re) return -EINVAL; for (i = 0; i < re_cnt; i++) { ret = ftrace_function_set_regexp(data->ops, filter, *reset, re[i], strlen(re[i])); if (ret) break; if (*reset) *reset = 0; } argv_free(re); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa13797.16%266.67%
Li Zefan42.84%133.33%
Total141100.00%3100.00%


static int ftrace_function_check_pred(struct filter_pred *pred, int leaf) { struct ftrace_event_field *field = pred->field; if (leaf) { /* * Check the leaf predicate for function trace, verify: * - only '==' and '!=' is used * - the 'ip' field is used */ if ((pred->op != OP_EQ) && (pred->op != OP_NE)) return -EINVAL; if (strcmp(field->name, "ip")) return -EINVAL; } else { /* * Check the non leaf predicate for function trace, verify: * - only '||' is used */ if (pred->op != OP_OR) return -EINVAL; } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa86100.00%1100.00%
Total86100.00%1100.00%


static int ftrace_function_set_filter_cb(enum move_type move, struct filter_pred *pred, int *err, void *data) { /* Checking the node is valid for function trace. */ if ((move != MOVE_DOWN) || (pred->left != FILTER_PRED_INVALID)) { *err = ftrace_function_check_pred(pred, 0); } else { *err = ftrace_function_check_pred(pred, 1); if (*err) return WALK_PRED_ABORT; *err = __ftrace_function_set_filter(pred->op == OP_EQ, pred->regex.pattern, pred->regex.len, data); } return (*err) ? WALK_PRED_ABORT : WALK_PRED_DEFAULT; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa109100.00%1100.00%
Total109100.00%1100.00%


static int ftrace_function_set_filter(struct perf_event *event, struct event_filter *filter) { struct function_filter_data data = { .first_filter = 1, .first_notrace = 1, .ops = &event->ftrace_ops, }; return walk_pred_tree(filter->preds, filter->root, ftrace_function_set_filter_cb, &data); }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa57100.00%1100.00%
Total57100.00%1100.00%

#else
static int ftrace_function_set_filter(struct perf_event *event, struct event_filter *filter) { return -ENODEV; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa20100.00%1100.00%
Total20100.00%1100.00%

#endif /* CONFIG_FUNCTION_TRACER */
int ftrace_profile_set_filter(struct perf_event *event, int event_id, char *filter_str) { int err; struct event_filter *filter; struct trace_event_call *call; mutex_lock(&event_mutex); call = event->tp_event; err = -EINVAL; if (!call) goto out_unlock; err = -EEXIST; if (event->filter) goto out_unlock; err = create_filter(call, filter_str, false, &filter); if (err) goto free_filter; if (ftrace_event_is_function(call)) err = ftrace_function_set_filter(event, filter); else event->filter = filter; free_filter: if (err || ftrace_event_is_function(call)) __free_filter(filter); out_unlock: mutex_unlock(&event_mutex); return err; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa12287.14%120.00%
Li Zefan1410.00%120.00%
Ingo Molnar21.43%120.00%
Steven Rostedt21.43%240.00%
Total140100.00%5100.00%

#endif /* CONFIG_PERF_EVENTS */ #ifdef CONFIG_FTRACE_STARTUP_TEST #include <linux/types.h> #include <linux/tracepoint.h> #define CREATE_TRACE_POINTS #include "trace_events_filter_test.h" #define DATA_REC(m, va, vb, vc, vd, ve, vf, vg, vh, nvisit) \ { \ .filter = FILTER, \ .rec = { .a = va, .b = vb, .c = vc, .d = vd, \ .e = ve, .f = vf, .g = vg, .h = vh }, \ .match = m, \ .not_visited = nvisit, \ } #define YES 1 #define NO 0 static struct test_filter_data_t { char *filter; struct trace_event_raw_ftrace_test_filter rec; int match; char *not_visited; } test_filter_data[] = { #define FILTER "a == 1 && b == 1 && c == 1 && d == 1 && " \ "e == 1 && f == 1 && g == 1 && h == 1" DATA_REC(YES, 1, 1, 1, 1, 1, 1, 1, 1, ""), DATA_REC(NO, 0, 1, 1, 1, 1, 1, 1, 1, "bcdefgh"), DATA_REC(NO, 1, 1, 1, 1, 1, 1, 1, 0, ""), #undef FILTER #define FILTER "a == 1 || b == 1 || c == 1 || d == 1 || " \ "e == 1 || f == 1 || g == 1 || h == 1" DATA_REC(NO, 0, 0, 0, 0, 0, 0, 0, 0, ""), DATA_REC(YES, 0, 0, 0, 0, 0, 0, 0, 1, ""), DATA_REC(YES, 1, 0, 0, 0, 0, 0, 0, 0, "bcdefgh"), #undef FILTER #define FILTER "(a == 1 || b == 1) && (c == 1 || d == 1) && " \ "(e == 1 || f == 1) && (g == 1 || h == 1)" DATA_REC(NO, 0, 0, 1, 1, 1, 1, 1, 1, "dfh"), DATA_REC(YES, 0, 1, 0, 1, 0, 1, 0, 1, ""), DATA_REC(YES, 1, 0, 1, 0, 0, 1, 0, 1, "bd"), DATA_REC(NO, 1, 0, 1, 0, 0, 1, 0, 0, "bd"), #undef FILTER #define FILTER "(a == 1 && b == 1) || (c == 1 && d == 1) || " \ "(e == 1 && f == 1) || (g == 1 && h == 1)" DATA_REC(YES, 1, 0, 1, 1, 1, 1, 1, 1, "efgh"), DATA_REC(YES, 0, 0, 0, 0, 0, 0, 1, 1, ""), DATA_REC(NO, 0, 0, 0, 0, 0, 0, 0, 1, ""), #undef FILTER #define FILTER "(a == 1 && b == 1) && (c == 1 && d == 1) && " \ "(e == 1 && f == 1) || (g == 1 && h == 1)" DATA_REC(YES, 1, 1, 1, 1, 1, 1, 0, 0, "gh"), DATA_REC(NO, 0, 0, 0, 0, 0, 0, 0, 1, ""), DATA_REC(YES, 1, 1, 1, 1, 1, 0, 1, 1, ""), #undef FILTER #define FILTER "((a == 1 || b == 1) || (c == 1 || d == 1) || " \ "(e == 1 || f == 1)) && (g == 1 || h == 1)" DATA_REC(YES, 1, 1, 1, 1, 1, 1, 0, 1, "bcdef"), DATA_REC(NO, 0, 0, 0, 0, 0, 0, 0, 0, ""), DATA_REC(YES, 1, 1, 1, 1, 1, 0, 1, 1, "h"), #undef FILTER #define FILTER "((((((((a == 1) && (b == 1)) || (c == 1)) && (d == 1)) || " \ "(e == 1)) && (f == 1)) || (g == 1)) && (h == 1))" DATA_REC(YES, 1, 1, 1, 1, 1, 1, 1, 1, "ceg"), DATA_REC(NO, 0, 1, 0, 1, 0, 1, 0, 1, ""), DATA_REC(NO, 1, 0, 1, 0, 1, 0, 1, 0, ""), #undef FILTER #define FILTER "((((((((a == 1) || (b == 1)) && (c == 1)) || (d == 1)) && " \ "(e == 1)) || (f == 1)) && (g == 1)) || (h == 1))" DATA_REC(YES, 1, 1, 1, 1, 1, 1, 1, 1, "bdfh"), DATA_REC(YES, 0, 1, 0, 1, 0, 1, 0, 1, ""), DATA_REC(YES, 1, 0, 1, 0, 1, 0, 1, 0, "bdfh"), }; #undef DATA_REC #undef FILTER #undef YES #undef NO #define DATA_CNT (sizeof(test_filter_data)/sizeof(struct test_filter_data_t)) static int test_pred_visited;
static int test_pred_visited_fn(struct filter_pred *pred, void *event) { struct ftrace_event_field *field = pred->field; test_pred_visited = 1; printk(KERN_INFO "\npred visited %s\n", field->name); return 1; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa41100.00%1100.00%
Total41100.00%1100.00%


static int test_walk_pred_cb(enum move_type move, struct filter_pred *pred, int *err, void *data) { char *fields = data; if ((move == MOVE_DOWN) && (pred->left == FILTER_PRED_INVALID)) { struct ftrace_event_field *field = pred->field; if (!field) { WARN(1, "all leafs should have field defined"); return WALK_PRED_DEFAULT; } if (!strchr(fields, *field->name)) return WALK_PRED_DEFAULT; WARN_ON(!pred->fn); pred->fn = test_pred_visited_fn; } return WALK_PRED_DEFAULT; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa106100.00%1100.00%
Total106100.00%1100.00%


static __init int ftrace_test_event_filter(void) { int i; printk(KERN_INFO "Testing ftrace filter: "); for (i = 0; i < DATA_CNT; i++) { struct event_filter *filter = NULL; struct test_filter_data_t *d = &test_filter_data[i]; int err; err = create_filter(&event_ftrace_test_filter, d->filter, false, &filter); if (err) { printk(KERN_INFO "Failed to get filter for '%s', err %d\n", d->filter, err); __free_filter(filter); break; } /* * The preemption disabling is not really needed for self * tests, but the rcu dereference will complain without it. */ preempt_disable(); if (*d->not_visited) walk_pred_tree(filter->preds, filter->root, test_walk_pred_cb, d->not_visited); test_pred_visited = 0; err = filter_match_preds(filter, &d->rec); preempt_enable(); __free_filter(filter); if (test_pred_visited) { printk(KERN_INFO "Failed, unwanted pred visited for filter %s\n", d->filter); break; } if (err != d->match) { printk(KERN_INFO "Failed to match filter '%s', expected %d\n", d->filter, d->match); break; } } if (i == DATA_CNT) printk(KERN_CONT "OK\n"); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Jiri Olsa18791.67%133.33%
Tejun Heo104.90%133.33%
Steven Rostedt73.43%133.33%
Total204100.00%3100.00%

late_initcall(ftrace_test_event_filter); #endif /* CONFIG_FTRACE_STARTUP_TEST */

Overall Contributors

PersonTokensPropCommitsCommitProp
Tom Zanussi359035.19%1110.58%
Jiri Olsa262325.71%1312.50%
Steven Rostedt209120.50%3432.69%
Frédéric Weisbecker5725.61%43.85%
Li Zefan5545.43%2524.04%
Tejun Heo4254.17%32.88%
Daniel Wagner1891.85%10.96%
Masami Hiramatsu840.82%10.96%
Oleg Nesterov320.31%21.92%
Ingo Molnar150.15%21.92%
Thomas Meyer110.11%10.96%
Yaowei Bai40.04%10.96%
Rasmus Villemoes30.03%10.96%
Jovi Zhangwei30.03%21.92%
Xiao Guangrong30.03%10.96%
Daniel Walter20.02%10.96%
Michal Hocko10.01%10.96%
Total10202100.00%104100.00%
Directory: kernel/trace
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.