Lower __builtin_fabs* to @llvm.fabs.*
authorReid Kleckner <reid@kleckner.net>
Mon, 3 Nov 2014 23:52:09 +0000 (23:52 +0000)
committerReid Kleckner <reid@kleckner.net>
Mon, 3 Nov 2014 23:52:09 +0000 (23:52 +0000)
mingw64's headers implement fabs by calling __builtin_fabs, so using the
library call results in an infinite loop. If the backend legalizes
@llvm.fabs as a call to fabs later, things should work out, as the crt
provides a definition.

llvm-svn: 221206

clang/lib/CodeGen/CGBuiltin.cpp
clang/test/CodeGen/builtins.c

index 0972e28..aa11a90 100644 (file)
@@ -245,6 +245,13 @@ RValue CodeGenFunction::EmitBuiltinExpr(const FunctionDecl *FD,
 
     return RValue::get(Result);
   }
+  case Builtin::BI__builtin_fabs:
+  case Builtin::BI__builtin_fabsf:
+  case Builtin::BI__builtin_fabsl: {
+    Value *Arg1 = EmitScalarExpr(E->getArg(0));
+    Value *Result = EmitFAbs(*this, Arg1);
+    return RValue::get(Result);
+  }
   case Builtin::BI__builtin_fmod:
   case Builtin::BI__builtin_fmodf:
   case Builtin::BI__builtin_fmodl: {
index 6dd749a..1ab29a6 100644 (file)
@@ -211,6 +211,13 @@ void test_float_builtin_ops(float F, double D, long double LD) {
 
   resld = __builtin_fmodl(LD,LD);
   // CHECK: frem x86_fp80
+
+  resf = __builtin_fabsf(F);
+  resd = __builtin_fabs(D);
+  resld = __builtin_fabsl(LD);
+  // CHECK: call float @llvm.fabs.f32(float
+  // CHECK: call double @llvm.fabs.f64(double
+  // CHECK: call x86_fp80 @llvm.fabs.f80(x86_fp80
 }
 
 // CHECK-LABEL: define void @test_builtin_longjmp