Qt5的插件机制(4)--Qt插件的元信息metaData

<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<

JSON 与Qt插件的元信息 MetaData


Qt插件的源码中,基本都能见到一个 xxx.json 的文件,这个文件中通常只包含一句:

{
    "Keys": [ "yyy" ]
}

我们可以猜到这个文件中的"Keys"应该是指定了与插件相关的关键字。那这个 .json 文件到底是如何起作用的?
先来认识一下 JSON .

JSON是一种存储结构化数据的格式,它有6中基本数据类型,分别是:

bool    布尔型,取值可以是 true 或 false
double    数字类型
string    字符串类型
array    数组类型
object    对象类型
null    空类型

具体可参见 Qt Assistant 中关于"JSON Support in Qt "的介绍。

A simple JSON document encoding a person, his/her age, address and phone numbers could look like:

[javascript] view plain
  1. {  
  2.     "FirstName""John",    # FirstName是变量(字段)的名称;John是变量的值  
  3.     "LastName""Doe",  
  4.     "Age": 43,  
  5.     "Address": {  
  6.         "Street""Downing Street 10",  
  7.         "City""London",  
  8.         "Country""Great Britain"  
  9.     },  
  10.     "Phone numbers": [  
  11.         "+44 1234567",  
  12.         "+44 2345678"  
  13.     ]  
  14. }  



值得一提的是,数组类型的字段在.json文件中赋值时应该用方括号 '[' 和 ']' 括起来,对象类型的字段在赋值时
应用花括号 '{' 和 '}' 括起来,普通类型的数据则不需要括。每一个 .json 文件描述了一个 JSON对象,而一个JSON
对象中的对象类型字段,又可以看做是一个子JSON对象(JSON对象的嵌套)。
再回过头来看看插件源码中的 xxx.json 文件,可以发现其中的变量 Keys 其实是个数组变量(因为它赋值时用方括号
括住了),只不过这个数组中通常只有一个元素而已,即一个插件一般只有一个关键字,当然也可以为一个插件设置多个
关键字。

.json在Qt插件中主要用于存储Qt插件的元信息(metaData),在Qt中,有一个专门的类 QJsonObject 来描述一个JSON。
QLibraryPrivate::metaData() 返回的是一个QJsonObject类的指针,通过这个指针可以访问该 QLibraryPrivate 对象对应的库的元信息;
QFactoryLoader::metaData() 返回的是一个 QList<QJsonObject> 列表,这个列表中顺序存放了与该 QFactoryLoader 对象相关的所有库
的元信息(动态库的元信息在前,静态库的元信息在后)。
另外值得注意的是,QLibraryPrivate::metaData() 返回的QJsonObject对象中一般都有一个 MetaData 字段,这个字段是对象类型的数据,
他可以看做是库的元信息的一个子JSON对象,而且它对应的就是 .json 文件中的内容。(.json文件中的内容只是一个库的元信息的一部分)

每个QFactoryLoader对象都有一个 d->iid 成员 (d是与QFactoryLoader对象关联的QFactoryLoaderPrivate实例),可用于描述插件的种类,
每一类插件都有一个独立的 IID 值, 比如平台输入法类插件的IDD都应该是 org.qt-project.Qt.QPlatformInputContextFactoryInterface,
平台类插件的IDD都应是 org.qt-project.Qt.QPA.QPlatformIntegrationFactoryInterface.5.2 等。



<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<

QLibraryPrivate的metaData(库的元信息)的产生过程



在编写Qt插件时需要用到 Q_PLUGIN_METADATA 这个宏来设置一个插件的元信息,但在Qt的源码中,发现这个宏是空的,所以不用说,这个
宏是被MOC解析的而不是C++编译器。那就看看MOC会怎么对付这个宏吧。找了个很简单的文件,用moc处理一下,看看都生成了什么。

main.cpp文件内容如下,其中使用了Q_PLUGIN_METADATA宏


