xref: /openbmc/qemu/audio/paaudio.c (revision bea29e9f2ea6061abc212809090548cba6d64081)
1b8e59f18Smalc /* public domain */
20b8fa32fSMarkus Armbruster 
36086a565SPeter Maydell #include "qemu/osdep.h"
40b8fa32fSMarkus Armbruster #include "qemu/module.h"
53443ad4eSKővágó, Zoltán #include "qemu-common.h"
6b8e59f18Smalc #include "audio.h"
72c324b28SKővágó, Zoltán #include "qapi/opts-visitor.h"
8b8e59f18Smalc 
9ea9ebc2cSMarc-André Lureau #include <pulse/pulseaudio.h>
10b8e59f18Smalc 
11b8e59f18Smalc #define AUDIO_CAP "pulseaudio"
12b8e59f18Smalc #include "audio_int.h"
13b8e59f18Smalc 
149d34e6d8SKővágó, Zoltán typedef struct PAConnection {
159d34e6d8SKővágó, Zoltán     char *server;
169d34e6d8SKővágó, Zoltán     int refcount;
179d34e6d8SKővágó, Zoltán     QTAILQ_ENTRY(PAConnection) list;
189d34e6d8SKővágó, Zoltán 
199a644c4bSKővágó, Zoltán     pa_threaded_mainloop *mainloop;
209a644c4bSKővágó, Zoltán     pa_context *context;
219d34e6d8SKővágó, Zoltán } PAConnection;
229d34e6d8SKővágó, Zoltán 
239d34e6d8SKővágó, Zoltán static QTAILQ_HEAD(PAConnectionHead, PAConnection) pa_conns =
249d34e6d8SKővágó, Zoltán     QTAILQ_HEAD_INITIALIZER(pa_conns);
259d34e6d8SKővágó, Zoltán 
269d34e6d8SKővágó, Zoltán typedef struct {
279d34e6d8SKővágó, Zoltán     Audiodev *dev;
289d34e6d8SKővágó, Zoltán     PAConnection *conn;
299a644c4bSKővágó, Zoltán } paaudio;
309a644c4bSKővágó, Zoltán 
319a644c4bSKővágó, Zoltán typedef struct {
32b8e59f18Smalc     HWVoiceOut hw;
33ea9ebc2cSMarc-André Lureau     pa_stream *stream;
349a644c4bSKővágó, Zoltán     paaudio *g;
35b8e59f18Smalc } PAVoiceOut;
36b8e59f18Smalc 
37b8e59f18Smalc typedef struct {
38b8e59f18Smalc     HWVoiceIn hw;
39ea9ebc2cSMarc-André Lureau     pa_stream *stream;
40ea9ebc2cSMarc-André Lureau     const void *read_data;
4149ddd7e1SKővágó, Zoltán     size_t read_length;
429a644c4bSKővágó, Zoltán     paaudio *g;
43b8e59f18Smalc } PAVoiceIn;
44b8e59f18Smalc 
459d34e6d8SKővágó, Zoltán static void qpa_conn_fini(PAConnection *c);
4649dd6d0dSKővágó, Zoltán 
47b8e59f18Smalc static void GCC_FMT_ATTR (2, 3) qpa_logerr (int err, const char *fmt, ...)
48b8e59f18Smalc {
49b8e59f18Smalc     va_list ap;
50b8e59f18Smalc 
51b8e59f18Smalc     va_start (ap, fmt);
52b8e59f18Smalc     AUD_vlog (AUDIO_CAP, fmt, ap);
53b8e59f18Smalc     va_end (ap);
54b8e59f18Smalc 
55b8e59f18Smalc     AUD_log (AUDIO_CAP, "Reason: %s\n", pa_strerror (err));
56b8e59f18Smalc }
57b8e59f18Smalc 
588f473dd1SGerd Hoffmann #ifndef PA_CONTEXT_IS_GOOD
598f473dd1SGerd Hoffmann static inline int PA_CONTEXT_IS_GOOD(pa_context_state_t x)
608f473dd1SGerd Hoffmann {
618f473dd1SGerd Hoffmann     return
628f473dd1SGerd Hoffmann         x == PA_CONTEXT_CONNECTING ||
638f473dd1SGerd Hoffmann         x == PA_CONTEXT_AUTHORIZING ||
648f473dd1SGerd Hoffmann         x == PA_CONTEXT_SETTING_NAME ||
658f473dd1SGerd Hoffmann         x == PA_CONTEXT_READY;
668f473dd1SGerd Hoffmann }
678f473dd1SGerd Hoffmann #endif
688f473dd1SGerd Hoffmann 
698f473dd1SGerd Hoffmann #ifndef PA_STREAM_IS_GOOD
708f473dd1SGerd Hoffmann static inline int PA_STREAM_IS_GOOD(pa_stream_state_t x)
718f473dd1SGerd Hoffmann {
728f473dd1SGerd Hoffmann     return
738f473dd1SGerd Hoffmann         x == PA_STREAM_CREATING ||
748f473dd1SGerd Hoffmann         x == PA_STREAM_READY;
758f473dd1SGerd Hoffmann }
768f473dd1SGerd Hoffmann #endif
778f473dd1SGerd Hoffmann 
7849ddd7e1SKővágó, Zoltán #define CHECK_SUCCESS_GOTO(c, expression, label, msg)           \
79ea9ebc2cSMarc-André Lureau     do {                                                        \
80ea9ebc2cSMarc-André Lureau         if (!(expression)) {                                    \
8149ddd7e1SKővágó, Zoltán             qpa_logerr(pa_context_errno((c)->context), msg);    \
82ea9ebc2cSMarc-André Lureau             goto label;                                         \
83ea9ebc2cSMarc-André Lureau         }                                                       \
842562755eSEric Blake     } while (0)
85ea9ebc2cSMarc-André Lureau 
8649ddd7e1SKővágó, Zoltán #define CHECK_DEAD_GOTO(c, stream, label, msg)                          \
87ea9ebc2cSMarc-André Lureau     do {                                                                \
88ea9ebc2cSMarc-André Lureau         if (!(c)->context || !PA_CONTEXT_IS_GOOD (pa_context_get_state((c)->context)) || \
89ea9ebc2cSMarc-André Lureau             !(stream) || !PA_STREAM_IS_GOOD (pa_stream_get_state ((stream)))) { \
90ea9ebc2cSMarc-André Lureau             if (((c)->context && pa_context_get_state ((c)->context) == PA_CONTEXT_FAILED) || \
91ea9ebc2cSMarc-André Lureau                 ((stream) && pa_stream_get_state ((stream)) == PA_STREAM_FAILED)) { \
9249ddd7e1SKővágó, Zoltán                 qpa_logerr(pa_context_errno((c)->context), msg);        \
93ea9ebc2cSMarc-André Lureau             } else {                                                    \
9449ddd7e1SKővágó, Zoltán                 qpa_logerr(PA_ERR_BADSTATE, msg);                       \
95ea9ebc2cSMarc-André Lureau             }                                                           \
96ea9ebc2cSMarc-André Lureau             goto label;                                                 \
97ea9ebc2cSMarc-André Lureau         }                                                               \
982562755eSEric Blake     } while (0)
99ea9ebc2cSMarc-André Lureau 
100337e8de6SKővágó, Zoltán static void *qpa_get_buffer_in(HWVoiceIn *hw, size_t *size)
101337e8de6SKővágó, Zoltán {
102337e8de6SKővágó, Zoltán     PAVoiceIn *p = (PAVoiceIn *) hw;
103337e8de6SKővágó, Zoltán     PAConnection *c = p->g->conn;
104337e8de6SKővágó, Zoltán     int r;
105337e8de6SKővágó, Zoltán 
106337e8de6SKővágó, Zoltán     pa_threaded_mainloop_lock(c->mainloop);
107337e8de6SKővágó, Zoltán 
108337e8de6SKővágó, Zoltán     CHECK_DEAD_GOTO(c, p->stream, unlock_and_fail,
109337e8de6SKővágó, Zoltán                     "pa_threaded_mainloop_lock failed\n");
110337e8de6SKővágó, Zoltán 
111337e8de6SKővágó, Zoltán     if (!p->read_length) {
112337e8de6SKővágó, Zoltán         r = pa_stream_peek(p->stream, &p->read_data, &p->read_length);
113337e8de6SKővágó, Zoltán         CHECK_SUCCESS_GOTO(c, r == 0, unlock_and_fail,
114337e8de6SKővágó, Zoltán                            "pa_stream_peek failed\n");
115337e8de6SKővágó, Zoltán     }
116337e8de6SKővágó, Zoltán 
117337e8de6SKővágó, Zoltán     *size = MIN(p->read_length, *size);
118337e8de6SKővágó, Zoltán 
119337e8de6SKővágó, Zoltán     pa_threaded_mainloop_unlock(c->mainloop);
120337e8de6SKővágó, Zoltán     return (void *) p->read_data;
121337e8de6SKővágó, Zoltán 
122337e8de6SKővágó, Zoltán unlock_and_fail:
123337e8de6SKővágó, Zoltán     pa_threaded_mainloop_unlock(c->mainloop);
124337e8de6SKővágó, Zoltán     *size = 0;
125337e8de6SKővágó, Zoltán     return NULL;
126337e8de6SKővágó, Zoltán }
127337e8de6SKővágó, Zoltán 
128337e8de6SKővágó, Zoltán static void qpa_put_buffer_in(HWVoiceIn *hw, void *buf, size_t size)
129337e8de6SKővágó, Zoltán {
130337e8de6SKővágó, Zoltán     PAVoiceIn *p = (PAVoiceIn *) hw;
131337e8de6SKővágó, Zoltán     PAConnection *c = p->g->conn;
132337e8de6SKővágó, Zoltán     int r;
133337e8de6SKővágó, Zoltán 
134337e8de6SKővágó, Zoltán     pa_threaded_mainloop_lock(c->mainloop);
135337e8de6SKővágó, Zoltán 
136337e8de6SKővágó, Zoltán     CHECK_DEAD_GOTO(c, p->stream, unlock,
137337e8de6SKővágó, Zoltán                     "pa_threaded_mainloop_lock failed\n");
138337e8de6SKővágó, Zoltán 
139337e8de6SKővágó, Zoltán     assert(buf == p->read_data && size <= p->read_length);
140337e8de6SKővágó, Zoltán 
141337e8de6SKővágó, Zoltán     p->read_data += size;
142337e8de6SKővágó, Zoltán     p->read_length -= size;
143337e8de6SKővágó, Zoltán 
144337e8de6SKővágó, Zoltán     if (size && !p->read_length) {
145337e8de6SKővágó, Zoltán         r = pa_stream_drop(p->stream);
146337e8de6SKővágó, Zoltán         CHECK_SUCCESS_GOTO(c, r == 0, unlock, "pa_stream_drop failed\n");
147337e8de6SKővágó, Zoltán     }
148337e8de6SKővágó, Zoltán 
149337e8de6SKővágó, Zoltán unlock:
150337e8de6SKővágó, Zoltán     pa_threaded_mainloop_unlock(c->mainloop);
151337e8de6SKővágó, Zoltán }
152337e8de6SKővágó, Zoltán 
15349ddd7e1SKővágó, Zoltán static size_t qpa_read(HWVoiceIn *hw, void *data, size_t length)
154ea9ebc2cSMarc-André Lureau {
15549ddd7e1SKővágó, Zoltán     PAVoiceIn *p = (PAVoiceIn *) hw;
1569d34e6d8SKővágó, Zoltán     PAConnection *c = p->g->conn;
157acc3b63eSVolker Rümelin     size_t total = 0;
158ea9ebc2cSMarc-André Lureau 
1599d34e6d8SKővágó, Zoltán     pa_threaded_mainloop_lock(c->mainloop);
160ea9ebc2cSMarc-André Lureau 
16149ddd7e1SKővágó, Zoltán     CHECK_DEAD_GOTO(c, p->stream, unlock_and_fail,
16249ddd7e1SKővágó, Zoltán                     "pa_threaded_mainloop_lock failed\n");
1637c9eb86eSVolker Rümelin     if (pa_stream_get_state(p->stream) != PA_STREAM_READY) {
1647c9eb86eSVolker Rümelin         /* wait for stream to become ready */
1657c9eb86eSVolker Rümelin         goto unlock;
1667c9eb86eSVolker Rümelin     }
167ea9ebc2cSMarc-André Lureau 
168acc3b63eSVolker Rümelin     while (total < length) {
169acc3b63eSVolker Rümelin         size_t l;
170acc3b63eSVolker Rümelin         int r;
171acc3b63eSVolker Rümelin 
17249ddd7e1SKővágó, Zoltán         if (!p->read_length) {
173ea9ebc2cSMarc-André Lureau             r = pa_stream_peek(p->stream, &p->read_data, &p->read_length);
17449ddd7e1SKővágó, Zoltán             CHECK_SUCCESS_GOTO(c, r == 0, unlock_and_fail,
17549ddd7e1SKővágó, Zoltán                                "pa_stream_peek failed\n");
176acc3b63eSVolker Rümelin             if (!p->read_length) {
177acc3b63eSVolker Rümelin                 /* buffer is empty */
178acc3b63eSVolker Rümelin                 break;
179acc3b63eSVolker Rümelin             }
180ea9ebc2cSMarc-André Lureau         }
181ea9ebc2cSMarc-André Lureau 
182acc3b63eSVolker Rümelin         l = MIN(p->read_length, length - total);
183acc3b63eSVolker Rümelin         memcpy((char *)data + total, p->read_data, l);
184ea9ebc2cSMarc-André Lureau 
18549ddd7e1SKővágó, Zoltán         p->read_data += l;
186ea9ebc2cSMarc-André Lureau         p->read_length -= l;
187acc3b63eSVolker Rümelin         total += l;
188ea9ebc2cSMarc-André Lureau 
189ea9ebc2cSMarc-André Lureau         if (!p->read_length) {
190ea9ebc2cSMarc-André Lureau             r = pa_stream_drop(p->stream);
19149ddd7e1SKővágó, Zoltán             CHECK_SUCCESS_GOTO(c, r == 0, unlock_and_fail,
19249ddd7e1SKővágó, Zoltán                                "pa_stream_drop failed\n");
193ea9ebc2cSMarc-André Lureau         }
194acc3b63eSVolker Rümelin     }
195ea9ebc2cSMarc-André Lureau 
1967c9eb86eSVolker Rümelin unlock:
1979d34e6d8SKővágó, Zoltán     pa_threaded_mainloop_unlock(c->mainloop);
198acc3b63eSVolker Rümelin     return total;
199ea9ebc2cSMarc-André Lureau 
200ea9ebc2cSMarc-André Lureau unlock_and_fail:
2019d34e6d8SKővágó, Zoltán     pa_threaded_mainloop_unlock(c->mainloop);
20249ddd7e1SKővágó, Zoltán     return 0;
203ea9ebc2cSMarc-André Lureau }
204ea9ebc2cSMarc-André Lureau 
205337e8de6SKővágó, Zoltán static void *qpa_get_buffer_out(HWVoiceOut *hw, size_t *size)
206337e8de6SKővágó, Zoltán {
207337e8de6SKővágó, Zoltán     PAVoiceOut *p = (PAVoiceOut *) hw;
208337e8de6SKővágó, Zoltán     PAConnection *c = p->g->conn;
209337e8de6SKővágó, Zoltán     void *ret;
210*bea29e9fSVolker Rümelin     size_t l;
211337e8de6SKővágó, Zoltán     int r;
212337e8de6SKővágó, Zoltán 
213337e8de6SKővágó, Zoltán     pa_threaded_mainloop_lock(c->mainloop);
214337e8de6SKővágó, Zoltán 
215337e8de6SKővágó, Zoltán     CHECK_DEAD_GOTO(c, p->stream, unlock_and_fail,
216337e8de6SKővágó, Zoltán                     "pa_threaded_mainloop_lock failed\n");
217337e8de6SKővágó, Zoltán 
218*bea29e9fSVolker Rümelin     l = pa_stream_writable_size(p->stream);
219*bea29e9fSVolker Rümelin     CHECK_SUCCESS_GOTO(c, l != (size_t) -1, unlock_and_fail,
220*bea29e9fSVolker Rümelin                        "pa_stream_writable_size failed\n");
221*bea29e9fSVolker Rümelin 
222337e8de6SKővágó, Zoltán     *size = -1;
223337e8de6SKővágó, Zoltán     r = pa_stream_begin_write(p->stream, &ret, size);
224337e8de6SKővágó, Zoltán     CHECK_SUCCESS_GOTO(c, r >= 0, unlock_and_fail,
225337e8de6SKővágó, Zoltán                        "pa_stream_begin_write failed\n");
226337e8de6SKővágó, Zoltán 
227337e8de6SKővágó, Zoltán     pa_threaded_mainloop_unlock(c->mainloop);
228*bea29e9fSVolker Rümelin     if (*size > l) {
229*bea29e9fSVolker Rümelin         *size = l;
230*bea29e9fSVolker Rümelin     }
231337e8de6SKővágó, Zoltán     return ret;
232337e8de6SKővágó, Zoltán 
233337e8de6SKővágó, Zoltán unlock_and_fail:
234337e8de6SKővágó, Zoltán     pa_threaded_mainloop_unlock(c->mainloop);
235337e8de6SKővágó, Zoltán     *size = 0;
236337e8de6SKővágó, Zoltán     return NULL;
237337e8de6SKővágó, Zoltán }
238337e8de6SKővágó, Zoltán 
239*bea29e9fSVolker Rümelin static size_t qpa_put_buffer_out(HWVoiceOut *hw, void *data, size_t length)
240*bea29e9fSVolker Rümelin {
241*bea29e9fSVolker Rümelin     PAVoiceOut *p = (PAVoiceOut *)hw;
242*bea29e9fSVolker Rümelin     PAConnection *c = p->g->conn;
243*bea29e9fSVolker Rümelin     int r;
244*bea29e9fSVolker Rümelin 
245*bea29e9fSVolker Rümelin     pa_threaded_mainloop_lock(c->mainloop);
246*bea29e9fSVolker Rümelin 
247*bea29e9fSVolker Rümelin     CHECK_DEAD_GOTO(c, p->stream, unlock_and_fail,
248*bea29e9fSVolker Rümelin                     "pa_threaded_mainloop_lock failed\n");
249*bea29e9fSVolker Rümelin 
250*bea29e9fSVolker Rümelin     r = pa_stream_write(p->stream, data, length, NULL, 0LL, PA_SEEK_RELATIVE);
251*bea29e9fSVolker Rümelin     CHECK_SUCCESS_GOTO(c, r >= 0, unlock_and_fail, "pa_stream_write failed\n");
252*bea29e9fSVolker Rümelin 
253*bea29e9fSVolker Rümelin     pa_threaded_mainloop_unlock(c->mainloop);
254*bea29e9fSVolker Rümelin     return length;
255*bea29e9fSVolker Rümelin 
256*bea29e9fSVolker Rümelin unlock_and_fail:
257*bea29e9fSVolker Rümelin     pa_threaded_mainloop_unlock(c->mainloop);
258*bea29e9fSVolker Rümelin     return 0;
259*bea29e9fSVolker Rümelin }
260*bea29e9fSVolker Rümelin 
26149ddd7e1SKővágó, Zoltán static size_t qpa_write(HWVoiceOut *hw, void *data, size_t length)
262ea9ebc2cSMarc-André Lureau {
26349ddd7e1SKővágó, Zoltán     PAVoiceOut *p = (PAVoiceOut *) hw;
2649d34e6d8SKővágó, Zoltán     PAConnection *c = p->g->conn;
265ea9ebc2cSMarc-André Lureau     size_t l;
266ea9ebc2cSMarc-André Lureau     int r;
267ea9ebc2cSMarc-André Lureau 
26849ddd7e1SKővágó, Zoltán     pa_threaded_mainloop_lock(c->mainloop);
269ea9ebc2cSMarc-André Lureau 
27049ddd7e1SKővágó, Zoltán     CHECK_DEAD_GOTO(c, p->stream, unlock_and_fail,
27149ddd7e1SKővágó, Zoltán                     "pa_threaded_mainloop_lock failed\n");
27249ddd7e1SKővágó, Zoltán 
27349ddd7e1SKővágó, Zoltán     l = pa_stream_writable_size(p->stream);
27449ddd7e1SKővágó, Zoltán 
27549ddd7e1SKővágó, Zoltán     CHECK_SUCCESS_GOTO(c, l != (size_t) -1, unlock_and_fail,
27649ddd7e1SKővágó, Zoltán                        "pa_stream_writable_size failed\n");
277ea9ebc2cSMarc-André Lureau 
278ea9ebc2cSMarc-André Lureau     if (l > length) {
279ea9ebc2cSMarc-André Lureau         l = length;
280ea9ebc2cSMarc-André Lureau     }
281ea9ebc2cSMarc-André Lureau 
282ea9ebc2cSMarc-André Lureau     r = pa_stream_write(p->stream, data, l, NULL, 0LL, PA_SEEK_RELATIVE);
28349ddd7e1SKővágó, Zoltán     CHECK_SUCCESS_GOTO(c, r >= 0, unlock_and_fail, "pa_stream_write failed\n");
284ea9ebc2cSMarc-André Lureau 
2859d34e6d8SKővágó, Zoltán     pa_threaded_mainloop_unlock(c->mainloop);
28649ddd7e1SKővágó, Zoltán     return l;
287ea9ebc2cSMarc-André Lureau 
288ea9ebc2cSMarc-André Lureau unlock_and_fail:
2899d34e6d8SKővágó, Zoltán     pa_threaded_mainloop_unlock(c->mainloop);
290b8e59f18Smalc     return 0;
291b8e59f18Smalc }
292b8e59f18Smalc 
29385bc5852SKővágó, Zoltán static pa_sample_format_t audfmt_to_pa (AudioFormat afmt, int endianness)
294b8e59f18Smalc {
295b8e59f18Smalc     int format;
296b8e59f18Smalc 
297b8e59f18Smalc     switch (afmt) {
29885bc5852SKővágó, Zoltán     case AUDIO_FORMAT_S8:
29985bc5852SKővágó, Zoltán     case AUDIO_FORMAT_U8:
300b8e59f18Smalc         format = PA_SAMPLE_U8;
301b8e59f18Smalc         break;
30285bc5852SKővágó, Zoltán     case AUDIO_FORMAT_S16:
30385bc5852SKővágó, Zoltán     case AUDIO_FORMAT_U16:
304b8e59f18Smalc         format = endianness ? PA_SAMPLE_S16BE : PA_SAMPLE_S16LE;
305b8e59f18Smalc         break;
30685bc5852SKővágó, Zoltán     case AUDIO_FORMAT_S32:
30785bc5852SKővágó, Zoltán     case AUDIO_FORMAT_U32:
308b8e59f18Smalc         format = endianness ? PA_SAMPLE_S32BE : PA_SAMPLE_S32LE;
309b8e59f18Smalc         break;
310ed2a4a79SKővágó, Zoltán     case AUDIO_FORMAT_F32:
311ed2a4a79SKővágó, Zoltán         format = endianness ? PA_SAMPLE_FLOAT32BE : PA_SAMPLE_FLOAT32LE;
312ed2a4a79SKővágó, Zoltán         break;
313b8e59f18Smalc     default:
314b8e59f18Smalc         dolog ("Internal logic error: Bad audio format %d\n", afmt);
315b8e59f18Smalc         format = PA_SAMPLE_U8;
316b8e59f18Smalc         break;
317b8e59f18Smalc     }
318b8e59f18Smalc     return format;
319b8e59f18Smalc }
320b8e59f18Smalc 
32185bc5852SKővágó, Zoltán static AudioFormat pa_to_audfmt (pa_sample_format_t fmt, int *endianness)
322b8e59f18Smalc {
323b8e59f18Smalc     switch (fmt) {
324b8e59f18Smalc     case PA_SAMPLE_U8:
32585bc5852SKővágó, Zoltán         return AUDIO_FORMAT_U8;
326b8e59f18Smalc     case PA_SAMPLE_S16BE:
327b8e59f18Smalc         *endianness = 1;
32885bc5852SKővágó, Zoltán         return AUDIO_FORMAT_S16;
329b8e59f18Smalc     case PA_SAMPLE_S16LE:
330b8e59f18Smalc         *endianness = 0;
33185bc5852SKővágó, Zoltán         return AUDIO_FORMAT_S16;
332b8e59f18Smalc     case PA_SAMPLE_S32BE:
333b8e59f18Smalc         *endianness = 1;
33485bc5852SKővágó, Zoltán         return AUDIO_FORMAT_S32;
335b8e59f18Smalc     case PA_SAMPLE_S32LE:
336b8e59f18Smalc         *endianness = 0;
33785bc5852SKővágó, Zoltán         return AUDIO_FORMAT_S32;
338ed2a4a79SKővágó, Zoltán     case PA_SAMPLE_FLOAT32BE:
339ed2a4a79SKővágó, Zoltán         *endianness = 1;
340ed2a4a79SKővágó, Zoltán         return AUDIO_FORMAT_F32;
341ed2a4a79SKővágó, Zoltán     case PA_SAMPLE_FLOAT32LE:
342ed2a4a79SKővágó, Zoltán         *endianness = 0;
343ed2a4a79SKővágó, Zoltán         return AUDIO_FORMAT_F32;
344b8e59f18Smalc     default:
345b8e59f18Smalc         dolog ("Internal logic error: Bad pa_sample_format %d\n", fmt);
34685bc5852SKővágó, Zoltán         return AUDIO_FORMAT_U8;
347b8e59f18Smalc     }
348b8e59f18Smalc }
349b8e59f18Smalc 
350ea9ebc2cSMarc-André Lureau static void context_state_cb (pa_context *c, void *userdata)
351ea9ebc2cSMarc-André Lureau {
3529d34e6d8SKővágó, Zoltán     PAConnection *conn = userdata;
353ea9ebc2cSMarc-André Lureau 
354ea9ebc2cSMarc-André Lureau     switch (pa_context_get_state(c)) {
355ea9ebc2cSMarc-André Lureau     case PA_CONTEXT_READY:
356ea9ebc2cSMarc-André Lureau     case PA_CONTEXT_TERMINATED:
357ea9ebc2cSMarc-André Lureau     case PA_CONTEXT_FAILED:
3589d34e6d8SKővágó, Zoltán         pa_threaded_mainloop_signal(conn->mainloop, 0);
359ea9ebc2cSMarc-André Lureau         break;
360ea9ebc2cSMarc-André Lureau 
361ea9ebc2cSMarc-André Lureau     case PA_CONTEXT_UNCONNECTED:
362ea9ebc2cSMarc-André Lureau     case PA_CONTEXT_CONNECTING:
363ea9ebc2cSMarc-André Lureau     case PA_CONTEXT_AUTHORIZING:
364ea9ebc2cSMarc-André Lureau     case PA_CONTEXT_SETTING_NAME:
365ea9ebc2cSMarc-André Lureau         break;
366ea9ebc2cSMarc-André Lureau     }
367ea9ebc2cSMarc-André Lureau }
368ea9ebc2cSMarc-André Lureau 
369ea9ebc2cSMarc-André Lureau static void stream_state_cb (pa_stream *s, void * userdata)
370ea9ebc2cSMarc-André Lureau {
3719d34e6d8SKővágó, Zoltán     PAConnection *c = userdata;
372ea9ebc2cSMarc-André Lureau 
373ea9ebc2cSMarc-André Lureau     switch (pa_stream_get_state (s)) {
374ea9ebc2cSMarc-André Lureau 
375ea9ebc2cSMarc-André Lureau     case PA_STREAM_READY:
376ea9ebc2cSMarc-André Lureau     case PA_STREAM_FAILED:
377ea9ebc2cSMarc-André Lureau     case PA_STREAM_TERMINATED:
3789d34e6d8SKővágó, Zoltán         pa_threaded_mainloop_signal(c->mainloop, 0);
379ea9ebc2cSMarc-André Lureau         break;
380ea9ebc2cSMarc-André Lureau 
381ea9ebc2cSMarc-André Lureau     case PA_STREAM_UNCONNECTED:
382ea9ebc2cSMarc-André Lureau     case PA_STREAM_CREATING:
383ea9ebc2cSMarc-André Lureau         break;
384ea9ebc2cSMarc-André Lureau     }
385ea9ebc2cSMarc-André Lureau }
386ea9ebc2cSMarc-André Lureau 
387ea9ebc2cSMarc-André Lureau static pa_stream *qpa_simple_new (
3889d34e6d8SKővágó, Zoltán         PAConnection *c,
389ea9ebc2cSMarc-André Lureau         const char *name,
390ea9ebc2cSMarc-André Lureau         pa_stream_direction_t dir,
391ea9ebc2cSMarc-André Lureau         const char *dev,
392ea9ebc2cSMarc-André Lureau         const pa_sample_spec *ss,
393ea9ebc2cSMarc-André Lureau         const pa_buffer_attr *attr,
394ea9ebc2cSMarc-André Lureau         int *rerror)
395ea9ebc2cSMarc-André Lureau {
396ea9ebc2cSMarc-André Lureau     int r;
3970cf13e36SKővágó, Zoltán     pa_stream *stream = NULL;
3989d34e6d8SKővágó, Zoltán     pa_stream_flags_t flags;
3990cf13e36SKővágó, Zoltán     pa_channel_map map;
400ea9ebc2cSMarc-André Lureau 
4019d34e6d8SKővágó, Zoltán     pa_threaded_mainloop_lock(c->mainloop);
402ea9ebc2cSMarc-André Lureau 
4030cf13e36SKővágó, Zoltán     pa_channel_map_init(&map);
4040cf13e36SKővágó, Zoltán     map.channels = ss->channels;
4050cf13e36SKővágó, Zoltán 
4060cf13e36SKővágó, Zoltán     /*
4070cf13e36SKővágó, Zoltán      * TODO: This currently expects the only frontend supporting more than 2
4080cf13e36SKővágó, Zoltán      * channels is the usb-audio.  We will need some means to set channel
4090cf13e36SKővágó, Zoltán      * order when a new frontend gains multi-channel support.
4100cf13e36SKővágó, Zoltán      */
4110cf13e36SKővágó, Zoltán     switch (ss->channels) {
4120cf13e36SKővágó, Zoltán     case 1:
4130cf13e36SKővágó, Zoltán         map.map[0] = PA_CHANNEL_POSITION_MONO;
4140cf13e36SKővágó, Zoltán         break;
4150cf13e36SKővágó, Zoltán 
4160cf13e36SKővágó, Zoltán     case 2:
4170cf13e36SKővágó, Zoltán         map.map[0] = PA_CHANNEL_POSITION_LEFT;
4180cf13e36SKővágó, Zoltán         map.map[1] = PA_CHANNEL_POSITION_RIGHT;
4190cf13e36SKővágó, Zoltán         break;
4200cf13e36SKővágó, Zoltán 
4210cf13e36SKővágó, Zoltán     case 6:
4220cf13e36SKővágó, Zoltán         map.map[0] = PA_CHANNEL_POSITION_FRONT_LEFT;
4230cf13e36SKővágó, Zoltán         map.map[1] = PA_CHANNEL_POSITION_FRONT_RIGHT;
4240cf13e36SKővágó, Zoltán         map.map[2] = PA_CHANNEL_POSITION_CENTER;
4250cf13e36SKővágó, Zoltán         map.map[3] = PA_CHANNEL_POSITION_LFE;
4260cf13e36SKővágó, Zoltán         map.map[4] = PA_CHANNEL_POSITION_REAR_LEFT;
4270cf13e36SKővágó, Zoltán         map.map[5] = PA_CHANNEL_POSITION_REAR_RIGHT;
4280cf13e36SKővágó, Zoltán         break;
4290cf13e36SKővágó, Zoltán 
4300cf13e36SKővágó, Zoltán     case 8:
4310cf13e36SKővágó, Zoltán         map.map[0] = PA_CHANNEL_POSITION_FRONT_LEFT;
4320cf13e36SKővágó, Zoltán         map.map[1] = PA_CHANNEL_POSITION_FRONT_RIGHT;
4330cf13e36SKővágó, Zoltán         map.map[2] = PA_CHANNEL_POSITION_CENTER;
4340cf13e36SKővágó, Zoltán         map.map[3] = PA_CHANNEL_POSITION_LFE;
4350cf13e36SKővágó, Zoltán         map.map[4] = PA_CHANNEL_POSITION_REAR_LEFT;
4360cf13e36SKővágó, Zoltán         map.map[5] = PA_CHANNEL_POSITION_REAR_RIGHT;
4370cf13e36SKővágó, Zoltán         map.map[6] = PA_CHANNEL_POSITION_SIDE_LEFT;
4380cf13e36SKővágó, Zoltán         map.map[7] = PA_CHANNEL_POSITION_SIDE_RIGHT;
43956089565SPaolo Bonzini         break;
4400cf13e36SKővágó, Zoltán 
4410cf13e36SKővágó, Zoltán     default:
4420cf13e36SKővágó, Zoltán         dolog("Internal error: unsupported channel count %d\n", ss->channels);
4430cf13e36SKővágó, Zoltán         goto fail;
4440cf13e36SKővágó, Zoltán     }
4450cf13e36SKővágó, Zoltán 
4460cf13e36SKővágó, Zoltán     stream = pa_stream_new(c->context, name, ss, &map);
447ea9ebc2cSMarc-André Lureau     if (!stream) {
448ea9ebc2cSMarc-André Lureau         goto fail;
449ea9ebc2cSMarc-André Lureau     }
450ea9ebc2cSMarc-André Lureau 
4519d34e6d8SKővágó, Zoltán     pa_stream_set_state_callback(stream, stream_state_cb, c);
4529d34e6d8SKővágó, Zoltán 
4539d34e6d8SKővágó, Zoltán     flags =
4549d34e6d8SKővágó, Zoltán         PA_STREAM_INTERPOLATE_TIMING
45510d5e750SKővágó, Zoltán         | PA_STREAM_AUTO_TIMING_UPDATE
45610d5e750SKővágó, Zoltán         | PA_STREAM_EARLY_REQUESTS;
457ea9ebc2cSMarc-André Lureau 
4588a435f74SKővágó, Zoltán     if (dev) {
4598a435f74SKővágó, Zoltán         /* don't move the stream if the user specified a sink/source */
4608a435f74SKővágó, Zoltán         flags |= PA_STREAM_DONT_MOVE;
4618a435f74SKővágó, Zoltán     }
4628a435f74SKővágó, Zoltán 
463ea9ebc2cSMarc-André Lureau     if (dir == PA_STREAM_PLAYBACK) {
4649d34e6d8SKővágó, Zoltán         r = pa_stream_connect_playback(stream, dev, attr, flags, NULL, NULL);
465ea9ebc2cSMarc-André Lureau     } else {
4669d34e6d8SKővágó, Zoltán         r = pa_stream_connect_record(stream, dev, attr, flags);
467ea9ebc2cSMarc-André Lureau     }
468ea9ebc2cSMarc-André Lureau 
469ea9ebc2cSMarc-André Lureau     if (r < 0) {
470ea9ebc2cSMarc-André Lureau       goto fail;
471ea9ebc2cSMarc-André Lureau     }
472ea9ebc2cSMarc-André Lureau 
4739d34e6d8SKővágó, Zoltán     pa_threaded_mainloop_unlock(c->mainloop);
474ea9ebc2cSMarc-André Lureau 
475ea9ebc2cSMarc-André Lureau     return stream;
476ea9ebc2cSMarc-André Lureau 
477ea9ebc2cSMarc-André Lureau fail:
4789d34e6d8SKővágó, Zoltán     pa_threaded_mainloop_unlock(c->mainloop);
479ea9ebc2cSMarc-André Lureau 
480ea9ebc2cSMarc-André Lureau     if (stream) {
481ea9ebc2cSMarc-André Lureau         pa_stream_unref (stream);
482ea9ebc2cSMarc-André Lureau     }
483ea9ebc2cSMarc-André Lureau 
4849d34e6d8SKővágó, Zoltán     *rerror = pa_context_errno(c->context);
485ea9ebc2cSMarc-André Lureau 
486ea9ebc2cSMarc-André Lureau     return NULL;
487ea9ebc2cSMarc-André Lureau }
488ea9ebc2cSMarc-André Lureau 
4895706db1dSKővágó, Zoltán static int qpa_init_out(HWVoiceOut *hw, struct audsettings *as,
4905706db1dSKővágó, Zoltán                         void *drv_opaque)
491b8e59f18Smalc {
492b8e59f18Smalc     int error;
4939a644c4bSKővágó, Zoltán     pa_sample_spec ss;
4949a644c4bSKővágó, Zoltán     pa_buffer_attr ba;
4951ea879e5Smalc     struct audsettings obt_as = *as;
496b8e59f18Smalc     PAVoiceOut *pa = (PAVoiceOut *) hw;
4979a644c4bSKővágó, Zoltán     paaudio *g = pa->g = drv_opaque;
4982c324b28SKővágó, Zoltán     AudiodevPaOptions *popts = &g->dev->u.pa;
4992c324b28SKővágó, Zoltán     AudiodevPaPerDirectionOptions *ppdo = popts->out;
5009d34e6d8SKővágó, Zoltán     PAConnection *c = g->conn;
501b8e59f18Smalc 
502b8e59f18Smalc     ss.format = audfmt_to_pa (as->fmt, as->endianness);
503b8e59f18Smalc     ss.channels = as->nchannels;
504b8e59f18Smalc     ss.rate = as->freq;
505b8e59f18Smalc 
506f6142777SMartin Schrodt     ba.tlength = pa_usec_to_bytes(ppdo->latency, &ss);
507f6142777SMartin Schrodt     ba.minreq = -1;
508e6d16fa4SGerd Hoffmann     ba.maxlength = -1;
509e6d16fa4SGerd Hoffmann     ba.prebuf = -1;
510e6d16fa4SGerd Hoffmann 
511b8e59f18Smalc     obt_as.fmt = pa_to_audfmt (ss.format, &obt_as.endianness);
512b8e59f18Smalc 
513ea9ebc2cSMarc-André Lureau     pa->stream = qpa_simple_new (
5149d34e6d8SKővágó, Zoltán         c,
515f47dffe8SKővágó, Zoltán         ppdo->has_stream_name ? ppdo->stream_name : g->dev->id,
516b8e59f18Smalc         PA_STREAM_PLAYBACK,
5172c324b28SKővágó, Zoltán         ppdo->has_name ? ppdo->name : NULL,
518b8e59f18Smalc         &ss,
519e6d16fa4SGerd Hoffmann         &ba,                    /* buffering attributes */
520b8e59f18Smalc         &error
521b8e59f18Smalc         );
522ea9ebc2cSMarc-André Lureau     if (!pa->stream) {
523b8e59f18Smalc         qpa_logerr (error, "pa_simple_new for playback failed\n");
524b8e59f18Smalc         goto fail1;
525b8e59f18Smalc     }
526b8e59f18Smalc 
527b8e59f18Smalc     audio_pcm_init_info (&hw->info, &obt_as);
528a76e6b87SVolker Rümelin     hw->samples = audio_buffer_samples(
529baea032eSMartin Schrodt         qapi_AudiodevPaPerDirectionOptions_base(ppdo),
530baea032eSMartin Schrodt         &obt_as, ppdo->buffer_length);
531b8e59f18Smalc 
532b8e59f18Smalc     return 0;
533b8e59f18Smalc 
534b8e59f18Smalc  fail1:
535b8e59f18Smalc     return -1;
536b8e59f18Smalc }
537b8e59f18Smalc 
5385706db1dSKővágó, Zoltán static int qpa_init_in(HWVoiceIn *hw, struct audsettings *as, void *drv_opaque)
539b8e59f18Smalc {
540b8e59f18Smalc     int error;
5419a644c4bSKővágó, Zoltán     pa_sample_spec ss;
542ade10301SMartin Schrodt     pa_buffer_attr ba;
5431ea879e5Smalc     struct audsettings obt_as = *as;
544b8e59f18Smalc     PAVoiceIn *pa = (PAVoiceIn *) hw;
5459a644c4bSKővágó, Zoltán     paaudio *g = pa->g = drv_opaque;
5462c324b28SKővágó, Zoltán     AudiodevPaOptions *popts = &g->dev->u.pa;
5472c324b28SKővágó, Zoltán     AudiodevPaPerDirectionOptions *ppdo = popts->in;
5489d34e6d8SKővágó, Zoltán     PAConnection *c = g->conn;
549b8e59f18Smalc 
550b8e59f18Smalc     ss.format = audfmt_to_pa (as->fmt, as->endianness);
551b8e59f18Smalc     ss.channels = as->nchannels;
552b8e59f18Smalc     ss.rate = as->freq;
553b8e59f18Smalc 
554ade10301SMartin Schrodt     ba.fragsize = pa_usec_to_bytes(ppdo->latency, &ss);
55558c15e52SMartin Schrodt     ba.maxlength = pa_usec_to_bytes(ppdo->latency * 2, &ss);
556ade10301SMartin Schrodt     ba.minreq = -1;
557ade10301SMartin Schrodt     ba.prebuf = -1;
558ade10301SMartin Schrodt 
559b8e59f18Smalc     obt_as.fmt = pa_to_audfmt (ss.format, &obt_as.endianness);
560b8e59f18Smalc 
561ea9ebc2cSMarc-André Lureau     pa->stream = qpa_simple_new (
5629d34e6d8SKővágó, Zoltán         c,
563f47dffe8SKővágó, Zoltán         ppdo->has_stream_name ? ppdo->stream_name : g->dev->id,
564b8e59f18Smalc         PA_STREAM_RECORD,
5652c324b28SKővágó, Zoltán         ppdo->has_name ? ppdo->name : NULL,
566b8e59f18Smalc         &ss,
567ade10301SMartin Schrodt         &ba,                    /* buffering attributes */
568b8e59f18Smalc         &error
569b8e59f18Smalc         );
570ea9ebc2cSMarc-André Lureau     if (!pa->stream) {
571b8e59f18Smalc         qpa_logerr (error, "pa_simple_new for capture failed\n");
572b8e59f18Smalc         goto fail1;
573b8e59f18Smalc     }
574b8e59f18Smalc 
575b8e59f18Smalc     audio_pcm_init_info (&hw->info, &obt_as);
576a76e6b87SVolker Rümelin     hw->samples = audio_buffer_samples(
577baea032eSMartin Schrodt         qapi_AudiodevPaPerDirectionOptions_base(ppdo),
578baea032eSMartin Schrodt         &obt_as, ppdo->buffer_length);
579b8e59f18Smalc 
580b8e59f18Smalc     return 0;
581b8e59f18Smalc 
582b8e59f18Smalc  fail1:
583b8e59f18Smalc     return -1;
584b8e59f18Smalc }
585b8e59f18Smalc 
5868692bf7dSKővágó, Zoltán static void qpa_simple_disconnect(PAConnection *c, pa_stream *stream)
5878692bf7dSKővágó, Zoltán {
5888692bf7dSKővágó, Zoltán     int err;
5898692bf7dSKővágó, Zoltán 
5908692bf7dSKővágó, Zoltán     /*
5918692bf7dSKővágó, Zoltán      * wait until actually connects. workaround pa bug #247
5928692bf7dSKővágó, Zoltán      * https://gitlab.freedesktop.org/pulseaudio/pulseaudio/issues/247
5938692bf7dSKővágó, Zoltán      */
5948692bf7dSKővágó, Zoltán     while (pa_stream_get_state(stream) == PA_STREAM_CREATING) {
5958692bf7dSKővágó, Zoltán         pa_threaded_mainloop_wait(c->mainloop);
5968692bf7dSKővágó, Zoltán     }
5978692bf7dSKővágó, Zoltán 
5988692bf7dSKővágó, Zoltán     err = pa_stream_disconnect(stream);
5998692bf7dSKővágó, Zoltán     if (err != 0) {
6008692bf7dSKővágó, Zoltán         dolog("Failed to disconnect! err=%d\n", err);
6018692bf7dSKővágó, Zoltán     }
6028692bf7dSKővágó, Zoltán     pa_stream_unref(stream);
6038692bf7dSKővágó, Zoltán }
6048692bf7dSKővágó, Zoltán 
605b8e59f18Smalc static void qpa_fini_out (HWVoiceOut *hw)
606b8e59f18Smalc {
607b8e59f18Smalc     PAVoiceOut *pa = (PAVoiceOut *) hw;
608b8e59f18Smalc 
609ea9ebc2cSMarc-André Lureau     if (pa->stream) {
6104db3e634SVolker Rümelin         PAConnection *c = pa->g->conn;
6114db3e634SVolker Rümelin 
6124db3e634SVolker Rümelin         pa_threaded_mainloop_lock(c->mainloop);
6134db3e634SVolker Rümelin         qpa_simple_disconnect(c, pa->stream);
614ea9ebc2cSMarc-André Lureau         pa->stream = NULL;
6154db3e634SVolker Rümelin         pa_threaded_mainloop_unlock(c->mainloop);
616b8e59f18Smalc     }
617b8e59f18Smalc }
618b8e59f18Smalc 
619b8e59f18Smalc static void qpa_fini_in (HWVoiceIn *hw)
620b8e59f18Smalc {
621b8e59f18Smalc     PAVoiceIn *pa = (PAVoiceIn *) hw;
622b8e59f18Smalc 
623ea9ebc2cSMarc-André Lureau     if (pa->stream) {
6244db3e634SVolker Rümelin         PAConnection *c = pa->g->conn;
6254db3e634SVolker Rümelin 
6264db3e634SVolker Rümelin         pa_threaded_mainloop_lock(c->mainloop);
6274db3e634SVolker Rümelin         if (pa->read_length) {
6284db3e634SVolker Rümelin             int r = pa_stream_drop(pa->stream);
6294db3e634SVolker Rümelin             if (r) {
6304db3e634SVolker Rümelin                 qpa_logerr(pa_context_errno(c->context),
6314db3e634SVolker Rümelin                            "pa_stream_drop failed\n");
6324db3e634SVolker Rümelin             }
6334db3e634SVolker Rümelin             pa->read_length = 0;
6344db3e634SVolker Rümelin         }
6354db3e634SVolker Rümelin         qpa_simple_disconnect(c, pa->stream);
636ea9ebc2cSMarc-André Lureau         pa->stream = NULL;
6374db3e634SVolker Rümelin         pa_threaded_mainloop_unlock(c->mainloop);
638b8e59f18Smalc     }
639b8e59f18Smalc }
640b8e59f18Smalc 
641cecc1e79SKővágó, Zoltán static void qpa_volume_out(HWVoiceOut *hw, Volume *vol)
642b8e59f18Smalc {
6436e7a7f3dSMarc-André Lureau     PAVoiceOut *pa = (PAVoiceOut *) hw;
6446e7a7f3dSMarc-André Lureau     pa_operation *op;
6456e7a7f3dSMarc-André Lureau     pa_cvolume v;
6469d34e6d8SKővágó, Zoltán     PAConnection *c = pa->g->conn;
647cecc1e79SKővágó, Zoltán     int i;
6486e7a7f3dSMarc-André Lureau 
6498f473dd1SGerd Hoffmann #ifdef PA_CHECK_VERSION    /* macro is present in 0.9.16+ */
6508f473dd1SGerd Hoffmann     pa_cvolume_init (&v);  /* function is present in 0.9.13+ */
6518f473dd1SGerd Hoffmann #endif
6526e7a7f3dSMarc-André Lureau 
653cecc1e79SKővágó, Zoltán     v.channels = vol->channels;
654cecc1e79SKővágó, Zoltán     for (i = 0; i < vol->channels; ++i) {
655cecc1e79SKővágó, Zoltán         v.values[i] = ((PA_VOLUME_NORM - PA_VOLUME_MUTED) * vol->vol[i]) / 255;
656cecc1e79SKővágó, Zoltán     }
6576e7a7f3dSMarc-André Lureau 
6589d34e6d8SKővágó, Zoltán     pa_threaded_mainloop_lock(c->mainloop);
6596e7a7f3dSMarc-André Lureau 
6609d34e6d8SKővágó, Zoltán     op = pa_context_set_sink_input_volume(c->context,
6616e7a7f3dSMarc-André Lureau                                           pa_stream_get_index(pa->stream),
6626e7a7f3dSMarc-André Lureau                                           &v, NULL, NULL);
6639d34e6d8SKővágó, Zoltán     if (!op) {
6649d34e6d8SKővágó, Zoltán         qpa_logerr(pa_context_errno(c->context),
6656e7a7f3dSMarc-André Lureau                    "set_sink_input_volume() failed\n");
6669d34e6d8SKővágó, Zoltán     } else {
6676e7a7f3dSMarc-André Lureau         pa_operation_unref(op);
6689d34e6d8SKővágó, Zoltán     }
6696e7a7f3dSMarc-André Lureau 
6709d34e6d8SKővágó, Zoltán     op = pa_context_set_sink_input_mute(c->context,
6716e7a7f3dSMarc-André Lureau                                         pa_stream_get_index(pa->stream),
672571a8c52SKővágó, Zoltán                                         vol->mute, NULL, NULL);
6736e7a7f3dSMarc-André Lureau     if (!op) {
6749d34e6d8SKővágó, Zoltán         qpa_logerr(pa_context_errno(c->context),
6756e7a7f3dSMarc-André Lureau                    "set_sink_input_mute() failed\n");
6766e7a7f3dSMarc-André Lureau     } else {
6776e7a7f3dSMarc-André Lureau         pa_operation_unref(op);
6786e7a7f3dSMarc-André Lureau     }
6796e7a7f3dSMarc-André Lureau 
6809d34e6d8SKővágó, Zoltán     pa_threaded_mainloop_unlock(c->mainloop);
6816e7a7f3dSMarc-André Lureau }
682b8e59f18Smalc 
683cecc1e79SKővágó, Zoltán static void qpa_volume_in(HWVoiceIn *hw, Volume *vol)
684b8e59f18Smalc {
6856e7a7f3dSMarc-André Lureau     PAVoiceIn *pa = (PAVoiceIn *) hw;
6866e7a7f3dSMarc-André Lureau     pa_operation *op;
6876e7a7f3dSMarc-André Lureau     pa_cvolume v;
6889d34e6d8SKővágó, Zoltán     PAConnection *c = pa->g->conn;
689cecc1e79SKővágó, Zoltán     int i;
6906e7a7f3dSMarc-André Lureau 
6918f473dd1SGerd Hoffmann #ifdef PA_CHECK_VERSION
6926e7a7f3dSMarc-André Lureau     pa_cvolume_init (&v);
6938f473dd1SGerd Hoffmann #endif
6946e7a7f3dSMarc-André Lureau 
695cecc1e79SKővágó, Zoltán     v.channels = vol->channels;
696cecc1e79SKővágó, Zoltán     for (i = 0; i < vol->channels; ++i) {
697cecc1e79SKővágó, Zoltán         v.values[i] = ((PA_VOLUME_NORM - PA_VOLUME_MUTED) * vol->vol[i]) / 255;
698cecc1e79SKővágó, Zoltán     }
6996e7a7f3dSMarc-André Lureau 
7009d34e6d8SKővágó, Zoltán     pa_threaded_mainloop_lock(c->mainloop);
7016e7a7f3dSMarc-André Lureau 
7029d34e6d8SKővágó, Zoltán     op = pa_context_set_source_output_volume(c->context,
703e58ff62dSPeter Krempa         pa_stream_get_index(pa->stream),
7046e7a7f3dSMarc-André Lureau         &v, NULL, NULL);
7056e7a7f3dSMarc-André Lureau     if (!op) {
7069d34e6d8SKővágó, Zoltán         qpa_logerr(pa_context_errno(c->context),
707e58ff62dSPeter Krempa                    "set_source_output_volume() failed\n");
7086e7a7f3dSMarc-André Lureau     } else {
7096e7a7f3dSMarc-André Lureau         pa_operation_unref(op);
7106e7a7f3dSMarc-André Lureau     }
7116e7a7f3dSMarc-André Lureau 
7129d34e6d8SKővágó, Zoltán     op = pa_context_set_source_output_mute(c->context,
7136e7a7f3dSMarc-André Lureau         pa_stream_get_index(pa->stream),
714571a8c52SKővágó, Zoltán         vol->mute, NULL, NULL);
7156e7a7f3dSMarc-André Lureau     if (!op) {
7169d34e6d8SKővágó, Zoltán         qpa_logerr(pa_context_errno(c->context),
717e58ff62dSPeter Krempa                    "set_source_output_mute() failed\n");
7186e7a7f3dSMarc-André Lureau     } else {
7196e7a7f3dSMarc-André Lureau         pa_operation_unref(op);
7206e7a7f3dSMarc-André Lureau     }
7216e7a7f3dSMarc-André Lureau 
7229d34e6d8SKővágó, Zoltán     pa_threaded_mainloop_unlock(c->mainloop);
7236e7a7f3dSMarc-André Lureau }
724b8e59f18Smalc 
725baea032eSMartin Schrodt static int qpa_validate_per_direction_opts(Audiodev *dev,
726baea032eSMartin Schrodt                                            AudiodevPaPerDirectionOptions *pdo)
727baea032eSMartin Schrodt {
728baea032eSMartin Schrodt     if (!pdo->has_buffer_length) {
729baea032eSMartin Schrodt         pdo->has_buffer_length = true;
730baea032eSMartin Schrodt         pdo->buffer_length = 46440;
731baea032eSMartin Schrodt     }
732f6142777SMartin Schrodt     if (!pdo->has_latency) {
733f6142777SMartin Schrodt         pdo->has_latency = true;
734f6142777SMartin Schrodt         pdo->latency = 15000;
735f6142777SMartin Schrodt     }
736baea032eSMartin Schrodt     return 1;
737baea032eSMartin Schrodt }
738baea032eSMartin Schrodt 
7399d34e6d8SKővágó, Zoltán /* common */
7409d34e6d8SKővágó, Zoltán static void *qpa_conn_init(const char *server)
7419d34e6d8SKővágó, Zoltán {
7423443ad4eSKővágó, Zoltán     const char *vm_name;
7439d34e6d8SKővágó, Zoltán     PAConnection *c = g_malloc0(sizeof(PAConnection));
7449d34e6d8SKővágó, Zoltán     QTAILQ_INSERT_TAIL(&pa_conns, c, list);
7459d34e6d8SKővágó, Zoltán 
7469d34e6d8SKővágó, Zoltán     c->mainloop = pa_threaded_mainloop_new();
7479d34e6d8SKővágó, Zoltán     if (!c->mainloop) {
7489d34e6d8SKővágó, Zoltán         goto fail;
7499d34e6d8SKővágó, Zoltán     }
7509d34e6d8SKővágó, Zoltán 
7513443ad4eSKővágó, Zoltán     vm_name = qemu_get_vm_name();
7529d34e6d8SKővágó, Zoltán     c->context = pa_context_new(pa_threaded_mainloop_get_api(c->mainloop),
7533443ad4eSKővágó, Zoltán                                 vm_name ? vm_name : "qemu");
7549d34e6d8SKővágó, Zoltán     if (!c->context) {
7559d34e6d8SKővágó, Zoltán         goto fail;
7569d34e6d8SKővágó, Zoltán     }
7579d34e6d8SKővágó, Zoltán 
7589d34e6d8SKővágó, Zoltán     pa_context_set_state_callback(c->context, context_state_cb, c);
7599d34e6d8SKővágó, Zoltán 
7609d34e6d8SKővágó, Zoltán     if (pa_context_connect(c->context, server, 0, NULL) < 0) {
7619d34e6d8SKővágó, Zoltán         qpa_logerr(pa_context_errno(c->context),
7629d34e6d8SKővágó, Zoltán                    "pa_context_connect() failed\n");
7639d34e6d8SKővágó, Zoltán         goto fail;
7649d34e6d8SKővágó, Zoltán     }
7659d34e6d8SKővágó, Zoltán 
7669d34e6d8SKővágó, Zoltán     pa_threaded_mainloop_lock(c->mainloop);
7679d34e6d8SKővágó, Zoltán 
7689d34e6d8SKővágó, Zoltán     if (pa_threaded_mainloop_start(c->mainloop) < 0) {
7699d34e6d8SKővágó, Zoltán         goto unlock_and_fail;
7709d34e6d8SKővágó, Zoltán     }
7719d34e6d8SKővágó, Zoltán 
7729d34e6d8SKővágó, Zoltán     for (;;) {
7739d34e6d8SKővágó, Zoltán         pa_context_state_t state;
7749d34e6d8SKővágó, Zoltán 
7759d34e6d8SKővágó, Zoltán         state = pa_context_get_state(c->context);
7769d34e6d8SKővágó, Zoltán 
7779d34e6d8SKővágó, Zoltán         if (state == PA_CONTEXT_READY) {
7789d34e6d8SKővágó, Zoltán             break;
7799d34e6d8SKővágó, Zoltán         }
7809d34e6d8SKővágó, Zoltán 
7819d34e6d8SKővágó, Zoltán         if (!PA_CONTEXT_IS_GOOD(state)) {
7829d34e6d8SKővágó, Zoltán             qpa_logerr(pa_context_errno(c->context),
7839d34e6d8SKővágó, Zoltán                        "Wrong context state\n");
7849d34e6d8SKővágó, Zoltán             goto unlock_and_fail;
7859d34e6d8SKővágó, Zoltán         }
7869d34e6d8SKővágó, Zoltán 
7879d34e6d8SKővágó, Zoltán         /* Wait until the context is ready */
7889d34e6d8SKővágó, Zoltán         pa_threaded_mainloop_wait(c->mainloop);
7899d34e6d8SKővágó, Zoltán     }
7909d34e6d8SKővágó, Zoltán 
7919d34e6d8SKővágó, Zoltán     pa_threaded_mainloop_unlock(c->mainloop);
7929d34e6d8SKővágó, Zoltán     return c;
7939d34e6d8SKővágó, Zoltán 
7949d34e6d8SKővágó, Zoltán unlock_and_fail:
7959d34e6d8SKővágó, Zoltán     pa_threaded_mainloop_unlock(c->mainloop);
7969d34e6d8SKővágó, Zoltán fail:
7979d34e6d8SKővágó, Zoltán     AUD_log (AUDIO_CAP, "Failed to initialize PA context");
7989d34e6d8SKővágó, Zoltán     qpa_conn_fini(c);
7999d34e6d8SKővágó, Zoltán     return NULL;
8009d34e6d8SKővágó, Zoltán }
8019d34e6d8SKővágó, Zoltán 
80271830221SKővágó, Zoltán static void *qpa_audio_init(Audiodev *dev)
803b8e59f18Smalc {
8042c324b28SKővágó, Zoltán     paaudio *g;
8052c324b28SKővágó, Zoltán     AudiodevPaOptions *popts = &dev->u.pa;
8062c324b28SKővágó, Zoltán     const char *server;
8079d34e6d8SKővágó, Zoltán     PAConnection *c;
8089d34e6d8SKővágó, Zoltán 
8099d34e6d8SKővágó, Zoltán     assert(dev->driver == AUDIODEV_DRIVER_PA);
8102c324b28SKővágó, Zoltán 
8112c324b28SKővágó, Zoltán     if (!popts->has_server) {
812d175505bSGerd Hoffmann         char pidfile[64];
813d175505bSGerd Hoffmann         char *runtime;
814d175505bSGerd Hoffmann         struct stat st;
815d175505bSGerd Hoffmann 
816d175505bSGerd Hoffmann         runtime = getenv("XDG_RUNTIME_DIR");
817d175505bSGerd Hoffmann         if (!runtime) {
818d175505bSGerd Hoffmann             return NULL;
819d175505bSGerd Hoffmann         }
820d175505bSGerd Hoffmann         snprintf(pidfile, sizeof(pidfile), "%s/pulse/pid", runtime);
821d175505bSGerd Hoffmann         if (stat(pidfile, &st) != 0) {
822d175505bSGerd Hoffmann             return NULL;
823d175505bSGerd Hoffmann         }
824d175505bSGerd Hoffmann     }
825d175505bSGerd Hoffmann 
826baea032eSMartin Schrodt     if (!qpa_validate_per_direction_opts(dev, popts->in)) {
8279d34e6d8SKővágó, Zoltán         return NULL;
828baea032eSMartin Schrodt     }
829baea032eSMartin Schrodt     if (!qpa_validate_per_direction_opts(dev, popts->out)) {
8309d34e6d8SKővágó, Zoltán         return NULL;
831baea032eSMartin Schrodt     }
832baea032eSMartin Schrodt 
8339d34e6d8SKővágó, Zoltán     g = g_malloc0(sizeof(paaudio));
8349d34e6d8SKővágó, Zoltán     server = popts->has_server ? popts->server : NULL;
8359d34e6d8SKővágó, Zoltán 
8362c324b28SKővágó, Zoltán     g->dev = dev;
837ea9ebc2cSMarc-André Lureau 
8389d34e6d8SKővágó, Zoltán     QTAILQ_FOREACH(c, &pa_conns, list) {
8399d34e6d8SKővágó, Zoltán         if (server == NULL || c->server == NULL ?
8409d34e6d8SKővágó, Zoltán             server == c->server :
8419d34e6d8SKővágó, Zoltán             strcmp(server, c->server) == 0) {
8429d34e6d8SKővágó, Zoltán             g->conn = c;
843ea9ebc2cSMarc-André Lureau             break;
844ea9ebc2cSMarc-André Lureau         }
845ea9ebc2cSMarc-André Lureau     }
8469d34e6d8SKővágó, Zoltán     if (!g->conn) {
8479d34e6d8SKővágó, Zoltán         g->conn = qpa_conn_init(server);
848ea9ebc2cSMarc-André Lureau     }
8499d34e6d8SKővágó, Zoltán     if (!g->conn) {
8509d34e6d8SKővágó, Zoltán         g_free(g);
851ea9ebc2cSMarc-André Lureau         return NULL;
852b8e59f18Smalc     }
853b8e59f18Smalc 
8549d34e6d8SKővágó, Zoltán     ++g->conn->refcount;
8559d34e6d8SKővágó, Zoltán     return g;
8569d34e6d8SKővágó, Zoltán }
8579d34e6d8SKővágó, Zoltán 
8589d34e6d8SKővágó, Zoltán static void qpa_conn_fini(PAConnection *c)
8599d34e6d8SKővágó, Zoltán {
8609d34e6d8SKővágó, Zoltán     if (c->mainloop) {
8619d34e6d8SKővágó, Zoltán         pa_threaded_mainloop_stop(c->mainloop);
8629d34e6d8SKővágó, Zoltán     }
8639d34e6d8SKővágó, Zoltán 
8649d34e6d8SKővágó, Zoltán     if (c->context) {
8659d34e6d8SKővágó, Zoltán         pa_context_disconnect(c->context);
8669d34e6d8SKővágó, Zoltán         pa_context_unref(c->context);
8679d34e6d8SKővágó, Zoltán     }
8689d34e6d8SKővágó, Zoltán 
8699d34e6d8SKővágó, Zoltán     if (c->mainloop) {
8709d34e6d8SKővágó, Zoltán         pa_threaded_mainloop_free(c->mainloop);
8719d34e6d8SKővágó, Zoltán     }
8729d34e6d8SKővágó, Zoltán 
8739d34e6d8SKővágó, Zoltán     QTAILQ_REMOVE(&pa_conns, c, list);
8749d34e6d8SKővágó, Zoltán     g_free(c);
8759d34e6d8SKővágó, Zoltán }
8769d34e6d8SKővágó, Zoltán 
877b8e59f18Smalc static void qpa_audio_fini (void *opaque)
878b8e59f18Smalc {
879ea9ebc2cSMarc-André Lureau     paaudio *g = opaque;
8809d34e6d8SKővágó, Zoltán     PAConnection *c = g->conn;
881ea9ebc2cSMarc-André Lureau 
8829d34e6d8SKővágó, Zoltán     if (--c->refcount == 0) {
8839d34e6d8SKővágó, Zoltán         qpa_conn_fini(c);
884ea9ebc2cSMarc-André Lureau     }
885ea9ebc2cSMarc-André Lureau 
8869a644c4bSKővágó, Zoltán     g_free(g);
887b8e59f18Smalc }
888b8e59f18Smalc 
88935f4b58cSblueswir1 static struct audio_pcm_ops qpa_pcm_ops = {
8901dd3e4d1SJuan Quintela     .init_out = qpa_init_out,
8911dd3e4d1SJuan Quintela     .fini_out = qpa_fini_out,
89249ddd7e1SKővágó, Zoltán     .write    = qpa_write,
893337e8de6SKővágó, Zoltán     .get_buffer_out = qpa_get_buffer_out,
894*bea29e9fSVolker Rümelin     .put_buffer_out = qpa_put_buffer_out,
895571a8c52SKővágó, Zoltán     .volume_out = qpa_volume_out,
8961dd3e4d1SJuan Quintela 
8971dd3e4d1SJuan Quintela     .init_in  = qpa_init_in,
8981dd3e4d1SJuan Quintela     .fini_in  = qpa_fini_in,
89949ddd7e1SKővágó, Zoltán     .read     = qpa_read,
900337e8de6SKővágó, Zoltán     .get_buffer_in = qpa_get_buffer_in,
901337e8de6SKővágó, Zoltán     .put_buffer_in = qpa_put_buffer_in,
902571a8c52SKővágó, Zoltán     .volume_in = qpa_volume_in
903b8e59f18Smalc };
904b8e59f18Smalc 
905d3893a39SGerd Hoffmann static struct audio_driver pa_audio_driver = {
906bee37f32SJuan Quintela     .name           = "pa",
907bee37f32SJuan Quintela     .descr          = "http://www.pulseaudio.org/",
908bee37f32SJuan Quintela     .init           = qpa_audio_init,
909bee37f32SJuan Quintela     .fini           = qpa_audio_fini,
910bee37f32SJuan Quintela     .pcm_ops        = &qpa_pcm_ops,
9111a4ea1e3SMichael S. Tsirkin     .can_be_default = 1,
912bee37f32SJuan Quintela     .max_voices_out = INT_MAX,
913bee37f32SJuan Quintela     .max_voices_in  = INT_MAX,
914bee37f32SJuan Quintela     .voice_size_out = sizeof (PAVoiceOut),
9156e7a7f3dSMarc-André Lureau     .voice_size_in  = sizeof (PAVoiceIn),
916b8e59f18Smalc };
917d3893a39SGerd Hoffmann 
918d3893a39SGerd Hoffmann static void register_audio_pa(void)
919d3893a39SGerd Hoffmann {
920d3893a39SGerd Hoffmann     audio_driver_register(&pa_audio_driver);
921d3893a39SGerd Hoffmann }
922d3893a39SGerd Hoffmann type_init(register_audio_pa);
923