[llvm] [AVR] Fix a bug in selection of ANY_EXTEND (PR #132398)
Patryk Wychowaniec via llvm-commits
llvm-commits at lists.llvm.org
Sat Mar 22 13:15:47 PDT 2025
https://github.com/Patryk27 requested changes to this pull request.
For others - the issue is that we end up generating a 16-bit instruction (subtraction, addition, ...) on a partially initialized register-pair.
Following my simplified example, after avr-isel we get:
```
bb.0 (%ir-block.0):
liveins: $r24
%0:gpr8 = COPY $r24
%2:dregs = IMPLICIT_DEF
%1:dldregs = INSERT_SUBREG %2:dregs(tied-def 0), %0:gpr8, %subreg.sub_lo
%3:dldregs = SUBIWRdK %1:dldregs(tied-def 0), 256, implicit-def dead $sreg
%4:dldregs = ANDIWRdK %3:dldregs(tied-def 0), 1, implicit-def dead $sreg
$r25r24 = COPY %4:dldregs
RET implicit $r25r24, implicit $r1
```
... which later gets expanded into:
```
0B bb.0 (%ir-block.0):
liveins: $r24
16B undef %4.sub_lo:dldregs = COPY $r24
64B %4:dldregs = SUBIWRdK %4:dldregs(tied-def 0), 256, implicit-def dead $sreg
96B %4:dldregs = ANDIWRdK %4:dldregs(tied-def 0), 1, implicit-def dead $sreg
112B $r25r24 = COPY %4:dldregs
128B RET implicit $r25r24, implicit $r1
```
... and into (via `virtregrewriter` - note the lost information about partially initialized `$r25r24`):
```
0B bb.0 (%ir-block.0):
liveins: $r24
64B $r25r24 = SUBIWRdK $r25r24(tied-def 0), 256, implicit-def dead $sreg
96B $r25r24 = ANDIWRdK killed $r25r24(tied-def 0), 1, implicit-def dead $sreg
128B RET implicit $r25r24, implicit $r1
```
... into:
```
bb.0 (%ir-block.0):
liveins: $r24
$r24 = SUBIRdK $r24(tied-def 0), 0, implicit-def $sreg
$r25 = SBCIRdK $r25(tied-def 0), 1, implicit-def dead $sreg, implicit killed $sreg
$r24 = ANDIRdK killed $r24(tied-def 0), 1, implicit-def dead $sreg
$r25 = ANDIRdK killed undef $r25(tied-def 0), 0, implicit-def dead $sreg
RET implicit $r25r24, implicit $r1
```
Note that the second `ANDIRdK` correctly says `undef $25`, but that's just a happy accident thanks to
https://github.com/llvm/llvm-project/blob/f3e8e805630e5092458a80e00ba83f7ad45e2302/llvm/lib/Target/AVR/AVRExpandPseudoInsts.cpp#L281
tl;dr it looks like somewhere between avr-isel and virtregrewriter we lose the information about "undef-ness"; I'm not yet sure what's the correct solution, but the proposed hotfix seems to be correct.
https://github.com/llvm/llvm-project/pull/132398
More information about the llvm-commits
mailing list