Skip to content

Commit 5fade35

Browse files
committed
Conditionally simplify use of if-then-else conditions
When symbolically executing (and dereferencing) statements like `*p = *p + 1;` where `p` may point to more than a single object we ended up with nested if-then-else expressions on the right-hand side that would use the very same conditions. In this particular case, we should propagate the conditions into nested if-then-else expressions. This simplification may be expensive, so we don't enable it in general.
1 parent 9220c3a commit 5fade35

File tree

6 files changed

+107
-79
lines changed

6 files changed

+107
-79
lines changed

regression/cbmc/Pointer32/main.c

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
struct S
2+
{
3+
int len;
4+
};
5+
6+
__CPROVER_bool nondet_bool();
7+
8+
int main()
9+
{
10+
struct S *s_p = 0;
11+
struct S s;
12+
if(nondet_bool())
13+
s_p = &s;
14+
s_p->len = s_p->len + 1;
15+
__CPROVER_assert(0, "");
16+
}

regression/cbmc/Pointer32/test.desc

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
CORE paths-lifo-expected-failure
2+
main.c
3+
--show-vcc
4+
main::1::s!0@1#2..len = .+ \? .+ : .+
5+
^SIGNAL=0$
6+
^EXIT=0$
7+
--
8+
main::1::s!0@1#2..len = .+ \? .+ \?
9+
--
10+
We should not find nested if-then-else expressions in the right-hand side
11+
evaluating `len` when simplification works as intended. (With --paths lifo we do
12+
not have any branching, so it is marked paths-lifo-expected-failure.)

src/goto-symex/symex_assign.cpp

+5-1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ Author: Daniel Kroening, [email protected]
1010
/// Symbolic Execution
1111

1212
#include "symex_assign.h"
13+
#include <util/simplify_expr_class.h>
1314

1415
#include <util/byte_operators.h>
1516
#include <util/expr_util.h>
@@ -203,7 +204,10 @@ void symex_assignt::assign_non_struct_symbol(
203204
assignmentt assignment{lhs, full_lhs, l2_rhs};
204205

205206
if(symex_config.simplify_opt)
206-
assignment.rhs = simplify_expr(std::move(assignment.rhs), ns);
207+
{
208+
simplify_exprt simp{ns, true};
209+
simp.simplify(assignment.rhs);
210+
}
207211

208212
const ssa_exprt l2_lhs = state
209213
.assignment(

src/util/simplify_expr.cpp

+8-9
Original file line numberDiff line numberDiff line change
@@ -3151,22 +3151,21 @@ simplify_exprt::resultt<> simplify_exprt::simplify_rec(const exprt &expr)
31513151
simplify_node_preorder_result.expr_changed;
31523152
}
31533153

3154-
#ifdef USE_LOCAL_REPLACE_MAP
3155-
exprt tmp = simplify_node_result.expr;
3156-
# if 1
3157-
replace_mapt::const_iterator it =
3158-
local_replace_map.find(simplify_node_result.expr);
3159-
if(it!=local_replace_map.end())
3160-
simplify_node_result = changed(it->second);
3154+
if(!local_replace_map.empty())
3155+
{
3156+
#if 1
3157+
replace_mapt::const_iterator it =
3158+
local_replace_map.find(simplify_node_result.expr);
3159+
if(it != local_replace_map.end())
3160+
simplify_node_result = changed(it->second);
31613161
# else
31623162
if(
3163-
!local_replace_map.empty() &&
31643163
!replace_expr(local_replace_map, simplify_node_result.expr))
31653164
{
31663165
simplify_node_result = changed(simplify_rec(simplify_node_result.expr));
31673166
}
3168-
# endif
31693167
#endif
3168+
}
31703169

31713170
if(!simplify_node_result.has_changed())
31723171
{

src/util/simplify_expr_class.h

+9-10
Original file line numberDiff line numberDiff line change
@@ -20,10 +20,7 @@ Author: Daniel Kroening, [email protected]
2020
#include "expr.h"
2121
#include "mp_arith.h"
2222
#include "type.h"
23-
// #define USE_LOCAL_REPLACE_MAP
24-
#ifdef USE_LOCAL_REPLACE_MAP
2523
#include "replace_expr.h"
26-
#endif
2724

2825
class abs_exprt;
2926
class address_of_exprt;
@@ -80,11 +77,12 @@ class with_exprt;
8077
class simplify_exprt
8178
{
8279
public:
83-
explicit simplify_exprt(const namespacet &_ns):
84-
do_simplify_if(true),
85-
ns(_ns)
80+
simplify_exprt(const namespacet &_ns, bool propagate_if_cond)
81+
: do_simplify_if(propagate_if_cond),
82+
ns(_ns)
8683
#ifdef DEBUG_ON_DEMAND
87-
, debug_on(false)
84+
,
85+
debug_on(false)
8886
#endif
8987
{
9088
#ifdef DEBUG_ON_DEMAND
@@ -93,6 +91,10 @@ class simplify_exprt
9391
#endif
9492
}
9593

94+
explicit simplify_exprt(const namespacet &_ns) : simplify_exprt(_ns, false)
95+
{
96+
}
97+
9698
virtual ~simplify_exprt()
9799
{
98100
}
@@ -287,10 +289,7 @@ class simplify_exprt
287289
#ifdef DEBUG_ON_DEMAND
288290
bool debug_on;
289291
#endif
290-
#ifdef USE_LOCAL_REPLACE_MAP
291292
replace_mapt local_replace_map;
292-
#endif
293-
294293
};
295294

296295
#endif // CPROVER_UTIL_SIMPLIFY_EXPR_CLASS_H

src/util/simplify_expr_if.cpp

+57-59
Original file line numberDiff line numberDiff line change
@@ -254,17 +254,16 @@ simplify_exprt::simplify_if_preorder(const if_exprt &expr)
254254
simplify_rec(r_cond.expr.is_true() ? truevalue : falsevalue));
255255
}
256256

