/**************************************************************************** ** ** 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 using namespace QV4; DEFINE_OBJECT_VTABLE(SymbolCtor); DEFINE_MANAGED_VTABLE(Symbol); void Heap::Symbol::init(const QString &s) { Q_ASSERT(s.at(0) == QLatin1Char('@')); identifier = Identifier::fromHeapObject(this); QString desc(s); text = desc.data_ptr(); text->ref.ref(); } void Heap::SymbolCtor::init(QV4::ExecutionContext *scope) { Heap::FunctionObject::init(scope, QStringLiteral("Symbol")); } ReturnedValue QV4::SymbolCtor::call(const QV4::FunctionObject *f, const QV4::Value *, const QV4::Value *argv, int argc) { Scope scope(f); ScopedString s(scope); if (argc) s = argv[0].toString(scope.engine); if (scope.hasException()) return Encode::undefined(); QString desc = QLatin1Char('@') + s->toQString(); return Symbol::create(scope.engine, desc)->asReturnedValue(); } ReturnedValue SymbolCtor::method_for(const FunctionObject *f, const Value *, const Value *argv, int argc) { Scope scope(f); ScopedValue k(scope, argc ? argv[0]: Primitive::undefinedValue()); ScopedString key(scope, k->toString(scope.engine)); if (scope.hasException()) return Encode::undefined(); QString desc = QLatin1Char('@') + key->toQString(); return scope.engine->identifierTable->insertSymbol(desc)->asReturnedValue(); } ReturnedValue SymbolCtor::method_keyFor(const FunctionObject *f, const Value *, const Value *argv, int argc) { ExecutionEngine *e = f->engine(); if (!argc || !argv[0].isSymbol()) return e->throwTypeError(QLatin1String("Symbol.keyFor: Argument is not a symbol.")); const Symbol &arg = static_cast(argv[0]); Heap::Symbol *s = e->identifierTable->symbolForId(arg.identifier()); Q_ASSERT(!s || s == arg.d()); if (s) return e->newString(arg.toQString().mid((1)))->asReturnedValue(); return Encode::undefined(); } void SymbolPrototype::init(ExecutionEngine *engine, Object *ctor) { Scope scope(engine); ScopedValue v(scope); ctor->defineReadonlyProperty(engine->id_prototype(), (v = this)); ctor->defineDefaultProperty(QStringLiteral("for"), SymbolCtor::method_for, 1); ctor->defineDefaultProperty(QStringLiteral("keyFor"), SymbolCtor::method_keyFor, 1); defineDefaultProperty(QStringLiteral("toString"), method_toString); defineDefaultProperty(QStringLiteral("valueOf"), method_valueOf); } ReturnedValue SymbolPrototype::method_toString(const FunctionObject *f, const Value *thisObject, const Value *, int) { ExecutionEngine *e = f->engine(); const Symbol *s = thisObject->as(); if (!s) return e->throwTypeError(); return e->newString(s->descriptiveString())->asReturnedValue(); } ReturnedValue SymbolPrototype::method_valueOf(const FunctionObject *f, const Value *thisObject, const Value *, int) { const Symbol *s = thisObject->as(); if (!s) { ExecutionEngine *e = f->engine(); return e->throwTypeError(); } return s->asReturnedValue(); } Heap::Symbol *Symbol::create(ExecutionEngine *e, const QString &s) { Q_ASSERT(s.at(0) == QLatin1Char('@')); return e->memoryManager->alloc(s); } QString Symbol::descriptiveString() const { return QLatin1String("Symbol(") + toQString().midRef(1) + QLatin1String(")"); }