[llvm] [BPF] Handle unreachable with a kfunc call (PR #131731)

via llvm-commits llvm-commits at lists.llvm.org
Fri Mar 21 18:35:17 PDT 2025


================
@@ -726,6 +733,38 @@ SDValue BPFTargetLowering::LowerATOMIC_LOAD_STORE(SDValue Op,
   return Op;
 }
 
+SDValue BPFTargetLowering::LowerTRAP(SDValue Op, SelectionDAG &DAG) const {
+  MachineFunction &MF = DAG.getMachineFunction();
+  Function &F = MF.getFunction();
+  if (F.hasFnAttribute(Attribute::Naked))
+    return Op;
+
+  TargetLowering::CallLoweringInfo CLI(DAG);
+  SmallVector<SDValue> InVals;
+  SDNode *N = Op.getNode();
+  SDLoc DL(N);
+
+  Module *M = MF.getFunction().getParent();
+  FunctionType *FT = FunctionType::get(Type::getVoidTy(M->getContext()), false);
+  Function *UnreachableHelper = M->getFunction("__unreachable_helper");
+  if (!UnreachableHelper) {
+    Function *NewF = Function::Create(FT, GlobalValue::ExternalWeakLinkage,
+                                      "__unreachable_helper", M);
+    NewF->setDSOLocal(true);
+    NewF->setCallingConv(CallingConv::C);
----------------
eddyz87 wrote:

I see, my example was broken because of calls to `bar()` that needed stack spill, sorry about that.
If changed to something like  [this](https://github.com/user-attachments/files/19399316/test6.ll.txt), there are no spills generated. But this makes me wonder how does it work:

```
$ llc -debug-only=isel -mtriple=bpfel -mcpu=v3 < test6.ll
...
*** MachineFunction at end of ISel ***
# Machine code for function foo: IsSSA, TracksLiveness
Function Live Ins: $w1 in %9, $r3 in %11

bb.0.entry:
  successors: %bb.2(0x80000000), %bb.1(0x00000000); %bb.2(100.00%), %bb.1(0.00%)
  liveins: $w1, $r3
  %11:gpr = COPY $r3
  %9:gpr32 = COPY $w1
  %0:gpr32 = LDW32 %11:gpr, 0 :: (volatile load (s32) from %ir.ap1)
  %1:gpr32 = LDW32 %11:gpr, 4 :: (volatile load (s32) from %ir.bp)
  %2:gpr32 = LDW32 %11:gpr, 8 :: (volatile load (s32) from %ir.cp)
  %3:gpr32 = LDW32 %11:gpr, 12 :: (volatile load (s32) from %ir.dp)
  %4:gpr32 = LDW32 %11:gpr, 16 :: (volatile load (s32) from %ir.ep)
  %5:gpr32 = LDW32 %11:gpr, 20 :: (volatile load (s32) from %ir.fp)
  %6:gpr32 = LDW32 %11:gpr, 24 :: (volatile load (s32) from %ir.gp)
  %7:gpr32 = LDW32 %11:gpr, 28 :: (volatile load (s32) from %ir.hp)
  %8:gpr32 = LDW32 %11:gpr, 32 :: (volatile load (s32) from %ir.kp)
  %12:gpr32 = AND_ri_32 %9:gpr32(tied-def 0), 1
  JNE_ri_32 killed %12:gpr32, 0, %bb.2
  JMP %bb.1

bb.1.flabel:
; predecessors: %bb.0

  ADJCALLSTACKDOWN 0, 0, implicit-def dead $r11, implicit $r11
  JAL &__unreachable_helper, <regmask $r6 $r7 $r8 $r9 $r10 $w6 $w7 $w8 $w9 $w10>, implicit $r11, implicit-def $r11
  ADJCALLSTACKUP 0, 0, implicit-def dead $r11, implicit $r11

bb.2.tlabel:
; predecessors: %bb.0

  %13:gpr32 = ADD_rr_32 %0:gpr32(tied-def 0), %1:gpr32
  %14:gpr32 = ADD_rr_32 %13:gpr32(tied-def 0), %2:gpr32
  %15:gpr32 = ADD_rr_32 %14:gpr32(tied-def 0), %3:gpr32
  %16:gpr32 = ADD_rr_32 %15:gpr32(tied-def 0), %4:gpr32
  %17:gpr32 = ADD_rr_32 %16:gpr32(tied-def 0), %5:gpr32
  %18:gpr32 = ADD_rr_32 %17:gpr32(tied-def 0), %6:gpr32
  %19:gpr32 = ADD_rr_32 %18:gpr32(tied-def 0), %7:gpr32
  %20:gpr32 = ADD_rr_32 %19:gpr32(tied-def 0), %8:gpr32
  ADJCALLSTACKDOWN 0, 0, implicit-def dead $r11, implicit $r11
  $w1 = COPY %20:gpr32
  JAL @buz, <regmask $r6 $r7 $r8 $r9 $r10 $w6 $w7 $w8 $w9 $w10>, implicit $r11, implicit $w1, implicit-def $r11
  ADJCALLSTACKUP 0, 0, implicit-def dead $r11, implicit $r11
  RET
...
```

Note that basic block `bb.1.flabel` does not have terminator after the `__unreachable_helper` call. Is it ok?
Is it necessary to mark to add `CLI.DoesNotReturn = true` in `LowerTRAP`?


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


More information about the llvm-commits mailing list