blob: a9d76a27436d76df1c39f76a3a9301b03f607d2f [file] [log] [blame]
// Copyright 2016 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef V8_DIAGNOSTICS_EH_FRAME_H_
#define V8_DIAGNOSTICS_EH_FRAME_H_
#include "src/base/compiler-specific.h"
#include "src/base/memory.h"
#include "src/codegen/register-arch.h"
#include "src/common/globals.h"
#include "src/zone/zone-containers.h"
namespace v8 {
namespace internal {
class CodeDesc;
class V8_EXPORT_PRIVATE EhFrameConstants final
: public NON_EXPORTED_BASE(AllStatic) {
public:
enum class DwarfOpcodes : byte {
kNop = 0x00,
kAdvanceLoc1 = 0x02,
kAdvanceLoc2 = 0x03,
kAdvanceLoc4 = 0x04,
kSameValue = 0x08,
kDefCfa = 0x0c,
kDefCfaRegister = 0x0d,
kDefCfaOffset = 0x0e,
kOffsetExtendedSf = 0x11,
};
enum DwarfEncodingSpecifiers : byte {
kUData4 = 0x03,
kSData4 = 0x0b,
kPcRel = 0x10,
kDataRel = 0x30,
kOmit = 0xff,
};
static const int kLocationTag = 1;
static const int kLocationMask = 0x3f;
static const int kLocationMaskSize = 6;
static const int kSavedRegisterTag = 2;
static const int kSavedRegisterMask = 0x3f;
static const int kSavedRegisterMaskSize = 6;
static const int kFollowInitialRuleTag = 3;
static const int kFollowInitialRuleMask = 0x3f;
static const int kFollowInitialRuleMaskSize = 6;
static const int kProcedureAddressOffsetInFde = 2 * kInt32Size;
static const int kProcedureSizeOffsetInFde = 3 * kInt32Size;
static const int kInitialStateOffsetInCie = 19;
static const int kEhFrameTerminatorSize = 4;
// Defined in eh-writer-<arch>.cc
static const int kCodeAlignmentFactor;
static const int kDataAlignmentFactor;
static const int kFdeVersionSize = 1;
static const int kFdeEncodingSpecifiersSize = 3;
static const int kEhFrameHdrVersion = 1;
static const int kEhFrameHdrSize = 20;
};
class V8_EXPORT_PRIVATE EhFrameWriter {
public:
explicit EhFrameWriter(Zone* zone);
// The empty frame is a hack to trigger fp-based unwinding in Linux perf
// compiled with libunwind support when processing DWARF-based call graphs.
//
// It is effectively a valid eh_frame_hdr with an empty look up table.
//
static void WriteEmptyEhFrame(std::ostream& stream); // NOLINT
// Write the CIE and FDE header. Call it before any other method.
void Initialize();
void AdvanceLocation(int pc_offset);
// The <base_address> is the one to which all <offset>s in SaveRegisterToStack
// directives are relative. It is given by <base_register> + <base_offset>.
//
// The <base_offset> must be positive or 0.
//
void SetBaseAddressRegister(Register base_register);
void SetBaseAddressOffset(int base_offset);
void IncreaseBaseAddressOffset(int base_delta) {
SetBaseAddressOffset(base_offset_ + base_delta);
}
void SetBaseAddressRegisterAndOffset(Register base_register, int base_offset);
// Register saved at location <base_address> + <offset>.
// The <offset> must be a multiple of EhFrameConstants::kDataAlignment.
void RecordRegisterSavedToStack(Register name, int offset) {
RecordRegisterSavedToStack(RegisterToDwarfCode(name), offset);
}
// The register has not been modified from the previous frame.
void RecordRegisterNotModified(Register name);
// The register follows the rule defined in the CIE.
void RecordRegisterFollowsInitialRule(Register name);
void Finish(int code_size);
// Remember to call Finish() before GetEhFrame().
//
// The EhFrameWriter instance owns the buffer pointed by
// CodeDesc::unwinding_info, and must outlive any use of the CodeDesc.
//
void GetEhFrame(CodeDesc* desc);
int last_pc_offset() const { return last_pc_offset_; }
Register base_register() const { return base_register_; }
int base_offset() const { return base_offset_; }
private:
enum class InternalState { kUndefined, kInitialized, kFinalized };
static const uint32_t kInt32Placeholder = 0xdeadc0de;
void WriteSLeb128(int32_t value);
void WriteULeb128(uint32_t value);
void WriteByte(byte value) { eh_frame_buffer_.push_back(value); }
void WriteOpcode(EhFrameConstants::DwarfOpcodes opcode) {
WriteByte(static_cast<byte>(opcode));
}
void WriteBytes(const byte* start, int size) {
eh_frame_buffer_.insert(eh_frame_buffer_.end(), start, start + size);
}
void WriteInt16(uint16_t value) {
WriteBytes(reinterpret_cast<const byte*>(&value), sizeof(value));
}
void WriteInt32(uint32_t value) {
WriteBytes(reinterpret_cast<const byte*>(&value), sizeof(value));
}
void PatchInt32(int base_offset, uint32_t value) {
DCHECK_EQ(
base::ReadUnalignedValue<uint32_t>(
reinterpret_cast<Address>(eh_frame_buffer_.data()) + base_offset),
kInt32Placeholder);
DCHECK_LT(base_offset + kInt32Size, eh_frame_offset());
base::WriteUnalignedValue<uint32_t>(
reinterpret_cast<Address>(eh_frame_buffer_.data()) + base_offset,
value);
}
// Write the common information entry, which includes encoding specifiers,
// alignment factors, the return address (pseudo) register code and the
// directives to construct the initial state of the unwinding table.
void WriteCie();
// Write the header of the function data entry, containing a pointer to the
// correspondent CIE and the position and size of the associated routine.
void WriteFdeHeader();
// Write the contents of the .eh_frame_hdr section, including encoding
// specifiers and the routine => FDE lookup table.
void WriteEhFrameHdr(int code_size);
// Write nops until the size reaches a multiple of 8 bytes.
void WritePaddingToAlignedSize(int unpadded_size);
// Internal version that directly accepts a DWARF register code, needed for
// handling pseudo-registers on some platforms.
void RecordRegisterSavedToStack(int register_code, int offset);
int GetProcedureAddressOffset() const {
return fde_offset() + EhFrameConstants::kProcedureAddressOffsetInFde;
}
int GetProcedureSizeOffset() const {
return fde_offset() + EhFrameConstants::kProcedureSizeOffsetInFde;
}
int eh_frame_offset() const {
return static_cast<int>(eh_frame_buffer_.size());
}
int fde_offset() const { return cie_size_; }
// Platform specific functions implemented in eh-frame-<arch>.cc
static int RegisterToDwarfCode(Register name);
// Write directives to build the initial state in the CIE.
void WriteInitialStateInCie();
// Write the return address (pseudo) register code.
void WriteReturnAddressRegisterCode();
int cie_size_;
int last_pc_offset_;
InternalState writer_state_;
Register base_register_;
int base_offset_;
ZoneVector<byte> eh_frame_buffer_;
DISALLOW_COPY_AND_ASSIGN(EhFrameWriter);
};
class V8_EXPORT_PRIVATE EhFrameIterator {
public:
EhFrameIterator(const byte* start, const byte* end)
: start_(start), next_(start), end_(end) {
DCHECK_LE(start, end);
}
void SkipCie() {
DCHECK_EQ(next_, start_);
next_ +=
base::ReadUnalignedValue<uint32_t>(reinterpret_cast<Address>(next_)) +
kInt32Size;
}
void SkipToFdeDirectives() {
SkipCie();
// Skip the FDE header.
Skip(kDirectivesOffsetInFde);
}
void Skip(int how_many) {
DCHECK_GE(how_many, 0);
next_ += how_many;
DCHECK_LE(next_, end_);
}
uint32_t GetNextUInt32() { return GetNextValue<uint32_t>(); }
uint16_t GetNextUInt16() { return GetNextValue<uint16_t>(); }
byte GetNextByte() { return GetNextValue<byte>(); }
EhFrameConstants::DwarfOpcodes GetNextOpcode() {
return static_cast<EhFrameConstants::DwarfOpcodes>(GetNextByte());
}
uint32_t GetNextULeb128();
int32_t GetNextSLeb128();
bool Done() const {
DCHECK_LE(next_, end_);
return next_ == end_;
}
int GetCurrentOffset() const {
DCHECK_GE(next_, start_);
return static_cast<int>(next_ - start_);
}
int GetBufferSize() { return static_cast<int>(end_ - start_); }
const void* current_address() const {
return reinterpret_cast<const void*>(next_);
}
private:
static const int kDirectivesOffsetInFde = 4 * kInt32Size + 1;
static uint32_t DecodeULeb128(const byte* encoded, int* encoded_size);
static int32_t DecodeSLeb128(const byte* encoded, int* encoded_size);
template <typename T>
T GetNextValue() {
T result;
DCHECK_LE(next_ + sizeof(result), end_);
result = base::ReadUnalignedValue<T>(reinterpret_cast<Address>(next_));
next_ += sizeof(result);
return result;
}
const byte* start_;
const byte* next_;
const byte* end_;
};
#ifdef ENABLE_DISASSEMBLER
class EhFrameDisassembler final {
public:
EhFrameDisassembler(const byte* start, const byte* end)
: start_(start), end_(end) {
DCHECK_LT(start, end);
}
void DisassembleToStream(std::ostream& stream); // NOLINT
private:
static void DumpDwarfDirectives(std::ostream& stream, // NOLINT
const byte* start, const byte* end);
static const char* DwarfRegisterCodeToString(int code);
const byte* start_;
const byte* end_;
DISALLOW_COPY_AND_ASSIGN(EhFrameDisassembler);
};
#endif
} // namespace internal
} // namespace v8
#endif // V8_DIAGNOSTICS_EH_FRAME_H_