| //===-- CodeGen/AsmPrinter/DwarfException.cpp - Dwarf Exception Impl ------===// | 
 | // | 
 | //                     The LLVM Compiler Infrastructure | 
 | // | 
 | // This file is distributed under the University of Illinois Open Source | 
 | // License. See LICENSE.TXT for details. | 
 | // | 
 | //===----------------------------------------------------------------------===// | 
 | // | 
 | // This file contains support for writing DWARF exception info into asm files. | 
 | // | 
 | //===----------------------------------------------------------------------===// | 
 |  | 
 | #include "DwarfException.h" | 
 | #include "llvm/Module.h" | 
 | #include "llvm/CodeGen/AsmPrinter.h" | 
 | #include "llvm/CodeGen/MachineModuleInfo.h" | 
 | #include "llvm/CodeGen/MachineFrameInfo.h" | 
 | #include "llvm/CodeGen/MachineFunction.h" | 
 | #include "llvm/MC/MachineLocation.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/Target/Mangler.h" | 
 | #include "llvm/Target/TargetData.h" | 
 | #include "llvm/Target/TargetFrameLowering.h" | 
 | #include "llvm/Target/TargetLoweringObjectFile.h" | 
 | #include "llvm/Target/TargetMachine.h" | 
 | #include "llvm/Target/TargetOptions.h" | 
 | #include "llvm/Target/TargetRegisterInfo.h" | 
 | #include "llvm/Support/Dwarf.h" | 
 | #include "llvm/Support/ErrorHandling.h" | 
 | #include "llvm/Support/FormattedStream.h" | 
 | #include "llvm/ADT/SmallString.h" | 
 | #include "llvm/ADT/StringExtras.h" | 
 | #include "llvm/ADT/Twine.h" | 
 | using namespace llvm; | 
 |  | 
 | DwarfCFIException::DwarfCFIException(AsmPrinter *A) | 
 |   : DwarfException(A), | 
 |     shouldEmitPersonality(false), shouldEmitLSDA(false), shouldEmitMoves(false), | 
 |     moveTypeModule(AsmPrinter::CFI_M_None) {} | 
 |  | 
 | DwarfCFIException::~DwarfCFIException() {} | 
 |  | 
 | /// EndModule - Emit all exception information that should come after the | 
 | /// content. | 
 | void DwarfCFIException::EndModule() { | 
 |   if (moveTypeModule == AsmPrinter::CFI_M_Debug) | 
 |     Asm->OutStreamer.EmitCFISections(false, true); | 
 |  | 
 |   if (!Asm->MAI->isExceptionHandlingDwarf()) | 
 |     return; | 
 |  | 
 |   const TargetLoweringObjectFile &TLOF = Asm->getObjFileLowering(); | 
 |  | 
 |   unsigned PerEncoding = TLOF.getPersonalityEncoding(); | 
 |  | 
 |   if ((PerEncoding & 0x70) != dwarf::DW_EH_PE_pcrel) | 
 |     return; | 
 |  | 
 |   // Emit references to all used personality functions | 
 |   bool AtLeastOne = false; | 
 |   const std::vector<const Function*> &Personalities = MMI->getPersonalities(); | 
 |   for (size_t i = 0, e = Personalities.size(); i != e; ++i) { | 
 |     if (!Personalities[i]) | 
 |       continue; | 
 |     MCSymbol *Sym = Asm->Mang->getSymbol(Personalities[i]); | 
 |     TLOF.emitPersonalityValue(Asm->OutStreamer, Asm->TM, Sym); | 
 |     AtLeastOne = true; | 
 |   } | 
 |  | 
 |   if (AtLeastOne && !TLOF.isFunctionEHFrameSymbolPrivate()) { | 
 |     // This is a temporary hack to keep sections in the same order they | 
 |     // were before. This lets us produce bit identical outputs while | 
 |     // transitioning to CFI. | 
 |     Asm->OutStreamer.SwitchSection( | 
 |                const_cast<TargetLoweringObjectFile&>(TLOF).getEHFrameSection()); | 
 |   } | 
 | } | 
 |  | 
 | /// BeginFunction - Gather pre-function exception information. Assumes it's | 
 | /// being emitted immediately after the function entry point. | 
 | void DwarfCFIException::BeginFunction(const MachineFunction *MF) { | 
 |   shouldEmitMoves = shouldEmitPersonality = shouldEmitLSDA = false; | 
 |  | 
 |   // If any landing pads survive, we need an EH table. | 
 |   bool hasLandingPads = !MMI->getLandingPads().empty(); | 
 |  | 
 |   // See if we need frame move info. | 
 |   AsmPrinter::CFIMoveType MoveType = Asm->needsCFIMoves(); | 
 |   if (MoveType == AsmPrinter::CFI_M_EH || | 
 |       (MoveType == AsmPrinter::CFI_M_Debug && | 
 |        moveTypeModule == AsmPrinter::CFI_M_None)) | 
 |     moveTypeModule = MoveType; | 
 |  | 
 |   shouldEmitMoves = MoveType != AsmPrinter::CFI_M_None; | 
 |  | 
 |   const TargetLoweringObjectFile &TLOF = Asm->getObjFileLowering(); | 
 |   unsigned PerEncoding = TLOF.getPersonalityEncoding(); | 
 |   const Function *Per = MMI->getPersonalities()[MMI->getPersonalityIndex()]; | 
 |  | 
 |   shouldEmitPersonality = hasLandingPads && | 
 |     PerEncoding != dwarf::DW_EH_PE_omit && Per; | 
 |  | 
 |   unsigned LSDAEncoding = TLOF.getLSDAEncoding(); | 
 |   shouldEmitLSDA = shouldEmitPersonality && | 
 |     LSDAEncoding != dwarf::DW_EH_PE_omit; | 
 |  | 
 |   if (!shouldEmitPersonality && !shouldEmitMoves) | 
 |     return; | 
 |  | 
 |   Asm->OutStreamer.EmitCFIStartProc(); | 
 |  | 
 |   // Indicate personality routine, if any. | 
 |   if (!shouldEmitPersonality) | 
 |     return; | 
 |  | 
 |   const MCSymbol *Sym = TLOF.getCFIPersonalitySymbol(Per, Asm->Mang, MMI); | 
 |   Asm->OutStreamer.EmitCFIPersonality(Sym, PerEncoding); | 
 |  | 
 |   Asm->OutStreamer.EmitLabel(Asm->GetTempSymbol("eh_func_begin", | 
 |                                                 Asm->getFunctionNumber())); | 
 |  | 
 |   // Provide LSDA information. | 
 |   if (!shouldEmitLSDA) | 
 |     return; | 
 |  | 
 |   Asm->OutStreamer.EmitCFILsda(Asm->GetTempSymbol("exception", | 
 |                                                   Asm->getFunctionNumber()), | 
 |                                LSDAEncoding); | 
 | } | 
 |  | 
 | /// EndFunction - Gather and emit post-function exception information. | 
 | /// | 
 | void DwarfCFIException::EndFunction() { | 
 |   if (!shouldEmitPersonality && !shouldEmitMoves) | 
 |     return; | 
 |  | 
 |   Asm->OutStreamer.EmitCFIEndProc(); | 
 |  | 
 |   Asm->OutStreamer.EmitLabel(Asm->GetTempSymbol("eh_func_end", | 
 |                                                 Asm->getFunctionNumber())); | 
 |  | 
 |   // Map all labels and get rid of any dead landing pads. | 
 |   MMI->TidyLandingPads(); | 
 |  | 
 |   if (shouldEmitPersonality) | 
 |     EmitExceptionTable(); | 
 | } |