[X86] Add test cases to show missed opportunities to remove AND mask from BTC/BTS/BTR instructions when LHS of AND has known zeros.

We can currently remove the mask if the immediate has all ones in the LSBs, but if the LHS of the AND is known zero, then the immediate might have had bits removed.

A similar issue also occurs with shifts and rotates. I'm preparing a common fix for all of them.

llvm-svn: 354520
This commit is contained in:
Craig Topper 2019-02-20 21:35:05 +00:00
parent 198cc305e9
commit 55cc7eb5cb
1 changed files with 187 additions and 0 deletions

View File

@ -949,3 +949,190 @@ define void @btc_64_dont_fold(i64* %x, i64 %n) {
store i64 %3, i64* %x
ret void
}
define i32 @btr_32_mask_zeros(i32 %x, i32 %n) {
; X64-LABEL: btr_32_mask_zeros:
; X64: # %bb.0:
; X64-NEXT: movl %edi, %eax
; X64-NEXT: shlb $2, %sil
; X64-NEXT: andb $28, %sil
; X64-NEXT: btrl %esi, %eax
; X64-NEXT: retq
;
; X86-LABEL: btr_32_mask_zeros:
; X86: # %bb.0:
; X86-NEXT: movl {{[0-9]+}}(%esp), %eax
; X86-NEXT: movb {{[0-9]+}}(%esp), %cl
; X86-NEXT: shlb $2, %cl
; X86-NEXT: andb $28, %cl
; X86-NEXT: btrl %ecx, %eax
; X86-NEXT: retl
%1 = shl i32 %n, 2
%2 = and i32 %1, 31
%3 = shl i32 1, %2
%4 = xor i32 %3, -1
%5 = and i32 %x, %4
ret i32 %5
}
define i32 @bts_32_mask_zeros(i32 %x, i32 %n) {
; X64-LABEL: bts_32_mask_zeros:
; X64: # %bb.0:
; X64-NEXT: movl %edi, %eax
; X64-NEXT: shlb $2, %sil
; X64-NEXT: andb $28, %sil
; X64-NEXT: btsl %esi, %eax
; X64-NEXT: retq
;
; X86-LABEL: bts_32_mask_zeros:
; X86: # %bb.0:
; X86-NEXT: movl {{[0-9]+}}(%esp), %eax
; X86-NEXT: movb {{[0-9]+}}(%esp), %cl
; X86-NEXT: shlb $2, %cl
; X86-NEXT: andb $28, %cl
; X86-NEXT: btsl %ecx, %eax
; X86-NEXT: retl
%1 = shl i32 %n, 2
%2 = and i32 %1, 31
%3 = shl i32 1, %2
%4 = or i32 %x, %3
ret i32 %4
}
define i32 @btc_32_mask_zeros(i32 %x, i32 %n) {
; X64-LABEL: btc_32_mask_zeros:
; X64: # %bb.0:
; X64-NEXT: movl %edi, %eax
; X64-NEXT: shlb $2, %sil
; X64-NEXT: andb $28, %sil
; X64-NEXT: btcl %esi, %eax
; X64-NEXT: retq
;
; X86-LABEL: btc_32_mask_zeros:
; X86: # %bb.0:
; X86-NEXT: movl {{[0-9]+}}(%esp), %eax
; X86-NEXT: movb {{[0-9]+}}(%esp), %cl
; X86-NEXT: shlb $2, %cl
; X86-NEXT: andb $28, %cl
; X86-NEXT: btcl %ecx, %eax
; X86-NEXT: retl
%1 = shl i32 %n, 2
%2 = and i32 %1, 31
%3 = shl i32 1, %2
%4 = xor i32 %x, %3
ret i32 %4
}
define i64 @btr_64_mask_zeros(i64 %x, i64 %n) {
; X64-LABEL: btr_64_mask_zeros:
; X64: # %bb.0:
; X64-NEXT: movq %rdi, %rax
; X64-NEXT: shlb $2, %sil
; X64-NEXT: andb $60, %sil
; X64-NEXT: btrq %rsi, %rax
; X64-NEXT: retq
;
; X86-LABEL: btr_64_mask_zeros:
; X86: # %bb.0:
; X86-NEXT: movb {{[0-9]+}}(%esp), %ch
; X86-NEXT: shlb $2, %ch
; X86-NEXT: movb %ch, %cl
; X86-NEXT: andb $60, %cl
; X86-NEXT: movl $1, %eax
; X86-NEXT: xorl %edx, %edx
; X86-NEXT: shldl %cl, %eax, %edx
; X86-NEXT: movb %ch, %cl
; X86-NEXT: andb $28, %cl
; X86-NEXT: shll %cl, %eax
; X86-NEXT: testb $32, %ch
; X86-NEXT: je .LBB39_2
; X86-NEXT: # %bb.1:
; X86-NEXT: movl %eax, %edx
; X86-NEXT: xorl %eax, %eax
; X86-NEXT: .LBB39_2:
; X86-NEXT: notl %edx
; X86-NEXT: notl %eax
; X86-NEXT: andl {{[0-9]+}}(%esp), %edx
; X86-NEXT: andl {{[0-9]+}}(%esp), %eax
; X86-NEXT: retl
%1 = shl i64 %n, 2
%2 = and i64 %1, 63
%3 = shl i64 1, %2
%4 = xor i64 %3, -1
%5 = and i64 %x, %4
ret i64 %5
}
define i64 @bts_64_mask_zeros(i64 %x, i64 %n) {
; X64-LABEL: bts_64_mask_zeros:
; X64: # %bb.0:
; X64-NEXT: movq %rdi, %rax
; X64-NEXT: shlb $2, %sil
; X64-NEXT: andb $60, %sil
; X64-NEXT: btsq %rsi, %rax
; X64-NEXT: retq
;
; X86-LABEL: bts_64_mask_zeros:
; X86: # %bb.0:
; X86-NEXT: movb {{[0-9]+}}(%esp), %ch
; X86-NEXT: shlb $2, %ch
; X86-NEXT: movb %ch, %cl
; X86-NEXT: andb $60, %cl
; X86-NEXT: movl $1, %eax
; X86-NEXT: xorl %edx, %edx
; X86-NEXT: shldl %cl, %eax, %edx
; X86-NEXT: movb %ch, %cl
; X86-NEXT: andb $28, %cl
; X86-NEXT: shll %cl, %eax
; X86-NEXT: testb $32, %ch
; X86-NEXT: je .LBB40_2
; X86-NEXT: # %bb.1:
; X86-NEXT: movl %eax, %edx
; X86-NEXT: xorl %eax, %eax
; X86-NEXT: .LBB40_2:
; X86-NEXT: orl {{[0-9]+}}(%esp), %edx
; X86-NEXT: orl {{[0-9]+}}(%esp), %eax
; X86-NEXT: retl
%1 = shl i64 %n, 2
%2 = and i64 %1, 63
%3 = shl i64 1, %2
%4 = or i64 %x, %3
ret i64 %4
}
define i64 @btc_64_mask_zeros(i64 %x, i64 %n) {
; X64-LABEL: btc_64_mask_zeros:
; X64: # %bb.0:
; X64-NEXT: movq %rdi, %rax
; X64-NEXT: shlb $2, %sil
; X64-NEXT: andb $60, %sil
; X64-NEXT: btcq %rsi, %rax
; X64-NEXT: retq
;
; X86-LABEL: btc_64_mask_zeros:
; X86: # %bb.0:
; X86-NEXT: movb {{[0-9]+}}(%esp), %ch
; X86-NEXT: shlb $2, %ch
; X86-NEXT: movb %ch, %cl
; X86-NEXT: andb $60, %cl
; X86-NEXT: movl $1, %eax
; X86-NEXT: xorl %edx, %edx
; X86-NEXT: shldl %cl, %eax, %edx
; X86-NEXT: movb %ch, %cl
; X86-NEXT: andb $28, %cl
; X86-NEXT: shll %cl, %eax
; X86-NEXT: testb $32, %ch
; X86-NEXT: je .LBB41_2
; X86-NEXT: # %bb.1:
; X86-NEXT: movl %eax, %edx
; X86-NEXT: xorl %eax, %eax
; X86-NEXT: .LBB41_2:
; X86-NEXT: xorl {{[0-9]+}}(%esp), %edx
; X86-NEXT: xorl {{[0-9]+}}(%esp), %eax
; X86-NEXT: retl
%1 = shl i64 %n, 2
%2 = and i64 %1, 63
%3 = shl i64 1, %2
%4 = xor i64 %x, %3
ret i64 %4
}