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 "WebBindings.h"
33
34 #include "V8Element.h"
35 #include "V8Range.h"
36 #include "WebArrayBufferView.h"
37 #include "WebElement.h"
38 #include "WebRange.h"
39 #include "bindings/v8/NPV8Object.h" // for PrivateIdentifier
40 #include "bindings/v8/ScriptController.h"
41 #include "bindings/v8/V8DOMWrapper.h"
42 #include "bindings/v8/V8NPObject.h"
43 #include "bindings/v8/V8NPUtils.h"
44 #include "bindings/v8/custom/V8ArrayBufferCustom.h"
45 #include "bindings/v8/custom/V8ArrayBufferViewCustom.h"
46 #include "bindings/v8/npruntime_impl.h"
47 #include "bindings/v8/npruntime_priv.h"
48 #include "core/dom/Range.h"
49 #include "core/frame/DOMWindow.h"
50 #include "core/frame/Frame.h"
51 #include "public/platform/WebArrayBuffer.h"
52 #include "wtf/ArrayBufferView.h"
53
54 using namespace WebCore;
55
56 namespace blink {
57
construct(NPP npp,NPObject * object,const NPVariant * args,uint32_t argCount,NPVariant * result)58 bool WebBindings::construct(NPP npp, NPObject* object, const NPVariant* args, uint32_t argCount, NPVariant* result)
59 {
60 return _NPN_Construct(npp, object, args, argCount, result);
61 }
62
createObject(NPP npp,NPClass * npClass)63 NPObject* WebBindings::createObject(NPP npp, NPClass* npClass)
64 {
65 return _NPN_CreateObject(npp, npClass);
66 }
67
enumerate(NPP npp,NPObject * object,NPIdentifier ** identifier,uint32_t * identifierCount)68 bool WebBindings::enumerate(NPP npp, NPObject* object, NPIdentifier** identifier, uint32_t* identifierCount)
69 {
70 return _NPN_Enumerate(npp, object, identifier, identifierCount);
71 }
72
evaluate(NPP npp,NPObject * object,NPString * script,NPVariant * result)73 bool WebBindings::evaluate(NPP npp, NPObject* object, NPString* script, NPVariant* result)
74 {
75 return _NPN_Evaluate(npp, object, script, result);
76 }
77
evaluateHelper(NPP npp,bool popupsAllowed,NPObject * object,NPString * script,NPVariant * result)78 bool WebBindings::evaluateHelper(NPP npp, bool popupsAllowed, NPObject* object, NPString* script, NPVariant* result)
79 {
80 return _NPN_EvaluateHelper(npp, popupsAllowed, object, script, result);
81 }
82
getIntIdentifier(int32_t number)83 NPIdentifier WebBindings::getIntIdentifier(int32_t number)
84 {
85 return _NPN_GetIntIdentifier(number);
86 }
87
getProperty(NPP npp,NPObject * object,NPIdentifier property,NPVariant * result)88 bool WebBindings::getProperty(NPP npp, NPObject* object, NPIdentifier property, NPVariant* result)
89 {
90 return _NPN_GetProperty(npp, object, property, result);
91 }
92
getStringIdentifier(const NPUTF8 * string)93 NPIdentifier WebBindings::getStringIdentifier(const NPUTF8* string)
94 {
95 return _NPN_GetStringIdentifier(string);
96 }
97
getStringIdentifiers(const NPUTF8 ** names,int32_t nameCount,NPIdentifier * identifiers)98 void WebBindings::getStringIdentifiers(const NPUTF8** names, int32_t nameCount, NPIdentifier* identifiers)
99 {
100 _NPN_GetStringIdentifiers(names, nameCount, identifiers);
101 }
102
hasMethod(NPP npp,NPObject * object,NPIdentifier method)103 bool WebBindings::hasMethod(NPP npp, NPObject* object, NPIdentifier method)
104 {
105 return _NPN_HasMethod(npp, object, method);
106 }
107
hasProperty(NPP npp,NPObject * object,NPIdentifier property)108 bool WebBindings::hasProperty(NPP npp, NPObject* object, NPIdentifier property)
109 {
110 return _NPN_HasProperty(npp, object, property);
111 }
112
identifierIsString(NPIdentifier identifier)113 bool WebBindings::identifierIsString(NPIdentifier identifier)
114 {
115 return _NPN_IdentifierIsString(identifier);
116 }
117
intFromIdentifier(NPIdentifier identifier)118 int32_t WebBindings::intFromIdentifier(NPIdentifier identifier)
119 {
120 return _NPN_IntFromIdentifier(identifier);
121 }
122
initializeVariantWithStringCopy(NPVariant * variant,const NPString * value)123 void WebBindings::initializeVariantWithStringCopy(NPVariant* variant, const NPString* value)
124 {
125 _NPN_InitializeVariantWithStringCopy(variant, value);
126 }
127
invoke(NPP npp,NPObject * object,NPIdentifier method,const NPVariant * args,uint32_t argCount,NPVariant * result)128 bool WebBindings::invoke(NPP npp, NPObject* object, NPIdentifier method, const NPVariant* args, uint32_t argCount, NPVariant* result)
129 {
130 return _NPN_Invoke(npp, object, method, args, argCount, result);
131 }
132
invokeDefault(NPP npp,NPObject * object,const NPVariant * args,uint32_t argCount,NPVariant * result)133 bool WebBindings::invokeDefault(NPP npp, NPObject* object, const NPVariant* args, uint32_t argCount, NPVariant* result)
134 {
135 return _NPN_InvokeDefault(npp, object, args, argCount, result);
136 }
137
releaseObject(NPObject * object)138 void WebBindings::releaseObject(NPObject* object)
139 {
140 return _NPN_ReleaseObject(object);
141 }
142
releaseVariantValue(NPVariant * variant)143 void WebBindings::releaseVariantValue(NPVariant* variant)
144 {
145 _NPN_ReleaseVariantValue(variant);
146 }
147
removeProperty(NPP npp,NPObject * object,NPIdentifier identifier)148 bool WebBindings::removeProperty(NPP npp, NPObject* object, NPIdentifier identifier)
149 {
150 return _NPN_RemoveProperty(npp, object, identifier);
151 }
152
retainObject(NPObject * object)153 NPObject* WebBindings::retainObject(NPObject* object)
154 {
155 return _NPN_RetainObject(object);
156 }
157
setException(NPObject * object,const NPUTF8 * message)158 void WebBindings::setException(NPObject* object, const NPUTF8* message)
159 {
160 _NPN_SetException(object, message);
161 }
162
setProperty(NPP npp,NPObject * object,NPIdentifier identifier,const NPVariant * value)163 bool WebBindings::setProperty(NPP npp, NPObject* object, NPIdentifier identifier, const NPVariant* value)
164 {
165 return _NPN_SetProperty(npp, object, identifier, value);
166 }
167
registerObjectOwner(NPP)168 void WebBindings::registerObjectOwner(NPP)
169 {
170 }
171
unregisterObjectOwner(NPP)172 void WebBindings::unregisterObjectOwner(NPP)
173 {
174 }
175
getObjectOwner(NPObject *)176 NPP WebBindings::getObjectOwner(NPObject*)
177 {
178 return 0;
179 }
180
unregisterObject(NPObject * object)181 void WebBindings::unregisterObject(NPObject* object)
182 {
183 _NPN_UnregisterObject(object);
184 }
185
dropV8WrapperForObject(NPObject * object)186 void WebBindings::dropV8WrapperForObject(NPObject* object)
187 {
188 WebCore::dropV8WrapperForNPObject(object);
189 }
190
utf8FromIdentifier(NPIdentifier identifier)191 NPUTF8* WebBindings::utf8FromIdentifier(NPIdentifier identifier)
192 {
193 return _NPN_UTF8FromIdentifier(identifier);
194 }
195
extractIdentifierData(const NPIdentifier & identifier,const NPUTF8 * & string,int32_t & number,bool & isString)196 void WebBindings::extractIdentifierData(const NPIdentifier& identifier, const NPUTF8*& string, int32_t& number, bool& isString)
197 {
198 PrivateIdentifier* data = static_cast<PrivateIdentifier*>(identifier);
199 if (!data) {
200 isString = false;
201 number = 0;
202 return;
203 }
204
205 isString = data->isString;
206 if (isString)
207 string = data->value.string;
208 else
209 number = data->value.number;
210 }
211
getRangeImpl(NPObject * object,WebRange * webRange,v8::Isolate * isolate)212 static bool getRangeImpl(NPObject* object, WebRange* webRange, v8::Isolate* isolate)
213 {
214 if (!object)
215 return false;
216
217 V8NPObject* v8NPObject = npObjectToV8NPObject(object);
218 if (!v8NPObject)
219 return false;
220
221 v8::HandleScope handleScope(isolate);
222 v8::Handle<v8::Object> v8Object = v8::Local<v8::Object>::New(isolate, v8NPObject->v8Object);
223 if (v8Object.IsEmpty())
224 return false;
225 if (!V8Range::wrapperTypeInfo.equals(toWrapperTypeInfo(v8Object)))
226 return false;
227
228 Range* native = V8Range::hasInstanceInAnyWorld(v8Object, isolate) ? V8Range::toNative(v8Object) : 0;
229 if (!native)
230 return false;
231
232 *webRange = WebRange(native);
233 return true;
234 }
235
getNodeImpl(NPObject * object,WebNode * webNode,v8::Isolate * isolate)236 static bool getNodeImpl(NPObject* object, WebNode* webNode, v8::Isolate* isolate)
237 {
238 if (!object)
239 return false;
240
241 V8NPObject* v8NPObject = npObjectToV8NPObject(object);
242 if (!v8NPObject)
243 return false;
244
245 v8::HandleScope handleScope(isolate);
246 v8::Handle<v8::Object> v8Object = v8::Local<v8::Object>::New(isolate, v8NPObject->v8Object);
247 if (v8Object.IsEmpty())
248 return false;
249 Node* native = V8Node::hasInstanceInAnyWorld(v8Object, isolate) ? V8Node::toNative(v8Object) : 0;
250 if (!native)
251 return false;
252
253 *webNode = WebNode(native);
254 return true;
255 }
256
getElementImpl(NPObject * object,WebElement * webElement,v8::Isolate * isolate)257 static bool getElementImpl(NPObject* object, WebElement* webElement, v8::Isolate* isolate)
258 {
259 if (!object)
260 return false;
261
262 V8NPObject* v8NPObject = npObjectToV8NPObject(object);
263 if (!v8NPObject)
264 return false;
265
266 v8::HandleScope handleScope(isolate);
267 v8::Handle<v8::Object> v8Object = v8::Local<v8::Object>::New(isolate, v8NPObject->v8Object);
268 if (v8Object.IsEmpty())
269 return false;
270 Element* native = V8Element::hasInstanceInAnyWorld(v8Object, isolate) ? V8Element::toNative(v8Object) : 0;
271 if (!native)
272 return false;
273
274 *webElement = WebElement(native);
275 return true;
276 }
277
getArrayBufferImpl(NPObject * object,WebArrayBuffer * arrayBuffer,v8::Isolate * isolate)278 static bool getArrayBufferImpl(NPObject* object, WebArrayBuffer* arrayBuffer, v8::Isolate* isolate)
279 {
280 if (!object)
281 return false;
282
283 V8NPObject* v8NPObject = npObjectToV8NPObject(object);
284 if (!v8NPObject)
285 return false;
286
287 v8::HandleScope handleScope(isolate);
288 v8::Handle<v8::Object> v8Object = v8::Local<v8::Object>::New(isolate, v8NPObject->v8Object);
289 if (v8Object.IsEmpty())
290 return false;
291 ArrayBuffer* native = V8ArrayBuffer::hasInstanceInAnyWorld(v8Object, isolate) ? V8ArrayBuffer::toNative(v8Object) : 0;
292 if (!native)
293 return false;
294
295 *arrayBuffer = WebArrayBuffer(native);
296 return true;
297 }
298
getArrayBufferViewImpl(NPObject * object,WebArrayBufferView * arrayBufferView,v8::Isolate * isolate)299 static bool getArrayBufferViewImpl(NPObject* object, WebArrayBufferView* arrayBufferView, v8::Isolate* isolate)
300 {
301 if (!object)
302 return false;
303
304 V8NPObject* v8NPObject = npObjectToV8NPObject(object);
305 if (!v8NPObject)
306 return false;
307
308 v8::HandleScope handleScope(isolate);
309 v8::Handle<v8::Object> v8Object = v8::Local<v8::Object>::New(isolate, v8NPObject->v8Object);
310 if (v8Object.IsEmpty())
311 return false;
312 ArrayBufferView* native = V8ArrayBufferView::hasInstanceInAnyWorld(v8Object, isolate) ? V8ArrayBufferView::toNative(v8Object) : 0;
313 if (!native)
314 return false;
315
316 *arrayBufferView = WebArrayBufferView(native);
317 return true;
318 }
319
makeIntArrayImpl(const WebVector<int> & data,v8::Isolate * isolate)320 static NPObject* makeIntArrayImpl(const WebVector<int>& data, v8::Isolate* isolate)
321 {
322 v8::HandleScope handleScope(isolate);
323 v8::Handle<v8::Array> result = v8::Array::New(isolate, data.size());
324 for (size_t i = 0; i < data.size(); ++i)
325 result->Set(i, v8::Number::New(isolate, data[i]));
326
327 DOMWindow* window = toDOMWindow(isolate->GetCurrentContext());
328 return npCreateV8ScriptObject(0, result, window, isolate);
329 }
330
makeStringArrayImpl(const WebVector<WebString> & data,v8::Isolate * isolate)331 static NPObject* makeStringArrayImpl(const WebVector<WebString>& data, v8::Isolate* isolate)
332 {
333 v8::HandleScope handleScope(isolate);
334 v8::Handle<v8::Array> result = v8::Array::New(isolate, data.size());
335 for (size_t i = 0; i < data.size(); ++i)
336 result->Set(i, v8String(isolate, data[i]));
337
338 DOMWindow* window = toDOMWindow(isolate->GetCurrentContext());
339 return npCreateV8ScriptObject(0, result, window, isolate);
340 }
341
getRange(NPObject * range,WebRange * webRange)342 bool WebBindings::getRange(NPObject* range, WebRange* webRange)
343 {
344 return getRangeImpl(range, webRange, v8::Isolate::GetCurrent());
345 }
346
getArrayBuffer(NPObject * arrayBuffer,WebArrayBuffer * webArrayBuffer)347 bool WebBindings::getArrayBuffer(NPObject* arrayBuffer, WebArrayBuffer* webArrayBuffer)
348 {
349 return getArrayBufferImpl(arrayBuffer, webArrayBuffer, v8::Isolate::GetCurrent());
350 }
351
getArrayBufferView(NPObject * arrayBufferView,WebArrayBufferView * webArrayBufferView)352 bool WebBindings::getArrayBufferView(NPObject* arrayBufferView, WebArrayBufferView* webArrayBufferView)
353 {
354 return getArrayBufferViewImpl(arrayBufferView, webArrayBufferView, v8::Isolate::GetCurrent());
355 }
356
getNode(NPObject * node,WebNode * webNode)357 bool WebBindings::getNode(NPObject* node, WebNode* webNode)
358 {
359 return getNodeImpl(node, webNode, v8::Isolate::GetCurrent());
360 }
361
getElement(NPObject * element,WebElement * webElement)362 bool WebBindings::getElement(NPObject* element, WebElement* webElement)
363 {
364 return getElementImpl(element, webElement, v8::Isolate::GetCurrent());
365 }
366
makeIntArray(const WebVector<int> & data)367 NPObject* WebBindings::makeIntArray(const WebVector<int>& data)
368 {
369 return makeIntArrayImpl(data, v8::Isolate::GetCurrent());
370 }
371
makeStringArray(const WebVector<WebString> & data)372 NPObject* WebBindings::makeStringArray(const WebVector<WebString>& data)
373 {
374 return makeStringArrayImpl(data, v8::Isolate::GetCurrent());
375 }
376
pushExceptionHandler(ExceptionHandler handler,void * data)377 void WebBindings::pushExceptionHandler(ExceptionHandler handler, void* data)
378 {
379 WebCore::pushExceptionHandler(handler, data);
380 }
381
popExceptionHandler()382 void WebBindings::popExceptionHandler()
383 {
384 WebCore::popExceptionHandler();
385 }
386
toNPVariant(v8::Local<v8::Value> object,NPObject * root,NPVariant * result)387 void WebBindings::toNPVariant(v8::Local<v8::Value> object, NPObject* root, NPVariant* result)
388 {
389 WebCore::convertV8ObjectToNPVariant(object, root, result, v8::Isolate::GetCurrent());
390 }
391
toV8Value(const NPVariant * variant)392 v8::Handle<v8::Value> WebBindings::toV8Value(const NPVariant* variant)
393 {
394 v8::Isolate* isolate = v8::Isolate::GetCurrent();
395 if (variant->type == NPVariantType_Object) {
396 NPObject* object = NPVARIANT_TO_OBJECT(*variant);
397 V8NPObject* v8Object = npObjectToV8NPObject(object);
398 if (!v8Object)
399 return v8::Undefined(isolate);
400 return convertNPVariantToV8Object(variant, v8Object->rootObject->frame()->script().windowScriptNPObject(), isolate);
401 }
402 // Safe to pass 0 since we have checked the script object class to make sure the
403 // argument is a primitive v8 type.
404 return convertNPVariantToV8Object(variant, 0, isolate);
405 }
406
407 } // namespace blink
408