Skip to content

Commit e5f6835

Browse files
committed
C and C++ parsers: construct with message handler
This both avoids an object of static lifetime as well as it fixes the (transitive) use of the deprecated messaget() constructor. The C scanner is now fully reentrant.
1 parent e18fead commit e5f6835

16 files changed

+775
-722
lines changed
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
KNOWNBUG
1+
FUTURE
22
main.c
33
--localize-faults
44
^EXIT=10$
@@ -8,4 +8,6 @@ line 9 function main$
88
--
99
--
1010
CBMC wrongly reports line 7 as the faulty statement when instead it should be
11-
line 9.
11+
line 9. The test is marked as "FUTURE" as it will sometimes pass, depending on
12+
what model the solver produces. It would, therefore, break our checking of
13+
"KNOWNBUG" tests.

src/ansi-c/ansi_c_language.cpp

+6-12
Original file line numberDiff line numberDiff line change
@@ -71,35 +71,28 @@ bool ansi_c_languaget::parse(
7171
ansi_c_internal_additions(code, config.ansi_c.ts_18661_3_Floatn_types);
7272
std::istringstream codestr(code);
7373

74-
ansi_c_parser.clear();
74+
ansi_c_parsert ansi_c_parser{message_handler};
7575
ansi_c_parser.set_file(ID_built_in);
7676
ansi_c_parser.in=&codestr;
77-
ansi_c_parser.log.set_message_handler(message_handler);
7877
ansi_c_parser.for_has_scope=config.ansi_c.for_has_scope;
7978
ansi_c_parser.ts_18661_3_Floatn_types=config.ansi_c.ts_18661_3_Floatn_types;
8079
ansi_c_parser.cpp98=false; // it's not C++
8180
ansi_c_parser.cpp11=false; // it's not C++
8281
ansi_c_parser.mode=config.ansi_c.mode;
8382

84-
ansi_c_scanner_init();
85-
8683
bool result=ansi_c_parser.parse();
8784

8885
if(!result)
8986
{
9087
ansi_c_parser.set_line_no(0);
9188
ansi_c_parser.set_file(path);
9289
ansi_c_parser.in=&i_preprocessed;
93-
ansi_c_scanner_init();
9490
result=ansi_c_parser.parse();
9591
}
9692

9793
// save result
9894
parse_tree.swap(ansi_c_parser.parse_tree);
9995

100-
// save some memory
101-
ansi_c_parser.clear();
102-
10396
return result;
10497
}
10598

@@ -197,13 +190,14 @@ bool ansi_c_languaget::to_expr(
197190

198191
// parsing
199192

200-
ansi_c_parser.clear();
193+
ansi_c_parsert ansi_c_parser{message_handler};
201194
ansi_c_parser.set_file(irep_idt());
202195
ansi_c_parser.in=&i_preprocessed;
203-
ansi_c_parser.log.set_message_handler(message_handler);
204-
ansi_c_parser.mode=config.ansi_c.mode;
196+
ansi_c_parser.for_has_scope = config.ansi_c.for_has_scope;
205197
ansi_c_parser.ts_18661_3_Floatn_types=config.ansi_c.ts_18661_3_Floatn_types;
206-
ansi_c_scanner_init();
198+
ansi_c_parser.cpp98 = false; // it's not C++
199+
ansi_c_parser.cpp11 = false; // it's not C++
200+
ansi_c_parser.mode = config.ansi_c.mode;
207201

208202
bool result=ansi_c_parser.parse();
209203

src/ansi-c/ansi_c_parser.cpp

+16-9
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,22 @@ Author: Daniel Kroening, [email protected]
1010

1111
#include "c_storage_spec.h"
1212

13-
ansi_c_parsert ansi_c_parser;
13+
int yyansi_clex_init_extra(ansi_c_parsert *, void **);
14+
int yyansi_clex_destroy(void *);
15+
int yyansi_cparse(ansi_c_parsert &, void *);
16+
void yyansi_cset_debug(int, void *);
17+
18+
bool ansi_c_parsert::parse()
19+
{
20+
void *scanner;
21+
yyansi_clex_init_extra(this, &scanner);
22+
#ifdef ANSI_C_DEBUG
23+
yyansi_cset_debug(1, scanner);
24+
#endif
25+
bool parse_fail = yyansi_cparse(*this, scanner) != 0;
26+
yyansi_clex_destroy(scanner);
27+
return parse_fail;
28+
}
1429

1530
ansi_c_id_classt ansi_c_parsert::lookup(
1631
const irep_idt &base_name,
@@ -73,14 +88,6 @@ void ansi_c_parsert::add_tag_with_body(irept &tag)
7388
}
7489
}
7590

