1*67e74705SXin Li // RUN: %clang_cc1 -fmath-errno -emit-llvm -o - %s -triple i386-unknown-unknown | FileCheck -check-prefix CHECK-YES %s
2*67e74705SXin Li // RUN: %clang_cc1 -emit-llvm -o - %s -triple i386-unknown-unknown | FileCheck -check-prefix CHECK-NO %s
3*67e74705SXin Li // RUN: %clang_cc1 -menable-unsafe-fp-math -emit-llvm -o - %s -triple i386-unknown-unknown | FileCheck -check-prefix CHECK-FAST %s
4*67e74705SXin Li
5*67e74705SXin Li // CHECK-YES-LABEL: define void @test_sqrt
6*67e74705SXin Li // CHECK-NO-LABEL: define void @test_sqrt
7*67e74705SXin Li // CHECK-FAST-LABEL: define void @test_sqrt
test_sqrt(float a0,double a1,long double a2)8*67e74705SXin Li void test_sqrt(float a0, double a1, long double a2) {
9*67e74705SXin Li // Following llvm-gcc's lead, we never emit these as intrinsics;
10*67e74705SXin Li // no-math-errno isn't good enough. We could probably use intrinsics
11*67e74705SXin Li // with appropriate guards if it proves worthwhile.
12*67e74705SXin Li
13*67e74705SXin Li // CHECK-YES: call float @sqrtf
14*67e74705SXin Li // CHECK-NO: call float @sqrtf
15*67e74705SXin Li float l0 = sqrtf(a0);
16*67e74705SXin Li
17*67e74705SXin Li // CHECK-YES: call double @sqrt
18*67e74705SXin Li // CHECK-NO: call double @sqrt
19*67e74705SXin Li double l1 = sqrt(a1);
20*67e74705SXin Li
21*67e74705SXin Li // CHECK-YES: call x86_fp80 @sqrtl
22*67e74705SXin Li // CHECK-NO: call x86_fp80 @sqrtl
23*67e74705SXin Li long double l2 = sqrtl(a2);
24*67e74705SXin Li }
25*67e74705SXin Li
26*67e74705SXin Li // CHECK-YES: declare float @sqrtf(float)
27*67e74705SXin Li // CHECK-YES: declare double @sqrt(double)
28*67e74705SXin Li // CHECK-YES: declare x86_fp80 @sqrtl(x86_fp80)
29*67e74705SXin Li // CHECK-NO: declare float @sqrtf(float) [[NUW_RN:#[0-9]+]]
30*67e74705SXin Li // CHECK-NO: declare double @sqrt(double) [[NUW_RN]]
31*67e74705SXin Li // CHECK-NO: declare x86_fp80 @sqrtl(x86_fp80) [[NUW_RN]]
32*67e74705SXin Li // CHECK-FAST: declare float @llvm.sqrt.f32(float)
33*67e74705SXin Li // CHECK-FAST: declare double @llvm.sqrt.f64(double)
34*67e74705SXin Li // CHECK-FAST: declare x86_fp80 @llvm.sqrt.f80(x86_fp80)
35*67e74705SXin Li
36*67e74705SXin Li // CHECK-YES-LABEL: define void @test_pow
37*67e74705SXin Li // CHECK-NO-LABEL: define void @test_pow
test_pow(float a0,double a1,long double a2)38*67e74705SXin Li void test_pow(float a0, double a1, long double a2) {
39*67e74705SXin Li // CHECK-YES: call float @powf
40*67e74705SXin Li // CHECK-NO: call float @llvm.pow.f32
41*67e74705SXin Li float l0 = powf(a0, a0);
42*67e74705SXin Li
43*67e74705SXin Li // CHECK-YES: call double @pow
44*67e74705SXin Li // CHECK-NO: call double @llvm.pow.f64
45*67e74705SXin Li double l1 = pow(a1, a1);
46*67e74705SXin Li
47*67e74705SXin Li // CHECK-YES: call x86_fp80 @powl
48*67e74705SXin Li // CHECK-NO: call x86_fp80 @llvm.pow.f80
49*67e74705SXin Li long double l2 = powl(a2, a2);
50*67e74705SXin Li }
51*67e74705SXin Li
52*67e74705SXin Li // CHECK-YES: declare float @powf(float, float)
53*67e74705SXin Li // CHECK-YES: declare double @pow(double, double)
54*67e74705SXin Li // CHECK-YES: declare x86_fp80 @powl(x86_fp80, x86_fp80)
55*67e74705SXin Li // CHECK-NO: declare float @llvm.pow.f32(float, float) [[NUW_RNI:#[0-9]+]]
56*67e74705SXin Li // CHECK-NO: declare double @llvm.pow.f64(double, double) [[NUW_RNI]]
57*67e74705SXin Li // CHECK-NO: declare x86_fp80 @llvm.pow.f80(x86_fp80, x86_fp80) [[NUW_RNI]]
58*67e74705SXin Li
59*67e74705SXin Li // CHECK-YES-LABEL: define void @test_fma
60*67e74705SXin Li // CHECK-NO-LABEL: define void @test_fma
test_fma(float a0,double a1,long double a2)61*67e74705SXin Li void test_fma(float a0, double a1, long double a2) {
62*67e74705SXin Li // CHECK-YES: call float @llvm.fma.f32
63*67e74705SXin Li // CHECK-NO: call float @llvm.fma.f32
64*67e74705SXin Li float l0 = fmaf(a0, a0, a0);
65*67e74705SXin Li
66*67e74705SXin Li // CHECK-YES: call double @llvm.fma.f64
67*67e74705SXin Li // CHECK-NO: call double @llvm.fma.f64
68*67e74705SXin Li double l1 = fma(a1, a1, a1);
69*67e74705SXin Li
70*67e74705SXin Li // CHECK-YES: call x86_fp80 @llvm.fma.f80
71*67e74705SXin Li // CHECK-NO: call x86_fp80 @llvm.fma.f80
72*67e74705SXin Li long double l2 = fmal(a2, a2, a2);
73*67e74705SXin Li }
74*67e74705SXin Li
75*67e74705SXin Li // CHECK-YES: declare float @llvm.fma.f32(float, float, float) [[NUW_RN:#[0-9]+]]
76*67e74705SXin Li // CHECK-YES: declare double @llvm.fma.f64(double, double, double) [[NUW_RN]]
77*67e74705SXin Li // CHECK-YES: declare x86_fp80 @llvm.fma.f80(x86_fp80, x86_fp80, x86_fp80) [[NUW_RN]]
78*67e74705SXin Li // CHECK-NO: declare float @llvm.fma.f32(float, float, float) [[NUW_RN2:#[0-9]+]]
79*67e74705SXin Li // CHECK-NO: declare double @llvm.fma.f64(double, double, double) [[NUW_RN2]]
80*67e74705SXin Li // CHECK-NO: declare x86_fp80 @llvm.fma.f80(x86_fp80, x86_fp80, x86_fp80) [[NUW_RN2]]
81*67e74705SXin Li
82*67e74705SXin Li // Just checking to make sure these library functions are marked readnone
test_builtins(double d,float f,long double ld)83*67e74705SXin Li void test_builtins(double d, float f, long double ld) {
84*67e74705SXin Li // CHECK-NO: @test_builtins
85*67e74705SXin Li // CHECK-YES: @test_builtins
86*67e74705SXin Li double atan_ = atan(d);
87*67e74705SXin Li long double atanl_ = atanl(ld);
88*67e74705SXin Li float atanf_ = atanf(f);
89*67e74705SXin Li // CHECK-NO: declare double @atan(double) [[NUW_RN]]
90*67e74705SXin Li // CHECK-NO: declare x86_fp80 @atanl(x86_fp80) [[NUW_RN]]
91*67e74705SXin Li // CHECK-NO: declare float @atanf(float) [[NUW_RN]]
92*67e74705SXin Li // CHECK-YES-NOT: declare double @atan(double) [[NUW_RN]]
93*67e74705SXin Li // CHECK-YES-NOT: declare x86_fp80 @atanl(x86_fp80) [[NUW_RN]]
94*67e74705SXin Li // CHECK-YES-NOT: declare float @atanf(float) [[NUW_RN]]
95*67e74705SXin Li
96*67e74705SXin Li double atan2_ = atan2(d, 2);
97*67e74705SXin Li long double atan2l_ = atan2l(ld, ld);
98*67e74705SXin Li float atan2f_ = atan2f(f, f);
99*67e74705SXin Li // CHECK-NO: declare double @atan2(double, double) [[NUW_RN]]
100*67e74705SXin Li // CHECK-NO: declare x86_fp80 @atan2l(x86_fp80, x86_fp80) [[NUW_RN]]
101*67e74705SXin Li // CHECK-NO: declare float @atan2f(float, float) [[NUW_RN]]
102*67e74705SXin Li // CHECK-YES-NOT: declare double @atan2(double, double) [[NUW_RN]]
103*67e74705SXin Li // CHECK-YES-NOT: declare x86_fp80 @atan2l(x86_fp80, x86_fp80) [[NUW_RN]]
104*67e74705SXin Li // CHECK-YES-NOT: declare float @atan2f(float, float) [[NUW_RN]]
105*67e74705SXin Li
106*67e74705SXin Li double exp_ = exp(d);
107*67e74705SXin Li long double expl_ = expl(ld);
108*67e74705SXin Li float expf_ = expf(f);
109*67e74705SXin Li // CHECK-NO: declare double @exp(double) [[NUW_RN]]
110*67e74705SXin Li // CHECK-NO: declare x86_fp80 @expl(x86_fp80) [[NUW_RN]]
111*67e74705SXin Li // CHECK-NO: declare float @expf(float) [[NUW_RN]]
112*67e74705SXin Li // CHECK-YES-NOT: declare double @exp(double) [[NUW_RN]]
113*67e74705SXin Li // CHECK-YES-NOT: declare x86_fp80 @expl(x86_fp80) [[NUW_RN]]
114*67e74705SXin Li // CHECK-YES-NOT: declare float @expf(float) [[NUW_RN]]
115*67e74705SXin Li
116*67e74705SXin Li double log_ = log(d);
117*67e74705SXin Li long double logl_ = logl(ld);
118*67e74705SXin Li float logf_ = logf(f);
119*67e74705SXin Li // CHECK-NO: declare double @log(double) [[NUW_RN]]
120*67e74705SXin Li // CHECK-NO: declare x86_fp80 @logl(x86_fp80) [[NUW_RN]]
121*67e74705SXin Li // CHECK-NO: declare float @logf(float) [[NUW_RN]]
122*67e74705SXin Li // CHECK-YES-NOT: declare double @log(double) [[NUW_RN]]
123*67e74705SXin Li // CHECK-YES-NOT: declare x86_fp80 @logl(x86_fp80) [[NUW_RN]]
124*67e74705SXin Li // CHECK-YES-NOT: declare float @logf(float) [[NUW_RN]]
125*67e74705SXin Li }
126*67e74705SXin Li
127*67e74705SXin Li // CHECK-YES: attributes [[NUW_RN]] = { nounwind readnone }
128*67e74705SXin Li
129*67e74705SXin Li // CHECK-NO: attributes [[NUW_RN]] = { nounwind readnone{{.*}} }
130*67e74705SXin Li // CHECK-NO: attributes [[NUW_RNI]] = { nounwind readnone }
131