summaryrefslogtreecommitdiffstats
path: root/src/plugins/multimedia/gstreamer/common/qgstreameraudioinput.cpp
blob: ad00903e82628575d7765a1fe53a637be27ab9c0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
// Copyright (C) 2021 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only

#include <qgstreameraudioinput_p.h>
#include <qgstreameraudiodevice_p.h>
#include <qaudiodevice.h>
#include <qaudioinput.h>

#include <QtCore/qloggingcategory.h>
#include <QtNetwork/qnetworkaccessmanager.h>
#include <QtNetwork/qnetworkreply.h>

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

#include <utility>

static Q_LOGGING_CATEGORY(qLcMediaAudioInput, "qt.multimedia.audioInput")

QT_BEGIN_NAMESPACE

QMaybe<QPlatformAudioInput *> QGstreamerAudioInput::create(QAudioInput *parent)
{
    QGstElement autoaudiosrc = QGstElement::createFromFactory("autoaudiosrc", "autoaudiosrc");
    if (!autoaudiosrc)
        return errorMessageCannotFindElement("autoaudiosrc");

    QGstElement volume = QGstElement::createFromFactory("volume", "volume");
    if (!volume)
        return errorMessageCannotFindElement("volume");

    return new QGstreamerAudioInput(autoaudiosrc, volume, parent);
}

QGstreamerAudioInput::QGstreamerAudioInput(QGstElement autoaudiosrc, QGstElement volume,
                                           QAudioInput *parent)
    : QObject(parent),
      QPlatformAudioInput(parent),
      gstAudioInput(QGstBin::create("audioInput")),
      audioSrc(std::move(autoaudiosrc)),
      audioVolume(std::move(volume))
{
    gstAudioInput.add(audioSrc, audioVolume);
    qLinkGstElements(audioSrc, audioVolume);

    gstAudioInput.addGhostPad(audioVolume, "src");
}

QGstreamerAudioInput::~QGstreamerAudioInput()
{
    gstAudioInput.setStateSync(GST_STATE_NULL);
}

void QGstreamerAudioInput::setVolume(float volume)
{
    audioVolume.set("volume", volume);
}

void QGstreamerAudioInput::setMuted(bool muted)
{
    audioVolume.set("mute", muted);
}

void QGstreamerAudioInput::setAudioDevice(const QAudioDevice &device)
{
    if (device == m_audioDevice)
        return;
    qCDebug(qLcMediaAudioInput) << "setAudioInput" << device.description() << device.isNull();
    m_audioDevice = device;

    QGstElement newSrc;
    if constexpr (QT_CONFIG(pulseaudio)) {
        auto id = m_audioDevice.id();
        newSrc = QGstElement::createFromFactory("pulsesrc", "audiosrc");
        if (!newSrc.isNull())
            newSrc.set("device", id.constData());
        else
            qCWarning(qLcMediaAudioInput) << "Invalid audio device";
    } else {
        auto *deviceInfo = static_cast<const QGStreamerAudioDeviceInfo *>(m_audioDevice.handle());
        if (deviceInfo && deviceInfo->gstDevice) {
            newSrc = QGstElement{
                gst_device_create_element(deviceInfo->gstDevice.get(), "audiosrc"),
                QGstElement::HasRef,
            };
        } else {
            qCWarning(qLcMediaAudioInput) << "Invalid audio device";
        }
    }

    if (newSrc.isNull()) {
        qCWarning(qLcMediaAudioInput) << "Failed to create a gst element for the audio device, using a default audio source";
        newSrc = QGstElement::createFromFactory("autoaudiosrc", "audiosrc");
    }

    // FIXME: most probably source can be disconnected outside of idle probe
    audioSrc.staticPad("src").doInIdleProbe([&]() { qUnlinkGstElements(audioSrc, audioVolume); });
    gstAudioInput.stopAndRemoveElements(audioSrc);
    audioSrc = newSrc;
    gstAudioInput.add(audioSrc);
    qLinkGstElements(audioSrc, audioVolume);
    audioSrc.syncStateWithParent();
}

QAudioDevice QGstreamerAudioInput::audioInput() const
{
    return m_audioDevice;
}

QT_END_NAMESPACE