AbstractOperations.cpp 46 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021
  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/TemporaryChange.h>
  11. #include <AK/Utf16View.h>
  12. #include <LibJS/Bytecode/Interpreter.h>
  13. #include <LibJS/Interpreter.h>
  14. #include <LibJS/Parser.h>
  15. #include <LibJS/Runtime/AbstractOperations.h>
  16. #include <LibJS/Runtime/Accessor.h>
  17. #include <LibJS/Runtime/ArgumentsObject.h>
  18. #include <LibJS/Runtime/Array.h>
  19. #include <LibJS/Runtime/BoundFunction.h>
  20. #include <LibJS/Runtime/Completion.h>
  21. #include <LibJS/Runtime/DeclarativeEnvironment.h>
  22. #include <LibJS/Runtime/ECMAScriptFunctionObject.h>
  23. #include <LibJS/Runtime/ErrorTypes.h>
  24. #include <LibJS/Runtime/FunctionEnvironment.h>
  25. #include <LibJS/Runtime/FunctionObject.h>
  26. #include <LibJS/Runtime/GlobalObject.h>
  27. #include <LibJS/Runtime/Object.h>
  28. #include <LibJS/Runtime/ObjectEnvironment.h>
  29. #include <LibJS/Runtime/PropertyDescriptor.h>
  30. #include <LibJS/Runtime/PropertyKey.h>
  31. #include <LibJS/Runtime/ProxyObject.h>
  32. #include <LibJS/Runtime/Reference.h>
  33. namespace JS {
  34. // 7.2.1 RequireObjectCoercible ( argument ), https://tc39.es/ecma262/#sec-requireobjectcoercible
  35. ThrowCompletionOr<Value> require_object_coercible(GlobalObject& global_object, Value value)
  36. {
  37. auto& vm = global_object.vm();
  38. if (value.is_nullish())
  39. return vm.throw_completion<TypeError>(global_object, ErrorType::NotObjectCoercible, value.to_string_without_side_effects());
  40. return value;
  41. }
  42. // 7.3.14 Call ( F, V [ , argumentsList ] ), https://tc39.es/ecma262/#sec-call
  43. ThrowCompletionOr<Value> call_impl(GlobalObject& global_object, Value function, Value this_value, Optional<MarkedValueList> arguments_list)
  44. {
  45. auto& vm = global_object.vm();
  46. // 1. If argumentsList is not present, set argumentsList to a new empty List.
  47. if (!arguments_list.has_value())
  48. arguments_list = MarkedValueList { global_object.heap() };
  49. // 2. If IsCallable(F) is false, throw a TypeError exception.
  50. if (!function.is_function())
  51. return vm.throw_completion<TypeError>(global_object, ErrorType::NotAFunction, function.to_string_without_side_effects());
  52. // 3. Return ? F.[[Call]](V, argumentsList).
  53. return function.as_function().internal_call(this_value, move(*arguments_list));
  54. }
  55. // 7.3.15 Construct ( F [ , argumentsList [ , newTarget ] ] ), https://tc39.es/ecma262/#sec-construct
  56. ThrowCompletionOr<Object*> construct(GlobalObject& global_object, FunctionObject& function, Optional<MarkedValueList> arguments_list, FunctionObject* new_target)
  57. {
  58. // 1. If newTarget is not present, set newTarget to F.
  59. if (!new_target)
  60. new_target = &function;
  61. // 2. If argumentsList is not present, set argumentsList to a new empty List.
  62. if (!arguments_list.has_value())
  63. arguments_list = MarkedValueList { global_object.heap() };
  64. // 3. Return ? F.[[Construct]](argumentsList, newTarget).
  65. return function.internal_construct(move(*arguments_list), *new_target);
  66. }
  67. // 7.3.19 LengthOfArrayLike ( obj ), https://tc39.es/ecma262/#sec-lengthofarraylike
  68. ThrowCompletionOr<size_t> length_of_array_like(GlobalObject& global_object, Object const& object)
  69. {
  70. auto& vm = global_object.vm();
  71. auto result = TRY(object.get(vm.names.length));
  72. return result.to_length(global_object);
  73. }
  74. // 7.3.20 CreateListFromArrayLike ( obj [ , elementTypes ] ), https://tc39.es/ecma262/#sec-createlistfromarraylike
  75. ThrowCompletionOr<MarkedValueList> create_list_from_array_like(GlobalObject& global_object, Value value, Function<ThrowCompletionOr<void>(Value)> check_value)
  76. {
  77. auto& vm = global_object.vm();
  78. auto& heap = global_object.heap();
  79. // 1. If elementTypes is not present, set elementTypes to « Undefined, Null, Boolean, String, Symbol, Number, BigInt, Object ».
  80. // 2. If Type(obj) is not Object, throw a TypeError exception.
  81. if (!value.is_object())
  82. return vm.throw_completion<TypeError>(global_object, ErrorType::NotAnObject, value.to_string_without_side_effects());
  83. auto& array_like = value.as_object();
  84. // 3. Let len be ? LengthOfArrayLike(obj).
  85. auto length = TRY(length_of_array_like(global_object, array_like));
  86. // 4. Let list be a new empty List.
  87. auto list = MarkedValueList { heap };
  88. // 5. Let index be 0.
  89. // 6. Repeat, while index < len,
  90. for (size_t i = 0; i < length; ++i) {
  91. // a. Let indexName be ! ToString(𝔽(index)).
  92. auto index_name = PropertyKey { i };
  93. // b. Let next be ? Get(obj, indexName).
  94. auto next = TRY(array_like.get(index_name));
  95. // c. If Type(next) is not an element of elementTypes, throw a TypeError exception.
  96. if (check_value)
  97. TRY(check_value(next));
  98. // d. Append next as the last element of list.
  99. list.append(next);
  100. }
  101. // 7. Return list.
  102. return ThrowCompletionOr(move(list));
  103. }
  104. // 7.3.23 SpeciesConstructor ( O, defaultConstructor ), https://tc39.es/ecma262/#sec-speciesconstructor
  105. ThrowCompletionOr<FunctionObject*> species_constructor(GlobalObject& global_object, Object const& object, FunctionObject& default_constructor)
  106. {
  107. auto& vm = global_object.vm();
  108. // 1. Let C be ? Get(O, "constructor").
  109. auto constructor = TRY(object.get(vm.names.constructor));
  110. // 2. If C is undefined, return defaultConstructor.
  111. if (constructor.is_undefined())
  112. return &default_constructor;
  113. // 3. If Type(C) is not Object, throw a TypeError exception.
  114. if (!constructor.is_object())
  115. return vm.throw_completion<TypeError>(global_object, ErrorType::NotAConstructor, constructor.to_string_without_side_effects());
  116. // 4. Let S be ? Get(C, @@species).
  117. auto species = TRY(constructor.as_object().get(*vm.well_known_symbol_species()));
  118. // 5. If S is either undefined or null, return defaultConstructor.
  119. if (species.is_nullish())
  120. return &default_constructor;
  121. // 6. If IsConstructor(S) is true, return S.
  122. if (species.is_constructor())
  123. return &species.as_function();
  124. // 7. Throw a TypeError exception.
  125. return vm.throw_completion<TypeError>(global_object, ErrorType::NotAConstructor, species.to_string_without_side_effects());
  126. }
  127. // 7.3.25 GetFunctionRealm ( obj ), https://tc39.es/ecma262/#sec-getfunctionrealm
  128. ThrowCompletionOr<Realm*> get_function_realm(GlobalObject& global_object, FunctionObject const& function)
  129. {
  130. auto& vm = global_object.vm();
  131. // 1. Assert: ! IsCallable(obj) is true.
  132. // 2. If obj has a [[Realm]] internal slot, then
  133. if (function.realm()) {
  134. // a. Return obj.[[Realm]].
  135. return function.realm();
  136. }
  137. // 3. If obj is a bound function exotic object, then
  138. if (is<BoundFunction>(function)) {
  139. auto& bound_function = static_cast<BoundFunction const&>(function);
  140. // a. Let target be obj.[[BoundTargetFunction]].
  141. auto& target = bound_function.bound_target_function();
  142. // b. Return ? GetFunctionRealm(target).
  143. return get_function_realm(global_object, target);
  144. }
  145. // 4. If obj is a Proxy exotic object, then
  146. if (is<ProxyObject>(function)) {
  147. auto& proxy = static_cast<ProxyObject const&>(function);
  148. // a. If obj.[[ProxyHandler]] is null, throw a TypeError exception.
  149. if (proxy.is_revoked())
  150. return vm.throw_completion<TypeError>(global_object, ErrorType::ProxyRevoked);
  151. // b. Let proxyTarget be obj.[[ProxyTarget]].
  152. auto& proxy_target = proxy.target();
  153. // c. Return ? GetFunctionRealm(proxyTarget).
  154. VERIFY(proxy_target.is_function());
  155. return get_function_realm(global_object, static_cast<FunctionObject const&>(proxy_target));
  156. }
  157. // 5. Return the current Realm Record.
  158. return vm.current_realm();
  159. }
  160. // 8.5.2.1 InitializeBoundName ( name, value, environment ), 8.5.2.1 InitializeBoundName ( name, value, environment )
  161. ThrowCompletionOr<void> initialize_bound_name(GlobalObject& global_object, FlyString const& name, Value value, Environment* environment)
  162. {
  163. auto& vm = global_object.vm();
  164. // 1. If environment is not undefined, then
  165. if (environment) {
  166. // a. Perform environment.InitializeBinding(name, value).
  167. MUST(environment->initialize_binding(global_object, name, value));
  168. // b. Return NormalCompletion(undefined).
  169. return {};
  170. }
  171. // 2. Else,
  172. else {
  173. // a. Let lhs be ResolveBinding(name).
  174. auto lhs = vm.resolve_binding(name);
  175. // b. Return ? PutValue(lhs, value).
  176. return TRY(lhs.put_value(global_object, value));
  177. }
  178. VERIFY_NOT_REACHED();
  179. }
  180. // 10.1.6.2 IsCompatiblePropertyDescriptor ( Extensible, Desc, Current ), https://tc39.es/ecma262/#sec-iscompatiblepropertydescriptor
  181. bool is_compatible_property_descriptor(bool extensible, PropertyDescriptor const& descriptor, Optional<PropertyDescriptor> const& current)
  182. {
  183. // 1. Return ValidateAndApplyPropertyDescriptor(undefined, undefined, Extensible, Desc, Current).
  184. return validate_and_apply_property_descriptor(nullptr, {}, extensible, descriptor, current);
  185. }
  186. // 10.1.6.3 ValidateAndApplyPropertyDescriptor ( O, P, extensible, Desc, current ), https://tc39.es/ecma262/#sec-validateandapplypropertydescriptor
  187. bool validate_and_apply_property_descriptor(Object* object, PropertyKey const& property_name, bool extensible, PropertyDescriptor const& descriptor, Optional<PropertyDescriptor> const& current)
  188. {
  189. // 1. Assert: If O is not undefined, then IsPropertyKey(P) is true.
  190. if (object)
  191. VERIFY(property_name.is_valid());
  192. // 2. If current is undefined, then
  193. if (!current.has_value()) {
  194. // a. If extensible is false, return false.
  195. if (!extensible)
  196. return false;
  197. // b. Assert: extensible is true.
  198. // c. If IsGenericDescriptor(Desc) is true or IsDataDescriptor(Desc) is true, then
  199. if (descriptor.is_generic_descriptor() || descriptor.is_data_descriptor()) {
  200. // i. If O is not undefined, create an own data property named P of object O whose [[Value]], [[Writable]],
  201. // [[Enumerable]], and [[Configurable]] attribute values are described by Desc.
  202. // If the value of an attribute field of Desc is absent, the attribute of the newly created property is set
  203. // to its default value.
  204. if (object) {
  205. auto value = descriptor.value.value_or(js_undefined());
  206. object->storage_set(property_name, { value, descriptor.attributes() });
  207. }
  208. }
  209. // d. Else,
  210. else {
  211. // i. Assert: ! IsAccessorDescriptor(Desc) is true.
  212. VERIFY(descriptor.is_accessor_descriptor());
  213. // ii. If O is not undefined, create an own accessor property named P of object O whose [[Get]], [[Set]],
  214. // [[Enumerable]], and [[Configurable]] attribute values are described by Desc.
  215. // If the value of an attribute field of Desc is absent, the attribute of the newly created property is set
  216. // to its default value.
  217. if (object) {
  218. auto accessor = Accessor::create(object->vm(), descriptor.get.value_or(nullptr), descriptor.set.value_or(nullptr));
  219. object->storage_set(property_name, { accessor, descriptor.attributes() });
  220. }
  221. }
  222. // e. Return true.
  223. return true;
  224. }
  225. // 3. If every field in Desc is absent, return true.
  226. if (descriptor.is_empty())
  227. return true;
  228. // 4. If current.[[Configurable]] is false, then
  229. if (!*current->configurable) {
  230. // a. If Desc.[[Configurable]] is present and its value is true, return false.
  231. if (descriptor.configurable.has_value() && *descriptor.configurable)
  232. return false;
  233. // b. If Desc.[[Enumerable]] is present and ! SameValue(Desc.[[Enumerable]], current.[[Enumerable]]) is false, return false.
  234. if (descriptor.enumerable.has_value() && *descriptor.enumerable != *current->enumerable)
  235. return false;
  236. }
  237. // 5. If ! IsGenericDescriptor(Desc) is true, then
  238. if (descriptor.is_generic_descriptor()) {
  239. // a. NOTE: No further validation is required.
  240. }
  241. // 6. Else if ! SameValue(! IsDataDescriptor(current), ! IsDataDescriptor(Desc)) is false, then
  242. else if (current->is_data_descriptor() != descriptor.is_data_descriptor()) {
  243. // a. If current.[[Configurable]] is false, return false.
  244. if (!*current->configurable)
  245. return false;
  246. // b. If IsDataDescriptor(current) is true, then
  247. if (current->is_data_descriptor()) {
  248. // If O is not undefined, convert the property named P of object O from a data property to an accessor property.
  249. // Preserve the existing values of the converted property's [[Configurable]] and [[Enumerable]] attributes and
  250. // set the rest of the property's attributes to their default values.
  251. if (object) {
  252. auto accessor = Accessor::create(object->vm(), nullptr, nullptr);
  253. object->storage_set(property_name, { accessor, current->attributes() });
  254. }
  255. }
  256. // c. Else,
  257. else {
  258. // If O is not undefined, convert the property named P of object O from an accessor property to a data property.
  259. // Preserve the existing values of the converted property's [[Configurable]] and [[Enumerable]] attributes and
  260. // set the rest of the property's attributes to their default values.
  261. if (object) {
  262. auto value = js_undefined();
  263. object->storage_set(property_name, { value, current->attributes() });
  264. }
  265. }
  266. }
  267. // 7. Else if IsDataDescriptor(current) and IsDataDescriptor(Desc) are both true, then
  268. else if (current->is_data_descriptor() && descriptor.is_data_descriptor()) {
  269. // a. If current.[[Configurable]] is false and current.[[Writable]] is false, then
  270. if (!*current->configurable && !*current->writable) {
  271. // i. If Desc.[[Writable]] is present and Desc.[[Writable]] is true, return false.
  272. if (descriptor.writable.has_value() && *descriptor.writable)
  273. return false;
  274. // ii. If Desc.[[Value]] is present and SameValue(Desc.[[Value]], current.[[Value]]) is false, return false.
  275. if (descriptor.value.has_value() && !same_value(*descriptor.value, *current->value))
  276. return false;
  277. // iii. Return true.
  278. return true;
  279. }
  280. }
  281. // 8. Else,
  282. else {
  283. // a. Assert: ! IsAccessorDescriptor(current) and ! IsAccessorDescriptor(Desc) are both true.
  284. VERIFY(current->is_accessor_descriptor());
  285. VERIFY(descriptor.is_accessor_descriptor());
  286. // b. If current.[[Configurable]] is false, then
  287. if (!*current->configurable) {
  288. // i. If Desc.[[Set]] is present and SameValue(Desc.[[Set]], current.[[Set]]) is false, return false.
  289. if (descriptor.set.has_value() && *descriptor.set != *current->set)
  290. return false;
  291. // ii. If Desc.[[Get]] is present and SameValue(Desc.[[Get]], current.[[Get]]) is false, return false.
  292. if (descriptor.get.has_value() && *descriptor.get != *current->get)
  293. return false;
  294. // iii. Return true.
  295. return true;
  296. }
  297. }
  298. // 9. If O is not undefined, then
  299. if (object) {
  300. // 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.
  301. Value value;
  302. if (descriptor.is_accessor_descriptor() || (current->is_accessor_descriptor() && !descriptor.is_data_descriptor())) {
  303. auto* getter = descriptor.get.value_or(current->get.value_or(nullptr));
  304. auto* setter = descriptor.set.value_or(current->set.value_or(nullptr));
  305. value = Accessor::create(object->vm(), getter, setter);
  306. } else {
  307. value = descriptor.value.value_or(current->value.value_or({}));
  308. }
  309. PropertyAttributes attributes;
  310. attributes.set_writable(descriptor.writable.value_or(current->writable.value_or(false)));
  311. attributes.set_enumerable(descriptor.enumerable.value_or(current->enumerable.value_or(false)));
  312. attributes.set_configurable(descriptor.configurable.value_or(current->configurable.value_or(false)));
  313. object->storage_set(property_name, { value, attributes });
  314. }
  315. // 10. Return true.
  316. return true;
  317. }
  318. // 10.1.14 GetPrototypeFromConstructor ( constructor, intrinsicDefaultProto ), https://tc39.es/ecma262/#sec-getprototypefromconstructor
  319. ThrowCompletionOr<Object*> get_prototype_from_constructor(GlobalObject& global_object, FunctionObject const& constructor, Object* (GlobalObject::*intrinsic_default_prototype)())
  320. {
  321. auto& vm = global_object.vm();
  322. // 1. Assert: intrinsicDefaultProto is this specification's name of an intrinsic object. The corresponding object must be an intrinsic that is intended to be used as the [[Prototype]] value of an object.
  323. // 2. Let proto be ? Get(constructor, "prototype").
  324. auto prototype = TRY(constructor.get(vm.names.prototype));
  325. // 3. If Type(proto) is not Object, then
  326. if (!prototype.is_object()) {
  327. // a. Let realm be ? GetFunctionRealm(constructor).
  328. auto* realm = TRY(get_function_realm(global_object, constructor));
  329. // b. Set proto to realm's intrinsic object named intrinsicDefaultProto.
  330. prototype = (realm->global_object().*intrinsic_default_prototype)();
  331. }
  332. // 4. Return proto.
  333. return &prototype.as_object();
  334. }
  335. // 9.1.2.2 NewDeclarativeEnvironment ( E ), https://tc39.es/ecma262/#sec-newdeclarativeenvironment
  336. DeclarativeEnvironment* new_declarative_environment(Environment& environment)
  337. {
  338. auto& global_object = environment.global_object();
  339. return global_object.heap().allocate<DeclarativeEnvironment>(global_object, &environment);
  340. }
  341. // 9.1.2.3 NewObjectEnvironment ( O, W, E ), https://tc39.es/ecma262/#sec-newobjectenvironment
  342. ObjectEnvironment* new_object_environment(Object& object, bool is_with_environment, Environment* environment)
  343. {
  344. auto& global_object = object.global_object();
  345. return global_object.heap().allocate<ObjectEnvironment>(global_object, object, is_with_environment ? ObjectEnvironment::IsWithEnvironment::Yes : ObjectEnvironment::IsWithEnvironment::No, environment);
  346. }
  347. // 9.1.2.4 NewFunctionEnvironment ( F, newTarget ), https://tc39.es/ecma262/#sec-newfunctionenvironment
  348. FunctionEnvironment* new_function_environment(ECMAScriptFunctionObject& function, Object* new_target)
  349. {
  350. auto& global_object = function.global_object();
  351. // 1. Let env be a new function Environment Record containing no bindings.
  352. auto* env = global_object.heap().allocate<FunctionEnvironment>(global_object, function.environment());
  353. // 2. Set env.[[FunctionObject]] to F.
  354. env->set_function_object(function);
  355. // 3. If F.[[ThisMode]] is lexical, set env.[[ThisBindingStatus]] to lexical.
  356. if (function.this_mode() == ECMAScriptFunctionObject::ThisMode::Lexical)
  357. env->set_this_binding_status(FunctionEnvironment::ThisBindingStatus::Lexical);
  358. // 4. Else, set env.[[ThisBindingStatus]] to uninitialized.
  359. else
  360. env->set_this_binding_status(FunctionEnvironment::ThisBindingStatus::Uninitialized);
  361. // 5. Set env.[[NewTarget]] to newTarget.
  362. env->set_new_target(new_target ?: js_undefined());
  363. // 6. Set env.[[OuterEnv]] to F.[[Environment]].
  364. // NOTE: Done in step 1 via the FunctionEnvironment constructor.
  365. // 7. Return env.
  366. return env;
  367. }
  368. PrivateEnvironment* new_private_environment(VM& vm, PrivateEnvironment* outer)
  369. {
  370. return vm.heap().allocate<PrivateEnvironment>(vm.current_realm()->global_object(), outer);
  371. }
  372. // 9.4.3 GetThisEnvironment ( ), https://tc39.es/ecma262/#sec-getthisenvironment
  373. Environment& get_this_environment(VM& vm)
  374. {
  375. for (auto* env = vm.lexical_environment(); env; env = env->outer_environment()) {
  376. if (env->has_this_binding())
  377. return *env;
  378. }
  379. VERIFY_NOT_REACHED();
  380. }
  381. // 13.3.7.2 GetSuperConstructor ( ), https://tc39.es/ecma262/#sec-getsuperconstructor
  382. Object* get_super_constructor(VM& vm)
  383. {
  384. // 1. Let envRec be GetThisEnvironment().
  385. auto& env = get_this_environment(vm);
  386. // 2. Assert: envRec is a function Environment Record.
  387. // 3. Let activeFunction be envRec.[[FunctionObject]].
  388. // 4. Assert: activeFunction is an ECMAScript function object.
  389. auto& active_function = verify_cast<FunctionEnvironment>(env).function_object();
  390. // 5. Let superConstructor be ! activeFunction.[[GetPrototypeOf]]().
  391. auto* super_constructor = MUST(active_function.internal_get_prototype_of());
  392. // 6. Return superConstructor.
  393. return super_constructor;
  394. }
  395. // 13.3.7.3 MakeSuperPropertyReference ( actualThis, propertyKey, strict ), https://tc39.es/ecma262/#sec-makesuperpropertyreference
  396. ThrowCompletionOr<Reference> make_super_property_reference(GlobalObject& global_object, Value actual_this, PropertyKey const& property_key, bool strict)
  397. {
  398. auto& vm = global_object.vm();
  399. // 1. Let env be GetThisEnvironment().
  400. auto& env = verify_cast<FunctionEnvironment>(get_this_environment(vm));
  401. // 2. Assert: env.HasSuperBinding() is true.
  402. VERIFY(env.has_super_binding());
  403. // 3. Let baseValue be ? env.GetSuperBase().
  404. auto base_value = TRY(env.get_super_base());
  405. // 4. Let bv be ? RequireObjectCoercible(baseValue).
  406. auto bv = TRY(require_object_coercible(global_object, base_value));
  407. // 5. Return the Reference Record { [[Base]]: bv, [[ReferencedName]]: propertyKey, [[Strict]]: strict, [[ThisValue]]: actualThis }.
  408. // 6. NOTE: This returns a Super Reference Record.
  409. return Reference { bv, property_key, actual_this, strict };
  410. }
  411. // 19.2.1.1 PerformEval ( x, callerRealm, strictCaller, direct ), https://tc39.es/ecma262/#sec-performeval
  412. ThrowCompletionOr<Value> perform_eval(Value x, GlobalObject& caller_realm, CallerMode strict_caller, EvalMode direct)
  413. {
  414. VERIFY(direct == EvalMode::Direct || strict_caller == CallerMode::NonStrict);
  415. if (!x.is_string())
  416. return x;
  417. auto& vm = caller_realm.vm();
  418. auto& eval_realm = vm.running_execution_context().realm;
  419. auto& code_string = x.as_string();
  420. Parser parser { Lexer { code_string.string() } };
  421. auto program = parser.parse_program(strict_caller == CallerMode::Strict);
  422. if (parser.has_errors()) {
  423. auto& error = parser.errors()[0];
  424. return vm.throw_completion<SyntaxError>(caller_realm, error.to_string());
  425. }
  426. auto strict_eval = strict_caller == CallerMode::Strict;
  427. if (program->is_strict_mode())
  428. strict_eval = true;
  429. auto& running_context = vm.running_execution_context();
  430. Environment* lexical_environment;
  431. Environment* variable_environment;
  432. PrivateEnvironment* private_environment;
  433. if (direct == EvalMode::Direct) {
  434. lexical_environment = new_declarative_environment(*running_context.lexical_environment);
  435. variable_environment = running_context.variable_environment;
  436. private_environment = running_context.private_environment;
  437. } else {
  438. lexical_environment = new_declarative_environment(eval_realm->global_environment());
  439. variable_environment = &eval_realm->global_environment();
  440. private_environment = nullptr;
  441. }
  442. if (strict_eval)
  443. variable_environment = lexical_environment;
  444. if (direct == EvalMode::Direct && !strict_eval) {
  445. // NOTE: Non-strict direct eval() forces us to deoptimize variable accesses.
  446. // Mark the variable environment chain as screwed since we will not be able
  447. // to rely on cached environment coordinates from this point on.
  448. variable_environment->set_permanently_screwed_by_eval();
  449. }
  450. // 18. If runningContext is not already suspended, suspend runningContext.
  451. // FIXME: We don't have this concept yet.
  452. ExecutionContext eval_context(vm.heap());
  453. eval_context.realm = eval_realm;
  454. eval_context.variable_environment = variable_environment;
  455. eval_context.lexical_environment = lexical_environment;
  456. eval_context.private_environment = private_environment;
  457. TRY(vm.push_execution_context(eval_context, eval_realm->global_object()));
  458. ScopeGuard pop_guard = [&] {
  459. vm.pop_execution_context();
  460. };
  461. TRY(eval_declaration_instantiation(vm, eval_realm->global_object(), program, variable_environment, lexical_environment, private_environment, strict_eval));
  462. TemporaryChange scope_change_strict(vm.running_execution_context().is_strict_mode, strict_eval);
  463. Value eval_result;
  464. if (auto* bytecode_interpreter = Bytecode::Interpreter::current()) {
  465. auto executable = JS::Bytecode::Generator::generate(program);
  466. executable.name = "eval"sv;
  467. if (JS::Bytecode::g_dump_bytecode)
  468. executable.dump();
  469. eval_result = TRY(bytecode_interpreter->run(executable));
  470. } else {
  471. auto& ast_interpreter = vm.interpreter();
  472. // FIXME: We need to use evaluate_statements() here because Program::execute() calls global_declaration_instantiation() when it shouldn't
  473. eval_result = program->evaluate_statements(ast_interpreter, caller_realm);
  474. }
  475. if (auto* exception = vm.exception())
  476. return throw_completion(exception->value());
  477. else
  478. return eval_result.value_or(js_undefined());
  479. }
  480. // 19.2.1.3 EvalDeclarationInstantiation ( body, varEnv, lexEnv, privateEnv, strict ), https://tc39.es/ecma262/#sec-evaldeclarationinstantiation
  481. ThrowCompletionOr<void> eval_declaration_instantiation(VM& vm, GlobalObject& global_object, Program const& program, Environment* variable_environment, Environment* lexical_environment, PrivateEnvironment* private_environment, bool strict)
  482. {
  483. // FIXME: I'm not sure if the global object is correct here. And this is quite a crucial spot!
  484. GlobalEnvironment* global_var_environment = variable_environment->is_global_environment() ? static_cast<GlobalEnvironment*>(variable_environment) : nullptr;
  485. if (!strict) {
  486. if (global_var_environment) {
  487. program.for_each_var_declared_name([&](auto const& name) {
  488. if (global_var_environment->has_lexical_declaration(name)) {
  489. vm.throw_exception<SyntaxError>(global_object, ErrorType::TopLevelVariableAlreadyDeclared, name);
  490. return IterationDecision::Break;
  491. }
  492. return IterationDecision::Continue;
  493. });
  494. }
  495. auto* this_environment = lexical_environment;
  496. while (this_environment != variable_environment) {
  497. if (!is<ObjectEnvironment>(*this_environment)) {
  498. program.for_each_var_declared_name([&](auto const& name) {
  499. if (MUST(this_environment->has_binding(name))) {
  500. vm.throw_exception<SyntaxError>(global_object, ErrorType::TopLevelVariableAlreadyDeclared, name);
  501. return IterationDecision::Break;
  502. }
  503. // FIXME: NOTE: Annex B.3.4 defines alternate semantics for the above step.
  504. // In particular it only throw the syntax error if it is not an environment from a catchclause.
  505. return IterationDecision::Continue;
  506. });
  507. if (auto* exception = vm.exception())
  508. return throw_completion(exception->value());
  509. }
  510. this_environment = this_environment->outer_environment();
  511. VERIFY(this_environment);
  512. }
  513. }
  514. // FIXME: Add Private identifiers check here.
  515. HashTable<FlyString> declared_function_names;
  516. Vector<FunctionDeclaration const&> functions_to_initialize;
  517. program.for_each_var_function_declaration_in_reverse_order([&](FunctionDeclaration const& function) {
  518. if (declared_function_names.set(function.name()) != AK::HashSetResult::InsertedNewEntry)
  519. return IterationDecision::Continue;
  520. if (global_var_environment) {
  521. auto function_definable_or_error = global_var_environment->can_declare_global_function(function.name());
  522. if (function_definable_or_error.is_error())
  523. return IterationDecision::Break;
  524. auto function_definable = function_definable_or_error.release_value();
  525. if (!function_definable) {
  526. vm.throw_exception<TypeError>(global_object, ErrorType::CannotDeclareGlobalFunction, function.name());
  527. return IterationDecision::Break;
  528. }
  529. }
  530. functions_to_initialize.append(function);
  531. return IterationDecision::Continue;
  532. });
  533. if (auto* exception = vm.exception())
  534. return throw_completion(exception->value());
  535. if (!strict) {
  536. // The spec here uses 'declaredVarNames' but that has not been declared yet.
  537. HashTable<FlyString> hoisted_functions;
  538. program.for_each_function_hoistable_with_annexB_extension([&](FunctionDeclaration& function_declaration) {
  539. auto& function_name = function_declaration.name();
  540. auto* this_environment = lexical_environment;
  541. while (this_environment != variable_environment) {
  542. if (!is<ObjectEnvironment>(*this_environment) && MUST(this_environment->has_binding(function_name)))
  543. return IterationDecision::Continue;
  544. this_environment = this_environment->outer_environment();
  545. VERIFY(this_environment);
  546. }
  547. if (global_var_environment) {
  548. if (global_var_environment->has_lexical_declaration(function_name))
  549. return IterationDecision::Continue;
  550. auto var_definable_or_error = global_var_environment->can_declare_global_var(function_name);
  551. if (var_definable_or_error.is_error())
  552. return IterationDecision::Break;
  553. auto var_definable = var_definable_or_error.release_value();
  554. if (!var_definable)
  555. return IterationDecision::Continue;
  556. }
  557. if (!declared_function_names.contains(function_name) && !hoisted_functions.contains(function_name)) {
  558. if (global_var_environment) {
  559. global_var_environment->create_global_var_binding(function_name, true);
  560. if (vm.exception())
  561. return IterationDecision::Break;
  562. } else {
  563. if (!MUST(variable_environment->has_binding(function_name))) {
  564. MUST(variable_environment->create_mutable_binding(global_object, function_name, true));
  565. MUST(variable_environment->initialize_binding(global_object, function_name, js_undefined()));
  566. }
  567. }
  568. hoisted_functions.set(function_name);
  569. }
  570. function_declaration.set_should_do_additional_annexB_steps();
  571. return IterationDecision::Continue;
  572. });
  573. if (auto* exception = vm.exception())
  574. return throw_completion(exception->value());
  575. }
  576. HashTable<FlyString> declared_var_names;
  577. program.for_each_var_scoped_variable_declaration([&](VariableDeclaration const& declaration) {
  578. declaration.for_each_bound_name([&](auto const& name) {
  579. if (!declared_function_names.contains(name)) {
  580. if (global_var_environment) {
  581. auto variable_definable_or_error = global_var_environment->can_declare_global_var(name);
  582. if (variable_definable_or_error.is_error())
  583. return IterationDecision::Break;
  584. auto variable_definable = variable_definable_or_error.release_value();
  585. if (!variable_definable) {
  586. vm.throw_exception<TypeError>(global_object, ErrorType::CannotDeclareGlobalVariable, name);
  587. return IterationDecision::Break;
  588. }
  589. }
  590. declared_var_names.set(name);
  591. }
  592. return IterationDecision::Continue;
  593. });
  594. if (vm.exception())
  595. return IterationDecision::Break;
  596. return IterationDecision::Continue;
  597. });
  598. if (auto* exception = vm.exception())
  599. return throw_completion(exception->value());
  600. // 14. NOTE: No abnormal terminations occur after this algorithm step unless varEnv is a global Environment Record and the global object is a Proxy exotic object.
  601. program.for_each_lexically_scoped_declaration([&](Declaration const& declaration) {
  602. declaration.for_each_bound_name([&](auto const& name) {
  603. if (declaration.is_constant_declaration())
  604. (void)lexical_environment->create_immutable_binding(global_object, name, true);
  605. else
  606. (void)lexical_environment->create_mutable_binding(global_object, name, false);
  607. if (vm.exception())
  608. return IterationDecision::Break;
  609. return IterationDecision::Continue;
  610. });
  611. if (vm.exception())
  612. return IterationDecision::Break;
  613. return IterationDecision::Continue;
  614. });
  615. if (auto* exception = vm.exception())
  616. return throw_completion(exception->value());
  617. for (auto& declaration : functions_to_initialize) {
  618. auto* function = ECMAScriptFunctionObject::create(global_object, declaration.name(), declaration.body(), declaration.parameters(), declaration.function_length(), lexical_environment, private_environment, declaration.kind(), declaration.is_strict_mode(), declaration.might_need_arguments_object());
  619. if (global_var_environment) {
  620. global_var_environment->create_global_function_binding(declaration.name(), function, true);
  621. if (auto* exception = vm.exception())
  622. return throw_completion(exception->value());
  623. } else {
  624. auto binding_exists = MUST(variable_environment->has_binding(declaration.name()));
  625. if (!binding_exists) {
  626. TRY(variable_environment->create_mutable_binding(global_object, declaration.name(), true));
  627. TRY(variable_environment->initialize_binding(global_object, declaration.name(), function));
  628. } else {
  629. TRY(variable_environment->set_mutable_binding(global_object, declaration.name(), function, false));
  630. }
  631. }
  632. }
  633. for (auto& var_name : declared_var_names) {
  634. if (global_var_environment) {
  635. global_var_environment->create_global_var_binding(var_name, true);
  636. if (auto* exception = vm.exception())
  637. return throw_completion(exception->value());
  638. } else {
  639. auto binding_exists = MUST(variable_environment->has_binding(var_name));
  640. if (!binding_exists) {
  641. TRY(variable_environment->create_mutable_binding(global_object, var_name, true));
  642. TRY(variable_environment->initialize_binding(global_object, var_name, js_undefined()));
  643. }
  644. }
  645. }
  646. return {};
  647. }
  648. // 10.4.4.6 CreateUnmappedArgumentsObject ( argumentsList ), https://tc39.es/ecma262/#sec-createunmappedargumentsobject
  649. Object* create_unmapped_arguments_object(GlobalObject& global_object, Span<Value> arguments)
  650. {
  651. auto& vm = global_object.vm();
  652. // 1. Let len be the number of elements in argumentsList.
  653. auto length = arguments.size();
  654. // 2. Let obj be ! OrdinaryObjectCreate(%Object.prototype%, « [[ParameterMap]] »).
  655. // 3. Set obj.[[ParameterMap]] to undefined.
  656. auto* object = Object::create(global_object, global_object.object_prototype());
  657. object->set_has_parameter_map();
  658. // 4. Perform DefinePropertyOrThrow(obj, "length", PropertyDescriptor { [[Value]]: 𝔽(len), [[Writable]]: true, [[Enumerable]]: false, [[Configurable]]: true }).
  659. MUST(object->define_property_or_throw(vm.names.length, { .value = Value(length), .writable = true, .enumerable = false, .configurable = true }));
  660. // 5. Let index be 0.
  661. // 6. Repeat, while index < len,
  662. for (size_t index = 0; index < length; ++index) {
  663. // a. Let val be argumentsList[index].
  664. auto value = arguments[index];
  665. // b. Perform ! CreateDataPropertyOrThrow(obj, ! ToString(𝔽(index)), val).
  666. MUST(object->create_data_property_or_throw(index, value));
  667. // c. Set index to index + 1.
  668. }
  669. // 7. Perform ! DefinePropertyOrThrow(obj, @@iterator, PropertyDescriptor { [[Value]]: %Array.prototype.values%, [[Writable]]: true, [[Enumerable]]: false, [[Configurable]]: true }).
  670. auto* array_prototype_values = global_object.array_prototype_values_function();
  671. MUST(object->define_property_or_throw(*vm.well_known_symbol_iterator(), { .value = array_prototype_values, .writable = true, .enumerable = false, .configurable = true }));
  672. // 8. Perform ! DefinePropertyOrThrow(obj, "callee", PropertyDescriptor { [[Get]]: %ThrowTypeError%, [[Set]]: %ThrowTypeError%, [[Enumerable]]: false, [[Configurable]]: false }).
  673. auto* throw_type_error = global_object.throw_type_error_function();
  674. MUST(object->define_property_or_throw(vm.names.callee, { .get = throw_type_error, .set = throw_type_error, .enumerable = false, .configurable = false }));
  675. // 9. Return obj.
  676. return object;
  677. }
  678. // 10.4.4.7 CreateMappedArgumentsObject ( func, formals, argumentsList, env ), https://tc39.es/ecma262/#sec-createmappedargumentsobject
  679. Object* create_mapped_arguments_object(GlobalObject& global_object, FunctionObject& function, Vector<FunctionNode::Parameter> const& formals, Span<Value> arguments, Environment& environment)
  680. {
  681. auto& vm = global_object.vm();
  682. // 1. Assert: formals does not contain a rest parameter, any binding patterns, or any initializers. It may contain duplicate identifiers.
  683. // 2. Let len be the number of elements in argumentsList.
  684. VERIFY(arguments.size() <= NumericLimits<i32>::max());
  685. i32 length = static_cast<i32>(arguments.size());
  686. // 3. Let obj be ! MakeBasicObject(« [[Prototype]], [[Extensible]], [[ParameterMap]] »).
  687. // 4. Set obj.[[GetOwnProperty]] as specified in 10.4.4.1.
  688. // 5. Set obj.[[DefineOwnProperty]] as specified in 10.4.4.2.
  689. // 6. Set obj.[[Get]] as specified in 10.4.4.3.
  690. // 7. Set obj.[[Set]] as specified in 10.4.4.4.
  691. // 8. Set obj.[[Delete]] as specified in 10.4.4.5.
  692. // 9. Set obj.[[Prototype]] to %Object.prototype%.
  693. auto* object = vm.heap().allocate<ArgumentsObject>(global_object, global_object, environment);
  694. VERIFY(!vm.exception());
  695. // 14. Let index be 0.
  696. // 15. Repeat, while index < len,
  697. for (i32 index = 0; index < length; ++index) {
  698. // a. Let val be argumentsList[index].
  699. auto value = arguments[index];
  700. // b. Perform ! CreateDataPropertyOrThrow(obj, ! ToString(𝔽(index)), val).
  701. MUST(object->create_data_property_or_throw(index, value));
  702. // c. Set index to index + 1.
  703. }
  704. // 16. Perform ! DefinePropertyOrThrow(obj, "length", PropertyDescriptor { [[Value]]: 𝔽(len), [[Writable]]: true, [[Enumerable]]: false, [[Configurable]]: true }).
  705. MUST(object->define_property_or_throw(vm.names.length, { .value = Value(length), .writable = true, .enumerable = false, .configurable = true }));
  706. // 17. Let mappedNames be a new empty List.
  707. HashTable<FlyString> mapped_names;
  708. // 18. Set index to numberOfParameters - 1.
  709. // 19. Repeat, while index ≥ 0,
  710. VERIFY(formals.size() <= NumericLimits<i32>::max());
  711. for (i32 index = static_cast<i32>(formals.size()) - 1; index >= 0; --index) {
  712. // a. Let name be parameterNames[index].
  713. auto const& name = formals[index].binding.get<FlyString>();
  714. // b. If name is not an element of mappedNames, then
  715. if (mapped_names.contains(name))
  716. continue;
  717. // i. Add name as an element of the list mappedNames.
  718. mapped_names.set(name);
  719. // ii. If index < len, then
  720. if (index < length) {
  721. // 1. Let g be MakeArgGetter(name, env).
  722. // 2. Let p be MakeArgSetter(name, env).
  723. // 3. Perform map.[[DefineOwnProperty]](! ToString(𝔽(index)), PropertyDescriptor { [[Set]]: p, [[Get]]: g, [[Enumerable]]: false, [[Configurable]]: true }).
  724. object->parameter_map().define_native_accessor(
  725. PropertyKey { index },
  726. [&environment, name](VM&, GlobalObject& global_object_getter) -> JS::ThrowCompletionOr<Value> {
  727. return MUST(environment.get_binding_value(global_object_getter, name, false));
  728. },
  729. [&environment, name](VM& vm, GlobalObject& global_object_setter) {
  730. MUST(environment.set_mutable_binding(global_object_setter, name, vm.argument(0), false));
  731. return js_undefined();
  732. },
  733. Attribute::Configurable);
  734. }
  735. }
  736. // 20. Perform ! DefinePropertyOrThrow(obj, @@iterator, PropertyDescriptor { [[Value]]: %Array.prototype.values%, [[Writable]]: true, [[Enumerable]]: false, [[Configurable]]: true }).
  737. auto* array_prototype_values = global_object.array_prototype_values_function();
  738. MUST(object->define_property_or_throw(*vm.well_known_symbol_iterator(), { .value = array_prototype_values, .writable = true, .enumerable = false, .configurable = true }));
  739. // 21. Perform ! DefinePropertyOrThrow(obj, "callee", PropertyDescriptor { [[Value]]: func, [[Writable]]: true, [[Enumerable]]: false, [[Configurable]]: true }).
  740. MUST(object->define_property_or_throw(vm.names.callee, { .value = &function, .writable = true, .enumerable = false, .configurable = true }));
  741. // 22. Return obj.
  742. return object;
  743. }
  744. // 7.1.21 CanonicalNumericIndexString ( argument ), https://tc39.es/ecma262/#sec-canonicalnumericindexstring
  745. Value canonical_numeric_index_string(GlobalObject& global_object, PropertyKey const& property_name)
  746. {
  747. // NOTE: If the property name is a number type (An implementation-defined optimized
  748. // property key type), it can be treated as a string property that has already been
  749. // converted successfully into a canonical numeric index.
  750. VERIFY(property_name.is_string() || property_name.is_number());
  751. if (property_name.is_number())
  752. return Value(property_name.as_number());
  753. // 1. Assert: Type(argument) is String.
  754. auto argument = Value(js_string(global_object.vm(), property_name.as_string()));
  755. // 2. If argument is "-0", return -0𝔽.
  756. if (argument.as_string().string() == "-0")
  757. return Value(-0.0);
  758. // 3. Let n be ! ToNumber(argument).
  759. auto n = MUST(argument.to_number(global_object));
  760. // 4. If SameValue(! ToString(n), argument) is false, return undefined.
  761. if (!same_value(MUST(n.to_primitive_string(global_object)), argument))
  762. return js_undefined();
  763. // 5. Return n.
  764. return n;
  765. }
  766. // 22.1.3.17.1 GetSubstitution ( matched, str, position, captures, namedCaptures, replacement ), https://tc39.es/ecma262/#sec-getsubstitution
  767. ThrowCompletionOr<String> get_substitution(GlobalObject& global_object, Utf16View const& matched, Utf16View const& str, size_t position, Span<Value> captures, Value named_captures, Value replacement)
  768. {
  769. auto replace_string = TRY(replacement.to_utf16_string(global_object));
  770. auto replace_view = replace_string.view();
  771. StringBuilder result;
  772. for (size_t i = 0; i < replace_view.length_in_code_units(); ++i) {
  773. u16 curr = replace_view.code_unit_at(i);
  774. if ((curr != '$') || (i + 1 >= replace_view.length_in_code_units())) {
  775. result.append(curr);
  776. continue;
  777. }
  778. u16 next = replace_view.code_unit_at(i + 1);
  779. if (next == '$') {
  780. result.append('$');
  781. ++i;
  782. } else if (next == '&') {
  783. result.append(matched);
  784. ++i;
  785. } else if (next == '`') {
  786. auto substring = str.substring_view(0, position);
  787. result.append(substring);
  788. ++i;
  789. } else if (next == '\'') {
  790. auto tail_pos = position + matched.length_in_code_units();
  791. if (tail_pos < str.length_in_code_units()) {
  792. auto substring = str.substring_view(tail_pos);
  793. result.append(substring);
  794. }
  795. ++i;
  796. } else if (is_ascii_digit(next)) {
  797. bool is_two_digits = (i + 2 < replace_view.length_in_code_units()) && is_ascii_digit(replace_view.code_unit_at(i + 2));
  798. auto capture_postition_string = replace_view.substring_view(i + 1, is_two_digits ? 2 : 1).to_utf8();
  799. auto capture_position = capture_postition_string.to_uint();
  800. if (capture_position.has_value() && (*capture_position > 0) && (*capture_position <= captures.size())) {
  801. auto& value = captures[*capture_position - 1];
  802. if (!value.is_undefined()) {
  803. auto value_string = TRY(value.to_string(global_object));
  804. result.append(value_string);
  805. }
  806. i += is_two_digits ? 2 : 1;
  807. } else {
  808. result.append(curr);
  809. }
  810. } else if (next == '<') {
  811. auto start_position = i + 2;
  812. Optional<size_t> end_position;
  813. for (size_t j = start_position; j < replace_view.length_in_code_units(); ++j) {
  814. if (replace_view.code_unit_at(j) == '>') {
  815. end_position = j;
  816. break;
  817. }
  818. }
  819. if (named_captures.is_undefined() || !end_position.has_value()) {
  820. result.append(curr);
  821. } else {
  822. auto group_name_view = replace_view.substring_view(start_position, *end_position - start_position);
  823. auto group_name = group_name_view.to_utf8(Utf16View::AllowInvalidCodeUnits::Yes);
  824. auto capture = TRY(named_captures.as_object().get(group_name));
  825. if (!capture.is_undefined()) {
  826. auto capture_string = TRY(capture.to_string(global_object));
  827. result.append(capture_string);
  828. }
  829. i = *end_position;
  830. }
  831. } else {
  832. result.append(curr);
  833. }
  834. }
  835. return result.build();
  836. }
  837. }