[cpp] view plain
  1. #include <qpa/qplatforminputcontextplugin_p.h>  
  2. #include <QtCore/QStringList>  
  3. #include "QtMinimalInputMethodFrame.h"  
  4. QT_BEGIN_NAMESPACE  
  5. class QtMinimalInputMethodFramePlugin : public QPlatformInputContextPlugin  
  6. {  
  7.     Q_OBJECT  
  8.     Q_PLUGIN_METADATA(IID "org.qt-project.Qt.QPlatformInputContextFactoryInterface" FILE "qtminimal.json")    // 指定了IID和.json文件  
  9.   
  10. public:  
  11.     QtMinimalInputMethodFrame *create(const QString &, const QStringList &);  
  12. };  
  13. QtMinimalInputMethodFrame *QtMinimalInputMethodFramePlugin::create(const QString &system, const QStringList ¶mList)  
  14. {  
  15.     Q_UNUSED(paramList);  
  16.     if (system.compare(system, QStringLiteral("qtminimal"), Qt::CaseInsensitive) == 0)  
  17.         return new QtMinimalInputMethodFrame;  
  18.     return 0;  
  19. }  
  20. QT_END_NAMESPACE  
  21. #include "main.moc"  



qtminimal.json文件内容如下:

[cpp] view plain
  1. {  
  2.     "Keys": [ "qtminimal" ]  
  3. }  

使用 "moc main.cpp moc_main.cpp" ,生成moc_main.cpp文件,打开,发现里面有一部分代码如下:

[cpp] view plain
  1.     ...  
  2.     ...  
  3.   
  4. static const unsigned char qt_pluginMetaData[] = {  
  5.     'Q''T''M''E''T''A''D''A''T''A'' '' ',        // "QTMETADATA",这段字符串可看做Qt的插件元信息的头,通过这个关键字能搜索到元信息的位置  
  6.     0x71, 0x62, 0x6a, 0x73, 0x01, 0x00, 0x00, 0x00,  
  7.     0x00, 0x01, 0x00, 0x00, 0x0b, 0x00, 0x00, 0x00,  
  8.     0xec, 0x00, 0x00, 0x00, 0x1b, 0x03, 0x00, 0x00,  
  9.     0x03, 0x00, 0x49, 0x49, 0x44, 0x00, 0x00, 0x00,  
  10.     0x37, 0x00, 0x6f, 0x72, 0x67, 0x2e, 0x71, 0x74,  
  11.     0x2d, 0x70, 0x72, 0x6f, 0x6a, 0x65, 0x63, 0x74,  
  12.     0x2e, 0x51, 0x74, 0x2e, 0x51, 0x50, 0x6c, 0x61,  
  13.     0x74, 0x66, 0x6f, 0x72, 0x6d, 0x49, 0x6e, 0x70,  
  14.     0x75, 0x74, 0x43, 0x6f, 0x6e, 0x74, 0x65, 0x78,  
  15.     0x74, 0x46, 0x61, 0x63, 0x74, 0x6f, 0x72, 0x79,  
  16.     0x49, 0x6e, 0x74, 0x65, 0x72, 0x66, 0x61, 0x63,  
  17.     0x65, 0x00, 0x00, 0x00, 0x9b, 0x0c, 0x00, 0x00,  
  18.     0x09, 0x00, 0x63, 0x6c, 0x61, 0x73, 0x73, 0x4e,  
  19.     0x61, 0x6d, 0x65, 0x00, 0x1f, 0x00, 0x51, 0x74,  
  20.     0x4d, 0x69, 0x6e, 0x69, 0x6d, 0x61, 0x6c, 0x49,  
  21.     0x6e, 0x70, 0x75, 0x74, 0x4d, 0x65, 0x74, 0x68,  
  22.     0x6f, 0x64, 0x46, 0x72, 0x61, 0x6d, 0x65, 0x50,  
  23.     0x6c, 0x75, 0x67, 0x69, 0x6e, 0x00, 0x00, 0x00,  
  24.     0x5a, 0x60, 0xa0, 0x00, 0x07, 0x00, 0x76, 0x65,  
  25.     0x72, 0x73, 0x69, 0x6f, 0x6e, 0x00, 0x00, 0x00,  
  26.     0x11, 0x00, 0x00, 0x00, 0x05, 0x00, 0x64, 0x65,  
  27.     0x62, 0x75, 0x67, 0x00, 0x95, 0x16, 0x00, 0x00,  
  28.     0x08, 0x00, 0x4d, 0x65, 0x74, 0x61, 0x44, 0x61,  
  29.     0x74, 0x61, 0x00, 0x00, 0x38, 0x00, 0x00, 0x00,  
  30.     0x03, 0x00, 0x00, 0x00, 0x34, 0x00, 0x00, 0x00,  
  31.     0x14, 0x03, 0x00, 0x00, 0x04, 0x00, 0x4b, 0x65,  
  32.     0x79, 0x73, 0x00, 0x00, 0x1c, 0x00, 0x00, 0x00,  
  33.     0x02, 0x00, 0x00, 0x00, 0x18, 0x00, 0x00, 0x00,  
  34.     0x09, 0x00, 0x71, 0x74, 0x6d, 0x69, 0x6e, 0x69,  
  35.     0x6d, 0x61, 0x6c, 0x00, 0x8b, 0x01, 0x00, 0x00,  
  36.     0x0c, 0x00, 0x00, 0x00, 0x0c, 0x00, 0x00, 0x00,  
  37.     0xa4, 0x00, 0x00, 0x00, 0x54, 0x00, 0x00, 0x00,  
  38.     0x98, 0x00, 0x00, 0x00, 0x88, 0x00, 0x00, 0x00  
  39. };  
  40.     ...  
  41.     ...  
  42.   
  43. // 留意一下QT_MOC_EXPORT_PLUGIN这个宏  
  44. QT_MOC_EXPORT_PLUGIN(QtMinimalInputMethodFramePlugin, QtMinimalInputMethodFramePlugin)      
  45.   
  46.     ...  
  47.     ...  



