GCC Middle and Back End API Reference
region-model.h
Go to the documentation of this file.
1/* Classes for modeling the state of memory.
2 Copyright (C) 2019-2026 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_REGION_MODEL_H
22#define GCC_ANALYZER_REGION_MODEL_H
23
24/* Implementation of the region-based ternary model described in:
25 "A Memory Model for Static Analysis of C Programs"
26 (Zhongxing Xu, Ted Kremenek, and Jian Zhang)
27 http://lcs.ios.ac.cn/~xuzb/canalyze/memmodel.pdf */
28
29#include "bitmap.h"
30#include "stringpool.h"
31#include "attribs.h" // for rdwr_map
32#include "selftest.h"
33#include "analyzer/svalue.h"
34#include "analyzer/region.h"
39#include "text-art/widget.h"
40#include "text-art/dump.h"
41
42using namespace ana;
43
44namespace inchash
45{
46 extern void add_path_var (path_var pv, hash &hstate);
47} // namespace inchash
48
49namespace ana {
50
51template <typename T>
53{
54 public:
55 one_way_id_map (int num_ids);
56 void put (T src, T dst);
57 T get_dst_for_src (T src) const;
58 void dump_to_pp (pretty_printer *pp) const;
59 void dump () const;
60 void update (T *) const;
61
62 private:
64 };
65
66/* class one_way_id_map. */
67
68/* one_way_id_map's ctor, which populates the map with dummy null values. */
69
70template <typename T>
71inline one_way_id_map<T>::one_way_id_map (int num_svalues)
72: m_src_to_dst (num_svalues)
73{
74 for (int i = 0; i < num_svalues; i++)
75 m_src_to_dst.quick_push (T::null ());
76}
77
78/* Record that SRC is to be mapped to DST. */
79
80template <typename T>
81inline void
83{
84 m_src_to_dst[src.as_int ()] = dst;
85}
86
87/* Get the new value for SRC within the map. */
88
89template <typename T>
90inline T
92{
93 if (src.null_p ())
94 return src;
95 return m_src_to_dst[src.as_int ()];
96}
97
98/* Dump this map to PP. */
99
100template <typename T>
101inline void
103{
104 pp_string (pp, "src to dst: {");
105 unsigned i;
106 T *dst;
108 {
109 if (i > 0)
110 pp_string (pp, ", ");
111 T src (T::from_int (i));
112 src.print (pp);
113 pp_string (pp, " -> ");
114 dst->print (pp);
115 }
116 pp_string (pp, "}");
117 pp_newline (pp);
118}
119
120/* Dump this map to stderr. */
121
122template <typename T>
123DEBUG_FUNCTION inline void
125{
127 pp.set_output_stream (stderr);
128 dump_to_pp (&pp);
129 pp_flush (&pp);
130}
131
132/* Update *ID from the old value to its new value in this map. */
133
134template <typename T>
135inline void
137{
138 *id = get_dst_for_src (*id);
139}
140
141/* A mapping from region to svalue for use when tracking state. */
142
144{
145public:
147 typedef hash_map_t::iterator iterator;
148
153
154 bool operator== (const region_to_value_map &other) const;
155 bool operator!= (const region_to_value_map &other) const
156 {
157 return !(*this == other);
158 }
159
160 iterator begin () const { return m_hash_map.begin (); }
161 iterator end () const { return m_hash_map.end (); }
162
163 const svalue * const *get (const region *reg) const
164 {
165 return const_cast <hash_map_t &> (m_hash_map).get (reg);
166 }
167 void put (const region *reg, const svalue *sval)
168 {
169 m_hash_map.put (reg, sval);
170 }
171 void remove (const region *reg)
172 {
173 m_hash_map.remove (reg);
174 }
175
176 bool is_empty () const { return m_hash_map.is_empty (); }
177
178 void dump_to_pp (pretty_printer *pp, bool simple, bool multiline) const;
179 void dump (bool simple) const;
180
181 std::unique_ptr<json::object> to_json () const;
182
183 std::unique_ptr<text_art::tree_widget>
185
187 region_to_value_map *out) const;
188
189 void purge_state_involving (const svalue *sval);
190
191private:
193};
194
195/* Various operations delete information from a region_model.
196
197 This struct tracks how many of each kind of entity were purged (e.g.
198 for selftests, and for debugging). */
199
218
219/* A base class for visiting regions and svalues, with do-nothing
220 base implementations of the per-subclass vfuncs. */
221
223{
224public:
225 virtual void visit_region_svalue (const region_svalue *) {}
226 virtual void visit_constant_svalue (const constant_svalue *) {}
227 virtual void visit_unknown_svalue (const unknown_svalue *) {}
228 virtual void visit_poisoned_svalue (const poisoned_svalue *) {}
229 virtual void visit_setjmp_svalue (const setjmp_svalue *) {}
230 virtual void visit_initial_svalue (const initial_svalue *) {}
231 virtual void visit_unaryop_svalue (const unaryop_svalue *) {}
232 virtual void visit_binop_svalue (const binop_svalue *) {}
233 virtual void visit_sub_svalue (const sub_svalue *) {}
234 virtual void visit_repeated_svalue (const repeated_svalue *) {}
238 virtual void visit_widening_svalue (const widening_svalue *) {}
239 virtual void visit_compound_svalue (const compound_svalue *) {}
240 virtual void visit_conjured_svalue (const conjured_svalue *) {}
243
244 virtual void visit_region (const region *) {}
245};
246
247struct append_regions_cb_data;
248
249/* Roughly equivalent to a struct __cxa_exception, except we store a std::vector
250 rather than a linked list. */
251
253{
254 exception_node (const svalue *exception_sval,
255 const svalue *typeinfo_sval,
256 const svalue *destructor_sval)
257 : m_exception_sval (exception_sval),
258 m_typeinfo_sval (typeinfo_sval),
259 m_destructor_sval (destructor_sval)
260 {
261 }
262
263 bool operator== (const exception_node &other) const;
264
265 void dump_to_pp (pretty_printer *pp, bool simple) const;
266 void dump (FILE *fp, bool simple) const;
267 void dump (bool simple) const;
268 void dump () const;
269
270 std::unique_ptr<json::object> to_json () const;
271
272 std::unique_ptr<text_art::tree_widget>
274
276
278
282};
283
284/* A region_model encapsulates a representation of the state of memory, with
285 a tree of regions, along with their associated values.
286 The representation is graph-like because values can be pointers to
287 regions.
288 It also stores:
289 - a constraint_manager, capturing relationships between the values, and
290 - dynamic extents, mapping dynamically-allocated regions to svalues (their
291 capacities). */
292
294{
295 public:
297
302
303 bool operator== (const region_model &other) const;
304 bool operator!= (const region_model &other) const
305 {
306 return !(*this == other);
307 }
308
309 hashval_t hash () const;
310
311 void print (pretty_printer *pp) const;
312
313 void dump_to_pp (pretty_printer *pp, bool simple, bool multiline) const;
314 void dump (FILE *fp, bool simple, bool multiline) const;
315 void dump (bool simple) const;
316 void dump () const;
317
318 void debug () const;
319
320 std::unique_ptr<json::object> to_json () const;
321
322 std::unique_ptr<text_art::tree_widget>
324
325 void validate () const;
326
328 bool canonicalized_p () const;
329
330 void
332 bool *out_unknown_side_effects,
334
336 const svalue *get_gassign_result (const gassign *assign,
338 void on_asm_stmt (const gasm *asm_stmt, region_model_context *ctxt);
340 void on_call_post (const gcall &stmt,
341 bool unknown_side_effects,
343
345
347
348 const svalue *maybe_get_copy_bounds (const region *src_reg,
349 const svalue *num_bytes_sval);
351 int retval,
352 bool unmergeable);
354 bool unmergeable);
356 bool unmergeable);
358
362 const svalue *extra_sval,
363 const uncertainty_t *uncertainty);
364
366 void on_setjmp (const gcall &stmt,
367 const exploded_node &enode,
368 const superedge &sedge,
370 void on_longjmp (const gcall &longjmp_call, const gcall &setjmp_call,
371 int setjmp_stack_depth, region_model_context *ctxt);
372
373 void update_for_gcall (const gcall &call_stmt,
375 function *callee = nullptr);
376
377 void update_for_return_gcall (const gcall &call_stmt,
379
380 const region *push_frame (const function &fun,
381 const gcall *call_stmt,
382 const vec<const svalue *> *arg_sids,
386 void pop_frame (tree result_lvalue,
387 const svalue **out_result,
389 const gcall *call_stmt,
390 bool eval_return_svalue = true);
391 int get_stack_depth () const;
392 const frame_region *get_frame_at_index (int index) const;
393
398
399 const region *deref_rvalue (const svalue *ptr_sval, tree ptr_tree,
401 bool add_nonnull_constraint = true) const;
402
404 const region *reg,
405 const bit_range &bits,
406 region_model_context *ctxt) const;
407
408 void set_value (const region *lhs_reg, const svalue *rhs_sval,
410 void set_value (tree lhs, tree rhs, region_model_context *ctxt);
411 void clobber_region (const region *reg);
412 void purge_region (const region *reg);
413 void fill_region (const region *reg,
414 const svalue *sval,
416 void zero_fill_region (const region *reg,
418 void write_bytes (const region *dest_reg,
419 const svalue *num_bytes_sval,
420 const svalue *sval,
422 const svalue *read_bytes (const region *src_reg,
423 tree src_ptr_expr,
424 const svalue *num_bytes_sval,
425 region_model_context *ctxt) const;
426 void copy_bytes (const region *dest_reg,
427 const region *src_reg,
428 tree src_ptr_expr,
429 const svalue *num_bytes_sval,
431 void mark_region_as_unknown (const region *reg, uncertainty_t *uncertainty);
432
434 enum tree_code op,
435 const svalue *rhs) const;
437 const region_svalue *ptr) const;
439 const svalue *b) const;
440 tristate structural_equality (const svalue *a, const svalue *b) const;
442 enum tree_code op,
443 tree rhs,
444 region_model_context *ctxt) const;
445 bool add_constraint (tree lhs, enum tree_code op, tree rhs,
447 bool add_constraint (tree lhs, enum tree_code op, tree rhs,
449 std::unique_ptr<rejected_constraint> *out);
450
451 const region *
454 bool update_state_machine = false,
455 const call_details *cd = nullptr);
456
460
462 logger *logger = nullptr) const;
464 logger *logger = nullptr) const;
468 logger *logger) const;
472 logger *logger) const;
473
474 /* For selftests. */
479
480 store *get_store () { return &m_store; }
481 const store *get_store () const { return &m_store; }
482
483 const dynamic_extents_t &
485 {
486 return m_dynamic_extents;
487 }
488 const svalue *get_dynamic_extents (const region *reg) const;
489 void set_dynamic_extents (const region *reg,
490 const svalue *size_in_bytes,
492 void unset_dynamic_extents (const region *reg);
493
496 {
497 return m_mgr->get_range_manager ();
498 }
499
501 enum poison_kind pkind);
502
503 bool can_merge_with_p (const region_model &other_model,
504 const program_point &point,
505 region_model *out_model,
506 const extrinsic_state *ext_state = nullptr,
507 const program_state *state_a = nullptr,
508 const program_state *state_b = nullptr) const;
509
512
514 static void append_regions_cb (const region *base_reg,
515 struct append_regions_cb_data *data);
516
517 const svalue *get_store_value (const region *reg,
518 region_model_context *ctxt) const;
519 const svalue *get_store_bytes (const region *base_reg,
520 const byte_range &bytes,
521 region_model_context *ctxt) const;
523 tree expr,
524 const svalue **out_sval,
525 region_model_context *ctxt) const;
527 tree expr,
528 const svalue **out_sval,
529 region_model_context *ctxt) const;
530
531 bool region_exists_p (const region *reg) const;
532
533 void loop_replay_fixup (const region_model *dst_state);
534
535 const svalue *get_capacity (const region *reg) const;
536
538 const region_model &summary);
539
541 const svalue *copied_sval,
542 const region *src_reg,
544
545 void set_errno (const call_details &cd);
546
547 /* Implemented in sm-fd.cc */
549
550 /* Implemented in sm-malloc.cc */
552 const svalue *old_ptr_sval,
553 const svalue *new_ptr_sval);
554
555 /* Implemented in sm-malloc.cc. */
556 void
558 const svalue *new_ptr_sval);
559
560 /* Implemented in sm-taint.cc. */
561 void mark_as_tainted (const svalue *sval,
563
564 bool add_constraint (const svalue *lhs,
565 enum tree_code op,
566 const svalue *rhs,
568
569 const svalue *check_for_poison (const svalue *sval,
570 tree expr,
571 const region *src_region,
572 region_model_context *ctxt) const;
573
574 void check_region_for_write (const region *dest_reg,
575 const svalue *sval_hint,
576 region_model_context *ctxt) const;
577
578 const svalue *
580 unsigned idx) const;
581 const svalue *
583 unsigned idx,
584 bool include_terminator,
585 const svalue **out_sval) const;
586
588 get_builtin_kf (const gcall &call,
589 region_model_context *ctxt = nullptr) const;
590
591 bool called_from_main_p () const;
592
594 {
596 }
598 {
599 if (m_thrown_exceptions_stack.empty ())
600 return nullptr;
601 return &m_thrown_exceptions_stack.back ();
602 }
604 {
606 const exception_node retval = m_thrown_exceptions_stack.back ();
607 m_thrown_exceptions_stack.pop_back ();
608 return retval;
609 }
610
612 {
614 }
616 {
617 if (m_caught_exceptions_stack.empty ())
618 return nullptr;
619 return &m_caught_exceptions_stack.back ();
620 }
622 {
624 const exception_node retval = m_caught_exceptions_stack.back ();
625 m_caught_exceptions_stack.pop_back ();
626 return retval;
627 }
628
629private:
632
636 logger *logger) const;
640 logger *logger) const;
641
643 const call_details &cd) const;
645
646 bool add_constraints_from_binop (const svalue *outer_lhs,
647 enum tree_code outer_op,
648 const svalue *outer_rhs,
649 bool *out,
651
653 enum poison_kind pkind);
654
656 bool nonnull,
658
659 const svalue *get_initial_value_for_global (const region *reg) const;
660
662
664 const svalue *size_in_bytes,
665 region_model_context *ctxt) const;
667 region_model_context *ctxt) const;
668
670 enum access_direction dir,
671 region_model_context *ctxt) const;
672
673 void check_for_writable_region (const region* dest_reg,
674 region_model_context *ctxt) const;
675 bool check_region_access (const region *reg,
676 enum access_direction dir,
677 const svalue *sval_hint,
678 region_model_context *ctxt) const;
679 bool check_region_for_read (const region *src_reg,
680 region_model_context *ctxt) const;
681 void check_region_size (const region *lhs_reg, const svalue *rhs_sval,
682 region_model_context *ctxt) const;
683
684 /* Implemented in bounds-checking.cc */
685 bool check_symbolic_bounds (const region *base_reg,
686 const svalue *sym_byte_offset,
687 const svalue *num_bytes_sval,
688 const svalue *capacity,
689 enum access_direction dir,
690 const svalue *sval_hint,
691 region_model_context *ctxt) const;
692 bool check_region_bounds (const region *reg, enum access_direction dir,
693 const svalue *sval_hint,
694 region_model_context *ctxt) const;
695
696 void check_call_args (const call_details &cd) const;
698 tree format_attr) const;
700 tree callee_fndecl,
702 rdwr_map &rdwr_idx) const;
704 tree callee_fndecl,
706 rdwr_map &rdwr_idx);
708 tree callee_fndecl,
710 rdwr_map &rdwr_idx,
711 tree attr);
712 void check_function_attrs (const gcall &call,
713 tree callee_fndecl,
715
717 tree fndecl,
719
720 /* Storing this here to avoid passing it around everywhere. */
722
724
725 constraint_manager *m_constraints; // TODO: embed, rather than dynalloc?
726
728
729 std::vector<exception_node> m_thrown_exceptions_stack;
730 std::vector<exception_node> m_caught_exceptions_stack;
731
732 /* Map from base region to size in bytes, for tracking the sizes of
733 dynamically-allocated regions.
734 This is part of the region_model rather than the region to allow for
735 memory regions to be resized (e.g. by realloc). */
737};
738
739/* Some region_model activity could lead to warnings (e.g. attempts to use an
740 uninitialized value). This abstract base class encapsulates an interface
741 for the region model to use when emitting such warnings.
742
743 Having this as an abstract base class allows us to support the various
744 operations needed by program_state in the analyzer within region_model,
745 whilst keeping them somewhat modularized. */
746
748{
749 public:
750 bool
751 warn (std::unique_ptr<pending_diagnostic> d,
752 std::unique_ptr<pending_location::fixer_for_epath> ploc_fixer = nullptr);
753
754 /* Hook for determining where diagnostics are to currently be emitted. */
755 virtual pending_location
757
758 /* Hook for clients to store pending diagnostics.
759 Return true if the diagnostic was stored, or false if it was deleted. */
760 virtual bool
761 warn_at (std::unique_ptr<pending_diagnostic> d,
762 pending_location &&ploc) = 0;
763
764 /* Hook for clients to add a note to the last previously stored
765 pending diagnostic. */
766 virtual void add_note (std::unique_ptr<pending_note> pn) = 0;
767
768 /* Hook for clients to add an event to the last previously stored
769 pending diagnostic. */
770 virtual void add_event (std::unique_ptr<checker_event> event) = 0;
771
772 /* Hook for clients to be notified when an SVAL that was reachable
773 in a previous state is no longer live, so that clients can emit warnings
774 about leaks. */
775 virtual void on_svalue_leak (const svalue *sval) = 0;
776
777 /* Hook for clients to be notified when the set of explicitly live
778 svalues changes, so that they can purge state relating to dead
779 svalues. */
780 virtual void on_liveness_change (const svalue_set &live_svalues,
781 const region_model *model) = 0;
782
783 virtual logger *get_logger () = 0;
784
785 /* Hook for clients to be notified when the condition
786 "LHS OP RHS" is added to the region model.
787 This exists so that state machines can detect tests on edges,
788 and use them to trigger sm-state transitions (e.g. transitions due
789 to ptrs becoming known to be NULL or non-NULL, rather than just
790 "unchecked") */
791 virtual void on_condition (const svalue *lhs,
792 enum tree_code op,
793 const svalue *rhs) = 0;
794
795 /* Hook for clients to be notified when the condition that
796 SVAL is within RANGES is added to the region model.
797 Similar to on_condition, but for use when handling switch statements.
798 RANGES is non-empty. */
799 virtual void on_bounded_ranges (const svalue &sval,
800 const bounded_ranges &ranges) = 0;
801
802 /* Hook for clients to be notified when a frame is popped from the stack. */
803 virtual void on_pop_frame (const frame_region *) = 0;
804
805 /* Hooks for clients to be notified when an unknown change happens
806 to SVAL (in response to a call to an unknown function). */
807 virtual void on_unknown_change (const svalue *sval, bool is_mutable) = 0;
808
809 /* Hooks for clients to be notified when a phi node is handled,
810 where RHS is the pertinent argument. */
811 virtual void on_phi (const gphi *phi, tree rhs) = 0;
812
813 /* Hooks for clients to be notified when the region model doesn't
814 know how to handle the tree code of T at LOC. */
816 const dump_location_t &loc) = 0;
817
818 /* Hook for clients to be notified when a function_decl escapes. */
819 virtual void on_escaped_function (tree fndecl) = 0;
820
822
823 /* Hook for clients to purge state involving SVAL. */
824 virtual void purge_state_involving (const svalue *sval) = 0;
825
826 /* Hook for clients to split state with a non-standard path. */
827 virtual void bifurcate (std::unique_ptr<custom_edge_info> info) = 0;
828
829 /* Hook for clients to terminate the standard path. */
830 virtual void terminate_path () = 0;
831
832 virtual const extrinsic_state *get_ext_state () const = 0;
833
834 /* Hook for clients to access the a specific state machine in
835 any underlying program_state. */
836 virtual bool
837 get_state_map_by_name (const char *name,
838 sm_state_map **out_smap,
839 const state_machine **out_sm,
840 unsigned *out_sm_idx,
841 std::unique_ptr<sm_context> *out_sm_context) = 0;
842
843 /* Precanned ways for clients to access specific state machines. */
844 bool get_fd_map (sm_state_map **out_smap,
845 const state_machine **out_sm,
846 unsigned *out_sm_idx,
847 std::unique_ptr<sm_context> *out_sm_context)
848 {
849 return get_state_map_by_name ("file-descriptor", out_smap, out_sm,
850 out_sm_idx, out_sm_context);
851 }
852 bool get_malloc_map (sm_state_map **out_smap,
853 const state_machine **out_sm,
854 unsigned *out_sm_idx)
855 {
856 return get_state_map_by_name ("malloc", out_smap, out_sm, out_sm_idx,
857 nullptr);
858 }
859 bool get_taint_map (sm_state_map **out_smap,
860 const state_machine **out_sm,
861 unsigned *out_sm_idx)
862 {
863 return get_state_map_by_name ("taint", out_smap, out_sm, out_sm_idx,
864 nullptr);
865 }
866
867 bool possibly_tainted_p (const svalue *sval);
868
869 /* Get the current statement, if any. */
870 virtual const gimple *get_stmt () const = 0;
871
872 virtual const exploded_graph *get_eg () const = 0;
873
874 virtual const program_state *get_state () const = 0;
875
876 /* Hooks for detecting infinite loops. */
877 virtual void maybe_did_work () = 0;
878 virtual bool checking_for_infinite_loop_p () const = 0;
879 virtual void on_unusable_in_infinite_loop () = 0;
880};
881
882/* A "do nothing" subclass of region_model_context. */
883
885{
886public:
889 {
890 return pending_location ();
891 }
892 bool
893 warn_at (std::unique_ptr<pending_diagnostic>,
894 pending_location &&) override
895 {
896 return false;
897 }
898 void add_note (std::unique_ptr<pending_note>) override;
899 void add_event (std::unique_ptr<checker_event>) override;
900 void on_svalue_leak (const svalue *) override {}
902 const region_model *) override {}
903 logger *get_logger () override { return nullptr; }
904 void on_condition (const svalue *lhs ATTRIBUTE_UNUSED,
905 enum tree_code op ATTRIBUTE_UNUSED,
906 const svalue *rhs ATTRIBUTE_UNUSED) override
907 {
908 }
910 const bounded_ranges &) override
911 {
912 }
913 void on_pop_frame (const frame_region *) override {}
914 void on_unknown_change (const svalue *sval ATTRIBUTE_UNUSED,
915 bool is_mutable ATTRIBUTE_UNUSED) override
916 {
917 }
918 void on_phi (const gphi *phi ATTRIBUTE_UNUSED,
919 tree rhs ATTRIBUTE_UNUSED) override
920 {
921 }
922 void on_unexpected_tree_code (tree, const dump_location_t &) override {}
923
924 void on_escaped_function (tree) override {}
925
926 uncertainty_t *get_uncertainty () override { return nullptr; }
927
928 void purge_state_involving (const svalue *sval ATTRIBUTE_UNUSED) override {}
929
930 void bifurcate (std::unique_ptr<custom_edge_info> info) override;
931 void terminate_path () override;
932
933 const extrinsic_state *get_ext_state () const override { return nullptr; }
934
935 bool get_state_map_by_name (const char *,
936 sm_state_map **,
937 const state_machine **,
938 unsigned *,
939 std::unique_ptr<sm_context> *) override
940 {
941 return false;
942 }
943
944 const gimple *get_stmt () const override { return nullptr; }
945 const exploded_graph *get_eg () const override { return nullptr; }
946 const program_state *get_state () const override { return nullptr; }
947
948 void maybe_did_work () override {}
949 bool checking_for_infinite_loop_p () const override { return false; }
951};
952
953/* A subclass of region_model_context for determining if operations fail
954 e.g. "can we generate a region for the lvalue of EXPR?". */
955
957{
958public:
960
962 final override
963 {
965 }
966
967 bool had_errors_p () const { return m_num_unexpected_codes > 0; }
968
969private:
971};
972
973/* Subclass of region_model_context that wraps another context, allowing
974 for extra code to be added to the various hooks. */
975
977{
978 public:
981 {
982 if (m_inner)
983 return m_inner->get_pending_location_for_diag ();
984 else
985 return pending_location ();
986 }
987
988 bool
989 warn_at (std::unique_ptr<pending_diagnostic> d,
990 pending_location &&ploc) override
991 {
992 if (m_inner)
993 return m_inner->warn_at (std::move (d), std::move (ploc));
994 else
995 return false;
996 }
997
998 void add_note (std::unique_ptr<pending_note> pn) override
999 {
1000 if (m_inner)
1001 m_inner->add_note (std::move (pn));
1002 }
1003 void add_event (std::unique_ptr<checker_event> event) override;
1004
1005 void on_svalue_leak (const svalue *sval) override
1006 {
1007 if (m_inner)
1008 m_inner->on_svalue_leak (sval);
1009 }
1010
1011 void on_liveness_change (const svalue_set &live_svalues,
1012 const region_model *model) override
1013 {
1014 if (m_inner)
1015 m_inner->on_liveness_change (live_svalues, model);
1016 }
1017
1018 logger *get_logger () override
1019 {
1020 if (m_inner)
1021 return m_inner->get_logger ();
1022 else
1023 return nullptr;
1024 }
1025
1026 void on_condition (const svalue *lhs,
1027 enum tree_code op,
1028 const svalue *rhs) override
1029 {
1030 if (m_inner)
1031 m_inner->on_condition (lhs, op, rhs);
1032 }
1033
1034 void on_bounded_ranges (const svalue &sval,
1035 const bounded_ranges &ranges) override
1036 {
1037 if (m_inner)
1038 m_inner->on_bounded_ranges (sval, ranges);
1039 }
1040
1041 void on_pop_frame (const frame_region *frame_reg) override
1042 {
1043 if (m_inner)
1044 m_inner->on_pop_frame (frame_reg);
1045 }
1046
1047 void on_unknown_change (const svalue *sval, bool is_mutable) override
1048 {
1049 if (m_inner)
1050 m_inner->on_unknown_change (sval, is_mutable);
1051 }
1052
1053 void on_phi (const gphi *phi, tree rhs) override
1054 {
1055 if (m_inner)
1056 m_inner->on_phi (phi, rhs);
1057 }
1058
1060 const dump_location_t &loc) override
1061 {
1062 if (m_inner)
1063 m_inner->on_unexpected_tree_code (t, loc);
1064 }
1065
1066 void on_escaped_function (tree fndecl) override
1067 {
1068 if (m_inner)
1069 m_inner->on_escaped_function (fndecl);
1070 }
1071
1073 {
1074 if (m_inner)
1075 return m_inner->get_uncertainty ();
1076 else
1077 return nullptr;
1078 }
1079
1080 void purge_state_involving (const svalue *sval) override
1081 {
1082 if (m_inner)
1083 m_inner->purge_state_involving (sval);
1084 }
1085
1086 void bifurcate (std::unique_ptr<custom_edge_info> info) override
1087 {
1088 if (m_inner)
1089 m_inner->bifurcate (std::move (info));
1090 }
1091
1092 void terminate_path () override
1093 {
1094 if (m_inner)
1095 m_inner->terminate_path ();
1096 }
1097
1098 const extrinsic_state *get_ext_state () const override
1099 {
1100 if (m_inner)
1101 return m_inner->get_ext_state ();
1102 else
1103 return nullptr;
1104 }
1105
1106 bool get_state_map_by_name (const char *name,
1107 sm_state_map **out_smap,
1108 const state_machine **out_sm,
1109 unsigned *out_sm_idx,
1110 std::unique_ptr<sm_context> *out_sm_context)
1111 override
1112 {
1113 if (m_inner)
1114 return m_inner->get_state_map_by_name (name, out_smap, out_sm, out_sm_idx,
1115 out_sm_context);
1116 else
1117 return false;
1118 }
1119
1120 const gimple *get_stmt () const override
1121 {
1122 if (m_inner)
1123 return m_inner->get_stmt ();
1124 else
1125 return nullptr;
1126 }
1127
1128 const exploded_graph *get_eg () const override
1129 {
1130 if (m_inner)
1131 return m_inner->get_eg ();
1132 else
1133 return nullptr;
1134 }
1135
1136 const program_state *get_state () const override
1137 {
1138 if (m_inner)
1139 return m_inner->get_state ();
1140 else
1141 return nullptr;
1142 }
1143
1144 void maybe_did_work () override
1145 {
1146 if (m_inner)
1147 m_inner->maybe_did_work ();
1148 }
1149
1150 bool checking_for_infinite_loop_p () const override
1151 {
1152 if (m_inner)
1153 return m_inner->checking_for_infinite_loop_p ();
1154 return false;
1155 }
1157 {
1158 if (m_inner)
1159 m_inner->on_unusable_in_infinite_loop ();
1160 }
1161
1162protected:
1164 : m_inner (inner)
1165 {
1166 }
1167
1169};
1170
1171/* Subclass of region_model_context_decorator with a hook for adding
1172 notes/events when saving diagnostics. */
1173
1175{
1176public:
1177 bool
1178 warn_at (std::unique_ptr<pending_diagnostic> d,
1179 pending_location &&ploc) override
1180 {
1181 if (m_inner)
1182 if (m_inner->warn_at (std::move (d), std::move (ploc)))
1183 {
1184 add_annotations ();
1185 return true;
1186 }
1187 return false;
1188 }
1189
1190 /* Hook to add new event(s)/note(s) */
1191 virtual void add_annotations () = 0;
1192
1193protected:
1198};
1199
1200/* A bundle of data for use when attempting to merge two region_model
1201 instances to make a third. */
1202
1204{
1206 const region_model *model_b,
1207 const program_point &point,
1208 region_model *merged_model,
1210 const program_state *state_a,
1211 const program_state *state_b)
1212 : m_model_a (model_a), m_model_b (model_b),
1213 m_point (point),
1214 m_merged_model (merged_model),
1216 m_state_a (state_a), m_state_b (state_b)
1217 {
1218 }
1219
1220 void dump_to_pp (pretty_printer *pp, bool simple) const;
1221 void dump (FILE *fp, bool simple) const;
1222 void dump (bool simple) const;
1223
1225 {
1226 return m_model_a->get_manager ();
1227 }
1228
1229 bool mergeable_svalue_p (const svalue *) const;
1230
1231 const supernode *get_supernode () const
1232 {
1233 return m_point.get_supernode ();
1234 }
1235
1236 void on_widening_reuse (const widening_svalue *widening_sval);
1237
1242
1246
1248};
1249
1250/* A record that can (optionally) be written out when
1251 region_model::add_constraint fails. */
1252
1254{
1255public:
1257 virtual void dump_to_pp (pretty_printer *pp) const = 0;
1258
1259 const region_model &get_model () const { return m_model; }
1260
1261protected:
1263 : m_model (model)
1264 {}
1265
1267};
1268
1270{
1271public:
1273 const svalue *lhs, enum tree_code op, const svalue *rhs)
1274 : rejected_constraint (model),
1275 m_lhs (lhs), m_op (op), m_rhs (rhs)
1276 {}
1277
1278 void dump_to_pp (pretty_printer *pp) const final override;
1279
1283};
1284
1286{
1287public:
1289 : rejected_constraint (model)
1290 {}
1291
1292 void dump_to_pp (pretty_printer *pp) const final override;
1293};
1294
1296{
1297public:
1299 tree expr, const bounded_ranges *ranges)
1300 : rejected_constraint (model),
1301 m_expr (expr), m_ranges (ranges)
1302 {}
1303
1304 void dump_to_pp (pretty_printer *pp) const final override;
1305
1306private:
1309};
1310
1311/* A bundle of state. */
1312
1314{
1315public:
1317 const supergraph *sg = nullptr);
1318 const supergraph *get_supergraph () { return m_sg; }
1321 {
1322 return m_mgr.get_known_function_manager ();
1323 }
1324
1325 void log_stats (logger *logger) const;
1326
1327private:
1330};
1331
1332/* Factory functions for various diagnostics. */
1333
1334extern std::unique_ptr<pending_diagnostic>
1336 const region *src_region,
1337 tree check_expr);
1338
1339extern std::unique_ptr<pending_diagnostic>
1341 tree count_cst,
1342 const region *src_region);
1343
1344extern std::unique_ptr<pending_diagnostic>
1346 int operand_precision,
1347 tree count_cst,
1348 const region *src_region);
1349
1350extern std::unique_ptr<pending_diagnostic>
1352
1353extern std::unique_ptr<pending_diagnostic>
1355
1356} // namespace ana
1357
1358extern void debug (const region_model &rmodel);
1359
1360namespace ana {
1361
1362#if CHECKING_P
1363
1364namespace selftest {
1365
1366using namespace ::selftest;
1367
1368/* An implementation of region_model_context for use in selftests, which
1369 stores any pending_diagnostic instances passed to it. */
1370
1371class test_region_model_context : public noop_region_model_context
1372{
1373public:
1374 bool
1375 warn_at (std::unique_ptr<pending_diagnostic> d,
1376 pending_location &&) final override
1377 {
1378 m_diagnostics.safe_push (d.release ());
1379 return true;
1380 }
1381
1382 unsigned get_num_diagnostics () const { return m_diagnostics.length (); }
1383
1384 void on_unexpected_tree_code (tree t, const dump_location_t &)
1385 final override
1386 {
1387 internal_error ("unhandled tree code: %qs",
1389 }
1390
1391private:
1392 /* Implicitly delete any diagnostics in the dtor. */
1393 auto_delete_vec<pending_diagnostic> m_diagnostics;
1394};
1395
1396/* Attempt to add the constraint (LHS OP RHS) to MODEL.
1397 Verify that MODEL remains satisfiable. */
1398
1399#define ADD_SAT_CONSTRAINT(MODEL, LHS, OP, RHS) \
1400 SELFTEST_BEGIN_STMT \
1401 bool sat = (MODEL).add_constraint (LHS, OP, RHS, nullptr); \
1402 ASSERT_TRUE (sat); \
1403 SELFTEST_END_STMT
1404
1405/* Attempt to add the constraint (LHS OP RHS) to MODEL.
1406 Verify that the result is not satisfiable. */
1407
1408#define ADD_UNSAT_CONSTRAINT(MODEL, LHS, OP, RHS) \
1409 SELFTEST_BEGIN_STMT \
1410 bool sat = (MODEL).add_constraint (LHS, OP, RHS, nullptr); \
1411 ASSERT_FALSE (sat); \
1412 SELFTEST_END_STMT
1413
1414/* Implementation detail of the ASSERT_CONDITION_* macros. */
1415
1416void assert_condition (const location &loc,
1417 region_model &model,
1418 const svalue *lhs, tree_code op, const svalue *rhs,
1419 tristate expected);
1420
1421void assert_condition (const location &loc,
1422 region_model &model,
1423 tree lhs, tree_code op, tree rhs,
1424 tristate expected);
1425
1426/* Assert that REGION_MODEL evaluates the condition "LHS OP RHS"
1427 as "true". */
1428
1429#define ASSERT_CONDITION_TRUE(REGION_MODEL, LHS, OP, RHS) \
1430 SELFTEST_BEGIN_STMT \
1431 assert_condition (SELFTEST_LOCATION, REGION_MODEL, LHS, OP, RHS, \
1432 tristate (tristate::TS_TRUE)); \
1433 SELFTEST_END_STMT
1434
1435/* Assert that REGION_MODEL evaluates the condition "LHS OP RHS"
1436 as "false". */
1437
1438#define ASSERT_CONDITION_FALSE(REGION_MODEL, LHS, OP, RHS) \
1439 SELFTEST_BEGIN_STMT \
1440 assert_condition (SELFTEST_LOCATION, REGION_MODEL, LHS, OP, RHS, \
1441 tristate (tristate::TS_FALSE)); \
1442 SELFTEST_END_STMT
1443
1444/* Assert that REGION_MODEL evaluates the condition "LHS OP RHS"
1445 as "unknown". */
1446
1447#define ASSERT_CONDITION_UNKNOWN(REGION_MODEL, LHS, OP, RHS) \
1448 SELFTEST_BEGIN_STMT \
1449 assert_condition (SELFTEST_LOCATION, REGION_MODEL, LHS, OP, RHS, \
1450 tristate (tristate::TS_UNKNOWN)); \
1451 SELFTEST_END_STMT
1452
1453} /* end of namespace selftest. */
1454
1455#endif /* #if CHECKING_P */
1456
1457} // namespace ana
1458
1459#endif /* GCC_ANALYZER_REGION_MODEL_H */
hash_map< rdwr_access_hash, attr_access > rdwr_map
Definition attribs.h:402
bool warn_at(std::unique_ptr< pending_diagnostic > d, pending_location &&ploc) override
Definition region-model.h:1178
virtual void add_annotations()=0
annotating_context(region_model_context *inner)
Definition region-model.h:1194
Definition svalue.h:1654
Definition svalue.h:814
Definition svalue.h:1100
Definition constraint-manager.h:178
Definition common.h:332
Definition call-details.h:31
Definition call-summary.h:68
Definition svalue.h:1417
Definition svalue.h:1552
Definition svalue.h:1800
Definition svalue.h:316
Definition constraint-manager.h:410
known_function_manager * get_known_function_manager()
Definition region-model.h:1320
const supergraph * get_supergraph()
Definition region-model.h:1318
const supergraph * m_sg
Definition region-model.h:1329
region_model_manager * get_model_manager()
Definition region-model.h:1319
void log_stats(logger *logger) const
engine(region_model_manager &mgr, const supergraph *sg=nullptr)
region_model_manager & m_mgr
Definition region-model.h:1328
Definition exploded-graph.h:794
Definition exploded-graph.h:206
Definition program-state.h:34
Definition region.h:320
Definition svalue.h:671
Definition known-function-manager.h:41
Definition common.h:303
Definition analyzer-logging.h:36
Definition region-model.h:885
const program_state * get_state() const override
Definition region-model.h:946
bool get_state_map_by_name(const char *, sm_state_map **, const state_machine **, unsigned *, std::unique_ptr< sm_context > *) override
Definition region-model.h:935
bool checking_for_infinite_loop_p() const override
Definition region-model.h:949
void maybe_did_work() override
Definition region-model.h:948
uncertainty_t * get_uncertainty() override
Definition region-model.h:926
void on_unexpected_tree_code(tree, const dump_location_t &) override
Definition region-model.h:922
void on_unknown_change(const svalue *sval, bool is_mutable) override
Definition region-model.h:914
void on_pop_frame(const frame_region *) override
Definition region-model.h:913
void on_phi(const gphi *phi, tree rhs) override
Definition region-model.h:918
void on_condition(const svalue *lhs, enum tree_code op, const svalue *rhs) override
Definition region-model.h:904
void add_event(std::unique_ptr< checker_event >) override
void on_liveness_change(const svalue_set &, const region_model *) override
Definition region-model.h:901
logger * get_logger() override
Definition region-model.h:903
void on_escaped_function(tree) override
Definition region-model.h:924
const exploded_graph * get_eg() const override
Definition region-model.h:945
const gimple * get_stmt() const override
Definition region-model.h:944
pending_location get_pending_location_for_diag() const override
Definition region-model.h:888
void on_unusable_in_infinite_loop() override
Definition region-model.h:950
bool warn_at(std::unique_ptr< pending_diagnostic >, pending_location &&) override
Definition region-model.h:893
void on_svalue_leak(const svalue *) override
Definition region-model.h:900
void add_note(std::unique_ptr< pending_note >) override
void purge_state_involving(const svalue *sval) override
Definition region-model.h:928
const extrinsic_state * get_ext_state() const override
Definition region-model.h:933
void on_bounded_ranges(const svalue &, const bounded_ranges &) override
Definition region-model.h:909
void bifurcate(std::unique_ptr< custom_edge_info > info) override
void put(T src, T dst)
Definition region-model.h:82
void dump() const
Definition region-model.h:124
T get_dst_for_src(T src) const
Definition region-model.h:91
void dump_to_pp(pretty_printer *pp) const
Definition region-model.h:102
auto_vec< T > m_src_to_dst
Definition region-model.h:63
void update(T *) const
Definition region-model.h:136
one_way_id_map(int num_ids)
Definition region-model.h:71
Definition common.h:179
Definition svalue.h:1249
Definition svalue.h:466
Definition program-point.h:54
Definition program-state.h:224
Definition region-model-reachability.h:36
region_model_context * m_inner
Definition region-model.h:1168
const exploded_graph * get_eg() const override
Definition region-model.h:1128
void on_bounded_ranges(const svalue &sval, const bounded_ranges &ranges) override
Definition region-model.h:1034
void on_phi(const gphi *phi, tree rhs) override
Definition region-model.h:1053
void add_event(std::unique_ptr< checker_event > event) override
const extrinsic_state * get_ext_state() const override
Definition region-model.h:1098
void on_condition(const svalue *lhs, enum tree_code op, const svalue *rhs) override
Definition region-model.h:1026
const gimple * get_stmt() const override
Definition region-model.h:1120
void maybe_did_work() override
Definition region-model.h:1144
const program_state * get_state() const override
Definition region-model.h:1136
void on_pop_frame(const frame_region *frame_reg) override
Definition region-model.h:1041
uncertainty_t * get_uncertainty() override
Definition region-model.h:1072
void bifurcate(std::unique_ptr< custom_edge_info > info) override
Definition region-model.h:1086
void add_note(std::unique_ptr< pending_note > pn) override
Definition region-model.h:998
void on_svalue_leak(const svalue *sval) override
Definition region-model.h:1005
pending_location get_pending_location_for_diag() const override
Definition region-model.h:980
bool checking_for_infinite_loop_p() const override
Definition region-model.h:1150
logger * get_logger() override
Definition region-model.h:1018
region_model_context_decorator(region_model_context *inner)
Definition region-model.h:1163
void purge_state_involving(const svalue *sval) override
Definition region-model.h:1080
void on_liveness_change(const svalue_set &live_svalues, const region_model *model) override
Definition region-model.h:1011
void on_unexpected_tree_code(tree t, const dump_location_t &loc) override
Definition region-model.h:1059
void on_escaped_function(tree fndecl) override
Definition region-model.h:1066
bool warn_at(std::unique_ptr< pending_diagnostic > d, pending_location &&ploc) override
Definition region-model.h:989
void terminate_path() override
Definition region-model.h:1092
void on_unknown_change(const svalue *sval, bool is_mutable) override
Definition region-model.h:1047
void on_unusable_in_infinite_loop() override
Definition region-model.h:1156
bool get_state_map_by_name(const char *name, sm_state_map **out_smap, const state_machine **out_sm, unsigned *out_sm_idx, std::unique_ptr< sm_context > *out_sm_context) override
Definition region-model.h:1106
Definition region-model.h:748
virtual void on_bounded_ranges(const svalue &sval, const bounded_ranges &ranges)=0
virtual bool warn_at(std::unique_ptr< pending_diagnostic > d, pending_location &&ploc)=0
virtual void bifurcate(std::unique_ptr< custom_edge_info > info)=0
virtual void purge_state_involving(const svalue *sval)=0
virtual void on_escaped_function(tree fndecl)=0
bool get_malloc_map(sm_state_map **out_smap, const state_machine **out_sm, unsigned *out_sm_idx)
Definition region-model.h:852
virtual void on_pop_frame(const frame_region *)=0
virtual void on_liveness_change(const svalue_set &live_svalues, const region_model *model)=0
bool get_fd_map(sm_state_map **out_smap, const state_machine **out_sm, unsigned *out_sm_idx, std::unique_ptr< sm_context > *out_sm_context)
Definition region-model.h:844
virtual void add_note(std::unique_ptr< pending_note > pn)=0
virtual pending_location get_pending_location_for_diag() const =0
virtual logger * get_logger()=0
virtual const extrinsic_state * get_ext_state() const =0
bool possibly_tainted_p(const svalue *sval)
virtual const program_state * get_state() const =0
virtual void on_unexpected_tree_code(tree t, const dump_location_t &loc)=0
virtual void on_phi(const gphi *phi, tree rhs)=0
bool get_taint_map(sm_state_map **out_smap, const state_machine **out_sm, unsigned *out_sm_idx)
Definition region-model.h:859
virtual void on_svalue_leak(const svalue *sval)=0
virtual void on_condition(const svalue *lhs, enum tree_code op, const svalue *rhs)=0
virtual void on_unusable_in_infinite_loop()=0
bool warn(std::unique_ptr< pending_diagnostic > d, std::unique_ptr< pending_location::fixer_for_epath > ploc_fixer=nullptr)
virtual const exploded_graph * get_eg() const =0
virtual void on_unknown_change(const svalue *sval, bool is_mutable)=0
virtual const gimple * get_stmt() const =0
virtual uncertainty_t * get_uncertainty()=0
virtual bool checking_for_infinite_loop_p() const =0
virtual void terminate_path()=0
virtual bool get_state_map_by_name(const char *name, sm_state_map **out_smap, const state_machine **out_sm, unsigned *out_sm_idx, std::unique_ptr< sm_context > *out_sm_context)=0
virtual void add_event(std::unique_ptr< checker_event > event)=0
virtual void maybe_did_work()=0
Definition region-model-manager.h:32
Definition region-model.h:294
path_var get_representative_path_var(const svalue *sval, svalue_set *visited, logger *logger) const
region_model(const region_model &other)
void impl_deallocation_call(const call_details &cd)
void update_for_zero_return(const call_details &cd, bool unmergeable)
bool add_constraint(tree lhs, enum tree_code op, tree rhs, region_model_context *ctxt)
store * get_store()
Definition region-model.h:480
constraint_manager * get_constraints()
Definition region-model.h:475
void update_for_nonzero_return(const call_details &cd)
bool add_constraint(tree lhs, enum tree_code op, tree rhs, region_model_context *ctxt, std::unique_ptr< rejected_constraint > *out)
void dump(FILE *fp, bool simple, bool multiline) const
bool replay_call_summary(call_summary_replay &r, const region_model &summary)
void check_region_size(const region *lhs_reg, const svalue *rhs_sval, region_model_context *ctxt) const
void zero_fill_region(const region *reg, region_model_context *ctxt)
void on_asm_stmt(const gasm *asm_stmt, region_model_context *ctxt)
bool check_region_access(const region *reg, enum access_direction dir, const svalue *sval_hint, region_model_context *ctxt) const
void update_for_return_gcall(const gcall &call_stmt, region_model_context *ctxt)
bounded_ranges_manager * get_range_manager() const
Definition region-model.h:495
void dump(bool simple) const
path_var get_representative_path_var_1(const svalue *sval, svalue_set *visited, logger *logger) const
static void append_regions_cb(const region *base_reg, struct append_regions_cb_data *data)
const region * deref_rvalue(const svalue *ptr_sval, tree ptr_tree, region_model_context *ctxt, bool add_nonnull_constraint=true) const
void debug() const
const builtin_known_function * get_builtin_kf(const gcall &call, region_model_context *ctxt=nullptr) const
void on_realloc_with_move(const call_details &cd, const svalue *old_ptr_sval, const svalue *new_ptr_sval)
constraint_manager * m_constraints
Definition region-model.h:725
const exception_node * get_current_thrown_exception() const
Definition region-model.h:597
const svalue * check_for_null_terminated_string_arg(const call_details &cd, unsigned idx) const
void set_errno(const call_details &cd)
void set_dynamic_extents(const region *reg, const svalue *size_in_bytes, region_model_context *ctxt)
void check_region_for_write(const region *dest_reg, const svalue *sval_hint, region_model_context *ctxt) const
const svalue * get_dynamic_extents(const region *reg) const
void clobber_region(const region *reg)
void dump_to_pp(pretty_printer *pp, bool simple, bool multiline) const
void transition_ptr_sval_non_null(region_model_context *ctxt, const svalue *new_ptr_sval)
const svalue * get_rvalue_1(path_var pv, region_model_context *ctxt) const
tristate eval_condition(const svalue *lhs, enum tree_code op, const svalue *rhs) const
const svalue * check_for_null_terminated_string_arg(const call_details &cd, unsigned idx, bool include_terminator, const svalue **out_sval) const
void mark_as_tainted(const svalue *sval, region_model_context *ctxt)
const svalue * get_capacity(const region *reg) const
bool add_constraint(const svalue *lhs, enum tree_code op, const svalue *rhs, region_model_context *ctxt)
void on_assignment(const gassign *stmt, region_model_context *ctxt)
tristate symbolic_greater_than(const binop_svalue *a, const svalue *b) const
region_model & operator=(const region_model &other)
const frame_region * get_frame_at_index(int index) const
store m_store
Definition region-model.h:723
void print(pretty_printer *pp) const
const region * create_region_for_alloca(const svalue *size_in_bytes, region_model_context *ctxt)
void purge_region(const region *reg)
void validate() const
const region * push_frame(const function &fun, const gcall *call_stmt, const vec< const svalue * > *arg_sids, region_model_context *ctxt)
bool operator!=(const region_model &other) const
Definition region-model.h:304
const svalue * get_store_value(const region *reg, region_model_context *ctxt) const
const region * get_region_for_poisoned_expr(tree expr) const
void push_caught_exception(const exception_node &node)
Definition region-model.h:611
void poison_any_pointers_to_descendents(const region *reg, enum poison_kind pkind)
void update_for_gcall(const gcall &call_stmt, region_model_context *ctxt, function *callee=nullptr)
hashval_t hash() const
void check_function_attrs(const gcall &call, tree callee_fndecl, region_model_context *ctxt)
void get_regions_for_current_frame(auto_vec< const decl_region * > *out) const
std::unique_ptr< text_art::tree_widget > make_dump_widget(const text_art::dump_widget_info &dwi) const
const svalue * get_rvalue_for_bits(tree type, const region *reg, const bit_range &bits, region_model_context *ctxt) const
const svalue * get_initial_value_for_global(const region *reg) const
std::vector< exception_node > m_caught_exceptions_stack
Definition region-model.h:730
tristate compare_initial_and_pointer(const initial_svalue *init, const region_svalue *ptr) const
exception_node pop_thrown_exception()
Definition region-model.h:603
const function * get_current_function() const
const svalue * get_store_bytes(const region *base_reg, const byte_range &bytes, region_model_context *ctxt) const
bool can_merge_with_p(const region_model &other_model, const program_point &point, region_model *out_model, const extrinsic_state *ext_state=nullptr, const program_state *state_a=nullptr, const program_state *state_b=nullptr) const
bool on_call_pre(const gcall &stmt, region_model_context *ctxt)
void loop_replay_fixup(const region_model *dst_state)
void check_dynamic_size_for_floats(const svalue *size_in_bytes, region_model_context *ctxt) const
bool operator==(const region_model &other) const
tree get_fndecl_for_call(const gcall &call, region_model_context *ctxt)
bool region_exists_p(const region *reg) const
tree get_representative_tree(const svalue *sval, logger *logger=nullptr) const
void check_function_attr_access(const gcall &call, tree callee_fndecl, region_model_context *ctxt, rdwr_map &rdwr_idx) const
region_model(region_model_manager *mgr)
void handle_unrecognized_call(const gcall &call, region_model_context *ctxt)
void push_thrown_exception(const exception_node &node)
Definition region-model.h:593
const svalue * maybe_get_copy_bounds(const region *src_reg, const svalue *num_bytes_sval)
region_to_value_map dynamic_extents_t
Definition region-model.h:296
tristate eval_condition(tree lhs, enum tree_code op, tree rhs, region_model_context *ctxt) const
const svalue * scan_for_null_terminator_1(const region *reg, tree expr, const svalue **out_sval, region_model_context *ctxt) const
bool called_from_main_p() const
bool add_constraints_from_binop(const svalue *outer_lhs, enum tree_code outer_op, const svalue *outer_rhs, bool *out, region_model_context *ctxt)
void check_one_function_attr_null_terminated_string_arg(const gcall &call, tree callee_fndecl, region_model_context *ctxt, rdwr_map &rdwr_idx, tree attr)
void set_value(const region *lhs_reg, const svalue *rhs_sval, region_model_context *ctxt)
void on_return(const greturn *stmt, region_model_context *ctxt)
void get_reachable_svalues(svalue_set *out, const svalue *extra_sval, const uncertainty_t *uncertainty)
tristate structural_equality(const svalue *a, const svalue *b) const
void copy_bytes(const region *dest_reg, const region *src_reg, tree src_ptr_expr, const svalue *num_bytes_sval, region_model_context *ctxt)
const frame_region * get_current_frame() const
Definition region-model.h:384
std::unique_ptr< json::object > to_json() const
region_model_manager *const m_mgr
Definition region-model.h:721
const frame_region * m_current_frame
Definition region-model.h:727
void dump() const
void check_region_for_taint(const region *reg, enum access_direction dir, region_model_context *ctxt) const
void on_top_level_param(tree param, bool nonnull, region_model_context *ctxt)
void check_function_attr_null_terminated_string_arg(const gcall &call, tree callee_fndecl, region_model_context *ctxt, rdwr_map &rdwr_idx)
void mark_region_as_unknown(const region *reg, uncertainty_t *uncertainty)
void on_stmt_pre(const gimple *stmt, bool *out_unknown_side_effects, region_model_context *ctxt)
void check_call_format_attr(const call_details &cd, tree format_attr) const
void check_for_throw_inside_call(const gcall &call, tree fndecl, region_model_context *ctxt)
void set_value(tree lhs, tree rhs, region_model_context *ctxt)
bool check_region_for_read(const region *src_reg, region_model_context *ctxt) const
void maybe_complain_about_infoleak(const region *dst_reg, const svalue *copied_sval, const region *src_reg, region_model_context *ctxt)
region_model_manager * get_manager() const
Definition region-model.h:494
const known_function * get_known_function(tree fndecl, const call_details &cd) const
const exception_node * get_current_caught_exception() const
Definition region-model.h:615
void write_bytes(const region *dest_reg, const svalue *num_bytes_sval, const svalue *sval, region_model_context *ctxt)
void on_longjmp(const gcall &longjmp_call, const gcall &setjmp_call, int setjmp_stack_depth, region_model_context *ctxt)
void unset_dynamic_extents(const region *reg)
void on_setjmp(const gcall &stmt, const exploded_node &enode, const superedge &sedge, region_model_context *ctxt)
void check_for_writable_region(const region *dest_reg, region_model_context *ctxt) const
void update_for_null_return(const call_details &cd, bool unmergeable)
const region * get_lvalue_1(path_var pv, region_model_context *ctxt) const
const known_function * get_known_function(enum internal_fn) const
path_var get_representative_path_var(const region *reg, svalue_set *visited, logger *logger) const
const region * get_lvalue(path_var pv, region_model_context *ctxt) const
void get_referenced_base_regions(auto_bitmap &out_ids) const
void update_for_int_cst_return(const call_details &cd, int retval, bool unmergeable)
const svalue * check_for_poison(const svalue *sval, tree expr, const region *src_region, region_model_context *ctxt) const
int get_stack_depth() const
std::vector< exception_node > m_thrown_exceptions_stack
Definition region-model.h:729
bool check_region_bounds(const region *reg, enum access_direction dir, const svalue *sval_hint, region_model_context *ctxt) const
tree get_representative_tree(const region *reg, logger *logger=nullptr) const
void on_call_post(const gcall &stmt, bool unknown_side_effects, region_model_context *ctxt)
void check_call_args(const call_details &cd) const
void fill_region(const region *reg, const svalue *sval, region_model_context *ctxt)
void check_dynamic_size_for_taint(enum memory_space mem_space, const svalue *size_in_bytes, region_model_context *ctxt) const
bool check_symbolic_bounds(const region *base_reg, const svalue *sym_byte_offset, const svalue *num_bytes_sval, const svalue *capacity, enum access_direction dir, const svalue *sval_hint, region_model_context *ctxt) const
void mark_as_valid_fd(const svalue *sval, region_model_context *ctxt)
const svalue * get_rvalue(path_var pv, region_model_context *ctxt) const
path_var get_representative_path_var_1(const region *reg, svalue_set *visited, logger *logger) const
const store * get_store() const
Definition region-model.h:481
dynamic_extents_t m_dynamic_extents
Definition region-model.h:736
const region * get_lvalue(tree expr, region_model_context *ctxt) const
const svalue * get_gassign_result(const gassign *assign, region_model_context *ctxt)
const svalue * read_bytes(const region *src_reg, tree src_ptr_expr, const svalue *num_bytes_sval, region_model_context *ctxt) const
void pop_frame(tree result_lvalue, const svalue **out_result, region_model_context *ctxt, const gcall *call_stmt, bool eval_return_svalue=true)
const svalue * scan_for_null_terminator(const region *reg, tree expr, const svalue **out_sval, region_model_context *ctxt) const
const region * get_or_create_region_for_heap_alloc(const svalue *size_in_bytes, region_model_context *ctxt, bool update_state_machine=false, const call_details *cd=nullptr)
exception_node pop_caught_exception()
Definition region-model.h:621
const svalue * get_rvalue(tree expr, region_model_context *ctxt) const
bool canonicalized_p() const
void purge_state_involving(const svalue *sval, region_model_context *ctxt)
void unbind_region_and_descendents(const region *reg, enum poison_kind pkind)
const dynamic_extents_t & get_dynamic_extents() const
Definition region-model.h:484
Definition svalue.h:229
Definition region-model.h:144
void remove(const region *reg)
Definition region-model.h:171
iterator begin() const
Definition region-model.h:160
const svalue *const * get(const region *reg) const
Definition region-model.h:163
hash_map_t::iterator iterator
Definition region-model.h:147
hash_map_t m_hash_map
Definition region-model.h:192
std::unique_ptr< text_art::tree_widget > make_dump_widget(const text_art::dump_widget_info &dwi) const
void purge_state_involving(const svalue *sval)
std::unique_ptr< json::object > to_json() const
hash_map< const region *, const svalue * > hash_map_t
Definition region-model.h:146
bool operator==(const region_to_value_map &other) const
iterator end() const
Definition region-model.h:161
void put(const region *reg, const svalue *sval)
Definition region-model.h:167
bool is_empty() const
Definition region-model.h:176
region_to_value_map & operator=(const region_to_value_map &other)
region_to_value_map()
Definition region-model.h:149
void dump_to_pp(pretty_printer *pp, bool simple, bool multiline) const
bool operator!=(const region_to_value_map &other) const
Definition region-model.h:155
void dump(bool simple) const
bool can_merge_with_p(const region_to_value_map &other, region_to_value_map *out) const
region_to_value_map(const region_to_value_map &other)
Definition region-model.h:150
Definition region.h:127
virtual ~rejected_constraint()
Definition region-model.h:1256
virtual void dump_to_pp(pretty_printer *pp) const =0
rejected_constraint(const region_model &model)
Definition region-model.h:1262
const region_model & get_model() const
Definition region-model.h:1259
region_model m_model
Definition region-model.h:1266
void dump_to_pp(pretty_printer *pp) const final override
rejected_default_case(const region_model &model)
Definition region-model.h:1288
const svalue * m_lhs
Definition region-model.h:1280
void dump_to_pp(pretty_printer *pp) const final override
enum tree_code m_op
Definition region-model.h:1281
rejected_op_constraint(const region_model &model, const svalue *lhs, enum tree_code op, const svalue *rhs)
Definition region-model.h:1272
const svalue * m_rhs
Definition region-model.h:1282
void dump_to_pp(pretty_printer *pp) const final override
tree m_expr
Definition region-model.h:1307
const bounded_ranges * m_ranges
Definition region-model.h:1308
rejected_ranges_constraint(const region_model &model, tree expr, const bounded_ranges *ranges)
Definition region-model.h:1298
Definition svalue.h:1004
Definition svalue.h:585
Definition program-state.h:92
Definition sm.h:43
Definition store.h:923
Definition svalue.h:918
Definition supergraph.h:281
Definition supergraph.h:105
Definition supergraph.h:224
Definition svalue.h:92
int m_num_unexpected_codes
Definition region-model.h:970
bool had_errors_p() const
Definition region-model.h:967
void on_unexpected_tree_code(tree, const dump_location_t &) final override
Definition region-model.h:961
tentative_region_model_context()
Definition region-model.h:959
Definition svalue.h:719
Definition store.h:162
Definition svalue.h:414
Definition svalue.h:1202
Definition region-model.h:223
virtual void visit_unknown_svalue(const unknown_svalue *)
Definition region-model.h:227
virtual void visit_poisoned_svalue(const poisoned_svalue *)
Definition region-model.h:228
virtual void visit_asm_output_svalue(const asm_output_svalue *)
Definition region-model.h:241
virtual void visit_unaryop_svalue(const unaryop_svalue *)
Definition region-model.h:231
virtual void visit_region_svalue(const region_svalue *)
Definition region-model.h:225
virtual void visit_initial_svalue(const initial_svalue *)
Definition region-model.h:230
virtual void visit_sub_svalue(const sub_svalue *)
Definition region-model.h:233
virtual void visit_setjmp_svalue(const setjmp_svalue *)
Definition region-model.h:229
virtual void visit_conjured_svalue(const conjured_svalue *)
Definition region-model.h:240
virtual void visit_placeholder_svalue(const placeholder_svalue *)
Definition region-model.h:237
virtual void visit_binop_svalue(const binop_svalue *)
Definition region-model.h:232
virtual void visit_const_fn_result_svalue(const const_fn_result_svalue *)
Definition region-model.h:242
virtual void visit_compound_svalue(const compound_svalue *)
Definition region-model.h:239
virtual void visit_unmergeable_svalue(const unmergeable_svalue *)
Definition region-model.h:236
virtual void visit_widening_svalue(const widening_svalue *)
Definition region-model.h:238
virtual void visit_bits_within_svalue(const bits_within_svalue *)
Definition region-model.h:235
virtual void visit_constant_svalue(const constant_svalue *)
Definition region-model.h:226
virtual void visit_region(const region *)
Definition region-model.h:244
virtual void visit_repeated_svalue(const repeated_svalue *)
Definition region-model.h:234
Definition svalue.h:1297
Definition bitmap.h:950
Definition vec.h:1667
Definition genoutput.cc:150
Definition dumpfile.h:446
Definition genmatch.cc:1506
Definition ree.cc:583
Definition hash-map.h:40
Definition hash-set.h:37
Definition inchash.h:38
Definition pretty-print.h:241
void set_output_stream(FILE *outfile)
Definition pretty-print.h:274
Definition tristate.h:26
bool debug
Definition collect-utils.cc:34
union tree_node * tree
Definition coretypes.h:97
void internal_error(const char *,...) ATTRIBUTE_GCC_DIAG(1
void final(rtx_insn *first, FILE *file, int optimize_p)
Definition final.cc:2009
internal_fn
Definition genmatch.cc:1015
tree_code
Definition genmatch.cc:1002
Definition access-diagram.h:30
std::unique_ptr< pending_diagnostic > make_write_to_const_diagnostic(const region *dest_reg, tree decl)
@ stmt
Definition checker-event.h:38
std::unique_ptr< pending_diagnostic > make_shift_count_negative_diagnostic(const gassign *assign, tree count_cst, const region *src_region)
access_direction
Definition common.h:377
std::unique_ptr< pending_diagnostic > make_poisoned_value_diagnostic(tree expr, enum poison_kind pkind, const region *src_region, tree check_expr)
std::unique_ptr< pending_diagnostic > make_write_to_string_literal_diagnostic(const region *reg)
poison_kind
Definition svalue.h:446
std::unique_ptr< pending_diagnostic > make_shift_count_overflow_diagnostic(const gassign *assign, int operand_precision, tree count_cst, const region *src_region)
hash_set< const svalue * > svalue_set
Definition common.h:76
memory_space
Definition region.h:33
Definition custom-sarif-properties/state-graphs.h:33
Definition fold-const.cc:4353
void add_path_var(path_var pv, hash &hstate)
Definition dump-context.h:31
poly_int< N, C > r
Definition poly-int.h:774
i
Definition poly-int.h:776
Ca const poly_int< N, Cb > & b
Definition poly-int.h:771
Ca & a
Definition poly-int.h:770
void pp_flush(pretty_printer *pp)
Definition pretty-print.cc:2462
void pp_newline(pretty_printer *pp)
Definition pretty-print.cc:2737
void pp_string(pretty_printer *pp, const char *str)
Definition pretty-print.cc:2764
Definition store.h:234
Definition constraint-manager.h:123
Definition store.h:328
Definition region-model.h:253
void dump() const
exception_node(const svalue *exception_sval, const svalue *typeinfo_sval, const svalue *destructor_sval)
Definition region-model.h:254
const svalue * m_exception_sval
Definition region-model.h:279
bool operator==(const exception_node &other) const
tree maybe_get_type() const
void dump(bool simple) const
const svalue * m_typeinfo_sval
Definition region-model.h:280
const svalue * m_destructor_sval
Definition region-model.h:281
void dump(FILE *fp, bool simple) const
std::unique_ptr< text_art::tree_widget > make_dump_widget(const text_art::dump_widget_info &dwi) const
void dump_to_pp(pretty_printer *pp, bool simple) const
std::unique_ptr< json::object > to_json() const
void add_to_reachable_regions(reachable_regions &) const
const program_point & m_point
Definition region-model.h:1240
const program_state * m_state_a
Definition region-model.h:1244
const region_model * m_model_a
Definition region-model.h:1238
const extrinsic_state * m_ext_state
Definition region-model.h:1243
hash_set< const svalue * > m_svals_changing_meaning
Definition region-model.h:1247
void dump_to_pp(pretty_printer *pp, bool simple) const
model_merger(const region_model *model_a, const region_model *model_b, const program_point &point, region_model *merged_model, const extrinsic_state *ext_state, const program_state *state_a, const program_state *state_b)
Definition region-model.h:1205
const program_state * m_state_b
Definition region-model.h:1245
const supernode * get_supernode() const
Definition region-model.h:1231
bool mergeable_svalue_p(const svalue *) const
void dump(bool simple) const
region_model_manager * get_manager() const
Definition region-model.h:1224
void on_widening_reuse(const widening_svalue *widening_sval)
region_model * m_merged_model
Definition region-model.h:1241
void dump(FILE *fp, bool simple) const
const region_model * m_model_b
Definition region-model.h:1239
Definition diagnostic-manager.h:35
int m_num_equiv_classes
Definition region-model.h:213
int m_num_bounded_ranges_constraints
Definition region-model.h:215
int m_num_client_items
Definition region-model.h:216
int m_num_svalues
Definition region-model.h:211
purge_stats()
Definition region-model.h:202
int m_num_constraints
Definition region-model.h:214
int m_num_regions
Definition region-model.h:212
Definition genautomata.cc:499
Definition function.h:249
Definition gimple.h:552
Definition gimple.h:910
Definition gimple.h:355
Definition gimple.h:224
Definition gimple.h:464
Definition gimple.h:920
Definition dump-widget-info.h:31
Definition gengtype.h:252
Definition vec.h:450
#define gcc_assert(EXPR)
Definition system.h:817
#define DEBUG_FUNCTION
Definition system.h:1191
static bitmap visited
Definition tree-ssa-dce.cc:664
static control_dependences * cd
Definition tree-ssa-dce.cc:105
const char * get_tree_code_name(enum tree_code code)
Definition tree.cc:13053
#define TREE_CODE(NODE)
Definition tree.h:325
tree size_in_bytes(const_tree t)
Definition tree.h:5283
#define FOR_EACH_VEC_ELT(V, I, P)
Definition vec.h:1895