257-
if(do_simplify_if)
258-
{
259-
bool swap_branches = false;
257+
bool swap_branches = false;
260258

261-
if(r_cond.expr.id() == ID_not)
262-
{
263-
r_cond = changed(to_not_expr(r_cond.expr).op());
264-
swap_branches = true;
265-
}
259+
if(r_cond.expr.id() == ID_not)
260+
{
261+
r_cond = changed(to_not_expr(r_cond.expr).op());
262+
swap_branches = true;
263+
}
266264

267-
#ifdef USE_LOCAL_REPLACE_MAP
265+
if(do_simplify_if)
266+
{
268267
replace_mapt map_before(local_replace_map);
269268

270269
// a ? b : c --> a ? b[a/true] : c
@@ -273,7 +272,10 @@ simplify_exprt::simplify_if_preorder(const if_exprt &expr)
273272
for(const auto &op : r_cond.expr.operands())
274273
{
275274
if(op.id() == ID_not)
276-
local_replace_map.insert(std::make_pair(op.op0(), false_exprt()));
275+
{
276+
local_replace_map.insert(
277+
std::make_pair(to_not_expr(op).op(), false_exprt()));
278+
}
277279
else
278280
local_replace_map.insert(std::make_pair(op, true_exprt()));
279281
}
@@ -291,7 +293,10 @@ simplify_exprt::simplify_if_preorder(const if_exprt &expr)
291293
for(const auto &op : r_cond.expr.operands())
292294
{
293295
if(op.id() == ID_not)
294-
local_replace_map.insert(std::make_pair(op.op0(), true_exprt()));
296+
{
297+
local_replace_map.insert(
298+
std::make_pair(to_not_expr(op).op(), true_exprt()));
299+
}
295300
else
296301
local_replace_map.insert(std::make_pair(op, false_exprt()));
297302
}
@@ -306,11 +311,13 @@ simplify_exprt::simplify_if_preorder(const if_exprt &expr)
306311
if(swap_branches)
307312
{
308313
// tell build_if_expr to replace truevalue and falsevalue
309-
r_truevalue.expr_changed = CHANGED;
310-
r_falsevalue.expr_changed = CHANGED;
314+
r_truevalue.expr_changed = resultt<>::CHANGED;
315+
r_falsevalue.expr_changed = resultt<>::CHANGED;
311316
}
312317
return build_if_expr(expr, r_cond, r_truevalue, r_falsevalue);
313-
#else
318+
}
319+
else
320+
{
314321
if(!swap_branches)
315322
{
316323
return build_if_expr(
@@ -324,12 +331,6 @@ simplify_exprt::simplify_if_preorder(const if_exprt &expr)
324331
changed(simplify_rec(falsevalue)),
325332
changed(simplify_rec(truevalue)));
326333
}
327-
#endif
328-
}
329-
else
330-
{
331-
return build_if_expr(
332-
expr, r_cond, simplify_rec(truevalue), simplify_rec(falsevalue));
333334
}
334335
}
335336

