|  | //===-- llvm/CodeGen/BinaryObject.h - Binary Object. -----------*- C++ -*-===// | 
|  | // | 
|  | //                     The LLVM Compiler Infrastructure | 
|  | // | 
|  | // This file is distributed under the University of Illinois Open Source | 
|  | // License. See LICENSE.TXT for details. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  | // | 
|  | // This file defines a Binary Object Aka. "blob" for holding data from code | 
|  | // generators, ready for data to the object module code writters. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #ifndef LLVM_CODEGEN_BINARYOBJECT_H | 
|  | #define LLVM_CODEGEN_BINARYOBJECT_H | 
|  |  | 
|  | #include "llvm/CodeGen/MachineRelocation.h" | 
|  | #include "llvm/Support/DataTypes.h" | 
|  |  | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | namespace llvm { | 
|  |  | 
|  | typedef std::vector<uint8_t> BinaryData; | 
|  |  | 
|  | class BinaryObject { | 
|  | protected: | 
|  | std::string Name; | 
|  | bool IsLittleEndian; | 
|  | bool Is64Bit; | 
|  | BinaryData Data; | 
|  | std::vector<MachineRelocation> Relocations; | 
|  |  | 
|  | public: | 
|  | /// Constructors and destructor | 
|  | BinaryObject() {} | 
|  |  | 
|  | BinaryObject(bool isLittleEndian, bool is64Bit) | 
|  | : IsLittleEndian(isLittleEndian), Is64Bit(is64Bit) {} | 
|  |  | 
|  | BinaryObject(const std::string &name, bool isLittleEndian, bool is64Bit) | 
|  | : Name(name), IsLittleEndian(isLittleEndian), Is64Bit(is64Bit) {} | 
|  |  | 
|  | ~BinaryObject() {} | 
|  |  | 
|  | /// getName - get name of BinaryObject | 
|  | inline std::string getName() const { return Name; } | 
|  |  | 
|  | /// get size of binary data | 
|  | size_t size() const { | 
|  | return Data.size(); | 
|  | } | 
|  |  | 
|  | /// get binary data | 
|  | BinaryData& getData() { | 
|  | return Data; | 
|  | } | 
|  |  | 
|  | /// get machine relocations | 
|  | const std::vector<MachineRelocation>& getRelocations() const { | 
|  | return Relocations; | 
|  | } | 
|  |  | 
|  | /// hasRelocations - Return true if 'Relocations' is not empty | 
|  | bool hasRelocations() const { | 
|  | return !Relocations.empty(); | 
|  | } | 
|  |  | 
|  | /// emitZeros - This callback is invoked to emit a arbitrary number | 
|  | /// of zero bytes to the data stream. | 
|  | inline void emitZeros(unsigned Size) { | 
|  | for (unsigned i=0; i < Size; ++i) | 
|  | emitByte(0); | 
|  | } | 
|  |  | 
|  | /// emitByte - This callback is invoked when a byte needs to be | 
|  | /// written to the data stream. | 
|  | inline void emitByte(uint8_t B) { | 
|  | Data.push_back(B); | 
|  | } | 
|  |  | 
|  | /// emitWord16 - This callback is invoked when a 16-bit word needs to be | 
|  | /// written to the data stream in correct endian format and correct size. | 
|  | inline void emitWord16(uint16_t W) { | 
|  | if (IsLittleEndian) | 
|  | emitWord16LE(W); | 
|  | else | 
|  | emitWord16BE(W); | 
|  | } | 
|  |  | 
|  | /// emitWord16LE - This callback is invoked when a 16-bit word needs to be | 
|  | /// written to the data stream in correct endian format and correct size. | 
|  | inline void emitWord16LE(uint16_t W) { | 
|  | Data.push_back((uint8_t)(W >> 0)); | 
|  | Data.push_back((uint8_t)(W >> 8)); | 
|  | } | 
|  |  | 
|  | /// emitWord16BE - This callback is invoked when a 16-bit word needs to be | 
|  | /// written to the data stream in correct endian format and correct size. | 
|  | inline void emitWord16BE(uint16_t W) { | 
|  | Data.push_back((uint8_t)(W >> 8)); | 
|  | Data.push_back((uint8_t)(W >> 0)); | 
|  | } | 
|  |  | 
|  | /// emitWord - This callback is invoked when a word needs to be | 
|  | /// written to the data stream in correct endian format and correct size. | 
|  | inline void emitWord(uint64_t W) { | 
|  | if (!Is64Bit) | 
|  | emitWord32(W); | 
|  | else | 
|  | emitWord64(W); | 
|  | } | 
|  |  | 
|  | /// emitWord32 - This callback is invoked when a 32-bit word needs to be | 
|  | /// written to the data stream in correct endian format. | 
|  | inline void emitWord32(uint32_t W) { | 
|  | if (IsLittleEndian) | 
|  | emitWordLE(W); | 
|  | else | 
|  | emitWordBE(W); | 
|  | } | 
|  |  | 
|  | /// emitWord64 - This callback is invoked when a 32-bit word needs to be | 
|  | /// written to the data stream in correct endian format. | 
|  | inline void emitWord64(uint64_t W) { | 
|  | if (IsLittleEndian) | 
|  | emitDWordLE(W); | 
|  | else | 
|  | emitDWordBE(W); | 
|  | } | 
|  |  | 
|  | /// emitWord64 - This callback is invoked when a x86_fp80 needs to be | 
|  | /// written to the data stream in correct endian format. | 
|  | inline void emitWordFP80(const uint64_t *W, unsigned PadSize) { | 
|  | if (IsLittleEndian) { | 
|  | emitWord64(W[0]); | 
|  | emitWord16(W[1]); | 
|  | } else { | 
|  | emitWord16(W[1]); | 
|  | emitWord64(W[0]); | 
|  | } | 
|  | emitZeros(PadSize); | 
|  | } | 
|  |  | 
|  | /// emitWordLE - This callback is invoked when a 32-bit word needs to be | 
|  | /// written to the data stream in little-endian format. | 
|  | inline void emitWordLE(uint32_t W) { | 
|  | Data.push_back((uint8_t)(W >>  0)); | 
|  | Data.push_back((uint8_t)(W >>  8)); | 
|  | Data.push_back((uint8_t)(W >> 16)); | 
|  | Data.push_back((uint8_t)(W >> 24)); | 
|  | } | 
|  |  | 
|  | /// emitWordBE - This callback is invoked when a 32-bit word needs to be | 
|  | /// written to the data stream in big-endian format. | 
|  | /// | 
|  | inline void emitWordBE(uint32_t W) { | 
|  | Data.push_back((uint8_t)(W >> 24)); | 
|  | Data.push_back((uint8_t)(W >> 16)); | 
|  | Data.push_back((uint8_t)(W >>  8)); | 
|  | Data.push_back((uint8_t)(W >>  0)); | 
|  | } | 
|  |  | 
|  | /// emitDWordLE - This callback is invoked when a 64-bit word needs to be | 
|  | /// written to the data stream in little-endian format. | 
|  | inline void emitDWordLE(uint64_t W) { | 
|  | Data.push_back((uint8_t)(W >>  0)); | 
|  | Data.push_back((uint8_t)(W >>  8)); | 
|  | Data.push_back((uint8_t)(W >> 16)); | 
|  | Data.push_back((uint8_t)(W >> 24)); | 
|  | Data.push_back((uint8_t)(W >> 32)); | 
|  | Data.push_back((uint8_t)(W >> 40)); | 
|  | Data.push_back((uint8_t)(W >> 48)); | 
|  | Data.push_back((uint8_t)(W >> 56)); | 
|  | } | 
|  |  | 
|  | /// emitDWordBE - This callback is invoked when a 64-bit word needs to be | 
|  | /// written to the data stream in big-endian format. | 
|  | inline void emitDWordBE(uint64_t W) { | 
|  | Data.push_back((uint8_t)(W >> 56)); | 
|  | Data.push_back((uint8_t)(W >> 48)); | 
|  | Data.push_back((uint8_t)(W >> 40)); | 
|  | Data.push_back((uint8_t)(W >> 32)); | 
|  | Data.push_back((uint8_t)(W >> 24)); | 
|  | Data.push_back((uint8_t)(W >> 16)); | 
|  | Data.push_back((uint8_t)(W >>  8)); | 
|  | Data.push_back((uint8_t)(W >>  0)); | 
|  | } | 
|  |  | 
|  | /// fixByte - This callback is invoked when a byte needs to be | 
|  | /// fixup the buffer. | 
|  | inline void fixByte(uint8_t B, uint32_t offset) { | 
|  | Data[offset] = B; | 
|  | } | 
|  |  | 
|  | /// fixWord16 - This callback is invoked when a 16-bit word needs to | 
|  | /// fixup the data stream in correct endian format. | 
|  | inline void fixWord16(uint16_t W, uint32_t offset) { | 
|  | if (IsLittleEndian) | 
|  | fixWord16LE(W, offset); | 
|  | else | 
|  | fixWord16BE(W, offset); | 
|  | } | 
|  |  | 
|  | /// emitWord16LE - This callback is invoked when a 16-bit word needs to | 
|  | /// fixup the data stream in little endian format. | 
|  | inline void fixWord16LE(uint16_t W, uint32_t offset) { | 
|  | Data[offset]   = (uint8_t)(W >> 0); | 
|  | Data[++offset] = (uint8_t)(W >> 8); | 
|  | } | 
|  |  | 
|  | /// fixWord16BE - This callback is invoked when a 16-bit word needs to | 
|  | /// fixup data stream in big endian format. | 
|  | inline void fixWord16BE(uint16_t W, uint32_t offset) { | 
|  | Data[offset]   = (uint8_t)(W >> 8); | 
|  | Data[++offset] = (uint8_t)(W >> 0); | 
|  | } | 
|  |  | 
|  | /// emitWord - This callback is invoked when a word needs to | 
|  | /// fixup the data in correct endian format and correct size. | 
|  | inline void fixWord(uint64_t W, uint32_t offset) { | 
|  | if (!Is64Bit) | 
|  | fixWord32(W, offset); | 
|  | else | 
|  | fixWord64(W, offset); | 
|  | } | 
|  |  | 
|  | /// fixWord32 - This callback is invoked when a 32-bit word needs to | 
|  | /// fixup the data in correct endian format. | 
|  | inline void fixWord32(uint32_t W, uint32_t offset) { | 
|  | if (IsLittleEndian) | 
|  | fixWord32LE(W, offset); | 
|  | else | 
|  | fixWord32BE(W, offset); | 
|  | } | 
|  |  | 
|  | /// fixWord32LE - This callback is invoked when a 32-bit word needs to | 
|  | /// fixup the data in little endian format. | 
|  | inline void fixWord32LE(uint32_t W, uint32_t offset) { | 
|  | Data[offset]   = (uint8_t)(W >>  0); | 
|  | Data[++offset] = (uint8_t)(W >>  8); | 
|  | Data[++offset] = (uint8_t)(W >> 16); | 
|  | Data[++offset] = (uint8_t)(W >> 24); | 
|  | } | 
|  |  | 
|  | /// fixWord32BE - This callback is invoked when a 32-bit word needs to | 
|  | /// fixup the data in big endian format. | 
|  | inline void fixWord32BE(uint32_t W, uint32_t offset) { | 
|  | Data[offset]   = (uint8_t)(W >> 24); | 
|  | Data[++offset] = (uint8_t)(W >> 16); | 
|  | Data[++offset] = (uint8_t)(W >>  8); | 
|  | Data[++offset] = (uint8_t)(W >>  0); | 
|  | } | 
|  |  | 
|  | /// fixWord64 - This callback is invoked when a 64-bit word needs to | 
|  | /// fixup the data in correct endian format. | 
|  | inline void fixWord64(uint64_t W, uint32_t offset) { | 
|  | if (IsLittleEndian) | 
|  | fixWord64LE(W, offset); | 
|  | else | 
|  | fixWord64BE(W, offset); | 
|  | } | 
|  |  | 
|  | /// fixWord64BE - This callback is invoked when a 64-bit word needs to | 
|  | /// fixup the data in little endian format. | 
|  | inline void fixWord64LE(uint64_t W, uint32_t offset) { | 
|  | Data[offset]   = (uint8_t)(W >>  0); | 
|  | Data[++offset] = (uint8_t)(W >>  8); | 
|  | Data[++offset] = (uint8_t)(W >> 16); | 
|  | Data[++offset] = (uint8_t)(W >> 24); | 
|  | Data[++offset] = (uint8_t)(W >> 32); | 
|  | Data[++offset] = (uint8_t)(W >> 40); | 
|  | Data[++offset] = (uint8_t)(W >> 48); | 
|  | Data[++offset] = (uint8_t)(W >> 56); | 
|  | } | 
|  |  | 
|  | /// fixWord64BE - This callback is invoked when a 64-bit word needs to | 
|  | /// fixup the data in big endian format. | 
|  | inline void fixWord64BE(uint64_t W, uint32_t offset) { | 
|  | Data[offset]   = (uint8_t)(W >> 56); | 
|  | Data[++offset] = (uint8_t)(W >> 48); | 
|  | Data[++offset] = (uint8_t)(W >> 40); | 
|  | Data[++offset] = (uint8_t)(W >> 32); | 
|  | Data[++offset] = (uint8_t)(W >> 24); | 
|  | Data[++offset] = (uint8_t)(W >> 16); | 
|  | Data[++offset] = (uint8_t)(W >>  8); | 
|  | Data[++offset] = (uint8_t)(W >>  0); | 
|  | } | 
|  |  | 
|  | /// emitAlignment - Pad the data to the specified alignment. | 
|  | void emitAlignment(unsigned Alignment, uint8_t fill = 0) { | 
|  | if (Alignment <= 1) return; | 
|  | unsigned PadSize = -Data.size() & (Alignment-1); | 
|  | for (unsigned i = 0; i<PadSize; ++i) | 
|  | Data.push_back(fill); | 
|  | } | 
|  |  | 
|  | /// emitULEB128Bytes - This callback is invoked when a ULEB128 needs to be | 
|  | /// written to the data stream. | 
|  | void emitULEB128Bytes(uint64_t Value) { | 
|  | do { | 
|  | uint8_t Byte = (uint8_t)(Value & 0x7f); | 
|  | Value >>= 7; | 
|  | if (Value) Byte |= 0x80; | 
|  | emitByte(Byte); | 
|  | } while (Value); | 
|  | } | 
|  |  | 
|  | /// emitSLEB128Bytes - This callback is invoked when a SLEB128 needs to be | 
|  | /// written to the data stream. | 
|  | void emitSLEB128Bytes(int64_t Value) { | 
|  | int Sign = Value >> (8 * sizeof(Value) - 1); | 
|  | bool IsMore; | 
|  |  | 
|  | do { | 
|  | uint8_t Byte = (uint8_t)(Value & 0x7f); | 
|  | Value >>= 7; | 
|  | IsMore = Value != Sign || ((Byte ^ Sign) & 0x40) != 0; | 
|  | if (IsMore) Byte |= 0x80; | 
|  | emitByte(Byte); | 
|  | } while (IsMore); | 
|  | } | 
|  |  | 
|  | /// emitString - This callback is invoked when a String needs to be | 
|  | /// written to the data stream. | 
|  | void emitString(const std::string &String) { | 
|  | for (unsigned i = 0, N = static_cast<unsigned>(String.size()); i<N; ++i) { | 
|  | unsigned char C = String[i]; | 
|  | emitByte(C); | 
|  | } | 
|  | emitByte(0); | 
|  | } | 
|  |  | 
|  | /// getCurrentPCOffset - Return the offset from the start of the emitted | 
|  | /// buffer that we are currently writing to. | 
|  | uintptr_t getCurrentPCOffset() const { | 
|  | return Data.size(); | 
|  | } | 
|  |  | 
|  | /// addRelocation - Whenever a relocatable address is needed, it should be | 
|  | /// noted with this interface. | 
|  | void addRelocation(const MachineRelocation& relocation) { | 
|  | Relocations.push_back(relocation); | 
|  | } | 
|  |  | 
|  | }; | 
|  |  | 
|  | } // end namespace llvm | 
|  |  | 
|  | #endif | 
|  |  |