aboutsummaryrefslogtreecommitdiffstats
path: root/scripts/gcc-plugins/gcc-generate-simple_ipa-pass.h
diff options
context:
space:
mode:
authorEmese Revfy <re.emese@gmail.com>2016-05-23 18:09:38 -0400
committerMichal Marek <mmarek@suse.com>2016-06-07 16:57:10 -0400
commit6b90bd4ba40b38dc13c2782469c1c77e4ed79915 (patch)
tree02d65b38b76e3543d33088ae9149010bae0290b0 /scripts/gcc-plugins/gcc-generate-simple_ipa-pass.h
parent24403874316a7180d367e51d7f7e25d5de1f78dd (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.h175
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
76namespace {
77static const pass_data _PASS_NAME_PASS_DATA = {
78#else
79static 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
110class _PASS_NAME_PASS : public simple_ipa_opt_pass {
111public:
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
134opt_pass *_MAKE_PASS_NAME_PASS(void)
135{
136 return new _PASS_NAME_PASS();
137}
138#else
139struct 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 */