ErrorConstructor.cpp 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106
  1. /*
  2. * Copyright (c) 2020-2021, Linus Groh <linusg@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibJS/Runtime/AbstractOperations.h>
  7. #include <LibJS/Runtime/Error.h>
  8. #include <LibJS/Runtime/ErrorConstructor.h>
  9. #include <LibJS/Runtime/GlobalObject.h>
  10. namespace JS {
  11. ErrorConstructor::ErrorConstructor(GlobalObject& global_object)
  12. : NativeFunction(vm().names.Error.as_string(), *global_object.function_prototype())
  13. {
  14. }
  15. void ErrorConstructor::initialize(GlobalObject& global_object)
  16. {
  17. auto& vm = this->vm();
  18. NativeFunction::initialize(global_object);
  19. // 20.5.2.1 Error.prototype, https://tc39.es/ecma262/#sec-error.prototype
  20. define_direct_property(vm.names.prototype, global_object.error_prototype(), 0);
  21. define_direct_property(vm.names.length, Value(1), Attribute::Configurable);
  22. }
  23. // 20.5.1.1 Error ( message ), https://tc39.es/ecma262/#sec-error-message
  24. Value ErrorConstructor::call()
  25. {
  26. return construct(*this);
  27. }
  28. // 20.5.1.1 Error ( message ), https://tc39.es/ecma262/#sec-error-message
  29. Value ErrorConstructor::construct(FunctionObject& new_target)
  30. {
  31. auto& vm = this->vm();
  32. auto& global_object = this->global_object();
  33. auto* error = TRY_OR_DISCARD(ordinary_create_from_constructor<Error>(global_object, new_target, &GlobalObject::error_prototype));
  34. if (!vm.argument(0).is_undefined()) {
  35. auto message = vm.argument(0).to_string(global_object);
  36. if (vm.exception())
  37. return {};
  38. error->create_non_enumerable_data_property_or_throw(vm.names.message, js_string(vm, message));
  39. }
  40. TRY_OR_DISCARD(error->install_error_cause(vm.argument(1)));
  41. return error;
  42. }
  43. #define __JS_ENUMERATE(ClassName, snake_name, PrototypeName, ConstructorName, ArrayType) \
  44. ConstructorName::ConstructorName(GlobalObject& global_object) \
  45. : NativeFunction(*static_cast<Object*>(global_object.error_constructor())) \
  46. { \
  47. } \
  48. \
  49. void ConstructorName::initialize(GlobalObject& global_object) \
  50. { \
  51. auto& vm = this->vm(); \
  52. NativeFunction::initialize(global_object); \
  53. \
  54. /* 20.5.6.2.1 NativeError.prototype, \
  55. https://tc39.es/ecma262/#sec-nativeerror.prototype */ \
  56. define_direct_property(vm.names.prototype, global_object.snake_name##_prototype(), 0); \
  57. \
  58. define_direct_property(vm.names.length, Value(1), Attribute::Configurable); \
  59. } \
  60. \
  61. ConstructorName::~ConstructorName() { } \
  62. \
  63. /* 20.5.6.1.1 NativeError ( message ), https://tc39.es/ecma262/#sec-nativeerror */ \
  64. Value ConstructorName::call() \
  65. { \
  66. return construct(*this); \
  67. } \
  68. \
  69. /* 20.5.6.1.1 NativeError ( message ), https://tc39.es/ecma262/#sec-nativeerror */ \
  70. Value ConstructorName::construct(FunctionObject& new_target) \
  71. { \
  72. auto& vm = this->vm(); \
  73. auto& global_object = this->global_object(); \
  74. \
  75. auto* error = TRY_OR_DISCARD(ordinary_create_from_constructor<ClassName>( \
  76. global_object, new_target, &GlobalObject::snake_name##_prototype)); \
  77. \
  78. if (!vm.argument(0).is_undefined()) { \
  79. auto message = vm.argument(0).to_string(global_object); \
  80. if (vm.exception()) \
  81. return {}; \
  82. error->create_non_enumerable_data_property_or_throw(vm.names.message, js_string(vm, message)); \
  83. } \
  84. \
  85. TRY_OR_DISCARD(error->install_error_cause(vm.argument(1))); \
  86. \
  87. return error; \
  88. }
  89. JS_ENUMERATE_NATIVE_ERRORS
  90. #undef __JS_ENUMERATE
  91. }