/* * Copyright (C) 2009 Google Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following disclaimer * in the documentation and/or other materials provided with the * distribution. * * Neither the name of Google Inc. nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "config.h" #if ENABLE(WORKERS) #include "Worker.h" #include "ExceptionCode.h" #include "Frame.h" #include "V8Binding.h" #include "V8CustomBinding.h" #include "V8ObjectEventListener.h" #include "V8Proxy.h" #include "V8Utilities.h" #include "WorkerContext.h" #include "WorkerContextExecutionProxy.h" namespace WebCore { CALLBACK_FUNC_DECL(WorkerConstructor) { INC_STATS(L"DOM.Worker.Constructor"); if (!WorkerContextExecutionProxy::isWebWorkersEnabled()) { return throwError("Worker is not enabled.", V8Proxy::SYNTAX_ERROR); } if (!args.IsConstructCall()) { return throwError("DOM object constructor cannot be called as a function."); } if (args.Length() == 0) { return throwError("Not enough arguments", V8Proxy::SYNTAX_ERROR); } v8::TryCatch tryCatch; v8::Handle scriptUrl = args[0]->ToString(); if (tryCatch.HasCaught()) { return throwError(tryCatch.Exception()); } if (scriptUrl.IsEmpty()) return v8::Undefined(); // Get the script execution context. ScriptExecutionContext* context = 0; WorkerContextExecutionProxy* proxy = WorkerContextExecutionProxy::retrieve(); if (proxy) context = proxy->workerContext(); else { Frame* frame = V8Proxy::retrieveFrame(); if (!frame) return v8::Undefined(); context = frame->document(); } // Create the worker object. // Note: it's OK to let this RefPtr go out of scope because we also call SetDOMWrapper(), which effectively holds a reference to obj. ExceptionCode ec = 0; RefPtr obj = Worker::create(toWebCoreString(scriptUrl), context, ec); // Setup the standard wrapper object internal fields. v8::Handle wrapperObject = args.Holder(); V8Proxy::SetDOMWrapper(wrapperObject, V8ClassIndex::WORKER, obj.get()); obj->ref(); V8Proxy::SetJSWrapperForActiveDOMObject(obj.get(), v8::Persistent::New(wrapperObject)); return wrapperObject; } PassRefPtr getEventListener(Worker* worker, v8::Local value, bool findOnly) { if (worker->scriptExecutionContext()->isWorkerContext()) { WorkerContextExecutionProxy* workerContextProxy = WorkerContextExecutionProxy::retrieve(); ASSERT(workerContextProxy); return workerContextProxy->findOrCreateObjectEventListener(value, false, findOnly); } V8Proxy* proxy = V8Proxy::retrieve(worker->scriptExecutionContext()); if (proxy) return findOnly ? proxy->FindObjectEventListener(value, false) : proxy->FindOrCreateObjectEventListener(value, false); return 0; } ACCESSOR_GETTER(WorkerOnmessage) { INC_STATS(L"DOM.Worker.onmessage._get"); Worker* worker = V8Proxy::ToNativeObject(V8ClassIndex::WORKER, info.Holder()); if (worker->onmessage()) { V8ObjectEventListener* listener = static_cast(worker->onmessage()); v8::Local v8Listener = listener->getListenerObject(); return v8Listener; } return v8::Undefined(); } ACCESSOR_SETTER(WorkerOnmessage) { INC_STATS(L"DOM.Worker.onmessage._set"); Worker* worker = V8Proxy::ToNativeObject(V8ClassIndex::WORKER, info.Holder()); V8ObjectEventListener* oldListener = static_cast(worker->onmessage()); if (value->IsNull()) { if (oldListener) { v8::Local oldV8Listener = oldListener->getListenerObject(); removeHiddenDependency(info.Holder(), oldV8Listener, V8Custom::kWorkerRequestCacheIndex); } // Clear the listener. worker->setOnmessage(0); } else { RefPtr listener = getEventListener(worker, value, false); if (listener) { if (oldListener) { v8::Local oldV8Listener = oldListener->getListenerObject(); removeHiddenDependency(info.Holder(), oldV8Listener, V8Custom::kWorkerRequestCacheIndex); } worker->setOnmessage(listener); createHiddenDependency(info.Holder(), value, V8Custom::kWorkerRequestCacheIndex); } } } ACCESSOR_GETTER(WorkerOnerror) { INC_STATS(L"DOM.Worker.onerror._get"); Worker* worker = V8Proxy::ToNativeObject(V8ClassIndex::WORKER, info.Holder()); if (worker->onerror()) { V8ObjectEventListener* listener = static_cast(worker->onerror()); v8::Local v8Listener = listener->getListenerObject(); return v8Listener; } return v8::Undefined(); } ACCESSOR_SETTER(WorkerOnerror) { INC_STATS(L"DOM.Worker.onerror._set"); Worker* worker = V8Proxy::ToNativeObject(V8ClassIndex::WORKER, info.Holder()); V8ObjectEventListener* oldListener = static_cast(worker->onerror()); if (value->IsNull()) { if (oldListener) { v8::Local oldV8Listener = oldListener->getListenerObject(); removeHiddenDependency(info.Holder(), oldV8Listener, V8Custom::kWorkerRequestCacheIndex); } // Clear the listener. worker->setOnerror(0); } else { RefPtr listener = getEventListener(worker, value, false); if (listener) { if (oldListener) { v8::Local oldV8Listener = oldListener->getListenerObject(); removeHiddenDependency(info.Holder(), oldV8Listener, V8Custom::kWorkerRequestCacheIndex); } worker->setOnerror(listener); createHiddenDependency(info.Holder(), value, V8Custom::kWorkerRequestCacheIndex); } } } CALLBACK_FUNC_DECL(WorkerAddEventListener) { INC_STATS(L"DOM.Worker.addEventListener()"); Worker* worker = V8Proxy::ToNativeObject(V8ClassIndex::WORKER, args.Holder()); RefPtr listener = getEventListener(worker, args[1], false); if (listener) { String type = toWebCoreString(args[0]); bool useCapture = args[2]->BooleanValue(); worker->addEventListener(type, listener, useCapture); createHiddenDependency(args.Holder(), args[1], V8Custom::kWorkerRequestCacheIndex); } return v8::Undefined(); } CALLBACK_FUNC_DECL(WorkerRemoveEventListener) { INC_STATS(L"DOM.Worker.removeEventListener()"); Worker* worker = V8Proxy::ToNativeObject(V8ClassIndex::WORKER, args.Holder()); RefPtr listener = getEventListener(worker, args[1], true); if (listener) { String type = toWebCoreString(args[0]); bool useCapture = args[2]->BooleanValue(); worker->removeEventListener(type, listener.get(), useCapture); removeHiddenDependency(args.Holder(), args[1], V8Custom::kWorkerRequestCacheIndex); } return v8::Undefined(); } } // namespace WebCore #endif // ENABLE(WORKERS)