/**************************************************************************** ** ** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of the QtQml module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL21$ ** 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 Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/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 2.1 or version 3 as published by the Free ** Software Foundation and appearing in the file LICENSE.LGPLv21 and ** LICENSE.LGPLv3 included in the packaging of this file. Please review the ** following information to ensure the GNU Lesser General Public License ** requirements will be met: https://www.gnu.org/licenses/lgpl.html and ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #ifndef QMLJS_MANAGED_H #define QMLJS_MANAGED_H #include #include #include #include "qv4global_p.h" #include "qv4value_p.h" #include "qv4internalclass_p.h" QT_BEGIN_NAMESPACE namespace QV4 { #define Q_MANAGED_CHECK \ template inline void qt_check_for_QMANAGED_macro(const T *_q_argument) const \ { int i = qYouForgotTheQ_MANAGED_Macro(this, _q_argument); i = i + 1; } template inline int qYouForgotTheQ_MANAGED_Macro(T, T) { return 0; } template inline void qYouForgotTheQ_MANAGED_Macro(T1, T2) {} #ifdef Q_COMPILER_STATIC_ASSERT #define V4_MANAGED_SIZE_TEST void __dataTest() { Q_STATIC_ASSERT(sizeof(*this) == sizeof(Managed)); } #else #define V4_MANAGED_SIZE_TEST #endif #define V4_MANAGED(superClass) \ public: \ Q_MANAGED_CHECK \ typedef superClass SuperClass; \ static const QV4::ManagedVTable static_vtbl; \ static inline const QV4::ManagedVTable *staticVTable() { return &static_vtbl; } \ template \ QV4::Returned *asReturned() { return QV4::Returned::create(this); } \ V4_MANAGED_SIZE_TEST \ const Data *d() const { return &static_cast(Managed::data); } \ Data *d() { return &static_cast(Managed::data); } #define V4_OBJECT(superClass) \ public: \ Q_MANAGED_CHECK \ typedef superClass SuperClass; \ static const QV4::ObjectVTable static_vtbl; \ static inline const QV4::ManagedVTable *staticVTable() { return &static_vtbl.managedVTable; } \ template \ QV4::Returned *asReturned() { return QV4::Returned::create(this); } \ V4_MANAGED_SIZE_TEST \ const Data *d() const { return &static_cast(Managed::data); } \ Data *d() { return &static_cast(Managed::data); } #define Q_MANAGED_TYPE(type) \ public: \ enum { MyType = Type_##type }; #define Q_VTABLE_FUNCTION(classname, func) \ (classname::func == QV4::Managed::func ? 0 : classname::func) struct GCDeletable { GCDeletable() : next(0), lastCall(false) {} virtual ~GCDeletable() {} GCDeletable *next; bool lastCall; }; struct ManagedVTable { const ManagedVTable * const parent; uint isExecutionContext : 1; uint isString : 1; uint isObject : 1; uint isFunctionObject : 1; uint isErrorObject : 1; uint isArrayData : 1; uint unused : 18; uint type : 8; const char *className; void (*destroy)(Managed *); void (*markObjects)(Managed *, ExecutionEngine *e); bool (*isEqualTo)(Managed *m, Managed *other); }; struct ObjectVTable { ManagedVTable managedVTable; ReturnedValue (*call)(Managed *, CallData *data); ReturnedValue (*construct)(Managed *, CallData *data); ReturnedValue (*get)(Managed *, String *name, bool *hasProperty); ReturnedValue (*getIndexed)(Managed *, uint index, bool *hasProperty); void (*put)(Managed *, String *name, const ValueRef value); void (*putIndexed)(Managed *, uint index, const ValueRef value); PropertyAttributes (*query)(const Managed *, String *name); PropertyAttributes (*queryIndexed)(const Managed *, uint index); bool (*deleteProperty)(Managed *m, String *name); bool (*deleteIndexedProperty)(Managed *m, uint index); ReturnedValue (*getLookup)(Managed *m, Lookup *l); void (*setLookup)(Managed *m, Lookup *l, const ValueRef v); uint (*getLength)(const Managed *m); void (*advanceIterator)(Managed *m, ObjectIterator *it, String *&name, uint *index, Property *p, PropertyAttributes *attributes); }; #define DEFINE_MANAGED_VTABLE_INT(classname, parentVTable) \ { \ parentVTable, \ classname::IsExecutionContext, \ classname::IsString, \ classname::IsObject, \ classname::IsFunctionObject, \ classname::IsErrorObject, \ classname::IsArrayData, \ 0, \ classname::MyType, \ #classname, \ Q_VTABLE_FUNCTION(classname, destroy), \ markObjects, \ isEqualTo \ } #define DEFINE_MANAGED_VTABLE(classname) \ const QV4::ManagedVTable classname::static_vtbl = DEFINE_MANAGED_VTABLE_INT(classname, 0) #define DEFINE_OBJECT_VTABLE(classname) \ const QV4::ObjectVTable classname::static_vtbl = \ { \ DEFINE_MANAGED_VTABLE_INT(classname, &classname::SuperClass::static_vtbl == &Object::static_vtbl ? 0 : &classname::SuperClass::static_vtbl.managedVTable), \ call, \ construct, \ get, \ getIndexed, \ put, \ putIndexed, \ query, \ queryIndexed, \ deleteProperty, \ deleteIndexedProperty, \ getLookup, \ setLookup, \ getLength, \ advanceIterator \ } struct Q_QML_PRIVATE_EXPORT Managed { struct Q_QML_PRIVATE_EXPORT Data : HeapObject { Data() {} Data(InternalClass *internal) : internalClass(internal) , markBit(0) , inUse(1) , extensible(1) { // #### // Q_ASSERT(internal && internal->vtable); } InternalClass *internalClass; struct { uchar markBit : 1; uchar inUse : 1; uchar extensible : 1; // used by Object uchar _unused : 1; uchar needsActivation : 1; // used by FunctionObject uchar strictMode : 1; // used by FunctionObject uchar bindingKeyFlag : 1; uchar hasAccessorProperty : 1; uchar _type; mutable uchar subtype; uchar _flags; }; void setVTable(const ManagedVTable *vt); ReturnedValue asReturnedValue() const { return reinterpret_cast(const_cast(this))->asReturnedValue(); } void *operator new(size_t, Managed *m) { return m; } void *operator new(size_t, Managed::Data *m) { return m; } void operator delete(void *, Managed::Data *) {} }; Data data; V4_MANAGED(Managed) enum { IsExecutionContext = false, IsString = false, IsObject = false, IsFunctionObject = false, IsErrorObject = false, IsArrayData = false }; private: void *operator new(size_t); Managed(const Managed &other); void operator = (const Managed &other); protected: Managed(InternalClass *internal) : data(internal) { } public: void *operator new(size_t size, MemoryManager *mm); void *operator new(size_t, Managed *m) { return m; } inline void mark(QV4::ExecutionEngine *engine); enum Type { Type_Invalid, Type_String, Type_Object, Type_ArrayObject, Type_FunctionObject, Type_BooleanObject, Type_NumberObject, Type_StringObject, Type_DateObject, Type_RegExpObject, Type_ErrorObject, Type_ArgumentsObject, Type_JsonObject, Type_MathObject, Type_ExecutionContext, Type_ForeachIteratorObject, Type_RegExp, Type_QmlSequence }; Q_MANAGED_TYPE(Invalid) ExecutionEngine *engine() const; template T *as() { Q_ASSERT(internalClass()); #if !defined(QT_NO_QOBJECT_CHECK) static_cast(this)->qt_check_for_QMANAGED_macro(static_cast(this)); #endif const ManagedVTable *vt = internalClass()->vtable; while (vt) { if (vt == T::staticVTable()) return static_cast(this); vt = vt->parent; } return 0; } template const T *as() const { Q_ASSERT(internalClass()); #if !defined(QT_NO_QOBJECT_CHECK) static_cast(this)->qt_check_for_QMANAGED_macro(static_cast(const_cast(this))); #endif const ManagedVTable *vt = internalClass()->vtable; while (vt) { if (vt == T::staticVTable()) return static_cast(this); vt = vt->parent; } return 0; } String *asString() { return internalClass()->vtable->isString ? reinterpret_cast(this) : 0; } Object *asObject() { return internalClass()->vtable->isObject ? reinterpret_cast(this) : 0; } ArrayObject *asArrayObject() { return internalClass()->vtable->type == Type_ArrayObject ? reinterpret_cast(this) : 0; } FunctionObject *asFunctionObject() { return internalClass()->vtable->isFunctionObject ? reinterpret_cast(this) : 0; } BooleanObject *asBooleanObject() { return internalClass()->vtable->type == Type_BooleanObject ? reinterpret_cast(this) : 0; } NumberObject *asNumberObject() { return internalClass()->vtable->type == Type_NumberObject ? reinterpret_cast(this) : 0; } StringObject *asStringObject() { return internalClass()->vtable->type == Type_StringObject ? reinterpret_cast(this) : 0; } DateObject *asDateObject() { return internalClass()->vtable->type == Type_DateObject ? reinterpret_cast(this) : 0; } ErrorObject *asErrorObject() { return internalClass()->vtable->isErrorObject ? reinterpret_cast(this) : 0; } ArgumentsObject *asArgumentsObject() { return internalClass()->vtable->type == Type_ArgumentsObject ? reinterpret_cast(this) : 0; } bool isListType() const { return internalClass()->vtable->type == Type_QmlSequence; } bool isArrayObject() const { return internalClass()->vtable->type == Type_ArrayObject; } bool isStringObject() const { return internalClass()->vtable->type == Type_StringObject; } QString className() const; Managed **nextFreeRef() { return reinterpret_cast(this); } Managed *nextFree() { return *reinterpret_cast(this); } void setNextFree(Managed *m) { *reinterpret_cast(this) = m; } void setVTable(const ManagedVTable *vt); bool isEqualTo(Managed *other) { return internalClass()->vtable->isEqualTo(this, other); } static bool isEqualTo(Managed *m, Managed *other); ReturnedValue asReturnedValue() { return Value::fromManaged(this).asReturnedValue(); } InternalClass *internalClass() const { return d()->internalClass; } void setInternalClass(InternalClass *ic) { d()->internalClass = ic; } uchar subtype() const { return d()->subtype; } void setSubtype(uchar subtype) const { d()->subtype = subtype; } bool inUse() const { return d()->inUse; } bool markBit() const { return d()->markBit; } static void destroy(Managed *) {} private: friend class MemoryManager; friend struct Identifiers; friend struct ObjectIterator; }; template<> inline Managed *value_cast(const Value &v) { return v.asManaged(); } template inline T *managed_cast(Managed *m) { return m ? m->as() : 0; } template<> inline String *managed_cast(Managed *m) { return m ? m->asString() : 0; } template<> inline Object *managed_cast(Managed *m) { return m ? m->asObject() : 0; } template<> inline FunctionObject *managed_cast(Managed *m) { return m ? m->asFunctionObject() : 0; } } QT_END_NAMESPACE #endif