AbstractOperations.cpp 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665
  1. /*
  2. * Copyright (c) 2020-2021, Linus Groh <linusg@serenityos.org>
  3. * Copyright (c) 2021, Andreas Kling <kling@serenityos.org>
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include <AK/CharacterTypes.h>
  8. #include <AK/Function.h>
  9. #include <AK/Optional.h>
  10. #include <AK/Result.h>
  11. #include <AK/TemporaryChange.h>
  12. #include <LibJS/Interpreter.h>
  13. #include <LibJS/Parser.h>
  14. #include <LibJS/Runtime/AbstractOperations.h>
  15. #include <LibJS/Runtime/Accessor.h>
  16. #include <LibJS/Runtime/ArgumentsObject.h>
  17. #include <LibJS/Runtime/Array.h>
  18. #include <LibJS/Runtime/BoundFunction.h>
  19. #include <LibJS/Runtime/DeclarativeEnvironment.h>
  20. #include <LibJS/Runtime/ErrorTypes.h>
  21. #include <LibJS/Runtime/FunctionEnvironment.h>
  22. #include <LibJS/Runtime/FunctionObject.h>
  23. #include <LibJS/Runtime/GlobalEnvironment.h>
  24. #include <LibJS/Runtime/GlobalObject.h>
  25. #include <LibJS/Runtime/Object.h>
  26. #include <LibJS/Runtime/ObjectEnvironment.h>
  27. #include <LibJS/Runtime/PropertyDescriptor.h>
  28. #include <LibJS/Runtime/PropertyName.h>
  29. #include <LibJS/Runtime/ProxyObject.h>
  30. #include <LibJS/Runtime/Reference.h>
  31. namespace JS {
  32. // Used in various abstract operations to make it obvious when a non-optional return value must be discarded.
  33. static constexpr double INVALID { 0 };
  34. // 7.2.1 RequireObjectCoercible ( argument ), https://tc39.es/ecma262/#sec-requireobjectcoercible
  35. Value require_object_coercible(GlobalObject& global_object, Value value)
  36. {
  37. auto& vm = global_object.vm();
  38. if (value.is_nullish()) {
  39. vm.throw_exception<TypeError>(global_object, ErrorType::NotObjectCoercible, value.to_string_without_side_effects());
  40. return {};
  41. }
  42. return value;
  43. }
  44. // 7.3.18 LengthOfArrayLike ( obj ), https://tc39.es/ecma262/#sec-lengthofarraylike
  45. size_t length_of_array_like(GlobalObject& global_object, Object const& object)
  46. {
  47. auto& vm = global_object.vm();
  48. auto result = object.get(vm.names.length);
  49. if (vm.exception())
  50. return INVALID;
  51. return result.to_length(global_object);
  52. }
  53. // 7.3.19 CreateListFromArrayLike ( obj [ , elementTypes ] ), https://tc39.es/ecma262/#sec-createlistfromarraylike
  54. MarkedValueList create_list_from_array_like(GlobalObject& global_object, Value value, Function<Result<void, ErrorType>(Value)> check_value)
  55. {
  56. auto& vm = global_object.vm();
  57. auto& heap = global_object.heap();
  58. if (!value.is_object()) {
  59. vm.throw_exception<TypeError>(global_object, ErrorType::NotAnObject, value.to_string_without_side_effects());
  60. return MarkedValueList { heap };
  61. }
  62. auto& array_like = value.as_object();
  63. auto length = length_of_array_like(global_object, array_like);
  64. if (vm.exception())
  65. return MarkedValueList { heap };
  66. auto list = MarkedValueList { heap };
  67. for (size_t i = 0; i < length; ++i) {
  68. auto index_name = String::number(i);
  69. auto next = array_like.get(index_name);
  70. if (vm.exception())
  71. return MarkedValueList { heap };
  72. if (check_value) {
  73. auto result = check_value(next);
  74. if (result.is_error()) {
  75. vm.throw_exception<TypeError>(global_object, result.release_error());
  76. return MarkedValueList { heap };
  77. }
  78. }
  79. list.append(next);
  80. }
  81. return list;
  82. }
  83. // 7.3.22 SpeciesConstructor ( O, defaultConstructor ), https://tc39.es/ecma262/#sec-speciesconstructor
  84. FunctionObject* species_constructor(GlobalObject& global_object, Object const& object, FunctionObject& default_constructor)
  85. {
  86. auto& vm = global_object.vm();
  87. auto constructor = object.get(vm.names.constructor);
  88. if (vm.exception())
  89. return nullptr;
  90. if (constructor.is_undefined())
  91. return &default_constructor;
  92. if (!constructor.is_object()) {
  93. vm.throw_exception<TypeError>(global_object, ErrorType::NotAConstructor, constructor.to_string_without_side_effects());
  94. return nullptr;
  95. }
  96. auto species = constructor.as_object().get(*vm.well_known_symbol_species());
  97. if (species.is_nullish())
  98. return &default_constructor;
  99. if (species.is_constructor())
  100. return &species.as_function();
  101. vm.throw_exception<TypeError>(global_object, ErrorType::NotAConstructor, species.to_string_without_side_effects());
  102. return nullptr;
  103. }
  104. // 7.3.24 GetFunctionRealm ( obj ), https://tc39.es/ecma262/#sec-getfunctionrealm
  105. GlobalObject* get_function_realm(GlobalObject& global_object, FunctionObject const& function)
  106. {
  107. auto& vm = global_object.vm();
  108. // 1. Assert: ! IsCallable(obj) is true.
  109. // 2. If obj has a [[Realm]] internal slot, then
  110. if (function.realm()) {
  111. // a. Return obj.[[Realm]].
  112. return function.realm();
  113. }
  114. // 3. If obj is a bound function exotic object, then
  115. if (is<BoundFunction>(function)) {
  116. auto& bound_function = static_cast<BoundFunction const&>(function);
  117. // a. Let target be obj.[[BoundTargetFunction]].
  118. auto& target = bound_function.target_function();
  119. // b. Return ? GetFunctionRealm(target).
  120. return get_function_realm(global_object, target);
  121. }
  122. // 4. If obj is a Proxy exotic object, then
  123. if (is<ProxyObject>(function)) {
  124. auto& proxy = static_cast<ProxyObject const&>(function);
  125. // a. If obj.[[ProxyHandler]] is null, throw a TypeError exception.
  126. if (proxy.is_revoked()) {
  127. vm.throw_exception<TypeError>(global_object, ErrorType::ProxyRevoked);
  128. return nullptr;
  129. }
  130. // b. Let proxyTarget be obj.[[ProxyTarget]].
  131. auto& proxy_target = proxy.target();
  132. // c. Return ? GetFunctionRealm(proxyTarget).
  133. VERIFY(proxy_target.is_function());
  134. return get_function_realm(global_object, static_cast<FunctionObject const&>(proxy_target));
  135. }
  136. // 5. Return the current Realm Record.
  137. return &global_object;
  138. }
  139. // 10.1.6.2 IsCompatiblePropertyDescriptor ( Extensible, Desc, Current ), https://tc39.es/ecma262/#sec-iscompatiblepropertydescriptor
  140. bool is_compatible_property_descriptor(bool extensible, PropertyDescriptor const& descriptor, Optional<PropertyDescriptor> const& current)
  141. {
  142. // 1. Return ValidateAndApplyPropertyDescriptor(undefined, undefined, Extensible, Desc, Current).
  143. return validate_and_apply_property_descriptor(nullptr, {}, extensible, descriptor, current);
  144. }
  145. // 10.1.6.3 ValidateAndApplyPropertyDescriptor ( O, P, extensible, Desc, current ),
  146. bool validate_and_apply_property_descriptor(Object* object, PropertyName const& property_name, bool extensible, PropertyDescriptor const& descriptor, Optional<PropertyDescriptor> const& current)
  147. {
  148. // 1. Assert: If O is not undefined, then IsPropertyKey(P) is true.
  149. if (object)
  150. VERIFY(property_name.is_valid());
  151. // 2. If current is undefined, then
  152. if (!current.has_value()) {
  153. // a. If extensible is false, return false.
  154. if (!extensible)
  155. return false;
  156. // b. Assert: extensible is true.
  157. // c. If IsGenericDescriptor(Desc) is true or IsDataDescriptor(Desc) is true, then
  158. if (descriptor.is_generic_descriptor() || descriptor.is_data_descriptor()) {
  159. // i. If O is not undefined, create an own data property named P of object O whose [[Value]], [[Writable]],
  160. // [[Enumerable]], and [[Configurable]] attribute values are described by Desc.
  161. // If the value of an attribute field of Desc is absent, the attribute of the newly created property is set
  162. // to its default value.
  163. if (object) {
  164. auto value = descriptor.value.value_or(js_undefined());
  165. object->storage_set(property_name, { value, descriptor.attributes() });
  166. }
  167. }
  168. // d. Else,
  169. else {
  170. // i. Assert: ! IsAccessorDescriptor(Desc) is true.
  171. VERIFY(descriptor.is_accessor_descriptor());
  172. // ii. If O is not undefined, create an own accessor property named P of object O whose [[Get]], [[Set]],
  173. // [[Enumerable]], and [[Configurable]] attribute values are described by Desc.
  174. // If the value of an attribute field of Desc is absent, the attribute of the newly created property is set
  175. // to its default value.
  176. if (object) {
  177. auto accessor = Accessor::create(object->vm(), descriptor.get.value_or(nullptr), descriptor.set.value_or(nullptr));
  178. object->storage_set(property_name, { accessor, descriptor.attributes() });
  179. }
  180. }
  181. // e. Return true.
  182. return true;
  183. }
  184. // 3. If every field in Desc is absent, return true.
  185. if (descriptor.is_empty())
  186. return true;
  187. // 4. If current.[[Configurable]] is false, then
  188. if (!*current->configurable) {
  189. // a. If Desc.[[Configurable]] is present and its value is true, return false.
  190. if (descriptor.configurable.has_value() && *descriptor.configurable)
  191. return false;
  192. // b. If Desc.[[Enumerable]] is present and ! SameValue(Desc.[[Enumerable]], current.[[Enumerable]]) is false, return false.
  193. if (descriptor.enumerable.has_value() && *descriptor.enumerable != *current->enumerable)
  194. return false;
  195. }
  196. // 5. If ! IsGenericDescriptor(Desc) is true, then
  197. if (descriptor.is_generic_descriptor()) {
  198. // a. NOTE: No further validation is required.
  199. }
  200. // 6. Else if ! SameValue(! IsDataDescriptor(current), ! IsDataDescriptor(Desc)) is false, then
  201. else if (current->is_data_descriptor() != descriptor.is_data_descriptor()) {
  202. // a. If current.[[Configurable]] is false, return false.
  203. if (!*current->configurable)
  204. return false;
  205. // b. If IsDataDescriptor(current) is true, then
  206. if (current->is_data_descriptor()) {
  207. // If O is not undefined, convert the property named P of object O from a data property to an accessor property.
  208. // Preserve the existing values of the converted property's [[Configurable]] and [[Enumerable]] attributes and
  209. // set the rest of the property's attributes to their default values.
  210. if (object) {
  211. auto accessor = Accessor::create(object->vm(), nullptr, nullptr);
  212. object->storage_set(property_name, { accessor, current->attributes() });
  213. }
  214. }
  215. // c. Else,
  216. else {
  217. // If O is not undefined, convert the property named P of object O from an accessor property to a data property.
  218. // Preserve the existing values of the converted property's [[Configurable]] and [[Enumerable]] attributes and
  219. // set the rest of the property's attributes to their default values.
  220. if (object) {
  221. auto value = js_undefined();
  222. object->storage_set(property_name, { value, current->attributes() });
  223. }
  224. }
  225. }
  226. // 7. Else if IsDataDescriptor(current) and IsDataDescriptor(Desc) are both true, then
  227. else if (current->is_data_descriptor() && descriptor.is_data_descriptor()) {
  228. // a. If current.[[Configurable]] is false and current.[[Writable]] is false, then
  229. if (!*current->configurable && !*current->writable) {
  230. // i. If Desc.[[Writable]] is present and Desc.[[Writable]] is true, return false.
  231. if (descriptor.writable.has_value() && *descriptor.writable)
  232. return false;
  233. // ii. If Desc.[[Value]] is present and SameValue(Desc.[[Value]], current.[[Value]]) is false, return false.
  234. if (descriptor.value.has_value() && !same_value(*descriptor.value, *current->value))
  235. return false;
  236. // iii. Return true.
  237. return true;
  238. }
  239. }
  240. // 8. Else,
  241. else {
  242. // a. Assert: ! IsAccessorDescriptor(current) and ! IsAccessorDescriptor(Desc) are both true.
  243. VERIFY(current->is_accessor_descriptor());
  244. VERIFY(descriptor.is_accessor_descriptor());
  245. // b. If current.[[Configurable]] is false, then
  246. if (!*current->configurable) {
  247. // i. If Desc.[[Set]] is present and SameValue(Desc.[[Set]], current.[[Set]]) is false, return false.
  248. if (descriptor.set.has_value() && *descriptor.set != *current->set)
  249. return false;
  250. // ii. If Desc.[[Get]] is present and SameValue(Desc.[[Get]], current.[[Get]]) is false, return false.
  251. if (descriptor.get.has_value() && *descriptor.get != *current->get)
  252. return false;
  253. // iii. Return true.
  254. return true;
  255. }
  256. }
  257. // 9. If O is not undefined, then
  258. if (object) {
  259. // a. For each field of Desc that is present, set the corresponding attribute of the property named P of object O to the value of the field.
  260. Value value;
  261. if (descriptor.is_accessor_descriptor() || (current->is_accessor_descriptor() && !descriptor.is_data_descriptor())) {
  262. auto* getter = descriptor.get.value_or(current->get.value_or(nullptr));
  263. auto* setter = descriptor.set.value_or(current->set.value_or(nullptr));
  264. value = Accessor::create(object->vm(), getter, setter);
  265. } else {
  266. value = descriptor.value.value_or(current->value.value_or({}));
  267. }
  268. PropertyAttributes attributes;
  269. attributes.set_writable(descriptor.writable.value_or(current->writable.value_or(false)));
  270. attributes.set_enumerable(descriptor.enumerable.value_or(current->enumerable.value_or(false)));
  271. attributes.set_configurable(descriptor.configurable.value_or(current->configurable.value_or(false)));
  272. object->storage_set(property_name, { value, attributes });
  273. }
  274. // 10. Return true.
  275. return true;
  276. }
  277. // 10.1.14 GetPrototypeFromConstructor ( constructor, intrinsicDefaultProto )
  278. Object* get_prototype_from_constructor(GlobalObject& global_object, FunctionObject const& constructor, Object* (GlobalObject::*intrinsic_default_prototype)())
  279. {
  280. auto& vm = global_object.vm();
  281. auto prototype = constructor.get(vm.names.prototype);
  282. if (vm.exception())
  283. return nullptr;
  284. if (!prototype.is_object()) {
  285. auto* realm = get_function_realm(global_object, constructor);
  286. if (vm.exception())
  287. return nullptr;
  288. prototype = (realm->*intrinsic_default_prototype)();
  289. }
  290. return &prototype.as_object();
  291. }
  292. // 9.1.2.2 NewDeclarativeEnvironment ( E ), https://tc39.es/ecma262/#sec-newdeclarativeenvironment
  293. DeclarativeEnvironment* new_declarative_environment(Environment& environment)
  294. {
  295. auto& global_object = environment.global_object();
  296. return global_object.heap().allocate<DeclarativeEnvironment>(global_object, &environment);
  297. }
  298. // 9.1.2.3 NewObjectEnvironment ( O, W, E ), https://tc39.es/ecma262/#sec-newobjectenvironment
  299. ObjectEnvironment* new_object_environment(Object& object, bool is_with_environment, Environment* environment)
  300. {
  301. auto& global_object = object.global_object();
  302. return global_object.heap().allocate<ObjectEnvironment>(global_object, object, is_with_environment ? ObjectEnvironment::IsWithEnvironment::Yes : ObjectEnvironment::IsWithEnvironment::No, environment);
  303. }
  304. // 9.4.3 GetThisEnvironment ( ), https://tc39.es/ecma262/#sec-getthisenvironment
  305. Environment& get_this_environment(VM& vm)
  306. {
  307. for (auto* env = vm.lexical_environment(); env; env = env->outer_environment()) {
  308. if (env->has_this_binding())
  309. return *env;
  310. }
  311. VERIFY_NOT_REACHED();
  312. }
  313. // 13.3.7.2 GetSuperConstructor ( ), https://tc39.es/ecma262/#sec-getsuperconstructor
  314. Object* get_super_constructor(VM& vm)
  315. {
  316. auto& env = get_this_environment(vm);
  317. auto& active_function = verify_cast<FunctionEnvironment>(env).function_object();
  318. auto* super_constructor = active_function.internal_get_prototype_of();
  319. return super_constructor;
  320. }
  321. // 13.3.7.3 MakeSuperPropertyReference ( actualThis, propertyKey, strict )
  322. Reference make_super_property_reference(GlobalObject& global_object, Value actual_this, StringOrSymbol const& property_key, bool strict)
  323. {
  324. auto& vm = global_object.vm();
  325. // 1. Let env be GetThisEnvironment().
  326. auto& env = verify_cast<FunctionEnvironment>(get_this_environment(vm));
  327. // 2. Assert: env.HasSuperBinding() is true.
  328. VERIFY(env.has_super_binding());
  329. // 3. Let baseValue be ? env.GetSuperBase().
  330. auto base_value = env.get_super_base();
  331. // 4. Let bv be ? RequireObjectCoercible(baseValue).
  332. auto bv = require_object_coercible(global_object, base_value);
  333. if (vm.exception())
  334. return {};
  335. // 5. Return the Reference Record { [[Base]]: bv, [[ReferencedName]]: propertyKey, [[Strict]]: strict, [[ThisValue]]: actualThis }.
  336. // 6. NOTE: This returns a Super Reference Record.
  337. return Reference { bv, property_key, actual_this, strict };
  338. }
  339. // 19.2.1.1 PerformEval ( x, callerRealm, strictCaller, direct ), https://tc39.es/ecma262/#sec-performeval
  340. Value perform_eval(Value x, GlobalObject& caller_realm, CallerMode strict_caller, EvalMode direct)
  341. {
  342. VERIFY(direct == EvalMode::Direct || strict_caller == CallerMode::NonStrict);
  343. if (!x.is_string())
  344. return x;
  345. auto& vm = caller_realm.vm();
  346. auto& code_string = x.as_string();
  347. Parser parser { Lexer { code_string.string() } };
  348. auto program = parser.parse_program(strict_caller == CallerMode::Strict);
  349. if (parser.has_errors()) {
  350. auto& error = parser.errors()[0];
  351. vm.throw_exception<SyntaxError>(caller_realm, error.to_string());
  352. return {};
  353. }
  354. auto& interpreter = vm.interpreter();
  355. if (direct == EvalMode::Direct)
  356. return interpreter.execute_statement(caller_realm, program).value_or(js_undefined());
  357. TemporaryChange scope_change(vm.running_execution_context().lexical_environment, static_cast<Environment*>(&caller_realm.environment()));
  358. return interpreter.execute_statement(caller_realm, program).value_or(js_undefined());
  359. }
  360. // 10.4.4.6 CreateUnmappedArgumentsObject ( argumentsList ), https://tc39.es/ecma262/#sec-createunmappedargumentsobject
  361. Object* create_unmapped_arguments_object(GlobalObject& global_object, Vector<Value> const& arguments)
  362. {
  363. auto& vm = global_object.vm();
  364. // 1. Let len be the number of elements in argumentsList.
  365. auto length = arguments.size();
  366. // 2. Let obj be ! OrdinaryObjectCreate(%Object.prototype%, « [[ParameterMap]] »).
  367. // 3. Set obj.[[ParameterMap]] to undefined.
  368. auto* object = Object::create(global_object, global_object.object_prototype());
  369. object->set_has_parameter_map();
  370. // 4. Perform DefinePropertyOrThrow(obj, "length", PropertyDescriptor { [[Value]]: 𝔽(len), [[Writable]]: true, [[Enumerable]]: false, [[Configurable]]: true }).
  371. object->define_property_or_throw(vm.names.length, { .value = Value(length), .writable = true, .enumerable = false, .configurable = true });
  372. VERIFY(!vm.exception());
  373. // 5. Let index be 0.
  374. // 6. Repeat, while index < len,
  375. for (size_t index = 0; index < length; ++index) {
  376. // a. Let val be argumentsList[index].
  377. auto value = arguments[index];
  378. // b. Perform ! CreateDataPropertyOrThrow(obj, ! ToString(𝔽(index)), val).
  379. object->create_data_property_or_throw(index, value);
  380. VERIFY(!vm.exception());
  381. // c. Set index to index + 1.
  382. }
  383. // 7. Perform ! DefinePropertyOrThrow(obj, @@iterator, PropertyDescriptor { [[Value]]: %Array.prototype.values%, [[Writable]]: true, [[Enumerable]]: false, [[Configurable]]: true }).
  384. // FIXME: This is not guaranteed to be %Array.prototype.values%!
  385. auto array_prototype_values = global_object.array_prototype()->get(vm.names.values);
  386. if (vm.exception())
  387. return {};
  388. object->define_property_or_throw(*vm.well_known_symbol_iterator(), { .value = array_prototype_values, .writable = true, .enumerable = false, .configurable = true });
  389. VERIFY(!vm.exception());
  390. // 8. Perform ! DefinePropertyOrThrow(obj, "callee", PropertyDescriptor { [[Get]]: %ThrowTypeError%, [[Set]]: %ThrowTypeError%, [[Enumerable]]: false, [[Configurable]]: false }).
  391. auto* throw_type_error = global_object.throw_type_error_function();
  392. object->define_property_or_throw(vm.names.callee, { .get = throw_type_error, .set = throw_type_error, .enumerable = false, .configurable = false });
  393. VERIFY(!vm.exception());
  394. // 9. Return obj.
  395. return object;
  396. }
  397. // 10.4.4.7 CreateMappedArgumentsObject ( func, formals, argumentsList, env ), https://tc39.es/ecma262/#sec-createmappedargumentsobject
  398. Object* create_mapped_arguments_object(GlobalObject& global_object, FunctionObject& function, Vector<FunctionNode::Parameter> const& formals, Vector<Value> const& arguments, Environment& environment)
  399. {
  400. auto& vm = global_object.vm();
  401. // 1. Assert: formals does not contain a rest parameter, any binding patterns, or any initializers. It may contain duplicate identifiers.
  402. // 2. Let len be the number of elements in argumentsList.
  403. VERIFY(arguments.size() <= NumericLimits<i32>::max());
  404. i32 length = static_cast<i32>(arguments.size());
  405. // 3. Let obj be ! MakeBasicObject(« [[Prototype]], [[Extensible]], [[ParameterMap]] »).
  406. // 4. Set obj.[[GetOwnProperty]] as specified in 10.4.4.1.
  407. // 5. Set obj.[[DefineOwnProperty]] as specified in 10.4.4.2.
  408. // 6. Set obj.[[Get]] as specified in 10.4.4.3.
  409. // 7. Set obj.[[Set]] as specified in 10.4.4.4.
  410. // 8. Set obj.[[Delete]] as specified in 10.4.4.5.
  411. // 9. Set obj.[[Prototype]] to %Object.prototype%.
  412. auto* object = vm.heap().allocate<ArgumentsObject>(global_object, global_object, environment);
  413. if (vm.exception())
  414. return nullptr;
  415. // 14. Let index be 0.
  416. // 15. Repeat, while index < len,
  417. for (i32 index = 0; index < length; ++index) {
  418. // a. Let val be argumentsList[index].
  419. auto value = arguments[index];
  420. // b. Perform ! CreateDataPropertyOrThrow(obj, ! ToString(𝔽(index)), val).
  421. object->create_data_property_or_throw(index, value);
  422. VERIFY(!vm.exception());
  423. // c. Set index to index + 1.
  424. }
  425. // 16. Perform ! DefinePropertyOrThrow(obj, "length", PropertyDescriptor { [[Value]]: 𝔽(len), [[Writable]]: true, [[Enumerable]]: false, [[Configurable]]: true }).
  426. object->define_property_or_throw(vm.names.length, { .value = Value(length), .writable = true, .enumerable = false, .configurable = true });
  427. VERIFY(!vm.exception());
  428. // 17. Let mappedNames be a new empty List.
  429. HashTable<FlyString> mapped_names;
  430. // 18. Set index to numberOfParameters - 1.
  431. // 19. Repeat, while index ≥ 0,
  432. VERIFY(formals.size() <= NumericLimits<i32>::max());
  433. for (i32 index = static_cast<i32>(formals.size()) - 1; index >= 0; --index) {
  434. // a. Let name be parameterNames[index].
  435. auto const& name = formals[index].binding.get<FlyString>();
  436. // b. If name is not an element of mappedNames, then
  437. if (mapped_names.contains(name))
  438. continue;
  439. // i. Add name as an element of the list mappedNames.
  440. mapped_names.set(name);
  441. // ii. If index < len, then
  442. if (index < length) {
  443. // 1. Let g be MakeArgGetter(name, env).
  444. // 2. Let p be MakeArgSetter(name, env).
  445. // 3. Perform map.[[DefineOwnProperty]](! ToString(𝔽(index)), PropertyDescriptor { [[Set]]: p, [[Get]]: g, [[Enumerable]]: false, [[Configurable]]: true }).
  446. object->parameter_map().define_native_accessor(
  447. String::number(index),
  448. [&environment, name](VM&, GlobalObject&) -> Value {
  449. auto variable = environment.get_from_environment(name);
  450. if (!variable.has_value())
  451. return {};
  452. return variable->value;
  453. },
  454. [&environment, name](VM& vm, GlobalObject&) {
  455. auto value = vm.argument(0);
  456. environment.put_into_environment(name, Variable { value, DeclarationKind::Var });
  457. return js_undefined();
  458. },
  459. Attribute::Configurable);
  460. }
  461. }
  462. // 20. Perform ! DefinePropertyOrThrow(obj, @@iterator, PropertyDescriptor { [[Value]]: %Array.prototype.values%, [[Writable]]: true, [[Enumerable]]: false, [[Configurable]]: true }).
  463. // FIXME: This is not guaranteed to be %Array.prototype.values%!
  464. auto array_prototype_values = global_object.array_prototype()->get(vm.names.values);
  465. if (vm.exception())
  466. return {};
  467. object->define_property_or_throw(*vm.well_known_symbol_iterator(), { .value = array_prototype_values, .writable = true, .enumerable = false, .configurable = true });
  468. VERIFY(!vm.exception());
  469. // 21. Perform ! DefinePropertyOrThrow(obj, "callee", PropertyDescriptor { [[Value]]: func, [[Writable]]: true, [[Enumerable]]: false, [[Configurable]]: true }).
  470. object->define_property_or_throw(vm.names.callee, { .value = &function, .writable = true, .enumerable = false, .configurable = true });
  471. VERIFY(!vm.exception());
  472. // 22. Return obj.
  473. return object;
  474. }
  475. // 7.1.21 CanonicalNumericIndexString ( argument ), https://tc39.es/ecma262/#sec-canonicalnumericindexstring
  476. Value canonical_numeric_index_string(GlobalObject& global_object, PropertyName const& property_name)
  477. {
  478. // NOTE: If the property name is a number type (An implementation-defined optimized
  479. // property key type), it can be treated as a string property that has already been
  480. // converted successfully into a canonical numeric index.
  481. VERIFY(property_name.is_string() || property_name.is_number());
  482. if (property_name.is_number())
  483. return Value(property_name.as_number());
  484. // 1. Assert: Type(argument) is String.
  485. auto argument = Value(js_string(global_object.vm(), property_name.as_string()));
  486. // 2. If argument is "-0", return -0𝔽.
  487. if (argument.as_string().string() == "-0")
  488. return Value(-0.0);
  489. // 3. Let n be ! ToNumber(argument).
  490. auto n = argument.to_number(global_object);
  491. // 4. If SameValue(! ToString(n), argument) is false, return undefined.
  492. if (!same_value(n.to_primitive_string(global_object), argument))
  493. return js_undefined();
  494. // 5. Return n.
  495. return n;
  496. }
  497. // 22.1.3.17.1 GetSubstitution ( matched, str, position, captures, namedCaptures, replacement ), https://tc39.es/ecma262/#sec-getsubstitution
  498. String get_substitution(GlobalObject& global_object, String const& matched, String const& str, size_t position, Vector<Value> const& captures, Value named_captures, Value replacement)
  499. {
  500. auto& vm = global_object.vm();
  501. auto replace_string = replacement.to_string(global_object);
  502. if (vm.exception())
  503. return {};
  504. StringBuilder result;
  505. for (size_t i = 0; i < replace_string.length(); ++i) {
  506. char curr = replace_string[i];
  507. if ((curr != '$') || (i + 1 >= replace_string.length())) {
  508. result.append(curr);
  509. continue;
  510. }
  511. char next = replace_string[i + 1];
  512. if (next == '$') {
  513. result.append(next);
  514. ++i;
  515. } else if (next == '&') {
  516. result.append(matched);
  517. ++i;
  518. } else if (next == '`') {
  519. result.append(str.substring_view(0, position));
  520. ++i;
  521. } else if (next == '\'') {
  522. auto tail_pos = position + matched.length();
  523. if (tail_pos < str.length())
  524. result.append(str.substring_view(tail_pos));
  525. ++i;
  526. } else if (is_ascii_digit(next)) {
  527. bool is_two_digits = (i + 2 < replace_string.length()) && is_ascii_digit(replace_string[i + 2]);
  528. auto capture_postition_string = replace_string.substring_view(i + 1, is_two_digits ? 2 : 1);
  529. auto capture_position = capture_postition_string.to_uint();
  530. if (capture_position.has_value() && (*capture_position > 0) && (*capture_position <= captures.size())) {
  531. auto& value = captures[*capture_position - 1];
  532. if (!value.is_undefined()) {
  533. auto value_string = value.to_string(global_object);
  534. if (vm.exception())
  535. return {};
  536. result.append(value_string);
  537. }
  538. i += is_two_digits ? 2 : 1;
  539. } else {
  540. result.append(curr);
  541. }
  542. } else if (next == '<') {
  543. auto start_position = i + 2;
  544. auto end_position = replace_string.find('>', start_position);
  545. if (named_captures.is_undefined() || !end_position.has_value()) {
  546. result.append(curr);
  547. } else {
  548. auto group_name = replace_string.substring(start_position, *end_position - start_position);
  549. auto capture = named_captures.as_object().get(group_name);
  550. if (vm.exception())
  551. return {};
  552. if (!capture.is_undefined()) {
  553. auto capture_string = capture.to_string(global_object);
  554. if (vm.exception())
  555. return {};
  556. result.append(capture_string);
  557. }
  558. i = *end_position;
  559. }
  560. } else {
  561. result.append(curr);
  562. }
  563. }
  564. return result.build();
  565. }
  566. }