[V8] Unify the V8GCController visitors
[WebKit-https.git] / Source / WebCore / bindings / v8 / V8DOMWrapper.cpp
1 /*
2  * Copyright (C) 2009 Google Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are
6  * met:
7  *
8  *     * Redistributions of source code must retain the above copyright
9  * notice, this list of conditions and the following disclaimer.
10  *     * Redistributions in binary form must reproduce the above
11  * copyright notice, this list of conditions and the following disclaimer
12  * in the documentation and/or other materials provided with the
13  * distribution.
14  *     * Neither the name of Google Inc. nor the names of its
15  * contributors may be used to endorse or promote products derived from
16  * this software without specific prior written permission.
17  *
18  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29  */
30
31 #include "config.h"
32 #include "V8DOMWrapper.h"
33
34 #include <wtf/ArrayBufferView.h>
35 #include "DocumentLoader.h"
36 #include "EventTargetHeaders.h"
37 #include "EventTargetInterfaces.h"
38 #include "Frame.h"
39 #include "FrameLoaderClient.h"
40 #include "StylePropertySet.h"
41 #include "V8AbstractEventListener.h"
42 #include "V8Binding.h"
43 #include "V8Collection.h"
44 #include "V8EventListener.h"
45 #include "V8EventListenerList.h"
46 #include "V8HTMLCollection.h"
47 #include "V8HTMLDocument.h"
48 #include "V8HiddenPropertyName.h"
49 #include "V8Location.h"
50 #include "V8NamedNodeMap.h"
51 #include "V8NodeFilterCondition.h"
52 #include "V8NodeList.h"
53 #include "V8ObjectConstructor.h"
54 #include "V8PerContextData.h"
55 #include "V8StyleSheet.h"
56 #include "V8WorkerContextEventListener.h"
57 #include "WebGLContextAttributes.h"
58 #include "WebGLUniformLocation.h"
59 #include "WorkerContextExecutionProxy.h"
60 #include "WrapperTypeInfo.h"
61 #include <algorithm>
62 #include <utility>
63 #include <v8-debug.h>
64 #include <wtf/Assertions.h>
65 #include <wtf/OwnArrayPtr.h>
66 #include <wtf/StdLibExtras.h>
67 #include <wtf/UnusedParam.h>
68
69 namespace WebCore {
70
71 v8::Persistent<v8::Object> V8DOMWrapper::setJSWrapperForDOMNode(PassRefPtr<Node> node, v8::Handle<v8::Object> wrapper, v8::Isolate* isolate)
72 {
73     v8::Persistent<v8::Object> wrapperHandle = v8::Persistent<v8::Object>::New(wrapper);
74     ASSERT(maybeDOMWrapper(wrapperHandle));
75     wrapperHandle.SetWrapperClassId(v8DOMNodeClassId);
76     getDOMNodeMap(isolate).set(node.leakRef(), wrapperHandle);
77     return wrapperHandle;
78 }
79
80 void V8DOMWrapper::setNamedHiddenReference(v8::Handle<v8::Object> parent, const char* name, v8::Handle<v8::Value> child)
81 {
82     ASSERT(name);
83     parent->SetHiddenValue(V8HiddenPropertyName::hiddenReferenceName(name, strlen(name)), child);
84 }
85
86 WrapperTypeInfo* V8DOMWrapper::domWrapperType(v8::Handle<v8::Object> object)
87 {
88     ASSERT(V8DOMWrapper::maybeDOMWrapper(object));
89     return toWrapperTypeInfo(object);
90 }
91
92 PassRefPtr<NodeFilter> V8DOMWrapper::wrapNativeNodeFilter(v8::Handle<v8::Value> filter)
93 {
94     // A NodeFilter is used when walking through a DOM tree or iterating tree
95     // nodes.
96     // FIXME: we may want to cache NodeFilterCondition and NodeFilter
97     // object, but it is minor.
98     // NodeFilter is passed to NodeIterator that has a ref counted pointer
99     // to NodeFilter. NodeFilter has a ref counted pointer to NodeFilterCondition.
100     // In NodeFilterCondition, filter object is persisted in its constructor,
101     // and disposed in its destructor.
102     return NodeFilter::create(V8NodeFilterCondition::create(filter));
103 }
104
105 v8::Local<v8::Object> V8DOMWrapper::instantiateV8Object(Document* deprecatedDocument, WrapperTypeInfo* type, void* impl)
106 {
107     V8PerContextData* perContextData = 0;
108
109     // Please don't add any more uses of deprecatedDocument. We want to remove it.
110
111     // If we have a pointer to the frame, we cna get the V8PerContextData
112     // directly, which is faster than going through V8.
113     if (deprecatedDocument && deprecatedDocument->frame())
114         perContextData = perContextDataForCurrentWorld(deprecatedDocument->frame());
115     else
116         perContextData = V8PerContextData::from(v8::Context::GetCurrent());
117
118     v8::Local<v8::Object> instance = perContextData ? perContextData->createWrapperFromCache(type) : V8ObjectConstructor::newInstance(type->getTemplate()->GetFunction());
119
120     // Avoid setting the DOM wrapper for failed allocations.
121     if (instance.IsEmpty())
122         return instance;
123
124     setDOMWrapper(instance, type, impl);
125     if (type == &V8HTMLDocument::info)
126         instance = V8HTMLDocument::wrapInShadowObject(instance, static_cast<Node*>(impl));
127
128     return instance;
129 }
130
131 #ifndef NDEBUG
132 bool V8DOMWrapper::maybeDOMWrapper(v8::Handle<v8::Value> value)
133 {
134     if (value.IsEmpty() || !value->IsObject())
135         return false;
136
137     v8::Handle<v8::Object> object = v8::Handle<v8::Object>::Cast(value);
138     if (!object->InternalFieldCount())
139         return false;
140
141     ASSERT(object->InternalFieldCount() >= v8DefaultWrapperInternalFieldCount);
142
143     v8::HandleScope scope;
144     v8::Handle<v8::Value> wrapper = object->GetInternalField(v8DOMWrapperObjectIndex);
145     ASSERT(wrapper->IsNumber() || wrapper->IsExternal());
146
147     return true;
148 }
149 #endif
150
151 bool V8DOMWrapper::isValidDOMObject(v8::Handle<v8::Value> value)
152 {
153     if (value.IsEmpty() || !value->IsObject())
154         return false;
155     return v8::Handle<v8::Object>::Cast(value)->InternalFieldCount();
156 }
157
158 bool V8DOMWrapper::isWrapperOfType(v8::Handle<v8::Value> value, WrapperTypeInfo* type)
159 {
160     if (!isValidDOMObject(value))
161         return false;
162
163     v8::Handle<v8::Object> object = v8::Handle<v8::Object>::Cast(value);
164     ASSERT(object->InternalFieldCount() >= v8DefaultWrapperInternalFieldCount);
165
166     v8::Handle<v8::Value> wrapper = object->GetInternalField(v8DOMWrapperObjectIndex);
167     ASSERT_UNUSED(wrapper, wrapper->IsNumber() || wrapper->IsExternal());
168
169     WrapperTypeInfo* typeInfo = static_cast<WrapperTypeInfo*>(object->GetPointerFromInternalField(v8DOMWrapperTypeIndex));
170     return typeInfo == type;
171 }
172
173 #define TRY_TO_WRAP_WITH_INTERFACE(interfaceName) \
174     if (eventNames().interfaceFor##interfaceName == desiredInterface) \
175         return toV8(static_cast<interfaceName*>(target), creationContext, isolate);
176
177 // A JS object of type EventTarget is limited to a small number of possible classes.
178 v8::Handle<v8::Value> V8DOMWrapper::convertEventTargetToV8Object(EventTarget* target, v8::Handle<v8::Object> creationContext, v8::Isolate* isolate)
179 {
180     if (!target)
181         return v8NullWithCheck(isolate);
182
183     AtomicString desiredInterface = target->interfaceName();
184     DOM_EVENT_TARGET_INTERFACES_FOR_EACH(TRY_TO_WRAP_WITH_INTERFACE)
185
186     ASSERT_NOT_REACHED();
187     return v8Undefined();
188 }
189
190 PassRefPtr<EventListener> V8DOMWrapper::getEventListener(v8::Local<v8::Value> value, bool isAttribute, ListenerLookupType lookup)
191 {
192     v8::Handle<v8::Context> context = v8::Context::GetCurrent();
193     if (context.IsEmpty())
194         return 0;
195     if (lookup == ListenerFindOnly)
196         return V8EventListenerList::findWrapper(value, isAttribute);
197     if (isWrapperOfType(toInnerGlobalObject(context), &V8DOMWindow::info))
198         return V8EventListenerList::findOrCreateWrapper<V8EventListener>(value, isAttribute);
199 #if ENABLE(WORKERS)
200     return V8EventListenerList::findOrCreateWrapper<V8WorkerContextEventListener>(value, isAttribute);
201 #else
202     return 0;
203 #endif
204 }
205
206 }  // namespace WebCore