TypedArray.cpp 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. /*
  2. * Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
  3. * All rights reserved.
  4. *
  5. * Redistribution and use in source and binary forms, with or without
  6. * modification, are permitted provided that the following conditions are met:
  7. *
  8. * 1. Redistributions of source code must retain the above copyright notice, this
  9. * list of conditions and the following disclaimer.
  10. *
  11. * 2. Redistributions in binary form must reproduce the above copyright notice,
  12. * this list of conditions and the following disclaimer in the documentation
  13. * and/or other materials provided with the distribution.
  14. *
  15. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  16. * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  17. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  18. * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
  19. * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  20. * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
  21. * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
  22. * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
  23. * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  24. * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  25. */
  26. #include <LibJS/Runtime/GlobalObject.h>
  27. #include <LibJS/Runtime/TypedArray.h>
  28. #include <LibJS/Runtime/TypedArrayConstructor.h>
  29. namespace JS {
  30. #define JS_DEFINE_TYPED_ARRAY(ClassName, snake_name, PrototypeName, ConstructorName, Type) \
  31. ClassName* ClassName::create(GlobalObject& global_object, u32 length) \
  32. { \
  33. return global_object.heap().allocate<ClassName>(global_object, length, *global_object.snake_name##_prototype()); \
  34. } \
  35. \
  36. ClassName::ClassName(u32 length, Object& prototype) \
  37. : TypedArray(length, prototype) \
  38. { \
  39. } \
  40. ClassName::~ClassName() { } \
  41. \
  42. PrototypeName::PrototypeName(GlobalObject& global_object) \
  43. : Object(*global_object.typed_array_prototype()) \
  44. { \
  45. } \
  46. void PrototypeName::initialize(GlobalObject& global_object) \
  47. { \
  48. auto& vm = this->vm(); \
  49. Object::initialize(global_object); \
  50. define_property(vm.names.length, Value(0), Attribute::Configurable); \
  51. } \
  52. PrototypeName::~PrototypeName() { } \
  53. \
  54. ConstructorName::ConstructorName(GlobalObject& global_object) \
  55. : TypedArrayConstructor(vm().names.ClassName, *global_object.typed_array_constructor()) \
  56. { \
  57. } \
  58. ConstructorName::~ConstructorName() { } \
  59. void ConstructorName::initialize(GlobalObject& global_object) \
  60. { \
  61. auto& vm = this->vm(); \
  62. NativeFunction::initialize(global_object); \
  63. define_property(vm.names.prototype, global_object.snake_name##_prototype(), 0); \
  64. define_property(vm.names.length, Value(1), Attribute::Configurable); \
  65. } \
  66. Value ConstructorName::call() \
  67. { \
  68. auto& vm = this->vm(); \
  69. vm.throw_exception<TypeError>(global_object(), ErrorType::ConstructorWithoutNew, vm.names.ClassName); \
  70. return {}; \
  71. } \
  72. Value ConstructorName::construct(Function&) \
  73. { \
  74. auto& vm = this->vm(); \
  75. if (vm.argument_count() == 0) \
  76. return ClassName::create(global_object(), 0); \
  77. \
  78. if (vm.argument(0).is_object()) { \
  79. /* FIXME: Initialize from TypedArray, ArrayBuffer, Iterable or Array-like object */ \
  80. TODO(); \
  81. } \
  82. /* FIXME: Use ToIndex() abstract operation */ \
  83. auto array_length_value = vm.argument(0); \
  84. if (!array_length_value.is_integer() || array_length_value.as_i32() < 0) { \
  85. vm.throw_exception<TypeError>(global_object(), ErrorType::ArrayInvalidLength); \
  86. return {}; \
  87. } \
  88. auto* array = ClassName::create(global_object(), array_length_value.as_i32()); \
  89. return array; \
  90. }
  91. #undef __JS_ENUMERATE
  92. #define __JS_ENUMERATE(ClassName, snake_name, PrototypeName, ConstructorName, ArrayType) \
  93. JS_DEFINE_TYPED_ARRAY(ClassName, snake_name, PrototypeName, ConstructorName, ArrayType);
  94. JS_ENUMERATE_TYPED_ARRAYS
  95. #undef __JS_ENUMERATE
  96. }