2 * Copyright (C) 2018 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 #include "smatch_extra.h"
23 static int suppress_multiple
= 1;
25 static int is_write(struct expression
*expr
)
30 static int is_read(struct expression
*expr
)
32 struct expression
*parent
, *last_parent
;
33 struct statement
*stmt
;
39 while ((parent
= expr_get_parent_expr(expr
))){
43 /* If we pass a value as a parameter that's a read, probably? */
44 // if (parent->type == EXPR_CALL)
47 if (parent
->type
== EXPR_ASSIGNMENT
) {
48 if (parent
->right
== expr
)
50 if (parent
->left
== expr
)
56 stmt
= expr_get_parent_stmt(last_parent
);
57 if (stmt
&& stmt
->type
== STMT_RETURN
)
63 static int is_harmless(struct expression
*expr
)
65 struct expression
*tmp
, *parent
;
66 struct statement
*stmt
;
70 while ((tmp
= expr_get_parent_expr(parent
))) {
71 if (tmp
->type
== EXPR_ASSIGNMENT
|| tmp
->type
== EXPR_CALL
)
78 stmt
= expr_get_parent_stmt(parent
);
81 if (stmt
->type
== STMT_IF
&& stmt
->if_conditional
== parent
)
83 if (stmt
->type
== STMT_ITERATOR
&&
84 (stmt
->iterator_pre_condition
== parent
||
85 stmt
->iterator_post_condition
== parent
))
91 static unsigned long long get_max_by_type(struct expression
*expr
)
94 .type
= &ullong_ctype
,
101 expr
= strip_parens(expr
);
102 type
= get_type(expr
);
103 if (type
&& sval_type_max(type
).uvalue
< max
.uvalue
)
104 max
= sval_type_max(type
);
105 if (expr
->type
== EXPR_PREOP
) {
107 } else if (expr
->type
== EXPR_BINOP
) {
108 if (expr
->op
== '%' || expr
->op
== '&')
113 expr
= get_assigned_expr(expr
);
124 static unsigned long long get_mask(struct expression
*expr
)
126 struct expression
*tmp
;
130 expr
= strip_expr(expr
);
132 tmp
= get_assigned_expr(expr
);
137 tmp
= get_assigned_expr(expr
);
140 if (expr
->type
== EXPR_BINOP
&& expr
->op
== '&') {
141 if (get_value(expr
->right
, &mask
)) /* right is the common case */
143 if (get_value(expr
->left
, &mask
))
150 static void array_check(struct expression
*expr
)
152 struct expression_list
*conditions
;
153 struct expression
*array_expr
, *offset
;
154 unsigned long long mask
;
158 expr
= strip_expr(expr
);
162 if (is_impossible_path())
164 if (is_harmless(expr
))
167 array_expr
= get_array_base(expr
);
168 if (suppress_multiple
&& is_ignored_expr(my_id
, array_expr
))
171 offset
= get_array_offset(expr
);
172 if (!is_user_rl(offset
))
174 if (is_nospec(offset
))
177 array_size
= get_array_size(array_expr
);
178 if (array_size
> 0 && get_max_by_type(offset
) < array_size
)
180 // binfo = get_bit_info(offset);
181 // if (array_size > 0 && binfo && binfo->possible < array_size)
184 mask
= get_mask(offset
);
185 if (mask
<= array_size
)
188 conditions
= get_conditions(offset
);
190 name
= expr_to_str(array_expr
);
191 sm_msg("warn: potential spectre issue '%s' [%s]%s",
193 is_read(expr
) ? "r" : "w",
194 conditions
? " (local cap)" : "");
195 if (suppress_multiple
)
196 add_ignore_expr(my_id
, array_expr
);
200 void check_spectre(int id
)
204 suppress_multiple
= getenv("FULL_SPECTRE") == NULL
;
206 if (option_project
!= PROJ_KERNEL
)
209 add_hook(&array_check
, OP_HOOK
);