1cabdff1aSopenharmony_ci/*
2cabdff1aSopenharmony_ci * Pulseaudio common
3cabdff1aSopenharmony_ci * Copyright (c) 2014 Lukasz Marek
4cabdff1aSopenharmony_ci * Copyright (c) 2011 Luca Barbato <lu_zero@gentoo.org>
5cabdff1aSopenharmony_ci *
6cabdff1aSopenharmony_ci * This file is part of FFmpeg.
7cabdff1aSopenharmony_ci *
8cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or
9cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public
10cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either
11cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version.
12cabdff1aSopenharmony_ci *
13cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful,
14cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of
15cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
16cabdff1aSopenharmony_ci * Lesser General Public License for more details.
17cabdff1aSopenharmony_ci *
18cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public
19cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software
20cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
21cabdff1aSopenharmony_ci */
22cabdff1aSopenharmony_ci
23cabdff1aSopenharmony_ci#include "pulse_audio_common.h"
24cabdff1aSopenharmony_ci#include "libavutil/attributes.h"
25cabdff1aSopenharmony_ci#include "libavutil/avstring.h"
26cabdff1aSopenharmony_ci#include "libavutil/mem.h"
27cabdff1aSopenharmony_ci#include "libavutil/avassert.h"
28cabdff1aSopenharmony_ci
29cabdff1aSopenharmony_cipa_sample_format_t av_cold ff_codec_id_to_pulse_format(enum AVCodecID codec_id)
30cabdff1aSopenharmony_ci{
31cabdff1aSopenharmony_ci    switch (codec_id) {
32cabdff1aSopenharmony_ci    case AV_CODEC_ID_PCM_U8:    return PA_SAMPLE_U8;
33cabdff1aSopenharmony_ci    case AV_CODEC_ID_PCM_ALAW:  return PA_SAMPLE_ALAW;
34cabdff1aSopenharmony_ci    case AV_CODEC_ID_PCM_MULAW: return PA_SAMPLE_ULAW;
35cabdff1aSopenharmony_ci    case AV_CODEC_ID_PCM_S16LE: return PA_SAMPLE_S16LE;
36cabdff1aSopenharmony_ci    case AV_CODEC_ID_PCM_S16BE: return PA_SAMPLE_S16BE;
37cabdff1aSopenharmony_ci    case AV_CODEC_ID_PCM_F32LE: return PA_SAMPLE_FLOAT32LE;
38cabdff1aSopenharmony_ci    case AV_CODEC_ID_PCM_F32BE: return PA_SAMPLE_FLOAT32BE;
39cabdff1aSopenharmony_ci    case AV_CODEC_ID_PCM_S32LE: return PA_SAMPLE_S32LE;
40cabdff1aSopenharmony_ci    case AV_CODEC_ID_PCM_S32BE: return PA_SAMPLE_S32BE;
41cabdff1aSopenharmony_ci    case AV_CODEC_ID_PCM_S24LE: return PA_SAMPLE_S24LE;
42cabdff1aSopenharmony_ci    case AV_CODEC_ID_PCM_S24BE: return PA_SAMPLE_S24BE;
43cabdff1aSopenharmony_ci    default:                    return PA_SAMPLE_INVALID;
44cabdff1aSopenharmony_ci    }
45cabdff1aSopenharmony_ci}
46cabdff1aSopenharmony_ci
47cabdff1aSopenharmony_cienum PulseAudioContextState {
48cabdff1aSopenharmony_ci    PULSE_CONTEXT_INITIALIZING,
49cabdff1aSopenharmony_ci    PULSE_CONTEXT_READY,
50cabdff1aSopenharmony_ci    PULSE_CONTEXT_FINISHED
51cabdff1aSopenharmony_ci};
52cabdff1aSopenharmony_ci
53cabdff1aSopenharmony_citypedef struct PulseAudioDeviceList {
54cabdff1aSopenharmony_ci    AVDeviceInfoList *devices;
55cabdff1aSopenharmony_ci    int error_code;
56cabdff1aSopenharmony_ci    int output;
57cabdff1aSopenharmony_ci    char *default_device;
58cabdff1aSopenharmony_ci} PulseAudioDeviceList;
59cabdff1aSopenharmony_ci
60cabdff1aSopenharmony_cistatic void pa_state_cb(pa_context *c, void *userdata)
61cabdff1aSopenharmony_ci{
62cabdff1aSopenharmony_ci    enum PulseAudioContextState *context_state = userdata;
63cabdff1aSopenharmony_ci
64cabdff1aSopenharmony_ci    switch  (pa_context_get_state(c)) {
65cabdff1aSopenharmony_ci    case PA_CONTEXT_FAILED:
66cabdff1aSopenharmony_ci    case PA_CONTEXT_TERMINATED:
67cabdff1aSopenharmony_ci        *context_state = PULSE_CONTEXT_FINISHED;
68cabdff1aSopenharmony_ci        break;
69cabdff1aSopenharmony_ci    case PA_CONTEXT_READY:
70cabdff1aSopenharmony_ci        *context_state = PULSE_CONTEXT_READY;
71cabdff1aSopenharmony_ci        break;
72cabdff1aSopenharmony_ci    default:
73cabdff1aSopenharmony_ci        break;
74cabdff1aSopenharmony_ci    }
75cabdff1aSopenharmony_ci}
76cabdff1aSopenharmony_ci
77cabdff1aSopenharmony_civoid ff_pulse_audio_disconnect_context(pa_mainloop **pa_ml, pa_context **pa_ctx)
78cabdff1aSopenharmony_ci{
79cabdff1aSopenharmony_ci    av_assert0(pa_ml);
80cabdff1aSopenharmony_ci    av_assert0(pa_ctx);
81cabdff1aSopenharmony_ci
82cabdff1aSopenharmony_ci    if (*pa_ctx) {
83cabdff1aSopenharmony_ci        pa_context_set_state_callback(*pa_ctx, NULL, NULL);
84cabdff1aSopenharmony_ci        pa_context_disconnect(*pa_ctx);
85cabdff1aSopenharmony_ci        pa_context_unref(*pa_ctx);
86cabdff1aSopenharmony_ci    }
87cabdff1aSopenharmony_ci    if (*pa_ml)
88cabdff1aSopenharmony_ci        pa_mainloop_free(*pa_ml);
89cabdff1aSopenharmony_ci    *pa_ml = NULL;
90cabdff1aSopenharmony_ci    *pa_ctx = NULL;
91cabdff1aSopenharmony_ci}
92cabdff1aSopenharmony_ci
93cabdff1aSopenharmony_ciint ff_pulse_audio_connect_context(pa_mainloop **pa_ml, pa_context **pa_ctx,
94cabdff1aSopenharmony_ci                                   const char *server, const char *description)
95cabdff1aSopenharmony_ci{
96cabdff1aSopenharmony_ci    int ret;
97cabdff1aSopenharmony_ci    pa_mainloop_api *pa_mlapi = NULL;
98cabdff1aSopenharmony_ci    enum PulseAudioContextState context_state = PULSE_CONTEXT_INITIALIZING;
99cabdff1aSopenharmony_ci
100cabdff1aSopenharmony_ci    av_assert0(pa_ml);
101cabdff1aSopenharmony_ci    av_assert0(pa_ctx);
102cabdff1aSopenharmony_ci
103cabdff1aSopenharmony_ci    *pa_ml = NULL;
104cabdff1aSopenharmony_ci    *pa_ctx = NULL;
105cabdff1aSopenharmony_ci
106cabdff1aSopenharmony_ci    if (!(*pa_ml = pa_mainloop_new()))
107cabdff1aSopenharmony_ci        return AVERROR(ENOMEM);
108cabdff1aSopenharmony_ci    if (!(pa_mlapi = pa_mainloop_get_api(*pa_ml))) {
109cabdff1aSopenharmony_ci        ret = AVERROR_EXTERNAL;
110cabdff1aSopenharmony_ci        goto fail;
111cabdff1aSopenharmony_ci    }
112cabdff1aSopenharmony_ci    if (!(*pa_ctx = pa_context_new(pa_mlapi, description))) {
113cabdff1aSopenharmony_ci        ret = AVERROR(ENOMEM);
114cabdff1aSopenharmony_ci        goto fail;
115cabdff1aSopenharmony_ci    }
116cabdff1aSopenharmony_ci    pa_context_set_state_callback(*pa_ctx, pa_state_cb, &context_state);
117cabdff1aSopenharmony_ci    if (pa_context_connect(*pa_ctx, server, 0, NULL) < 0) {
118cabdff1aSopenharmony_ci        ret = AVERROR_EXTERNAL;
119cabdff1aSopenharmony_ci        goto fail;
120cabdff1aSopenharmony_ci    }
121cabdff1aSopenharmony_ci
122cabdff1aSopenharmony_ci    while (context_state == PULSE_CONTEXT_INITIALIZING)
123cabdff1aSopenharmony_ci        pa_mainloop_iterate(*pa_ml, 1, NULL);
124cabdff1aSopenharmony_ci    if (context_state == PULSE_CONTEXT_FINISHED) {
125cabdff1aSopenharmony_ci        ret = AVERROR_EXTERNAL;
126cabdff1aSopenharmony_ci        goto fail;
127cabdff1aSopenharmony_ci    }
128cabdff1aSopenharmony_ci    return 0;
129cabdff1aSopenharmony_ci
130cabdff1aSopenharmony_ci  fail:
131cabdff1aSopenharmony_ci    ff_pulse_audio_disconnect_context(pa_ml, pa_ctx);
132cabdff1aSopenharmony_ci    return ret;
133cabdff1aSopenharmony_ci}
134cabdff1aSopenharmony_ci
135cabdff1aSopenharmony_cistatic void pulse_add_detected_device(PulseAudioDeviceList *info,
136cabdff1aSopenharmony_ci                                      const char *name, const char *description)
137cabdff1aSopenharmony_ci{
138cabdff1aSopenharmony_ci    int ret;
139cabdff1aSopenharmony_ci    AVDeviceInfo *new_device = NULL;
140cabdff1aSopenharmony_ci
141cabdff1aSopenharmony_ci    if (info->error_code)
142cabdff1aSopenharmony_ci        return;
143cabdff1aSopenharmony_ci
144cabdff1aSopenharmony_ci    new_device = av_mallocz(sizeof(AVDeviceInfo));
145cabdff1aSopenharmony_ci    if (!new_device) {
146cabdff1aSopenharmony_ci        info->error_code = AVERROR(ENOMEM);
147cabdff1aSopenharmony_ci        return;
148cabdff1aSopenharmony_ci    }
149cabdff1aSopenharmony_ci
150cabdff1aSopenharmony_ci    new_device->device_description = av_strdup(description);
151cabdff1aSopenharmony_ci    new_device->device_name = av_strdup(name);
152cabdff1aSopenharmony_ci
153cabdff1aSopenharmony_ci    if (!new_device->device_description || !new_device->device_name) {
154cabdff1aSopenharmony_ci        info->error_code = AVERROR(ENOMEM);
155cabdff1aSopenharmony_ci        goto fail;
156cabdff1aSopenharmony_ci    }
157cabdff1aSopenharmony_ci
158cabdff1aSopenharmony_ci    if ((ret = av_dynarray_add_nofree(&info->devices->devices,
159cabdff1aSopenharmony_ci                                      &info->devices->nb_devices, new_device)) < 0) {
160cabdff1aSopenharmony_ci        info->error_code = ret;
161cabdff1aSopenharmony_ci        goto fail;
162cabdff1aSopenharmony_ci    }
163cabdff1aSopenharmony_ci    return;
164cabdff1aSopenharmony_ci
165cabdff1aSopenharmony_ci  fail:
166cabdff1aSopenharmony_ci    av_freep(&new_device->device_description);
167cabdff1aSopenharmony_ci    av_freep(&new_device->device_name);
168cabdff1aSopenharmony_ci    av_free(new_device);
169cabdff1aSopenharmony_ci
170cabdff1aSopenharmony_ci}
171cabdff1aSopenharmony_ci
172cabdff1aSopenharmony_cistatic void pulse_audio_source_device_cb(pa_context *c, const pa_source_info *dev,
173cabdff1aSopenharmony_ci                                         int eol, void *userdata)
174cabdff1aSopenharmony_ci{
175cabdff1aSopenharmony_ci    if (!eol)
176cabdff1aSopenharmony_ci        pulse_add_detected_device(userdata, dev->name, dev->description);
177cabdff1aSopenharmony_ci}
178cabdff1aSopenharmony_ci
179cabdff1aSopenharmony_cistatic void pulse_audio_sink_device_cb(pa_context *c, const pa_sink_info *dev,
180cabdff1aSopenharmony_ci                                       int eol, void *userdata)
181cabdff1aSopenharmony_ci{
182cabdff1aSopenharmony_ci    if (!eol)
183cabdff1aSopenharmony_ci        pulse_add_detected_device(userdata, dev->name, dev->description);
184cabdff1aSopenharmony_ci}
185cabdff1aSopenharmony_ci
186cabdff1aSopenharmony_cistatic void pulse_server_info_cb(pa_context *c, const pa_server_info *i, void *userdata)
187cabdff1aSopenharmony_ci{
188cabdff1aSopenharmony_ci    PulseAudioDeviceList *info = userdata;
189cabdff1aSopenharmony_ci    if (info->output)
190cabdff1aSopenharmony_ci        info->default_device = av_strdup(i->default_sink_name);
191cabdff1aSopenharmony_ci    else
192cabdff1aSopenharmony_ci        info->default_device = av_strdup(i->default_source_name);
193cabdff1aSopenharmony_ci    if (!info->default_device)
194cabdff1aSopenharmony_ci        info->error_code = AVERROR(ENOMEM);
195cabdff1aSopenharmony_ci}
196cabdff1aSopenharmony_ci
197cabdff1aSopenharmony_ciint ff_pulse_audio_get_devices(AVDeviceInfoList *devices, const char *server, int output)
198cabdff1aSopenharmony_ci{
199cabdff1aSopenharmony_ci    pa_mainloop *pa_ml = NULL;
200cabdff1aSopenharmony_ci    pa_operation *pa_op = NULL;
201cabdff1aSopenharmony_ci    pa_context *pa_ctx = NULL;
202cabdff1aSopenharmony_ci    enum pa_operation_state op_state;
203cabdff1aSopenharmony_ci    PulseAudioDeviceList dev_list = { 0 };
204cabdff1aSopenharmony_ci    int i;
205cabdff1aSopenharmony_ci
206cabdff1aSopenharmony_ci    dev_list.output = output;
207cabdff1aSopenharmony_ci    dev_list.devices = devices;
208cabdff1aSopenharmony_ci    if (!devices)
209cabdff1aSopenharmony_ci        return AVERROR(EINVAL);
210cabdff1aSopenharmony_ci    devices->nb_devices = 0;
211cabdff1aSopenharmony_ci    devices->devices = NULL;
212cabdff1aSopenharmony_ci
213cabdff1aSopenharmony_ci    if ((dev_list.error_code = ff_pulse_audio_connect_context(&pa_ml, &pa_ctx, server, "Query devices")) < 0)
214cabdff1aSopenharmony_ci        goto fail;
215cabdff1aSopenharmony_ci
216cabdff1aSopenharmony_ci    if (output)
217cabdff1aSopenharmony_ci        pa_op = pa_context_get_sink_info_list(pa_ctx, pulse_audio_sink_device_cb, &dev_list);
218cabdff1aSopenharmony_ci    else
219cabdff1aSopenharmony_ci        pa_op = pa_context_get_source_info_list(pa_ctx, pulse_audio_source_device_cb, &dev_list);
220cabdff1aSopenharmony_ci    while ((op_state = pa_operation_get_state(pa_op)) == PA_OPERATION_RUNNING)
221cabdff1aSopenharmony_ci        pa_mainloop_iterate(pa_ml, 1, NULL);
222cabdff1aSopenharmony_ci    if (op_state != PA_OPERATION_DONE)
223cabdff1aSopenharmony_ci        dev_list.error_code = AVERROR_EXTERNAL;
224cabdff1aSopenharmony_ci    pa_operation_unref(pa_op);
225cabdff1aSopenharmony_ci    if (dev_list.error_code < 0)
226cabdff1aSopenharmony_ci        goto fail;
227cabdff1aSopenharmony_ci
228cabdff1aSopenharmony_ci    pa_op = pa_context_get_server_info(pa_ctx, pulse_server_info_cb, &dev_list);
229cabdff1aSopenharmony_ci    while ((op_state = pa_operation_get_state(pa_op)) == PA_OPERATION_RUNNING)
230cabdff1aSopenharmony_ci        pa_mainloop_iterate(pa_ml, 1, NULL);
231cabdff1aSopenharmony_ci    if (op_state != PA_OPERATION_DONE)
232cabdff1aSopenharmony_ci        dev_list.error_code = AVERROR_EXTERNAL;
233cabdff1aSopenharmony_ci    pa_operation_unref(pa_op);
234cabdff1aSopenharmony_ci    if (dev_list.error_code < 0)
235cabdff1aSopenharmony_ci        goto fail;
236cabdff1aSopenharmony_ci
237cabdff1aSopenharmony_ci    devices->default_device = -1;
238cabdff1aSopenharmony_ci    for (i = 0; i < devices->nb_devices; i++) {
239cabdff1aSopenharmony_ci        if (!strcmp(devices->devices[i]->device_name, dev_list.default_device)) {
240cabdff1aSopenharmony_ci            devices->default_device = i;
241cabdff1aSopenharmony_ci            break;
242cabdff1aSopenharmony_ci        }
243cabdff1aSopenharmony_ci    }
244cabdff1aSopenharmony_ci
245cabdff1aSopenharmony_ci  fail:
246cabdff1aSopenharmony_ci    av_free(dev_list.default_device);
247cabdff1aSopenharmony_ci    ff_pulse_audio_disconnect_context(&pa_ml, &pa_ctx);
248cabdff1aSopenharmony_ci    return dev_list.error_code;
249cabdff1aSopenharmony_ci}
250