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
qandroidaudiodevices.cpp
Go to the documentation of this file.
1// Copyright (C) 2021 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 <QtMultimedia/private/qandroidaudiodevices_p.h>
5
6#include <QtMultimedia/private/qandroidaudiodevice_p.h>
7#include <QtMultimedia/private/qandroidaudiojnitypes_p.h>
8#include <QtMultimedia/private/qandroidaudiosink_p.h>
9#include <QtMultimedia/private/qandroidaudiosource_p.h>
10
11#include <QtMultimedia/private/qplatformmediaintegration_p.h>
12
13#include <QtCore/qjniobject.h>
14
15QT_BEGIN_NAMESPACE
16
17using namespace QtJniTypes;
18
19namespace {
20
21QAudioFormat preferredFormatForDevice(const QtJniTypes::AudioDeviceInfo &deviceInfo)
22{
23 QAudioFormat preferredFormat;
24
25 // Set preferred channel count based on what device reports, with default set to stereo (2)
26 QJniArray<jint> channelCounts = deviceInfo.callMethod<QJniArray<jint>>("getChannelCounts");
27 if (channelCounts.isEmpty()) {
28 preferredFormat.setChannelConfig(QAudioFormat::ChannelConfigStereo);
29 } else {
30 const auto [minIt, maxIt] = std::minmax_element(channelCounts.begin(), channelCounts.end());
31 const int channelCount = std::clamp(2, *minIt, *maxIt);
32 preferredFormat.setChannelConfig(
33 QAudioFormat::defaultChannelConfigForChannelCount(channelCount));
34 }
35
36 // Get optimal sample rate from AudioManager
37 preferredFormat.setSampleRate(
38 QtAudioDeviceManager::callStaticMethod<jint>("getDefaultSampleRate"));
39
40 // Using Float avoids conversions for processing, so we should prefer that instead of whatever
41 // the device uses natively
42 preferredFormat.setSampleFormat(QAudioFormat::Float);
43
44 return preferredFormat;
45};
46
47QList<QAudioDevice> availableDevices(QAudioDevice::Mode mode)
48{
49 if (mode == QAudioDevice::Null)
50 return {};
51
52 QList<QAudioDevice> devices;
53 const char *getMethod =
54 mode == QAudioDevice::Input ? "getAudioInputDevices" : "getAudioOutputDevices";
55 auto deviceInfos =
56 QtAudioDeviceManager::callStaticMethod<QJniArray<AudioDeviceInfo>>(getMethod);
57
58 if (!deviceInfos.isValid())
59 return {};
60
61 for (int i = 0; i < deviceInfos.size(); ++i) {
62 AudioDeviceInfo deviceInfo = deviceInfos.at(i);
63 int id = deviceInfo.callMethod<jint>("getId");
64 jint deviceType = deviceInfo.callMethod<jint>("getType");
65 auto description = QtAudioDeviceManager::callStaticMethod<QString>(
66 "audioDeviceTypeToString", deviceType);
67 bool isBluetoothDevice =
68 QtAudioDeviceManager::callStaticMethod<jboolean>("isBluetoothDevice", deviceInfo);
69 devices << QAudioDevicePrivate::createQAudioDevice(std::make_unique<QAndroidAudioDevice>(
70 QString::number(id).toUtf8(), description, mode,
71 preferredFormatForDevice(deviceInfo), isBluetoothDevice, i == 0));
72 }
73
74 return devices;
75}
76
77} // namespace
78
79QAndroidAudioDevices::QAndroidAudioDevices() : QPlatformAudioDevices()
80{
81 QtAudioDeviceManager::callStaticMethod<void>("registerAudioHeadsetStateReceiver");
82}
83
84QAndroidAudioDevices::~QAndroidAudioDevices()
85{
86 // Object of QAndroidAudioDevices type is static. Unregistering will happend only when closing
87 // the application. In such case it is probably not needed, but let's leave it for
88 // compatibility with Android documentation
89 QtAudioDeviceManager::callStaticMethod<void>("unregisterAudioHeadsetStateReceiver");
90}
91
92QList<QAudioDevice> QAndroidAudioDevices::findAudioInputs() const
93{
94 return availableDevices(QAudioDevice::Input);
95}
96
97QList<QAudioDevice> QAndroidAudioDevices::findAudioOutputs() const
98{
99 return availableDevices(QAudioDevice::Output);
100}
101
102QPlatformAudioSource *QAndroidAudioDevices::createAudioSource(const QAudioDevice &deviceInfo,
103 const QAudioFormat &fmt,
104 QObject *parent)
105{
106 return new QtAAudio::QAndroidAudioSource(deviceInfo, fmt, parent);
107}
108
109QPlatformAudioSink *QAndroidAudioDevices::createAudioSink(const QAudioDevice &deviceInfo,
110 const QAudioFormat &fmt, QObject *parent)
111{
112 return new QtAAudio::QAndroidAudioSink(deviceInfo, fmt, parent);
113}
114
115static void onAudioInputDevicesUpdated(JNIEnv * /*env*/, jobject /*thiz*/)
116{
117 static_cast<QAndroidAudioDevices *>(QPlatformMediaIntegration::instance()->audioDevices())
118 ->onAudioInputsChanged();
119}
120Q_DECLARE_JNI_NATIVE_METHOD(onAudioInputDevicesUpdated)
121
122static void onAudioOutputDevicesUpdated(JNIEnv * /*env*/, jobject /*thiz*/)
123{
124 static_cast<QAndroidAudioDevices *>(QPlatformMediaIntegration::instance()->audioDevices())
125 ->onAudioOutputsChanged();
126}
127Q_DECLARE_JNI_NATIVE_METHOD(onAudioOutputDevicesUpdated)
128
129bool QAndroidAudioDevices::registerNativeMethods()
130{
131 static const bool registered = []{
132 const auto context = QNativeInterface::QAndroidApplication::context();
133 QtAudioDeviceManager::callStaticMethod<void>("setContext", context);
134
135 return QtJniTypes::QtAudioDeviceManager::registerNativeMethods({
136 Q_JNI_NATIVE_METHOD(onAudioInputDevicesUpdated),
137 Q_JNI_NATIVE_METHOD(onAudioOutputDevicesUpdated),
138 });
139 }();
140 return registered;
141}
142
143QT_END_NAMESPACE
144
145extern "C" Q_DECL_EXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void * /*reserved*/)
146{
147 static bool initialized = false;
148 if (initialized)
149 return JNI_VERSION_1_6;
150 initialized = true;
151
152 QT_USE_NAMESPACE
153 typedef union {
154 JNIEnv *nativeEnvironment;
155 void *venv;
156 } UnionJNIEnvToVoid;
157
158 UnionJNIEnvToVoid uenv;
159 uenv.venv = NULL;
160
161 if (vm->GetEnv(&uenv.venv, JNI_VERSION_1_6) != JNI_OK)
162 return JNI_ERR;
163
164 if (!QAndroidAudioDevices::registerNativeMethods())
165 return JNI_ERR;
166
167 return JNI_VERSION_1_6;
168}
static void onAudioInputDevicesUpdated(JNIEnv *, jobject)
static void onAudioOutputDevicesUpdated(JNIEnv *, jobject)