BindingPatternAssignmentExpression.cs 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421
  1. using System;
  2. using System.Collections.Generic;
  3. using Esprima.Ast;
  4. using Jint.Native;
  5. using Jint.Native.Array;
  6. using Jint.Native.Function;
  7. using Jint.Native.Iterator;
  8. using Jint.Runtime.Environments;
  9. using Jint.Runtime.References;
  10. namespace Jint.Runtime.Interpreter.Expressions
  11. {
  12. internal sealed class BindingPatternAssignmentExpression : JintExpression
  13. {
  14. private readonly BindingPattern _pattern;
  15. private JintExpression _right;
  16. public BindingPatternAssignmentExpression(
  17. Engine engine,
  18. AssignmentExpression expression) : base(engine, expression)
  19. {
  20. _pattern = (BindingPattern) expression.Left;
  21. _initialized = false;
  22. }
  23. protected override void Initialize()
  24. {
  25. _right = Build(_engine, ((AssignmentExpression) _expression).Right);
  26. }
  27. protected override object EvaluateInternal()
  28. {
  29. var rightValue = _right.GetValue();
  30. ProcessPatterns(_engine, _pattern, rightValue, null);
  31. return rightValue;
  32. }
  33. internal static void ProcessPatterns(
  34. Engine engine,
  35. BindingPattern pattern,
  36. JsValue argument,
  37. EnvironmentRecord environment,
  38. bool checkObjectPatternPropertyReference = true)
  39. {
  40. if (pattern is ArrayPattern ap)
  41. {
  42. HandleArrayPattern(engine, ap, argument, environment);
  43. }
  44. else if (pattern is ObjectPattern op)
  45. {
  46. HandleObjectPattern(engine, op, argument, environment, checkObjectPatternPropertyReference);
  47. }
  48. }
  49. private static bool ConsumeFromIterator(IIterator it, out JsValue value, out bool done)
  50. {
  51. value = JsValue.Undefined;
  52. done = false;
  53. if (!it.TryIteratorStep(out var d))
  54. {
  55. done = true;
  56. return false;
  57. }
  58. d.TryGetValue(CommonProperties.Value, out value);
  59. return true;
  60. }
  61. private static void HandleArrayPattern(Engine engine, ArrayPattern pattern, JsValue argument, EnvironmentRecord environment)
  62. {
  63. var realm = engine.Realm;
  64. var obj = TypeConverter.ToObject(realm, argument);
  65. ArrayOperations arrayOperations = null;
  66. IIterator iterator = null;
  67. // optimize for array unless someone has touched the iterator
  68. if (obj.IsArrayLike && obj.HasOriginalIterator)
  69. {
  70. arrayOperations = ArrayOperations.For(obj);
  71. }
  72. else
  73. {
  74. if (!obj.TryGetIterator(realm, out iterator))
  75. {
  76. ExceptionHelper.ThrowTypeError(realm);
  77. return;
  78. }
  79. }
  80. var completionType = CompletionType.Normal;
  81. var close = false;
  82. var done = false;
  83. uint i = 0;
  84. try
  85. {
  86. for (; i < pattern.Elements.Count; i++)
  87. {
  88. var left = pattern.Elements[(int) i];
  89. if (left is null)
  90. {
  91. if (arrayOperations != null)
  92. {
  93. arrayOperations.TryGetValue(i, out _);
  94. }
  95. else
  96. {
  97. if (!ConsumeFromIterator(iterator, out _, out done))
  98. {
  99. break;
  100. }
  101. }
  102. // skip assignment
  103. continue;
  104. }
  105. if (left is Identifier identifier)
  106. {
  107. JsValue value;
  108. if (arrayOperations != null)
  109. {
  110. arrayOperations.TryGetValue(i, out value);
  111. }
  112. else
  113. {
  114. if (!ConsumeFromIterator(iterator, out value, out done))
  115. {
  116. break;
  117. }
  118. }
  119. AssignToIdentifier(engine, identifier.Name, value, environment);
  120. }
  121. else if (left is MemberExpression me)
  122. {
  123. close = true;
  124. var reference = GetReferenceFromMember(engine, me);
  125. JsValue value;
  126. if (arrayOperations != null)
  127. {
  128. arrayOperations.TryGetValue(i, out value);
  129. }
  130. else
  131. {
  132. ConsumeFromIterator(iterator, out value, out done);
  133. }
  134. AssignToReference(engine, reference, value, environment);
  135. }
  136. else if (left is BindingPattern bindingPattern)
  137. {
  138. JsValue value;
  139. if (arrayOperations != null)
  140. {
  141. arrayOperations.TryGetValue(i, out value);
  142. }
  143. else
  144. {
  145. iterator.TryIteratorStep(out var temp);
  146. value = temp;
  147. }
  148. ProcessPatterns(engine, bindingPattern, value, environment);
  149. }
  150. else if (left is RestElement restElement)
  151. {
  152. close = true;
  153. Reference reference = null;
  154. if (restElement.Argument is MemberExpression memberExpression)
  155. {
  156. reference = GetReferenceFromMember(engine, memberExpression);
  157. }
  158. ArrayInstance array;
  159. if (arrayOperations != null)
  160. {
  161. var length = arrayOperations.GetLength();
  162. array = engine.Realm.Intrinsics.Array.ConstructFast(length - i);
  163. for (uint j = i; j < length; ++j)
  164. {
  165. arrayOperations.TryGetValue(j, out var indexValue);
  166. array.SetIndexValue(j - i, indexValue, updateLength: false);
  167. }
  168. }
  169. else
  170. {
  171. array = engine.Realm.Intrinsics.Array.ConstructFast(0);
  172. uint index = 0;
  173. done = true;
  174. do
  175. {
  176. if (!iterator.TryIteratorStep(out var item))
  177. {
  178. done = true;
  179. break;
  180. }
  181. var value = item.Get(CommonProperties.Value);
  182. array.SetIndexValue(index++, value, updateLength: false);
  183. } while (true);
  184. array.SetLength(index);
  185. }
  186. if (restElement.Argument is Identifier leftIdentifier)
  187. {
  188. AssignToIdentifier(engine, leftIdentifier.Name, array, environment);
  189. }
  190. else if (restElement.Argument is BindingPattern bp)
  191. {
  192. ProcessPatterns(engine, bp, array, environment);
  193. }
  194. else
  195. {
  196. AssignToReference(engine, reference, array, environment);
  197. }
  198. }
  199. else if (left is AssignmentPattern assignmentPattern)
  200. {
  201. JsValue value;
  202. if (arrayOperations != null)
  203. {
  204. arrayOperations.TryGetValue(i, out value);
  205. }
  206. else
  207. {
  208. ConsumeFromIterator(iterator, out value, out done);
  209. }
  210. if (value.IsUndefined())
  211. {
  212. var jintExpression = Build(engine, assignmentPattern.Right);
  213. value = jintExpression.GetValue();
  214. }
  215. if (assignmentPattern.Left is Identifier leftIdentifier)
  216. {
  217. if (assignmentPattern.Right.IsFunctionDefinition())
  218. {
  219. ((FunctionInstance) value).SetFunctionName(new JsString(leftIdentifier.Name));
  220. }
  221. AssignToIdentifier(engine, leftIdentifier.Name, value, environment);
  222. }
  223. else if (assignmentPattern.Left is BindingPattern bp)
  224. {
  225. ProcessPatterns(engine, bp, value, environment);
  226. }
  227. }
  228. else
  229. {
  230. ExceptionHelper.ThrowArgumentOutOfRangeException("pattern",
  231. "Unable to determine how to handle array pattern element " + left);
  232. break;
  233. }
  234. }
  235. close = true;
  236. }
  237. catch
  238. {
  239. completionType = CompletionType.Throw;
  240. throw;
  241. }
  242. finally
  243. {
  244. if (close && !done)
  245. {
  246. iterator?.Close(completionType);
  247. }
  248. }
  249. }
  250. private static void HandleObjectPattern(Engine engine, ObjectPattern pattern, JsValue argument, EnvironmentRecord environment, bool checkReference)
  251. {
  252. var processedProperties = pattern.Properties.Count > 0 && pattern.Properties[pattern.Properties.Count - 1] is RestElement
  253. ? new HashSet<JsValue>()
  254. : null;
  255. var source = TypeConverter.ToObject(engine.Realm, argument);
  256. for (var i = 0; i < pattern.Properties.Count; i++)
  257. {
  258. if (pattern.Properties[i] is Property p)
  259. {
  260. JsValue sourceKey;
  261. var identifier = p.Key as Identifier;
  262. if (identifier == null || p.Computed)
  263. {
  264. var keyExpression = Build(engine, p.Key);
  265. sourceKey = TypeConverter.ToPropertyKey(keyExpression.GetValue());
  266. }
  267. else
  268. {
  269. sourceKey = identifier.Name;
  270. }
  271. processedProperties?.Add(sourceKey);
  272. if (p.Value is AssignmentPattern assignmentPattern)
  273. {
  274. source.TryGetValue(sourceKey, out var value);
  275. if (value.IsUndefined())
  276. {
  277. var jintExpression = Build(engine, assignmentPattern.Right);
  278. value = jintExpression.GetValue();
  279. }
  280. if (assignmentPattern.Left is BindingPattern bp)
  281. {
  282. ProcessPatterns(engine, bp, value, environment);
  283. continue;
  284. }
  285. var target = assignmentPattern.Left as Identifier ?? identifier;
  286. if (assignmentPattern.Right.IsFunctionDefinition())
  287. {
  288. ((FunctionInstance) value).SetFunctionName(target.Name);
  289. }
  290. AssignToIdentifier(engine, target.Name, value, environment);
  291. }
  292. else if (p.Value is BindingPattern bindingPattern)
  293. {
  294. source.TryGetValue(sourceKey, out var value);
  295. ProcessPatterns(engine, bindingPattern, value, environment);
  296. }
  297. else if (p.Value is MemberExpression memberExpression)
  298. {
  299. var reference = GetReferenceFromMember(engine, memberExpression);
  300. source.TryGetValue(sourceKey, out var value);
  301. AssignToReference(engine, reference, value, environment);
  302. }
  303. else
  304. {
  305. var identifierReference = p.Value as Identifier;
  306. var target = identifierReference ?? identifier;
  307. source.TryGetValue(sourceKey, out var v);
  308. AssignToIdentifier(engine, target.Name, v, environment, checkReference);
  309. }
  310. }
  311. else
  312. {
  313. var restElement = (RestElement) pattern.Properties[i];
  314. if (restElement.Argument is Identifier leftIdentifier)
  315. {
  316. var count = Math.Max(0, source.Properties?.Count ?? 0) - processedProperties.Count;
  317. var rest = engine.Realm.Intrinsics.Object.Construct(count);
  318. source.CopyDataProperties(rest, processedProperties);
  319. AssignToIdentifier(engine, leftIdentifier.Name, rest, environment);
  320. }
  321. else if (restElement.Argument is BindingPattern bp)
  322. {
  323. ProcessPatterns(engine, bp, argument, environment);
  324. }
  325. else if (restElement.Argument is MemberExpression memberExpression)
  326. {
  327. var left = GetReferenceFromMember(engine, memberExpression);
  328. var rest = engine.Realm.Intrinsics.Object.Construct(0);
  329. source.CopyDataProperties(rest, processedProperties);
  330. AssignToReference(engine, left, rest, environment);
  331. }
  332. else
  333. {
  334. ExceptionHelper.ThrowArgumentException("cannot handle parameter type " + restElement.Argument);
  335. }
  336. }
  337. }
  338. }
  339. private static void AssignToReference(
  340. Engine engine,
  341. Reference lhs,
  342. JsValue v,
  343. EnvironmentRecord environment)
  344. {
  345. if (environment is null)
  346. {
  347. engine.PutValue(lhs, v);
  348. }
  349. else
  350. {
  351. lhs.InitializeReferencedBinding(v);
  352. }
  353. engine._referencePool.Return(lhs);
  354. }
  355. private static Reference GetReferenceFromMember(Engine engine, MemberExpression memberExpression)
  356. {
  357. var expression = new JintMemberExpression(engine, memberExpression);
  358. var reference = expression.Evaluate() as Reference;
  359. if (reference is null)
  360. {
  361. ExceptionHelper.ThrowReferenceError(engine.Realm, "invalid reference");
  362. }
  363. reference.AssertValid(engine);
  364. return reference;
  365. }
  366. private static void AssignToIdentifier(
  367. Engine engine,
  368. string name,
  369. JsValue rval,
  370. EnvironmentRecord environment,
  371. bool checkReference = true)
  372. {
  373. var lhs = engine.ResolveBinding(name, environment);
  374. if (environment is not null)
  375. {
  376. lhs.InitializeReferencedBinding(rval);
  377. }
  378. else
  379. {
  380. if (checkReference && lhs.IsUnresolvableReference() && StrictModeScope.IsStrictModeCode)
  381. {
  382. ExceptionHelper.ThrowReferenceError(engine.Realm, "invalid reference");
  383. }
  384. engine.PutValue(lhs, rval);
  385. }
  386. }
  387. }
  388. }