2 * sparse/smatch_extra.c
4 * Copyright (C) 2008 Dan Carpenter.
6 * Licensed under the Open Software License version 1.1
15 #include "smatch_slist.h"
16 #include "smatch_extra.h"
20 struct data_range whole_range
= {
25 static struct smatch_state
*alloc_extra_state_no_name(int val
)
27 struct smatch_state
*state
;
29 state
= __alloc_smatch_state(0);
31 state
->data
= (void *)alloc_dinfo_range(whole_range
.min
, whole_range
.max
);
33 state
->data
= (void *)alloc_dinfo_range(val
, val
);
37 /* We do this because ->value_ranges is a list */
38 struct smatch_state
*extra_undefined()
40 static struct data_info
*dinfo
;
41 static struct smatch_state
*ret
;
43 dinfo
= alloc_dinfo_range(whole_range
.min
, whole_range
.max
);
44 ret
= __alloc_smatch_state(0);
45 ret
->name
= "unknown";
50 struct smatch_state
*alloc_extra_state(int val
)
52 struct smatch_state
*state
;
55 return extra_undefined();
56 state
= alloc_extra_state_no_name(val
);
57 state
->name
= show_ranges(((struct data_info
*)state
->data
)->value_ranges
);
61 struct smatch_state
*filter_ranges(struct smatch_state
*orig
,
62 long long filter_min
, long long filter_max
)
64 struct smatch_state
*ret
;
65 struct data_info
*orig_info
;
66 struct data_info
*ret_info
;
69 orig
= extra_undefined();
70 orig_info
= (struct data_info
*)orig
->data
;
71 ret
= alloc_extra_state_no_name(UNDEFINED
);
72 ret_info
= (struct data_info
*)ret
->data
;
73 ret_info
->value_ranges
= remove_range(orig_info
->value_ranges
, filter_min
, filter_max
);
74 ret
->name
= show_ranges(ret_info
->value_ranges
);
78 struct smatch_state
*add_filter(struct smatch_state
*orig
, long long num
)
80 return filter_ranges(orig
, num
, num
);
83 static struct smatch_state
*merge_func(const char *name
, struct symbol
*sym
,
84 struct smatch_state
*s1
,
85 struct smatch_state
*s2
)
87 struct data_info
*info1
= (struct data_info
*)s1
->data
;
88 struct data_info
*info2
= (struct data_info
*)s2
->data
;
89 struct data_info
*ret_info
;
90 struct smatch_state
*tmp
;
92 tmp
= alloc_extra_state_no_name(UNDEFINED
);
93 tmp
->name
= "extra_merged";
94 ret_info
= (struct data_info
*)tmp
->data
;
96 ret_info
->value_ranges
= range_list_union(info1
->value_ranges
, info2
->value_ranges
);
100 struct sm_state
*__extra_merge(struct sm_state
*one
, struct state_list
*slist1
,
101 struct sm_state
*two
, struct state_list
*slist2
)
103 struct data_info
*info1
;
104 struct data_info
*info2
;
106 if (!one
->state
->data
|| !two
->state
->data
) {
107 smatch_msg("internal error in smatch extra '%s = %s or %s'",
108 one
->name
, show_state(one
->state
),
109 show_state(two
->state
));
110 return alloc_state(one
->name
, one
->owner
, one
->sym
,
114 info1
= (struct data_info
*)one
->state
->data
;
115 info2
= (struct data_info
*)two
->state
->data
;
118 free_stack(&one
->my_pools
);
120 free_stack(&two
->my_pools
);
122 if (one
== two
&& !one
->my_pools
) {
123 add_pool(&one
->my_pools
, slist1
);
124 add_pool(&one
->my_pools
, slist2
);
127 add_pool(&one
->my_pools
, slist1
);
129 add_pool(&two
->my_pools
, slist2
);
132 add_pool(&one
->all_pools
, slist1
);
133 add_pool(&two
->all_pools
, slist2
);
134 return merge_sm_states(one
, two
);
137 struct sm_state
*__extra_and_merge(struct sm_state
*sm
,
138 struct state_list_stack
*stack
)
140 struct state_list
*slist
;
141 struct sm_state
*ret
= NULL
;
142 struct sm_state
*tmp
;
145 FOR_EACH_PTR(stack
, slist
) {
147 ret
= get_sm_state_slist(slist
, sm
->name
, sm
->owner
,
150 tmp
= get_sm_state_slist(slist
, sm
->name
, sm
->owner
,
152 ret
= merge_sm_states(ret
, tmp
);
154 } END_FOR_EACH_PTR(slist
);
156 smatch_msg("Internal error in __extra_and_merge");
159 ret
->my_pools
= stack
;
160 ret
->all_pools
= clone_stack(stack
);
164 static struct smatch_state
*unmatched_state(struct sm_state
*sm
)
166 return extra_undefined();
169 static void match_function_call(struct expression
*expr
)
171 struct expression
*tmp
;
176 FOR_EACH_PTR(expr
->args
, tmp
) {
177 if (tmp
->op
== '&') {
178 name
= get_variable_from_expr(tmp
->unop
, &sym
);
180 set_state(name
, my_id
, sym
, extra_undefined());
185 } END_FOR_EACH_PTR(tmp
);
188 static void match_assign(struct expression
*expr
)
190 struct expression
*left
;
194 left
= strip_expr(expr
->left
);
195 name
= get_variable_from_expr(left
, &sym
);
198 set_state(name
, my_id
, sym
, alloc_extra_state(get_value(expr
->right
)));
202 static void undef_expr(struct expression
*expr
)
207 name
= get_variable_from_expr(expr
->unop
, &sym
);
210 if (!get_state(name
, my_id
, sym
)) {
214 set_state(name
, my_id
, sym
, extra_undefined());
218 static void match_declarations(struct symbol
*sym
)
223 name
= sym
->ident
->name
;
224 if (sym
->initializer
) {
225 set_state(name
, my_id
, sym
, alloc_extra_state(get_value(sym
->initializer
)));
227 set_state(name
, my_id
, sym
, extra_undefined());
232 static void match_function_def(struct symbol
*sym
)
236 FOR_EACH_PTR(sym
->ctype
.base_type
->arguments
, arg
) {
240 set_state(arg
->ident
->name
, my_id
, arg
, extra_undefined());
241 } END_FOR_EACH_PTR(arg
);
244 static void match_unop(struct expression
*expr
)
251 name
= get_variable_from_expr(expr
->unop
, &sym
);
255 tmp
= show_special(expr
->op
);
256 if ((!strcmp(tmp
, "--")) || (!strcmp(tmp
, "++")))
257 set_state(name
, my_id
, sym
, extra_undefined());
261 int get_implied_value(struct expression
*expr
)
263 struct smatch_state
*state
;
268 val
= get_value(expr
);
269 if (val
!= UNDEFINED
)
272 name
= get_variable_from_expr(expr
, &sym
);
275 state
= get_state(name
, my_id
, sym
);
277 if (!state
|| !state
->data
)
279 return get_single_value_from_range((struct data_info
*)state
->data
);
282 int true_comparison(int left
, int comparison
, int right
)
286 case SPECIAL_UNSIGNED_LT
:
290 case SPECIAL_UNSIGNED_LTE
:
298 case SPECIAL_UNSIGNED_GTE
:
303 case SPECIAL_UNSIGNED_GT
:
307 case SPECIAL_NOTEQUAL
:
312 smatch_msg("unhandled comparison %d\n", comparison
);
318 int true_comparison_range(struct data_range
*left
, int comparison
, struct data_range
*right
)
322 case SPECIAL_UNSIGNED_LT
:
323 if (left
->min
< right
->max
)
326 case SPECIAL_UNSIGNED_LTE
:
328 if (left
->min
<= right
->max
)
332 if (left
->max
< right
->min
)
334 if (left
->min
> right
->max
)
337 case SPECIAL_UNSIGNED_GTE
:
339 if (left
->max
>= right
->min
)
343 case SPECIAL_UNSIGNED_GT
:
344 if (left
->max
> right
->min
)
347 case SPECIAL_NOTEQUAL
:
348 if (left
->min
!= left
->max
)
350 if (right
->min
!= right
->max
)
352 if (left
->min
!= right
->min
)
356 smatch_msg("unhandled comparison %d\n", comparison
);
362 int false_comparison_range(struct data_range
*left
, int comparison
, struct data_range
*right
)
366 case SPECIAL_UNSIGNED_LT
:
367 if (left
->max
>= right
->min
)
370 case SPECIAL_UNSIGNED_LTE
:
372 if (left
->max
> right
->min
)
376 if (left
->min
!= left
->max
)
378 if (right
->min
!= right
->max
)
380 if (left
->min
!= right
->min
)
383 case SPECIAL_UNSIGNED_GTE
:
385 if (left
->min
< right
->max
)
389 case SPECIAL_UNSIGNED_GT
:
390 if (left
->min
>= right
->max
)
393 case SPECIAL_NOTEQUAL
:
394 if (left
->max
< right
->min
)
396 if (left
->min
> right
->max
)
400 smatch_msg("unhandled comparison %d\n", comparison
);
406 static int do_comparison(struct expression
*expr
)
408 int left
, right
, ret
;
410 if ((left
= get_implied_value(expr
->left
)) == UNDEFINED
)
413 if ((right
= get_implied_value(expr
->right
)) == UNDEFINED
)
416 ret
= true_comparison(left
, expr
->op
, right
);
418 SM_DEBUG("%d known condition: %d %s %d => true\n",
419 get_lineno(), left
, show_special(expr
->op
), right
);
420 } else if (ret
== 0) {
421 SM_DEBUG("%d known condition: %d %s %d => false\n",
422 get_lineno(), left
, show_special(expr
->op
), right
);
427 int last_stmt_val(struct statement
*stmt
)
429 struct expression
*expr
;
431 stmt
= last_ptr_list((struct ptr_list
*)stmt
->stmts
);
432 if (stmt
->type
!= STMT_EXPRESSION
)
434 expr
= stmt
->expression
;
435 return get_value(expr
);
438 static void match_comparison(struct expression
*expr
)
443 struct smatch_state
*eq_state
;
444 struct smatch_state
*neq_state
;
446 if (expr
->op
!= SPECIAL_EQUAL
&& expr
->op
!= SPECIAL_NOTEQUAL
)
448 value
= get_value(expr
->left
);
449 if (value
!= UNDEFINED
) {
450 name
= get_variable_from_expr(expr
->right
, &sym
);
452 value
= get_value(expr
->right
);
453 name
= get_variable_from_expr(expr
->left
, &sym
);
455 if (value
== UNDEFINED
|| !name
|| !sym
)
457 eq_state
= alloc_extra_state(value
);
458 neq_state
= add_filter(extra_undefined(), value
);
459 if (expr
->op
== SPECIAL_EQUAL
)
460 set_true_false_states(name
, my_id
, sym
, eq_state
, neq_state
);
462 set_true_false_states(name
, my_id
, sym
, neq_state
, eq_state
);
467 /* this is actually hooked from smatch_implied.c... it's hacky, yes */
468 void __extra_match_condition(struct expression
*expr
)
472 struct smatch_state
*pre_state
;
473 struct smatch_state
*true_state
;
474 struct smatch_state
*false_state
;
476 expr
= strip_expr(expr
);
481 name
= get_variable_from_expr(expr
, &sym
);
484 pre_state
= get_state(name
, my_id
, sym
);
485 true_state
= add_filter(pre_state
, 0);
486 false_state
= alloc_extra_state(0);
487 set_true_false_states(name
, my_id
, sym
, true_state
, false_state
);
491 match_comparison(expr
);
496 static int variable_non_zero(struct expression
*expr
)
500 struct smatch_state
*state
;
503 name
= get_variable_from_expr(expr
, &sym
);
506 state
= get_state(name
, my_id
, sym
);
507 if (!state
|| !state
->data
)
509 ret
= true_comparison(get_single_value_from_range((struct data_info
*)state
->data
),
510 SPECIAL_NOTEQUAL
, 0);
516 int known_condition_true(struct expression
*expr
)
523 tmp
= get_value(expr
);
524 if (tmp
&& tmp
!= UNDEFINED
)
527 expr
= strip_expr(expr
);
530 if (expr
->op
== '!') {
531 if (known_condition_false(expr
->unop
))
542 int known_condition_false(struct expression
*expr
)
552 if (expr
->op
== '!') {
553 if (known_condition_true(expr
->unop
))
564 int implied_condition_true(struct expression
*expr
)
566 struct statement
*stmt
;
572 tmp
= get_value(expr
);
573 if (tmp
&& tmp
!= UNDEFINED
)
576 expr
= strip_expr(expr
);
579 if (do_comparison(expr
) == 1)
583 if (expr
->op
== '!') {
584 if (implied_condition_false(expr
->unop
))
588 stmt
= get_block_thing(expr
);
589 if (stmt
&& (last_stmt_val(stmt
) == 1))
593 if (variable_non_zero(expr
) == 1)
600 int implied_condition_false(struct expression
*expr
)
602 struct statement
*stmt
;
603 struct expression
*tmp
;
613 if (do_comparison(expr
) == 0)
616 if (expr
->op
== '!') {
617 if (implied_condition_true(expr
->unop
))
621 stmt
= get_block_thing(expr
);
622 if (stmt
&& (last_stmt_val(stmt
) == 0))
624 tmp
= strip_expr(expr
);
626 return implied_condition_false(tmp
);
629 if (variable_non_zero(expr
) == 0)
636 void register_smatch_extra(int id
)
639 add_merge_hook(my_id
, &merge_func
);
640 add_unmatched_state_hook(my_id
, &unmatched_state
);
641 add_hook(&undef_expr
, OP_HOOK
);
642 add_hook(&match_function_def
, FUNC_DEF_HOOK
);
643 add_hook(&match_function_call
, FUNCTION_CALL_HOOK
);
644 add_hook(&match_assign
, ASSIGNMENT_HOOK
);
645 add_hook(&match_declarations
, DECLARATION_HOOK
);
646 add_hook(&match_unop
, OP_HOOK
);
647 add_hook(&free_data_info_allocs
, END_FUNC_HOOK
);
650 /* I don't know how to test for the ATTRIB_NORET attribute. :( */
651 add_function_hook("panic", &__match_nullify_path_hook
, NULL
);
652 add_function_hook("do_exit", &__match_nullify_path_hook
, NULL
);
653 add_function_hook("complete_and_exit", &__match_nullify_path_hook
, NULL
);
654 add_function_hook("__module_put_and_exit", &__match_nullify_path_hook
, NULL
);
655 add_function_hook("do_group_exit", &__match_nullify_path_hook
, NULL
);