JavascriptPromise.cpp 81 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033
  1. //-------------------------------------------------------------------------------------------------------
  2. // Copyright (C) Microsoft. All rights reserved.
  3. // Licensed under the MIT license. See LICENSE.txt file in the project root for full license information.
  4. //-------------------------------------------------------------------------------------------------------
  5. #include "RuntimeLibraryPch.h"
  6. namespace Js
  7. {
  8. JavascriptPromise::JavascriptPromise(DynamicType * type)
  9. : DynamicObject(type)
  10. {
  11. Assert(type->GetTypeId() == TypeIds_Promise);
  12. this->status = PromiseStatusCode_Undefined;
  13. this->isHandled = false;
  14. this->result = nullptr;
  15. this->resolveReactions = nullptr;
  16. this->rejectReactions = nullptr;
  17. }
  18. // Promise() as defined by ES 2016 Sections 25.4.3.1
  19. Var JavascriptPromise::NewInstance(RecyclableObject* function, CallInfo callInfo, ...)
  20. {
  21. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  22. ARGUMENTS(args, callInfo);
  23. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  24. ScriptContext* scriptContext = function->GetScriptContext();
  25. JavascriptLibrary* library = scriptContext->GetLibrary();
  26. CHAKRATEL_LANGSTATS_INC_DATACOUNT(ES6_Promise);
  27. // SkipDefaultNewObject function flag should have prevented the default object from
  28. // being created, except when call true a host dispatch
  29. Var newTarget = args.GetNewTarget();
  30. bool isCtorSuperCall = JavascriptOperators::GetAndAssertIsConstructorSuperCall(args);
  31. AUTO_TAG_NATIVE_LIBRARY_ENTRY(function, callInfo, _u("Promise"));
  32. // 1. If NewTarget is undefined, throw a TypeError exception.
  33. if ((callInfo.Flags & CallFlags_New) != CallFlags_New || (newTarget != nullptr && JavascriptOperators::IsUndefined(newTarget)))
  34. {
  35. JavascriptError::ThrowTypeError(scriptContext, JSERR_ClassConstructorCannotBeCalledWithoutNew, _u("Promise"));
  36. }
  37. // 2. If IsCallable(executor) is false, throw a TypeError exception.
  38. if (args.Info.Count < 2 || !JavascriptConversion::IsCallable(args[1]))
  39. {
  40. JavascriptError::ThrowTypeError(scriptContext, JSERR_FunctionArgument_NeedFunction, _u("Promise"));
  41. }
  42. RecyclableObject* executor = RecyclableObject::FromVar(args[1]);
  43. // 3. Let promise be ? OrdinaryCreateFromConstructor(NewTarget, "%PromisePrototype%", <<[[PromiseState]], [[PromiseResult]], [[PromiseFulfillReactions]], [[PromiseRejectReactions]], [[PromiseIsHandled]] >>).
  44. JavascriptPromise* promise = library->CreatePromise();
  45. if (isCtorSuperCall)
  46. {
  47. JavascriptOperators::OrdinaryCreateFromConstructor(RecyclableObject::FromVar(newTarget), promise, library->GetPromisePrototype(), scriptContext);
  48. }
  49. JavascriptPromiseResolveOrRejectFunction* resolve;
  50. JavascriptPromiseResolveOrRejectFunction* reject;
  51. // 4. Set promise's [[PromiseState]] internal slot to "pending".
  52. // 5. Set promise's [[PromiseFulfillReactions]] internal slot to a new empty List.
  53. // 6. Set promise's [[PromiseRejectReactions]] internal slot to a new empty List.
  54. // 7. Set promise's [[PromiseIsHandled]] internal slot to false.
  55. // 8. Let resolvingFunctions be CreateResolvingFunctions(promise).
  56. InitializePromise(promise, &resolve, &reject, scriptContext);
  57. JavascriptExceptionObject* exception = nullptr;
  58. // 9. Let completion be Call(executor, undefined, << resolvingFunctions.[[Resolve]], resolvingFunctions.[[Reject]] >>).
  59. try
  60. {
  61. CALL_FUNCTION(scriptContext->GetThreadContext(),
  62. executor, CallInfo(CallFlags_Value, 3),
  63. library->GetUndefined(),
  64. resolve,
  65. reject);
  66. }
  67. catch (const JavascriptException& err)
  68. {
  69. exception = err.GetAndClear();
  70. }
  71. if (exception != nullptr)
  72. {
  73. // 10. If completion is an abrupt completion, then
  74. // a. Perform ? Call(resolvingFunctions.[[Reject]], undefined, << completion.[[Value]] >>).
  75. TryRejectWithExceptionObject(exception, reject, scriptContext);
  76. }
  77. // 11. Return promise.
  78. return promise;
  79. }
  80. void JavascriptPromise::InitializePromise(JavascriptPromise* promise, JavascriptPromiseResolveOrRejectFunction** resolve, JavascriptPromiseResolveOrRejectFunction** reject, ScriptContext* scriptContext)
  81. {
  82. Assert(promise->status == PromiseStatusCode_Undefined);
  83. Assert(resolve);
  84. Assert(reject);
  85. Recycler* recycler = scriptContext->GetRecycler();
  86. JavascriptLibrary* library = scriptContext->GetLibrary();
  87. promise->status = PromiseStatusCode_Unresolved;
  88. promise->resolveReactions = RecyclerNew(recycler, JavascriptPromiseReactionList, recycler);
  89. promise->rejectReactions = RecyclerNew(recycler, JavascriptPromiseReactionList, recycler);
  90. JavascriptPromiseResolveOrRejectFunctionAlreadyResolvedWrapper* alreadyResolvedRecord = RecyclerNewStructZ(scriptContext->GetRecycler(), JavascriptPromiseResolveOrRejectFunctionAlreadyResolvedWrapper);
  91. alreadyResolvedRecord->alreadyResolved = false;
  92. *resolve = library->CreatePromiseResolveOrRejectFunction(EntryResolveOrRejectFunction, promise, false, alreadyResolvedRecord);
  93. *reject = library->CreatePromiseResolveOrRejectFunction(EntryResolveOrRejectFunction, promise, true, alreadyResolvedRecord);
  94. }
  95. bool JavascriptPromise::Is(Var aValue)
  96. {
  97. return Js::JavascriptOperators::GetTypeId(aValue) == TypeIds_Promise;
  98. }
  99. JavascriptPromise* JavascriptPromise::FromVar(Js::Var aValue)
  100. {
  101. AssertOrFailFastMsg(Is(aValue), "Ensure var is actually a 'JavascriptPromise'");
  102. return static_cast<JavascriptPromise *>(aValue);
  103. }
  104. JavascriptPromise* JavascriptPromise::UnsafeFromVar(Js::Var aValue)
  105. {
  106. AssertMsg(Is(aValue), "Ensure var is actually a 'JavascriptPromise'");
  107. return static_cast<JavascriptPromise *>(aValue);
  108. }
  109. BOOL JavascriptPromise::GetDiagValueString(StringBuilder<ArenaAllocator>* stringBuilder, ScriptContext* requestContext)
  110. {
  111. stringBuilder->AppendCppLiteral(_u("[...]"));
  112. return TRUE;
  113. }
  114. BOOL JavascriptPromise::GetDiagTypeString(StringBuilder<ArenaAllocator>* stringBuilder, ScriptContext* requestContext)
  115. {
  116. stringBuilder->AppendCppLiteral(_u("Promise"));
  117. return TRUE;
  118. }
  119. JavascriptPromiseReactionList* JavascriptPromise::GetResolveReactions()
  120. {
  121. return this->resolveReactions;
  122. }
  123. JavascriptPromiseReactionList* JavascriptPromise::GetRejectReactions()
  124. {
  125. return this->rejectReactions;
  126. }
  127. // Promise.all as described in ES 2015 Section 25.4.4.1
  128. Var JavascriptPromise::EntryAll(RecyclableObject* function, CallInfo callInfo, ...)
  129. {
  130. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  131. ARGUMENTS(args, callInfo);
  132. Assert(!(callInfo.Flags & CallFlags_New));
  133. ScriptContext* scriptContext = function->GetScriptContext();
  134. AUTO_TAG_NATIVE_LIBRARY_ENTRY(function, callInfo, _u("Promise.all"));
  135. // 1. Let C be the this value.
  136. Var constructor = args[0];
  137. // 2. If Type(C) is not Object, throw a TypeError exception.
  138. if (!JavascriptOperators::IsObject(constructor))
  139. {
  140. JavascriptError::ThrowTypeError(scriptContext, JSERR_This_NeedObject, _u("Promise.all"));
  141. }
  142. JavascriptLibrary* library = scriptContext->GetLibrary();
  143. Var iterable;
  144. if (args.Info.Count > 1)
  145. {
  146. iterable = args[1];
  147. }
  148. else
  149. {
  150. iterable = library->GetUndefined();
  151. }
  152. // 3. Let promiseCapability be NewPromiseCapability(C).
  153. JavascriptPromiseCapability* promiseCapability = NewPromiseCapability(constructor, scriptContext);
  154. // We know that constructor is an object at this point - further, we even know that it is a constructor - because NewPromiseCapability
  155. // would throw otherwise. That means we can safely cast constructor into a RecyclableObject* now and avoid having to perform ToObject
  156. // as part of the Invoke operation performed inside the loop below.
  157. RecyclableObject* constructorObject = RecyclableObject::FromVar(constructor);
  158. uint32 index = 0;
  159. JavascriptArray* values = nullptr;
  160. // We can't use a simple counter for the remaining element count since each Promise.all Resolve Element Function needs to know how many
  161. // elements are remaining when it runs and needs to update that counter for all other functions created by this call to Promise.all.
  162. // We can't just use a static variable, either, since this element count is only used for the Promise.all Resolve Element Functions created
  163. // by this call to Promise.all.
  164. JavascriptPromiseAllResolveElementFunctionRemainingElementsWrapper* remainingElementsWrapper = RecyclerNewStructZ(scriptContext->GetRecycler(), JavascriptPromiseAllResolveElementFunctionRemainingElementsWrapper);
  165. remainingElementsWrapper->remainingElements = 1;
  166. JavascriptExceptionObject* exception = nullptr;
  167. try
  168. {
  169. // 4. Let iterator be GetIterator(iterable).
  170. RecyclableObject* iterator = JavascriptOperators::GetIterator(iterable, scriptContext);
  171. values = library->CreateArray(0);
  172. JavascriptOperators::DoIteratorStepAndValue(iterator, scriptContext, [&](Var next)
  173. {
  174. Var resolveVar = JavascriptOperators::GetProperty(constructorObject, Js::PropertyIds::resolve, scriptContext);
  175. if (!JavascriptConversion::IsCallable(resolveVar))
  176. {
  177. JavascriptError::ThrowTypeError(scriptContext, JSERR_NeedFunction);
  178. }
  179. RecyclableObject* resolveFunc = RecyclableObject::FromVar(resolveVar);
  180. Var nextPromise = CALL_FUNCTION(scriptContext->GetThreadContext(),
  181. resolveFunc, Js::CallInfo(CallFlags_Value, 2),
  182. constructorObject,
  183. next);
  184. JavascriptPromiseAllResolveElementFunction* resolveElement = library->CreatePromiseAllResolveElementFunction(EntryAllResolveElementFunction, index, values, promiseCapability, remainingElementsWrapper);
  185. remainingElementsWrapper->remainingElements++;
  186. RecyclableObject* nextPromiseObject;
  187. if (!JavascriptConversion::ToObject(nextPromise, scriptContext, &nextPromiseObject))
  188. {
  189. JavascriptError::ThrowTypeError(scriptContext, JSERR_NeedObject);
  190. }
  191. Var thenVar = JavascriptOperators::GetProperty(nextPromiseObject, Js::PropertyIds::then, scriptContext);
  192. if (!JavascriptConversion::IsCallable(thenVar))
  193. {
  194. JavascriptError::ThrowTypeError(scriptContext, JSERR_NeedFunction);
  195. }
  196. RecyclableObject* thenFunc = RecyclableObject::FromVar(thenVar);
  197. CALL_FUNCTION(scriptContext->GetThreadContext(),
  198. thenFunc, Js::CallInfo(CallFlags_Value, 3),
  199. nextPromiseObject,
  200. resolveElement,
  201. promiseCapability->GetReject());
  202. index++;
  203. });
  204. }
  205. catch (const JavascriptException& err)
  206. {
  207. exception = err.GetAndClear();
  208. }
  209. if (exception != nullptr)
  210. {
  211. TryRejectWithExceptionObject(exception, promiseCapability->GetReject(), scriptContext);
  212. // We need to explicitly return here to make sure we don't resolve in case index == 0 here.
  213. // That would happen if GetIterator or IteratorValue throws an exception in the first iteration.
  214. return promiseCapability->GetPromise();
  215. }
  216. remainingElementsWrapper->remainingElements--;
  217. // We want this call to happen outside the try statement because if it throws, we aren't supposed to reject the promise.
  218. if (remainingElementsWrapper->remainingElements == 0)
  219. {
  220. Assert(values != nullptr);
  221. TryCallResolveOrRejectHandler(promiseCapability->GetResolve(), values, scriptContext);
  222. }
  223. return promiseCapability->GetPromise();
  224. }
  225. // Promise.prototype.catch as defined in ES 2015 Section 25.4.5.1
  226. Var JavascriptPromise::EntryCatch(RecyclableObject* function, CallInfo callInfo, ...)
  227. {
  228. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  229. ARGUMENTS(args, callInfo);
  230. Assert(!(callInfo.Flags & CallFlags_New));
  231. ScriptContext* scriptContext = function->GetScriptContext();
  232. AUTO_TAG_NATIVE_LIBRARY_ENTRY(function, callInfo, _u("Promise.prototype.catch"));
  233. RecyclableObject* promise;
  234. if (!JavascriptConversion::ToObject(args[0], scriptContext, &promise))
  235. {
  236. JavascriptError::ThrowTypeError(scriptContext, JSERR_This_NeedObject, _u("Promise.prototype.catch"));
  237. }
  238. Var funcVar = JavascriptOperators::GetProperty(promise, Js::PropertyIds::then, scriptContext);
  239. if (!JavascriptConversion::IsCallable(funcVar))
  240. {
  241. JavascriptError::ThrowTypeError(scriptContext, JSERR_FunctionArgument_NeedFunction, _u("Promise.prototype.catch"));
  242. }
  243. Var onRejected;
  244. RecyclableObject* undefinedVar = scriptContext->GetLibrary()->GetUndefined();
  245. if (args.Info.Count > 1)
  246. {
  247. onRejected = args[1];
  248. }
  249. else
  250. {
  251. onRejected = undefinedVar;
  252. }
  253. RecyclableObject* func = RecyclableObject::FromVar(funcVar);
  254. return CALL_FUNCTION(scriptContext->GetThreadContext(),
  255. func, Js::CallInfo(CallFlags_Value, 3),
  256. promise,
  257. undefinedVar,
  258. onRejected);
  259. }
  260. // Promise.race as described in ES 2015 Section 25.4.4.3
  261. Var JavascriptPromise::EntryRace(RecyclableObject* function, CallInfo callInfo, ...)
  262. {
  263. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  264. ARGUMENTS(args, callInfo);
  265. Assert(!(callInfo.Flags & CallFlags_New));
  266. ScriptContext* scriptContext = function->GetScriptContext();
  267. AUTO_TAG_NATIVE_LIBRARY_ENTRY(function, callInfo, _u("Promise.race"));
  268. // 1. Let C be the this value.
  269. Var constructor = args[0];
  270. // 2. If Type(C) is not Object, throw a TypeError exception.
  271. if (!JavascriptOperators::IsObject(constructor))
  272. {
  273. JavascriptError::ThrowTypeError(scriptContext, JSERR_This_NeedObject, _u("Promise.race"));
  274. }
  275. Var undefinedVar = scriptContext->GetLibrary()->GetUndefined();
  276. Var iterable;
  277. if (args.Info.Count > 1)
  278. {
  279. iterable = args[1];
  280. }
  281. else
  282. {
  283. iterable = undefinedVar;
  284. }
  285. // 3. Let promiseCapability be NewPromiseCapability(C).
  286. JavascriptPromiseCapability* promiseCapability = NewPromiseCapability(constructor, scriptContext);
  287. // We know that constructor is an object at this point - further, we even know that it is a constructor - because NewPromiseCapability
  288. // would throw otherwise. That means we can safely cast constructor into a RecyclableObject* now and avoid having to perform ToObject
  289. // as part of the Invoke operation performed inside the loop below.
  290. RecyclableObject* constructorObject = RecyclableObject::FromVar(constructor);
  291. JavascriptExceptionObject* exception = nullptr;
  292. try
  293. {
  294. // 4. Let iterator be GetIterator(iterable).
  295. RecyclableObject* iterator = JavascriptOperators::GetIterator(iterable, scriptContext);
  296. JavascriptOperators::DoIteratorStepAndValue(iterator, scriptContext, [&](Var next)
  297. {
  298. Var resolveVar = JavascriptOperators::GetProperty(constructorObject, Js::PropertyIds::resolve, scriptContext);
  299. if (!JavascriptConversion::IsCallable(resolveVar))
  300. {
  301. JavascriptError::ThrowTypeError(scriptContext, JSERR_NeedFunction);
  302. }
  303. RecyclableObject* resolveFunc = RecyclableObject::FromVar(resolveVar);
  304. Var nextPromise = CALL_FUNCTION(scriptContext->GetThreadContext(),
  305. resolveFunc, Js::CallInfo(CallFlags_Value, 2),
  306. constructorObject,
  307. next);
  308. RecyclableObject* nextPromiseObject;
  309. if (!JavascriptConversion::ToObject(nextPromise, scriptContext, &nextPromiseObject))
  310. {
  311. JavascriptError::ThrowTypeError(scriptContext, JSERR_NeedObject);
  312. }
  313. Var thenVar = JavascriptOperators::GetProperty(nextPromiseObject, Js::PropertyIds::then, scriptContext);
  314. if (!JavascriptConversion::IsCallable(thenVar))
  315. {
  316. JavascriptError::ThrowTypeError(scriptContext, JSERR_NeedFunction);
  317. }
  318. RecyclableObject* thenFunc = RecyclableObject::FromVar(thenVar);
  319. CALL_FUNCTION(scriptContext->GetThreadContext(),
  320. thenFunc, Js::CallInfo(CallFlags_Value, 3),
  321. nextPromiseObject,
  322. promiseCapability->GetResolve(),
  323. promiseCapability->GetReject());
  324. });
  325. }
  326. catch (const JavascriptException& err)
  327. {
  328. exception = err.GetAndClear();
  329. }
  330. if (exception != nullptr)
  331. {
  332. TryRejectWithExceptionObject(exception, promiseCapability->GetReject(), scriptContext);
  333. }
  334. return promiseCapability->GetPromise();
  335. }
  336. // Promise.reject as described in ES 2015 Section 25.4.4.4
  337. Var JavascriptPromise::EntryReject(RecyclableObject* function, CallInfo callInfo, ...)
  338. {
  339. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  340. ARGUMENTS(args, callInfo);
  341. Assert(!(callInfo.Flags & CallFlags_New));
  342. ScriptContext* scriptContext = function->GetScriptContext();
  343. AUTO_TAG_NATIVE_LIBRARY_ENTRY(function, callInfo, _u("Promise.reject"));
  344. // 1. Let C be the this value.
  345. Var constructor = args[0];
  346. // 2. If Type(C) is not Object, throw a TypeError exception.
  347. if (!JavascriptOperators::IsObject(constructor))
  348. {
  349. JavascriptError::ThrowTypeError(scriptContext, JSERR_This_NeedObject, _u("Promise.reject"));
  350. }
  351. Var r;
  352. if (args.Info.Count > 1)
  353. {
  354. r = args[1];
  355. }
  356. else
  357. {
  358. r = scriptContext->GetLibrary()->GetUndefined();
  359. }
  360. // 3. Let promiseCapability be NewPromiseCapability(C).
  361. // 4. Perform ? Call(promiseCapability.[[Reject]], undefined, << r >>).
  362. // 5. Return promiseCapability.[[Promise]].
  363. return CreateRejectedPromise(r, scriptContext, constructor);
  364. }
  365. // Promise.resolve as described in ES 2015 Section 25.4.4.5
  366. Var JavascriptPromise::EntryResolve(RecyclableObject* function, CallInfo callInfo, ...)
  367. {
  368. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  369. ARGUMENTS(args, callInfo);
  370. Assert(!(callInfo.Flags & CallFlags_New));
  371. ScriptContext* scriptContext = function->GetScriptContext();
  372. AUTO_TAG_NATIVE_LIBRARY_ENTRY(function, callInfo, _u("Promise.resolve"));
  373. Var x;
  374. // 1. Let C be the this value.
  375. Var constructor = args[0];
  376. // 2. If Type(C) is not Object, throw a TypeError exception.
  377. if (!JavascriptOperators::IsObject(constructor))
  378. {
  379. JavascriptError::ThrowTypeError(scriptContext, JSERR_This_NeedObject, _u("Promise.resolve"));
  380. }
  381. if (args.Info.Count > 1)
  382. {
  383. x = args[1];
  384. }
  385. else
  386. {
  387. x = scriptContext->GetLibrary()->GetUndefined();
  388. }
  389. // 3. If IsPromise(x) is true,
  390. if (JavascriptPromise::Is(x))
  391. {
  392. // a. Let xConstructor be Get(x, "constructor").
  393. Var xConstructor = JavascriptOperators::GetProperty((RecyclableObject*)x, PropertyIds::constructor, scriptContext);
  394. // b. If SameValue(xConstructor, C) is true, return x.
  395. if (JavascriptConversion::SameValue(xConstructor, constructor))
  396. {
  397. return x;
  398. }
  399. }
  400. // 4. Let promiseCapability be NewPromiseCapability(C).
  401. // 5. Perform ? Call(promiseCapability.[[Resolve]], undefined, << x >>).
  402. // 6. Return promiseCapability.[[Promise]].
  403. return CreateResolvedPromise(x, scriptContext, constructor);
  404. }
  405. // Promise.prototype.then as described in ES 2015 Section 25.4.5.3
  406. Var JavascriptPromise::EntryThen(RecyclableObject* function, CallInfo callInfo, ...)
  407. {
  408. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  409. ARGUMENTS(args, callInfo);
  410. Assert(!(callInfo.Flags & CallFlags_New));
  411. ScriptContext* scriptContext = function->GetScriptContext();
  412. AUTO_TAG_NATIVE_LIBRARY_ENTRY(function, callInfo, _u("Promise.prototype.then"));
  413. if (args.Info.Count < 1 || !JavascriptPromise::Is(args[0]))
  414. {
  415. JavascriptError::ThrowTypeError(scriptContext, JSERR_This_NeedPromise, _u("Promise.prototype.then"));
  416. }
  417. JavascriptLibrary* library = scriptContext->GetLibrary();
  418. JavascriptPromise* promise = JavascriptPromise::FromVar(args[0]);
  419. RecyclableObject* rejectionHandler;
  420. RecyclableObject* fulfillmentHandler;
  421. if (args.Info.Count > 1 && JavascriptConversion::IsCallable(args[1]))
  422. {
  423. fulfillmentHandler = RecyclableObject::FromVar(args[1]);
  424. }
  425. else
  426. {
  427. fulfillmentHandler = library->GetIdentityFunction();
  428. }
  429. if (args.Info.Count > 2 && JavascriptConversion::IsCallable(args[2]))
  430. {
  431. rejectionHandler = RecyclableObject::FromVar(args[2]);
  432. }
  433. else
  434. {
  435. rejectionHandler = library->GetThrowerFunction();
  436. }
  437. return CreateThenPromise(promise, fulfillmentHandler, rejectionHandler, scriptContext);
  438. }
  439. // Promise Reject and Resolve Functions as described in ES 2015 Section 25.4.1.4.1 and 25.4.1.4.2
  440. Var JavascriptPromise::EntryResolveOrRejectFunction(RecyclableObject* function, CallInfo callInfo, ...)
  441. {
  442. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  443. ARGUMENTS(args, callInfo);
  444. Assert(!(callInfo.Flags & CallFlags_New));
  445. ScriptContext* scriptContext = function->GetScriptContext();
  446. JavascriptLibrary* library = scriptContext->GetLibrary();
  447. Var undefinedVar = library->GetUndefined();
  448. Var resolution;
  449. if (args.Info.Count > 1)
  450. {
  451. resolution = args[1];
  452. }
  453. else
  454. {
  455. resolution = undefinedVar;
  456. }
  457. JavascriptPromiseResolveOrRejectFunction* resolveOrRejectFunction = JavascriptPromiseResolveOrRejectFunction::FromVar(function);
  458. if (resolveOrRejectFunction->IsAlreadyResolved())
  459. {
  460. return undefinedVar;
  461. }
  462. resolveOrRejectFunction->SetAlreadyResolved(true);
  463. bool rejecting = resolveOrRejectFunction->IsRejectFunction();
  464. JavascriptPromise* promise = resolveOrRejectFunction->GetPromise();
  465. return promise->ResolveHelper(resolution, rejecting, scriptContext);
  466. }
  467. Var JavascriptPromise::Resolve(Var resolution, ScriptContext* scriptContext)
  468. {
  469. return this->ResolveHelper(resolution, false, scriptContext);
  470. }
  471. Var JavascriptPromise::Reject(Var resolution, ScriptContext* scriptContext)
  472. {
  473. return this->ResolveHelper(resolution, true, scriptContext);
  474. }
  475. Var JavascriptPromise::ResolveHelper(Var resolution, bool isRejecting, ScriptContext* scriptContext)
  476. {
  477. JavascriptLibrary* library = scriptContext->GetLibrary();
  478. Var undefinedVar = library->GetUndefined();
  479. // We only need to check SameValue and check for thenable resolution in the Resolve function case (not Reject)
  480. if (!isRejecting)
  481. {
  482. if (JavascriptConversion::SameValue(resolution, this))
  483. {
  484. JavascriptError* selfResolutionError = scriptContext->GetLibrary()->CreateTypeError();
  485. JavascriptError::SetErrorMessage(selfResolutionError, JSERR_PromiseSelfResolution, _u(""), scriptContext);
  486. resolution = selfResolutionError;
  487. isRejecting = true;
  488. }
  489. else if (RecyclableObject::Is(resolution))
  490. {
  491. try
  492. {
  493. RecyclableObject* thenable = RecyclableObject::FromVar(resolution);
  494. Var then = JavascriptOperators::GetPropertyNoCache(thenable, Js::PropertyIds::then, scriptContext);
  495. if (JavascriptConversion::IsCallable(then))
  496. {
  497. JavascriptPromiseResolveThenableTaskFunction* resolveThenableTaskFunction = library->CreatePromiseResolveThenableTaskFunction(EntryResolveThenableTaskFunction, this, thenable, RecyclableObject::FromVar(then));
  498. library->EnqueueTask(resolveThenableTaskFunction);
  499. return undefinedVar;
  500. }
  501. }
  502. catch (const JavascriptException& err)
  503. {
  504. resolution = err.GetAndClear()->GetThrownObject(scriptContext);
  505. if (resolution == nullptr)
  506. {
  507. resolution = undefinedVar;
  508. }
  509. isRejecting = true;
  510. }
  511. }
  512. }
  513. JavascriptPromiseReactionList* reactions;
  514. PromiseStatus newStatus;
  515. // Need to check rejecting state again as it might have changed due to failures
  516. if (isRejecting)
  517. {
  518. reactions = this->GetRejectReactions();
  519. newStatus = PromiseStatusCode_HasRejection;
  520. if (!GetIsHandled())
  521. {
  522. scriptContext->GetLibrary()->CallNativeHostPromiseRejectionTracker(this, resolution, false);
  523. }
  524. }
  525. else
  526. {
  527. reactions = this->GetResolveReactions();
  528. newStatus = PromiseStatusCode_HasResolution;
  529. }
  530. Assert(resolution != nullptr);
  531. this->result = resolution;
  532. this->resolveReactions = nullptr;
  533. this->rejectReactions = nullptr;
  534. this->status = newStatus;
  535. return TriggerPromiseReactions(reactions, resolution, scriptContext);
  536. }
  537. // Promise Capabilities Executor Function as described in ES 2015 Section 25.4.1.6.2
  538. Var JavascriptPromise::EntryCapabilitiesExecutorFunction(RecyclableObject* function, CallInfo callInfo, ...)
  539. {
  540. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  541. ARGUMENTS(args, callInfo);
  542. Assert(!(callInfo.Flags & CallFlags_New));
  543. ScriptContext* scriptContext = function->GetScriptContext();
  544. Var undefinedVar = scriptContext->GetLibrary()->GetUndefined();
  545. Var resolve = undefinedVar;
  546. Var reject = undefinedVar;
  547. if (args.Info.Count > 1)
  548. {
  549. resolve = args[1];
  550. if (args.Info.Count > 2)
  551. {
  552. reject = args[2];
  553. }
  554. }
  555. JavascriptPromiseCapabilitiesExecutorFunction* capabilitiesExecutorFunction = JavascriptPromiseCapabilitiesExecutorFunction::FromVar(function);
  556. JavascriptPromiseCapability* promiseCapability = capabilitiesExecutorFunction->GetCapability();
  557. if (!JavascriptOperators::IsUndefined(promiseCapability->GetResolve()) || !JavascriptOperators::IsUndefined(promiseCapability->GetReject()))
  558. {
  559. JavascriptError::ThrowTypeErrorVar(scriptContext, JSERR_UnexpectedMetadataFailure, _u("Promise"));
  560. }
  561. promiseCapability->SetResolve(resolve);
  562. promiseCapability->SetReject(reject);
  563. return undefinedVar;
  564. }
  565. // Promise Reaction Task Function as described in ES 2015 Section 25.4.2.1
  566. Var JavascriptPromise::EntryReactionTaskFunction(RecyclableObject* function, CallInfo callInfo, ...)
  567. {
  568. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  569. ARGUMENTS(args, callInfo);
  570. Assert(!(callInfo.Flags & CallFlags_New));
  571. ScriptContext* scriptContext = function->GetScriptContext();
  572. Var undefinedVar = scriptContext->GetLibrary()->GetUndefined();
  573. JavascriptPromiseReactionTaskFunction* reactionTaskFunction = JavascriptPromiseReactionTaskFunction::FromVar(function);
  574. JavascriptPromiseReaction* reaction = reactionTaskFunction->GetReaction();
  575. Var argument = reactionTaskFunction->GetArgument();
  576. JavascriptPromiseCapability* promiseCapability = reaction->GetCapabilities();
  577. RecyclableObject* handler = reaction->GetHandler();
  578. Var handlerResult = nullptr;
  579. JavascriptExceptionObject* exception = nullptr;
  580. {
  581. Js::JavascriptExceptionOperators::AutoCatchHandlerExists autoCatchHandlerExists(scriptContext);
  582. try
  583. {
  584. handlerResult = CALL_FUNCTION(scriptContext->GetThreadContext(),
  585. handler, Js::CallInfo(Js::CallFlags::CallFlags_Value, 2),
  586. undefinedVar,
  587. argument);
  588. }
  589. catch (const JavascriptException& err)
  590. {
  591. exception = err.GetAndClear();
  592. }
  593. }
  594. if (exception != nullptr)
  595. {
  596. return TryRejectWithExceptionObject(exception, promiseCapability->GetReject(), scriptContext);
  597. }
  598. Assert(handlerResult != nullptr);
  599. return TryCallResolveOrRejectHandler(promiseCapability->GetResolve(), handlerResult, scriptContext);
  600. }
  601. Var JavascriptPromise::TryCallResolveOrRejectHandler(Var handler, Var value, ScriptContext* scriptContext)
  602. {
  603. Var undefinedVar = scriptContext->GetLibrary()->GetUndefined();
  604. if (!JavascriptConversion::IsCallable(handler))
  605. {
  606. JavascriptError::ThrowTypeError(scriptContext, JSERR_NeedFunction);
  607. }
  608. RecyclableObject* handlerFunc = RecyclableObject::FromVar(handler);
  609. return CALL_FUNCTION(scriptContext->GetThreadContext(),
  610. handlerFunc, CallInfo(CallFlags_Value, 2),
  611. undefinedVar,
  612. value);
  613. }
  614. Var JavascriptPromise::TryRejectWithExceptionObject(JavascriptExceptionObject* exceptionObject, Var handler, ScriptContext* scriptContext)
  615. {
  616. Var thrownObject = exceptionObject->GetThrownObject(scriptContext);
  617. if (thrownObject == nullptr)
  618. {
  619. thrownObject = scriptContext->GetLibrary()->GetUndefined();
  620. }
  621. return TryCallResolveOrRejectHandler(handler, thrownObject, scriptContext);
  622. }
  623. Var JavascriptPromise::CreateRejectedPromise(Var resolution, ScriptContext* scriptContext, Var promiseConstructor)
  624. {
  625. if (promiseConstructor == nullptr)
  626. {
  627. promiseConstructor = scriptContext->GetLibrary()->GetPromiseConstructor();
  628. }
  629. JavascriptPromiseCapability* promiseCapability = NewPromiseCapability(promiseConstructor, scriptContext);
  630. TryCallResolveOrRejectHandler(promiseCapability->GetReject(), resolution, scriptContext);
  631. return promiseCapability->GetPromise();
  632. }
  633. Var JavascriptPromise::CreateResolvedPromise(Var resolution, ScriptContext* scriptContext, Var promiseConstructor)
  634. {
  635. if (promiseConstructor == nullptr)
  636. {
  637. promiseConstructor = scriptContext->GetLibrary()->GetPromiseConstructor();
  638. }
  639. JavascriptPromiseCapability* promiseCapability = NewPromiseCapability(promiseConstructor, scriptContext);
  640. TryCallResolveOrRejectHandler(promiseCapability->GetResolve(), resolution, scriptContext);
  641. return promiseCapability->GetPromise();
  642. }
  643. Var JavascriptPromise::CreatePassThroughPromise(JavascriptPromise* sourcePromise, ScriptContext* scriptContext)
  644. {
  645. JavascriptLibrary* library = scriptContext->GetLibrary();
  646. return CreateThenPromise(sourcePromise, library->GetIdentityFunction(), library->GetThrowerFunction(), scriptContext);
  647. }
  648. Var JavascriptPromise::CreateThenPromise(JavascriptPromise* sourcePromise, RecyclableObject* fulfillmentHandler, RecyclableObject* rejectionHandler, ScriptContext* scriptContext)
  649. {
  650. RecyclableObject* constructor = JavascriptOperators::SpeciesConstructor(sourcePromise, scriptContext->GetLibrary()->GetPromiseConstructor(), scriptContext);
  651. JavascriptPromiseCapability* promiseCapability = NewPromiseCapability(constructor, scriptContext);
  652. JavascriptPromiseReaction* resolveReaction = JavascriptPromiseReaction::New(promiseCapability, fulfillmentHandler, scriptContext);
  653. JavascriptPromiseReaction* rejectReaction = JavascriptPromiseReaction::New(promiseCapability, rejectionHandler, scriptContext);
  654. switch (sourcePromise->status)
  655. {
  656. case PromiseStatusCode_Unresolved:
  657. sourcePromise->resolveReactions->Add(resolveReaction);
  658. sourcePromise->rejectReactions->Add(rejectReaction);
  659. break;
  660. case PromiseStatusCode_HasResolution:
  661. EnqueuePromiseReactionTask(resolveReaction, sourcePromise->result, scriptContext);
  662. break;
  663. case PromiseStatusCode_HasRejection:
  664. if (!sourcePromise->GetIsHandled())
  665. {
  666. scriptContext->GetLibrary()->CallNativeHostPromiseRejectionTracker(sourcePromise, sourcePromise->result, true);
  667. }
  668. EnqueuePromiseReactionTask(rejectReaction, sourcePromise->result, scriptContext);
  669. break;
  670. default:
  671. AssertMsg(false, "Promise status is in an invalid state");
  672. break;
  673. }
  674. sourcePromise->SetIsHandled();
  675. return promiseCapability->GetPromise();
  676. }
  677. // Promise Resolve Thenable Job as described in ES 2015 Section 25.4.2.2
  678. Var JavascriptPromise::EntryResolveThenableTaskFunction(RecyclableObject* function, CallInfo callInfo, ...)
  679. {
  680. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  681. ARGUMENTS(args, callInfo);
  682. Assert(!(callInfo.Flags & CallFlags_New));
  683. ScriptContext* scriptContext = function->GetScriptContext();
  684. JavascriptLibrary* library = scriptContext->GetLibrary();
  685. JavascriptPromiseResolveThenableTaskFunction* resolveThenableTaskFunction = JavascriptPromiseResolveThenableTaskFunction::FromVar(function);
  686. JavascriptPromise* promise = resolveThenableTaskFunction->GetPromise();
  687. RecyclableObject* thenable = resolveThenableTaskFunction->GetThenable();
  688. RecyclableObject* thenFunction = resolveThenableTaskFunction->GetThenFunction();
  689. JavascriptPromiseResolveOrRejectFunctionAlreadyResolvedWrapper* alreadyResolvedRecord = RecyclerNewStructZ(scriptContext->GetRecycler(), JavascriptPromiseResolveOrRejectFunctionAlreadyResolvedWrapper);
  690. alreadyResolvedRecord->alreadyResolved = false;
  691. JavascriptPromiseResolveOrRejectFunction* resolve = library->CreatePromiseResolveOrRejectFunction(EntryResolveOrRejectFunction, promise, false, alreadyResolvedRecord);
  692. JavascriptPromiseResolveOrRejectFunction* reject = library->CreatePromiseResolveOrRejectFunction(EntryResolveOrRejectFunction, promise, true, alreadyResolvedRecord);
  693. JavascriptExceptionObject* exception = nullptr;
  694. {
  695. Js::JavascriptExceptionOperators::AutoCatchHandlerExists autoCatchHandlerExists(scriptContext);
  696. try
  697. {
  698. return CALL_FUNCTION(scriptContext->GetThreadContext(),
  699. thenFunction, Js::CallInfo(Js::CallFlags::CallFlags_Value, 3),
  700. thenable,
  701. resolve,
  702. reject);
  703. }
  704. catch (const JavascriptException& err)
  705. {
  706. exception = err.GetAndClear();
  707. }
  708. }
  709. Assert(exception != nullptr);
  710. return TryRejectWithExceptionObject(exception, reject, scriptContext);
  711. }
  712. // Promise Identity Function as described in ES 2015Section 25.4.5.3.1
  713. Var JavascriptPromise::EntryIdentityFunction(RecyclableObject* function, CallInfo callInfo, ...)
  714. {
  715. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  716. ARGUMENTS(args, callInfo);
  717. Assert(!(callInfo.Flags & CallFlags_New));
  718. if (args.Info.Count > 1)
  719. {
  720. Assert(args[1] != nullptr);
  721. return args[1];
  722. }
  723. else
  724. {
  725. return function->GetScriptContext()->GetLibrary()->GetUndefined();
  726. }
  727. }
  728. // Promise Thrower Function as described in ES 2015Section 25.4.5.3.3
  729. Var JavascriptPromise::EntryThrowerFunction(RecyclableObject* function, CallInfo callInfo, ...)
  730. {
  731. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  732. ARGUMENTS(args, callInfo);
  733. Assert(!(callInfo.Flags & CallFlags_New));
  734. ScriptContext* scriptContext = function->GetScriptContext();
  735. Var arg;
  736. if (args.Info.Count > 1)
  737. {
  738. Assert(args[1] != nullptr);
  739. arg = args[1];
  740. }
  741. else
  742. {
  743. arg = scriptContext->GetLibrary()->GetUndefined();
  744. }
  745. JavascriptExceptionOperators::Throw(arg, scriptContext);
  746. }
  747. // Promise.all Resolve Element Function as described in ES6.0 (Release Candidate 3) Section 25.4.4.1.2
  748. Var JavascriptPromise::EntryAllResolveElementFunction(RecyclableObject* function, CallInfo callInfo, ...)
  749. {
  750. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  751. ARGUMENTS(args, callInfo);
  752. Assert(!(callInfo.Flags & CallFlags_New));
  753. ScriptContext* scriptContext = function->GetScriptContext();
  754. Var undefinedVar = scriptContext->GetLibrary()->GetUndefined();
  755. Var x;
  756. if (args.Info.Count > 1)
  757. {
  758. x = args[1];
  759. }
  760. else
  761. {
  762. x = undefinedVar;
  763. }
  764. JavascriptPromiseAllResolveElementFunction* allResolveElementFunction = JavascriptPromiseAllResolveElementFunction::FromVar(function);
  765. if (allResolveElementFunction->IsAlreadyCalled())
  766. {
  767. return undefinedVar;
  768. }
  769. allResolveElementFunction->SetAlreadyCalled(true);
  770. uint32 index = allResolveElementFunction->GetIndex();
  771. JavascriptArray* values = allResolveElementFunction->GetValues();
  772. JavascriptPromiseCapability* promiseCapability = allResolveElementFunction->GetCapabilities();
  773. JavascriptExceptionObject* exception = nullptr;
  774. try
  775. {
  776. values->SetItem(index, x, PropertyOperation_None);
  777. }
  778. catch (const JavascriptException& err)
  779. {
  780. exception = err.GetAndClear();
  781. }
  782. if (exception != nullptr)
  783. {
  784. return TryRejectWithExceptionObject(exception, promiseCapability->GetReject(), scriptContext);
  785. }
  786. if (allResolveElementFunction->DecrementRemainingElements() == 0)
  787. {
  788. return TryCallResolveOrRejectHandler(promiseCapability->GetResolve(), values, scriptContext);
  789. }
  790. return undefinedVar;
  791. }
  792. Var JavascriptPromise::EntryJavascriptPromiseAsyncSpawnExecutorFunction(RecyclableObject* function, CallInfo callInfo, ...)
  793. {
  794. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  795. ARGUMENTS(args, callInfo);
  796. ScriptContext* scriptContext = function->GetScriptContext();
  797. JavascriptLibrary* library = scriptContext->GetLibrary();
  798. Var undefinedVar = library->GetUndefined();
  799. Var resolve = undefinedVar;
  800. Var reject = undefinedVar;
  801. Assert(args.Info.Count == 3);
  802. resolve = args[1];
  803. reject = args[2];
  804. Assert(JavascriptPromiseAsyncSpawnExecutorFunction::Is(function));
  805. JavascriptPromiseAsyncSpawnExecutorFunction* asyncSpawnExecutorFunction = JavascriptPromiseAsyncSpawnExecutorFunction::FromVar(function);
  806. Var self = asyncSpawnExecutorFunction->GetTarget();
  807. Var varCallArgs[] = { undefinedVar, self };
  808. JavascriptGenerator* gen = asyncSpawnExecutorFunction->GetGenerator();
  809. JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction* nextFunction = library->CreatePromiseAsyncSpawnStepArgumentExecutorFunction(EntryJavascriptPromiseAsyncSpawnStepNextExecutorFunction, gen, varCallArgs);
  810. Assert(JavascriptConversion::IsCallable(resolve) && JavascriptConversion::IsCallable(reject));
  811. AsyncSpawnStep(nextFunction, gen, resolve, reject);
  812. return undefinedVar;
  813. }
  814. Var JavascriptPromise::EntryJavascriptPromiseAsyncSpawnStepNextExecutorFunction(RecyclableObject* function, CallInfo callInfo, ...)
  815. {
  816. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  817. JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction* asyncSpawnStepArgumentExecutorFunction = JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::FromVar(function);
  818. Var argument = asyncSpawnStepArgumentExecutorFunction->GetArgument();
  819. JavascriptFunction* next = function->GetScriptContext()->GetLibrary()->EnsureGeneratorNextFunction();
  820. return CALL_FUNCTION(function->GetScriptContext()->GetThreadContext(), next, CallInfo(CallFlags_Value, 2), asyncSpawnStepArgumentExecutorFunction->GetGenerator(), argument);
  821. }
  822. Var JavascriptPromise::EntryJavascriptPromiseAsyncSpawnStepThrowExecutorFunction(RecyclableObject* function, CallInfo callInfo, ...)
  823. {
  824. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  825. JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction* asyncSpawnStepArgumentExecutorFunction = JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::FromVar(function);
  826. JavascriptFunction* throw_ = function->GetScriptContext()->GetLibrary()->EnsureGeneratorThrowFunction();
  827. return CALL_FUNCTION(function->GetScriptContext()->GetThreadContext(), throw_, CallInfo(CallFlags_Value, 2), asyncSpawnStepArgumentExecutorFunction->GetGenerator(), asyncSpawnStepArgumentExecutorFunction->GetArgument());
  828. }
  829. Var JavascriptPromise::EntryJavascriptPromiseAsyncSpawnCallStepExecutorFunction(RecyclableObject* function, CallInfo callInfo, ...)
  830. {
  831. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  832. ARGUMENTS(args, callInfo);
  833. ScriptContext* scriptContext = function->GetScriptContext();
  834. JavascriptLibrary* library = scriptContext->GetLibrary();
  835. Var undefinedVar = library->GetUndefined();
  836. Var argument = undefinedVar;
  837. if (args.Info.Count > 1)
  838. {
  839. argument = args[1];
  840. }
  841. JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction* asyncSpawnStepExecutorFunction = JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::FromVar(function);
  842. JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction* functionArg;
  843. JavascriptGenerator* gen = asyncSpawnStepExecutorFunction->GetGenerator();
  844. Var reject = asyncSpawnStepExecutorFunction->GetReject();
  845. Var resolve = asyncSpawnStepExecutorFunction->GetResolve();
  846. if (asyncSpawnStepExecutorFunction->GetIsReject())
  847. {
  848. functionArg = library->CreatePromiseAsyncSpawnStepArgumentExecutorFunction(EntryJavascriptPromiseAsyncSpawnStepThrowExecutorFunction, gen, argument, NULL, NULL, false);
  849. }
  850. else
  851. {
  852. functionArg = library->CreatePromiseAsyncSpawnStepArgumentExecutorFunction(EntryJavascriptPromiseAsyncSpawnStepNextExecutorFunction, gen, argument, NULL, NULL, false);
  853. }
  854. AsyncSpawnStep(functionArg, gen, resolve, reject);
  855. return undefinedVar;
  856. }
  857. void JavascriptPromise::AsyncSpawnStep(JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction* nextFunction, JavascriptGenerator* gen, Var resolve, Var reject)
  858. {
  859. ScriptContext* scriptContext = gen->GetScriptContext();
  860. JavascriptLibrary* library = scriptContext->GetLibrary();
  861. Var undefinedVar = library->GetUndefined();
  862. JavascriptExceptionObject* exception = nullptr;
  863. Var value = nullptr;
  864. RecyclableObject* next = nullptr;
  865. bool done;
  866. try
  867. {
  868. Var nextVar = CALL_FUNCTION(scriptContext->GetThreadContext(), nextFunction, CallInfo(CallFlags_Value, 1), undefinedVar);
  869. next = RecyclableObject::FromVar(nextVar);
  870. }
  871. catch (const JavascriptException& err)
  872. {
  873. exception = err.GetAndClear();
  874. }
  875. if (exception != nullptr)
  876. {
  877. // finished with failure, reject the promise
  878. TryRejectWithExceptionObject(exception, reject, scriptContext);
  879. return;
  880. }
  881. Assert(next != nullptr);
  882. done = JavascriptConversion::ToBool(JavascriptOperators::GetProperty(next, PropertyIds::done, scriptContext), scriptContext);
  883. if (done)
  884. {
  885. // finished with success, resolve the promise
  886. value = JavascriptOperators::GetProperty(next, PropertyIds::value, scriptContext);
  887. if (!JavascriptConversion::IsCallable(resolve))
  888. {
  889. JavascriptError::ThrowTypeError(scriptContext, JSERR_NeedFunction);
  890. }
  891. CALL_FUNCTION(scriptContext->GetThreadContext(), RecyclableObject::FromVar(resolve), CallInfo(CallFlags_Value, 2), undefinedVar, value);
  892. return;
  893. }
  894. // not finished, chain off the yielded promise and `step` again
  895. JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction* successFunction = library->CreatePromiseAsyncSpawnStepArgumentExecutorFunction(EntryJavascriptPromiseAsyncSpawnCallStepExecutorFunction, gen, undefinedVar, resolve, reject);
  896. JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction* failFunction = library->CreatePromiseAsyncSpawnStepArgumentExecutorFunction(EntryJavascriptPromiseAsyncSpawnCallStepExecutorFunction, gen, undefinedVar, resolve, reject, true);
  897. JavascriptFunction* promiseResolve = library->EnsurePromiseResolveFunction();
  898. value = JavascriptOperators::GetProperty(next, PropertyIds::value, scriptContext);
  899. Var promiseVar = CALL_FUNCTION(scriptContext->GetThreadContext(), promiseResolve, CallInfo(CallFlags_Value, 2), library->GetPromiseConstructor(), value);
  900. JavascriptPromise* promise = FromVar(promiseVar);
  901. Var promiseThen = JavascriptOperators::GetProperty(promise, PropertyIds::then, scriptContext);
  902. if (!JavascriptConversion::IsCallable(promiseThen))
  903. {
  904. JavascriptError::ThrowTypeError(scriptContext, JSERR_NeedFunction);
  905. }
  906. CALL_FUNCTION(scriptContext->GetThreadContext(), RecyclableObject::FromVar(promiseThen), CallInfo(CallFlags_Value, 3), promise, successFunction, failFunction);
  907. }
  908. #if ENABLE_TTD
  909. void JavascriptPromise::MarkVisitKindSpecificPtrs(TTD::SnapshotExtractor* extractor)
  910. {
  911. if(this->result != nullptr)
  912. {
  913. extractor->MarkVisitVar(this->result);
  914. }
  915. if(this->resolveReactions != nullptr)
  916. {
  917. for(int32 i = 0; i < this->resolveReactions->Count(); ++i)
  918. {
  919. this->resolveReactions->Item(i)->MarkVisitPtrs(extractor);
  920. }
  921. }
  922. if(this->rejectReactions != nullptr)
  923. {
  924. for(int32 i = 0; i < this->rejectReactions->Count(); ++i)
  925. {
  926. this->rejectReactions->Item(i)->MarkVisitPtrs(extractor);
  927. }
  928. }
  929. }
  930. TTD::NSSnapObjects::SnapObjectType JavascriptPromise::GetSnapTag_TTD() const
  931. {
  932. return TTD::NSSnapObjects::SnapObjectType::SnapPromiseObject;
  933. }
  934. void JavascriptPromise::ExtractSnapObjectDataInto(TTD::NSSnapObjects::SnapObject* objData, TTD::SlabAllocator& alloc)
  935. {
  936. JsUtil::List<TTD_PTR_ID, HeapAllocator> depOnList(&HeapAllocator::Instance);
  937. TTD::NSSnapObjects::SnapPromiseInfo* spi = alloc.SlabAllocateStruct<TTD::NSSnapObjects::SnapPromiseInfo>();
  938. spi->Result = this->result;
  939. //Primitive kinds always inflated first so we only need to deal with complex kinds as depends on
  940. if(this->result != nullptr && TTD::JsSupport::IsVarComplexKind(this->result))
  941. {
  942. depOnList.Add(TTD_CONVERT_VAR_TO_PTR_ID(this->result));
  943. }
  944. spi->Status = this->status;
  945. spi->ResolveReactionCount = (this->resolveReactions != nullptr) ? this->resolveReactions->Count() : 0;
  946. spi->ResolveReactions = nullptr;
  947. if(spi->ResolveReactionCount != 0)
  948. {
  949. spi->ResolveReactions = alloc.SlabAllocateArray<TTD::NSSnapValues::SnapPromiseReactionInfo>(spi->ResolveReactionCount);
  950. for(uint32 i = 0; i < spi->ResolveReactionCount; ++i)
  951. {
  952. this->resolveReactions->Item(i)->ExtractSnapPromiseReactionInto(spi->ResolveReactions + i, depOnList, alloc);
  953. }
  954. }
  955. spi->RejectReactionCount = (this->rejectReactions != nullptr) ? this->rejectReactions->Count() : 0;
  956. spi->RejectReactions = nullptr;
  957. if(spi->RejectReactionCount != 0)
  958. {
  959. spi->RejectReactions = alloc.SlabAllocateArray<TTD::NSSnapValues::SnapPromiseReactionInfo>(spi->RejectReactionCount);
  960. for(uint32 i = 0; i < spi->RejectReactionCount; ++i)
  961. {
  962. this->rejectReactions->Item(i)->ExtractSnapPromiseReactionInto(spi->RejectReactions+ i, depOnList, alloc);
  963. }
  964. }
  965. //see what we need to do wrt dependencies
  966. if(depOnList.Count() == 0)
  967. {
  968. TTD::NSSnapObjects::StdExtractSetKindSpecificInfo<TTD::NSSnapObjects::SnapPromiseInfo*, TTD::NSSnapObjects::SnapObjectType::SnapPromiseObject>(objData, spi);
  969. }
  970. else
  971. {
  972. uint32 depOnCount = depOnList.Count();
  973. TTD_PTR_ID* depOnArray = alloc.SlabAllocateArray<TTD_PTR_ID>(depOnCount);
  974. for(uint32 i = 0; i < depOnCount; ++i)
  975. {
  976. depOnArray[i] = depOnList.Item(i);
  977. }
  978. TTD::NSSnapObjects::StdExtractSetKindSpecificInfo<TTD::NSSnapObjects::SnapPromiseInfo*, TTD::NSSnapObjects::SnapObjectType::SnapPromiseObject>(objData, spi, alloc, depOnCount, depOnArray);
  979. }
  980. }
  981. JavascriptPromise* JavascriptPromise::InitializePromise_TTD(ScriptContext* scriptContext, uint32 status, Var result, JsUtil::List<Js::JavascriptPromiseReaction*, HeapAllocator>& resolveReactions, JsUtil::List<Js::JavascriptPromiseReaction*, HeapAllocator>& rejectReactions)
  982. {
  983. Recycler* recycler = scriptContext->GetRecycler();
  984. JavascriptLibrary* library = scriptContext->GetLibrary();
  985. JavascriptPromise* promise = library->CreatePromise();
  986. promise->status = (PromiseStatus)status;
  987. promise->result = result;
  988. promise->resolveReactions = RecyclerNew(recycler, JavascriptPromiseReactionList, recycler);
  989. promise->resolveReactions->Copy(&resolveReactions);
  990. promise->rejectReactions = RecyclerNew(recycler, JavascriptPromiseReactionList, recycler);
  991. promise->rejectReactions->Copy(&rejectReactions);
  992. return promise;
  993. }
  994. #endif
  995. // NewPromiseCapability as described in ES6.0 (draft 29) Section 25.4.1.6
  996. JavascriptPromiseCapability* JavascriptPromise::NewPromiseCapability(Var constructor, ScriptContext* scriptContext)
  997. {
  998. if (!JavascriptOperators::IsConstructor(constructor))
  999. {
  1000. JavascriptError::ThrowTypeError(scriptContext, JSERR_NeedFunction);
  1001. }
  1002. RecyclableObject* constructorFunc = RecyclableObject::FromVar(constructor);
  1003. return CreatePromiseCapabilityRecord(constructorFunc, scriptContext);
  1004. }
  1005. // CreatePromiseCapabilityRecord as described in ES6.0 (draft 29) Section 25.4.1.6.1
  1006. JavascriptPromiseCapability* JavascriptPromise::CreatePromiseCapabilityRecord(RecyclableObject* constructor, ScriptContext* scriptContext)
  1007. {
  1008. JavascriptLibrary* library = scriptContext->GetLibrary();
  1009. Var undefinedVar = library->GetUndefined();
  1010. JavascriptPromiseCapability* promiseCapability = JavascriptPromiseCapability::New(undefinedVar, undefinedVar, undefinedVar, scriptContext);
  1011. JavascriptPromiseCapabilitiesExecutorFunction* executor = library->CreatePromiseCapabilitiesExecutorFunction(EntryCapabilitiesExecutorFunction, promiseCapability);
  1012. CallInfo callinfo = Js::CallInfo((Js::CallFlags)(Js::CallFlags::CallFlags_Value | Js::CallFlags::CallFlags_New), 2);
  1013. Var argVars[] = { constructor, executor };
  1014. Arguments args(callinfo, argVars);
  1015. Var promise = JavascriptFunction::CallAsConstructor(constructor, nullptr, args, scriptContext);
  1016. if (!JavascriptConversion::IsCallable(promiseCapability->GetResolve()) || !JavascriptConversion::IsCallable(promiseCapability->GetReject()))
  1017. {
  1018. JavascriptError::ThrowTypeError(scriptContext, JSERR_NeedFunction, _u("Promise"));
  1019. }
  1020. promiseCapability->SetPromise(promise);
  1021. return promiseCapability;
  1022. }
  1023. // TriggerPromiseReactions as defined in ES 2015 Section 25.4.1.7
  1024. Var JavascriptPromise::TriggerPromiseReactions(JavascriptPromiseReactionList* reactions, Var resolution, ScriptContext* scriptContext)
  1025. {
  1026. JavascriptLibrary* library = scriptContext->GetLibrary();
  1027. if (reactions != nullptr)
  1028. {
  1029. for (int i = 0; i < reactions->Count(); i++)
  1030. {
  1031. JavascriptPromiseReaction* reaction = reactions->Item(i);
  1032. EnqueuePromiseReactionTask(reaction, resolution, scriptContext);
  1033. }
  1034. }
  1035. return library->GetUndefined();
  1036. }
  1037. void JavascriptPromise::EnqueuePromiseReactionTask(JavascriptPromiseReaction* reaction, Var resolution, ScriptContext* scriptContext)
  1038. {
  1039. Assert(resolution != nullptr);
  1040. JavascriptLibrary* library = scriptContext->GetLibrary();
  1041. JavascriptPromiseReactionTaskFunction* reactionTaskFunction = library->CreatePromiseReactionTaskFunction(EntryReactionTaskFunction, reaction, resolution);
  1042. library->EnqueueTask(reactionTaskFunction);
  1043. }
  1044. JavascriptPromiseResolveOrRejectFunction::JavascriptPromiseResolveOrRejectFunction(DynamicType* type)
  1045. : RuntimeFunction(type, &Js::JavascriptPromise::EntryInfo::ResolveOrRejectFunction), promise(nullptr), isReject(false), alreadyResolvedWrapper(nullptr)
  1046. { }
  1047. JavascriptPromiseResolveOrRejectFunction::JavascriptPromiseResolveOrRejectFunction(DynamicType* type, FunctionInfo* functionInfo, JavascriptPromise* promise, bool isReject, JavascriptPromiseResolveOrRejectFunctionAlreadyResolvedWrapper* alreadyResolvedRecord)
  1048. : RuntimeFunction(type, functionInfo), promise(promise), isReject(isReject), alreadyResolvedWrapper(alreadyResolvedRecord)
  1049. { }
  1050. bool JavascriptPromiseResolveOrRejectFunction::Is(Var var)
  1051. {
  1052. if (JavascriptFunction::Is(var))
  1053. {
  1054. JavascriptFunction* obj = JavascriptFunction::UnsafeFromVar(var);
  1055. return VirtualTableInfo<JavascriptPromiseResolveOrRejectFunction>::HasVirtualTable(obj)
  1056. || VirtualTableInfo<CrossSiteObject<JavascriptPromiseResolveOrRejectFunction>>::HasVirtualTable(obj);
  1057. }
  1058. return false;
  1059. }
  1060. JavascriptPromiseResolveOrRejectFunction* JavascriptPromiseResolveOrRejectFunction::FromVar(Var var)
  1061. {
  1062. AssertOrFailFast(JavascriptPromiseResolveOrRejectFunction::Is(var));
  1063. return static_cast<JavascriptPromiseResolveOrRejectFunction*>(var);
  1064. }
  1065. JavascriptPromiseResolveOrRejectFunction* JavascriptPromiseResolveOrRejectFunction::UnsafeFromVar(Var var)
  1066. {
  1067. Assert(JavascriptPromiseResolveOrRejectFunction::Is(var));
  1068. return static_cast<JavascriptPromiseResolveOrRejectFunction*>(var);
  1069. }
  1070. JavascriptPromise* JavascriptPromiseResolveOrRejectFunction::GetPromise()
  1071. {
  1072. return this->promise;
  1073. }
  1074. bool JavascriptPromiseResolveOrRejectFunction::IsRejectFunction()
  1075. {
  1076. return this->isReject;
  1077. }
  1078. bool JavascriptPromiseResolveOrRejectFunction::IsAlreadyResolved()
  1079. {
  1080. Assert(this->alreadyResolvedWrapper);
  1081. return this->alreadyResolvedWrapper->alreadyResolved;
  1082. }
  1083. void JavascriptPromiseResolveOrRejectFunction::SetAlreadyResolved(bool is)
  1084. {
  1085. Assert(this->alreadyResolvedWrapper);
  1086. this->alreadyResolvedWrapper->alreadyResolved = is;
  1087. }
  1088. #if ENABLE_TTD
  1089. void JavascriptPromiseResolveOrRejectFunction::MarkVisitKindSpecificPtrs(TTD::SnapshotExtractor* extractor)
  1090. {
  1091. TTDAssert(this->promise != nullptr, "Was not expecting that!!!");
  1092. extractor->MarkVisitVar(this->promise);
  1093. }
  1094. TTD::NSSnapObjects::SnapObjectType JavascriptPromiseResolveOrRejectFunction::GetSnapTag_TTD() const
  1095. {
  1096. return TTD::NSSnapObjects::SnapObjectType::SnapPromiseResolveOrRejectFunctionObject;
  1097. }
  1098. void JavascriptPromiseResolveOrRejectFunction::ExtractSnapObjectDataInto(TTD::NSSnapObjects::SnapObject* objData, TTD::SlabAllocator& alloc)
  1099. {
  1100. TTD::NSSnapObjects::SnapPromiseResolveOrRejectFunctionInfo* sprri = alloc.SlabAllocateStruct<TTD::NSSnapObjects::SnapPromiseResolveOrRejectFunctionInfo>();
  1101. uint32 depOnCount = 1;
  1102. TTD_PTR_ID* depOnArray = alloc.SlabAllocateArray<TTD_PTR_ID>(depOnCount);
  1103. sprri->PromiseId = TTD_CONVERT_VAR_TO_PTR_ID(this->promise);
  1104. depOnArray[0] = sprri->PromiseId;
  1105. sprri->IsReject = this->isReject;
  1106. sprri->AlreadyResolvedWrapperId = TTD_CONVERT_PROMISE_INFO_TO_PTR_ID(this->alreadyResolvedWrapper);
  1107. sprri->AlreadyResolvedValue = this->alreadyResolvedWrapper->alreadyResolved;
  1108. TTD::NSSnapObjects::StdExtractSetKindSpecificInfo<TTD::NSSnapObjects::SnapPromiseResolveOrRejectFunctionInfo*, TTD::NSSnapObjects::SnapObjectType::SnapPromiseResolveOrRejectFunctionObject>(objData, sprri, alloc, depOnCount, depOnArray);
  1109. }
  1110. #endif
  1111. JavascriptPromiseAsyncSpawnExecutorFunction::JavascriptPromiseAsyncSpawnExecutorFunction(DynamicType* type, FunctionInfo* functionInfo, JavascriptGenerator* generator, Var target)
  1112. : RuntimeFunction(type, functionInfo), generator(generator), target(target)
  1113. { }
  1114. bool JavascriptPromiseAsyncSpawnExecutorFunction::Is(Var var)
  1115. {
  1116. if (JavascriptFunction::Is(var))
  1117. {
  1118. JavascriptFunction* obj = JavascriptFunction::UnsafeFromVar(var);
  1119. return VirtualTableInfo<JavascriptPromiseAsyncSpawnExecutorFunction>::HasVirtualTable(obj)
  1120. || VirtualTableInfo<CrossSiteObject<JavascriptPromiseAsyncSpawnExecutorFunction>>::HasVirtualTable(obj);
  1121. }
  1122. return false;
  1123. }
  1124. JavascriptPromiseAsyncSpawnExecutorFunction* JavascriptPromiseAsyncSpawnExecutorFunction::FromVar(Var var)
  1125. {
  1126. AssertOrFailFast(JavascriptPromiseAsyncSpawnExecutorFunction::Is(var));
  1127. return static_cast<JavascriptPromiseAsyncSpawnExecutorFunction*>(var);
  1128. }
  1129. JavascriptPromiseAsyncSpawnExecutorFunction* JavascriptPromiseAsyncSpawnExecutorFunction::UnsafeFromVar(Var var)
  1130. {
  1131. Assert(JavascriptPromiseAsyncSpawnExecutorFunction::Is(var));
  1132. return static_cast<JavascriptPromiseAsyncSpawnExecutorFunction*>(var);
  1133. }
  1134. JavascriptGenerator* JavascriptPromiseAsyncSpawnExecutorFunction::GetGenerator()
  1135. {
  1136. return this->generator;
  1137. }
  1138. Var JavascriptPromiseAsyncSpawnExecutorFunction::GetTarget()
  1139. {
  1140. return this->target;
  1141. }
  1142. #if ENABLE_TTD
  1143. void JavascriptPromiseAsyncSpawnExecutorFunction::MarkVisitKindSpecificPtrs(TTD::SnapshotExtractor* extractor)
  1144. {
  1145. if (this->generator != nullptr)
  1146. {
  1147. extractor->MarkVisitVar(this->generator);
  1148. }
  1149. if (this->target != nullptr)
  1150. {
  1151. extractor->MarkVisitVar(this->target);
  1152. }
  1153. }
  1154. TTD::NSSnapObjects::SnapObjectType JavascriptPromiseAsyncSpawnExecutorFunction::GetSnapTag_TTD() const
  1155. {
  1156. return TTD::NSSnapObjects::SnapObjectType::JavascriptPromiseAsyncSpawnExecutorFunction;
  1157. }
  1158. void JavascriptPromiseAsyncSpawnExecutorFunction::ExtractSnapObjectDataInto(TTD::NSSnapObjects::SnapObject* objData, TTD::SlabAllocator& alloc)
  1159. {
  1160. TTD::NSSnapObjects::SnapJavascriptPromiseAsyncSpawnExecutorFunctionInfo* info = alloc.SlabAllocateStruct<TTD::NSSnapObjects::SnapJavascriptPromiseAsyncSpawnExecutorFunctionInfo>();
  1161. info->generator= TTD_CONVERT_VAR_TO_PTR_ID(this->generator);
  1162. info->target = TTD_CONVERT_JSVAR_TO_TTDVAR(this->target);
  1163. TTD::NSSnapObjects::StdExtractSetKindSpecificInfo<TTD::NSSnapObjects::SnapJavascriptPromiseAsyncSpawnExecutorFunctionInfo*, TTD::NSSnapObjects::SnapObjectType::JavascriptPromiseAsyncSpawnExecutorFunction>(objData, info);
  1164. }
  1165. #endif
  1166. JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction(DynamicType* type, FunctionInfo* functionInfo, JavascriptGenerator* generator, Var argument, Var resolve, Var reject, bool isReject)
  1167. : RuntimeFunction(type, functionInfo), generator(generator), argument(argument), resolve(resolve), reject(reject), isReject(isReject)
  1168. { }
  1169. bool JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::Is(Var var)
  1170. {
  1171. if (JavascriptFunction::Is(var))
  1172. {
  1173. JavascriptFunction* obj = JavascriptFunction::UnsafeFromVar(var);
  1174. return VirtualTableInfo<JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction>::HasVirtualTable(obj)
  1175. || VirtualTableInfo<CrossSiteObject<JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction>>::HasVirtualTable(obj);
  1176. }
  1177. return false;
  1178. }
  1179. JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction* JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::FromVar(Var var)
  1180. {
  1181. AssertOrFailFast(JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::Is(var));
  1182. return static_cast<JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction*>(var);
  1183. }
  1184. JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction* JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::UnsafeFromVar(Var var)
  1185. {
  1186. Assert(JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::Is(var));
  1187. return static_cast<JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction*>(var);
  1188. }
  1189. JavascriptGenerator* JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::GetGenerator()
  1190. {
  1191. return this->generator;
  1192. }
  1193. Var JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::GetResolve()
  1194. {
  1195. return this->resolve;
  1196. }
  1197. Var JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::GetReject()
  1198. {
  1199. return this->reject;
  1200. }
  1201. bool JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::GetIsReject()
  1202. {
  1203. return this->isReject;
  1204. }
  1205. Var JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::GetArgument()
  1206. {
  1207. return this->argument;
  1208. }
  1209. #if ENABLE_TTD
  1210. void JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::MarkVisitKindSpecificPtrs(TTD::SnapshotExtractor* extractor)
  1211. {
  1212. if (this->generator != nullptr)
  1213. {
  1214. extractor->MarkVisitVar(this->generator);
  1215. }
  1216. if (this->reject != nullptr)
  1217. {
  1218. extractor->MarkVisitVar(this->reject);
  1219. }
  1220. if (this->resolve != nullptr)
  1221. {
  1222. extractor->MarkVisitVar(this->resolve);
  1223. }
  1224. if (this->argument != nullptr)
  1225. {
  1226. extractor->MarkVisitVar(this->argument);
  1227. }
  1228. }
  1229. TTD::NSSnapObjects::SnapObjectType JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::GetSnapTag_TTD() const
  1230. {
  1231. return TTD::NSSnapObjects::SnapObjectType::JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction;
  1232. }
  1233. void JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction::ExtractSnapObjectDataInto(TTD::NSSnapObjects::SnapObject* objData, TTD::SlabAllocator& alloc)
  1234. {
  1235. TTD::NSSnapObjects::SnapJavascriptPromiseAsyncSpawnStepArgumentExecutorFunctionInfo* info = alloc.SlabAllocateStruct<TTD::NSSnapObjects::SnapJavascriptPromiseAsyncSpawnStepArgumentExecutorFunctionInfo>();
  1236. info->generator = TTD_CONVERT_VAR_TO_PTR_ID(this->generator);
  1237. info->reject = this->reject;
  1238. info->resolve = this->resolve;
  1239. info->argument = this->argument;
  1240. info->isReject = this->isReject;
  1241. info->entryPoint = 0;
  1242. JavascriptMethod entryPoint = this->GetFunctionInfo()->GetOriginalEntryPoint();
  1243. if (entryPoint == JavascriptPromise::EntryJavascriptPromiseAsyncSpawnStepNextExecutorFunction)
  1244. {
  1245. info->entryPoint = 1;
  1246. }
  1247. else if (entryPoint == JavascriptPromise::EntryJavascriptPromiseAsyncSpawnStepThrowExecutorFunction)
  1248. {
  1249. info->entryPoint = 2;
  1250. }
  1251. else if (entryPoint == JavascriptPromise::EntryJavascriptPromiseAsyncSpawnCallStepExecutorFunction)
  1252. {
  1253. info->entryPoint = 3;
  1254. }
  1255. else
  1256. {
  1257. TTDAssert(false, "Unexpected entrypoint found JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction");
  1258. }
  1259. const uint32 maxDeps = 4;
  1260. uint32 depCount = 0;
  1261. TTD_PTR_ID* depArray = alloc.SlabReserveArraySpace<TTD_PTR_ID>(maxDeps);
  1262. if (this->reject != nullptr && TTD::JsSupport::IsVarComplexKind(this->reject))
  1263. {
  1264. depArray[depCount] = TTD_CONVERT_VAR_TO_PTR_ID(this->reject);
  1265. depCount++;
  1266. }
  1267. if (this->resolve != nullptr && TTD::JsSupport::IsVarComplexKind(this->resolve))
  1268. {
  1269. depArray[depCount] = TTD_CONVERT_VAR_TO_PTR_ID(this->resolve);
  1270. depCount++;
  1271. }
  1272. if (this->argument != nullptr && TTD::JsSupport::IsVarComplexKind(this->argument))
  1273. {
  1274. depArray[depCount] = TTD_CONVERT_VAR_TO_PTR_ID(this->argument);
  1275. depCount++;
  1276. }
  1277. if (this->generator != nullptr)
  1278. {
  1279. depArray[depCount] = TTD_CONVERT_VAR_TO_PTR_ID(this->generator);
  1280. depCount++;
  1281. }
  1282. if (depCount > 0)
  1283. {
  1284. alloc.SlabCommitArraySpace<TTD_PTR_ID>(depCount, maxDeps);
  1285. }
  1286. else
  1287. {
  1288. alloc.SlabAbortArraySpace<TTD_PTR_ID>(maxDeps);
  1289. }
  1290. if (depCount == 0)
  1291. {
  1292. TTD::NSSnapObjects::StdExtractSetKindSpecificInfo<TTD::NSSnapObjects::SnapJavascriptPromiseAsyncSpawnStepArgumentExecutorFunctionInfo*, TTD::NSSnapObjects::SnapObjectType::JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction>(objData, info);
  1293. }
  1294. else
  1295. {
  1296. TTDAssert(depArray != nullptr, "depArray should be non-null if depCount is > 0");
  1297. TTD::NSSnapObjects::StdExtractSetKindSpecificInfo<TTD::NSSnapObjects::SnapJavascriptPromiseAsyncSpawnStepArgumentExecutorFunctionInfo*, TTD::NSSnapObjects::SnapObjectType::JavascriptPromiseAsyncSpawnStepArgumentExecutorFunction>(objData, info, alloc, depCount, depArray);
  1298. }
  1299. }
  1300. #endif
  1301. JavascriptPromiseCapabilitiesExecutorFunction::JavascriptPromiseCapabilitiesExecutorFunction(DynamicType* type, FunctionInfo* functionInfo, JavascriptPromiseCapability* capability)
  1302. : RuntimeFunction(type, functionInfo), capability(capability)
  1303. { }
  1304. bool JavascriptPromiseCapabilitiesExecutorFunction::Is(Var var)
  1305. {
  1306. if (JavascriptFunction::Is(var))
  1307. {
  1308. JavascriptFunction* obj = JavascriptFunction::UnsafeFromVar(var);
  1309. return VirtualTableInfo<JavascriptPromiseCapabilitiesExecutorFunction>::HasVirtualTable(obj)
  1310. || VirtualTableInfo<CrossSiteObject<JavascriptPromiseCapabilitiesExecutorFunction>>::HasVirtualTable(obj);
  1311. }
  1312. return false;
  1313. }
  1314. JavascriptPromiseCapabilitiesExecutorFunction* JavascriptPromiseCapabilitiesExecutorFunction::FromVar(Var var)
  1315. {
  1316. AssertOrFailFast(JavascriptPromiseCapabilitiesExecutorFunction::Is(var));
  1317. return static_cast<JavascriptPromiseCapabilitiesExecutorFunction*>(var);
  1318. }
  1319. JavascriptPromiseCapabilitiesExecutorFunction* JavascriptPromiseCapabilitiesExecutorFunction::UnsafeFromVar(Var var)
  1320. {
  1321. Assert(JavascriptPromiseCapabilitiesExecutorFunction::Is(var));
  1322. return static_cast<JavascriptPromiseCapabilitiesExecutorFunction*>(var);
  1323. }
  1324. JavascriptPromiseCapability* JavascriptPromiseCapabilitiesExecutorFunction::GetCapability()
  1325. {
  1326. return this->capability;
  1327. }
  1328. #if ENABLE_TTD
  1329. void JavascriptPromiseCapabilitiesExecutorFunction::MarkVisitKindSpecificPtrs(TTD::SnapshotExtractor* extractor)
  1330. {
  1331. TTDAssert(false, "Not Implemented Yet");
  1332. }
  1333. TTD::NSSnapObjects::SnapObjectType JavascriptPromiseCapabilitiesExecutorFunction::GetSnapTag_TTD() const
  1334. {
  1335. TTDAssert(false, "Not Implemented Yet");
  1336. return TTD::NSSnapObjects::SnapObjectType::Invalid;
  1337. }
  1338. void JavascriptPromiseCapabilitiesExecutorFunction::ExtractSnapObjectDataInto(TTD::NSSnapObjects::SnapObject* objData, TTD::SlabAllocator& alloc)
  1339. {
  1340. TTDAssert(false, "Not Implemented Yet");
  1341. }
  1342. #endif
  1343. JavascriptPromiseCapability* JavascriptPromiseCapability::New(Var promise, Var resolve, Var reject, ScriptContext* scriptContext)
  1344. {
  1345. return RecyclerNew(scriptContext->GetRecycler(), JavascriptPromiseCapability, promise, resolve, reject);
  1346. }
  1347. Var JavascriptPromiseCapability::GetResolve()
  1348. {
  1349. return this->resolve;
  1350. }
  1351. Var JavascriptPromiseCapability::GetReject()
  1352. {
  1353. return this->reject;
  1354. }
  1355. Var JavascriptPromiseCapability::GetPromise()
  1356. {
  1357. return this->promise;
  1358. }
  1359. void JavascriptPromiseCapability::SetPromise(Var promise)
  1360. {
  1361. this->promise = promise;
  1362. }
  1363. void JavascriptPromiseCapability::SetResolve(Var resolve)
  1364. {
  1365. this->resolve = resolve;
  1366. }
  1367. void JavascriptPromiseCapability::SetReject(Var reject)
  1368. {
  1369. this->reject = reject;
  1370. }
  1371. #if ENABLE_TTD
  1372. void JavascriptPromiseCapability::MarkVisitPtrs(TTD::SnapshotExtractor* extractor)
  1373. {
  1374. TTDAssert(this->promise != nullptr && this->resolve != nullptr && this->reject != nullptr, "Seems odd, I was not expecting this!!!");
  1375. extractor->MarkVisitVar(this->promise);
  1376. extractor->MarkVisitVar(this->resolve);
  1377. extractor->MarkVisitVar(this->reject);
  1378. }
  1379. void JavascriptPromiseCapability::ExtractSnapPromiseCapabilityInto(TTD::NSSnapValues::SnapPromiseCapabilityInfo* snapPromiseCapability, JsUtil::List<TTD_PTR_ID, HeapAllocator>& depOnList, TTD::SlabAllocator& alloc)
  1380. {
  1381. snapPromiseCapability->CapabilityId = TTD_CONVERT_PROMISE_INFO_TO_PTR_ID(this);
  1382. snapPromiseCapability->PromiseVar = this->promise;
  1383. if(TTD::JsSupport::IsVarComplexKind(this->promise))
  1384. {
  1385. depOnList.Add(TTD_CONVERT_VAR_TO_PTR_ID(this->resolve));
  1386. }
  1387. snapPromiseCapability->ResolveVar = this->resolve;
  1388. if(TTD::JsSupport::IsVarComplexKind(this->resolve))
  1389. {
  1390. depOnList.Add(TTD_CONVERT_VAR_TO_PTR_ID(this->resolve));
  1391. }
  1392. snapPromiseCapability->RejectVar = this->reject;
  1393. if(TTD::JsSupport::IsVarComplexKind(this->reject))
  1394. {
  1395. depOnList.Add(TTD_CONVERT_VAR_TO_PTR_ID(this->reject));
  1396. }
  1397. }
  1398. #endif
  1399. JavascriptPromiseReaction* JavascriptPromiseReaction::New(JavascriptPromiseCapability* capabilities, RecyclableObject* handler, ScriptContext* scriptContext)
  1400. {
  1401. return RecyclerNew(scriptContext->GetRecycler(), JavascriptPromiseReaction, capabilities, handler);
  1402. }
  1403. JavascriptPromiseCapability* JavascriptPromiseReaction::GetCapabilities()
  1404. {
  1405. return this->capabilities;
  1406. }
  1407. RecyclableObject* JavascriptPromiseReaction::GetHandler()
  1408. {
  1409. return this->handler;
  1410. }
  1411. #if ENABLE_TTD
  1412. void JavascriptPromiseReaction::MarkVisitPtrs(TTD::SnapshotExtractor* extractor)
  1413. {
  1414. TTDAssert(this->handler != nullptr && this->capabilities != nullptr, "Seems odd, I was not expecting this!!!");
  1415. extractor->MarkVisitVar(this->handler);
  1416. this->capabilities->MarkVisitPtrs(extractor);
  1417. }
  1418. void JavascriptPromiseReaction::ExtractSnapPromiseReactionInto(TTD::NSSnapValues::SnapPromiseReactionInfo* snapPromiseReaction, JsUtil::List<TTD_PTR_ID, HeapAllocator>& depOnList, TTD::SlabAllocator& alloc)
  1419. {
  1420. TTDAssert(this->handler != nullptr && this->capabilities != nullptr, "Seems odd, I was not expecting this!!!");
  1421. snapPromiseReaction->PromiseReactionId = TTD_CONVERT_PROMISE_INFO_TO_PTR_ID(this);
  1422. snapPromiseReaction->HandlerObjId = TTD_CONVERT_VAR_TO_PTR_ID(this->handler);
  1423. depOnList.Add(snapPromiseReaction->HandlerObjId);
  1424. this->capabilities->ExtractSnapPromiseCapabilityInto(&snapPromiseReaction->Capabilities, depOnList, alloc);
  1425. }
  1426. #endif
  1427. JavascriptPromiseReaction* JavascriptPromiseReactionTaskFunction::GetReaction()
  1428. {
  1429. return this->reaction;
  1430. }
  1431. Var JavascriptPromiseReactionTaskFunction::GetArgument()
  1432. {
  1433. return this->argument;
  1434. }
  1435. #if ENABLE_TTD
  1436. void JavascriptPromiseReactionTaskFunction::MarkVisitKindSpecificPtrs(TTD::SnapshotExtractor* extractor)
  1437. {
  1438. TTDAssert(this->argument != nullptr && this->reaction != nullptr, "Was not expecting this!!!");
  1439. extractor->MarkVisitVar(this->argument);
  1440. this->reaction->MarkVisitPtrs(extractor);
  1441. }
  1442. TTD::NSSnapObjects::SnapObjectType JavascriptPromiseReactionTaskFunction::GetSnapTag_TTD() const
  1443. {
  1444. return TTD::NSSnapObjects::SnapObjectType::SnapPromiseReactionTaskFunctionObject;
  1445. }
  1446. void JavascriptPromiseReactionTaskFunction::ExtractSnapObjectDataInto(TTD::NSSnapObjects::SnapObject* objData, TTD::SlabAllocator& alloc)
  1447. {
  1448. TTD::NSSnapObjects::SnapPromiseReactionTaskFunctionInfo* sprtfi = alloc.SlabAllocateStruct<TTD::NSSnapObjects::SnapPromiseReactionTaskFunctionInfo>();
  1449. JsUtil::List<TTD_PTR_ID, HeapAllocator> depOnList(&HeapAllocator::Instance);
  1450. sprtfi->Argument = this->argument;
  1451. if(this->argument != nullptr && TTD::JsSupport::IsVarComplexKind(this->argument))
  1452. {
  1453. depOnList.Add(TTD_CONVERT_VAR_TO_PTR_ID(this->argument));
  1454. }
  1455. this->reaction->ExtractSnapPromiseReactionInto(&sprtfi->Reaction, depOnList, alloc);
  1456. //see what we need to do wrt dependencies
  1457. if(depOnList.Count() == 0)
  1458. {
  1459. TTD::NSSnapObjects::StdExtractSetKindSpecificInfo<TTD::NSSnapObjects::SnapPromiseReactionTaskFunctionInfo*, TTD::NSSnapObjects::SnapObjectType::SnapPromiseReactionTaskFunctionObject>(objData, sprtfi);
  1460. }
  1461. else
  1462. {
  1463. uint32 depOnCount = depOnList.Count();
  1464. TTD_PTR_ID* depOnArray = alloc.SlabAllocateArray<TTD_PTR_ID>(depOnCount);
  1465. for(uint32 i = 0; i < depOnCount; ++i)
  1466. {
  1467. depOnArray[i] = depOnList.Item(i);
  1468. }
  1469. TTD::NSSnapObjects::StdExtractSetKindSpecificInfo<TTD::NSSnapObjects::SnapPromiseReactionTaskFunctionInfo*, TTD::NSSnapObjects::SnapObjectType::SnapPromiseReactionTaskFunctionObject>(objData, sprtfi, alloc, depOnCount, depOnArray);
  1470. }
  1471. }
  1472. #endif
  1473. JavascriptPromise* JavascriptPromiseResolveThenableTaskFunction::GetPromise()
  1474. {
  1475. return this->promise;
  1476. }
  1477. RecyclableObject* JavascriptPromiseResolveThenableTaskFunction::GetThenable()
  1478. {
  1479. return this->thenable;
  1480. }
  1481. RecyclableObject* JavascriptPromiseResolveThenableTaskFunction::GetThenFunction()
  1482. {
  1483. return this->thenFunction;
  1484. }
  1485. #if ENABLE_TTD
  1486. void JavascriptPromiseResolveThenableTaskFunction::MarkVisitKindSpecificPtrs(TTD::SnapshotExtractor* extractor)
  1487. {
  1488. TTDAssert(false, "Not Implemented Yet");
  1489. }
  1490. TTD::NSSnapObjects::SnapObjectType JavascriptPromiseResolveThenableTaskFunction::GetSnapTag_TTD() const
  1491. {
  1492. TTDAssert(false, "Not Implemented Yet");
  1493. return TTD::NSSnapObjects::SnapObjectType::Invalid;
  1494. }
  1495. void JavascriptPromiseResolveThenableTaskFunction::ExtractSnapObjectDataInto(TTD::NSSnapObjects::SnapObject* objData, TTD::SlabAllocator& alloc)
  1496. {
  1497. TTDAssert(false, "Not Implemented Yet");
  1498. }
  1499. #endif
  1500. JavascriptPromiseAllResolveElementFunction::JavascriptPromiseAllResolveElementFunction(DynamicType* type)
  1501. : RuntimeFunction(type, &Js::JavascriptPromise::EntryInfo::AllResolveElementFunction), index(0), values(nullptr), capabilities(nullptr), remainingElementsWrapper(nullptr), alreadyCalled(false)
  1502. { }
  1503. JavascriptPromiseAllResolveElementFunction::JavascriptPromiseAllResolveElementFunction(DynamicType* type, FunctionInfo* functionInfo, uint32 index, JavascriptArray* values, JavascriptPromiseCapability* capabilities, JavascriptPromiseAllResolveElementFunctionRemainingElementsWrapper* remainingElementsWrapper)
  1504. : RuntimeFunction(type, functionInfo), index(index), values(values), capabilities(capabilities), remainingElementsWrapper(remainingElementsWrapper), alreadyCalled(false)
  1505. { }
  1506. bool JavascriptPromiseAllResolveElementFunction::Is(Var var)
  1507. {
  1508. if (JavascriptFunction::Is(var))
  1509. {
  1510. JavascriptFunction* obj = JavascriptFunction::UnsafeFromVar(var);
  1511. return VirtualTableInfo<JavascriptPromiseAllResolveElementFunction>::HasVirtualTable(obj)
  1512. || VirtualTableInfo<CrossSiteObject<JavascriptPromiseAllResolveElementFunction>>::HasVirtualTable(obj);
  1513. }
  1514. return false;
  1515. }
  1516. JavascriptPromiseAllResolveElementFunction* JavascriptPromiseAllResolveElementFunction::FromVar(Var var)
  1517. {
  1518. AssertOrFailFast(JavascriptPromiseAllResolveElementFunction::Is(var));
  1519. return static_cast<JavascriptPromiseAllResolveElementFunction*>(var);
  1520. }
  1521. JavascriptPromiseAllResolveElementFunction* JavascriptPromiseAllResolveElementFunction::UnsafeFromVar(Var var)
  1522. {
  1523. Assert(JavascriptPromiseAllResolveElementFunction::Is(var));
  1524. return static_cast<JavascriptPromiseAllResolveElementFunction*>(var);
  1525. }
  1526. JavascriptPromiseCapability* JavascriptPromiseAllResolveElementFunction::GetCapabilities()
  1527. {
  1528. return this->capabilities;
  1529. }
  1530. uint32 JavascriptPromiseAllResolveElementFunction::GetIndex()
  1531. {
  1532. return this->index;
  1533. }
  1534. uint32 JavascriptPromiseAllResolveElementFunction::GetRemainingElements()
  1535. {
  1536. return this->remainingElementsWrapper->remainingElements;
  1537. }
  1538. JavascriptArray* JavascriptPromiseAllResolveElementFunction::GetValues()
  1539. {
  1540. return this->values;
  1541. }
  1542. uint32 JavascriptPromiseAllResolveElementFunction::DecrementRemainingElements()
  1543. {
  1544. return --(this->remainingElementsWrapper->remainingElements);
  1545. }
  1546. bool JavascriptPromiseAllResolveElementFunction::IsAlreadyCalled() const
  1547. {
  1548. return this->alreadyCalled;
  1549. }
  1550. void JavascriptPromiseAllResolveElementFunction::SetAlreadyCalled(const bool is)
  1551. {
  1552. this->alreadyCalled = is;
  1553. }
  1554. #if ENABLE_TTD
  1555. void JavascriptPromiseAllResolveElementFunction::MarkVisitKindSpecificPtrs(TTD::SnapshotExtractor* extractor)
  1556. {
  1557. TTDAssert(this->capabilities != nullptr && this->remainingElementsWrapper != nullptr && this->values != nullptr, "Don't think these can be null");
  1558. this->capabilities->MarkVisitPtrs(extractor);
  1559. extractor->MarkVisitVar(this->values);
  1560. }
  1561. TTD::NSSnapObjects::SnapObjectType JavascriptPromiseAllResolveElementFunction::GetSnapTag_TTD() const
  1562. {
  1563. return TTD::NSSnapObjects::SnapObjectType::SnapPromiseAllResolveElementFunctionObject;
  1564. }
  1565. void JavascriptPromiseAllResolveElementFunction::ExtractSnapObjectDataInto(TTD::NSSnapObjects::SnapObject* objData, TTD::SlabAllocator& alloc)
  1566. {
  1567. TTD::NSSnapObjects::SnapPromiseAllResolveElementFunctionInfo* sprai = alloc.SlabAllocateStruct<TTD::NSSnapObjects::SnapPromiseAllResolveElementFunctionInfo>();
  1568. JsUtil::List<TTD_PTR_ID, HeapAllocator> depOnList(&HeapAllocator::Instance);
  1569. this->capabilities->ExtractSnapPromiseCapabilityInto(&sprai->Capabilities, depOnList, alloc);
  1570. sprai->Index = this->index;
  1571. sprai->RemainingElementsWrapperId = TTD_CONVERT_PROMISE_INFO_TO_PTR_ID(this->remainingElementsWrapper);
  1572. sprai->RemainingElementsValue = this->remainingElementsWrapper->remainingElements;
  1573. sprai->Values = TTD_CONVERT_VAR_TO_PTR_ID(this->values);
  1574. depOnList.Add(sprai->Values);
  1575. sprai->AlreadyCalled = this->alreadyCalled;
  1576. uint32 depOnCount = depOnList.Count();
  1577. TTD_PTR_ID* depOnArray = alloc.SlabAllocateArray<TTD_PTR_ID>(depOnCount);
  1578. for(uint32 i = 0; i < depOnCount; ++i)
  1579. {
  1580. depOnArray[i] = depOnList.Item(i);
  1581. }
  1582. TTD::NSSnapObjects::StdExtractSetKindSpecificInfo<TTD::NSSnapObjects::SnapPromiseAllResolveElementFunctionInfo*, TTD::NSSnapObjects::SnapObjectType::SnapPromiseAllResolveElementFunctionObject>(objData, sprai, alloc, depOnCount, depOnArray);
  1583. }
  1584. #endif
  1585. Var JavascriptPromise::EntryGetterSymbolSpecies(RecyclableObject* function, CallInfo callInfo, ...)
  1586. {
  1587. ARGUMENTS(args, callInfo);
  1588. Assert(args.Info.Count > 0);
  1589. return args[0];
  1590. }
  1591. //static
  1592. JavascriptPromise* JavascriptPromise::CreateEnginePromise(ScriptContext *scriptContext)
  1593. {
  1594. JavascriptPromiseResolveOrRejectFunction *resolve = nullptr;
  1595. JavascriptPromiseResolveOrRejectFunction *reject = nullptr;
  1596. JavascriptPromise *promise = scriptContext->GetLibrary()->CreatePromise();
  1597. JavascriptPromise::InitializePromise(promise, &resolve, &reject, scriptContext);
  1598. return promise;
  1599. }
  1600. } // namespace Js