Skip to content

Commit 10d922c

Browse files
Merge pull request #8089 from thomasspriggs/tas/smt_stl
Use the STL versions of `variant` and `optional` in incremental smt2 decision procedure
2 parents c0fe5a8 + 062d6c6 commit 10d922c

15 files changed

+46
-57
lines changed

src/solvers/smt2_incremental/ast/smt_commands.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ class smt_commandt : protected irept
1515
{
1616
public:
1717
// smt_commandt does not support the notion of an empty / null state. Use
18-
// optionalt<smt_commandt> instead if an empty command is required.
18+
// std::optional<smt_commandt> instead if an empty command is required.
1919
smt_commandt() = delete;
2020

2121
using irept::pretty;

src/solvers/smt2_incremental/ast/smt_index.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ class smt_indext : protected irept
1414
{
1515
public:
1616
// smt_indext does not support the notion of an empty / null state. Use
17-
// optionalt<smt_indext> instead if an empty index is required.
17+
// std::optional<smt_indext> instead if an empty index is required.
1818
smt_indext() = delete;
1919

2020
using irept::pretty;

src/solvers/smt2_incremental/ast/smt_logics.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ class smt_logict : protected irept
1111
{
1212
public:
1313
// smt_logict does not support the notion of an empty / null state. Use
14-
// optionalt<smt_logict> instead if an empty logic is required.
14+
// std::optional<smt_logict> instead if an empty logic is required.
1515
smt_logict() = delete;
1616

1717
using irept::pretty;

src/solvers/smt2_incremental/ast/smt_options.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ class smt_optiont : protected irept
1111
{
1212
public:
1313
// smt_optiont does not support the notion of an empty / null state. Use
14-
// optionalt<smt_optiont> instead if an empty option is required.
14+
// std::optional<smt_optiont> instead if an empty option is required.
1515
smt_optiont() = delete;
1616

1717
using irept::pretty;

src/solvers/smt2_incremental/ast/smt_responses.h

+2-2
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ class smt_responset : protected irept
1111
{
1212
public:
1313
// smt_responset does not support the notion of an empty / null state. Use
14-
// optionalt<smt_responset> instead if an empty response is required.
14+
// std::optional<smt_responset> instead if an empty response is required.
1515
smt_responset() = delete;
1616

1717
using irept::pretty;
@@ -36,7 +36,7 @@ class smt_check_sat_response_kindt : protected irept
3636
{
3737
public:
3838
// smt_responset does not support the notion of an empty / null state. Use
39-
// optionalt<smt_responset> instead if an empty response is required.
39+
// std::optional<smt_responset> instead if an empty response is required.
4040
smt_check_sat_response_kindt() = delete;
4141

4242
using irept::pretty;

src/solvers/smt2_incremental/ast/smt_sorts.cpp

+2-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,8 @@
2424

2525
#define SORT_ID(the_id) \
2626
template <> \
27-
optionalt<smt_##the_id##_sortt> smt_sortt::cast<smt_##the_id##_sortt>() && \
27+
std::optional<smt_##the_id##_sortt> smt_sortt::cast<smt_##the_id##_sortt>() \
28+
&& \
2829
{ \
2930
if(id() == ID_smt_##the_id##_sort) \
3031
return {std::move(*static_cast<const smt_##the_id##_sortt *>(this))}; \

src/solvers/smt2_incremental/ast/smt_sorts.h

+3-3
Original file line numberDiff line numberDiff line change
@@ -8,8 +8,8 @@
88
#define CPROVER_SOLVERS_SMT2_INCREMENTAL_SMT_SORTS_H
99

1010
#include <util/irep.h>
11-
#include <util/optional.h>
1211

12+
#include <optional>
1313
#include <type_traits>
1414

1515
class smt_sort_const_downcast_visitort;
@@ -18,7 +18,7 @@ class smt_sortt : protected irept
1818
{
1919
public:
2020
// smt_sortt does not support the notion of an empty / null state. Use
21-
// optionalt<smt_sortt> instead if an empty sort is required.
21+
// std::optional<smt_sortt> instead if an empty sort is required.
2222
smt_sortt() = delete;
2323

2424
using irept::pretty;
@@ -47,7 +47,7 @@ class smt_sortt : protected irept
4747
const sub_classt *cast() const &;
4848

4949
template <typename sub_classt>
50-
optionalt<sub_classt> cast() &&;
50+
std::optional<sub_classt> cast() &&;
5151

5252
protected:
5353
using irept::irept;

src/solvers/smt2_incremental/ast/smt_terms.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@ class smt_termt : protected irept, private smt_sortt::storert<smt_termt>
2121
{
2222
public:
2323
// smt_termt does not support the notion of an empty / null state. Use
24-
// optionalt<smt_termt> instead if an empty term is required.
24+
// std::optional<smt_termt> instead if an empty term is required.
2525
smt_termt() = delete;
2626

2727
using irept::pretty;

src/solvers/smt2_incremental/construct_value_expr_from_smt.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ class value_expr_from_smt_factoryt : public smt_term_const_downcast_visitort
1616
private:
1717
const typet &type_to_construct;
1818
const namespacet &ns;
19-
optionalt<exprt> result;
19+
std::optional<exprt> result;
2020

2121
explicit value_expr_from_smt_factoryt(
2222
const typet &type_to_construct,

src/solvers/smt2_incremental/convert_expr_to_smt.cpp

+3-3
Original file line numberDiff line numberDiff line change
@@ -206,7 +206,7 @@ struct sort_based_cast_to_bit_vector_convertert final
206206
const smt_termt &from_term;
207207
const typet &from_type;
208208
const bitvector_typet &to_type;
209-
optionalt<smt_termt> result;
209+
std::optional<smt_termt> result;
210210

211211
sort_based_cast_to_bit_vector_convertert(
212212
const smt_termt &from_term,
@@ -299,7 +299,7 @@ static smt_termt convert_expr_to_smt(
299299
struct sort_based_literal_convertert : public smt_sort_const_downcast_visitort
300300
{
301301
const constant_exprt &member_input;
302-
optionalt<smt_termt> result;
302+
std::optional<smt_termt> result;
303303

304304
explicit sort_based_literal_convertert(const constant_exprt &input)
305305
: member_input{input}
@@ -591,7 +591,7 @@ static smt_termt convert_relational_to_smt(
591591
binary_relation.pretty());
592592
}
593593

594-
static optionalt<smt_termt> try_relational_conversion(
594+
static std::optional<smt_termt> try_relational_conversion(
595595
const exprt &expr,
596596
const sub_expression_mapt &converted)
597597
{

src/solvers/smt2_incremental/encoding/struct_encoding.cpp

+3-2
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@
1515

1616
#include <algorithm>
1717
#include <numeric>
18+
#include <optional>
1819
#include <queue>
1920

2021
struct_encodingt::struct_encodingt(const namespacet &ns)
@@ -33,7 +34,7 @@ struct_encodingt::~struct_encodingt() = default;
3334
/// If the given \p type needs re-encoding as a bit-vector then this function
3435
/// \returns the width of the new bitvector type. The width calculation is
3536
/// delegated to \p boolbv_width.
36-
static optionalt<std::size_t>
37+
static std::optional<std::size_t>
3738
needs_width(const typet &type, const boolbv_widtht &boolbv_width)
3839
{
3940
if(const auto struct_tag = type_try_dynamic_cast<struct_tag_typet>(type))
@@ -222,7 +223,7 @@ exprt struct_encodingt::encode(exprt expr) const
222223
if(can_cast_type<struct_tag_typet>(current.type()))
223224
current = ::encode(*with_expr, ns);
224225
current.type() = encode(current.type());
225-
optionalt<exprt> update;
226+
std::optional<exprt> update;
226227
if(const auto struct_expr = expr_try_dynamic_cast<struct_exprt>(current))
227228
update = ::encode(*struct_expr);
228229
if(const auto union_expr = expr_try_dynamic_cast<union_exprt>(current))

src/solvers/smt2_incremental/response_or_error.h

+7-20
Original file line numberDiff line numberDiff line change
@@ -4,9 +4,9 @@
44
#define CPROVER_SOLVERS_SMT2_INCREMENTAL_RESPONSE_OR_ERROR_H
55

66
#include <util/invariant.h>
7-
#include <util/optional.h>
87

98
#include <string>
9+
#include <variant>
1010
#include <vector>
1111

1212
/// Holds either a valid parsed response or response sub-tree of type \tparam
@@ -16,49 +16,36 @@ template <class smtt>
1616
class response_or_errort final
1717
{
1818
public:
19-
explicit response_or_errort(smtt smt) : smt{std::move(smt)}
19+
explicit response_or_errort(smtt smt) : smt_or_messages{std::move(smt)}
2020
{
2121
}
2222

2323
explicit response_or_errort(std::string message)
24-
: messages{std::move(message)}
24+
: smt_or_messages{std::vector<std::string>{std::move(message)}}
2525
{
2626
}
2727

2828
explicit response_or_errort(std::vector<std::string> messages)
29-
: messages{std::move(messages)}
29+
: smt_or_messages{std::move(messages)}
3030
{
3131
}
3232

3333
/// \brief Gets the smt response if the response is valid, or returns nullptr
3434
/// otherwise.
3535
const smtt *get_if_valid() const
3636
{
37-
INVARIANT(
38-
smt.has_value() == messages.empty(),
39-
"The response_or_errort class must be in the valid state or error state, "
40-
"exclusively.");
41-
return smt.has_value() ? &smt.value() : nullptr;
37+
return std::get_if<smtt>(&smt_or_messages);
4238
}
4339

4440
/// \brief Gets the error messages if the response is invalid, or returns
4541
/// nullptr otherwise.
4642
const std::vector<std::string> *get_if_error() const
4743
{
48-
INVARIANT(
49-
smt.has_value() == messages.empty(),
50-
"The response_or_errort class must be in the valid state or error state, "
51-
"exclusively.");
52-
return smt.has_value() ? nullptr : &messages;
44+
return std::get_if<std::vector<std::string>>(&smt_or_messages);
5345
}
5446

5547
private:
56-
// The below two fields could be a single `std::variant` field, if there was
57-
// an implementation of it available in the cbmc repository. However at the
58-
// time of writing we are targeting C++11, `std::variant` was introduced in
59-
// C++17 and we have no backported version.
60-
optionalt<smtt> smt;
61-
std::vector<std::string> messages;
48+
std::variant<smtt, std::vector<std::string>> smt_or_messages;
6249
};
6350

6451
#endif // CPROVER_SOLVERS_SMT2_INCREMENTAL_RESPONSE_OR_ERROR_H

src/solvers/smt2_incremental/smt2_incremental_decision_procedure.cpp

+7-7
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,7 @@ static smt_responset get_response_to_command(
4444

4545
/// Returns a message string describing the problem in the case where the
4646
/// response from the solver is an error status. Returns empty otherwise.
47-
static optionalt<std::string>
47+
static std::optional<std::string>
4848
get_problem_messages(const smt_responset &response)
4949
{
5050
if(const auto error = response.cast<smt_error_responset>())
@@ -413,7 +413,7 @@ exprt smt2_incremental_decision_proceduret::handle(const exprt &expr)
413413
return expr;
414414
}
415415

416-
optionalt<smt_termt>
416+
std::optional<smt_termt>
417417
smt2_incremental_decision_proceduret::get_identifier(const exprt &expr) const
418418
{
419419
// Lookup the non-lowered form first.
@@ -437,7 +437,7 @@ smt2_incremental_decision_proceduret::get_identifier(const exprt &expr) const
437437
return {};
438438
}
439439

440-
optionalt<exprt> smt2_incremental_decision_proceduret::get_expr(
440+
std::optional<exprt> smt2_incremental_decision_proceduret::get_expr(
441441
const smt_termt &array,
442442
const array_typet &type) const
443443
{
@@ -467,7 +467,7 @@ optionalt<exprt> smt2_incremental_decision_proceduret::get_expr(
467467
return array_exprt{elements, type};
468468
}
469469

470-
optionalt<exprt> smt2_incremental_decision_proceduret::get_expr(
470+
std::optional<exprt> smt2_incremental_decision_proceduret::get_expr(
471471
const smt_termt &struct_term,
472472
const struct_tag_typet &type) const
473473
{
@@ -478,7 +478,7 @@ optionalt<exprt> smt2_incremental_decision_proceduret::get_expr(
478478
return {struct_encoding.decode(*encoded_result, type)};
479479
}
480480

481-
optionalt<exprt> smt2_incremental_decision_proceduret::get_expr(
481+
std::optional<exprt> smt2_incremental_decision_proceduret::get_expr(
482482
const smt_termt &union_term,
483483
const union_tag_typet &type) const
484484
{
@@ -489,7 +489,7 @@ optionalt<exprt> smt2_incremental_decision_proceduret::get_expr(
489489
return {struct_encoding.decode(*encoded_result, type)};
490490
}
491491

492-
optionalt<exprt> smt2_incremental_decision_proceduret::get_expr(
492+
std::optional<exprt> smt2_incremental_decision_proceduret::get_expr(
493493
const smt_termt &descriptor,
494494
const typet &type) const
495495
{
@@ -556,7 +556,7 @@ exprt smt2_incremental_decision_proceduret::get(const exprt &expr) const
556556
log.conditional_output(log.debug(), [&](messaget::mstreamt &debug) {
557557
debug << "`get` - \n " + expr.pretty(2, 0) << messaget::eom;
558558
});
559-
auto descriptor = [&]() -> optionalt<smt_termt> {
559+
auto descriptor = [&]() -> std::optional<smt_termt> {
560560
if(const auto index_expr = expr_try_dynamic_cast<index_exprt>(expr))
561561
{
562562
const auto array = get_identifier(index_expr->array());

src/solvers/smt2_incremental/smt2_incremental_decision_procedure.h

+5-5
Original file line numberDiff line numberDiff line change
@@ -56,13 +56,13 @@ class smt2_incremental_decision_proceduret final
5656
/// Gets the value of \p descriptor from the solver and returns the solver
5757
/// response expressed as an exprt of type \p type. This is an implementation
5858
/// detail of the `get(exprt)` member function.
59-
optionalt<exprt>
59+
std::optional<exprt>
6060
get_expr(const smt_termt &descriptor, const typet &type) const;
61-
optionalt<exprt>
61+
std::optional<exprt>
6262
get_expr(const smt_termt &struct_term, const struct_tag_typet &type) const;
63-
optionalt<exprt>
63+
std::optional<exprt>
6464
get_expr(const smt_termt &union_term, const union_tag_typet &type) const;
65-
optionalt<exprt>
65+
std::optional<exprt>
6666
get_expr(const smt_termt &array, const array_typet &type) const;
6767

6868
protected:
@@ -117,7 +117,7 @@ class smt2_incremental_decision_proceduret final
117117
/// possible expression forms by expressing these in terms of the remaining
118118
/// language features.
119119
exprt lower(exprt expression) const;
120-
optionalt<smt_termt> get_identifier(const exprt &expr) const;
120+
std::optional<smt_termt> get_identifier(const exprt &expr) const;
121121

122122
/// Namespace for looking up the expressions which symbol_exprts relate to.
123123
/// This includes the symbols defined outside of the decision procedure but

src/solvers/smt2_incremental/smt_response_validation.cpp

+8-8
Original file line numberDiff line numberDiff line change
@@ -125,7 +125,7 @@ validate_string_literal(const irept &parse_tree)
125125
/// \note: Because this kind of response does not start with an identifying
126126
/// keyword, it will be considered that the response is intended to be a
127127
/// get-value response if it is composed of a collection of one or more pairs.
128-
static optionalt<response_or_errort<smt_responset>>
128+
static std::optional<response_or_errort<smt_responset>>
129129
valid_smt_error_response(const irept &parse_tree)
130130
{
131131
// Check if the parse tree looks to be an error response.
@@ -162,7 +162,7 @@ static bool all_subs_are_pairs(const irept &parse_tree)
162162

163163
/// Checks for valid bit vector constants of the form `(_ bv(value) (width))`
164164
/// for example - `(_ bv4 64)`.
165-
static optionalt<smt_termt>
165+
static std::optional<smt_termt>
166166
valid_smt_indexed_bit_vector(const irept &parse_tree)
167167
{
168168
if(parse_tree.get_sub().size() != 3)
@@ -189,7 +189,7 @@ valid_smt_indexed_bit_vector(const irept &parse_tree)
189189
return smt_bit_vector_constant_termt{value, bit_width};
190190
}
191191

192-
static optionalt<smt_termt> valid_smt_bool(const irept &parse_tree)
192+
static std::optional<smt_termt> valid_smt_bool(const irept &parse_tree)
193193
{
194194
if(!parse_tree.get_sub().empty())
195195
return {};
@@ -200,7 +200,7 @@ static optionalt<smt_termt> valid_smt_bool(const irept &parse_tree)
200200
return {};
201201
}
202202

203-
static optionalt<smt_termt> valid_smt_binary(const std::string &text)
203+
static std::optional<smt_termt> valid_smt_binary(const std::string &text)
204204
{
205205
static const std::regex binary_format{"#b[01]+"};
206206
if(!std::regex_match(text, binary_format))
@@ -211,7 +211,7 @@ static optionalt<smt_termt> valid_smt_binary(const std::string &text)
211211
return {smt_bit_vector_constant_termt{value, width}};
212212
}
213213

214-
static optionalt<smt_termt> valid_smt_hex(const std::string &text)
214+
static std::optional<smt_termt> valid_smt_hex(const std::string &text)
215215
{
216216
static const std::regex hex_format{"#x[0-9A-Za-z]+"};
217217
if(!std::regex_match(text, hex_format))
@@ -225,7 +225,7 @@ static optionalt<smt_termt> valid_smt_hex(const std::string &text)
225225
return {smt_bit_vector_constant_termt{value, width}};
226226
}
227227

228-
static optionalt<smt_termt>
228+
static std::optional<smt_termt>
229229
valid_smt_bit_vector_constant(const irept &parse_tree)
230230
{
231231
if(const auto indexed = valid_smt_indexed_bit_vector(parse_tree))
@@ -240,7 +240,7 @@ valid_smt_bit_vector_constant(const irept &parse_tree)
240240
return {};
241241
}
242242

243-
static optionalt<response_or_errort<smt_termt>> try_select_validation(
243+
static std::optional<response_or_errort<smt_termt>> try_select_validation(
244244
const irept &parse_tree,
245245
const std::unordered_map<irep_idt, smt_identifier_termt> &identifier_table)
246246
{
@@ -321,7 +321,7 @@ validate_valuation_pair(
321321
/// \note: Because this kind of response does not start with an identifying
322322
/// keyword, it will be considered that the response is intended to be a
323323
/// get-value response if it is composed of a collection of one or more pairs.
324-
static optionalt<response_or_errort<smt_responset>>
324+
static std::optional<response_or_errort<smt_responset>>
325325
valid_smt_get_value_response(
326326
const irept &parse_tree,
327327
const std::unordered_map<irep_idt, smt_identifier_termt> &identifier_table)

0 commit comments

Comments
 (0)