Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qpipewire_audiodevicemonitor_p.h
Go to the documentation of this file.
1// Copyright (C) 2025 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#ifndef QPIPEWIRE_AUDIODEVICEMONITOR_P_H
5#define QPIPEWIRE_AUDIODEVICEMONITOR_P_H
6
7//
8// W A R N I N G
9// -------------
10//
11// This file is not part of the Qt API. It exists purely as an
12// implementation detail. This header file may change from version to
13// version without notice, or even be removed.
14//
15// We mean it.
16//
17
22
23#include <QtCore/qfuture.h>
24#include <QtCore/qtimer.h>
25#include <QtCore/qreadwritelock.h>
26#include <QtMultimedia/qaudiodevice.h>
27
28#include <list>
29#include <vector>
30
31#include <pipewire/pipewire.h>
32
33QT_BEGIN_NAMESPACE
34
35namespace QtPipeWire {
36
38{
40public:
42 ObjectSerial serial() const;
43
46
47private:
48 const ObjectSerial m_observedSerial;
49};
50
52
53// TODO: can we make use of COW here?
55{
57
58 enum class Direction : uint8_t
59 {
60 sink,
61 source,
62 };
63
64public:
66 void objectAdded(ObjectId, uint32_t permissions, PipewireRegistryType, uint32_t version,
67 const spa_dict &props);
69
71 {
72 };
76
77 std::optional<ObjectSerial> findSinkNodeSerial(std::string_view nodeName) const;
78 std::optional<ObjectSerial> findSourceNodeSerial(std::string_view nodeName) const;
79
80 // ObjectId/ObjectSerial mapping
83
86
87 // Obtaining device lists
93 DeviceLists getDeviceLists(bool verifyThreading = true);
94
98
99private:
100 struct DeviceRecord
101 {
102 ObjectSerial serial;
103 PwPropertyDict properties;
104 };
105
106 struct PendingNodeRecord
107 {
108 PendingNodeRecord(ObjectId, ObjectSerial serial, ObjectSerial deviceSerial,
109 PwPropertyDict properties);
110
111 ObjectSerial serial;
112 ObjectSerial deviceSerial;
113 PwPropertyDict properties;
114 std::unique_ptr<NodeEventListener> enumFormatListener;
115 std::unique_ptr<CoreEventDoneListener> enumFormatDoneListener;
116 QFuture<std::optional<SpaObjectAudioFormat>> formatFuture;
117 };
118
119 struct NodeRecord
120 {
121 ObjectSerial serial;
122 ObjectSerial deviceSerial;
123 PwPropertyDict properties;
125 };
126
127 // discovered, but format not resolved
128 struct PendingRecords
129 {
130 std::list<PendingNodeRecord> m_sources;
131 std::list<PendingNodeRecord> m_sinks;
132 std::vector<ObjectSerial> m_removals;
133
134 std::optional<std::variant<QByteArray, NoDefaultDeviceType>> m_defaultSource;
135 std::optional<std::variant<QByteArray, NoDefaultDeviceType>> m_defaultSink;
136
137 void removeRecordsForObject(ObjectSerial);
138 };
139 QMutex m_pendingRecordsMutex;
140 PendingRecords m_pendingRecords QT_MM_GUARDED_BY(m_pendingRecordsMutex);
141
142 // discovered, format resolved. living on application thread
143 mutable QReadWriteLock m_mutex;
144 std::map<ObjectSerial, DeviceRecord> m_devices QT_MM_GUARDED_BY(m_mutex);
145 std::vector<NodeRecord> m_sources QT_MM_GUARDED_BY(m_mutex);
146 std::vector<NodeRecord> m_sinks QT_MM_GUARDED_BY(m_mutex);
147 std::optional<QByteArray> m_defaultSourceName;
148 std::optional<QByteArray> m_defaultSinkName;
149
150 QTimer m_compressionTimer;
151 void startCompressionTimer();
152
153 // Device list updates
154 void audioDevicesChanged(bool verifyThreading = true);
155 void updateSources(std::list<PendingNodeRecord> addedNodes,
156 QSpan<const ObjectSerial> removedObjects);
157 void updateSinks(std::list<PendingNodeRecord> addedNodes,
158 QSpan<const ObjectSerial> removedObjects);
159 template <Direction>
160 void updateSourcesOrSinks(std::list<PendingNodeRecord>, QSpan<const ObjectSerial>);
161
162 QList<QAudioDevice> m_sourceDeviceList;
163 QList<QAudioDevice> m_sinkDeviceList;
164
165 std::optional<ObjectSerial> findDeviceSerial(std::string_view deviceName) const;
166
167 template <Direction>
168 std::optional<ObjectSerial> findNodeSerialForNodeName(std::string_view nodeName) const;
169
170 // ObjectId/ObjectSerial mapping
171 // CHECK: can we completely rely on only accessing ObjectId under the pipewire event loop lock?
172 mutable QReadWriteLock m_objectDictMutex;
173 std::map<ObjectId, ObjectSerial> m_objectSerialDict QT_MM_GUARDED_BY(m_objectDictMutex);
174 std::map<ObjectSerial, ObjectId> m_serialObjectDict QT_MM_GUARDED_BY(m_objectDictMutex);
175
176 std::vector<SharedObjectRemoveObserver>
177 m_objectRemoveObserver QT_MM_GUARDED_BY(m_objectDictMutex);
178};
179
180} // namespace QtPipeWire
181
182QT_END_NAMESPACE
183
184#endif // QPIPEWIRE_AUDIODEVICEMONITOR_P_H
void registerStreamReference(std::shared_ptr< QPipewireAudioStream >)
void unregisterStreamReference(const std::shared_ptr< QPipewireAudioStream > &)
const PwCoreConnectionHandle & coreConnection() const
void audioSourcesChanged(QList< QAudioDevice >)
void unregisterObserver(const SharedObjectRemoveObserver &)
std::optional< ObjectSerial > findSourceNodeSerial(std::string_view nodeName) const
void setDefaultAudioSink(std::variant< QByteArray, NoDefaultDeviceType >)
DeviceLists getDeviceLists(bool verifyThreading=true)
void setDefaultAudioSource(std::variant< QByteArray, NoDefaultDeviceType >)
std::optional< ObjectId > findObjectId(ObjectSerial)
std::optional< ObjectSerial > findSinkNodeSerial(std::string_view nodeName) const
static constexpr NoDefaultDeviceType NoDefaultDevice
void objectAdded(ObjectId, uint32_t permissions, PipewireRegistryType, uint32_t version, const spa_dict &props)
std::optional< ObjectSerial > findObjectSerial(ObjectId)
bool registerObserver(SharedObjectRemoveObserver)
static std::atomic_int s_sequenceNumberAllocator
std::unique_ptr< pw_core, PwCoreConnectionDeleter > PwCoreConnectionHandle
StrongIdType< uint32_t, ObjectIdTag > ObjectId
StrongIdType< uint64_t, ObjectSerialTag > ObjectSerial
std::shared_ptr< ObjectRemoveObserver > SharedObjectRemoveObserver
std::error_code make_error_code(int errnoValue=errno)
bool pw_check_library_version(int major, int minor, int micro)
QDebug operator<<(QDebug dbg, const pw_time &state)
QDebug operator<<(QDebug dbg, const spa_dict &dict)
QDebug operator<<(QDebug dbg, enum pw_stream_state)
#define QT_MM_GUARDED_BY(Mutex)
QDebug operator<<(QDebug dbg, const spa_pod &pod)
q23::expected< void, int > asyncWait(pw_core *coreConnection, std::function< void()> handler)
q23::expected< bool, int > sync(pw_core *coreConnection, std::optional< std::chrono::nanoseconds > timeout={})
void operator()(Type *handle) const
NodeEventListener(PwNodeHandle, NodeHandler)
std::function< void(const struct pw_node_info *)> InfoHandler
std::function< void(int, uint32_t, uint32_t, uint32_t, const struct spa_pod *)> ParamHandler
void operator()(pw_core *handle) const
friend QDebug operator<<(QDebug dbg, const StrongIdType &self)
friend bool comparesEqual(const StrongIdType &lhs, const StrongIdType &rhs) noexcept
friend Qt::strong_ordering compareThreeWay(const StrongIdType &lhs, const StrongIdType &rhs) noexcept