// Copyright (c) 2009 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "config.h" #include "base/compiler_specific.h" #include "GenericWorkerTask.h" #include "KURL.h" #include "ScriptExecutionContext.h" #include "SecurityOrigin.h" #include "SubstituteData.h" #include "WorkerContext.h" #include "WorkerThread.h" #include #include #undef LOG #include "base/logging.h" #include "webkit/api/public/WebScreenInfo.h" #include "webkit/api/public/WebString.h" #include "webkit/api/public/WebURL.h" #include "webkit/api/public/WebWorkerClient.h" #include "webkit/glue/glue_util.h" #include "webkit/glue/webdatasource_impl.h" #include "webkit/glue/webframe_impl.h" #include "webkit/glue/webpreferences.h" #include "webkit/glue/webview.h" #include "webkit/glue/webview_delegate.h" #include "webkit/glue/webworker_impl.h" using WebKit::WebWorker; using WebKit::WebWorkerClient; using WebKit::WebString; using WebKit::WebURL; #if ENABLE(WORKERS) // Dummy WebViewDelegate - we only need it in Worker process to load a // 'shadow page' which will initialize WebCore loader. class WorkerWebViewDelegate : public WebViewDelegate { public: WorkerWebViewDelegate() {} virtual void AddRef() { } virtual void Release() { } virtual void Blur(WebWidget *webwidget) { } virtual void CloseWidgetSoon(WebWidget *webwidget) { } virtual void DidInvalidateRect(WebWidget *webwidget, const WebKit::WebRect &rect) { } virtual void DidMove(WebWidget *webwidget, const WebPluginGeometry &move) { } virtual void DidScrollRect(WebWidget *webwidget, int dx, int dy, const WebKit::WebRect &clip_rect) { } virtual void Focus(WebWidget *webwidget) { } virtual gfx::NativeViewId GetContainingView(WebWidget *webwidget) { return gfx::NativeViewId(); } virtual void GetRootWindowRect(WebWidget *webwidget, WebKit::WebRect *rect) { } virtual void GetRootWindowResizerRect(WebWidget *webwidget, WebKit::WebRect *rect) { } virtual WebKit::WebScreenInfo GetScreenInfo(WebWidget *webwidget) { WebKit::WebScreenInfo info; return info; } virtual void GetWindowRect(WebWidget *webwidget, WebKit::WebRect *rect) { } virtual bool IsHidden(WebWidget *webwidget) { return true; } virtual void RunModal(WebWidget *webwidget) { } virtual void SetCursor(WebWidget *webwidget, const WebCursor &cursor) { } virtual void SetWindowRect(WebWidget *webwidget, const WebKit::WebRect &rect) { } virtual void Show(WebWidget *webwidget, WindowOpenDisposition disposition) { } virtual void ShowAsPopupWithItems(WebWidget *webwidget, const WebKit::WebRect &bounds, int item_height, int selected_index, const std::vector &items) { } // Tell the loader to load the data into the 'shadow page' synchronously, // so we can grab the resulting Document right after load. virtual void DidCreateDataSource(WebFrame* frame, WebKit::WebDataSource* ds) { static_cast(ds)->setDeferMainResourceDataLoad(false); } // Lazy allocate and leak this instance. static WorkerWebViewDelegate* worker_delegate() { static WorkerWebViewDelegate* worker_delegate = new WorkerWebViewDelegate(); return worker_delegate; } private: DISALLOW_COPY_AND_ASSIGN(WorkerWebViewDelegate); }; namespace WebKit { WebWorker* WebWorker::create(WebWorkerClient* client) { return new WebWorkerImpl(client); } } // This function is called on the main thread to force to initialize some static // values used in WebKit before any worker thread is started. This is because in // our worker processs, we do not run any WebKit code in main thread and thus // when multiple workers try to start at the same time, we might hit crash due // to contention for initializing static values. void InitializeWebKitStaticValues() { static bool initialized = false; if (!initialized) { initialized= true; // Note that we have to pass a URL with valid protocol in order to follow // the path to do static value initializations. WTF::RefPtr origin = WebCore::SecurityOrigin::create(WebCore::KURL("http://localhost")); origin.release(); } } WebWorkerImpl::WebWorkerImpl(WebWorkerClient* client) : client_(client), web_view_(NULL) { InitializeWebKitStaticValues(); } WebWorkerImpl::~WebWorkerImpl() { web_view_->Close(); } void WebWorkerImpl::PostMessageToWorkerContextTask( WebCore::ScriptExecutionContext* context, WebWorkerImpl* this_ptr, const WebCore::String& message) { DCHECK(context->isWorkerContext()); WebCore::WorkerContext* worker_context = static_cast(context); worker_context->dispatchMessage(message); this_ptr->confirmMessageFromWorkerObject( worker_context->hasPendingActivity()); } // WebWorker ------------------------------------------------------------------- void WebWorkerImpl::startWorkerContext(const WebURL& script_url, const WebString& user_agent, const WebString& source_code) { // Create 'shadow page'. This page is never displayed, it is used to proxy the // loading requests from the worker context to the rest of WebKit and Chromium // infrastructure. DCHECK(!web_view_); web_view_ = WebView::Create(WorkerWebViewDelegate::worker_delegate(), WebPreferences()); WebFrameImpl* web_frame = static_cast(web_view_->GetMainFrame()); // Construct substitute data source for the 'shadow page'. We only need it // to have same origin as the worker so the loading checks work correctly. WebCore::CString content(""); int len = static_cast(content.length()); RefPtr buf( WebCore::SharedBuffer::create(content.data(), len)); WebCore::SubstituteData subst_data(buf, WebCore::String("text/html"), WebCore::String("UTF-8"), WebCore::KURL()); WebCore::ResourceRequest request(webkit_glue::GURLToKURL(script_url), WebCore::CString()); web_frame->frame()->loader()->load(request, subst_data, false); // This document will be used as 'loading context' for the worker. loading_document_ = web_frame->frame()->document(); worker_thread_ = WebCore::WorkerThread::create( webkit_glue::WebURLToKURL(script_url), webkit_glue::WebStringToString(user_agent), webkit_glue::WebStringToString(source_code), *this, *this); // Worker initialization means a pending activity. reportPendingActivity(true); worker_thread_->start(); } void WebWorkerImpl::terminateWorkerContext() { if (worker_thread_) worker_thread_->stop(); } void WebWorkerImpl::postMessageToWorkerContext(const WebString& message) { worker_thread_->runLoop().postTask(WebCore::createCallbackTask( &PostMessageToWorkerContextTask, this, webkit_glue::WebStringToString(message))); } void WebWorkerImpl::workerObjectDestroyed() { } void WebWorkerImpl::DispatchTaskToMainThread( PassRefPtr task) { return WTF::callOnMainThread(InvokeTaskMethod, task.releaseRef()); } void WebWorkerImpl::InvokeTaskMethod(void* param) { WebCore::ScriptExecutionContext::Task* task = static_cast(param); task->performTask(NULL); task->deref(); } // WorkerObjectProxy ----------------------------------------------------------- void WebWorkerImpl::postMessageToWorkerObject(const WebCore::String& message) { DispatchTaskToMainThread(WebCore::createCallbackTask( &PostMessageTask, this, message)); } void WebWorkerImpl::PostMessageTask( WebCore::ScriptExecutionContext* context, WebWorkerImpl* this_ptr, WebCore::String message) { this_ptr->client_->postMessageToWorkerObject( webkit_glue::StringToWebString(message)); } void WebWorkerImpl::postExceptionToWorkerObject( const WebCore::String& error_message, int line_number, const WebCore::String& source_url) { DispatchTaskToMainThread(WebCore::createCallbackTask( &PostExceptionTask, this, error_message, line_number, source_url)); } void WebWorkerImpl::PostExceptionTask( WebCore::ScriptExecutionContext* context, WebWorkerImpl* this_ptr, const WebCore::String& error_message, int line_number, const WebCore::String& source_url) { this_ptr->client_->postExceptionToWorkerObject( webkit_glue::StringToWebString(error_message), line_number, webkit_glue::StringToWebString(source_url)); } void WebWorkerImpl::postConsoleMessageToWorkerObject( WebCore::MessageDestination destination, WebCore::MessageSource source, WebCore::MessageLevel level, const WebCore::String& message, int line_number, const WebCore::String& source_url) { DispatchTaskToMainThread(WebCore::createCallbackTask( &PostConsoleMessageTask, this, static_cast(destination), static_cast(source), static_cast(level), message, line_number, source_url)); } void WebWorkerImpl::PostConsoleMessageTask( WebCore::ScriptExecutionContext* context, WebWorkerImpl* this_ptr, int destination, int source, int level, const WebCore::String& message, int line_number, const WebCore::String& source_url) { this_ptr->client_->postConsoleMessageToWorkerObject( destination, source, level, webkit_glue::StringToWebString(message), line_number, webkit_glue::StringToWebString(source_url)); } void WebWorkerImpl::confirmMessageFromWorkerObject(bool has_pending_activity) { DispatchTaskToMainThread(WebCore::createCallbackTask( &ConfirmMessageTask, this, has_pending_activity)); } void WebWorkerImpl::ConfirmMessageTask( WebCore::ScriptExecutionContext* context, WebWorkerImpl* this_ptr, bool has_pending_activity) { this_ptr->client_->confirmMessageFromWorkerObject(has_pending_activity); } void WebWorkerImpl::reportPendingActivity(bool has_pending_activity) { DispatchTaskToMainThread(WebCore::createCallbackTask( &ReportPendingActivityTask, this, has_pending_activity)); } void WebWorkerImpl::ReportPendingActivityTask( WebCore::ScriptExecutionContext* context, WebWorkerImpl* this_ptr, bool has_pending_activity) { this_ptr->client_->reportPendingActivity(has_pending_activity); } void WebWorkerImpl::workerContextDestroyed() { DispatchTaskToMainThread(WebCore::createCallbackTask( &WorkerContextDestroyedTask, this)); } // WorkerLoaderProxy ----------------------------------------------------------- void WebWorkerImpl::postTaskToLoader( PassRefPtr task) { ASSERT(loading_document_->isDocument()); loading_document_->postTask(task); } void WebWorkerImpl::postTaskForModeToWorkerContext( PassRefPtr task, const WebCore::String& mode) { worker_thread_->runLoop().postTaskForMode(task, mode); } void WebWorkerImpl::WorkerContextDestroyedTask( WebCore::ScriptExecutionContext* context, WebWorkerImpl* this_ptr) { this_ptr->client_->workerContextDestroyed(); // The lifetime of this proxy is controlled by the worker context. delete this_ptr; } #else namespace WebKit { WebWorker* WebWorker::create(WebWorkerClient* client) { return NULL; } } #endif