2 * Copyright (C) 2014 Oracle.
4 * This program is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU General Public License
6 * as published by the Free Software Foundation; either version 2
7 * of the License, or (at your option) any later version.
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with this program; if not, see http://www.gnu.org/copyleft/gpl.txt
19 * Sometimes we aren't able to track a variable through a function call. This
20 * usually happens because a function changes too many variables so we give up.
21 * Another reason this happens is because we call a function pointer and there
22 * are too many functions which implement that function pointer so we give up.
23 * Also maybe we don't have the database enabled.
25 * The goal here is to make a call back so what if we call:
29 * but we're not able to say what happens to "foo", then let's assume that we
30 * don't know anything about "foo" if it's an untracked call.
35 #include "smatch_slist.h"
36 #include "smatch_extra.h"
43 typedef void (untracked_hook
)(struct expression
*call
, int param
);
44 DECLARE_PTR_LIST(untracked_hook_list
, untracked_hook
*);
45 static struct untracked_hook_list
*untracked_hooks
;
47 DECLARE_PTR_LIST(int_stack
, int);
48 struct int_stack
*tracked_stack
;
50 static void push_int(struct int_stack
**stack
, int num
)
55 * Just put the int on directly instead of a pointer to the int.
56 * Shift it to the left because Sparse uses the last two bits.
57 * This is sort of a dirty hack, yes.
60 munged
= INT_PTR(tracked
<< 2);
62 add_ptr_list(stack
, munged
);
65 static int pop_int(struct int_stack
**stack
)
69 num
= last_ptr_list((struct ptr_list
*)*stack
);
70 delete_ptr_list_last((struct ptr_list
**)stack
);
72 return PTR_INT(num
) >> 2;
75 void add_untracked_param_hook(void (func
)(struct expression
*call
, int param
))
77 untracked_hook
**p
= malloc(sizeof(untracked_hook
*));
79 add_ptr_list(&untracked_hooks
, p
);
82 static void call_untracked_callbacks(struct expression
*expr
, int param
)
86 FOR_EACH_PTR(untracked_hooks
, fn
) {
88 } END_FOR_EACH_PTR(fn
);
91 static void assume_tracked(struct expression
*call_expr
, int param
, char *key
, char *value
)
96 static void mark_untracked(struct expression
*expr
, int param
, char *key
, char *value
)
101 while (expr
->type
== EXPR_ASSIGNMENT
)
102 expr
= strip_expr(expr
->right
);
103 if (expr
->type
!= EXPR_CALL
)
106 name
= return_state_to_var_sym(expr
, param
, key
, &sym
);
110 call_untracked_callbacks(expr
, param
);
111 set_state(my_id
, name
, sym
, &untracked
);
116 static int lost_in_va_args(struct expression
*expr
)
122 fn
= get_type(expr
->fn
);
123 if (!fn
|| !fn
->variadic
)
127 name
= expr_to_var(expr
->fn
);
128 if (strstr(name
, "print"))
135 static void match_after_call(struct expression
*expr
)
137 struct expression
*arg
;
141 if (lost_in_va_args(expr
))
150 FOR_EACH_PTR(expr
->args
, arg
) {
153 type
= get_type(arg
);
154 if (!type
|| type
->type
!= SYM_PTR
)
157 call_untracked_callbacks(expr
, i
);
158 set_state_expr(my_id
, arg
, &untracked
);
159 } END_FOR_EACH_PTR(arg
);
162 static void print_untracked_params(int return_id
, char *return_ranges
, struct expression
*expr
)
168 FOR_EACH_PTR(cur_func_sym
->ctype
.base_type
->arguments
, arg
) {
173 if (!get_state(my_id
, arg
->ident
->name
, arg
))
176 sql_insert_return_states(return_id
, return_ranges
,
177 UNTRACKED_PARAM
, param
, "$", "");
178 } END_FOR_EACH_PTR(arg
);
181 static void match_param_assign(struct expression
*expr
)
183 struct expression
*right
;
187 if (__in_fake_assign
)
190 right
= strip_expr(expr
->right
);
191 type
= get_type(right
);
192 if (!type
|| type
->type
!= SYM_PTR
)
195 param
= get_param_num(right
);
199 set_state_expr(my_id
, right
, &untracked
);
203 static void match_param_assign_in_asm(struct statement
*stmt
)
206 struct expression
*expr
;
211 FOR_EACH_PTR(stmt
->asm_inputs
, expr
) {
213 case 0: /* identifier */
214 case 1: /* constraint */
217 case 2: /* expression */
220 expr
= strip_expr(expr
);
221 type
= get_type(expr
);
222 if (!type
|| type
->type
!= SYM_PTR
)
224 param
= get_param_num(expr
);
227 set_state_expr(my_id
, expr
, &untracked
);
230 } END_FOR_EACH_PTR(expr
);
233 static void match_inline_start(struct expression
*expr
)
235 push_int(&tracked_stack
, tracked
);
238 static void match_inline_end(struct expression
*expr
)
240 tracked
= pop_int(&tracked_stack
);
243 void register_untracked_param(int id
)
247 select_return_states_hook(INTERNAL
, &assume_tracked
);
248 select_return_states_hook(UNTRACKED_PARAM
, &mark_untracked
);
249 add_hook(&match_after_call
, FUNCTION_CALL_HOOK_AFTER_DB
);
251 add_split_return_callback(&print_untracked_params
);
253 add_hook(&match_param_assign
, ASSIGNMENT_HOOK
);
254 add_hook(&match_param_assign_in_asm
, ASM_HOOK
);
256 add_hook(&match_inline_start
, INLINE_FN_START
);
257 add_hook(&match_inline_end
, INLINE_FN_END
);