Skip to content

Commit 7834dd5

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 f2a7665 commit 7834dd5

File tree

3 files changed

+50
-5
lines changed

3 files changed

+50
-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

+16-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,16 @@ 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+
value_set.assign(l1_lhs, l1_rhs, ns, true, true);
130+
value_set.assign(l1_rhs, l1_lhs, ns, true, true);
131+
}
121132
}
122133
}
123134
else if(

0 commit comments

Comments
 (0)