// Copyright (C) 2016 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only #ifndef QV4QOBJECTWRAPPER_P_H #define QV4QOBJECTWRAPPER_P_H // // W A R N I N G // ------------- // // This file is not part of the Qt API. It exists purely as an // implementation detail. This header file may change from version to // version without notice, or even be removed. // // We mean it. // #include #include #include #include #include #include #include #include #include #include QT_BEGIN_NAMESPACE class QObject; class QQmlData; class QQmlPropertyCache; class QQmlPropertyData; namespace QV4 { struct QObjectSlotDispatcher; namespace Heap { struct QQmlValueTypeWrapper; struct Q_QML_EXPORT QObjectWrapper : Object { void init(QObject *object) { Object::init(); qObj.init(object); } void destroy() { qObj.destroy(); Object::destroy(); } QObject *object() const { return qObj.data(); } static void markObjects(Heap::Base *that, MarkStack *markStack); private: QV4QPointer qObj; }; #define QObjectMethodMembers(class, Member) \ Member(class, Pointer, QQmlValueTypeWrapper *, valueTypeWrapper) DECLARE_HEAP_OBJECT(QObjectMethod, FunctionObject) { DECLARE_MARKOBJECTS(QObjectMethod) QV4QPointer qObj; QQmlPropertyData *methods; alignas(alignof(QQmlPropertyData)) std::byte _singleMethod[sizeof(QQmlPropertyData)]; int methodCount; int index; void init(QV4::ExecutionContext *scope); void destroy() { if (methods != reinterpret_cast(&_singleMethod)) delete[] methods; qObj.destroy(); FunctionObject::destroy(); } void ensureMethodsCache(const QMetaObject *thisMeta); QString name() const; const QMetaObject *metaObject() const; QObject *object() const { return qObj.data(); } void setObject(QObject *o) { qObj = o; } bool isDetached() const; bool isAttachedTo(QObject *o) const; enum ThisObjectMode { Invalid, Included, Explicit, }; QV4::Heap::QObjectMethod::ThisObjectMode checkThisObject(const QMetaObject *thisMeta) const; }; struct QMetaObjectWrapper : FunctionObject { const QMetaObject* metaObject; QQmlPropertyData *constructors; int constructorCount; void init(const QMetaObject* metaObject); void destroy(); void ensureConstructorsCache(); }; struct QmlSignalHandler : Object { void init(QObject *object, int signalIndex); void destroy() { qObj.destroy(); Object::destroy(); } int signalIndex; QObject *object() const { return qObj.data(); } void setObject(QObject *o) { qObj = o; } private: QV4QPointer qObj; }; } struct Q_QML_EXPORT QObjectWrapper : public Object { V4_OBJECT2(QObjectWrapper, Object) V4_NEEDS_DESTROY enum Flag { NoFlag = 0x0, CheckRevision = 0x1, AttachMethods = 0x2, AllowOverride = 0x4, IncludeImports = 0x8, }; Q_DECLARE_FLAGS(Flags, Flag); static void initializeBindings(ExecutionEngine *engine); QObject *object() const { return d()->object(); } ReturnedValue getQmlProperty( const QQmlRefPointer &qmlContext, String *name, Flags flags, bool *hasProperty = nullptr) const; static ReturnedValue getQmlProperty( ExecutionEngine *engine, const QQmlRefPointer &qmlContext, Heap::Object *wrapper, QObject *object, String *name, Flags flags, bool *hasProperty = nullptr, const QQmlPropertyData **property = nullptr); static bool setQmlProperty( ExecutionEngine *engine, const QQmlRefPointer &qmlContext, QObject *object, String *name, Flags flags, const Value &value); static ReturnedValue wrap(ExecutionEngine *engine, QObject *object); static ReturnedValue wrapConst(ExecutionEngine *engine, QObject *object); static void markWrapper(QObject *object, MarkStack *markStack); using Object::get; static void setProperty(ExecutionEngine *engine, QObject *object, int propertyIndex, const Value &value); void setProperty(ExecutionEngine *engine, int propertyIndex, const Value &value); static void setProperty( ExecutionEngine *engine, QObject *object, const QQmlPropertyData *property, const Value &value); void destroyObject(bool lastCall); static ReturnedValue getProperty( ExecutionEngine *engine, Heap::Object *wrapper, QObject *object, const QQmlPropertyData *property, Flags flags); static ReturnedValue virtualResolveLookupGetter(const Object *object, ExecutionEngine *engine, Lookup *lookup); static ReturnedValue lookupAttached(Lookup *l, ExecutionEngine *engine, const Value &object); template static ReturnedValue lookupPropertyGetterImpl( Lookup *l, ExecutionEngine *engine, const Value &object, Flags flags, ReversalFunctor revert); template static ReturnedValue lookupMethodGetterImpl( Lookup *l, ExecutionEngine *engine, const Value &object, Flags flags, ReversalFunctor revert); static bool virtualResolveLookupSetter( Object *object, ExecutionEngine *engine, Lookup *lookup, const Value &value); static OwnPropertyKeyIterator *virtualOwnPropertyKeys(const Object *m, Value *target); static int virtualMetacall(Object *object, QMetaObject::Call call, int index, void **a); static QString objectToString( ExecutionEngine *engine, const QMetaObject *metaObject, QObject *object); protected: static bool virtualIsEqualTo(Managed *that, Managed *o); static ReturnedValue create(ExecutionEngine *engine, QObject *object); static const QQmlPropertyData *findProperty( QObject *o, const QQmlRefPointer &qmlContext, String *name, Flags flags, QQmlPropertyData *local); const QQmlPropertyData *findProperty( const QQmlRefPointer &qmlContext, String *name, Flags flags, QQmlPropertyData *local) const; static ReturnedValue virtualGet( const Managed *m, PropertyKey id, const Value *receiver, bool *hasProperty); static bool virtualPut(Managed *m, PropertyKey id, const Value &value, Value *receiver); static PropertyAttributes virtualGetOwnProperty(const Managed *m, PropertyKey id, Property *p); static ReturnedValue method_connect( const FunctionObject *, const Value *thisObject, const Value *argv, int argc); static ReturnedValue method_disconnect( const FunctionObject *, const Value *thisObject, const Value *argv, int argc); private: Q_NEVER_INLINE static ReturnedValue wrap_slowPath(ExecutionEngine *engine, QObject *object); Q_NEVER_INLINE static ReturnedValue wrapConst_slowPath(ExecutionEngine *engine, QObject *object); static Heap::QObjectMethod *cloneMethod( ExecutionEngine *engine, Heap::QObjectMethod *cloneFrom, Heap::Object *wrapper, QObject *object); }; Q_DECLARE_OPERATORS_FOR_FLAGS(QObjectWrapper::Flags) inline ReturnedValue QObjectWrapper::wrap(ExecutionEngine *engine, QObject *object) { if (Q_UNLIKELY(QQmlData::wasDeleted(object))) return QV4::Encode::null(); auto ddata = QQmlData::get(object); if (Q_LIKELY(ddata && ddata->jsEngineId == engine->m_engineId && !ddata->jsWrapper.isUndefined())) { // We own the JS object return ddata->jsWrapper.value(); } return wrap_slowPath(engine, object); } // Unfortunately we still need a non-const QObject* here because QQmlData needs to register itself in QObjectPrivate. inline ReturnedValue QObjectWrapper::wrapConst(ExecutionEngine *engine, QObject *object) { if (Q_UNLIKELY(QQmlData::wasDeleted(object))) return QV4::Encode::null(); return wrapConst_slowPath(engine, object); } inline bool canConvert(const QQmlPropertyCache *fromMo, const QQmlPropertyCache *toMo) { while (fromMo) { if (fromMo == toMo) return true; fromMo = fromMo->parent().data(); } return false; } template inline ReturnedValue QObjectWrapper::lookupPropertyGetterImpl( Lookup *lookup, ExecutionEngine *engine, const Value &object, QObjectWrapper::Flags flags, ReversalFunctor revertLookup) { // we can safely cast to a QV4::Object here. If object is something else, // the internal class won't match Heap::Object *o = static_cast(object.heapObject()); if (!o || o->internalClass != lookup->qobjectLookup.ic) return revertLookup(); Heap::QObjectWrapper *This = static_cast(o); QObject *qobj = This->object(); if (QQmlData::wasDeleted(qobj)) return QV4::Encode::undefined(); QQmlData *ddata = QQmlData::get(qobj, /*create*/false); if (!ddata) return revertLookup(); const QQmlPropertyData *property = lookup->qobjectLookup.propertyData; if (ddata->propertyCache.data() != lookup->qobjectLookup.propertyCache) { // If the property is overridden and the lookup allows overrides to be considered, // we have to revert here and redo the lookup from scratch. if (property->isOverridden() && ((flags & AllowOverride) || property->isFunction() || property->isSignalHandler())) { return revertLookup(); } if (!canConvert(ddata->propertyCache.data(), lookup->qobjectLookup.propertyCache)) return revertLookup(); } return getProperty(engine, This, qobj, property, flags); } template inline ReturnedValue QObjectWrapper::lookupMethodGetterImpl( Lookup *lookup, ExecutionEngine *engine, const Value &object, QObjectWrapper::Flags flags, ReversalFunctor revertLookup) { // we can safely cast to a QV4::Object here. If object is something else, // the internal class won't match Heap::Object *o = static_cast(object.heapObject()); if (!o || o->internalClass != lookup->qobjectMethodLookup.ic) return revertLookup(); Heap::QObjectWrapper *This = static_cast(o); QObject *qobj = This->object(); if (QQmlData::wasDeleted(qobj)) return QV4::Encode::undefined(); QQmlData *ddata = QQmlData::get(qobj, /*create*/false); if (!ddata) return revertLookup(); const QQmlPropertyData *property = lookup->qobjectMethodLookup.propertyData; if (ddata->propertyCache.data() != lookup->qobjectMethodLookup.propertyCache) { if (property && property->isOverridden()) return revertLookup(); if (!canConvert(ddata->propertyCache.data(), lookup->qobjectMethodLookup.propertyCache)) return revertLookup(); } if (Heap::QObjectMethod *method = lookup->qobjectMethodLookup.method) { if (lookup->forCall && !method->isDetached()) { method = lookup->qobjectMethodLookup.method = cloneMethod(engine, method, nullptr, nullptr); } else if (!lookup->forCall && !method->isAttachedTo(qobj)) { method = lookup->qobjectMethodLookup.method = cloneMethod(engine, method, This, qobj); } return method ? method->asReturnedValue() : revertLookup(); } if (!property) // was toString() or destroy() return revertLookup(); QV4::Scope scope(engine); QV4::ScopedValue v(scope, getProperty(engine, This, qobj, property, flags)); if (!v->as()) return revertLookup(); lookup->qobjectMethodLookup.method = static_cast(v->heapObject()); return v->asReturnedValue(); } struct QQmlValueTypeWrapper; struct Q_QML_EXPORT QObjectMethod : public QV4::FunctionObject { V4_OBJECT2(QObjectMethod, QV4::FunctionObject) V4_NEEDS_DESTROY enum { DestroyMethod = -1, ToStringMethod = -2 }; static ReturnedValue create( QV4::ExecutionContext *scope, QObject *object, int index); static ReturnedValue create( QV4::ExecutionContext *scope, Heap::QQmlValueTypeWrapper *valueType, int index); static ReturnedValue create( QV4::ExecutionEngine *engine, Heap::QObjectMethod *cloneFrom, Heap::Object *wrapper, QObject *object); int methodIndex() const { return d()->index; } QObject *object() const { return d()->object(); } QV4::ReturnedValue method_toString(QV4::ExecutionEngine *engine, QObject *o) const; QV4::ReturnedValue method_destroy( QV4::ExecutionEngine *ctx, QObject *o, const Value *args, int argc) const; static ReturnedValue virtualCall(const FunctionObject *, const Value *thisObject, const Value *argv, int argc); ReturnedValue callInternal(const Value *thisObject, const Value *argv, int argc) const; static QPair extractQtMethod(const QV4::FunctionObject *function); }; struct Q_QML_EXPORT QMetaObjectWrapper : public QV4::FunctionObject { V4_OBJECT2(QMetaObjectWrapper, QV4::FunctionObject) V4_NEEDS_DESTROY static ReturnedValue create(ExecutionEngine *engine, const QMetaObject* metaObject); const QMetaObject *metaObject() const { return d()->metaObject; } protected: static ReturnedValue virtualCallAsConstructor(const FunctionObject *, const Value *argv, int argc, const Value *); static bool virtualIsEqualTo(Managed *a, Managed *b); private: void init(ExecutionEngine *engine); ReturnedValue constructInternal(const Value *argv, int argc) const; }; struct Q_QML_EXPORT QmlSignalHandler : public QV4::Object { V4_OBJECT2(QmlSignalHandler, QV4::Object) V4_PROTOTYPE(signalHandlerPrototype) V4_NEEDS_DESTROY int signalIndex() const { return d()->signalIndex; } QObject *object() const { return d()->object(); } static void initProto(ExecutionEngine *v4); }; using QObjectBiPointer = QBiPointer; class MultiplyWrappedQObjectMap : public QObject, private QHash { Q_OBJECT public: typedef QHash::ConstIterator ConstIterator; typedef QHash::Iterator Iterator; using value_type = QHash::value_type; ConstIterator begin() const { return QHash::constBegin(); } Iterator begin() { return QHash::begin(); } ConstIterator end() const { return QHash::constEnd(); } Iterator end() { return QHash::end(); } template void insert(Pointer key, Heap::Object *value) { QHash::operator[](key).set(value->internalClass->engine, value); connect(key, SIGNAL(destroyed(QObject*)), this, SLOT(removeDestroyedObject(QObject*))); } template ReturnedValue value(Pointer key) const { ConstIterator it = find(key); return it == end() ? QV4::WeakValue().value() : it->value(); } Iterator erase(Iterator it); template void remove(Pointer key) { Iterator it = find(key); if (it == end()) return; erase(it); } template void mark(Pointer key, MarkStack *markStack) { Iterator it = find(key); if (it == end()) return; it->markOnce(markStack); } private Q_SLOTS: void removeDestroyedObject(QObject*); }; } QT_END_NAMESPACE #endif // QV4QOBJECTWRAPPER_P_H