我试图将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();
}

10-01 08:36