From 22418ab259412c9dabe90eb94e85e0bd03d1cf5a Mon Sep 17 00:00:00 2001 From: Daniel Kroening Date: Sun, 2 Jun 2024 15:08:16 -0700 Subject: [PATCH] property instantiation now returns largest timeframe used The word-level property instantiation now returns the largest timeframe used, in addition to the instantiated property. This enables making the counterexample have the right length. --- regression/verilog/SVA/sequence1.desc | 7 +- .../instantiate_word_level.cpp | 104 +++++++++++------- src/trans-word-level/instantiate_word_level.h | 6 + src/trans-word-level/property.cpp | 82 +------------- 4 files changed, 79 insertions(+), 120 deletions(-) diff --git a/regression/verilog/SVA/sequence1.desc b/regression/verilog/SVA/sequence1.desc index 5b2482b32..e1d993b6c 100644 --- a/regression/verilog/SVA/sequence1.desc +++ b/regression/verilog/SVA/sequence1.desc @@ -1,9 +1,12 @@ -KNOWNBUG +CORE sequence1.sv --bound 20 --numbered-trace +^\[main\.property\.1\] ##\[0:9\] main\.x == 100: REFUTED$ +^Counterexample with 10 states:$ +^main\.x@0 = 0$ +^main\.x@9 = 9$ ^EXIT=10$ ^SIGNAL=0$ -- ^warning: ignoring -- -The trace shown only has one state, but 10 are expected. diff --git a/src/trans-word-level/instantiate_word_level.cpp b/src/trans-word-level/instantiate_word_level.cpp index 459e2d595..77656cc2b 100644 --- a/src/trans-word-level/instantiate_word_level.cpp +++ b/src/trans-word-level/instantiate_word_level.cpp @@ -77,7 +77,8 @@ class wl_instantiatet } /// Instantiate the given expression for timeframe t - [[nodiscard]] exprt operator()(exprt expr, const mp_integer &t) const + [[nodiscard]] std::pair + operator()(exprt expr, const mp_integer &t) const { return instantiate_rec(std::move(expr), t); } @@ -86,7 +87,8 @@ class wl_instantiatet const mp_integer &no_timeframes; const namespacet &ns; - [[nodiscard]] exprt instantiate_rec(exprt, const mp_integer &t) const; + [[nodiscard]] std::pair + instantiate_rec(exprt, const mp_integer &t) const; [[nodiscard]] typet instantiate_rec(typet, const mp_integer &t) const; }; @@ -123,18 +125,20 @@ Function: wl_instantiatet::instantiate_rec \*******************************************************************/ -exprt wl_instantiatet::instantiate_rec(exprt expr, const mp_integer &t) const +std::pair +wl_instantiatet::instantiate_rec(exprt expr, const mp_integer &t) const { expr.type() = instantiate_rec(expr.type(), t); if(expr.id() == ID_next_symbol) { expr.id(ID_symbol); - return timeframe_symbol(t + 1, to_symbol_expr(std::move(expr))); + auto u = t + 1; + return {u, timeframe_symbol(u, to_symbol_expr(std::move(expr)))}; } else if(expr.id() == ID_symbol) { - return timeframe_symbol(t, to_symbol_expr(std::move(expr))); + return {t, timeframe_symbol(t, to_symbol_expr(std::move(expr)))}; } else if(expr.id()==ID_sva_cycle_delay) // ##[1:2] something { @@ -150,7 +154,7 @@ exprt wl_instantiatet::instantiate_rec(exprt expr, const mp_integer &t) const // Do we exceed the bound? Make it 'true' if(u >= no_timeframes) - return true_exprt(); + return {no_timeframes - 1, true_exprt()}; else return instantiate_rec(sva_cycle_delay_expr.op(), u); } @@ -168,35 +172,29 @@ exprt wl_instantiatet::instantiate_rec(exprt expr, const mp_integer &t) const else if(to_integer_non_constant(sva_cycle_delay_expr.to(), to)) throw "failed to convert sva_cycle_delay offsets"; - // This is an 'or', and we let it fail if the bound is too small. + auto lower = t + from; + auto upper = t + to; + + // Do we exceed the bound? Make it 'true' + if(upper >= no_timeframes) + return {no_timeframes - 1, true_exprt()}; exprt::operandst disjuncts; - for(mp_integer offset = from; offset < to; ++offset) + for(mp_integer u = lower; u <= upper; ++u) { - auto u = t + offset; - - if(u >= no_timeframes) - { - } - else - { - disjuncts.push_back(instantiate_rec(sva_cycle_delay_expr.op(), u)); - } + disjuncts.push_back( + instantiate_rec(sva_cycle_delay_expr.op(), u).second); } - return disjunction(disjuncts); + return {upper, disjunction(disjuncts)}; } } else if(expr.id()==ID_sva_sequence_concatenation) { // much like regular 'and' expr.id(ID_and); - - for(auto &op : expr.operands()) - op = instantiate_rec(op, t); - - return expr; + return instantiate_rec(expr, t); } else if(expr.id()==ID_sva_always) { @@ -206,10 +204,10 @@ exprt wl_instantiatet::instantiate_rec(exprt expr, const mp_integer &t) const for(auto u = t; u < no_timeframes; ++u) { - conjuncts.push_back(instantiate_rec(op, u)); + conjuncts.push_back(instantiate_rec(op, u).second); } - return conjunction(conjuncts); + return {no_timeframes - 1, conjunction(conjuncts)}; } else if(expr.id() == ID_X) { @@ -220,7 +218,7 @@ exprt wl_instantiatet::instantiate_rec(exprt expr, const mp_integer &t) const return instantiate_rec(to_X_expr(expr).op(), next); } else - return true_exprt(); // works on NNF only + return {no_timeframes - 1, true_exprt()}; // works on NNF only } else if(expr.id() == ID_sva_eventually) { @@ -238,14 +236,14 @@ exprt wl_instantiatet::instantiate_rec(exprt expr, const mp_integer &t) const // This is weak, and passes if any of the timeframes // does not exist. if(t + lower >= no_timeframes || t + upper >= no_timeframes) - return true_exprt(); + return {no_timeframes - 1, true_exprt()}; exprt::operandst disjuncts = {}; for(mp_integer u = t + lower; u <= t + upper; ++u) - disjuncts.push_back(instantiate_rec(op, u)); + disjuncts.push_back(instantiate_rec(op, u).second); - return disjunction(disjuncts); + return {no_timeframes - 1, disjunction(disjuncts)}; } else if( expr.id() == ID_sva_s_eventually || expr.id() == ID_F || expr.id() == ID_AF) @@ -273,13 +271,13 @@ exprt wl_instantiatet::instantiate_rec(exprt expr, const mp_integer &t) const for(mp_integer j = k; j <= i; ++j) { - disjuncts.push_back(instantiate_rec(p, j)); + disjuncts.push_back(instantiate_rec(p, j).second); } conjuncts.push_back(disjunction(disjuncts)); } - return conjunction(conjuncts); + return {no_timeframes - 1, conjunction(conjuncts)}; } else if(expr.id()==ID_sva_until || expr.id()==ID_sva_s_until) @@ -292,19 +290,19 @@ exprt wl_instantiatet::instantiate_rec(exprt expr, const mp_integer &t) const // we expand: p U q <=> q || (p && X(p U q)) exprt tmp_q = to_binary_expr(expr).op1(); - tmp_q = instantiate_rec(tmp_q, t); + tmp_q = instantiate_rec(tmp_q, t).second; exprt expansion = to_binary_expr(expr).op0(); - expansion = instantiate_rec(expansion, t); + expansion = instantiate_rec(expansion, t).second; const auto next = t + 1; if(next < no_timeframes) { - expansion = and_exprt(expansion, instantiate_rec(expr, next)); + expansion = and_exprt(expansion, instantiate_rec(expr, next).second); } - return or_exprt(tmp_q, expansion); + return {no_timeframes - 1, or_exprt(tmp_q, expansion)}; } else if(expr.id()==ID_sva_until_with || expr.id()==ID_sva_s_until_with) @@ -335,7 +333,7 @@ exprt wl_instantiatet::instantiate_rec(exprt expr, const mp_integer &t) const if(ticks > t) { // return the 'default value' for the type - return default_value(verilog_past.type()); + return {t, default_value(verilog_past.type())}; } else { @@ -344,9 +342,15 @@ exprt wl_instantiatet::instantiate_rec(exprt expr, const mp_integer &t) const } else { + mp_integer max = t; for(auto &op : expr.operands()) - op = instantiate_rec(op, t); - return expr; + { + auto tmp = instantiate_rec(op, t); + op = tmp.second; + max = std::max(max, tmp.first); + } + + return {max, expr}; } } @@ -386,5 +390,27 @@ exprt instantiate( const namespacet &ns) { wl_instantiatet wl_instantiate(no_timeframes, ns); - return wl_instantiate(expr, t); + return wl_instantiate(expr, t).second; +} + +/*******************************************************************\ + +Function: instantiate_property + + Inputs: + + Outputs: + + Purpose: + +\*******************************************************************/ + +std::pair instantiate_property( + const exprt &expr, + const mp_integer ¤t, + const mp_integer &no_timeframes, + const namespacet &ns) +{ + wl_instantiatet wl_instantiate(no_timeframes, ns); + return wl_instantiate(expr, current); } diff --git a/src/trans-word-level/instantiate_word_level.h b/src/trans-word-level/instantiate_word_level.h index 7c05c087f..8ff13c927 100644 --- a/src/trans-word-level/instantiate_word_level.h +++ b/src/trans-word-level/instantiate_word_level.h @@ -20,6 +20,12 @@ exprt instantiate( const mp_integer &no_timeframes, const namespacet &); +std::pair instantiate_property( + const exprt &, + const mp_integer ¤t, + const mp_integer &no_timeframes, + const namespacet &); + std::string timeframe_identifier(const mp_integer &timeframe, const irep_idt &identifier); diff --git a/src/trans-word-level/property.cpp b/src/trans-word-level/property.cpp index cc9d7ba20..cb4cf73f8 100644 --- a/src/trans-word-level/property.cpp +++ b/src/trans-word-level/property.cpp @@ -78,54 +78,6 @@ bool bmc_supports_property(const exprt &expr) /*******************************************************************\ -Function: max_property_obligation - - Inputs: - - Outputs: - - Purpose: - -\*******************************************************************/ - -static void property_obligations_rec( - const exprt &property_expr, - decision_proceduret &, - const mp_integer ¤t, - const mp_integer &no_timeframes, - const namespacet &, - std::map &obligations); - -static std::pair max_property_obligation( - const exprt &property_expr, - decision_proceduret &solver, - const mp_integer ¤t, - const mp_integer &no_timeframes, - const namespacet &ns) -{ - // Generate one obligation, equivalent to the conjunction - // for the maximum timeframe. - - std::map obligations; - - property_obligations_rec( - property_expr, solver, current, no_timeframes, ns, obligations); - - exprt::operandst conjuncts; - mp_integer max_timeframe = 0; - - for(auto &[timeframe, exprs] : obligations) - { - max_timeframe = std::max(max_timeframe, timeframe); - for(auto &conjunct : exprs) - conjuncts.push_back(conjunct); - } - - return std::pair{max_timeframe, conjunction(conjuncts)}; -} - -/*******************************************************************\ - Function: property_obligations_rec Inputs: @@ -146,17 +98,7 @@ static void property_obligations_rec( { PRECONDITION(current >= 0 && current < no_timeframes); - if(property_expr.id() == ID_X) - { - auto next = current + 1; - if(next < no_timeframes) - { - auto &op = to_X_expr(property_expr).op(); - property_obligations_rec( - op, solver, next, no_timeframes, ns, obligations); - } - } - else if( + if( property_expr.id() == ID_AG || property_expr.id() == ID_G || property_expr.id() == ID_sva_always) { @@ -254,28 +196,10 @@ static void property_obligations_rec( property_obligations_rec( op, solver, current, no_timeframes, ns, obligations); } - else if(property_expr.id() == ID_or) - { - // generate one obligation, equivalent to the disjunction, - // for the maximum timeframe - mp_integer max_timeframe = 0; - exprt::operandst disjuncts; - - for(auto &op : to_or_expr(property_expr).operands()) - { - auto obligation = - max_property_obligation(op, solver, current, no_timeframes, ns); - max_timeframe = std::max(max_timeframe, obligation.first); - disjuncts.push_back(obligation.second); - } - - obligations[max_timeframe].push_back(disjunction(disjuncts)); - } else { - // current state property - exprt tmp = instantiate(property_expr, current, no_timeframes, ns); - obligations[current].push_back(tmp); + auto tmp = instantiate_property(property_expr, current, no_timeframes, ns); + obligations[tmp.first].push_back(tmp.second); } }