WinCOFFObjectWriter.cpp revision 808ecfce6a312625bee5c0f4f9831a0d0ed01b16
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
16b162290e39afd49d4c7d342333b331bc96232720Chris Lattner#include "llvm/MC/MCObjectWriter.h"
17801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/MC/MCSection.h"
18801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/MC/MCContext.h"
19801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/MC/MCSymbol.h"
20801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/MC/MCExpr.h"
21b162290e39afd49d4c7d342333b331bc96232720Chris Lattner#include "llvm/MC/MCValue.h"
22b162290e39afd49d4c7d342333b331bc96232720Chris Lattner#include "llvm/MC/MCAssembler.h"
23b162290e39afd49d4c7d342333b331bc96232720Chris Lattner#include "llvm/MC/MCAsmLayout.h"
24801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/MC/MCSectionCOFF.h"
25801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
26801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/ADT/DenseMap.h"
27801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/ADT/StringMap.h"
28801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/ADT/StringRef.h"
29801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
30801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/Support/COFF.h"
31801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/Support/Debug.h"
32801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/Support/ErrorHandling.h"
33801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
34801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include <cstdio>
35801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
36b162290e39afd49d4c7d342333b331bc96232720Chris Lattnerusing namespace llvm;
37b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
38b162290e39afd49d4c7d342333b331bc96232720Chris Lattnernamespace {
39801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencertypedef llvm::SmallString<COFF::NameSize> name;
40801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
41801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerenum AuxiliaryType {
42801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  ATFunctionDefinition,
43801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  ATbfAndefSymbol,
44801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  ATWeakExternal,
45801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  ATFile,
46801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  ATSectionDefinition
47801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer};
48801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
49801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerstruct AuxSymbol {
50801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  AuxiliaryType   AuxType;
51801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFF::Auxiliary Aux;
52801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer};
53801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
54801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerclass COFFSymbol {
55801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerpublic:
56801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFF::symbol Data;
57801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
58801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  typedef llvm::SmallVector<AuxSymbol, 1> AuxiliarySymbols;
59801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
60801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  name             Name;
61801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  size_t           Index;
62801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  AuxiliarySymbols Aux;
63801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSymbol      *Other;
64801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
65801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  MCSymbolData const *MCData;
66801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
67801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSymbol(llvm::StringRef name, size_t index);
68801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  size_t size() const;
69801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void set_name_offset(uint32_t Offset);
70801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer};
71801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
72801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// This class contains staging data for a COFF relocation entry.
73801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerstruct COFFRelocation {
74801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFF::relocation Data;
75801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSymbol          *Symb;
76801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
77801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFRelocation() : Symb(NULL) {}
78801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  static size_t size() { return COFF::RelocationSize; }
79801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer};
80801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
81801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencertypedef std::vector<COFFRelocation> relocations;
82801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
83801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerclass COFFSection {
84801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerpublic:
85801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFF::section Header;
86801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
87801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  std::string          Name;
88801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  size_t               Number;
89801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  MCSectionData const *MCData;
90801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSymbol              *Symb;
91801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  relocations          Relocations;
92801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
93801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSection(llvm::StringRef name, size_t Index);
94801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  static size_t size();
95801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer};
96801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
97801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// This class holds the COFF string table.
98801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerclass StringTable {
99801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  typedef llvm::StringMap<size_t> map;
100801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  map Map;
101801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
102801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void update_length();
103801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerpublic:
104801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  std::vector<char> Data;
105801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
106801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  StringTable();
107801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  size_t size() const;
108801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  size_t insert(llvm::StringRef String);
109801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer};
110801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
111801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerclass WinCOFFObjectWriter : public MCObjectWriter {
112801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerpublic:
113801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
114801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  typedef std::vector<COFFSymbol*>  symbols;
115801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  typedef std::vector<COFFSection*> sections;
116b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
117801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  typedef StringMap<COFFSymbol *>  name_symbol_map;
118801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  typedef StringMap<COFFSection *> name_section_map;
119b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
120801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  typedef DenseMap<MCSymbolData const *, COFFSymbol *>   symbol_map;
121801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  typedef DenseMap<MCSectionData const *, COFFSection *> section_map;
122b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
123801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Root level file contents.
124801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFF::header Header;
125801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  sections     Sections;
126801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  symbols      Symbols;
127801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  StringTable  Strings;
128b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
129801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Maps used during object file creation.
130801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  section_map SectionMap;
131801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  symbol_map  SymbolMap;
132801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
133801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WinCOFFObjectWriter(raw_ostream &OS);
134808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer  ~WinCOFFObjectWriter();
135801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
136801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSymbol *createSymbol(llvm::StringRef Name);
137801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSection *createSection(llvm::StringRef Name);
138801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
139801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void InitCOFFEntity(COFFSymbol &Symbol);
140801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void InitCOFFEntity(COFFSection &Section);
141801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
142801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  template <typename object_t, typename list_t>
143801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  object_t *createCOFFEntity(llvm::StringRef Name, list_t &List);
144801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
145801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void DefineSection(MCSectionData const &SectionData);
146801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void DefineSymbol(MCSymbolData const &SymbolData, MCAssembler &Assembler);
147801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
148801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  bool ExportSection(COFFSection *S);
149801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  bool ExportSymbol(MCSymbolData const &SymbolData, MCAssembler &Asm);
150801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
151801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Entity writing methods.
152801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
153801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void WriteFileHeader(const COFF::header &Header);
154801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void WriteSymbol(const COFFSymbol *S);
155801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void WriteAuxiliarySymbols(const COFFSymbol::AuxiliarySymbols &S);
156801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void WriteSectionHeader(const COFF::section &S);
157801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void WriteRelocation(const COFF::relocation &R);
158801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
159801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // MCObjectWriter interface implementation.
160801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
161801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void ExecutePostLayoutBinding(MCAssembler &Asm);
162801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
163801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void RecordRelocation(const MCAssembler &Asm,
164801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                        const MCAsmLayout &Layout,
165801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                        const MCFragment *Fragment,
166801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                        const MCFixup &Fixup,
167801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                        MCValue Target,
168801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                        uint64_t &FixedValue);
169801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
170801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  void WriteObject(const MCAssembler &Asm, const MCAsmLayout &Layout);
171801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer};
172801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
173801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
174801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerstatic inline void write_uint32_le(void *Data, uint32_t const &Value) {
175801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  uint8_t *Ptr = reinterpret_cast<uint8_t *>(Data);
176801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[0] = (Value & 0x000000FF) >>  0;
177801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[1] = (Value & 0x0000FF00) >>  8;
178801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[2] = (Value & 0x00FF0000) >> 16;
179801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[3] = (Value & 0xFF000000) >> 24;
180801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
181801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
182801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerstatic inline void write_uint16_le(void *Data, uint16_t const &Value) {
183801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  uint8_t *Ptr = reinterpret_cast<uint8_t *>(Data);
184801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[0] = (Value & 0x00FF) >> 0;
185801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[1] = (Value & 0xFF00) >> 8;
186801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
187801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
188801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerstatic inline void write_uint8_le(void *Data, uint8_t const &Value) {
189801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  uint8_t *Ptr = reinterpret_cast<uint8_t *>(Data);
190801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Ptr[0] = (Value & 0xFF) >> 0;
191801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
192801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
193801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------
194801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// Symbol class implementation
195801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
196801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. SpencerCOFFSymbol::COFFSymbol(llvm::StringRef name, size_t index)
197801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      : Name(name.begin(), name.end()), Index(-1)
198801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      , Other(NULL), MCData(NULL) {
199801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  memset(&Data, 0, sizeof(Data));
200801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
201b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
202801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencersize_t COFFSymbol::size() const {
203801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return COFF::SymbolSize + (Data.NumberOfAuxSymbols * COFF::SymbolSize);
204b162290e39afd49d4c7d342333b331bc96232720Chris Lattner}
205b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
206801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// In the case that the name does not fit within 8 bytes, the offset
207801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// into the string table is stored in the last 4 bytes instead, leaving
208801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// the first 4 bytes as 0.
209801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid COFFSymbol::set_name_offset(uint32_t Offset) {
210801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  write_uint32_le(Data.Name + 0, 0);
211801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  write_uint32_le(Data.Name + 4, Offset);
212801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
213801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
214801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------
215801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// Section class implementation
216801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
217801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. SpencerCOFFSection::COFFSection(llvm::StringRef name, size_t Index)
218801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer       : Name(name), Number(Index + 1)
219801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer       , MCData(NULL), Symb(NULL) {
220801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  memset(&Header, 0, sizeof(Header));
221801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
222801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
223801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencersize_t COFFSection::size() {
224801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return COFF::SectionSize;
225801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
226801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
227801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------
228801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// StringTable class implementation
229801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
230801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// Write the length of the string table into Data.
231801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// The length of the string table includes uint32 length header.
232801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid StringTable::update_length() {
233801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  write_uint32_le(&Data.front(), Data.size());
234801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
235801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
236801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. SpencerStringTable::StringTable() {
237801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // The string table data begins with the length of the entire string table
238801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // including the length header. Allocate space for this header.
239801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Data.resize(4);
240801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
241801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
242801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencersize_t StringTable::size() const {
243801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return Data.size();
244801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
245801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
246801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// Add String to the table iff it is not already there.
247801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// @returns the index into the string table where the string is now located.
248801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencersize_t StringTable::insert(llvm::StringRef String) {
249801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  map::iterator i = Map.find(String);
250801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
251801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  if (i != Map.end())
252801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    return i->second;
253801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
254801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  size_t Offset = Data.size();
255801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
256801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Insert string data into string table.
257801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Data.insert(Data.end(), String.begin(), String.end());
258801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Data.push_back('\0');
259801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
260801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Put a reference to it in the map.
261801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Map[String] = Offset;
262801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
263801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Update the internal length field.
264801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  update_length();
265801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
266801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return Offset;
267801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
268801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
269801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------
270801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// WinCOFFObjectWriter class implementation
271801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
272b162290e39afd49d4c7d342333b331bc96232720Chris LattnerWinCOFFObjectWriter::WinCOFFObjectWriter(raw_ostream &OS)
273b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                : MCObjectWriter(OS, true) {
274801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  memset(&Header, 0, sizeof(Header));
275801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // TODO: Move magic constant out to COFF.h
276801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Header.Machine = 0x14C; // x86
277801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
278801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
279808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin KramerWinCOFFObjectWriter::~WinCOFFObjectWriter() {
280808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer  for (symbols::iterator I = Symbols.begin(), E = Symbols.end(); I != E; ++I)
281808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer    delete *I;
282808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer  for (sections::iterator I = Sections.begin(), E = Sections.end(); I != E; ++I)
283808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer    delete *I;
284808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer}
285808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer
286801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. SpencerCOFFSymbol *WinCOFFObjectWriter::createSymbol(llvm::StringRef Name) {
287801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return createCOFFEntity<COFFSymbol>(Name, Symbols);
288801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
289801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
290801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. SpencerCOFFSection *WinCOFFObjectWriter::createSection(llvm::StringRef Name) {
291801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return createCOFFEntity<COFFSection>(Name, Sections);
292801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
293801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
294801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// This function initializes a symbol by entering its name into the string
295801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// table if it is too long to fit in the symbol table header.
296801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::InitCOFFEntity(COFFSymbol &S) {
297801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  if (S.Name.size() > COFF::NameSize) {
298801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    size_t StringTableEntry = Strings.insert(S.Name.c_str());
299801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
300801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    S.set_name_offset(StringTableEntry);
301801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  } else
302801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    memcpy(S.Data.Name, S.Name.c_str(), S.Name.size());
303801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
304801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
305801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// This function initializes a section by entering its name into the string
306801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// table if it is too long to fit in the section table header.
307801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::InitCOFFEntity(COFFSection &S) {
308801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  if (S.Name.size() > COFF::NameSize) {
309801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    size_t StringTableEntry = Strings.insert(S.Name.c_str());
310801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
311801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    // FIXME: Why is this number 999999? This number is never mentioned in the
312801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    // spec. I'm assuming this is due to the printed value needing to fit into
313801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    // the S.Header.Name field. In which case why not 9999999 (7 9's instead of
314801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    // 6)? The spec does not state if this entry should be null terminated in
315801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    // this case, and thus this seems to be the best way to do it. I think I
316801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    // just solved my own FIXME...
317801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    if (StringTableEntry > 999999)
318801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      report_fatal_error("COFF string table is greater than 999999 bytes.");
319801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
320fd2878c8d0e876e849f795bb3f5d5e2f82aa59bdDouglas Gregor    sprintf(S.Header.Name, "/%d", (unsigned)StringTableEntry);
321801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  } else
322801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    memcpy(S.Header.Name, S.Name.c_str(), S.Name.size());
323801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
324801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
325801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// A template used to lookup or create a symbol/section, and initialize it if
326801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// needed.
327801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencertemplate <typename object_t, typename list_t>
328801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerobject_t *WinCOFFObjectWriter::createCOFFEntity(llvm::StringRef Name,
329801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                                list_t &List) {
330801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  object_t *Object = new object_t(Name, List.size());
331801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
332801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  InitCOFFEntity(*Object);
333801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
334801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  List.push_back(Object);
335801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
336801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return Object;
337801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
338801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
339801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// This function takes a section data object from the assembler
340801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// and creates the associated COFF section staging object.
341801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::DefineSection(MCSectionData const &SectionData) {
342801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // FIXME: Not sure how to verify this (at least in a debug build).
343801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  MCSectionCOFF const &Sec =
344801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    static_cast<MCSectionCOFF const &>(SectionData.getSection());
345801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
346801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSection *coff_section = createSection(Sec.getSectionName());
347801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSymbol  *coff_symbol = createSymbol(Sec.getSectionName());
348801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
349801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_section->Symb = coff_symbol;
350801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->Data.StorageClass = COFF::IMAGE_SYM_CLASS_STATIC;
351801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->Data.SectionNumber = coff_section->Number;
352801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
353801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // In this case the auxiliary symbol is a Section Definition.
354801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->Aux.resize(1);
355801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  memset(&coff_symbol->Aux[0], 0, sizeof(coff_symbol->Aux[0]));
356801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->Aux[0].AuxType = ATSectionDefinition;
357801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->Aux[0].Aux.SectionDefinition.Number = coff_section->Number;
358801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->Aux[0].Aux.SectionDefinition.Selection = Sec.getSelection();
359801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
360801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_section->Header.Characteristics = Sec.getCharacteristics();
361801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
362801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  uint32_t &Characteristics = coff_section->Header.Characteristics;
363801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  switch (SectionData.getAlignment()) {
364801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 1:    Characteristics |= COFF::IMAGE_SCN_ALIGN_1BYTES;    break;
365801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 2:    Characteristics |= COFF::IMAGE_SCN_ALIGN_2BYTES;    break;
366801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 4:    Characteristics |= COFF::IMAGE_SCN_ALIGN_4BYTES;    break;
367801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 8:    Characteristics |= COFF::IMAGE_SCN_ALIGN_8BYTES;    break;
368801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 16:   Characteristics |= COFF::IMAGE_SCN_ALIGN_16BYTES;   break;
369801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 32:   Characteristics |= COFF::IMAGE_SCN_ALIGN_32BYTES;   break;
370801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 64:   Characteristics |= COFF::IMAGE_SCN_ALIGN_64BYTES;   break;
371801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 128:  Characteristics |= COFF::IMAGE_SCN_ALIGN_128BYTES;  break;
372801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 256:  Characteristics |= COFF::IMAGE_SCN_ALIGN_256BYTES;  break;
373801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 512:  Characteristics |= COFF::IMAGE_SCN_ALIGN_512BYTES;  break;
374801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 1024: Characteristics |= COFF::IMAGE_SCN_ALIGN_1024BYTES; break;
375801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 2048: Characteristics |= COFF::IMAGE_SCN_ALIGN_2048BYTES; break;
376801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 4096: Characteristics |= COFF::IMAGE_SCN_ALIGN_4096BYTES; break;
377801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case 8192: Characteristics |= COFF::IMAGE_SCN_ALIGN_8192BYTES; break;
378801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  default:
379801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    llvm_unreachable("unsupported section alignment");
380801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
381801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
382801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Bind internal COFF section to MC section.
383801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_section->MCData = &SectionData;
384801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  SectionMap[&SectionData] = coff_section;
385801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
386801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
387801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// This function takes a section data object from the assembler
388801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// and creates the associated COFF symbol staging object.
389801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::DefineSymbol(MCSymbolData const &SymbolData,
390801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                        MCAssembler &Assembler) {
391801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSymbol *coff_symbol = createSymbol(SymbolData.getSymbol().getName());
392801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
393801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->Data.Type         = (SymbolData.getFlags() & 0x0000FFFF) >>  0;
394801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->Data.StorageClass = (SymbolData.getFlags() & 0x00FF0000) >> 16;
395801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
396801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // If no storage class was specified in the streamer, define it here.
397801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  if (coff_symbol->Data.StorageClass == 0) {
398801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    bool external = SymbolData.isExternal() || (SymbolData.Fragment == NULL);
399801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
400801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Data.StorageClass =
401801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      external ? COFF::IMAGE_SYM_CLASS_EXTERNAL : COFF::IMAGE_SYM_CLASS_STATIC;
402801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
403801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
404801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  if (SymbolData.getFlags() & COFF::SF_WeakReference) {
405801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Data.StorageClass = COFF::IMAGE_SYM_CLASS_WEAK_EXTERNAL;
406801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
407801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    const MCExpr *Value = SymbolData.getSymbol().getVariableValue();
408801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
409801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    // FIXME: This assert message isn't very good.
410801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    assert(Value->getKind() == MCExpr::SymbolRef &&
411801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer           "Value must be a SymbolRef!");
412801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
413801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    const MCSymbolRefExpr *SymbolRef =
414801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      static_cast<const MCSymbolRefExpr *>(Value);
415801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
416801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    const MCSymbolData &OtherSymbolData =
417801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      Assembler.getSymbolData(SymbolRef->getSymbol());
418801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
419801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    // FIXME: This assert message isn't very good.
420801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    assert(SymbolMap.find(&OtherSymbolData) != SymbolMap.end() &&
421801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer           "OtherSymbolData must be in the symbol map!");
422801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
423801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Other = SymbolMap[&OtherSymbolData];
424801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
425801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    // Setup the Weak External auxiliary symbol.
426801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Aux.resize(1);
427801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    memset(&coff_symbol->Aux[0], 0, sizeof(coff_symbol->Aux[0]));
428801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Aux[0].AuxType = ATWeakExternal;
429801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Aux[0].Aux.WeakExternal.TagIndex = 0;
430801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Aux[0].Aux.WeakExternal.Characteristics =
431801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                        COFF::IMAGE_WEAK_EXTERN_SEARCH_LIBRARY;
432801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
433801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
434801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Bind internal COFF symbol to MC symbol.
435801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_symbol->MCData = &SymbolData;
436801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  SymbolMap[&SymbolData] = coff_symbol;
437801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
438801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
439801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerbool WinCOFFObjectWriter::ExportSection(COFFSection *S) {
440801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return (S->Header.Characteristics
441801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer         & COFF::IMAGE_SCN_CNT_UNINITIALIZED_DATA) == 0;
442801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
443801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
444801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerbool WinCOFFObjectWriter::ExportSymbol(MCSymbolData const &SymbolData,
445801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                       MCAssembler &Asm) {
446801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // This doesn't seem to be right. Strings referred to from the .data section
447801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // need symbols so they can be linked to code in the .text section right?
448801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
449801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // return Asm.isSymbolLinkerVisible (&SymbolData);
450801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
451801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // For now, all symbols are exported, the linker will sort it out for us.
452801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  return true;
453801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
454801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
455801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------
456801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// entity writing methods
457801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
458801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::WriteFileHeader(const COFF::header &Header) {
459801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(Header.Machine);
460801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(Header.NumberOfSections);
461801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(Header.TimeDateStamp);
462801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(Header.PointerToSymbolTable);
463801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(Header.NumberOfSymbols);
464801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(Header.SizeOfOptionalHeader);
465801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(Header.Characteristics);
466801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
467801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
468801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::WriteSymbol(const COFFSymbol *S) {
469801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteBytes(StringRef(S->Data.Name, COFF::NameSize));
470801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S->Data.Value);
471801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(S->Data.SectionNumber);
472801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(S->Data.Type);
473801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Write8(S->Data.StorageClass);
474801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Write8(S->Data.NumberOfAuxSymbols);
475801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteAuxiliarySymbols(S->Aux);
476801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
477801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
478801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::WriteAuxiliarySymbols(
479801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                        const COFFSymbol::AuxiliarySymbols &S) {
480801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for(COFFSymbol::AuxiliarySymbols::const_iterator i = S.begin(), e = S.end();
481801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      i != e; ++i) {
482801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    switch(i->AuxType) {
483801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    case ATFunctionDefinition:
484801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.FunctionDefinition.TagIndex);
485801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.FunctionDefinition.TotalSize);
486801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.FunctionDefinition.PointerToLinenumber);
487801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.FunctionDefinition.PointerToNextFunction);
488801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.FunctionDefinition.unused));
489801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      break;
490801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    case ATbfAndefSymbol:
491801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.bfAndefSymbol.unused1));
492801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE16(i->Aux.bfAndefSymbol.Linenumber);
493801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.bfAndefSymbol.unused2));
494801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.bfAndefSymbol.PointerToNextFunction);
495801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.bfAndefSymbol.unused3));
496801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      break;
497801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    case ATWeakExternal:
498801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.WeakExternal.TagIndex);
499801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.WeakExternal.Characteristics);
500801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.WeakExternal.unused));
501801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      break;
502801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    case ATFile:
503801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteBytes(StringRef(reinterpret_cast<const char *>(i->Aux.File.FileName),
504801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                 sizeof(i->Aux.File.FileName)));
505801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      break;
506801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    case ATSectionDefinition:
507801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.SectionDefinition.Length);
508801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE16(i->Aux.SectionDefinition.NumberOfRelocations);
509801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE16(i->Aux.SectionDefinition.NumberOfLinenumbers);
510801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE32(i->Aux.SectionDefinition.CheckSum);
511801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteLE16(i->Aux.SectionDefinition.Number);
512801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      Write8(i->Aux.SectionDefinition.Selection);
513801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteZeros(sizeof(i->Aux.SectionDefinition.unused));
514801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      break;
515801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
516801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
517801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
518801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
519801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::WriteSectionHeader(const COFF::section &S) {
520801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteBytes(StringRef(S.Name, COFF::NameSize));
521801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
522801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.VirtualSize);
523801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.VirtualAddress);
524801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.SizeOfRawData);
525801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.PointerToRawData);
526801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.PointerToRelocations);
527801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.PointerToLineNumbers);
528801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(S.NumberOfRelocations);
529801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(S.NumberOfLineNumbers);
530801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(S.Characteristics);
531801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}
532801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
533801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::WriteRelocation(const COFF::relocation &R) {
534801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(R.VirtualAddress);
535801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE32(R.SymbolTableIndex);
536801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteLE16(R.Type);
537b162290e39afd49d4c7d342333b331bc96232720Chris Lattner}
538b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
539b162290e39afd49d4c7d342333b331bc96232720Chris Lattner////////////////////////////////////////////////////////////////////////////////
540b162290e39afd49d4c7d342333b331bc96232720Chris Lattner// MCObjectWriter interface implementations
541b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
542b162290e39afd49d4c7d342333b331bc96232720Chris Lattnervoid WinCOFFObjectWriter::ExecutePostLayoutBinding(MCAssembler &Asm) {
543801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // "Define" each section & symbol. This creates section & symbol
544801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // entries in the staging area and gives them their final indexes.
545801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
546801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (MCAssembler::const_iterator i = Asm.begin(), e = Asm.end(); i != e; i++)
547801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    DefineSection(*i);
548801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
549801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (MCAssembler::const_symbol_iterator i = Asm.symbol_begin(),
550801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                          e = Asm.symbol_end(); i != e; i++) {
551801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    if (ExportSymbol(*i, Asm))
552801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      DefineSymbol(*i, Asm);
553801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
554b162290e39afd49d4c7d342333b331bc96232720Chris Lattner}
555b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
556b162290e39afd49d4c7d342333b331bc96232720Chris Lattnervoid WinCOFFObjectWriter::RecordRelocation(const MCAssembler &Asm,
557b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                           const MCAsmLayout &Layout,
558b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                           const MCFragment *Fragment,
559b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                           const MCFixup &Fixup,
560b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                           MCValue Target,
561b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                           uint64_t &FixedValue) {
562801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  assert(Target.getSymA() != NULL && "Relocation must reference a symbol!");
563801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  assert(Target.getSymB() == NULL &&
564801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer         "Relocation must reference only one symbol!");
565801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
566801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  MCSectionData const *SectionData = Fragment->getParent();
567801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  MCSymbolData const *SymbolData =
568801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                              &Asm.getSymbolData(Target.getSymA()->getSymbol());
569801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
570801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  assert(SectionMap.find(SectionData) != SectionMap.end() &&
571801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer         "Section must already have been defined in ExecutePostLayoutBinding!");
572801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  assert(SymbolMap.find(SymbolData) != SymbolMap.end() &&
573801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer         "Symbol must already have been defined in ExecutePostLayoutBinding!");
574801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
575801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSection *coff_section = SectionMap[SectionData];
576801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFSymbol *coff_symbol = SymbolMap[SymbolData];
577801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
578801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  FixedValue = Target.getConstant();
579801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
580801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  COFFRelocation Reloc;
581801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
582801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Reloc.Data.VirtualAddress = Layout.getFragmentOffset(Fragment);
583801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Reloc.Symb = coff_symbol;
584801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
585801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Reloc.Data.VirtualAddress += Fixup.getOffset();
586801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
587801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  switch (Fixup.getKind()) {
588801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case FirstTargetFixupKind: // reloc_pcrel_4byte
589801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    Reloc.Data.Type = COFF::IMAGE_REL_I386_REL32;
590801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    FixedValue += 4;
591801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    break;
592801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  case FK_Data_4:
593801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    Reloc.Data.Type = COFF::IMAGE_REL_I386_DIR32;
594801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    break;
595801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  default:
596801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    llvm_unreachable("unsupported relocation type");
597801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
598801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
599801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  coff_section->Relocations.push_back(Reloc);
600b162290e39afd49d4c7d342333b331bc96232720Chris Lattner}
601b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
602b162290e39afd49d4c7d342333b331bc96232720Chris Lattnervoid WinCOFFObjectWriter::WriteObject(const MCAssembler &Asm,
603b162290e39afd49d4c7d342333b331bc96232720Chris Lattner                                      const MCAsmLayout &Layout) {
604801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Assign symbol and section indexes and offsets.
605801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
606801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Header.NumberOfSymbols = 0;
607801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
608801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (symbols::iterator i = Symbols.begin(), e = Symbols.end(); i != e; i++) {
609801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    COFFSymbol *coff_symbol = *i;
610801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    MCSymbolData const *SymbolData = coff_symbol->MCData;
611801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
612801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Index = Header.NumberOfSymbols++;
613801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
614801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    // Update section number & offset for symbols that have them.
615801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    if ((SymbolData != NULL) && (SymbolData->Fragment != NULL)) {
616801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      COFFSection *coff_section = SectionMap[SymbolData->Fragment->getParent()];
617801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
618801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      coff_symbol->Data.SectionNumber = coff_section->Number;
619801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      coff_symbol->Data.Value = Layout.getFragmentOffset(SymbolData->Fragment);
620801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
621801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
622801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    // Update auxiliary symbol info.
623801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    coff_symbol->Data.NumberOfAuxSymbols = coff_symbol->Aux.size();
624801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    Header.NumberOfSymbols += coff_symbol->Data.NumberOfAuxSymbols;
625801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
626801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
627801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Fixup weak external references.
628801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (symbols::iterator i = Symbols.begin(), e = Symbols.end(); i != e; i++) {
629801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    COFFSymbol *symb = *i;
630801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
631801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    if (symb->Other != NULL) {
632801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      assert(symb->Aux.size() == 1 &&
633801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer             "Symbol must contain one aux symbol!");
634801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      assert(symb->Aux[0].AuxType == ATWeakExternal &&
635801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer             "Symbol's aux symbol must be a Weak External!");
636801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      symb->Aux[0].Aux.WeakExternal.TagIndex = symb->Other->Index;
637801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
638801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
639801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
640801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Assign file offsets to COFF object file structures.
641801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
642801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  unsigned offset = 0;
643801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
644801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  offset += COFF::HeaderSize;
645801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  offset += COFF::SectionSize * Asm.size();
646801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
647801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Header.NumberOfSections = Sections.size();
648801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
649801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (MCAssembler::const_iterator i = Asm.begin(),
650801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                   e = Asm.end();
651801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                   i != e; i++) {
652801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    COFFSection *Sec = SectionMap[i];
653801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
654801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    Sec->Header.SizeOfRawData = Layout.getSectionFileSize(i);
655801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
656801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    if (ExportSection(Sec)) {
657801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      Sec->Header.PointerToRawData = offset;
658801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
659801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      offset += Sec->Header.SizeOfRawData;
660801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
661801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
662801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    if (Sec->Relocations.size() > 0) {
663801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      Sec->Header.NumberOfRelocations = Sec->Relocations.size();
664801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      Sec->Header.PointerToRelocations = offset;
665801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
666801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      offset += COFF::RelocationSize * Sec->Relocations.size();
667801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
668801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      for (relocations::iterator cr = Sec->Relocations.begin(),
669801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                 er = Sec->Relocations.end();
670801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                 cr != er; cr++) {
671801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        (*cr).Data.SymbolTableIndex = (*cr).Symb->Index;
672801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      }
673801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
674801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
675801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    assert(Sec->Symb->Aux.size() == 1 && "Section's symbol must have one aux!");
676801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    AuxSymbol &Aux = Sec->Symb->Aux[0];
677801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    assert(Aux.AuxType == ATSectionDefinition &&
678801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer           "Section's symbol's aux symbol must be a Section Definition!");
679801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    Aux.Aux.SectionDefinition.Length = Sec->Header.SizeOfRawData;
680801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    Aux.Aux.SectionDefinition.NumberOfRelocations =
681801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                                Sec->Header.NumberOfRelocations;
682801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    Aux.Aux.SectionDefinition.NumberOfLinenumbers =
683801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                                Sec->Header.NumberOfLineNumbers;
684801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
685801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
686801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  Header.PointerToSymbolTable = offset;
687801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
688801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  // Write it all to disk...
689801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  WriteFileHeader(Header);
690801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
691801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  {
692801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    sections::iterator i, ie;
693801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    MCAssembler::const_iterator j, je;
694801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
695801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    for (i = Sections.begin(), ie = Sections.end(); i != ie; i++)
696801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      WriteSectionHeader((*i)->Header);
697801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
698801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    for (i = Sections.begin(), ie = Sections.end(),
699801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer         j = Asm.begin(), je = Asm.end();
700801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer         (i != ie) && (j != je); i++, j++) {
701801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      if ((*i)->Header.PointerToRawData != 0) {
702801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        assert(OS.tell() == (*i)->Header.PointerToRawData &&
703801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer               "Section::PointerToRawData is insane!");
704801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
705801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        Asm.WriteSectionData(j, Layout, this);
706801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      }
707801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
708801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      if ((*i)->Relocations.size() > 0) {
709801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        assert(OS.tell() == (*i)->Header.PointerToRelocations &&
710801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer               "Section::PointerToRelocations is insane!");
711801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
712801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        for (relocations::const_iterator k = (*i)->Relocations.begin(),
713801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                               ke = (*i)->Relocations.end();
714801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer                                               k != ke; k++) {
715801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer          WriteRelocation(k->Data);
716801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        }
717801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer      } else
718801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer        assert((*i)->Header.PointerToRelocations == 0 &&
719801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer               "Section::PointerToRelocations is insane!");
720801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    }
721801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  }
722801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
723801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  assert(OS.tell() == Header.PointerToSymbolTable &&
724801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer         "Header::PointerToSymbolTable is insane!");
725801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
726801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  for (symbols::iterator i = Symbols.begin(), e = Symbols.end(); i != e; i++)
727801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer    WriteSymbol(*i);
728801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer
729801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer  OS.write((char const *)&Strings.Data.front(), Strings.Data.size());
730b162290e39afd49d4c7d342333b331bc96232720Chris Lattner}
731b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
732b162290e39afd49d4c7d342333b331bc96232720Chris Lattner//------------------------------------------------------------------------------
733b162290e39afd49d4c7d342333b331bc96232720Chris Lattner// WinCOFFObjectWriter factory function
734b162290e39afd49d4c7d342333b331bc96232720Chris Lattner
735b162290e39afd49d4c7d342333b331bc96232720Chris Lattnernamespace llvm {
736b162290e39afd49d4c7d342333b331bc96232720Chris Lattner  MCObjectWriter *createWinCOFFObjectWriter(raw_ostream &OS) {
737b162290e39afd49d4c7d342333b331bc96232720Chris Lattner    return new WinCOFFObjectWriter(OS);
738b162290e39afd49d4c7d342333b331bc96232720Chris Lattner  }
739933304ef0c3ec18c23d0b385c2117a6eae790430Michael J. Spencer}
740