Skip to content

Commit 83b210e

Browse files
committed
C++ front-end: support explicit type conversion with braced-init-list
Now follows the grammar described in the C++ standard: Explicit type conversions are postfix expressions. This required adding rules for C++ cast expressions, which in turn simplifies type checking for we no longer treat them as function calls with template arguments.
1 parent f529e30 commit 83b210e

File tree

9 files changed

+336
-189
lines changed

9 files changed

+336
-189
lines changed
+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
#include <cassert>
2+
3+
int main(int argc, char * argv[])
4+
{
5+
struct S {
6+
S() : x(42)
7+
{
8+
}
9+
10+
int x;
11+
};
12+
S s = S{};
13+
14+
__CPROVER_assert(s.x == 42, "");
15+
assert(s.x == 42);
16+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
CORE
2+
main.cpp
3+
4+
^EXIT=0$
5+
^SIGNAL=0$
6+
^VERIFICATION SUCCESSFUL$
7+
--
8+
^warning: ignoring
+24
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
struct __invoke_memfun_ref {};
2+
constexpr bool __call_is_nt(__invoke_memfun_ref)
3+
{
4+
return false;
5+
}
6+
7+
template<typename _Result>
8+
struct __call_is_nothrow
9+
{
10+
constexpr static bool is_nt =
11+
__call_is_nt(typename _Result::__invoke_type{});
12+
};
13+
14+
int main(int argc, char * argv[])
15+
{
16+
struct S {
17+
S() : x(42)
18+
{
19+
}
20+
21+
int x;
22+
};
23+
S s = S{};
24+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
CORE
2+
main.cpp
3+
-std=c++11
4+
^EXIT=0$
5+
^SIGNAL=0$
6+
--
7+
^warning: ignoring
8+
^CONVERSION ERROR$

src/ansi-c/parser.y

+4
Original file line numberDiff line numberDiff line change
@@ -279,6 +279,10 @@ int yyansi_cerror(const std::string &error);
279279
%token TOK_MSC_IF_EXISTS "__if_exists"
280280
%token TOK_MSC_IF_NOT_EXISTS "__if_not_exists"
281281
%token TOK_UNDERLYING_TYPE "__underlying_type"
282+
%token TOK_DYNAMIC_CAST "dynamic_cast"
283+
%token TOK_STATIC_CAST "static_cast"
284+
%token TOK_REINTERPRET_CAST "reinterpret_cast"
285+
%token TOK_CONST_CAST "const_cast"
282286

283287
/*** priority, associativity, etc. definitions **************************/
284288

src/ansi-c/scanner.l

+5
Original file line numberDiff line numberDiff line change
@@ -1316,6 +1316,11 @@ __decltype { if(PARSER.cpp98 &&
13161316
return make_identifier();
13171317
}
13181318

1319+
"dynamic_cast" { return cpp98_keyword(TOK_DYNAMIC_CAST); }
1320+
"static_cast" { return cpp98_keyword(TOK_STATIC_CAST); }
1321+
"reinterpret_cast" { return cpp98_keyword(TOK_REINTERPRET_CAST); }
1322+
"const_cast" { return cpp98_keyword(TOK_CONST_CAST); }
1323+
13191324
{CPROVER_PREFIX}"atomic" { loc(); return TOK_CPROVER_ATOMIC; }
13201325
{CPROVER_PREFIX}"forall" { loc(); return TOK_FORALL; }
13211326
{CPROVER_PREFIX}"exists" { loc(); return TOK_EXISTS; }

src/cpp/cpp_is_pod.cpp

+4
Original file line numberDiff line numberDiff line change
@@ -73,6 +73,10 @@ bool cpp_typecheckt::cpp_is_pod(const typet &type) const
7373
{
7474
return cpp_is_pod(to_array_type(type).element_type());
7575
}
76+
else if(type.id()==ID_vector)
77+
{
78+
return cpp_is_pod(to_vector_type(type).element_type());
79+
}
7680
else if(type.id()==ID_pointer)
7781
{
7882
if(is_reference(type)) // references are not PODs

src/cpp/cpp_typecheck_expr.cpp

+12-66
Original file line numberDiff line numberDiff line change
@@ -127,6 +127,13 @@ void cpp_typecheckt::typecheck_expr_main(exprt &expr)
127127
{
128128
expr.type().id(ID_initializer_list);
129129
}
130+
else if(expr.id() == ID_const_cast ||
131+
expr.id() == ID_dynamic_cast ||
132+
expr.id() == ID_reinterpret_cast ||
133+
expr.id() == ID_static_cast)
134+
{
135+
typecheck_cast_expr(expr);
136+
}
130137
else
131138
c_typecheck_baset::typecheck_expr_main(expr);
132139
}
@@ -967,13 +974,8 @@ void cpp_typecheckt::typecheck_expr_explicit_constructor_call(exprt &expr)
967974
}
968975
else
969976
{
970-
CHECK_RETURN(expr.type().id() == ID_struct);
971-
972-
struct_tag_typet tag(expr.type().get(ID_name));
973-
tag.add_source_location() = expr.source_location();
974-
975977
exprt e=expr;
976-
new_temporary(e.source_location(), tag, e.operands(), expr);
978+
new_temporary(e.source_location(), e.type(), e.operands(), expr);
977979
}
978980
}
979981

@@ -1275,53 +1277,20 @@ void cpp_typecheckt::typecheck_expr_ptrmember(
12751277

12761278
void cpp_typecheckt::typecheck_cast_expr(exprt &expr)
12771279
{
1278-
side_effect_expr_function_callt e =
1279-
to_side_effect_expr_function_call(expr);
1280-
1281-
if(e.arguments().size() != 1)
1280+
if(expr.operands().size() != 1)
12821281
{
12831282
error().source_location=expr.find_source_location();
12841283
error() << "cast expressions expect one operand" << eom;
12851284
throw 0;
12861285
}
12871286

1288-
exprt &f_op=e.function();
1289-
exprt &cast_op=e.arguments().front();
1287+
exprt &cast_op = to_unary_expr(expr).op();
12901288

12911289
add_implicit_dereference(cast_op);
12921290

1293-
const irep_idt &id=
1294-
f_op.get_sub().front().get(ID_identifier);
1295-
1296-
if(f_op.get_sub().size()!=2 ||
1297-
f_op.get_sub()[1].id()!=ID_template_args)
1298-
{
1299-
error().source_location=expr.find_source_location();
1300-
error() << id << " expects template argument" << eom;
1301-
throw 0;
1302-
}
1303-
1304-
irept &template_arguments=f_op.get_sub()[1].add(ID_arguments);
1305-
1306-
if(template_arguments.get_sub().size()!=1)
1307-
{
1308-
error().source_location=expr.find_source_location();
1309-
error() << id << " expects one template argument" << eom;
1310-
throw 0;
1311-
}
1312-
1313-
irept &template_arg=template_arguments.get_sub().front();
1314-
1315-
if(template_arg.id() != ID_type && template_arg.id() != ID_ambiguous)
1316-
{
1317-
error().source_location=expr.find_source_location();
1318-
error() << id << " expects a type as template argument" << eom;
1319-
throw 0;
1320-
}
1321-
1322-
typet &type=static_cast<typet &>(
1323-
template_arguments.get_sub().front().add(ID_type));
1291+
const irep_idt &id = expr.id();
13241292

1293+
typet &type = expr.type();
13251294
typecheck_type(type);
13261295

13271296
source_locationt source_location=expr.source_location();
@@ -1415,21 +1384,6 @@ void cpp_typecheckt::typecheck_expr_cpp_name(
14151384
}
14161385
}
14171386

1418-
if(expr.get_sub().size()>=1 &&
1419-
expr.get_sub().front().id()==ID_name)
1420-
{
1421-
const irep_idt &id=expr.get_sub().front().get(ID_identifier);
1422-
1423-
if(id==ID_const_cast ||
1424-
id==ID_dynamic_cast ||
1425-
id==ID_reinterpret_cast ||
1426-
id==ID_static_cast)
1427-
{
1428-
expr.id(ID_cast_expression);
1429-
return;
1430-
}
1431-
}
1432-
14331387
exprt symbol_expr=
14341388
resolve(
14351389
to_cpp_name(expr),
@@ -1556,14 +1510,6 @@ void cpp_typecheckt::typecheck_side_effect_function_call(
15561510

15571511
return;
15581512
}
1559-
else if(expr.function().id() == ID_cast_expression)
1560-
{
1561-
// These are not really function calls,
1562-
// but usually just type adjustments.
1563-
typecheck_cast_expr(expr);
1564-
add_implicit_dereference(expr);
1565-
return;
1566-
}
15671513
else if(expr.function().id() == ID_cpp_dummy_destructor)
15681514
{
15691515
// these don't do anything, e.g., (char*)->~char()

0 commit comments

Comments
 (0)