aboutsummaryrefslogtreecommitdiffstats
path: root/src/qml/jsruntime/qv4referenceobject_p.h
diff options
context:
space:
mode:
Diffstat (limited to 'src/qml/jsruntime/qv4referenceobject_p.h')
-rw-r--r--src/qml/jsruntime/qv4referenceobject_p.h171
1 files changed, 171 insertions, 0 deletions
diff --git a/src/qml/jsruntime/qv4referenceobject_p.h b/src/qml/jsruntime/qv4referenceobject_p.h
new file mode 100644
index 0000000000..094749ce6e
--- /dev/null
+++ b/src/qml/jsruntime/qv4referenceobject_p.h
@@ -0,0 +1,171 @@
+// Copyright (C) 2022 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 QV4REFERENCEOBJECT_P_H
+#define QV4REFERENCEOBJECT_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 <private/qv4object_p.h>
+#include <private/qv4stackframe_p.h>
+
+QT_BEGIN_NAMESPACE
+
+namespace QV4 {
+namespace Heap {
+
+
+#define ReferenceObjectMembers(class, Member) \
+ Member(class, Pointer, Object *, m_object)
+
+DECLARE_HEAP_OBJECT(ReferenceObject, Object) {
+ DECLARE_MARKOBJECTS(ReferenceObject);
+
+ enum Flag : quint8 {
+ NoFlag = 0,
+ CanWriteBack = 1 << 0,
+ IsVariant = 1 << 1,
+ EnforcesLocation = 1 << 2,
+ };
+ Q_DECLARE_FLAGS(Flags, Flag);
+
+ void init(Object *object, int property, Flags flags)
+ {
+ setObject(object);
+ m_property = property;
+ m_flags = flags;
+ Object::init();
+ }
+
+ Flags flags() const { return Flags(m_flags); }
+
+ Object *object() const { return m_object.get(); }
+ void setObject(Object *object) { m_object.set(internalClass->engine, object); }
+
+ int property() const { return m_property; }
+
+ bool canWriteBack() const { return hasFlag(CanWriteBack); }
+ bool isVariant() const { return hasFlag(IsVariant); }
+ bool enforcesLocation() const { return hasFlag(EnforcesLocation); }
+
+ void setLocation(const Function *function, quint16 statement)
+ {
+ m_function = function;
+ m_statementIndex = statement;
+ }
+
+ const Function *function() const { return m_function; }
+ quint16 statementIndex() const { return m_statementIndex; }
+
+ bool isAttachedToProperty() const
+ {
+ if (enforcesLocation()) {
+ if (CppStackFrame *frame = internalClass->engine->currentStackFrame) {
+ if (frame->v4Function != function() || frame->statementNumber() != statementIndex())
+ return false;
+ } else {
+ return false;
+ }
+ }
+
+ return true;
+ }
+
+ bool isReference() const { return m_object; }
+
+private:
+
+ bool hasFlag(Flag flag) const
+ {
+ return m_flags & quint8(flag);
+ }
+
+ void setFlag(Flag flag, bool set)
+ {
+ m_flags = set ? (m_flags | quint8(flag)) : (m_flags & ~quint8(flag));
+ }
+
+ const Function *m_function;
+ int m_property;
+ quint16 m_statementIndex;
+ quint8 m_flags;
+};
+
+Q_DECLARE_OPERATORS_FOR_FLAGS(ReferenceObject::Flags)
+
+} // namespace Heap
+
+
+struct ReferenceObject : public Object
+{
+ V4_OBJECT2(ReferenceObject, Object)
+ V4_NEEDS_DESTROY
+
+public:
+ static constexpr const int AllProperties = -1;
+
+ template<typename HeapObject>
+ static bool readReference(HeapObject *ref)
+ {
+ if (!ref->object())
+ return false;
+
+ QV4::Scope scope(ref->internalClass->engine);
+ QV4::ScopedObject object(scope, ref->object());
+
+ if (ref->isVariant()) {
+ QVariant variant;
+ void *a[] = { &variant };
+ return object->metacall(QMetaObject::ReadProperty, ref->property(), a)
+ && ref->setVariant(variant);
+ }
+
+ void *a[] = { ref->storagePointer() };
+ return object->metacall(QMetaObject::ReadProperty, ref->property(), a);
+ }
+
+ template<typename HeapObject>
+ static bool writeBack(HeapObject *ref, int internalIndex = AllProperties)
+ {
+ if (!ref->object() || !ref->canWriteBack())
+ return false;
+
+ QV4::Scope scope(ref->internalClass->engine);
+ QV4::ScopedObject object(scope, ref->object());
+
+ int flags = QQmlPropertyData::HasInternalIndex;
+ int status = -1;
+ if (ref->isVariant()) {
+ QVariant variant = ref->toVariant();
+ void *a[] = { &variant, nullptr, &status, &flags, &internalIndex };
+ return object->metacall(QMetaObject::WriteProperty, ref->property(), a);
+ }
+
+ void *a[] = { ref->storagePointer(), nullptr, &status, &flags, &internalIndex };
+ return object->metacall(QMetaObject::WriteProperty, ref->property(), a);
+ }
+
+ template<typename HeapObject>
+ static HeapObject *detached(HeapObject *ref)
+ {
+ if (ref->object() && !ref->enforcesLocation() && !readReference(ref))
+ return ref; // It's dead. No point in detaching it anymore
+
+ return ref->detached();
+ }
+};
+
+} // namespace QV4
+
+QT_END_NAMESPACE
+
+#endif // QV4REFERENCEOBJECT_P_H