math: handle MOD operation better
[smatch.git] / smatch_param_set.c
blobdd215eb04a901391840d9f56398ea04a01551fd2
1 /*
2 * sparse/smatch_param_set.c
4 * Copyright (C) 2012 Oracle.
6 * Licensed under the Open Software License version 1.1
8 */
11 * This is for functions like:
13 * int foo(int *x)
14 * {
15 * if (*x == 42) {
16 * *x = 0;
17 * return 1;
18 * }
19 * return 0;
20 * }
22 * If we return 1 that means the value of *x has been set to 0. If we return
23 * 0 then we have left *x alone.
27 #include "scope.h"
28 #include "smatch.h"
29 #include "smatch_slist.h"
30 #include "smatch_extra.h"
32 static int my_id;
34 static struct smatch_state *unmatched_state(struct sm_state *sm)
36 return alloc_estate_empty();
39 static void extra_mod_hook(const char *name, struct symbol *sym, struct smatch_state *state)
41 set_state(my_id, name, sym, state);
44 static const char *get_param_name(struct sm_state *sm)
46 char *param_name;
47 int name_len;
48 static char buf[256];
50 param_name = sm->sym->ident->name;
51 name_len = strlen(param_name);
53 if (sm->name[name_len] == '-' && /* check for '-' from "->" */
54 strncmp(sm->name, param_name, name_len) == 0) {
55 snprintf(buf, sizeof(buf), "$$%s", sm->name + name_len);
56 return buf;
57 } else if (sm->name[0] == '*' && strcmp(sm->name + 1, param_name) == 0) {
58 return "*$$";
60 return NULL;
63 static void print_one_return_value_param(int return_id, char *return_ranges,
64 int param, struct sm_state *sm, char *implied_rl,
65 struct state_list *slist)
67 const char *param_name;
69 param_name = get_param_name(sm);
70 if (!param_name)
71 return;
73 sm_msg("info: return_param_add %d %d '%s' '%s' '%s' %s",
74 return_id, param, return_ranges,
75 param_name, implied_rl, global_static());
78 static void print_return_value_param(int return_id, char *return_ranges, struct expression *expr, struct state_list *slist)
80 struct state_list *my_slist;
81 struct sm_state *sm;
82 struct smatch_state *extra;
83 int param;
84 struct range_list *rl;
86 my_slist = get_all_states_slist(my_id, slist);
88 FOR_EACH_PTR(my_slist, sm) {
89 if (!estate_rl(sm->state))
90 continue;
91 extra = get_state_slist(slist, SMATCH_EXTRA, sm->name, sm->sym);
92 if (!estate_rl(extra))
93 continue;
94 rl = rl_intersection(estate_rl(sm->state), estate_rl(extra));
95 if (!rl)
96 continue;
98 param = get_param_num_from_sym(sm->sym);
99 if (param < 0)
100 continue;
101 if (!sm->sym->ident)
102 continue;
103 print_one_return_value_param(return_id, return_ranges, param, sm, show_rl(rl), slist);
104 } END_FOR_EACH_PTR(sm);
107 void register_param_set(int id)
109 if (!option_info)
110 return;
112 my_id = id;
114 add_extra_mod_hook(&extra_mod_hook);
115 add_unmatched_state_hook(my_id, &unmatched_state);
116 add_merge_hook(my_id, &merge_estates);
117 add_returned_state_callback(&print_return_value_param);