diff options
author | Emese Revfy <re.emese@gmail.com> | 2016-05-23 18:09:38 -0400 |
---|---|---|
committer | Michal Marek <mmarek@suse.com> | 2016-06-07 16:57:10 -0400 |
commit | 6b90bd4ba40b38dc13c2782469c1c77e4ed79915 (patch) | |
tree | 02d65b38b76e3543d33088ae9149010bae0290b0 /scripts/gcc-plugins/gcc-generate-simple_ipa-pass.h | |
parent | 24403874316a7180d367e51d7f7e25d5de1f78dd (diff) |
GCC plugin infrastructure
This patch allows to build the whole kernel with GCC plugins. It was ported from
grsecurity/PaX. The infrastructure supports building out-of-tree modules and
building in a separate directory. Cross-compilation is supported too.
Currently the x86, arm, arm64 and uml architectures enable plugins.
The directory of the gcc plugins is scripts/gcc-plugins. You can use a file or a directory
there. The plugins compile with these options:
* -fno-rtti: gcc is compiled with this option so the plugins must use it too
* -fno-exceptions: this is inherited from gcc too
* -fasynchronous-unwind-tables: this is inherited from gcc too
* -ggdb: it is useful for debugging a plugin (better backtrace on internal
errors)
* -Wno-narrowing: to suppress warnings from gcc headers (ipa-utils.h)
* -Wno-unused-variable: to suppress warnings from gcc headers (gcc_version
variable, plugin-version.h)
The infrastructure introduces a new Makefile target called gcc-plugins. It
supports all gcc versions from 4.5 to 6.0. The scripts/gcc-plugin.sh script
chooses the proper host compiler (gcc-4.7 can be built by either gcc or g++).
This script also checks the availability of the included headers in
scripts/gcc-plugins/gcc-common.h.
The gcc-common.h header contains frequently included headers for GCC plugins
and it has a compatibility layer for the supported gcc versions.
The gcc-generate-*-pass.h headers automatically generate the registration
structures for GIMPLE, SIMPLE_IPA, IPA and RTL passes.
Note that 'make clean' keeps the *.so files (only the distclean or mrproper
targets clean all) because they are needed for out-of-tree modules.
Based on work created by the PaX Team.
Signed-off-by: Emese Revfy <re.emese@gmail.com>
Acked-by: Kees Cook <keescook@chromium.org>
Signed-off-by: Michal Marek <mmarek@suse.com>
Diffstat (limited to 'scripts/gcc-plugins/gcc-generate-simple_ipa-pass.h')
-rw-r--r-- | scripts/gcc-plugins/gcc-generate-simple_ipa-pass.h | 175 |
1 files changed, 175 insertions, 0 deletions
diff --git a/scripts/gcc-plugins/gcc-generate-simple_ipa-pass.h b/scripts/gcc-plugins/gcc-generate-simple_ipa-pass.h new file mode 100644 index 000000000000..a27e2b36afaa --- /dev/null +++ b/scripts/gcc-plugins/gcc-generate-simple_ipa-pass.h | |||
@@ -0,0 +1,175 @@ | |||
1 | /* | ||
2 | * Generator for SIMPLE_IPA pass related boilerplate code/data | ||
3 | * | ||
4 | * Supports gcc 4.5-6 | ||
5 | * | ||
6 | * Usage: | ||
7 | * | ||
8 | * 1. before inclusion define PASS_NAME | ||
9 | * 2. before inclusion define NO_* for unimplemented callbacks | ||
10 | * NO_GATE | ||
11 | * NO_EXECUTE | ||
12 | * 3. before inclusion define PROPERTIES_* and TODO_FLAGS_* to override | ||
13 | * the default 0 values | ||
14 | * 4. for convenience, all the above will be undefined after inclusion! | ||
15 | * 5. the only exported name is make_PASS_NAME_pass() to register with gcc | ||
16 | */ | ||
17 | |||
18 | #ifndef PASS_NAME | ||
19 | #error at least PASS_NAME must be defined | ||
20 | #else | ||
21 | #define __GCC_PLUGIN_STRINGIFY(n) #n | ||
22 | #define _GCC_PLUGIN_STRINGIFY(n) __GCC_PLUGIN_STRINGIFY(n) | ||
23 | #define _GCC_PLUGIN_CONCAT2(x, y) x ## y | ||
24 | #define _GCC_PLUGIN_CONCAT3(x, y, z) x ## y ## z | ||
25 | |||
26 | #define __PASS_NAME_PASS_DATA(n) _GCC_PLUGIN_CONCAT2(n, _pass_data) | ||
27 | #define _PASS_NAME_PASS_DATA __PASS_NAME_PASS_DATA(PASS_NAME) | ||
28 | |||
29 | #define __PASS_NAME_PASS(n) _GCC_PLUGIN_CONCAT2(n, _pass) | ||
30 | #define _PASS_NAME_PASS __PASS_NAME_PASS(PASS_NAME) | ||
31 | |||
32 | #define _PASS_NAME_NAME _GCC_PLUGIN_STRINGIFY(PASS_NAME) | ||
33 | |||
34 | #define __MAKE_PASS_NAME_PASS(n) _GCC_PLUGIN_CONCAT3(make_, n, _pass) | ||
35 | #define _MAKE_PASS_NAME_PASS __MAKE_PASS_NAME_PASS(PASS_NAME) | ||
36 | |||
37 | #ifdef NO_GATE | ||
38 | #define _GATE NULL | ||
39 | #define _HAS_GATE false | ||
40 | #else | ||
41 | #define __GATE(n) _GCC_PLUGIN_CONCAT2(n, _gate) | ||
42 | #define _GATE __GATE(PASS_NAME) | ||
43 | #define _HAS_GATE true | ||
44 | #endif | ||
45 | |||
46 | #ifdef NO_EXECUTE | ||
47 | #define _EXECUTE NULL | ||
48 | #define _HAS_EXECUTE false | ||
49 | #else | ||
50 | #define __EXECUTE(n) _GCC_PLUGIN_CONCAT2(n, _execute) | ||
51 | #define _EXECUTE __EXECUTE(PASS_NAME) | ||
52 | #define _HAS_EXECUTE true | ||
53 | #endif | ||
54 | |||
55 | #ifndef PROPERTIES_REQUIRED | ||
56 | #define PROPERTIES_REQUIRED 0 | ||
57 | #endif | ||
58 | |||
59 | #ifndef PROPERTIES_PROVIDED | ||
60 | #define PROPERTIES_PROVIDED 0 | ||
61 | #endif | ||
62 | |||
63 | #ifndef PROPERTIES_DESTROYED | ||
64 | #define PROPERTIES_DESTROYED 0 | ||
65 | #endif | ||
66 | |||
67 | #ifndef TODO_FLAGS_START | ||
68 | #define TODO_FLAGS_START 0 | ||
69 | #endif | ||
70 | |||
71 | #ifndef TODO_FLAGS_FINISH | ||
72 | #define TODO_FLAGS_FINISH 0 | ||
73 | #endif | ||
74 | |||
75 | #if BUILDING_GCC_VERSION >= 4009 | ||
76 | namespace { | ||
77 | static const pass_data _PASS_NAME_PASS_DATA = { | ||
78 | #else | ||
79 | static struct simple_ipa_opt_pass _PASS_NAME_PASS = { | ||
80 | .pass = { | ||
81 | #endif | ||
82 | .type = SIMPLE_IPA_PASS, | ||
83 | .name = _PASS_NAME_NAME, | ||
84 | #if BUILDING_GCC_VERSION >= 4008 | ||
85 | .optinfo_flags = OPTGROUP_NONE, | ||
86 | #endif | ||
87 | #if BUILDING_GCC_VERSION >= 5000 | ||
88 | #elif BUILDING_GCC_VERSION == 4009 | ||
89 | .has_gate = _HAS_GATE, | ||
90 | .has_execute = _HAS_EXECUTE, | ||
91 | #else | ||
92 | .gate = _GATE, | ||
93 | .execute = _EXECUTE, | ||
94 | .sub = NULL, | ||
95 | .next = NULL, | ||
96 | .static_pass_number = 0, | ||
97 | #endif | ||
98 | .tv_id = TV_NONE, | ||
99 | .properties_required = PROPERTIES_REQUIRED, | ||
100 | .properties_provided = PROPERTIES_PROVIDED, | ||
101 | .properties_destroyed = PROPERTIES_DESTROYED, | ||
102 | .todo_flags_start = TODO_FLAGS_START, | ||
103 | .todo_flags_finish = TODO_FLAGS_FINISH, | ||
104 | #if BUILDING_GCC_VERSION < 4009 | ||
105 | } | ||
106 | #endif | ||
107 | }; | ||
108 | |||
109 | #if BUILDING_GCC_VERSION >= 4009 | ||
110 | class _PASS_NAME_PASS : public simple_ipa_opt_pass { | ||
111 | public: | ||
112 | _PASS_NAME_PASS() : simple_ipa_opt_pass(_PASS_NAME_PASS_DATA, g) {} | ||
113 | |||
114 | #ifndef NO_GATE | ||
115 | #if BUILDING_GCC_VERSION >= 5000 | ||
116 | virtual bool gate(function *) { return _GATE(); } | ||
117 | #else | ||
118 | virtual bool gate(void) { return _GATE(); } | ||
119 | #endif | ||
120 | #endif | ||
121 | |||
122 | virtual opt_pass *clone() { return new _PASS_NAME_PASS(); } | ||
123 | |||
124 | #ifndef NO_EXECUTE | ||
125 | #if BUILDING_GCC_VERSION >= 5000 | ||
126 | virtual unsigned int execute(function *) { return _EXECUTE(); } | ||
127 | #else | ||
128 | virtual unsigned int execute(void) { return _EXECUTE(); } | ||
129 | #endif | ||
130 | #endif | ||
131 | }; | ||
132 | } | ||
133 | |||
134 | opt_pass *_MAKE_PASS_NAME_PASS(void) | ||
135 | { | ||
136 | return new _PASS_NAME_PASS(); | ||
137 | } | ||
138 | #else | ||
139 | struct opt_pass *_MAKE_PASS_NAME_PASS(void) | ||
140 | { | ||
141 | return &_PASS_NAME_PASS.pass; | ||
142 | } | ||
143 | #endif | ||
144 | |||
145 | /* clean up user provided defines */ | ||
146 | #undef PASS_NAME | ||
147 | #undef NO_GATE | ||
148 | #undef NO_EXECUTE | ||
149 | |||
150 | #undef PROPERTIES_DESTROYED | ||
151 | #undef PROPERTIES_PROVIDED | ||
152 | #undef PROPERTIES_REQUIRED | ||
153 | #undef TODO_FLAGS_FINISH | ||
154 | #undef TODO_FLAGS_START | ||
155 | |||
156 | /* clean up generated defines */ | ||
157 | #undef _EXECUTE | ||
158 | #undef __EXECUTE | ||
159 | #undef _GATE | ||
160 | #undef __GATE | ||
161 | #undef _GCC_PLUGIN_CONCAT2 | ||
162 | #undef _GCC_PLUGIN_CONCAT3 | ||
163 | #undef _GCC_PLUGIN_STRINGIFY | ||
164 | #undef __GCC_PLUGIN_STRINGIFY | ||
165 | #undef _HAS_EXECUTE | ||
166 | #undef _HAS_GATE | ||
167 | #undef _MAKE_PASS_NAME_PASS | ||
168 | #undef __MAKE_PASS_NAME_PASS | ||
169 | #undef _PASS_NAME_NAME | ||
170 | #undef _PASS_NAME_PASS | ||
171 | #undef __PASS_NAME_PASS | ||
172 | #undef _PASS_NAME_PASS_DATA | ||
173 | #undef __PASS_NAME_PASS_DATA | ||
174 | |||
175 | #endif /* PASS_NAME */ | ||