Support constant-evaluation of __builtin_nans* as well as the correct constant

evaluation of __builtin_nan*.  Most of the work to make this work is in LLVM.

Fixes <rdar://problem/7696712> and part of PR 5255.

llvm-svn: 97383
This commit is contained in:
John McCall 2010-02-28 13:00:19 +00:00
parent c12b133d00
commit 1629149103
2 changed files with 75 additions and 16 deletions

View File

@ -1560,6 +1560,31 @@ static bool EvaluateFloat(const Expr* E, APFloat& Result, EvalInfo &Info) {
return FloatExprEvaluator(Info, Result).Visit(const_cast<Expr*>(E)); return FloatExprEvaluator(Info, Result).Visit(const_cast<Expr*>(E));
} }
static bool TryEvaluateBuiltinNaN(ASTContext &Context,
QualType ResultTy,
const Expr *Arg,
bool SNaN,
llvm::APFloat &Result) {
const StringLiteral *S = dyn_cast<StringLiteral>(Arg->IgnoreParenCasts());
if (!S) return false;
const llvm::fltSemantics &Sem = Context.getFloatTypeSemantics(ResultTy);
llvm::APInt fill;
// Treat empty strings as if they were zero.
if (S->getString().empty())
fill = llvm::APInt(32, 0);
else if (S->getString().getAsInteger(0, fill))
return false;
if (SNaN)
Result = llvm::APFloat::getSNaN(Sem, false, &fill);
else
Result = llvm::APFloat::getQNaN(Sem, false, &fill);
return true;
}
bool FloatExprEvaluator::VisitCallExpr(const CallExpr *E) { bool FloatExprEvaluator::VisitCallExpr(const CallExpr *E) {
switch (E->isBuiltinCall(Info.Ctx)) { switch (E->isBuiltinCall(Info.Ctx)) {
default: return false; default: return false;
@ -1575,24 +1600,19 @@ bool FloatExprEvaluator::VisitCallExpr(const CallExpr *E) {
return true; return true;
} }
case Builtin::BI__builtin_nans:
case Builtin::BI__builtin_nansf:
case Builtin::BI__builtin_nansl:
return TryEvaluateBuiltinNaN(Info.Ctx, E->getType(), E->getArg(0),
true, Result);
case Builtin::BI__builtin_nan: case Builtin::BI__builtin_nan:
case Builtin::BI__builtin_nanf: case Builtin::BI__builtin_nanf:
case Builtin::BI__builtin_nanl: case Builtin::BI__builtin_nanl:
// If this is __builtin_nan() turn this into a nan, otherwise we // If this is __builtin_nan() turn this into a nan, otherwise we
// can't constant fold it. // can't constant fold it.
if (const StringLiteral *S = return TryEvaluateBuiltinNaN(Info.Ctx, E->getType(), E->getArg(0),
dyn_cast<StringLiteral>(E->getArg(0)->IgnoreParenCasts())) { false, Result);
if (!S->isWide()) {
const llvm::fltSemantics &Sem =
Info.Ctx.getFloatTypeSemantics(E->getType());
unsigned Type = 0;
if (!S->getString().empty() && S->getString().getAsInteger(0, Type))
return false;
Result = llvm::APFloat::getNaN(Sem, false, Type);
return true;
}
}
return false;
case Builtin::BI__builtin_fabs: case Builtin::BI__builtin_fabs:
case Builtin::BI__builtin_fabsf: case Builtin::BI__builtin_fabsf:

View File

@ -1,5 +1,6 @@
// RUN: %clang_cc1 -emit-llvm -o %t %s // RUN: %clang_cc1 -emit-llvm -o %t %s
// RUN: not grep __builtin %t // RUN: not grep __builtin %t
// RUN: %clang_cc1 %s -emit-llvm -o - -triple x86_64-darwin-apple | FileCheck %s
int printf(const char *, ...); int printf(const char *, ...);
@ -9,11 +10,17 @@ void p(char *str, int x) {
void q(char *str, double x) { void q(char *str, double x) {
printf("%s: %f\n", str, x); printf("%s: %f\n", str, x);
} }
void r(char *str, void *ptr) {
printf("%s: %p\n", str, ptr);
}
int random(void);
int main() { int main() {
int N = random(); int N = random();
#define P(n,args) p(#n #args, __builtin_##n args) #define P(n,args) p(#n #args, __builtin_##n args)
#define Q(n,args) q(#n #args, __builtin_##n args) #define Q(n,args) q(#n #args, __builtin_##n args)
#define R(n,args) r(#n #args, __builtin_##n args)
#define V(n,args) p(#n #args, (__builtin_##n args, 0)) #define V(n,args) p(#n #args, (__builtin_##n args, 0))
P(types_compatible_p, (int, float)); P(types_compatible_p, (int, float));
P(choose_expr, (0, 10, 20)); P(choose_expr, (0, 10, 20));
@ -110,16 +117,48 @@ int main() {
// FIXME // FIXME
// V(clear_cache, (&N, &N+1)); // V(clear_cache, (&N, &N+1));
V(trap, ()); V(trap, ());
P(extract_return_addr, (&N)); R(extract_return_addr, (&N));
return 0; return 0;
} }
void strcat() {}
void foo() { void foo() {
__builtin_strcat(0, 0); __builtin_strcat(0, 0);
} }
// CHECK: define void @bar(
void bar() {
float f;
double d;
long double ld;
// LLVM's hex representation of float constants is really unfortunate;
// basically it does a float-to-double "conversion" and then prints the
// hex form of that. That gives us wierd artifacts like exponents
// that aren't numerically similar to the original exponent and
// significand bit-patterns that are offset by three bits (because
// the exponent was expanded from 8 bits to 11).
//
// 0xAE98 == 1010111010011000
// 0x15D3 == 1010111010011
f = __builtin_huge_valf(); // CHECK: float 0x7FF0000000000000
d = __builtin_huge_val(); // CHECK: double 0x7FF0000000000000
ld = __builtin_huge_vall(); // CHECK: x86_fp80 0xK7FFF8000000000000000
f = __builtin_nanf(""); // CHECK: float 0x7FF8000000000000
d = __builtin_nan(""); // CHECK: double 0x7FF8000000000000
ld = __builtin_nanl(""); // CHECK: x86_fp80 0xK7FFFC000000000000000
f = __builtin_nanf("0xAE98"); // CHECK: float 0x7FF815D300000000
d = __builtin_nan("0xAE98"); // CHECK: double 0x7FF800000000AE98
ld = __builtin_nanl("0xAE98"); // CHECK: x86_fp80 0xK7FFFC00000000000AE98
f = __builtin_nansf(""); // CHECK: float 0x7FF4000000000000
d = __builtin_nans(""); // CHECK: double 0x7FF4000000000000
ld = __builtin_nansl(""); // CHECK: x86_fp80 0xK7FFFA000000000000000
f = __builtin_nansf("0xAE98"); // CHECK: float 0x7FF015D300000000
d = __builtin_nans("0xAE98"); // CHECK: double 0x7FF000000000AE98
ld = __builtin_nansl("0xAE98");// CHECK: x86_fp80 0xK7FFF800000000000AE98
}
// CHECK: }