// 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_