annotate gcc/analyzer/sm.h @ 145:1830386684a0

gcc-9.2.0
author anatofuz
date Thu, 13 Feb 2020 11:34:05 +0900
parents
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
145
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
1 /* Modeling API uses and misuses via state machines.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
2 Copyright (C) 2019-2020 Free Software Foundation, Inc.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
3 Contributed by David Malcolm <dmalcolm@redhat.com>.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
4
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
5 This file is part of GCC.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
6
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
7 GCC is free software; you can redistribute it and/or modify it
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
8 under the terms of the GNU General Public License as published by
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
9 the Free Software Foundation; either version 3, or (at your option)
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
10 any later version.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
11
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
12 GCC is distributed in the hope that it will be useful, but
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
13 WITHOUT ANY WARRANTY; without even the implied warranty of
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
15 General Public License for more details.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
16
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
17 You should have received a copy of the GNU General Public License
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
18 along with GCC; see the file COPYING3. If not see
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
19 <http://www.gnu.org/licenses/>. */
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
20
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
21 #ifndef GCC_ANALYZER_SM_H
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
22 #define GCC_ANALYZER_SM_H
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
23
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
24 /* Utility functions for use by state machines. */
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
25
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
26 extern tree is_zero_assignment (const gimple *stmt);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
27 extern bool any_pointer_p (tree var);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
28
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
29 namespace ana {
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
30
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
31 class state_machine;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
32 class sm_context;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
33 class pending_diagnostic;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
34
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
35 /* An abstract base class for a state machine describing an API.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
36 A mapping from state IDs to names, and various virtual functions
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
37 for pattern-matching on statements. */
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
38
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
39 class state_machine : public log_user
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
40 {
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
41 public:
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
42 typedef unsigned state_t;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
43
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
44 state_machine (const char *name, logger *logger)
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
45 : log_user (logger), m_name (name) {}
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
46
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
47 virtual ~state_machine () {}
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
48
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
49 /* Should states be inherited from a parent region to a child region,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
50 when first accessing a child region?
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
51 For example we should inherit the taintedness of a subregion,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
52 but we should not inherit the "malloc:non-null" state of a field
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
53 within a heap-allocated struct. */
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
54 virtual bool inherited_state_p () const = 0;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
55
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
56 const char *get_name () const { return m_name; }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
57
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
58 const char *get_state_name (state_t s) const;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
59
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
60 /* Return true if STMT is a function call recognized by this sm. */
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
61 virtual bool on_stmt (sm_context *sm_ctxt,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
62 const supernode *node,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
63 const gimple *stmt) const = 0;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
64
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
65 virtual void on_phi (sm_context *sm_ctxt ATTRIBUTE_UNUSED,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
66 const supernode *node ATTRIBUTE_UNUSED,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
67 const gphi *phi ATTRIBUTE_UNUSED,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
68 tree rhs ATTRIBUTE_UNUSED) const
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
69 {
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
70 }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
71
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
72 virtual void on_condition (sm_context *sm_ctxt,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
73 const supernode *node,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
74 const gimple *stmt,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
75 tree lhs, enum tree_code op, tree rhs) const = 0;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
76
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
77 /* Return true if it safe to discard the given state (to help
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
78 when simplifying state objects).
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
79 States that need leak detection should return false. */
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
80 virtual bool can_purge_p (state_t s) const = 0;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
81
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
82 /* Called when VAR leaks (and !can_purge_p). */
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
83 virtual pending_diagnostic *on_leak (tree var ATTRIBUTE_UNUSED) const
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
84 {
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
85 return NULL;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
86 }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
87
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
88 void validate (state_t s) const;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
89
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
90 void dump_to_pp (pretty_printer *pp) const;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
91
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
92 protected:
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
93 state_t add_state (const char *name);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
94
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
95 private:
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
96 DISABLE_COPY_AND_ASSIGN (state_machine);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
97
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
98 const char *m_name;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
99 auto_vec<const char *> m_state_names;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
100 };
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
101
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
102 /* Is STATE the start state? (zero is hardcoded as the start state). */
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
103
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
104 static inline bool
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
105 start_start_p (state_machine::state_t state)
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
106 {
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
107 return state == 0;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
108 }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
109
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
110 /* Abstract base class for state machines to pass to
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
111 sm_context::on_custom_transition for handling non-standard transitions
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
112 (e.g. adding a node and edge to simulate registering a callback and having
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
113 the callback be called later). */
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
114
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
115 class custom_transition
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
116 {
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
117 public:
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
118 virtual ~custom_transition () {}
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
119 virtual void impl_transition (exploded_graph *eg,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
120 exploded_node *src_enode,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
121 int sm_idx) = 0;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
122 };
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
123
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
124 /* Abstract base class giving an interface for the state machine to call
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
125 the checker engine, at a particular stmt. */
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
126
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
127 class sm_context
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
128 {
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
129 public:
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
130 virtual ~sm_context () {}
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
131
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
132 /* Get the fndecl used at call, or NULL_TREE.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
133 Use in preference to gimple_call_fndecl (and gimple_call_addr_fndecl),
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
134 since it can look through function pointer assignments and
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
135 other callback handling. */
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
136 virtual tree get_fndecl_for_call (const gcall *call) = 0;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
137
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
138 /* Called by state_machine in response to pattern matches:
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
139 if VAR is in state FROM, transition it to state TO, potentially
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
140 recording the "origin" of the state as ORIGIN.
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
141 Use NODE and STMT for location information. */
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
142 virtual void on_transition (const supernode *node, const gimple *stmt,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
143 tree var,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
144 state_machine::state_t from,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
145 state_machine::state_t to,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
146 tree origin = NULL_TREE) = 0;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
147
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
148 /* Called by state_machine in response to pattern matches:
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
149 issue a diagnostic D if VAR is in state STATE, using NODE and STMT
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
150 for location information. */
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
151 virtual void warn_for_state (const supernode *node, const gimple *stmt,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
152 tree var, state_machine::state_t state,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
153 pending_diagnostic *d) = 0;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
154
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
155 virtual tree get_readable_tree (tree expr)
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
156 {
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
157 return expr;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
158 }
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
159
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
160 virtual state_machine::state_t get_global_state () const = 0;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
161 virtual void set_global_state (state_machine::state_t) = 0;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
162
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
163 /* A vfunc for handling custom transitions, such as when registering
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
164 a signal handler. */
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
165 virtual void on_custom_transition (custom_transition *transition) = 0;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
166
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
167 protected:
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
168 sm_context (int sm_idx, const state_machine &sm)
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
169 : m_sm_idx (sm_idx), m_sm (sm) {}
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
170
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
171 int m_sm_idx;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
172 const state_machine &m_sm;
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
173 };
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
174
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
175
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
176 /* The various state_machine subclasses are hidden in their respective
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
177 implementation files. */
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
178
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
179 extern void make_checkers (auto_delete_vec <state_machine> &out,
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
180 logger *logger);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
181
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
182 extern state_machine *make_malloc_state_machine (logger *logger);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
183 extern state_machine *make_fileptr_state_machine (logger *logger);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
184 extern state_machine *make_taint_state_machine (logger *logger);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
185 extern state_machine *make_sensitive_state_machine (logger *logger);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
186 extern state_machine *make_signal_state_machine (logger *logger);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
187 extern state_machine *make_pattern_test_state_machine (logger *logger);
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
188
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
189 } // namespace ana
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
190
1830386684a0 gcc-9.2.0
anatofuz
parents:
diff changeset
191 #endif /* GCC_ANALYZER_SM_H */