#include "smatch.h"
#include "smatch_slist.h"
#include "smatch_extra.h"
static int my_id;
STATE(capped);
STATE(uncapped);
static void set_uncapped(struct sm_state *sm, struct expression *mod_expr)
{
set_state(my_id, sm->name, sm->sym, &uncapped);
}
static struct smatch_state *unmatched_state(struct sm_state *sm)
{
struct smatch_state *state;
state = __get_state(SMATCH_EXTRA, sm->name, sm->sym);
if (state && !estate_is_whole(state))
return &capped;
return &uncapped;
}
static int is_capped_macro(struct expression *expr)
{
char *name;
name = get_macro_name(expr->pos);
if (!name)
return 0;
if (strcmp(name, "min") == 0)
return 1;
if (strcmp(name, "MIN") == 0)
return 1;
if (strcmp(name, "min_t") == 0)
return 1;
return 0;
}
int is_capped(struct expression *expr)
{
struct symbol *type;
sval_t dummy;
expr = strip_expr(expr);
while (expr && expr->type == EXPR_POSTOP) {
expr = strip_expr(expr->unop);
}
if (!expr)
return 0;
type = get_type(expr);
if (is_ptr_type(type))
return 0;
if (type == &bool_ctype)
return 0;
if (type_bits(type) >= 0 && type_bits(type) <= 2)
return 0;
if (get_hard_max(expr, &dummy))
return 1;
if (is_capped_macro(expr))
return 1;
if (expr->type == EXPR_BINOP) {
struct range_list *left_rl, *right_rl;
sval_t sval;
if (expr->op == '&' && !get_value(expr->right, &sval))
return 1;
if (expr->op == SPECIAL_RIGHTSHIFT)
return 0;
if (expr->op == '%' &&
!get_value(expr->right, &sval) && is_capped(expr->right))
return 1;
if (!is_capped(expr->left))
return 0;
if (expr->op == '/')
return 1;
if (!is_capped(expr->right))
return 0;
if (expr->op == '*') {
get_absolute_rl(expr->left, &left_rl);
get_absolute_rl(expr->right, &right_rl);
if (sval_is_negative(rl_min(left_rl)) ||
sval_is_negative(rl_min(right_rl)))
return 0;
}
return 1;
}
if (get_state_expr(my_id, expr) == &capped)
return 1;
return 0;
}
int is_capped_var_sym(const char *name, struct symbol *sym)
{
if (get_state(my_id, name, sym) == &capped)
return 1;
return 0;
}
void set_param_capped_data(const char *name, struct symbol *sym, char *key, char *value)
{
char fullname[256];
if (strncmp(key, "$", 1))
return;
snprintf(fullname, 256, "%s%s", name, key + 1);
set_state(my_id, fullname, sym, &capped);
}
static void match_condition(struct expression *expr)
{
struct expression *left, *right;
struct smatch_state *left_true = NULL;
struct smatch_state *left_false = NULL;
struct smatch_state *right_true = NULL;
struct smatch_state *right_false = NULL;
sval_t sval;
if (expr->type != EXPR_COMPARE)
return;
left = strip_expr(expr->left);
right = strip_expr(expr->right);
while (left->type == EXPR_ASSIGNMENT)
left = strip_expr(left->left);
if (get_implied_value(left, &sval) ||
get_implied_value(right, &sval))
return;
switch (expr->op) {
case '<':
case SPECIAL_LTE:
case SPECIAL_UNSIGNED_LT:
case SPECIAL_UNSIGNED_LTE:
left_true = &capped;
right_false = &capped;
break;
case '>':
case SPECIAL_GTE:
case SPECIAL_UNSIGNED_GT:
case SPECIAL_UNSIGNED_GTE:
left_false = &capped;
right_true = &capped;
break;
case SPECIAL_EQUAL:
left_true = &capped;
right_true = &capped;
break;
case SPECIAL_NOTEQUAL:
left_false = &capped;
right_false = &capped;
break;
default:
return;
}
set_true_false_states_expr(my_id, left, left_true, left_false);
set_true_false_states_expr(my_id, right, right_true, right_false);
}
static void match_assign(struct expression *expr)
{
struct symbol *type;
type = get_type(expr);
if (is_ptr_type(type))
return;
if (type == &bool_ctype)
return;
if (type_bits(type) >= 0 && type_bits(type) <= 2)
return;
if (is_capped(expr->right)) {
set_state_expr(my_id, expr->left, &capped);
} else {
if (get_state_expr(my_id, expr->left))
set_state_expr(my_id, expr->left, &uncapped);
}
}
static void match_caller_info(struct expression *expr)
{
struct expression *tmp;
sval_t sval;
int i;
i = -1;
FOR_EACH_PTR(expr->args, tmp) {
i++;
if (get_implied_value(tmp, &sval))
continue;
if (!is_capped(tmp))
continue;
sql_insert_caller_info(expr, CAPPED_DATA, i, "$", "1");
} END_FOR_EACH_PTR(tmp);
}
static void struct_member_callback(struct expression *call, int param, char *printed_name, struct sm_state *sm)
{
struct smatch_state *estate;
sval_t sval;
if (sm->state != &capped)
return;
estate = __get_state(SMATCH_EXTRA, sm->name, sm->sym);
if (estate_get_single_value(estate, &sval))
return;
sql_insert_caller_info(call, CAPPED_DATA, param, printed_name, "1");
}
static void print_return_implies_capped(int return_id, char *return_ranges, struct expression *expr)
{
struct smatch_state *orig, *estate;
struct sm_state *sm;
struct symbol *ret_sym;
const char *param_name;
char *return_str;
int param;
sval_t sval;
bool return_found = false;
expr = strip_expr(expr);
return_str = expr_to_str(expr);
ret_sym = expr_to_sym(expr);
FOR_EACH_MY_SM(my_id, __get_cur_stree(), sm) {
if (sm->state != &capped)
continue;
param = get_param_num_from_sym(sm->sym);
if (param < 0)
continue;
estate = __get_state(SMATCH_EXTRA, sm->name, sm->sym);
if (estate_get_single_value(estate, &sval))
continue;
orig = get_state_stree(get_start_states(), my_id, sm->name, sm->sym);
if (orig == &capped && !param_was_set_var_sym(sm->name, sm->sym))
continue;
param_name = get_param_name(sm);
if (!param_name)
continue;
sql_insert_return_states(return_id, return_ranges, CAPPED_DATA,
param, param_name, "1");
} END_FOR_EACH_SM(sm);
FOR_EACH_MY_SM(my_id, __get_cur_stree(), sm) {
if (!ret_sym)
break;
if (sm->state != &capped)
continue;
if (ret_sym != sm->sym)
continue;
estate = __get_state(SMATCH_EXTRA, sm->name, sm->sym);
if (estate_get_single_value(estate, &sval))
continue;
param_name = state_name_to_param_name(sm->name, return_str);
if (!param_name)
continue;
if (strcmp(param_name, "$") == 0)
return_found = true;
sql_insert_return_states(return_id, return_ranges, CAPPED_DATA,
-1, param_name, "1");
} END_FOR_EACH_SM(sm);
if (return_found)
goto free_string;
if (option_project == PROJ_KERNEL && get_function() &&
strstr(get_function(), "nla_get_"))
sql_insert_return_states(return_id, return_ranges, CAPPED_DATA,
-1, "$", "1");
free_string:
free_string(return_str);
}
static void db_return_states_capped(struct expression *expr, int param, char *key, char *value)
{
char *name;
struct symbol *sym;
name = return_state_to_var_sym(expr, param, key, &sym);
if (!name || !sym)
goto free;
set_state(my_id, name, sym, &capped);
free:
free_string(name);
}
void register_capped(int id)
{
my_id = id;
add_unmatched_state_hook(my_id, &unmatched_state);
select_caller_info_hook(set_param_capped_data, CAPPED_DATA);
add_hook(&match_condition, CONDITION_HOOK);
add_hook(&match_assign, ASSIGNMENT_HOOK);
add_modification_hook(my_id, &set_uncapped);
add_hook(&match_caller_info, FUNCTION_CALL_HOOK);
add_member_info_callback(my_id, struct_member_callback);
add_split_return_callback(print_return_implies_capped);
select_return_states_hook(CAPPED_DATA, &db_return_states_capped);
}