172 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			172 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			C++
		
	
	
	
//===-- RecordSerialization.cpp -------------------------------------------===//
 | 
						|
//
 | 
						|
//                     The LLVM Compiler Infrastructure
 | 
						|
//
 | 
						|
// This file is distributed under the University of Illinois Open Source
 | 
						|
// License. See LICENSE.TXT for details.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
//
 | 
						|
// Utilities for serializing and deserializing CodeView records.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
 | 
						|
#include "llvm/DebugInfo/CodeView/RecordSerialization.h"
 | 
						|
#include "llvm/ADT/APInt.h"
 | 
						|
#include "llvm/ADT/APSInt.h"
 | 
						|
#include "llvm/DebugInfo/CodeView/TypeRecord.h"
 | 
						|
 | 
						|
using namespace llvm;
 | 
						|
using namespace llvm::codeview;
 | 
						|
using namespace llvm::support;
 | 
						|
 | 
						|
/// Reinterpret a byte array as an array of characters. Does not interpret as
 | 
						|
/// a C string, as StringRef has several helpers (split) that make that easy.
 | 
						|
StringRef llvm::codeview::getBytesAsCharacters(ArrayRef<uint8_t> LeafData) {
 | 
						|
  return StringRef(reinterpret_cast<const char *>(LeafData.data()),
 | 
						|
                   LeafData.size());
 | 
						|
}
 | 
						|
 | 
						|
StringRef llvm::codeview::getBytesAsCString(ArrayRef<uint8_t> LeafData) {
 | 
						|
  return getBytesAsCharacters(LeafData).split('\0').first;
 | 
						|
}
 | 
						|
 | 
						|
std::error_code llvm::codeview::consume(ArrayRef<uint8_t> &Data, APSInt &Num) {
 | 
						|
  // Used to avoid overload ambiguity on APInt construtor.
 | 
						|
  bool FalseVal = false;
 | 
						|
  if (Data.size() < 2)
 | 
						|
    return std::make_error_code(std::errc::illegal_byte_sequence);
 | 
						|
  uint16_t Short = *reinterpret_cast<const ulittle16_t *>(Data.data());
 | 
						|
  Data = Data.drop_front(2);
 | 
						|
  if (Short < LF_NUMERIC) {
 | 
						|
    Num = APSInt(APInt(/*numBits=*/16, Short, /*isSigned=*/false),
 | 
						|
                 /*isUnsigned=*/true);
 | 
						|
    return std::error_code();
 | 
						|
  }
 | 
						|
  switch (Short) {
 | 
						|
  case LF_CHAR:
 | 
						|
    if (Data.size() < 1)
 | 
						|
      return std::make_error_code(std::errc::illegal_byte_sequence);
 | 
						|
    Num = APSInt(APInt(/*numBits=*/8,
 | 
						|
                       *reinterpret_cast<const int8_t *>(Data.data()),
 | 
						|
                       /*isSigned=*/true),
 | 
						|
                 /*isUnsigned=*/false);
 | 
						|
    Data = Data.drop_front(1);
 | 
						|
    return std::error_code();
 | 
						|
  case LF_SHORT:
 | 
						|
    if (Data.size() < 2)
 | 
						|
      return std::make_error_code(std::errc::illegal_byte_sequence);
 | 
						|
    Num = APSInt(APInt(/*numBits=*/16,
 | 
						|
                       *reinterpret_cast<const little16_t *>(Data.data()),
 | 
						|
                       /*isSigned=*/true),
 | 
						|
                 /*isUnsigned=*/false);
 | 
						|
    Data = Data.drop_front(2);
 | 
						|
    return std::error_code();
 | 
						|
  case LF_USHORT:
 | 
						|
    if (Data.size() < 2)
 | 
						|
      return std::make_error_code(std::errc::illegal_byte_sequence);
 | 
						|
    Num = APSInt(APInt(/*numBits=*/16,
 | 
						|
                       *reinterpret_cast<const ulittle16_t *>(Data.data()),
 | 
						|
                       /*isSigned=*/false),
 | 
						|
                 /*isUnsigned=*/true);
 | 
						|
    Data = Data.drop_front(2);
 | 
						|
    return std::error_code();
 | 
						|
  case LF_LONG:
 | 
						|
    if (Data.size() < 4)
 | 
						|
      return std::make_error_code(std::errc::illegal_byte_sequence);
 | 
						|
    Num = APSInt(APInt(/*numBits=*/32,
 | 
						|
                       *reinterpret_cast<const little32_t *>(Data.data()),
 | 
						|
                       /*isSigned=*/true),
 | 
						|
                 /*isUnsigned=*/false);
 | 
						|
    Data = Data.drop_front(4);
 | 
						|
    return std::error_code();
 | 
						|
  case LF_ULONG:
 | 
						|
    if (Data.size() < 4)
 | 
						|
      return std::make_error_code(std::errc::illegal_byte_sequence);
 | 
						|
    Num = APSInt(APInt(/*numBits=*/32,
 | 
						|
                       *reinterpret_cast<const ulittle32_t *>(Data.data()),
 | 
						|
                       /*isSigned=*/FalseVal),
 | 
						|
                 /*isUnsigned=*/true);
 | 
						|
    Data = Data.drop_front(4);
 | 
						|
    return std::error_code();
 | 
						|
  case LF_QUADWORD:
 | 
						|
    if (Data.size() < 8)
 | 
						|
      return std::make_error_code(std::errc::illegal_byte_sequence);
 | 
						|
    Num = APSInt(APInt(/*numBits=*/64,
 | 
						|
                       *reinterpret_cast<const little64_t *>(Data.data()),
 | 
						|
                       /*isSigned=*/true),
 | 
						|
                 /*isUnsigned=*/false);
 | 
						|
    Data = Data.drop_front(8);
 | 
						|
    return std::error_code();
 | 
						|
  case LF_UQUADWORD:
 | 
						|
    if (Data.size() < 8)
 | 
						|
      return std::make_error_code(std::errc::illegal_byte_sequence);
 | 
						|
    Num = APSInt(APInt(/*numBits=*/64,
 | 
						|
                       *reinterpret_cast<const ulittle64_t *>(Data.data()),
 | 
						|
                       /*isSigned=*/false),
 | 
						|
                 /*isUnsigned=*/true);
 | 
						|
    Data = Data.drop_front(8);
 | 
						|
    return std::error_code();
 | 
						|
  }
 | 
						|
  return std::make_error_code(std::errc::illegal_byte_sequence);
 | 
						|
}
 | 
						|
 | 
						|
