Skip to content

Commit 2df4916

Browse files
committed
Adapt c nondet symbol factory for use in function body generation
1 parent 8a0b1de commit 2df4916

File tree

3 files changed

+137
-67
lines changed

3 files changed

+137
-67
lines changed

src/ansi-c/ansi_c_entry_point.cpp

Lines changed: 2 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,8 @@ exprt::operandst build_function_environment(
4242
base_name,
4343
p.type(),
4444
p.source_location(),
45-
object_factory_parameters);
45+
object_factory_parameters,
46+
lifetimet::AUTOMATIC_LOCAL);
4647
}
4748

4849
return main_arguments;

src/ansi-c/c_nondet_symbol_factory.cpp

Lines changed: 29 additions & 65 deletions
Original file line numberDiff line numberDiff line change
@@ -25,79 +25,29 @@ Author: Diffblue Ltd.
2525

2626
#include <goto-programs/goto_functions.h>
2727

28-
class symbol_factoryt
29-
{
30-
symbol_tablet &symbol_table;
31-
const source_locationt &loc;
32-
namespacet ns;
33-
const c_object_factory_parameterst &object_factory_params;
34-
35-
allocate_objectst allocate_objects;
36-
37-
typedef std::set<irep_idt> recursion_sett;
38-
39-
public:
40-
symbol_factoryt(
41-
symbol_tablet &_symbol_table,
42-
const source_locationt &loc,
43-
const c_object_factory_parameterst &object_factory_params)
44-
: symbol_table(_symbol_table),
45-
loc(loc),
46-
ns(_symbol_table),
47-
object_factory_params(object_factory_params),
48-
allocate_objects(ID_C, loc, loc.get_function(), symbol_table)
49-
{}
50-
51-
void gen_nondet_init(
52-
code_blockt &assignments,
53-
const exprt &expr,
54-
const std::size_t depth = 0,
55-
recursion_sett recursion_set = recursion_sett());
56-
57-
void add_created_symbol(const symbolt *symbol_ptr)
58-
{
59-
allocate_objects.add_created_symbol(symbol_ptr);
60-
}
61-
62-
void declare_created_symbols(code_blockt &init_code)
63-
{
64-
allocate_objects.declare_created_symbols(init_code);
65-
}
66-
67-
void mark_created_symbols_as_input(code_blockt &init_code)
68-
{
69-
allocate_objects.mark_created_symbols_as_input(init_code);
70-
}
71-
72-
private:
73-
/// Generate initialisation code for each array element
74-
/// \param assignments: The code block to add code to
75-
/// \param expr: An expression of array type
76-
/// \param depth: The struct recursion depth
77-
/// \param recursion_set: The struct recursion set
78-
/// \see gen_nondet_init For the meaning of the last 2 parameters
79-
void gen_nondet_array_init(
80-
code_blockt &assignments,
81-
const exprt &expr,
82-
std::size_t depth,
83-
const recursion_sett &recursion_set);
84-
};
85-
8628
/// Creates a nondet for expr, including calling itself recursively to make
8729
/// appropriate symbols to point to if expr is a pointer.
8830
/// \param assignments: The code block to add code to
8931
/// \param expr: The expression which we are generating a non-determinate value
9032
/// for
9133
/// \param depth number of pointers followed so far during initialisation
9234
/// \param recursion_set names of structs seen so far on current pointer chain
35+
/// \param assign_const Indicates whether const objects should be nondet
36+
/// initialized
9337
void symbol_factoryt::gen_nondet_init(
9438
code_blockt &assignments,
9539
const exprt &expr,
9640
const std::size_t depth,
97-
recursion_sett recursion_set)
41+
recursion_sett recursion_set,
42+
const bool assign_const)
9843
{
9944
const typet &type=ns.follow(expr.type());
10045

46+
if(!assign_const && expr.type().get_bool(ID_C_constant))
47+
{
48+
return;
49+
}
50+
10151
if(type.id()==ID_pointer)
10252
{
10353
// dereferenced type
@@ -122,10 +72,10 @@ void symbol_factoryt::gen_nondet_init(
12272

12373
code_blockt non_null_inst;
12474

125-
exprt init_expr = allocate_objects.allocate_automatic_local_object(
126-
non_null_inst, expr, subtype);
75+
exprt init_expr = allocate_objects.allocate_object(
76+
non_null_inst, expr, subtype, lifetime);
12777

128-
gen_nondet_init(non_null_inst, init_expr, depth + 1, recursion_set);
78+
gen_nondet_init(non_null_inst, init_expr, depth + 1, recursion_set, true);
12979

13080
if(depth < object_factory_params.min_null_tree_depth)
13181
{
@@ -165,12 +115,18 @@ void symbol_factoryt::gen_nondet_init(
165115
for(const auto &component : struct_type.components())
166116
{
167117
const typet &component_type = component.type();
118+
119+
if(!assign_const && component_type.get_bool(ID_C_constant))
120+
{
121+
continue;
122+
}
123+
168124
const irep_idt name = component.get_name();
169125

170126
member_exprt me(expr, name, component_type);
171127
me.add_source_location() = loc;
172128

173-
gen_nondet_init(assignments, me, depth, recursion_set);
129+
gen_nondet_init(assignments, me, depth, recursion_set, assign_const);
174130
}
175131
}
176132
else if(type.id() == ID_array)
@@ -221,14 +177,17 @@ void symbol_factoryt::gen_nondet_array_init(
221177
/// \param loc: The location to assign to generated code
222178
/// \param object_factory_parameters configuration parameters for the object
223179
/// factory
180+
/// \param allocation_type Indicates whether the new objects should be allocated
181+
/// locally, globally, or dynamically
224182
/// \return Returns the symbol_exprt for the symbol created
225183
symbol_exprt c_nondet_symbol_factory(
226184
code_blockt &init_code,
227185
symbol_tablet &symbol_table,
228186
const irep_idt base_name,
229187
const typet &type,
230188
const source_locationt &loc,
231-
const c_object_factory_parameterst &object_factory_parameters)
189+
const c_object_factory_parameterst &object_factory_parameters,
190+
const lifetimet lifetime)
232191
{
233192
irep_idt identifier=id2string(goto_functionst::entry_point())+
234193
"::"+id2string(base_name);
@@ -247,7 +206,12 @@ symbol_exprt c_nondet_symbol_factory(
247206
bool moving_symbol_failed=symbol_table.move(main_symbol, main_symbol_ptr);
248207
CHECK_RETURN(!moving_symbol_failed);
249208

250-
symbol_factoryt state(symbol_table, loc, object_factory_parameters);
209+
symbol_factoryt state(
210+
symbol_table,
211+
loc,
212+
object_factory_parameters,
213+
lifetime);
214+
251215
code_blockt assignments;
252216
state.gen_nondet_init(assignments, main_symbol_expr);
253217

src/ansi-c/c_nondet_symbol_factory.h

Lines changed: 106 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -14,15 +14,120 @@ Author: Diffblue Ltd.
1414

1515
#include "c_object_factory_parameters.h"
1616

17+
#include <set>
18+
#include <vector>
19+
20+
#include <util/allocate_objects.h>
1721
#include <util/std_code.h>
1822
#include <util/symbol_table.h>
1923

24+
#if 0
25+
class symbol_factoryt
26+
{
27+
std::vector<const symbolt *> &symbols_created;
28+
symbol_tablet &symbol_table;
29+
const source_locationt &loc;
30+
namespacet ns;
31+
const c_object_factory_parameterst &object_factory_params;
32+
const lifetimet lifetime;
33+
34+
public:
35+
typedef std::set<irep_idt> recursion_sett;
36+
37+
symbol_factoryt(
38+
std::vector<const symbolt *> &symbols_created,
39+
symbol_tablet &symbol_table,
40+
const source_locationt &loc,
41+
const c_object_factory_parameterst &object_factory_params,
42+
const lifetimet lifetime)
43+
: symbols_created(symbols_created),
44+
symbol_table(symbol_table),
45+
loc(loc),
46+
ns(symbol_table),
47+
object_factory_params(object_factory_params),
48+
lifetime(lifetime)
49+
{
50+
}
51+
52+
void gen_nondet_init(
53+
code_blockt &assignments,
54+
const exprt &expr,
55+
const std::size_t depth = 0,
56+
recursion_sett recursion_set = recursion_sett(),
57+
const bool assign_const = true);
58+
};
59+
#endif
60+
61+
class symbol_factoryt
62+
{
63+
symbol_tablet &symbol_table;
64+
const source_locationt &loc;
65+
namespacet ns;
66+
const c_object_factory_parameterst &object_factory_params;
67+
68+
allocate_objectst allocate_objects;
69+
70+
const lifetimet lifetime;
71+
72+
public:
73+
typedef std::set<irep_idt> recursion_sett;
74+
75+
symbol_factoryt(
76+
symbol_tablet &_symbol_table,
77+
const source_locationt &loc,
78+
const c_object_factory_parameterst &object_factory_params,
79+
const lifetimet lifetime)
80+
: symbol_table(_symbol_table),
81+
loc(loc),
82+
ns(_symbol_table),
83+
object_factory_params(object_factory_params),
84+
allocate_objects(ID_C, loc, loc.get_function(), symbol_table),
85+
lifetime(lifetime)
86+
{}
87+
88+
void gen_nondet_init(
89+
code_blockt &assignments,
90+
const exprt &expr,
91+
const std::size_t depth = 0,
92+
recursion_sett recursion_set = recursion_sett(),
93+
const bool assign_const = true);
94+
95+
void add_created_symbol(const symbolt *symbol_ptr)
96+
{
97+
allocate_objects.add_created_symbol(symbol_ptr);
98+
}
99+
100+
void declare_created_symbols(code_blockt &init_code)
101+
{
102+
allocate_objects.declare_created_symbols(init_code);
103+
}
104+
105+
void mark_created_symbols_as_input(code_blockt &init_code)
106+
{
107+
allocate_objects.mark_created_symbols_as_input(init_code);
108+
}
109+
110+
private:
111+
/// Generate initialisation code for each array element
112+
/// \param assignments: The code block to add code to
113+
/// \param expr: An expression of array type
114+
/// \param depth: The struct recursion depth
115+
/// \param recursion_set: The struct recursion set
116+
/// \see gen_nondet_init For the meaning of the last 2 parameters
117+
void gen_nondet_array_init(
118+
code_blockt &assignments,
119+
const exprt &expr,
120+
std::size_t depth,
121+
const recursion_sett &recursion_set);
122+
};
123+
20124
symbol_exprt c_nondet_symbol_factory(
21125
code_blockt &init_code,
22126
symbol_tablet &symbol_table,
23127
const irep_idt base_name,
24128
const typet &type,
25129
const source_locationt &,
26-
const c_object_factory_parameterst &object_factory_parameters);
130+
const c_object_factory_parameterst &object_factory_parameters,
131+
const lifetimet lifetime);
27132

28133
#endif // CPROVER_ANSI_C_C_NONDET_SYMBOL_FACTORY_H

0 commit comments

Comments
 (0)