| 1 | // Copyright (C) 2020 The Qt Company Ltd. |
| 2 | // Copyright (C) 2021 Intel Corporation. |
| 3 | // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only |
| 4 | |
| 5 | #ifndef QPLUGIN_H |
| 6 | #define QPLUGIN_H |
| 7 | |
| 8 | #if 0 |
| 9 | #pragma qt_class(QtPlugin) |
| 10 | #endif |
| 11 | |
| 12 | #include <QtCore/qobject.h> |
| 13 | #include <QtCore/qpointer.h> |
| 14 | #include <QtCore/qjsonobject.h> |
| 15 | |
| 16 | #include <QtCore/q20algorithm.h> |
| 17 | |
| 18 | QT_BEGIN_NAMESPACE |
| 19 | |
| 20 | // Used up to Qt 6.2 |
| 21 | inline constexpr unsigned char qPluginArchRequirements() |
| 22 | { |
| 23 | return 0 |
| 24 | #ifndef QT_NO_DEBUG |
| 25 | | 1 |
| 26 | #endif |
| 27 | #ifdef __AVX2__ |
| 28 | | 2 |
| 29 | # ifdef __AVX512F__ |
| 30 | | 4 |
| 31 | # endif |
| 32 | #endif |
| 33 | ; |
| 34 | } |
| 35 | |
| 36 | typedef QObject *(*QtPluginInstanceFunction)(); |
| 37 | struct QPluginMetaData |
| 38 | { |
| 39 | static constexpr quint8 CurrentMetaDataVersion = 1; |
| 40 | static constexpr char MagicString[] = { |
| 41 | 'Q', 'T', 'M', 'E', 'T', 'A', 'D', 'A', 'T', 'A', ' ', '!' |
| 42 | }; |
| 43 | |
| 44 | template <size_t OSize, typename OO, size_t ISize, typename II> |
| 45 | static constexpr void copy(OO (&out)[OSize], II (&in)[ISize]) |
| 46 | { |
| 47 | static_assert(OSize <= ISize, "Output would not be fully initialized" ); |
| 48 | q20::copy_n(in, OSize, out); |
| 49 | } |
| 50 | |
| 51 | static constexpr quint8 archRequirements() |
| 52 | { |
| 53 | quint8 v = 0; |
| 54 | #if defined(__AVX512F__) |
| 55 | v = 4; // x86-64-v4: AVX512F, AVX512BW, AVX512CD, AVX512DQ and AVX512VL |
| 56 | #elif defined(__AVX__) || defined(__BMI__) || defined(__BMI2__) || defined(__MOVBE__) |
| 57 | v = 3; // x86-64-v3: AVX, AVX2, BMI1, BMI2, F16C, FMA, LZCNT, MOVBE, XSAVE |
| 58 | #elif defined(__SSE3__) |
| 59 | v = 2; // x86-64-v2: POPCNT, SSE3, SSSE3, SSE4.1 and SSE4.2. |
| 60 | #elif defined(__SSE__) || defined(__MMX___) |
| 61 | v = 1; // x86-64 baseline: SSE and SSE2 |
| 62 | #endif |
| 63 | #ifndef QT_NO_DEBUG |
| 64 | v |= 0x80; |
| 65 | #endif |
| 66 | return v; |
| 67 | } |
| 68 | |
| 69 | struct { |
| 70 | quint8 = CurrentMetaDataVersion; |
| 71 | quint8 = QT_VERSION_MAJOR; |
| 72 | quint8 = QT_VERSION_MINOR; |
| 73 | quint8 = archRequirements(); |
| 74 | }; |
| 75 | static_assert(alignof(Header) == 1, "Alignment of header incorrect with this compiler" ); |
| 76 | |
| 77 | struct { |
| 78 | char [sizeof(QPluginMetaData::MagicString)] = {}; |
| 79 | constexpr () { copy(out&: magic, in: QPluginMetaData::MagicString); } |
| 80 | Header = {}; |
| 81 | }; |
| 82 | static_assert(alignof(MagicHeader) == 1, "Alignment of header incorrect with this compiler" ); |
| 83 | |
| 84 | struct { |
| 85 | static constexpr quint32 = 0x74510001; |
| 86 | static constexpr char [] = "qt-project!" ; |
| 87 | |
| 88 | // ELF note header |
| 89 | quint32 = sizeof(name); |
| 90 | quint32 ; |
| 91 | quint32 = NoteType; |
| 92 | char [sizeof(NoteName)] = {}; |
| 93 | |
| 94 | // payload |
| 95 | alignas(void *) // mandatory alignment as per ELF note requirements |
| 96 | Header = {}; |
| 97 | constexpr (quint32 payloadSize) : n_descsz(sizeof(header) + payloadSize) |
| 98 | { QPluginMetaData::copy(out&: name, in: NoteName); } |
| 99 | }; |
| 100 | static_assert(alignof(ElfNoteHeader) == alignof(void*), "Alignment of header incorrect with this compiler" ); |
| 101 | static_assert((sizeof(ElfNoteHeader::name) % 4) == 0, "ELF note name length not a multiple of 4" ); |
| 102 | |
| 103 | const void *data; |
| 104 | size_t size; |
| 105 | }; |
| 106 | typedef QPluginMetaData (*QtPluginMetaDataFunction)(); |
| 107 | |
| 108 | |
| 109 | struct Q_CORE_EXPORT QStaticPlugin |
| 110 | { |
| 111 | public: |
| 112 | constexpr QStaticPlugin(QtPluginInstanceFunction i, QtPluginMetaDataFunction m) |
| 113 | : instance(i), rawMetaDataSize(m().size), rawMetaData(m().data) |
| 114 | {} |
| 115 | QtPluginInstanceFunction instance; |
| 116 | QJsonObject metaData() const; |
| 117 | |
| 118 | private: |
| 119 | qsizetype rawMetaDataSize; |
| 120 | const void *rawMetaData; |
| 121 | friend class QFactoryLoader; |
| 122 | }; |
| 123 | Q_DECLARE_TYPEINFO(QStaticPlugin, Q_PRIMITIVE_TYPE); |
| 124 | |
| 125 | void Q_CORE_EXPORT qRegisterStaticPluginFunction(QStaticPlugin staticPlugin); |
| 126 | |
| 127 | #if defined(Q_OF_ELF) || (defined(Q_OS_WIN) && (defined (Q_CC_GNU) || defined(Q_CC_CLANG))) |
| 128 | # define QT_PLUGIN_METADATA_SECTION \ |
| 129 | __attribute__ ((section (".qtmetadata"))) __attribute__((used)) |
| 130 | #elif defined(Q_OS_DARWIN) |
| 131 | # define QT_PLUGIN_METADATA_SECTION \ |
| 132 | __attribute__ ((section ("__TEXT,qtmetadata"))) __attribute__((used)) |
| 133 | #elif defined(Q_CC_MSVC) |
| 134 | // TODO: Implement section parsing for MSVC |
| 135 | #pragma section(".qtmetadata",read,shared) |
| 136 | # define QT_PLUGIN_METADATA_SECTION \ |
| 137 | __declspec(allocate(".qtmetadata")) |
| 138 | #else |
| 139 | # define QT_PLUGIN_METADATA_SECTION |
| 140 | #endif |
| 141 | |
| 142 | // Since Qt 6.3 |
| 143 | template <auto (&PluginMetaData)> class QPluginMetaDataV2 |
| 144 | { |
| 145 | struct ElfNotePayload : QPluginMetaData::ElfNoteHeader { |
| 146 | static constexpr size_t = offsetof(QPluginMetaData::ElfNoteHeader, header); |
| 147 | quint8 payload[sizeof(PluginMetaData)] = {}; |
| 148 | constexpr ElfNotePayload() : ElfNoteHeader(sizeof(PluginMetaData)) |
| 149 | { QPluginMetaData::copy(payload, PluginMetaData); } |
| 150 | }; |
| 151 | |
| 152 | struct RegularPayload : QPluginMetaData::MagicHeader { |
| 153 | static constexpr size_t = offsetof(QPluginMetaData::MagicHeader, header); |
| 154 | quint8 payload[sizeof(PluginMetaData)] = {}; |
| 155 | constexpr RegularPayload() { QPluginMetaData::copy(payload, PluginMetaData); } |
| 156 | }; |
| 157 | |
| 158 | struct StaticPayload { |
| 159 | static constexpr size_t = 0; |
| 160 | QPluginMetaData::Header = {}; |
| 161 | quint8 payload[sizeof(PluginMetaData)] = {}; |
| 162 | constexpr StaticPayload() { QPluginMetaData::copy(payload, PluginMetaData); } |
| 163 | }; |
| 164 | |
| 165 | #if defined(QT_STATICPLUGIN) |
| 166 | # define QT_PLUGIN_METADATAV2_SECTION |
| 167 | using Payload = StaticPayload; |
| 168 | #elif defined(Q_OF_ELF) |
| 169 | # ifdef Q_CC_CLANG |
| 170 | # define QT_PLUGIN_METADATAV2_SECTION \ |
| 171 | __attribute__((section(".note.qt.metadata"), used, aligned(alignof(void *)), \ |
| 172 | no_sanitize("address"))) |
| 173 | # else |
| 174 | # define QT_PLUGIN_METADATAV2_SECTION \ |
| 175 | __attribute__((section(".note.qt.metadata"), used, aligned(alignof(void *)))) |
| 176 | # endif |
| 177 | using Payload = ElfNotePayload; |
| 178 | #else |
| 179 | # define QT_PLUGIN_METADATAV2_SECTION QT_PLUGIN_METADATA_SECTION |
| 180 | using Payload = RegularPayload; |
| 181 | #endif |
| 182 | |
| 183 | Payload payload = {}; |
| 184 | |
| 185 | public: |
| 186 | operator QPluginMetaData() const |
| 187 | { |
| 188 | Q_ASSERT(reinterpret_cast<const char *>(&payload) + Payload::HeaderOffset == |
| 189 | reinterpret_cast<const char *>(&payload.header)); |
| 190 | return { &payload.header, sizeof(payload) - Payload::HeaderOffset }; |
| 191 | } |
| 192 | }; |
| 193 | |
| 194 | #define Q_IMPORT_PLUGIN(PLUGIN) \ |
| 195 | extern const QT_PREPEND_NAMESPACE(QStaticPlugin) qt_static_plugin_##PLUGIN(); \ |
| 196 | class Static##PLUGIN##PluginInstance{ \ |
| 197 | public: \ |
| 198 | Static##PLUGIN##PluginInstance() { \ |
| 199 | qRegisterStaticPluginFunction(qt_static_plugin_##PLUGIN()); \ |
| 200 | } \ |
| 201 | }; \ |
| 202 | static Static##PLUGIN##PluginInstance static##PLUGIN##Instance; |
| 203 | |
| 204 | #if defined(QT_PLUGIN_RESOURCE_INIT_FUNCTION) |
| 205 | # define QT_PLUGIN_RESOURCE_INIT \ |
| 206 | extern void QT_PLUGIN_RESOURCE_INIT_FUNCTION(); \ |
| 207 | QT_PLUGIN_RESOURCE_INIT_FUNCTION(); |
| 208 | #else |
| 209 | # define QT_PLUGIN_RESOURCE_INIT |
| 210 | #endif |
| 211 | |
| 212 | #define Q_PLUGIN_INSTANCE(IMPLEMENTATION) \ |
| 213 | { \ |
| 214 | static QT_PREPEND_NAMESPACE(QPointer)<QT_PREPEND_NAMESPACE(QObject)> _instance; \ |
| 215 | if (!_instance) { \ |
| 216 | QT_PLUGIN_RESOURCE_INIT \ |
| 217 | _instance = new IMPLEMENTATION; \ |
| 218 | } \ |
| 219 | return _instance; \ |
| 220 | } |
| 221 | |
| 222 | #if defined(QT_STATICPLUGIN) |
| 223 | # define QT_MOC_EXPORT_PLUGIN_COMMON(PLUGINCLASS, MANGLEDNAME) \ |
| 224 | static QT_PREPEND_NAMESPACE(QObject) *qt_plugin_instance_##MANGLEDNAME() \ |
| 225 | Q_PLUGIN_INSTANCE(PLUGINCLASS) \ |
| 226 | const QT_PREPEND_NAMESPACE(QStaticPlugin) qt_static_plugin_##MANGLEDNAME() \ |
| 227 | { return { qt_plugin_instance_##MANGLEDNAME, qt_plugin_query_metadata_##MANGLEDNAME}; } \ |
| 228 | /**/ |
| 229 | |
| 230 | # define QT_MOC_EXPORT_PLUGIN(PLUGINCLASS, PLUGINCLASSNAME) \ |
| 231 | static QPluginMetaData qt_plugin_query_metadata_##PLUGINCLASSNAME() \ |
| 232 | { return { qt_pluginMetaData_##PLUGINCLASSNAME, sizeof qt_pluginMetaData_##PLUGINCLASSNAME }; } \ |
| 233 | QT_MOC_EXPORT_PLUGIN_COMMON(PLUGINCLASS, PLUGINCLASSNAME) |
| 234 | |
| 235 | # define QT_MOC_EXPORT_PLUGIN_V2(PLUGINCLASS, MANGLEDNAME, MD) \ |
| 236 | static QT_PREPEND_NAMESPACE(QPluginMetaData) qt_plugin_query_metadata_##MANGLEDNAME() \ |
| 237 | { static constexpr QPluginMetaDataV2<MD> md{}; return md; } \ |
| 238 | QT_MOC_EXPORT_PLUGIN_COMMON(PLUGINCLASS, MANGLEDNAME) |
| 239 | #else |
| 240 | # define QT_MOC_EXPORT_PLUGIN_COMMON(PLUGINCLASS, MANGLEDNAME) \ |
| 241 | extern "C" Q_DECL_EXPORT QT_PREPEND_NAMESPACE(QObject) *qt_plugin_instance() \ |
| 242 | Q_PLUGIN_INSTANCE(PLUGINCLASS) \ |
| 243 | /**/ |
| 244 | |
| 245 | # define QT_MOC_EXPORT_PLUGIN(PLUGINCLASS, PLUGINCLASSNAME) \ |
| 246 | extern "C" Q_DECL_EXPORT \ |
| 247 | QPluginMetaData qt_plugin_query_metadata() \ |
| 248 | { return { qt_pluginMetaData_##PLUGINCLASSNAME, sizeof qt_pluginMetaData_##PLUGINCLASSNAME }; } \ |
| 249 | QT_MOC_EXPORT_PLUGIN_COMMON(PLUGINCLASS, PLUGINCLASSNAME) |
| 250 | |
| 251 | # define QT_MOC_EXPORT_PLUGIN_V2(PLUGINCLASS, MANGLEDNAME, MD) \ |
| 252 | extern "C" Q_DECL_EXPORT QT_PREPEND_NAMESPACE(QPluginMetaData) qt_plugin_query_metadata_v2()\ |
| 253 | { static constexpr QT_PLUGIN_METADATAV2_SECTION QPluginMetaDataV2<MD> md{}; return md; } \ |
| 254 | QT_MOC_EXPORT_PLUGIN_COMMON(PLUGINCLASS, MANGLEDNAME) |
| 255 | #endif |
| 256 | |
| 257 | #define Q_EXPORT_PLUGIN(PLUGIN) \ |
| 258 | Q_EXPORT_PLUGIN2(PLUGIN, PLUGIN) |
| 259 | # define Q_EXPORT_PLUGIN2(PLUGIN, PLUGINCLASS) \ |
| 260 | static_assert(false, "Old plugin system used") |
| 261 | |
| 262 | # define Q_EXPORT_STATIC_PLUGIN2(PLUGIN, PLUGINCLASS) \ |
| 263 | static_assert(false, "Old plugin system used") |
| 264 | |
| 265 | |
| 266 | QT_END_NAMESPACE |
| 267 | |
| 268 | #endif // Q_PLUGIN_H |
| 269 | |