mirror of
https://github.com/RPCS3/llvm-mirror.git
synced 2024-11-22 18:54:02 +01:00
d12f0adda0
The full GSYM patch started with: https://reviews.llvm.org/D53379 This patch add the ability to encode data using the new llvm::gsym::FileWriter class. FileWriter is a simplified binary data writer class that doesn't require targets, target definitions, architectures, or require any other optional compile time libraries to be enabled via the build process. This class needs the ability to seek to different spots in the binary data that it produces to fix up offsets and sizes in GSYM data. It currently uses std::ostream over llvm::raw_ostream because llvm::raw_ostream doesn't support seeking which is required when encoding and decoding GSYM data. AddressRange objects are encoded and decoded to be relative to a base address. This will be the FunctionInfo's start address if the AddressRange is directly contained in a FunctionInfo, or a base address of the containing parent AddressRange or AddressRanges. This allows address ranges to be efficiently encoded using ULEB128 encodings as we encode the offset and size of each range instead of full addresses. This also makes encoded addresses easy to relocate as we just need to relocate one base address. Differential Revision: https://reviews.llvm.org/D63828 llvm-svn: 369587
79 lines
2.3 KiB
C++
79 lines
2.3 KiB
C++
//===- FileWriter.cpp -------------------------------------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "llvm/DebugInfo/GSYM/FileWriter.h"
|
|
#include "llvm/Support/LEB128.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
#include <cassert>
|
|
|
|
using namespace llvm;
|
|
using namespace gsym;
|
|
|
|
FileWriter::~FileWriter() { OS.flush(); }
|
|
|
|
void FileWriter::writeSLEB(int64_t S) {
|
|
uint8_t Bytes[32];
|
|
auto Length = encodeSLEB128(S, Bytes);
|
|
assert(Length < sizeof(Bytes));
|
|
OS.write(reinterpret_cast<const char *>(Bytes), Length);
|
|
}
|
|
|
|
void FileWriter::writeULEB(uint64_t U) {
|
|
uint8_t Bytes[32];
|
|
auto Length = encodeULEB128(U, Bytes);
|
|
assert(Length < sizeof(Bytes));
|
|
OS.write(reinterpret_cast<const char *>(Bytes), Length);
|
|
}
|
|
|
|
void FileWriter::writeU8(uint8_t U) {
|
|
OS.write(reinterpret_cast<const char *>(&U), sizeof(U));
|
|
}
|
|
|
|
void FileWriter::writeU16(uint16_t U) {
|
|
const uint16_t Swapped = support::endian::byte_swap(U, ByteOrder);
|
|
OS.write(reinterpret_cast<const char *>(&Swapped), sizeof(Swapped));
|
|
}
|
|
|
|
void FileWriter::writeU32(uint32_t U) {
|
|
const uint32_t Swapped = support::endian::byte_swap(U, ByteOrder);
|
|
OS.write(reinterpret_cast<const char *>(&Swapped), sizeof(Swapped));
|
|
}
|
|
|
|
void FileWriter::writeU64(uint64_t U) {
|
|
const uint64_t Swapped = support::endian::byte_swap(U, ByteOrder);
|
|
OS.write(reinterpret_cast<const char *>(&Swapped), sizeof(Swapped));
|
|
}
|
|
|
|
void FileWriter::fixup32(uint32_t U, uint64_t Offset) {
|
|
const uint32_t Swapped = support::endian::byte_swap(U, ByteOrder);
|
|
OS.pwrite(reinterpret_cast<const char *>(&Swapped), sizeof(Swapped),
|
|
Offset);
|
|
}
|
|
|
|
void FileWriter::writeData(llvm::ArrayRef<uint8_t> Data) {
|
|
OS.write(reinterpret_cast<const char *>(Data.data()), Data.size());
|
|
}
|
|
|
|
void FileWriter::writeNullTerminated(llvm::StringRef Str) {
|
|
OS << Str << '\0';
|
|
}
|
|
|
|
uint64_t FileWriter::tell() {
|
|
return OS.tell();
|
|
}
|
|
|
|
void FileWriter::alignTo(size_t Align) {
|
|
off_t Offset = OS.tell();
|
|
off_t AlignedOffset = (Offset + Align - 1) / Align * Align;
|
|
if (AlignedOffset == Offset)
|
|
return;
|
|
off_t PadCount = AlignedOffset - Offset;
|
|
OS.write_zeros(PadCount);
|
|
}
|