]> git.ipfire.org Git - thirdparty/gcc.git/blob - gcc/analyzer/diagnostic-manager.h
analyzer: early rejection of disabled warnings [PR104955]
[thirdparty/gcc.git] / gcc / analyzer / diagnostic-manager.h
1 /* Classes for saving, deduplicating, and emitting analyzer diagnostics.
2 Copyright (C) 2019-2022 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 #ifndef GCC_ANALYZER_DIAGNOSTIC_MANAGER_H
22 #define GCC_ANALYZER_DIAGNOSTIC_MANAGER_H
23
24 namespace ana {
25
26 class epath_finder;
27
28 /* A to-be-emitted diagnostic stored within diagnostic_manager. */
29
30 class saved_diagnostic
31 {
32 public:
33 saved_diagnostic (const state_machine *sm,
34 const exploded_node *enode,
35 const supernode *snode, const gimple *stmt,
36 stmt_finder *stmt_finder,
37 tree var, const svalue *sval,
38 state_machine::state_t state,
39 pending_diagnostic *d,
40 unsigned idx);
41 ~saved_diagnostic ();
42
43 bool operator== (const saved_diagnostic &other) const;
44
45 void add_note (pending_note *pn);
46
47 json::object *to_json () const;
48
49 const feasibility_problem *get_feasibility_problem () const
50 {
51 return m_problem;
52 }
53
54 bool calc_best_epath (epath_finder *pf);
55 const exploded_path *get_best_epath () const { return m_best_epath; }
56 unsigned get_epath_length () const;
57
58 void add_duplicate (saved_diagnostic *other);
59 unsigned get_num_dupes () const { return m_duplicates.length (); }
60
61 unsigned get_index () const { return m_idx; }
62
63 bool supercedes_p (const saved_diagnostic &other) const;
64
65 void emit_any_notes () const;
66
67 //private:
68 const state_machine *m_sm;
69 const exploded_node *m_enode;
70 const supernode *m_snode;
71 const gimple *m_stmt;
72 stmt_finder *m_stmt_finder;
73 tree m_var;
74 const svalue *m_sval;
75 state_machine::state_t m_state;
76 pending_diagnostic *m_d; // owned
77 const exploded_edge *m_trailing_eedge;
78
79 private:
80 DISABLE_COPY_AND_ASSIGN (saved_diagnostic);
81
82 unsigned m_idx;
83 exploded_path *m_best_epath; // owned
84 feasibility_problem *m_problem; // owned
85
86 auto_vec<const saved_diagnostic *> m_duplicates;
87 auto_delete_vec <pending_note> m_notes;
88 };
89
90 class path_builder;
91
92 /* A class with responsibility for saving pending diagnostics, so that
93 they can be emitted after the exploded_graph is complete.
94 This lets us de-duplicate diagnostics, and find the shortest path
95 for each similar diagnostic, potentially using edges that might
96 not have been found when each diagnostic was first saved.
97
98 This also lets us compute shortest_paths once, rather than
99 per-diagnostic. */
100
101 class diagnostic_manager : public log_user
102 {
103 public:
104 diagnostic_manager (logger *logger, engine *eng, int verbosity);
105
106 engine *get_engine () const { return m_eng; }
107
108 json::object *to_json () const;
109
110 void add_diagnostic (const state_machine *sm,
111 exploded_node *enode,
112 const supernode *snode, const gimple *stmt,
113 stmt_finder *finder,
114 tree var,
115 const svalue *sval,
116 state_machine::state_t state,
117 pending_diagnostic *d);
118
119 void add_diagnostic (exploded_node *enode,
120 const supernode *snode, const gimple *stmt,
121 stmt_finder *finder,
122 pending_diagnostic *d);
123
124 void add_note (pending_note *pn);
125
126 void emit_saved_diagnostics (const exploded_graph &eg);
127
128 void emit_saved_diagnostic (const exploded_graph &eg,
129 const saved_diagnostic &sd);
130
131 unsigned get_num_diagnostics () const
132 {
133 return m_saved_diagnostics.length ();
134 }
135 saved_diagnostic *get_saved_diagnostic (unsigned idx)
136 {
137 return m_saved_diagnostics[idx];
138 }
139 const saved_diagnostic *get_saved_diagnostic (unsigned idx) const
140 {
141 return m_saved_diagnostics[idx];
142 }
143
144 private:
145 void build_emission_path (const path_builder &pb,
146 const exploded_path &epath,
147 checker_path *emission_path) const;
148
149 void add_events_for_eedge (const path_builder &pb,
150 const exploded_edge &eedge,
151 checker_path *emission_path,
152 interesting_t *interest) const;
153
154 bool significant_edge_p (const path_builder &pb,
155 const exploded_edge &eedge) const;
156
157 void add_events_for_superedge (const path_builder &pb,
158 const exploded_edge &eedge,
159 checker_path *emission_path) const;
160
161 void prune_path (checker_path *path,
162 const state_machine *sm,
163 const svalue *sval,
164 state_machine::state_t state) const;
165
166 void prune_for_sm_diagnostic (checker_path *path,
167 const state_machine *sm,
168 tree var,
169 state_machine::state_t state) const;
170 void prune_for_sm_diagnostic (checker_path *path,
171 const state_machine *sm,
172 const svalue *sval,
173 state_machine::state_t state) const;
174 void update_for_unsuitable_sm_exprs (tree *expr) const;
175 void prune_interproc_events (checker_path *path) const;
176 void consolidate_conditions (checker_path *path) const;
177 void finish_pruning (checker_path *path) const;
178
179 engine *m_eng;
180 auto_delete_vec<saved_diagnostic> m_saved_diagnostics;
181 const int m_verbosity;
182 int m_num_disabled_diagnostics;
183 };
184
185 } // namespace ana
186
187 #endif /* GCC_ANALYZER_DIAGNOSTIC_MANAGER_H */