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 | |