blob: 7dcf2cd0b8563b11b36f2fe87db55dbcc449a71a (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
|
// Copyright (C) 2020 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
#include "qv4resolvedtypereference_p.h"
#include <QtQml/private/qqmlengine_p.h>
#include <QtQml/qqmlpropertymap.h>
#include <QtCore/qcryptographichash.h>
QT_BEGIN_NAMESPACE
namespace QV4 {
template <typename T>
bool qtTypeInherits(const QMetaObject *mo) {
while (mo) {
if (mo == &T::staticMetaObject)
return true;
mo = mo->superClass();
}
return false;
}
void ResolvedTypeReference::doDynamicTypeCheck()
{
const QMetaObject *mo = nullptr;
if (m_typePropertyCache)
mo = m_typePropertyCache->firstCppMetaObject();
else if (m_type.isValid())
mo = m_type.metaObject();
else if (m_compilationUnit)
mo = m_compilationUnit->rootPropertyCache()->firstCppMetaObject();
m_isFullyDynamicType = qtTypeInherits<QQmlPropertyMap>(mo);
}
/*!
Returns the property cache, if one alread exists. The cache is not referenced.
*/
QQmlPropertyCache::ConstPtr ResolvedTypeReference::propertyCache() const
{
if (m_type.isValid())
return m_typePropertyCache;
else
return m_compilationUnit->rootPropertyCache();
}
/*!
Returns the property cache, creating one if it doesn't already exist. The cache is not referenced.
*/
QQmlPropertyCache::ConstPtr ResolvedTypeReference::createPropertyCache()
{
if (m_typePropertyCache) {
return m_typePropertyCache;
} else if (m_type.isValid()) {
const QMetaObject *metaObject = m_type.metaObject();
if (!metaObject) // value type of non-Q_GADGET base with extension
metaObject = m_type.extensionMetaObject();
if (metaObject)
m_typePropertyCache = QQmlMetaType::propertyCache(metaObject, m_version);
return m_typePropertyCache;
} else {
Q_ASSERT(m_compilationUnit);
return m_compilationUnit->rootPropertyCache();
}
}
bool ResolvedTypeReference::addToHash(
QCryptographicHash *hash, QHash<quintptr, QByteArray> *checksums)
{
if (m_type.isInlineComponentType()) {
// A reference to an inline component in the same file will have
// - no compilation unit since we cannot resolve the compilation unit before it's built.
// - a property cache since we've assigned one in buildMetaObjectsIncrementally().
// - a QQmlType that says it's an inline component.
// We don't have to add such a thing to the hash since if it changes, the QML document
// itself changes, leading to a new timestamp, which is checked before the checksum.
if (!m_compilationUnit)
return !m_typePropertyCache.isNull();
} else if (m_type.isValid()) {
bool ok = false;
if (QQmlPropertyCache::ConstPtr propertyCache = createPropertyCache())
hash->addData(propertyCache->checksum(checksums, &ok));
else
Q_ASSERT(m_type.module() == QLatin1String("QML")); // a builtin without metaobject
return ok;
}
if (!m_compilationUnit)
return false;
hash->addData({m_compilationUnit->unitData()->md5Checksum,
sizeof(m_compilationUnit->unitData()->md5Checksum)});
return true;
}
} // namespace QV4
QT_END_NAMESPACE
|