GCC Middle and Back End API Reference
program-state.h
Go to the documentation of this file.
1/* Classes for representing the state of interest at a given path of analysis.
2 Copyright (C) 2019-2025 Free Software Foundation, Inc.
3 Contributed by David Malcolm <dmalcolm@redhat.com>.
4
5This file is part of GCC.
6
7GCC is free software; you can redistribute it and/or modify it
8under the terms of the GNU General Public License as published by
9the Free Software Foundation; either version 3, or (at your option)
10any later version.
11
12GCC is distributed in the hope that it will be useful, but
13WITHOUT ANY WARRANTY; without even the implied warranty of
14MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15General Public License for more details.
16
17You should have received a copy of the GNU General Public License
18along with GCC; see the file COPYING3. If not see
19<http://www.gnu.org/licenses/>. */
20
21#ifndef GCC_ANALYZER_PROGRAM_STATE_H
22#define GCC_ANALYZER_PROGRAM_STATE_H
23
24#include "text-art/widget.h"
25
26namespace ana {
27
28/* Data shared by all program_state instances. */
29
31{
32public:
33 extrinsic_state (std::vector<std::unique_ptr<state_machine>> &&checkers,
34 engine *eng,
36 : m_checkers (std::move (checkers)),
38 m_engine (eng)
39 {
40 }
41
42 // For use in selftests that use just one state machine
43 extrinsic_state (std::unique_ptr<state_machine> sm,
44 engine *eng,
46 : m_logger (logger),
47 m_engine (eng)
48 {
49 m_checkers.push_back (std::move (sm));
50 }
51
52 const state_machine &get_sm (int idx) const
53 {
54 return *m_checkers[idx];
55 }
56
57 const char *get_name (int idx) const
58 {
59 return m_checkers[idx]->get_name ();
60 }
61
62 unsigned get_num_checkers () const { return m_checkers.size (); }
63
64 logger *get_logger () const { return m_logger; }
65
66 void dump_to_pp (pretty_printer *pp) const;
67 void dump_to_file (FILE *outf) const;
68 void dump () const;
69
70 std::unique_ptr<json::object> to_json () const;
71
72 engine *get_engine () const { return m_engine; }
74
75 bool get_sm_idx_by_name (const char *name, unsigned *out) const;
76
77private:
78 /* The state machines. */
79 std::vector<std::unique_ptr<state_machine>> m_checkers;
80
83};
84
85/* Map from svalue * to state machine state, also capturing the origin of
86 each state. */
87
89{
90public:
91 /* An entry in the hash_map. */
92 struct entry_t
93 {
94 /* Default ctor needed by hash_map::empty. */
96 : m_state (0), m_origin (NULL)
97 {
98 }
99
101 const svalue *origin)
102 : m_state (state), m_origin (origin)
103 {}
104
105 bool operator== (const entry_t &other) const
106 {
107 return (m_state == other.m_state
108 && m_origin == other.m_origin);
109 }
110 bool operator!= (const entry_t &other) const
111 {
112 return !(*this == other);
113 }
114
115 static int cmp (const entry_t &entry_a, const entry_t &entry_b);
116
119 };
120 typedef hash_map <const svalue *, entry_t> map_t;
121 typedef map_t::iterator iterator_t;
122
124
126
127 void print (const region_model *model,
128 bool simple, bool multiline,
129 pretty_printer *pp) const;
130 void dump (bool simple) const;
131
132 std::unique_ptr<json::object> to_json () const;
133
134 std::unique_ptr<text_art::tree_widget>
135 make_dump_widget (const text_art::dump_widget_info &dwi,
136 const region_model *model) const;
137
138 bool is_empty_p () const;
139
140 hashval_t hash () const;
141
142 bool operator== (const sm_state_map &other) const;
143 bool operator!= (const sm_state_map &other) const
144 {
145 return !(*this == other);
146 }
147
149 const extrinsic_state &ext_state) const;
150 const svalue *get_origin (const svalue *sval,
151 const extrinsic_state &ext_state) const;
152
154 const svalue *sval,
156 const svalue *origin,
158 bool set_state (const equiv_class &ec,
160 const svalue *origin,
162 bool impl_set_state (const svalue *sval,
164 const svalue *origin,
166 void clear_any_state (const svalue *sval);
168
171
172 void on_svalue_leak (const svalue *sval,
174 void on_liveness_change (const svalue_set &live_svalues,
175 const region_model *model,
178
179 void on_unknown_change (const svalue *sval,
180 bool is_mutable,
182
183 void purge_state_involving (const svalue *sval,
185
186 iterator_t begin () const { return m_map.begin (); }
187 iterator_t end () const { return m_map.end (); }
188 size_t elements () const { return m_map.elements (); }
189
190 static int cmp (const sm_state_map &smap_a, const sm_state_map &smap_b);
191
192 static const svalue *
194
196 const sm_state_map &summary);
197
198 bool can_merge_with_p (const sm_state_map &other,
199 const state_machine &sm,
201 sm_state_map **out) const;
202
203private:
207};
208
209/* A class for representing the state of interest at a given path of
210 analysis.
211
212 Currently this is a combination of:
213 (a) a region_model, giving:
214 (a.1) a hierarchy of memory regions
215 (a.2) values for the regions
216 (a.3) inequalities between values
217 (b) sm_state_maps per state machine, giving a sparse mapping of
218 values to states. */
219
221{
222public:
228
229 hashval_t hash () const;
230 bool operator== (const program_state &other) const;
231 bool operator!= (const program_state &other) const
232 {
233 return !(*this == other);
234 }
235
237 pretty_printer *pp) const;
238
239 void dump_to_pp (const extrinsic_state &ext_state, bool simple,
240 bool multiline, pretty_printer *pp) const;
241 void dump_to_file (const extrinsic_state &ext_state, bool simple,
242 bool multiline, FILE *outf) const;
243 void dump (const extrinsic_state &ext_state, bool simple) const;
244 void dump () const;
245
246 std::unique_ptr<json::object>
248
249 std::unique_ptr<text_art::tree_widget>
250 make_dump_widget (const text_art::dump_widget_info &dwi) const;
251
252 void push_frame (const extrinsic_state &ext_state, const function &fun);
254
256 exploded_node *enode,
257 const gcall &call_stmt,
258 uncertainty_t *uncertainty);
259
261 exploded_node *enode,
262 const gcall &call_stmt,
263 uncertainty_t *uncertainty);
264
265
267 exploded_node *enode,
268 const superedge *succ,
269 uncertainty_t *uncertainty);
270
272 const program_point &point,
273 exploded_node *enode_for_diag,
274 uncertainty_t *uncertainty) const;
275
277
279 const svalue *sval) const
280 {
281 /* Don't purge vars that have non-purgeable sm state, to avoid
282 generating false "leak" complaints. */
283 int i;
284 sm_state_map *smap;
286 {
287 const state_machine &sm = ext_state.get_sm (i);
288 if (!sm.can_purge_p (smap->get_state (sval, ext_state)))
289 return false;
290 }
291 return true;
292 }
293
295 const region *base_reg) const;
296
297 bool can_merge_with_p (const program_state &other,
299 const program_point &point,
300 program_state *out) const;
301
302 void validate (const extrinsic_state &ext_state) const;
303
304 static void detect_leaks (const program_state &src_state,
305 const program_state &dest_state,
306 const svalue *extra_sval,
309
311 const program_state &summary);
312
316
317 /* TODO: lose the pointer here (const-correctness issues?). */
320
321 /* If false, then don't attempt to explore further states along this path.
322 For use in "handling" lvalues for tree codes we haven't yet
323 implemented. */
325};
326
327/* An abstract base class for use with for_each_state_change. */
328
330{
331public:
333
334 /* Return true for early exit, false to keep iterating. */
335 virtual bool on_global_state_change (const state_machine &sm,
336 state_machine::state_t src_sm_val,
337 state_machine::state_t dst_sm_val) = 0;
338
339 /* Return true for early exit, false to keep iterating. */
340 virtual bool on_state_change (const state_machine &sm,
341 state_machine::state_t src_sm_val,
342 state_machine::state_t dst_sm_val,
343 const svalue *dst_sval,
344 const svalue *dst_origin_sval) = 0;
345};
346
347extern bool for_each_state_change (const program_state &src_state,
348 const program_state &dst_state,
351
352} // namespace ana
353
354#endif /* GCC_ANALYZER_PROGRAM_STATE_H */
Definition call-summary.h:68
Definition region-model.h:1338
Definition constraint-manager.h:255
Definition exploded-graph.h:804
Definition exploded-graph.h:203
Definition program-state.h:31
void dump_to_file(FILE *outf) const
bool get_sm_idx_by_name(const char *name, unsigned *out) const
std::vector< std::unique_ptr< state_machine > > m_checkers
Definition program-state.h:79
unsigned get_num_checkers() const
Definition program-state.h:62
logger * get_logger() const
Definition program-state.h:64
std::unique_ptr< json::object > to_json() const
const char * get_name(int idx) const
Definition program-state.h:57
logger * m_logger
Definition program-state.h:81
void dump_to_pp(pretty_printer *pp) const
region_model_manager * get_model_manager() const
extrinsic_state(std::unique_ptr< state_machine > sm, engine *eng, logger *logger=NULL)
Definition program-state.h:43
engine * m_engine
Definition program-state.h:82
engine * get_engine() const
Definition program-state.h:72
const state_machine & get_sm(int idx) const
Definition program-state.h:52
extrinsic_state(std::vector< std::unique_ptr< state_machine > > &&checkers, engine *eng, logger *logger=NULL)
Definition program-state.h:33
Definition exploded-graph.h:39
Definition analyzer-logging.h:34
Definition program-point.h:175
Definition program-state.h:221
void returning_call(exploded_graph &eg, exploded_node *enode, const gcall &call_stmt, uncertainty_t *uncertainty)
bool can_purge_base_region_p(const extrinsic_state &ext_state, const region *base_reg) const
program_state prune_for_point(exploded_graph &eg, const program_point &point, exploded_node *enode_for_diag, uncertainty_t *uncertainty) const
void dump() const
std::unique_ptr< json::object > to_json(const extrinsic_state &ext_state) const
void dump(const extrinsic_state &ext_state, bool simple) const
bool operator!=(const program_state &other) const
Definition program-state.h:231
program_state(const program_state &other)
void push_frame(const extrinsic_state &ext_state, const function &fun)
bool operator==(const program_state &other) const
static void detect_leaks(const program_state &src_state, const program_state &dest_state, const svalue *extra_sval, const extrinsic_state &ext_state, region_model_context *ctxt)
void dump_to_file(const extrinsic_state &ext_state, bool simple, bool multiline, FILE *outf) const
void print(const extrinsic_state &ext_state, pretty_printer *pp) const
bool replay_call_summary(call_summary_replay &r, const program_state &summary)
program_state & operator=(const program_state &other)
bool on_edge(exploded_graph &eg, exploded_node *enode, const superedge *succ, uncertainty_t *uncertainty)
program_state(program_state &&other)
void validate(const extrinsic_state &ext_state) const
auto_delete_vec< sm_state_map > m_checker_states
Definition program-state.h:319
bool can_merge_with_p(const program_state &other, const extrinsic_state &ext_state, const program_point &point, program_state *out) const
region_model * m_region_model
Definition program-state.h:318
void push_call(exploded_graph &eg, exploded_node *enode, const gcall &call_stmt, uncertainty_t *uncertainty)
bool can_purge_p(const extrinsic_state &ext_state, const svalue *sval) const
Definition program-state.h:278
std::unique_ptr< text_art::tree_widget > make_dump_widget(const text_art::dump_widget_info &dwi) const
const function * get_current_function() const
tree get_representative_tree(const svalue *sval) const
hashval_t hash() const
void impl_call_analyzer_dump_state(const gcall &call, const extrinsic_state &ext_state, region_model_context *ctxt)
void dump_to_pp(const extrinsic_state &ext_state, bool simple, bool multiline, pretty_printer *pp) const
bool m_valid
Definition program-state.h:324
program_state(const extrinsic_state &ext_state)
Definition region-model.h:815
Definition region-model-manager.h:32
Definition region-model.h:298
Definition region.h:126
Definition program-state.h:89
map_t::iterator iterator_t
Definition program-state.h:121
bool operator==(const sm_state_map &other) const
bool can_merge_with_p(const sm_state_map &other, const state_machine &sm, const extrinsic_state &ext_state, sm_state_map **out) const
static const svalue * canonicalize_svalue(const svalue *sval, const extrinsic_state &ext_state)
static int cmp(const sm_state_map &smap_a, const sm_state_map &smap_b)
iterator_t begin() const
Definition program-state.h:186
state_machine::state_t m_global_state
Definition program-state.h:206
state_machine::state_t get_state(const svalue *sval, const extrinsic_state &ext_state) const
const state_machine & m_sm
Definition program-state.h:204
sm_state_map * clone() const
size_t elements() const
Definition program-state.h:188
hash_map< const svalue *, entry_t > map_t
Definition program-state.h:120
void print(const region_model *model, bool simple, bool multiline, pretty_printer *pp) const
map_t m_map
Definition program-state.h:205
std::unique_ptr< json::object > to_json() const
void set_state(region_model *model, const svalue *sval, state_machine::state_t state, const svalue *origin, const extrinsic_state &ext_state)
bool operator!=(const sm_state_map &other) const
Definition program-state.h:143
bool impl_set_state(const svalue *sval, state_machine::state_t state, const svalue *origin, const extrinsic_state &ext_state)
hashval_t hash() const
void clear_any_state(const svalue *sval)
void on_svalue_leak(const svalue *sval, impl_region_model_context *ctxt)
state_machine::state_t get_global_state() const
const svalue * get_origin(const svalue *sval, const extrinsic_state &ext_state) const
iterator_t end() const
Definition program-state.h:187
void on_unknown_change(const svalue *sval, bool is_mutable, const extrinsic_state &ext_state)
void clear_all_per_svalue_state()
void purge_state_involving(const svalue *sval, const extrinsic_state &ext_state)
bool set_state(const equiv_class &ec, state_machine::state_t state, const svalue *origin, const extrinsic_state &ext_state)
std::unique_ptr< text_art::tree_widget > make_dump_widget(const text_art::dump_widget_info &dwi, const region_model *model) const
void set_global_state(state_machine::state_t state)
bool is_empty_p() const
void on_liveness_change(const svalue_set &live_svalues, const region_model *model, const extrinsic_state &ext_state, impl_region_model_context *ctxt)
bool replay_call_summary(call_summary_replay &r, const sm_state_map &summary)
sm_state_map(const state_machine &sm)
void dump(bool simple) const
Definition program-state.h:330
virtual bool on_state_change(const state_machine &sm, state_machine::state_t src_sm_val, state_machine::state_t dst_sm_val, const svalue *dst_sval, const svalue *dst_origin_sval)=0
virtual ~state_change_visitor()
Definition program-state.h:332
virtual bool on_global_state_change(const state_machine &sm, state_machine::state_t src_sm_val, state_machine::state_t dst_sm_val)=0
Definition sm.h:40
const state_machine::state * state_t
Definition sm.h:60
virtual bool can_purge_p(state_t s) const =0
Definition supergraph.h:318
Definition svalue.h:92
Definition store.h:161
Definition region-model.h:222
Definition vec.h:1813
Definition ree.cc:583
Definition pretty-print.h:241
union tree_node * tree
Definition coretypes.h:97
Definition access-diagram.h:30
bool for_each_state_change(const program_state &src_state, const program_state &dst_state, const extrinsic_state &ext_state, state_change_visitor *visitor)
hash_set< const svalue * > svalue_set
Definition common.h:78
poly_int< N, C > r
Definition poly-int.h:774
i
Definition poly-int.h:776
static int cmp(const entry_t &entry_a, const entry_t &entry_b)
bool operator==(const entry_t &other) const
Definition program-state.h:105
entry_t(state_machine::state_t state, const svalue *origin)
Definition program-state.h:100
entry_t()
Definition program-state.h:95
state_machine::state_t m_state
Definition program-state.h:117
bool operator!=(const entry_t &other) const
Definition program-state.h:110
const svalue * m_origin
Definition program-state.h:118
Definition function.h:249
Definition gimple.h:352
Definition ira-emit.cc:158
Definition gengtype.h:377
Definition genautomata.cc:669
#define NULL
Definition system.h:50
#define FOR_EACH_VEC_ELT(V, I, P)
Definition vec.h:1895