22ReturnedValue GeneratorFunctionCtor::virtualCallAsConstructor(
const FunctionObject *f,
const Value *argv,
int argc,
const Value *newTarget)
24 ExecutionEngine *engine = f->engine();
26 QQmlRefPointer<ExecutableCompilationUnit> compilationUnit = parse(engine, argv, argc, Type_Generator);
27 if (engine->hasException)
28 return Encode::undefined();
30 Function *vmf = compilationUnit->rootFunction();
31 ExecutionContext *global = engine->scriptContext();
32 ReturnedValue o = Encode(GeneratorFunction::create(global, vmf));
37 ScopedObject obj(scope, o);
38 obj->setProtoFromNewTarget(newTarget);
39 return obj->asReturnedValue();
48Heap::FunctionObject *GeneratorFunction::create(ExecutionContext *context, Function *function)
51 Scoped<GeneratorFunction> g(scope, context->engine()->memoryManager->allocate<GeneratorFunction>(context, function));
52 ScopedObject proto(scope, scope.engine->newObject());
53 proto->setPrototypeOf(scope.engine->generatorPrototype());
54 g->defineDefaultProperty(scope.engine->id_prototype(), proto, Attr_NotConfigurable|Attr_NotEnumerable);
55 g->setPrototypeOf(ScopedObject(scope, scope.engine->generatorFunctionCtor()->get(scope.engine->id_prototype())));
59ReturnedValue GeneratorFunction::virtualCall(
const FunctionObject *f,
const Value *thisObject,
const Value *argv,
int argc)
61 const GeneratorFunction *gf =
static_cast<
const GeneratorFunction *>(f);
62 Function *function = gf->function();
63 ExecutionEngine *engine = gf->engine();
66 Scoped<GeneratorObject> g(scope, engine->memoryManager->allocManaged<GeneratorObject>(engine->classes[EngineBase::Class_GeneratorObject]));
67 g->setPrototypeOf(ScopedObject(scope, gf->get(scope.engine->id_prototype())));
70 Heap::GeneratorObject *gp = g->d();
71 gp->values.set(engine, engine->newArrayObject(argc));
72 gp->jsFrame.set(engine, engine->newArrayObject(
73 JSTypesStackFrame::requiredJSStackFrameSize(function)));
76 for (
int i = 0; i < argc; i++)
77 gp->values->arrayData->setArrayData(engine, i, argv[i]);
79 gp->cppFrame.init(function, gp->values->arrayData->values.values, argc);
80 gp->cppFrame.setupJSFrame(gp->jsFrame->arrayData->values.values, *gf, gf->scope(),
81 thisObject ? *thisObject : Value::undefinedValue(),
82 Value::undefinedValue());
84 gp->cppFrame.push(engine);
87 Moth::VME::interpret(&gp->cppFrame, engine, function->codeData);
89 gp->state = GeneratorState::SuspendedStart;
91 gp->cppFrame.pop(engine);
92 return g->asReturnedValue();
102void GeneratorPrototype::init(ExecutionEngine *engine, Object *ctor)
105 ScopedValue v(scope);
107 Scoped<InternalClass> ic(scope, engine->newInternalClass(
108 Object::staticVTable(), engine->functionPrototype()));
109 ScopedObject ctorProto(scope, engine->newObject(ic->d()));
111 ctor->defineReadonlyConfigurableProperty(engine->id_length(), Value::fromInt32(1));
112 ctor->defineReadonlyProperty(engine->id_prototype(), ctorProto);
114 ctorProto->defineDefaultProperty(QStringLiteral(
"constructor"), (v = ctor), Attr_ReadOnly_ButConfigurable);
115 ctorProto->defineDefaultProperty(engine->symbol_toStringTag(), (v = engine->newIdentifier(QStringLiteral(
"GeneratorFunction"))), Attr_ReadOnly_ButConfigurable);
116 ctorProto->defineDefaultProperty(engine->id_prototype(), (v =
this), Attr_ReadOnly_ButConfigurable);
118 setPrototypeOf(engine->iteratorPrototype());
119 defineDefaultProperty(QStringLiteral(
"constructor"), ctorProto, Attr_ReadOnly_ButConfigurable);
120 defineDefaultProperty(QStringLiteral(
"next"), method_next, 1);
121 defineDefaultProperty(QStringLiteral(
"return"), method_return, 1);
122 defineDefaultProperty(QStringLiteral(
"throw"), method_throw, 1);
123 defineDefaultProperty(engine->symbol_toStringTag(), (v = engine->newString(QStringLiteral(
"Generator"))), Attr_ReadOnly_ButConfigurable);
126ReturnedValue GeneratorPrototype::method_next(
const FunctionObject *f,
const Value *thisObject,
const Value *argv,
int argc)
128 ExecutionEngine *engine = f->engine();
129 const GeneratorObject *g = thisObject->as<GeneratorObject>();
130 if (!g || g->d()->state == GeneratorState::Executing)
131 return engine->throwTypeError();
132 Heap::GeneratorObject *gp = g->d();
134 if (gp->state == GeneratorState::Completed)
135 return IteratorPrototype::createIterResultObject(engine, Value::undefinedValue(),
true);
137 return g->resume(engine, argc ? argv[0] : Value::undefinedValue(), {});
140ReturnedValue GeneratorPrototype::method_return(
const FunctionObject *f,
const Value *thisObject,
const Value *argv,
int argc)
142 ExecutionEngine *engine = f->engine();
143 const GeneratorObject *g = thisObject->as<GeneratorObject>();
144 if (!g || g->d()->state == GeneratorState::Executing)
145 return engine->throwTypeError();
147 Heap::GeneratorObject *gp = g->d();
149 if (gp->state == GeneratorState::SuspendedStart)
150 gp->state = GeneratorState::Completed;
152 if (gp->state == GeneratorState::Completed)
153 return IteratorPrototype::createIterResultObject(engine, argc ? argv[0] : Value::undefinedValue(),
true);
155 return g->resume(engine, argc ? argv[0] : Value::undefinedValue(), Value::emptyValue());
158ReturnedValue GeneratorPrototype::method_throw(
const FunctionObject *f,
const Value *thisObject,
const Value *argv,
int argc)
160 ExecutionEngine *engine = f->engine();
161 const GeneratorObject *g = thisObject->as<GeneratorObject>();
162 if (!g || g->d()->state == GeneratorState::Executing)
163 return engine->throwTypeError();
165 Heap::GeneratorObject *gp = g->d();
168 if (gp->state == GeneratorState::SuspendedStart || gp->state == GeneratorState::Completed) {
169 gp->state = GeneratorState::Completed;
170 engine->throwError(argc ? argv[0] : Value::undefinedValue());
171 return Encode::undefined();
174 return g->resume(engine, Value::undefinedValue(), argc ? argv[0] : Value::undefinedValue());
177ReturnedValue GeneratorObject::resume(ExecutionEngine *engine,
const Value &arg, std::optional<Value> exception)
const
179 Heap::GeneratorObject *gp = d();
180 gp->state = GeneratorState::Executing;
181 gp->cppFrame.setParentFrame(engine->currentStackFrame);
182 engine->currentStackFrame = &gp->cppFrame;
184 Q_ASSERT(gp->cppFrame.yield() !=
nullptr);
185 const char *code = gp->cppFrame.yield();
186 gp->cppFrame.setYield(
nullptr);
187 gp->cppFrame.jsFrame->accumulator = arg;
188 gp->cppFrame.setYieldIsIterator(
false);
203 engine->throwError(*exception);
204 ScopedValue result(scope, Moth::VME::interpret(&gp->cppFrame, engine, code));
206 engine->currentStackFrame = gp->cppFrame.parentFrame();
208 bool done = (gp->cppFrame.yield() ==
nullptr);
209 gp->state = done ? GeneratorState::Completed : GeneratorState::SuspendedYield;
210 if (engine->hasException)
211 return Encode::undefined();
212 if (gp->cppFrame.yieldIsIterator())
213 return result->asReturnedValue();
214 return IteratorPrototype::createIterResultObject(engine, result, done);
219Heap::FunctionObject *MemberGeneratorFunction::create(ExecutionContext *context, Function *function, Object *homeObject, String *name)
221 Scope scope(context);
222 Scoped<MemberGeneratorFunction> g(scope, context->engine()->memoryManager->allocate<MemberGeneratorFunction>(context, function, name));
223 g->d()->homeObject.set(scope.engine, homeObject->d());
224 ScopedObject proto(scope, scope.engine->newObject());
225 proto->setPrototypeOf(scope.engine->generatorPrototype());
226 g->defineDefaultProperty(scope.engine->id_prototype(), proto, Attr_NotConfigurable|Attr_NotEnumerable);
227 g->setPrototypeOf(ScopedObject(scope, scope.engine->generatorFunctionCtor()->get(scope.engine->id_prototype())));