| /* |
| * Copyright (C) 2008-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. |
| */ |
| |
| #include "config.h" |
| #include "CallFrame.h" |
| |
| #include "CodeBlock.h" |
| #include "InlineCallFrame.h" |
| #include "Interpreter.h" |
| #include "JSCInlines.h" |
| #include "JSWebAssemblyInstance.h" |
| #include "VMEntryScope.h" |
| #include "WasmContextInlines.h" |
| #include "WasmInstance.h" |
| #include <wtf/StringPrintStream.h> |
| |
| namespace JSC { |
| |
| void CallFrame::initDeprecatedCallFrameForDebugger(CallFrame* globalExec, JSCallee* globalCallee) |
| { |
| globalExec->setCodeBlock(nullptr); |
| globalExec->setCallerFrame(noCaller()); |
| globalExec->setReturnPC(0); |
| globalExec->setArgumentCountIncludingThis(0); |
| globalExec->setCallee(globalCallee); |
| ASSERT(globalExec->isDeprecatedCallFrameForDebugger()); |
| } |
| |
| bool CallFrame::callSiteBitsAreBytecodeOffset() const |
| { |
| ASSERT(codeBlock()); |
| switch (codeBlock()->jitType()) { |
| case JITType::InterpreterThunk: |
| case JITType::BaselineJIT: |
| return true; |
| case JITType::None: |
| case JITType::HostCallThunk: |
| RELEASE_ASSERT_NOT_REACHED(); |
| return false; |
| default: |
| return false; |
| } |
| |
| RELEASE_ASSERT_NOT_REACHED(); |
| return false; |
| } |
| |
| bool CallFrame::callSiteBitsAreCodeOriginIndex() const |
| { |
| ASSERT(codeBlock()); |
| switch (codeBlock()->jitType()) { |
| case JITType::DFGJIT: |
| case JITType::FTLJIT: |
| return true; |
| case JITType::None: |
| case JITType::HostCallThunk: |
| RELEASE_ASSERT_NOT_REACHED(); |
| return false; |
| default: |
| return false; |
| } |
| |
| RELEASE_ASSERT_NOT_REACHED(); |
| return false; |
| } |
| |
| unsigned CallFrame::callSiteAsRawBits() const |
| { |
| return this[CallFrameSlot::argumentCount].tag(); |
| } |
| |
| SUPPRESS_ASAN unsigned CallFrame::unsafeCallSiteAsRawBits() const |
| { |
| return this[CallFrameSlot::argumentCount].unsafeTag(); |
| } |
| |
| CallSiteIndex CallFrame::callSiteIndex() const |
| { |
| return CallSiteIndex(callSiteAsRawBits()); |
| } |
| |
| SUPPRESS_ASAN CallSiteIndex CallFrame::unsafeCallSiteIndex() const |
| { |
| return CallSiteIndex(unsafeCallSiteAsRawBits()); |
| } |
| |
| #if USE(JSVALUE32_64) |
| const Instruction* CallFrame::currentVPC() const |
| { |
| return bitwise_cast<Instruction*>(callSiteIndex().bits()); |
| } |
| |
| void CallFrame::setCurrentVPC(const Instruction* vpc) |
| { |
| CallSiteIndex callSite(vpc); |
| this[CallFrameSlot::argumentCount].tag() = callSite.bits(); |
| } |
| |
| unsigned CallFrame::callSiteBitsAsBytecodeOffset() const |
| { |
| ASSERT(codeBlock()); |
| ASSERT(callSiteBitsAreBytecodeOffset()); |
| return codeBlock()->bytecodeOffset(currentVPC()); |
| } |
| |
| #else // USE(JSVALUE32_64) |
| const Instruction* CallFrame::currentVPC() const |
| { |
| ASSERT(callSiteBitsAreBytecodeOffset()); |
| return codeBlock()->instructions().at(callSiteBitsAsBytecodeOffset()).ptr(); |
| } |
| |
| void CallFrame::setCurrentVPC(const Instruction* vpc) |
| { |
| CallSiteIndex callSite(codeBlock()->bytecodeOffset(vpc)); |
| this[CallFrameSlot::argumentCount].tag() = static_cast<int32_t>(callSite.bits()); |
| } |
| |
| unsigned CallFrame::callSiteBitsAsBytecodeOffset() const |
| { |
| ASSERT(codeBlock()); |
| ASSERT(callSiteBitsAreBytecodeOffset()); |
| return callSiteIndex().bits(); |
| } |
| |
| #endif |
| |
| unsigned CallFrame::bytecodeOffset() |
| { |
| ASSERT(!callee().isWasm()); |
| if (!codeBlock()) |
| return 0; |
| #if ENABLE(DFG_JIT) |
| if (callSiteBitsAreCodeOriginIndex()) { |
| ASSERT(codeBlock()); |
| CodeOrigin codeOrigin = this->codeOrigin(); |
| for (InlineCallFrame* inlineCallFrame = codeOrigin.inlineCallFrame(); inlineCallFrame;) { |
| codeOrigin = inlineCallFrame->directCaller; |
| inlineCallFrame = codeOrigin.inlineCallFrame(); |
| } |
| return codeOrigin.bytecodeIndex(); |
| } |
| #endif |
| ASSERT(callSiteBitsAreBytecodeOffset()); |
| return callSiteBitsAsBytecodeOffset(); |
| } |
| |
| CodeOrigin CallFrame::codeOrigin() |
| { |
| if (!codeBlock()) |
| return CodeOrigin(0); |
| #if ENABLE(DFG_JIT) |
| if (callSiteBitsAreCodeOriginIndex()) { |
| CallSiteIndex index = callSiteIndex(); |
| ASSERT(codeBlock()->canGetCodeOrigin(index)); |
| return codeBlock()->codeOrigin(index); |
| } |
| #endif |
| return CodeOrigin(callSiteBitsAsBytecodeOffset()); |
| } |
| |
| Register* CallFrame::topOfFrameInternal() |
| { |
| CodeBlock* codeBlock = this->codeBlock(); |
| ASSERT(codeBlock); |
| return registers() + codeBlock->stackPointerOffset(); |
| } |
| |
| JSGlobalObject* CallFrame::wasmAwareLexicalGlobalObject(VM& vm) |
| { |
| #if ENABLE(WEBASSEMBLY) |
| if (!callee().isWasm()) |
| return lexicalGlobalObject(); |
| return vm.wasmContext.load()->owner<JSWebAssemblyInstance>()->globalObject(); |
| #else |
| UNUSED_PARAM(vm); |
| return lexicalGlobalObject(); |
| #endif |
| } |
| |
| bool CallFrame::isAnyWasmCallee() |
| { |
| CalleeBits callee = this->callee(); |
| if (callee.isWasm()) |
| return true; |
| |
| ASSERT(callee.isCell()); |
| if (!!callee.rawPtr() && isWebAssemblyToJSCallee(callee.asCell())) |
| return true; |
| |
| return false; |
| } |
| |
| CallFrame* CallFrame::callerFrame(EntryFrame*& currEntryFrame) const |
| { |
| if (callerFrameOrEntryFrame() == currEntryFrame) { |
| VMEntryRecord* currVMEntryRecord = vmEntryRecord(currEntryFrame); |
| currEntryFrame = currVMEntryRecord->prevTopEntryFrame(); |
| return currVMEntryRecord->prevTopCallFrame(); |
| } |
| return static_cast<CallFrame*>(callerFrameOrEntryFrame()); |
| } |
| |
| SUPPRESS_ASAN CallFrame* CallFrame::unsafeCallerFrame(EntryFrame*& currEntryFrame) const |
| { |
| if (unsafeCallerFrameOrEntryFrame() == currEntryFrame) { |
| VMEntryRecord* currVMEntryRecord = vmEntryRecord(currEntryFrame); |
| currEntryFrame = currVMEntryRecord->unsafePrevTopEntryFrame(); |
| return currVMEntryRecord->unsafePrevTopCallFrame(); |
| } |
| return static_cast<CallFrame*>(unsafeCallerFrameOrEntryFrame()); |
| } |
| |
| SourceOrigin CallFrame::callerSourceOrigin() |
| { |
| RELEASE_ASSERT(callee().isCell()); |
| VM* vm = &this->vm(); |
| SourceOrigin sourceOrigin; |
| bool haveSkippedFirstFrame = false; |
| StackVisitor::visit(this, vm, [&](StackVisitor& visitor) { |
| if (!std::exchange(haveSkippedFirstFrame, true)) |
| return StackVisitor::Status::Continue; |
| |
| switch (visitor->codeType()) { |
| case StackVisitor::Frame::CodeType::Function: |
| // Skip the builtin functions since they should not pass the source origin to the dynamic code generation calls. |
| // Consider the following code. |
| // |
| // [ "42 + 44" ].forEach(eval); |
| // |
| // In the above case, the eval function will be interpreted as the indirect call to eval inside forEach function. |
| // At that time, the generated eval code should have the source origin to the original caller of the forEach function |
| // instead of the source origin of the forEach function. |
| if (static_cast<FunctionExecutable*>(visitor->codeBlock()->ownerExecutable())->isBuiltinFunction()) |
| return StackVisitor::Status::Continue; |
| FALLTHROUGH; |
| |
| case StackVisitor::Frame::CodeType::Eval: |
| case StackVisitor::Frame::CodeType::Module: |
| case StackVisitor::Frame::CodeType::Global: |
| sourceOrigin = visitor->codeBlock()->ownerExecutable()->sourceOrigin(); |
| return StackVisitor::Status::Done; |
| |
| case StackVisitor::Frame::CodeType::Native: |
| return StackVisitor::Status::Continue; |
| |
| case StackVisitor::Frame::CodeType::Wasm: |
| // FIXME: Should return the source origin for WASM. |
| return StackVisitor::Status::Done; |
| } |
| |
| RELEASE_ASSERT_NOT_REACHED(); |
| return StackVisitor::Status::Done; |
| }); |
| return sourceOrigin; |
| } |
| |
| String CallFrame::friendlyFunctionName() |
| { |
| CodeBlock* codeBlock = this->codeBlock(); |
| if (!codeBlock) |
| return emptyString(); |
| |
| switch (codeBlock->codeType()) { |
| case EvalCode: |
| return "eval code"_s; |
| case ModuleCode: |
| return "module code"_s; |
| case GlobalCode: |
| return "global code"_s; |
| case FunctionCode: |
| if (jsCallee()) |
| return getCalculatedDisplayName(vm(), jsCallee()); |
| return emptyString(); |
| } |
| |
| ASSERT_NOT_REACHED(); |
| return emptyString(); |
| } |
| |
| void CallFrame::dump(PrintStream& out) |
| { |
| if (CodeBlock* codeBlock = this->codeBlock()) { |
| out.print(codeBlock->inferredName(), "#", codeBlock->hashAsStringIfPossible(), " [", codeBlock->jitType(), " bc#", bytecodeOffset(), "]"); |
| |
| out.print("("); |
| thisValue().dumpForBacktrace(out); |
| |
| for (size_t i = 0; i < argumentCount(); ++i) { |
| out.print(", "); |
| JSValue value = argument(i); |
| value.dumpForBacktrace(out); |
| } |
| |
| out.print(")"); |
| |
| return; |
| } |
| |
| out.print(returnPC()); |
| } |
| |
| const char* CallFrame::describeFrame() |
| { |
| const size_t bufferSize = 200; |
| static char buffer[bufferSize + 1]; |
| |
| WTF::StringPrintStream stringStream; |
| |
| dump(stringStream); |
| |
| strncpy(buffer, stringStream.toCString().data(), bufferSize); |
| buffer[bufferSize] = '\0'; |
| |
| return buffer; |
| } |
| |
| void CallFrame::convertToStackOverflowFrame(VM& vm, CodeBlock* codeBlockToKeepAliveUntilFrameIsUnwound) |
| { |
| ASSERT(!isDeprecatedCallFrameForDebugger()); |
| ASSERT(codeBlockToKeepAliveUntilFrameIsUnwound->inherits<CodeBlock>(vm)); |
| |
| EntryFrame* entryFrame = vm.topEntryFrame; |
| CallFrame* throwOriginFrame = this; |
| do { |
| throwOriginFrame = throwOriginFrame->callerFrame(entryFrame); |
| } while (throwOriginFrame && throwOriginFrame->callee().isWasm()); |
| |
| JSObject* originCallee = throwOriginFrame ? throwOriginFrame->jsCallee() : vmEntryRecord(vm.topEntryFrame)->callee(); |
| JSObject* stackOverflowCallee = originCallee->globalObject()->stackOverflowFrameCallee(); |
| |
| setCodeBlock(codeBlockToKeepAliveUntilFrameIsUnwound); |
| setCallee(stackOverflowCallee); |
| setArgumentCountIncludingThis(0); |
| } |
| |
| } // namespace JSC |