21ReturnedValue GeneratorFunctionCtor::virtualCallAsConstructor(
const FunctionObject *f,
const Value *argv,
int argc,
const Value *newTarget)
23 ExecutionEngine *engine = f->engine();
25 QQmlRefPointer<ExecutableCompilationUnit> compilationUnit = parse(engine, argv, argc, Type_Generator);
26 if (engine->hasException)
27 return Encode::undefined();
29 Function *vmf = compilationUnit->rootFunction();
30 ExecutionContext *global = engine->scriptContext();
31 ReturnedValue o = Encode(GeneratorFunction::create(global, vmf));
36 ScopedObject obj(scope, o);
37 obj->setProtoFromNewTarget(newTarget);
38 return obj->asReturnedValue();
47Heap::FunctionObject *GeneratorFunction::create(ExecutionContext *context, Function *function)
50 Scoped<GeneratorFunction> g(scope, context->engine()->memoryManager->allocate<GeneratorFunction>(context, function));
51 ScopedObject proto(scope, scope.engine->newObject());
52 proto->setPrototypeOf(scope.engine->generatorPrototype());
53 g->defineDefaultProperty(scope.engine->id_prototype(), proto, Attr_NotConfigurable|Attr_NotEnumerable);
54 g->setPrototypeOf(ScopedObject(scope, scope.engine->generatorFunctionCtor()->get(scope.engine->id_prototype())));
58ReturnedValue GeneratorFunction::virtualCall(
const FunctionObject *f,
const Value *thisObject,
const Value *argv,
int argc)
60 const GeneratorFunction *gf =
static_cast<
const GeneratorFunction *>(f);
61 Function *function = gf->function();
62 ExecutionEngine *engine = gf->engine();
65 Scoped<GeneratorObject> g(scope, engine->memoryManager->allocManaged<GeneratorObject>(engine->classes[EngineBase::Class_GeneratorObject]));
66 g->setPrototypeOf(ScopedObject(scope, gf->get(scope.engine->id_prototype())));
69 Heap::GeneratorObject *gp = g->d();
70 gp->values.set(engine, engine->newArrayObject(argc));
71 gp->jsFrame.set(engine, engine->newArrayObject(
72 JSTypesStackFrame::requiredJSStackFrameSize(function)));
75 for (
int i = 0; i < argc; i++)
76 gp->values->arrayData->setArrayData(engine, i, argv[i]);
78 gp->cppFrame.init(function, gp->values->arrayData->values.values, argc);
79 gp->cppFrame.setupJSFrame(gp->jsFrame->arrayData->values.values, *gf, gf->scope(),
80 thisObject ? *thisObject : Value::undefinedValue(),
81 Value::undefinedValue());
83 gp->cppFrame.push(engine);
86 Moth::VME::interpret(&gp->cppFrame, engine, function->codeData);
88 gp->state = GeneratorState::SuspendedStart;
90 gp->cppFrame.pop(engine);
91 return g->asReturnedValue();
101void GeneratorPrototype::init(ExecutionEngine *engine, Object *ctor)
104 ScopedValue v(scope);
106 Scoped<InternalClass> ic(scope, engine->newInternalClass(
107 Object::staticVTable(), engine->functionPrototype()));
108 ScopedObject ctorProto(scope, engine->newObject(ic->d()));
110 ctor->defineReadonlyConfigurableProperty(engine->id_length(), Value::fromInt32(1));
111 ctor->defineReadonlyProperty(engine->id_prototype(), ctorProto);
113 ctorProto->defineDefaultProperty(QStringLiteral(
"constructor"), (v = ctor), Attr_ReadOnly_ButConfigurable);
114 ctorProto->defineDefaultProperty(engine->symbol_toStringTag(), (v = engine->newIdentifier(QStringLiteral(
"GeneratorFunction"))), Attr_ReadOnly_ButConfigurable);
115 ctorProto->defineDefaultProperty(engine->id_prototype(), (v =
this), Attr_ReadOnly_ButConfigurable);
117 setPrototypeOf(engine->iteratorPrototype());
118 defineDefaultProperty(QStringLiteral(
"constructor"), ctorProto, Attr_ReadOnly_ButConfigurable);
119 defineDefaultProperty(QStringLiteral(
"next"), method_next, 1);
120 defineDefaultProperty(QStringLiteral(
"return"), method_return, 1);
121 defineDefaultProperty(QStringLiteral(
"throw"), method_throw, 1);
122 defineDefaultProperty(engine->symbol_toStringTag(), (v = engine->newString(QStringLiteral(
"Generator"))), Attr_ReadOnly_ButConfigurable);
125ReturnedValue GeneratorPrototype::method_next(
const FunctionObject *f,
const Value *thisObject,
const Value *argv,
int argc)
127 ExecutionEngine *engine = f->engine();
128 const GeneratorObject *g = thisObject->as<GeneratorObject>();
129 if (!g || g->d()->state == GeneratorState::Executing)
130 return engine->throwTypeError();
131 Heap::GeneratorObject *gp = g->d();
133 if (gp->state == GeneratorState::Completed)
134 return IteratorPrototype::createIterResultObject(engine, Value::undefinedValue(),
true);
136 return g->resume(engine, argc ? argv[0] : Value::undefinedValue(), {});
139ReturnedValue GeneratorPrototype::method_return(
const FunctionObject *f,
const Value *thisObject,
const Value *argv,
int argc)
141 ExecutionEngine *engine = f->engine();
142 const GeneratorObject *g = thisObject->as<GeneratorObject>();
143 if (!g || g->d()->state == GeneratorState::Executing)
144 return engine->throwTypeError();
146 Heap::GeneratorObject *gp = g->d();
148 if (gp->state == GeneratorState::SuspendedStart)
149 gp->state = GeneratorState::Completed;
151 if (gp->state == GeneratorState::Completed)
152 return IteratorPrototype::createIterResultObject(engine, argc ? argv[0] : Value::undefinedValue(),
true);
154 return g->resume(engine, argc ? argv[0] : Value::undefinedValue(), Value::emptyValue());
157ReturnedValue GeneratorPrototype::method_throw(
const FunctionObject *f,
const Value *thisObject,
const Value *argv,
int argc)
159 ExecutionEngine *engine = f->engine();
160 const GeneratorObject *g = thisObject->as<GeneratorObject>();
161 if (!g || g->d()->state == GeneratorState::Executing)
162 return engine->throwTypeError();
164 Heap::GeneratorObject *gp = g->d();
167 if (gp->state == GeneratorState::SuspendedStart || gp->state == GeneratorState::Completed) {
168 gp->state = GeneratorState::Completed;
169 engine->throwError(argc ? argv[0] : Value::undefinedValue());
170 return Encode::undefined();
173 return g->resume(engine, Value::undefinedValue(), argc ? argv[0] : Value::undefinedValue());
176ReturnedValue GeneratorObject::resume(ExecutionEngine *engine,
const Value &arg, std::optional<Value> exception)
const
178 Heap::GeneratorObject *gp = d();
179 gp->state = GeneratorState::Executing;
180 gp->cppFrame.setParentFrame(engine->currentStackFrame);
181 engine->currentStackFrame = &gp->cppFrame;
183 Q_ASSERT(gp->cppFrame.yield() !=
nullptr);
184 const char *code = gp->cppFrame.yield();
185 gp->cppFrame.setYield(
nullptr);
186 gp->cppFrame.jsFrame->accumulator = arg;
187 gp->cppFrame.setYieldIsIterator(
false);
202 engine->throwError(*exception);
203 ScopedValue result(scope, Moth::VME::interpret(&gp->cppFrame, engine, code));
205 engine->currentStackFrame = gp->cppFrame.parentFrame();
207 bool done = (gp->cppFrame.yield() ==
nullptr);
208 gp->state = done ? GeneratorState::Completed : GeneratorState::SuspendedYield;
209 if (engine->hasException)
210 return Encode::undefined();
211 if (gp->cppFrame.yieldIsIterator())
212 return result->asReturnedValue();
213 return IteratorPrototype::createIterResultObject(engine, result, done);
218Heap::FunctionObject *MemberGeneratorFunction::create(ExecutionContext *context, Function *function, Object *homeObject, String *name)
220 Scope scope(context);
221 Scoped<MemberGeneratorFunction> g(scope, context->engine()->memoryManager->allocate<MemberGeneratorFunction>(context, function, name));
222 g->d()->homeObject.set(scope.engine, homeObject->d());
223 ScopedObject proto(scope, scope.engine->newObject());
224 proto->setPrototypeOf(scope.engine->generatorPrototype());
225 g->defineDefaultProperty(scope.engine->id_prototype(), proto, Attr_NotConfigurable|Attr_NotEnumerable);
226 g->setPrototypeOf(ScopedObject(scope, scope.engine->generatorFunctionCtor()->get(scope.engine->id_prototype())));