blob: 5ae7a9c7f739ab5742d0c2372dca891ece0d92e3 [file]
//===- HeaderV2.cpp ---------------------------------------------*- C++ -*-===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "llvm/DebugInfo/GSYM/HeaderV2.h"
#include "llvm/DebugInfo/GSYM/FileWriter.h"
#include "llvm/DebugInfo/GSYM/GsymDataExtractor.h"
#include "llvm/Support/Format.h"
#include "llvm/Support/raw_ostream.h"
#define HEX8(v) llvm::format_hex(v, 4)
#define HEX16(v) llvm::format_hex(v, 6)
#define HEX32(v) llvm::format_hex(v, 10)
#define HEX64(v) llvm::format_hex(v, 18)
using namespace llvm;
using namespace gsym;
raw_ostream &llvm::gsym::operator<<(raw_ostream &OS, const HeaderV2 &H) {
OS << "Header:\n";
OS << " Magic = " << HEX32(H.Magic) << "\n";
OS << " Version = " << HEX16(H.Version) << '\n';
OS << " AddrOffSize = " << HEX8(H.AddrOffSize) << '\n';
OS << " StrTableEnc = " << HEX8(static_cast<uint8_t>(H.StrTableEncoding))
<< '\n';
OS << " BaseAddress = " << HEX64(H.BaseAddress) << '\n';
OS << " NumAddresses = " << HEX32(H.NumAddresses) << '\n';
return OS;
}
llvm::Error HeaderV2::checkForError() const {
if (Magic != GSYM_MAGIC)
return createStringError(std::errc::invalid_argument,
"invalid GSYM magic 0x%8.8x", Magic);
if (Version != HeaderV2::getVersion())
return createStringError(std::errc::invalid_argument,
"unsupported GSYM version %u", Version);
if (AddrOffSize < 1 || AddrOffSize > 8)
return createStringError(std::errc::invalid_argument,
"invalid address offset size %u", AddrOffSize);
uint8_t Encoding = static_cast<uint8_t>(StrTableEncoding);
switch (Encoding) {
case static_cast<uint8_t>(StringTableEncoding::Default):
break;
default:
return createStringError(std::errc::invalid_argument,
"unsupported string table encoding %u", Encoding);
}
return Error::success();
}
llvm::Expected<HeaderV2> HeaderV2::decode(GsymDataExtractor &Data) {
uint64_t Offset = 0;
// The fixed portion of the HeaderV2 is 20 bytes:
// Magic(4) + Version(2) + AddrOffSize(1) + StrTableEncoding(1) +
// BaseAddress(8) + NumAddresses(4)
const uint64_t FixedHeaderSize = HeaderV2::getEncodedSize();
if (!Data.isValidOffsetForDataOfSize(Offset, FixedHeaderSize))
return createStringError(std::errc::invalid_argument,
"not enough data for a gsym::HeaderV2");
HeaderV2 H;
H.Magic = Data.getU32(&Offset);
H.Version = Data.getU16(&Offset);
H.AddrOffSize = Data.getU8(&Offset);
H.StrTableEncoding = static_cast<StringTableEncoding>(Data.getU8(&Offset));
H.BaseAddress = Data.getU64(&Offset);
H.NumAddresses = Data.getU32(&Offset);
if (llvm::Error Err = H.checkForError())
return std::move(Err);
return H;
}
llvm::Error HeaderV2::encode(FileWriter &O) const {
if (llvm::Error Err = checkForError())
return Err;
O.writeU32(Magic);
O.writeU16(Version);
O.writeU8(AddrOffSize);
O.writeU8(static_cast<uint8_t>(StrTableEncoding));
O.writeU64(BaseAddress);
O.writeU32(NumAddresses);
return Error::success();
}
bool llvm::gsym::operator==(const HeaderV2 &LHS, const HeaderV2 &RHS) {
return LHS.Magic == RHS.Magic && LHS.Version == RHS.Version &&
LHS.AddrOffSize == RHS.AddrOffSize &&
LHS.StrTableEncoding == RHS.StrTableEncoding &&
LHS.BaseAddress == RHS.BaseAddress &&
LHS.NumAddresses == RHS.NumAddresses;
}