diff options
| author | Chris Lattner <sabre@nondot.org> | 2010-01-11 04:05:13 +0000 | 
|---|---|---|
| committer | Chris Lattner <sabre@nondot.org> | 2010-01-11 04:05:13 +0000 | 
| commit | 0a85420409670db429b1d70a1801f0a543541b3a (patch) | |
| tree | bdce77cf9fe295f2c031e0d38189a8407fa81bc1 /llvm/lib/Transforms | |
| parent | 12bd8992b3eca2abb2f391dfbd14791117d09fe5 (diff) | |
| download | bcm5719-llvm-0a85420409670db429b1d70a1801f0a543541b3a.tar.gz bcm5719-llvm-0a85420409670db429b1d70a1801f0a543541b3a.zip | |
Extend CanEvaluateZExtd to handle and/or/xor more aggressively in the
BitsToClear case.  This allows it to promote expressions which have an
and/or/xor after the lshr, promoting cases like test2 (from PR4216) 
and test3 (random extample extracted from a spec benchmark).
clang now compiles the code in PR4216 into:
_test_bitfield:                                             ## @test_bitfield
	movl	%edi, %eax
	orl	$194, %eax
	movl	$4294902010, %ecx
	andq	%rax, %rcx
	orl	$32768, %edi
	andq	$39936, %rdi
	movq	%rdi, %rax
	orq	%rcx, %rax
	ret
instead of:
_test_bitfield:                                             ## @test_bitfield
	movl	%edi, %eax
	orl	$194, %eax
	movl	$4294902010, %ecx
	andq	%rax, %rcx
	shrl	$8, %edi
	orl	$128, %edi
	shlq	$8, %rdi
	andq	$39936, %rdi
	movq	%rdi, %rax
	orq	%rcx, %rax
	ret
which is still not great, but is progress.
llvm-svn: 93145
Diffstat (limited to 'llvm/lib/Transforms')
| -rw-r--r-- | llvm/lib/Transforms/InstCombine/InstCombineCasts.cpp | 18 | 
1 files changed, 18 insertions, 0 deletions
| diff --git a/llvm/lib/Transforms/InstCombine/InstCombineCasts.cpp b/llvm/lib/Transforms/InstCombine/InstCombineCasts.cpp index be0734c7a43..939c2b1937e 100644 --- a/llvm/lib/Transforms/InstCombine/InstCombineCasts.cpp +++ b/llvm/lib/Transforms/InstCombine/InstCombineCasts.cpp @@ -634,6 +634,20 @@ static bool CanEvaluateZExtd(Value *V, const Type *Ty, unsigned &BitsToClear) {      if (BitsToClear == 0 && Tmp == 0)        return true; +    // If the operation is an AND/OR/XOR and the bits to clear are zero in the +    // other side, BitsToClear is ok. +    if (Tmp == 0 && +        (Opc == Instruction::And || Opc == Instruction::Or || +         Opc == Instruction::Xor)) { +      // We use MaskedValueIsZero here for generality, but the case we care +      // about the most is constant RHS. +      unsigned VSize = V->getType()->getScalarSizeInBits(); +      if (MaskedValueIsZero(I->getOperand(1), +                            APInt::getHighBitsSet(VSize, BitsToClear))) +        return true; +    } +       +    // Otherwise, we don't know how to analyze this BitsToClear case yet.      return false;    case Instruction::LShr: @@ -652,6 +666,8 @@ static bool CanEvaluateZExtd(Value *V, const Type *Ty, unsigned &BitsToClear) {    case Instruction::Select:      if (!CanEvaluateZExtd(I->getOperand(1), Ty, Tmp) ||          !CanEvaluateZExtd(I->getOperand(2), Ty, BitsToClear) || +        // TODO: If important, we could handle the case when the BitsToClear are +        // known zero in the disagreeing side.          Tmp != BitsToClear)        return false;      return true; @@ -665,6 +681,8 @@ static bool CanEvaluateZExtd(Value *V, const Type *Ty, unsigned &BitsToClear) {        return false;      for (unsigned i = 1, e = PN->getNumIncomingValues(); i != e; ++i)        if (!CanEvaluateZExtd(PN->getIncomingValue(i), Ty, Tmp) || +          // TODO: If important, we could handle the case when the BitsToClear +          // are known zero in the disagreeing input.            Tmp != BitsToClear)          return false;      return true; | 

