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