Run DCE after a LoopFlatten test to reduce spurious output [nfc]
[llvm-project.git] / clang / test / CodeGen / pragma-weak.c
blob52328bf9ff1be755459efd4dbd5fde9b8dc14955
1 // RUN: %clang_cc1 -triple x86_64-pc-linux-gnu -emit-llvm %s -o - -verify | FileCheck %s
3 // CHECK: @weakvar = weak{{.*}} global
4 // CHECK: @__weakvar_alias ={{.*}} global
5 // CHECK: @correct_linkage = weak{{.*}} global
8 // CHECK-DAG: @both ={{.*}} alias void (), ptr @__both
9 // CHECK-DAG: @both2 ={{.*}} alias void (), ptr @__both2
10 // CHECK-DAG: @weakvar_alias = weak{{.*}} alias i32, ptr @__weakvar_alias
11 // CHECK-DAG: @foo = weak{{.*}} alias void (), ptr @__foo
12 // CHECK-DAG: @foo2 = weak{{.*}} alias void (), ptr @__foo2
13 // CHECK-DAG: @stutter = weak{{.*}} alias void (), ptr @__stutter
14 // CHECK-DAG: @stutter2 = weak{{.*}} alias void (), ptr @__stutter2
15 // CHECK-DAG: @declfirst = weak{{.*}} alias void (), ptr @__declfirst
16 // CHECK-DAG: @declfirstattr = weak{{.*}} alias void (), ptr @__declfirstattr
17 // CHECK-DAG: @mix2 = weak{{.*}} alias void (), ptr @__mix2
18 // CHECK-DAG: @a1 = weak{{.*}} alias void (), ptr @__a1
19 // CHECK-DAG: @xxx = weak{{.*}} alias void (), ptr @__xxx
20 // CHECK-DAG: @undecfunc_alias1 = weak{{.*}} alias void (), ptr @undecfunc
21 // CHECK-DAG: @undecfunc_alias2 = weak{{.*}} alias void (), ptr @undecfunc
22 // CHECK-DAG: @undecfunc_alias3 = weak{{.*}} alias void (), ptr @undecfunc
23 // CHECK-DAG: @undecfunc_alias4 = weak{{.*}} alias void (), ptr @undecfunc
27 // CHECK-LABEL: define weak{{.*}} void @weakdef()
30 #pragma weak weakvar
31 int weakvar;
33 #pragma weak weakdef
34 void weakdef(void) {}
36 #pragma weak param // expected-warning {{weak identifier 'param' never declared}}
37 #pragma weak correct_linkage
38 void f(int param) {
39 int correct_linkage;
42 #pragma weak weakvar_alias = __weakvar_alias
43 int __weakvar_alias;
45 #pragma weak foo = __foo
46 void __foo(void) {}
47 // CHECK-LABEL: define{{.*}} void @__foo()
50 void __foo2(void) {}
51 #pragma weak foo2 = __foo2
52 // CHECK-LABEL: define{{.*}} void @__foo2()
55 ///// test errors
57 #pragma weak unused // expected-warning {{weak identifier 'unused' never declared}}
58 #pragma weak unused_alias = __unused_alias // expected-warning {{weak identifier '__unused_alias' never declared}}
60 #pragma weak td // expected-warning {{'weak' attribute only applies to variables and functions}}
61 typedef int td;
63 #pragma weak td2 = __td2 // expected-warning {{'weak' attribute only applies to variables and functions}}
64 typedef int __td2;
66 typedef int __td3;
67 #pragma weak td3 = __td3 // expected-warning {{'weak' attribute only applies to variables and functions}}
69 ///// test weird cases
71 // test repeats
73 #pragma weak stutter = __stutter
74 #pragma weak stutter = __stutter
75 void __stutter(void) {}
76 // CHECK-LABEL: define{{.*}} void @__stutter()
78 void __stutter2(void) {}
79 #pragma weak stutter2 = __stutter2
80 #pragma weak stutter2 = __stutter2
81 // CHECK-LABEL: define{{.*}} void @__stutter2()
84 // test decl/pragma weak order
86 void __declfirst(void);
87 #pragma weak declfirst = __declfirst
88 void __declfirst(void) {}
89 // CHECK-LABEL: define{{.*}} void @__declfirst()
91 void __declfirstattr(void) __attribute((noinline));
92 #pragma weak declfirstattr = __declfirstattr
93 void __declfirstattr(void) {}
94 // CHECK-LABEL: define{{.*}} void @__declfirstattr()
96 //// test that other attributes are preserved
98 //// ensure that pragma weak/__attribute((weak)) play nice
100 void mix(void);
101 #pragma weak mix
102 __attribute((weak)) void mix(void) { }
103 // CHECK-LABEL: define weak{{.*}} void @mix()
105 // ensure following __attributes are preserved and that only a single
106 // alias is generated
107 #pragma weak mix2 = __mix2
108 void __mix2(void) __attribute((noinline));
109 void __mix2(void) __attribute((noinline));
110 void __mix2(void) {}
111 // CHECK-LABEL: define{{.*}} void @__mix2()
113 ////////////// test #pragma weak/__attribute combinations
115 // if the SAME ALIAS is already declared then it overrides #pragma weak
116 // resulting in a non-weak alias in this case
117 void both(void) __attribute((alias("__both")));
118 #pragma weak both = __both
119 void __both(void) {}
120 // CHECK-LABEL: define{{.*}} void @__both()
122 // if the TARGET is previously declared then whichever aliasing method
123 // comes first applies and subsequent aliases are discarded.
124 // TODO: warn about this
126 void __both2(void);
127 void both2(void) __attribute((alias("__both2"))); // first, wins
128 #pragma weak both2 = __both2
129 void __both2(void) {}
130 // CHECK-LABEL: define{{.*}} void @__both2()
132 ///////////// ensure that #pragma weak does not alter existing __attributes()
134 void __a1(void) __attribute((noinline));
135 #pragma weak a1 = __a1
136 void __a1(void) {}
137 // CHECK: define{{.*}} void @__a1() [[NI:#[0-9]+]]
139 #pragma weak xxx = __xxx
140 __attribute((pure,noinline,const)) void __xxx(void) { }
141 // CHECK: void @__xxx() [[RN:#[0-9]+]]
143 ///////////// PR28611: Try multiple aliases of same undeclared symbol or alias
144 #pragma weak undecfunc_alias1 = undecfunc
145 #pragma weak undecfunc_alias1 = undecfunc // Try specifying same alias/target pair a second time.
146 #pragma weak undecfunc_alias3 = undecfunc_alias2 // expected-warning {{alias will always resolve to undecfunc}}
147 #pragma weak undecfunc_alias4 = undecfunc_alias2 // expected-warning {{alias will always resolve to undecfunc}}
148 #pragma weak undecfunc_alias2 = undecfunc
149 void undecfunc_alias2(void);
150 void undecfunc(void) { }
152 ///////////// PR10878: Make sure we can call a weak alias
153 void SHA512Pad(void *context) {}
154 #pragma weak SHA384Pad = SHA512Pad
155 void PR10878(void) { SHA384Pad(0); }
156 // CHECK: call void @SHA384Pad(ptr noundef null)
159 // PR14046: Parse #pragma weak in function-local context
160 extern int PR14046e(void);
161 void PR14046f(void) {
162 #pragma weak PR14046e
163 PR14046e();
165 // CHECK: declare extern_weak i32 @PR14046e()
167 // Parse #pragma weak after a label or case statement
168 extern int PR16705a(void);
169 extern int PR16705b(void);
170 extern int PR16705c(void);
171 void PR16705f(int a) {
172 switch(a) {
173 case 1:
174 #pragma weak PR16705a
175 PR16705a();
176 default:
177 #pragma weak PR16705b
178 PR16705b();
180 label:
181 #pragma weak PR16705c
182 PR16705c();
185 // CHECK: declare extern_weak i32 @PR16705a()
186 // CHECK: declare extern_weak i32 @PR16705b()
187 // CHECK: declare extern_weak i32 @PR16705c()
190 ///////////// TODO: stuff that still doesn't work
192 // due to the fact that disparate TopLevelDecls cannot affect each other
193 // (due to clang's Parser and ASTConsumer behavior, and quite reasonable)
194 // #pragma weak must appear before or within the same TopLevelDecl as it
195 // references.
196 void yyy(void){}
197 void zzz(void){}
198 #pragma weak yyy
199 // NOTE: weak doesn't apply, not before or in same TopLevelDec(!)
200 // CHECK-LABEL: define{{.*}} void @yyy()
202 int correct_linkage;
204 // CHECK: attributes [[NI]] = { noinline nounwind{{.*}} }
205 // CHECK: attributes [[RN]] = { noinline nounwind optnone willreturn memory(none){{.*}} }