diff options
author | Alon Zakai <alonzakai@gmail.com> | 2017-03-21 18:25:18 -0700 |
---|---|---|
committer | GitHub <noreply@github.com> | 2017-03-21 18:25:18 -0700 |
commit | 7b71bb6b0d3966ce42b631d433c772e24d6e68be (patch) | |
tree | b8392d1b5f1f827247d5afc48d2a93479683a860 /src | |
parent | ee501dfb427e675adee7790a6dbc7e90f9f5a4ca (diff) | |
download | binaryen-7b71bb6b0d3966ce42b631d433c772e24d6e68be.tar.gz binaryen-7b71bb6b0d3966ce42b631d433c772e24d6e68be.tar.bz2 binaryen-7b71bb6b0d3966ce42b631d433c772e24d6e68be.zip |
Fix comparisons of sign-extends to weird constants (#956)
* fix eq/ne of sign-ext with a constant, when the constant can never be equal to it as it has the effective sign bit but not the upper bits above it set, which the sign-ext would emit
Diffstat (limited to 'src')
-rw-r--r-- | src/passes/OptimizeInstructions.cpp | 37 |
1 files changed, 31 insertions, 6 deletions
diff --git a/src/passes/OptimizeInstructions.cpp b/src/passes/OptimizeInstructions.cpp index ad828dafc..c1787b57d 100644 --- a/src/passes/OptimizeInstructions.cpp +++ b/src/passes/OptimizeInstructions.cpp @@ -423,18 +423,43 @@ struct OptimizeInstructions : public WalkerPass<PostWalker<OptimizeInstructions, } } else if (binary->op == EqInt32 || binary->op == NeInt32) { if (auto* c = binary->right->dynCast<Const>()) { + if (binary->op == EqInt32 && c->value.geti32() == 0) { + // equal 0 => eqz + return Builder(*getModule()).makeUnary(EqZInt32, binary->left); + } if (auto* ext = Properties::getSignExtValue(binary->left)) { // we are comparing a sign extend to a constant, which means we can use a cheaper zext auto bits = Properties::getSignExtBits(binary->left); binary->left = makeZeroExt(ext, bits); - // the const we compare to only needs the relevant bits - c->value = c->value.and_(Literal(Bits::lowBitMask(bits))); + // when we replace the sign-ext of the non-constant with a zero-ext, we are forcing + // the high bits to be all zero, instead of all zero or all one depending on the + // sign bit. so we may be changing the high bits from all one to all zero: + // * if the constant value's higher bits are mixed, then it can't be equal anyhow + // * if they are all zero, we may get a false true if the non-constant's upper bits + // were one. this can only happen if the non-constant's sign bit is set, so this + // false true is a risk only if the constant's sign bit is set (otherwise, false). + // But a constant with a sign bit but with upper bits zero is impossible to be + // equal to a sign-extended value anyhow, so the entire thing is false. + // * if they were all one, we may get a false false, if the only difference is in + // those upper bits. that means we are equal on the other bits, including the sign + // bit. so we can just mask off the upper bits in the constant value, in this + // case, forcing them to zero like we do in the zero-extend. + int32_t constValue = c->value.geti32(); + auto upperConstValue = constValue & ~Bits::lowBitMask(bits); + uint32_t count = PopCount(upperConstValue); + auto constSignBit = constValue & (1 << (bits - 1)); + if ((count > 0 && count < 32 - bits) || (constSignBit && count == 0)) { + // mixed or [zero upper const bits with sign bit set]; the compared values can never be identical, so + // force something definitely impossible even after zext + assert(bits < 31); + c->value = Literal(int32_t(0x80000000)); + // TODO: if no side effects, we can just replace it all with 1 or 0 + } else { + // otherwise, they are all ones, so we can mask them off as mentioned before + c->value = c->value.and_(Literal(Bits::lowBitMask(bits))); + } return binary; } - if (binary->op == EqInt32 && c->value.geti32() == 0) { - // equal 0 => eqz - return Builder(*getModule()).makeUnary(EqZInt32, binary->left); - } } else if (auto* left = Properties::getSignExtValue(binary->left)) { if (auto* right = Properties::getSignExtValue(binary->right)) { auto bits = Properties::getSignExtBits(binary->left); |