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 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 63 NPObject* WebBindings::createObject(NPP npp, NPClass* npClass) 64 { 65 return _NPN_CreateObject(npp, npClass); 66 } 67 68 bool WebBindings::enumerate(NPP npp, NPObject* object, NPIdentifier** identifier, uint32_t* identifierCount) 69 { 70 return _NPN_Enumerate(npp, object, identifier, identifierCount); 71 } 72 73 bool WebBindings::evaluate(NPP npp, NPObject* object, NPString* script, NPVariant* result) 74 { 75 return _NPN_Evaluate(npp, object, script, result); 76 } 77 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 83 NPIdentifier WebBindings::getIntIdentifier(int32_t number) 84 { 85 return _NPN_GetIntIdentifier(number); 86 } 87 88 bool WebBindings::getProperty(NPP npp, NPObject* object, NPIdentifier property, NPVariant* result) 89 { 90 return _NPN_GetProperty(npp, object, property, result); 91 } 92 93 NPIdentifier WebBindings::getStringIdentifier(const NPUTF8* string) 94 { 95 return _NPN_GetStringIdentifier(string); 96 } 97 98 void WebBindings::getStringIdentifiers(const NPUTF8** names, int32_t nameCount, NPIdentifier* identifiers) 99 { 100 _NPN_GetStringIdentifiers(names, nameCount, identifiers); 101 } 102 103 bool WebBindings::hasMethod(NPP npp, NPObject* object, NPIdentifier method) 104 { 105 return _NPN_HasMethod(npp, object, method); 106 } 107 108 bool WebBindings::hasProperty(NPP npp, NPObject* object, NPIdentifier property) 109 { 110 return _NPN_HasProperty(npp, object, property); 111 } 112 113 bool WebBindings::identifierIsString(NPIdentifier identifier) 114 { 115 return _NPN_IdentifierIsString(identifier); 116 } 117 118 int32_t WebBindings::intFromIdentifier(NPIdentifier identifier) 119 { 120 return _NPN_IntFromIdentifier(identifier); 121 } 122 123 void WebBindings::initializeVariantWithStringCopy(NPVariant* variant, const NPString* value) 124 { 125 _NPN_InitializeVariantWithStringCopy(variant, value); 126 } 127 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 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 138 void WebBindings::releaseObject(NPObject* object) 139 { 140 return _NPN_ReleaseObject(object); 141 } 142 143 void WebBindings::releaseVariantValue(NPVariant* variant) 144 { 145 _NPN_ReleaseVariantValue(variant); 146 } 147 148 bool WebBindings::removeProperty(NPP npp, NPObject* object, NPIdentifier identifier) 149 { 150 return _NPN_RemoveProperty(npp, object, identifier); 151 } 152 153 NPObject* WebBindings::retainObject(NPObject* object) 154 { 155 return _NPN_RetainObject(object); 156 } 157 158 void WebBindings::setException(NPObject* object, const NPUTF8* message) 159 { 160 _NPN_SetException(object, message); 161 } 162 163 bool WebBindings::setProperty(NPP npp, NPObject* object, NPIdentifier identifier, const NPVariant* value) 164 { 165 return _NPN_SetProperty(npp, object, identifier, value); 166 } 167 168 void WebBindings::registerObjectOwner(NPP) 169 { 170 } 171 172 void WebBindings::unregisterObjectOwner(NPP) 173 { 174 } 175 176 NPP WebBindings::getObjectOwner(NPObject*) 177 { 178 return 0; 179 } 180 181 void WebBindings::unregisterObject(NPObject* object) 182 { 183 _NPN_UnregisterObject(object); 184 } 185 186 void WebBindings::dropV8WrapperForObject(NPObject* object) 187 { 188 WebCore::dropV8WrapperForNPObject(object); 189 } 190 191 NPUTF8* WebBindings::utf8FromIdentifier(NPIdentifier identifier) 192 { 193 return _NPN_UTF8FromIdentifier(identifier); 194 } 195 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 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 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 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 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 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 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 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 342 bool WebBindings::getRange(NPObject* range, WebRange* webRange) 343 { 344 return getRangeImpl(range, webRange, v8::Isolate::GetCurrent()); 345 } 346 347 bool WebBindings::getArrayBuffer(NPObject* arrayBuffer, WebArrayBuffer* webArrayBuffer) 348 { 349 return getArrayBufferImpl(arrayBuffer, webArrayBuffer, v8::Isolate::GetCurrent()); 350 } 351 352 bool WebBindings::getArrayBufferView(NPObject* arrayBufferView, WebArrayBufferView* webArrayBufferView) 353 { 354 return getArrayBufferViewImpl(arrayBufferView, webArrayBufferView, v8::Isolate::GetCurrent()); 355 } 356 357 bool WebBindings::getNode(NPObject* node, WebNode* webNode) 358 { 359 return getNodeImpl(node, webNode, v8::Isolate::GetCurrent()); 360 } 361 362 bool WebBindings::getElement(NPObject* element, WebElement* webElement) 363 { 364 return getElementImpl(element, webElement, v8::Isolate::GetCurrent()); 365 } 366 367 NPObject* WebBindings::makeIntArray(const WebVector<int>& data) 368 { 369 return makeIntArrayImpl(data, v8::Isolate::GetCurrent()); 370 } 371 372 NPObject* WebBindings::makeStringArray(const WebVector<WebString>& data) 373 { 374 return makeStringArrayImpl(data, v8::Isolate::GetCurrent()); 375 } 376 377 void WebBindings::pushExceptionHandler(ExceptionHandler handler, void* data) 378 { 379 WebCore::pushExceptionHandler(handler, data); 380 } 381 382 void WebBindings::popExceptionHandler() 383 { 384 WebCore::popExceptionHandler(); 385 } 386 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 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