/**************************************************************************** ** ** Copyright (C) 2018 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the QtQml module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 3 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL3 included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 3 requirements ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 2.0 or (at your option) the GNU General ** Public license version 3 or any later version approved by the KDE Free ** Qt Foundation. The licenses are as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-2.0.html and ** https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include #include #include #include #include using namespace QV4; DEFINE_OBJECT_VTABLE(GeneratorFunctionCtor); DEFINE_OBJECT_VTABLE(GeneratorFunction); DEFINE_OBJECT_VTABLE(GeneratorObject); void Heap::GeneratorFunctionCtor::init(QV4::ExecutionContext *scope) { Heap::FunctionObject::init(scope, QStringLiteral("GeneratorFunction")); } ReturnedValue GeneratorFunctionCtor::callAsConstructor(const FunctionObject *f, const Value *argv, int argc) { ExecutionEngine *engine = f->engine(); QQmlRefPointer compilationUnit = parse(engine, argv, argc, Type_Generator); if (engine->hasException) return Encode::undefined(); Function *vmf = compilationUnit->linkToEngine(engine); ExecutionContext *global = engine->scriptContext(); return Encode(GeneratorFunction::create(global, vmf)); } // 15.3.1: This is equivalent to new Function(...) ReturnedValue GeneratorFunctionCtor::call(const FunctionObject *f, const Value *, const Value *argv, int argc) { return callAsConstructor(f, argv, argc); } Heap::FunctionObject *GeneratorFunction::create(ExecutionContext *context, Function *function) { Scope scope(context); Scoped g(scope, context->engine()->memoryManager->allocate(context, function)); ScopedObject proto(scope, scope.engine->newObject()); proto->setPrototype(scope.engine->generatorPrototype()); g->defineDefaultProperty(scope.engine->id_prototype(), proto, Attr_NotConfigurable|Attr_NotEnumerable); g->setPrototype(ScopedObject(scope, scope.engine->generatorFunctionCtor()->get(scope.engine->id_prototype()))); return g->d(); } ReturnedValue GeneratorFunction::callAsConstructor(const FunctionObject *f, const Value *, int) { return f->engine()->throwTypeError(); } ReturnedValue GeneratorFunction::call(const FunctionObject *f, const Value *thisObject, const Value *argv, int argc) { const GeneratorFunction *gf = static_cast(f); Function *function = gf->function(); ExecutionEngine *engine = gf->engine(); // We need to set up a separate stack for the generator, as it's being re-entered uint stackSize = argc; // space for the original arguments int jsStackFrameSize = offsetof(CallData, args)/sizeof(Value) + function->compiledFunction->nRegisters; stackSize += jsStackFrameSize; size_t requiredMemory = sizeof(GeneratorObject::Data) - sizeof(Value) + sizeof(Value) * stackSize; Scope scope(gf); Scoped g(scope, scope.engine->memoryManager->allocManaged(requiredMemory, scope.engine->classes[EngineBase::Class_GeneratorObject])); g->setPrototype(ScopedObject(scope, gf->get(scope.engine->id_prototype()))); Heap::GeneratorObject *gp = g->d(); gp->stack.size = stackSize; gp->stack.alloc = stackSize; // copy original arguments memcpy(gp->stack.values, argv, argc*sizeof(Value)); gp->cppFrame.originalArguments = gp->stack.values; gp->cppFrame.originalArgumentsCount = argc; // setup JS stack frame CallData *callData = reinterpret_cast(&gp->stack.values[argc]); callData->function = *gf; callData->context = gf->scope(); callData->accumulator = Encode::undefined(); callData->thisObject = thisObject ? *thisObject : Primitive::undefinedValue(); if (argc > int(function->nFormals)) argc = int(function->nFormals); callData->setArgc(argc); memcpy(callData->args, argv, argc*sizeof(Value)); gp->cppFrame.v4Function = function; gp->cppFrame.instructionPointer = 0; gp->cppFrame.jsFrame = callData; gp->cppFrame.parent = engine->currentStackFrame; engine->currentStackFrame = &gp->cppFrame; Moth::VME::interpret(gp->cppFrame, function->codeData); gp->state = GeneratorState::SuspendedStart; engine->currentStackFrame = gp->cppFrame.parent; return g->asReturnedValue(); } void Heap::GeneratorPrototype::init() { Heap::FunctionObject::init(); } void GeneratorPrototype::init(ExecutionEngine *engine, Object *ctor) { Scope scope(engine); ScopedValue v(scope); ScopedObject ctorProto(scope, engine->newObject(engine->newInternalClass(Object::staticVTable(), engine->functionPrototype()))); ctor->defineReadonlyConfigurableProperty(engine->id_length(), Primitive::fromInt32(1)); ctor->defineReadonlyProperty(engine->id_prototype(), ctorProto); ctorProto->defineDefaultProperty(QStringLiteral("constructor"), (v = ctor), Attr_ReadOnly_ButConfigurable); ctorProto->defineDefaultProperty(engine->symbol_toStringTag(), (v = engine->newIdentifier(QStringLiteral("GeneratorFunction"))), Attr_ReadOnly_ButConfigurable); ctorProto->defineDefaultProperty(engine->id_prototype(), (v = this), Attr_ReadOnly_ButConfigurable); setPrototype(engine->iteratorPrototype()); defineDefaultProperty(QStringLiteral("constructor"), ctorProto, Attr_ReadOnly_ButConfigurable); defineDefaultProperty(QStringLiteral("next"), method_next, 1); defineDefaultProperty(QStringLiteral("return"), method_return, 1); defineDefaultProperty(QStringLiteral("throw"), method_throw, 1); defineDefaultProperty(engine->symbol_toStringTag(), (v = engine->newString(QStringLiteral("Generator"))), Attr_ReadOnly_ButConfigurable); } ReturnedValue GeneratorPrototype::method_next(const FunctionObject *f, const Value *thisObject, const Value *argv, int argc) { ExecutionEngine *engine = f->engine(); const GeneratorObject *g = thisObject->as(); if (!g || g->d()->state == GeneratorState::Executing) return engine->throwTypeError(); Heap::GeneratorObject *gp = g->d(); if (gp->state == GeneratorState::Completed) return IteratorPrototype::createIterResultObject(engine, Primitive::undefinedValue(), true); return g->resume(engine, argc ? argv[0] : Primitive::undefinedValue()); } ReturnedValue GeneratorPrototype::method_return(const FunctionObject *f, const Value *thisObject, const Value *argv, int argc) { ExecutionEngine *engine = f->engine(); const GeneratorObject *g = thisObject->as(); if (!g || g->d()->state == GeneratorState::Executing) return engine->throwTypeError(); Heap::GeneratorObject *gp = g->d(); if (gp->state == GeneratorState::SuspendedStart) gp->state = GeneratorState::Completed; if (gp->state == GeneratorState::Completed) return IteratorPrototype::createIterResultObject(engine, argc ? argv[0] : Primitive::undefinedValue(), true); // the bytecode interpreter interprets an exception with empty value as // a yield called with return() engine->throwError(Primitive::emptyValue()); return g->resume(engine, argc ? argv[0]: Primitive::undefinedValue()); } ReturnedValue GeneratorPrototype::method_throw(const FunctionObject *f, const Value *thisObject, const Value *argv, int argc) { ExecutionEngine *engine = f->engine(); const GeneratorObject *g = thisObject->as(); if (!g || g->d()->state == GeneratorState::Executing) return engine->throwTypeError(); Heap::GeneratorObject *gp = g->d(); engine->throwError(argc ? argv[0]: Primitive::undefinedValue()); if (gp->state == GeneratorState::SuspendedStart || gp->state == GeneratorState::Completed) { gp->state = GeneratorState::Completed; return Encode::undefined(); } return g->resume(engine, Primitive::undefinedValue()); } ReturnedValue GeneratorObject::resume(ExecutionEngine *engine, const Value &arg) const { Heap::GeneratorObject *gp = d(); gp->state = GeneratorState::Executing; gp->cppFrame.parent = engine->currentStackFrame; engine->currentStackFrame = &gp->cppFrame; Q_ASSERT(gp->cppFrame.yield != nullptr); const char *code = gp->cppFrame.yield; gp->cppFrame.yield = nullptr; gp->cppFrame.jsFrame->accumulator = arg; Scope scope(engine); ScopedValue result(scope, Moth::VME::interpret(gp->cppFrame, code)); engine->currentStackFrame = gp->cppFrame.parent; bool done = (gp->cppFrame.yield == nullptr); gp->state = done ? GeneratorState::Completed : GeneratorState::SuspendedYield; if (engine->hasException) return Encode::undefined(); return IteratorPrototype::createIterResultObject(engine, result, done); }