我试图将UDisks2与Qt应用程序接口(interface),并读取自定义属性会使程序崩溃。
对于SMART属性,一切正常(包括custom properties),但是当我尝试读取RAID设备属性ActiveDevices时,出现Qt致命错误,并且应用程序崩溃
以下是一个说明行为的测试程序:
#include <QApplication>
#include <QDBusMetaType>
#include <QDBusConnection>
#include <QDBusInterface>
#include <QDebug>
//create the structure as defined in
// http://udisks.freedesktop.org/docs/latest/gdbus-org.freedesktop.UDisks2.MDRaid.html#gdbus-property-org-freedesktop-UDisks2-MDRaid.ActiveDevices
struct MDRaidMember {
QDBusObjectPath block;
qint32 slot;
QStringList state;
qint64 numReadErrors;
QVariantMap expansion;
};
Q_DECLARE_METATYPE(MDRaidMember)
//marshalling operator
QDBusArgument &operator<<(QDBusArgument &argument, const MDRaidMember& raidMember)
{
argument.beginStructure();
argument << raidMember.block;
argument << raidMember.slot;
argument << raidMember.state;
argument << raidMember.numReadErrors;
argument << raidMember.expansion;
argument.endStructure();
return argument;
}
//unmarshall operator
const QDBusArgument &operator>>(const QDBusArgument &argument, MDRaidMember& raidMember)
{
argument.beginStructure();
argument >> raidMember.block;
argument >> raidMember.slot;
argument >> raidMember.state;
argument >> raidMember.numReadErrors;
argument >> raidMember.expansion;
argument.endStructure();
return argument;
}
int main(int argc, char *argv[]) {
QApplication app(argc, argv);
//register the type, everything looks fine
qDebug() << qRegisterMetaType<MDRaidMember>("MDRaidMember");
qDebug() << qDBusRegisterMetaType<MDRaidMember>();
QDBusInterface iface("org.freedesktop.UDisks2",
"/org/freedesktop/UDisks2/mdraid/9aec6784_86ec37d4_8c1c6add_4dc9fe81",
"org.freedesktop.UDisks2.MDRaid",
QDBusConnection::systemBus());
//read some properties
qDebug() << iface.property("UUID");
qDebug() << iface.property("ActiveDevices"); //crash the test program!
return app.exec();
}
执行此代码将产生以下输出
1057
1057
QVariant(QString, "9aec6784:86ec37d4:8c1c6add:4dc9fe81")
Cannot construct placeholder type QDBusRawType
zsh: abort (core dumped) ./app/test_members
任何人都喜欢这种情况?我完全不认为这与我的代码有关,因为如果不声明自定义类型(仅保留主函数的内容),结果将是相同的。
最佳答案
我没有找到正确的答案,但是这里有一个解决方法:在Get
接口(interface)上调用org.freedesktop.DBus.Properties
方法,然后手动解码结果。棘手的部分是弄清楚结果的内容(这是一个包含QDBus的QVariant包含一个QDBusArgument的Variant)
更新的main
函数:
//...
int main(int argc, char *argv[]) {
QApplication app(argc, argv);
//register the type, everything looks fine
qDebug() << qRegisterMetaType<MDRaidMember>("MDRaidMember");
qDebug() << qDBusRegisterMetaType<MDRaidMember>();
QDBusInterface iface("org.freedesktop.UDisks2",
"/org/freedesktop/UDisks2/mdraid/9aec6784_86ec37d4_8c1c6add_4dc9fe81",
"org.freedesktop.DBus.Properties",
QDBusConnection::systemBus());
QDBusMessage reply = iface.call("Get", "org.freedesktop.UDisks2.MDRaid", "ActiveDevices");
QVariant v = reply.arguments().first();
QDBusArgument arg = v.value<QDBusVariant>().variant().value<QDBusArgument>();
QList<MDRaidMember> members;
arg.beginArray();
while(!arg.atEnd()) {
MDRaidMember m;
arg >> m;
members << m;
}
qDebug() << members;
return app.exec();
}