diff options
author | Chris Lattner <sabre@nondot.org> | 2007-01-14 19:42:17 +0000 |
---|---|---|
committer | Chris Lattner <sabre@nondot.org> | 2007-01-14 19:42:17 +0000 |
commit | 6ee923f3bb68a243682949678ff9a0f02fb03ba0 (patch) | |
tree | e65ec7f277fb1d5049b892b2b3ccc923f37db8a5 /llvm/lib/Transforms/Scalar/InstructionCombining.cpp | |
parent | 1ca997078c2cf607dfc074a19f822e38fa716f58 (diff) | |
download | bcm5719-llvm-6ee923f3bb68a243682949678ff9a0f02fb03ba0.tar.gz bcm5719-llvm-6ee923f3bb68a243682949678ff9a0f02fb03ba0.zip |
instcombine has always been miscompiling fcmp x, x, disregarding possible
NANs. This fixes PR1111 and Transforms/InstCombine/2007-01-14-FcmpSelf.ll
llvm-svn: 33208
Diffstat (limited to 'llvm/lib/Transforms/Scalar/InstructionCombining.cpp')
-rw-r--r-- | llvm/lib/Transforms/Scalar/InstructionCombining.cpp | 54 |
1 files changed, 39 insertions, 15 deletions
diff --git a/llvm/lib/Transforms/Scalar/InstructionCombining.cpp b/llvm/lib/Transforms/Scalar/InstructionCombining.cpp index 743502240d4..e4db98ecb69 100644 --- a/llvm/lib/Transforms/Scalar/InstructionCombining.cpp +++ b/llvm/lib/Transforms/Scalar/InstructionCombining.cpp @@ -1457,16 +1457,6 @@ static bool isTrueWhenEqual(ICmpInst &ICI) { pred == ICmpInst::ICMP_SLE; } -/// @returns true if the specified compare instruction is -/// true when both operands are equal... -/// @brief Determine if the FCmpInst returns true if both operands are equal -static bool isTrueWhenEqual(FCmpInst &FCI) { - FCmpInst::Predicate pred = FCI.getPredicate(); - return pred == FCmpInst::FCMP_OEQ || pred == FCmpInst::FCMP_UEQ || - pred == FCmpInst::FCMP_OGE || pred == FCmpInst::FCMP_UGE || - pred == FCmpInst::FCMP_OLE || pred == FCmpInst::FCMP_ULE; -} - /// AssociativeOpt - Perform an optimization on an associative operator. This /// function is designed to check a chain of associative operators for a /// potential to apply a certain optimization. Since the optimization may be @@ -4281,11 +4271,45 @@ Instruction *InstCombiner::visitFCmpInst(FCmpInst &I) { bool Changed = SimplifyCompare(I); Value *Op0 = I.getOperand(0), *Op1 = I.getOperand(1); - // fcmp pred X, X - if (Op0 == Op1) - return ReplaceInstUsesWith(I, ConstantInt::get(Type::Int1Ty, - isTrueWhenEqual(I))); - + // Fold trivial predicates. + if (I.getPredicate() == FCmpInst::FCMP_FALSE) + return ReplaceInstUsesWith(I, Constant::getNullValue(Type::Int1Ty)); + if (I.getPredicate() == FCmpInst::FCMP_TRUE) + return ReplaceInstUsesWith(I, ConstantInt::get(Type::Int1Ty, 1)); + + // Simplify 'fcmp pred X, X' + if (Op0 == Op1) { + switch (I.getPredicate()) { + default: assert(0 && "Unknown predicate!"); + case FCmpInst::FCMP_UEQ: // True if unordered or equal + case FCmpInst::FCMP_UGE: // True if unordered, greater than, or equal + case FCmpInst::FCMP_ULE: // True if unordered, less than, or equal + return ReplaceInstUsesWith(I, ConstantInt::get(Type::Int1Ty, 1)); + case FCmpInst::FCMP_OGT: // True if ordered and greater than + case FCmpInst::FCMP_OLT: // True if ordered and less than + case FCmpInst::FCMP_ONE: // True if ordered and operands are unequal + return ReplaceInstUsesWith(I, ConstantInt::get(Type::Int1Ty, 0)); + + case FCmpInst::FCMP_UNO: // True if unordered: isnan(X) | isnan(Y) + case FCmpInst::FCMP_ULT: // True if unordered or less than + case FCmpInst::FCMP_UGT: // True if unordered or greater than + case FCmpInst::FCMP_UNE: // True if unordered or not equal + // Canonicalize these to be 'fcmp uno %X, 0.0'. + I.setPredicate(FCmpInst::FCMP_UNO); + I.setOperand(1, Constant::getNullValue(Op0->getType())); + return &I; + + case FCmpInst::FCMP_ORD: // True if ordered (no nans) + case FCmpInst::FCMP_OEQ: // True if ordered and equal + case FCmpInst::FCMP_OGE: // True if ordered and greater than or equal + case FCmpInst::FCMP_OLE: // True if ordered and less than or equal + // Canonicalize these to be 'fcmp ord %X, 0.0'. + I.setPredicate(FCmpInst::FCMP_ORD); + I.setOperand(1, Constant::getNullValue(Op0->getType())); + return &I; + } + } + if (isa<UndefValue>(Op1)) // fcmp pred X, undef -> undef return ReplaceInstUsesWith(I, UndefValue::get(Type::Int1Ty)); |