Home | History | Annotate | Download | only in AsmPrinter
      1 //===-- CodeGen/AsmPrinter/Win64Exception.cpp - Dwarf Exception Impl ------===//
      2 //
      3 //                     The LLVM Compiler Infrastructure
      4 //
      5 // This file is distributed under the University of Illinois Open Source
      6 // License. See LICENSE.TXT for details.
      7 //
      8 //===----------------------------------------------------------------------===//
      9 //
     10 // This file contains support for writing Win64 exception info into asm files.
     11 //
     12 //===----------------------------------------------------------------------===//
     13 
     14 #include "DwarfException.h"
     15 #include "llvm/ADT/SmallString.h"
     16 #include "llvm/ADT/StringExtras.h"
     17 #include "llvm/ADT/Twine.h"
     18 #include "llvm/CodeGen/AsmPrinter.h"
     19 #include "llvm/CodeGen/MachineFrameInfo.h"
     20 #include "llvm/CodeGen/MachineFunction.h"
     21 #include "llvm/CodeGen/MachineModuleInfo.h"
     22 #include "llvm/IR/DataLayout.h"
     23 #include "llvm/IR/Module.h"
     24 #include "llvm/MC/MCAsmInfo.h"
     25 #include "llvm/MC/MCContext.h"
     26 #include "llvm/MC/MCExpr.h"
     27 #include "llvm/MC/MCSection.h"
     28 #include "llvm/MC/MCStreamer.h"
     29 #include "llvm/MC/MCSymbol.h"
     30 #include "llvm/Support/Dwarf.h"
     31 #include "llvm/Support/ErrorHandling.h"
     32 #include "llvm/Support/FormattedStream.h"
     33 #include "llvm/Target/Mangler.h"
     34 #include "llvm/Target/TargetFrameLowering.h"
     35 #include "llvm/Target/TargetLoweringObjectFile.h"
     36 #include "llvm/Target/TargetOptions.h"
     37 #include "llvm/Target/TargetRegisterInfo.h"
     38 using namespace llvm;
     39 
     40 Win64Exception::Win64Exception(AsmPrinter *A)
     41   : DwarfException(A),
     42     shouldEmitPersonality(false), shouldEmitLSDA(false), shouldEmitMoves(false)
     43     {}
     44 
     45 Win64Exception::~Win64Exception() {}
     46 
     47 /// EndModule - Emit all exception information that should come after the
     48 /// content.
     49 void Win64Exception::EndModule() {
     50 }
     51 
     52 /// BeginFunction - Gather pre-function exception information. Assumes it's
     53 /// being emitted immediately after the function entry point.
     54 void Win64Exception::BeginFunction(const MachineFunction *MF) {
     55   shouldEmitMoves = shouldEmitPersonality = shouldEmitLSDA = false;
     56 
     57   // If any landing pads survive, we need an EH table.
     58   bool hasLandingPads = !MMI->getLandingPads().empty();
     59 
     60   shouldEmitMoves = Asm->needsSEHMoves();
     61 
     62   const TargetLoweringObjectFile &TLOF = Asm->getObjFileLowering();
     63   unsigned PerEncoding = TLOF.getPersonalityEncoding();
     64   const Function *Per = MMI->getPersonalities()[MMI->getPersonalityIndex()];
     65 
     66   shouldEmitPersonality = hasLandingPads &&
     67     PerEncoding != dwarf::DW_EH_PE_omit && Per;
     68 
     69   unsigned LSDAEncoding = TLOF.getLSDAEncoding();
     70   shouldEmitLSDA = shouldEmitPersonality &&
     71     LSDAEncoding != dwarf::DW_EH_PE_omit;
     72 
     73   if (!shouldEmitPersonality && !shouldEmitMoves)
     74     return;
     75 
     76   Asm->OutStreamer.EmitWin64EHStartProc(Asm->CurrentFnSym);
     77 
     78   if (!shouldEmitPersonality)
     79     return;
     80 
     81   MCSymbol *GCCHandlerSym =
     82     Asm->GetExternalSymbolSymbol("_GCC_specific_handler");
     83   Asm->OutStreamer.EmitWin64EHHandler(GCCHandlerSym, true, true);
     84 
     85   Asm->OutStreamer.EmitLabel(Asm->GetTempSymbol("eh_func_begin",
     86                                                 Asm->getFunctionNumber()));
     87 }
     88 
     89 /// EndFunction - Gather and emit post-function exception information.
     90 ///
     91 void Win64Exception::EndFunction() {
     92   if (!shouldEmitPersonality && !shouldEmitMoves)
     93     return;
     94 
     95   Asm->OutStreamer.EmitLabel(Asm->GetTempSymbol("eh_func_end",
     96                                                 Asm->getFunctionNumber()));
     97 
     98   // Map all labels and get rid of any dead landing pads.
     99   MMI->TidyLandingPads();
    100 
    101   if (shouldEmitPersonality) {
    102     const TargetLoweringObjectFile &TLOF = Asm->getObjFileLowering();
    103     const Function *Per = MMI->getPersonalities()[MMI->getPersonalityIndex()];
    104     const MCSymbol *Sym = TLOF.getCFIPersonalitySymbol(Per, Asm->Mang, MMI);
    105 
    106     Asm->OutStreamer.PushSection();
    107     Asm->OutStreamer.EmitWin64EHHandlerData();
    108     Asm->OutStreamer.EmitValue(MCSymbolRefExpr::Create(Sym, Asm->OutContext),
    109                                4);
    110     EmitExceptionTable();
    111     Asm->OutStreamer.PopSection();
    112   }
    113   Asm->OutStreamer.EmitWin64EHEndProc();
    114 }
    115