OLD | NEW |
| (Empty) |
1 // Copyright (c) 2009, Google Inc. | |
2 // 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 #include "config.h" | |
31 #include "DartController.h" | |
32 | |
33 #include "DOMTimer.h" | |
34 #include "DOMWindow.h" | |
35 #include "DartApplicationLoader.h" | |
36 #include "DartBuiltinLibrarySource.h" | |
37 #include "DartClassInfo.h" | |
38 #include "DartDOMWindow.h" | |
39 #include "DartDOMWrapper.h" | |
40 #include "DartIsolateState.h" | |
41 #include "DartUtilities.h" | |
42 #include "Document.h" | |
43 #include "Frame.h" | |
44 #include "HTMLNames.h" | |
45 #include "HTMLScriptElement.h" | |
46 #include "IDBPendingTransactionMonitor.h" | |
47 #include "NodeList.h" | |
48 #include "Page.h" | |
49 #include "PageGroup.h" | |
50 #include "ScheduledAction.h" | |
51 #include "ScriptExecutionContext.h" | |
52 #include "Settings.h" | |
53 #include "StorageNamespace.h" | |
54 | |
55 #include "npruntime_impl.h" | |
56 #include <bindings/npruntime.h> | |
57 | |
58 namespace WebCore { | |
59 | |
60 static void initDOMIsolate() | |
61 { | |
62 DartApiScope dartApiScope; | |
63 | |
64 // Fix the dom library. | |
65 Dart_Handle dom = Dart_LookupLibrary(Dart_NewString(DartUtilities::domLibrar
yName)); | |
66 | |
67 // Inject builtin library to forward core functionality to dom. | |
68 // FIXME: We need to provide something for non-DOM isolates as well. | |
69 Dart_Handle core = Dart_LookupLibrary(Dart_NewString("dart:core")); | |
70 Dart_LoadSource(core, Dart_NewString("dart:builtin"), Dart_NewString(dartBui
ltinLibrarySource)); | |
71 Dart_LibraryImportLibrary(core, dom); | |
72 | |
73 Dart_Handle domimpl = Dart_LookupLibrary(Dart_NewString("dart:domimpl")); | |
74 ASSERT(!Dart_IsError(domimpl)); | |
75 Dart_SetNativeResolver(domimpl, domResolver); | |
76 } | |
77 | |
78 DartPerScriptState::DartPerScriptState(Document* document, PassRefPtr<DartApplic
ationLoader> applicationLoader) | |
79 : m_dartApplicationLoader(applicationLoader) | |
80 , m_isolate(DartIsolateState::create(document, m_dartApplicationLoader.get()
)) | |
81 { | |
82 initDOMIsolate(); | |
83 isolateToDartApplicationLoaderMap().set(m_isolate, m_dartApplicationLoader.g
et()); | |
84 // DartIsolateState::create pushes newly create isolate, undo it. | |
85 DartIsolateState::pop(); | |
86 } | |
87 | |
88 DartPerScriptState::~DartPerScriptState() | |
89 { | |
90 *DartUtilities::recursionForIsolate(m_isolate) = 0; | |
91 isolateToDartApplicationLoaderMap().remove(m_isolate); | |
92 DartIsolateState::shutdown(m_isolate); | |
93 } | |
94 | |
95 DartController::DartController(Frame* frame) | |
96 : m_frame(frame) | |
97 , m_scriptsLoaded(false) | |
98 , m_states() | |
99 , m_npObjectMap() | |
100 { | |
101 // The DartController's constructor must be called in the Frame's | |
102 // constructor, so it can properly maintain the unit of related | |
103 // browsing contexts. | |
104 | |
105 // The DartController must be created after the frame's loader and | |
106 // tree nodes are initialized. | |
107 ASSERT(frame->loader()); | |
108 ASSERT(frame->tree()); | |
109 } | |
110 | |
111 void DartController::clearWindowShell() | |
112 { | |
113 m_scriptsLoaded = false; | |
114 m_states.clear(); | |
115 } | |
116 | |
117 class PostMessageTask : public ScriptExecutionContext::Task { | |
118 public: | |
119 PostMessageTask(Dart_Isolate destinationIsolate, Dart_Port destinationPort,
Dart_Port replyPort, Dart_Message message) | |
120 : m_destinationIsolate(destinationIsolate) | |
121 , m_destinationPort(destinationPort) | |
122 , m_replyPort(replyPort) | |
123 , m_message(message) { } | |
124 | |
125 ~PostMessageTask() | |
126 { | |
127 free(m_message); | |
128 } | |
129 | |
130 virtual void performTask(ScriptExecutionContext* context) | |
131 { | |
132 // FIXME: one shouldn't trust isFullDomIsolate as another | |
133 // isolate with the same address might be allocated. Apparently better | |
134 // way would be to maintain a way to tell all the tasks that they are | |
135 // cancelled from now on. For example, we may maintain a list of all | |
136 // pending tasks and iterate over it. | |
137 // destinationIsolate might have been shut down before. | |
138 if (!DartUtilities::isFullDomIsolate(m_destinationIsolate)) | |
139 return; | |
140 DartIsolateState::Scope scope(m_destinationIsolate); | |
141 DartApiScope apiScope; | |
142 Dart_Handle result = Dart_HandleMessage(m_destinationPort, m_replyPort,
m_message); | |
143 if (Dart_IsError(result)) | |
144 DartUtilities::reportProblem(context, result); | |
145 } | |
146 | |
147 private: | |
148 Dart_Isolate m_destinationIsolate; | |
149 Dart_Port m_destinationPort; | |
150 Dart_Port m_replyPort; | |
151 Dart_Message m_message; | |
152 }; | |
153 | |
154 static bool postMessageCallback(Dart_Isolate destinationIsolate, Dart_Port desti
nationPort, Dart_Port replyPort, Dart_Message message) | |
155 { | |
156 ASSERT(DartUtilities::isFullDomIsolate(destinationIsolate)); | |
157 ScriptExecutionContext* destinationContext = DartUtilities::isolateContext(d
estinationIsolate); | |
158 destinationContext->postTask(adoptPtr(new PostMessageTask(destinationIsolate
, destinationPort, replyPort, message))); | |
159 return true; | |
160 } | |
161 | |
162 static void closePortCallback(Dart_Isolate, Dart_Port) | |
163 { | |
164 } | |
165 | |
166 void DartController::setupDOMEnabledIsolate(ScriptExecutionContext* context) | |
167 { | |
168 ASSERT(context); | |
169 Dart_SetMessageCallbacks(&postMessageCallback, &closePortCallback); | |
170 DartUtilities::registerIsolateContext(Dart_CurrentIsolate(), context); | |
171 } | |
172 | |
173 bool DartController::createPureIsolateCallback(void* data, char** errorMsg) | |
174 { | |
175 Dart_Isolate isolate = Dart_CreateIsolate(DartUtilities::fullSnapshot(), dat
a, errorMsg); | |
176 if (!isolate) | |
177 return false; | |
178 | |
179 DartApiScope dartApiScope; | |
180 // It's safe to initialize DOM for pure Dart isolates: unless Dart isolate i
s | |
181 // registered with DartUtilities::registerIsolatecContext, its DOM functiona
lity | |
182 // will be disabled: we won't be able to resolve top level accessors. | |
183 initDOMIsolate(); | |
184 | |
185 DartApplicationLoader* dartApplicationLoader = reinterpret_cast<DartApplicat
ionLoader*>(data); | |
186 // FIXME: when DartVM has shutdown callback, we'll be able to deref it. | |
187 dartApplicationLoader->ref(); | |
188 ASSERT(!dartApplicationLoader->isLoadingMainIsolate()); | |
189 dartApplicationLoader->reinjectSources(); | |
190 | |
191 return true; | |
192 } | |
193 | |
194 void DartController::initVMIfNeeded() | |
195 { | |
196 static bool hasBeenInitialized = false; | |
197 if (hasBeenInitialized) | |
198 return; | |
199 | |
200 // FIXME: make sure Dart_SetVMFlags is called even if there was no | |
201 // --dart-flags command line switch. | |
202 Dart_SetVMFlags(0, 0); | |
203 Dart_Initialize(&createPureIsolateCallback, 0); | |
204 hasBeenInitialized = true; | |
205 } | |
206 | |
207 bool DartController::isDartMimeType(const String& mimeType) | |
208 { | |
209 DEFINE_STATIC_LOCAL(HashSet<String>, types, ()); | |
210 if (types.isEmpty()) { | |
211 types.add("application/dart"); | |
212 types.add("application/dart-app"); | |
213 types.add("application/dart-script"); | |
214 } | |
215 return !mimeType.isEmpty() && types.contains(mimeType); | |
216 } | |
217 | |
218 | |
219 class DartScriptRunner : public EventListener { | |
220 public: | |
221 static PassRefPtr<DartScriptRunner> create() | |
222 { | |
223 return adoptRef(new DartScriptRunner()); | |
224 } | |
225 | |
226 virtual void handleEvent(ScriptExecutionContext* context, Event*) | |
227 { | |
228 ASSERT(context->isDocument()); | |
229 Document* document = static_cast<Document*>(context); | |
230 | |
231 // this gets removed below, so protect it while handler runs. | |
232 RefPtr<DartScriptRunner> protect = this; | |
233 document->domWindow()->removeEventListener(String("DOMContentLoaded"), t
his, false); | |
234 | |
235 DartController::retrieve(context)->loadScripts(); | |
236 } | |
237 | |
238 virtual bool operator==(const EventListener& other) | |
239 { | |
240 return this == &other; | |
241 } | |
242 | |
243 private: | |
244 DartScriptRunner() | |
245 : EventListener(EventListener::NativeEventListenerType) | |
246 { | |
247 } | |
248 }; | |
249 | |
250 void DartController::evaluate(const ScriptSourceCode& sourceCode) | |
251 { | |
252 if (!m_scriptsLoaded) { | |
253 m_scriptsLoaded = true; | |
254 // FIXME: We defer loading Dart scripts until document is fully parsed t
o make | |
255 // sure DartController::loadScripts will find all the script tags in doc
ument. | |
256 // To make startup faster, we should start loading Dart scripts immediat
ely. | |
257 Document* document = m_frame->document(); | |
258 if (document->readyState() == "loading") | |
259 document->domWindow()->addEventListener(String("DOMContentLoaded"),
DartScriptRunner::create(), false); | |
260 else | |
261 loadScripts(); | |
262 } | |
263 } | |
264 | |
265 | |
266 void DartController::loadScripts() | |
267 { | |
268 initVMIfNeeded(); | |
269 | |
270 Vector<RefPtr<HTMLScriptElement> > dartScripts; | |
271 RefPtr<NodeList> scripts = m_frame->document()->getElementsByTagName("script
"); | |
272 for (unsigned i = 0; i < scripts->length(); ++i) { | |
273 Node* scriptNode = scripts->item(i); | |
274 HTMLScriptElement* scriptElement = static_cast<HTMLScriptElement*>(scrip
tNode); | |
275 String typeAttr = scriptElement->getAttribute(HTMLNames::typeAttr).strin
g(); | |
276 if (isDartMimeType(typeAttr.stripWhiteSpace().lower())) | |
277 dartScripts.append(scriptElement); | |
278 } | |
279 Document* document = frame()->document(); | |
280 for (size_t i = 0; i < dartScripts.size(); ++i) { | |
281 RefPtr<HTMLScriptElement> scriptElement = dartScripts.at(i); | |
282 | |
283 // FIXME: it may make sense to ensure that we'll never call evaluate mor
e than once for the same script tag. | |
284 DartPerScriptState* state = new DartPerScriptState(document, adoptRef(ne
w DartApplicationLoader(document))); | |
285 ASSERT(state->isolate()); | |
286 m_states.append(adoptPtr(state)); | |
287 | |
288 DartIsolateState::Scope scope(state->isolate()); | |
289 if (scriptElement->src().isEmpty()) | |
290 state->dartApplicationLoader()->load(m_frame->document()->url(), scr
iptElement->scriptContent()); | |
291 else | |
292 state->dartApplicationLoader()->loadScriptResource(scriptElement->sr
c()); | |
293 } | |
294 } | |
295 | |
296 void DartController::bindToWindowObject(Frame* frame, const String& key, NPObjec
t* object) | |
297 { | |
298 // FIXME: proper management of lifetime. | |
299 m_npObjectMap.set(key, object); | |
300 } | |
301 | |
302 NPObject* DartController::npObject(const String& key) | |
303 { | |
304 return m_npObjectMap.get(key); | |
305 } | |
306 | |
307 Dart_Handle DartController::callFunction(Dart_Handle function, int argc, Dart_Ha
ndle* argv) | |
308 { | |
309 // FIXME: Introduce Dart variant of V8GCController::checkMemoryUsage(); | |
310 const int kMaxRecursionDepth = 22; | |
311 | |
312 int* recursion = DartUtilities::recursionForCurrentIsolate(); | |
313 | |
314 if (*recursion >= kMaxRecursionDepth) | |
315 return Dart_Error("Maximum call stack size exceeded"); | |
316 | |
317 // FIXME: implement InspectorInstrumentationCookie stuff a la v8. | |
318 (*recursion)++; | |
319 Dart_Handle result = Dart_InvokeClosure(function, argc, argv); | |
320 (*recursion)--; | |
321 | |
322 // Release the storage mutex if applicable. | |
323 didLeaveScriptContext(*recursion); | |
324 | |
325 // Handle fatal error in Dart VM a la v8. | |
326 | |
327 return result; | |
328 } | |
329 | |
330 class StartIsolateAction : public ScheduledAction { | |
331 public: | |
332 StartIsolateAction(Dart_Isolate isolate, const String& mainLibURL, const Str
ing& entryPoint) | |
333 : ScheduledAction(v8::Handle<v8::Context>(), String()) | |
334 , m_isolate(isolate) | |
335 , m_mainLibURL(mainLibURL) | |
336 , m_entryPoint(entryPoint) | |
337 { } | |
338 | |
339 virtual void execute(ScriptExecutionContext* context) | |
340 { | |
341 DartIsolateState::Scope scope(m_isolate); | |
342 DartApiScope apiScope; | |
343 | |
344 Dart_Port mainPort = Dart_GetMainPortId(); | |
345 Dart_Handle port = Dart_GetReceivePort(mainPort); | |
346 if (Dart_IsError(port)) | |
347 DartUtilities::reportProblem(context, port); | |
348 | |
349 Dart_Handle mainLib = Dart_LookupLibrary(DartUtilities::stringToDartStri
ng(m_mainLibURL)); | |
350 if (Dart_IsError(mainLib)) | |
351 DartUtilities::reportProblem(context, mainLib); | |
352 | |
353 Dart_Handle result = Dart_InvokeStatic( | |
354 mainLib, | |
355 Dart_NewString(""), | |
356 DartUtilities::stringToDartString(m_entryPoint), | |
357 1, &port); | |
358 // FIXME: consider communicating exceptions back to parent isolate someh
ow. | |
359 if (Dart_IsError(result)) | |
360 DartUtilities::reportProblem(context, result); | |
361 } | |
362 | |
363 private: | |
364 Dart_Isolate m_isolate; | |
365 const String m_mainLibURL; | |
366 const String m_entryPoint; | |
367 }; | |
368 | |
369 Dart_Handle DartController::spawnDomIsolate(DOMWindow* targetWindow, const Strin
g& entryPoint) | |
370 { | |
371 Dart_Isolate parentIsolate = Dart_CurrentIsolate(); | |
372 | |
373 RefPtr<DartApplicationLoader> applicationLoader = isolateToDartApplicationLo
aderMap().get(parentIsolate); | |
374 ASSERT(applicationLoader); | |
375 | |
376 Dart_Isolate childIsolate; | |
377 Dart_Port childIsolatePort; | |
378 { | |
379 DartPerScriptState* state = new DartPerScriptState(targetWindow->documen
t(), applicationLoader); | |
380 m_states.append(adoptPtr(state)); // FIXME: should we add to m_states? S
hould navigation clear it? | |
381 childIsolate = state->isolate(); | |
382 | |
383 DartIsolateState::Scope scope(childIsolate); | |
384 DartApiScope apiScope; | |
385 applicationLoader->reinjectSources(); | |
386 childIsolatePort = Dart_GetMainPortId(); | |
387 } | |
388 | |
389 Dart_Handle sendPort = Dart_NewSendPort(childIsolatePort); | |
390 if (Dart_IsError(sendPort)) | |
391 return sendPort; | |
392 | |
393 OwnPtr<StartIsolateAction> action = adoptPtr(new StartIsolateAction(childIso
late, applicationLoader->mainLibraryURL(), entryPoint)); | |
394 DOMTimer::install(m_frame->document(), action.release(), 0, true); | |
395 | |
396 return sendPort; | |
397 } | |
398 | |
399 void DartController::didLeaveScriptContext(int recursion) | |
400 { | |
401 // FIXME: implement this. | |
402 } | |
403 | |
404 DartController* DartController::retrieve(Frame* frame) | |
405 { | |
406 if (!frame) | |
407 return 0; | |
408 return frame->script()->scriptControllerDelegate(); | |
409 } | |
410 | |
411 DartController* DartController::retrieve(ScriptExecutionContext* context) | |
412 { | |
413 if (!context || !context->isDocument()) | |
414 return 0; | |
415 return retrieve(static_cast<Document*>(context)->frame()); | |
416 } | |
417 | |
418 } | |
OLD | NEW |