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"
22 extern int second_half_id
;
23 extern void set_spectre_first_half(struct expression
*expr
);
25 static int suppress_multiple
= 1;
27 static int is_write(struct expression
*expr
)
32 static int is_read(struct expression
*expr
)
34 struct expression
*parent
, *last_parent
;
35 struct statement
*stmt
;
41 while ((parent
= expr_get_parent_expr(expr
))){
45 /* If we pass a value as a parameter that's a read, probably? */
46 // if (parent->type == EXPR_CALL)
49 if (parent
->type
== EXPR_ASSIGNMENT
) {
50 if (parent
->right
== expr
)
52 if (parent
->left
== expr
)
58 stmt
= expr_get_parent_stmt(last_parent
);
59 if (stmt
&& stmt
->type
== STMT_RETURN
)
65 static int is_harmless(struct expression
*expr
)
67 struct expression
*tmp
, *parent
;
68 struct statement
*stmt
;
72 while ((tmp
= expr_get_parent_expr(parent
))) {
73 if (tmp
->type
== EXPR_ASSIGNMENT
|| tmp
->type
== EXPR_CALL
)
80 stmt
= expr_get_parent_stmt(parent
);
83 if (stmt
->type
== STMT_IF
&& stmt
->if_conditional
== parent
)
85 if (stmt
->type
== STMT_ITERATOR
&&
86 (stmt
->iterator_pre_condition
== parent
||
87 stmt
->iterator_post_condition
== parent
))
93 static unsigned long long get_max_by_type(struct expression
*expr
)
96 .type
= &ullong_ctype
,
103 expr
= strip_parens(expr
);
104 type
= get_type(expr
);
105 if (type
&& sval_type_max(type
).uvalue
< max
.uvalue
)
106 max
= sval_type_max(type
);
107 if (expr
->type
== EXPR_PREOP
) {
109 } else if (expr
->type
== EXPR_BINOP
) {
110 if (expr
->op
== '%' || expr
->op
== '&')
115 expr
= get_assigned_expr(expr
);
126 static unsigned long long get_mask(struct expression
*expr
)
128 struct expression
*tmp
;
132 expr
= strip_expr(expr
);
134 tmp
= get_assigned_expr(expr
);
139 tmp
= get_assigned_expr(expr
);
142 if (expr
->type
== EXPR_BINOP
&& expr
->op
== '&') {
143 if (get_value(expr
->right
, &mask
)) /* right is the common case */
145 if (get_value(expr
->left
, &mask
))
152 static void array_check(struct expression
*expr
)
154 struct expression_list
*conditions
;
155 struct expression
*array_expr
, *offset
;
156 unsigned long long mask
;
160 expr
= strip_expr(expr
);
164 if (is_impossible_path())
166 if (is_harmless(expr
))
169 array_expr
= get_array_base(expr
);
170 if (suppress_multiple
&& is_ignored_expr(my_id
, array_expr
)) {
171 set_spectre_first_half(expr
);
175 offset
= get_array_offset(expr
);
176 if (!is_user_rl(offset
))
178 if (is_nospec(offset
))
181 array_size
= get_array_size(array_expr
);
182 if (array_size
> 0 && get_max_by_type(offset
) < array_size
)
184 // binfo = get_bit_info(offset);
185 // if (array_size > 0 && binfo && binfo->possible < array_size)
188 mask
= get_mask(offset
);
189 if (mask
<= array_size
)
192 conditions
= get_conditions(offset
);
194 name
= expr_to_str(array_expr
);
195 sm_warning("potential spectre issue '%s' [%s]%s",
197 is_read(expr
) ? "r" : "w",
198 conditions
? " (local cap)" : "");
200 set_spectre_first_half(expr
);
201 if (suppress_multiple
)
202 add_ignore_expr(my_id
, array_expr
);
206 void check_spectre(int id
)
210 suppress_multiple
= getenv("FULL_SPECTRE") == NULL
;
212 if (option_project
!= PROJ_KERNEL
)
215 add_hook(&array_check
, OP_HOOK
);