blob: bb32f49e5d8966bafc079d274ad9600f75f85759 [file] [log] [blame]
// Copyright 2019 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_UNWINDING_INFO_WIN64_H_
#define V8_DIAGNOSTICS_UNWINDING_INFO_WIN64_H_
#include <vector>
#include "include/v8-callbacks.h"
#include "include/v8config.h"
#include "src/common/globals.h"
#if defined(V8_OS_WIN64)
#include "src/base/win32-headers.h"
namespace v8 {
namespace internal {
namespace win64_unwindinfo {
#define CRASH_HANDLER_FUNCTION_NAME CrashForExceptionInNonABICompliantCodeRange
#define CRASH_HANDLER_FUNCTION_NAME_STRING \
"CrashForExceptionInNonABICompliantCodeRange"
static const int kOSPageSize = 4096;
/**
* Returns true if V8 is configured to emit unwinding data for embedded in the
* pdata/xdata sections of the executable. Currently, this happens when V8 is
* built with "v8_win64_unwinding_info = true".
*/
bool CanEmitUnwindInfoForBuiltins();
/**
* Returns true if V8 if we can register unwinding data for the whole code range
* of an isolate or Wasm module. The first page of the code range is reserved
* and writable, to be used to store unwind data, as documented in:
* https://docs.microsoft.com/en-us/cpp/build/exception-handling-x64.
* In jitless mode V8 does not allocate any executable memory itself so the only
* non-abi-compliant code range is in the embedded blob.
*/
bool CanRegisterUnwindInfoForNonABICompliantCodeRange();
/**
* Registers a custom exception handler for exceptions in V8-generated code.
*/
void SetUnhandledExceptionCallback(
v8::UnhandledExceptionCallback unhandled_exception_callback);
void RegisterNonABICompliantCodeRange(void* start, size_t size_in_bytes);
void UnregisterNonABICompliantCodeRange(void* start);
/**
* Default count of RUNTIME_FUNCTION needed. For Windows X64, 1 RUNTIME_FUNCTION
* covers 4GB range which is sufficient to cover the whole code range of an
* isolate or Wasm module. For Windows ARM64, 1 RUNTIME_FUNCTION covers
* kMaxFunctionLength bytes so multiple RUNTIME_FUNCTION structs could be needed
* to cover the whole code range of an isolate or Wasm module. The extra
* RUNTIME_FUNCTIONs are assumed following the first one in the reserved page.
*/
static const uint32_t kDefaultRuntimeFunctionCount = 1;
#if defined(V8_OS_WIN_X64)
static const int kPushRbpInstructionLength = 1;
static const int kMovRbpRspInstructionLength = 3;
static const int kRbpPrefixCodes = 2;
static const int kRbpPrefixLength =
kPushRbpInstructionLength + kMovRbpRspInstructionLength;
/**
* Returns a vector of bytes that contains the Win X64 unwind data used for all
* V8 builtin functions.
*/
std::vector<uint8_t> GetUnwindInfoForBuiltinFunctions();
class BuiltinUnwindInfo {
public:
BuiltinUnwindInfo() : is_leaf_function_(true) {}
explicit BuiltinUnwindInfo(const std::vector<int>& fp_offsets)
: is_leaf_function_(false), fp_offsets_(fp_offsets) {}
bool is_leaf_function() const { return is_leaf_function_; }
const std::vector<int>& fp_offsets() const { return fp_offsets_; }
private:
bool is_leaf_function_;
std::vector<int> fp_offsets_;
};
class XdataEncoder {
public:
explicit XdataEncoder(const Assembler& assembler)
: assembler_(assembler), current_frame_code_offset_(-1) {}
void onPushRbp();
void onMovRbpRsp();
BuiltinUnwindInfo unwinding_info() const {
return BuiltinUnwindInfo(fp_offsets_);
}
private:
const Assembler& assembler_;
std::vector<int> fp_offsets_;
int current_frame_code_offset_;
};
#elif defined(V8_OS_WIN_ARM64)
/**
* Base on below doc, unwind record has 18 bits (unsigned) to encode function
* length, besides 2 LSB which are always 0.
* https://docs.microsoft.com/en-us/cpp/build/arm64-exception-handling#xdata-records
*/
static const int kMaxFunctionLength = ((1 << 18) - 1) << 2;
struct FrameOffsets {
FrameOffsets();
bool IsDefault() const;
int fp_to_saved_caller_fp;
int fp_to_caller_sp;
};
/**
* Returns a vector of bytes that contains the Win ARM64 unwind data used for
* all V8 builtin functions.
*
* func_len: length in bytes of current function/region to unwind.
* fp_adjustment: offset of the saved caller's fp based on fp in current frame.
* this is necessary to encode unwind data for Windows stack
* unwinder to find correct caller's fp.
*/
std::vector<uint8_t> GetUnwindInfoForBuiltinFunction(
uint32_t func_len, FrameOffsets fp_adjustment);
class BuiltinUnwindInfo {
public:
BuiltinUnwindInfo() : is_leaf_function_(true) {}
explicit BuiltinUnwindInfo(const std::vector<int>& fp_offsets,
const std::vector<FrameOffsets>& fp_adjustments)
: is_leaf_function_(false),
fp_offsets_(fp_offsets),
fp_adjustments_(fp_adjustments) {}
const std::vector<FrameOffsets>& fp_adjustments() const {
return fp_adjustments_;
}
bool is_leaf_function() const { return is_leaf_function_; }
const std::vector<int>& fp_offsets() const { return fp_offsets_; }
private:
bool is_leaf_function_;
std::vector<int> fp_offsets_;
std::vector<FrameOffsets> fp_adjustments_;
};
class XdataEncoder {
public:
explicit XdataEncoder(const Assembler& assembler)
: assembler_(assembler), current_frame_code_offset_(-1) {}
void onSaveFpLr();
void onFramePointerAdjustment(int fp_to_saved_caller_fp, int fp_to_caller_sp);
BuiltinUnwindInfo unwinding_info() const {
return BuiltinUnwindInfo(fp_offsets_, fp_adjustments_);
}
private:
const Assembler& assembler_;
std::vector<int> fp_offsets_;
int current_frame_code_offset_;
FrameOffsets current_frame_adjustment_;
std::vector<FrameOffsets> fp_adjustments_;
};
#endif
} // namespace win64_unwindinfo
} // namespace internal
} // namespace v8
#endif // V8_OS_WIN64
#endif // V8_DIAGNOSTICS_UNWINDING_INFO_WIN64_H_