diff options
author | Steven Rostedt <srostedt@redhat.com> | 2010-02-16 22:55:59 -0500 |
---|---|---|
committer | Steven Rostedt <rostedt@goodmis.org> | 2010-02-16 22:55:59 -0500 |
commit | 40abb355ed1a82514f698c1ae6df4b74ba0b4011 (patch) | |
tree | 416fb52d59e543c32315fabd0971ce96f69c4561 | |
parent | ca296f1d6481a9051e3499d42c5975c9781b86e7 (diff) |
parse-events: Add regex to string matches for filter
Add =~ and !~ for filtering string matches.
Also fixed the == and != string matching since it was using
strncmp() due to values may not being NULL terminated. Copies
the data instead and compares the copy.
Signed-off-by: Steven Rostedt <rostedt@goodmis.org>
-rw-r--r-- | parse-events.c | 10 | ||||
-rw-r--r-- | parse-events.h | 8 | ||||
-rw-r--r-- | parse-filter.c | 71 |
3 files changed, 83 insertions, 6 deletions
diff --git a/parse-events.c b/parse-events.c index 679c0c9..7e0336b 100644 --- a/parse-events.c +++ b/parse-events.c | |||
@@ -696,6 +696,16 @@ static int __peek_char(void) | |||
696 | return input_buf[input_buf_ptr]; | 696 | return input_buf[input_buf_ptr]; |
697 | } | 697 | } |
698 | 698 | ||
699 | /** | ||
700 | * pevent_peek_char - peek at the next character that will be read | ||
701 | * | ||
702 | * Returns the next character read, or -1 if end of buffer. | ||
703 | */ | ||
704 | int pevent_peek_char(void) | ||
705 | { | ||
706 | return __peek_char(); | ||
707 | } | ||
708 | |||
699 | static enum event_type force_token(const char *str, char **tok); | 709 | static enum event_type force_token(const char *str, char **tok); |
700 | 710 | ||
701 | static enum event_type __read_token(char **tok) | 711 | static enum event_type __read_token(char **tok) |
diff --git a/parse-events.h b/parse-events.h index 13b1f3a..1573869 100644 --- a/parse-events.h +++ b/parse-events.h | |||
@@ -22,6 +22,7 @@ | |||
22 | #define _PARSE_EVENTS_H | 22 | #define _PARSE_EVENTS_H |
23 | 23 | ||
24 | #include <stdarg.h> | 24 | #include <stdarg.h> |
25 | #include <regex.h> | ||
25 | 26 | ||
26 | #ifndef __unused | 27 | #ifndef __unused |
27 | #define __unused __attribute__ ((unused)) | 28 | #define __unused __attribute__ ((unused)) |
@@ -485,6 +486,7 @@ void pevent_unref(struct pevent *pevent); | |||
485 | void pevent_buffer_init(const char *buf, unsigned long long size); | 486 | void pevent_buffer_init(const char *buf, unsigned long long size); |
486 | enum event_type pevent_read_token(char **tok); | 487 | enum event_type pevent_read_token(char **tok); |
487 | void pevent_free_token(char *token); | 488 | void pevent_free_token(char *token); |
489 | int pevent_peek_char(void); | ||
488 | 490 | ||
489 | /* for debugging */ | 491 | /* for debugging */ |
490 | void pevent_print_funcs(struct pevent *pevent); | 492 | void pevent_print_funcs(struct pevent *pevent); |
@@ -513,8 +515,8 @@ enum filter_cmp_type { | |||
513 | FILTER_CMP_LE, | 515 | FILTER_CMP_LE, |
514 | FILTER_CMP_MATCH, | 516 | FILTER_CMP_MATCH, |
515 | FILTER_CMP_NOT_MATCH, | 517 | FILTER_CMP_NOT_MATCH, |
516 | FILTER_CMP_CONTAIN, | 518 | FILTER_CMP_REGEX, |
517 | FILTER_CMP_NOT_CONTAIN, | 519 | FILTER_CMP_NOT_REGEX, |
518 | }; | 520 | }; |
519 | 521 | ||
520 | enum filter_arg_type { | 522 | enum filter_arg_type { |
@@ -547,6 +549,8 @@ struct filter_arg_str { | |||
547 | enum filter_cmp_type type; | 549 | enum filter_cmp_type type; |
548 | struct format_field *field; | 550 | struct format_field *field; |
549 | char *val; | 551 | char *val; |
552 | char *buffer; | ||
553 | regex_t reg; | ||
550 | }; | 554 | }; |
551 | 555 | ||
552 | struct filter_arg { | 556 | struct filter_arg { |
diff --git a/parse-filter.c b/parse-filter.c index 3e47b2a..8944644 100644 --- a/parse-filter.c +++ b/parse-filter.c | |||
@@ -23,7 +23,6 @@ | |||
23 | #include <string.h> | 23 | #include <string.h> |
24 | #include <stdarg.h> | 24 | #include <stdarg.h> |
25 | #include <ctype.h> | 25 | #include <ctype.h> |
26 | #include <regex.h> | ||
27 | #include <errno.h> | 26 | #include <errno.h> |
28 | #include <sys/types.h> | 27 | #include <sys/types.h> |
29 | 28 | ||
@@ -89,7 +88,20 @@ static enum event_type read_token(char **tok) | |||
89 | type = pevent_read_token(&token); | 88 | type = pevent_read_token(&token); |
90 | } while (type == EVENT_NEWLINE || type == EVENT_SPACE); | 89 | } while (type == EVENT_NEWLINE || type == EVENT_SPACE); |
91 | 90 | ||
92 | *tok = token; | 91 | /* If token is = or ! check to see if the next char is ~ */ |
92 | if (token && | ||
93 | (strcmp(token, "=") == 0 || strcmp(token, "!") == 0) && | ||
94 | pevent_peek_char() == '~') { | ||
95 | /* append it */ | ||
96 | *tok = malloc(3); | ||
97 | sprintf(*tok, "%c%c", *token, '~'); | ||
98 | free_token(token); | ||
99 | /* Now remove the '~' from the buffer */ | ||
100 | pevent_read_token(&token); | ||
101 | free_token(token); | ||
102 | } else | ||
103 | *tok = token; | ||
104 | |||
93 | return type; | 105 | return type; |
94 | } | 106 | } |
95 | 107 | ||
@@ -205,6 +217,8 @@ static void free_arg(struct filter_arg *arg) | |||
205 | 217 | ||
206 | case FILTER_ARG_STR: | 218 | case FILTER_ARG_STR: |
207 | free(arg->str.val); | 219 | free(arg->str.val); |
220 | regfree(&arg->str.reg); | ||
221 | free(arg->str.buffer); | ||
208 | break; | 222 | break; |
209 | 223 | ||
210 | case FILTER_ARG_OP: | 224 | case FILTER_ARG_OP: |
@@ -314,6 +328,8 @@ static int process_valid_field(struct filter_arg *arg, | |||
314 | char *val, | 328 | char *val, |
315 | char **error_str) | 329 | char **error_str) |
316 | { | 330 | { |
331 | int ret; | ||
332 | |||
317 | switch (type) { | 333 | switch (type) { |
318 | 334 | ||
319 | case EVENT_SQUOTE: | 335 | case EVENT_SQUOTE: |
@@ -332,6 +348,13 @@ static int process_valid_field(struct filter_arg *arg, | |||
332 | op_type = FILTER_CMP_NOT_MATCH; | 348 | op_type = FILTER_CMP_NOT_MATCH; |
333 | break; | 349 | break; |
334 | 350 | ||
351 | case FILTER_CMP_REGEX: | ||
352 | case FILTER_CMP_NOT_REGEX: | ||
353 | ret = regcomp(&arg->str.reg, val, REG_ICASE|REG_NOSUB); | ||
354 | if (ret) | ||
355 | return -1; | ||
356 | break; | ||
357 | |||
335 | default: | 358 | default: |
336 | show_error(error_str, | 359 | show_error(error_str, |
337 | "Op not allowed with string"); | 360 | "Op not allowed with string"); |
@@ -343,9 +366,24 @@ static int process_valid_field(struct filter_arg *arg, | |||
343 | arg->str.val = strdup(val); | 366 | arg->str.val = strdup(val); |
344 | if (!arg->str.val) | 367 | if (!arg->str.val) |
345 | die("Can't allocate arg value"); | 368 | die("Can't allocate arg value"); |
369 | |||
370 | /* Need a buffer to copy data int for tests */ | ||
371 | arg->str.buffer = malloc_or_die(arg->str.field->size + 1); | ||
372 | /* Null terminate this buffer */ | ||
373 | arg->str.buffer[arg->str.field->size] = 0; | ||
374 | |||
346 | break; | 375 | break; |
347 | case EVENT_ITEM: | 376 | case EVENT_ITEM: |
348 | as_int: | 377 | as_int: |
378 | switch (op_type) { | ||
379 | case FILTER_CMP_REGEX: | ||
380 | case FILTER_CMP_NOT_REGEX: | ||
381 | show_error(error_str, | ||
382 | "Op not allowed with integers"); | ||
383 | return -1; | ||
384 | default: | ||
385 | break; | ||
386 | } | ||
349 | arg->type = FILTER_ARG_NUM; | 387 | arg->type = FILTER_ARG_NUM; |
350 | arg->num.field = field; | 388 | arg->num.field = field; |
351 | arg->num.type = op_type; | 389 | arg->num.type = op_type; |
@@ -477,6 +515,10 @@ process_filter(struct event_format *event, struct filter_arg **parg, | |||
477 | etype = FILTER_CMP_LE; | 515 | etype = FILTER_CMP_LE; |
478 | } else if (strcmp(token, ">=") == 0) { | 516 | } else if (strcmp(token, ">=") == 0) { |
479 | etype = FILTER_CMP_GE; | 517 | etype = FILTER_CMP_GE; |
518 | } else if (strcmp(token, "=~") == 0) { | ||
519 | etype = FILTER_CMP_REGEX; | ||
520 | } else if (strcmp(token, "!~") == 0) { | ||
521 | etype = FILTER_CMP_NOT_REGEX; | ||
480 | } else { | 522 | } else { |
481 | show_error(error_str, | 523 | show_error(error_str, |
482 | "Unknown op '%s' after '%s'", | 524 | "Unknown op '%s' after '%s'", |
@@ -897,13 +939,34 @@ static int test_str(struct event_format *event, | |||
897 | struct filter_arg *arg, struct record *record) | 939 | struct filter_arg *arg, struct record *record) |
898 | { | 940 | { |
899 | const char *val = record->data + arg->str.field->offset; | 941 | const char *val = record->data + arg->str.field->offset; |
942 | const char *buffer; | ||
943 | |||
944 | /* | ||
945 | * We need to copy the data since we can't be sure the field | ||
946 | * is null terminated. | ||
947 | */ | ||
948 | if (*(val + arg->str.field->size - 1)) { | ||
949 | /* copy it */ | ||
950 | memcpy(arg->str.buffer, val, arg->str.field->size); | ||
951 | /* the buffer is already NULL terminated */ | ||
952 | buffer = arg->str.buffer; | ||
953 | } else | ||
954 | /* OK, it's NULL terminated */ | ||
955 | buffer = val; | ||
900 | 956 | ||
901 | switch (arg->str.type) { | 957 | switch (arg->str.type) { |
902 | case FILTER_CMP_MATCH: | 958 | case FILTER_CMP_MATCH: |
903 | return strncmp(val, arg->str.val, arg->str.field->size) == 0; | 959 | return strcmp(buffer, arg->str.val) == 0; |
904 | 960 | ||
905 | case FILTER_CMP_NOT_MATCH: | 961 | case FILTER_CMP_NOT_MATCH: |
906 | return strncmp(val, arg->str.val, arg->str.field->size) != 0; | 962 | return strcmp(buffer, arg->str.val) != 0; |
963 | |||
964 | case FILTER_CMP_REGEX: | ||
965 | /* Returns zero on match */ | ||
966 | return !regexec(&arg->str.reg, buffer, 0, NULL, 0); | ||
967 | |||
968 | case FILTER_CMP_NOT_REGEX: | ||
969 | return regexec(&arg->str.reg, buffer, 0, NULL, 0); | ||
907 | 970 | ||
908 | default: | 971 | default: |
909 | /* ?? */ | 972 | /* ?? */ |