WinCOFFObjectWriter.cpp revision 6057eb7ab697fcd0feb3cdd55e9a497cfe0aff72
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);
141808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer  ~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);
1514974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne  void DefineSymbol(MCSymbolData const &SymbolData,
1520933134a304b47d3767aad202df9f0e09743da6dMichael J. Spencer                    MCAssembler &Assembler);
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
157a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  bool ExportSection(COFFSection const *S);
158801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  bool ExportSymbol(MCSymbolData const &SymbolData, MCAssembler &Asm);
159801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
160a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  bool IsPhysicalSection(COFFSection *S);
161a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
162801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Entity writing methods.
163801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
164801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void WriteFileHeader(const COFF::header &Header);
165801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void WriteSymbol(const COFFSymbol *S);
166801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void WriteAuxiliarySymbols(const COFFSymbol::AuxiliarySymbols &S);
167801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void WriteSectionHeader(const COFF::section &S);
168801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void WriteRelocation(const COFF::relocation &R);
169801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
170801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // MCObjectWriter interface implementation.
171801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
17285f2ecc697a8ca6c8cf08093054cbbb9d2060ccfRafael Espindola  void ExecutePostLayoutBinding(MCAssembler &Asm, const MCAsmLayout &Layout);
173801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
174801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void RecordRelocation(const MCAssembler &Asm,
175801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                        const MCAsmLayout &Layout,
176801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                        const MCFragment *Fragment,
177801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                        const MCFixup &Fixup,
178801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                        MCValue Target,
179801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                        uint64_t &FixedValue);
180801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
1818f413fa9c00dbaea000ddfe265ab5edd285ea3abRafael Espindola  void WriteObject(MCAssembler &Asm, const MCAsmLayout &Layout);
182801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer};
183801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
184801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
185801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerstatic inline void write_uint32_le(void *Data, uint32_t const &Value) {
186801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  uint8_t *Ptr = reinterpret_cast<uint8_t *>(Data);
187801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[0] = (Value & 0x000000FF) >>  0;
188801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[1] = (Value & 0x0000FF00) >>  8;
189801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[2] = (Value & 0x00FF0000) >> 16;
190801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[3] = (Value & 0xFF000000) >> 24;
191801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
192801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
193801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerstatic inline void write_uint16_le(void *Data, uint16_t const &Value) {
194801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  uint8_t *Ptr = reinterpret_cast<uint8_t *>(Data);
195801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[0] = (Value & 0x00FF) >> 0;
196801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[1] = (Value & 0xFF00) >> 8;
197801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
198801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
199801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerstatic inline void write_uint8_le(void *Data, uint8_t const &Value) {
200801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  uint8_t *Ptr = reinterpret_cast<uint8_t *>(Data);
201801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[0] = (Value & 0xFF) >> 0;
202801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
203801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
204801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------
205801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// Symbol class implementation
206801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
20796f498bd9f140a98321c478f517877c4767b94faDmitri GribenkoCOFFSymbol::COFFSymbol(StringRef name)
208a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  : Name(name.begin(), name.end())
209a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  , Other(NULL)
210a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  , Section(NULL)
211a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  , Relocations(0)
212a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  , MCData(NULL) {
213801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  memset(&Data, 0, sizeof(Data));
214801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
215b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
216801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencersize_t COFFSymbol::size() const {
217801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return COFF::SymbolSize + (Data.NumberOfAuxSymbols * COFF::SymbolSize);
218b162290e39afd49d4c7d342333b331bc96232720Chris Lattner}
219b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
220801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// In the case that the name does not fit within 8 bytes, the offset
221801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// into the string table is stored in the last 4 bytes instead, leaving
222801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// the first 4 bytes as 0.
223801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid COFFSymbol::set_name_offset(uint32_t Offset) {
224801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  write_uint32_le(Data.Name + 0, 0);
225801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  write_uint32_le(Data.Name + 4, Offset);
226801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
227801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
228a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer/// logic to decide if the symbol should be reported in the symbol table
229a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencerbool COFFSymbol::should_keep() const {
230a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  // no section means its external, keep it
231a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  if (Section == NULL)
232a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    return true;
233a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
234a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  // if it has relocations pointing at it, keep it
235a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  if (Relocations > 0)   {
236a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    assert(Section->Number != -1 && "Sections with relocations must be real!");
237a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    return true;
238a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  }
239a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
240a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  // if the section its in is being droped, drop it
241a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  if (Section->Number == -1)
242a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      return false;
243a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
244a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  // if it is the section symbol, keep it
245a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  if (Section->Symbol == this)
246a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    return true;
247a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
248a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  // if its temporary, drop it
249a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  if (MCData && MCData->getSymbol().isTemporary())
250a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      return false;
251a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
252a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  // otherwise, keep it
253a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  return true;
254a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer}
255a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
256801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------
257801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// Section class implementation
258801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
25996f498bd9f140a98321c478f517877c4767b94faDmitri GribenkoCOFFSection::COFFSection(StringRef name)
260a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  : Name(name)
261a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  , MCData(NULL)
262a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  , Symbol(NULL) {
263801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  memset(&Header, 0, sizeof(Header));
264801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
265801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
266801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencersize_t COFFSection::size() {
267801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return COFF::SectionSize;
268801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
269801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
270801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------
271801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// StringTable class implementation
272801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
273801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// Write the length of the string table into Data.
274801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// The length of the string table includes uint32 length header.
275801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid StringTable::update_length() {
276801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  write_uint32_le(&Data.front(), Data.size());
277801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
278801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
279801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. SpencerStringTable::StringTable() {
280801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // The string table data begins with the length of the entire string table
281801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // including the length header. Allocate space for this header.
282801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Data.resize(4);
2830d64632c9e8a2632b8804bcc3fbc3523588862a7Michael J. Spencer  update_length();
284801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
285801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
286801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencersize_t StringTable::size() const {
287801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return Data.size();
288801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
289801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
29094c22716d60ff5edf6a98a3c67e0faa001be1142Sylvestre Ledru/// Add String to the table iff it is not already there.
291801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// @returns the index into the string table where the string is now located.
29296f498bd9f140a98321c478f517877c4767b94faDmitri Gribenkosize_t StringTable::insert(StringRef String) {
293801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  map::iterator i = Map.find(String);
294801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
295801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  if (i != Map.end())
296801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    return i->second;
297801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
298801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  size_t Offset = Data.size();
299801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
300801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Insert string data into string table.
301801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Data.insert(Data.end(), String.begin(), String.end());
302801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Data.push_back('\0');
303801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
304801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Put a reference to it in the map.
305801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Map[String] = Offset;
306801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
307801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Update the internal length field.
308801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  update_length();
309801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
310801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return Offset;
311801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
312801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
313801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------
314801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// WinCOFFObjectWriter class implementation
315801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
316df09270ae897e7fa64a7c162de163c32ee181a03Rafael EspindolaWinCOFFObjectWriter::WinCOFFObjectWriter(MCWinCOFFObjectTargetWriter *MOTW,
317df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola                                         raw_ostream &OS)
31882c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer  : MCObjectWriter(OS, true)
319df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola  , TargetObjectWriter(MOTW) {
320801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  memset(&Header, 0, sizeof(Header));
321da0bfcdaf95d95a66e306ef6d45f638939272d34Michael J. Spencer
322df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola  Header.Machine = TargetObjectWriter->getMachine();
323801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
324801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
325808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin KramerWinCOFFObjectWriter::~WinCOFFObjectWriter() {
326808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer  for (symbols::iterator I = Symbols.begin(), E = Symbols.end(); I != E; ++I)
327808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer    delete *I;
328808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer  for (sections::iterator I = Sections.begin(), E = Sections.end(); I != E; ++I)
329808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer    delete *I;
330808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer}
331808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer
3324cee2890a66974af506f2125243114cc14bd5556Michael J. SpencerCOFFSymbol *WinCOFFObjectWriter::createSymbol(StringRef Name) {
333801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return createCOFFEntity<COFFSymbol>(Name, Symbols);
334801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
335801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
3364cee2890a66974af506f2125243114cc14bd5556Michael J. SpencerCOFFSymbol *WinCOFFObjectWriter::GetOrCreateCOFFSymbol(const MCSymbol * Symbol){
3374cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  symbol_map::iterator i = SymbolMap.find(Symbol);
3384cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  if (i != SymbolMap.end())
3394cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer    return i->second;
3404cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  COFFSymbol *RetSymbol
3414cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer    = createCOFFEntity<COFFSymbol>(Symbol->getName(), Symbols);
3424cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  SymbolMap[Symbol] = RetSymbol;
3434cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  return RetSymbol;
3444cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer}
3454cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer
34696f498bd9f140a98321c478f517877c4767b94faDmitri GribenkoCOFFSection *WinCOFFObjectWriter::createSection(StringRef Name) {
347801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return createCOFFEntity<COFFSection>(Name, Sections);
348801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
349801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
350801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// A template used to lookup or create a symbol/section, and initialize it if
351801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// needed.
352801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencertemplate <typename object_t, typename list_t>
35396f498bd9f140a98321c478f517877c4767b94faDmitri Gribenkoobject_t *WinCOFFObjectWriter::createCOFFEntity(StringRef Name,
354801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                                list_t &List) {
355a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  object_t *Object = new object_t(Name);
356801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
357801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  List.push_back(Object);
358801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
359801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return Object;
360801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
361801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
362801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// This function takes a section data object from the assembler
363801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// and creates the associated COFF section staging object.
364801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::DefineSection(MCSectionData const &SectionData) {
365d47f4a9c982d264e46a6a2fe0f357288768bb5b9Michael J. Spencer  assert(SectionData.getSection().getVariant() == MCSection::SV_COFF
366d47f4a9c982d264e46a6a2fe0f357288768bb5b9Michael J. Spencer    && "Got non COFF section in the COFF backend!");
367801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // FIXME: Not sure how to verify this (at least in a debug build).
368801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  MCSectionCOFF const &Sec =
369801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    static_cast<MCSectionCOFF const &>(SectionData.getSection());
370801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
371801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSection *coff_section = createSection(Sec.getSectionName());
372801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSymbol  *coff_symbol = createSymbol(Sec.getSectionName());
373801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
374a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  coff_section->Symbol = coff_symbol;
375a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  coff_symbol->Section = coff_section;
376801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->Data.StorageClass = COFF::IMAGE_SYM_CLASS_STATIC;
377801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
378801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // In this case the auxiliary symbol is a Section Definition.
379801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->Aux.resize(1);
380801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  memset(&coff_symbol->Aux[0], 0, sizeof(coff_symbol->Aux[0]));
381801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->Aux[0].AuxType = ATSectionDefinition;
382801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->Aux[0].Aux.SectionDefinition.Selection = Sec.getSelection();
383801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
384801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_section->Header.Characteristics = Sec.getCharacteristics();
385801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
386801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  uint32_t &Characteristics = coff_section->Header.Characteristics;
387801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  switch (SectionData.getAlignment()) {
388801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 1:    Characteristics |= COFF::IMAGE_SCN_ALIGN_1BYTES;    break;
389801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 2:    Characteristics |= COFF::IMAGE_SCN_ALIGN_2BYTES;    break;
390801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 4:    Characteristics |= COFF::IMAGE_SCN_ALIGN_4BYTES;    break;
391801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 8:    Characteristics |= COFF::IMAGE_SCN_ALIGN_8BYTES;    break;
392801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 16:   Characteristics |= COFF::IMAGE_SCN_ALIGN_16BYTES;   break;
393801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 32:   Characteristics |= COFF::IMAGE_SCN_ALIGN_32BYTES;   break;
394801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 64:   Characteristics |= COFF::IMAGE_SCN_ALIGN_64BYTES;   break;
395801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 128:  Characteristics |= COFF::IMAGE_SCN_ALIGN_128BYTES;  break;
396801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 256:  Characteristics |= COFF::IMAGE_SCN_ALIGN_256BYTES;  break;
397801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 512:  Characteristics |= COFF::IMAGE_SCN_ALIGN_512BYTES;  break;
398801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 1024: Characteristics |= COFF::IMAGE_SCN_ALIGN_1024BYTES; break;
399801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 2048: Characteristics |= COFF::IMAGE_SCN_ALIGN_2048BYTES; break;
400801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 4096: Characteristics |= COFF::IMAGE_SCN_ALIGN_4096BYTES; break;
401801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 8192: Characteristics |= COFF::IMAGE_SCN_ALIGN_8192BYTES; break;
402801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  default:
403801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    llvm_unreachable("unsupported section alignment");
404801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
405801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
406801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Bind internal COFF section to MC section.
407801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_section->MCData = &SectionData;
4084cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  SectionMap[&SectionData.getSection()] = coff_section;
409801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
410801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
411801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// This function takes a section data object from the assembler
412801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// and creates the associated COFF symbol staging object.
4134974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbournevoid WinCOFFObjectWriter::DefineSymbol(MCSymbolData const &SymbolData,
414a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer                                       MCAssembler &Assembler) {
4154974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne  MCSymbol const &Symbol = SymbolData.getSymbol();
4160933134a304b47d3767aad202df9f0e09743da6dMichael J. Spencer  COFFSymbol *coff_symbol = GetOrCreateCOFFSymbol(&Symbol);
4174974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne  SymbolMap[&Symbol] = coff_symbol;
418801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
4194cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  if (SymbolData.getFlags() & COFF::SF_WeakExternal) {
420801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Data.StorageClass = COFF::IMAGE_SYM_CLASS_WEAK_EXTERNAL;
421801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
4220933134a304b47d3767aad202df9f0e09743da6dMichael J. Spencer    if (Symbol.isVariable()) {
4234974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne      const MCSymbolRefExpr *SymRef =
4244974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne        dyn_cast<MCSymbolRefExpr>(Symbol.getVariableValue());
425801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
4264974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne      if (!SymRef)
4274974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne        report_fatal_error("Weak externals may only alias symbols");
428801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
4294974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne      coff_symbol->Other = GetOrCreateCOFFSymbol(&SymRef->getSymbol());
4304cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer    } else {
4314cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer      std::string WeakName = std::string(".weak.")
4320933134a304b47d3767aad202df9f0e09743da6dMichael J. Spencer                           +  Symbol.getName().str()
4334cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer                           + ".default";
4344cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer      COFFSymbol *WeakDefault = createSymbol(WeakName);
4354cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer      WeakDefault->Data.SectionNumber = COFF::IMAGE_SYM_ABSOLUTE;
4364cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer      WeakDefault->Data.StorageClass  = COFF::IMAGE_SYM_CLASS_EXTERNAL;
4374cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer      WeakDefault->Data.Type          = 0;
4384cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer      WeakDefault->Data.Value         = 0;
4394cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer      coff_symbol->Other = WeakDefault;
4404cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer    }
441801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
442801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    // Setup the Weak External auxiliary symbol.
443801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Aux.resize(1);
444801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    memset(&coff_symbol->Aux[0], 0, sizeof(coff_symbol->Aux[0]));
445801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Aux[0].AuxType = ATWeakExternal;
446801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Aux[0].Aux.WeakExternal.TagIndex = 0;
447801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Aux[0].Aux.WeakExternal.Characteristics =
4484cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer      COFF::IMAGE_WEAK_EXTERN_SEARCH_LIBRARY;
4494cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer
4504974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    coff_symbol->MCData = &SymbolData;
4514974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne  } else {
4524974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    const MCSymbolData &ResSymData =
4534974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne      Assembler.getSymbolData(Symbol.AliasedSymbol());
4544cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer
4554974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    coff_symbol->Data.Type         = (ResSymData.getFlags() & 0x0000FFFF) >>  0;
4564974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    coff_symbol->Data.StorageClass = (ResSymData.getFlags() & 0x00FF0000) >> 16;
457801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
4584974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    // If no storage class was specified in the streamer, define it here.
4594974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    if (coff_symbol->Data.StorageClass == 0) {
4604974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne      bool external = ResSymData.isExternal() || (ResSymData.Fragment == NULL);
461a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
4624974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne      coff_symbol->Data.StorageClass =
4634974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne       external ? COFF::IMAGE_SYM_CLASS_EXTERNAL : COFF::IMAGE_SYM_CLASS_STATIC;
4644974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    }
4654974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne
4664974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    if (ResSymData.Fragment != NULL)
4674974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne      coff_symbol->Section =
4684974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne        SectionMap[&ResSymData.Fragment->getParent()->getSection()];
4694974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne
4704974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne    coff_symbol->MCData = &ResSymData;
4714974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne  }
472801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
473801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
474a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer/// making a section real involves assigned it a number and putting
475a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer/// name into the string table if needed
476a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencervoid WinCOFFObjectWriter::MakeSectionReal(COFFSection &S, size_t Number) {
477a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  if (S.Name.size() > COFF::NameSize) {
478a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    size_t StringTableEntry = Strings.insert(S.Name.c_str());
479a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
480a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    // FIXME: Why is this number 999999? This number is never mentioned in the
481a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    // spec. I'm assuming this is due to the printed value needing to fit into
482a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    // the S.Header.Name field. In which case why not 9999999 (7 9's instead of
483a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    // 6)? The spec does not state if this entry should be null terminated in
484a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    // this case, and thus this seems to be the best way to do it. I think I
485a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    // just solved my own FIXME...
486a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    if (StringTableEntry > 999999)
487a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      report_fatal_error("COFF string table is greater than 999999 bytes.");
488a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
489a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    std::sprintf(S.Header.Name, "/%d", unsigned(StringTableEntry));
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
508a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencerbool WinCOFFObjectWriter::ExportSection(COFFSection const *S) {
509a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  return !S->MCData->getFragmentList().empty();
510801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
511801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
512801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerbool WinCOFFObjectWriter::ExportSymbol(MCSymbolData const &SymbolData,
513801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                       MCAssembler &Asm) {
514801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // This doesn't seem to be right. Strings referred to from the .data section
515801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // need symbols so they can be linked to code in the .text section right?
516801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
517801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // return Asm.isSymbolLinkerVisible (&SymbolData);
518801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
519a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  // For now, all non-variable symbols are exported,
520a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  // the linker will sort the rest out for us.
5214cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  return SymbolData.isExternal() || !SymbolData.getSymbol().isVariable();
522a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer}
523a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
524a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencerbool WinCOFFObjectWriter::IsPhysicalSection(COFFSection *S) {
525a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  return (S->Header.Characteristics
526a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer         & COFF::IMAGE_SCN_CNT_UNINITIALIZED_DATA) == 0;
527801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
528801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
529801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------
530801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// entity writing methods
531801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
532801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::WriteFileHeader(const COFF::header &Header) {
533801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(Header.Machine);
534801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(Header.NumberOfSections);
535801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(Header.TimeDateStamp);
536801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(Header.PointerToSymbolTable);
537801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(Header.NumberOfSymbols);
538801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(Header.SizeOfOptionalHeader);
539801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(Header.Characteristics);
540801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
541801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
542801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::WriteSymbol(const COFFSymbol *S) {
543801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteBytes(StringRef(S->Data.Name, COFF::NameSize));
544801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S->Data.Value);
545801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(S->Data.SectionNumber);
546801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(S->Data.Type);
547801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Write8(S->Data.StorageClass);
548801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Write8(S->Data.NumberOfAuxSymbols);
549801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteAuxiliarySymbols(S->Aux);
550801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
551801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
552801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::WriteAuxiliarySymbols(
553801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                        const COFFSymbol::AuxiliarySymbols &S) {
554801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for(COFFSymbol::AuxiliarySymbols::const_iterator i = S.begin(), e = S.end();
555801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      i != e; ++i) {
556801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    switch(i->AuxType) {
557801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    case ATFunctionDefinition:
558801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.FunctionDefinition.TagIndex);
559801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.FunctionDefinition.TotalSize);
560801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.FunctionDefinition.PointerToLinenumber);
561801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.FunctionDefinition.PointerToNextFunction);
562801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.FunctionDefinition.unused));
563801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      break;
564801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    case ATbfAndefSymbol:
565801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.bfAndefSymbol.unused1));
566801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE16(i->Aux.bfAndefSymbol.Linenumber);
567801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.bfAndefSymbol.unused2));
568801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.bfAndefSymbol.PointerToNextFunction);
569801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.bfAndefSymbol.unused3));
570801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      break;
571801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    case ATWeakExternal:
572801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.WeakExternal.TagIndex);
573801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.WeakExternal.Characteristics);
574801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.WeakExternal.unused));
575801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      break;
576801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    case ATFile:
577801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteBytes(StringRef(reinterpret_cast<const char *>(i->Aux.File.FileName),
578801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                 sizeof(i->Aux.File.FileName)));
579801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      break;
580801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    case ATSectionDefinition:
581801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.SectionDefinition.Length);
582801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE16(i->Aux.SectionDefinition.NumberOfRelocations);
583801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE16(i->Aux.SectionDefinition.NumberOfLinenumbers);
584801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.SectionDefinition.CheckSum);
585801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE16(i->Aux.SectionDefinition.Number);
586801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      Write8(i->Aux.SectionDefinition.Selection);
587801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.SectionDefinition.unused));
588801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      break;
589801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
590801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
591801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
592801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
593801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::WriteSectionHeader(const COFF::section &S) {
594801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteBytes(StringRef(S.Name, COFF::NameSize));
595801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
596801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.VirtualSize);
597801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.VirtualAddress);
598801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.SizeOfRawData);
599801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.PointerToRawData);
600801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.PointerToRelocations);
601801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.PointerToLineNumbers);
602801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(S.NumberOfRelocations);
603801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(S.NumberOfLineNumbers);
604801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.Characteristics);
605801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
606801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
607801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::WriteRelocation(const COFF::relocation &R) {
608801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(R.VirtualAddress);
609801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(R.SymbolTableIndex);
610801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(R.Type);
611b162290e39afd49d4c7d342333b331bc96232720Chris Lattner}
612b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
613b162290e39afd49d4c7d342333b331bc96232720Chris Lattner////////////////////////////////////////////////////////////////////////////////
614b162290e39afd49d4c7d342333b331bc96232720Chris Lattner// MCObjectWriter interface implementations
615b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
61685f2ecc697a8ca6c8cf08093054cbbb9d2060ccfRafael Espindolavoid WinCOFFObjectWriter::ExecutePostLayoutBinding(MCAssembler &Asm,
61785f2ecc697a8ca6c8cf08093054cbbb9d2060ccfRafael Espindola                                                   const MCAsmLayout &Layout) {
618801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // "Define" each section & symbol. This creates section & symbol
619a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  // entries in the staging area.
620801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
621801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (MCAssembler::const_iterator i = Asm.begin(), e = Asm.end(); i != e; i++)
622801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    DefineSection(*i);
623801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
624801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (MCAssembler::const_symbol_iterator i = Asm.symbol_begin(),
625801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                          e = Asm.symbol_end(); i != e; i++) {
6260933134a304b47d3767aad202df9f0e09743da6dMichael J. Spencer    if (ExportSymbol(*i, Asm)) {
6274974b972e7dd94fad74ada4df32a12aba09c4de0Peter Collingbourne      DefineSymbol(*i, Asm);
6280933134a304b47d3767aad202df9f0e09743da6dMichael J. Spencer    }
629801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
630b162290e39afd49d4c7d342333b331bc96232720Chris Lattner}
631b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
632b162290e39afd49d4c7d342333b331bc96232720Chris Lattnervoid WinCOFFObjectWriter::RecordRelocation(const MCAssembler &Asm,
633b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                           const MCAsmLayout &Layout,
634b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                           const MCFragment *Fragment,
635b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                           const MCFixup &Fixup,
636b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                           MCValue Target,
637b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                           uint64_t &FixedValue) {
638801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  assert(Target.getSymA() != NULL && "Relocation must reference a symbol!");
63982c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer
6406057eb7ab697fcd0feb3cdd55e9a497cfe0aff72Reid Kleckner  const MCSymbol &Symbol = Target.getSymA()->getSymbol();
6416057eb7ab697fcd0feb3cdd55e9a497cfe0aff72Reid Kleckner  const MCSymbol &A = Symbol.AliasedSymbol();
6426057eb7ab697fcd0feb3cdd55e9a497cfe0aff72Reid Kleckner  MCSymbolData &A_SD = Asm.getSymbolData(A);
643801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
644801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  MCSectionData const *SectionData = Fragment->getParent();
645801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
64682c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer  // Mark this symbol as requiring an entry in the symbol table.
6474cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  assert(SectionMap.find(&SectionData->getSection()) != SectionMap.end() &&
648801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer         "Section must already have been defined in ExecutePostLayoutBinding!");
6494cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  assert(SymbolMap.find(&A_SD.getSymbol()) != SymbolMap.end() &&
650801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer         "Symbol must already have been defined in ExecutePostLayoutBinding!");
651801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
6524cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  COFFSection *coff_section = SectionMap[&SectionData->getSection()];
6534cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer  COFFSymbol *coff_symbol = SymbolMap[&A_SD.getSymbol()];
6543660a847f1820d73847539f3959dc069396f8e44Rafael Espindola  const MCSymbolRefExpr *SymA = Target.getSymA();
6553660a847f1820d73847539f3959dc069396f8e44Rafael Espindola  const MCSymbolRefExpr *SymB = Target.getSymB();
6563660a847f1820d73847539f3959dc069396f8e44Rafael Espindola  const bool CrossSection = SymB &&
6573660a847f1820d73847539f3959dc069396f8e44Rafael Espindola    &SymA->getSymbol().getSection() != &SymB->getSymbol().getSection();
65882c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer
65982c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer  if (Target.getSymB()) {
66082c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer    const MCSymbol *B = &Target.getSymB()->getSymbol();
66182c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer    MCSymbolData &B_SD = Asm.getSymbolData(*B);
662801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
6631ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola    // Offset of the symbol in the section
6641ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola    int64_t a = Layout.getSymbolOffset(&B_SD);
66582c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer
6661ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola    // Ofeset of the relocation in the section
6671ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola    int64_t b = Layout.getFragmentOffset(Fragment) + Fixup.getOffset();
6681ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola
6691ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola    FixedValue = b - a;
67082c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer    // In the case where we have SymbA and SymB, we just need to store the delta
67182c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer    // between the two symbols.  Update FixedValue to account for the delta, and
67282c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer    // skip recording the relocation.
6733660a847f1820d73847539f3959dc069396f8e44Rafael Espindola    if (!CrossSection)
6743660a847f1820d73847539f3959dc069396f8e44Rafael Espindola      return;
67582c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer  } else {
67682c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer    FixedValue = Target.getConstant();
67782c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer  }
678801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
679801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFRelocation Reloc;
680801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
681425f634917542d7f09c189e2eb130752c6a12d2cDaniel Dunbar  Reloc.Data.SymbolTableIndex = 0;
682801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Reloc.Data.VirtualAddress = Layout.getFragmentOffset(Fragment);
683a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
684ea1104a4c376634ed8eb693f6c71e0ac51074949Michael J. Spencer  // Turn relocations for temporary symbols into section relocations.
6853660a847f1820d73847539f3959dc069396f8e44Rafael Espindola  if (coff_symbol->MCData->getSymbol().isTemporary() || CrossSection) {
686a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    Reloc.Symb = coff_symbol->Section->Symbol;
687eb6e77f8cccd14cdba995ff8231f2c9faea9bfccMichael J. Spencer    FixedValue += Layout.getFragmentOffset(coff_symbol->MCData->Fragment)
688eb6e77f8cccd14cdba995ff8231f2c9faea9bfccMichael J. Spencer                + coff_symbol->MCData->getOffset();
689a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  } else
690a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    Reloc.Symb = coff_symbol;
691a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
692a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  ++Reloc.Symb->Relocations;
693801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
694801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Reloc.Data.VirtualAddress += Fixup.getOffset();
69518d49acdab79d6f0966b47182b6c3a2ba3d9f80fNico Rieck  Reloc.Data.Type = TargetObjectWriter->getRelocType(Target, Fixup,
69618d49acdab79d6f0966b47182b6c3a2ba3d9f80fNico Rieck                                                     CrossSection);
697b156c5d3765637393eda28b04b7fc1e5c55675ebRafael Espindola
698b156c5d3765637393eda28b04b7fc1e5c55675ebRafael Espindola  // FIXME: Can anyone explain what this does other than adjust for the size
699b156c5d3765637393eda28b04b7fc1e5c55675ebRafael Espindola  // of the offset?
700b156c5d3765637393eda28b04b7fc1e5c55675ebRafael Espindola  if (Reloc.Data.Type == COFF::IMAGE_REL_AMD64_REL32 ||
701b156c5d3765637393eda28b04b7fc1e5c55675ebRafael Espindola      Reloc.Data.Type == COFF::IMAGE_REL_I386_REL32)
70282c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer    FixedValue += 4;
703801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
704801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_section->Relocations.push_back(Reloc);
705b162290e39afd49d4c7d342333b331bc96232720Chris Lattner}
706b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
7078f413fa9c00dbaea000ddfe265ab5edd285ea3abRafael Espindolavoid WinCOFFObjectWriter::WriteObject(MCAssembler &Asm,
708b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                      const MCAsmLayout &Layout) {
709801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Assign symbol and section indexes and offsets.
710a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  Header.NumberOfSections = 0;
711a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
71280646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck  DenseMap<COFFSection *, uint16_t> SectionIndices;
713a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  for (sections::iterator i = Sections.begin(),
714a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer                          e = Sections.end(); i != e; i++) {
71585f2ecc697a8ca6c8cf08093054cbbb9d2060ccfRafael Espindola    if (Layout.getSectionAddressSize((*i)->MCData) > 0) {
71680646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck      size_t Number = ++Header.NumberOfSections;
71780646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck      SectionIndices[*i] = Number;
71880646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck      MakeSectionReal(**i, Number);
719a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    } else {
720a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      (*i)->Number = -1;
721a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    }
722a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  }
723801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
724801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Header.NumberOfSymbols = 0;
725801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
726801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (symbols::iterator i = Symbols.begin(), e = Symbols.end(); i != e; i++) {
727801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    COFFSymbol *coff_symbol = *i;
728801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    MCSymbolData const *SymbolData = coff_symbol->MCData;
729801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
730801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    // Update section number & offset for symbols that have them.
731801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    if ((SymbolData != NULL) && (SymbolData->Fragment != NULL)) {
732a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      assert(coff_symbol->Section != NULL);
733801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
734a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      coff_symbol->Data.SectionNumber = coff_symbol->Section->Number;
735237f8fe5df628065874b8590b364d04dfc2686fdMichael J. Spencer      coff_symbol->Data.Value = Layout.getFragmentOffset(SymbolData->Fragment)
736237f8fe5df628065874b8590b364d04dfc2686fdMichael J. Spencer                              + SymbolData->Offset;
737801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
738801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
739a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    if (coff_symbol->should_keep()) {
740a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      MakeSymbolReal(*coff_symbol, Header.NumberOfSymbols++);
741a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
742a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      // Update auxiliary symbol info.
743a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      coff_symbol->Data.NumberOfAuxSymbols = coff_symbol->Aux.size();
744a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      Header.NumberOfSymbols += coff_symbol->Data.NumberOfAuxSymbols;
745a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    } else
746a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      coff_symbol->Index = -1;
747801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
748801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
749801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Fixup weak external references.
750801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (symbols::iterator i = Symbols.begin(), e = Symbols.end(); i != e; i++) {
751a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    COFFSymbol *coff_symbol = *i;
752a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    if (coff_symbol->Other != NULL) {
753a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      assert(coff_symbol->Index != -1);
754a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      assert(coff_symbol->Aux.size() == 1 &&
755801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer             "Symbol must contain one aux symbol!");
756a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      assert(coff_symbol->Aux[0].AuxType == ATWeakExternal &&
757801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer             "Symbol's aux symbol must be a Weak External!");
758a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      coff_symbol->Aux[0].Aux.WeakExternal.TagIndex = coff_symbol->Other->Index;
759801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
760801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
761801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
76280646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck  // Fixup associative COMDAT sections.
76380646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck  for (sections::iterator i = Sections.begin(),
76480646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck                          e = Sections.end(); i != e; i++) {
76580646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck    if ((*i)->Symbol->Aux[0].Aux.SectionDefinition.Selection !=
76680646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck        COFF::IMAGE_COMDAT_SELECT_ASSOCIATIVE)
76780646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck      continue;
76880646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck
76980646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck    const MCSectionCOFF &MCSec = static_cast<const MCSectionCOFF &>(
77080646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck                                                    (*i)->MCData->getSection());
77180646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck
77280646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck    COFFSection *Assoc = SectionMap.lookup(MCSec.getAssocSection());
77380646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck    if (!Assoc) {
77480646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck      report_fatal_error(Twine("Missing associated COMDAT section ") +
77580646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck                         MCSec.getAssocSection()->getSectionName() +
77680646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck                         " for section " + MCSec.getSectionName());
77780646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck    }
77880646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck
77980646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck    // Skip this section if the associated section is unused.
78080646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck    if (Assoc->Number == -1)
78180646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck      continue;
78280646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck
78380646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck    (*i)->Symbol->Aux[0].Aux.SectionDefinition.Number = SectionIndices[Assoc];
78480646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck  }
78580646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck
78680646283796b20c6a1b7d8eb69ce6f0478d54383Nico Rieck
787801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Assign file offsets to COFF object file structures.
788801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
789801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  unsigned offset = 0;
790801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
791801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  offset += COFF::HeaderSize;
792a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer  offset += COFF::SectionSize * Header.NumberOfSections;
793801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
794801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (MCAssembler::const_iterator i = Asm.begin(),
795801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                   e = Asm.end();
796801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                   i != e; i++) {
7974cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer    COFFSection *Sec = SectionMap[&i->getSection()];
798801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
799a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    if (Sec->Number == -1)
800a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      continue;
801a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
80228ca86aa19fe2a5493573164ef0c2c54542ed9daMichael J. Spencer    Sec->Header.SizeOfRawData = Layout.getSectionAddressSize(i);
803801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
804a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    if (IsPhysicalSection(Sec)) {
805801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      Sec->Header.PointerToRawData = offset;
806801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
807801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      offset += Sec->Header.SizeOfRawData;
808801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
809801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
810801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    if (Sec->Relocations.size() > 0) {
811d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer      bool RelocationsOverflow = Sec->Relocations.size() >= 0xffff;
812d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer
813d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer      if (RelocationsOverflow) {
814d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        // Signal overflow by setting NumberOfSections to max value. Actual
815d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        // size is found in reloc #0. Microsoft tools understand this.
816d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        Sec->Header.NumberOfRelocations = 0xffff;
817d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer      } else {
818d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        Sec->Header.NumberOfRelocations = Sec->Relocations.size();
819d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer      }
820801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      Sec->Header.PointerToRelocations = offset;
821801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
822d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer      if (RelocationsOverflow) {
823d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        // Reloc #0 will contain actual count, so make room for it.
824d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        offset += COFF::RelocationSize;
825d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer      }
826d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer
827801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      offset += COFF::RelocationSize * Sec->Relocations.size();
828801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
829801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      for (relocations::iterator cr = Sec->Relocations.begin(),
830801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                 er = Sec->Relocations.end();
831a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer                                 cr != er; ++cr) {
832a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer        assert((*cr).Symb->Index != -1);
833801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        (*cr).Data.SymbolTableIndex = (*cr).Symb->Index;
834801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      }
835801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
836801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
837a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    assert(Sec->Symbol->Aux.size() == 1
838a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      && "Section's symbol must have one aux!");
839a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    AuxSymbol &Aux = Sec->Symbol->Aux[0];
840801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    assert(Aux.AuxType == ATSectionDefinition &&
841801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer           "Section's symbol's aux symbol must be a Section Definition!");
842801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    Aux.Aux.SectionDefinition.Length = Sec->Header.SizeOfRawData;
843801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    Aux.Aux.SectionDefinition.NumberOfRelocations =
844801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                                Sec->Header.NumberOfRelocations;
845801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    Aux.Aux.SectionDefinition.NumberOfLinenumbers =
846801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                                Sec->Header.NumberOfLineNumbers;
847801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
848801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
849801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Header.PointerToSymbolTable = offset;
850801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
851ab3de49c48bd3282421ce24323fb6b868a3da6ccMichael J. Spencer  Header.TimeDateStamp = sys::TimeValue::now().toEpochTime();
852ab3de49c48bd3282421ce24323fb6b868a3da6ccMichael J. Spencer
853801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Write it all to disk...
854801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteFileHeader(Header);
855801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
856801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  {
857801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    sections::iterator i, ie;
858801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    MCAssembler::const_iterator j, je;
859801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
860801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    for (i = Sections.begin(), ie = Sections.end(); i != ie; i++)
861d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer      if ((*i)->Number != -1) {
862d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        if ((*i)->Relocations.size() >= 0xffff) {
863d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer          (*i)->Header.Characteristics |= COFF::IMAGE_SCN_LNK_NRELOC_OVFL;
864d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        }
865a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer        WriteSectionHeader((*i)->Header);
866d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer      }
867801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
868801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    for (i = Sections.begin(), ie = Sections.end(),
869801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer         j = Asm.begin(), je = Asm.end();
870a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer         (i != ie) && (j != je); ++i, ++j) {
871a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
872a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      if ((*i)->Number == -1)
873a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer        continue;
874a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer
875801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      if ((*i)->Header.PointerToRawData != 0) {
876801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        assert(OS.tell() == (*i)->Header.PointerToRawData &&
877801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer               "Section::PointerToRawData is insane!");
878801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
879f77d5b14af6b92403b93f7ed249f9023e99028ccJim Grosbach        Asm.writeSectionData(j, Layout);
880801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      }
881801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
882801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      if ((*i)->Relocations.size() > 0) {
883801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        assert(OS.tell() == (*i)->Header.PointerToRelocations &&
884801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer               "Section::PointerToRelocations is insane!");
885801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
886d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        if ((*i)->Relocations.size() >= 0xffff) {
887d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer          // In case of overflow, write actual relocation count as first
888d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer          // relocation. Including the synthetic reloc itself (+ 1).
889d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer          COFF::relocation r;
890d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer          r.VirtualAddress = (*i)->Relocations.size() + 1;
891d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer          r.SymbolTableIndex = 0;
892d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer          r.Type = 0;
893d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer          WriteRelocation(r);
894d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer        }
895d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer
896801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        for (relocations::const_iterator k = (*i)->Relocations.begin(),
897801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                               ke = (*i)->Relocations.end();
898801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                               k != ke; k++) {
899801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer          WriteRelocation(k->Data);
900801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        }
901801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      } else
902801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        assert((*i)->Header.PointerToRelocations == 0 &&
903801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer               "Section::PointerToRelocations is insane!");
904801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
905801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
906801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
907801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  assert(OS.tell() == Header.PointerToSymbolTable &&
908801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer         "Header::PointerToSymbolTable is insane!");
909801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
910801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (symbols::iterator i = Symbols.begin(), e = Symbols.end(); i != e; i++)
911a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer    if ((*i)->Index != -1)
912a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer      WriteSymbol(*i);
913801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
914801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  OS.write((char const *)&Strings.Data.front(), Strings.Data.size());
915b162290e39afd49d4c7d342333b331bc96232720Chris Lattner}
916b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
917df09270ae897e7fa64a7c162de163c32ee181a03Rafael EspindolaMCWinCOFFObjectTargetWriter::MCWinCOFFObjectTargetWriter(unsigned Machine_) :
918df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola  Machine(Machine_) {
919df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola}
920df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola
921b162290e39afd49d4c7d342333b331bc96232720Chris Lattner//------------------------------------------------------------------------------
922b162290e39afd49d4c7d342333b331bc96232720Chris Lattner// WinCOFFObjectWriter factory function
923b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
924b162290e39afd49d4c7d342333b331bc96232720Chris Lattnernamespace llvm {
925df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola  MCObjectWriter *createWinCOFFObjectWriter(MCWinCOFFObjectTargetWriter *MOTW,
926df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola                                            raw_ostream &OS) {
927df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola    return new WinCOFFObjectWriter(MOTW, OS);
928b162290e39afd49d4c7d342333b331bc96232720Chris Lattner  }
929933304ef0c3ec18c23d0b385c2117a6eae790430Michael J. Spencer}
930