2015-05-29 03:09:30 +08:00
|
|
|
//===- Writer.h -----------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Linker
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef LLD_COFF_WRITER_H
|
|
|
|
#define LLD_COFF_WRITER_H
|
|
|
|
|
2015-06-07 09:15:04 +08:00
|
|
|
#include "DLL.h"
|
2015-05-29 03:09:30 +08:00
|
|
|
#include "InputFiles.h"
|
|
|
|
#include "SymbolTable.h"
|
|
|
|
#include "llvm/Support/FileOutputBuffer.h"
|
|
|
|
#include <memory>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
namespace lld {
|
|
|
|
namespace coff {
|
|
|
|
|
2015-06-16 02:03:47 +08:00
|
|
|
// Mask for section types (code, data, bss, disacardable, etc.)
|
|
|
|
// and permissions (writable, readable or executable).
|
|
|
|
const uint32_t PermMask = 0xFF0000F0;
|
2015-05-29 03:09:30 +08:00
|
|
|
|
2015-06-25 04:40:03 +08:00
|
|
|
// Implemented in ICF.cpp.
|
|
|
|
void doICF(const std::vector<Chunk *> &Chunks);
|
|
|
|
|
2015-05-29 03:09:30 +08:00
|
|
|
// OutputSection represents a section in an output file. It's a
|
|
|
|
// container of chunks. OutputSection and Chunk are 1:N relationship.
|
|
|
|
// Chunks cannot belong to more than one OutputSections. The writer
|
|
|
|
// creates multiple OutputSections and assign them unique,
|
|
|
|
// non-overlapping file offsets and RVAs.
|
|
|
|
class OutputSection {
|
|
|
|
public:
|
2015-05-31 03:28:58 +08:00
|
|
|
OutputSection(StringRef N, uint32_t SI)
|
|
|
|
: Name(N), SectionIndex(SI), Header({}) {}
|
2015-05-29 03:09:30 +08:00
|
|
|
void setRVA(uint64_t);
|
|
|
|
void setFileOffset(uint64_t);
|
|
|
|
void addChunk(Chunk *C);
|
|
|
|
StringRef getName() { return Name; }
|
|
|
|
uint64_t getSectionIndex() { return SectionIndex; }
|
|
|
|
std::vector<Chunk *> &getChunks() { return Chunks; }
|
|
|
|
void addPermissions(uint32_t C);
|
|
|
|
uint32_t getPermissions() { return Header.Characteristics & PermMask; }
|
|
|
|
uint32_t getCharacteristics() { return Header.Characteristics; }
|
|
|
|
uint64_t getRVA() { return Header.VirtualAddress; }
|
|
|
|
uint64_t getFileOff() { return Header.PointerToRawData; }
|
2015-06-07 07:19:38 +08:00
|
|
|
void writeHeaderTo(uint8_t *Buf);
|
2015-05-29 03:09:30 +08:00
|
|
|
|
|
|
|
// Returns the size of this section in an executable memory image.
|
|
|
|
// This may be smaller than the raw size (the raw size is multiple
|
|
|
|
// of disk sector size, so there may be padding at end), or may be
|
|
|
|
// larger (if that's the case, the loader reserves spaces after end
|
|
|
|
// of raw data).
|
|
|
|
uint64_t getVirtualSize() { return Header.VirtualSize; }
|
|
|
|
|
|
|
|
// Returns the size of the section in the output file.
|
|
|
|
uint64_t getRawSize() { return Header.SizeOfRawData; }
|
|
|
|
|
2015-05-31 03:09:50 +08:00
|
|
|
// Set offset into the string table storing this section name.
|
|
|
|
// Used only when the name is longer than 8 bytes.
|
|
|
|
void setStringTableOff(uint32_t V) { StringTableOff = V; }
|
|
|
|
|
2015-05-29 03:09:30 +08:00
|
|
|
private:
|
|
|
|
StringRef Name;
|
|
|
|
uint32_t SectionIndex;
|
2015-05-31 03:28:58 +08:00
|
|
|
coff_section Header;
|
2015-05-31 03:09:50 +08:00
|
|
|
uint32_t StringTableOff = 0;
|
2015-05-29 03:09:30 +08:00
|
|
|
std::vector<Chunk *> Chunks;
|
|
|
|
};
|
|
|
|
|
|
|
|
// The writer writes a SymbolTable result to a file.
|
|
|
|
class Writer {
|
|
|
|
public:
|
|
|
|
explicit Writer(SymbolTable *T) : Symtab(T) {}
|
|
|
|
std::error_code write(StringRef Path);
|
|
|
|
|
|
|
|
private:
|
|
|
|
void markLive();
|
2015-06-25 04:40:03 +08:00
|
|
|
void dedupCOMDATs();
|
2015-05-29 03:09:30 +08:00
|
|
|
void createSections();
|
2015-06-25 11:31:47 +08:00
|
|
|
void createMiscChunks();
|
2015-05-29 03:09:30 +08:00
|
|
|
void createImportTables();
|
2015-06-17 08:16:33 +08:00
|
|
|
void createExportTable();
|
2015-05-29 03:09:30 +08:00
|
|
|
void assignAddresses();
|
|
|
|
void removeEmptySections();
|
|
|
|
std::error_code openFile(StringRef OutputPath);
|
|
|
|
void writeHeader();
|
|
|
|
void writeSections();
|
2015-06-21 12:00:54 +08:00
|
|
|
void sortExceptionTable();
|
2015-05-29 03:09:30 +08:00
|
|
|
void applyRelocations();
|
|
|
|
|
|
|
|
OutputSection *findSection(StringRef Name);
|
|
|
|
OutputSection *createSection(StringRef Name);
|
2015-06-15 09:23:58 +08:00
|
|
|
void addBaserels(OutputSection *Dest);
|
|
|
|
void addBaserelBlocks(OutputSection *Dest, std::vector<uint32_t> &V);
|
2015-05-29 03:09:30 +08:00
|
|
|
|
|
|
|
uint32_t getSizeOfInitializedData();
|
|
|
|
std::map<StringRef, std::vector<DefinedImportData *>> binImports();
|
|
|
|
|
|
|
|
SymbolTable *Symtab;
|
|
|
|
std::unique_ptr<llvm::FileOutputBuffer> Buffer;
|
2015-06-04 00:44:00 +08:00
|
|
|
llvm::SpecificBumpPtrAllocator<OutputSection> CAlloc;
|
2015-06-15 09:23:58 +08:00
|
|
|
llvm::SpecificBumpPtrAllocator<BaserelChunk> BAlloc;
|
2015-06-04 00:44:00 +08:00
|
|
|
std::vector<OutputSection *> OutputSections;
|
2015-06-22 06:31:52 +08:00
|
|
|
IdataContents Idata;
|
|
|
|
DelayLoadContents DelayIdata;
|
|
|
|
EdataContents Edata;
|
2015-05-29 03:09:30 +08:00
|
|
|
|
|
|
|
uint64_t FileSize;
|
|
|
|
uint64_t SizeOfImage;
|
|
|
|
uint64_t SizeOfHeaders;
|
|
|
|
|
|
|
|
std::vector<std::unique_ptr<Chunk>> Chunks;
|
|
|
|
};
|
|
|
|
|
2015-06-13 05:37:55 +08:00
|
|
|
} // namespace coff
|
2015-05-29 03:09:30 +08:00
|
|
|
} // namespace lld
|
|
|
|
|
|
|
|
#endif
|