std::error_code llvm::codeview::consume(StringRef &Data, APSInt &Num) {
 | 
						|
  ArrayRef<uint8_t> Bytes(Data.bytes_begin(), Data.bytes_end());
 | 
						|
  auto EC = consume(Bytes, Num);
 | 
						|
  Data = StringRef(reinterpret_cast<const char *>(Bytes.data()), Bytes.size());
 | 
						|
  return EC;
 | 
						|
}
 | 
						|
 | 
						|
/// Decode a numeric leaf value that is known to be a uint64_t.
 | 
						|
std::error_code llvm::codeview::consume_numeric(ArrayRef<uint8_t> &Data,
 | 
						|
                                                uint64_t &Num) {
 | 
						|
  APSInt N;
 | 
						|
  if (auto EC = consume(Data, N))
 | 
						|
    return EC;
 | 
						|
  if (N.isSigned() || !N.isIntN(64))
 | 
						|
    return std::make_error_code(std::errc::illegal_byte_sequence);
 | 
						|
  Num = N.getLimitedValue();
 | 
						|
  return std::error_code();
 | 
						|
}
 | 
						|
 | 
						|
std::error_code llvm::codeview::consume(ArrayRef<uint8_t> &Data,
 | 
						|
                                        uint32_t &Item) {
 | 
						|
  const support::ulittle32_t *IntPtr;
 | 
						|
  if (auto EC = consumeObject(Data, IntPtr))
 | 
						|
    return EC;
 | 
						|
  Item = *IntPtr;
 | 
						|
  return std::error_code();
 | 
						|
}
 | 
						|
 | 
						|
std::error_code llvm::codeview::consume(StringRef &Data, uint32_t &Item) {
 | 
						|
  ArrayRef<uint8_t> Bytes(Data.bytes_begin(), Data.bytes_end());
 | 
						|
  auto EC = consume(Bytes, Item);
 | 
						|
  Data = StringRef(reinterpret_cast<const char *>(Bytes.data()), Bytes.size());
 | 
						|
  return EC;
 | 
						|
}
 | 
						|
 | 
						|
std::error_code llvm::codeview::consume(ArrayRef<uint8_t> &Data,
 | 
						|
                                        int32_t &Item) {
 | 
						|
  const support::little32_t *IntPtr;
 | 
						|
  if (auto EC = consumeObject(Data, IntPtr))
 | 
						|
    return EC;
 | 
						|
  Item = *IntPtr;
 | 
						|
  return std::error_code();
 | 
						|
}
 | 
						|
 | 
						|
std::error_code llvm::codeview::consume(ArrayRef<uint8_t> &Data,
 | 
						|
                                        StringRef &Item) {
 | 
						|
  if (Data.empty())
 | 
						|
    return std::make_error_code(std::errc::illegal_byte_sequence);
 | 
						|
 | 
						|
  StringRef Rest;
 | 
						|
  std::tie(Item, Rest) = getBytesAsCharacters(Data).split('\0');
 | 
						|
  // We expect this to be null terminated.  If it was not, it is an error.
 | 
						|
  if (Data.size() == Item.size())
 | 
						|
    return std::make_error_code(std::errc::illegal_byte_sequence);
 | 
						|
 | 
						|
  Data = ArrayRef<uint8_t>(Rest.bytes_begin(), Rest.bytes_end());
 | 
						|
  return std::error_code();
 | 
						|
}
 |