blob: 5ce607cb867ce077cd97bf629fca7ec997d134ca [file] [log] [blame]
/*
* Copyright (C) 2010 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.
*/
#ifndef THIRD_PARTY_BLINK_RENDERER_PLATFORM_BINDINGS_WRAPPER_TYPE_INFO_H_
#define THIRD_PARTY_BLINK_RENDERER_PLATFORM_BINDINGS_WRAPPER_TYPE_INFO_H_
#include "gin/public/wrapper_info.h"
#include "third_party/blink/renderer/platform/bindings/v8_interface_bridge_base.h"
#include "third_party/blink/renderer/platform/heap/visitor.h"
#include "third_party/blink/renderer/platform/platform_export.h"
#include "third_party/blink/renderer/platform/wtf/allocator/allocator.h"
#include "third_party/blink/renderer/platform/wtf/assertions.h"
#include "v8/include/v8.h"
namespace blink {
class CustomWrappable;
class DOMWrapperWorld;
class ScriptWrappable;
static const int kV8DOMWrapperTypeIndex =
static_cast<int>(gin::kWrapperInfoIndex);
static const int kV8DOMWrapperObjectIndex =
static_cast<int>(gin::kEncodedValueIndex);
static const int kV8DefaultWrapperInternalFieldCount =
static_cast<int>(gin::kNumberOfInternalFields);
// The value of the following field isn't used (only its presence), hence no
// corresponding Index constant exists for it.
static const int kV8PrototypeInternalFieldcount = 1;
// This struct provides a way to store a bunch of information that is helpful
// when unwrapping v8 objects. Each v8 bindings class has exactly one static
// WrapperTypeInfo member, so comparing pointers is a safe way to determine if
// types match.
struct PLATFORM_EXPORT WrapperTypeInfo final {
DISALLOW_NEW();
enum WrapperTypePrototype {
kWrapperTypeObjectPrototype,
kWrapperTypeNoPrototype, // For legacy callback interface
};
enum WrapperClassId {
kNodeClassId = 1, // NodeClassId must be smaller than ObjectClassId.
kObjectClassId,
kCustomWrappableId,
};
enum ActiveScriptWrappableInheritance {
kNotInheritFromActiveScriptWrappable,
kInheritFromActiveScriptWrappable,
};
enum IdlDefinitionKind {
kIdlInterface,
kIdlNamespace,
kIdlCallbackInterface,
kCustomWrappableKind,
};
static const WrapperTypeInfo* Unwrap(v8::Local<v8::Value> type_info_wrapper) {
return reinterpret_cast<const WrapperTypeInfo*>(
v8::External::Cast(*type_info_wrapper)->Value());
}
bool Equals(const WrapperTypeInfo* that) const { return this == that; }
bool IsSubclass(const WrapperTypeInfo* that) const {
for (const WrapperTypeInfo* current = this; current;
current = current->parent_class) {
if (current == that)
return true;
}
return false;
}
void ConfigureWrapper(v8::TracedReference<v8::Object>* wrapper) const {
wrapper->SetWrapperClassId(wrapper_class_id);
}
// Returns a v8::Template of interface object, namespace object, or the
// counterpart of the IDL definition.
//
// - kIdlInterface: v8::FunctionTemplate of interface object
// - kIdlNamespace: v8::ObjectTemplate of namespace object
// - kIdlCallbackInterface: v8::FunctionTemplate of legacy callback interface
// object
// - kCustomWrappableKind: v8::FunctionTemplate
v8::Local<v8::Template> GetV8ClassTemplate(
v8::Isolate* isolate,
const DOMWrapperWorld& world) const;
void InstallConditionalFeatures(
v8::Local<v8::Context> context,
const DOMWrapperWorld& world,
v8::Local<v8::Object> instance_object,
v8::Local<v8::Object> prototype_object,
v8::Local<v8::Object> interface_object,
v8::Local<v8::Template> interface_template) const {
if (!install_context_dependent_props_func)
return;
install_context_dependent_props_func(
context, world, instance_object, prototype_object, interface_object,
interface_template, bindings::V8InterfaceBridgeBase::FeatureSelector());
}
bool IsActiveScriptWrappable() const {
return active_script_wrappable_inheritance ==
kInheritFromActiveScriptWrappable;
}
// Garbage collection support for when the type depends the WrapperTypeInfo
// object.
void Trace(Visitor*, const void*) const;
// This field must be the first member of the struct WrapperTypeInfo.
// See also static_assert() in .cpp file.
const gin::GinEmbedder gin_embedder;
bindings::V8InterfaceBridgeBase::InstallInterfaceTemplateFuncType
install_interface_template_func;
bindings::V8InterfaceBridgeBase::InstallContextDependentPropertiesFuncType
install_context_dependent_props_func;
const char* interface_name;
const WrapperTypeInfo* parent_class;
unsigned wrapper_type_prototype : 2; // WrapperTypePrototype
unsigned wrapper_class_id : 2; // WrapperClassId
unsigned // ActiveScriptWrappableInheritance
active_script_wrappable_inheritance : 1;
unsigned idl_definition_kind : 2; // IdlDefinitionKind
};
template <typename T, int offset>
inline T* GetInternalField(const v8::TracedReference<v8::Object>& global) {
DCHECK_LT(offset, v8::Object::InternalFieldCount(global));
return reinterpret_cast<T*>(
v8::Object::GetAlignedPointerFromInternalField(global, offset));
}
template <typename T, int offset>
inline T* GetInternalField(v8::Local<v8::Object> wrapper) {
DCHECK_LT(offset, wrapper->InternalFieldCount());
return reinterpret_cast<T*>(
wrapper->GetAlignedPointerFromInternalField(offset));
}
template <typename T, int offset>
inline T* GetInternalField(v8::Object* wrapper) {
DCHECK_LT(offset, wrapper->InternalFieldCount());
return reinterpret_cast<T*>(
wrapper->GetAlignedPointerFromInternalField(offset));
}
// The return value can be null if |wrapper| is a global proxy, which points to
// nothing while a navigation.
inline ScriptWrappable* ToScriptWrappable(
const v8::TracedReference<v8::Object>& wrapper) {
return GetInternalField<ScriptWrappable, kV8DOMWrapperObjectIndex>(wrapper);
}
inline ScriptWrappable* ToScriptWrappable(v8::Local<v8::Object> wrapper) {
return GetInternalField<ScriptWrappable, kV8DOMWrapperObjectIndex>(wrapper);
}
inline ScriptWrappable* ToScriptWrappable(v8::Object* wrapper) {
return GetInternalField<ScriptWrappable, kV8DOMWrapperObjectIndex>(wrapper);
}
inline CustomWrappable* ToCustomWrappable(v8::Local<v8::Object> wrapper) {
return GetInternalField<CustomWrappable, kV8DOMWrapperObjectIndex>(wrapper);
}
inline void* ToUntypedWrappable(
const v8::TracedReference<v8::Object>& wrapper) {
return GetInternalField<void, kV8DOMWrapperObjectIndex>(wrapper);
}
inline void* ToUntypedWrappable(v8::Local<v8::Object> wrapper) {
return GetInternalField<void, kV8DOMWrapperObjectIndex>(wrapper);
}
inline const WrapperTypeInfo* ToWrapperTypeInfo(
const v8::TracedReference<v8::Object>& wrapper) {
return GetInternalField<WrapperTypeInfo, kV8DOMWrapperTypeIndex>(wrapper);
}
inline const WrapperTypeInfo* ToWrapperTypeInfo(v8::Local<v8::Object> wrapper) {
return GetInternalField<WrapperTypeInfo, kV8DOMWrapperTypeIndex>(wrapper);
}
} // namespace blink
#endif // THIRD_PARTY_BLINK_RENDERER_PLATFORM_BINDINGS_WRAPPER_TYPE_INFO_H_