2013-08-12 20:51:05 +08:00
|
|
|
// RUN: %clang_cc1 -fmath-errno -emit-llvm -o - %s -triple i386-unknown-unknown | FileCheck -check-prefix CHECK-YES %s
|
|
|
|
// RUN: %clang_cc1 -emit-llvm -o - %s -triple i386-unknown-unknown | FileCheck -check-prefix CHECK-NO %s
|
2013-09-13 07:57:55 +08:00
|
|
|
// RUN: %clang_cc1 -menable-unsafe-fp-math -emit-llvm -o - %s -triple i386-unknown-unknown | FileCheck -check-prefix CHECK-FAST %s
|
2009-02-17 06:43:43 +08:00
|
|
|
|
2020-02-04 02:09:39 +08:00
|
|
|
// CHECK-YES-LABEL: define void @test_sqrt
|
|
|
|
// CHECK-NO-LABEL: define void @test_sqrt
|
|
|
|
// CHECK-FAST-LABEL: define void @test_sqrt
|
2009-02-17 06:43:43 +08:00
|
|
|
void test_sqrt(float a0, double a1, long double a2) {
|
2010-04-07 16:20:20 +08:00
|
|
|
// CHECK-YES: call float @sqrtf
|
2017-11-01 04:19:39 +08:00
|
|
|
// CHECK-NO: call float @llvm.sqrt.f32(float
|
2020-06-02 09:02:02 +08:00
|
|
|
// CHECK-FAST: call reassoc nsz arcp afn float @llvm.sqrt.f32(float
|
2009-02-17 06:43:43 +08:00
|
|
|
float l0 = sqrtf(a0);
|
2010-04-07 16:20:20 +08:00
|
|
|
|
|
|
|
// CHECK-YES: call double @sqrt
|
2017-11-01 04:19:39 +08:00
|
|
|
// CHECK-NO: call double @llvm.sqrt.f64(double
|
2020-06-02 09:02:02 +08:00
|
|
|
// CHECK-FAST: call reassoc nsz arcp afn double @llvm.sqrt.f64(double
|
2009-02-17 06:43:43 +08:00
|
|
|
double l1 = sqrt(a1);
|
2010-04-07 16:20:20 +08:00
|
|
|
|
|
|
|
// CHECK-YES: call x86_fp80 @sqrtl
|
2017-11-01 04:19:39 +08:00
|
|
|
// CHECK-NO: call x86_fp80 @llvm.sqrt.f80(x86_fp80
|
2020-06-02 09:02:02 +08:00
|
|
|
// CHECK-FAST: call reassoc nsz arcp afn x86_fp80 @llvm.sqrt.f80(x86_fp80
|
2009-02-17 06:43:43 +08:00
|
|
|
long double l2 = sqrtl(a2);
|
|
|
|
}
|
|
|
|
|
2010-04-07 16:20:20 +08:00
|
|
|
// CHECK-YES: declare float @sqrtf(float)
|
|
|
|
// CHECK-YES: declare double @sqrt(double)
|
|
|
|
// CHECK-YES: declare x86_fp80 @sqrtl(x86_fp80)
|
2017-11-01 04:19:39 +08:00
|
|
|
// CHECK-NO: declare float @llvm.sqrt.f32(float)
|
|
|
|
// CHECK-NO: declare double @llvm.sqrt.f64(double)
|
|
|
|
// CHECK-NO: declare x86_fp80 @llvm.sqrt.f80(x86_fp80)
|
2013-09-13 07:57:55 +08:00
|
|
|
// CHECK-FAST: declare float @llvm.sqrt.f32(float)
|
|
|
|
// CHECK-FAST: declare double @llvm.sqrt.f64(double)
|
|
|
|
// CHECK-FAST: declare x86_fp80 @llvm.sqrt.f80(x86_fp80)
|
2010-04-07 16:20:20 +08:00
|
|
|
|
2020-02-04 02:09:39 +08:00
|
|
|
// CHECK-YES-LABEL: define void @test_pow
|
|
|
|
// CHECK-NO-LABEL: define void @test_pow
|
2009-02-17 06:43:43 +08:00
|
|
|
void test_pow(float a0, double a1, long double a2) {
|
2010-04-07 16:20:20 +08:00
|
|
|
// CHECK-YES: call float @powf
|
|
|
|
// CHECK-NO: call float @llvm.pow.f32
|
2009-02-17 06:43:43 +08:00
|
|
|
float l0 = powf(a0, a0);
|
2010-04-07 16:20:20 +08:00
|
|
|
|
|
|
|
// CHECK-YES: call double @pow
|
|
|
|
// CHECK-NO: call double @llvm.pow.f64
|
2009-02-17 06:43:43 +08:00
|
|
|
double l1 = pow(a1, a1);
|
2010-04-07 16:20:20 +08:00
|
|
|
|
|
|
|
// CHECK-YES: call x86_fp80 @powl
|
|
|
|
// CHECK-NO: call x86_fp80 @llvm.pow.f80
|
2009-02-17 06:43:43 +08:00
|
|
|
long double l2 = powl(a2, a2);
|
|
|
|
}
|
2010-04-07 16:20:20 +08:00
|
|
|
|
|
|
|
// CHECK-YES: declare float @powf(float, float)
|
|
|
|
// CHECK-YES: declare double @pow(double, double)
|
|
|
|
// CHECK-YES: declare x86_fp80 @powl(x86_fp80, x86_fp80)
|
2014-03-06 09:37:10 +08:00
|
|
|
// CHECK-NO: declare float @llvm.pow.f32(float, float) [[NUW_RNI:#[0-9]+]]
|
|
|
|
// CHECK-NO: declare double @llvm.pow.f64(double, double) [[NUW_RNI]]
|
|
|
|
// CHECK-NO: declare x86_fp80 @llvm.pow.f80(x86_fp80, x86_fp80) [[NUW_RNI]]
|
2011-07-09 05:39:34 +08:00
|
|
|
|
2020-02-04 02:09:39 +08:00
|
|
|
// CHECK-YES-LABEL: define void @test_fma
|
|
|
|
// CHECK-NO-LABEL: define void @test_fma
|
2011-07-09 05:39:34 +08:00
|
|
|
void test_fma(float a0, double a1, long double a2) {
|
2017-11-14 06:11:49 +08:00
|
|
|
// CHECK-YES: call float @fmaf
|
2011-07-09 05:39:34 +08:00
|
|
|
// CHECK-NO: call float @llvm.fma.f32
|
|
|
|
float l0 = fmaf(a0, a0, a0);
|
|
|
|
|
2017-11-14 06:11:49 +08:00
|
|
|
// CHECK-YES: call double @fma
|
2011-07-09 05:39:34 +08:00
|
|
|
// CHECK-NO: call double @llvm.fma.f64
|
|
|
|
double l1 = fma(a1, a1, a1);
|
|
|
|
|
2017-11-14 06:11:49 +08:00
|
|
|
// CHECK-YES: call x86_fp80 @fmal
|
2011-07-09 05:39:34 +08:00
|
|
|
// CHECK-NO: call x86_fp80 @llvm.fma.f80
|
|
|
|
long double l2 = fmal(a2, a2, a2);
|
|
|
|
}
|
|
|
|
|
2017-11-14 06:11:49 +08:00
|
|
|
// CHECK-YES: declare float @fmaf(float, float, float)
|
|
|
|
// CHECK-YES: declare double @fma(double, double, double)
|
|
|
|
// CHECK-YES: declare x86_fp80 @fmal(x86_fp80, x86_fp80, x86_fp80)
|
2013-02-27 07:08:48 +08:00
|
|
|
// CHECK-NO: declare float @llvm.fma.f32(float, float, float) [[NUW_RN2:#[0-9]+]]
|
|
|
|
// CHECK-NO: declare double @llvm.fma.f64(double, double, double) [[NUW_RN2]]
|
|
|
|
// CHECK-NO: declare x86_fp80 @llvm.fma.f80(x86_fp80, x86_fp80, x86_fp80) [[NUW_RN2]]
|
2012-04-25 06:40:01 +08:00
|
|
|
|
|
|
|
// Just checking to make sure these library functions are marked readnone
|
|
|
|
void test_builtins(double d, float f, long double ld) {
|
2014-02-16 15:29:41 +08:00
|
|
|
// CHECK-NO: @test_builtins
|
|
|
|
// CHECK-YES: @test_builtins
|
2012-04-25 06:40:01 +08:00
|
|
|
double atan_ = atan(d);
|
|
|
|
long double atanl_ = atanl(ld);
|
|
|
|
float atanf_ = atanf(f);
|
2017-11-01 04:19:39 +08:00
|
|
|
// CHECK-NO: declare double @atan(double) [[NUW_RN:#[0-9]+]]
|
2013-02-27 08:06:04 +08:00
|
|
|
// CHECK-NO: declare x86_fp80 @atanl(x86_fp80) [[NUW_RN]]
|
|
|
|
// CHECK-NO: declare float @atanf(float) [[NUW_RN]]
|
2013-02-27 07:08:48 +08:00
|
|
|
// CHECK-YES-NOT: declare double @atan(double) [[NUW_RN]]
|
|
|
|
// CHECK-YES-NOT: declare x86_fp80 @atanl(x86_fp80) [[NUW_RN]]
|
|
|
|
// CHECK-YES-NOT: declare float @atanf(float) [[NUW_RN]]
|
2012-04-25 06:40:01 +08:00
|
|
|
|
|
|
|
double atan2_ = atan2(d, 2);
|
|
|
|
long double atan2l_ = atan2l(ld, ld);
|
|
|
|
float atan2f_ = atan2f(f, f);
|
2013-02-27 08:06:04 +08:00
|
|
|
// CHECK-NO: declare double @atan2(double, double) [[NUW_RN]]
|
|
|
|
// CHECK-NO: declare x86_fp80 @atan2l(x86_fp80, x86_fp80) [[NUW_RN]]
|
|
|
|
// CHECK-NO: declare float @atan2f(float, float) [[NUW_RN]]
|
2013-02-27 07:08:48 +08:00
|
|
|
// CHECK-YES-NOT: declare double @atan2(double, double) [[NUW_RN]]
|
|
|
|
// CHECK-YES-NOT: declare x86_fp80 @atan2l(x86_fp80, x86_fp80) [[NUW_RN]]
|
|
|
|
// CHECK-YES-NOT: declare float @atan2f(float, float) [[NUW_RN]]
|
2012-04-25 06:40:01 +08:00
|
|
|
|
|
|
|
double exp_ = exp(d);
|
|
|
|
long double expl_ = expl(ld);
|
|
|
|
float expf_ = expf(f);
|
[CodeGen] convert math libcalls/builtins to equivalent LLVM intrinsics
There are 20 LLVM math intrinsics that correspond to mathlib calls according to the LangRef:
http://llvm.org/docs/LangRef.html#standard-c-library-intrinsics
We were only converting 3 mathlib calls (sqrt, fma, pow) and 12 builtin calls (ceil, copysign,
fabs, floor, fma, fmax, fmin, nearbyint, pow, rint, round, trunc) to their intrinsic-equivalents.
This patch pulls the transforms together and handles all 20 cases. The switch is guarded by a
check for const-ness to make sure we're not doing the transform if errno could possibly be set by
the libcall or builtin.
Differential Revision: https://reviews.llvm.org/D40044
llvm-svn: 319593
2017-12-02 07:15:52 +08:00
|
|
|
// CHECK-NO: declare double @llvm.exp.f64(double) [[NUW_RNI]]
|
|
|
|
// CHECK-NO: declare x86_fp80 @llvm.exp.f80(x86_fp80) [[NUW_RNI]]
|
|
|
|
// CHECK-NO: declare float @llvm.exp.f32(float) [[NUW_RNI]]
|
2013-02-27 07:08:48 +08:00
|
|
|
// CHECK-YES-NOT: declare double @exp(double) [[NUW_RN]]
|
|
|
|
// CHECK-YES-NOT: declare x86_fp80 @expl(x86_fp80) [[NUW_RN]]
|
|
|
|
// CHECK-YES-NOT: declare float @expf(float) [[NUW_RN]]
|
2012-04-25 06:40:01 +08:00
|
|
|
|
|
|
|
double log_ = log(d);
|
|
|
|
long double logl_ = logl(ld);
|
|
|
|
float logf_ = logf(f);
|
[CodeGen] convert math libcalls/builtins to equivalent LLVM intrinsics
There are 20 LLVM math intrinsics that correspond to mathlib calls according to the LangRef:
http://llvm.org/docs/LangRef.html#standard-c-library-intrinsics
We were only converting 3 mathlib calls (sqrt, fma, pow) and 12 builtin calls (ceil, copysign,
fabs, floor, fma, fmax, fmin, nearbyint, pow, rint, round, trunc) to their intrinsic-equivalents.
This patch pulls the transforms together and handles all 20 cases. The switch is guarded by a
check for const-ness to make sure we're not doing the transform if errno could possibly be set by
the libcall or builtin.
Differential Revision: https://reviews.llvm.org/D40044
llvm-svn: 319593
2017-12-02 07:15:52 +08:00
|
|
|
// CHECK-NO: declare double @llvm.log.f64(double) [[NUW_RNI]]
|
|
|
|
// CHECK-NO: declare x86_fp80 @llvm.log.f80(x86_fp80) [[NUW_RNI]]
|
|
|
|
// CHECK-NO: declare float @llvm.log.f32(float) [[NUW_RNI]]
|
2013-02-27 07:08:48 +08:00
|
|
|
// CHECK-YES-NOT: declare double @log(double) [[NUW_RN]]
|
|
|
|
// CHECK-YES-NOT: declare x86_fp80 @logl(x86_fp80) [[NUW_RN]]
|
|
|
|
// CHECK-YES-NOT: declare float @logf(float) [[NUW_RN]]
|
2012-04-25 06:40:01 +08:00
|
|
|
}
|
2013-02-20 15:22:19 +08:00
|
|
|
|
2017-11-01 04:19:39 +08:00
|
|
|
// CHECK-NO-DAG: attributes [[NUW_RN]] = { nounwind readnone{{.*}} }
|
2020-10-15 19:04:36 +08:00
|
|
|
// CHECK-NO-DAG: attributes [[NUW_RNI]] = { nofree nosync nounwind readnone speculatable willreturn }
|