WinCOFFObjectWriter.cpp revision d03a29b69d7285ed2ca523d70a45174e33716727
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" 25df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola#include "llvm/MC/MCWinCOFFObjectWriter.h" 26801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 27801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/ADT/DenseMap.h" 28df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola#include "llvm/ADT/OwningPtr.h" 29801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/ADT/StringMap.h" 30801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/ADT/StringRef.h" 31801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 32801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/Support/COFF.h" 33801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/Support/Debug.h" 34801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include "llvm/Support/ErrorHandling.h" 35801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 361f6efa3996dd1929fbc129203ce5009b620e6969Michael J. Spencer#include "llvm/Support/TimeValue.h" 37ab3de49c48bd3282421ce24323fb6b868a3da6ccMichael J. Spencer 38801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer#include <cstdio> 39801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 40b162290e39afd49d4c7d342333b331bc96232720Chris Lattnerusing namespace llvm; 41b162290e39afd49d4c7d342333b331bc96232720Chris Lattner 42b162290e39afd49d4c7d342333b331bc96232720Chris Lattnernamespace { 43801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencertypedef llvm::SmallString<COFF::NameSize> name; 44801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 45801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerenum AuxiliaryType { 46801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer ATFunctionDefinition, 47801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer ATbfAndefSymbol, 48801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer ATWeakExternal, 49801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer ATFile, 50801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer ATSectionDefinition 51801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}; 52801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 53801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerstruct AuxSymbol { 54801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer AuxiliaryType AuxType; 55801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer COFF::Auxiliary Aux; 56801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}; 57801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 58a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencerclass COFFSymbol; 59a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencerclass COFFSection; 60a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 61801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerclass COFFSymbol { 62801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerpublic: 63801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer COFF::symbol Data; 64801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 65801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer typedef llvm::SmallVector<AuxSymbol, 1> AuxiliarySymbols; 66801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 67801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer name Name; 689cf23a9ab466a900bc0f937bc930d398d6097766Michael J. Spencer int Index; 69801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer AuxiliarySymbols Aux; 70801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer COFFSymbol *Other; 71a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer COFFSection *Section; 72a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer int Relocations; 73801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 74801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer MCSymbolData const *MCData; 75801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 76a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer COFFSymbol(llvm::StringRef name); 77801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer size_t size() const; 78801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer void set_name_offset(uint32_t Offset); 79a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 80a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer bool should_keep() const; 81801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}; 82801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 83801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// This class contains staging data for a COFF relocation entry. 84801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerstruct COFFRelocation { 85801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer COFF::relocation Data; 86801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer COFFSymbol *Symb; 87801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 88801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer COFFRelocation() : Symb(NULL) {} 89801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer static size_t size() { return COFF::RelocationSize; } 90801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}; 91801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 92801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencertypedef std::vector<COFFRelocation> relocations; 93801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 94801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerclass COFFSection { 95801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerpublic: 96801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer COFF::section Header; 97801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 98801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer std::string Name; 999cf23a9ab466a900bc0f937bc930d398d6097766Michael J. Spencer int Number; 100801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer MCSectionData const *MCData; 101a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer COFFSymbol *Symbol; 102801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer relocations Relocations; 103801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 104a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer COFFSection(llvm::StringRef name); 105801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer static size_t size(); 106801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}; 107801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 108801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// This class holds the COFF string table. 109801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerclass StringTable { 110801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer typedef llvm::StringMap<size_t> map; 111801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer map Map; 112801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 113801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer void update_length(); 114801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerpublic: 115801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer std::vector<char> Data; 116801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 117801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer StringTable(); 118801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer size_t size() const; 119801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer size_t insert(llvm::StringRef String); 120801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}; 121801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 122801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerclass WinCOFFObjectWriter : public MCObjectWriter { 123801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerpublic: 124801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 125801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer typedef std::vector<COFFSymbol*> symbols; 126801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer typedef std::vector<COFFSection*> sections; 127b162290e39afd49d4c7d342333b331bc96232720Chris Lattner 1284cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer typedef DenseMap<MCSymbol const *, COFFSymbol *> symbol_map; 1294cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer typedef DenseMap<MCSection const *, COFFSection *> section_map; 130b162290e39afd49d4c7d342333b331bc96232720Chris Lattner 131df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola llvm::OwningPtr<MCWinCOFFObjectTargetWriter> TargetObjectWriter; 132df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola 133801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // Root level file contents. 134801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer COFF::header Header; 135801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer sections Sections; 136801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer symbols Symbols; 137801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer StringTable Strings; 138b162290e39afd49d4c7d342333b331bc96232720Chris Lattner 139801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // Maps used during object file creation. 140801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer section_map SectionMap; 141801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer symbol_map SymbolMap; 142801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 143df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola WinCOFFObjectWriter(MCWinCOFFObjectTargetWriter *MOTW, raw_ostream &OS); 144808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer ~WinCOFFObjectWriter(); 145801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 1464cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer COFFSymbol *createSymbol(StringRef Name); 1474cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer COFFSymbol *GetOrCreateCOFFSymbol(const MCSymbol * Symbol); 1484cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer COFFSection *createSection(StringRef Name); 149801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 150801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer template <typename object_t, typename list_t> 151801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer object_t *createCOFFEntity(llvm::StringRef Name, list_t &List); 152801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 153801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer void DefineSection(MCSectionData const &SectionData); 154801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer void DefineSymbol(MCSymbolData const &SymbolData, MCAssembler &Assembler); 155801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 156a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer void MakeSymbolReal(COFFSymbol &S, size_t Index); 157a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer void MakeSectionReal(COFFSection &S, size_t Number); 158a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 159a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer bool ExportSection(COFFSection const *S); 160801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer bool ExportSymbol(MCSymbolData const &SymbolData, MCAssembler &Asm); 161801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 162a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer bool IsPhysicalSection(COFFSection *S); 163a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 164801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // Entity writing methods. 165801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 166801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer void WriteFileHeader(const COFF::header &Header); 167801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer void WriteSymbol(const COFFSymbol *S); 168801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer void WriteAuxiliarySymbols(const COFFSymbol::AuxiliarySymbols &S); 169801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer void WriteSectionHeader(const COFF::section &S); 170801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer void WriteRelocation(const COFF::relocation &R); 171801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 172801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // MCObjectWriter interface implementation. 173801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 17485f2ecc697a8ca6c8cf08093054cbbb9d2060ccfRafael Espindola void ExecutePostLayoutBinding(MCAssembler &Asm, const MCAsmLayout &Layout); 175801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 176801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer void RecordRelocation(const MCAssembler &Asm, 177801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer const MCAsmLayout &Layout, 178801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer const MCFragment *Fragment, 179801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer const MCFixup &Fixup, 180801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer MCValue Target, 181801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer uint64_t &FixedValue); 182801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 1838f413fa9c00dbaea000ddfe265ab5edd285ea3abRafael Espindola void WriteObject(MCAssembler &Asm, const MCAsmLayout &Layout); 184801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer}; 185801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 186801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 187801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerstatic inline void write_uint32_le(void *Data, uint32_t const &Value) { 188801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer uint8_t *Ptr = reinterpret_cast<uint8_t *>(Data); 189801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Ptr[0] = (Value & 0x000000FF) >> 0; 190801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Ptr[1] = (Value & 0x0000FF00) >> 8; 191801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Ptr[2] = (Value & 0x00FF0000) >> 16; 192801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Ptr[3] = (Value & 0xFF000000) >> 24; 193801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 194801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 195801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerstatic inline void write_uint16_le(void *Data, uint16_t const &Value) { 196801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer uint8_t *Ptr = reinterpret_cast<uint8_t *>(Data); 197801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Ptr[0] = (Value & 0x00FF) >> 0; 198801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Ptr[1] = (Value & 0xFF00) >> 8; 199801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 200801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 201801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerstatic inline void write_uint8_le(void *Data, uint8_t const &Value) { 202801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer uint8_t *Ptr = reinterpret_cast<uint8_t *>(Data); 203801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Ptr[0] = (Value & 0xFF) >> 0; 204801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 205801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 206801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------ 207801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// Symbol class implementation 208801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 209a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. SpencerCOFFSymbol::COFFSymbol(llvm::StringRef name) 210a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer : Name(name.begin(), name.end()) 211a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer , Other(NULL) 212a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer , Section(NULL) 213a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer , Relocations(0) 214a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer , MCData(NULL) { 215801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer memset(&Data, 0, sizeof(Data)); 216801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 217b162290e39afd49d4c7d342333b331bc96232720Chris Lattner 218801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencersize_t COFFSymbol::size() const { 219801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer return COFF::SymbolSize + (Data.NumberOfAuxSymbols * COFF::SymbolSize); 220b162290e39afd49d4c7d342333b331bc96232720Chris Lattner} 221b162290e39afd49d4c7d342333b331bc96232720Chris Lattner 222801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// In the case that the name does not fit within 8 bytes, the offset 223801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// into the string table is stored in the last 4 bytes instead, leaving 224801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// the first 4 bytes as 0. 225801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid COFFSymbol::set_name_offset(uint32_t Offset) { 226801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer write_uint32_le(Data.Name + 0, 0); 227801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer write_uint32_le(Data.Name + 4, Offset); 228801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 229801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 230a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer/// logic to decide if the symbol should be reported in the symbol table 231a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencerbool COFFSymbol::should_keep() const { 232a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer // no section means its external, keep it 233a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer if (Section == NULL) 234a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer return true; 235a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 236a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer // if it has relocations pointing at it, keep it 237a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer if (Relocations > 0) { 238a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer assert(Section->Number != -1 && "Sections with relocations must be real!"); 239a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer return true; 240a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer } 241a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 242a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer // if the section its in is being droped, drop it 243a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer if (Section->Number == -1) 244a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer return false; 245a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 246a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer // if it is the section symbol, keep it 247a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer if (Section->Symbol == this) 248a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer return true; 249a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 250a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer // if its temporary, drop it 251a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer if (MCData && MCData->getSymbol().isTemporary()) 252a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer return false; 253a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 254a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer // otherwise, keep it 255a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer return true; 256a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer} 257a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 258801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------ 259801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// Section class implementation 260801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 261a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. SpencerCOFFSection::COFFSection(llvm::StringRef name) 262a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer : Name(name) 263a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer , MCData(NULL) 264a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer , Symbol(NULL) { 265801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer memset(&Header, 0, sizeof(Header)); 266801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 267801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 268801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencersize_t COFFSection::size() { 269801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer return COFF::SectionSize; 270801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 271801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 272801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------ 273801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// StringTable class implementation 274801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 275801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// Write the length of the string table into Data. 276801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// The length of the string table includes uint32 length header. 277801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid StringTable::update_length() { 278801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer write_uint32_le(&Data.front(), Data.size()); 279801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 280801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 281801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. SpencerStringTable::StringTable() { 282801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // The string table data begins with the length of the entire string table 283801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // including the length header. Allocate space for this header. 284801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Data.resize(4); 2850d64632c9e8a2632b8804bcc3fbc3523588862a7Michael J. Spencer update_length(); 286801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 287801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 288801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencersize_t StringTable::size() const { 289801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer return Data.size(); 290801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 291801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 292801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// Add String to the table iff it is not already there. 293801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// @returns the index into the string table where the string is now located. 294801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencersize_t StringTable::insert(llvm::StringRef String) { 295801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer map::iterator i = Map.find(String); 296801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 297801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer if (i != Map.end()) 298801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer return i->second; 299801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 300801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer size_t Offset = Data.size(); 301801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 302801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // Insert string data into string table. 303801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Data.insert(Data.end(), String.begin(), String.end()); 304801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Data.push_back('\0'); 305801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 306801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // Put a reference to it in the map. 307801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Map[String] = Offset; 308801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 309801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // Update the internal length field. 310801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer update_length(); 311801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 312801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer return Offset; 313801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 314801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 315801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer//------------------------------------------------------------------------------ 316801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer// WinCOFFObjectWriter class implementation 317801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 318df09270ae897e7fa64a7c162de163c32ee181a03Rafael EspindolaWinCOFFObjectWriter::WinCOFFObjectWriter(MCWinCOFFObjectTargetWriter *MOTW, 319df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola raw_ostream &OS) 32082c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer : MCObjectWriter(OS, true) 321df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola , TargetObjectWriter(MOTW) { 322801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer memset(&Header, 0, sizeof(Header)); 323da0bfcdaf95d95a66e306ef6d45f638939272d34Michael J. Spencer 324df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola Header.Machine = TargetObjectWriter->getMachine(); 325801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 326801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 327808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin KramerWinCOFFObjectWriter::~WinCOFFObjectWriter() { 328808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer for (symbols::iterator I = Symbols.begin(), E = Symbols.end(); I != E; ++I) 329808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer delete *I; 330808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer for (sections::iterator I = Sections.begin(), E = Sections.end(); I != E; ++I) 331808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer delete *I; 332808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer} 333808ecfce6a312625bee5c0f4f9831a0d0ed01b16Benjamin Kramer 3344cee2890a66974af506f2125243114cc14bd5556Michael J. SpencerCOFFSymbol *WinCOFFObjectWriter::createSymbol(StringRef Name) { 335801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer return createCOFFEntity<COFFSymbol>(Name, Symbols); 336801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 337801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 3384cee2890a66974af506f2125243114cc14bd5556Michael J. SpencerCOFFSymbol *WinCOFFObjectWriter::GetOrCreateCOFFSymbol(const MCSymbol * Symbol){ 3394cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer symbol_map::iterator i = SymbolMap.find(Symbol); 3404cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer if (i != SymbolMap.end()) 3414cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer return i->second; 3424cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer COFFSymbol *RetSymbol 3434cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer = createCOFFEntity<COFFSymbol>(Symbol->getName(), Symbols); 3444cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer SymbolMap[Symbol] = RetSymbol; 3454cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer return RetSymbol; 3464cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer} 3474cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer 348801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. SpencerCOFFSection *WinCOFFObjectWriter::createSection(llvm::StringRef Name) { 349801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer return createCOFFEntity<COFFSection>(Name, Sections); 350801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 351801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 352801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// A template used to lookup or create a symbol/section, and initialize it if 353801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// needed. 354801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencertemplate <typename object_t, typename list_t> 355801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencerobject_t *WinCOFFObjectWriter::createCOFFEntity(llvm::StringRef Name, 356801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer list_t &List) { 357a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer object_t *Object = new object_t(Name); 358801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 359801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer List.push_back(Object); 360801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 361801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer return Object; 362801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 363801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 364801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// This function takes a section data object from the assembler 365801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// and creates the associated COFF section staging object. 366801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::DefineSection(MCSectionData const &SectionData) { 367d47f4a9c982d264e46a6a2fe0f357288768bb5b9Michael J. Spencer assert(SectionData.getSection().getVariant() == MCSection::SV_COFF 368d47f4a9c982d264e46a6a2fe0f357288768bb5b9Michael J. Spencer && "Got non COFF section in the COFF backend!"); 369801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // FIXME: Not sure how to verify this (at least in a debug build). 370801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer MCSectionCOFF const &Sec = 371801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer static_cast<MCSectionCOFF const &>(SectionData.getSection()); 372801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 373801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer COFFSection *coff_section = createSection(Sec.getSectionName()); 374801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer COFFSymbol *coff_symbol = createSymbol(Sec.getSectionName()); 375801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 376a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer coff_section->Symbol = coff_symbol; 377a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer coff_symbol->Section = coff_section; 378801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer coff_symbol->Data.StorageClass = COFF::IMAGE_SYM_CLASS_STATIC; 379801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 380801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // In this case the auxiliary symbol is a Section Definition. 381801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer coff_symbol->Aux.resize(1); 382801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer memset(&coff_symbol->Aux[0], 0, sizeof(coff_symbol->Aux[0])); 383801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer coff_symbol->Aux[0].AuxType = ATSectionDefinition; 384801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer coff_symbol->Aux[0].Aux.SectionDefinition.Selection = Sec.getSelection(); 385801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 386801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer coff_section->Header.Characteristics = Sec.getCharacteristics(); 387801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 388801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer uint32_t &Characteristics = coff_section->Header.Characteristics; 389801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer switch (SectionData.getAlignment()) { 390801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer case 1: Characteristics |= COFF::IMAGE_SCN_ALIGN_1BYTES; break; 391801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer case 2: Characteristics |= COFF::IMAGE_SCN_ALIGN_2BYTES; break; 392801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer case 4: Characteristics |= COFF::IMAGE_SCN_ALIGN_4BYTES; break; 393801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer case 8: Characteristics |= COFF::IMAGE_SCN_ALIGN_8BYTES; break; 394801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer case 16: Characteristics |= COFF::IMAGE_SCN_ALIGN_16BYTES; break; 395801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer case 32: Characteristics |= COFF::IMAGE_SCN_ALIGN_32BYTES; break; 396801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer case 64: Characteristics |= COFF::IMAGE_SCN_ALIGN_64BYTES; break; 397801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer case 128: Characteristics |= COFF::IMAGE_SCN_ALIGN_128BYTES; break; 398801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer case 256: Characteristics |= COFF::IMAGE_SCN_ALIGN_256BYTES; break; 399801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer case 512: Characteristics |= COFF::IMAGE_SCN_ALIGN_512BYTES; break; 400801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer case 1024: Characteristics |= COFF::IMAGE_SCN_ALIGN_1024BYTES; break; 401801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer case 2048: Characteristics |= COFF::IMAGE_SCN_ALIGN_2048BYTES; break; 402801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer case 4096: Characteristics |= COFF::IMAGE_SCN_ALIGN_4096BYTES; break; 403801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer case 8192: Characteristics |= COFF::IMAGE_SCN_ALIGN_8192BYTES; break; 404801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer default: 405801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer llvm_unreachable("unsupported section alignment"); 406801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer } 407801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 408801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // Bind internal COFF section to MC section. 409801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer coff_section->MCData = &SectionData; 4104cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer SectionMap[&SectionData.getSection()] = coff_section; 411801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer} 412801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 413801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// This function takes a section data object from the assembler 414801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer/// and creates the associated COFF symbol staging object. 415801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencervoid WinCOFFObjectWriter::DefineSymbol(MCSymbolData const &SymbolData, 416a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer MCAssembler &Assembler) { 4174cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer COFFSymbol *coff_symbol = GetOrCreateCOFFSymbol(&SymbolData.getSymbol()); 418801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 419801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer coff_symbol->Data.Type = (SymbolData.getFlags() & 0x0000FFFF) >> 0; 420801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer coff_symbol->Data.StorageClass = (SymbolData.getFlags() & 0x00FF0000) >> 16; 421801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 4224cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer if (SymbolData.getFlags() & COFF::SF_WeakExternal) { 423801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer coff_symbol->Data.StorageClass = COFF::IMAGE_SYM_CLASS_WEAK_EXTERNAL; 424801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 4254cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer if (SymbolData.getSymbol().isVariable()) { 4264cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer coff_symbol->Data.StorageClass = COFF::IMAGE_SYM_CLASS_WEAK_EXTERNAL; 4274cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer const MCExpr *Value = SymbolData.getSymbol().getVariableValue(); 428801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 4294cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer // FIXME: This assert message isn't very good. 4304cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer assert(Value->getKind() == MCExpr::SymbolRef && 4314cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer "Value must be a SymbolRef!"); 432801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 4334cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer const MCSymbolRefExpr *SymbolRef = 4344cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer static_cast<const MCSymbolRefExpr *>(Value); 4354cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer coff_symbol->Other = GetOrCreateCOFFSymbol(&SymbolRef->getSymbol()); 4364cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer } else { 4374cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer std::string WeakName = std::string(".weak.") 4384cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer + SymbolData.getSymbol().getName().str() 4394cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer + ".default"; 4404cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer COFFSymbol *WeakDefault = createSymbol(WeakName); 4414cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer WeakDefault->Data.SectionNumber = COFF::IMAGE_SYM_ABSOLUTE; 4424cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer WeakDefault->Data.StorageClass = COFF::IMAGE_SYM_CLASS_EXTERNAL; 4434cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer WeakDefault->Data.Type = 0; 4444cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer WeakDefault->Data.Value = 0; 4454cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer coff_symbol->Other = WeakDefault; 4464cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer } 447801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 448801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // Setup the Weak External auxiliary symbol. 449801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer coff_symbol->Aux.resize(1); 450801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer memset(&coff_symbol->Aux[0], 0, sizeof(coff_symbol->Aux[0])); 451801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer coff_symbol->Aux[0].AuxType = ATWeakExternal; 452801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer coff_symbol->Aux[0].Aux.WeakExternal.TagIndex = 0; 453801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer coff_symbol->Aux[0].Aux.WeakExternal.Characteristics = 4544cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer COFF::IMAGE_WEAK_EXTERN_SEARCH_LIBRARY; 4554cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer } 4564cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer 4574cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer // If no storage class was specified in the streamer, define it here. 4584cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer if (coff_symbol->Data.StorageClass == 0) { 4594cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer bool external = SymbolData.isExternal() || (SymbolData.Fragment == NULL); 4604cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer 4614cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer coff_symbol->Data.StorageClass = 4624cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer external ? COFF::IMAGE_SYM_CLASS_EXTERNAL : COFF::IMAGE_SYM_CLASS_STATIC; 463801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer } 464801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 465a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer if (SymbolData.Fragment != NULL) 4664cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer coff_symbol->Section = 4674cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer SectionMap[&SymbolData.Fragment->getParent()->getSection()]; 468a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 469801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // Bind internal COFF symbol to MC symbol. 470801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer coff_symbol->MCData = &SymbolData; 4714cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer SymbolMap[&SymbolData.getSymbol()] = coff_symbol; 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++) { 626801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer if (ExportSymbol(*i, Asm)) 627801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer DefineSymbol(*i, Asm); 628801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer } 629b162290e39afd49d4c7d342333b331bc96232720Chris Lattner} 630b162290e39afd49d4c7d342333b331bc96232720Chris Lattner 631b162290e39afd49d4c7d342333b331bc96232720Chris Lattnervoid WinCOFFObjectWriter::RecordRelocation(const MCAssembler &Asm, 632b162290e39afd49d4c7d342333b331bc96232720Chris Lattner const MCAsmLayout &Layout, 633b162290e39afd49d4c7d342333b331bc96232720Chris Lattner const MCFragment *Fragment, 634b162290e39afd49d4c7d342333b331bc96232720Chris Lattner const MCFixup &Fixup, 635b162290e39afd49d4c7d342333b331bc96232720Chris Lattner MCValue Target, 636b162290e39afd49d4c7d342333b331bc96232720Chris Lattner uint64_t &FixedValue) { 637801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer assert(Target.getSymA() != NULL && "Relocation must reference a symbol!"); 63882c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer 63982c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer const MCSymbol *A = &Target.getSymA()->getSymbol(); 64082c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer MCSymbolData &A_SD = Asm.getSymbolData(*A); 641801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 642801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer MCSectionData const *SectionData = Fragment->getParent(); 643801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 64482c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer // Mark this symbol as requiring an entry in the symbol table. 6454cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer assert(SectionMap.find(&SectionData->getSection()) != SectionMap.end() && 646801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer "Section must already have been defined in ExecutePostLayoutBinding!"); 6474cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer assert(SymbolMap.find(&A_SD.getSymbol()) != SymbolMap.end() && 648801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer "Symbol must already have been defined in ExecutePostLayoutBinding!"); 649801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 6504cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer COFFSection *coff_section = SectionMap[&SectionData->getSection()]; 6514cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer COFFSymbol *coff_symbol = SymbolMap[&A_SD.getSymbol()]; 6523660a847f1820d73847539f3959dc069396f8e44Rafael Espindola const MCSymbolRefExpr *SymA = Target.getSymA(); 6533660a847f1820d73847539f3959dc069396f8e44Rafael Espindola const MCSymbolRefExpr *SymB = Target.getSymB(); 6543660a847f1820d73847539f3959dc069396f8e44Rafael Espindola const bool CrossSection = SymB && 6553660a847f1820d73847539f3959dc069396f8e44Rafael Espindola &SymA->getSymbol().getSection() != &SymB->getSymbol().getSection(); 65682c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer 65782c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer if (Target.getSymB()) { 65882c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer const MCSymbol *B = &Target.getSymB()->getSymbol(); 65982c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer MCSymbolData &B_SD = Asm.getSymbolData(*B); 660801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 6611ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola // Offset of the symbol in the section 6621ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola int64_t a = Layout.getSymbolOffset(&B_SD); 66382c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer 6641ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola // Ofeset of the relocation in the section 6651ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola int64_t b = Layout.getFragmentOffset(Fragment) + Fixup.getOffset(); 6661ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola 6671ac7fe0f4dae8a9266fa6ff31ea4939ec64a3e5eRafael Espindola FixedValue = b - a; 66882c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer // In the case where we have SymbA and SymB, we just need to store the delta 66982c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer // between the two symbols. Update FixedValue to account for the delta, and 67082c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer // skip recording the relocation. 6713660a847f1820d73847539f3959dc069396f8e44Rafael Espindola if (!CrossSection) 6723660a847f1820d73847539f3959dc069396f8e44Rafael Espindola return; 67382c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer } else { 67482c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer FixedValue = Target.getConstant(); 67582c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer } 676801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 677801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer COFFRelocation Reloc; 678801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 679425f634917542d7f09c189e2eb130752c6a12d2cDaniel Dunbar Reloc.Data.SymbolTableIndex = 0; 680801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Reloc.Data.VirtualAddress = Layout.getFragmentOffset(Fragment); 681a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 682ea1104a4c376634ed8eb693f6c71e0ac51074949Michael J. Spencer // Turn relocations for temporary symbols into section relocations. 6833660a847f1820d73847539f3959dc069396f8e44Rafael Espindola if (coff_symbol->MCData->getSymbol().isTemporary() || CrossSection) { 684a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer Reloc.Symb = coff_symbol->Section->Symbol; 685eb6e77f8cccd14cdba995ff8231f2c9faea9bfccMichael J. Spencer FixedValue += Layout.getFragmentOffset(coff_symbol->MCData->Fragment) 686eb6e77f8cccd14cdba995ff8231f2c9faea9bfccMichael J. Spencer + coff_symbol->MCData->getOffset(); 687a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer } else 688a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer Reloc.Symb = coff_symbol; 689a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 690a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer ++Reloc.Symb->Relocations; 691801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 692801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Reloc.Data.VirtualAddress += Fixup.getOffset(); 693801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 6943660a847f1820d73847539f3959dc069396f8e44Rafael Espindola unsigned FixupKind = Fixup.getKind(); 6953660a847f1820d73847539f3959dc069396f8e44Rafael Espindola 6963660a847f1820d73847539f3959dc069396f8e44Rafael Espindola if (CrossSection) 6973660a847f1820d73847539f3959dc069396f8e44Rafael Espindola FixupKind = FK_PCRel_4; 6983660a847f1820d73847539f3959dc069396f8e44Rafael Espindola 699df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola Reloc.Data.Type = TargetObjectWriter->getRelocType(FixupKind); 700b156c5d3765637393eda28b04b7fc1e5c55675ebRafael Espindola 701b156c5d3765637393eda28b04b7fc1e5c55675ebRafael Espindola // FIXME: Can anyone explain what this does other than adjust for the size 702b156c5d3765637393eda28b04b7fc1e5c55675ebRafael Espindola // of the offset? 703b156c5d3765637393eda28b04b7fc1e5c55675ebRafael Espindola if (Reloc.Data.Type == COFF::IMAGE_REL_AMD64_REL32 || 704b156c5d3765637393eda28b04b7fc1e5c55675ebRafael Espindola Reloc.Data.Type == COFF::IMAGE_REL_I386_REL32) 70582c84fdd23669d23c02a07498c83b83702979829Michael J. Spencer FixedValue += 4; 706801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 707801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer coff_section->Relocations.push_back(Reloc); 708b162290e39afd49d4c7d342333b331bc96232720Chris Lattner} 709b162290e39afd49d4c7d342333b331bc96232720Chris Lattner 7108f413fa9c00dbaea000ddfe265ab5edd285ea3abRafael Espindolavoid WinCOFFObjectWriter::WriteObject(MCAssembler &Asm, 711b162290e39afd49d4c7d342333b331bc96232720Chris Lattner const MCAsmLayout &Layout) { 712801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // Assign symbol and section indexes and offsets. 713a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer Header.NumberOfSections = 0; 714a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 715a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer for (sections::iterator i = Sections.begin(), 716a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer e = Sections.end(); i != e; i++) { 71785f2ecc697a8ca6c8cf08093054cbbb9d2060ccfRafael Espindola if (Layout.getSectionAddressSize((*i)->MCData) > 0) { 718a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer MakeSectionReal(**i, ++Header.NumberOfSections); 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 762801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // Assign file offsets to COFF object file structures. 763801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 764801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer unsigned offset = 0; 765801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 766801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer offset += COFF::HeaderSize; 767a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer offset += COFF::SectionSize * Header.NumberOfSections; 768801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 769801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer for (MCAssembler::const_iterator i = Asm.begin(), 770801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer e = Asm.end(); 771801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer i != e; i++) { 7724cee2890a66974af506f2125243114cc14bd5556Michael J. Spencer COFFSection *Sec = SectionMap[&i->getSection()]; 773801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 774a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer if (Sec->Number == -1) 775a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer continue; 776a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 77728ca86aa19fe2a5493573164ef0c2c54542ed9daMichael J. Spencer Sec->Header.SizeOfRawData = Layout.getSectionAddressSize(i); 778801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 779a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer if (IsPhysicalSection(Sec)) { 780801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Sec->Header.PointerToRawData = offset; 781801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 782801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer offset += Sec->Header.SizeOfRawData; 783801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer } 784801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 785801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer if (Sec->Relocations.size() > 0) { 786d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer bool RelocationsOverflow = Sec->Relocations.size() >= 0xffff; 787d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer 788d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer if (RelocationsOverflow) { 789d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer // Signal overflow by setting NumberOfSections to max value. Actual 790d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer // size is found in reloc #0. Microsoft tools understand this. 791d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer Sec->Header.NumberOfRelocations = 0xffff; 792d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer } else { 793d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer Sec->Header.NumberOfRelocations = Sec->Relocations.size(); 794d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer } 795801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Sec->Header.PointerToRelocations = offset; 796801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 797d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer if (RelocationsOverflow) { 798d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer // Reloc #0 will contain actual count, so make room for it. 799d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer offset += COFF::RelocationSize; 800d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer } 801d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer 802801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer offset += COFF::RelocationSize * Sec->Relocations.size(); 803801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 804801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer for (relocations::iterator cr = Sec->Relocations.begin(), 805801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer er = Sec->Relocations.end(); 806a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer cr != er; ++cr) { 807a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer assert((*cr).Symb->Index != -1); 808801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer (*cr).Data.SymbolTableIndex = (*cr).Symb->Index; 809801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer } 810801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer } 811801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 812a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer assert(Sec->Symbol->Aux.size() == 1 813a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer && "Section's symbol must have one aux!"); 814a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer AuxSymbol &Aux = Sec->Symbol->Aux[0]; 815801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer assert(Aux.AuxType == ATSectionDefinition && 816801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer "Section's symbol's aux symbol must be a Section Definition!"); 817801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Aux.Aux.SectionDefinition.Length = Sec->Header.SizeOfRawData; 818801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Aux.Aux.SectionDefinition.NumberOfRelocations = 819801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Sec->Header.NumberOfRelocations; 820801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Aux.Aux.SectionDefinition.NumberOfLinenumbers = 821801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Sec->Header.NumberOfLineNumbers; 822801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer } 823801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 824801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer Header.PointerToSymbolTable = offset; 825801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 826ab3de49c48bd3282421ce24323fb6b868a3da6ccMichael J. Spencer Header.TimeDateStamp = sys::TimeValue::now().toEpochTime(); 827ab3de49c48bd3282421ce24323fb6b868a3da6ccMichael J. Spencer 828801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer // Write it all to disk... 829801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer WriteFileHeader(Header); 830801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 831801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer { 832801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer sections::iterator i, ie; 833801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer MCAssembler::const_iterator j, je; 834801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 835801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer for (i = Sections.begin(), ie = Sections.end(); i != ie; i++) 836d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer if ((*i)->Number != -1) { 837d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer if ((*i)->Relocations.size() >= 0xffff) { 838d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer (*i)->Header.Characteristics |= COFF::IMAGE_SCN_LNK_NRELOC_OVFL; 839d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer } 840a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer WriteSectionHeader((*i)->Header); 841d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer } 842801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 843801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer for (i = Sections.begin(), ie = Sections.end(), 844801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer j = Asm.begin(), je = Asm.end(); 845a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer (i != ie) && (j != je); ++i, ++j) { 846a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 847a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer if ((*i)->Number == -1) 848a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer continue; 849a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer 850801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer if ((*i)->Header.PointerToRawData != 0) { 851801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer assert(OS.tell() == (*i)->Header.PointerToRawData && 852801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer "Section::PointerToRawData is insane!"); 853801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 854f77d5b14af6b92403b93f7ed249f9023e99028ccJim Grosbach Asm.writeSectionData(j, Layout); 855801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer } 856801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 857801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer if ((*i)->Relocations.size() > 0) { 858801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer assert(OS.tell() == (*i)->Header.PointerToRelocations && 859801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer "Section::PointerToRelocations is insane!"); 860801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 861d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer if ((*i)->Relocations.size() >= 0xffff) { 862d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer // In case of overflow, write actual relocation count as first 863d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer // relocation. Including the synthetic reloc itself (+ 1). 864d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer COFF::relocation r; 865d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer r.VirtualAddress = (*i)->Relocations.size() + 1; 866d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer r.SymbolTableIndex = 0; 867d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer r.Type = 0; 868d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer WriteRelocation(r); 869d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer } 870d03a29b69d7285ed2ca523d70a45174e33716727Michael J. Spencer 871801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer for (relocations::const_iterator k = (*i)->Relocations.begin(), 872801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer ke = (*i)->Relocations.end(); 873801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer k != ke; k++) { 874801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer WriteRelocation(k->Data); 875801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer } 876801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer } else 877801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer assert((*i)->Header.PointerToRelocations == 0 && 878801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer "Section::PointerToRelocations is insane!"); 879801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer } 880801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer } 881801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 882801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer assert(OS.tell() == Header.PointerToSymbolTable && 883801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer "Header::PointerToSymbolTable is insane!"); 884801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 885801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer for (symbols::iterator i = Symbols.begin(), e = Symbols.end(); i != e; i++) 886a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer if ((*i)->Index != -1) 887a72d87899bc1bfdb17744aae2545a21b3630d3c1Michael J. Spencer WriteSymbol(*i); 888801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer 889801a3591161789e9716b0ec519f0f950b2d0c2acMichael J. Spencer OS.write((char const *)&Strings.Data.front(), Strings.Data.size()); 890b162290e39afd49d4c7d342333b331bc96232720Chris Lattner} 891b162290e39afd49d4c7d342333b331bc96232720Chris Lattner 892df09270ae897e7fa64a7c162de163c32ee181a03Rafael EspindolaMCWinCOFFObjectTargetWriter::MCWinCOFFObjectTargetWriter(unsigned Machine_) : 893df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola Machine(Machine_) { 894df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola} 895df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola 896b162290e39afd49d4c7d342333b331bc96232720Chris Lattner//------------------------------------------------------------------------------ 897b162290e39afd49d4c7d342333b331bc96232720Chris Lattner// WinCOFFObjectWriter factory function 898b162290e39afd49d4c7d342333b331bc96232720Chris Lattner 899b162290e39afd49d4c7d342333b331bc96232720Chris Lattnernamespace llvm { 900df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola MCObjectWriter *createWinCOFFObjectWriter(MCWinCOFFObjectTargetWriter *MOTW, 901df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola raw_ostream &OS) { 902df09270ae897e7fa64a7c162de163c32ee181a03Rafael Espindola return new WinCOFFObjectWriter(MOTW, OS); 903b162290e39afd49d4c7d342333b331bc96232720Chris Lattner } 904933304ef0c3ec18c23d0b385c2117a6eae790430Michael J. Spencer} 905