/* * 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" #include "bindings/v8/V8DOMWrapper.h" #include "V8HTMLCollection.h" #include "V8HTMLDocument.h" #include "V8Window.h" #include "bindings/v8/V8Binding.h" #include "bindings/v8/V8HiddenPropertyName.h" #include "bindings/v8/V8ObjectConstructor.h" #include "bindings/v8/V8PerContextData.h" #include "bindings/v8/V8ScriptRunner.h" namespace WebCore { static v8::Local wrapInShadowTemplate(v8::Local wrapper, Node* impl, v8::Isolate* isolate) { // This is only for getting a unique pointer which we can pass to privateTemplate. static int shadowTemplateUniqueKey; WrapperWorldType currentWorldType = worldType(isolate); V8PerIsolateData* data = V8PerIsolateData::from(isolate); v8::Handle shadowTemplate = data->privateTemplateIfExists(currentWorldType, &shadowTemplateUniqueKey); if (shadowTemplate.IsEmpty()) { shadowTemplate = v8::FunctionTemplate::New(isolate); if (shadowTemplate.IsEmpty()) return v8::Local(); shadowTemplate->SetClassName(v8AtomicString(isolate, "HTMLDocument")); shadowTemplate->Inherit(V8HTMLDocument::domTemplate(isolate, currentWorldType)); shadowTemplate->InstanceTemplate()->SetInternalFieldCount(V8HTMLDocument::internalFieldCount); data->setPrivateTemplate(currentWorldType, &shadowTemplateUniqueKey, shadowTemplate); } v8::Local shadowConstructor = shadowTemplate->GetFunction(); if (shadowConstructor.IsEmpty()) return v8::Local(); v8::Local shadow = V8ScriptRunner::instantiateObject(shadowConstructor); if (shadow.IsEmpty()) return v8::Local(); shadow->SetPrototype(wrapper); V8DOMWrapper::setNativeInfo(wrapper, &V8HTMLDocument::wrapperTypeInfo, impl); return shadow; } v8::Local V8DOMWrapper::createWrapper(v8::Handle creationContext, const WrapperTypeInfo* type, void* impl, v8::Isolate* isolate) { V8WrapperInstantiationScope scope(creationContext, isolate); V8PerContextData* perContextData = V8PerContextData::from(scope.context()); v8::Local wrapper = perContextData ? perContextData->createWrapperFromCache(type) : V8ObjectConstructor::newInstance(type->domTemplate(isolate, worldTypeInMainThread(isolate))->GetFunction()); if (type == &V8HTMLDocument::wrapperTypeInfo && !wrapper.IsEmpty()) wrapper = wrapInShadowTemplate(wrapper, static_cast(impl), isolate); return wrapper; } static bool hasInternalField(v8::Handle value) { if (value.IsEmpty() || !value->IsObject()) return false; return v8::Handle::Cast(value)->InternalFieldCount(); } #ifndef NDEBUG bool V8DOMWrapper::maybeDOMWrapper(v8::Handle value) { if (!hasInternalField(value)) return false; v8::Handle object = v8::Handle::Cast(value); ASSERT(object->InternalFieldCount() >= v8DefaultWrapperInternalFieldCount); v8::HandleScope scope(v8::Isolate::GetCurrent()); ASSERT(object->GetAlignedPointerFromInternalField(v8DOMWrapperObjectIndex)); const WrapperTypeInfo* typeInfo = static_cast(object->GetAlignedPointerFromInternalField(v8DOMWrapperTypeIndex)); return typeInfo->ginEmbedder == gin::kEmbedderBlink; } #endif bool V8DOMWrapper::isDOMWrapper(v8::Handle value) { if (value.IsEmpty() || !value->IsObject()) return false; v8::Handle wrapper = v8::Handle::Cast(value); if (wrapper->InternalFieldCount() < v8DefaultWrapperInternalFieldCount) return false; ASSERT(wrapper->GetAlignedPointerFromInternalField(v8DOMWrapperObjectIndex)); ASSERT(wrapper->GetAlignedPointerFromInternalField(v8DOMWrapperTypeIndex)); const WrapperTypeInfo* typeInfo = static_cast(wrapper->GetAlignedPointerFromInternalField(v8DOMWrapperTypeIndex)); // FIXME: Add class id checks. return typeInfo->ginEmbedder == gin::kEmbedderBlink; } bool V8DOMWrapper::isWrapperOfType(v8::Handle value, const WrapperTypeInfo* type) { if (!hasInternalField(value)) return false; v8::Handle wrapper = v8::Handle::Cast(value); ASSERT(wrapper->InternalFieldCount() >= v8DefaultWrapperInternalFieldCount); ASSERT(wrapper->GetAlignedPointerFromInternalField(v8DOMWrapperObjectIndex)); const WrapperTypeInfo* typeInfo = static_cast(wrapper->GetAlignedPointerFromInternalField(v8DOMWrapperTypeIndex)); return typeInfo->ginEmbedder == gin::kEmbedderBlink && typeInfo == type; } } // namespace WebCore