[llvm] Combine (X ^ Y) and (X == Y) where appropriate (PR #130922)

Craig Topper via llvm-commits llvm-commits at lists.llvm.org
Wed Mar 12 14:46:24 PDT 2025


topperc wrote:

> I addressed the above comments, but I came across one issue in my tests:
> 
> When working with riscv64 with i32 an i32 input (a0 is i32), the following code generation occurs:
> 
> ```
> entry:
>   %cmp.not = icmp eq i32 %x, 2048
>   br i1 %cmp.not, label %if.end, label %if.then
> if.then:
>   %xor = xor i32 %x, 2048
>   ret i32 %xor
> if.end:
>     tail call void @abort() #2
>   unreachable
> 
> ----->
> 
> # %bb.0:
>    li a1, 1
>    slli a1, a1, 11
>    xor a0, a0, a1
>    sext.w a1, a0
>    beqz a1, .LBB0_2
> # %bb.1: 
>         ret
> .LBB0_2: 
> ```
> 
> Because the result of the sign extension is being used for the comparison, it does not matching the XOR folding pattern.
> 
> One though is that this sign extension is not needed because we are comparing against 0, so we can fold away the sign-extension. However, I'm not sure if that is legal or not (i.e. is an LLVM i32 guaranteed to have the upper bits 0'ed when compiling to a 64 bit architecture).

If %x is a function attribute with the `signext` attribute we can assume the i32 is sign extended to the full register size. Likewise the `zeroext` attribute allows us to assume it is zero extended. If neither attribute is present, we can't make any assumptions.



https://github.com/llvm/llvm-project/pull/130922


More information about the llvm-commits mailing list