Branch data Line data Source code
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 : :
5 : : This file is part of GCC.
6 : :
7 : : GCC is free software; you can redistribute it and/or modify it
8 : : under the terms of the GNU General Public License as published by
9 : : the Free Software Foundation; either version 3, or (at your option)
10 : : any later version.
11 : :
12 : : GCC is distributed in the hope that it will be useful, but
13 : : WITHOUT ANY WARRANTY; without even the implied warranty of
14 : : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 : : General Public License for more details.
16 : :
17 : : You should have received a copy of the GNU General Public License
18 : : along with GCC; see the file COPYING3. If not see
19 : : <http://www.gnu.org/licenses/>. */
20 : :
21 : : #include "config.h"
22 : : #define INCLUDE_VECTOR
23 : : #include "system.h"
24 : : #include "coretypes.h"
25 : : #include "tree.h"
26 : : #include "diagnostic-core.h"
27 : : #include "diagnostic.h"
28 : : #include "analyzer/analyzer.h"
29 : : #include "analyzer/analyzer-logging.h"
30 : : #include "analyzer/sm.h"
31 : : #include "sbitmap.h"
32 : : #include "bitmap.h"
33 : : #include "ordered-hash-map.h"
34 : : #include "selftest.h"
35 : : #include "analyzer/call-string.h"
36 : : #include "analyzer/program-point.h"
37 : : #include "analyzer/store.h"
38 : : #include "analyzer/region-model.h"
39 : : #include "analyzer/program-state.h"
40 : : #include "analyzer/constraint-manager.h"
41 : : #include "diagnostic-event-id.h"
42 : : #include "analyzer/pending-diagnostic.h"
43 : : #include "analyzer/diagnostic-manager.h"
44 : : #include "cfg.h"
45 : : #include "basic-block.h"
46 : : #include "gimple.h"
47 : : #include "gimple-iterator.h"
48 : : #include "cgraph.h"
49 : : #include "digraph.h"
50 : : #include "analyzer/supergraph.h"
51 : : #include "analyzer/program-state.h"
52 : : #include "analyzer/exploded-graph.h"
53 : : #include "analyzer/state-purge.h"
54 : : #include "analyzer/call-summary.h"
55 : : #include "analyzer/analyzer-selftests.h"
56 : : #include "text-art/tree-widget.h"
57 : : #include "text-art/dump.h"
58 : : #include "make-unique.h"
59 : :
60 : : #if ENABLE_ANALYZER
61 : :
62 : : namespace ana {
63 : :
64 : : /* class extrinsic_state. */
65 : :
66 : : /* Dump a multiline representation of this state to PP. */
67 : :
68 : : void
69 : 0 : extrinsic_state::dump_to_pp (pretty_printer *pp) const
70 : : {
71 : 0 : pp_printf (pp, "extrinsic_state: %i checker(s)\n", get_num_checkers ());
72 : 0 : unsigned i;
73 : 0 : state_machine *checker;
74 : 0 : FOR_EACH_VEC_ELT (m_checkers, i, checker)
75 : : {
76 : 0 : pp_printf (pp, "m_checkers[%i]: %qs\n", i, checker->get_name ());
77 : 0 : checker->dump_to_pp (pp);
78 : : }
79 : 0 : }
80 : :
81 : : /* Dump a multiline representation of this state to OUTF. */
82 : :
83 : : void
84 : 0 : extrinsic_state::dump_to_file (FILE *outf) const
85 : : {
86 : 0 : tree_dump_pretty_printer pp (outf);
87 : 0 : dump_to_pp (&pp);
88 : 0 : }
89 : :
90 : : /* Dump a multiline representation of this state to stderr. */
91 : :
92 : : DEBUG_FUNCTION void
93 : 0 : extrinsic_state::dump () const
94 : : {
95 : 0 : dump_to_file (stderr);
96 : 0 : }
97 : :
98 : : /* Return a new json::object of the form
99 : : {"checkers" : array of objects, one for each state_machine}. */
100 : :
101 : : std::unique_ptr<json::object>
102 : 0 : extrinsic_state::to_json () const
103 : : {
104 : 0 : auto ext_state_obj = ::make_unique<json::object> ();
105 : :
106 : 0 : {
107 : 0 : auto checkers_arr = ::make_unique<json::array> ();
108 : 0 : unsigned i;
109 : 0 : state_machine *sm;
110 : 0 : FOR_EACH_VEC_ELT (m_checkers, i, sm)
111 : 0 : checkers_arr->append (sm->to_json ());
112 : 0 : ext_state_obj->set ("checkers", std::move (checkers_arr));
113 : 0 : }
114 : :
115 : 0 : return ext_state_obj;
116 : : }
117 : :
118 : : /* Get the region_model_manager for this extrinsic_state. */
119 : :
120 : : region_model_manager *
121 : 10943798 : extrinsic_state::get_model_manager () const
122 : : {
123 : 10943798 : if (m_engine)
124 : 10943798 : return m_engine->get_model_manager ();
125 : : else
126 : : return NULL; /* for selftests. */
127 : : }
128 : :
129 : : /* Try to find a state machine named NAME.
130 : : If found, return true and write its index to *OUT.
131 : : Otherwise return false. */
132 : :
133 : : bool
134 : 664240 : extrinsic_state::get_sm_idx_by_name (const char *name, unsigned *out) const
135 : : {
136 : 664240 : unsigned i;
137 : 664240 : state_machine *sm;
138 : 2654123 : FOR_EACH_VEC_ELT (m_checkers, i, sm)
139 : 2653620 : if (0 == strcmp (name, sm->get_name ()))
140 : : {
141 : : /* Found NAME. */
142 : 663737 : *out = i;
143 : 663737 : return true;
144 : : }
145 : :
146 : : /* NAME not found. */
147 : : return false;
148 : : }
149 : :
150 : : /* struct sm_state_map::entry_t. */
151 : :
152 : : int
153 : 91811 : sm_state_map::entry_t::cmp (const entry_t &entry_a, const entry_t &entry_b)
154 : : {
155 : 91811 : gcc_assert (entry_a.m_state);
156 : 91811 : gcc_assert (entry_b.m_state);
157 : 91811 : if (int cmp_state = ((int)entry_a.m_state->get_id ()
158 : 91811 : - (int)entry_b.m_state->get_id ()))
159 : : return cmp_state;
160 : 86188 : if (entry_a.m_origin && entry_b.m_origin)
161 : 0 : return svalue::cmp_ptr (entry_a.m_origin, entry_b.m_origin);
162 : 86188 : if (entry_a.m_origin)
163 : : return 1;
164 : 86188 : if (entry_b.m_origin)
165 : 0 : return -1;
166 : : return 0;
167 : : }
168 : :
169 : : /* class sm_state_map. */
170 : :
171 : : /* sm_state_map's ctor. */
172 : :
173 : 2456264 : sm_state_map::sm_state_map (const state_machine &sm)
174 : 2456264 : : m_sm (sm), m_map (), m_global_state (sm.get_start_state ())
175 : : {
176 : 2456264 : }
177 : :
178 : : /* Clone the sm_state_map. */
179 : :
180 : : sm_state_map *
181 : 19290647 : sm_state_map::clone () const
182 : : {
183 : 19290647 : return new sm_state_map (*this);
184 : : }
185 : :
186 : : /* Print this sm_state_map to PP.
187 : : If MODEL is non-NULL, print representative tree values where
188 : : available. */
189 : :
190 : : void
191 : 498 : sm_state_map::print (const region_model *model,
192 : : bool simple, bool multiline,
193 : : pretty_printer *pp) const
194 : : {
195 : 498 : bool first = true;
196 : 498 : if (!multiline)
197 : 98 : pp_string (pp, "{");
198 : 498 : if (m_global_state != m_sm.get_start_state ())
199 : : {
200 : 0 : if (multiline)
201 : 0 : pp_string (pp, " ");
202 : 0 : pp_string (pp, "global: ");
203 : 0 : m_global_state->dump_to_pp (pp);
204 : 0 : if (multiline)
205 : 0 : pp_newline (pp);
206 : : first = false;
207 : : }
208 : 498 : auto_vec <const svalue *> keys (m_map.elements ());
209 : 498 : for (map_t::iterator iter = m_map.begin ();
210 : 1275 : iter != m_map.end ();
211 : 777 : ++iter)
212 : 777 : keys.quick_push ((*iter).first);
213 : 498 : keys.qsort (svalue::cmp_ptr_ptr);
214 : : unsigned i;
215 : : const svalue *sval;
216 : 1275 : FOR_EACH_VEC_ELT (keys, i, sval)
217 : : {
218 : 777 : if (multiline)
219 : 676 : pp_string (pp, " ");
220 : 101 : else if (!first)
221 : 3 : pp_string (pp, ", ");
222 : 777 : first = false;
223 : 777 : if (!flag_dump_noaddr)
224 : : {
225 : 777 : pp_pointer (pp, sval);
226 : 777 : pp_string (pp, ": ");
227 : : }
228 : 777 : sval->dump_to_pp (pp, simple);
229 : :
230 : 777 : entry_t e = *const_cast <map_t &> (m_map).get (sval);
231 : 777 : pp_string (pp, ": ");
232 : 777 : e.m_state->dump_to_pp (pp);
233 : 777 : if (model)
234 : 777 : if (tree rep = model->get_representative_tree (sval))
235 : : {
236 : 735 : pp_string (pp, " (");
237 : 735 : dump_quoted_tree (pp, rep);
238 : 735 : pp_character (pp, ')');
239 : : }
240 : 777 : if (e.m_origin)
241 : : {
242 : 0 : pp_string (pp, " (origin: ");
243 : 0 : if (!flag_dump_noaddr)
244 : : {
245 : 0 : pp_pointer (pp, e.m_origin);
246 : 0 : pp_string (pp, ": ");
247 : : }
248 : 0 : e.m_origin->dump_to_pp (pp, simple);
249 : 0 : if (model)
250 : 0 : if (tree rep = model->get_representative_tree (e.m_origin))
251 : : {
252 : 0 : pp_string (pp, " (");
253 : 0 : dump_quoted_tree (pp, rep);
254 : 0 : pp_character (pp, ')');
255 : : }
256 : 0 : pp_string (pp, ")");
257 : : }
258 : 777 : if (multiline)
259 : 676 : pp_newline (pp);
260 : : }
261 : 498 : if (!multiline)
262 : 98 : pp_string (pp, "}");
263 : 498 : }
264 : :
265 : : /* Dump this object to stderr. */
266 : :
267 : : DEBUG_FUNCTION void
268 : 0 : sm_state_map::dump (bool simple) const
269 : : {
270 : 0 : tree_dump_pretty_printer pp (stderr);
271 : 0 : print (NULL, simple, true, &pp);
272 : 0 : pp_newline (&pp);
273 : 0 : }
274 : :
275 : : /* Return a new json::object of the form
276 : : {"global" : (optional) value for global state,
277 : : SVAL_DESC : value for state}. */
278 : :
279 : : std::unique_ptr<json::object>
280 : 0 : sm_state_map::to_json () const
281 : : {
282 : 0 : auto map_obj = ::make_unique<json::object> ();
283 : :
284 : 0 : if (m_global_state != m_sm.get_start_state ())
285 : 0 : map_obj->set ("global", m_global_state->to_json ());
286 : 0 : for (map_t::iterator iter = m_map.begin ();
287 : 0 : iter != m_map.end ();
288 : 0 : ++iter)
289 : : {
290 : 0 : const svalue *sval = (*iter).first;
291 : 0 : entry_t e = (*iter).second;
292 : :
293 : 0 : label_text sval_desc = sval->get_desc ();
294 : 0 : map_obj->set (sval_desc.get (), e.m_state->to_json ());
295 : :
296 : : /* This doesn't yet JSONify e.m_origin. */
297 : 0 : }
298 : 0 : return map_obj;
299 : : }
300 : :
301 : : /* Make a text_art::tree_widget describing this sm_state_map,
302 : : using MODEL if non-null to describe svalues. */
303 : :
304 : : std::unique_ptr<text_art::tree_widget>
305 : 0 : sm_state_map::make_dump_widget (const text_art::dump_widget_info &dwi,
306 : : const region_model *model) const
307 : : {
308 : 0 : using text_art::styled_string;
309 : 0 : using text_art::tree_widget;
310 : 0 : std::unique_ptr<tree_widget> state_widget
311 : : (tree_widget::from_fmt (dwi, nullptr,
312 : 0 : "%qs state machine", m_sm.get_name ()));
313 : :
314 : 0 : if (m_global_state != m_sm.get_start_state ())
315 : : {
316 : 0 : pretty_printer the_pp;
317 : 0 : pretty_printer * const pp = &the_pp;
318 : 0 : pp_format_decoder (pp) = default_tree_printer;
319 : 0 : pp_string (pp, "Global State: ");
320 : 0 : m_global_state->dump_to_pp (pp);
321 : 0 : state_widget->add_child (tree_widget::make (dwi, pp));
322 : 0 : }
323 : :
324 : 0 : auto_vec <const svalue *> keys (m_map.elements ());
325 : 0 : for (map_t::iterator iter = m_map.begin ();
326 : 0 : iter != m_map.end ();
327 : 0 : ++iter)
328 : 0 : keys.quick_push ((*iter).first);
329 : 0 : keys.qsort (svalue::cmp_ptr_ptr);
330 : : unsigned i;
331 : : const svalue *sval;
332 : 0 : FOR_EACH_VEC_ELT (keys, i, sval)
333 : : {
334 : 0 : pretty_printer the_pp;
335 : 0 : pretty_printer * const pp = &the_pp;
336 : 0 : const bool simple = true;
337 : 0 : pp_format_decoder (pp) = default_tree_printer;
338 : 0 : if (!flag_dump_noaddr)
339 : : {
340 : 0 : pp_pointer (pp, sval);
341 : 0 : pp_string (pp, ": ");
342 : : }
343 : 0 : sval->dump_to_pp (pp, simple);
344 : :
345 : 0 : entry_t e = *const_cast <map_t &> (m_map).get (sval);
346 : 0 : pp_string (pp, ": ");
347 : 0 : e.m_state->dump_to_pp (pp);
348 : 0 : if (model)
349 : 0 : if (tree rep = model->get_representative_tree (sval))
350 : : {
351 : 0 : pp_string (pp, " (");
352 : 0 : dump_quoted_tree (pp, rep);
353 : 0 : pp_character (pp, ')');
354 : : }
355 : 0 : if (e.m_origin)
356 : : {
357 : 0 : pp_string (pp, " (origin: ");
358 : 0 : if (!flag_dump_noaddr)
359 : : {
360 : 0 : pp_pointer (pp, e.m_origin);
361 : 0 : pp_string (pp, ": ");
362 : : }
363 : 0 : e.m_origin->dump_to_pp (pp, simple);
364 : 0 : if (model)
365 : 0 : if (tree rep = model->get_representative_tree (e.m_origin))
366 : : {
367 : 0 : pp_string (pp, " (");
368 : 0 : dump_quoted_tree (pp, rep);
369 : 0 : pp_character (pp, ')');
370 : : }
371 : 0 : pp_string (pp, ")");
372 : : }
373 : :
374 : 0 : state_widget->add_child (tree_widget::make (dwi, pp));
375 : 0 : }
376 : :
377 : 0 : return state_widget;
378 : 0 : }
379 : :
380 : : /* Return true if no states have been set within this map
381 : : (all expressions are for the start state). */
382 : :
383 : : bool
384 : 11020 : sm_state_map::is_empty_p () const
385 : : {
386 : 11020 : return m_map.elements () == 0 && m_global_state == m_sm.get_start_state ();
387 : : }
388 : :
389 : : /* Generate a hash value for this sm_state_map. */
390 : :
391 : : hashval_t
392 : 8140518 : sm_state_map::hash () const
393 : : {
394 : 8140518 : hashval_t result = 0;
395 : :
396 : : /* Accumulate the result by xoring a hash for each slot, so that the
397 : : result doesn't depend on the ordering of the slots in the map. */
398 : :
399 : 8140518 : for (map_t::iterator iter = m_map.begin ();
400 : 9279712 : iter != m_map.end ();
401 : 1139194 : ++iter)
402 : : {
403 : 1139194 : inchash::hash hstate;
404 : 1139194 : hstate.add_ptr ((*iter).first);
405 : 1139194 : entry_t e = (*iter).second;
406 : 1139194 : hstate.add_int (e.m_state->get_id ());
407 : 1139194 : hstate.add_ptr (e.m_origin);
408 : 1139194 : result ^= hstate.end ();
409 : : }
410 : 8140518 : result ^= m_global_state->get_id ();
411 : :
412 : 8140518 : return result;
413 : : }
414 : :
415 : : /* Equality operator for sm_state_map. */
416 : :
417 : : bool
418 : 2778161 : sm_state_map::operator== (const sm_state_map &other) const
419 : : {
420 : 2778161 : if (m_global_state != other.m_global_state)
421 : : return false;
422 : :
423 : 2776012 : if (m_map.elements () != other.m_map.elements ())
424 : : return false;
425 : :
426 : 2608831 : for (map_t::iterator iter = m_map.begin ();
427 : 3015165 : iter != m_map.end ();
428 : 406334 : ++iter)
429 : : {
430 : 439383 : const svalue *sval = (*iter).first;
431 : 439383 : entry_t e = (*iter).second;
432 : 439383 : entry_t *other_slot = const_cast <map_t &> (other.m_map).get (sval);
433 : 439383 : if (other_slot == NULL)
434 : 33049 : return false;
435 : 845717 : if (e != *other_slot)
436 : : return false;
437 : : }
438 : :
439 : 2575782 : gcc_checking_assert (hash () == other.hash ());
440 : :
441 : : return true;
442 : : }
443 : :
444 : : /* Get the state of SVAL within this object.
445 : : States default to the start state. */
446 : :
447 : : state_machine::state_t
448 : 8349183 : sm_state_map::get_state (const svalue *sval,
449 : : const extrinsic_state &ext_state) const
450 : : {
451 : 8349183 : gcc_assert (sval);
452 : :
453 : 8349183 : sval = canonicalize_svalue (sval, ext_state);
454 : :
455 : 16698366 : if (entry_t *slot
456 : 8349183 : = const_cast <map_t &> (m_map).get (sval))
457 : 648986 : return slot->m_state;
458 : :
459 : : /* SVAL has no explicit sm-state.
460 : : If this sm allows for state inheritance, then SVAL might have implicit
461 : : sm-state inherited via a parent.
462 : : For example INIT_VAL(foo.field) might inherit taintedness state from
463 : : INIT_VAL(foo). */
464 : 7700197 : if (m_sm.inherited_state_p ())
465 : 2394851 : if (region_model_manager *mgr = ext_state.get_model_manager ())
466 : : {
467 : 2394851 : if (const initial_svalue *init_sval = sval->dyn_cast_initial_svalue ())
468 : : {
469 : 273432 : const region *reg = init_sval->get_region ();
470 : : /* Try recursing upwards (up to the base region for the
471 : : cluster). */
472 : 273432 : if (!reg->base_region_p ())
473 : 63528 : if (const region *parent_reg = reg->get_parent_region ())
474 : : {
475 : 63528 : const svalue *parent_init_sval
476 : 63528 : = mgr->get_or_create_initial_value (parent_reg);
477 : 63528 : state_machine::state_t parent_state
478 : 63528 : = get_state (parent_init_sval, ext_state);
479 : 63528 : if (parent_state)
480 : : return parent_state;
481 : : }
482 : : }
483 : 2121419 : else if (const sub_svalue *sub_sval = sval->dyn_cast_sub_svalue ())
484 : : {
485 : 9779 : const svalue *parent_sval = sub_sval->get_parent ();
486 : 19558 : if (state_machine::state_t parent_state
487 : 9779 : = get_state (parent_sval, ext_state))
488 : : return parent_state;
489 : : }
490 : : }
491 : :
492 : 15253780 : if (state_machine::state_t state
493 : 7626890 : = m_sm.alt_get_inherited_state (*this, sval, ext_state))
494 : : return state;
495 : :
496 : 7498152 : return m_sm.get_default_state (sval);
497 : : }
498 : :
499 : : /* Get the "origin" svalue for any state of SVAL. */
500 : :
501 : : const svalue *
502 : 12 : sm_state_map::get_origin (const svalue *sval,
503 : : const extrinsic_state &ext_state) const
504 : : {
505 : 12 : gcc_assert (sval);
506 : :
507 : 12 : sval = canonicalize_svalue (sval, ext_state);
508 : :
509 : 12 : entry_t *slot
510 : 12 : = const_cast <map_t &> (m_map).get (sval);
511 : 12 : if (slot)
512 : 12 : return slot->m_origin;
513 : : else
514 : : return NULL;
515 : : }
516 : :
517 : : /* Set the state of SID within MODEL to STATE, recording that
518 : : the state came from ORIGIN. */
519 : :
520 : : void
521 : 42745 : sm_state_map::set_state (region_model *model,
522 : : const svalue *sval,
523 : : state_machine::state_t state,
524 : : const svalue *origin,
525 : : const extrinsic_state &ext_state)
526 : : {
527 : 42745 : if (model == NULL)
528 : : return;
529 : :
530 : : /* Reject attempts to set state on UNKNOWN/POISONED. */
531 : 42745 : if (!sval->can_have_associated_state_p ())
532 : : return;
533 : :
534 : 37099 : equiv_class &ec = model->get_constraints ()->get_equiv_class (sval);
535 : 37099 : if (!set_state (ec, state, origin, ext_state))
536 : : return;
537 : : }
538 : :
539 : : /* Set the state of EC to STATE, recording that the state came from
540 : : ORIGIN.
541 : : Return true if any states of svalue_ids within EC changed. */
542 : :
543 : : bool
544 : 37099 : sm_state_map::set_state (const equiv_class &ec,
545 : : state_machine::state_t state,
546 : : const svalue *origin,
547 : : const extrinsic_state &ext_state)
548 : : {
549 : 37099 : bool any_changed = false;
550 : 150090 : for (const svalue *sval : ec.m_vars)
551 : 38793 : any_changed |= impl_set_state (sval, state, origin, ext_state);
552 : 37099 : return any_changed;
553 : : }
554 : :
555 : : /* Set state of SVAL to STATE, bypassing equivalence classes.
556 : : Return true if the state changed. */
557 : :
558 : : bool
559 : 186792 : sm_state_map::impl_set_state (const svalue *sval,
560 : : state_machine::state_t state,
561 : : const svalue *origin,
562 : : const extrinsic_state &ext_state)
563 : : {
564 : 186792 : sval = canonicalize_svalue (sval, ext_state);
565 : :
566 : 186792 : if (get_state (sval, ext_state) == state)
567 : : return false;
568 : :
569 : 145012 : gcc_assert (sval->can_have_associated_state_p ());
570 : :
571 : 145012 : if (m_sm.inherited_state_p ())
572 : : {
573 : 29604 : if (const compound_svalue *compound_sval
574 : 14802 : = sval->dyn_cast_compound_svalue ())
575 : 0 : for (auto iter : *compound_sval)
576 : : {
577 : 0 : const svalue *inner_sval = iter.second;
578 : 0 : if (inner_sval->can_have_associated_state_p ())
579 : 0 : impl_set_state (inner_sval, state, origin, ext_state);
580 : : }
581 : : }
582 : :
583 : : /* Special-case state 0 as the default value. */
584 : 145012 : if (state == 0)
585 : : {
586 : 1401 : if (m_map.get (sval))
587 : 1397 : m_map.remove (sval);
588 : 1401 : return true;
589 : : }
590 : 143611 : gcc_assert (sval);
591 : 143611 : m_map.put (sval, entry_t (state, origin));
592 : 143611 : return true;
593 : : }
594 : :
595 : : /* Clear any state for SVAL from this state map. */
596 : :
597 : : void
598 : 2442 : sm_state_map::clear_any_state (const svalue *sval)
599 : : {
600 : 2442 : m_map.remove (sval);
601 : 2442 : }
602 : :
603 : : /* Clear all per-svalue state within this state map. */
604 : :
605 : : void
606 : 8920 : sm_state_map::clear_all_per_svalue_state ()
607 : : {
608 : 8920 : m_map.empty ();
609 : 8920 : }
610 : :
611 : : /* Set the "global" state within this state map to STATE. */
612 : :
613 : : void
614 : 202582 : sm_state_map::set_global_state (state_machine::state_t state)
615 : : {
616 : 202582 : m_global_state = state;
617 : 202582 : }
618 : :
619 : : /* Get the "global" state within this state map. */
620 : :
621 : : state_machine::state_t
622 : 1297443 : sm_state_map::get_global_state () const
623 : : {
624 : 1297443 : return m_global_state;
625 : : }
626 : :
627 : : /* Purge any state for SVAL.
628 : : If !SM::can_purge_p, then report the state as leaking,
629 : : using CTXT. */
630 : :
631 : : void
632 : 497447 : sm_state_map::on_svalue_leak (const svalue *sval,
633 : : impl_region_model_context *ctxt)
634 : : {
635 : 497447 : if (state_machine::state_t state = get_state (sval, ctxt->m_ext_state))
636 : : {
637 : 497447 : if (m_sm.can_purge_p (state))
638 : 496782 : m_map.remove (sval);
639 : : else
640 : 665 : ctxt->on_state_leak (m_sm, sval, state);
641 : : }
642 : 497447 : }
643 : :
644 : : /* Purge any state for svalues that aren't live with respect to LIVE_SVALUES
645 : : and MODEL. */
646 : :
647 : : void
648 : 4007514 : sm_state_map::on_liveness_change (const svalue_set &live_svalues,
649 : : const region_model *model,
650 : : const extrinsic_state &ext_state,
651 : : impl_region_model_context *ctxt)
652 : : {
653 : 4007514 : svalue_set svals_to_unset;
654 : 4007514 : uncertainty_t *uncertainty = ctxt->get_uncertainty ();
655 : :
656 : 4007514 : auto_vec<const svalue *> leaked_svals (m_map.elements ());
657 : 4593181 : for (map_t::iterator iter = m_map.begin ();
658 : 4593181 : iter != m_map.end ();
659 : 585667 : ++iter)
660 : : {
661 : 585667 : const svalue *iter_sval = (*iter).first;
662 : 585667 : if (!iter_sval->live_p (&live_svalues, model))
663 : : {
664 : 12558 : svals_to_unset.add (iter_sval);
665 : 12558 : entry_t e = (*iter).second;
666 : 12558 : if (!m_sm.can_purge_p (e.m_state))
667 : 1165 : leaked_svals.quick_push (iter_sval);
668 : : }
669 : 585667 : if (uncertainty)
670 : 585667 : if (uncertainty->unknown_sm_state_p (iter_sval))
671 : 613 : svals_to_unset.add (iter_sval);
672 : : }
673 : :
674 : 4007514 : leaked_svals.qsort (svalue::cmp_ptr_ptr);
675 : :
676 : : unsigned i;
677 : : const svalue *sval;
678 : 4008679 : FOR_EACH_VEC_ELT (leaked_svals, i, sval)
679 : : {
680 : 1165 : entry_t e = *m_map.get (sval);
681 : 1165 : ctxt->on_state_leak (m_sm, sval, e.m_state);
682 : : }
683 : :
684 : 4007514 : sm_state_map old_sm_map = *this;
685 : :
686 : 4007514 : for (svalue_set::iterator iter = svals_to_unset.begin ();
687 : 4033856 : iter != svals_to_unset.end (); ++iter)
688 : 13171 : m_map.remove (*iter);
689 : :
690 : : /* For state machines like "taint" where states can be
691 : : alt-inherited from other svalues, ensure that state purging doesn't
692 : : make us lose sm-state.
693 : :
694 : : Consider e.g.:
695 : :
696 : : make_tainted(foo);
697 : : if (foo.field > 128)
698 : : return;
699 : : arr[foo.field].f1 = v1;
700 : :
701 : : where the last line is:
702 : :
703 : : (A): _t1 = foo.field;
704 : : (B): _t2 = _t1 * sizeof(arr[0]);
705 : : (C): [arr + _t2].f1 = val;
706 : :
707 : : At (A), foo is 'tainted' and foo.field is 'has_ub'.
708 : : After (B), foo.field's value (in _t1) is no longer directly
709 : : within LIVE_SVALUES, so with state purging enabled, we would
710 : : erroneously purge the "has_ub" state from the svalue.
711 : :
712 : : Given that _t2's value's state comes from _t1's value's state,
713 : : we need to preserve that information.
714 : :
715 : : Hence for all svalues that have had their explicit sm-state unset,
716 : : having their sm-state being unset, determine if doing so has changed
717 : : their effective state, and if so, explicitly set their state.
718 : :
719 : : For example, in the above, unsetting the "has_ub" for _t1's value means
720 : : that _t2's effective value changes from "has_ub" (from alt-inherited
721 : : from _t1's value) to "tainted" (inherited from "foo"'s value).
722 : :
723 : : For such cases, preserve the effective state by explicitly setting the
724 : : new state. In the above example, this means explicitly setting _t2's
725 : : value to the value ("has_ub") it was previously alt-inheriting from _t1's
726 : : value. */
727 : 4007514 : if (m_sm.has_alt_get_inherited_state_p ())
728 : : {
729 : 572370 : auto_vec<const svalue *> svalues_needing_state;
730 : 593048 : for (auto unset_sval : svals_to_unset)
731 : : {
732 : 10339 : const state_machine::state_t old_state
733 : 10339 : = old_sm_map.get_state (unset_sval, ext_state);
734 : 10339 : const state_machine::state_t new_state
735 : 10339 : = get_state (unset_sval, ext_state);
736 : 10339 : if (new_state != old_state)
737 : 10339 : svalues_needing_state.safe_push (unset_sval);
738 : : }
739 : 597973 : for (auto sval : svalues_needing_state)
740 : : {
741 : 10339 : const state_machine::state_t old_state
742 : 10339 : = old_sm_map.get_state (sval, ext_state);
743 : 10339 : impl_set_state (sval, old_state, nullptr, ext_state);
744 : : }
745 : 572370 : }
746 : 4007514 : }
747 : :
748 : : /* Purge state from SVAL (in response to a call to an unknown function). */
749 : :
750 : : void
751 : 328047 : sm_state_map::on_unknown_change (const svalue *sval,
752 : : bool is_mutable,
753 : : const extrinsic_state &ext_state)
754 : : {
755 : 328047 : svalue_set svals_to_unset;
756 : :
757 : 359815 : for (map_t::iterator iter = m_map.begin ();
758 : 359815 : iter != m_map.end ();
759 : 31768 : ++iter)
760 : : {
761 : 31768 : const svalue *key = (*iter).first;
762 : 31768 : entry_t e = (*iter).second;
763 : : /* We only want to purge state for some states when things
764 : : are mutable. For example, in sm-malloc.cc, an on-stack ptr
765 : : doesn't stop being stack-allocated when passed to an unknown fn. */
766 : 31768 : if (!m_sm.reset_when_passed_to_unknown_fn_p (e.m_state, is_mutable))
767 : 14636 : continue;
768 : 17132 : if (key == sval)
769 : 1255 : svals_to_unset.add (key);
770 : : /* If we have INIT_VAL(BASE_REG), then unset any INIT_VAL(REG)
771 : : for REG within BASE_REG. */
772 : 17132 : if (const initial_svalue *init_sval = sval->dyn_cast_initial_svalue ())
773 : 3431 : if (const initial_svalue *init_key = key->dyn_cast_initial_svalue ())
774 : : {
775 : 1491 : const region *changed_reg = init_sval->get_region ();
776 : 1491 : const region *changed_key = init_key->get_region ();
777 : 1491 : if (changed_key->get_base_region () == changed_reg)
778 : 223 : svals_to_unset.add (key);
779 : : }
780 : : }
781 : :
782 : 329303 : for (svalue_set::iterator iter = svals_to_unset.begin ();
783 : 658606 : iter != svals_to_unset.end (); ++iter)
784 : 1256 : impl_set_state (*iter, (state_machine::state_t)0, NULL, ext_state);
785 : 328047 : }
786 : :
787 : : /* Purge state for things involving SVAL.
788 : : For use when SVAL changes meaning, at the def_stmt on an SSA_NAME. */
789 : :
790 : : void
791 : 126119 : sm_state_map::purge_state_involving (const svalue *sval,
792 : : const extrinsic_state &ext_state)
793 : : {
794 : : /* Currently svalue::involves_p requires this. */
795 : 252238 : if (!(sval->get_kind () == SK_INITIAL
796 : 126119 : || sval->get_kind () == SK_CONJURED))
797 : 0 : return;
798 : :
799 : 126119 : svalue_set svals_to_unset;
800 : :
801 : 149462 : for (map_t::iterator iter = m_map.begin ();
802 : 149462 : iter != m_map.end ();
803 : 23343 : ++iter)
804 : : {
805 : 23343 : const svalue *key = (*iter).first;
806 : 23343 : entry_t e = (*iter).second;
807 : 23343 : if (!m_sm.can_purge_p (e.m_state))
808 : 9069 : continue;
809 : 14274 : if (key->involves_p (sval))
810 : 137 : svals_to_unset.add (key);
811 : : }
812 : :
813 : 126256 : for (svalue_set::iterator iter = svals_to_unset.begin ();
814 : 252512 : iter != svals_to_unset.end (); ++iter)
815 : 137 : impl_set_state (*iter, (state_machine::state_t)0, NULL, ext_state);
816 : 126119 : }
817 : :
818 : : /* Comparator for imposing an order on sm_state_map instances. */
819 : :
820 : : int
821 : 502589 : sm_state_map::cmp (const sm_state_map &smap_a, const sm_state_map &smap_b)
822 : : {
823 : 502589 : if (int cmp_count = smap_a.elements () - smap_b.elements ())
824 : : return cmp_count;
825 : :
826 : 437139 : auto_vec <const svalue *> keys_a (smap_a.elements ());
827 : 437139 : for (map_t::iterator iter = smap_a.begin ();
828 : 560561 : iter != smap_a.end ();
829 : 123422 : ++iter)
830 : 123422 : keys_a.quick_push ((*iter).first);
831 : 437139 : keys_a.qsort (svalue::cmp_ptr_ptr);
832 : :
833 : 437139 : auto_vec <const svalue *> keys_b (smap_b.elements ());
834 : 437139 : for (map_t::iterator iter = smap_b.begin ();
835 : 560561 : iter != smap_b.end ();
836 : 123422 : ++iter)
837 : 123422 : keys_b.quick_push ((*iter).first);
838 : 437139 : keys_b.qsort (svalue::cmp_ptr_ptr);
839 : :
840 : : unsigned i;
841 : : const svalue *sval_a;
842 : 584388 : FOR_EACH_VEC_ELT (keys_a, i, sval_a)
843 : : {
844 : 104750 : const svalue *sval_b = keys_b[i];
845 : 104750 : if (int cmp_sval = svalue::cmp_ptr (sval_a, sval_b))
846 : 18562 : return cmp_sval;
847 : 91811 : const entry_t *e_a = const_cast <map_t &> (smap_a.m_map).get (sval_a);
848 : 91811 : const entry_t *e_b = const_cast <map_t &> (smap_b.m_map).get (sval_b);
849 : 91811 : if (int cmp_entry = entry_t::cmp (*e_a, *e_b))
850 : : return cmp_entry;
851 : : }
852 : :
853 : : return 0;
854 : 437139 : }
855 : :
856 : : /* Canonicalize SVAL before getting/setting it within the map.
857 : : Convert all NULL pointers to (void *) to avoid state explosions
858 : : involving all of the various (foo *)NULL vs (bar *)NULL. */
859 : :
860 : : const svalue *
861 : 8535987 : sm_state_map::canonicalize_svalue (const svalue *sval,
862 : : const extrinsic_state &ext_state)
863 : : {
864 : 8535987 : region_model_manager *mgr = ext_state.get_model_manager ();
865 : 8535987 : if (mgr && sval->get_type () && POINTER_TYPE_P (sval->get_type ()))
866 : 3106291 : if (tree cst = sval->maybe_get_constant ())
867 : 56992 : if (zerop (cst))
868 : 53075 : return mgr->get_or_create_constant_svalue (null_pointer_node);
869 : :
870 : : return sval;
871 : : }
872 : :
873 : : /* Attempt to merge this state map with OTHER, writing the result
874 : : into *OUT.
875 : : Return true if the merger was possible, false otherwise.
876 : :
877 : : Normally, only identical state maps can be merged, so that
878 : : differences between state maps lead to different enodes
879 : :
880 : : However some state machines may support merging states to
881 : : allow for discarding of less important states, and thus avoid
882 : : blow-up of the exploded graph. */
883 : :
884 : : bool
885 : 1253161 : sm_state_map::can_merge_with_p (const sm_state_map &other,
886 : : const state_machine &sm,
887 : : const extrinsic_state &ext_state,
888 : : sm_state_map **out) const
889 : : {
890 : : /* If identical, then they merge trivially, with a copy. */
891 : 1253161 : if (*this == other)
892 : : {
893 : 2140310 : delete *out;
894 : 1070155 : *out = clone ();
895 : 1070155 : return true;
896 : : }
897 : :
898 : 366012 : delete *out;
899 : 183006 : *out = new sm_state_map (sm);
900 : :
901 : : /* Otherwise, attempt to merge element by element. */
902 : :
903 : : /* Try to merge global state. */
904 : 366012 : if (state_machine::state_t merged_global_state
905 : 184662 : = sm.maybe_get_merged_state (get_global_state (),
906 : : other.get_global_state ()))
907 : 181350 : (*out)->set_global_state (merged_global_state);
908 : : else
909 : : return false;
910 : :
911 : : /* Try to merge state each svalue's state (for the union
912 : : of svalues represented by each smap).
913 : : Ignore the origin information. */
914 : 181350 : hash_set<const svalue *> svals;
915 : 633495 : for (auto kv : *this)
916 : 452145 : svals.add (kv.first);
917 : 499656 : for (auto kv : other)
918 : 318306 : svals.add (kv.first);
919 : 453780 : for (auto sval : svals)
920 : : {
921 : 296565 : state_machine::state_t this_state = get_state (sval, ext_state);
922 : 296565 : state_machine::state_t other_state = other.get_state (sval, ext_state);
923 : 296565 : if (state_machine::state_t merged_state
924 : 296565 : = sm.maybe_get_merged_state (this_state, other_state))
925 : 136215 : (*out)->impl_set_state (sval, merged_state, NULL, ext_state);
926 : : else
927 : 160350 : return false;
928 : : }
929 : :
930 : : /* Successfully merged all elements. */
931 : 21000 : return true;
932 : 181350 : }
933 : :
934 : : /* class program_state. */
935 : :
936 : : /* program_state's ctor. */
937 : :
938 : 324814 : program_state::program_state (const extrinsic_state &ext_state)
939 : 324814 : : m_region_model (NULL),
940 : 649624 : m_checker_states (ext_state.get_num_checkers ()),
941 : 324814 : m_valid (true)
942 : : {
943 : 324814 : engine *eng = ext_state.get_engine ();
944 : 324814 : region_model_manager *mgr = eng->get_model_manager ();
945 : 324814 : m_region_model = new region_model (mgr);
946 : 324814 : const int num_states = ext_state.get_num_checkers ();
947 : 2598040 : for (int i = 0; i < num_states; i++)
948 : : {
949 : 2273226 : sm_state_map *sm = new sm_state_map (ext_state.get_sm (i));
950 : 2273226 : m_checker_states.quick_push (sm);
951 : : }
952 : 324814 : }
953 : :
954 : : /* Attempt to use R to replay SUMMARY into this object.
955 : : Return true if it is possible. */
956 : :
957 : : bool
958 : 9807 : sm_state_map::replay_call_summary (call_summary_replay &r,
959 : : const sm_state_map &summary)
960 : : {
961 : 14037 : for (auto kv : summary.m_map)
962 : : {
963 : 2115 : const svalue *summary_sval = kv.first;
964 : 2115 : const svalue *caller_sval = r.convert_svalue_from_summary (summary_sval);
965 : 2115 : if (!caller_sval)
966 : 556 : continue;
967 : 2115 : if (!caller_sval->can_have_associated_state_p ())
968 : 556 : continue;
969 : 1559 : const svalue *summary_origin = kv.second.m_origin;
970 : 1559 : const svalue *caller_origin
971 : : = (summary_origin
972 : 1559 : ? r.convert_svalue_from_summary (summary_origin)
973 : 1559 : : NULL);
974 : : // caller_origin can be NULL.
975 : 1559 : m_map.put (caller_sval, entry_t (kv.second.m_state, caller_origin));
976 : : }
977 : 9807 : m_global_state = summary.m_global_state;
978 : 9807 : return true;
979 : : }
980 : :
981 : : /* program_state's copy ctor. */
982 : :
983 : 2308015 : program_state::program_state (const program_state &other)
984 : 2308015 : : m_region_model (new region_model (*other.m_region_model)),
985 : 4616030 : m_checker_states (other.m_checker_states.length ()),
986 : 2308015 : m_valid (true)
987 : : {
988 : 2308015 : int i;
989 : 2308015 : sm_state_map *smap;
990 : 18441661 : FOR_EACH_VEC_ELT (other.m_checker_states, i, smap)
991 : 16133646 : m_checker_states.quick_push (smap->clone ());
992 : 2308015 : }
993 : :
994 : : /* program_state's assignment operator. */
995 : :
996 : : program_state&
997 : 298341 : program_state::operator= (const program_state &other)
998 : : {
999 : 298341 : delete m_region_model;
1000 : 298341 : m_region_model = new region_model (*other.m_region_model);
1001 : :
1002 : 298341 : int i;
1003 : 298341 : sm_state_map *smap;
1004 : 2385187 : FOR_EACH_VEC_ELT (m_checker_states, i, smap)
1005 : 4173692 : delete smap;
1006 : 298341 : m_checker_states.truncate (0);
1007 : 895023 : gcc_assert (m_checker_states.space (other.m_checker_states.length ()));
1008 : :
1009 : 2385187 : FOR_EACH_VEC_ELT (other.m_checker_states, i, smap)
1010 : 2086846 : m_checker_states.quick_push (smap->clone ());
1011 : :
1012 : 298341 : m_valid = other.m_valid;
1013 : :
1014 : 298341 : return *this;
1015 : : }
1016 : :
1017 : : /* Move constructor for program_state (when building with C++11). */
1018 : 598388 : program_state::program_state (program_state &&other)
1019 : 598388 : : m_region_model (other.m_region_model),
1020 : 1196776 : m_checker_states (other.m_checker_states.length ())
1021 : : {
1022 : 598388 : other.m_region_model = NULL;
1023 : :
1024 : 598388 : int i;
1025 : 598388 : sm_state_map *smap;
1026 : 4781718 : FOR_EACH_VEC_ELT (other.m_checker_states, i, smap)
1027 : 4183330 : m_checker_states.quick_push (smap);
1028 : 598388 : other.m_checker_states.truncate (0);
1029 : :
1030 : 598388 : m_valid = other.m_valid;
1031 : 598388 : }
1032 : :
1033 : : /* program_state's dtor. */
1034 : :
1035 : 3231217 : program_state::~program_state ()
1036 : : {
1037 : 3231217 : delete m_region_model;
1038 : 3231217 : }
1039 : :
1040 : : /* Generate a hash value for this program_state. */
1041 : :
1042 : : hashval_t
1043 : 427572 : program_state::hash () const
1044 : : {
1045 : 427572 : hashval_t result = m_region_model->hash ();
1046 : :
1047 : 427572 : int i;
1048 : 427572 : sm_state_map *smap;
1049 : 3844058 : FOR_EACH_VEC_ELT (m_checker_states, i, smap)
1050 : 2988914 : result ^= smap->hash ();
1051 : 427572 : return result;
1052 : : }
1053 : :
1054 : : /* Equality operator for program_state.
1055 : : All parts of the program_state (region model, checker states) must
1056 : : equal their counterparts in OTHER for the two program_states to be
1057 : : considered equal. */
1058 : :
1059 : : bool
1060 : 71973 : program_state::operator== (const program_state &other) const
1061 : : {
1062 : 71973 : if (!(*m_region_model == *other.m_region_model))
1063 : : return false;
1064 : :
1065 : : int i;
1066 : : sm_state_map *smap;
1067 : 69879 : FOR_EACH_VEC_ELT (m_checker_states, i, smap)
1068 : 61411 : if (!(*smap == *other.m_checker_states[i]))
1069 : : return false;
1070 : :
1071 : 8468 : gcc_checking_assert (hash () == other.hash ());
1072 : :
1073 : : return true;
1074 : : }
1075 : :
1076 : : /* Print a compact representation of this state to PP. */
1077 : :
1078 : : void
1079 : 0 : program_state::print (const extrinsic_state &ext_state,
1080 : : pretty_printer *pp) const
1081 : : {
1082 : 0 : pp_printf (pp, "rmodel: ");
1083 : 0 : m_region_model->dump_to_pp (pp, true, false);
1084 : 0 : pp_newline (pp);
1085 : :
1086 : 0 : int i;
1087 : 0 : sm_state_map *smap;
1088 : 0 : FOR_EACH_VEC_ELT (m_checker_states, i, smap)
1089 : : {
1090 : 0 : if (!smap->is_empty_p ())
1091 : : {
1092 : 0 : pp_printf (pp, "%s: ", ext_state.get_name (i));
1093 : 0 : smap->print (m_region_model, true, false, pp);
1094 : 0 : pp_newline (pp);
1095 : : }
1096 : : }
1097 : 0 : if (!m_valid)
1098 : : {
1099 : 0 : pp_printf (pp, "invalid state");
1100 : 0 : pp_newline (pp);
1101 : : }
1102 : 0 : }
1103 : :
1104 : : /* Dump a representation of this state to PP. */
1105 : :
1106 : : void
1107 : 1572 : program_state::dump_to_pp (const extrinsic_state &ext_state,
1108 : : bool /*summarize*/, bool multiline,
1109 : : pretty_printer *pp) const
1110 : : {
1111 : 1572 : if (!multiline)
1112 : 964 : pp_string (pp, "{");
1113 : 1572 : {
1114 : 1572 : pp_printf (pp, "rmodel:");
1115 : 1572 : if (multiline)
1116 : 608 : pp_newline (pp);
1117 : : else
1118 : 964 : pp_string (pp, " {");
1119 : 1572 : m_region_model->dump_to_pp (pp, true, multiline);
1120 : 1572 : if (!multiline)
1121 : 964 : pp_string (pp, "}");
1122 : : }
1123 : :
1124 : : int i;
1125 : : sm_state_map *smap;
1126 : 12576 : FOR_EACH_VEC_ELT (m_checker_states, i, smap)
1127 : : {
1128 : 11004 : if (!smap->is_empty_p ())
1129 : : {
1130 : 498 : if (!multiline)
1131 : 98 : pp_string (pp, " {");
1132 : 498 : pp_printf (pp, "%s: ", ext_state.get_name (i));
1133 : 498 : if (multiline)
1134 : 400 : pp_newline (pp);
1135 : 498 : smap->print (m_region_model, true, multiline, pp);
1136 : 498 : if (!multiline)
1137 : 98 : pp_string (pp, "}");
1138 : : }
1139 : : }
1140 : :
1141 : 1572 : if (!m_valid)
1142 : : {
1143 : 0 : if (!multiline)
1144 : 0 : pp_space (pp);
1145 : 0 : pp_printf (pp, "invalid state");
1146 : 0 : if (multiline)
1147 : 0 : pp_newline (pp);
1148 : : }
1149 : 1572 : if (!multiline)
1150 : 964 : pp_string (pp, "}");
1151 : 1572 : }
1152 : :
1153 : : /* Dump a representation of this state to OUTF. */
1154 : :
1155 : : void
1156 : 0 : program_state::dump_to_file (const extrinsic_state &ext_state,
1157 : : bool summarize, bool multiline,
1158 : : FILE *outf) const
1159 : : {
1160 : 0 : tree_dump_pretty_printer pp (outf);
1161 : 0 : dump_to_pp (ext_state, summarize, multiline, &pp);
1162 : 0 : }
1163 : :
1164 : : /* Dump a multiline representation of this state to stderr. */
1165 : :
1166 : : DEBUG_FUNCTION void
1167 : 0 : program_state::dump (const extrinsic_state &ext_state,
1168 : : bool summarize) const
1169 : : {
1170 : 0 : dump_to_file (ext_state, summarize, true, stderr);
1171 : 0 : }
1172 : :
1173 : : /* Dump a tree-like representation of this state to stderr. */
1174 : :
1175 : : DEBUG_FUNCTION void
1176 : 0 : program_state::dump () const
1177 : : {
1178 : 0 : text_art::dump (*this);
1179 : 0 : }
1180 : :
1181 : : /* Return a new json::object of the form
1182 : : {"store" : object for store,
1183 : : "constraints" : object for constraint_manager,
1184 : : "curr_frame" : (optional) str for current frame,
1185 : : "checkers" : { STATE_NAME : object per sm_state_map },
1186 : : "valid" : true/false}. */
1187 : :
1188 : : std::unique_ptr<json::object>
1189 : 0 : program_state::to_json (const extrinsic_state &ext_state) const
1190 : : {
1191 : 0 : auto state_obj = ::make_unique<json::object> ();
1192 : :
1193 : 0 : state_obj->set ("store", m_region_model->get_store ()->to_json ());
1194 : 0 : state_obj->set ("constraints",
1195 : 0 : m_region_model->get_constraints ()->to_json ());
1196 : 0 : if (m_region_model->get_current_frame ())
1197 : 0 : state_obj->set ("curr_frame",
1198 : 0 : m_region_model->get_current_frame ()->to_json ());
1199 : :
1200 : : /* Provide m_checker_states as an object, using names as keys. */
1201 : 0 : {
1202 : 0 : auto checkers_obj = ::make_unique<json::object> ();
1203 : :
1204 : 0 : int i;
1205 : 0 : sm_state_map *smap;
1206 : 0 : FOR_EACH_VEC_ELT (m_checker_states, i, smap)
1207 : 0 : if (!smap->is_empty_p ())
1208 : 0 : checkers_obj->set (ext_state.get_name (i), smap->to_json ());
1209 : :
1210 : 0 : state_obj->set ("checkers", std::move (checkers_obj));
1211 : 0 : }
1212 : :
1213 : 0 : state_obj->set_bool ("valid", m_valid);
1214 : :
1215 : 0 : return state_obj;
1216 : : }
1217 : :
1218 : :
1219 : : std::unique_ptr<text_art::tree_widget>
1220 : 0 : program_state::make_dump_widget (const text_art::dump_widget_info &dwi) const
1221 : : {
1222 : 0 : using text_art::tree_widget;
1223 : 0 : std::unique_ptr<tree_widget> state_widget
1224 : 0 : (tree_widget::from_fmt (dwi, nullptr, "State"));
1225 : :
1226 : 0 : state_widget->add_child (m_region_model->make_dump_widget (dwi));
1227 : :
1228 : : /* Add nodes for any sm_state_maps with state. */
1229 : 0 : {
1230 : 0 : int i;
1231 : 0 : sm_state_map *smap;
1232 : 0 : FOR_EACH_VEC_ELT (m_checker_states, i, smap)
1233 : 0 : if (!smap->is_empty_p ())
1234 : 0 : state_widget->add_child (smap->make_dump_widget (dwi, m_region_model));
1235 : : }
1236 : :
1237 : 0 : return state_widget;
1238 : : }
1239 : :
1240 : : /* Update this program_state to reflect a top-level call to FUN.
1241 : : The params will have initial_svalues. */
1242 : :
1243 : : void
1244 : 9586 : program_state::push_frame (const extrinsic_state &ext_state ATTRIBUTE_UNUSED,
1245 : : const function &fun)
1246 : : {
1247 : 9586 : m_region_model->push_frame (fun, NULL, NULL);
1248 : 9586 : }
1249 : :
1250 : : /* Get the current function of this state. */
1251 : :
1252 : : const function *
1253 : 22 : program_state::get_current_function () const
1254 : : {
1255 : 22 : return m_region_model->get_current_function ();
1256 : : }
1257 : :
1258 : : /* Determine if following edge SUCC from ENODE is valid within the graph EG
1259 : : and update this state accordingly in-place.
1260 : :
1261 : : Return true if the edge can be followed, or false otherwise.
1262 : :
1263 : : Check for relevant conditionals and switch-values for conditionals
1264 : : and switch statements, adding the relevant conditions to this state.
1265 : : Push/pop frames for interprocedural edges and update params/returned
1266 : : values.
1267 : :
1268 : : This is the "state" half of exploded_node::on_edge. */
1269 : :
1270 : : bool
1271 : 123899 : program_state::on_edge (exploded_graph &eg,
1272 : : exploded_node *enode,
1273 : : const superedge *succ,
1274 : : uncertainty_t *uncertainty)
1275 : : {
1276 : 247798 : class my_path_context : public path_context
1277 : : {
1278 : : public:
1279 : 123899 : my_path_context (bool &terminated) : m_terminated (terminated) {}
1280 : 0 : void bifurcate (std::unique_ptr<custom_edge_info>) final override
1281 : : {
1282 : 0 : gcc_unreachable ();
1283 : : }
1284 : :
1285 : 28 : void terminate_path () final override
1286 : : {
1287 : 28 : m_terminated = true;
1288 : 28 : }
1289 : :
1290 : 0 : bool terminate_path_p () const final override
1291 : : {
1292 : 0 : return m_terminated;
1293 : : }
1294 : : bool &m_terminated;
1295 : : };
1296 : :
1297 : : /* Update state. */
1298 : 123899 : const program_point &point = enode->get_point ();
1299 : 123899 : const gimple *last_stmt = point.get_supernode ()->get_last_stmt ();
1300 : :
1301 : : /* For conditionals and switch statements, add the
1302 : : relevant conditions (for the specific edge) to new_state;
1303 : : skip edges for which the resulting constraints
1304 : : are impossible.
1305 : : This also updates frame information for call/return superedges.
1306 : : Adding the relevant conditions for the edge could also trigger
1307 : : sm-state transitions (e.g. transitions due to ptrs becoming known
1308 : : to be NULL or non-NULL) */
1309 : 123899 : bool terminated = false;
1310 : 123899 : my_path_context path_ctxt (terminated);
1311 : 123899 : impl_region_model_context ctxt (eg, enode,
1312 : 123899 : &enode->get_state (),
1313 : : this,
1314 : : uncertainty, &path_ctxt,
1315 : 123899 : last_stmt);
1316 : 123899 : std::unique_ptr<rejected_constraint> rc;
1317 : 123899 : logger * const logger = eg.get_logger ();
1318 : 247757 : if (!m_region_model->maybe_update_for_edge (*succ,
1319 : : last_stmt,
1320 : : &ctxt,
1321 : : logger ? &rc : nullptr))
1322 : : {
1323 : 6019 : if (logger)
1324 : : {
1325 : 0 : logger->start_log_line ();
1326 : 0 : logger->log_partial ("edge to SN: %i is impossible"
1327 : : " due to region_model constraint: ",
1328 : 0 : succ->m_dest->m_index);
1329 : 0 : rc->dump_to_pp (logger->get_printer ());
1330 : 0 : logger->end_log_line ();
1331 : : }
1332 : 6019 : return false;
1333 : : }
1334 : 117880 : if (terminated)
1335 : : return false;
1336 : :
1337 : 117852 : program_state::detect_leaks (enode->get_state (), *this,
1338 : : NULL, eg.get_ext_state (),
1339 : : &ctxt);
1340 : :
1341 : 117852 : return true;
1342 : 123899 : }
1343 : :
1344 : : /* Update this program_state to reflect a call to function
1345 : : represented by CALL_STMT.
1346 : : currently used only when the call doesn't have a superedge representing
1347 : : the call ( like call via a function pointer ) */
1348 : : void
1349 : 77 : program_state::push_call (exploded_graph &eg,
1350 : : exploded_node *enode,
1351 : : const gcall *call_stmt,
1352 : : uncertainty_t *uncertainty)
1353 : : {
1354 : : /* Update state. */
1355 : 77 : const program_point &point = enode->get_point ();
1356 : 77 : const gimple *last_stmt = point.get_supernode ()->get_last_stmt ();
1357 : :
1358 : 77 : impl_region_model_context ctxt (eg, enode,
1359 : 77 : &enode->get_state (),
1360 : : this,
1361 : : uncertainty,
1362 : : NULL,
1363 : 77 : last_stmt);
1364 : 77 : m_region_model->update_for_gcall (call_stmt, &ctxt);
1365 : 77 : }
1366 : :
1367 : : /* Update this program_state to reflect a return from function
1368 : : call to which is represented by CALL_STMT.
1369 : : currently used only when the call doesn't have a superedge representing
1370 : : the return */
1371 : : void
1372 : 67 : program_state::returning_call (exploded_graph &eg,
1373 : : exploded_node *enode,
1374 : : const gcall *call_stmt,
1375 : : uncertainty_t *uncertainty)
1376 : : {
1377 : : /* Update state. */
1378 : 67 : const program_point &point = enode->get_point ();
1379 : 67 : const gimple *last_stmt = point.get_supernode ()->get_last_stmt ();
1380 : :
1381 : 67 : impl_region_model_context ctxt (eg, enode,
1382 : 67 : &enode->get_state (),
1383 : : this,
1384 : : uncertainty,
1385 : : NULL,
1386 : 67 : last_stmt);
1387 : 67 : m_region_model->update_for_return_gcall (call_stmt, &ctxt);
1388 : 67 : }
1389 : :
1390 : : /* Generate a simpler version of THIS, discarding state that's no longer
1391 : : relevant at POINT.
1392 : : The idea is that we're more likely to be able to consolidate
1393 : : multiple (point, state) into single exploded_nodes if we discard
1394 : : irrelevant state (e.g. at the end of functions). */
1395 : :
1396 : : program_state
1397 : 601568 : program_state::prune_for_point (exploded_graph &eg,
1398 : : const program_point &point,
1399 : : exploded_node *enode_for_diag,
1400 : : uncertainty_t *uncertainty) const
1401 : : {
1402 : 601568 : logger * const logger = eg.get_logger ();
1403 : 601568 : LOG_SCOPE (logger);
1404 : :
1405 : 601568 : function *fun = point.get_function ();
1406 : 601568 : if (!fun)
1407 : 3180 : return *this;
1408 : :
1409 : 598388 : program_state new_state (*this);
1410 : :
1411 : 598388 : const state_purge_map *pm = eg.get_purge_map ();
1412 : 598388 : if (pm)
1413 : : {
1414 : 597640 : unsigned num_ssas_purged = 0;
1415 : 597640 : unsigned num_decls_purged = 0;
1416 : 597640 : auto_vec<const decl_region *> regs;
1417 : 597640 : new_state.m_region_model->get_regions_for_current_frame (®s);
1418 : 597640 : regs.qsort (region::cmp_ptr_ptr);
1419 : : unsigned i;
1420 : : const decl_region *reg;
1421 : 1928622 : FOR_EACH_VEC_ELT (regs, i, reg)
1422 : : {
1423 : 1330982 : const tree node = reg->get_decl ();
1424 : 1330982 : if (TREE_CODE (node) == SSA_NAME)
1425 : : {
1426 : 1117492 : const tree ssa_name = node;
1427 : 1117492 : const state_purge_per_ssa_name &per_ssa
1428 : 1117492 : = pm->get_data_for_ssa_name (node);
1429 : 1117492 : if (!per_ssa.needed_at_point_p (point.get_function_point ()))
1430 : : {
1431 : : /* Don't purge bindings of SSA names to svalues
1432 : : that have unpurgable sm-state, so that leaks are
1433 : : reported at the end of the function, rather than
1434 : : at the last place that such an SSA name is referred to.
1435 : :
1436 : : But do purge them for temporaries (when SSA_NAME_VAR is
1437 : : NULL), so that we report for cases where a leak happens when
1438 : : a variable is overwritten with another value, so that the leak
1439 : : is reported at the point of overwrite, rather than having
1440 : : temporaries keep the value reachable until the frame is
1441 : : popped. */
1442 : 211786 : const svalue *sval
1443 : 211786 : = new_state.m_region_model->get_store_value (reg, NULL);
1444 : 211786 : if (!new_state.can_purge_p (eg.get_ext_state (), sval)
1445 : 256879 : && SSA_NAME_VAR (ssa_name))
1446 : : {
1447 : : /* (currently only state maps can keep things
1448 : : alive). */
1449 : 37943 : if (logger)
1450 : 0 : logger->log ("not purging binding for %qE"
1451 : : " (used by state map)", ssa_name);
1452 : 37943 : continue;
1453 : : }
1454 : :
1455 : 173843 : new_state.m_region_model->purge_region (reg);
1456 : 173843 : num_ssas_purged++;
1457 : : }
1458 : : }
1459 : : else
1460 : : {
1461 : 213490 : const tree decl = node;
1462 : 213490 : gcc_assert (TREE_CODE (node) == VAR_DECL
1463 : : || TREE_CODE (node) == PARM_DECL
1464 : : || TREE_CODE (node) == RESULT_DECL);
1465 : 426980 : if (const state_purge_per_decl *per_decl
1466 : 213490 : = pm->get_any_data_for_decl (decl))
1467 : 170110 : if (!per_decl->needed_at_point_p (point.get_function_point ()))
1468 : : {
1469 : : /* Don't purge bindings of decls if there are svalues
1470 : : that have unpurgable sm-state within the decl's cluster,
1471 : : so that leaks are reported at the end of the function,
1472 : : rather than at the last place that such a decl is
1473 : : referred to. */
1474 : 6138 : if (!new_state.can_purge_base_region_p (eg.get_ext_state (),
1475 : : reg))
1476 : : {
1477 : : /* (currently only state maps can keep things
1478 : : alive). */
1479 : 1382 : if (logger)
1480 : 0 : logger->log ("not purging binding for %qE"
1481 : : " (value in binding used by state map)",
1482 : : decl);
1483 : 1382 : continue;
1484 : : }
1485 : :
1486 : 4756 : new_state.m_region_model->purge_region (reg);
1487 : 4756 : num_decls_purged++;
1488 : : }
1489 : : }
1490 : : }
1491 : :
1492 : 597640 : if (num_ssas_purged > 0 || num_decls_purged > 0)
1493 : : {
1494 : 140848 : if (logger)
1495 : : {
1496 : 77 : logger->log ("num_ssas_purged: %i", num_ssas_purged);
1497 : 77 : logger->log ("num_decl_purged: %i", num_decls_purged);
1498 : : }
1499 : 140848 : impl_region_model_context ctxt (eg, enode_for_diag,
1500 : : this,
1501 : : &new_state,
1502 : : uncertainty, NULL,
1503 : 140848 : point.get_stmt ());
1504 : 140848 : detect_leaks (*this, new_state, NULL, eg.get_ext_state (), &ctxt);
1505 : 140848 : }
1506 : 597640 : }
1507 : :
1508 : 598388 : new_state.m_region_model->canonicalize ();
1509 : :
1510 : 598388 : return new_state;
1511 : 601568 : }
1512 : :
1513 : : /* Return true if there are no unpurgeable bindings within BASE_REG. */
1514 : :
1515 : : bool
1516 : 6138 : program_state::can_purge_base_region_p (const extrinsic_state &ext_state,
1517 : : const region *base_reg) const
1518 : : {
1519 : 6138 : binding_cluster *cluster
1520 : 6138 : = m_region_model->get_store ()->get_cluster (base_reg);
1521 : 6138 : if (!cluster)
1522 : : return true;
1523 : :
1524 : 26176 : for (auto iter : *cluster)
1525 : : {
1526 : 11401 : const svalue *sval = iter.second;
1527 : 11401 : if (!can_purge_p (ext_state, sval))
1528 : 1382 : return false;
1529 : : }
1530 : :
1531 : 4756 : return true;
1532 : : }
1533 : :
1534 : : /* Get a representative tree to use for describing SVAL. */
1535 : :
1536 : : tree
1537 : 0 : program_state::get_representative_tree (const svalue *sval) const
1538 : : {
1539 : 0 : gcc_assert (m_region_model);
1540 : 0 : return m_region_model->get_representative_tree (sval);
1541 : : }
1542 : :
1543 : : /* Attempt to merge this state with OTHER, both at POINT.
1544 : : Write the result to *OUT.
1545 : : If the states were merged successfully, return true. */
1546 : :
1547 : : bool
1548 : 312018 : program_state::can_merge_with_p (const program_state &other,
1549 : : const extrinsic_state &ext_state,
1550 : : const program_point &point,
1551 : : program_state *out) const
1552 : : {
1553 : 312018 : gcc_assert (out);
1554 : 312018 : gcc_assert (m_region_model);
1555 : :
1556 : : /* Attempt to merge the sm-states. */
1557 : : int i;
1558 : : sm_state_map *smap;
1559 : 1403173 : FOR_EACH_VEC_ELT (out->m_checker_states, i, smap)
1560 : 2506322 : if (!m_checker_states[i]->can_merge_with_p (*other.m_checker_states[i],
1561 : : ext_state.get_sm (i),
1562 : : ext_state,
1563 : 1253161 : &out->m_checker_states[i]))
1564 : : return false;
1565 : :
1566 : : /* Attempt to merge the region_models. */
1567 : 150012 : if (!m_region_model->can_merge_with_p (*other.m_region_model,
1568 : : point,
1569 : : out->m_region_model,
1570 : : &ext_state,
1571 : : this, &other))
1572 : : return false;
1573 : :
1574 : 60666 : out->m_region_model->canonicalize ();
1575 : :
1576 : 60666 : return true;
1577 : : }
1578 : :
1579 : : /* Assert that this object is valid. */
1580 : :
1581 : : void
1582 : 1641136 : program_state::validate (const extrinsic_state &ext_state) const
1583 : : {
1584 : : /* Skip this in a release build. */
1585 : : #if !CHECKING_P
1586 : : return;
1587 : : #endif
1588 : :
1589 : 4923408 : gcc_assert (m_checker_states.length () == ext_state.get_num_checkers ());
1590 : 1641136 : m_region_model->validate ();
1591 : 1641136 : }
1592 : :
1593 : : static void
1594 : 496 : log_set_of_svalues (logger *logger, const char *name,
1595 : : const svalue_set &set)
1596 : : {
1597 : 496 : logger->log (name);
1598 : 496 : logger->inc_indent ();
1599 : 496 : auto_vec<const svalue *> sval_vecs (set.elements ());
1600 : 496 : for (svalue_set::iterator iter = set.begin ();
1601 : 4556 : iter != set.end (); ++iter)
1602 : 2030 : sval_vecs.quick_push (*iter);
1603 : 496 : sval_vecs.qsort (svalue::cmp_ptr_ptr);
1604 : : unsigned i;
1605 : : const svalue *sval;
1606 : 2526 : FOR_EACH_VEC_ELT (sval_vecs, i, sval)
1607 : : {
1608 : 2030 : logger->start_log_line ();
1609 : 2030 : pretty_printer *pp = logger->get_printer ();
1610 : 2030 : if (!flag_dump_noaddr)
1611 : : {
1612 : 2030 : pp_pointer (pp, sval);
1613 : 2030 : pp_string (pp, ": ");
1614 : : }
1615 : 2030 : sval->dump_to_pp (pp, false);
1616 : 2030 : logger->end_log_line ();
1617 : : }
1618 : 496 : logger->dec_indent ();
1619 : 496 : }
1620 : :
1621 : : /* Compare the sets of svalues reachable from each of SRC_STATE and DEST_STATE.
1622 : : For all svalues that are reachable in SRC_STATE and are not live in
1623 : : DEST_STATE (whether explicitly reachable in DEST_STATE, or implicitly live
1624 : : based on the former set), call CTXT->on_svalue_leak for them.
1625 : :
1626 : : Call on_liveness_change on both the CTXT and on the DEST_STATE's
1627 : : constraint_manager, purging dead svalues from sm-state and from
1628 : : constraints, respectively.
1629 : :
1630 : : This function should be called at each fine-grained state change, not
1631 : : just at exploded edges. */
1632 : :
1633 : : void
1634 : 573110 : program_state::detect_leaks (const program_state &src_state,
1635 : : const program_state &dest_state,
1636 : : const svalue *extra_sval,
1637 : : const extrinsic_state &ext_state,
1638 : : region_model_context *ctxt)
1639 : : {
1640 : 573110 : logger *logger = ext_state.get_logger ();
1641 : 573110 : LOG_SCOPE (logger);
1642 : 573110 : const uncertainty_t *uncertainty = ctxt->get_uncertainty ();
1643 : 573110 : if (logger)
1644 : : {
1645 : 248 : pretty_printer *pp = logger->get_printer ();
1646 : 248 : logger->start_log_line ();
1647 : 248 : pp_string (pp, "src_state: ");
1648 : 248 : src_state.dump_to_pp (ext_state, true, false, pp);
1649 : 248 : logger->end_log_line ();
1650 : 248 : logger->start_log_line ();
1651 : 248 : pp_string (pp, "dest_state: ");
1652 : 248 : dest_state.dump_to_pp (ext_state, true, false, pp);
1653 : 248 : logger->end_log_line ();
1654 : 248 : if (extra_sval)
1655 : : {
1656 : 0 : logger->start_log_line ();
1657 : 0 : pp_string (pp, "extra_sval: ");
1658 : 0 : extra_sval->dump_to_pp (pp, true);
1659 : 0 : logger->end_log_line ();
1660 : : }
1661 : 248 : if (uncertainty)
1662 : : {
1663 : 248 : logger->start_log_line ();
1664 : 248 : pp_string (pp, "uncertainty: ");
1665 : 248 : uncertainty->dump_to_pp (pp, true);
1666 : 248 : logger->end_log_line ();
1667 : : }
1668 : : }
1669 : :
1670 : : /* Get svalues reachable from each of src_state and dest_state.
1671 : : Get svalues *known* to be reachable in src_state.
1672 : : Pass in uncertainty for dest_state so that we additionally get svalues that
1673 : : *might* still be reachable in dst_state. */
1674 : 573110 : svalue_set known_src_svalues;
1675 : 573110 : src_state.m_region_model->get_reachable_svalues (&known_src_svalues,
1676 : : NULL, NULL);
1677 : 573110 : svalue_set maybe_dest_svalues;
1678 : 573110 : dest_state.m_region_model->get_reachable_svalues (&maybe_dest_svalues,
1679 : : extra_sval, uncertainty);
1680 : :
1681 : 573110 : if (logger)
1682 : : {
1683 : 248 : log_set_of_svalues (logger, "src_state known reachable svalues:",
1684 : : known_src_svalues);
1685 : 248 : log_set_of_svalues (logger, "dest_state maybe reachable svalues:",
1686 : : maybe_dest_svalues);
1687 : : }
1688 : :
1689 : 573110 : auto_vec <const svalue *> dead_svals (known_src_svalues.elements ());
1690 : 3606101 : for (svalue_set::iterator iter = known_src_svalues.begin ();
1691 : 6639092 : iter != known_src_svalues.end (); ++iter)
1692 : : {
1693 : 3032991 : const svalue *sval = (*iter);
1694 : : /* For each sval reachable from SRC_STATE, determine if it is
1695 : : live in DEST_STATE: either explicitly reachable, implicitly
1696 : : live based on the set of explicitly reachable svalues,
1697 : : or possibly reachable as recorded in uncertainty.
1698 : : Record those that have ceased to be live i.e. were known
1699 : : to be live, and are now not known to be even possibly-live. */
1700 : 3032991 : if (!sval->live_p (&maybe_dest_svalues, dest_state.m_region_model))
1701 : 71166 : dead_svals.quick_push (sval);
1702 : : }
1703 : :
1704 : : /* Call CTXT->on_svalue_leak on all svals in SRC_STATE that have ceased
1705 : : to be live, sorting them first to ensure deterministic behavior. */
1706 : 573110 : dead_svals.qsort (svalue::cmp_ptr_ptr);
1707 : : unsigned i;
1708 : : const svalue *sval;
1709 : 644276 : FOR_EACH_VEC_ELT (dead_svals, i, sval)
1710 : 71166 : ctxt->on_svalue_leak (sval);
1711 : :
1712 : : /* Purge dead svals from sm-state. */
1713 : 573110 : ctxt->on_liveness_change (maybe_dest_svalues,
1714 : 573110 : dest_state.m_region_model);
1715 : :
1716 : : /* Purge dead svals from constraints. */
1717 : 573110 : dest_state.m_region_model->get_constraints ()->on_liveness_change
1718 : 573110 : (maybe_dest_svalues, dest_state.m_region_model);
1719 : :
1720 : : /* Purge dead heap-allocated regions from dynamic extents. */
1721 : 1641448 : for (const svalue *sval : dead_svals)
1722 : 71166 : if (const region *reg = sval->maybe_get_region ())
1723 : 11954 : if (reg->get_kind () == RK_HEAP_ALLOCATED)
1724 : 6632 : dest_state.m_region_model->unset_dynamic_extents (reg);
1725 : 573110 : }
1726 : :
1727 : : /* Attempt to use R to replay SUMMARY into this object.
1728 : : Return true if it is possible. */
1729 : :
1730 : : bool
1731 : 1486 : program_state::replay_call_summary (call_summary_replay &r,
1732 : : const program_state &summary)
1733 : : {
1734 : 1486 : if (!m_region_model->replay_call_summary (r, *summary.m_region_model))
1735 : : return false;
1736 : :
1737 : 11208 : for (unsigned sm_idx = 0; sm_idx < m_checker_states.length (); sm_idx++)
1738 : : {
1739 : 9807 : const sm_state_map *summary_sm_map = summary.m_checker_states[sm_idx];
1740 : 9807 : m_checker_states[sm_idx]->replay_call_summary (r, *summary_sm_map);
1741 : : }
1742 : :
1743 : 1401 : if (!summary.m_valid)
1744 : 0 : m_valid = false;
1745 : :
1746 : : return true;
1747 : : }
1748 : :
1749 : : /* Handle calls to "__analyzer_dump_state". */
1750 : :
1751 : : void
1752 : 361 : program_state::impl_call_analyzer_dump_state (const gcall *call,
1753 : : const extrinsic_state &ext_state,
1754 : : region_model_context *ctxt)
1755 : : {
1756 : 361 : call_details cd (call, m_region_model, ctxt);
1757 : 361 : const char *sm_name = cd.get_arg_string_literal (0);
1758 : 361 : if (!sm_name)
1759 : : {
1760 : 4 : error_at (call->location, "cannot determine state machine");
1761 : 16 : return;
1762 : : }
1763 : 357 : unsigned sm_idx;
1764 : 357 : if (!ext_state.get_sm_idx_by_name (sm_name, &sm_idx))
1765 : : {
1766 : 8 : error_at (call->location, "unrecognized state machine %qs", sm_name);
1767 : 8 : return;
1768 : : }
1769 : 349 : const sm_state_map *smap = m_checker_states[sm_idx];
1770 : :
1771 : 349 : const svalue *sval = cd.get_arg_svalue (1);
1772 : :
1773 : : /* Strip off cast to int (due to variadic args). */
1774 : 349 : if (const svalue *cast = sval->maybe_undo_cast ())
1775 : 5 : sval = cast;
1776 : :
1777 : 349 : state_machine::state_t state = smap->get_state (sval, ext_state);
1778 : 349 : warning_at (call->location, 0, "state: %qs", state->get_name ());
1779 : : }
1780 : :
1781 : : #if CHECKING_P
1782 : :
1783 : : namespace selftest {
1784 : :
1785 : : /* Tests for sm_state_map. */
1786 : :
1787 : : static void
1788 : 4 : test_sm_state_map ()
1789 : : {
1790 : 4 : tree x = build_global_decl ("x", integer_type_node);
1791 : 4 : tree y = build_global_decl ("y", integer_type_node);
1792 : 4 : tree z = build_global_decl ("z", integer_type_node);
1793 : :
1794 : 4 : state_machine *sm = make_malloc_state_machine (NULL);
1795 : 4 : auto_delete_vec <state_machine> checkers;
1796 : 4 : checkers.safe_push (sm);
1797 : 4 : engine eng;
1798 : 4 : extrinsic_state ext_state (checkers, &eng);
1799 : 4 : state_machine::state_t start = sm->get_start_state ();
1800 : :
1801 : : /* Test setting states on svalue_id instances directly. */
1802 : 4 : {
1803 : 4 : const state_machine::state test_state_42 ("test state 42", 42);
1804 : 4 : const state_machine::state_t TEST_STATE_42 = &test_state_42;
1805 : 4 : region_model_manager mgr;
1806 : 4 : region_model model (&mgr);
1807 : 4 : const svalue *x_sval = model.get_rvalue (x, NULL);
1808 : 4 : const svalue *y_sval = model.get_rvalue (y, NULL);
1809 : 4 : const svalue *z_sval = model.get_rvalue (z, NULL);
1810 : :
1811 : 4 : sm_state_map map (*sm);
1812 : 4 : ASSERT_TRUE (map.is_empty_p ());
1813 : 4 : ASSERT_EQ (map.get_state (x_sval, ext_state), start);
1814 : :
1815 : 4 : map.impl_set_state (x_sval, TEST_STATE_42, z_sval, ext_state);
1816 : 4 : ASSERT_EQ (map.get_state (x_sval, ext_state), TEST_STATE_42);
1817 : 4 : ASSERT_EQ (map.get_origin (x_sval, ext_state), z_sval);
1818 : 4 : ASSERT_EQ (map.get_state (y_sval, ext_state), start);
1819 : 4 : ASSERT_FALSE (map.is_empty_p ());
1820 : :
1821 : 4 : map.impl_set_state (y_sval, 0, z_sval, ext_state);
1822 : 4 : ASSERT_EQ (map.get_state (y_sval, ext_state), start);
1823 : :
1824 : 4 : map.impl_set_state (x_sval, 0, z_sval, ext_state);
1825 : 4 : ASSERT_EQ (map.get_state (x_sval, ext_state), start);
1826 : 4 : ASSERT_TRUE (map.is_empty_p ());
1827 : 4 : }
1828 : :
1829 : 4 : const state_machine::state test_state_5 ("test state 5", 5);
1830 : 4 : const state_machine::state_t TEST_STATE_5 = &test_state_5;
1831 : :
1832 : : /* Test setting states via equivalence classes. */
1833 : 4 : {
1834 : 4 : region_model_manager mgr;
1835 : 4 : region_model model (&mgr);
1836 : 4 : const svalue *x_sval = model.get_rvalue (x, NULL);
1837 : 4 : const svalue *y_sval = model.get_rvalue (y, NULL);
1838 : 4 : const svalue *z_sval = model.get_rvalue (z, NULL);
1839 : :
1840 : 4 : sm_state_map map (*sm);
1841 : 4 : ASSERT_TRUE (map.is_empty_p ());
1842 : 4 : ASSERT_EQ (map.get_state (x_sval, ext_state), start);
1843 : 4 : ASSERT_EQ (map.get_state (y_sval, ext_state), start);
1844 : :
1845 : 4 : model.add_constraint (x, EQ_EXPR, y, NULL);
1846 : :
1847 : : /* Setting x to a state should also update y, as they
1848 : : are in the same equivalence class. */
1849 : 4 : map.set_state (&model, x_sval, TEST_STATE_5, z_sval, ext_state);
1850 : 4 : ASSERT_EQ (map.get_state (x_sval, ext_state), TEST_STATE_5);
1851 : 4 : ASSERT_EQ (map.get_state (y_sval, ext_state), TEST_STATE_5);
1852 : 4 : ASSERT_EQ (map.get_origin (x_sval, ext_state), z_sval);
1853 : 4 : ASSERT_EQ (map.get_origin (y_sval, ext_state), z_sval);
1854 : 4 : }
1855 : :
1856 : : /* Test equality and hashing. */
1857 : 4 : {
1858 : 4 : region_model_manager mgr;
1859 : 4 : region_model model (&mgr);
1860 : 4 : const svalue *y_sval = model.get_rvalue (y, NULL);
1861 : 4 : const svalue *z_sval = model.get_rvalue (z, NULL);
1862 : :
1863 : 4 : sm_state_map map0 (*sm);
1864 : 4 : sm_state_map map1 (*sm);
1865 : 4 : sm_state_map map2 (*sm);
1866 : :
1867 : 4 : ASSERT_EQ (map0.hash (), map1.hash ());
1868 : 4 : ASSERT_EQ (map0, map1);
1869 : :
1870 : 4 : map1.impl_set_state (y_sval, TEST_STATE_5, z_sval, ext_state);
1871 : 4 : ASSERT_NE (map0.hash (), map1.hash ());
1872 : 4 : ASSERT_NE (map0, map1);
1873 : :
1874 : : /* Make the same change to map2. */
1875 : 4 : map2.impl_set_state (y_sval, TEST_STATE_5, z_sval, ext_state);
1876 : 4 : ASSERT_EQ (map1.hash (), map2.hash ());
1877 : 4 : ASSERT_EQ (map1, map2);
1878 : 4 : }
1879 : :
1880 : : /* Equality and hashing shouldn't depend on ordering. */
1881 : 4 : {
1882 : 4 : const state_machine::state test_state_2 ("test state 2", 2);
1883 : 4 : const state_machine::state_t TEST_STATE_2 = &test_state_2;
1884 : 4 : const state_machine::state test_state_3 ("test state 3", 3);
1885 : 4 : const state_machine::state_t TEST_STATE_3 = &test_state_3;
1886 : 4 : sm_state_map map0 (*sm);
1887 : 4 : sm_state_map map1 (*sm);
1888 : 4 : sm_state_map map2 (*sm);
1889 : :
1890 : 4 : ASSERT_EQ (map0.hash (), map1.hash ());
1891 : 4 : ASSERT_EQ (map0, map1);
1892 : :
1893 : 4 : region_model_manager mgr;
1894 : 4 : region_model model (&mgr);
1895 : 4 : const svalue *x_sval = model.get_rvalue (x, NULL);
1896 : 4 : const svalue *y_sval = model.get_rvalue (y, NULL);
1897 : 4 : const svalue *z_sval = model.get_rvalue (z, NULL);
1898 : :
1899 : 4 : map1.impl_set_state (x_sval, TEST_STATE_2, NULL, ext_state);
1900 : 4 : map1.impl_set_state (y_sval, TEST_STATE_3, NULL, ext_state);
1901 : 4 : map1.impl_set_state (z_sval, TEST_STATE_2, NULL, ext_state);
1902 : :
1903 : 4 : map2.impl_set_state (z_sval, TEST_STATE_2, NULL, ext_state);
1904 : 4 : map2.impl_set_state (y_sval, TEST_STATE_3, NULL, ext_state);
1905 : 4 : map2.impl_set_state (x_sval, TEST_STATE_2, NULL, ext_state);
1906 : :
1907 : 4 : ASSERT_EQ (map1.hash (), map2.hash ());
1908 : 4 : ASSERT_EQ (map1, map2);
1909 : 4 : }
1910 : :
1911 : : // TODO: coverage for purging
1912 : 4 : }
1913 : :
1914 : : /* Check program_state works as expected. */
1915 : :
1916 : : static void
1917 : 4 : test_program_state_1 ()
1918 : : {
1919 : : /* Create a program_state for a global ptr "p" that has
1920 : : malloc sm-state, pointing to a region on the heap. */
1921 : 4 : tree p = build_global_decl ("p", ptr_type_node);
1922 : :
1923 : 4 : state_machine *sm = make_malloc_state_machine (NULL);
1924 : 4 : const state_machine::state_t UNCHECKED_STATE
1925 : 4 : = sm->get_state_by_name ("unchecked");
1926 : 4 : auto_delete_vec <state_machine> checkers;
1927 : 4 : checkers.safe_push (sm);
1928 : :
1929 : 4 : engine eng;
1930 : 4 : extrinsic_state ext_state (checkers, &eng);
1931 : 4 : region_model_manager *mgr = eng.get_model_manager ();
1932 : 4 : program_state s (ext_state);
1933 : 4 : region_model *model = s.m_region_model;
1934 : 4 : const svalue *size_in_bytes
1935 : 4 : = mgr->get_or_create_unknown_svalue (size_type_node);
1936 : 4 : const region *new_reg
1937 : 4 : = model->get_or_create_region_for_heap_alloc (size_in_bytes, NULL);
1938 : 4 : const svalue *ptr_sval = mgr->get_ptr_svalue (ptr_type_node, new_reg);
1939 : 4 : model->set_value (model->get_lvalue (p, NULL),
1940 : : ptr_sval, NULL);
1941 : 4 : sm_state_map *smap = s.m_checker_states[0];
1942 : :
1943 : 4 : smap->impl_set_state (ptr_sval, UNCHECKED_STATE, NULL, ext_state);
1944 : 4 : ASSERT_EQ (smap->get_state (ptr_sval, ext_state), UNCHECKED_STATE);
1945 : 4 : }
1946 : :
1947 : : /* Check that program_state works for string literals. */
1948 : :
1949 : : static void
1950 : 4 : test_program_state_2 ()
1951 : : {
1952 : : /* Create a program_state for a global ptr "p" that points to
1953 : : a string constant. */
1954 : 4 : tree p = build_global_decl ("p", ptr_type_node);
1955 : :
1956 : 4 : tree string_cst_ptr = build_string_literal (4, "foo");
1957 : :
1958 : 4 : auto_delete_vec <state_machine> checkers;
1959 : 4 : engine eng;
1960 : 4 : extrinsic_state ext_state (checkers, &eng);
1961 : :
1962 : 4 : program_state s (ext_state);
1963 : 4 : region_model *model = s.m_region_model;
1964 : 4 : const region *p_reg = model->get_lvalue (p, NULL);
1965 : 4 : const svalue *str_sval = model->get_rvalue (string_cst_ptr, NULL);
1966 : 4 : model->set_value (p_reg, str_sval, NULL);
1967 : 4 : }
1968 : :
1969 : : /* Verify that program_states with identical sm-state can be merged,
1970 : : and that the merged program_state preserves the sm-state. */
1971 : :
1972 : : static void
1973 : 4 : test_program_state_merging ()
1974 : : {
1975 : : /* Create a program_state for a global ptr "p" that has
1976 : : malloc sm-state, pointing to a region on the heap. */
1977 : 4 : tree p = build_global_decl ("p", ptr_type_node);
1978 : :
1979 : 4 : engine eng;
1980 : 4 : region_model_manager *mgr = eng.get_model_manager ();
1981 : 4 : program_point point (program_point::origin (*mgr));
1982 : 4 : auto_delete_vec <state_machine> checkers;
1983 : 4 : checkers.safe_push (make_malloc_state_machine (NULL));
1984 : 4 : extrinsic_state ext_state (checkers, &eng);
1985 : :
1986 : 4 : program_state s0 (ext_state);
1987 : 4 : uncertainty_t uncertainty;
1988 : 4 : impl_region_model_context ctxt (&s0, ext_state, &uncertainty);
1989 : :
1990 : 4 : region_model *model0 = s0.m_region_model;
1991 : 4 : const svalue *size_in_bytes
1992 : 4 : = mgr->get_or_create_unknown_svalue (size_type_node);
1993 : 4 : const region *new_reg
1994 : 4 : = model0->get_or_create_region_for_heap_alloc (size_in_bytes, NULL);
1995 : 4 : const svalue *ptr_sval = mgr->get_ptr_svalue (ptr_type_node, new_reg);
1996 : 4 : model0->set_value (model0->get_lvalue (p, &ctxt),
1997 : : ptr_sval, &ctxt);
1998 : 4 : sm_state_map *smap = s0.m_checker_states[0];
1999 : 4 : const state_machine::state test_state ("test state", 0);
2000 : 4 : const state_machine::state_t TEST_STATE = &test_state;
2001 : 4 : smap->impl_set_state (ptr_sval, TEST_STATE, NULL, ext_state);
2002 : 4 : ASSERT_EQ (smap->get_state (ptr_sval, ext_state), TEST_STATE);
2003 : :
2004 : 4 : model0->canonicalize ();
2005 : :
2006 : : /* Verify that canonicalization preserves sm-state. */
2007 : 4 : ASSERT_EQ (smap->get_state (model0->get_rvalue (p, NULL), ext_state),
2008 : : TEST_STATE);
2009 : :
2010 : : /* Make a copy of the program_state. */
2011 : 4 : program_state s1 (s0);
2012 : 4 : ASSERT_EQ (s0, s1);
2013 : :
2014 : : /* We have two identical states with "p" pointing to a heap region
2015 : : with the given sm-state.
2016 : : They ought to be mergeable, preserving the sm-state. */
2017 : 4 : program_state merged (ext_state);
2018 : 4 : ASSERT_TRUE (s0.can_merge_with_p (s1, ext_state, point, &merged));
2019 : 4 : merged.validate (ext_state);
2020 : :
2021 : : /* Verify that the merged state has the sm-state for "p". */
2022 : 4 : region_model *merged_model = merged.m_region_model;
2023 : 4 : sm_state_map *merged_smap = merged.m_checker_states[0];
2024 : 4 : ASSERT_EQ (merged_smap->get_state (merged_model->get_rvalue (p, NULL),
2025 : : ext_state),
2026 : : TEST_STATE);
2027 : :
2028 : : /* Try canonicalizing. */
2029 : 4 : merged.m_region_model->canonicalize ();
2030 : 4 : merged.validate (ext_state);
2031 : :
2032 : : /* Verify that the merged state still has the sm-state for "p". */
2033 : 4 : ASSERT_EQ (merged_smap->get_state (merged_model->get_rvalue (p, NULL),
2034 : : ext_state),
2035 : : TEST_STATE);
2036 : :
2037 : : /* After canonicalization, we ought to have equality with the inputs. */
2038 : 4 : ASSERT_EQ (s0, merged);
2039 : 8 : }
2040 : :
2041 : : /* Verify that program_states with different global-state in an sm-state
2042 : : can't be merged. */
2043 : :
2044 : : static void
2045 : 4 : test_program_state_merging_2 ()
2046 : : {
2047 : 4 : engine eng;
2048 : 4 : region_model_manager *mgr = eng.get_model_manager ();
2049 : 4 : program_point point (program_point::origin (*mgr));
2050 : 4 : auto_delete_vec <state_machine> checkers;
2051 : 4 : checkers.safe_push (make_signal_state_machine (NULL));
2052 : 4 : extrinsic_state ext_state (checkers, &eng);
2053 : :
2054 : 4 : const state_machine::state test_state_0 ("test state 0", 0);
2055 : 4 : const state_machine::state test_state_1 ("test state 1", 1);
2056 : 4 : const state_machine::state_t TEST_STATE_0 = &test_state_0;
2057 : 4 : const state_machine::state_t TEST_STATE_1 = &test_state_1;
2058 : :
2059 : 4 : program_state s0 (ext_state);
2060 : 4 : {
2061 : 4 : sm_state_map *smap0 = s0.m_checker_states[0];
2062 : 4 : smap0->set_global_state (TEST_STATE_0);
2063 : 4 : ASSERT_EQ (smap0->get_global_state (), TEST_STATE_0);
2064 : : }
2065 : :
2066 : 4 : program_state s1 (ext_state);
2067 : 4 : {
2068 : 4 : sm_state_map *smap1 = s1.m_checker_states[0];
2069 : 4 : smap1->set_global_state (TEST_STATE_1);
2070 : 4 : ASSERT_EQ (smap1->get_global_state (), TEST_STATE_1);
2071 : : }
2072 : :
2073 : 4 : ASSERT_NE (s0, s1);
2074 : :
2075 : : /* They ought to not be mergeable. */
2076 : 4 : program_state merged (ext_state);
2077 : 4 : ASSERT_FALSE (s0.can_merge_with_p (s1, ext_state, point, &merged));
2078 : 4 : }
2079 : :
2080 : : /* Run all of the selftests within this file. */
2081 : :
2082 : : void
2083 : 4 : analyzer_program_state_cc_tests ()
2084 : : {
2085 : 4 : test_sm_state_map ();
2086 : 4 : test_program_state_1 ();
2087 : 4 : test_program_state_2 ();
2088 : 4 : test_program_state_merging ();
2089 : 4 : test_program_state_merging_2 ();
2090 : 4 : }
2091 : :
2092 : : } // namespace selftest
2093 : :
2094 : : #endif /* CHECKING_P */
2095 : :
2096 : : } // namespace ana
2097 : :
2098 : : #endif /* #if ENABLE_ANALYZER */
|