1 | // Copyright (C) 2024 The Qt Company Ltd. |
---|---|
2 | // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only |
3 | |
4 | #include "qgstreamer_qiodevice_handler_p.h" |
5 | |
6 | #include <QtCore/qdebug.h> |
7 | #include <QtCore/qglobal.h> |
8 | #include <QtCore/qiodevice.h> |
9 | #include <QtCore/qmap.h> |
10 | #include <QtCore/qmutex.h> |
11 | #include <QtCore/qobject.h> |
12 | #include <QtCore/qspan.h> |
13 | #include <QtCore/qurl.h> |
14 | #include <QtCore/quuid.h> |
15 | |
16 | #include <gst/base/gstbasesrc.h> |
17 | #include <map> |
18 | #include <memory> |
19 | #include <utility> |
20 | |
21 | QT_BEGIN_NAMESPACE |
22 | |
23 | namespace { |
24 | |
25 | using namespace Qt::Literals; |
26 | |
27 | // QIODeviceRegistry |
28 | |
29 | class QIODeviceRegistry : public QObject |
30 | { |
31 | public: |
32 | struct Record |
33 | { |
34 | explicit Record(QByteArray, QIODevice *); |
35 | |
36 | void unsetDevice(); |
37 | bool isValid() const; |
38 | |
39 | const QByteArray id; |
40 | |
41 | template <typename Functor> |
42 | auto runWhileLocked(Functor &&f) |
43 | { |
44 | QMutexLocker guard(&mutex); |
45 | return f(device); |
46 | } |
47 | |
48 | private: |
49 | QIODevice *device; |
50 | mutable QMutex mutex; |
51 | }; |
52 | using SharedRecord = std::shared_ptr<Record>; |
53 | |
54 | QByteArray registerQIODevice(QIODevice *); |
55 | SharedRecord findRecord(QByteArrayView); |
56 | |
57 | private: |
58 | void unregisterDevice(QIODevice *); |
59 | void deviceDestroyed(QIODevice *); |
60 | |
61 | QMutex m_registryMutex; |
62 | std::map<QByteArray, SharedRecord, std::less<>> m_registry; |
63 | QMap<QIODevice *, QByteArray> m_reverseLookupTable; |
64 | }; |
65 | |
66 | QByteArray QIODeviceRegistry::registerQIODevice(QIODevice *device) |
67 | { |
68 | Q_ASSERT(device); |
69 | |
70 | if (device->isSequential()) |
71 | qWarning() << "GStreamer: sequential QIODevices are not fully supported"; |
72 | |
73 | QMutexLocker lock(&m_registryMutex); |
74 | |
75 | auto it = m_reverseLookupTable.find(key: device); |
76 | if (it != m_reverseLookupTable.end()) |
77 | return it.value(); |
78 | |
79 | QByteArray identifier = |
80 | "qiodevice:/"_ba+ QUuid::createUuid().toByteArray(mode: QUuid::StringFormat::Id128); |
81 | |
82 | m_registry.emplace(args&: identifier, args: std::make_shared<Record>(args&: identifier, args&: device)); |
83 | |
84 | QMetaObject::Connection destroyedConnection = QObject::connect( |
85 | sender: device, signal: &QObject::destroyed, context: this, |
86 | slot: [this, device] { |
87 | // Caveat: if the QIODevice has not been closed, we unregister the device, however gstreamer |
88 | // worker threads have a chance to briefly read from a partially destroyed QIODevice. |
89 | // There's nothing we can do about it |
90 | unregisterDevice(device); |
91 | }, |
92 | type: Qt::DirectConnection); |
93 | |
94 | QObject::connect( |
95 | sender: device, signal: &QIODevice::aboutToClose, context: this, |
96 | slot: [this, device, destroyedConnection = std::move(destroyedConnection)] { |
97 | unregisterDevice(device); |
98 | disconnect(destroyedConnection); |
99 | }, |
100 | type: Qt::DirectConnection); |
101 | |
102 | m_reverseLookupTable.insert(key: device, value: identifier); |
103 | return identifier; |
104 | } |
105 | |
106 | QIODeviceRegistry::SharedRecord QIODeviceRegistry::findRecord(QByteArrayView id) |
107 | { |
108 | QMutexLocker registryLock(&m_registryMutex); |
109 | |
110 | auto it = m_registry.find(x: id); |
111 | if (it != m_registry.end()) |
112 | return it->second; |
113 | return {}; |
114 | } |
115 | |
116 | void QIODeviceRegistry::unregisterDevice(QIODevice *device) |
117 | { |
118 | QMutexLocker registryLock(&m_registryMutex); |
119 | auto reverseLookupIt = m_reverseLookupTable.find(key: device); |
120 | if (reverseLookupIt == m_reverseLookupTable.end()) |
121 | return; |
122 | |
123 | auto it = m_registry.find(x: reverseLookupIt.value()); |
124 | Q_ASSERT(it != m_registry.end()); |
125 | |
126 | it->second->unsetDevice(); |
127 | m_reverseLookupTable.erase(it: reverseLookupIt); |
128 | m_registry.erase(position: it); |
129 | } |
130 | |
131 | QIODeviceRegistry::Record::Record(QByteArray id, QIODevice *device) |
132 | : id { |
133 | std::move(id), |
134 | }, |
135 | device { |
136 | device, |
137 | } |
138 | { |
139 | if (!device->isOpen()) |
140 | device->open(mode: QIODevice::ReadOnly); |
141 | } |
142 | |
143 | void QIODeviceRegistry::Record::unsetDevice() |
144 | { |
145 | QMutexLocker lock(&mutex); |
146 | device = nullptr; |
147 | } |
148 | |
149 | bool QIODeviceRegistry::Record::isValid() const |
150 | { |
151 | QMutexLocker lock(&mutex); |
152 | return device; |
153 | } |
154 | |
155 | Q_GLOBAL_STATIC(QIODeviceRegistry, gQIODeviceRegistry); |
156 | |
157 | // qt helpers |
158 | |
159 | // glib / gstreamer object |
160 | enum PropertyId : uint8_t { |
161 | PROP_NONE, |
162 | PROP_URI, |
163 | }; |
164 | |
165 | struct QGstQIODeviceSrc |
166 | { |
167 | void getProperty(guint propId, GValue *value, const GParamSpec *pspec) const; |
168 | void setProperty(guint propId, const GValue *value, const GParamSpec *pspec); |
169 | auto lockObject() const; |
170 | |
171 | bool start(); |
172 | bool stop(); |
173 | |
174 | bool isSeekable(); |
175 | std::optional<guint64> size(); |
176 | GstFlowReturn fill(guint64 offset, guint length, GstBuffer *buf); |
177 | void getURI(GValue *value) const; |
178 | bool setURI(const char *location, GError **err = nullptr); |
179 | |
180 | GstBaseSrc baseSrc; |
181 | QIODeviceRegistry::SharedRecord record; |
182 | }; |
183 | |
184 | void QGstQIODeviceSrc::getProperty(guint propId, GValue *value, const GParamSpec *pspec) const |
185 | { |
186 | switch (propId) { |
187 | case PROP_URI: |
188 | return getURI(value); |
189 | |
190 | default: |
191 | G_OBJECT_WARN_INVALID_PROPERTY_ID(this, propId, pspec); |
192 | break; |
193 | } |
194 | } |
195 | |
196 | void QGstQIODeviceSrc::setProperty(guint propId, const GValue *value, const GParamSpec *pspec) |
197 | { |
198 | switch (propId) { |
199 | case PROP_URI: |
200 | setURI(location: g_value_get_string(value)); |
201 | break; |
202 | default: |
203 | G_OBJECT_WARN_INVALID_PROPERTY_ID(this, propId, pspec); |
204 | break; |
205 | } |
206 | } |
207 | |
208 | auto QGstQIODeviceSrc::lockObject() const |
209 | { |
210 | GST_OBJECT_LOCK(this); |
211 | return qScopeGuard(f: [this] { |
212 | GST_OBJECT_UNLOCK(this); |
213 | }); |
214 | } |
215 | |
216 | bool QGstQIODeviceSrc::start() |
217 | { |
218 | auto lock = lockObject(); |
219 | |
220 | return record && record->isValid(); |
221 | } |
222 | |
223 | bool QGstQIODeviceSrc::stop() |
224 | { |
225 | return true; |
226 | } |
227 | |
228 | bool QGstQIODeviceSrc::isSeekable() |
229 | { |
230 | auto lock = lockObject(); |
231 | return record->runWhileLocked(f: [&](QIODevice *device) { |
232 | return !device->isSequential(); |
233 | }); |
234 | } |
235 | |
236 | std::optional<guint64> QGstQIODeviceSrc::size() |
237 | { |
238 | auto lock = lockObject(); |
239 | if (!record) |
240 | return std::nullopt; |
241 | |
242 | qint64 size = record->runWhileLocked(f: [&](QIODevice *device) { |
243 | return device->size(); |
244 | }); |
245 | |
246 | if (size == -1) |
247 | return std::nullopt; |
248 | return size; |
249 | } |
250 | |
251 | GstFlowReturn QGstQIODeviceSrc::fill(guint64 offset, guint length, GstBuffer *buf) |
252 | { |
253 | auto lock = lockObject(); |
254 | |
255 | if (!record) |
256 | return GST_FLOW_ERROR; |
257 | |
258 | GstMapInfo info; |
259 | if (!gst_buffer_map(buffer: buf, info: &info, flags: GST_MAP_WRITE)) { |
260 | GST_ELEMENT_ERROR(this, RESOURCE, WRITE, (nullptr), ("Can't map buffer for writing")); |
261 | return GST_FLOW_ERROR; |
262 | } |
263 | |
264 | int64_t totalRead = 0; |
265 | GstFlowReturn ret = record->runWhileLocked(f: [&](QIODevice *device) -> GstFlowReturn { |
266 | if (device->pos() != qint64(offset)) { |
267 | bool success = device->seek(pos: offset); |
268 | if (!success) { |
269 | qWarning() << "seek on iodevice failed"; |
270 | return GST_FLOW_ERROR; |
271 | } |
272 | } |
273 | |
274 | int64_t remain = length; |
275 | while (remain) { |
276 | int64_t bytesRead = |
277 | device->read(data: reinterpret_cast<char *>(info.data + totalRead), maxlen: remain); |
278 | if (bytesRead == -1) { |
279 | if (device->atEnd()) { |
280 | return GST_FLOW_EOS; |
281 | } |
282 | GST_ELEMENT_ERROR(this, RESOURCE, READ, (nullptr), GST_ERROR_SYSTEM); |
283 | return GST_FLOW_ERROR; |
284 | } |
285 | |
286 | remain -= bytesRead; |
287 | totalRead += bytesRead; |
288 | } |
289 | |
290 | return GST_FLOW_OK; |
291 | }); |
292 | |
293 | if (ret != GST_FLOW_OK) { |
294 | gst_buffer_unmap(buffer: buf, info: &info); |
295 | gst_buffer_resize(buffer: buf, offset: 0, size: 0); |
296 | return ret; |
297 | } |
298 | |
299 | gst_buffer_unmap(buffer: buf, info: &info); |
300 | if (totalRead != length) |
301 | gst_buffer_resize(buffer: buf, offset: 0, size: totalRead); |
302 | |
303 | GST_BUFFER_OFFSET(buf) = offset; |
304 | GST_BUFFER_OFFSET_END(buf) = offset + totalRead; |
305 | |
306 | return GST_FLOW_OK; |
307 | } |
308 | |
309 | void QGstQIODeviceSrc::getURI(GValue *value) const |
310 | { |
311 | auto lock = lockObject(); |
312 | if (record) |
313 | g_value_set_string(value, v_string: record->id); |
314 | else |
315 | g_value_set_string(value, v_string: nullptr); |
316 | } |
317 | |
318 | bool QGstQIODeviceSrc::setURI(const char *location, GError **err) |
319 | { |
320 | Q_ASSERT(QLatin1StringView(location).startsWith("qiodevice:/"_L1)); |
321 | |
322 | { |
323 | auto lock = lockObject(); |
324 | GstState state = GST_STATE(this); |
325 | if (state != GST_STATE_READY && state != GST_STATE_NULL) { |
326 | g_warning( |
327 | "Changing the `uri' property on qiodevicesrc when the resource is open is not " |
328 | "supported."); |
329 | if (err) |
330 | g_set_error(err, GST_URI_ERROR, code: GST_URI_ERROR_BAD_STATE, |
331 | format: "Changing the `uri' property on qiodevicesrc when the resource is open " |
332 | "is not supported."); |
333 | return false; |
334 | } |
335 | } |
336 | |
337 | auto newRecord = gQIODeviceRegistry->findRecord(id: QByteArrayView{ location }); |
338 | |
339 | { |
340 | auto lock = lockObject(); |
341 | record = std::move(newRecord); |
342 | } |
343 | |
344 | g_object_notify(G_OBJECT(this), property_name: "uri"); |
345 | |
346 | return true; |
347 | } |
348 | |
349 | struct QGstQIODeviceSrcClass |
350 | { |
351 | GstBaseSrcClass parent_class; |
352 | }; |
353 | |
354 | // GObject |
355 | static void gst_qiodevice_src_class_init(QGstQIODeviceSrcClass *klass); |
356 | static void gst_qiodevice_src_init(QGstQIODeviceSrc *self); |
357 | |
358 | GType gst_qiodevice_src_get_type(); |
359 | |
360 | template <typename T> |
361 | QGstQIODeviceSrc *asQGstQIODeviceSrc(T *obj) |
362 | { |
363 | return (G_TYPE_CHECK_INSTANCE_CAST((obj), gst_qiodevice_src_get_type(), QGstQIODeviceSrc)); |
364 | } |
365 | |
366 | // URI handler |
367 | void qGstInitQIODeviceURIHandler(gpointer, gpointer); |
368 | |
369 | #define gst_qiodevice_src_parent_class parent_class |
370 | G_DEFINE_TYPE_WITH_CODE(QGstQIODeviceSrc, gst_qiodevice_src, GST_TYPE_BASE_SRC, |
371 | G_IMPLEMENT_INTERFACE(GST_TYPE_URI_HANDLER, qGstInitQIODeviceURIHandler)); |
372 | |
373 | // implementations |
374 | |
375 | void gst_qiodevice_src_init(QGstQIODeviceSrc *self) |
376 | { |
377 | using SharedRecord = QIODeviceRegistry::SharedRecord; |
378 | |
379 | new (reinterpret_cast<void *>(&self->record)) SharedRecord; |
380 | |
381 | static constexpr guint defaultBlockSize = 16384; |
382 | gst_base_src_set_blocksize(src: &self->baseSrc, blocksize: defaultBlockSize); |
383 | } |
384 | |
385 | void gst_qiodevice_src_class_init(QGstQIODeviceSrcClass *klass) |
386 | { |
387 | // GObject |
388 | GObjectClass *gobjectClass = G_OBJECT_CLASS(klass); |
389 | gobjectClass->set_property = [](GObject *instance, guint propId, const GValue *value, |
390 | GParamSpec *pspec) { |
391 | QGstQIODeviceSrc *src = asQGstQIODeviceSrc(obj: instance); |
392 | return src->setProperty(propId, value, pspec); |
393 | }; |
394 | |
395 | gobjectClass->get_property = [](GObject *instance, guint propId, GValue *value, |
396 | GParamSpec *pspec) { |
397 | QGstQIODeviceSrc *src = asQGstQIODeviceSrc(obj: instance); |
398 | return src->getProperty(propId, value, pspec); |
399 | }; |
400 | |
401 | g_object_class_install_property( |
402 | oclass: gobjectClass, property_id: PROP_URI, |
403 | pspec: g_param_spec_string(name: "uri", nick: "QRC Location", blurb: "Path of the qrc to read", default_value: nullptr, |
404 | flags: GParamFlags(G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS |
405 | | GST_PARAM_MUTABLE_READY))); |
406 | |
407 | gobjectClass->finalize = [](GObject *instance) { |
408 | QGstQIODeviceSrc *src = asQGstQIODeviceSrc(obj: instance); |
409 | using SharedRecord = QIODeviceRegistry::SharedRecord; |
410 | src->record.~SharedRecord(); |
411 | G_OBJECT_CLASS(parent_class)->finalize(instance); |
412 | }; |
413 | |
414 | // GstElement |
415 | GstElementClass *gstelementClass = GST_ELEMENT_CLASS(klass); |
416 | gst_element_class_set_static_metadata(klass: gstelementClass, longname: "QRC Source", classification: "Source/QRC", |
417 | description: "Read from arbitrary point in QRC resource", |
418 | author: "Tim Blechmann <tim.blechmann@qt.io>"); |
419 | |
420 | static GstStaticPadTemplate srctemplate = |
421 | GST_STATIC_PAD_TEMPLATE("src", GST_PAD_SRC, GST_PAD_ALWAYS, GST_STATIC_CAPS_ANY); |
422 | |
423 | gst_element_class_add_static_pad_template(klass: gstelementClass, static_templ: &srctemplate); |
424 | |
425 | // GstBaseSrc |
426 | GstBaseSrcClass *gstbasesrcClass = GST_BASE_SRC_CLASS(klass); |
427 | gstbasesrcClass->start = [](GstBaseSrc *basesrc) -> gboolean { |
428 | QGstQIODeviceSrc *src = asQGstQIODeviceSrc(obj: basesrc); |
429 | return src->start(); |
430 | }; |
431 | gstbasesrcClass->stop = [](GstBaseSrc *basesrc) -> gboolean { |
432 | QGstQIODeviceSrc *src = asQGstQIODeviceSrc(obj: basesrc); |
433 | return src->stop(); |
434 | }; |
435 | |
436 | gstbasesrcClass->is_seekable = [](GstBaseSrc *basesrc) -> gboolean { |
437 | QGstQIODeviceSrc *src = asQGstQIODeviceSrc(obj: basesrc); |
438 | return src->isSeekable(); |
439 | }; |
440 | gstbasesrcClass->get_size = [](GstBaseSrc *basesrc, guint64 *size) -> gboolean { |
441 | QGstQIODeviceSrc *src = asQGstQIODeviceSrc(obj: basesrc); |
442 | auto optionalSize = src->size(); |
443 | if (!optionalSize) |
444 | return false; |
445 | *size = optionalSize.value(); |
446 | return true; |
447 | }; |
448 | gstbasesrcClass->fill = [](GstBaseSrc *basesrc, guint64 offset, guint length, |
449 | GstBuffer *buf) -> GstFlowReturn { |
450 | QGstQIODeviceSrc *src = asQGstQIODeviceSrc(obj: basesrc); |
451 | return src->fill(offset, length, buf); |
452 | }; |
453 | } |
454 | |
455 | void qGstInitQIODeviceURIHandler(gpointer g_handlerInterface, gpointer) |
456 | { |
457 | GstURIHandlerInterface *iface = (GstURIHandlerInterface *)g_handlerInterface; |
458 | |
459 | iface->get_type = [](GType) { |
460 | return GST_URI_SRC; |
461 | }; |
462 | iface->get_protocols = [](GType) { |
463 | static constexpr const gchar *protocols[] = { |
464 | "qiodevice", |
465 | nullptr, |
466 | }; |
467 | return protocols; |
468 | }; |
469 | iface->get_uri = [](GstURIHandler *handler) -> gchar * { |
470 | QGstQIODeviceSrc *src = asQGstQIODeviceSrc(obj: handler); |
471 | auto lock = src->lockObject(); |
472 | if (src->record) |
473 | return g_strdup(str: src->record->id.constData()); |
474 | return nullptr; |
475 | }; |
476 | iface->set_uri = [](GstURIHandler *handler, const gchar *uri, GError **err) -> gboolean { |
477 | QGstQIODeviceSrc *src = asQGstQIODeviceSrc(obj: handler); |
478 | return src->setURI(location: uri, err); |
479 | }; |
480 | } |
481 | |
482 | } // namespace |
483 | |
484 | // plugin registration |
485 | |
486 | void qGstRegisterQIODeviceHandler(GstPlugin *plugin) |
487 | { |
488 | gst_element_register(plugin, name: "qiodevicesrc", rank: GST_RANK_PRIMARY, type: gst_qiodevice_src_get_type()); |
489 | } |
490 | |
491 | QUrl qGstRegisterQIODevice(QIODevice *device) |
492 | { |
493 | return QUrl{ |
494 | QString::fromLatin1(ba: gQIODeviceRegistry->registerQIODevice(device)), |
495 | }; |
496 | } |
497 | |
498 | QT_END_NAMESPACE |
499 |
Definitions
- QIODeviceRegistry
- Record
- runWhileLocked
- registerQIODevice
- findRecord
- unregisterDevice
- Record
- unsetDevice
- isValid
- gQIODeviceRegistry
- PropertyId
- QGstQIODeviceSrc
- getProperty
- setProperty
- lockObject
- start
- stop
- isSeekable
- size
- fill
- getURI
- setURI
- QGstQIODeviceSrcClass
- asQGstQIODeviceSrc
- parent_class
- gst_qiodevice_src_init
- gst_qiodevice_src_class_init
- qGstInitQIODeviceURIHandler
- qGstRegisterQIODeviceHandler
Learn Advanced QML with KDAB
Find out more