/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- * vim: set ts=8 sts=2 et sw=2 tw=80: * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #ifndef jit_Bailouts_h #define jit_Bailouts_h #include "mozilla/Assertions.h" // MOZ_ASSERT #include // size_t #include // uint8_t, uint32_t #include "jstypes.h" #include "jit/IonTypes.h" // js::jit::Bailout{Id,Kind}, js::jit::SnapshotOffset #include "jit/Registers.h" // js::jit::MachineState #include "js/TypeDecls.h" // jsbytecode namespace js { class AbstractFramePtr; namespace jit { // [SMDOC] IonMonkey Bailouts // // A "bailout" is the process of recovering a baseline interpreter frame from an // IonFrame. Bailouts are implemented in js::jit::BailoutIonToBaseline, which // has the following callers: // // * js::jit::Bailout - This is used when a guard fails in the Ion code // itself; for example, an LGuardShape fails or an LAddI overflows. See // callers of CodeGenerator::bailoutFrom() for more examples. // // * js::jit::ExceptionHandlerBailout - Ion doesn't implement `catch` or // `finally`. If an exception is thrown and would be caught by an Ion frame, // we bail out instead. // // * js::jit::InvalidationBailout - We returned to Ion code that was // invalidated while it was on the stack. See "OSI" below. Ion code can be // invalidated for several reasons: when GC evicts Ion code to save memory, // for example, or when assumptions baked into the jitted code are // invalidated by the VM. // // (Some stack inspection can be done without bailing out, including GC stack // marking, Error object construction, and Gecko profiler sampling.) // // Consider the first case. When an Ion guard fails, we can't continue in // Ion. There's no IC fallback case coming to save us; we've got a broken // assumption baked into the code we're running. So we jump to an out-of-line // code path that's responsible for abandoning Ion execution and resuming in // the baseline interpreter: the bailout path. // // We were in the midst of optimized Ion code, so bits of program state may be // in registers or spilled to the native stack; values may be unboxed; some // objects may have been optimized away; thanks to inlining, whole call frames // may be missing. The bailout path must put all these pieces back together // into the structure the baseline interpreter expects. // // The data structure that makes this possible is called a *snapshot*. // Snapshots are created during Ion codegen and associated with the IonScript; // they tell how to recover each value in a BaselineFrame from the current // machine state at a given point in the Ion JIT code. This is potentially // different at every place in an Ion script where we might bail out. (See // Snapshots.h.) // // The bailout path performs roughly the following steps: // // 1. Push a snapshot index and the frame size to the native stack. // 2. Spill all registers. // 3. Call js::jit::Bailout to reconstruct the baseline frame(s). // 4. memmove() those to the right place on the native stack. // 5. Jump into the baseline interpreter. // // When C++ code invalidates Ion code, we do on-stack invalidation, or OSI, to // arrange for every affected Ion frame on the stack to bail out as soon as // control returns to it. OSI patches every instruction in the JIT code that's // at a return address currently on the stack. See InvalidateActivation. // // // ## Bailout path implementation details // // Ion code has a lot of guards, so each bailout path must be small. Steps 2 // and 3 above are therefore implemented by a shared per-Runtime trampoline, // rt->jitRuntime()->getGenericBailoutHandler(). // // Naively, we could implement step 1 like: // // _bailout_ID_1: // push 1 // jmp _deopt // _bailout_ID_2: // push 2 // jmp _deopt // ... // _deopt: // push imm(FrameSize) // call _global_bailout_handler // // This takes about 10 extra bytes per guard. On some platforms, we can reduce // this overhead to 4 bytes by creating a global jump table, shared again in // the compartment: // // call _global_bailout_handler // call _global_bailout_handler // call _global_bailout_handler // call _global_bailout_handler // ... // _global_bailout_handler: // // In the bailout handler, we can recompute which entry in the table was // selected by subtracting the return addressed pushed by the call, from the // start of the table, and then dividing by the size of a (call X) entry in the // table. This gives us a number in [0, TableSize), which we call a // "BailoutId". // // Then, we can provide a per-script mapping from BailoutIds to snapshots, // which takes only four bytes per entry. // // This strategy does not work as given, because the bailout handler has no way // to compute the location of an IonScript. Currently, we do not use frame // pointers. To account for this we segregate frames into a limited set of // "frame sizes", and create a table for each frame size. We also have the // option of not using bailout tables, for platforms or situations where the // 10 byte cost is more optimal than a bailout table. See JitFrames.h for more // detail. static const BailoutId INVALID_BAILOUT_ID = BailoutId(-1); // Keep this arbitrarily small for now, for testing. static const uint32_t BAILOUT_TABLE_SIZE = 16; // BailoutStack is an architecture specific pointer to the stack, given by the // bailout handler. class BailoutStack; class InvalidationBailoutStack; class IonScript; class InlineFrameIterator; class JitActivation; class JitActivationIterator; class JSJitFrameIter; struct ResumeFromException; // Must be implemented by each architecture. // This structure is constructed before recovering the baseline frames for a // bailout. It records all information extracted from the stack, and which are // needed for the JSJitFrameIter. class BailoutFrameInfo { MachineState machine_; uint8_t* framePointer_; size_t topFrameSize_; IonScript* topIonScript_; uint32_t snapshotOffset_; JitActivation* activation_; void attachOnJitActivation(const JitActivationIterator& activations); public: BailoutFrameInfo(const JitActivationIterator& activations, BailoutStack* sp); BailoutFrameInfo(const JitActivationIterator& activations, InvalidationBailoutStack* sp); BailoutFrameInfo(const JitActivationIterator& activations, const JSJitFrameIter& frame); ~BailoutFrameInfo(); uint8_t* fp() const { return framePointer_; } SnapshotOffset snapshotOffset() const { return snapshotOffset_; } const MachineState* machineState() const { return &machine_; } size_t topFrameSize() const { return topFrameSize_; } IonScript* ionScript() const { return topIonScript_; } JitActivation* activation() const { return activation_; } }; [[nodiscard]] bool EnsureHasEnvironmentObjects(JSContext* cx, AbstractFramePtr fp); struct BaselineBailoutInfo; // Called from a bailout thunk. [[nodiscard]] bool Bailout(BailoutStack* sp, BaselineBailoutInfo** info); // Called from the invalidation thunk. [[nodiscard]] bool InvalidationBailout(InvalidationBailoutStack* sp, size_t* frameSizeOut, BaselineBailoutInfo** info); class ExceptionBailoutInfo { size_t frameNo_; jsbytecode* resumePC_; size_t numExprSlots_; public: ExceptionBailoutInfo(size_t frameNo, jsbytecode* resumePC, size_t numExprSlots) : frameNo_(frameNo), resumePC_(resumePC), numExprSlots_(numExprSlots) {} ExceptionBailoutInfo() : frameNo_(0), resumePC_(nullptr), numExprSlots_(0) {} bool catchingException() const { return !!resumePC_; } bool propagatingIonExceptionForDebugMode() const { return !resumePC_; } size_t frameNo() const { MOZ_ASSERT(catchingException()); return frameNo_; } jsbytecode* resumePC() const { MOZ_ASSERT(catchingException()); return resumePC_; } size_t numExprSlots() const { MOZ_ASSERT(catchingException()); return numExprSlots_; } }; // Called from the exception handler to enter a catch or finally block. [[nodiscard]] bool ExceptionHandlerBailout(JSContext* cx, const InlineFrameIterator& frame, ResumeFromException* rfe, const ExceptionBailoutInfo& excInfo); [[nodiscard]] bool FinishBailoutToBaseline(BaselineBailoutInfo* bailoutInfoArg); } // namespace jit } // namespace js #endif /* jit_Bailouts_h */