76-
extern char *yyansi_ctext;
77-
78-
int yyansi_cerror(const std::string &error)
79-
{
80-
ansi_c_parser.parse_error(error, yyansi_ctext);
81-
return 0;
82-
}
83-
8491
void ansi_c_parsert::add_declarator(
8592
exprt &declaration,
8693
irept &declarator)

src/ansi-c/ansi_c_parser.h

+8-14
Original file line numberDiff line numberDiff line change
@@ -18,15 +18,14 @@ Author: Daniel Kroening, [email protected]
1818
#include "ansi_c_parse_tree.h"
1919
#include "ansi_c_scope.h"
2020

21-
int yyansi_cparse();
22-
2321
class ansi_c_parsert:public parsert
2422
{
2523
public:
2624
ansi_c_parse_treet parse_tree;
2725

28-
ansi_c_parsert()
29-
: tag_following(false),
26+
explicit ansi_c_parsert(message_handlert &message_handler)
27+
: parsert(message_handler),
28+
tag_following(false),
3029
asm_block_following(false),
3130
parenthesis_counter(0),
3231
mode(modet::NONE),
@@ -35,14 +34,14 @@ class ansi_c_parsert:public parsert
3534
for_has_scope(false),
3635
ts_18661_3_Floatn_types(false)
3736
{
37+
// set up global scope
38+
scopes.clear();
39+
scopes.push_back(scopet());
3840
}
3941

40-
virtual bool parse() override
41-
{
42-
return yyansi_cparse()!=0;
43-
}
42+
bool parse() override;
4443

45-
virtual void clear() override
44+
void clear() override
4645
{
4746
parsert::clear();
4847
parse_tree.clear();
@@ -166,9 +165,4 @@ class ansi_c_parsert:public parsert
166165
std::list<std::map<const irep_idt, bool>> pragma_cprover_stack;
167166
};
168167

169-
extern ansi_c_parsert ansi_c_parser;
170-
171-
int yyansi_cerror(const std::string &error);
172-
void ansi_c_scanner_init();
173-
174168
#endif // CPROVER_ANSI_C_ANSI_C_PARSER_H

src/ansi-c/builtin_factory.cpp

+2-4
Original file line numberDiff line numberDiff line change
@@ -47,17 +47,15 @@ static bool convert(
4747
{
4848
std::istringstream in(s.str());
4949

50-
ansi_c_parser.clear();
50+
ansi_c_parsert ansi_c_parser{message_handler};
5151
ansi_c_parser.set_file(ID_built_in);
5252
ansi_c_parser.in=&in;
53-
ansi_c_parser.log.set_message_handler(message_handler);
5453
ansi_c_parser.for_has_scope=config.ansi_c.for_has_scope;
54+
ansi_c_parser.ts_18661_3_Floatn_types = config.ansi_c.ts_18661_3_Floatn_types;
5555
ansi_c_parser.cpp98=false; // it's not C++
5656
ansi_c_parser.cpp11=false; // it's not C++
5757
ansi_c_parser.mode=config.ansi_c.mode;
5858

59-
ansi_c_scanner_init();
60-
6159
if(ansi_c_parser.parse())
6260
return true;
6361

0 commit comments

Comments
 (0)