Skip to content

Commit abb26db

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 dc35ca7 commit abb26db

15 files changed

+783
-718
lines changed

src/ansi-c/ansi_c_language.cpp

+6-15
Original file line numberDiff line numberDiff line change
@@ -71,10 +71,9 @@ bool ansi_c_languaget::parse(
7171
ansi_c_internal_additions(code, config.ansi_c.float16_type);
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.float16_type = config.ansi_c.float16_type;
@@ -83,25 +82,19 @@ bool ansi_c_languaget::parse(
8382
ansi_c_parser.cpp11=false; // it's not C++
8483
ansi_c_parser.mode=config.ansi_c.mode;
8584

86-
ansi_c_scanner_init();
87-
8885
bool result=ansi_c_parser.parse();
8986

9087
if(!result)
9188
{
9289
ansi_c_parser.set_line_no(0);
9390
ansi_c_parser.set_file(path);
9491
ansi_c_parser.in=&i_preprocessed;
95-
ansi_c_scanner_init();
9692
result=ansi_c_parser.parse();
9793
}
9894

9995
// save result
10096
parse_tree.swap(ansi_c_parser.parse_tree);
10197

102-
// save some memory
103-
ansi_c_parser.clear();
104-
10598
return result;
10699
}
107100

@@ -199,15 +192,16 @@ bool ansi_c_languaget::to_expr(
199192

200193
// parsing
201194

202-
ansi_c_parser.clear();
195+
ansi_c_parsert ansi_c_parser{message_handler};
203196
ansi_c_parser.set_file(irep_idt());
204197
ansi_c_parser.in=&i_preprocessed;
205-
ansi_c_parser.log.set_message_handler(message_handler);
206-
ansi_c_parser.mode=config.ansi_c.mode;
198+
ansi_c_parser.for_has_scope = config.ansi_c.for_has_scope;
207199
ansi_c_parser.ts_18661_3_Floatn_types=config.ansi_c.ts_18661_3_Floatn_types;
208200
ansi_c_parser.float16_type = config.ansi_c.float16_type;
209201
ansi_c_parser.bf16_type = config.ansi_c.bf16_type;
210-
ansi_c_scanner_init();
202+
ansi_c_parser.cpp98 = false; // it's not C++
203+
ansi_c_parser.cpp11 = false; // it's not C++
204+
ansi_c_parser.mode = config.ansi_c.mode;
211205

212206
bool result=ansi_c_parser.parse();
213207

@@ -221,9 +215,6 @@ bool ansi_c_languaget::to_expr(
221215
result = ansi_c_typecheck(expr, message_handler, ns);
222216
}
223217

224-
// save some memory
225-
ansi_c_parser.clear();
226-
227218
// now remove that (void) cast
228219
if(expr.id()==ID_typecast &&
229220
expr.type().id()==ID_empty &&

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),
@@ -37,14 +36,14 @@ class ansi_c_parsert:public parsert
3736
float16_type(false),
3837
bf16_type(false)
3938
{
39+
// set up global scope
40+
scopes.clear();
41+
scopes.push_back(scopet());
4042
}
4143

42-
virtual bool parse() override
43-
{
44-
return yyansi_cparse()!=0;
45-
}
44+
bool parse() override;
4645

47-
virtual void clear() override
46+
void clear() override
4847
{
4948
parsert::clear();
5049
parse_tree.clear();
@@ -170,9 +169,4 @@ class ansi_c_parsert:public parsert
170169
std::list<std::map<const irep_idt, bool>> pragma_cprover_stack;
171170
};
172171

173-
extern ansi_c_parsert ansi_c_parser;
174-
175-
int yyansi_cerror(const std::string &error);
176-
void ansi_c_scanner_init();
177-
178172
#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)