| //===-- CodeGen/AsmPrinter/WinException.cpp - Dwarf Exception Impl ------===// |
| // |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| // See https://llvm.org/LICENSE.txt for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // This file contains support for writing Win64 exception info into asm files. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "WinException.h" |
| #include "llvm/ADT/Twine.h" |
| #include "llvm/BinaryFormat/COFF.h" |
| #include "llvm/BinaryFormat/Dwarf.h" |
| #include "llvm/CodeGen/AsmPrinter.h" |
| #include "llvm/CodeGen/MachineFrameInfo.h" |
| #include "llvm/CodeGen/MachineFunction.h" |
| #include "llvm/CodeGen/MachineModuleInfo.h" |
| #include "llvm/CodeGen/TargetFrameLowering.h" |
| #include "llvm/CodeGen/TargetLowering.h" |
| #include "llvm/CodeGen/TargetSubtargetInfo.h" |
| #include "llvm/CodeGen/WinEHFuncInfo.h" |
| #include "llvm/IR/DataLayout.h" |
| #include "llvm/IR/Mangler.h" |
| #include "llvm/IR/Module.h" |
| #include "llvm/MC/MCAsmInfo.h" |
| #include "llvm/MC/MCContext.h" |
| #include "llvm/MC/MCExpr.h" |
| #include "llvm/MC/MCSection.h" |
| #include "llvm/MC/MCStreamer.h" |
| #include "llvm/MC/MCSymbol.h" |
| #include "llvm/Support/ErrorHandling.h" |
| #include "llvm/Support/FormattedStream.h" |
| #include "llvm/Target/TargetLoweringObjectFile.h" |
| #include "llvm/Target/TargetOptions.h" |
| using namespace llvm; |
| |
| WinException::WinException(AsmPrinter *A) : EHStreamer(A) { |
| // MSVC's EH tables are always composed of 32-bit words. All known 64-bit |
| // platforms use an imagerel32 relocation to refer to symbols. |
| useImageRel32 = (A->getDataLayout().getPointerSizeInBits() == 64); |
| isAArch64 = Asm->TM.getTargetTriple().isAArch64(); |
| } |
| |
| WinException::~WinException() {} |
| |
| /// endModule - Emit all exception information that should come after the |
| /// content. |
| void WinException::endModule() { |
| auto &OS = *Asm->OutStreamer; |
| const Module *M = MMI->getModule(); |
| for (const Function &F : *M) |
| if (F.hasFnAttribute("safeseh")) |
| OS.EmitCOFFSafeSEH(Asm->getSymbol(&F)); |
| } |
| |
| void WinException::beginFunction(const MachineFunction *MF) { |
| shouldEmitMoves = shouldEmitPersonality = shouldEmitLSDA = false; |
| |
| // If any landing pads survive, we need an EH table. |
| bool hasLandingPads = !MF->getLandingPads().empty(); |
| bool hasEHFunclets = MF->hasEHFunclets(); |
| |
| const Function &F = MF->getFunction(); |
| |
| shouldEmitMoves = Asm->needsSEHMoves() && MF->hasWinCFI(); |
| |
| const TargetLoweringObjectFile &TLOF = Asm->getObjFileLowering(); |
| unsigned PerEncoding = TLOF.getPersonalityEncoding(); |
| |
| EHPersonality Per = EHPersonality::Unknown; |
| const Function *PerFn = nullptr; |
| if (F.hasPersonalityFn()) { |
| PerFn = dyn_cast<Function>(F.getPersonalityFn()->stripPointerCasts()); |
| Per = classifyEHPersonality(PerFn); |
| } |
| |
| bool forceEmitPersonality = F.hasPersonalityFn() && |
| !isNoOpWithoutInvoke(Per) && |
| F.needsUnwindTableEntry(); |
| |
| shouldEmitPersonality = |
| forceEmitPersonality || ((hasLandingPads || hasEHFunclets) && |
| PerEncoding != dwarf::DW_EH_PE_omit && PerFn); |
| |
| unsigned LSDAEncoding = TLOF.getLSDAEncoding(); |
| shouldEmitLSDA = shouldEmitPersonality && |
| LSDAEncoding != dwarf::DW_EH_PE_omit; |
| |
| // If we're not using CFI, we don't want the CFI or the personality, but we |
| // might want EH tables if we had EH pads. |
| if (!Asm->MAI->usesWindowsCFI()) { |
| if (Per == EHPersonality::MSVC_X86SEH && !hasEHFunclets) { |
| // If this is 32-bit SEH and we don't have any funclets (really invokes), |
| // make sure we emit the parent offset label. Some unreferenced filter |
| // functions may still refer to it. |
| const WinEHFuncInfo &FuncInfo = *MF->getWinEHFuncInfo(); |
| StringRef FLinkageName = |
| GlobalValue::dropLLVMManglingEscape(MF->getFunction().getName()); |
| emitEHRegistrationOffsetLabel(FuncInfo, FLinkageName); |
| } |
| shouldEmitLSDA = hasEHFunclets; |
| shouldEmitPersonality = false; |
| return; |
| } |
| |
| beginFunclet(MF->front(), Asm->CurrentFnSym); |
| } |
| |
| void WinException::markFunctionEnd() { |
| if (isAArch64 && CurrentFuncletEntry && |
| (shouldEmitMoves || shouldEmitPersonality)) |
| Asm->OutStreamer->EmitWinCFIFuncletOrFuncEnd(); |
| } |
| |
| /// endFunction - Gather and emit post-function exception information. |
| /// |
| void WinException::endFunction(const MachineFunction *MF) { |
| if (!shouldEmitPersonality && !shouldEmitMoves && !shouldEmitLSDA) |
| return; |
| |
| const Function &F = MF->getFunction(); |
| EHPersonality Per = EHPersonality::Unknown; |
| if (F.hasPersonalityFn()) |
| Per = classifyEHPersonality(F.getPersonalityFn()->stripPointerCasts()); |
| |
| // Get rid of any dead landing pads if we're not using funclets. In funclet |
| // schemes, the landing pad is not actually reachable. It only exists so |
| // that we can emit the right table data. |
| if (!isFuncletEHPersonality(Per)) { |
| MachineFunction *NonConstMF = const_cast<MachineFunction*>(MF); |
| NonConstMF->tidyLandingPads(); |
| } |
| |
| endFuncletImpl(); |
| |
| // endFunclet will emit the necessary .xdata tables for x64 SEH. |
| if (Per == EHPersonality::MSVC_Win64SEH && MF->hasEHFunclets()) |
| return; |
| |
| if (shouldEmitPersonality || shouldEmitLSDA) { |
| Asm->OutStreamer->PushSection(); |
| |
| // Just switch sections to the right xdata section. |
| MCSection *XData = Asm->OutStreamer->getAssociatedXDataSection( |
| Asm->OutStreamer->getCurrentSectionOnly()); |
| Asm->OutStreamer->SwitchSection(XData); |
| |
| // Emit the tables appropriate to the personality function in use. If we |
| // don't recognize the personality, assume it uses an Itanium-style LSDA. |
| if (Per == EHPersonality::MSVC_Win64SEH) |
| emitCSpecificHandlerTable(MF); |
| else if (Per == EHPersonality::MSVC_X86SEH) |
| emitExceptHandlerTable(MF); |
| else if (Per == EHPersonality::MSVC_CXX) |
| emitCXXFrameHandler3Table(MF); |
| else if (Per == EHPersonality::CoreCLR) |
| emitCLRExceptionTable(MF); |
| else |
| emitExceptionTable(); |
| |
| Asm->OutStreamer->PopSection(); |
| } |
| } |
| |
| /// Retrieve the MCSymbol for a GlobalValue or MachineBasicBlock. |
| static MCSymbol *getMCSymbolForMBB(AsmPrinter *Asm, |
| const MachineBasicBlock *MBB) { |
| if (!MBB) |
| return nullptr; |
| |
| assert(MBB->isEHFuncletEntry()); |
| |
| // Give catches and cleanups a name based off of their parent function and |
| // their funclet entry block's number. |
| const MachineFunction *MF = MBB->getParent(); |
| const Function &F = MF->getFunction(); |
| StringRef FuncLinkageName = GlobalValue::dropLLVMManglingEscape(F.getName()); |
| MCContext &Ctx = MF->getContext(); |
| StringRef HandlerPrefix = MBB->isCleanupFuncletEntry() ? "dtor" : "catch"; |
| return Ctx.getOrCreateSymbol("?" + HandlerPrefix + "$" + |
| Twine(MBB->getNumber()) + "@?0?" + |
| FuncLinkageName + "@4HA"); |
| } |
| |
| void WinException::beginFunclet(const MachineBasicBlock &MBB, |
| MCSymbol *Sym) { |
| CurrentFuncletEntry = &MBB; |
| |
| const Function &F = Asm->MF->getFunction(); |
| // If a symbol was not provided for the funclet, invent one. |
| if (!Sym) { |
| Sym = getMCSymbolForMBB(Asm, &MBB); |
| |
| // Describe our funclet symbol as a function with internal linkage. |
| Asm->OutStreamer->BeginCOFFSymbolDef(Sym); |
| Asm->OutStreamer->EmitCOFFSymbolStorageClass(COFF::IMAGE_SYM_CLASS_STATIC); |
| Asm->OutStreamer->EmitCOFFSymbolType(COFF::IMAGE_SYM_DTYPE_FUNCTION |
| << COFF::SCT_COMPLEX_TYPE_SHIFT); |
| Asm->OutStreamer->EndCOFFSymbolDef(); |
| |
| // We want our funclet's entry point to be aligned such that no nops will be |
| // present after the label. |
| Asm->EmitAlignment(std::max(Asm->MF->getAlignment(), MBB.getAlignment()), |
| &F); |
| |
| // Now that we've emitted the alignment directive, point at our funclet. |
| Asm->OutStreamer->EmitLabel(Sym); |
| } |
| |
| // Mark 'Sym' as starting our funclet. |
| if (shouldEmitMoves || shouldEmitPersonality) { |
| CurrentFuncletTextSection = Asm->OutStreamer->getCurrentSectionOnly(); |
| Asm->OutStreamer->EmitWinCFIStartProc(Sym); |
| } |
| |
| if (shouldEmitPersonality) { |
| const TargetLoweringObjectFile &TLOF = Asm->getObjFileLowering(); |
| const Function *PerFn = nullptr; |
| |
| // Determine which personality routine we are using for this funclet. |
| if (F.hasPersonalityFn()) |
| PerFn = dyn_cast<Function>(F.getPersonalityFn()->stripPointerCasts()); |
| const MCSymbol *PersHandlerSym = |
| TLOF.getCFIPersonalitySymbol(PerFn, Asm->TM, MMI); |
| |
| // Do not emit a .seh_handler directives for cleanup funclets. |
| // FIXME: This means cleanup funclets cannot handle exceptions. Given that |
| // Clang doesn't produce EH constructs inside cleanup funclets and LLVM's |
| // inliner doesn't allow inlining them, this isn't a major problem in |
| // practice. |
| if (!CurrentFuncletEntry->isCleanupFuncletEntry()) |
| Asm->OutStreamer->EmitWinEHHandler(PersHandlerSym, true, true); |
| } |
| } |
| |
| void WinException::endFunclet() { |
| if (isAArch64 && CurrentFuncletEntry && |
| (shouldEmitMoves || shouldEmitPersonality)) { |
| Asm->OutStreamer->SwitchSection(CurrentFuncletTextSection); |
| Asm->OutStreamer->EmitWinCFIFuncletOrFuncEnd(); |
| } |
| endFuncletImpl(); |
| } |
| |
| void WinException::endFuncletImpl() { |
| // No funclet to process? Great, we have nothing to do. |
| if (!CurrentFuncletEntry) |
| return; |
| |
| const MachineFunction *MF = Asm->MF; |
| if (shouldEmitMoves || shouldEmitPersonality) { |
| const Function &F = MF->getFunction(); |
| EHPersonality Per = EHPersonality::Unknown; |
| if (F.hasPersonalityFn()) |
| Per = classifyEHPersonality(F.getPersonalityFn()->stripPointerCasts()); |
| |
| // On funclet exit, we emit a fake "function" end marker, so that the call |
| // to EmitWinEHHandlerData below can calculate the size of the funclet or |
| // function. |
| if (isAArch64) { |
| MCSection *XData = Asm->OutStreamer->getAssociatedXDataSection( |
| Asm->OutStreamer->getCurrentSectionOnly()); |
| Asm->OutStreamer->SwitchSection(XData); |
| } |
| |
| // Emit an UNWIND_INFO struct describing the prologue. |
| Asm->OutStreamer->EmitWinEHHandlerData(); |
| |
| if (Per == EHPersonality::MSVC_CXX && shouldEmitPersonality && |
| !CurrentFuncletEntry->isCleanupFuncletEntry()) { |
| // If this is a C++ catch funclet (or the parent function), |
| // emit a reference to the LSDA for the parent function. |
| StringRef FuncLinkageName = GlobalValue::dropLLVMManglingEscape(F.getName()); |
| MCSymbol *FuncInfoXData = Asm->OutContext.getOrCreateSymbol( |
| Twine("$cppxdata$", FuncLinkageName)); |
| Asm->OutStreamer->EmitValue(create32bitRef(FuncInfoXData), 4); |
| } else if (Per == EHPersonality::MSVC_Win64SEH && MF->hasEHFunclets() && |
| !CurrentFuncletEntry->isEHFuncletEntry()) { |
| // If this is the parent function in Win64 SEH, emit the LSDA immediately |
| // following .seh_handlerdata. |
| emitCSpecificHandlerTable(MF); |
| } |
| |
| // Switch back to the funclet start .text section now that we are done |
| // writing to .xdata, and emit an .seh_endproc directive to mark the end of |
| // the function. |
| Asm->OutStreamer->SwitchSection(CurrentFuncletTextSection); |
| Asm->OutStreamer->EmitWinCFIEndProc(); |
| } |
| |
| // Let's make sure we don't try to end the same funclet twice. |
| CurrentFuncletEntry = nullptr; |
| } |
| |
| const MCExpr *WinException::create32bitRef(const MCSymbol *Value) { |
| if (!Value) |
| return MCConstantExpr::create(0, Asm->OutContext); |
| return MCSymbolRefExpr::create(Value, useImageRel32 |
| ? MCSymbolRefExpr::VK_COFF_IMGREL32 |
| : MCSymbolRefExpr::VK_None, |
| Asm->OutContext); |
| } |
| |
| const MCExpr *WinException::create32bitRef(const GlobalValue *GV) { |
| if (!GV) |
| return MCConstantExpr::create(0, Asm->OutContext); |
| return create32bitRef(Asm->getSymbol(GV)); |
| } |
| |
| const MCExpr *WinException::getLabel(const MCSymbol *Label) { |
| if (isAArch64) |
| return MCSymbolRefExpr::create(Label, MCSymbolRefExpr::VK_COFF_IMGREL32, |
| Asm->OutContext); |
| return MCBinaryExpr::createAdd(create32bitRef(Label), |
| MCConstantExpr::create(1, Asm->OutContext), |
| Asm->OutContext); |
| } |
| |
| const MCExpr *WinException::getOffset(const MCSymbol *OffsetOf, |
| const MCSymbol *OffsetFrom) { |
| return MCBinaryExpr::createSub( |
| MCSymbolRefExpr::create(OffsetOf, Asm->OutContext), |
| MCSymbolRefExpr::create(OffsetFrom, Asm->OutContext), Asm->OutContext); |
| } |
| |
| const MCExpr *WinException::getOffsetPlusOne(const MCSymbol *OffsetOf, |
| const MCSymbol *OffsetFrom) { |
| return MCBinaryExpr::createAdd(getOffset(OffsetOf, OffsetFrom), |
| MCConstantExpr::create(1, Asm->OutContext), |
| Asm->OutContext); |
| } |
| |
| int WinException::getFrameIndexOffset(int FrameIndex, |
| const WinEHFuncInfo &FuncInfo) { |
| const TargetFrameLowering &TFI = *Asm->MF->getSubtarget().getFrameLowering(); |
| unsigned UnusedReg; |
| if (Asm->MAI->usesWindowsCFI()) { |
| int Offset = |
| TFI.getFrameIndexReferencePreferSP(*Asm->MF, FrameIndex, UnusedReg, |
| /*IgnoreSPUpdates*/ true); |
| assert(UnusedReg == |
| Asm->MF->getSubtarget() |
| .getTargetLowering() |
| ->getStackPointerRegisterToSaveRestore()); |
| return Offset; |
| } |
| |
| // For 32-bit, offsets should be relative to the end of the EH registration |
| // node. For 64-bit, it's relative to SP at the end of the prologue. |
| assert(FuncInfo.EHRegNodeEndOffset != INT_MAX); |
| int Offset = TFI.getFrameIndexReference(*Asm->MF, FrameIndex, UnusedReg); |
| Offset += FuncInfo.EHRegNodeEndOffset; |
| return Offset; |
| } |
| |
| namespace { |
| |
| /// Top-level state used to represent unwind to caller |
| const int NullState = -1; |
| |
| struct InvokeStateChange { |
| /// EH Label immediately after the last invoke in the previous state, or |
| /// nullptr if the previous state was the null state. |
| const MCSymbol *PreviousEndLabel; |
| |
| /// EH label immediately before the first invoke in the new state, or nullptr |
| /// if the new state is the null state. |
| const MCSymbol *NewStartLabel; |
| |
| /// State of the invoke following NewStartLabel, or NullState to indicate |
| /// the presence of calls which may unwind to caller. |
| int NewState; |
| }; |
| |
| /// Iterator that reports all the invoke state changes in a range of machine |
| /// basic blocks. Changes to the null state are reported whenever a call that |
| /// may unwind to caller is encountered. The MBB range is expected to be an |
| /// entire function or funclet, and the start and end of the range are treated |
| /// as being in the NullState even if there's not an unwind-to-caller call |
| /// before the first invoke or after the last one (i.e., the first state change |
| /// reported is the first change to something other than NullState, and a |
| /// change back to NullState is always reported at the end of iteration). |
| class InvokeStateChangeIterator { |
| InvokeStateChangeIterator(const WinEHFuncInfo &EHInfo, |
| MachineFunction::const_iterator MFI, |
| MachineFunction::const_iterator MFE, |
| MachineBasicBlock::const_iterator MBBI, |
| int BaseState) |
| : EHInfo(EHInfo), MFI(MFI), MFE(MFE), MBBI(MBBI), BaseState(BaseState) { |
| LastStateChange.PreviousEndLabel = nullptr; |
| LastStateChange.NewStartLabel = nullptr; |
| LastStateChange.NewState = BaseState; |
| scan(); |
| } |
| |
| public: |
| static iterator_range<InvokeStateChangeIterator> |
| range(const WinEHFuncInfo &EHInfo, MachineFunction::const_iterator Begin, |
| MachineFunction::const_iterator End, int BaseState = NullState) { |
| // Reject empty ranges to simplify bookkeeping by ensuring that we can get |
| // the end of the last block. |
| assert(Begin != End); |
| auto BlockBegin = Begin->begin(); |
| auto BlockEnd = std::prev(End)->end(); |
| return make_range( |
| InvokeStateChangeIterator(EHInfo, Begin, End, BlockBegin, BaseState), |
| InvokeStateChangeIterator(EHInfo, End, End, BlockEnd, BaseState)); |
| } |
| |
| // Iterator methods. |
| bool operator==(const InvokeStateChangeIterator &O) const { |
| assert(BaseState == O.BaseState); |
| // Must be visiting same block. |
| if (MFI != O.MFI) |
| return false; |
| // Must be visiting same isntr. |
| if (MBBI != O.MBBI) |
| return false; |
| // At end of block/instr iteration, we can still have two distinct states: |
| // one to report the final EndLabel, and another indicating the end of the |
| // state change iteration. Check for CurrentEndLabel equality to |
| // distinguish these. |
| return CurrentEndLabel == O.CurrentEndLabel; |
| } |
| |
| bool operator!=(const InvokeStateChangeIterator &O) const { |
| return !operator==(O); |
| } |
| InvokeStateChange &operator*() { return LastStateChange; } |
| InvokeStateChange *operator->() { return &LastStateChange; } |
| InvokeStateChangeIterator &operator++() { return scan(); } |
| |
| private: |
| InvokeStateChangeIterator &scan(); |
| |
| const WinEHFuncInfo &EHInfo; |
| const MCSymbol *CurrentEndLabel = nullptr; |
| MachineFunction::const_iterator MFI; |
| MachineFunction::const_iterator MFE; |
| MachineBasicBlock::const_iterator MBBI; |
| InvokeStateChange LastStateChange; |
| bool VisitingInvoke = false; |
| int BaseState; |
| }; |
| |
| } // end anonymous namespace |
| |
| InvokeStateChangeIterator &InvokeStateChangeIterator::scan() { |
| bool IsNewBlock = false; |
| for (; MFI != MFE; ++MFI, IsNewBlock = true) { |
| if (IsNewBlock) |
| MBBI = MFI->begin(); |
| for (auto MBBE = MFI->end(); MBBI != MBBE; ++MBBI) { |
| const MachineInstr &MI = *MBBI; |
| if (!VisitingInvoke && LastStateChange.NewState != BaseState && |
| MI.isCall() && !EHStreamer::callToNoUnwindFunction(&MI)) { |
| // Indicate a change of state to the null state. We don't have |
| // start/end EH labels handy but the caller won't expect them for |
| // null state regions. |
| LastStateChange.PreviousEndLabel = CurrentEndLabel; |
| LastStateChange.NewStartLabel = nullptr; |
| LastStateChange.NewState = BaseState; |
| CurrentEndLabel = nullptr; |
| // Don't re-visit this instr on the next scan |
| ++MBBI; |
| return *this; |
| } |
| |
| // All other state changes are at EH labels before/after invokes. |
| if (!MI.isEHLabel()) |
| continue; |
| MCSymbol *Label = MI.getOperand(0).getMCSymbol(); |
| if (Label == CurrentEndLabel) { |
| VisitingInvoke = false; |
| continue; |
| } |
| auto InvokeMapIter = EHInfo.LabelToStateMap.find(Label); |
| // Ignore EH labels that aren't the ones inserted before an invoke |
| if (InvokeMapIter == EHInfo.LabelToStateMap.end()) |
| continue; |
| auto &StateAndEnd = InvokeMapIter->second; |
| int NewState = StateAndEnd.first; |
| // Keep track of the fact that we're between EH start/end labels so |
| // we know not to treat the inoke we'll see as unwinding to caller. |
| VisitingInvoke = true; |
| if (NewState == LastStateChange.NewState) { |
| // The state isn't actually changing here. Record the new end and |
| // keep going. |
| CurrentEndLabel = StateAndEnd.second; |
| continue; |
| } |
| // Found a state change to report |
| LastStateChange.PreviousEndLabel = CurrentEndLabel; |
| LastStateChange.NewStartLabel = Label; |
| LastStateChange.NewState = NewState; |
| // Start keeping track of the new current end |
| CurrentEndLabel = StateAndEnd.second; |
| // Don't re-visit this instr on the next scan |
| ++MBBI; |
| return *this; |
| } |
| } |
| // Iteration hit the end of the block range. |
| if (LastStateChange.NewState != BaseState) { |
| // Report the end of the last new state |
| LastStateChange.PreviousEndLabel = CurrentEndLabel; |
| LastStateChange.NewStartLabel = nullptr; |
| LastStateChange.NewState = BaseState; |
| // Leave CurrentEndLabel non-null to distinguish this state from end. |
| assert(CurrentEndLabel != nullptr); |
| return *this; |
| } |
| // We've reported all state changes and hit the end state. |
| CurrentEndLabel = nullptr; |
| return *this; |
| } |
| |
| /// Emit the language-specific data that __C_specific_handler expects. This |
| /// handler lives in the x64 Microsoft C runtime and allows catching or cleaning |
| /// up after faults with __try, __except, and __finally. The typeinfo values |
| /// are not really RTTI data, but pointers to filter functions that return an |
| /// integer (1, 0, or -1) indicating how to handle the exception. For __finally |
| /// blocks and other cleanups, the landing pad label is zero, and the filter |
| /// function is actually a cleanup handler with the same prototype. A catch-all |
| /// entry is modeled with a null filter function field and a non-zero landing |
| /// pad label. |
| /// |
| /// Possible filter function return values: |
| /// EXCEPTION_EXECUTE_HANDLER (1): |
| /// Jump to the landing pad label after cleanups. |
| /// EXCEPTION_CONTINUE_SEARCH (0): |
| /// Continue searching this table or continue unwinding. |
| /// EXCEPTION_CONTINUE_EXECUTION (-1): |
| /// Resume execution at the trapping PC. |
| /// |
| /// Inferred table structure: |
| /// struct Table { |
| /// int NumEntries; |
| /// struct Entry { |
| /// imagerel32 LabelStart; |
| /// imagerel32 LabelEnd; |
| /// imagerel32 FilterOrFinally; // One means catch-all. |
| /// imagerel32 LabelLPad; // Zero means __finally. |
| /// } Entries[NumEntries]; |
| /// }; |
| void WinException::emitCSpecificHandlerTable(const MachineFunction *MF) { |
| auto &OS = *Asm->OutStreamer; |
| MCContext &Ctx = Asm->OutContext; |
| const WinEHFuncInfo &FuncInfo = *MF->getWinEHFuncInfo(); |
| |
| bool VerboseAsm = OS.isVerboseAsm(); |
| auto AddComment = [&](const Twine &Comment) { |
| if (VerboseAsm) |
| OS.AddComment(Comment); |
| }; |
| |
| if (!isAArch64) { |
| // Emit a label assignment with the SEH frame offset so we can use it for |
| // llvm.eh.recoverfp. |
| StringRef FLinkageName = |
| GlobalValue::dropLLVMManglingEscape(MF->getFunction().getName()); |
| MCSymbol *ParentFrameOffset = |
| Ctx.getOrCreateParentFrameOffsetSymbol(FLinkageName); |
| const MCExpr *MCOffset = |
| MCConstantExpr::create(FuncInfo.SEHSetFrameOffset, Ctx); |
| Asm->OutStreamer->EmitAssignment(ParentFrameOffset, MCOffset); |
| } |
| |
| // Use the assembler to compute the number of table entries through label |
| // difference and division. |
| MCSymbol *TableBegin = |
| Ctx.createTempSymbol("lsda_begin", /*AlwaysAddSuffix=*/true); |
| MCSymbol *TableEnd = |
| Ctx.createTempSymbol("lsda_end", /*AlwaysAddSuffix=*/true); |
| const MCExpr *LabelDiff = getOffset(TableEnd, TableBegin); |
| const MCExpr *EntrySize = MCConstantExpr::create(16, Ctx); |
| const MCExpr *EntryCount = MCBinaryExpr::createDiv(LabelDiff, EntrySize, Ctx); |
| AddComment("Number of call sites"); |
| OS.EmitValue(EntryCount, 4); |
| |
| OS.EmitLabel(TableBegin); |
| |
| // Iterate over all the invoke try ranges. Unlike MSVC, LLVM currently only |
| // models exceptions from invokes. LLVM also allows arbitrary reordering of |
| // the code, so our tables end up looking a bit different. Rather than |
| // trying to match MSVC's tables exactly, we emit a denormalized table. For |
| // each range of invokes in the same state, we emit table entries for all |
| // the actions that would be taken in that state. This means our tables are |
| // slightly bigger, which is OK. |
| const MCSymbol *LastStartLabel = nullptr; |
| int LastEHState = -1; |
| // Break out before we enter into a finally funclet. |
| // FIXME: We need to emit separate EH tables for cleanups. |
| MachineFunction::const_iterator End = MF->end(); |
| MachineFunction::const_iterator Stop = std::next(MF->begin()); |
| while (Stop != End && !Stop->isEHFuncletEntry()) |
| ++Stop; |
| for (const auto &StateChange : |
| InvokeStateChangeIterator::range(FuncInfo, MF->begin(), Stop)) { |
| // Emit all the actions for the state we just transitioned out of |
| // if it was not the null state |
| if (LastEHState != -1) |
| emitSEHActionsForRange(FuncInfo, LastStartLabel, |
| StateChange.PreviousEndLabel, LastEHState); |
| LastStartLabel = StateChange.NewStartLabel; |
| LastEHState = StateChange.NewState; |
| } |
| |
| OS.EmitLabel(TableEnd); |
| } |
| |
| void WinException::emitSEHActionsForRange(const WinEHFuncInfo &FuncInfo, |
| const MCSymbol *BeginLabel, |
| const MCSymbol *EndLabel, int State) { |
| auto &OS = *Asm->OutStreamer; |
| MCContext &Ctx = Asm->OutContext; |
| bool VerboseAsm = OS.isVerboseAsm(); |
| auto AddComment = [&](const Twine &Comment) { |
| if (VerboseAsm) |
| OS.AddComment(Comment); |
| }; |
| |
| assert(BeginLabel && EndLabel); |
| while (State != -1) { |
| const SEHUnwindMapEntry &UME = FuncInfo.SEHUnwindMap[State]; |
| const MCExpr *FilterOrFinally; |
| const MCExpr *ExceptOrNull; |
| auto *Handler = UME.Handler.get<MachineBasicBlock *>(); |
| if (UME.IsFinally) { |
| FilterOrFinally = create32bitRef(getMCSymbolForMBB(Asm, Handler)); |
| ExceptOrNull = MCConstantExpr::create(0, Ctx); |
| } else { |
| // For an except, the filter can be 1 (catch-all) or a function |
| // label. |
| FilterOrFinally = UME.Filter ? create32bitRef(UME.Filter) |
| : MCConstantExpr::create(1, Ctx); |
| ExceptOrNull = create32bitRef(Handler->getSymbol()); |
| } |
| |
| AddComment("LabelStart"); |
| OS.EmitValue(getLabel(BeginLabel), 4); |
| AddComment("LabelEnd"); |
| OS.EmitValue(getLabel(EndLabel), 4); |
| AddComment(UME.IsFinally ? "FinallyFunclet" : UME.Filter ? "FilterFunction" |
| : "CatchAll"); |
| OS.EmitValue(FilterOrFinally, 4); |
| AddComment(UME.IsFinally ? "Null" : "ExceptionHandler"); |
| OS.EmitValue(ExceptOrNull, 4); |
| |
| assert(UME.ToState < State && "states should decrease"); |
| State = UME.ToState; |
| } |
| } |
| |
| void WinException::emitCXXFrameHandler3Table(const MachineFunction *MF) { |
| const Function &F = MF->getFunction(); |
| auto &OS = *Asm->OutStreamer; |
| const WinEHFuncInfo &FuncInfo = *MF->getWinEHFuncInfo(); |
| |
| StringRef FuncLinkageName = GlobalValue::dropLLVMManglingEscape(F.getName()); |
| |
| SmallVector<std::pair<const MCExpr *, int>, 4> IPToStateTable; |
| MCSymbol *FuncInfoXData = nullptr; |
| if (shouldEmitPersonality) { |
| // If we're 64-bit, emit a pointer to the C++ EH data, and build a map from |
| // IPs to state numbers. |
| FuncInfoXData = |
| Asm->OutContext.getOrCreateSymbol(Twine("$cppxdata$", FuncLinkageName)); |
| computeIP2StateTable(MF, FuncInfo, IPToStateTable); |
| } else { |
| FuncInfoXData = Asm->OutContext.getOrCreateLSDASymbol(FuncLinkageName); |
| } |
| |
| int UnwindHelpOffset = 0; |
| if (Asm->MAI->usesWindowsCFI()) |
| UnwindHelpOffset = |
| getFrameIndexOffset(FuncInfo.UnwindHelpFrameIdx, FuncInfo); |
| |
| MCSymbol *UnwindMapXData = nullptr; |
| MCSymbol *TryBlockMapXData = nullptr; |
| MCSymbol *IPToStateXData = nullptr; |
| if (!FuncInfo.CxxUnwindMap.empty()) |
| UnwindMapXData = Asm->OutContext.getOrCreateSymbol( |
| Twine("$stateUnwindMap$", FuncLinkageName)); |
| if (!FuncInfo.TryBlockMap.empty()) |
| TryBlockMapXData = |
| Asm->OutContext.getOrCreateSymbol(Twine("$tryMap$", FuncLinkageName)); |
| if (!IPToStateTable.empty()) |
| IPToStateXData = |
| Asm->OutContext.getOrCreateSymbol(Twine("$ip2state$", FuncLinkageName)); |
| |
| bool VerboseAsm = OS.isVerboseAsm(); |
| auto AddComment = [&](const Twine &Comment) { |
| if (VerboseAsm) |
| OS.AddComment(Comment); |
| }; |
| |
| // FuncInfo { |
| // uint32_t MagicNumber |
| // int32_t MaxState; |
| // UnwindMapEntry *UnwindMap; |
| // uint32_t NumTryBlocks; |
| // TryBlockMapEntry *TryBlockMap; |
| // uint32_t IPMapEntries; // always 0 for x86 |
| // IPToStateMapEntry *IPToStateMap; // always 0 for x86 |
| // uint32_t UnwindHelp; // non-x86 only |
| // ESTypeList *ESTypeList; |
| // int32_t EHFlags; |
| // } |
| // EHFlags & 1 -> Synchronous exceptions only, no async exceptions. |
| // EHFlags & 2 -> ??? |
| // EHFlags & 4 -> The function is noexcept(true), unwinding can't continue. |
| OS.EmitValueToAlignment(4); |
| OS.EmitLabel(FuncInfoXData); |
| |
| AddComment("MagicNumber"); |
| OS.EmitIntValue(0x19930522, 4); |
| |
| AddComment("MaxState"); |
| OS.EmitIntValue(FuncInfo.CxxUnwindMap.size(), 4); |
| |
| AddComment("UnwindMap"); |
| OS.EmitValue(create32bitRef(UnwindMapXData), 4); |
| |
| AddComment("NumTryBlocks"); |
| OS.EmitIntValue(FuncInfo.TryBlockMap.size(), 4); |
| |
| AddComment("TryBlockMap"); |
| OS.EmitValue(create32bitRef(TryBlockMapXData), 4); |
| |
| AddComment("IPMapEntries"); |
| OS.EmitIntValue(IPToStateTable.size(), 4); |
| |
| AddComment("IPToStateXData"); |
| OS.EmitValue(create32bitRef(IPToStateXData), 4); |
| |
| if (Asm->MAI->usesWindowsCFI()) { |
| AddComment("UnwindHelp"); |
| OS.EmitIntValue(UnwindHelpOffset, 4); |
| } |
| |
| AddComment("ESTypeList"); |
| OS.EmitIntValue(0, 4); |
| |
| AddComment("EHFlags"); |
| OS.EmitIntValue(1, 4); |
| |
| // UnwindMapEntry { |
| // int32_t ToState; |
| // void (*Action)(); |
| // }; |
| if (UnwindMapXData) { |
| OS.EmitLabel(UnwindMapXData); |
| for (const CxxUnwindMapEntry &UME : FuncInfo.CxxUnwindMap) { |
| MCSymbol *CleanupSym = |
| getMCSymbolForMBB(Asm, UME.Cleanup.dyn_cast<MachineBasicBlock *>()); |
| AddComment("ToState"); |
| OS.EmitIntValue(UME.ToState, 4); |
| |
| AddComment("Action"); |
| OS.EmitValue(create32bitRef(CleanupSym), 4); |
| } |
| } |
| |
| // TryBlockMap { |
| // int32_t TryLow; |
| // int32_t TryHigh; |
| // int32_t CatchHigh; |
| // int32_t NumCatches; |
| // HandlerType *HandlerArray; |
| // }; |
| if (TryBlockMapXData) { |
| OS.EmitLabel(TryBlockMapXData); |
| SmallVector<MCSymbol *, 1> HandlerMaps; |
| for (size_t I = 0, E = FuncInfo.TryBlockMap.size(); I != E; ++I) { |
| const WinEHTryBlockMapEntry &TBME = FuncInfo.TryBlockMap[I]; |
| |
| MCSymbol *HandlerMapXData = nullptr; |
| if (!TBME.HandlerArray.empty()) |
| HandlerMapXData = |
| Asm->OutContext.getOrCreateSymbol(Twine("$handlerMap$") |
| .concat(Twine(I)) |
| .concat("$") |
| .concat(FuncLinkageName)); |
| HandlerMaps.push_back(HandlerMapXData); |
| |
| // TBMEs should form intervals. |
| assert(0 <= TBME.TryLow && "bad trymap interval"); |
| assert(TBME.TryLow <= TBME.TryHigh && "bad trymap interval"); |
| assert(TBME.TryHigh < TBME.CatchHigh && "bad trymap interval"); |
| assert(TBME.CatchHigh < int(FuncInfo.CxxUnwindMap.size()) && |
| "bad trymap interval"); |
| |
| AddComment("TryLow"); |
| OS.EmitIntValue(TBME.TryLow, 4); |
| |
| AddComment("TryHigh"); |
| OS.EmitIntValue(TBME.TryHigh, 4); |
| |
| AddComment("CatchHigh"); |
| OS.EmitIntValue(TBME.CatchHigh, 4); |
| |
| AddComment("NumCatches"); |
| OS.EmitIntValue(TBME.HandlerArray.size(), 4); |
| |
| AddComment("HandlerArray"); |
| OS.EmitValue(create32bitRef(HandlerMapXData), 4); |
| } |
| |
| // All funclets use the same parent frame offset currently. |
| unsigned ParentFrameOffset = 0; |
| if (shouldEmitPersonality) { |
| const TargetFrameLowering *TFI = MF->getSubtarget().getFrameLowering(); |
| ParentFrameOffset = TFI->getWinEHParentFrameOffset(*MF); |
| } |
| |
| for (size_t I = 0, E = FuncInfo.TryBlockMap.size(); I != E; ++I) { |
| const WinEHTryBlockMapEntry &TBME = FuncInfo.TryBlockMap[I]; |
| MCSymbol *HandlerMapXData = HandlerMaps[I]; |
| if (!HandlerMapXData) |
| continue; |
| // HandlerType { |
| // int32_t Adjectives; |
| // TypeDescriptor *Type; |
| // int32_t CatchObjOffset; |
| // void (*Handler)(); |
| // int32_t ParentFrameOffset; // x64 and AArch64 only |
| // }; |
| OS.EmitLabel(HandlerMapXData); |
| for (const WinEHHandlerType &HT : TBME.HandlerArray) { |
| // Get the frame escape label with the offset of the catch object. If |
| // the index is INT_MAX, then there is no catch object, and we should |
| // emit an offset of zero, indicating that no copy will occur. |
| const MCExpr *FrameAllocOffsetRef = nullptr; |
| if (HT.CatchObj.FrameIndex != INT_MAX) { |
| int Offset = getFrameIndexOffset(HT.CatchObj.FrameIndex, FuncInfo); |
| assert(Offset != 0 && "Illegal offset for catch object!"); |
| FrameAllocOffsetRef = MCConstantExpr::create(Offset, Asm->OutContext); |
| } else { |
| FrameAllocOffsetRef = MCConstantExpr::create(0, Asm->OutContext); |
| } |
| |
| MCSymbol *HandlerSym = |
| getMCSymbolForMBB(Asm, HT.Handler.dyn_cast<MachineBasicBlock *>()); |
| |
| AddComment("Adjectives"); |
| OS.EmitIntValue(HT.Adjectives, 4); |
| |
| AddComment("Type"); |
| OS.EmitValue(create32bitRef(HT.TypeDescriptor), 4); |
| |
| AddComment("CatchObjOffset"); |
| OS.EmitValue(FrameAllocOffsetRef, 4); |
| |
| AddComment("Handler"); |
| OS.EmitValue(create32bitRef(HandlerSym), 4); |
| |
| if (shouldEmitPersonality) { |
| AddComment("ParentFrameOffset"); |
| OS.EmitIntValue(ParentFrameOffset, 4); |
| } |
| } |
| } |
| } |
| |
| // IPToStateMapEntry { |
| // void *IP; |
| // int32_t State; |
| // }; |
| if (IPToStateXData) { |
| OS.EmitLabel(IPToStateXData); |
| for (auto &IPStatePair : IPToStateTable) { |
| AddComment("IP"); |
| OS.EmitValue(IPStatePair.first, 4); |
| AddComment("ToState"); |
| OS.EmitIntValue(IPStatePair.second, 4); |
| } |
| } |
| } |
| |
| void WinException::computeIP2StateTable( |
| const MachineFunction *MF, const WinEHFuncInfo &FuncInfo, |
| SmallVectorImpl<std::pair<const MCExpr *, int>> &IPToStateTable) { |
| |
| for (MachineFunction::const_iterator FuncletStart = MF->begin(), |
| FuncletEnd = MF->begin(), |
| End = MF->end(); |
| FuncletStart != End; FuncletStart = FuncletEnd) { |
| // Find the end of the funclet |
| while (++FuncletEnd != End) { |
| if (FuncletEnd->isEHFuncletEntry()) { |
| break; |
| } |
| } |
| |
| // Don't emit ip2state entries for cleanup funclets. Any interesting |
| // exceptional actions in cleanups must be handled in a separate IR |
| // function. |
| if (FuncletStart->isCleanupFuncletEntry()) |
| continue; |
| |
| MCSymbol *StartLabel; |
| int BaseState; |
| if (FuncletStart == MF->begin()) { |
| BaseState = NullState; |
| StartLabel = Asm->getFunctionBegin(); |
| } else { |
| auto *FuncletPad = |
| cast<FuncletPadInst>(FuncletStart->getBasicBlock()->getFirstNonPHI()); |
| assert(FuncInfo.FuncletBaseStateMap.count(FuncletPad) != 0); |
| BaseState = FuncInfo.FuncletBaseStateMap.find(FuncletPad)->second; |
| StartLabel = getMCSymbolForMBB(Asm, &*FuncletStart); |
| } |
| assert(StartLabel && "need local function start label"); |
| IPToStateTable.push_back( |
| std::make_pair(create32bitRef(StartLabel), BaseState)); |
| |
| for (const auto &StateChange : InvokeStateChangeIterator::range( |
| FuncInfo, FuncletStart, FuncletEnd, BaseState)) { |
| // Compute the label to report as the start of this entry; use the EH |
| // start label for the invoke if we have one, otherwise (this is a call |
| // which may unwind to our caller and does not have an EH start label, so) |
| // use the previous end label. |
| const MCSymbol *ChangeLabel = StateChange.NewStartLabel; |
| if (!ChangeLabel) |
| ChangeLabel = StateChange.PreviousEndLabel; |
| // Emit an entry indicating that PCs after 'Label' have this EH state. |
| IPToStateTable.push_back( |
| std::make_pair(getLabel(ChangeLabel), StateChange.NewState)); |
| // FIXME: assert that NewState is between CatchLow and CatchHigh. |
| } |
| } |
| } |
| |
| void WinException::emitEHRegistrationOffsetLabel(const WinEHFuncInfo &FuncInfo, |
| StringRef FLinkageName) { |
| // Outlined helpers called by the EH runtime need to know the offset of the EH |
| // registration in order to recover the parent frame pointer. Now that we know |
| // we've code generated the parent, we can emit the label assignment that |
| // those helpers use to get the offset of the registration node. |
| |
| // Compute the parent frame offset. The EHRegNodeFrameIndex will be invalid if |
| // after optimization all the invokes were eliminated. We still need to emit |
| // the parent frame offset label, but it should be garbage and should never be |
| // used. |
| int64_t Offset = 0; |
| int FI = FuncInfo.EHRegNodeFrameIndex; |
| if (FI != INT_MAX) { |
| const TargetFrameLowering *TFI = Asm->MF->getSubtarget().getFrameLowering(); |
| Offset = TFI->getNonLocalFrameIndexReference(*Asm->MF, FI); |
| } |
| |
| MCContext &Ctx = Asm->OutContext; |
| MCSymbol *ParentFrameOffset = |
| Ctx.getOrCreateParentFrameOffsetSymbol(FLinkageName); |
| Asm->OutStreamer->EmitAssignment(ParentFrameOffset, |
| MCConstantExpr::create(Offset, Ctx)); |
| } |
| |
| /// Emit the language-specific data that _except_handler3 and 4 expect. This is |
| /// functionally equivalent to the __C_specific_handler table, except it is |
| /// indexed by state number instead of IP. |
| void WinException::emitExceptHandlerTable(const MachineFunction *MF) { |
| MCStreamer &OS = *Asm->OutStreamer; |
| const Function &F = MF->getFunction(); |
| StringRef FLinkageName = GlobalValue::dropLLVMManglingEscape(F.getName()); |
| |
| bool VerboseAsm = OS.isVerboseAsm(); |
| auto AddComment = [&](const Twine &Comment) { |
| if (VerboseAsm) |
| OS.AddComment(Comment); |
| }; |
| |
| const WinEHFuncInfo &FuncInfo = *MF->getWinEHFuncInfo(); |
| emitEHRegistrationOffsetLabel(FuncInfo, FLinkageName); |
| |
| // Emit the __ehtable label that we use for llvm.x86.seh.lsda. |
| MCSymbol *LSDALabel = Asm->OutContext.getOrCreateLSDASymbol(FLinkageName); |
| OS.EmitValueToAlignment(4); |
| OS.EmitLabel(LSDALabel); |
| |
| const auto *Per = cast<Function>(F.getPersonalityFn()->stripPointerCasts()); |
| StringRef PerName = Per->getName(); |
| int BaseState = -1; |
| if (PerName == "_except_handler4") { |
| // The LSDA for _except_handler4 starts with this struct, followed by the |
| // scope table: |
| // |
| // struct EH4ScopeTable { |
| // int32_t GSCookieOffset; |
| // int32_t GSCookieXOROffset; |
| // int32_t EHCookieOffset; |
| // int32_t EHCookieXOROffset; |
| // ScopeTableEntry ScopeRecord[]; |
| // }; |
| // |
| // Offsets are %ebp relative. |
| // |
| // The GS cookie is present only if the function needs stack protection. |
| // GSCookieOffset = -2 means that GS cookie is not used. |
| // |
| // The EH cookie is always present. |
| // |
| // Check is done the following way: |
| // (ebp+CookieXOROffset) ^ [ebp+CookieOffset] == _security_cookie |
| |
| // Retrieve the Guard Stack slot. |
| int GSCookieOffset = -2; |
| const MachineFrameInfo &MFI = MF->getFrameInfo(); |
| if (MFI.hasStackProtectorIndex()) { |
| unsigned UnusedReg; |
| const TargetFrameLowering *TFI = MF->getSubtarget().getFrameLowering(); |
| int SSPIdx = MFI.getStackProtectorIndex(); |
| GSCookieOffset = TFI->getFrameIndexReference(*MF, SSPIdx, UnusedReg); |
| } |
| |
| // Retrieve the EH Guard slot. |
| // TODO(etienneb): Get rid of this value and change it for and assertion. |
| int EHCookieOffset = 9999; |
| if (FuncInfo.EHGuardFrameIndex != INT_MAX) { |
| unsigned UnusedReg; |
| const TargetFrameLowering *TFI = MF->getSubtarget().getFrameLowering(); |
| int EHGuardIdx = FuncInfo.EHGuardFrameIndex; |
| EHCookieOffset = TFI->getFrameIndexReference(*MF, EHGuardIdx, UnusedReg); |
| } |
| |
| AddComment("GSCookieOffset"); |
| OS.EmitIntValue(GSCookieOffset, 4); |
| AddComment("GSCookieXOROffset"); |
| OS.EmitIntValue(0, 4); |
| AddComment("EHCookieOffset"); |
| OS.EmitIntValue(EHCookieOffset, 4); |
| AddComment("EHCookieXOROffset"); |
| OS.EmitIntValue(0, 4); |
| BaseState = -2; |
| } |
| |
| assert(!FuncInfo.SEHUnwindMap.empty()); |
| for (const SEHUnwindMapEntry &UME : FuncInfo.SEHUnwindMap) { |
| auto *Handler = UME.Handler.get<MachineBasicBlock *>(); |
| const MCSymbol *ExceptOrFinally = |
| UME.IsFinally ? getMCSymbolForMBB(Asm, Handler) : Handler->getSymbol(); |
| // -1 is usually the base state for "unwind to caller", but for |
| // _except_handler4 it's -2. Do that replacement here if necessary. |
| int ToState = UME.ToState == -1 ? BaseState : UME.ToState; |
| AddComment("ToState"); |
| OS.EmitIntValue(ToState, 4); |
| AddComment(UME.IsFinally ? "Null" : "FilterFunction"); |
| OS.EmitValue(create32bitRef(UME.Filter), 4); |
| AddComment(UME.IsFinally ? "FinallyFunclet" : "ExceptionHandler"); |
| OS.EmitValue(create32bitRef(ExceptOrFinally), 4); |
| } |
| } |
| |
| static int getTryRank(const WinEHFuncInfo &FuncInfo, int State) { |
| int Rank = 0; |
| while (State != -1) { |
| ++Rank; |
| State = FuncInfo.ClrEHUnwindMap[State].TryParentState; |
| } |
| return Rank; |
| } |
| |
| static int getTryAncestor(const WinEHFuncInfo &FuncInfo, int Left, int Right) { |
| int LeftRank = getTryRank(FuncInfo, Left); |
| int RightRank = getTryRank(FuncInfo, Right); |
| |
| while (LeftRank < RightRank) { |
| Right = FuncInfo.ClrEHUnwindMap[Right].TryParentState; |
| --RightRank; |
| } |
| |
| while (RightRank < LeftRank) { |
| Left = FuncInfo.ClrEHUnwindMap[Left].TryParentState; |
| --LeftRank; |
| } |
| |
| while (Left != Right) { |
| Left = FuncInfo.ClrEHUnwindMap[Left].TryParentState; |
| Right = FuncInfo.ClrEHUnwindMap[Right].TryParentState; |
| } |
| |
| return Left; |
| } |
| |
| void WinException::emitCLRExceptionTable(const MachineFunction *MF) { |
| // CLR EH "states" are really just IDs that identify handlers/funclets; |
| // states, handlers, and funclets all have 1:1 mappings between them, and a |
| // handler/funclet's "state" is its index in the ClrEHUnwindMap. |
| MCStreamer &OS = *Asm->OutStreamer; |
| const WinEHFuncInfo &FuncInfo = *MF->getWinEHFuncInfo(); |
| MCSymbol *FuncBeginSym = Asm->getFunctionBegin(); |
| MCSymbol *FuncEndSym = Asm->getFunctionEnd(); |
| |
| // A ClrClause describes a protected region. |
| struct ClrClause { |
| const MCSymbol *StartLabel; // Start of protected region |
| const MCSymbol *EndLabel; // End of protected region |
| int State; // Index of handler protecting the protected region |
| int EnclosingState; // Index of funclet enclosing the protected region |
| }; |
| SmallVector<ClrClause, 8> Clauses; |
| |
| // Build a map from handler MBBs to their corresponding states (i.e. their |
| // indices in the ClrEHUnwindMap). |
| int NumStates = FuncInfo.ClrEHUnwindMap.size(); |
| assert(NumStates > 0 && "Don't need exception table!"); |
| DenseMap<const MachineBasicBlock *, int> HandlerStates; |
| for (int State = 0; State < NumStates; ++State) { |
| MachineBasicBlock *HandlerBlock = |
| FuncInfo.ClrEHUnwindMap[State].Handler.get<MachineBasicBlock *>(); |
| HandlerStates[HandlerBlock] = State; |
| // Use this loop through all handlers to verify our assumption (used in |
| // the MinEnclosingState computation) that enclosing funclets have lower |
| // state numbers than their enclosed funclets. |
| assert(FuncInfo.ClrEHUnwindMap[State].HandlerParentState < State && |
| "ill-formed state numbering"); |
| } |
| // Map the main function to the NullState. |
| HandlerStates[&MF->front()] = NullState; |
| |
| // Write out a sentinel indicating the end of the standard (Windows) xdata |
| // and the start of the additional (CLR) info. |
| OS.EmitIntValue(0xffffffff, 4); |
| // Write out the number of funclets |
| OS.EmitIntValue(NumStates, 4); |
| |
| // Walk the machine blocks/instrs, computing and emitting a few things: |
| // 1. Emit a list of the offsets to each handler entry, in lexical order. |
| // 2. Compute a map (EndSymbolMap) from each funclet to the symbol at its end. |
| // 3. Compute the list of ClrClauses, in the required order (inner before |
| // outer, earlier before later; the order by which a forward scan with |
| // early termination will find the innermost enclosing clause covering |
| // a given address). |
| // 4. A map (MinClauseMap) from each handler index to the index of the |
| // outermost funclet/function which contains a try clause targeting the |
| // key handler. This will be used to determine IsDuplicate-ness when |
| // emitting ClrClauses. The NullState value is used to indicate that the |
| // top-level function contains a try clause targeting the key handler. |
| // HandlerStack is a stack of (PendingStartLabel, PendingState) pairs for |
| // try regions we entered before entering the PendingState try but which |
| // we haven't yet exited. |
| SmallVector<std::pair<const MCSymbol *, int>, 4> HandlerStack; |
| // EndSymbolMap and MinClauseMap are maps described above. |
| std::unique_ptr<MCSymbol *[]> EndSymbolMap(new MCSymbol *[NumStates]); |
| SmallVector<int, 4> MinClauseMap((size_t)NumStates, NumStates); |
| |
| // Visit the root function and each funclet. |
| for (MachineFunction::const_iterator FuncletStart = MF->begin(), |
| FuncletEnd = MF->begin(), |
| End = MF->end(); |
| FuncletStart != End; FuncletStart = FuncletEnd) { |
| int FuncletState = HandlerStates[&*FuncletStart]; |
| // Find the end of the funclet |
| MCSymbol *EndSymbol = FuncEndSym; |
| while (++FuncletEnd != End) { |
| if (FuncletEnd->isEHFuncletEntry()) { |
| EndSymbol = getMCSymbolForMBB(Asm, &*FuncletEnd); |
| break; |
| } |
| } |
| // Emit the function/funclet end and, if this is a funclet (and not the |
| // root function), record it in the EndSymbolMap. |
| OS.EmitValue(getOffset(EndSymbol, FuncBeginSym), 4); |
| if (FuncletState != NullState) { |
| // Record the end of the handler. |
| EndSymbolMap[FuncletState] = EndSymbol; |
| } |
| |
| // Walk the state changes in this function/funclet and compute its clauses. |
| // Funclets always start in the null state. |
| const MCSymbol *CurrentStartLabel = nullptr; |
| int CurrentState = NullState; |
| assert(HandlerStack.empty()); |
| for (const auto &StateChange : |
| InvokeStateChangeIterator::range(FuncInfo, FuncletStart, FuncletEnd)) { |
| // Close any try regions we're not still under |
| int StillPendingState = |
| getTryAncestor(FuncInfo, CurrentState, StateChange.NewState); |
| while (CurrentState != StillPendingState) { |
| assert(CurrentState != NullState && |
| "Failed to find still-pending state!"); |
| // Close the pending clause |
| Clauses.push_back({CurrentStartLabel, StateChange.PreviousEndLabel, |
| CurrentState, FuncletState}); |
| // Now the next-outer try region is current |
| CurrentState = FuncInfo.ClrEHUnwindMap[CurrentState].TryParentState; |
| // Pop the new start label from the handler stack if we've exited all |
| // inner try regions of the corresponding try region. |
| if (HandlerStack.back().second == CurrentState) |
| CurrentStartLabel = HandlerStack.pop_back_val().first; |
| } |
| |
| if (StateChange.NewState != CurrentState) { |
| // For each clause we're starting, update the MinClauseMap so we can |
| // know which is the topmost funclet containing a clause targeting |
| // it. |
| for (int EnteredState = StateChange.NewState; |
| EnteredState != CurrentState; |
| EnteredState = |
| FuncInfo.ClrEHUnwindMap[EnteredState].TryParentState) { |
| int &MinEnclosingState = MinClauseMap[EnteredState]; |
| if (FuncletState < MinEnclosingState) |
| MinEnclosingState = FuncletState; |
| } |
| // Save the previous current start/label on the stack and update to |
| // the newly-current start/state. |
| HandlerStack.emplace_back(CurrentStartLabel, CurrentState); |
| CurrentStartLabel = StateChange.NewStartLabel; |
| CurrentState = StateChange.NewState; |
| } |
| } |
| assert(HandlerStack.empty()); |
| } |
| |
| // Now emit the clause info, starting with the number of clauses. |
| OS.EmitIntValue(Clauses.size(), 4); |
| for (ClrClause &Clause : Clauses) { |
| // Emit a CORINFO_EH_CLAUSE : |
| /* |
| struct CORINFO_EH_CLAUSE |
| { |
| CORINFO_EH_CLAUSE_FLAGS Flags; // actually a CorExceptionFlag |
| DWORD TryOffset; |
| DWORD TryLength; // actually TryEndOffset |
| DWORD HandlerOffset; |
| DWORD HandlerLength; // actually HandlerEndOffset |
| union |
| { |
| DWORD ClassToken; // use for catch clauses |
| DWORD FilterOffset; // use for filter clauses |
| }; |
| }; |
| |
| enum CORINFO_EH_CLAUSE_FLAGS |
| { |
| CORINFO_EH_CLAUSE_NONE = 0, |
| CORINFO_EH_CLAUSE_FILTER = 0x0001, // This clause is for a filter |
| CORINFO_EH_CLAUSE_FINALLY = 0x0002, // This clause is a finally clause |
| CORINFO_EH_CLAUSE_FAULT = 0x0004, // This clause is a fault clause |
| }; |
| typedef enum CorExceptionFlag |
| { |
| COR_ILEXCEPTION_CLAUSE_NONE, |
| COR_ILEXCEPTION_CLAUSE_FILTER = 0x0001, // This is a filter clause |
| COR_ILEXCEPTION_CLAUSE_FINALLY = 0x0002, // This is a finally clause |
| COR_ILEXCEPTION_CLAUSE_FAULT = 0x0004, // This is a fault clause |
| COR_ILEXCEPTION_CLAUSE_DUPLICATED = 0x0008, // duplicated clause. This |
| // clause was duplicated |
| // to a funclet which was |
| // pulled out of line |
| } CorExceptionFlag; |
| */ |
| // Add 1 to the start/end of the EH clause; the IP associated with a |
| // call when the runtime does its scan is the IP of the next instruction |
| // (the one to which control will return after the call), so we need |
| // to add 1 to the end of the clause to cover that offset. We also add |
| // 1 to the start of the clause to make sure that the ranges reported |
| // for all clauses are disjoint. Note that we'll need some additional |
| // logic when machine traps are supported, since in that case the IP |
| // that the runtime uses is the offset of the faulting instruction |
| // itself; if such an instruction immediately follows a call but the |
| // two belong to different clauses, we'll need to insert a nop between |
| // them so the runtime can distinguish the point to which the call will |
| // return from the point at which the fault occurs. |
| |
| const MCExpr *ClauseBegin = |
| getOffsetPlusOne(Clause.StartLabel, FuncBeginSym); |
| const MCExpr *ClauseEnd = getOffsetPlusOne(Clause.EndLabel, FuncBeginSym); |
| |
| const ClrEHUnwindMapEntry &Entry = FuncInfo.ClrEHUnwindMap[Clause.State]; |
| MachineBasicBlock *HandlerBlock = Entry.Handler.get<MachineBasicBlock *>(); |
| MCSymbol *BeginSym = getMCSymbolForMBB(Asm, HandlerBlock); |
| const MCExpr *HandlerBegin = getOffset(BeginSym, FuncBeginSym); |
| MCSymbol *EndSym = EndSymbolMap[Clause.State]; |
| const MCExpr *HandlerEnd = getOffset(EndSym, FuncBeginSym); |
| |
| uint32_t Flags = 0; |
| switch (Entry.HandlerType) { |
| case ClrHandlerType::Catch: |
| // Leaving bits 0-2 clear indicates catch. |
| break; |
| case ClrHandlerType::Filter: |
| Flags |= 1; |
| break; |
| case ClrHandlerType::Finally: |
| Flags |= 2; |
| break; |
| case ClrHandlerType::Fault: |
| Flags |= 4; |
| break; |
| } |
| if (Clause.EnclosingState != MinClauseMap[Clause.State]) { |
| // This is a "duplicate" clause; the handler needs to be entered from a |
| // frame above the one holding the invoke. |
| assert(Clause.EnclosingState > MinClauseMap[Clause.State]); |
| Flags |= 8; |
| } |
| OS.EmitIntValue(Flags, 4); |
| |
| // Write the clause start/end |
| OS.EmitValue(ClauseBegin, 4); |
| OS.EmitValue(ClauseEnd, 4); |
| |
| // Write out the handler start/end |
| OS.EmitValue(HandlerBegin, 4); |
| OS.EmitValue(HandlerEnd, 4); |
| |
| // Write out the type token or filter offset |
| assert(Entry.HandlerType != ClrHandlerType::Filter && "NYI: filters"); |
| OS.EmitIntValue(Entry.TypeToken, 4); |
| } |
| } |