aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorSteven Rostedt <srostedt@redhat.com>2010-02-16 22:55:59 -0500
committerSteven Rostedt <rostedt@goodmis.org>2010-02-16 22:55:59 -0500
commit40abb355ed1a82514f698c1ae6df4b74ba0b4011 (patch)
tree416fb52d59e543c32315fabd0971ce96f69c4561
parentca296f1d6481a9051e3499d42c5975c9781b86e7 (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.c10
-rw-r--r--parse-events.h8
-rw-r--r--parse-filter.c71
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 */
704int pevent_peek_char(void)
705{
706 return __peek_char();
707}
708
699static enum event_type force_token(const char *str, char **tok); 709static enum event_type force_token(const char *str, char **tok);
700 710
701static enum event_type __read_token(char **tok) 711static 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);
485void pevent_buffer_init(const char *buf, unsigned long long size); 486void pevent_buffer_init(const char *buf, unsigned long long size);
486enum event_type pevent_read_token(char **tok); 487enum event_type pevent_read_token(char **tok);
487void pevent_free_token(char *token); 488void pevent_free_token(char *token);
489int pevent_peek_char(void);
488 490
489/* for debugging */ 491/* for debugging */
490void pevent_print_funcs(struct pevent *pevent); 492void 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
520enum filter_arg_type { 522enum 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
552struct filter_arg { 556struct 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 /* ?? */