/* * Copyright (C) 2008, 2009, 2010, 2012 Apple Inc. All rights reserved. * Copyright (C) 2008 Cameron Zwarich * * 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. * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of * its contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "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 OR ITS 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 "Interpreter.h" #include "Arguments.h" #include "BatchedTransitionOptimizer.h" #include "CallFrame.h" #include "CallFrameClosure.h" #include "CodeBlock.h" #include "Heap.h" #include "Debugger.h" #include "DebuggerCallFrame.h" #include "ErrorInstance.h" #include "EvalCodeCache.h" #include "ExceptionHelpers.h" #include "GetterSetter.h" #include "JSActivation.h" #include "JSArray.h" #include "JSBoundFunction.h" #include "JSNameScope.h" #include "JSNotAnObject.h" #include "JSPropertyNameIterator.h" #include "JSStackInlines.h" #include "JSString.h" #include "JSWithScope.h" #include "LLIntCLoop.h" #include "LiteralParser.h" #include "NameInstance.h" #include "ObjectPrototype.h" #include "Operations.h" #include "Parser.h" #include "Profiler.h" #include "RegExpObject.h" #include "RegExpPrototype.h" #include "Register.h" #include "SamplingTool.h" #include "StrictEvalActivation.h" #include "StrongInlines.h" #include #include #include #include #include #include #if ENABLE(JIT) #include "JIT.h" #endif #define WTF_USE_GCC_COMPUTED_GOTO_WORKAROUND (ENABLE(LLINT) && !defined(__llvm__)) using namespace std; namespace JSC { Interpreter::ErrorHandlingMode::ErrorHandlingMode(ExecState *exec) : m_interpreter(*exec->interpreter()) { if (!m_interpreter.m_errorHandlingModeReentry) m_interpreter.stack().enableErrorStackReserve(); m_interpreter.m_errorHandlingModeReentry++; } Interpreter::ErrorHandlingMode::~ErrorHandlingMode() { m_interpreter.m_errorHandlingModeReentry--; ASSERT(m_interpreter.m_errorHandlingModeReentry >= 0); if (!m_interpreter.m_errorHandlingModeReentry) m_interpreter.stack().disableErrorStackReserve(); } // The Interpreter::StackPolicy class is used to compute a stack capacity // requirement to ensure that we have enough room on the native stack for: // 1. the max cumulative stack used by the interpreter and all code // paths sub of it up till leaf functions. // 2. the max cumulative stack used by the interpreter before it reaches // the next checkpoint (execute...() function) in the interpreter. // // The interpreter can be run on different threads and hence, different // native stacks (with different sizes) before exiting out of the first // frame. Hence, the required capacity needs to be re-computed on every // entry into the interpreter. // // Currently the requiredStack is computed based on a policy. See comments // in StackPolicy::StackPolicy() for details. Interpreter::StackPolicy::StackPolicy(Interpreter& interpreter, const StackBounds& stack) : m_interpreter(interpreter) { const size_t size = stack.size(); const size_t DEFAULT_REQUIRED_STACK = 1024 * 1024; const size_t DEFAULT_MINIMUM_USEABLE_STACK = 128 * 1024; const size_t DEFAULT_ERROR_MODE_REQUIRED_STACK = 32 * 1024; // Here's the policy in a nutshell: // // 1. If we have a large stack, let JS use as much stack as possible // but require that we have at least DEFAULT_REQUIRED_STACK capacity // remaining on the stack: // // stack grows this way --> // --------------------------------------------------------- // | ... | <-- DEFAULT_REQUIRED_STACK --> | ... // --------------------------------------------------------- // ^ ^ // start current sp // // 2. In event that we're re-entering the interpreter to handle // exceptions (in error mode), we'll be a little more generous and // require less stack capacity for the interpreter to be re-entered. // // This is needed because we may have just detected an eminent stack // overflow based on the normally computed required stack capacity. // However, the normal required capacity far exceeds what is needed // for exception handling work. Hence, in error mode, we only require // DEFAULT_ERROR_MODE_REQUIRED_STACK capacity. // // stack grows this way --> // ----------------------------------------------------------------- // | ... | <-- DEFAULT_ERROR_MODE_REQUIRED_STACK --> | ... // ----------------------------------------------------------------- // ^ ^ // start current sp // // This smaller required capacity also means that we won't re-trigger // a stack overflow for processing the exception caused by the original // StackOverflowError. // // 3. If the stack is not large enough, give JS at least a minimum // amount of useable stack: // // stack grows this way --> // -------------------------------------------------------------------- // | <-- DEFAULT_MINIMUM_USEABLE_STACK --> | <-- requiredCapacity --> | // -------------------------------------------------------------------- // ^ ^ // start current sp // // The minimum useable capacity is DEFAULT_MINIMUM_USEABLE_STACK. // In this case, the requiredCapacity is whatever is left of the // total stack capacity after we have give JS its minimum stack // i.e. requiredCapacity can even be 0 if there's not enough stack. // Policy 1: Normal mode: required = DEFAULT_REQUIRED_STACK. // Policy 2: Error mode: required = DEFAULT_ERROR_MODE_REQUIRED_STACK. size_t requiredCapacity = !m_interpreter.m_errorHandlingModeReentry ? DEFAULT_REQUIRED_STACK : DEFAULT_ERROR_MODE_REQUIRED_STACK; size_t useableStack = (requiredCapacity <= size) ? size - requiredCapacity : DEFAULT_MINIMUM_USEABLE_STACK; // Policy 3: Ensure the useable stack is not too small: if (useableStack < DEFAULT_MINIMUM_USEABLE_STACK) useableStack = DEFAULT_MINIMUM_USEABLE_STACK; // Sanity check: Make sure we do not use more space than the stack's // total capacity: if (useableStack > size) useableStack = size; // Re-compute the requiredCapacity based on the adjusted useable stack // size: requiredCapacity = size - useableStack; ASSERT(requiredCapacity < size); m_requiredCapacity = requiredCapacity; } static CallFrame* getCallerInfo(JSGlobalData*, CallFrame*, int& lineNumber, unsigned& bytecodeOffset); // Returns the depth of the scope chain within a given call frame. static int depth(CodeBlock* codeBlock, JSScope* sc) { if (!codeBlock->needsFullScopeChain()) return 0; return sc->localDepth(); } JSValue eval(CallFrame* callFrame) { if (!callFrame->argumentCount()) return jsUndefined(); JSValue program = callFrame->argument(0); if (!program.isString()) return program; TopCallFrameSetter topCallFrame(callFrame->globalData(), callFrame); String programSource = asString(program)->value(callFrame); if (callFrame->hadException()) return JSValue(); CallFrame* callerFrame = callFrame->callerFrame(); CodeBlock* callerCodeBlock = callerFrame->codeBlock(); JSScope* callerScopeChain = callerFrame->scope(); EvalExecutable* eval = callerCodeBlock->evalCodeCache().tryGet(callerCodeBlock->isStrictMode(), programSource, callerScopeChain); if (!eval) { if (!callerCodeBlock->isStrictMode()) { // FIXME: We can use the preparser in strict mode, we just need additional logic // to prevent duplicates. if (programSource.is8Bit()) { LiteralParser preparser(callFrame, programSource.characters8(), programSource.length(), NonStrictJSON); if (JSValue parsedObject = preparser.tryLiteralParse()) return parsedObject; } else { LiteralParser preparser(callFrame, programSource.characters16(), programSource.length(), NonStrictJSON); if (JSValue parsedObject = preparser.tryLiteralParse()) return parsedObject; } } // If the literal parser bailed, it should not have thrown exceptions. ASSERT(!callFrame->globalData().exception); JSValue exceptionValue; eval = callerCodeBlock->evalCodeCache().getSlow(callFrame, callerCodeBlock->ownerExecutable(), callerCodeBlock->isStrictMode(), programSource, callerScopeChain, exceptionValue); ASSERT(!eval == exceptionValue); if (UNLIKELY(!eval)) return throwError(callFrame, exceptionValue); } JSValue thisValue = callerFrame->thisValue(); ASSERT(isValidThisObject(thisValue, callFrame)); Interpreter* interpreter = callFrame->globalData().interpreter; return interpreter->execute(eval, callFrame, thisValue, callerScopeChain); } CallFrame* loadVarargs(CallFrame* callFrame, JSStack* stack, JSValue thisValue, JSValue arguments, int firstFreeRegister) { if (!arguments) { // f.apply(x, arguments), with arguments unmodified. unsigned argumentCountIncludingThis = callFrame->argumentCountIncludingThis(); CallFrame* newCallFrame = CallFrame::create(callFrame->registers() + firstFreeRegister + argumentCountIncludingThis + JSStack::CallFrameHeaderSize); if (argumentCountIncludingThis > Arguments::MaxArguments + 1 || !stack->grow(newCallFrame->registers())) { callFrame->globalData().exception = createStackOverflowError(callFrame); return 0; } newCallFrame->setArgumentCountIncludingThis(argumentCountIncludingThis); newCallFrame->setThisValue(thisValue); for (size_t i = 0; i < callFrame->argumentCount(); ++i) newCallFrame->setArgument(i, callFrame->argumentAfterCapture(i)); return newCallFrame; } if (arguments.isUndefinedOrNull()) { CallFrame* newCallFrame = CallFrame::create(callFrame->registers() + firstFreeRegister + 1 + JSStack::CallFrameHeaderSize); if (!stack->grow(newCallFrame->registers())) { callFrame->globalData().exception = createStackOverflowError(callFrame); return 0; } newCallFrame->setArgumentCountIncludingThis(1); newCallFrame->setThisValue(thisValue); return newCallFrame; } if (!arguments.isObject()) { callFrame->globalData().exception = createInvalidParamError(callFrame, "Function.prototype.apply", arguments); return 0; } if (asObject(arguments)->classInfo() == &Arguments::s_info) { Arguments* argsObject = asArguments(arguments); unsigned argCount = argsObject->length(callFrame); CallFrame* newCallFrame = CallFrame::create(callFrame->registers() + firstFreeRegister + CallFrame::offsetFor(argCount + 1)); if (argCount > Arguments::MaxArguments || !stack->grow(newCallFrame->registers())) { callFrame->globalData().exception = createStackOverflowError(callFrame); return 0; } newCallFrame->setArgumentCountIncludingThis(argCount + 1); newCallFrame->setThisValue(thisValue); argsObject->copyToArguments(callFrame, newCallFrame, argCount); return newCallFrame; } if (isJSArray(arguments)) { JSArray* array = asArray(arguments); unsigned argCount = array->length(); CallFrame* newCallFrame = CallFrame::create(callFrame->registers() + firstFreeRegister + CallFrame::offsetFor(argCount + 1)); if (argCount > Arguments::MaxArguments || !stack->grow(newCallFrame->registers())) { callFrame->globalData().exception = createStackOverflowError(callFrame); return 0; } newCallFrame->setArgumentCountIncludingThis(argCount + 1); newCallFrame->setThisValue(thisValue); array->copyToArguments(callFrame, newCallFrame, argCount); return newCallFrame; } JSObject* argObject = asObject(arguments); unsigned argCount = argObject->get(callFrame, callFrame->propertyNames().length).toUInt32(callFrame); CallFrame* newCallFrame = CallFrame::create(callFrame->registers() + firstFreeRegister + CallFrame::offsetFor(argCount + 1)); if (argCount > Arguments::MaxArguments || !stack->grow(newCallFrame->registers())) { callFrame->globalData().exception = createStackOverflowError(callFrame); return 0; } newCallFrame->setArgumentCountIncludingThis(argCount + 1); newCallFrame->setThisValue(thisValue); for (size_t i = 0; i < argCount; ++i) { newCallFrame->setArgument(i, asObject(arguments)->get(callFrame, i)); if (UNLIKELY(callFrame->globalData().exception)) return 0; } return newCallFrame; } Interpreter::Interpreter(JSGlobalData& globalData) : m_sampleEntryDepth(0) , m_stack(globalData) , m_errorHandlingModeReentry(0) #if !ASSERT_DISABLED , m_initialized(false) #endif { } Interpreter::~Interpreter() { } void Interpreter::initialize(bool canUseJIT) { UNUSED_PARAM(canUseJIT); #if ENABLE(COMPUTED_GOTO_OPCODES) && ENABLE(LLINT) m_opcodeTable = LLInt::opcodeMap(); for (int i = 0; i < numOpcodeIDs; ++i) m_opcodeIDTable.add(m_opcodeTable[i], static_cast(i)); #endif #if !ASSERT_DISABLED m_initialized = true; #endif #if ENABLE(OPCODE_SAMPLING) enableSampler(); #endif } #ifdef NDEBUG void Interpreter::dumpCallFrame(CallFrame*) { } #else void Interpreter::dumpCallFrame(CallFrame* callFrame) { callFrame->codeBlock()->dumpBytecode(); dumpRegisters(callFrame); } void Interpreter::dumpRegisters(CallFrame* callFrame) { dataLogF("Register frame: \n\n"); dataLogF("-----------------------------------------------------------------------------\n"); dataLogF(" use | address | value \n"); dataLogF("-----------------------------------------------------------------------------\n"); CodeBlock* codeBlock = callFrame->codeBlock(); const Register* it; const Register* end; it = callFrame->registers() - JSStack::CallFrameHeaderSize - callFrame->argumentCountIncludingThis(); end = callFrame->registers() - JSStack::CallFrameHeaderSize; while (it < end) { JSValue v = it->jsValue(); int registerNumber = it - callFrame->registers(); String name = codeBlock->nameForRegister(registerNumber); dataLogF("[r% 3d %14s] | %10p | %-16s 0x%lld \n", registerNumber, name.ascii().data(), it, v.description(), (long long)JSValue::encode(v)); it++; } dataLogF("-----------------------------------------------------------------------------\n"); dataLogF("[ArgumentCount] | %10p | %lu \n", it, (unsigned long) callFrame->argumentCount()); ++it; dataLogF("[CallerFrame] | %10p | %p \n", it, callFrame->callerFrame()); ++it; dataLogF("[Callee] | %10p | %p \n", it, callFrame->callee()); ++it; dataLogF("[ScopeChain] | %10p | %p \n", it, callFrame->scope()); ++it; #if ENABLE(JIT) AbstractPC pc = callFrame->abstractReturnPC(callFrame->globalData()); if (pc.hasJITReturnAddress()) dataLogF("[ReturnJITPC] | %10p | %p \n", it, pc.jitReturnAddress().value()); #endif unsigned bytecodeOffset = 0; int line = 0; getCallerInfo(&callFrame->globalData(), callFrame, line, bytecodeOffset); dataLogF("[ReturnVPC] | %10p | %d (line %d)\n", it, bytecodeOffset, line); ++it; dataLogF("[CodeBlock] | %10p | %p \n", it, callFrame->codeBlock()); ++it; dataLogF("-----------------------------------------------------------------------------\n"); int registerCount = 0; end = it + codeBlock->m_numVars; if (it != end) { do { JSValue v = it->jsValue(); int registerNumber = it - callFrame->registers(); String name = codeBlock->nameForRegister(registerNumber); dataLogF("[r% 3d %14s] | %10p | %-16s 0x%lld \n", registerNumber, name.ascii().data(), it, v.description(), (long long)JSValue::encode(v)); ++it; ++registerCount; } while (it != end); } dataLogF("-----------------------------------------------------------------------------\n"); end = it + codeBlock->m_numCalleeRegisters - codeBlock->m_numVars; if (it != end) { do { JSValue v = (*it).jsValue(); dataLogF("[r% 3d] | %10p | %-16s 0x%lld \n", registerCount, it, v.description(), (long long)JSValue::encode(v)); ++it; ++registerCount; } while (it != end); } dataLogF("-----------------------------------------------------------------------------\n"); } #endif bool Interpreter::isOpcode(Opcode opcode) { #if ENABLE(COMPUTED_GOTO_OPCODES) #if !ENABLE(LLINT) return static_cast(bitwise_cast(opcode)) <= op_end; #else return opcode != HashTraits::emptyValue() && !HashTraits::isDeletedValue(opcode) && m_opcodeIDTable.contains(opcode); #endif #else return opcode >= 0 && opcode <= op_end; #endif } NEVER_INLINE bool Interpreter::unwindCallFrame(CallFrame*& callFrame, JSValue exceptionValue, unsigned& bytecodeOffset, CodeBlock*& codeBlock) { CodeBlock* oldCodeBlock = codeBlock; JSScope* scope = callFrame->scope(); if (Debugger* debugger = callFrame->dynamicGlobalObject()->debugger()) { DebuggerCallFrame debuggerCallFrame(callFrame, exceptionValue); if (callFrame->callee()) debugger->returnEvent(debuggerCallFrame, codeBlock->ownerExecutable()->sourceID(), codeBlock->ownerExecutable()->lastLine(), 0); else debugger->didExecuteProgram(debuggerCallFrame, codeBlock->ownerExecutable()->sourceID(), codeBlock->ownerExecutable()->lastLine(), 0); } JSValue activation; if (oldCodeBlock->codeType() == FunctionCode && oldCodeBlock->needsActivation()) { activation = callFrame->uncheckedR(oldCodeBlock->activationRegister()).jsValue(); if (activation) jsCast(activation)->tearOff(*scope->globalData()); } if (oldCodeBlock->codeType() == FunctionCode && oldCodeBlock->usesArguments()) { if (JSValue arguments = callFrame->uncheckedR(unmodifiedArgumentsRegister(oldCodeBlock->argumentsRegister())).jsValue()) { if (activation) jsCast(arguments)->didTearOffActivation(callFrame, jsCast(activation)); else jsCast(arguments)->tearOff(callFrame); } } CallFrame* callerFrame = callFrame->callerFrame(); callFrame->globalData().topCallFrame = callerFrame; if (callerFrame->hasHostCallFrameFlag()) return false; codeBlock = callerFrame->codeBlock(); // Because of how the JIT records call site->bytecode offset // information the JIT reports the bytecodeOffset for the returnPC // to be at the beginning of the opcode that has caused the call. #if ENABLE(JIT) || ENABLE(LLINT) bytecodeOffset = codeBlock->bytecodeOffset(callerFrame, callFrame->returnPC()); #endif callFrame = callerFrame; return true; } static void appendSourceToError(CallFrame* callFrame, ErrorInstance* exception, unsigned bytecodeOffset) { exception->clearAppendSourceToMessage(); if (!callFrame->codeBlock()->hasExpressionInfo()) return; int startOffset = 0; int endOffset = 0; int divotPoint = 0; CodeBlock* codeBlock = callFrame->codeBlock(); codeBlock->expressionRangeForBytecodeOffset(bytecodeOffset, divotPoint, startOffset, endOffset); int expressionStart = divotPoint - startOffset; int expressionStop = divotPoint + endOffset; const String& sourceString = codeBlock->source()->source(); if (!expressionStop || expressionStart > static_cast(sourceString.length())) return; JSGlobalData* globalData = &callFrame->globalData(); JSValue jsMessage = exception->getDirect(*globalData, globalData->propertyNames->message); if (!jsMessage || !jsMessage.isString()) return; String message = asString(jsMessage)->value(callFrame); if (expressionStart < expressionStop) message = makeString(message, " (evaluating '", codeBlock->source()->getRange(expressionStart, expressionStop), "')"); else { // No range information, so give a few characters of context const StringImpl* data = sourceString.impl(); int dataLength = sourceString.length(); int start = expressionStart; int stop = expressionStart; // Get up to 20 characters of context to the left and right of the divot, clamping to the line. // then strip whitespace. while (start > 0 && (expressionStart - start < 20) && (*data)[start - 1] != '\n') start--; while (start < (expressionStart - 1) && isStrWhiteSpace((*data)[start])) start++; while (stop < dataLength && (stop - expressionStart < 20) && (*data)[stop] != '\n') stop++; while (stop > expressionStart && isStrWhiteSpace((*data)[stop - 1])) stop--; message = makeString(message, " (near '...", codeBlock->source()->getRange(start, stop), "...')"); } exception->putDirect(*globalData, globalData->propertyNames->message, jsString(globalData, message)); } static int getLineNumberForCallFrame(JSGlobalData* globalData, CallFrame* callFrame) { UNUSED_PARAM(globalData); callFrame = callFrame->removeHostCallFrameFlag(); CodeBlock* codeBlock = callFrame->codeBlock(); if (!codeBlock) return -1; #if ENABLE(JIT) || ENABLE(LLINT) #if ENABLE(DFG_JIT) if (codeBlock->getJITType() == JITCode::DFGJIT) return codeBlock->lineNumberForBytecodeOffset(codeBlock->codeOrigin(callFrame->codeOriginIndexForDFG()).bytecodeIndex); #endif return codeBlock->lineNumberForBytecodeOffset(callFrame->bytecodeOffsetForNonDFGCode()); #endif } static CallFrame* getCallerInfo(JSGlobalData* globalData, CallFrame* callFrame, int& lineNumber, unsigned& bytecodeOffset) { UNUSED_PARAM(globalData); bytecodeOffset = 0; lineNumber = -1; ASSERT(!callFrame->hasHostCallFrameFlag()); CallFrame* callerFrame = callFrame->codeBlock() ? callFrame->trueCallerFrame() : callFrame->callerFrame()->removeHostCallFrameFlag(); bool callframeIsHost = callerFrame->addHostCallFrameFlag() == callFrame->callerFrame(); ASSERT(!callerFrame->hasHostCallFrameFlag()); if (callerFrame == CallFrame::noCaller() || !callerFrame || !callerFrame->codeBlock()) return callerFrame; CodeBlock* callerCodeBlock = callerFrame->codeBlock(); #if ENABLE(JIT) || ENABLE(LLINT) if (!callFrame->hasReturnPC()) callframeIsHost = true; #endif #if ENABLE(DFG_JIT) if (callFrame->isInlineCallFrame()) callframeIsHost = false; #endif if (callframeIsHost) { // Don't need to deal with inline callframes here as by definition we haven't // inlined a call with an intervening native call frame. #if ENABLE(JIT) || ENABLE(LLINT) #if ENABLE(DFG_JIT) if (callerCodeBlock && callerCodeBlock->getJITType() == JITCode::DFGJIT) { unsigned codeOriginIndex = callerFrame->codeOriginIndexForDFG(); bytecodeOffset = callerCodeBlock->codeOrigin(codeOriginIndex).bytecodeIndex; } else #endif bytecodeOffset = callerFrame->bytecodeOffsetForNonDFGCode(); #endif } else { #if ENABLE(JIT) || ENABLE(LLINT) #if ENABLE(DFG_JIT) if (callFrame->isInlineCallFrame()) { InlineCallFrame* icf = callFrame->inlineCallFrame(); bytecodeOffset = icf->caller.bytecodeIndex; if (InlineCallFrame* parentCallFrame = icf->caller.inlineCallFrame) { FunctionExecutable* executable = static_cast(parentCallFrame->executable.get()); CodeBlock* newCodeBlock = executable->baselineCodeBlockFor(parentCallFrame->isCall ? CodeForCall : CodeForConstruct); ASSERT(newCodeBlock); ASSERT(newCodeBlock->instructionCount() > bytecodeOffset); callerCodeBlock = newCodeBlock; } } else if (callerCodeBlock && callerCodeBlock->getJITType() == JITCode::DFGJIT) { CodeOrigin origin; if (!callerCodeBlock->codeOriginForReturn(callFrame->returnPC(), origin)) ASSERT_NOT_REACHED(); bytecodeOffset = origin.bytecodeIndex; if (InlineCallFrame* icf = origin.inlineCallFrame) { FunctionExecutable* executable = static_cast(icf->executable.get()); CodeBlock* newCodeBlock = executable->baselineCodeBlockFor(icf->isCall ? CodeForCall : CodeForConstruct); ASSERT(newCodeBlock); ASSERT(newCodeBlock->instructionCount() > bytecodeOffset); callerCodeBlock = newCodeBlock; } } else #endif bytecodeOffset = callerCodeBlock->bytecodeOffset(callerFrame, callFrame->returnPC()); #endif } lineNumber = callerCodeBlock->lineNumberForBytecodeOffset(bytecodeOffset); return callerFrame; } static ALWAYS_INLINE const String getSourceURLFromCallFrame(CallFrame* callFrame) { ASSERT(!callFrame->hasHostCallFrameFlag()); return callFrame->codeBlock()->ownerExecutable()->sourceURL(); } static StackFrameCodeType getStackFrameCodeType(CallFrame* callFrame) { ASSERT(!callFrame->hasHostCallFrameFlag()); switch (callFrame->codeBlock()->codeType()) { case EvalCode: return StackFrameEvalCode; case FunctionCode: return StackFrameFunctionCode; case GlobalCode: return StackFrameGlobalCode; } ASSERT_NOT_REACHED(); return StackFrameGlobalCode; } void Interpreter::getStackTrace(JSGlobalData* globalData, Vector& results) { CallFrame* callFrame = globalData->topCallFrame->removeHostCallFrameFlag(); if (!callFrame || callFrame == CallFrame::noCaller()) return; int line = getLineNumberForCallFrame(globalData, callFrame); callFrame = callFrame->trueCallFrameFromVMCode(); while (callFrame && callFrame != CallFrame::noCaller()) { String sourceURL; if (callFrame->codeBlock()) { sourceURL = getSourceURLFromCallFrame(callFrame); StackFrame s = { Strong(*globalData, callFrame->callee()), getStackFrameCodeType(callFrame), Strong(*globalData, callFrame->codeBlock()->ownerExecutable()), line, sourceURL}; results.append(s); } else { StackFrame s = { Strong(*globalData, callFrame->callee()), StackFrameNativeCode, Strong(), -1, String()}; results.append(s); } unsigned unusedBytecodeOffset = 0; callFrame = getCallerInfo(globalData, callFrame, line, unusedBytecodeOffset); } } void Interpreter::addStackTraceIfNecessary(CallFrame* callFrame, JSObject* error) { JSGlobalData* globalData = &callFrame->globalData(); ASSERT(callFrame == globalData->topCallFrame || callFrame == callFrame->lexicalGlobalObject()->globalExec() || callFrame == callFrame->dynamicGlobalObject()->globalExec()); if (error->hasProperty(callFrame, globalData->propertyNames->stack)) return; Vector stackTrace; getStackTrace(&callFrame->globalData(), stackTrace); if (stackTrace.isEmpty()) return; JSGlobalObject* globalObject = 0; if (isTerminatedExecutionException(error) || isInterruptedExecutionException(error)) globalObject = globalData->dynamicGlobalObject; else globalObject = error->globalObject(); // FIXME: JSStringJoiner could be more efficient than StringBuilder here. StringBuilder builder; for (unsigned i = 0; i < stackTrace.size(); i++) { builder.append(String(stackTrace[i].toString(globalObject->globalExec()).impl())); if (i != stackTrace.size() - 1) builder.append('\n'); } error->putDirect(*globalData, globalData->propertyNames->stack, jsString(globalData, builder.toString()), ReadOnly | DontDelete); } NEVER_INLINE HandlerInfo* Interpreter::throwException(CallFrame*& callFrame, JSValue& exceptionValue, unsigned bytecodeOffset) { CodeBlock* codeBlock = callFrame->codeBlock(); bool isInterrupt = false; ASSERT(!exceptionValue.isEmpty()); ASSERT(!exceptionValue.isCell() || exceptionValue.asCell()); // This shouldn't be possible (hence the assertions), but we're already in the slowest of // slow cases, so let's harden against it anyway to be safe. if (exceptionValue.isEmpty() || (exceptionValue.isCell() && !exceptionValue.asCell())) exceptionValue = jsNull(); // Set up the exception object if (exceptionValue.isObject()) { JSObject* exception = asObject(exceptionValue); if (exception->isErrorInstance() && static_cast(exception)->appendSourceToMessage()) appendSourceToError(callFrame, static_cast(exception), bytecodeOffset); if (!hasErrorInfo(callFrame, exception)) { // FIXME: should only really be adding these properties to VM generated exceptions, // but the inspector currently requires these for all thrown objects. addErrorInfo(callFrame, exception, codeBlock->lineNumberForBytecodeOffset(bytecodeOffset), codeBlock->ownerExecutable()->source()); } isInterrupt = isInterruptedExecutionException(exception) || isTerminatedExecutionException(exception); } if (Debugger* debugger = callFrame->dynamicGlobalObject()->debugger()) { DebuggerCallFrame debuggerCallFrame(callFrame, exceptionValue); bool hasHandler = codeBlock->handlerForBytecodeOffset(bytecodeOffset); debugger->exception(debuggerCallFrame, codeBlock->ownerExecutable()->sourceID(), codeBlock->lineNumberForBytecodeOffset(bytecodeOffset), 0, hasHandler); } // Calculate an exception handler vPC, unwinding call frames as necessary. HandlerInfo* handler = 0; while (isInterrupt || !(handler = codeBlock->handlerForBytecodeOffset(bytecodeOffset))) { if (!unwindCallFrame(callFrame, exceptionValue, bytecodeOffset, codeBlock)) { if (Profiler* profiler = callFrame->globalData().enabledProfiler()) profiler->exceptionUnwind(callFrame); return 0; } } if (Profiler* profiler = callFrame->globalData().enabledProfiler()) profiler->exceptionUnwind(callFrame); // Unwind the scope chain within the exception handler's call frame. JSScope* scope = callFrame->scope(); int scopeDelta = 0; if (!codeBlock->needsFullScopeChain() || codeBlock->codeType() != FunctionCode || callFrame->uncheckedR(codeBlock->activationRegister()).jsValue()) { int currentDepth = depth(codeBlock, scope); int targetDepth = handler->scopeDepth; scopeDelta = currentDepth - targetDepth; ASSERT(scopeDelta >= 0); } while (scopeDelta--) scope = scope->next(); callFrame->setScope(scope); return handler; } static inline JSValue checkedReturn(JSValue returnValue) { ASSERT(returnValue); return returnValue; } static inline JSObject* checkedReturn(JSObject* returnValue) { ASSERT(returnValue); return returnValue; } class SamplingScope { public: SamplingScope(Interpreter* interpreter) : m_interpreter(interpreter) { interpreter->startSampling(); } ~SamplingScope() { m_interpreter->stopSampling(); } private: Interpreter* m_interpreter; }; JSValue Interpreter::execute(ProgramExecutable* program, CallFrame* callFrame, JSObject* thisObj) { SamplingScope samplingScope(this); JSScope* scope = callFrame->scope(); JSGlobalData& globalData = *scope->globalData(); ASSERT(isValidThisObject(thisObj, callFrame)); ASSERT(!globalData.exception); ASSERT(!globalData.isCollectorBusy()); if (globalData.isCollectorBusy()) CRASH(); StackStats::CheckPoint stackCheckPoint; const StackBounds& nativeStack = wtfThreadData().stack(); StackPolicy policy(*this, nativeStack); if (!nativeStack.isSafeToRecurse(policy.requiredCapacity())) return checkedReturn(throwStackOverflowError(callFrame)); // First check if the "program" is actually just a JSON object. If so, // we'll handle the JSON object here. Else, we'll handle real JS code // below at failedJSONP. DynamicGlobalObjectScope globalObjectScope(globalData, scope->globalObject()); Vector JSONPData; bool parseResult; const String programSource = program->source().toString(); if (programSource.isNull()) return jsUndefined(); if (programSource.is8Bit()) { LiteralParser literalParser(callFrame, programSource.characters8(), programSource.length(), JSONP); parseResult = literalParser.tryJSONPParse(JSONPData, scope->globalObject()->globalObjectMethodTable()->supportsRichSourceInfo(scope->globalObject())); } else { LiteralParser literalParser(callFrame, programSource.characters16(), programSource.length(), JSONP); parseResult = literalParser.tryJSONPParse(JSONPData, scope->globalObject()->globalObjectMethodTable()->supportsRichSourceInfo(scope->globalObject())); } if (parseResult) { JSGlobalObject* globalObject = scope->globalObject(); JSValue result; for (unsigned entry = 0; entry < JSONPData.size(); entry++) { Vector JSONPPath; JSONPPath.swap(JSONPData[entry].m_path); JSValue JSONPValue = JSONPData[entry].m_value.get(); if (JSONPPath.size() == 1 && JSONPPath[0].m_type == JSONPPathEntryTypeDeclare) { if (globalObject->hasProperty(callFrame, JSONPPath[0].m_pathEntryName)) { PutPropertySlot slot; globalObject->methodTable()->put(globalObject, callFrame, JSONPPath[0].m_pathEntryName, JSONPValue, slot); } else globalObject->methodTable()->putDirectVirtual(globalObject, callFrame, JSONPPath[0].m_pathEntryName, JSONPValue, DontEnum | DontDelete); // var declarations return undefined result = jsUndefined(); continue; } JSValue baseObject(globalObject); for (unsigned i = 0; i < JSONPPath.size() - 1; i++) { ASSERT(JSONPPath[i].m_type != JSONPPathEntryTypeDeclare); switch (JSONPPath[i].m_type) { case JSONPPathEntryTypeDot: { if (i == 0) { PropertySlot slot(globalObject); if (!globalObject->getPropertySlot(callFrame, JSONPPath[i].m_pathEntryName, slot)) { if (entry) return throwError(callFrame, createUndefinedVariableError(globalObject->globalExec(), JSONPPath[i].m_pathEntryName)); goto failedJSONP; } baseObject = slot.getValue(callFrame, JSONPPath[i].m_pathEntryName); } else baseObject = baseObject.get(callFrame, JSONPPath[i].m_pathEntryName); if (callFrame->hadException()) return jsUndefined(); continue; } case JSONPPathEntryTypeLookup: { baseObject = baseObject.get(callFrame, JSONPPath[i].m_pathIndex); if (callFrame->hadException()) return jsUndefined(); continue; } default: ASSERT_NOT_REACHED(); return jsUndefined(); } } PutPropertySlot slot; switch (JSONPPath.last().m_type) { case JSONPPathEntryTypeCall: { JSValue function = baseObject.get(callFrame, JSONPPath.last().m_pathEntryName); if (callFrame->hadException()) return jsUndefined(); CallData callData; CallType callType = getCallData(function, callData); if (callType == CallTypeNone) return throwError(callFrame, createNotAFunctionError(callFrame, function)); MarkedArgumentBuffer jsonArg; jsonArg.append(JSONPValue); JSValue thisValue = JSONPPath.size() == 1 ? jsUndefined(): baseObject; JSONPValue = JSC::call(callFrame, function, callType, callData, thisValue, jsonArg); if (callFrame->hadException()) return jsUndefined(); break; } case JSONPPathEntryTypeDot: { baseObject.put(callFrame, JSONPPath.last().m_pathEntryName, JSONPValue, slot); if (callFrame->hadException()) return jsUndefined(); break; } case JSONPPathEntryTypeLookup: { baseObject.putByIndex(callFrame, JSONPPath.last().m_pathIndex, JSONPValue, slot.isStrictMode()); if (callFrame->hadException()) return jsUndefined(); break; } default: ASSERT_NOT_REACHED(); return jsUndefined(); } result = JSONPValue; } return result; } failedJSONP: // If we get here, then we have already proven that the script is not a JSON // object. // Compile source to bytecode if necessary: if (JSObject* error = program->initalizeGlobalProperties(globalData, callFrame, scope)) return checkedReturn(throwError(callFrame, error)); if (JSObject* error = program->compile(callFrame, scope)) return checkedReturn(throwError(callFrame, error)); ProgramCodeBlock* codeBlock = &program->generatedBytecode(); // Push the call frame for this invocation: ASSERT(codeBlock->numParameters() == 1); // 1 parameter for 'this'. CallFrame* newCallFrame = m_stack.pushFrame(callFrame, codeBlock, scope, 1, 0); if (UNLIKELY(!newCallFrame)) return checkedReturn(throwStackOverflowError(callFrame)); // Set the arguments for the callee: newCallFrame->setThisValue(thisObj); if (Profiler* profiler = globalData.enabledProfiler()) profiler->willExecute(callFrame, program->sourceURL(), program->lineNo()); // Execute the code: JSValue result; { SamplingTool::CallRecord callRecord(m_sampler.get()); #if ENABLE(LLINT_C_LOOP) result = LLInt::CLoop::execute(newCallFrame, llint_program_prologue); #elif ENABLE(JIT) result = program->generatedJITCode().execute(&m_stack, newCallFrame, &globalData); #endif // ENABLE(JIT) } if (Profiler* profiler = globalData.enabledProfiler()) profiler->didExecute(callFrame, program->sourceURL(), program->lineNo()); m_stack.popFrame(newCallFrame); return checkedReturn(result); } JSValue Interpreter::executeCall(CallFrame* callFrame, JSObject* function, CallType callType, const CallData& callData, JSValue thisValue, const ArgList& args) { JSGlobalData& globalData = callFrame->globalData(); ASSERT(isValidThisObject(thisValue, callFrame)); ASSERT(!callFrame->hadException()); ASSERT(!globalData.isCollectorBusy()); if (globalData.isCollectorBusy()) return jsNull(); StackStats::CheckPoint stackCheckPoint; const StackBounds& nativeStack = wtfThreadData().stack(); StackPolicy policy(*this, nativeStack); if (!nativeStack.isSafeToRecurse(policy.requiredCapacity())) return checkedReturn(throwStackOverflowError(callFrame)); bool isJSCall = (callType == CallTypeJS); JSScope* scope; CodeBlock* newCodeBlock; size_t argsCount = 1 + args.size(); // implicit "this" parameter if (isJSCall) scope = callData.js.scope; else { ASSERT(callType == CallTypeHost); scope = callFrame->scope(); } DynamicGlobalObjectScope globalObjectScope(globalData, scope->globalObject()); if (isJSCall) { // Compile the callee: JSObject* compileError = callData.js.functionExecutable->compileForCall(callFrame, scope); if (UNLIKELY(!!compileError)) { return checkedReturn(throwError(callFrame, compileError)); } newCodeBlock = &callData.js.functionExecutable->generatedBytecodeForCall(); ASSERT(!!newCodeBlock); } else newCodeBlock = 0; CallFrame* newCallFrame = m_stack.pushFrame(callFrame, newCodeBlock, scope, argsCount, function); if (UNLIKELY(!newCallFrame)) return checkedReturn(throwStackOverflowError(callFrame)); // Set the arguments for the callee: newCallFrame->setThisValue(thisValue); for (size_t i = 0; i < args.size(); ++i) newCallFrame->setArgument(i, args.at(i)); if (Profiler* profiler = globalData.enabledProfiler()) profiler->willExecute(callFrame, function); JSValue result; { SamplingTool::CallRecord callRecord(m_sampler.get(), !isJSCall); // Execute the code: if (isJSCall) { #if ENABLE(LLINT_C_LOOP) result = LLInt::CLoop::execute(newCallFrame, llint_function_for_call_prologue); #elif ENABLE(JIT) result = callData.js.functionExecutable->generatedJITCodeForCall().execute(&m_stack, newCallFrame, &globalData); #endif // ENABLE(JIT) } else result = JSValue::decode(callData.native.function(newCallFrame)); } if (Profiler* profiler = globalData.enabledProfiler()) profiler->didExecute(callFrame, function); m_stack.popFrame(newCallFrame); return checkedReturn(result); } JSObject* Interpreter::executeConstruct(CallFrame* callFrame, JSObject* constructor, ConstructType constructType, const ConstructData& constructData, const ArgList& args) { JSGlobalData& globalData = callFrame->globalData(); ASSERT(!callFrame->hadException()); ASSERT(!globalData.isCollectorBusy()); // We throw in this case because we have to return something "valid" but we're // already in an invalid state. if (globalData.isCollectorBusy()) return checkedReturn(throwStackOverflowError(callFrame)); StackStats::CheckPoint stackCheckPoint; const StackBounds& nativeStack = wtfThreadData().stack(); StackPolicy policy(*this, nativeStack); if (!nativeStack.isSafeToRecurse(policy.requiredCapacity())) return checkedReturn(throwStackOverflowError(callFrame)); bool isJSConstruct = (constructType == ConstructTypeJS); JSScope* scope; CodeBlock* newCodeBlock; size_t argsCount = 1 + args.size(); // implicit "this" parameter if (isJSConstruct) scope = constructData.js.scope; else { ASSERT(constructType == ConstructTypeHost); scope = callFrame->scope(); } DynamicGlobalObjectScope globalObjectScope(globalData, scope->globalObject()); if (isJSConstruct) { // Compile the callee: JSObject* compileError = constructData.js.functionExecutable->compileForConstruct(callFrame, scope); if (UNLIKELY(!!compileError)) { return checkedReturn(throwError(callFrame, compileError)); } newCodeBlock = &constructData.js.functionExecutable->generatedBytecodeForConstruct(); ASSERT(!!newCodeBlock); } else newCodeBlock = 0; CallFrame* newCallFrame = m_stack.pushFrame(callFrame, newCodeBlock, scope, argsCount, constructor); if (UNLIKELY(!newCallFrame)) return checkedReturn(throwStackOverflowError(callFrame)); // Set the arguments for the callee: newCallFrame->setThisValue(jsUndefined()); for (size_t i = 0; i < args.size(); ++i) newCallFrame->setArgument(i, args.at(i)); if (Profiler* profiler = globalData.enabledProfiler()) profiler->willExecute(callFrame, constructor); JSValue result; { SamplingTool::CallRecord callRecord(m_sampler.get(), !isJSConstruct); // Execute the code. if (isJSConstruct) { #if ENABLE(LLINT_C_LOOP) result = LLInt::CLoop::execute(newCallFrame, llint_function_for_construct_prologue); #elif ENABLE(JIT) result = constructData.js.functionExecutable->generatedJITCodeForConstruct().execute(&m_stack, newCallFrame, &globalData); #endif // ENABLE(JIT) } else { result = JSValue::decode(constructData.native.function(newCallFrame)); } } if (Profiler* profiler = globalData.enabledProfiler()) profiler->didExecute(callFrame, constructor); m_stack.popFrame(newCallFrame); if (callFrame->hadException()) return 0; ASSERT(result.isObject()); return checkedReturn(asObject(result)); } CallFrameClosure Interpreter::prepareForRepeatCall(FunctionExecutable* functionExecutable, CallFrame* callFrame, JSFunction* function, int argumentCountIncludingThis, JSScope* scope) { JSGlobalData& globalData = *scope->globalData(); ASSERT(!globalData.exception); if (globalData.isCollectorBusy()) return CallFrameClosure(); StackStats::CheckPoint stackCheckPoint; const StackBounds& nativeStack = wtfThreadData().stack(); StackPolicy policy(*this, nativeStack); if (!nativeStack.isSafeToRecurse(policy.requiredCapacity())) { throwStackOverflowError(callFrame); return CallFrameClosure(); } // Compile the callee: JSObject* error = functionExecutable->compileForCall(callFrame, scope); if (error) { throwError(callFrame, error); return CallFrameClosure(); } CodeBlock* newCodeBlock = &functionExecutable->generatedBytecodeForCall(); size_t argsCount = argumentCountIncludingThis; CallFrame* newCallFrame = m_stack.pushFrame(callFrame, newCodeBlock, scope, argsCount, function); if (UNLIKELY(!newCallFrame)) { throwStackOverflowError(callFrame); return CallFrameClosure(); } if (UNLIKELY(!newCallFrame)) { throwStackOverflowError(callFrame); return CallFrameClosure(); } // Return the successful closure: CallFrameClosure result = { callFrame, newCallFrame, function, functionExecutable, &globalData, scope, newCodeBlock->numParameters(), argumentCountIncludingThis }; return result; } JSValue Interpreter::execute(CallFrameClosure& closure) { JSGlobalData& globalData = *closure.globalData; SamplingScope samplingScope(this); ASSERT(!globalData.isCollectorBusy()); if (globalData.isCollectorBusy()) return jsNull(); StackStats::CheckPoint stackCheckPoint; m_stack.validateFence(closure.newCallFrame, "BEFORE"); closure.resetCallFrame(); m_stack.validateFence(closure.newCallFrame, "STEP 1"); if (Profiler* profiler = globalData.enabledProfiler()) profiler->willExecute(closure.oldCallFrame, closure.function); // The code execution below may push more frames and point the topCallFrame // to those newer frames, or it may pop to the top frame to the caller of // the current repeat frame, or it may leave the top frame pointing to the // current repeat frame. // // Hence, we need to preserve the topCallFrame here ourselves before // repeating this call on a second callback function. TopCallFrameSetter topCallFrame(globalData, closure.newCallFrame); // Execute the code: JSValue result; { SamplingTool::CallRecord callRecord(m_sampler.get()); #if ENABLE(LLINT_C_LOOP) result = LLInt::CLoop::execute(closure.newCallFrame, llint_function_for_call_prologue); #elif ENABLE(JIT) result = closure.functionExecutable->generatedJITCodeForCall().execute(&m_stack, closure.newCallFrame, &globalData); #endif // ENABLE(JIT) } if (Profiler* profiler = globalData.enabledProfiler()) profiler->didExecute(closure.oldCallFrame, closure.function); m_stack.validateFence(closure.newCallFrame, "AFTER"); return checkedReturn(result); } void Interpreter::endRepeatCall(CallFrameClosure& closure) { m_stack.popFrame(closure.newCallFrame); } JSValue Interpreter::execute(EvalExecutable* eval, CallFrame* callFrame, JSValue thisValue, JSScope* scope) { JSGlobalData& globalData = *scope->globalData(); SamplingScope samplingScope(this); ASSERT(scope->globalData() == &callFrame->globalData()); ASSERT(isValidThisObject(thisValue, callFrame)); ASSERT(!globalData.exception); ASSERT(!globalData.isCollectorBusy()); if (globalData.isCollectorBusy()) return jsNull(); DynamicGlobalObjectScope globalObjectScope(globalData, scope->globalObject()); StackStats::CheckPoint stackCheckPoint; const StackBounds& nativeStack = wtfThreadData().stack(); StackPolicy policy(*this, nativeStack); if (!nativeStack.isSafeToRecurse(policy.requiredCapacity())) return checkedReturn(throwStackOverflowError(callFrame)); // Compile the callee: JSObject* compileError = eval->compile(callFrame, scope); if (UNLIKELY(!!compileError)) return checkedReturn(throwError(callFrame, compileError)); EvalCodeBlock* codeBlock = &eval->generatedBytecode(); JSObject* variableObject; for (JSScope* node = scope; ; node = node->next()) { ASSERT(node); if (node->isVariableObject() && !node->isNameScopeObject()) { variableObject = node; break; } } unsigned numVariables = codeBlock->numVariables(); int numFunctions = codeBlock->numberOfFunctionDecls(); if (numVariables || numFunctions) { if (codeBlock->isStrictMode()) { scope = StrictEvalActivation::create(callFrame); variableObject = scope; } // Scope for BatchedTransitionOptimizer BatchedTransitionOptimizer optimizer(globalData, variableObject); for (unsigned i = 0; i < numVariables; ++i) { const Identifier& ident = codeBlock->variable(i); if (!variableObject->hasProperty(callFrame, ident)) { PutPropertySlot slot; variableObject->methodTable()->put(variableObject, callFrame, ident, jsUndefined(), slot); } } for (int i = 0; i < numFunctions; ++i) { FunctionExecutable* function = codeBlock->functionDecl(i); PutPropertySlot slot; variableObject->methodTable()->put(variableObject, callFrame, function->name(), JSFunction::create(callFrame, function, scope), slot); } } // Push the frame: ASSERT(codeBlock->numParameters() == 1); // 1 parameter for 'this'. CallFrame* newCallFrame = m_stack.pushFrame(callFrame, codeBlock, scope, 1, 0); if (UNLIKELY(!newCallFrame)) return checkedReturn(throwStackOverflowError(callFrame)); // Set the arguments for the callee: newCallFrame->setThisValue(thisValue); if (Profiler* profiler = globalData.enabledProfiler()) profiler->willExecute(callFrame, eval->sourceURL(), eval->lineNo()); // Execute the code: JSValue result; { SamplingTool::CallRecord callRecord(m_sampler.get()); #if ENABLE(LLINT_C_LOOP) result = LLInt::CLoop::execute(newCallFrame, llint_eval_prologue); #elif ENABLE(JIT) result = eval->generatedJITCode().execute(&m_stack, newCallFrame, &globalData); #endif // ENABLE(JIT) } if (Profiler* profiler = globalData.enabledProfiler()) profiler->didExecute(callFrame, eval->sourceURL(), eval->lineNo()); m_stack.popFrame(newCallFrame); return checkedReturn(result); } NEVER_INLINE void Interpreter::debug(CallFrame* callFrame, DebugHookID debugHookID, int firstLine, int lastLine, int column) { Debugger* debugger = callFrame->dynamicGlobalObject()->debugger(); if (!debugger) return; switch (debugHookID) { case DidEnterCallFrame: debugger->callEvent(callFrame, callFrame->codeBlock()->ownerExecutable()->sourceID(), firstLine, column); return; case WillLeaveCallFrame: debugger->returnEvent(callFrame, callFrame->codeBlock()->ownerExecutable()->sourceID(), lastLine, column); return; case WillExecuteStatement: debugger->atStatement(callFrame, callFrame->codeBlock()->ownerExecutable()->sourceID(), firstLine, column); return; case WillExecuteProgram: debugger->willExecuteProgram(callFrame, callFrame->codeBlock()->ownerExecutable()->sourceID(), firstLine, column); return; case DidExecuteProgram: debugger->didExecuteProgram(callFrame, callFrame->codeBlock()->ownerExecutable()->sourceID(), lastLine, column); return; case DidReachBreakpoint: debugger->didReachBreakpoint(callFrame, callFrame->codeBlock()->ownerExecutable()->sourceID(), lastLine, column); return; } } JSValue Interpreter::retrieveArgumentsFromVMCode(CallFrame* callFrame, JSFunction* function) const { CallFrame* functionCallFrame = findFunctionCallFrameFromVMCode(callFrame, function); if (!functionCallFrame) return jsNull(); Arguments* arguments = Arguments::create(functionCallFrame->globalData(), functionCallFrame); arguments->tearOff(functionCallFrame); return JSValue(arguments); } JSValue Interpreter::retrieveCallerFromVMCode(CallFrame* callFrame, JSFunction* function) const { CallFrame* functionCallFrame = findFunctionCallFrameFromVMCode(callFrame, function); if (!functionCallFrame) return jsNull(); int lineNumber; unsigned bytecodeOffset; CallFrame* callerFrame = getCallerInfo(&callFrame->globalData(), functionCallFrame, lineNumber, bytecodeOffset); if (!callerFrame) return jsNull(); JSValue caller = callerFrame->callee(); if (!caller) return jsNull(); // Skip over function bindings. ASSERT(caller.isObject()); while (asObject(caller)->inherits(&JSBoundFunction::s_info)) { callerFrame = getCallerInfo(&callFrame->globalData(), callerFrame, lineNumber, bytecodeOffset); if (!callerFrame) return jsNull(); caller = callerFrame->callee(); if (!caller) return jsNull(); } return caller; } void Interpreter::retrieveLastCaller(CallFrame* callFrame, int& lineNumber, intptr_t& sourceID, String& sourceURL, JSValue& function) const { function = JSValue(); lineNumber = -1; sourceURL = String(); CallFrame* callerFrame = callFrame->callerFrame(); if (callerFrame->hasHostCallFrameFlag()) return; CodeBlock* callerCodeBlock = callerFrame->codeBlock(); if (!callerCodeBlock) return; unsigned bytecodeOffset = 0; bytecodeOffset = callerCodeBlock->bytecodeOffset(callerFrame, callFrame->returnPC()); lineNumber = callerCodeBlock->lineNumberForBytecodeOffset(bytecodeOffset - 1); sourceID = callerCodeBlock->ownerExecutable()->sourceID(); sourceURL = callerCodeBlock->ownerExecutable()->sourceURL(); function = callerFrame->callee(); } CallFrame* Interpreter::findFunctionCallFrameFromVMCode(CallFrame* callFrame, JSFunction* function) { for (CallFrame* candidate = callFrame->trueCallFrameFromVMCode(); candidate; candidate = candidate->trueCallerFrame()) { if (candidate->callee() == function) return candidate; } return 0; } void Interpreter::enableSampler() { #if ENABLE(OPCODE_SAMPLING) if (!m_sampler) { m_sampler = adoptPtr(new SamplingTool(this)); m_sampler->setup(); } #endif } void Interpreter::dumpSampleData(ExecState* exec) { #if ENABLE(OPCODE_SAMPLING) if (m_sampler) m_sampler->dump(exec); #else UNUSED_PARAM(exec); #endif } void Interpreter::startSampling() { #if ENABLE(SAMPLING_THREAD) if (!m_sampleEntryDepth) SamplingThread::start(); m_sampleEntryDepth++; #endif } void Interpreter::stopSampling() { #if ENABLE(SAMPLING_THREAD) m_sampleEntryDepth--; if (!m_sampleEntryDepth) SamplingThread::stop(); #endif } } // namespace JSC