2de045936f
* basic_filters.c (lookup_class, qualify_syscall_class, qualify_syscall, qualify_syscall_tokens, parse_syscall_filter, qualify_tokens, parse_fd_filter, parse_path_filter): Add qualify_mode argument. (qualify_tokens, qualify_syscall_tokens): Use set inversion only in qualify mode. (lookup_class): Use deprecated class names only in qualify mode. * defs.h (qualify): Remove declaration. (filtering_parse): Add new declaration. * filter.c (struct filter_type, parse_filter): Add bool argument to parse_*_filter declarations. * filter.h (parse_filter_action, parse_qualify_action, parse_filter_expression): Add new declarations. (parse_filter, qualify_tokens, qualify_syscall_tokens): Add qualify_mode argument. (DECL_FILTER): Add bool argument to parse_*_filter declarations. * filter_action.c (parse_filter_action): Add new parsing function. (inject_path_tracing): Use filtering_parse instead of qualify. * filter_expression.c (parse_filter_expression): Implement parsing of filter expression. (parse_operator, push_operator, is_higher_priority): Add helper functions. (is_space_ascii, is_allowed_in_name): Add new declarations. * filter_parse.c: New file. * filter_qualify.c (qualify_read, qualify_write, qualify_signals, qualify_trace, qualify_abbrev, qualify_verbose, qualify_raw, qualify_inject_common, qualify_fault, qualify_inject): Use main_part and args arguments. * strace.c (init): Use filtering_parse instead of qualify. * Makefile.am (strace_SOURCES): Add filter_parse.c.
253 lines
5.9 KiB
C
253 lines
5.9 KiB
C
/*
|
|
* Copyright (c) 2017 Nikolay Marchuk <marchuk.nikolay.a@gmail.com>
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. The name of the author may not be used to endorse or promote products
|
|
* derived from this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#include "defs.h"
|
|
#include "filter.h"
|
|
|
|
bool
|
|
is_space_ascii(char c)
|
|
{
|
|
return (c == ' ') || (c == '\t') || (c == '\n') || (c == '\r') ||
|
|
(c == '\v') || (c == '\f');
|
|
}
|
|
|
|
bool
|
|
is_allowed_in_name(char c)
|
|
{
|
|
return (c >= 'A' && c <= 'Z') || (c >= 'a' && c <= 'z')
|
|
|| (c >= '0' && c <= '9') || (c == '_');
|
|
}
|
|
|
|
/*
|
|
* Split expression into action name, filter expression or qualify set
|
|
* and action arguments.
|
|
*/
|
|
void
|
|
filtering_parse(const char *str)
|
|
{
|
|
enum parsing_states {
|
|
F_EMPTY,
|
|
F_BEGIN,
|
|
F_QUAL_SET,
|
|
F_FILT_EXPR,
|
|
F_QUAL_ARGS,
|
|
F_FILT_ARGS,
|
|
F_END,
|
|
} state = F_EMPTY;
|
|
const char *begin = NULL;
|
|
const char *action_name = "trace";
|
|
const char *main_part = NULL;
|
|
const char *args = NULL;
|
|
bool action_specified = false;
|
|
bool escaped = false;
|
|
int parentheses_count = 0;
|
|
/* Used to store position of last terminating parenthesis. */
|
|
char *expression_end = NULL;
|
|
/* Used to provide diagnostics. */
|
|
unsigned int pos = 0;
|
|
char *buf = xstrdup(str);
|
|
char *p;
|
|
|
|
for (p = buf; *p; ++p, ++pos) {
|
|
switch (state) {
|
|
case F_EMPTY:
|
|
switch (*p) {
|
|
/* trace(), action name omitted */
|
|
case '(':
|
|
parentheses_count++;
|
|
main_part = p;
|
|
state = F_FILT_EXPR;
|
|
break;
|
|
/* missing action name */
|
|
case '=':
|
|
*p = '\0';
|
|
error_msg_and_die("invalid filter action '%s'",
|
|
buf);
|
|
default:
|
|
if (is_space_ascii(*p)) {
|
|
break;
|
|
} else if (!strncmp(p, "not", 3) && *(p + 3) &&
|
|
(is_space_ascii(*(p + 3)) ||
|
|
*(p + 3) == '(')) {
|
|
main_part = p;
|
|
state = F_FILT_EXPR;
|
|
break;
|
|
} else {
|
|
begin = p;
|
|
state = F_BEGIN;
|
|
}
|
|
}
|
|
if (state != F_BEGIN)
|
|
break;
|
|
/* else fall through to check for qualify set */
|
|
|
|
case F_BEGIN:
|
|
switch (*p) {
|
|
/* action(...) */
|
|
case '(':
|
|
if (*begin == '!') {
|
|
main_part = begin;
|
|
} else {
|
|
action_name = begin;
|
|
action_specified = true;
|
|
*p = '\0';
|
|
main_part = p + 1;
|
|
}
|
|
state = F_FILT_EXPR;
|
|
parentheses_count++;
|
|
break;
|
|
/* action=... */
|
|
case '=':
|
|
action_name = begin;
|
|
action_specified = true;
|
|
*p = '\0';
|
|
main_part = p + 1;
|
|
state = F_QUAL_SET;
|
|
break;
|
|
case ':':
|
|
main_part = begin;
|
|
*p = '\0';
|
|
args = p + 1;
|
|
state = F_QUAL_ARGS;
|
|
break;
|
|
case ';':
|
|
error_msg_and_die("invalid arguments position "
|
|
"'%s':%u",
|
|
str, pos);
|
|
/* qualify set without action. */
|
|
case ',':
|
|
case '?':
|
|
case '/':
|
|
case '%':
|
|
case '-':
|
|
main_part = begin;
|
|
state = F_QUAL_SET;
|
|
break;
|
|
default:
|
|
/* new expression without action. */
|
|
if (is_space_ascii(*p)) {
|
|
main_part = begin;
|
|
state = F_FILT_EXPR;
|
|
}
|
|
}
|
|
break;
|
|
|
|
case F_QUAL_SET:
|
|
if (*p == ':') {
|
|
*p = '\0';
|
|
args = p + 1;
|
|
state = F_QUAL_ARGS;
|
|
}
|
|
break;
|
|
|
|
case F_FILT_EXPR:
|
|
if (!escaped) {
|
|
switch (*p) {
|
|
case ';':
|
|
if (parentheses_count != 1 ||
|
|
!action_specified)
|
|
error_msg_and_die("invalid "
|
|
"arguments "
|
|
"position "
|
|
"'%s':%u",
|
|
str, pos);
|
|
*p = '\0';
|
|
args = p + 1;
|
|
state = F_FILT_ARGS;
|
|
break;
|
|
case '(':
|
|
parentheses_count++;
|
|
break;
|
|
case ')':
|
|
if (parentheses_count <= 0)
|
|
error_msg_and_die("unexpected "
|
|
"')' at "
|
|
"'%s':%u",
|
|
str, pos);
|
|
parentheses_count--;
|
|
expression_end = p;
|
|
if (action_specified &&
|
|
parentheses_count == 0)
|
|
state = F_END;
|
|
break;
|
|
case '\\':
|
|
escaped = true;
|
|
break;
|
|
}
|
|
} else
|
|
escaped = false;
|
|
break;
|
|
|
|
case F_QUAL_ARGS:
|
|
break;
|
|
case F_FILT_ARGS:
|
|
if (!escaped) {
|
|
switch (*p) {
|
|
case ')':
|
|
parentheses_count--;
|
|
expression_end = p;
|
|
state = F_END;
|
|
break;
|
|
case '\\':
|
|
escaped = true;
|
|
break;
|
|
}
|
|
} else
|
|
escaped = false;
|
|
break;
|
|
case F_END:
|
|
if (!is_space_ascii(*p))
|
|
error_msg_and_die("unexpected '%c' at "
|
|
"'%s':%u", *p, str, pos);
|
|
}
|
|
}
|
|
|
|
switch (state) {
|
|
case F_EMPTY:
|
|
main_part = buf;
|
|
parse_qualify_action(action_name, main_part, args);
|
|
break;
|
|
case F_BEGIN:
|
|
main_part = begin;
|
|
/* fall through */
|
|
case F_QUAL_SET:
|
|
case F_QUAL_ARGS:
|
|
parse_qualify_action(action_name, main_part, args);
|
|
break;
|
|
case F_FILT_EXPR:
|
|
case F_FILT_ARGS:
|
|
case F_END:
|
|
if (parentheses_count != 0)
|
|
error_msg_and_die("missing ')' in '%s'", str);
|
|
if (action_specified && expression_end)
|
|
*expression_end = '\0';
|
|
parse_filter_action(action_name, main_part, args);
|
|
break;
|
|
}
|
|
free(buf);
|
|
}
|