2 * sparse/smatch_extra.c
4 * Copyright (C) 2008 Dan Carpenter.
6 * Licensed under the Open Software License version 1.1
13 #include "smatch_slist.h"
14 #include "smatch_extra.h"
18 struct data_info unknown_num
= {
24 static struct smatch_state extra_undefined
= {
29 static struct smatch_state
*alloc_extra_state_no_name(int val
)
31 struct smatch_state
*state
;
33 state
= __alloc_smatch_state(0);
34 state
->data
= (void *)alloc_data_info(val
);
38 static struct smatch_state
*alloc_extra_state(int val
)
40 struct smatch_state
*state
;
44 return &extra_undefined
;
46 state
= alloc_extra_state_no_name(val
);
47 snprintf(name
, 20, "%d", val
);
48 state
->name
= alloc_string(name
);
52 static struct smatch_state
*merge_func(const char *name
, struct symbol
*sym
,
53 struct smatch_state
*s1
,
54 struct smatch_state
*s2
)
56 struct data_info
*info1
= (struct data_info
*)s1
->data
;
57 struct data_info
*info2
= (struct data_info
*)s2
->data
;
58 struct smatch_state
*tmp
;
60 tmp
= alloc_extra_state_no_name(UNDEFINED
);
61 tmp
->name
= "extra_merged";
62 ((struct data_info
*)tmp
->data
)->merged
= 1;
63 ((struct data_info
*)tmp
->data
)->values
=
64 num_list_union(info1
->values
, info2
->values
);
68 struct sm_state
*__extra_merge(struct sm_state
*one
, struct state_list
*slist1
,
69 struct sm_state
*two
, struct state_list
*slist2
)
71 struct data_info
*info1
;
72 struct data_info
*info2
;
74 if (!one
->state
->data
|| !two
->state
->data
) {
75 smatch_msg("internal error in smatch extra '%s = %s or %s'",
76 one
->name
, show_state(one
->state
),
77 show_state(two
->state
));
78 return alloc_state(one
->name
, one
->owner
, one
->sym
,
82 info1
= (struct data_info
*)one
->state
->data
;
83 info2
= (struct data_info
*)two
->state
->data
;
86 free_stack(&one
->my_pools
);
88 free_stack(&two
->my_pools
);
90 if (one
== two
&& !one
->my_pools
) {
91 add_pool(&one
->my_pools
, slist1
);
92 add_pool(&one
->my_pools
, slist2
);
95 add_pool(&one
->my_pools
, slist1
);
97 add_pool(&two
->my_pools
, slist2
);
100 add_pool(&one
->all_pools
, slist1
);
101 add_pool(&two
->all_pools
, slist2
);
102 return merge_sm_states(one
, two
);
105 struct sm_state
*__extra_and_merge(struct sm_state
*sm
,
106 struct state_list_stack
*stack
)
108 struct state_list
*slist
;
109 struct sm_state
*ret
= NULL
;
110 struct sm_state
*tmp
;
113 FOR_EACH_PTR(stack
, slist
) {
115 ret
= get_sm_state_slist(slist
, sm
->name
, sm
->owner
,
118 tmp
= get_sm_state_slist(slist
, sm
->name
, sm
->owner
,
120 ret
= merge_sm_states(ret
, tmp
);
122 } END_FOR_EACH_PTR(slist
);
124 smatch_msg("Internal error in __extra_and_merge");
127 ret
->my_pools
= stack
;
128 ret
->all_pools
= clone_stack(stack
);
132 static struct smatch_state
*unmatched_state(struct sm_state
*sm
)
134 return &extra_undefined
;
137 static void match_function_call(struct expression
*expr
)
139 struct expression
*tmp
;
144 FOR_EACH_PTR(expr
->args
, tmp
) {
145 if (tmp
->op
== '&') {
146 name
= get_variable_from_expr(tmp
->unop
, &sym
);
148 set_state(name
, my_id
, sym
, &extra_undefined
);
153 } END_FOR_EACH_PTR(tmp
);
156 static void match_assign(struct expression
*expr
)
158 struct expression
*left
;
162 left
= strip_expr(expr
->left
);
163 name
= get_variable_from_expr(left
, &sym
);
166 set_state(name
, my_id
, sym
, alloc_extra_state(get_value(expr
->right
)));
170 static void undef_expr(struct expression
*expr
)
175 name
= get_variable_from_expr(expr
->unop
, &sym
);
178 if (!get_state(name
, my_id
, sym
)) {
182 set_state(name
, my_id
, sym
, &extra_undefined
);
186 static void match_declarations(struct symbol
*sym
)
191 name
= sym
->ident
->name
;
192 if (sym
->initializer
) {
193 set_state(name
, my_id
, sym
, alloc_extra_state(get_value(sym
->initializer
)));
195 set_state(name
, my_id
, sym
, &extra_undefined
);
200 static void match_function_def(struct symbol
*sym
)
204 FOR_EACH_PTR(sym
->ctype
.base_type
->arguments
, arg
) {
208 set_state(arg
->ident
->name
, my_id
, arg
, &extra_undefined
);
209 } END_FOR_EACH_PTR(arg
);
212 static void match_unop(struct expression
*expr
)
219 name
= get_variable_from_expr(expr
->unop
, &sym
);
223 tmp
= show_special(expr
->op
);
224 if ((!strcmp(tmp
, "--")) || (!strcmp(tmp
, "++")))
225 set_state(name
, my_id
, sym
, &extra_undefined
);
229 static int expr_to_val(struct expression
*expr
)
231 struct smatch_state
*state
;
236 val
= get_value(expr
);
237 if (val
!= UNDEFINED
)
240 name
= get_variable_from_expr(expr
, &sym
);
243 state
= get_state(name
, my_id
, sym
);
245 if (!state
|| !state
->data
)
247 return get_single_value((struct data_info
*)state
->data
);
250 int true_comparison(int left
, int comparison
, int right
)
254 case SPECIAL_UNSIGNED_LT
:
258 case SPECIAL_UNSIGNED_LTE
:
266 case SPECIAL_UNSIGNED_GTE
:
271 case SPECIAL_UNSIGNED_GT
:
275 case SPECIAL_NOTEQUAL
:
280 smatch_msg("unhandled comparison %d\n", comparison
);
286 static int do_comparison(struct expression
*expr
)
288 int left
, right
, ret
;
290 if ((left
= expr_to_val(expr
->left
)) == UNDEFINED
)
293 if ((right
= expr_to_val(expr
->right
)) == UNDEFINED
)
296 ret
= true_comparison(left
, expr
->op
, right
);
298 SM_DEBUG("%d known condition: %d %s %d => true\n",
299 get_lineno(), left
, show_special(expr
->op
), right
);
300 } else if (ret
== 0) {
301 SM_DEBUG("%d known condition: %d %s %d => false\n",
302 get_lineno(), left
, show_special(expr
->op
), right
);
307 int last_stmt_val(struct statement
*stmt
)
309 struct expression
*expr
;
311 stmt
= last_ptr_list((struct ptr_list
*)stmt
->stmts
);
312 if (stmt
->type
!= STMT_EXPRESSION
)
314 expr
= stmt
->expression
;
315 return get_value(expr
);
318 static int variable_non_zero(struct expression
*expr
)
322 struct smatch_state
*state
;
325 name
= get_variable_from_expr(expr
, &sym
);
328 state
= get_state(name
, my_id
, sym
);
329 if (!state
|| !state
->data
)
331 ret
= true_comparison(get_single_value((struct data_info
*)state
->data
),
332 SPECIAL_NOTEQUAL
, 0);
338 int known_condition_true(struct expression
*expr
)
340 struct statement
*stmt
;
346 tmp
= get_value(expr
);
347 if (tmp
&& tmp
!= UNDEFINED
)
350 expr
= strip_expr(expr
);
353 if (do_comparison(expr
) == 1)
357 if (expr
->op
== '!') {
358 if (known_condition_false(expr
->unop
))
362 stmt
= get_block_thing(expr
);
363 if (stmt
&& (last_stmt_val(stmt
) == 1))
367 if (variable_non_zero(expr
) == 1)
374 int known_condition_false(struct expression
*expr
)
376 struct statement
*stmt
;
377 struct expression
*tmp
;
387 if (do_comparison(expr
) == 0)
390 if (expr
->op
== '!') {
391 if (known_condition_true(expr
->unop
))
395 stmt
= get_block_thing(expr
);
396 if (stmt
&& (last_stmt_val(stmt
) == 0))
398 tmp
= strip_expr(expr
);
400 return known_condition_false(tmp
);
403 if (variable_non_zero(expr
) == 0)
410 void register_smatch_extra(int id
)
413 add_merge_hook(my_id
, &merge_func
);
414 add_unmatched_state_hook(my_id
, &unmatched_state
);
415 add_hook(&undef_expr
, OP_HOOK
);
416 add_hook(&match_function_def
, FUNC_DEF_HOOK
);
417 add_hook(&match_function_call
, FUNCTION_CALL_HOOK
);
418 add_hook(&match_assign
, ASSIGNMENT_HOOK
);
419 add_hook(&match_declarations
, DECLARATION_HOOK
);
420 add_hook(&match_unop
, OP_HOOK
);
422 /* I don't know how to test for the ATTRIB_NORET attribute. :( */
423 add_function_hook("panic", &__match_nullify_path_hook
, NULL
);
424 add_function_hook("do_exit", &__match_nullify_path_hook
, NULL
);
425 add_function_hook("complete_and_exit", &__match_nullify_path_hook
, NULL
);
426 add_function_hook("do_group_exit", &__match_nullify_path_hook
, NULL
);