@@ -339,49 +340,46 @@ simplify_exprt::resultt<> simplify_exprt::simplify_if(const if_exprt &expr)
339340
const exprt &truevalue = expr.true_case();
340341
const exprt &falsevalue = expr.false_case();
341342

342-
if(do_simplify_if)
343-
{
344343
#if 0
345-
no_change = simplify_if_cond(cond) && no_change;
346-
no_change = simplify_if_branch(truevalue, falsevalue, cond) && no_change;
344+
no_change = simplify_if_cond(cond) && no_change;
345+
no_change = simplify_if_branch(truevalue, falsevalue, cond) && no_change;
347346
#endif
348347

349-
if(expr.type() == bool_typet())
350-
{
351-
// a?b:c <-> (a && b) || (!a && c)
348+
if(expr.type() == bool_typet())
349+
{
350+
// a?b:c <-> (a && b) || (!a && c)
352351

353-
if(truevalue.is_true() && falsevalue.is_false())
354-
{
355-
// a?1:0 <-> a
356-
return cond;
357-
}
358-
else if(truevalue.is_false() && falsevalue.is_true())
359-
{
360-
// a?0:1 <-> !a
361-
return changed(simplify_not(not_exprt(cond)));
362-
}
363-
else if(falsevalue.is_false())
364-
{
365-
// a?b:0 <-> a AND b
366-
return changed(simplify_boolean(and_exprt(cond, truevalue)));
367-
}
368-
else if(falsevalue.is_true())
369-
{
370-
// a?b:1 <-> !a OR b
371-
return changed(
372-
simplify_boolean(or_exprt(simplify_not(not_exprt(cond)), truevalue)));
373-
}
374-
else if(truevalue.is_true())
375-
{
376-
// a?1:b <-> a||(!a && b) <-> a OR b
377-
return changed(simplify_boolean(or_exprt(cond, falsevalue)));
378-
}
379-
else if(truevalue.is_false())
380-
{
381-
// a?0:b <-> !a && b
382-
return changed(simplify_boolean(
383-
and_exprt(simplify_not(not_exprt(cond)), falsevalue)));
384-
}
352+
if(truevalue.is_true() && falsevalue.is_false())
353+
{
354+
// a?1:0 <-> a
355+
return cond;
356+
}
357+
else if(truevalue.is_false() && falsevalue.is_true())
358+
{
359+
// a?0:1 <-> !a
360+
return changed(simplify_not(not_exprt(cond)));
361+
}
362+
else if(falsevalue.is_false())
363+
{
364+
// a?b:0 <-> a AND b
365+
return changed(simplify_boolean(and_exprt(cond, truevalue)));
366+
}
367+
else if(falsevalue.is_true())
368+
{
369+
// a?b:1 <-> !a OR b
370+
return changed(
371+
simplify_boolean(or_exprt(simplify_not(not_exprt(cond)), truevalue)));
372+
}
373+
else if(truevalue.is_true())
374+
{
375+
// a?1:b <-> a||(!a && b) <-> a OR b
376+
return changed(simplify_boolean(or_exprt(cond, falsevalue)));
377+
}
378+
else if(truevalue.is_false())
379+
{
380+
// a?0:b <-> !a && b
381+
return changed(
382+
simplify_boolean(and_exprt(simplify_not(not_exprt(cond)), falsevalue)));
385383
}
386384
}
387385

0 commit comments

Comments
 (0)