From 64e87326d9cec962018d3f6274a8f2b84d0274a1 Mon Sep 17 00:00:00 2001 From: Bill Wendling Date: Fri, 16 Jan 2009 19:25:27 +0000 Subject: [PATCH] Add support for non-zero __builtin_return_address values on X86. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@62338 91177308-0d34-0410-b5e6-96231b3b80d8 --- lib/Target/X86/X86ISelLowering.cpp | 25 +++++++++++++++++-------- test/CodeGen/X86/ret-addr.ll | 22 ++++++++++++++++++++++ 2 files changed, 39 insertions(+), 8 deletions(-) create mode 100644 test/CodeGen/X86/ret-addr.ll diff --git a/lib/Target/X86/X86ISelLowering.cpp b/lib/Target/X86/X86ISelLowering.cpp index e39036fb5b7..73bd4b384b5 100644 --- a/lib/Target/X86/X86ISelLowering.cpp +++ b/lib/Target/X86/X86ISelLowering.cpp @@ -1466,7 +1466,7 @@ X86TargetLowering::LowerMemOpCallTo(CallSDNode *TheCall, SelectionDAG &DAG, PseudoSourceValue::getStack(), LocMemOffset); } -/// EmitTailCallLoadRetAddr - Emit a load of return adress if tail call +/// EmitTailCallLoadRetAddr - Emit a load of return address if tail call /// optimization is performed and it is required. SDValue X86TargetLowering::EmitTailCallLoadRetAddr(SelectionDAG &DAG, @@ -1480,8 +1480,9 @@ X86TargetLowering::EmitTailCallLoadRetAddr(SelectionDAG &DAG, // Adjust the Return address stack slot. MVT VT = getPointerTy(); OutRetAddr = getReturnAddressFrameIndex(DAG); + // Load the "old" Return address. - OutRetAddr = DAG.getLoad(VT, Chain,OutRetAddr, NULL, 0); + OutRetAddr = DAG.getLoad(VT, Chain, OutRetAddr, NULL, 0); return SDValue(OutRetAddr.getNode(), 1); } @@ -1949,10 +1950,10 @@ SDValue X86TargetLowering::getReturnAddressFrameIndex(SelectionDAG &DAG) { MachineFunction &MF = DAG.getMachineFunction(); X86MachineFunctionInfo *FuncInfo = MF.getInfo(); int ReturnAddrIndex = FuncInfo->getRAIndex(); - uint64_t SlotSize = TD->getPointerSize(); if (ReturnAddrIndex == 0) { // Set up a frame object for the return address. + uint64_t SlotSize = TD->getPointerSize(); ReturnAddrIndex = MF.getFrameInfo()->CreateFixedObject(SlotSize, -SlotSize); FuncInfo->setRAIndex(ReturnAddrIndex); } @@ -5891,11 +5892,19 @@ X86TargetLowering::LowerINTRINSIC_WO_CHAIN(SDValue Op, SelectionDAG &DAG) { } SDValue X86TargetLowering::LowerRETURNADDR(SDValue Op, SelectionDAG &DAG) { - // Depths > 0 not supported yet! - if (cast(Op.getOperand(0))->getZExtValue() > 0) - return SDValue(); - - // Just load the return address + unsigned Depth = cast(Op.getOperand(0))->getZExtValue(); + + if (Depth > 0) { + SDValue FrameAddr = LowerFRAMEADDR(Op, DAG); + SDValue Offset = + DAG.getConstant(TD->getPointerSize(), + Subtarget->is64Bit() ? MVT::i64 : MVT::i32); + return DAG.getLoad(getPointerTy(), DAG.getEntryNode(), + DAG.getNode(ISD::ADD, getPointerTy(), FrameAddr, Offset), + NULL, 0); + } + + // Just load the return address. SDValue RetAddrFI = getReturnAddressFrameIndex(DAG); return DAG.getLoad(getPointerTy(), DAG.getEntryNode(), RetAddrFI, NULL, 0); } diff --git a/test/CodeGen/X86/ret-addr.ll b/test/CodeGen/X86/ret-addr.ll new file mode 100644 index 00000000000..06a10c6a30f --- /dev/null +++ b/test/CodeGen/X86/ret-addr.ll @@ -0,0 +1,22 @@ +; RUN: llvm-as < %s | llc -disable-fp-elim -march=x86 | not grep xor +; RUN: llvm-as < %s | llc -disable-fp-elim -march=x86-64 | not grep xor + +define i8* @h() nounwind readnone optsize { +entry: + %0 = tail call i8* @llvm.returnaddress(i32 2) ; [#uses=1] + ret i8* %0 +} + +declare i8* @llvm.returnaddress(i32) nounwind readnone + +define i8* @g() nounwind readnone optsize { +entry: + %0 = tail call i8* @llvm.returnaddress(i32 1) ; [#uses=1] + ret i8* %0 +} + +define i8* @f() nounwind readnone optsize { +entry: + %0 = tail call i8* @llvm.returnaddress(i32 0) ; [#uses=1] + ret i8* %0 +}