Reviewed by Adam.
[WebKit-https.git] / WebKitTools / Drosera / win / DebuggerDocumentPlatform.cpp
1 /*
2  * Copyright (C) 2007 Apple, 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
6  * are met:
7  *
8  * 1.  Redistributions of source code must retain the above copyright
9  *     notice, this list of conditions and the following disclaimer. 
10  * 2.  Redistributions in binary form must reproduce the above copyright
11  *     notice, this list of conditions and the following disclaimer in the
12  *     documentation and/or other materials provided with the distribution. 
13  * 3.  Neither the name of Apple Computer, Inc. ("Apple") nor the names of
14  *     its contributors may be used to endorse or promote products derived
15  *     from this software without specific prior written permission. 
16  *
17  * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
18  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
19  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
20  * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
21  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
22  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
23  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
24  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  */
28 #include "config.h"
29 #include "DebuggerDocument.h"
30
31 #include "ServerConnection.h"
32
33 #include <JavaScriptCore/JSRetainPtr.h>
34 #include <JavaScriptCore/JSStringRef.h>
35 #include <JavaScriptCore/JSStringRefBSTR.h>
36 #include <WebKit/IWebScriptCallFrame.h>
37 #include <WebKit/IWebScriptScope.h>
38
39 JSValueRef JSValueRefCreateWithBSTR(JSContextRef context, BSTR string)
40 {
41     JSRetainPtr<JSStringRef> jsString(Adopt, JSStringCreateWithBSTR(string));
42     return JSValueMakeString(context, jsString.get());
43 }
44
45 // DebuggerDocument platform specific implementations
46
47 void DebuggerDocument::platformPause()
48 {
49     m_server->pause();
50 }
51
52 void DebuggerDocument::platformResume()
53 {
54     m_server->resume();
55 }
56
57 void DebuggerDocument::platformStepInto()
58 {
59     m_server->stepInto();
60 }
61
62 JSValueRef DebuggerDocument::platformEvaluateScript(JSContextRef context, JSStringRef script, int callFrame)
63 {
64     HRESULT ret = S_OK;
65
66     COMPtr<IWebScriptCallFrame> cframe = m_server->getCallerFrame(callFrame);
67     if (!cframe)
68         return JSValueMakeUndefined(context);
69
70     // Convert script to BSTR
71     BSTR scriptBSTR = JSStringCopyBSTR(script);
72     BSTR value = 0;
73     ret = cframe->stringByEvaluatingJavaScriptFromString(scriptBSTR, &value);
74     SysFreeString(scriptBSTR);
75     if (FAILED(ret)) {
76         SysFreeString(value);
77         return JSValueMakeUndefined(context);
78     }
79
80     JSValueRef returnValue = JSValueRefCreateWithBSTR(context, value);
81     SysFreeString(value);
82     return returnValue;
83
84 }
85
86 void DebuggerDocument::getPlatformCurrentFunctionStack(JSContextRef context, Vector<JSValueRef>& currentStack)
87 {
88     HRESULT ret = S_OK;
89
90     COMPtr<IWebScriptCallFrame> caller;
91     for (COMPtr<IWebScriptCallFrame> frame = m_server->currentFrame(); frame; frame = caller) {
92         BSTR function = 0;
93         ret = frame->functionName(&function);
94         if (FAILED(ret)) {
95             SysFreeString(function);
96             return;
97         }
98
99         ret = frame->caller(&caller);
100         if (FAILED(ret))
101             return;
102
103         bool needToFree = true;
104         if (!function) {
105             needToFree = false;
106             if (caller)
107                 function = L"(anonymous function)";
108             else
109                 function = L"(global scope)";
110         }
111
112         currentStack.append(JSValueRefCreateWithBSTR(context, function));
113         if (needToFree)
114             SysFreeString(function);
115     }
116 }
117
118 void DebuggerDocument::getPlatformLocalScopeVariableNamesForCallFrame(JSContextRef context, int callFrame, Vector<JSValueRef>& variableNames)
119 {
120     HRESULT ret = S_OK;
121
122     COMPtr<IWebScriptCallFrame> cframe = m_server->getCallerFrame(callFrame);
123     if (!cframe)
124         return;
125
126     COMPtr<IEnumVARIANT> scopeChain;
127     ret = cframe->scopeChain(&scopeChain);
128     if (FAILED(ret))
129         return;
130
131     VARIANT var;
132     VariantInit(&var);
133     ret = scopeChain->Next(1, &var, 0); // local is always first
134     if (FAILED(ret)) {
135         VariantClear(&var);
136         return;
137     }
138
139     ASSERT(V_VT(&var) == VT_UNKNOWN);
140     COMPtr<IWebScriptScope> scope;
141     scope.query(V_UNKNOWN(&var));
142     VariantClear(&var);
143
144     COMPtr<IEnumVARIANT> localScopeVariableNames;
145     ret = scope->variableNames(&localScopeVariableNames);
146     if (FAILED(ret))
147         return;
148
149     while (localScopeVariableNames->Next(1, &var, 0) == S_OK) {
150         ASSERT(V_VT(&var) == VT_BSTR);
151         BSTR variableName;
152
153         variableName = V_BSTR(&var);
154         variableNames.append(JSValueRefCreateWithBSTR(context, variableName));
155
156         SysFreeString(variableName);
157         VariantClear(&var);
158     }
159 }
160
161 JSValueRef DebuggerDocument::platformValueForScopeVariableNamed(JSContextRef context, JSStringRef key, int callFrame)
162 {
163     HRESULT ret = S_OK;
164
165     COMPtr<IWebScriptCallFrame> cframe = m_server->getCallerFrame(callFrame);
166     if (!cframe)
167         return JSValueMakeUndefined(context);
168
169     COMPtr<IEnumVARIANT> scopeChain;
170     ret = cframe->scopeChain(&scopeChain);
171     if (FAILED(ret))
172         return JSValueMakeUndefined(context);
173
174     VARIANT var;
175     VariantInit(&var);
176
177     BSTR resultString = 0;
178     BSTR bstrKey = JSStringCopyBSTR(key);
179     while (scopeChain->Next(1, &var, 0) == S_OK && resultString == 0) {
180         ASSERT(V_VT(&var) == VT_UNKNOWN);
181
182         COMPtr<IWebScriptScope> scope;
183         scope.query(V_UNKNOWN(&var));
184         ret = scope->valueForVariable(bstrKey, &resultString);
185         VariantClear(&var);
186
187         if (FAILED(ret)) {
188             SysFreeString(bstrKey);
189             SysFreeString(resultString);
190             return JSValueMakeUndefined(context);
191         }
192     }
193     SysFreeString(bstrKey);
194
195     JSValueRef returnValue = JSValueRefCreateWithBSTR(context, resultString);
196     SysFreeString(resultString);
197
198     return returnValue;
199 }
200
201
202 void DebuggerDocument::platformLog(JSStringRef msg)
203 {
204     printf("%S\n", JSStringGetCharactersPtr(msg));
205 }