| /* |
| * Copyright (C) 2013 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 "StackIterator.h" |
| |
| #include "Arguments.h" |
| #include "CallFrameInlines.h" |
| #include "Executable.h" |
| #include "Operations.h" |
| #include <wtf/DataLog.h> |
| |
| namespace JSC { |
| |
| StackIterator::StackIterator(CallFrame* frame, StackIterator::FrameFilter filter) |
| : m_filter(filter) |
| { |
| ASSERT(frame); |
| m_frame = Frame::create(frame); |
| m_frame = m_frame->logicalFrame(); |
| } |
| |
| void StackIterator::gotoNextFrame() |
| { |
| Frame* frame = m_frame; |
| while (frame) { |
| frame = frame->logicalCallerFrame(); |
| if (!frame || !m_filter || !m_filter(frame)) |
| break; |
| } |
| m_frame = frame; |
| } |
| |
| void StackIterator::find(JSFunction* functionObj) |
| { |
| ASSERT(functionObj); |
| JSObject* targetCallee = jsDynamicCast<JSObject*>(functionObj); |
| while (m_frame) { |
| if (m_frame->callee() == targetCallee) |
| break; |
| gotoNextFrame(); |
| } |
| } |
| |
| StackIterator::Frame::CodeType StackIterator::Frame::codeType() const |
| { |
| if (!isJSFrame()) |
| return CodeType::Native; |
| |
| switch (codeBlock()->codeType()) { |
| case EvalCode: |
| return CodeType::Eval; |
| case FunctionCode: |
| return CodeType::Function; |
| case GlobalCode: |
| return CodeType::Global; |
| } |
| RELEASE_ASSERT_NOT_REACHED(); |
| return CodeType::Global; |
| } |
| |
| String StackIterator::Frame::functionName() |
| { |
| String traceLine; |
| JSObject* callee = this->callee(); |
| |
| switch (codeType()) { |
| case CodeType::Eval: |
| traceLine = "eval code"; |
| break; |
| case CodeType::Native: |
| if (callee) |
| traceLine = getCalculatedDisplayName(callFrame(), callee).impl(); |
| break; |
| case CodeType::Function: |
| traceLine = getCalculatedDisplayName(callFrame(), callee).impl(); |
| break; |
| case CodeType::Global: |
| traceLine = "global code"; |
| break; |
| } |
| return traceLine.isNull() ? emptyString() : traceLine; |
| } |
| |
| String StackIterator::Frame::sourceURL() |
| { |
| String traceLine; |
| |
| switch (codeType()) { |
| case CodeType::Eval: |
| case CodeType::Function: |
| case CodeType::Global: { |
| String sourceURL = codeBlock()->ownerExecutable()->sourceURL(); |
| if (!sourceURL.isEmpty()) |
| traceLine = sourceURL.impl(); |
| break; |
| } |
| case CodeType::Native: |
| traceLine = "[native code]"; |
| break; |
| } |
| return traceLine.isNull() ? emptyString() : traceLine; |
| } |
| |
| String StackIterator::Frame::toString() |
| { |
| StringBuilder traceBuild; |
| String functionName = this->functionName(); |
| String sourceURL = this->sourceURL(); |
| traceBuild.append(functionName); |
| if (!sourceURL.isEmpty()) { |
| if (!functionName.isEmpty()) |
| traceBuild.append('@'); |
| traceBuild.append(sourceURL); |
| if (isJSFrame()) { |
| unsigned line = 0; |
| unsigned column = 0; |
| computeLineAndColumn(line, column); |
| traceBuild.append(':'); |
| traceBuild.appendNumber(line); |
| traceBuild.append(':'); |
| traceBuild.appendNumber(column); |
| } |
| } |
| return traceBuild.toString().impl(); |
| } |
| |
| unsigned StackIterator::Frame::bytecodeOffset() |
| { |
| if (!isJSFrame()) |
| return 0; |
| if (hasLocationAsCodeOriginIndex()) |
| return bytecodeOffsetFromCodeOriginIndex(); |
| return locationAsBytecodeOffset(); |
| } |
| |
| Arguments* StackIterator::Frame::arguments() |
| { |
| CallFrame* callFrame = this->callFrame(); |
| Arguments* arguments = Arguments::create(vm(), callFrame); |
| arguments->tearOff(callFrame); |
| return arguments; |
| } |
| |
| void StackIterator::Frame::computeLineAndColumn(unsigned& line, unsigned& column) |
| { |
| CodeBlock* codeBlock = this->codeBlock(); |
| if (!codeBlock) { |
| line = 0; |
| column = 0; |
| return; |
| } |
| |
| int divot = 0; |
| int unusedStartOffset = 0; |
| int unusedEndOffset = 0; |
| unsigned divotLine = 0; |
| unsigned divotColumn = 0; |
| retrieveExpressionInfo(divot, unusedStartOffset, unusedEndOffset, divotLine, divotColumn); |
| |
| line = divotLine + codeBlock->ownerExecutable()->lineNo(); |
| column = divotColumn + (divotLine ? 1 : codeBlock->firstLineColumnOffset()); |
| } |
| |
| void StackIterator::Frame::retrieveExpressionInfo(int& divot, int& startOffset, int& endOffset, unsigned& line, unsigned& column) |
| { |
| CodeBlock* codeBlock = this->codeBlock(); |
| codeBlock->unlinkedCodeBlock()->expressionRangeForBytecodeOffset(bytecodeOffset(), divot, startOffset, endOffset, line, column); |
| divot += codeBlock->sourceOffset(); |
| } |
| |
| |
| StackIterator::Frame* StackIterator::Frame::logicalFrame() |
| { |
| #if !ENABLE(DFG_JIT) |
| return this; |
| |
| #else // !ENABLE(DFG_JIT) |
| if (isInlinedFrame()) |
| return this; |
| |
| // If I don't have a code block, then I'm not DFG code, so I'm the true call frame. |
| CodeBlock* codeBlock = this->codeBlock(); |
| if (!codeBlock) |
| return this; |
| |
| // If the code block does not have any code origins, then there was no inlining, so |
| // I'm done. |
| if (!codeBlock->hasCodeOrigins()) |
| return this; |
| |
| CodeBlock* outerMostCodeBlock = codeBlock; |
| unsigned index = locationAsCodeOriginIndex(); |
| ASSERT(outerMostCodeBlock->canGetCodeOrigin(index)); |
| if (!outerMostCodeBlock->canGetCodeOrigin(index)) { |
| // See above. In release builds, we try to protect ourselves from crashing even |
| // though stack walking will be goofed up. |
| return nullptr; |
| } |
| |
| CodeOrigin codeOrigin = outerMostCodeBlock->codeOrigin(index); |
| if (!codeOrigin.inlineCallFrame) |
| return this; // Not currently in inlined code. |
| |
| // We've got inlined frames. So, reify them so that the iterator can walk through them. |
| CallFrame* currFrame = this->callFrame(); |
| CallFrame* innerMostLogicalFrame = currFrame + codeOrigin.inlineCallFrame->stackOffset; |
| |
| CallFrame* logicalFrame = innerMostLogicalFrame; |
| while (logicalFrame != currFrame) { |
| InlineCallFrame* inlinedFrameInfo = codeOrigin.inlineCallFrame; |
| |
| // Fill in the logical (i.e. inlined) frame |
| logicalFrame->setCodeBlock(inlinedFrameInfo->baselineCodeBlock()); |
| logicalFrame->setInlineCallFrame(inlinedFrameInfo); |
| logicalFrame->setArgumentCountIncludingThis(inlinedFrameInfo->arguments.size()); |
| logicalFrame->setLocationAsBytecodeOffset(codeOrigin.bytecodeIndex); |
| logicalFrame->setIsInlinedFrame(); |
| |
| JSFunction* callee = inlinedFrameInfo->callee.get(); |
| if (callee) { |
| logicalFrame->setScope(callee->scope()); |
| logicalFrame->setCallee(callee); |
| } |
| |
| CodeOrigin* callerCodeOrigin = &inlinedFrameInfo->caller; |
| InlineCallFrame* callerInlinedFrameInfo = callerCodeOrigin->inlineCallFrame; |
| unsigned callerFrameOffset = callerInlinedFrameInfo ? callerInlinedFrameInfo->stackOffset : 0; |
| CallFrame* callerFrame = currFrame + callerFrameOffset; |
| logicalFrame->setCallerFrame(callerFrame); |
| |
| codeOrigin = *callerCodeOrigin; |
| logicalFrame = callerFrame; |
| } |
| |
| ASSERT(!innerMostLogicalFrame->hasHostCallFrameFlag()); |
| return Frame::create(innerMostLogicalFrame); |
| #endif // !ENABLE(DFG_JIT) |
| } |
| |
| StackIterator::Frame* StackIterator::Frame::logicalCallerFrame() |
| { |
| Frame* callerFrame = create(this->callerFrame()->removeHostCallFrameFlag()); |
| #if !ENABLE(DFG_JIT) |
| return callerFrame; |
| |
| #else // !ENABLE(DFG_JIT) |
| if (!isJSFrame() || !callerFrame) |
| return callerFrame; |
| |
| // If I am known to be an inlined frame, then I've been reified already and |
| // have my caller. |
| if (isInlinedFrame()) |
| return callerFrame; |
| |
| // I am not an inlined frame. So the question is: is my caller a CallFrame |
| // that has inlines or a CallFrame that doesn't? |
| |
| // If my caller is not a JS frame, it cannot have inlines, and we're done. |
| if (!callerFrame->isJSFrame()) |
| return callerFrame; |
| |
| ASSERT(!callerFrame->isInlinedFrame()); |
| return callerFrame->logicalFrame(); |
| |
| #endif // !ENABLE(DFG_JIT) |
| } |
| |
| #ifndef NDEBUG |
| |
| static const char* jitTypeName(JITCode::JITType jitType) |
| { |
| switch (jitType) { |
| case JITCode::None: return "None"; |
| case JITCode::HostCallThunk: return "HostCallThunk"; |
| case JITCode::InterpreterThunk: return "InterpreterThunk"; |
| case JITCode::BaselineJIT: return "BaselineJIT"; |
| case JITCode::DFGJIT: return "DFGJIT"; |
| case JITCode::FTLJIT: return "FTLJIT"; |
| } |
| return "<unknown>"; |
| } |
| |
| static void printIndents(int levels) |
| { |
| while (levels--) |
| dataLogFString(" "); |
| } |
| |
| static void printif(int indentLevels, const char* format, ...) |
| { |
| va_list argList; |
| va_start(argList, format); |
| |
| if (indentLevels) |
| printIndents(indentLevels); |
| |
| #if COMPILER(CLANG) || (COMPILER(GCC) && GCC_VERSION_AT_LEAST(4, 6, 0)) |
| #pragma GCC diagnostic push |
| #pragma GCC diagnostic ignored "-Wformat-nonliteral" |
| #endif |
| |
| WTF::dataLogFV(format, argList); |
| |
| #if COMPILER(CLANG) || (COMPILER(GCC) && GCC_VERSION_AT_LEAST(4, 6, 0)) |
| #pragma GCC diagnostic pop |
| #endif |
| |
| va_end(argList); |
| } |
| |
| void StackIterator::Frame::print(int indentLevel) |
| { |
| int i = indentLevel; |
| |
| CodeBlock* codeBlock = this->codeBlock(); |
| printif(i, "frame %p {\n", this); |
| |
| CallFrame* callerFrame = this->callerFrame(); |
| void* returnPC = hasReturnPC()? this->returnPC().value() : nullptr; |
| |
| printif(i, " name '%s'\n", functionName().utf8().data()); |
| printif(i, " sourceURL '%s'\n", sourceURL().utf8().data()); |
| printif(i, " hostFlag %d\n", callerFrame->hasHostCallFrameFlag()); |
| printif(i, " isInlinedFrame %d\n", isInlinedFrame()); |
| |
| if (isInlinedFrame()) |
| printif(i, " InlineCallFrame %p\n", this->inlineCallFrame()); |
| |
| printif(i, " callee %p\n", callee()); |
| printif(i, " returnPC %p\n", returnPC); |
| printif(i, " callerFrame %p\n", callerFrame->removeHostCallFrameFlag()); |
| printif(i, " logicalCallerFrame %p\n", logicalCallerFrame()); |
| printif(i, " rawLocationBits %u 0x%x\n", locationAsRawBits(), locationAsRawBits()); |
| printif(i, " codeBlock %p\n", codeBlock); |
| if (codeBlock) { |
| JITCode::JITType jitType = codeBlock->jitType(); |
| if (hasLocationAsBytecodeOffset()) { |
| unsigned bytecodeOffset = locationAsBytecodeOffset(); |
| printif(i, " bytecodeOffset %u %p / %zu\n", bytecodeOffset, reinterpret_cast<void*>(bytecodeOffset), codeBlock->instructions().size()); |
| } else { |
| unsigned codeOriginIndex = locationAsCodeOriginIndex(); |
| printif(i, " codeOriginIdex %u %p / %zu\n", codeOriginIndex, reinterpret_cast<void*>(codeOriginIndex), codeBlock->codeOrigins().size()); |
| } |
| unsigned line = 0; |
| unsigned column = 0; |
| computeLineAndColumn(line, column); |
| printif(i, " line %d\n", line); |
| printif(i, " column %d\n", column); |
| printif(i, " jitType %d <%s> isOptimizingJIT %d\n", jitType, jitTypeName(jitType), JITCode::isOptimizingJIT(jitType)); |
| printif(i, " hasCodeOrigins %d\n", codeBlock->hasCodeOrigins()); |
| if (codeBlock->hasCodeOrigins()) { |
| JITCode* jitCode = codeBlock->jitCode().get(); |
| printif(i, " jitCode %p start %p end %p\n", jitCode, jitCode->start(), jitCode->end()); |
| } |
| } |
| printif(i, "}\n"); |
| } |
| |
| #endif // NDEBUG |
| |
| } // namespace JSC |
| |
| #ifndef NDEBUG |
| // For use in the debugger |
| void debugPrintCallFrame(JSC::CallFrame*); |
| |
| void debugPrintCallFrame(JSC::CallFrame* callFrame) |
| { |
| if (!callFrame) |
| return; |
| JSC::StackIterator::Frame* frame = JSC::StackIterator::Frame::create(callFrame); |
| frame->print(2); |
| } |
| #endif // !NDEBUG |