blob: 9cf2edf2a56ca5eefa50b5f38ecd6c5364791bb4 [file] [log] [blame]
Saleem Abdulrasool64a8cc72014-08-07 02:59:41 +00001//===- lib/MC/MCWinEH.cpp - Windows EH implementation ---------------------===//
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#include "llvm/ADT/StringRef.h"
11#include "llvm/MC/MCContext.h"
12#include "llvm/MC/MCObjectFileInfo.h"
13#include "llvm/MC/MCSectionCOFF.h"
Pete Cooperef21bd42015-03-04 01:24:11 +000014#include "llvm/MC/MCStreamer.h"
Saleem Abdulrasool64a8cc72014-08-07 02:59:41 +000015#include "llvm/MC/MCSymbol.h"
16#include "llvm/MC/MCWinEH.h"
17#include "llvm/Support/COFF.h"
18
19namespace llvm {
20namespace WinEH {
Saleem Abdulrasool64a8cc72014-08-07 02:59:41 +000021
Reid Kleckner5aabc062014-12-22 22:10:08 +000022/// We can't have one section for all .pdata or .xdata because the Microsoft
23/// linker seems to want all code relocations to refer to the same object file
24/// section. If the code described is comdat, create a new comdat section
25/// associated with that comdat. If the code described is not in the main .text
26/// section, make a new section for it. Otherwise use the main unwind info
27/// section.
Rafael Espindola0709a7b2015-05-21 19:20:38 +000028static MCSection *getUnwindInfoSection(StringRef SecName,
29 MCSectionCOFF *UnwindSec,
30 const MCSymbol *Function,
31 MCContext &Context) {
Reid Kleckner7c4059e2014-09-04 17:42:03 +000032 if (Function && Function->isInSection()) {
Reid Kleckner5aabc062014-12-22 22:10:08 +000033 // If Function is in a COMDAT, get or create an unwind info section in that
34 // COMDAT group.
Reid Kleckner7c4059e2014-09-04 17:42:03 +000035 const MCSectionCOFF *FunctionSection =
36 cast<MCSectionCOFF>(&Function->getSection());
37 if (FunctionSection->getCharacteristics() & COFF::IMAGE_SCN_LNK_COMDAT) {
38 return Context.getAssociativeCOFFSection(
39 UnwindSec, FunctionSection->getCOMDATSymbol());
40 }
Reid Kleckner5aabc062014-12-22 22:10:08 +000041
42 // If Function is in a section other than .text, create a new .pdata section.
43 // Otherwise use the plain .pdata section.
44 if (const auto *Section = dyn_cast<MCSectionCOFF>(FunctionSection)) {
45 StringRef CodeSecName = Section->getSectionName();
46 if (CodeSecName == ".text")
47 return UnwindSec;
48
49 if (CodeSecName.startswith(".text$"))
50 CodeSecName = CodeSecName.substr(6);
51
52 return Context.getCOFFSection(
53 (SecName + Twine('$') + CodeSecName).str(),
54 COFF::IMAGE_SCN_CNT_INITIALIZED_DATA | COFF::IMAGE_SCN_MEM_READ,
55 SectionKind::getDataRel());
56 }
Reid Kleckner7c4059e2014-09-04 17:42:03 +000057 }
58
Reid Kleckner5aabc062014-12-22 22:10:08 +000059 return UnwindSec;
60
Reid Kleckner7c4059e2014-09-04 17:42:03 +000061}
62
Rafael Espindola0709a7b2015-05-21 19:20:38 +000063MCSection *UnwindEmitter::getPDataSection(const MCSymbol *Function,
64 MCContext &Context) {
65 MCSectionCOFF *PData =
Reid Kleckner7c4059e2014-09-04 17:42:03 +000066 cast<MCSectionCOFF>(Context.getObjectFileInfo()->getPDataSection());
67 return getUnwindInfoSection(".pdata", PData, Function, Context);
68}
69
Rafael Espindola0709a7b2015-05-21 19:20:38 +000070MCSection *UnwindEmitter::getXDataSection(const MCSymbol *Function,
71 MCContext &Context) {
72 MCSectionCOFF *XData =
Reid Kleckner7c4059e2014-09-04 17:42:03 +000073 cast<MCSectionCOFF>(Context.getObjectFileInfo()->getXDataSection());
74 return getUnwindInfoSection(".xdata", XData, Function, Context);
75}
76
Alexander Kornienko70bc5f12015-06-19 15:57:42 +000077} // namespace WinEH
78} // namespace llvm
Saleem Abdulrasool64a8cc72014-08-07 02:59:41 +000079