Masked merge has a pattern of: ((x ^ y) & M) ^ y.
But, there is no difference between ((x ^ y) & M) ^ y and ((x ^ y) & ~M) ^ x,
We should canonicalize it and drop that xor.
Details
Diff Detail
- Repository
- rL LLVM
Event Timeline
Slightly cleanup the matcher by using m_CombineAnd(),
which allows to use m_c_And() when looking for A part.
I'm not sure whether it actually matter because i was not able
to come up with a testcase, but it looks a bit cleaner anyway..
lib/Transforms/InstCombine/InstCombineAndOrXor.cpp | ||
---|---|---|
2493–2495 | Make sure I haven't missed anything, but it seems easier to match this inline here more simply as: // Eliminate a 'not' from xor form of masked-merge (aka, bitwise select). // The 'not' must be operand 1 of the 'and' (complexity-based op ordering). // ((Op1 ^ X) & ~Y) ^ Op1 --> ((Op1 ^ X) & Y) ^ X if (match(Op0, m_OneUse(m_And(m_c_Xor(m_Specific(Op1), m_Value(X)), m_Not(m_Value(Y)))))) { Value *Op00 = cast<BinaryOperator>(Op0)->getOperand(0); return BinaryOperator::CreateXor(Builder.CreateAnd(Op00, Y), X); } // Op0 ^ ((Op0 ^ X) & ~Y) --> ((Op0 ^ X) & Y) ^ X if (match(Op1, m_OneUse(m_And(m_c_Xor(m_Specific(Op0), m_Value(X)), m_Not(m_Value(Y)))))) { Value *Op10 = cast<BinaryOperator>(Op1)->getOperand(0); return BinaryOperator::CreateXor(Builder.CreateAnd(Op10, Y), X); } |
lib/Transforms/InstCombine/InstCombineAndOrXor.cpp | ||
---|---|---|
2493–2495 | I'm sorry, i don't really like this, especially after seeing the DAGCombiner :) |
- Rebased.
- Ping. This differential is the base, and has no unmerged dependencies, so this is the one to review :)
The comment seems inverted from our usual description of:
Starting pattern --> Ending pattern