• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2021-2022 Huawei Device Co., Ltd.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  * http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 
16 #include "functionBuilder.h"
17 
18 #include <binder/binder.h>
19 #include <compiler/base/iterators.h>
20 #include <compiler/core/pandagen.h>
21 #include <ir/base/classDefinition.h>
22 #include <ir/base/scriptFunction.h>
23 #include <ir/statement.h>
24 #include <util/helpers.h>
25 
26 namespace panda::es2panda::compiler {
27 
FunctionBuilder(PandaGen * pg,CatchTable * catchTable)28 FunctionBuilder::FunctionBuilder(PandaGen *pg, CatchTable *catchTable)
29     : pg_(pg), catchTable_(catchTable), funcObj_(pg_->AllocReg())
30 {
31 }
32 
GeneratorKind() const33 IteratorType FunctionBuilder::GeneratorKind() const
34 {
35     return IteratorType::SYNC;
36 }
37 
DirectReturn(const ir::AstNode * node) const38 void FunctionBuilder::DirectReturn(const ir::AstNode *node) const
39 {
40     pg_->NotifyConcurrentResult(node);
41     pg_->EmitReturn(node);
42 }
43 
ImplicitReturn(const ir::AstNode * node) const44 void FunctionBuilder::ImplicitReturn(const ir::AstNode *node) const
45 {
46     const auto *rootNode = pg_->RootNode();
47 
48     if (!rootNode->IsScriptFunction() || !rootNode->AsScriptFunction()->IsConstructor()) {
49         if (pg_->isDebuggerEvaluateExpressionMode()) {
50             pg_->NotifyConcurrentResult(node);
51             pg_->EmitReturn(node);
52             return;
53         }
54         pg_->LoadConst(node, Constant::JS_UNDEFINED);
55         pg_->NotifyConcurrentResult(node);
56         pg_->EmitReturnUndefined(node);
57         return;
58     }
59 
60     pg_->GetThis(rootNode);
61     if (rootNode->AsScriptFunction()->IsConstructor() &&
62         util::Helpers::GetClassDefiniton(rootNode->AsScriptFunction())->Super()) {
63             pg_->ThrowIfSuperNotCorrectCall(rootNode, 0);
64     }
65 
66     pg_->NotifyConcurrentResult(node);
67     pg_->EmitReturn(node);
68 }
69 
ExplicitReturn(const ir::AstNode * node) const70 void FunctionBuilder::ExplicitReturn(const ir::AstNode *node) const
71 {
72     DirectReturn(node);
73 }
74 
AsyncYield(const ir::AstNode * node,VReg value,VReg completionType,VReg completionValue) const75 void FunctionBuilder::AsyncYield(const ir::AstNode *node, VReg value, VReg completionType, VReg completionValue) const
76 {
77     ASSERT(BuilderKind() == BuilderType::ASYNC_GENERATOR);
78     RegScope rs(pg_);
79     VReg done = pg_->AllocReg();
80     // 27.6.3.8.6 Set generator.[[AsyncGeneratorState]] to suspendedYield.
81     pg_->GeneratorYield(node, funcObj_);
82     /** 27.6.3.8.7 Remove genContext from the execution context stack and restore the execution context that
83      *  is at the top of the execution context stack as the running execution context.
84      *  27.6.3.8.9 Return ! AsyncGeneratorResolve(generator, value, false).
85      */
86     pg_->StoreConst(node, done, Constant::JS_FALSE);
87     pg_->AsyncGeneratorResolve(node, funcObj_, value, done);
88 
89     resumeGenerator(node, completionType, completionValue);
90 }
91 
SuspendResumeExecution(const ir::AstNode * node,VReg completionType,VReg completionValue) const92 void FunctionBuilder::SuspendResumeExecution(const ir::AstNode *node, VReg completionType, VReg completionValue) const
93 {
94     ASSERT(BuilderKind() == BuilderType::ASYNC || BuilderKind() == BuilderType::ASYNC_GENERATOR ||
95            BuilderKind() == BuilderType::GENERATOR);
96 
97     pg_->SuspendGenerator(node, funcObj_); // iterResult is in acc
98     resumeGenerator(node, completionType, completionValue);
99 }
100 
resumeGenerator(const ir::AstNode * node,VReg completionType,VReg completionValue) const101 void FunctionBuilder::resumeGenerator(const ir::AstNode *node, VReg completionType, VReg completionValue) const
102 {
103     ASSERT(BuilderKind() == BuilderType::ASYNC || BuilderKind() == BuilderType::ASYNC_GENERATOR ||
104            BuilderKind() == BuilderType::GENERATOR);
105 
106     pg_->ResumeGenerator(node, funcObj_);
107     pg_->StoreAccumulator(node, completionValue);
108     pg_->GetResumeMode(node, funcObj_);
109     pg_->StoreAccumulator(node, completionType);
110 }
111 
FunctionReg(const ir::ScriptFunction * node) const112 VReg FunctionBuilder::FunctionReg(const ir::ScriptFunction *node) const
113 {
114     binder::FunctionScope *scope = node->Scope();
115     auto res = scope->Find(binder::Binder::MANDATORY_PARAM_FUNC);
116     ASSERT(res.level == 0 && res.variable->IsLocalVariable());
117     return res.variable->AsLocalVariable()->Vreg();
118 }
119 
Await(const ir::AstNode * node)120 void FunctionBuilder::Await(const ir::AstNode *node)
121 {
122     if (BuilderKind() == BuilderType::NORMAL) {
123         // TODO(frobert): Implement top-level await
124         PandaGen::Unimplemented();
125     }
126 
127     ASSERT(BuilderKind() == BuilderType::ASYNC || BuilderKind() == BuilderType::ASYNC_GENERATOR);
128 
129     RegScope rs(pg_);
130     VReg completionType = pg_->AllocReg();
131     VReg completionValue = pg_->AllocReg();
132 
133     pg_->AsyncFunctionAwait(node, funcObj_);
134     SuspendResumeExecution(node, completionType, completionValue);
135 
136     HandleCompletion(node, completionType, completionValue);
137 }
138 
HandleCompletion(const ir::AstNode * node,VReg completionType,VReg completionValue)139 void FunctionBuilder::HandleCompletion(const ir::AstNode *node, VReg completionType, VReg completionValue)
140 {
141     if (BuilderKind() == BuilderType::GENERATOR) {
142         // .return(value)
143         pg_->LoadAccumulatorInt(node, static_cast<int32_t>(ResumeMode::RETURN));
144 
145         auto *notRetLabel = pg_->AllocLabel();
146         pg_->Condition(node, lexer::TokenType::PUNCTUATOR_EQUAL, completionType, notRetLabel);
147         if (!handleReturn_) {
148             handleReturn_ = true;
149             pg_->ControlFlowChangeReturn();
150             handleReturn_ = false;
151         }
152 
153         pg_->LoadAccumulator(node, completionValue);
154         pg_->DirectReturn(node);
155 
156         // .throw(value)
157         pg_->SetLabel(node, notRetLabel);
158     }
159 
160     pg_->LoadAccumulatorInt(node, static_cast<int32_t>(ResumeMode::THROW));
161 
162     auto *not_throw_label = pg_->AllocLabel();
163     pg_->Condition(node, lexer::TokenType::PUNCTUATOR_EQUAL, completionType, not_throw_label);
164     pg_->LoadAccumulator(node, completionValue);
165     pg_->EmitThrow(node);
166 
167     // .next(value)
168     pg_->SetLabel(node, not_throw_label);
169     pg_->LoadAccumulator(node, completionValue);
170 }
171 
YieldStar(const ir::AstNode * node)172 void FunctionBuilder::YieldStar(const ir::AstNode *node)
173 {
174     ASSERT(BuilderKind() == BuilderType::GENERATOR || BuilderKind() == BuilderType::ASYNC_GENERATOR);
175 
176     RegScope rs(pg_);
177 
178     auto *loopStart = pg_->AllocLabel();
179     auto *returnCompletion = pg_->AllocLabel();
180     auto *throwCompletion = pg_->AllocLabel();
181     auto *callMethod = pg_->AllocLabel();
182     auto *normalOrThrowCompletion = pg_->AllocLabel();
183     auto *iteratorComplete = pg_->AllocLabel();
184 
185     // 4. Let iteratorRecord be ? GetIterator(value, generatorKind).
186     Iterator iterator(pg_, node, GeneratorKind());
187 
188     // 6. Let received be NormalCompletion(undefined).
189     VReg receivedValue = iterator.NextResult();
190     VReg receivedType = pg_->AllocReg();
191     VReg nextMethod = pg_->AllocReg();
192     VReg exitReturn = pg_->AllocReg();
193     VReg iterValue = pg_->AllocReg();
194 
195     pg_->StoreConst(node, receivedValue, Constant::JS_UNDEFINED);
196     pg_->LoadAccumulatorInt(node, static_cast<int32_t>(ResumeMode::NEXT));
197     pg_->StoreAccumulator(node, receivedType);
198     pg_->MoveVreg(node, nextMethod, iterator.Method());
199 
200     // 7. Repeat
201     pg_->SetLabel(node, loopStart);
202     pg_->StoreConst(node, exitReturn, Constant::JS_FALSE);
203 
204     // a. If received.[[Type]] is normal, then
205     pg_->LoadAccumulatorInt(node, static_cast<int32_t>(ResumeMode::NEXT));
206     pg_->Condition(node, lexer::TokenType::PUNCTUATOR_STRICT_EQUAL, receivedType, throwCompletion);
207     pg_->MoveVreg(node, iterator.Method(), nextMethod);
208     pg_->Branch(node, callMethod);
209 
210     // b. Else if received.[[Type]] is throw, then
211     pg_->SetLabel(node, throwCompletion);
212     pg_->LoadAccumulatorInt(node, static_cast<int32_t>(ResumeMode::THROW));
213     pg_->Condition(node, lexer::TokenType::PUNCTUATOR_STRICT_EQUAL, receivedType, returnCompletion);
214 
215     // i. Let throw be ? GetMethod(iterator, "throw").
216     iterator.GetMethod("throw");
217 
218     // ii. If throw is not undefined, then
219     pg_->BranchIfNotUndefined(node, callMethod);
220 
221     // iii. Else,
222     // 1. NOTE: If iterator does not have a throw method, this throw is going to terminate the yield* loop. But first we
223     // need to give iterator a chance to clean up.
224     // 2. Let closeCompletion be Completion { [[Type]]: normal, [[Value]]: empty, [[Target]]: empty }.
225     // 3. If generatorKind is async, perform ? AsyncIteratorClose(iteratorRecord, closeCompletion).
226     // 4. Else, perform ? IteratorClose(iteratorRecord, closeCompletion).
227     iterator.Close(false);
228     // 5. NOTE: The next step throws a TypeError to indicate that there was a yield* protocol violation: iterator does
229     // not have a throw method.
230     // 6. Throw a TypeError exception.
231     pg_->ThrowThrowNotExist(node);
232 
233     // c. Else,
234     // i. Assert: received.[[Type]] is return.
235     pg_->SetLabel(node, returnCompletion);
236     pg_->StoreConst(node, exitReturn, Constant::JS_TRUE);
237     // ii. Let return be ? GetMethod(iterator, "return").
238     iterator.GetMethod("return");
239 
240     // iii. If return is undefined, then
241     pg_->BranchIfNotUndefined(node, callMethod);
242 
243     // 1. If generatorKind is async, set received.[[Value]] to ? Await(received.[[Value]]).
244     pg_->ControlFlowChangeReturn();
245     pg_->LoadAccumulator(node, receivedValue);
246 
247     if (GeneratorKind() == IteratorType::ASYNC) {
248         Await(node);
249     }
250 
251     // 2. Return Completion(received).
252     pg_->DirectReturn(node);
253 
254     pg_->SetLabel(node, callMethod);
255     // i. Let innerResult be ? Call(iteratorRecord.[[NextMethod]], iteratorRecord.[[Iterator]], « received.[[Value]] »).
256     // 1. Let innerResult be ? Call(throw, iterator, « received.[[Value]] »).
257     // iv. Let innerReturnResult be ? Call(return, iterator, « received.[[Value]] »).
258     iterator.CallMethodWithValue();
259 
260     // ii. ii. If generatorKind is async, set innerResult to ? Await(innerResult).
261     // 2. If generatorKind is async, set innerResult to ? Await(innerResult).
262     // v. If generatorKind is async, set innerReturnResult to ? Await(innerReturnResult).
263     if (GeneratorKind() == IteratorType::ASYNC) {
264         Await(node);
265     }
266 
267     pg_->StoreAccumulator(node, receivedValue);
268 
269     // ii. If Type(innerResult) is not Object, throw a TypeError exception.
270     // 4. If Type(innerResult) is not Object, throw a TypeError exception.
271     // vi. If Type(innerReturnResult) is not Object, throw a TypeError exception.
272     pg_->ThrowIfNotObject(node, receivedValue);
273 
274     // iv. Let done be ? IteratorComplete(innerResult).
275     // v. Let done be ? IteratorComplete(innerResult).
276     // vii. Let done be ? IteratorComplete(innerReturnResult).
277     iterator.Complete();
278     pg_->BranchIfTrue(node, iteratorComplete);
279 
280     pg_->LoadAccumulator(node, receivedValue);
281     // vi. If generatorKind is async, set received to AsyncGeneratorYield(? IteratorValue(innerResult)).
282     // 7. If generatorKind is async, set received to AsyncGeneratorYield(? IteratorValue(innerResult)).
283     // ix. If generatorKind is async, set received to AsyncGeneratorYield(? IteratorValue(innerReturnResult)).
284     if (GeneratorKind() == IteratorType::ASYNC) {
285         iterator.Value();
286         // 27.6.3.8 AsyncGeneratorYield
287         // 5. Set value to ? Await(value).
288         Await(node);
289         // 6. Set generator.[[AsyncGeneratorState]] to suspendedYield.
290         pg_->StoreAccumulator(node, iterValue);
291         AsyncYield(node, iterValue, receivedType, receivedValue);
292 
293         // a. If resumptionValue.[[Type]] is not return
294         pg_->LoadAccumulatorInt(node, static_cast<int32_t>(ResumeMode::RETURN));
295         pg_->Condition(node, lexer::TokenType::PUNCTUATOR_EQUAL, receivedType, loopStart);
296 
297         // b. Let awaited be Await(resumptionValue.[[Value]]).
298         pg_->LoadAccumulator(node, receivedValue);
299         pg_->AsyncFunctionAwait(node, funcObj_);
300         SuspendResumeExecution(node, receivedType, receivedValue);
301 
302         // c. If awaited.[[Type]] is throw, return Completion(awaited).
303         pg_->LoadAccumulatorInt(node, static_cast<int32_t>(ResumeMode::THROW));
304         // d. Assert: awaited.[[Type]] is normal.
305         // e. Return Completion { [[Type]]: return, [[Value]]: awaited.[[Value]], [[Target]]: empty }.
306         pg_->Condition(node, lexer::TokenType::PUNCTUATOR_EQUAL, receivedType, returnCompletion);
307     } else {
308         // vii. Else, set received to GeneratorYield(innerResult).
309         // 8. Else, set received to GeneratorYield(innerResult).
310         // x. Else, set received to GeneratorYield(innerReturnResult).
311         SuspendResumeExecution(node, receivedType, receivedValue);
312     }
313 
314     pg_->Branch(node, loopStart);
315 
316     // v. If done is true, then
317     // 6. If done is true, then
318     // viii. If done is true, then
319     pg_->SetLabel(node, iteratorComplete);
320 
321     pg_->LoadAccumulator(node, exitReturn);
322     pg_->BranchIfFalse(node, normalOrThrowCompletion);
323 
324     // 1. Let value be ? IteratorValue(innerReturnResult).
325     iterator.Value();
326 
327     if (pg_->CheckControlFlowChange()) {
328         pg_->StoreAccumulator(node, receivedValue);
329         pg_->ControlFlowChangeReturn();
330         pg_->LoadAccumulator(node, receivedValue);
331     }
332 
333     // 2. Return Completion { [[Type]]: return, [[Value]]: value, [[Target]]: empty }.
334     pg_->DirectReturn(node);
335 
336     pg_->SetLabel(node, normalOrThrowCompletion);
337     // 1. Return ? IteratorValue(innerResult).
338     // a. Return ? IteratorValue(innerResult).
339     iterator.Value();
340 }
341 
342 }  // namespace panda::es2panda::compiler
343