Index: llvm/trunk/lib/Analysis/InstructionSimplify.cpp =================================================================== --- llvm/trunk/lib/Analysis/InstructionSimplify.cpp +++ llvm/trunk/lib/Analysis/InstructionSimplify.cpp @@ -1752,6 +1752,24 @@ (A == Op0 || B == Op0)) return Op0; + // A mask that only clears known zeros of a shifted value is a no-op. + Value *X; + const APInt *Mask; + const APInt *ShAmt; + if (match(Op1, m_APInt(Mask))) { + // If all bits in the inverted and shifted mask are clear: + // and (shl X, ShAmt), Mask --> shl X, ShAmt + if (match(Op0, m_Shl(m_Value(X), m_APInt(ShAmt))) && + (~(*Mask)).lshr(*ShAmt).isNullValue()) + return Op0; + + // If all bits in the inverted and shifted mask are clear: + // and (lshr X, ShAmt), Mask --> lshr X, ShAmt + if (match(Op0, m_LShr(m_Value(X), m_APInt(ShAmt))) && + (~(*Mask)).shl(*ShAmt).isNullValue()) + return Op0; + } + // A & (-A) = A if A is a power of two or zero. if (match(Op0, m_Neg(m_Specific(Op1))) || match(Op1, m_Neg(m_Specific(Op0)))) { Index: llvm/trunk/test/Transforms/InstSimplify/AndOrXor.ll =================================================================== --- llvm/trunk/test/Transforms/InstSimplify/AndOrXor.ll +++ llvm/trunk/test/Transforms/InstSimplify/AndOrXor.ll @@ -738,8 +738,7 @@ define i8 @lshr_perfect_mask(i8 %x) { ; CHECK-LABEL: @lshr_perfect_mask( ; CHECK-NEXT: [[SH:%.*]] = lshr i8 %x, 5 -; CHECK-NEXT: [[MASK:%.*]] = and i8 [[SH]], 7 -; CHECK-NEXT: ret i8 [[MASK]] +; CHECK-NEXT: ret i8 [[SH]] ; %sh = lshr i8 %x, 5 %mask = and i8 %sh, 7 ; 0x07 @@ -749,8 +748,7 @@ define <2 x i8> @lshr_oversized_mask_splat(<2 x i8> %x) { ; CHECK-LABEL: @lshr_oversized_mask_splat( ; CHECK-NEXT: [[SH:%.*]] = lshr <2 x i8> %x, -; CHECK-NEXT: [[MASK:%.*]] = and <2 x i8> [[SH]], -; CHECK-NEXT: ret <2 x i8> [[MASK]] +; CHECK-NEXT: ret <2 x i8> [[SH]] ; %sh = lshr <2 x i8> %x, %mask = and <2 x i8> %sh, ; 0x87 @@ -771,8 +769,7 @@ define <2 x i8> @shl_perfect_mask_splat(<2 x i8> %x) { ; CHECK-LABEL: @shl_perfect_mask_splat( ; CHECK-NEXT: [[SH:%.*]] = shl <2 x i8> %x, -; CHECK-NEXT: [[MASK:%.*]] = and <2 x i8> [[SH]], -; CHECK-NEXT: ret <2 x i8> [[MASK]] +; CHECK-NEXT: ret <2 x i8> [[SH]] ; %sh = shl <2 x i8> %x, %mask = and <2 x i8> %sh, ; 0xC0 @@ -782,8 +779,7 @@ define i8 @shl_oversized_mask(i8 %x) { ; CHECK-LABEL: @shl_oversized_mask( ; CHECK-NEXT: [[SH:%.*]] = shl i8 %x, 6 -; CHECK-NEXT: [[MASK:%.*]] = and i8 [[SH]], -61 -; CHECK-NEXT: ret i8 [[MASK]] +; CHECK-NEXT: ret i8 [[SH]] ; %sh = shl i8 %x, 6 %mask = and i8 %sh, 195 ; 0xC3