| /* | 
 |  * Copyright (C) 2013-2018 Apple Inc. All rights reserved. | 
 |  * | 
 |  * Redistribution and use in source and binary forms, with or without | 
 |  * modification, are permitted provided that the following conditions | 
 |  * are met: | 
 |  * 1. Redistributions of source code must retain the above copyright | 
 |  *    notice, this list of conditions and the following disclaimer. | 
 |  * 2. Redistributions in binary form must reproduce the above copyright | 
 |  *    notice, this list of conditions and the following disclaimer in the | 
 |  *    documentation and/or other materials provided with the distribution. | 
 |  * | 
 |  * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY | 
 |  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
 |  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR | 
 |  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR | 
 |  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, | 
 |  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, | 
 |  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR | 
 |  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY | 
 |  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
 |  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
 |  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.  | 
 |  */ | 
 |  | 
 | #pragma once | 
 |  | 
 | #if ENABLE(FTL_JIT) | 
 |  | 
 | #include "B3ValueRep.h" | 
 | #include "CodeOrigin.h" | 
 | #include "DFGExitProfile.h" | 
 | #include "DFGNodeOrigin.h" | 
 | #include "DFGOSRExitBase.h" | 
 | #include "FTLAbbreviatedTypes.h" | 
 | #include "FTLExitTimeObjectMaterialization.h" | 
 | #include "FTLExitValue.h" | 
 | #include "FTLFormattedValue.h" | 
 | #include "FTLOSRExitHandle.h" | 
 | #include "FTLStackmapArgumentList.h" | 
 | #include "HandlerInfo.h" | 
 | #include "MethodOfGettingAValueProfile.h" | 
 | #include "Operands.h" | 
 | #include "Reg.h" | 
 | #include "ValueProfile.h" | 
 | #include "VirtualRegister.h" | 
 | #include <wtf/FixedVector.h> | 
 |  | 
 | namespace JSC { | 
 |  | 
 | class TrackedReferences; | 
 |  | 
 | namespace B3 { | 
 | class StackmapGenerationParams; | 
 | namespace Air { | 
 | struct GenerationContext; | 
 | } // namespace Air | 
 | } // namespace B3 | 
 |  | 
 | namespace DFG { | 
 | struct NodeOrigin; | 
 | } // namespace DFG; | 
 |  | 
 | namespace FTL { | 
 |  | 
 | class State; | 
 | struct OSRExitDescriptorImpl; | 
 | struct OSRExitHandle; | 
 |  | 
 | struct OSRExitDescriptor { | 
 | private: | 
 |     WTF_MAKE_NONCOPYABLE(OSRExitDescriptor); | 
 | public: | 
 |     OSRExitDescriptor( | 
 |         DataFormat profileDataFormat, MethodOfGettingAValueProfile, | 
 |         unsigned numberOfArguments, unsigned numberOfLocals, unsigned numberOfTmps); | 
 |  | 
 |     // The first argument to the exit call may be a value we wish to profile. | 
 |     // If that's the case, the format will be not Invalid and we'll have a | 
 |     // method of getting a value profile. Note that all of the ExitArgument's | 
 |     // are already aware of this possible off-by-one, so there is no need to | 
 |     // correct them. | 
 |     DataFormat m_profileDataFormat; | 
 |     MethodOfGettingAValueProfile m_valueProfile; | 
 |      | 
 |     FixedOperands<ExitValue> m_values; | 
 |     Bag<ExitTimeObjectMaterialization> m_materializations; | 
 |  | 
 |     void validateReferences(const TrackedReferences&); | 
 |  | 
 |     // Call this once we have a place to emit the OSR exit jump and we have data about how the state | 
 |     // should be recovered. This effectively emits code that does the exit, though the code is really a | 
 |     // patchable jump and we emit the real code lazily. The description of how to emit the real code is | 
 |     // up to the OSRExit object, which this creates. Note that it's OK to drop the OSRExitHandle object | 
 |     // on the ground. It contains information that is mostly not useful if you use this API, since after | 
 |     // this call, the OSRExit is simply ready to go. | 
 |     Ref<OSRExitHandle> emitOSRExit( | 
 |         State&, ExitKind, const DFG::NodeOrigin&, CCallHelpers&, const B3::StackmapGenerationParams&, | 
 |         uint32_t dfgNodeIndex, unsigned offset); | 
 |  | 
 |     // In some cases you want an OSRExit to come into existence, but you don't want to emit it right now. | 
 |     // This will emit the OSR exit in a late path. You can't be sure exactly when that will happen, but | 
 |     // you know that it will be done by the time late path emission is done. So, a linker task will | 
 |     // surely happen after that. You can use the OSRExitHandle to retrieve the exit's label. | 
 |     // | 
 |     // This API is meant to be used for things like exception handling, where some patchpoint wants to | 
 |     // have a place to jump to for OSR exit. It doesn't care where that OSR exit is emitted so long as it | 
 |     // eventually gets access to its label. | 
 |     Ref<OSRExitHandle> emitOSRExitLater( | 
 |         State&, ExitKind, const DFG::NodeOrigin&, const B3::StackmapGenerationParams&, | 
 |         uint32_t dfgNodeIndex, unsigned offset); | 
 |  | 
 | private: | 
 |     // This is the low-level interface. It will create a handle representing the desire to emit code for | 
 |     // an OSR exit. You can call OSRExitHandle::emitExitThunk() once you have a place to emit it. Note | 
 |     // that the above two APIs are written in terms of this and OSRExitHandle::emitExitThunk(). | 
 |     Ref<OSRExitHandle> prepareOSRExitHandle( | 
 |         State&, ExitKind, const DFG::NodeOrigin&, const B3::StackmapGenerationParams&, | 
 |         uint32_t dfgNodeIndex, unsigned offset); | 
 | }; | 
 |  | 
 | struct OSRExit : public DFG::OSRExitBase { | 
 |     OSRExit(OSRExitDescriptor*, ExitKind, CodeOrigin, CodeOrigin codeOriginForExitProfile, bool wasHoisted, uint32_t dfgNodeIndex, FixedVector<B3::ValueRep>&&); | 
 |  | 
 |     OSRExitDescriptor* m_descriptor; | 
 |     MacroAssemblerCodeRef<OSRExitPtrTag> m_code; | 
 |     // This tells us where to place a jump. | 
 |     CodeLocationJump<JSInternalPtrTag> m_patchableJump; | 
 |     FixedVector<B3::ValueRep> m_valueReps; | 
 |  | 
 |     CodeLocationJump<JSInternalPtrTag> codeLocationForRepatch(CodeBlock* ftlCodeBlock) const; | 
 |     void considerAddingAsFrequentExitSite(CodeBlock* profiledCodeBlock) | 
 |     { | 
 |         OSRExitBase::considerAddingAsFrequentExitSite(profiledCodeBlock, ExitFromFTL); | 
 |     } | 
 | }; | 
 |  | 
 | } } // namespace JSC::FTL | 
 |  | 
 | #endif // ENABLE(FTL_JIT) |