 Chromium Code Reviews
 Chromium Code Reviews Issue 9114050:
  Add primitive WebGL ArrayBuffer() support to d8  (Closed) 
  Base URL: https://v8.googlecode.com/svn/branches/bleeding_edge
    
  
    Issue 9114050:
  Add primitive WebGL ArrayBuffer() support to d8  (Closed) 
  Base URL: https://v8.googlecode.com/svn/branches/bleeding_edge| OLD | NEW | 
|---|---|
| 1 // Copyright 2011 the V8 project authors. All rights reserved. | 1 // Copyright 2012 the V8 project authors. All rights reserved. | 
| 2 // Redistribution and use in source and binary forms, with or without | 2 // Redistribution and use in source and binary forms, with or without | 
| 3 // modification, are permitted provided that the following conditions are | 3 // modification, are permitted provided that the following conditions are | 
| 4 // met: | 4 // met: | 
| 5 // | 5 // | 
| 6 // * Redistributions of source code must retain the above copyright | 6 // * Redistributions of source code must retain the above copyright | 
| 7 // notice, this list of conditions and the following disclaimer. | 7 // notice, this list of conditions and the following disclaimer. | 
| 8 // * Redistributions in binary form must reproduce the above | 8 // * Redistributions in binary form must reproduce the above | 
| 9 // copyright notice, this list of conditions and the following | 9 // copyright notice, this list of conditions and the following | 
| 10 // disclaimer in the documentation and/or other materials provided | 10 // disclaimer in the documentation and/or other materials provided | 
| 11 // with the distribution. | 11 // with the distribution. | 
| (...skipping 262 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 274 if (source.IsEmpty()) { | 274 if (source.IsEmpty()) { | 
| 275 return ThrowException(String::New("Error loading file")); | 275 return ThrowException(String::New("Error loading file")); | 
| 276 } | 276 } | 
| 277 if (!ExecuteString(source, String::New(*file), false, true)) { | 277 if (!ExecuteString(source, String::New(*file), false, true)) { | 
| 278 return ThrowException(String::New("Error executing file")); | 278 return ThrowException(String::New("Error executing file")); | 
| 279 } | 279 } | 
| 280 } | 280 } | 
| 281 return Undefined(); | 281 return Undefined(); | 
| 282 } | 282 } | 
| 283 | 283 | 
| 284 static size_t convertToUint(Local<Value> value_in, TryCatch* try_catch) { | |
| 285 if (value_in->IsUint32()) { | |
| 286 return value_in->Uint32Value(); | |
| 287 } | |
| 288 | |
| 289 Local<Value> number = value_in->ToNumber(); | |
| 290 if (try_catch->HasCaught()) return 0; | |
| 291 | |
| 292 ASSERT(number->IsNumber()); | |
| 293 Local<Int32> int32 = number->ToInt32(); | |
| 294 if (try_catch->HasCaught() || int32.IsEmpty()) return 0; | |
| 295 | |
| 296 int32_t raw_value = int32->Int32Value(); | |
| 297 if (try_catch->HasCaught()) return 0; | |
| 298 | |
| 299 if (raw_value < 0) { | |
| 300 ThrowException(String::New("Array length must not be negative.")); | |
| 301 return 0; | |
| 302 } | |
| 303 | |
| 304 static const int kMaxLength = 0x3fffffff; | |
| 305 #ifndef V8_SHARED | |
| 306 ASSERT(kMaxLength == i::ExternalArray::kMaxLength); | |
| 307 #endif // V8_SHARED | |
| 308 if (raw_value > static_cast<int32_t>(kMaxLength)) { | |
| 309 ThrowException( | |
| 310 String::New("Array length exceeds maximum length.")); | |
| 311 } | |
| 312 return static_cast<size_t>(raw_value); | |
| 313 } | |
| 314 | |
| 315 | |
| 316 const char kArrayBufferReferencePropName[] = "_is_array_buffer_"; | |
| 317 const char kArrayBufferMarkerPropName[] = "_array_buffer_ref_"; | |
| 318 | |
| 284 | 319 | 
| 285 Handle<Value> Shell::CreateExternalArray(const Arguments& args, | 320 Handle<Value> Shell::CreateExternalArray(const Arguments& args, | 
| 286 ExternalArrayType type, | 321 ExternalArrayType type, | 
| 287 size_t element_size) { | 322 size_t element_size) { | 
| 323 TryCatch try_catch; | |
| 324 bool is_array_buffer_construct = element_size == 0; | |
| 325 if (is_array_buffer_construct) { | |
| 326 type = v8::kExternalByteArray; | |
| 327 element_size = 1; | |
| 328 } | |
| 288 ASSERT(element_size == 1 || element_size == 2 || element_size == 4 || | 329 ASSERT(element_size == 1 || element_size == 2 || element_size == 4 || | 
| 289 element_size == 8); | 330 element_size == 8); | 
| 290 if (args.Length() != 1) { | 331 if (args.Length() == 0) { | 
| 291 return ThrowException( | 332 return ThrowException( | 
| 292 String::New("Array constructor needs one parameter.")); | 333 String::New("Array constructor must have at least one " | 
| 334 "parameter.")); | |
| 293 } | 335 } | 
| 294 static const int kMaxLength = 0x3fffffff; | 336 bool first_arg_is_array_buffer = | 
| 295 #ifndef V8_SHARED | 337 args[0]->IsObject() && | 
| 296 ASSERT(kMaxLength == i::ExternalArray::kMaxLength); | 338 args[0]->ToObject()->Get( | 
| 297 #endif // V8_SHARED | 339 String::New(kArrayBufferMarkerPropName))->IsTrue(); | 
| 298 size_t length = 0; | 340 // Currently, only the following constructors are supported: | 
| 299 TryCatch try_catch; | 341 // TypedArray(unsigned long length) | 
| 300 if (args[0]->IsUint32()) { | 342 // TypedArray(ArrayBuffer buffer, | 
| 301 length = args[0]->Uint32Value(); | 343 // optional unsigned long byteOffset, | 
| 302 } else { | 344 // optional unsigned long length) | 
| 303 Local<Number> number = args[0]->ToNumber(); | 345 if (args.Length() > 3) { | 
| 304 if (number.IsEmpty()) { | 346 return ThrowException( | 
| 305 ASSERT(try_catch.HasCaught()); | 347 String::New("Array constructor from ArrayBuffer must " | 
| 306 return try_catch.Exception(); | 348 "have 1-3 parameters.")); | 
| 349 } | |
| 350 | |
| 351 Local<Value> length_value = (args.Length() < 3) | |
| 352 ? (first_arg_is_array_buffer | |
| 353 ? args[0]->ToObject()->Get(String::New("length")) | |
| 354 : args[0]) | |
| 355 : args[2]; | |
| 356 size_t length = convertToUint(length_value, &try_catch); | |
| 357 if (try_catch.HasCaught()) return try_catch.Exception(); | |
| 358 | |
| 359 void* data = NULL; | |
| 360 size_t offset = 0; | |
| 361 | |
| 362 Handle<Object> array = Object::New(); | |
| 363 if (first_arg_is_array_buffer) { | |
| 364 Handle<Object> derived_from = args[0]->ToObject(); | |
| 365 data = derived_from->GetIndexedPropertiesExternalArrayData(); | |
| 366 | |
| 367 size_t array_buffer_length = convertToUint( | |
| 368 derived_from->Get(String::New("length")), | |
| 369 &try_catch); | |
| 370 if (try_catch.HasCaught()) return try_catch.Exception(); | |
| 371 | |
| 372 if (data == NULL && array_buffer_length != 0) { | |
| 373 return ThrowException( | |
| 374 String::New("ArrayBuffer doesn't have data")); | |
| 307 } | 375 } | 
| 308 ASSERT(number->IsNumber()); | 376 | 
| 309 Local<Int32> int32 = number->ToInt32(); | 377 if (args.Length() > 1) { | 
| 310 if (int32.IsEmpty()) { | 378 offset = convertToUint(args[1], &try_catch); | 
| 311 if (try_catch.HasCaught()) { | 379 if (try_catch.HasCaught()) return try_catch.Exception(); | 
| 312 return try_catch.Exception(); | 380 | 
| 381 // The given byteOffset must be a multiple of the element size of the | |
| 382 // specific type, otherwise an exception is raised. | |
| 383 if (offset % element_size != 0) { | |
| 384 return ThrowException( | |
| 385 String::New("offset must be multiple of element_size")); | |
| 313 } | 386 } | 
| 314 } | 387 } | 
| 315 int32_t raw_length = int32->Int32Value(); | 388 | 
| 316 if (try_catch.HasCaught()) { | 389 if (args.Length() == 2) { | 
| 317 return try_catch.Exception(); | 390 // If length is not explicitly specified, the length of the ArrayBuffer | 
| 391 // minus the byteOffset must be a multiple of the element size of the | |
| 392 // specific type, or an exception is raised. | |
| 393 length = array_buffer_length - offset; | |
| 318 } | 394 } | 
| 319 if (raw_length < 0) { | 395 | 
| 320 return ThrowException(String::New("Array length must not be negative.")); | 396 if (args.Length() == 3) { | 
| 397 // If a given byteOffset and length references an area beyond the end of | |
| 398 // the ArrayBuffer an exception is raised. | |
| 399 if (offset + (length * element_size) > array_buffer_length) { | |
| 
Jakob Kummerow
2012/01/11 13:54:27
You need an equivalent check (offset > array_buffe
 
danno
2012/01/11 15:10:20
Done.
 | |
| 400 return ThrowException( | |
| 401 String::New("length references an area beyond the end of the " | |
| 402 "ArrayBuffer")); | |
| 403 } | |
| 404 } else { | |
| 405 if (length % element_size != 0) { | |
| 406 return ThrowException( | |
| 407 String::New("ArrayBuffer minus the byteOffset must be a " | |
| 
Jakob Kummerow
2012/01/11 13:54:27
nit: s/ArrayBuffer/ArrayBuffer length/
 
danno
2012/01/11 15:10:20
Done.
 | |
| 408 "multiple of the element size")); | |
| 409 } | |
| 410 length /= element_size; | |
| 321 } | 411 } | 
| 322 if (raw_length > static_cast<int32_t>(kMaxLength)) { | 412 | 
| 323 return ThrowException( | 413 // Hold a reference to the ArrayBuffer so its buffer doesn't get collected. | 
| 324 String::New("Array length exceeds maximum length.")); | 414 array->Set(String::New(kArrayBufferReferencePropName), args[0], ReadOnly); | 
| 325 } | |
| 326 length = static_cast<size_t>(raw_length); | |
| 327 } | 415 } | 
| 328 if (length > static_cast<size_t>(kMaxLength)) { | 416 | 
| 329 return ThrowException(String::New("Array length exceeds maximum length.")); | 417 if (is_array_buffer_construct) { | 
| 418 array->Set(String::New(kArrayBufferMarkerPropName), True(), ReadOnly); | |
| 330 } | 419 } | 
| 331 void* data = calloc(length, element_size); | 420 | 
| 332 if (data == NULL) { | |
| 333 return ThrowException(String::New("Memory allocation failed.")); | |
| 334 } | |
| 335 Handle<Object> array = Object::New(); | |
| 336 Persistent<Object> persistent_array = Persistent<Object>::New(array); | 421 Persistent<Object> persistent_array = Persistent<Object>::New(array); | 
| 337 persistent_array.MakeWeak(data, ExternalArrayWeakCallback); | 422 persistent_array.MakeWeak(data, ExternalArrayWeakCallback); | 
| 338 persistent_array.MarkIndependent(); | 423 persistent_array.MarkIndependent(); | 
| 339 array->SetIndexedPropertiesToExternalArrayData(data, type, | 424 if (data == NULL && length != 0) { | 
| 340 static_cast<int>(length)); | 425 data = calloc(length, element_size); | 
| 426 if (data == NULL) { | |
| 427 return ThrowException(String::New("Memory allocation failed.")); | |
| 428 } | |
| 429 } | |
| 430 | |
| 431 array->SetIndexedPropertiesToExternalArrayData( | |
| 432 reinterpret_cast<uint8_t*>(data) + offset, type, | |
| 433 static_cast<int>(length)); | |
| 341 array->Set(String::New("length"), | 434 array->Set(String::New("length"), | 
| 342 Int32::New(static_cast<int32_t>(length)), ReadOnly); | 435 Int32::New(static_cast<int32_t>(length)), ReadOnly); | 
| 343 array->Set(String::New("BYTES_PER_ELEMENT"), | 436 array->Set(String::New("BYTES_PER_ELEMENT"), | 
| 344 Int32::New(static_cast<int32_t>(element_size))); | 437 Int32::New(static_cast<int32_t>(element_size))); | 
| 345 return array; | 438 return array; | 
| 346 } | 439 } | 
| 347 | 440 | 
| 348 | 441 | 
| 349 void Shell::ExternalArrayWeakCallback(Persistent<Value> object, void* data) { | 442 void Shell::ExternalArrayWeakCallback(Persistent<Value> object, void* data) { | 
| 350 free(data); | 443 HandleScope scope; | 
| 444 Handle<String> prop_name = String::New(kArrayBufferReferencePropName); | |
| 445 Handle<Object> converted_object = object->ToObject(); | |
| 446 Local<Value> prop_value = converted_object->Get(prop_name); | |
| 447 if (data != NULL && !prop_value->IsObject()) { | |
| 448 free(data); | |
| 449 } | |
| 351 object.Dispose(); | 450 object.Dispose(); | 
| 352 } | 451 } | 
| 353 | 452 | 
| 354 | 453 | 
| 454 Handle<Value> Shell::ArrayBuffer(const Arguments& args) { | |
| 455 Handle<Value> result = CreateExternalArray(args, v8::kExternalByteArray, 0); | |
| 456 return result; | |
| 
Jakob Kummerow
2012/01/11 13:54:27
nit: you don't need this line, could return direct
 
danno
2012/01/11 15:10:20
Done.
 | |
| 457 } | |
| 458 | |
| 459 | |
| 355 Handle<Value> Shell::Int8Array(const Arguments& args) { | 460 Handle<Value> Shell::Int8Array(const Arguments& args) { | 
| 356 return CreateExternalArray(args, v8::kExternalByteArray, sizeof(int8_t)); | 461 return CreateExternalArray(args, v8::kExternalByteArray, sizeof(int8_t)); | 
| 357 } | 462 } | 
| 358 | 463 | 
| 359 | 464 | 
| 360 Handle<Value> Shell::Uint8Array(const Arguments& args) { | 465 Handle<Value> Shell::Uint8Array(const Arguments& args) { | 
| 361 return CreateExternalArray(args, kExternalUnsignedByteArray, sizeof(uint8_t)); | 466 return CreateExternalArray(args, kExternalUnsignedByteArray, sizeof(uint8_t)); | 
| 362 } | 467 } | 
| 363 | 468 | 
| 364 | 469 | 
| (...skipping 321 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 686 FunctionTemplate::New(ReadLine)); | 791 FunctionTemplate::New(ReadLine)); | 
| 687 global_template->Set(String::New("load"), FunctionTemplate::New(Load)); | 792 global_template->Set(String::New("load"), FunctionTemplate::New(Load)); | 
| 688 global_template->Set(String::New("quit"), FunctionTemplate::New(Quit)); | 793 global_template->Set(String::New("quit"), FunctionTemplate::New(Quit)); | 
| 689 global_template->Set(String::New("version"), FunctionTemplate::New(Version)); | 794 global_template->Set(String::New("version"), FunctionTemplate::New(Version)); | 
| 690 global_template->Set(String::New("enableProfiler"), | 795 global_template->Set(String::New("enableProfiler"), | 
| 691 FunctionTemplate::New(EnableProfiler)); | 796 FunctionTemplate::New(EnableProfiler)); | 
| 692 global_template->Set(String::New("disableProfiler"), | 797 global_template->Set(String::New("disableProfiler"), | 
| 693 FunctionTemplate::New(DisableProfiler)); | 798 FunctionTemplate::New(DisableProfiler)); | 
| 694 | 799 | 
| 695 // Bind the handlers for external arrays. | 800 // Bind the handlers for external arrays. | 
| 801 global_template->Set(String::New("ArrayBuffer"), | |
| 802 FunctionTemplate::New(ArrayBuffer)); | |
| 696 global_template->Set(String::New("Int8Array"), | 803 global_template->Set(String::New("Int8Array"), | 
| 697 FunctionTemplate::New(Int8Array)); | 804 FunctionTemplate::New(Int8Array)); | 
| 698 global_template->Set(String::New("Uint8Array"), | 805 global_template->Set(String::New("Uint8Array"), | 
| 699 FunctionTemplate::New(Uint8Array)); | 806 FunctionTemplate::New(Uint8Array)); | 
| 700 global_template->Set(String::New("Int16Array"), | 807 global_template->Set(String::New("Int16Array"), | 
| 701 FunctionTemplate::New(Int16Array)); | 808 FunctionTemplate::New(Int16Array)); | 
| 702 global_template->Set(String::New("Uint16Array"), | 809 global_template->Set(String::New("Uint16Array"), | 
| 703 FunctionTemplate::New(Uint16Array)); | 810 FunctionTemplate::New(Uint16Array)); | 
| 704 global_template->Set(String::New("Int32Array"), | 811 global_template->Set(String::New("Int32Array"), | 
| 705 FunctionTemplate::New(Int32Array)); | 812 FunctionTemplate::New(Int32Array)); | 
| (...skipping 687 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 1393 } | 1500 } | 
| 1394 | 1501 | 
| 1395 } // namespace v8 | 1502 } // namespace v8 | 
| 1396 | 1503 | 
| 1397 | 1504 | 
| 1398 #ifndef GOOGLE3 | 1505 #ifndef GOOGLE3 | 
| 1399 int main(int argc, char* argv[]) { | 1506 int main(int argc, char* argv[]) { | 
| 1400 return v8::Shell::Main(argc, argv); | 1507 return v8::Shell::Main(argc, argv); | 
| 1401 } | 1508 } | 
| 1402 #endif | 1509 #endif | 
| OLD | NEW |