| # NOTE: Assertions have been autogenerated by utils/update_mir_test_checks.py UTC_ARGS: --version 4 |
| # RUN: llc %s -mtriple=riscv64 -run-pass=greedy,virtregrewriter,stack-slot-coloring -o - | FileCheck %s |
| |
| --- |
| name: foo |
| alignment: 4 |
| tracksRegLiveness: true |
| frameInfo: |
| maxAlignment: 4 |
| localFrameSize: 4 |
| stack: |
| - { id: 0, size: 1, alignment: 4, local-offset: -4 } |
| machineFunctionInfo: |
| varArgsFrameIndex: 0 |
| varArgsSaveSize: 0 |
| body: | |
| bb.0.entry: |
| ; To trick stack-slot-colouring to run its dead-store-elimination phase, |
| ; which is at fault, we need the register allocator to run, and spill in two |
| ; places that can have their slots merged. Achieve this by volatile-loading |
| ; data into all allocatable GPRs except $x31. Then, volatile storing them |
| ; later, leaving regalloc only $x31 to play with in the middle. |
| $x1 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x5 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x6 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x7 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x8 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x9 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x10 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x11 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x12 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x13 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x14 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x15 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x16 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x17 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x18 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x19 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x20 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x21 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x22 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x23 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x24 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x25 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x26 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x27 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x28 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x29 = LW %stack.0, 0 :: (volatile load (s32)) |
| $x30 = LW %stack.0, 0 :: (volatile load (s32)) |
| |
| ; Force the first spill. |
| %0:gpr = LW %stack.0, 0 :: (volatile load (s32)) |
| %1:gpr = LW %stack.0, 0 :: (volatile load (s32)) |
| SW %1, %stack.0, 0 :: (volatile store (s32)) |
| SW %0, %stack.0, 0 :: (volatile store (s32)) |
| |
| ; CHECK: renamable $x31 = LW %stack.0, 0 :: (volatile load (s32)) |
| ; CHECK-NEXT: SD killed renamable $x31, %stack.1, 0 :: (store (s64) into %stack.1) |
| ; CHECK-NEXT: renamable $x31 = LW %stack.0, 0 :: (volatile load (s32)) |
| ; CHECK-NEXT: SW killed renamable $x31, %stack.0, 0 :: (volatile store (s32)) |
| ; CHECK-NEXT: renamable $x31 = LD %stack.1, 0 :: (load (s64) from %stack.1) |
| ; CHECK-NEXT: SW killed renamable $x31, %stack.0, 0 :: (volatile store (s32)) |
| |
| ; stack-slot-coloring doesn't know that a write to $x0 is discarded. |
| dead $x0 = LW %stack.0, 0 :: (volatile load (s32)) |
| ; This stores 0 rather than the result of the preceding load since $x0 |
| ; is special. |
| ; We don't want this store to be deleted. |
| SW $x0, %stack.0, 0 :: (volatile store (s32)) |
| |
| ; CHECK-NEXT: dead $x0 = LW %stack.0, 0 :: (volatile load (s32)) |
| ; CHECK-NEXT: SW $x0, %stack.0, 0 :: (volatile store (s32)) |
| |
| ; Force a second spill |
| %2:gpr = LW %stack.0, 0 :: (volatile load (s32)) |
| %3:gpr = LW %stack.0, 0 :: (volatile load (s32)) |
| SW %3, %stack.0, 0 :: (volatile store (s32)) |
| SW %2, %stack.0, 0 :: (volatile store (s32)) |
| |
| ; CHECK-NEXT: renamable $x31 = LW %stack.0, 0 :: (volatile load (s32)) |
| ; CHECK-NEXT: SD killed renamable $x31, %stack.1, 0 :: (store (s64) into %stack.1) |
| ; CHECK-NEXT: renamable $x31 = LW %stack.0, 0 :: (volatile load (s32)) |
| ; CHECK-NEXT: SW killed renamable $x31, %stack.0, 0 :: (volatile store (s32)) |
| ; CHECK-NEXT: renamable $x31 = LD %stack.1, 0 :: (load (s64) from %stack.1) |
| ; CHECK-NEXT: SW killed renamable $x31, %stack.0, 0 :: (volatile store (s32)) |
| |
| SW $x1, %stack.0, 0 :: (volatile store (s32)) |
| SW $x5, %stack.0, 0 :: (volatile store (s32)) |
| SW $x6, %stack.0, 0 :: (volatile store (s32)) |
| SW $x7, %stack.0, 0 :: (volatile store (s32)) |
| SW $x8, %stack.0, 0 :: (volatile store (s32)) |
| SW $x9, %stack.0, 0 :: (volatile store (s32)) |
| SW $x10, %stack.0, 0 :: (volatile store (s32)) |
| SW $x11, %stack.0, 0 :: (volatile store (s32)) |
| SW $x12, %stack.0, 0 :: (volatile store (s32)) |
| SW $x13, %stack.0, 0 :: (volatile store (s32)) |
| SW $x14, %stack.0, 0 :: (volatile store (s32)) |
| SW $x15, %stack.0, 0 :: (volatile store (s32)) |
| SW $x16, %stack.0, 0 :: (volatile store (s32)) |
| SW $x17, %stack.0, 0 :: (volatile store (s32)) |
| SW $x18, %stack.0, 0 :: (volatile store (s32)) |
| SW $x19, %stack.0, 0 :: (volatile store (s32)) |
| SW $x20, %stack.0, 0 :: (volatile store (s32)) |
| SW $x21, %stack.0, 0 :: (volatile store (s32)) |
| SW $x22, %stack.0, 0 :: (volatile store (s32)) |
| SW $x23, %stack.0, 0 :: (volatile store (s32)) |
| SW $x24, %stack.0, 0 :: (volatile store (s32)) |
| SW $x25, %stack.0, 0 :: (volatile store (s32)) |
| SW $x26, %stack.0, 0 :: (volatile store (s32)) |
| SW $x27, %stack.0, 0 :: (volatile store (s32)) |
| SW $x28, %stack.0, 0 :: (volatile store (s32)) |
| SW $x29, %stack.0, 0 :: (volatile store (s32)) |
| SW $x30, %stack.0, 0 :: (volatile store (s32)) |
| PseudoRET |
| |
| ... |