/* * 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 (!args.IsConstructCall()) return throwError("DOM object constructor cannot be called as a function."); if (!args.Length()) return throwError("Not enough arguments", V8Proxy::SyntaxError); 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::retrieveFrameForCurrentContext(); 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); if (ec) return throwError(ec); // Setup the standard wrapper object internal fields. v8::Handle wrapperObject = args.Holder(); V8DOMWrapper::setDOMWrapper(wrapperObject, V8ClassIndex::WORKER, obj.get()); obj->ref(); V8DOMWrapper::setJSWrapperForActiveDOMObject(obj.get(), v8::Persistent::New(wrapperObject)); return wrapperObject; } PassRefPtr getEventListener(Worker* worker, v8::Local value, bool isAttribute, bool findOnly) { if (worker->scriptExecutionContext()->isWorkerContext()) { WorkerContextExecutionProxy* workerContextProxy = WorkerContextExecutionProxy::retrieve(); ASSERT(workerContextProxy); return workerContextProxy->findOrCreateObjectEventListener(value, isAttribute, findOnly); } V8Proxy* proxy = V8Proxy::retrieve(worker->scriptExecutionContext()); if (proxy) { V8EventListenerList* list = proxy->objectListeners(); return findOnly ? list->findWrapper(value, isAttribute) : list->findOrCreateWrapper(proxy->frame(), value, isAttribute); } return 0; } ACCESSOR_GETTER(WorkerOnmessage) { INC_STATS(L"DOM.Worker.onmessage._get"); Worker* worker = V8DOMWrapper::convertToNativeObject(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 = V8DOMWrapper::convertToNativeObject(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, true, 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); } } } } // namespace WebCore #endif // ENABLE(WORKERS)