2001-02-01 20:31:21 +00:00
|
|
|
/*
|
2004-07-18 00:24:25 +00:00
|
|
|
* $Id$
|
2001-02-01 20:31:21 +00:00
|
|
|
*
|
2006-05-21 05:12:17 +00:00
|
|
|
* Wireshark - Network traffic analyzer
|
|
|
|
* By Gerald Combs <gerald@wireshark.org>
|
2001-02-01 20:31:21 +00:00
|
|
|
* Copyright 2001 Gerald Combs
|
|
|
|
*
|
|
|
|
* 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.
|
2002-08-28 20:41:00 +00:00
|
|
|
*
|
2001-02-01 20:31:21 +00:00
|
|
|
* 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.
|
2002-08-28 20:41:00 +00:00
|
|
|
*
|
2001-02-01 20:31:21 +00:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2001-02-01 20:21:25 +00:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "dfvm.h"
|
|
|
|
|
|
|
|
dfvm_insn_t*
|
|
|
|
dfvm_insn_new(dfvm_opcode_t op)
|
|
|
|
{
|
|
|
|
dfvm_insn_t *insn;
|
|
|
|
|
|
|
|
insn = g_new(dfvm_insn_t, 1);
|
|
|
|
insn->op = op;
|
|
|
|
insn->arg1 = NULL;
|
|
|
|
insn->arg2 = NULL;
|
|
|
|
insn->arg3 = NULL;
|
2007-11-28 22:44:37 +00:00
|
|
|
insn->arg4 = NULL;
|
2001-02-01 20:21:25 +00:00
|
|
|
return insn;
|
|
|
|
}
|
|
|
|
|
2002-04-08 20:11:31 +00:00
|
|
|
static void
|
|
|
|
dfvm_value_free(dfvm_value_t *v)
|
|
|
|
{
|
|
|
|
switch (v->type) {
|
|
|
|
case FVALUE:
|
2006-12-22 09:01:12 +00:00
|
|
|
FVALUE_FREE(v->value.fvalue);
|
2002-04-08 20:11:31 +00:00
|
|
|
break;
|
|
|
|
case DRANGE:
|
|
|
|
drange_free(v->value.drange);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* nothing */
|
|
|
|
;
|
|
|
|
}
|
|
|
|
g_free(v);
|
|
|
|
}
|
|
|
|
|
2001-02-01 20:21:25 +00:00
|
|
|
void
|
|
|
|
dfvm_insn_free(dfvm_insn_t *insn)
|
|
|
|
{
|
|
|
|
if (insn->arg1) {
|
|
|
|
dfvm_value_free(insn->arg1);
|
|
|
|
}
|
|
|
|
if (insn->arg2) {
|
|
|
|
dfvm_value_free(insn->arg2);
|
|
|
|
}
|
|
|
|
if (insn->arg3) {
|
|
|
|
dfvm_value_free(insn->arg3);
|
|
|
|
}
|
2007-11-28 22:44:37 +00:00
|
|
|
if (insn->arg4) {
|
|
|
|
dfvm_value_free(insn->arg4);
|
|
|
|
}
|
2001-02-01 20:21:25 +00:00
|
|
|
g_free(insn);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
dfvm_value_t*
|
|
|
|
dfvm_value_new(dfvm_value_type_t type)
|
|
|
|
{
|
|
|
|
dfvm_value_t *v;
|
|
|
|
|
|
|
|
v = g_new(dfvm_value_t, 1);
|
|
|
|
v->type = type;
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
dfvm_dump(FILE *f, GPtrArray *insns)
|
|
|
|
{
|
|
|
|
int id, length;
|
|
|
|
dfvm_insn_t *insn;
|
|
|
|
dfvm_value_t *arg1;
|
|
|
|
dfvm_value_t *arg2;
|
|
|
|
dfvm_value_t *arg3;
|
|
|
|
dfvm_value_t *arg4;
|
2005-10-31 02:42:22 +00:00
|
|
|
char *value_str;
|
2005-10-31 03:33:14 +00:00
|
|
|
GSList *range_list;
|
|
|
|
drange_node *range_item;
|
2001-02-01 20:21:25 +00:00
|
|
|
|
|
|
|
length = insns->len;
|
|
|
|
|
|
|
|
for (id = 0; id < length; id++) {
|
|
|
|
|
|
|
|
insn = g_ptr_array_index(insns, id);
|
|
|
|
arg1 = insn->arg1;
|
|
|
|
arg2 = insn->arg2;
|
|
|
|
arg3 = insn->arg3;
|
|
|
|
arg4 = insn->arg4;
|
|
|
|
|
|
|
|
switch (insn->op) {
|
|
|
|
case CHECK_EXISTS:
|
|
|
|
fprintf(f, "%05d CHECK_EXISTS\t%s\n",
|
2002-10-16 16:32:59 +00:00
|
|
|
id, arg1->value.hfinfo->abbrev);
|
2001-02-01 20:21:25 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case READ_TREE:
|
2005-10-31 02:42:22 +00:00
|
|
|
fprintf(f, "%05d READ_TREE\t\t%s -> reg#%u\n",
|
2002-10-16 16:32:59 +00:00
|
|
|
id, arg1->value.hfinfo->abbrev,
|
2001-02-01 20:21:25 +00:00
|
|
|
arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
2009-01-21 05:26:40 +00:00
|
|
|
case CALL_FUNCTION:
|
|
|
|
fprintf(f, "%05d CALL_FUNCTION\t%s (",
|
|
|
|
id, arg1->value.funcdef->name);
|
|
|
|
if (arg3) {
|
|
|
|
fprintf(f, "reg#%u", arg3->value.numeric);
|
|
|
|
}
|
|
|
|
if (arg4) {
|
|
|
|
fprintf(f, ", reg#%u", arg4->value.numeric);
|
|
|
|
}
|
|
|
|
fprintf(f, ") --> reg#%u\n", arg2->value.numeric);
|
|
|
|
break;
|
2006-05-02 14:26:17 +00:00
|
|
|
|
2001-02-01 20:21:25 +00:00
|
|
|
case PUT_FVALUE:
|
2005-10-31 02:42:22 +00:00
|
|
|
value_str = fvalue_to_string_repr(arg1->value.fvalue,
|
|
|
|
FTREPR_DFILTER, NULL);
|
|
|
|
fprintf(f, "%05d PUT_FVALUE\t%s <%s> -> reg#%u\n",
|
|
|
|
id, value_str,
|
|
|
|
fvalue_type_name(arg1->value.fvalue),
|
2001-02-01 20:21:25 +00:00
|
|
|
arg2->value.numeric);
|
2005-10-31 02:42:22 +00:00
|
|
|
g_free(value_str);
|
2001-02-01 20:21:25 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case MK_RANGE:
|
2005-10-31 03:33:14 +00:00
|
|
|
arg3 = insn->arg3;
|
|
|
|
fprintf(f, "%05d MK_RANGE\t\treg#%u[",
|
2001-02-01 20:21:25 +00:00
|
|
|
id,
|
2005-10-31 03:33:14 +00:00
|
|
|
arg1->value.numeric);
|
|
|
|
for (range_list = arg3->value.drange->range_list;
|
|
|
|
range_list != NULL;
|
|
|
|
range_list = range_list->next) {
|
|
|
|
range_item = range_list->data;
|
|
|
|
switch (range_item->ending) {
|
|
|
|
|
2010-01-19 19:37:03 +00:00
|
|
|
case DRANGE_NODE_END_T_UNINITIALIZED:
|
2005-10-31 03:33:14 +00:00
|
|
|
fprintf(f, "?");
|
|
|
|
break;
|
|
|
|
|
2010-01-19 19:37:03 +00:00
|
|
|
case DRANGE_NODE_END_T_LENGTH:
|
2005-10-31 03:33:14 +00:00
|
|
|
fprintf(f, "%d:%d",
|
|
|
|
range_item->start_offset,
|
|
|
|
range_item->length);
|
|
|
|
break;
|
|
|
|
|
2010-01-19 19:37:03 +00:00
|
|
|
case DRANGE_NODE_END_T_OFFSET:
|
2005-10-31 03:33:14 +00:00
|
|
|
fprintf(f, "%d-%d",
|
|
|
|
range_item->start_offset,
|
|
|
|
range_item->end_offset);
|
|
|
|
break;
|
|
|
|
|
2010-01-19 19:37:03 +00:00
|
|
|
case DRANGE_NODE_END_T_TO_THE_END:
|
2005-10-31 03:33:14 +00:00
|
|
|
fprintf(f, "%d:",
|
|
|
|
range_item->start_offset);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (range_list->next != NULL)
|
|
|
|
fprintf(f, ",");
|
|
|
|
}
|
|
|
|
fprintf(f, "] -> reg#%u\n",
|
2001-02-01 20:21:25 +00:00
|
|
|
arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ANY_EQ:
|
2005-10-31 02:42:22 +00:00
|
|
|
fprintf(f, "%05d ANY_EQ\t\treg#%u == reg#%u\n",
|
2001-02-01 20:21:25 +00:00
|
|
|
id, arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ANY_NE:
|
2005-10-31 02:42:22 +00:00
|
|
|
fprintf(f, "%05d ANY_NE\t\treg#%u == reg#%u\n",
|
2001-02-01 20:21:25 +00:00
|
|
|
id, arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ANY_GT:
|
2005-10-31 02:42:22 +00:00
|
|
|
fprintf(f, "%05d ANY_GT\t\treg#%u == reg#%u\n",
|
2001-02-01 20:21:25 +00:00
|
|
|
id, arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ANY_GE:
|
2005-10-31 02:42:22 +00:00
|
|
|
fprintf(f, "%05d ANY_GE\t\treg#%u == reg#%u\n",
|
2001-02-01 20:21:25 +00:00
|
|
|
id, arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ANY_LT:
|
2005-10-31 02:42:22 +00:00
|
|
|
fprintf(f, "%05d ANY_LT\t\treg#%u == reg#%u\n",
|
2001-02-01 20:21:25 +00:00
|
|
|
id, arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ANY_LE:
|
2005-10-31 02:42:22 +00:00
|
|
|
fprintf(f, "%05d ANY_LE\t\treg#%u == reg#%u\n",
|
2001-02-01 20:21:25 +00:00
|
|
|
id, arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
2004-02-27 12:00:32 +00:00
|
|
|
case ANY_BITWISE_AND:
|
2005-10-31 02:42:22 +00:00
|
|
|
fprintf(f, "%05d ANY_BITWISE_AND\t\treg#%u == reg#%u\n",
|
2004-02-27 12:00:32 +00:00
|
|
|
id, arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
2003-12-19 04:40:24 +00:00
|
|
|
case ANY_CONTAINS:
|
2005-10-31 02:42:22 +00:00
|
|
|
fprintf(f, "%05d ANY_CONTAINS\treg#%u contains reg#%u\n",
|
2003-12-19 04:40:24 +00:00
|
|
|
id, arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ANY_MATCHES:
|
2005-10-31 02:42:22 +00:00
|
|
|
fprintf(f, "%05d ANY_MATCHES\treg#%u matches reg#%u\n",
|
2003-12-19 04:40:24 +00:00
|
|
|
id, arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
2001-02-01 20:21:25 +00:00
|
|
|
case NOT:
|
|
|
|
fprintf(f, "%05d NOT\n", id);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RETURN:
|
|
|
|
fprintf(f, "%05d RETURN\n", id);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case IF_TRUE_GOTO:
|
|
|
|
fprintf(f, "%05d IF-TRUE-GOTO\t%d\n",
|
|
|
|
id, arg1->value.numeric);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case IF_FALSE_GOTO:
|
|
|
|
fprintf(f, "%05d IF-FALSE-GOTO\t%d\n",
|
|
|
|
id, arg1->value.numeric);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
g_assert_not_reached();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Reads a field from the proto_tree and loads the fvalues into a register,
|
|
|
|
* if that field has not already been read. */
|
|
|
|
static gboolean
|
2002-10-16 16:32:59 +00:00
|
|
|
read_tree(dfilter_t *df, proto_tree *tree, header_field_info *hfinfo, int reg)
|
2001-02-01 20:21:25 +00:00
|
|
|
{
|
|
|
|
GPtrArray *finfos;
|
|
|
|
field_info *finfo;
|
|
|
|
int i, len;
|
|
|
|
GList *fvalues = NULL;
|
2002-10-16 16:32:59 +00:00
|
|
|
gboolean found_something = FALSE;
|
2001-02-01 20:21:25 +00:00
|
|
|
|
|
|
|
/* Already loaded in this run of the dfilter? */
|
|
|
|
if (df->attempted_load[reg]) {
|
|
|
|
if (df->registers[reg]) {
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
df->attempted_load[reg] = TRUE;
|
|
|
|
|
2002-10-16 16:32:59 +00:00
|
|
|
while (hfinfo) {
|
|
|
|
finfos = proto_get_finfo_ptr_array(tree, hfinfo->id);
|
|
|
|
if (!finfos) {
|
|
|
|
hfinfo = hfinfo->same_name_next;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else if (g_ptr_array_len(finfos) == 0) {
|
|
|
|
hfinfo = hfinfo->same_name_next;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
found_something = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
len = finfos->len;
|
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
finfo = g_ptr_array_index(finfos, i);
|
2003-12-02 21:15:49 +00:00
|
|
|
fvalues = g_list_prepend(fvalues, &finfo->value);
|
2002-10-16 16:32:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
hfinfo = hfinfo->same_name_next;
|
2001-02-01 20:21:25 +00:00
|
|
|
}
|
2002-10-16 16:32:59 +00:00
|
|
|
|
|
|
|
if (!found_something) {
|
|
|
|
return FALSE;
|
2001-02-01 20:21:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
df->registers[reg] = fvalues;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
put_fvalue(dfilter_t *df, fvalue_t *fv, int reg)
|
|
|
|
{
|
|
|
|
df->registers[reg] = g_list_append(NULL, fv);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef gboolean (*FvalueCmpFunc)(fvalue_t*, fvalue_t*);
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
any_test(dfilter_t *df, FvalueCmpFunc cmp, int reg1, int reg2)
|
|
|
|
{
|
|
|
|
GList *list_a, *list_b;
|
|
|
|
|
|
|
|
list_a = df->registers[reg1];
|
|
|
|
|
|
|
|
while (list_a) {
|
|
|
|
list_b = df->registers[reg2];
|
|
|
|
while (list_b) {
|
|
|
|
if (cmp(list_a->data, list_b->data)) {
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
list_b = g_list_next(list_b);
|
|
|
|
}
|
|
|
|
list_a = g_list_next(list_a);
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Free the list nodes w/o freeing the memory that each
|
|
|
|
* list node points to. */
|
|
|
|
static void
|
|
|
|
free_register_overhead(dfilter_t* df)
|
|
|
|
{
|
2009-10-18 23:25:33 +00:00
|
|
|
guint i;
|
2001-02-01 20:21:25 +00:00
|
|
|
|
|
|
|
for (i = 0; i < df->num_registers; i++) {
|
2007-11-28 22:13:50 +00:00
|
|
|
df->attempted_load[i] = FALSE;
|
2001-02-01 20:21:25 +00:00
|
|
|
if (df->registers[i]) {
|
|
|
|
g_list_free(df->registers[i]);
|
2007-11-28 22:13:50 +00:00
|
|
|
df->registers[i] = NULL;
|
2001-02-01 20:21:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Takes the list of fvalue_t's in a register, uses fvalue_slice()
|
|
|
|
* to make a new list of fvalue_t's (which are ranges, or byte-slices),
|
|
|
|
* and puts the new list into a new register. */
|
|
|
|
static void
|
2008-06-23 20:24:52 +00:00
|
|
|
mk_range(dfilter_t *df, int from_reg, int to_reg, drange *d_range)
|
2001-02-01 20:21:25 +00:00
|
|
|
{
|
|
|
|
GList *from_list, *to_list;
|
|
|
|
fvalue_t *old_fv, *new_fv;
|
|
|
|
|
|
|
|
to_list = NULL;
|
|
|
|
from_list = df->registers[from_reg];
|
|
|
|
|
|
|
|
while (from_list) {
|
|
|
|
old_fv = from_list->data;
|
2008-06-23 20:24:52 +00:00
|
|
|
new_fv = fvalue_slice(old_fv, d_range);
|
2001-02-27 19:23:30 +00:00
|
|
|
/* Assert here because semcheck.c should have
|
2001-02-01 20:21:25 +00:00
|
|
|
* already caught the cases in which a slice
|
|
|
|
* cannot be made. */
|
|
|
|
g_assert(new_fv);
|
|
|
|
to_list = g_list_append(to_list, new_fv);
|
|
|
|
|
|
|
|
from_list = g_list_next(from_list);
|
|
|
|
}
|
|
|
|
|
|
|
|
df->registers[to_reg] = to_list;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
gboolean
|
2002-05-09 23:50:34 +00:00
|
|
|
dfvm_apply(dfilter_t *df, proto_tree *tree)
|
2001-02-01 20:21:25 +00:00
|
|
|
{
|
2007-11-28 22:13:50 +00:00
|
|
|
int id, length;
|
2001-02-01 20:21:25 +00:00
|
|
|
gboolean accum = TRUE;
|
|
|
|
dfvm_insn_t *insn;
|
|
|
|
dfvm_value_t *arg1;
|
|
|
|
dfvm_value_t *arg2;
|
2006-05-02 14:26:17 +00:00
|
|
|
dfvm_value_t *arg3 = NULL;
|
|
|
|
dfvm_value_t *arg4 = NULL;
|
2002-10-16 16:32:59 +00:00
|
|
|
header_field_info *hfinfo;
|
2009-01-21 05:26:40 +00:00
|
|
|
GList *param1;
|
|
|
|
GList *param2;
|
2001-02-01 20:21:25 +00:00
|
|
|
|
|
|
|
g_assert(tree);
|
|
|
|
|
|
|
|
length = df->insns->len;
|
|
|
|
|
|
|
|
for (id = 0; id < length; id++) {
|
|
|
|
|
|
|
|
AGAIN:
|
|
|
|
insn = g_ptr_array_index(df->insns, id);
|
|
|
|
arg1 = insn->arg1;
|
|
|
|
arg2 = insn->arg2;
|
|
|
|
|
|
|
|
switch (insn->op) {
|
|
|
|
case CHECK_EXISTS:
|
2002-10-16 16:32:59 +00:00
|
|
|
hfinfo = arg1->value.hfinfo;
|
|
|
|
while(hfinfo) {
|
|
|
|
accum = proto_check_for_protocol_or_field(tree,
|
2007-04-10 18:31:36 +00:00
|
|
|
hfinfo->id);
|
2002-10-16 16:32:59 +00:00
|
|
|
if (accum) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
hfinfo = hfinfo->same_name_next;
|
|
|
|
}
|
|
|
|
}
|
2001-02-01 20:21:25 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case READ_TREE:
|
|
|
|
accum = read_tree(df, tree,
|
2002-10-16 16:32:59 +00:00
|
|
|
arg1->value.hfinfo, arg2->value.numeric);
|
2001-02-01 20:21:25 +00:00
|
|
|
break;
|
|
|
|
|
2006-05-02 14:26:17 +00:00
|
|
|
case CALL_FUNCTION:
|
|
|
|
arg3 = insn->arg3;
|
|
|
|
arg4 = insn->arg4;
|
2009-01-21 05:26:40 +00:00
|
|
|
param1 = NULL;
|
|
|
|
param2 = NULL;
|
|
|
|
if (arg3) {
|
|
|
|
param1 = df->registers[arg3->value.numeric];
|
|
|
|
}
|
|
|
|
if (arg4) {
|
|
|
|
param2 = df->registers[arg4->value.numeric];
|
|
|
|
}
|
|
|
|
accum = arg1->value.funcdef->function(param1, param2,
|
|
|
|
&df->registers[arg2->value.numeric]);
|
2006-05-02 14:26:17 +00:00
|
|
|
break;
|
|
|
|
|
2001-02-01 20:21:25 +00:00
|
|
|
case MK_RANGE:
|
|
|
|
arg3 = insn->arg3;
|
|
|
|
mk_range(df,
|
|
|
|
arg1->value.numeric, arg2->value.numeric,
|
2001-02-27 19:23:30 +00:00
|
|
|
arg3->value.drange);
|
2001-02-01 20:21:25 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ANY_EQ:
|
|
|
|
accum = any_test(df, fvalue_eq,
|
|
|
|
arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ANY_NE:
|
|
|
|
accum = any_test(df, fvalue_ne,
|
|
|
|
arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ANY_GT:
|
|
|
|
accum = any_test(df, fvalue_gt,
|
|
|
|
arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ANY_GE:
|
|
|
|
accum = any_test(df, fvalue_ge,
|
|
|
|
arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ANY_LT:
|
|
|
|
accum = any_test(df, fvalue_lt,
|
|
|
|
arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ANY_LE:
|
|
|
|
accum = any_test(df, fvalue_le,
|
|
|
|
arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
2004-02-27 12:00:32 +00:00
|
|
|
case ANY_BITWISE_AND:
|
|
|
|
accum = any_test(df, fvalue_bitwise_and,
|
|
|
|
arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
2003-08-27 15:23:11 +00:00
|
|
|
case ANY_CONTAINS:
|
|
|
|
accum = any_test(df, fvalue_contains,
|
|
|
|
arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
2003-12-06 16:35:20 +00:00
|
|
|
case ANY_MATCHES:
|
|
|
|
accum = any_test(df, fvalue_matches,
|
|
|
|
arg1->value.numeric, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
|
2001-02-01 20:21:25 +00:00
|
|
|
case NOT:
|
|
|
|
accum = !accum;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RETURN:
|
|
|
|
free_register_overhead(df);
|
|
|
|
return accum;
|
|
|
|
|
|
|
|
case IF_TRUE_GOTO:
|
|
|
|
if (accum) {
|
|
|
|
id = arg1->value.numeric;
|
|
|
|
goto AGAIN;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case IF_FALSE_GOTO:
|
|
|
|
if (!accum) {
|
|
|
|
id = arg1->value.numeric;
|
|
|
|
goto AGAIN;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2007-11-28 22:44:37 +00:00
|
|
|
case PUT_FVALUE:
|
|
|
|
#if 0
|
|
|
|
accum = put_fvalue(df,
|
|
|
|
arg1->value.fvalue, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
#endif
|
2001-02-01 20:21:25 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
g_assert_not_reached();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
g_assert_not_reached();
|
|
|
|
return FALSE; /* to appease the compiler */
|
|
|
|
}
|
2007-11-28 22:44:37 +00:00
|
|
|
|
|
|
|
void
|
|
|
|
dfvm_init_const(dfilter_t *df)
|
|
|
|
{
|
|
|
|
int id, length;
|
|
|
|
dfvm_insn_t *insn;
|
|
|
|
dfvm_value_t *arg1;
|
|
|
|
dfvm_value_t *arg2;
|
|
|
|
|
|
|
|
length = df->consts->len;
|
|
|
|
|
|
|
|
for (id = 0; id < length; id++) {
|
|
|
|
|
|
|
|
insn = g_ptr_array_index(df->consts, id);
|
|
|
|
arg1 = insn->arg1;
|
|
|
|
arg2 = insn->arg2;
|
|
|
|
|
|
|
|
switch (insn->op) {
|
|
|
|
case PUT_FVALUE:
|
|
|
|
put_fvalue(df,
|
|
|
|
arg1->value.fvalue, arg2->value.numeric);
|
|
|
|
break;
|
|
|
|
case CHECK_EXISTS:
|
|
|
|
case READ_TREE:
|
|
|
|
case CALL_FUNCTION:
|
|
|
|
case MK_RANGE:
|
|
|
|
case ANY_EQ:
|
|
|
|
case ANY_NE:
|
|
|
|
case ANY_GT:
|
|
|
|
case ANY_GE:
|
|
|
|
case ANY_LT:
|
|
|
|
case ANY_LE:
|
|
|
|
case ANY_BITWISE_AND:
|
|
|
|
case ANY_CONTAINS:
|
|
|
|
case ANY_MATCHES:
|
|
|
|
case NOT:
|
|
|
|
case RETURN:
|
|
|
|
case IF_TRUE_GOTO:
|
|
|
|
case IF_FALSE_GOTO:
|
|
|
|
default:
|
|
|
|
g_assert_not_reached();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|