Objective-C API: Rename JSValue.h/APIJSValue.h to JSCJSValue.h/JSValue.h
[WebKit-https.git] / Source / WebCore / bindings / js / ScriptCallStackFactory.cpp
1 /*
2  * Copyright (c) 2010 Google Inc. All rights reserved.
3  * Copyright (C) 2012 Research In Motion Limited. All rights reserved.
4  * 
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are
7  * met:
8  * 
9  *     * Redistributions of source code must retain the above copyright
10  * notice, this list of conditions and the following disclaimer.
11  *     * Redistributions in binary form must reproduce the above
12  * copyright notice, this list of conditions and the following disclaimer
13  * in the documentation and/or other materials provided with the
14  * distribution.
15  *     * Neither the name of Google Inc. nor the names of its
16  * contributors may be used to endorse or promote products derived from
17  * this software without specific prior written permission.
18  * 
19  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30  */
31
32 #include "config.h"
33 #include "ScriptCallStackFactory.h"
34
35 #include "InspectorInstrumentation.h"
36 #include "JSDOMBinding.h"
37 #include "JSMainThreadExecState.h"
38 #include "ScriptArguments.h"
39 #include "ScriptCallFrame.h"
40 #include "ScriptCallStack.h"
41 #include "ScriptValue.h"
42 #include <interpreter/CallFrame.h>
43 #include <interpreter/Interpreter.h>
44 #include <runtime/ArgList.h>
45 #include <runtime/JSCJSValue.h>
46 #include <runtime/JSFunction.h>
47 #include <runtime/JSGlobalData.h>
48 #include <wtf/text/WTFString.h>
49
50 using namespace JSC;
51
52 namespace WebCore {
53
54 class ScriptExecutionContext;
55
56 PassRefPtr<ScriptCallStack> createScriptCallStack(size_t maxStackSize, bool emptyIsAllowed)
57 {
58     Vector<ScriptCallFrame> frames;
59     if (JSC::ExecState* exec = JSMainThreadExecState::currentState()) {
60         Vector<StackFrame> stackTrace;
61         Interpreter::getStackTrace(&exec->globalData(), stackTrace);
62         for (Vector<StackFrame>::const_iterator iter = stackTrace.begin(); iter < stackTrace.end(); iter++) {
63             frames.append(ScriptCallFrame(iter->friendlyFunctionName(exec), iter->friendlySourceURL(), iter->friendlyLineNumber()));
64             if (frames.size() >= maxStackSize)
65                 break;
66         }
67     }
68     if (frames.isEmpty() && !emptyIsAllowed) {
69         // No frames found. It may happen in the case where
70         // a bound function is called from native code for example.
71         // Fallback to setting lineNumber to 0, and source and function name to "undefined".
72         frames.append(ScriptCallFrame("undefined", "undefined", 0));
73     }
74     return ScriptCallStack::create(frames);
75 }
76
77 PassRefPtr<ScriptCallStack> createScriptCallStack(JSC::ExecState* exec, size_t maxStackSize)
78 {
79     Vector<ScriptCallFrame> frames;
80     CallFrame* callFrame = exec;
81     while (true) {
82         ASSERT(callFrame);
83         int signedLineNumber;
84         intptr_t sourceID;
85         String urlString;
86         JSValue function;
87
88         exec->interpreter()->retrieveLastCaller(callFrame, signedLineNumber, sourceID, urlString, function);
89         String functionName;
90         if (function)
91             functionName = jsCast<JSFunction*>(function)->name(exec);
92         else {
93             // Caller is unknown, but if frames is empty we should still add the frame, because
94             // something called us, and gave us arguments.
95             if (!frames.isEmpty())
96                 break;
97         }
98         unsigned lineNumber = signedLineNumber >= 0 ? signedLineNumber : 0;
99         frames.append(ScriptCallFrame(functionName, urlString, lineNumber));
100         if (!function || frames.size() == maxStackSize)
101             break;
102         callFrame = callFrame->callerFrame();
103     }
104     return ScriptCallStack::create(frames);
105 }
106
107 PassRefPtr<ScriptCallStack> createScriptCallStackForConsole(JSC::ExecState* exec)
108 {
109     size_t maxStackSize = 1;
110     if (InspectorInstrumentation::hasFrontends()) {
111         ScriptExecutionContext* scriptExecutionContext = jsCast<JSDOMGlobalObject*>(exec->lexicalGlobalObject())->scriptExecutionContext();
112         if (InspectorInstrumentation::consoleAgentEnabled(scriptExecutionContext))
113             maxStackSize = ScriptCallStack::maxCallStackSizeToCapture;
114     }
115     return createScriptCallStack(exec, maxStackSize);
116 }
117
118 PassRefPtr<ScriptArguments> createScriptArguments(JSC::ExecState* exec, unsigned skipArgumentCount)
119 {
120     Vector<ScriptValue> arguments;
121     size_t argumentCount = exec->argumentCount();
122     for (size_t i = skipArgumentCount; i < argumentCount; ++i)
123         arguments.append(ScriptValue(exec->globalData(), exec->argument(i)));
124     return ScriptArguments::create(exec, arguments);
125 }
126
127 } // namespace WebCore