db/fixup_kernel.sh: commit all my stuff
[smatch.git] / check_free_strict.c
blob3dcac4d79a5fe772da5cc5686dab487ddbc33bf0
1 /*
2 * Copyright (C) 2010 Dan Carpenter.
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 * check_memory() is getting too big and messy.
23 #include <string.h>
24 #include "smatch.h"
25 #include "smatch_slist.h"
26 #include "smatch_extra.h"
28 static int my_id;
30 STATE(freed);
31 STATE(ok);
33 static void ok_to_use(struct sm_state *sm, struct expression *mod_expr)
35 if (sm->state != &ok)
36 set_state(my_id, sm->name, sm->sym, &ok);
39 static void pre_merge_hook(struct sm_state *sm)
41 if (is_impossible_path())
42 set_state(my_id, sm->name, sm->sym, &ok);
45 static int is_freed(struct expression *expr)
47 struct sm_state *sm;
49 sm = get_sm_state_expr(my_id, expr);
50 if (sm && slist_has_state(sm->possible, &freed))
51 return 1;
52 return 0;
55 static void match_symbol(struct expression *expr)
57 struct expression *parent;
58 char *name;
60 if (is_impossible_path())
61 return;
63 parent = expr_get_parent_expr(expr);
64 while (parent && parent->type == EXPR_PREOP && parent->op == '(')
65 parent = expr_get_parent_expr(parent);
66 if (parent && parent->type == EXPR_PREOP && parent->op == '&')
67 return;
69 if (!is_freed(expr))
70 return;
71 name = expr_to_var(expr);
72 sm_msg("warn: '%s' was already freed.", name);
73 free_string(name);
76 static void match_dereferences(struct expression *expr)
78 char *name;
80 if (expr->type != EXPR_PREOP)
81 return;
83 if (is_impossible_path())
84 return;
86 expr = strip_expr(expr->unop);
87 if (!is_freed(expr))
88 return;
89 name = expr_to_var(expr);
90 sm_msg("error: dereferencing freed memory '%s'", name);
91 set_state_expr(my_id, expr, &ok);
92 free_string(name);
95 static int ignored_params[16];
97 static void set_ignored_params(struct expression *call)
99 struct expression *arg;
100 const char *p;
101 int i;
103 memset(&ignored_params, 0, sizeof(ignored_params));
105 i = -1;
106 FOR_EACH_PTR(call->args, arg) {
107 i++;
108 if (arg->type != EXPR_STRING)
109 continue;
110 goto found;
111 } END_FOR_EACH_PTR(arg);
113 return;
115 found:
116 i++;
117 p = arg->string->data;
118 while ((p = strchr(p, '%'))) {
119 if (i >= ARRAY_SIZE(ignored_params))
120 return;
121 p++;
122 if (*p == '%') {
123 p++;
124 continue;
126 if (*p == '.')
127 p++;
128 if (*p == '*')
129 i++;
130 if (*p == 'p')
131 ignored_params[i] = 1;
132 i++;
136 static int is_free_func(struct expression *fn)
138 char *name;
139 int ret = 0;
141 name = expr_to_str(fn);
142 if (!name)
143 return 0;
144 if (strstr(name, "free"))
145 ret = 1;
146 free_string(name);
148 return ret;
151 static void match_call(struct expression *expr)
153 struct expression *arg;
154 char *name;
155 int i;
157 if (is_impossible_path())
158 return;
160 set_ignored_params(expr);
162 i = -1;
163 FOR_EACH_PTR(expr->args, arg) {
164 i++;
165 if (!is_pointer(arg))
166 continue;
167 if (!is_freed(arg))
168 continue;
169 if (ignored_params[i])
170 continue;
172 name = expr_to_var(arg);
173 if (is_free_func(expr->fn))
174 sm_msg("error: double free of '%s'", name);
175 else
176 sm_msg("warn: passing freed memory '%s'", name);
177 set_state_expr(my_id, arg, &ok);
178 free_string(name);
179 } END_FOR_EACH_PTR(arg);
182 static void match_return(struct expression *expr)
184 char *name;
186 if (is_impossible_path())
187 return;
189 if (!expr)
190 return;
191 if (!is_freed(expr))
192 return;
194 name = expr_to_var(expr);
195 sm_msg("warn: returning freed memory '%s'", name);
196 set_state_expr(my_id, expr, &ok);
197 free_string(name);
200 static void match_free(const char *fn, struct expression *expr, void *param)
202 struct expression *arg;
204 if (is_impossible_path())
205 return;
207 arg = get_argument_from_call_expr(expr->args, PTR_INT(param));
208 if (!arg)
209 return;
210 if (is_freed(arg)) {
211 char *name = expr_to_var(arg);
213 sm_msg("error: double free of '%s'", name);
214 free_string(name);
216 set_state_expr(my_id, arg, &freed);
219 static void set_param_freed(struct expression *expr, int param, char *key, char *value)
221 struct expression *arg;
222 char *name;
223 struct symbol *sym;
224 struct sm_state *sm;
226 while (expr->type == EXPR_ASSIGNMENT)
227 expr = strip_expr(expr->right);
228 if (expr->type != EXPR_CALL)
229 return;
231 arg = get_argument_from_call_expr(expr->args, param);
232 if (!arg)
233 return;
234 name = get_variable_from_key(arg, key, &sym);
235 if (!name || !sym)
236 goto free;
238 if (!is_impossible_path()) {
239 sm = get_sm_state(my_id, name, sym);
240 if (sm && slist_has_state(sm->possible, &freed)) {
241 sm_msg("warn: '%s' double freed", name);
242 set_state(my_id, name, sym, &ok); /* fixme: doesn't silence anything. I know */
246 set_state(my_id, name, sym, &freed);
247 free:
248 free_string(name);
251 int parent_is_free_var_sym_strict(const char *name, struct symbol *sym)
253 char buf[256];
254 char *start;
255 char *end;
256 struct smatch_state *state;
258 strncpy(buf, name, sizeof(buf) - 1);
259 buf[sizeof(buf) - 1] = '\0';
261 start = &buf[0];
262 while ((*start == '&'))
263 start++;
265 while ((end = strrchr(start, '-'))) {
266 *end = '\0';
267 state = __get_state(my_id, start, sym);
268 if (state == &freed)
269 return 1;
271 return 0;
274 int parent_is_free_strict(struct expression *expr)
276 struct symbol *sym;
277 char *var;
278 int ret = 0;
280 expr = strip_expr(expr);
281 var = expr_to_var_sym(expr, &sym);
282 if (!var || !sym)
283 goto free;
284 ret = parent_is_free_var_sym_strict(var, sym);
285 free:
286 free_string(var);
287 return ret;
290 void check_free_strict(int id)
292 my_id = id;
294 if (option_project != PROJ_KERNEL)
295 return;
297 add_function_hook("kfree", &match_free, INT_PTR(0));
298 add_function_hook("kmem_cache_free", &match_free, INT_PTR(1));
300 if (option_spammy)
301 add_hook(&match_symbol, SYM_HOOK);
302 add_hook(&match_dereferences, DEREF_HOOK);
303 add_hook(&match_call, FUNCTION_CALL_HOOK);
304 add_hook(&match_return, RETURN_HOOK);
306 add_modification_hook_late(my_id, &ok_to_use);
307 add_pre_merge_hook(my_id, &pre_merge_hook);
309 select_return_states_hook(PARAM_FREED, &set_param_freed);