Merge integration refs/builds/qtci/dev/1616690051

This commit is contained in:
Qt CI Bot 2021-03-25 20:23:15 +00:00
commit 87308fb95d
7 changed files with 98 additions and 54 deletions

View File

@ -384,6 +384,33 @@ QString QMetaObject::tr(const char *s, const char *c, int n) const
{ {
return QCoreApplication::translate(objectClassName(this), s, c, n); return QCoreApplication::translate(objectClassName(this), s, c, n);
} }
/*!
\since 6.2
Returns the metatype corresponding to this metaobject.
If the metaobject originates from a namespace, an invalid metatype is returned.
*/
QMetaType QMetaObject::metaType() const
{
const QMetaObjectPrivate *d = priv(this->d.data);
if (d->revision < 10) {
// before revision 10, we did not store the metatype in the metatype array
return QMetaType::fromName(className());
} else {
/* in the metatype array, we store
idx: 0 propertyCount - 1 propertyCount
data:QMetaType(prop0), ..., QMetaType(propPropCount-1), QMetaType(class),...
*/
auto iface = this->d.metaTypes[d->propertyCount];
if (iface == QtPrivate::qMetaTypeInterfaceForType<void>())
return QMetaType(); // return invalid meta-type for namespaces
if (iface)
return QMetaType(iface);
else // in case of a dynamic metaobject, we might have no metatype stored
return QMetaType::fromName(className()); // try lookup by name in that case
}
}
#endif // QT_NO_TRANSLATION #endif // QT_NO_TRANSLATION
/*! /*!

View File

@ -172,7 +172,7 @@ struct QMetaObjectPrivate
// revision 7 is Qt 5.0 everything lower is not supported // revision 7 is Qt 5.0 everything lower is not supported
// revision 8 is Qt 5.12: It adds the enum name to QMetaEnum // revision 8 is Qt 5.12: It adds the enum name to QMetaEnum
// revision 9 is Qt 6.0: It adds the metatype of properties and methods // revision 9 is Qt 6.0: It adds the metatype of properties and methods
enum { OutputRevision = 9 }; // Used by moc, qmetaobjectbuilder and qdbus enum { OutputRevision = 10 }; // Used by moc, qmetaobjectbuilder and qdbus
enum { IntsPerMethod = QMetaMethod::Data::Size }; enum { IntsPerMethod = QMetaMethod::Data::Size };
enum { IntsPerEnum = QMetaEnum::Data::Size }; enum { IntsPerEnum = QMetaEnum::Data::Size };
enum { IntsPerProperty = QMetaProperty::Data::Size }; enum { IntsPerProperty = QMetaProperty::Data::Size };

View File

@ -1203,7 +1203,7 @@ static int buildMetaObject(QMetaObjectBuilderPrivate *d, char *buf,
- int(d->methods.size()) // return "parameters" don't have names - int(d->methods.size()) // return "parameters" don't have names
- int(d->constructors.size()); // "this" parameters don't have names - int(d->constructors.size()); // "this" parameters don't have names
if constexpr (mode == Construct) { if constexpr (mode == Construct) {
static_assert(QMetaObjectPrivate::OutputRevision == 9, "QMetaObjectBuilder should generate the same version as moc"); static_assert(QMetaObjectPrivate::OutputRevision == 10, "QMetaObjectBuilder should generate the same version as moc");
pmeta->revision = QMetaObjectPrivate::OutputRevision; pmeta->revision = QMetaObjectPrivate::OutputRevision;
pmeta->flags = d->flags; pmeta->flags = d->flags;
pmeta->className = 0; // Class name is always the first string. pmeta->className = 0; // Class name is always the first string.
@ -1281,7 +1281,8 @@ static int buildMetaObject(QMetaObjectBuilderPrivate *d, char *buf,
// Output the methods in the class. // Output the methods in the class.
Q_ASSERT(!buf || dataIndex == pmeta->methodData); Q_ASSERT(!buf || dataIndex == pmeta->methodData);
int parameterMetaTypesIndex = int(d->properties.size()); // + 1 for metatype of this metaobject
int parameterMetaTypesIndex = int(d->properties.size()) + 1;
for (const auto &method : d->methods) { for (const auto &method : d->methods) {
[[maybe_unused]] int name = strings.enter(method.name()); [[maybe_unused]] int name = strings.enter(method.name());
int argc = method.parameterCount(); int argc = method.parameterCount();
@ -1448,6 +1449,10 @@ static int buildMetaObject(QMetaObjectBuilderPrivate *d, char *buf,
*types = reinterpret_cast<QtPrivate::QMetaTypeInterface *&>(mt); *types = reinterpret_cast<QtPrivate::QMetaTypeInterface *&>(mt);
types++; types++;
} }
// add metatype interface for this metaobject - must be null
// as we can't know our metatype
*types = nullptr;
types++;
for (const auto &method: d->methods) { for (const auto &method: d->methods) {
QMetaType mt(QMetaType::fromName(method.returnType).id()); QMetaType mt(QMetaType::fromName(method.returnType).id());
*types = reinterpret_cast<QtPrivate::QMetaTypeInterface *&>(mt); *types = reinterpret_cast<QtPrivate::QMetaTypeInterface *&>(mt);

View File

@ -173,6 +173,8 @@ struct Q_CORE_EXPORT QMetaObject
QString tr(const char *s, const char *c, int n = -1) const; QString tr(const char *s, const char *c, int n = -1) const;
#endif // QT_NO_TRANSLATION #endif // QT_NO_TRANSLATION
QMetaType metaType() const;
int methodOffset() const; int methodOffset() const;
int enumeratorOffset() const; int enumeratorOffset() const;
int propertyOffset() const; int propertyOffset() const;

View File

@ -422,7 +422,7 @@ void QDBusMetaObjectGenerator::write(QDBusMetaObject *obj)
- methods.count(); // ditto - methods.count(); // ditto
QDBusMetaObjectPrivate *header = reinterpret_cast<QDBusMetaObjectPrivate *>(idata.data()); QDBusMetaObjectPrivate *header = reinterpret_cast<QDBusMetaObjectPrivate *>(idata.data());
static_assert(QMetaObjectPrivate::OutputRevision == 9, "QtDBus meta-object generator should generate the same version as moc"); static_assert(QMetaObjectPrivate::OutputRevision == 10, "QtDBus meta-object generator should generate the same version as moc");
header->revision = QMetaObjectPrivate::OutputRevision; header->revision = QMetaObjectPrivate::OutputRevision;
header->className = 0; header->className = 0;
header->classInfoCount = 0; header->classInfoCount = 0;
@ -459,6 +459,7 @@ void QDBusMetaObjectGenerator::write(QDBusMetaObject *obj)
idata[typeidOffset++] = 0; // eod idata[typeidOffset++] = 0; // eod
int totalMetaTypeCount = properties.count(); int totalMetaTypeCount = properties.count();
++totalMetaTypeCount; // + 1 for metatype of dynamic metaobject
for (const auto& methodContainer: {signals_, methods}) { for (const auto& methodContainer: {signals_, methods}) {
for (const auto& method: methodContainer) { for (const auto& method: methodContainer) {
int argc = method.inputTypes.size() + qMax(qsizetype(0), method.outputTypes.size() - 1); int argc = method.inputTypes.size() + qMax(qsizetype(0), method.outputTypes.size() - 1);
@ -469,7 +470,7 @@ void QDBusMetaObjectGenerator::write(QDBusMetaObject *obj)
int propertyId = 0; int propertyId = 0;
// add each method: // add each method:
int currentMethodMetaTypeOffset = properties.count(); int currentMethodMetaTypeOffset = properties.count() + 1;
for (int x = 0; x < 2; ++x) { for (int x = 0; x < 2; ++x) {
// Signals must be added before other methods, to match moc. // Signals must be added before other methods, to match moc.
QMap<QByteArray, Method> &map = (x == 0) ? signals_ : methods; QMap<QByteArray, Method> &map = (x == 0) ? signals_ : methods;
@ -559,6 +560,7 @@ void QDBusMetaObjectGenerator::write(QDBusMetaObject *obj)
metaTypes[propertyId++] = QMetaType(mp.type); metaTypes[propertyId++] = QMetaType(mp.type);
} }
metaTypes[propertyId] = QMetaType(); // we can't know our own metatype
Q_ASSERT(offset == header->propertyDBusData); Q_ASSERT(offset == header->propertyDBusData);
Q_ASSERT(signatureOffset == header->methodDBusData); Q_ASSERT(signatureOffset == header->methodDBusData);

View File

@ -396,7 +396,8 @@ void Generator::generateCode()
// //
generateClassInfos(); generateClassInfos();
int initialMetaTypeOffset = cdef->propertyList.count(); // all property metatypes, + 1 for the type of the current class itself
int initialMetaTypeOffset = cdef->propertyList.count() + 1;
// //
// Build signals array first, otherwise the signal indices would be wrong // Build signals array first, otherwise the signal indices would be wrong
@ -560,17 +561,6 @@ void Generator::generateCode()
else else
fprintf(out, " qt_meta_extradata_%s,\n", qualifiedClassNameIdentifier.constData()); fprintf(out, " qt_meta_extradata_%s,\n", qualifiedClassNameIdentifier.constData());
bool constructorListContainsArgument = false;
for (int i = 0; i< cdef->constructorList.count(); ++i) {
const FunctionDef& fdef = cdef->constructorList.at(i);
if (fdef.arguments.count()) {
constructorListContainsArgument = true;
break;
}
}
if (cdef->propertyList.isEmpty() && cdef->signalList.isEmpty() && cdef->slotList.isEmpty() && cdef->methodList.isEmpty() && !constructorListContainsArgument) {
fprintf(out, " nullptr,\n");
} else {
bool needsComma = false; bool needsComma = false;
const bool requireCompleteness = requireCompleteTypes || cdef->requireCompleteMethodTypes; const bool requireCompleteness = requireCompleteTypes || cdef->requireCompleteMethodTypes;
if (!requireCompleteness) { if (!requireCompleteness) {
@ -579,6 +569,7 @@ void Generator::generateCode()
} else { } else {
fprintf(out, "qt_metaTypeArray<\n"); fprintf(out, "qt_metaTypeArray<\n");
} }
// metatypes for properties
for (int i = 0; i < cdef->propertyList.count(); ++i) { for (int i = 0; i < cdef->propertyList.count(); ++i) {
const PropertyDef &p = cdef->propertyList.at(i); const PropertyDef &p = cdef->propertyList.at(i);
if (requireCompleteness) if (requireCompleteness)
@ -587,15 +578,23 @@ void Generator::generateCode()
fprintf(out, "%sQtPrivate::TypeAndForceComplete<%s, std::true_type>", needsComma ? ", " : "", p.type.data()); fprintf(out, "%sQtPrivate::TypeAndForceComplete<%s, std::true_type>", needsComma ? ", " : "", p.type.data());
needsComma = true; needsComma = true;
} }
// type name for the Q_OJBECT/GADGET itself, void for namespaces
auto ownType = !cdef->hasQNamespace ? cdef->classname.data() : "void";
if (requireCompleteness)
fprintf(out, "%s%s", needsComma ? ", " : "", ownType);
else
fprintf(out, "%sQtPrivate::TypeAndForceComplete<%s, std::true_type>", needsComma ? ", " : "", ownType);
// metatypes for all exposed methods
// no need to check for needsComma any longer, as we always need one due to the classname being present
for (const QList<FunctionDef> &methodContainer : for (const QList<FunctionDef> &methodContainer :
{ cdef->signalList, cdef->slotList, cdef->methodList }) { { cdef->signalList, cdef->slotList, cdef->methodList }) {
for (int i = 0; i< methodContainer.count(); ++i) { for (int i = 0; i< methodContainer.count(); ++i) {
const FunctionDef& fdef = methodContainer.at(i); const FunctionDef& fdef = methodContainer.at(i);
if (requireCompleteness) if (requireCompleteness)
fprintf(out, "%s%s", needsComma ? ", " : "", fdef.type.name.data()); fprintf(out, ", %s", fdef.type.name.data());
else else
fprintf(out, "%sQtPrivate::TypeAndForceComplete<%s, std::false_type>", needsComma ? ", " : "", fdef.type.name.data()); fprintf(out, ", QtPrivate::TypeAndForceComplete<%s, std::false_type>", fdef.type.name.data());
needsComma = true;
for (const auto &argument: fdef.arguments) { for (const auto &argument: fdef.arguments) {
if (requireCompleteness) if (requireCompleteness)
fprintf(out, ", %s", argument.type.name.data()); fprintf(out, ", %s", argument.type.name.data());
@ -609,15 +608,13 @@ void Generator::generateCode()
const FunctionDef& fdef = cdef->constructorList.at(i); const FunctionDef& fdef = cdef->constructorList.at(i);
for (const auto &argument: fdef.arguments) { for (const auto &argument: fdef.arguments) {
if (requireCompleteness) if (requireCompleteness)
fprintf(out, "%s%s", needsComma ? ", " : "", argument.type.name.data()); fprintf(out, ", %s", argument.type.name.data());
else else
fprintf(out, "%sQtPrivate::TypeAndForceComplete<%s, std::false_type>", needsComma ? ", " : "", argument.type.name.data()); fprintf(out, ", QtPrivate::TypeAndForceComplete<%s, std::false_type>", argument.type.name.data());
needsComma = true;
} }
} }
fprintf(out, "\n"); fprintf(out, "\n");
fprintf(out, ">,\n"); fprintf(out, ">,\n");
}
fprintf(out, " nullptr\n} };\n\n"); fprintf(out, " nullptr\n} };\n\n");

View File

@ -315,6 +315,8 @@ private slots:
void propertyConstant(); void propertyConstant();
void propertyFinal(); void propertyFinal();
void metaType();
void stdSet(); void stdSet();
void classInfo(); void classInfo();
@ -1612,6 +1614,15 @@ void tst_QMetaObject::propertyFinal()
QVERIFY(!prop.isFinal()); QVERIFY(!prop.isFinal());
} }
void tst_QMetaObject::metaType()
{
QCOMPARE(QObject::staticMetaObject.metaType(), QMetaType::fromType<QObject>());
QCOMPARE(MyGadget::staticMetaObject.metaType(), QMetaType::fromType<MyGadget>());
QCOMPARE(QAbstractProxyModel::staticMetaObject.metaType(), QMetaType::fromType<QAbstractProxyModel>());
auto qtNameSpaceMetaType = Qt::staticMetaObject.metaType();
QVERIFY2(!qtNameSpaceMetaType.isValid(), qtNameSpaceMetaType.name());
}
class ClassInfoTestObjectA : public QObject class ClassInfoTestObjectA : public QObject
{ {
Q_OBJECT Q_OBJECT