Skip to content

Commit 2a27074

Browse files
committed
goto-symex: assumed pointer equalities must update value set
Value sets and constant propagation are our only sources of points-to information when resolving dereferences in goto-symex. Therefore, assuming or branching on equalities of pointer-typed variables must have us consider both of their value sets to be equal. Else we may end up with spurious counterexamples as seen with the enclosed regression test (where we previously did not have any known value for `a`). Fixes: #8492
1 parent 018c61c commit 2a27074

File tree

3 files changed

+54
-5
lines changed

3 files changed

+54
-5
lines changed
+27
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
int main()
2+
{
3+
int *r;
4+
int *a;
5+
#if 1
6+
_Bool tmp_if_expr;
7+
if(r == 0)
8+
{
9+
tmp_if_expr = 0;
10+
}
11+
else
12+
{
13+
r = __CPROVER_allocate(sizeof(int), 0);
14+
tmp_if_expr = 1;
15+
}
16+
17+
__CPROVER_assume(tmp_if_expr);
18+
#else
19+
// this works, because we constant-propagate r as an address-of expression
20+
__CPROVER_assume(r != 0);
21+
r = __CPROVER_allocate(sizeof(int), 0);
22+
#endif
23+
__CPROVER_assume(r != 0);
24+
__CPROVER_assume(r == a);
25+
__CPROVER_assert(r == a, " r == a before");
26+
__CPROVER_assert(*r == *a, "*r == *a before");
27+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
CORE
2+
main.c
3+
4+
^EXIT=0$
5+
^SIGNAL=0$
6+
^VERIFICATION SUCCESSFUL$
7+
--

src/goto-symex/goto_state.cpp

+20-5
Original file line numberDiff line numberDiff line change
@@ -92,19 +92,20 @@ void goto_statet::apply_condition(
9292
if(is_ssa_expr(rhs))
9393
std::swap(lhs, rhs);
9494

95-
if(is_ssa_expr(lhs) && goto_symex_can_forward_propagatet(ns)(rhs))
95+
if(is_ssa_expr(lhs))
9696
{
9797
const ssa_exprt &ssa_lhs = to_ssa_expr(lhs);
9898
INVARIANT(
9999
!ssa_lhs.get_level_2().empty(),
100100
"apply_condition operand should be L2 renamed");
101+
const ssa_exprt l1_lhs = remove_level_2(ssa_lhs);
101102

102103
if(
103-
previous_state.threads.size() == 1 ||
104-
previous_state.write_is_shared(ssa_lhs, ns) !=
105-
goto_symex_statet::write_is_shared_resultt::SHARED)
104+
goto_symex_can_forward_propagatet(ns)(rhs) &&
105+
(previous_state.threads.size() == 1 ||
106+
previous_state.write_is_shared(ssa_lhs, ns) !=
107+
goto_symex_statet::write_is_shared_resultt::SHARED))
106108
{
107-
const ssa_exprt l1_lhs = remove_level_2(ssa_lhs);
108109
const irep_idt &l1_identifier = l1_lhs.get_identifier();
109110

110111
level2.increase_generation(
@@ -118,6 +119,20 @@ void goto_statet::apply_condition(
118119

119120
value_set.assign(l1_lhs, rhs, ns, true, false);
120121
}
122+
else if(is_ssa_expr(rhs))
123+
{
124+
const ssa_exprt &ssa_rhs = to_ssa_expr(rhs);
125+
INVARIANT(
126+
!ssa_rhs.get_level_2().empty(),
127+
"apply_condition operand should be L2 renamed");
128+
const ssa_exprt l1_rhs = remove_level_2(ssa_rhs);
129+
130+
// We have a condition a == b. Make both a's and b's value sets the
131+
// union of their previous value sets (the last "true" argument makes
132+
// sure we add rather than replace value sets).
133+
value_set.assign(l1_lhs, l1_rhs, ns, true, true);
134+
value_set.assign(l1_rhs, l1_lhs, ns, true, true);
135+
}
121136
}
122137
}
123138
else if(

0 commit comments

Comments
 (0)