上面的代码中生成了一个名为 qt_pluginMetaData 的数组,看其名字就能猜出来这个数组是与插件的元信息有关的,
将该数组用字符显示,结果为:
[cpp] view plain
  1. QTMETADATA  qbjs^A^@^@^@^@^A^@^@^K^@^@^@ì^@^@^@^[^C^@^@^C^@  
  2. IID^@^@^@7^@org.qt-project.Qt.QPlatformInputContextFactoryInterface    // IDD  
  3. ^@^@^@<9b>^L^@^@^@          
  4. className^@^_^@QtMinimalInputMethodFramePlugin        // 类名  
  5. ^@^@^@Z` ^@^G^@  
  6. version^@^@^@^Q^@^@^@^E^@            // 版本  
  7. debug^@<95>^V^@^@^H^@                // debug  
  8. MetaData^@^@8^@^@^@^C^@^@^@4^@^@^@^T^C^@^@^D^@    // MetaData  
  9. Keys^@^@^\^@^@^@^B^@^@^@^X^@^@^@^@qtminimal    // Keys  
  10. ^@<8b>^A^@^@^L^@^@^@^L^@^@^@¤^@^@^@T^@^@^@<98>^@^@^@<88>^@^@^@  


可以看到其中已经包含了IDD、Keys等信息,因此可以确定与插件相关的元信息就是存在这个数组中。
上面给出的代码最后,使用了QT_MOC_EXPORT_PLUGIN宏,它的作用是定义两个函数,qt_plugin_instance() 和
qt_plugin_query_metadata(),前者用于返回插件类(QtMinimalInputMethodFramePlugin)的一个实例,后者用于
返回数组qt_pluginMetaData的指针。因此,当这个库被加载后,通过库中的qt_plugin_query_metadata()函数就可以获得
库的元信息数组(二进制数据)。

[cpp] view plain
  1. #define Q_PLUGIN_INSTANCE(IMPLEMENTATION) \    // 这个宏用于返回一个类名是IMPLEMENTATION的类的静态实例  
  2.         { \  
  3.             static QT_PREPEND_NAMESPACE(QPointer)<QT_PREPEND_NAMESPACE(QObject)> _instance; \  
  4.             if (!_instance)      \  
  5.                 _instance = new IMPLEMENTATION; \  
  6.             return _instance; \  
  7.         }  
  8.   
  9. #if defined(QT_STATICPLUGIN)  
  10.     // 这部分是静态编译时用的宏,我们暂不管它  
  11. #  define QT_MOC_EXPORT_PLUGIN(PLUGINCLASS, PLUGINCLASSNAME) \  
  12.     static QT_PREPEND_NAMESPACE(QObject) *qt_plugin_instance_##PLUGINCLASSNAME() \  
  13.     Q_PLUGIN_INSTANCE(PLUGINCLASS) \  
  14.     static const char *qt_plugin_query_metadata_##PLUGINCLASSNAME() { return (const char *)qt_pluginMetaData; } \  
  15.     const QT_PREPEND_NAMESPACE(QStaticPlugin) qt_static_plugin_##PLUGINCLASSNAME() { \  
  16.         QT_PREPEND_NAMESPACE(QStaticPlugin) plugin = { qt_plugin_instance_##PLUGINCLASSNAME, qt_plugin_query_metadata_##PLUGINCLASSNAME}; \  
  17.         return plugin; \  
  18.     }  
  19.   
  20. #else  
  21.     // 这部分才是动态编译时用的宏  
  22. #  define QT_MOC_EXPORT_PLUGIN(PLUGINCLASS, PLUGINCLASSNAME)      \  
  23.             Q_EXTERN_C Q_DECL_EXPORT \  
  24.             const char *qt_plugin_query_metadata() \            // 这个函数返回一个指针,指向元信息数组  
  25.             { return (const char *)qt_pluginMetaData; } \  
  26.             Q_EXTERN_C Q_DECL_EXPORT QT_PREPEND_NAMESPACE(QObject) *qt_plugin_instance() \ 这个函数返回插件类PLUGINCLASS的一个静态实例  
  27.             Q_PLUGIN_INSTANCE(PLUGINCLASS)  
  28.   
  29. #endif  


我们虽然可以通过qt_plugin_query_metadata()函数获得库的元信息数组(二进制数据),但是还需要把这些
二进制数据转换成 QJsonObject 对象,这样才方便在程序中对其各个字段进行访问。QLibraryPrivate类有个
静态成员函数fromRawMetaData,可以从二进制数据中构造一个 QJsonDocument 对象,
而QJsonDocument对象的 object() 方法就能返回所需的 QJsonObject 对象(库的元信息)。
通过这些可以猜测,QLibraryPrivate类的 metaData 成员是这样被设置的:
加载一个库文件后,将库中的qt_plugin_query_metadata()这个符号resolve出来,并通过它得到数组 qt_pluginMetaData 的指针,
然后调用QLibraryPrivate类的fromRawMetaData方法讲二进制存储的元信息转换到一个QJsonObject 对象中,最后将这个对象赋值
给QLibraryPrivate类的metaData成员。


现在问题是, QLibraryPrivate的metaData是在哪里,被谁设置的?

当第一次使用 QLibraryPrivate类的 isPlugin() 方法调查一个库是否是插件时,这个方法内部会调用 QLibraryPrivate::updatePluginState() 。

[cpp] view plain
  1. bool QLibraryPrivate::isPlugin()  
  2. {  
  3.     if (pluginState == MightBeAPlugin)    // pluginState在构造函数中会被初始化为MightBeAPlugin  
  4.         updatePluginState();  
  5.   
  6.     return pluginState == IsAPlugin;  
  7. }  

而QLibraryPrivate::updatePluginState()函数中,需要读取QLibraryPrivate类的 metaData 信息来确认一个库是不是插件,而 metaData 也就是在这时
被设置的。从这个函数中还可以看出,只有Qt插件才有 metaData 元信息, 普通的库是没有的(这一点我上不能打保票,不过从下面这个函数的代码上看,应该
是这样的)。

[cpp] view plain
  1. void QLibraryPrivate::updatePluginState()  
  2. {  
  3.     errorString.clear();  
  4.     if (pluginState != MightBeAPlugin)  
  5.         return;        // 如果pluginState的状态已经确定了,则直接返回  
  6.   
  7.     bool success = false;  
  8.   
  9. #if defined(Q_OS_UNIX) && !defined(Q_OS_MAC)  
  10.     if (fileName.endsWith(QLatin1String(".debug"))) {  
  11.         // refuse to load a file that ends in .debug  
  12.         // these are the debug symbols from the libraries  
  13.         // the problem is that they are valid shared library files  
  14.         // and dlopen is known to crash while opening them  
  15.   
  16.         // pretend we didn't see the file  
  17.         errorString = QLibrary::tr("The shared library was not found.");  
  18.         pluginState = IsNotAPlugin;  
  19.         return;  
  20.     }  
  21. #endif  
  22.   
  23.     // 这里开始设置 metaData 。findPatternUnloaded() 和 qt_get_metadata() 两个函数的内部都会设置 metaData  
  24.     if (!pHnd) {  
  25.     // 如果库还没有被加载,(pHnd==NULL说明库未被加载),则调用findPatternUnloaded函数,它内部会  
  26.     // 讲对应的库文件打开,读取其内容,在其中寻找Qt的插件元信息的头Header(就是前面提到的"QTMETADATA"),  
  27.     // 如果找到了,就解析元信息。这种方法的好处是在不加载库的情况下也能获得其元信息。  
  28.         // scan for the plugin metadata without loading  
  29.         success = findPatternUnloaded(fileName, this);      
  30.     } else {  
  31.     // 如果库已被加载,则调用qt_get_metadata获得元信息。  
  32.     // QtPluginQueryVerificationDataFunction是一个函数指针类型,其具体类型是 :  char*(*)(), 它返回  
  33.     // 一个字符指针(指向一个二进制数组),准确的说,他应该返回上面提到的qt_pluginMetaData数组的指针 。  
  34.   
  35.         // library is already loaded (probably via QLibrary)  
  36.         // simply get the target function and call it.  
  37.         QtPluginQueryVerificationDataFunction getMetaData = NULL;  
  38.         getMetaData = (QtPluginQueryVerificationDataFunction) resolve("qt_plugin_query_metadata");      
  39.         // 上面这行将库中的qt_plugin_query_metadata符号resolve出来  
  40.         success = qt_get_metadata(getMetaData, this);  
  41.     }  
  42.   
  43.     if (!success) {  
  44.     // 如果获取 元信息 失败,即当前库没有元信息,就将pluginState设置为IsNotAPlugin,代表非Qt插件。  
  45.     // 这里因为当前库没有元信息,就认为它不是Qt插件,所以这是不是意味着,只有Qt插件才有元信息而普通库是没有的?  
  46.         if (errorString.isEmpty()){  
  47.             if (fileName.isEmpty())  
  48.                 errorString = QLibrary::tr("The shared library was not found.");  
  49.             else  
  50.                 errorString = QLibrary::tr("The file '%1' is not a valid Qt plugin.").arg(fileName);  
  51.         }  
  52.         pluginState = IsNotAPlugin;  
  53.         return;  
  54.     }  
  55.   
  56.     pluginState = IsNotAPlugin; // be pessimistic  
  57.   
  58.     // 如果获取 元信息 成功,再看版本号是否符合要求,如果版本号合适,则将pluginState设置为IsAPlugin,代表是Qt插件  
  59.     uint qt_version = (uint)metaData.value(QLatin1String("version")).toDouble();  
  60.     bool debug = metaData.value(QLatin1String("debug")).toBool();  
  61.     if ((qt_version & 0x00ff00) > (QT_VERSION & 0x00ff00) || (qt_version & 0xff0000) != (QT_VERSION & 0xff0000)) {  
  62.         if (qt_debug_component()) {  
  63.             qWarning("In %s:\n"  
  64.                  "  Plugin uses incompatible Qt library (%d.%d.%d) [%s]",  
  65.                  (const char*) QFile::encodeName(fileName),  
  66.                  (qt_version&0xff0000) >> 16, (qt_version&0xff00) >> 8, qt_version&0xff,  
  67.                  debug ? "debug" : "release");  
  68.         }  
  69.         errorString = QLibrary::tr("The plugin '%1' uses incompatible Qt library. (%2.%3.%4) [%5]")  
  70.             .arg(fileName)  
  71.             .arg((qt_version&0xff0000) >> 16)  
  72.             .arg((qt_version&0xff00) >> 8)  
  73.             .arg(qt_version&0xff)  
  74.             .arg(debug ? QLatin1String("debug") : QLatin1String("release"));  
  75. #ifndef QT_NO_DEBUG_PLUGIN_CHECK  
  76.     } else if(debug != QLIBRARY_AS_DEBUG) {  
  77.         //don't issue a qWarning since we will hopefully find a non-debug? --Sam  
  78.         errorString = QLibrary::tr("The plugin '%1' uses incompatible Qt library."  
  79.                  " (Cannot mix debug and release libraries.)").arg(fileName);  
  80. #endif  
  81.     } else {  
  82.         pluginState = IsAPlugin;  
  83.     }  

你可能感兴趣的:(Qt5的插件机制(4)--Qt插件的元信息metaData)