WinCOFFObjectWriter.cpp revision a7b7a7d629c3101f6f6c87e6848e865734e0238c
1b162290e39afd49d4c7d342333b331bc96232720Chris Lattner//===-- llvm/MC/WinCOFFObjectWriter.cpp -------------------------*- C++ -*-===//
2b162290e39afd49d4c7d342333b331bc96232720Chris Lattner//
3b162290e39afd49d4c7d342333b331bc96232720Chris Lattner//                     The LLVM Compiler Infrastructure
4b162290e39afd49d4c7d342333b331bc96232720Chris Lattner//
5b162290e39afd49d4c7d342333b331bc96232720Chris Lattner// This file is distributed under the University of Illinois Open Source
6b162290e39afd49d4c7d342333b331bc96232720Chris Lattner// License. See LICENSE.TXT for details.
7b162290e39afd49d4c7d342333b331bc96232720Chris Lattner//
8b162290e39afd49d4c7d342333b331bc96232720Chris Lattner//===----------------------------------------------------------------------===//
9b162290e39afd49d4c7d342333b331bc96232720Chris Lattner//
10b162290e39afd49d4c7d342333b331bc96232720Chris Lattner// This file contains an implementation of a Win32 COFF object file writer.
11b162290e39afd49d4c7d342333b331bc96232720Chris Lattner//
12b162290e39afd49d4c7d342333b331bc96232720Chris Lattner//===----------------------------------------------------------------------===//
13b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
14b162290e39afd49d4c7d342333b331bc96232720Chris Lattner#define DEBUG_TYPE "WinCOFFObjectWriter"
15801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
16df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola#include "llvm/MC/MCWinCOFFObjectWriter.h"
17801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/ADT/DenseMap.h"
18df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola#include "llvm/ADT/OwningPtr.h"
19801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/ADT/StringMap.h"
20801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/ADT/StringRef.h"
2180646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck#include "llvm/ADT/Twine.h"
22d04a8d4b33ff316ca4cf961e06c9e312eff8e64fChandler Carruth#include "llvm/MC/MCAsmLayout.h"
23d04a8d4b33ff316ca4cf961e06c9e312eff8e64fChandler Carruth#include "llvm/MC/MCAssembler.h"
24d04a8d4b33ff316ca4cf961e06c9e312eff8e64fChandler Carruth#include "llvm/MC/MCContext.h"
25d04a8d4b33ff316ca4cf961e06c9e312eff8e64fChandler Carruth#include "llvm/MC/MCExpr.h"
26d04a8d4b33ff316ca4cf961e06c9e312eff8e64fChandler Carruth#include "llvm/MC/MCObjectWriter.h"
27d04a8d4b33ff316ca4cf961e06c9e312eff8e64fChandler Carruth#include "llvm/MC/MCSection.h"
28d04a8d4b33ff316ca4cf961e06c9e312eff8e64fChandler Carruth#include "llvm/MC/MCSectionCOFF.h"
29d04a8d4b33ff316ca4cf961e06c9e312eff8e64fChandler Carruth#include "llvm/MC/MCSymbol.h"
30d04a8d4b33ff316ca4cf961e06c9e312eff8e64fChandler Carruth#include "llvm/MC/MCValue.h"
31801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/Support/COFF.h"
32801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/Support/Debug.h"
33801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/Support/ErrorHandling.h"
341f6efa3996dd1929fbc129203ce5009b620e6969Michael J. Spencer#include "llvm/Support/TimeValue.h"
35801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include <cstdio>
36801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
37b162290e39afd49d4c7d342333b331bc96232720Chris Lattnerusing namespace llvm;
38b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
39b162290e39afd49d4c7d342333b331bc96232720Chris Lattnernamespace {
4096f498bd9f140a98321c478f517877c4767b94faDmitri Gribenkotypedef SmallString<COFF::NameSize> name;
41801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
42801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerenum AuxiliaryType {
43801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  ATFunctionDefinition,
44801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  ATbfAndefSymbol,
45801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  ATWeakExternal,
46801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  ATFile,
47801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  ATSectionDefinition
48801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer};
49801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
50801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerstruct AuxSymbol {
51801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  AuxiliaryType   AuxType;
52801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFF::Auxiliary Aux;
53801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer};
54801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
55a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencerclass COFFSymbol;
56a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencerclass COFFSection;
57a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
58801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerclass COFFSymbol {
59801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerpublic:
60801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFF::symbol Data;
61801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
6296f498bd9f140a98321c478f517877c4767b94faDmitri Gribenko  typedef SmallVector<AuxSymbol, 1> AuxiliarySymbols;
63801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
64801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  name             Name;
659cf23a9ab466a900bc0f937bc930d398d6097766Michael J. Spencer  int              Index;
66801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  AuxiliarySymbols Aux;
67801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSymbol      *Other;
68a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  COFFSection     *Section;
69a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  int              Relocations;
70801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
71801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  MCSymbolData const *MCData;
72801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
7396f498bd9f140a98321c478f517877c4767b94faDmitri Gribenko  COFFSymbol(StringRef name);
74801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  size_t size() const;
75801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void set_name_offset(uint32_t Offset);
76a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
77a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  bool should_keep() const;
78801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer};
79801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
80801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// This class contains staging data for a COFF relocation entry.
81801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerstruct COFFRelocation {
82801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFF::relocation Data;
83801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSymbol          *Symb;
84801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
85801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFRelocation() : Symb(NULL) {}
86801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  static size_t size() { return COFF::RelocationSize; }
87801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer};
88801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
89801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencertypedef std::vector<COFFRelocation> relocations;
90801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
91801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerclass COFFSection {
92801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerpublic:
93801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFF::section Header;
94801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
95801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  std::string          Name;
969cf23a9ab466a900bc0f937bc930d398d6097766Michael J. Spencer  int                  Number;
97801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  MCSectionData const *MCData;
98a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  COFFSymbol          *Symbol;
99801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  relocations          Relocations;
100801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
10196f498bd9f140a98321c478f517877c4767b94faDmitri Gribenko  COFFSection(StringRef name);
102801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  static size_t size();
103801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer};
104801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
105801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// This class holds the COFF string table.
106801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerclass StringTable {
10796f498bd9f140a98321c478f517877c4767b94faDmitri Gribenko  typedef StringMap<size_t> map;
108801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  map Map;
109801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
110801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void update_length();
111801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerpublic:
112801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  std::vector<char> Data;
113801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
114801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  StringTable();
115801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  size_t size() const;
11696f498bd9f140a98321c478f517877c4767b94faDmitri Gribenko  size_t insert(StringRef String);
117801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer};
118801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
119801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerclass WinCOFFObjectWriter : public MCObjectWriter {
120801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerpublic:
121801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
122801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  typedef std::vector<COFFSymbol*>  symbols;
123801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  typedef std::vector<COFFSection*> sections;
124b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
1254cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  typedef DenseMap<MCSymbol  const *, COFFSymbol *>   symbol_map;
1264cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  typedef DenseMap<MCSection const *, COFFSection *> section_map;
127b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
128df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola  llvm::OwningPtr<MCWinCOFFObjectTargetWriter> TargetObjectWriter;
129df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola
130801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Root level file contents.
131801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFF::header Header;
132801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  sections     Sections;
133801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  symbols      Symbols;
134801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  StringTable  Strings;
135b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
136801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Maps used during object file creation.
137801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  section_map SectionMap;
138801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  symbol_map  SymbolMap;
139801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
140df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola  WinCOFFObjectWriter(MCWinCOFFObjectTargetWriter *MOTW, raw_ostream &OS);
141b21ab43cfc3fa0dacf5c95f04e58b6d804b59a16Alexey Samsonov  ~WinCOFFObjectWriter();
142801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
1434cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  COFFSymbol *createSymbol(StringRef Name);
1444cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  COFFSymbol *GetOrCreateCOFFSymbol(const MCSymbol * Symbol);
1454cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  COFFSection *createSection(StringRef Name);
146801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
147801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  template <typename object_t, typename list_t>
14896f498bd9f140a98321c478f517877c4767b94faDmitri Gribenko  object_t *createCOFFEntity(StringRef Name, list_t &List);
149801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
150801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void DefineSection(MCSectionData const &SectionData);
15128860823ad34d41d4f58561dc14a982fb0843fddReid Kleckner  void DefineSymbol(MCSymbolData const &SymbolData, MCAssembler &Assembler,
15228860823ad34d41d4f58561dc14a982fb0843fddReid Kleckner                    const MCAsmLayout &Layout);
153801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
154a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  void MakeSymbolReal(COFFSymbol &S, size_t Index);
155a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  void MakeSectionReal(COFFSection &S, size_t Number);
156a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
157a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner  bool ExportSymbol(MCSymbolData const &SymbolData, MCAssembler &Asm);
158a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner
159a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  bool IsPhysicalSection(COFFSection *S);
160a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
161801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Entity writing methods.
162801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
163801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void WriteFileHeader(const COFF::header &Header);
164801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void WriteSymbol(const COFFSymbol *S);
165801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void WriteAuxiliarySymbols(const COFFSymbol::AuxiliarySymbols &S);
166801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void WriteSectionHeader(const COFF::section &S);
167801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void WriteRelocation(const COFF::relocation &R);
168801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
169801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // MCObjectWriter interface implementation.
170801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
17185f2ecc697a8ca6c8cf08093054cbbb9d2060ccfRafael Espindola  void ExecutePostLayoutBinding(MCAssembler &Asm, const MCAsmLayout &Layout);
172801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
173801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void RecordRelocation(const MCAssembler &Asm,
174801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                        const MCAsmLayout &Layout,
175801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                        const MCFragment *Fragment,
176801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                        const MCFixup &Fixup,
177801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                        MCValue Target,
178801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                        uint64_t &FixedValue);
179801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
1808f413fa9c00dbaea000ddfe265ab5edd285ea3abRafael Espindola  void WriteObject(MCAssembler &Asm, const MCAsmLayout &Layout);
181801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer};
182801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
183801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
184801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerstatic inline void write_uint32_le(void *Data, uint32_t const &Value) {
185801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  uint8_t *Ptr = reinterpret_cast<uint8_t *>(Data);
186801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[0] = (Value & 0x000000FF) >>  0;
187801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[1] = (Value & 0x0000FF00) >>  8;
188801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[2] = (Value & 0x00FF0000) >> 16;
189801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[3] = (Value & 0xFF000000) >> 24;
190801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
191801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
192801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------
193801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// Symbol class implementation
194801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
19596f498bd9f140a98321c478f517877c4767b94faDmitri GribenkoCOFFSymbol::COFFSymbol(StringRef name)
196a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  : Name(name.begin(), name.end())
197a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  , Other(NULL)
198a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  , Section(NULL)
199a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  , Relocations(0)
200a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  , MCData(NULL) {
201801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  memset(&Data, 0, sizeof(Data));
202801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
203b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
204801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencersize_t COFFSymbol::size() const {
205801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return COFF::SymbolSize + (Data.NumberOfAuxSymbols * COFF::SymbolSize);
206b162290e39afd49d4c7d342333b331bc96232720Chris Lattner}
207b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
208801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// In the case that the name does not fit within 8 bytes, the offset
209801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// into the string table is stored in the last 4 bytes instead, leaving
210801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// the first 4 bytes as 0.
211801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid COFFSymbol::set_name_offset(uint32_t Offset) {
212801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  write_uint32_le(Data.Name + 0, 0);
213801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  write_uint32_le(Data.Name + 4, Offset);
214801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
215801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
216a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer/// logic to decide if the symbol should be reported in the symbol table
217a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencerbool COFFSymbol::should_keep() const {
218a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  // no section means its external, keep it
219a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  if (Section == NULL)
220a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    return true;
221a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
222a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  // if it has relocations pointing at it, keep it
223a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  if (Relocations > 0)   {
224a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    assert(Section->Number != -1 && "Sections with relocations must be real!");
225a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    return true;
226a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  }
227a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
228a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  // if the section its in is being droped, drop it
229a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  if (Section->Number == -1)
230a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      return false;
231a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
232a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  // if it is the section symbol, keep it
233a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  if (Section->Symbol == this)
234a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    return true;
235a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
236a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  // if its temporary, drop it
237a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  if (MCData && MCData->getSymbol().isTemporary())
238a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      return false;
239a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
240a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  // otherwise, keep it
241a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  return true;
242a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer}
243a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
244801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------
245801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// Section class implementation
246801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
24796f498bd9f140a98321c478f517877c4767b94faDmitri GribenkoCOFFSection::COFFSection(StringRef name)
248a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  : Name(name)
249a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  , MCData(NULL)
250a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  , Symbol(NULL) {
251801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  memset(&Header, 0, sizeof(Header));
252801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
253801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
254801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencersize_t COFFSection::size() {
255801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return COFF::SectionSize;
256801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
257801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
258801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------
259801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// StringTable class implementation
260801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
261801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// Write the length of the string table into Data.
262801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// The length of the string table includes uint32 length header.
263801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid StringTable::update_length() {
264801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  write_uint32_le(&Data.front(), Data.size());
265801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
266801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
267801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. SpencerStringTable::StringTable() {
268801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // The string table data begins with the length of the entire string table
269801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // including the length header. Allocate space for this header.
270801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Data.resize(4);
2710d64632c9e8a2632b8804bcc3fbc3523588862a7Michael J. Spencer  update_length();
272801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
273801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
274801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencersize_t StringTable::size() const {
275801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return Data.size();
276801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
277801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
27894c22716d60ff5edf6a98a3c67e0faa001be1142Sylvestre Ledru/// Add String to the table iff it is not already there.
279801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// @returns the index into the string table where the string is now located.
28096f498bd9f140a98321c478f517877c4767b94faDmitri Gribenkosize_t StringTable::insert(StringRef String) {
281801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  map::iterator i = Map.find(String);
282801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
283801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  if (i != Map.end())
284801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    return i->second;
285801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
286801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  size_t Offset = Data.size();
287801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
288801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Insert string data into string table.
289801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Data.insert(Data.end(), String.begin(), String.end());
290801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Data.push_back('\0');
291801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
292801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Put a reference to it in the map.
293801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Map[String] = Offset;
294801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
295801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Update the internal length field.
296801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  update_length();
297801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
298801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return Offset;
299801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
300801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
301801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------
302801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// WinCOFFObjectWriter class implementation
303801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
304df09270ae897e7fa64a7c162de163c32ee181a03Rafael EspindolaWinCOFFObjectWriter::WinCOFFObjectWriter(MCWinCOFFObjectTargetWriter *MOTW,
305df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola                                         raw_ostream &OS)
30682c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer  : MCObjectWriter(OS, true)
307df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola  , TargetObjectWriter(MOTW) {
308801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  memset(&Header, 0, sizeof(Header));
309da0bfcdaf95d95a66e306ef6d45f638939272d34Michael J. Spencer
310df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola  Header.Machine = TargetObjectWriter->getMachine();
311801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
312801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
313808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin KramerWinCOFFObjectWriter::~WinCOFFObjectWriter() {
314808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer  for (symbols::iterator I = Symbols.begin(), E = Symbols.end(); I != E; ++I)
315808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer    delete *I;
316808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer  for (sections::iterator I = Sections.begin(), E = Sections.end(); I != E; ++I)
317808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer    delete *I;
318808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer}
319808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer
3204cee2890a66974af506f2125243114cc14bd5556Michael J. SpencerCOFFSymbol *WinCOFFObjectWriter::createSymbol(StringRef Name) {
321801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return createCOFFEntity<COFFSymbol>(Name, Symbols);
322801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
323801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
3244cee2890a66974af506f2125243114cc14bd5556Michael J. SpencerCOFFSymbol *WinCOFFObjectWriter::GetOrCreateCOFFSymbol(const MCSymbol * Symbol){
3254cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  symbol_map::iterator i = SymbolMap.find(Symbol);
3264cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  if (i != SymbolMap.end())
3274cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer    return i->second;
3284cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  COFFSymbol *RetSymbol
3294cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer    = createCOFFEntity<COFFSymbol>(Symbol->getName(), Symbols);
3304cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  SymbolMap[Symbol] = RetSymbol;
3314cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  return RetSymbol;
3324cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer}
3334cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer
33496f498bd9f140a98321c478f517877c4767b94faDmitri GribenkoCOFFSection *WinCOFFObjectWriter::createSection(StringRef Name) {
335801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return createCOFFEntity<COFFSection>(Name, Sections);
336801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
337801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
338801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// A template used to lookup or create a symbol/section, and initialize it if
339801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// needed.
340801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencertemplate <typename object_t, typename list_t>
34196f498bd9f140a98321c478f517877c4767b94faDmitri Gribenkoobject_t *WinCOFFObjectWriter::createCOFFEntity(StringRef Name,
342801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                                list_t &List) {
343a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  object_t *Object = new object_t(Name);
344801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
345801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  List.push_back(Object);
346801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
347801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return Object;
348801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
349801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
350801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// This function takes a section data object from the assembler
351801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// and creates the associated COFF section staging object.
352801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::DefineSection(MCSectionData const &SectionData) {
353d47f4a9c982d264e46a6a2fe0f357288768bb5b9Michael J. Spencer  assert(SectionData.getSection().getVariant() == MCSection::SV_COFF
354d47f4a9c982d264e46a6a2fe0f357288768bb5b9Michael J. Spencer    && "Got non COFF section in the COFF backend!");
355801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // FIXME: Not sure how to verify this (at least in a debug build).
356801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  MCSectionCOFF const &Sec =
357801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    static_cast<MCSectionCOFF const &>(SectionData.getSection());
358801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
359801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSection *coff_section = createSection(Sec.getSectionName());
360801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSymbol  *coff_symbol = createSymbol(Sec.getSectionName());
361801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
362a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  coff_section->Symbol = coff_symbol;
363a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  coff_symbol->Section = coff_section;
364801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->Data.StorageClass = COFF::IMAGE_SYM_CLASS_STATIC;
365801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
366801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // In this case the auxiliary symbol is a Section Definition.
367801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->Aux.resize(1);
368801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  memset(&coff_symbol->Aux[0], 0, sizeof(coff_symbol->Aux[0]));
369801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->Aux[0].AuxType = ATSectionDefinition;
370801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->Aux[0].Aux.SectionDefinition.Selection = Sec.getSelection();
371801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
372801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_section->Header.Characteristics = Sec.getCharacteristics();
373801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
374801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  uint32_t &Characteristics = coff_section->Header.Characteristics;
375801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  switch (SectionData.getAlignment()) {
376801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 1:    Characteristics |= COFF::IMAGE_SCN_ALIGN_1BYTES;    break;
377801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 2:    Characteristics |= COFF::IMAGE_SCN_ALIGN_2BYTES;    break;
378801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 4:    Characteristics |= COFF::IMAGE_SCN_ALIGN_4BYTES;    break;
379801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 8:    Characteristics |= COFF::IMAGE_SCN_ALIGN_8BYTES;    break;
380801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 16:   Characteristics |= COFF::IMAGE_SCN_ALIGN_16BYTES;   break;
381801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 32:   Characteristics |= COFF::IMAGE_SCN_ALIGN_32BYTES;   break;
382801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 64:   Characteristics |= COFF::IMAGE_SCN_ALIGN_64BYTES;   break;
383801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 128:  Characteristics |= COFF::IMAGE_SCN_ALIGN_128BYTES;  break;
384801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 256:  Characteristics |= COFF::IMAGE_SCN_ALIGN_256BYTES;  break;
385801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 512:  Characteristics |= COFF::IMAGE_SCN_ALIGN_512BYTES;  break;
386801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 1024: Characteristics |= COFF::IMAGE_SCN_ALIGN_1024BYTES; break;
387801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 2048: Characteristics |= COFF::IMAGE_SCN_ALIGN_2048BYTES; break;
388801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 4096: Characteristics |= COFF::IMAGE_SCN_ALIGN_4096BYTES; break;
389801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 8192: Characteristics |= COFF::IMAGE_SCN_ALIGN_8192BYTES; break;
390801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  default:
391801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    llvm_unreachable("unsupported section alignment");
392801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
393801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
394801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Bind internal COFF section to MC section.
395801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_section->MCData = &SectionData;
3964cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  SectionMap[&SectionData.getSection()] = coff_section;
397801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
398801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
399801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// This function takes a section data object from the assembler
400801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// and creates the associated COFF symbol staging object.
4014974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbournevoid WinCOFFObjectWriter::DefineSymbol(MCSymbolData const &SymbolData,
40228860823ad34d41d4f58561dc14a982fb0843fddReid Kleckner                                       MCAssembler &Assembler,
40328860823ad34d41d4f58561dc14a982fb0843fddReid Kleckner                                       const MCAsmLayout &Layout) {
4044974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne  MCSymbol const &Symbol = SymbolData.getSymbol();
4050933134a304b47d3767aad202df9f0e09743da6dMichael J. Spencer  COFFSymbol *coff_symbol = GetOrCreateCOFFSymbol(&Symbol);
4064974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne  SymbolMap[&Symbol] = coff_symbol;
407801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
4084cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  if (SymbolData.getFlags() & COFF::SF_WeakExternal) {
409801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Data.StorageClass = COFF::IMAGE_SYM_CLASS_WEAK_EXTERNAL;
410801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
4110933134a304b47d3767aad202df9f0e09743da6dMichael J. Spencer    if (Symbol.isVariable()) {
4124974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne      const MCSymbolRefExpr *SymRef =
4134974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne        dyn_cast<MCSymbolRefExpr>(Symbol.getVariableValue());
414801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
4154974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne      if (!SymRef)
4164974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne        report_fatal_error("Weak externals may only alias symbols");
417801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
4184974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne      coff_symbol->Other = GetOrCreateCOFFSymbol(&SymRef->getSymbol());
4194cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer    } else {
4204cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer      std::string WeakName = std::string(".weak.")
4210933134a304b47d3767aad202df9f0e09743da6dMichael J. Spencer                           +  Symbol.getName().str()
4224cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer                           + ".default";
4234cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer      COFFSymbol *WeakDefault = createSymbol(WeakName);
4244cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer      WeakDefault->Data.SectionNumber = COFF::IMAGE_SYM_ABSOLUTE;
4254cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer      WeakDefault->Data.StorageClass  = COFF::IMAGE_SYM_CLASS_EXTERNAL;
4264cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer      WeakDefault->Data.Type          = 0;
4274cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer      WeakDefault->Data.Value         = 0;
4284cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer      coff_symbol->Other = WeakDefault;
4294cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer    }
430801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
431801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    // Setup the Weak External auxiliary symbol.
432801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Aux.resize(1);
433801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    memset(&coff_symbol->Aux[0], 0, sizeof(coff_symbol->Aux[0]));
434801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Aux[0].AuxType = ATWeakExternal;
435801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Aux[0].Aux.WeakExternal.TagIndex = 0;
436801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Aux[0].Aux.WeakExternal.Characteristics =
4374cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer      COFF::IMAGE_WEAK_EXTERN_SEARCH_LIBRARY;
4384cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer
4394974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    coff_symbol->MCData = &SymbolData;
4404974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne  } else {
4414974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    const MCSymbolData &ResSymData =
4424974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne      Assembler.getSymbolData(Symbol.AliasedSymbol());
4434cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer
44428860823ad34d41d4f58561dc14a982fb0843fddReid Kleckner    if (Symbol.isVariable()) {
44528860823ad34d41d4f58561dc14a982fb0843fddReid Kleckner      int64_t Addr;
44628860823ad34d41d4f58561dc14a982fb0843fddReid Kleckner      if (Symbol.getVariableValue()->EvaluateAsAbsolute(Addr, Layout))
44728860823ad34d41d4f58561dc14a982fb0843fddReid Kleckner        coff_symbol->Data.Value = Addr;
44828860823ad34d41d4f58561dc14a982fb0843fddReid Kleckner    }
44928860823ad34d41d4f58561dc14a982fb0843fddReid Kleckner
4504974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    coff_symbol->Data.Type         = (ResSymData.getFlags() & 0x0000FFFF) >>  0;
4514974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    coff_symbol->Data.StorageClass = (ResSymData.getFlags() & 0x00FF0000) >> 16;
452801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
4534974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    // If no storage class was specified in the streamer, define it here.
4544974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    if (coff_symbol->Data.StorageClass == 0) {
4554974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne      bool external = ResSymData.isExternal() || (ResSymData.Fragment == NULL);
456a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
4574974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne      coff_symbol->Data.StorageClass =
4584974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne       external ? COFF::IMAGE_SYM_CLASS_EXTERNAL : COFF::IMAGE_SYM_CLASS_STATIC;
4594974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    }
4604974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne
46128860823ad34d41d4f58561dc14a982fb0843fddReid Kleckner    if (Symbol.isAbsolute() || Symbol.AliasedSymbol().isVariable())
46228860823ad34d41d4f58561dc14a982fb0843fddReid Kleckner      coff_symbol->Data.SectionNumber = COFF::IMAGE_SYM_ABSOLUTE;
46328860823ad34d41d4f58561dc14a982fb0843fddReid Kleckner    else if (ResSymData.Fragment != NULL)
4644974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne      coff_symbol->Section =
4654974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne        SectionMap[&ResSymData.Fragment->getParent()->getSection()];
4664974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne
4674974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    coff_symbol->MCData = &ResSymData;
4684974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne  }
469801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
470801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
471a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer/// making a section real involves assigned it a number and putting
472a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer/// name into the string table if needed
473a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencervoid WinCOFFObjectWriter::MakeSectionReal(COFFSection &S, size_t Number) {
474a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  if (S.Name.size() > COFF::NameSize) {
475c63dce3c59ac24b2656e06f7017cd4dce4bf733cNico Rieck    const unsigned Max6DecimalSize = 999999;
476c63dce3c59ac24b2656e06f7017cd4dce4bf733cNico Rieck    const unsigned Max7DecimalSize = 9999999;
477c63dce3c59ac24b2656e06f7017cd4dce4bf733cNico Rieck    uint64_t StringTableEntry = Strings.insert(S.Name.c_str());
478c63dce3c59ac24b2656e06f7017cd4dce4bf733cNico Rieck
479c63dce3c59ac24b2656e06f7017cd4dce4bf733cNico Rieck    if (StringTableEntry <= Max6DecimalSize) {
480c63dce3c59ac24b2656e06f7017cd4dce4bf733cNico Rieck      std::sprintf(S.Header.Name, "/%d", unsigned(StringTableEntry));
481c63dce3c59ac24b2656e06f7017cd4dce4bf733cNico Rieck    } else if (StringTableEntry <= Max7DecimalSize) {
482c63dce3c59ac24b2656e06f7017cd4dce4bf733cNico Rieck      // With seven digits, we have to skip the terminating null. Because
483c63dce3c59ac24b2656e06f7017cd4dce4bf733cNico Rieck      // sprintf always appends it, we use a larger temporary buffer.
484c63dce3c59ac24b2656e06f7017cd4dce4bf733cNico Rieck      char buffer[9] = { };
485c63dce3c59ac24b2656e06f7017cd4dce4bf733cNico Rieck      std::sprintf(buffer, "/%d", unsigned(StringTableEntry));
486c63dce3c59ac24b2656e06f7017cd4dce4bf733cNico Rieck      std::memcpy(S.Header.Name, buffer, 8);
487c63dce3c59ac24b2656e06f7017cd4dce4bf733cNico Rieck    } else {
488c63dce3c59ac24b2656e06f7017cd4dce4bf733cNico Rieck      report_fatal_error("COFF string table is greater than 9,999,999 bytes.");
489c63dce3c59ac24b2656e06f7017cd4dce4bf733cNico Rieck    }
490a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  } else
491a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    std::memcpy(S.Header.Name, S.Name.c_str(), S.Name.size());
492a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
493a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  S.Number = Number;
494a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  S.Symbol->Data.SectionNumber = S.Number;
495a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  S.Symbol->Aux[0].Aux.SectionDefinition.Number = S.Number;
496a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer}
497a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
498a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencervoid WinCOFFObjectWriter::MakeSymbolReal(COFFSymbol &S, size_t Index) {
499a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  if (S.Name.size() > COFF::NameSize) {
500a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    size_t StringTableEntry = Strings.insert(S.Name.c_str());
501a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
502a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    S.set_name_offset(StringTableEntry);
503a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  } else
504a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    std::memcpy(S.Data.Name, S.Name.c_str(), S.Name.size());
505a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  S.Index = Index;
506a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer}
507a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
508a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Klecknerbool WinCOFFObjectWriter::ExportSymbol(MCSymbolData const &SymbolData,
509a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner                                       MCAssembler &Asm) {
510a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner  // This doesn't seem to be right. Strings referred to from the .data section
511a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner  // need symbols so they can be linked to code in the .text section right?
512a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner
513a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner  // return Asm.isSymbolLinkerVisible (&SymbolData);
514a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner
515a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner  // For now, all non-variable symbols are exported,
516a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner  // the linker will sort the rest out for us.
517a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner  return SymbolData.isExternal() || !SymbolData.getSymbol().isVariable();
518a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner}
519a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner
520a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencerbool WinCOFFObjectWriter::IsPhysicalSection(COFFSection *S) {
521a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  return (S->Header.Characteristics
522a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer         & COFF::IMAGE_SCN_CNT_UNINITIALIZED_DATA) == 0;
523801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
524801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
525801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------
526801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// entity writing methods
527801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
528801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::WriteFileHeader(const COFF::header &Header) {
529801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(Header.Machine);
530801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(Header.NumberOfSections);
531801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(Header.TimeDateStamp);
532801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(Header.PointerToSymbolTable);
533801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(Header.NumberOfSymbols);
534801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(Header.SizeOfOptionalHeader);
535801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(Header.Characteristics);
536801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
537801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
538801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::WriteSymbol(const COFFSymbol *S) {
539801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteBytes(StringRef(S->Data.Name, COFF::NameSize));
540801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S->Data.Value);
541801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(S->Data.SectionNumber);
542801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(S->Data.Type);
543801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Write8(S->Data.StorageClass);
544801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Write8(S->Data.NumberOfAuxSymbols);
545801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteAuxiliarySymbols(S->Aux);
546801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
547801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
548801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::WriteAuxiliarySymbols(
549801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                        const COFFSymbol::AuxiliarySymbols &S) {
550801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for(COFFSymbol::AuxiliarySymbols::const_iterator i = S.begin(), e = S.end();
551801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      i != e; ++i) {
552801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    switch(i->AuxType) {
553801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    case ATFunctionDefinition:
554801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.FunctionDefinition.TagIndex);
555801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.FunctionDefinition.TotalSize);
556801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.FunctionDefinition.PointerToLinenumber);
557801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.FunctionDefinition.PointerToNextFunction);
558801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.FunctionDefinition.unused));
559801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      break;
560801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    case ATbfAndefSymbol:
561801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.bfAndefSymbol.unused1));
562801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE16(i->Aux.bfAndefSymbol.Linenumber);
563801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.bfAndefSymbol.unused2));
564801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.bfAndefSymbol.PointerToNextFunction);
565801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.bfAndefSymbol.unused3));
566801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      break;
567801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    case ATWeakExternal:
568801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.WeakExternal.TagIndex);
569801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.WeakExternal.Characteristics);
570801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.WeakExternal.unused));
571801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      break;
572801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    case ATFile:
573801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteBytes(StringRef(reinterpret_cast<const char *>(i->Aux.File.FileName),
574801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                 sizeof(i->Aux.File.FileName)));
575801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      break;
576801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    case ATSectionDefinition:
577801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.SectionDefinition.Length);
578801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE16(i->Aux.SectionDefinition.NumberOfRelocations);
579801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE16(i->Aux.SectionDefinition.NumberOfLinenumbers);
580801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.SectionDefinition.CheckSum);
581801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE16(i->Aux.SectionDefinition.Number);
582801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      Write8(i->Aux.SectionDefinition.Selection);
583801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.SectionDefinition.unused));
584801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      break;
585801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
586801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
587801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
588801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
589801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::WriteSectionHeader(const COFF::section &S) {
590801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteBytes(StringRef(S.Name, COFF::NameSize));
591801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
592801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.VirtualSize);
593801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.VirtualAddress);
594801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.SizeOfRawData);
595801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.PointerToRawData);
596801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.PointerToRelocations);
597801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.PointerToLineNumbers);
598801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(S.NumberOfRelocations);
599801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(S.NumberOfLineNumbers);
600801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.Characteristics);
601801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
602801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
603801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::WriteRelocation(const COFF::relocation &R) {
604801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(R.VirtualAddress);
605801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(R.SymbolTableIndex);
606801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(R.Type);
607b162290e39afd49d4c7d342333b331bc96232720Chris Lattner}
608b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
609b162290e39afd49d4c7d342333b331bc96232720Chris Lattner////////////////////////////////////////////////////////////////////////////////
610b162290e39afd49d4c7d342333b331bc96232720Chris Lattner// MCObjectWriter interface implementations
611b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
61285f2ecc697a8ca6c8cf08093054cbbb9d2060ccfRafael Espindolavoid WinCOFFObjectWriter::ExecutePostLayoutBinding(MCAssembler &Asm,
61385f2ecc697a8ca6c8cf08093054cbbb9d2060ccfRafael Espindola                                                   const MCAsmLayout &Layout) {
614801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // "Define" each section & symbol. This creates section & symbol
615a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  // entries in the staging area.
616801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
617801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (MCAssembler::const_iterator i = Asm.begin(), e = Asm.end(); i != e; i++)
618801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    DefineSection(*i);
619801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
620801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (MCAssembler::const_symbol_iterator i = Asm.symbol_begin(),
6213168868bb91ac871dbb83c879e763d39a39e607eReid Kleckner                                          e = Asm.symbol_end();
622a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner       i != e; i++) {
623a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner    if (ExportSymbol(*i, Asm)) {
624a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner      DefineSymbol(*i, Asm, Layout);
625a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner    }
626a7b7a7d629c3101f6f6c87e6848e865734e0238cReid Kleckner  }
627b162290e39afd49d4c7d342333b331bc96232720Chris Lattner}
628b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
629b162290e39afd49d4c7d342333b331bc96232720Chris Lattnervoid WinCOFFObjectWriter::RecordRelocation(const MCAssembler &Asm,
630b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                           const MCAsmLayout &Layout,
631b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                           const MCFragment *Fragment,
632b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                           const MCFixup &Fixup,
633b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                           MCValue Target,
634b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                           uint64_t &FixedValue) {
635801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  assert(Target.getSymA() != NULL && "Relocation must reference a symbol!");
63682c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer
6376057eb7ab697fcd0feb3cdd55e9a497cfe0aff72Reid Kleckner  const MCSymbol &Symbol = Target.getSymA()->getSymbol();
6386057eb7ab697fcd0feb3cdd55e9a497cfe0aff72Reid Kleckner  const MCSymbol &A = Symbol.AliasedSymbol();
6396057eb7ab697fcd0feb3cdd55e9a497cfe0aff72Reid Kleckner  MCSymbolData &A_SD = Asm.getSymbolData(A);
640801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
641801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  MCSectionData const *SectionData = Fragment->getParent();
642801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
64382c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer  // Mark this symbol as requiring an entry in the symbol table.
6444cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  assert(SectionMap.find(&SectionData->getSection()) != SectionMap.end() &&
645801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer         "Section must already have been defined in ExecutePostLayoutBinding!");
6464cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  assert(SymbolMap.find(&A_SD.getSymbol()) != SymbolMap.end() &&
647801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer         "Symbol must already have been defined in ExecutePostLayoutBinding!");
648801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
6494cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  COFFSection *coff_section = SectionMap[&SectionData->getSection()];
6504cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  COFFSymbol *coff_symbol = SymbolMap[&A_SD.getSymbol()];
6513660a847f1820d73847539f3959dc069396f8e44Rafael Espindola  const MCSymbolRefExpr *SymA = Target.getSymA();
6523660a847f1820d73847539f3959dc069396f8e44Rafael Espindola  const MCSymbolRefExpr *SymB = Target.getSymB();
6533660a847f1820d73847539f3959dc069396f8e44Rafael Espindola  const bool CrossSection = SymB &&
6543660a847f1820d73847539f3959dc069396f8e44Rafael Espindola    &SymA->getSymbol().getSection() != &SymB->getSymbol().getSection();
65582c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer
65682c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer  if (Target.getSymB()) {
65782c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer    const MCSymbol *B = &Target.getSymB()->getSymbol();
65882c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer    MCSymbolData &B_SD = Asm.getSymbolData(*B);
659801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
6601ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola    // Offset of the symbol in the section
6611ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola    int64_t a = Layout.getSymbolOffset(&B_SD);
66282c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer
6631ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola    // Ofeset of the relocation in the section
6641ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola    int64_t b = Layout.getFragmentOffset(Fragment) + Fixup.getOffset();
6651ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola
6661ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola    FixedValue = b - a;
66782c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer    // In the case where we have SymbA and SymB, we just need to store the delta
66882c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer    // between the two symbols.  Update FixedValue to account for the delta, and
66982c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer    // skip recording the relocation.
6703660a847f1820d73847539f3959dc069396f8e44Rafael Espindola    if (!CrossSection)
6713660a847f1820d73847539f3959dc069396f8e44Rafael Espindola      return;
67282c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer  } else {
67382c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer    FixedValue = Target.getConstant();
67482c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer  }
675801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
676801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFRelocation Reloc;
677801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
678425f634917542d7f09c189e2eb130752c6a12d2cDaniel Dunbar  Reloc.Data.SymbolTableIndex = 0;
679801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Reloc.Data.VirtualAddress = Layout.getFragmentOffset(Fragment);
680a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
681ea1104a4c376634ed8eb693f6c71e0ac51074949Michael J. Spencer  // Turn relocations for temporary symbols into section relocations.
6823660a847f1820d73847539f3959dc069396f8e44Rafael Espindola  if (coff_symbol->MCData->getSymbol().isTemporary() || CrossSection) {
683a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    Reloc.Symb = coff_symbol->Section->Symbol;
684eb6e77f8cccd14cdba995ff8231f2c9faea9bfccMichael J. Spencer    FixedValue += Layout.getFragmentOffset(coff_symbol->MCData->Fragment)
685eb6e77f8cccd14cdba995ff8231f2c9faea9bfccMichael J. Spencer                + coff_symbol->MCData->getOffset();
686a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  } else
687a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    Reloc.Symb = coff_symbol;
688a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
689a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  ++Reloc.Symb->Relocations;
690801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
691801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Reloc.Data.VirtualAddress += Fixup.getOffset();
69218d49acdab79d6f0966b47182b6c3a2ba3d9f80fNico Rieck  Reloc.Data.Type = TargetObjectWriter->getRelocType(Target, Fixup,
69318d49acdab79d6f0966b47182b6c3a2ba3d9f80fNico Rieck                                                     CrossSection);
694b156c5d3765637393eda28b04b7fc1e5c55675ebRafael Espindola
695b156c5d3765637393eda28b04b7fc1e5c55675ebRafael Espindola  // FIXME: Can anyone explain what this does other than adjust for the size
696b156c5d3765637393eda28b04b7fc1e5c55675ebRafael Espindola  // of the offset?
697b156c5d3765637393eda28b04b7fc1e5c55675ebRafael Espindola  if (Reloc.Data.Type == COFF::IMAGE_REL_AMD64_REL32 ||
698b156c5d3765637393eda28b04b7fc1e5c55675ebRafael Espindola      Reloc.Data.Type == COFF::IMAGE_REL_I386_REL32)
69982c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer    FixedValue += 4;
700801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
701801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_section->Relocations.push_back(Reloc);
702b162290e39afd49d4c7d342333b331bc96232720Chris Lattner}
703b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
7048f413fa9c00dbaea000ddfe265ab5edd285ea3abRafael Espindolavoid WinCOFFObjectWriter::WriteObject(MCAssembler &Asm,
705b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                      const MCAsmLayout &Layout) {
706801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Assign symbol and section indexes and offsets.
707a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  Header.NumberOfSections = 0;
708a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
70980646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck  DenseMap<COFFSection *, uint16_t> SectionIndices;
710a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  for (sections::iterator i = Sections.begin(),
711a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer                          e = Sections.end(); i != e; i++) {
71285f2ecc697a8ca6c8cf08093054cbbb9d2060ccfRafael Espindola    if (Layout.getSectionAddressSize((*i)->MCData) > 0) {
71380646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck      size_t Number = ++Header.NumberOfSections;
71480646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck      SectionIndices[*i] = Number;
71580646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck      MakeSectionReal(**i, Number);
716a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    } else {
717a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      (*i)->Number = -1;
718a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    }
719a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  }
720801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
721801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Header.NumberOfSymbols = 0;
722801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
723801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (symbols::iterator i = Symbols.begin(), e = Symbols.end(); i != e; i++) {
724801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    COFFSymbol *coff_symbol = *i;
725801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    MCSymbolData const *SymbolData = coff_symbol->MCData;
726801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
727801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    // Update section number & offset for symbols that have them.
728801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    if ((SymbolData != NULL) && (SymbolData->Fragment != NULL)) {
729a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      assert(coff_symbol->Section != NULL);
730801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
731a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      coff_symbol->Data.SectionNumber = coff_symbol->Section->Number;
732237f8fe5df628065874b8590b364d04dfc2686fdMichael J. Spencer      coff_symbol->Data.Value = Layout.getFragmentOffset(SymbolData->Fragment)
733237f8fe5df628065874b8590b364d04dfc2686fdMichael J. Spencer                              + SymbolData->Offset;
734801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
735801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
736a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    if (coff_symbol->should_keep()) {
737a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      MakeSymbolReal(*coff_symbol, Header.NumberOfSymbols++);
738a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
739a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      // Update auxiliary symbol info.
740a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      coff_symbol->Data.NumberOfAuxSymbols = coff_symbol->Aux.size();
741a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      Header.NumberOfSymbols += coff_symbol->Data.NumberOfAuxSymbols;
742a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    } else
743a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      coff_symbol->Index = -1;
744801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
745801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
746801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Fixup weak external references.
747801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (symbols::iterator i = Symbols.begin(), e = Symbols.end(); i != e; i++) {
748a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    COFFSymbol *coff_symbol = *i;
749a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    if (coff_symbol->Other != NULL) {
750a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      assert(coff_symbol->Index != -1);
751a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      assert(coff_symbol->Aux.size() == 1 &&
752801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer             "Symbol must contain one aux symbol!");
753a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      assert(coff_symbol->Aux[0].AuxType == ATWeakExternal &&
754801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer             "Symbol's aux symbol must be a Weak External!");
755a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      coff_symbol->Aux[0].Aux.WeakExternal.TagIndex = coff_symbol->Other->Index;
756801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
757801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
758801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
75980646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck  // Fixup associative COMDAT sections.
76080646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck  for (sections::iterator i = Sections.begin(),
76180646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck                          e = Sections.end(); i != e; i++) {
76280646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck    if ((*i)->Symbol->Aux[0].Aux.SectionDefinition.Selection !=
76380646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck        COFF::IMAGE_COMDAT_SELECT_ASSOCIATIVE)
76480646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck      continue;
76580646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck
76680646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck    const MCSectionCOFF &MCSec = static_cast<const MCSectionCOFF &>(
76780646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck                                                    (*i)->MCData->getSection());
76880646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck
76980646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck    COFFSection *Assoc = SectionMap.lookup(MCSec.getAssocSection());
77080646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck    if (!Assoc) {
77180646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck      report_fatal_error(Twine("Missing associated COMDAT section ") +
77280646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck                         MCSec.getAssocSection()->getSectionName() +
77380646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck                         " for section " + MCSec.getSectionName());
77480646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck    }
77580646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck
77680646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck    // Skip this section if the associated section is unused.
77780646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck    if (Assoc->Number == -1)
77880646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck      continue;
77980646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck
78080646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck    (*i)->Symbol->Aux[0].Aux.SectionDefinition.Number = SectionIndices[Assoc];
78180646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck  }
78280646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck
78380646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck
784801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Assign file offsets to COFF object file structures.
785801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
786801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  unsigned offset = 0;
787801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
788801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  offset += COFF::HeaderSize;
789a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  offset += COFF::SectionSize * Header.NumberOfSections;
790801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
791801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (MCAssembler::const_iterator i = Asm.begin(),
792801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                   e = Asm.end();
793801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                   i != e; i++) {
7944cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer    COFFSection *Sec = SectionMap[&i->getSection()];
795801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
796a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    if (Sec->Number == -1)
797a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      continue;
798a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
79928ca86aa19fe2a5493573164ef0c2c54542ed9daMichael J. Spencer    Sec->Header.SizeOfRawData = Layout.getSectionAddressSize(i);
800801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
801a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    if (IsPhysicalSection(Sec)) {
802801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      Sec->Header.PointerToRawData = offset;
803801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
804801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      offset += Sec->Header.SizeOfRawData;
805801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
806801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
807801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    if (Sec->Relocations.size() > 0) {
808d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer      bool RelocationsOverflow = Sec->Relocations.size() >= 0xffff;
809d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer
810d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer      if (RelocationsOverflow) {
811d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        // Signal overflow by setting NumberOfSections to max value. Actual
812d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        // size is found in reloc #0. Microsoft tools understand this.
813d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        Sec->Header.NumberOfRelocations = 0xffff;
814d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer      } else {
815d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        Sec->Header.NumberOfRelocations = Sec->Relocations.size();
816d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer      }
817801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      Sec->Header.PointerToRelocations = offset;
818801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
819d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer      if (RelocationsOverflow) {
820d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        // Reloc #0 will contain actual count, so make room for it.
821d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        offset += COFF::RelocationSize;
822d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer      }
823d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer
824801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      offset += COFF::RelocationSize * Sec->Relocations.size();
825801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
826801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      for (relocations::iterator cr = Sec->Relocations.begin(),
827801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                 er = Sec->Relocations.end();
828a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer                                 cr != er; ++cr) {
829a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer        assert((*cr).Symb->Index != -1);
830801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        (*cr).Data.SymbolTableIndex = (*cr).Symb->Index;
831801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      }
832801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
833801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
834a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    assert(Sec->Symbol->Aux.size() == 1
835a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      && "Section's symbol must have one aux!");
836a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    AuxSymbol &Aux = Sec->Symbol->Aux[0];
837801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    assert(Aux.AuxType == ATSectionDefinition &&
838801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer           "Section's symbol's aux symbol must be a Section Definition!");
839801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    Aux.Aux.SectionDefinition.Length = Sec->Header.SizeOfRawData;
840801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    Aux.Aux.SectionDefinition.NumberOfRelocations =
841801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                                Sec->Header.NumberOfRelocations;
842801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    Aux.Aux.SectionDefinition.NumberOfLinenumbers =
843801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                                Sec->Header.NumberOfLineNumbers;
844801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
845801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
846801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Header.PointerToSymbolTable = offset;
847801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
848ab3de49c48bd3282421ce24323fb6b868a3da6ccMichael J. Spencer  Header.TimeDateStamp = sys::TimeValue::now().toEpochTime();
849ab3de49c48bd3282421ce24323fb6b868a3da6ccMichael J. Spencer
850801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Write it all to disk...
851801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteFileHeader(Header);
852801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
853801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  {
854801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    sections::iterator i, ie;
855801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    MCAssembler::const_iterator j, je;
856801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
857801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    for (i = Sections.begin(), ie = Sections.end(); i != ie; i++)
858d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer      if ((*i)->Number != -1) {
859d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        if ((*i)->Relocations.size() >= 0xffff) {
860d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer          (*i)->Header.Characteristics |= COFF::IMAGE_SCN_LNK_NRELOC_OVFL;
861d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        }
862a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer        WriteSectionHeader((*i)->Header);
863d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer      }
864801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
865801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    for (i = Sections.begin(), ie = Sections.end(),
866801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer         j = Asm.begin(), je = Asm.end();
867a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer         (i != ie) && (j != je); ++i, ++j) {
868a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
869a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      if ((*i)->Number == -1)
870a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer        continue;
871a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
872801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      if ((*i)->Header.PointerToRawData != 0) {
873801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        assert(OS.tell() == (*i)->Header.PointerToRawData &&
874801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer               "Section::PointerToRawData is insane!");
875801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
876f77d5b14af6b92403b93f7ed249f9023e99028ccJim Grosbach        Asm.writeSectionData(j, Layout);
877801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      }
878801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
879801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      if ((*i)->Relocations.size() > 0) {
880801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        assert(OS.tell() == (*i)->Header.PointerToRelocations &&
881801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer               "Section::PointerToRelocations is insane!");
882801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
883d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        if ((*i)->Relocations.size() >= 0xffff) {
884d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer          // In case of overflow, write actual relocation count as first
885d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer          // relocation. Including the synthetic reloc itself (+ 1).
886d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer          COFF::relocation r;
887d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer          r.VirtualAddress = (*i)->Relocations.size() + 1;
888d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer          r.SymbolTableIndex = 0;
889d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer          r.Type = 0;
890d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer          WriteRelocation(r);
891d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        }
892d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer
893801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        for (relocations::const_iterator k = (*i)->Relocations.begin(),
894801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                               ke = (*i)->Relocations.end();
895801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                               k != ke; k++) {
896801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer          WriteRelocation(k->Data);
897801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        }
898801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      } else
899801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        assert((*i)->Header.PointerToRelocations == 0 &&
900801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer               "Section::PointerToRelocations is insane!");
901801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
902801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
903801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
904801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  assert(OS.tell() == Header.PointerToSymbolTable &&
905801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer         "Header::PointerToSymbolTable is insane!");
906801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
907801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (symbols::iterator i = Symbols.begin(), e = Symbols.end(); i != e; i++)
908a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    if ((*i)->Index != -1)
909a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      WriteSymbol(*i);
910801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
911801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  OS.write((char const *)&Strings.Data.front(), Strings.Data.size());
912b162290e39afd49d4c7d342333b331bc96232720Chris Lattner}
913b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
914df09270ae897e7fa64a7c162de163c32ee181a03Rafael EspindolaMCWinCOFFObjectTargetWriter::MCWinCOFFObjectTargetWriter(unsigned Machine_) :
915df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola  Machine(Machine_) {
916df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola}
917df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola
918b162290e39afd49d4c7d342333b331bc96232720Chris Lattner//------------------------------------------------------------------------------
919b162290e39afd49d4c7d342333b331bc96232720Chris Lattner// WinCOFFObjectWriter factory function
920b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
921b162290e39afd49d4c7d342333b331bc96232720Chris Lattnernamespace llvm {
922df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola  MCObjectWriter *createWinCOFFObjectWriter(MCWinCOFFObjectTargetWriter *MOTW,
923df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola                                            raw_ostream &OS) {
924df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola    return new WinCOFFObjectWriter(MOTW, OS);
925b162290e39afd49d4c7d342333b331bc96232720Chris Lattner  }
926933304ef0c3ec18c23d0b385c2117a6eae790430Michael J. Spencer}
927