forked from OSchip/llvm-project
235 lines
8.7 KiB
C
235 lines
8.7 KiB
C
// RUN: %clang_cc1 -triple x86_64-linux-gnu -emit-llvm %s -o - | FileCheck %s --check-prefix=LINUX
|
|
// RUN: %clang_cc1 -triple x86_64-windows-pc -emit-llvm %s -o - | FileCheck %s --check-prefix=WINDOWS
|
|
|
|
int __attribute__((target("sse4.2"))) foo(void) { return 0; }
|
|
int __attribute__((target("arch=sandybridge"))) foo(void);
|
|
int __attribute__((target("arch=ivybridge"))) foo(void) {return 1;}
|
|
int __attribute__((target("default"))) foo(void) { return 2; }
|
|
|
|
int bar() {
|
|
return foo();
|
|
}
|
|
|
|
inline int __attribute__((target("sse4.2"))) foo_inline(void) { return 0; }
|
|
inline int __attribute__((target("arch=sandybridge"))) foo_inline(void);
|
|
inline int __attribute__((target("arch=ivybridge"))) foo_inline(void) {return 1;}
|
|
inline int __attribute__((target("default"))) foo_inline(void) { return 2; }
|
|
|
|
int bar2() {
|
|
return foo_inline();
|
|
}
|
|
|
|
inline __attribute__((target("default"))) void foo_decls(void);
|
|
inline __attribute__((target("sse4.2"))) void foo_decls(void);
|
|
void bar3() {
|
|
foo_decls();
|
|
}
|
|
inline __attribute__((target("default"))) void foo_decls(void) {}
|
|
inline __attribute__((target("sse4.2"))) void foo_decls(void) {}
|
|
|
|
inline __attribute__((target("default"))) void foo_multi(int i, double d) {}
|
|
inline __attribute__((target("avx,sse4.2"))) void foo_multi(int i, double d) {}
|
|
inline __attribute__((target("sse4.2,fma4"))) void foo_multi(int i, double d) {}
|
|
inline __attribute__((target("arch=ivybridge,fma4,sse4.2"))) void foo_multi(int i, double d) {}
|
|
void bar4() {
|
|
foo_multi(1, 5.0);
|
|
}
|
|
|
|
int fwd_decl_default(void);
|
|
int __attribute__((target("default"))) fwd_decl_default(void) { return 2; }
|
|
|
|
int fwd_decl_avx(void);
|
|
int __attribute__((target("avx"))) fwd_decl_avx(void) { return 2; }
|
|
int __attribute__((target("default"))) fwd_decl_avx(void) { return 2; }
|
|
|
|
void bar5() {
|
|
fwd_decl_default();
|
|
fwd_decl_avx();
|
|
}
|
|
|
|
// LINUX: @foo.ifunc = ifunc i32 (), i32 ()* ()* @foo.resolver
|
|
// LINUX: @foo_inline.ifunc = ifunc i32 (), i32 ()* ()* @foo_inline.resolver
|
|
// LINUX: @foo_decls.ifunc = ifunc void (), void ()* ()* @foo_decls.resolver
|
|
// LINUX: @foo_multi.ifunc = ifunc void (i32, double), void (i32, double)* ()* @foo_multi.resolver
|
|
// LINUX: @fwd_decl_default.ifunc = ifunc i32 (), i32 ()* ()* @fwd_decl_default.resolver
|
|
// LINUX: @fwd_decl_avx.ifunc = ifunc i32 (), i32 ()* ()* @fwd_decl_avx.resolver
|
|
|
|
// LINUX: define i32 @foo.sse4.2()
|
|
// LINUX: ret i32 0
|
|
// LINUX: define i32 @foo.arch_ivybridge()
|
|
// LINUX: ret i32 1
|
|
// LINUX: define i32 @foo()
|
|
// LINUX: ret i32 2
|
|
// LINUX: define i32 @bar()
|
|
// LINUX: call i32 @foo.ifunc()
|
|
|
|
// WINDOWS: define dso_local i32 @foo.sse4.2()
|
|
// WINDOWS: ret i32 0
|
|
// WINDOWS: define dso_local i32 @foo.arch_ivybridge()
|
|
// WINDOWS: ret i32 1
|
|
// WINDOWS: define dso_local i32 @foo()
|
|
// WINDOWS: ret i32 2
|
|
// WINDOWS: define dso_local i32 @bar()
|
|
// WINDOWS: call i32 @foo.resolver()
|
|
|
|
// LINUX: define i32 ()* @foo.resolver() comdat
|
|
// LINUX: call void @__cpu_indicator_init()
|
|
// LINUX: ret i32 ()* @foo.arch_sandybridge
|
|
// LINUX: ret i32 ()* @foo.arch_ivybridge
|
|
// LINUX: ret i32 ()* @foo.sse4.2
|
|
// LINUX: ret i32 ()* @foo
|
|
|
|
// WINDOWS: define dso_local i32 @foo.resolver() comdat
|
|
// WINDOWS: call void @__cpu_indicator_init()
|
|
// WINDOWS: call i32 @foo.arch_sandybridge
|
|
// WINDOWS: call i32 @foo.arch_ivybridge
|
|
// WINDOWS: call i32 @foo.sse4.2
|
|
// WINDOWS: call i32 @foo
|
|
|
|
// LINUX: define i32 @bar2()
|
|
// LINUX: call i32 @foo_inline.ifunc()
|
|
|
|
// WINDOWS: define dso_local i32 @bar2()
|
|
// WINDOWS: call i32 @foo_inline.resolver()
|
|
|
|
// LINUX: define i32 ()* @foo_inline.resolver() comdat
|
|
// LINUX: call void @__cpu_indicator_init()
|
|
// LINUX: ret i32 ()* @foo_inline.arch_sandybridge
|
|
// LINUX: ret i32 ()* @foo_inline.arch_ivybridge
|
|
// LINUX: ret i32 ()* @foo_inline.sse4.2
|
|
// LINUX: ret i32 ()* @foo_inline
|
|
|
|
// WINDOWS: define dso_local i32 @foo_inline.resolver() comdat
|
|
// WINDOWS: call void @__cpu_indicator_init()
|
|
// WINDOWS: call i32 @foo_inline.arch_sandybridge
|
|
// WINDOWS: call i32 @foo_inline.arch_ivybridge
|
|
// WINDOWS: call i32 @foo_inline.sse4.2
|
|
// WINDOWS: call i32 @foo_inline
|
|
|
|
// LINUX: define void @bar3()
|
|
// LINUX: call void @foo_decls.ifunc()
|
|
|
|
// WINDOWS: define dso_local void @bar3()
|
|
// WINDOWS: call void @foo_decls.resolver()
|
|
|
|
// LINUX: define void ()* @foo_decls.resolver() comdat
|
|
// LINUX: ret void ()* @foo_decls.sse4.2
|
|
// LINUX: ret void ()* @foo_decls
|
|
|
|
// WINDOWS: define dso_local void @foo_decls.resolver() comdat
|
|
// WINDOWS: call void @foo_decls.sse4.2
|
|
// WINDOWS: call void @foo_decls
|
|
|
|
// LINUX: define void @bar4()
|
|
// LINUX: call void @foo_multi.ifunc(i32 1, double 5.{{[0+e]*}})
|
|
|
|
// WINDOWS: define dso_local void @bar4()
|
|
// WINDOWS: call void @foo_multi.resolver(i32 1, double 5.{{[0+e]*}})
|
|
|
|
// LINUX: define void (i32, double)* @foo_multi.resolver() comdat
|
|
// LINUX: and i32 %{{.*}}, 4352
|
|
// LINUX: icmp eq i32 %{{.*}}, 4352
|
|
// LINUX: ret void (i32, double)* @foo_multi.fma4_sse4.2
|
|
// LINUX: icmp eq i32 %{{.*}}, 12
|
|
// LINUX: and i32 %{{.*}}, 4352
|
|
// LINUX: icmp eq i32 %{{.*}}, 4352
|
|
// LINUX: ret void (i32, double)* @foo_multi.arch_ivybridge_fma4_sse4.2
|
|
// LINUX: and i32 %{{.*}}, 768
|
|
// LINUX: icmp eq i32 %{{.*}}, 768
|
|
// LINUX: ret void (i32, double)* @foo_multi.avx_sse4.2
|
|
// LINUX: ret void (i32, double)* @foo_multi
|
|
|
|
// WINDOWS: define dso_local void @foo_multi.resolver(i32 %0, double %1) comdat
|
|
// WINDOWS: and i32 %{{.*}}, 4352
|
|
// WINDOWS: icmp eq i32 %{{.*}}, 4352
|
|
// WINDOWS: call void @foo_multi.fma4_sse4.2(i32 %0, double %1)
|
|
// WINDOWS-NEXT: ret void
|
|
// WINDOWS: icmp eq i32 %{{.*}}, 12
|
|
// WINDOWS: and i32 %{{.*}}, 4352
|
|
// WINDOWS: icmp eq i32 %{{.*}}, 4352
|
|
// WINDOWS: call void @foo_multi.arch_ivybridge_fma4_sse4.2(i32 %0, double %1)
|
|
// WINDOWS-NEXT: ret void
|
|
// WINDOWS: and i32 %{{.*}}, 768
|
|
// WINDOWS: icmp eq i32 %{{.*}}, 768
|
|
// WINDOWS: call void @foo_multi.avx_sse4.2(i32 %0, double %1)
|
|
// WINDOWS-NEXT: ret void
|
|
// WINDOWS: call void @foo_multi(i32 %0, double %1)
|
|
// WINDOWS-NEXT: ret void
|
|
|
|
// LINUX: define i32 @fwd_decl_default()
|
|
// LINUX: ret i32 2
|
|
// LINUX: define i32 @fwd_decl_avx.avx()
|
|
// LINUX: ret i32 2
|
|
// LINUX: define i32 @fwd_decl_avx()
|
|
// LINUX: ret i32 2
|
|
|
|
// WINDOWS: define dso_local i32 @fwd_decl_default()
|
|
// WINDOWS: ret i32 2
|
|
// WINDOWS: define dso_local i32 @fwd_decl_avx.avx()
|
|
// WINDOWS: ret i32 2
|
|
// WINDOWS: define dso_local i32 @fwd_decl_avx()
|
|
// WINDOWS: ret i32 2
|
|
|
|
// LINUX: define void @bar5()
|
|
// LINUX: call i32 @fwd_decl_default.ifunc()
|
|
// LINUX: call i32 @fwd_decl_avx.ifunc()
|
|
|
|
// WINDOWS: define dso_local void @bar5()
|
|
// WINDOWS: call i32 @fwd_decl_default.resolver()
|
|
// WINDOWS: call i32 @fwd_decl_avx.resolver()
|
|
|
|
// LINUX: define i32 ()* @fwd_decl_default.resolver() comdat
|
|
// LINUX: call void @__cpu_indicator_init()
|
|
// LINUX: ret i32 ()* @fwd_decl_default
|
|
// LINUX: define i32 ()* @fwd_decl_avx.resolver() comdat
|
|
// LINUX: call void @__cpu_indicator_init()
|
|
// LINUX: ret i32 ()* @fwd_decl_avx.avx
|
|
// LINUX: ret i32 ()* @fwd_decl_avx
|
|
|
|
// WINDOWS: define dso_local i32 @fwd_decl_default.resolver() comdat
|
|
// WINDOWS: call void @__cpu_indicator_init()
|
|
// WINDOWS: call i32 @fwd_decl_default
|
|
// WINDOWS: define dso_local i32 @fwd_decl_avx.resolver() comdat
|
|
// WINDOWS: call void @__cpu_indicator_init()
|
|
// WINDOWS: call i32 @fwd_decl_avx.avx
|
|
// WINDOWS: call i32 @fwd_decl_avx
|
|
|
|
// LINUX: declare i32 @foo.arch_sandybridge()
|
|
// WINDOWS: declare dso_local i32 @foo.arch_sandybridge()
|
|
|
|
// LINUX: define linkonce i32 @foo_inline.sse4.2()
|
|
// LINUX: ret i32 0
|
|
|
|
// WINDOWS: define linkonce_odr dso_local i32 @foo_inline.sse4.2()
|
|
// WINDOWS: ret i32 0
|
|
|
|
// LINUX: declare i32 @foo_inline.arch_sandybridge()
|
|
|
|
// WINDOWS: declare dso_local i32 @foo_inline.arch_sandybridge()
|
|
|
|
// LINUX: define linkonce i32 @foo_inline.arch_ivybridge()
|
|
// LINUX: ret i32 1
|
|
// LINUX: define linkonce i32 @foo_inline()
|
|
// LINUX: ret i32 2
|
|
|
|
// WINDOWS: define linkonce_odr dso_local i32 @foo_inline.arch_ivybridge()
|
|
// WINDOWS: ret i32 1
|
|
// WINDOWS: define linkonce_odr dso_local i32 @foo_inline()
|
|
// WINDOWS: ret i32 2
|
|
|
|
// LINUX: define linkonce void @foo_decls()
|
|
// LINUX: define linkonce void @foo_decls.sse4.2()
|
|
|
|
// WINDOWS: define linkonce_odr dso_local void @foo_decls()
|
|
// WINDOWS: define linkonce_odr dso_local void @foo_decls.sse4.2()
|
|
|
|
// LINUX: define linkonce void @foo_multi(i32 %{{[^,]+}}, double %{{[^\)]+}})
|
|
// LINUX: define linkonce void @foo_multi.avx_sse4.2(i32 %{{[^,]+}}, double %{{[^\)]+}})
|
|
// LINUX: define linkonce void @foo_multi.fma4_sse4.2(i32 %{{[^,]+}}, double %{{[^\)]+}})
|
|
// LINUX: define linkonce void @foo_multi.arch_ivybridge_fma4_sse4.2(i32 %{{[^,]+}}, double %{{[^\)]+}})
|
|
|
|
// WINDOWS: define linkonce_odr dso_local void @foo_multi(i32 %{{[^,]+}}, double %{{[^\)]+}})
|
|
// WINDOWS: define linkonce_odr dso_local void @foo_multi.avx_sse4.2(i32 %{{[^,]+}}, double %{{[^\)]+}})
|
|
// WINDOWS: define linkonce_odr dso_local void @foo_multi.fma4_sse4.2(i32 %{{[^,]+}}, double %{{[^\)]+}})
|
|
// WINDOWS: define linkonce_odr dso_local void @foo_multi.arch_ivybridge_fma4_sse4.2(i32 %{{[^,]+}}, double %{{[^\)]+}})
|