]> git.sesse.net Git - vlc/blob - modules/audio_output/oss.c
OSS: updates and fixes (untested)
[vlc] / modules / audio_output / oss.c
1 /*****************************************************************************
2  * oss.c: Open Sound System audio output plugin for VLC
3  *****************************************************************************
4  * Copyright (C) 2000-2002 VLC authors and VideoLAN
5  * Copyright (C) 2007-2012 RĂ©mi Denis-Courmont
6  *
7  * Authors: Michel Kaempf <maxx@via.ecp.fr>
8  *          Sam Hocevar <sam@zoy.org>
9  *          Christophe Massiot <massiot@via.ecp.fr>
10  *
11  * This program is free software; you can redistribute it and/or modify it
12  * under the terms of the GNU Lesser General Public License as published by
13  * the Free Software Foundation; either version 2.1 of the License, or
14  * (at your option) any later version.
15  *
16  * This program is distributed in the hope that it will be useful,
17  * but WITHOUT ANY WARRANTY; without even the implied warranty of
18  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19  * GNU Lesser General Public License for more details.
20  *
21  * You should have received a copy of the GNU Lesser General Public License
22  * along with this program; if not, write to the Free Software Foundation,
23  * Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
24  *****************************************************************************/
25
26 #ifdef HAVE_CONFIG_H
27 # include "config.h"
28 #endif
29
30 #include <stdlib.h>
31 #include <math.h>
32 #include <sys/types.h>
33 #include <fcntl.h>
34 #include <sys/ioctl.h>
35 #ifdef HAVE_SOUNDCARD_H
36 # include <soundcard.h>
37 #else
38 # include <sys/soundcard.h>
39 #endif
40
41 #ifndef SNDCTL_DSP_HALT
42 # define SNDCTL_DSP_HALT SNDCTL_DSP_RESET
43 #endif
44
45 #include <vlc_common.h>
46 #include <vlc_plugin.h>
47 #include <vlc_fs.h>
48 #include <vlc_cpu.h>
49 #include <vlc_aout.h>
50
51 #define A52_FRAME_NB 1536
52
53 struct aout_sys_t
54 {
55     int fd;
56     audio_sample_format_t format;
57     bool starting;
58
59     bool mute;
60     uint8_t level;
61     char *device;
62 };
63
64 static int Open (vlc_object_t *);
65 static void Close (vlc_object_t *);
66
67 #define AUDIO_DEV_TEXT N_("Audio output device")
68 #define AUDIO_DEV_LONGTEXT N_("OSS device node path.")
69
70 vlc_module_begin ()
71     set_shortname( "OSS" )
72     set_description (N_("Open Sound System audio output"))
73     set_category( CAT_AUDIO )
74     set_subcategory( SUBCAT_AUDIO_AOUT )
75     add_string ("oss-audio-device", "",
76                 AUDIO_DEV_TEXT, AUDIO_DEV_LONGTEXT, false)
77     set_capability( "audio output", 100 )
78     set_callbacks (Open, Close)
79 vlc_module_end ()
80
81 static int TimeGet (audio_output_t *, mtime_t *);
82 static void Play (audio_output_t *, block_t *);
83 static void Pause (audio_output_t *, bool, mtime_t);
84 static void Flush (audio_output_t *, bool);
85 static int VolumeSync (audio_output_t *);
86
87 static int Start (audio_output_t *aout, audio_sample_format_t *restrict fmt)
88 {
89     aout_sys_t* sys = aout->sys;
90
91     /* Open the device */
92     const char *device = sys->device;
93     if (device == NULL)
94         device = getenv ("OSS_AUDIODEV");
95     if (device == NULL)
96         device = "/dev/dsp";
97
98     int fd = vlc_open (device, O_WRONLY);
99     if (fd == -1)
100     {
101         msg_Err (aout, "cannot open OSS device %s: %m", device);
102         return VLC_EGENERIC;
103     }
104     sys->fd = fd;
105     msg_Dbg (aout, "using OSS device: %s", device);
106
107     /* Select audio format */
108     int format;
109     bool spdif = false;
110
111     switch (fmt->i_format)
112     {
113 #ifdef AFMT_FLOAT
114         case VLC_CODEC_F64B:
115         case VLC_CODEC_F64L:
116         case VLC_CODEC_F32B:
117         case VLC_CODEC_F32L:
118             format = AFMT_FLOAT;
119             break;
120 #endif
121         case VLC_CODEC_S32B:
122             format = AFMT_S32_BE;
123             break;
124         case VLC_CODEC_S32L:
125             format = AFMT_S32_LE;
126             break;
127         case VLC_CODEC_S16B:
128             format = AFMT_S16_BE;
129             break;
130         case VLC_CODEC_S16L:
131             format = AFMT_S16_LE;
132             break;
133         case VLC_CODEC_S8:
134         case VLC_CODEC_U8:
135             format = AFMT_U8;
136             break;
137         default:
138             if (AOUT_FMT_SPDIF(fmt))
139                 spdif = var_InheritBool (aout, "spdif");
140             if (spdif)
141                 format = AFMT_AC3;
142 #ifdef AFMT_FLOAT
143             else if (HAVE_FPU)
144                 format = AFMT_FLOAT;
145 #endif
146             else
147                 format = AFMT_S16_NE;
148     }
149
150     if (ioctl (fd, SNDCTL_DSP_SETFMT, &format) < 0)
151     {
152         msg_Err (aout, "cannot set audio format 0x%X: %m", format);
153         goto error;
154     }
155
156     switch (format)
157     {
158         case AFMT_S8:     fmt->i_format = VLC_CODEC_S8;   break;
159         case AFMT_U8:     fmt->i_format = VLC_CODEC_U8;   break;
160         case AFMT_S16_BE: fmt->i_format = VLC_CODEC_S16B; break;
161         case AFMT_S16_LE: fmt->i_format = VLC_CODEC_S16L; break;
162         //case AFMT_S24_BE:
163         //case AFMT_S24_LE:
164         case AFMT_S32_BE: fmt->i_format = VLC_CODEC_S32B; break;
165         case AFMT_S32_LE: fmt->i_format = VLC_CODEC_S32L; break;
166 #ifdef AFMT_FLOAT
167         case AFMT_FLOAT:  fmt->i_format = VLC_CODEC_FL32; break;
168 #endif
169         case AFMT_AC3:
170             if (spdif)
171             {
172                 fmt->i_format = VLC_CODEC_SPDIFL;
173                 break;
174             }
175         default:
176             msg_Err (aout, "unsupported audio format 0x%X", format);
177             goto error;
178     }
179
180     /* Select channels count */
181     int channels = spdif ? 2 : aout_FormatNbChannels (fmt);
182     if (ioctl (fd, SNDCTL_DSP_CHANNELS, &channels) < 0)
183     {
184         msg_Err (aout, "cannot set %d channels: %m", channels);
185         goto error;
186     }
187
188     switch (channels)
189     {
190         case 1: channels = AOUT_CHAN_CENTER;  break;
191         case 2: channels = AOUT_CHANS_STEREO; break;
192         case 4: channels = AOUT_CHANS_4_0;    break;
193         case 6: channels = AOUT_CHANS_5_1;    break;
194         case 8: channels = AOUT_CHANS_7_1;    break;
195         default:
196             msg_Err (aout, "unsupported channels count %d", channels);
197             goto error;
198     }
199
200     /* Select sample rate */
201     int rate = spdif ? 48000 : fmt->i_rate;
202     if (ioctl (fd, SNDCTL_DSP_SPEED, &rate) < 0)
203     {
204         msg_Err (aout, "cannot set %d Hz sample rate: %m", rate);
205         goto error;
206     }
207
208     /* Setup audio_output_t */
209     aout->time_get = TimeGet;
210     aout->play = Play;
211     aout->pause = Pause;
212     aout->flush = Flush;
213
214     if (spdif)
215     {
216         fmt->i_bytes_per_frame = AOUT_SPDIF_SIZE;
217         fmt->i_frame_length = A52_FRAME_NB;
218     }
219     else
220     {
221         fmt->i_rate = rate;
222         fmt->i_original_channels =
223         fmt->i_physical_channels = channels;
224     }
225
226     VolumeSync (aout):
227     sys->starting = true;
228     sys->format = *fmt;
229     return VLC_SUCCESS;
230 error:
231     close (fd);
232     return VLC_EGENERIC;
233 }
234
235 static int TimeGet (audio_output_t *aout, mtime_t *restrict pts)
236 {
237     aout_sys_t *sys = aout->sys;
238     int delay;
239
240     if (ioctl (sys->fd, SNDCTL_DSP_GETODELAY, &delay) < 0)
241     {
242         msg_Warn (aout, "cannot get delay: %m");
243         return -1;
244     }
245
246     *pts = (delay * CLOCK_FREQ * sys->format.i_frame_length)
247                         / (sys->format.i_rate * sys->format.i_bytes_per_frame);
248     return 0;
249 }
250
251 /**
252  * Queues one audio buffer to the hardware.
253  */
254 static void Play (audio_output_t *aout, block_t *block)
255 {
256     aout_sys_t *sys = aout->sys;
257     int fd = sys->fd;
258
259     while (block->i_buffer > 0)
260     {
261         ssize_t bytes = write (fd, block->p_buffer, block->i_buffer);
262         if (bytes >= 0)
263         {
264             block->p_buffer += bytes;
265             block->i_buffer -= bytes;
266         }
267         else
268             msg_Err (aout, "cannot write samples: %m");
269     }
270     block_Release (block);
271
272     /* Dumb OSS cannot send any kind of events for this... */
273     VolumeSync (aout);
274 }
275
276 /**
277  * Pauses/resumes the audio playback.
278  */
279 static void Pause (audio_output_t *aout, bool pause, mtime_t date)
280 {
281     aout_sys_t *sys = aout->sys;
282     int fd = sys->fd;
283
284     (void) date;
285     ioctl (fd, pause ? SNDCTL_DSP_SILENCE : SNDCTL_DSP_SKIP, NULL);
286 }
287
288 /**
289  * Flushes/drains the audio playback buffer.
290  */
291 static void Flush (audio_output_t *aout, bool wait)
292 {
293     aout_sys_t *sys = aout->sys;
294     int fd = sys->fd;
295
296     if (wait)
297         return; /* drain is implicit with OSS */
298     ioctl (fd, SNDCTL_DSP_HALT_OUTPUT, NULL);
299 }
300
301 static int VolumeSync (audio_output_t *aout)
302 {
303     aout_sys_t *sys = aout->sys;
304     int fd = sys->fd;
305
306     int level;
307     if (ioctl (fd, SNDCTL_DSP_GETPLAYVOL, &level) < 0)
308         return -1;
309
310     sys->mute = !level;
311     if (level) /* try to keep last volume before mute */
312         sys->level = level;
313     aout_MuteReport (aout, !level);
314     aout_VolumeReport (aout, (float)(level & 0xFF) / 100.f);
315     return 0;
316 }
317
318 /**
319  * Releases the audio output device.
320  */
321 static void Stop (audio_output_t *aout)
322 {
323     aout_sys_t *sys = aout->sys;
324     int fd = sys->fd;
325
326     ioctl (fd, SNDCTL_DSP_HALT, NULL);
327     close (fd);
328     sys->fd = -1;
329 }
330
331 static int VolumeSet (audio_output_t *aout, float vol)
332 {
333     aout_sys_t *sys = aout->sys;
334     int fd = sys->fd;
335
336     int level = lroundf (vol * 100.f);
337     if (level > 0xFF)
338         level = 0xFFFF;
339     else
340         level |= level << 8;
341     if (!sys->mute && ioctl (fd, SNDCTL_DSP_SETPLAYVOL, &level) < 0)
342     {
343         msg_Err (aout, "cannot set volume: %m");
344         return -1;
345     }
346
347     sys->level = level;
348     aout_VolumeReport (aout, (float)(level & 0xFF) / 100.f);
349     return 0;
350 }
351
352 static int MuteSet (audio_output_t *aout, bool mute)
353 {
354     aout_sys_t *sys = aout->sys;
355     int fd = sys->fd;
356
357     int level = mute ? 0 : (sys->level | (sys->level << 8));
358     if (ioctl (fd, SNDCTL_DSP_SETPLAYVOL, &level) < 0)
359     {
360         msg_Err (aout, "cannot mute: %m");
361         return -1;
362     }
363
364     sys->mute = mute;
365     aout_MuteReport (aout, mute);
366     return 0;
367 }
368
369 static int DevicesEnum (audio_output_t *aout, char ***idp, char ***namep)
370 {
371     aout_sys_t *sys = sys;
372     int fd = sys->fd;
373     oss_sysinfo si;
374
375     if (fd == -1)
376         return -1;
377     if (ioctl (fd, SNDCTL_SYSINFO, &si) < 0)
378     {
379         msg_Err (aout, "cannot get system infos: %m");
380         return -1;
381     }
382
383     msg_Dbg (aout, "using %s version %s (0x%06X) under %s", si.product,
384              si.version, si.versionnum, si.license);
385
386     char **ids = xmalloc (sizeof (*ids) * si.numaudios);
387     char **names = xmalloc (sizeof (*names) * si.numaudios);
388     int n = 0;
389
390     for (int i = 0; i < si.numaudios; i++)
391     {
392         oss_audioinfo ai = { .dev = i };
393
394         if (ioctl (fd, SNDCTL_AUDIOINFO, &ai) < 0)
395         {
396             msg_Warn (aout, "cannot get device %d infos: %m", i);
397             continue;
398         }
399         if (ai.caps & (PCM_CAP_HIDDEN|PCM_CAP_MODEM))
400             continue;
401         if (!(ai.caps & PCM_CAP_OUTPUT))
402             continue;
403         if (!ai.enabled)
404             continue;
405
406         ids[n] = xstrdup (ai.devnode);
407         names[n] = xstrdup (ai.name);
408         n++;
409     }
410     *idp = ids;
411     *namep = names;
412     return n;
413 }
414
415 static int DeviceSelect (audio_output_t *aout, const char *id)
416 {
417     aout_sys_t *sys = aout->sys;
418     char *path = NULL;
419
420     if (id != NULL)
421     {
422         path = strdup (id);
423         if (unlikely(path == NULL))
424             return -1;
425     }
426
427     free (sys->device);
428     sys->device = path;
429     aout_DeviceReport (aout, path);
430     aout_RestartRequest (aout, AOUT_RESTART_OUTPUT);
431     return 0;
432 }
433
434 static int Open (vlc_object_t *obj)
435 {
436     audio_output_t *aout = (audio_output_t *)obj;
437
438     aout_sys_t *sys = malloc (sizeof (*sys));
439     if(unlikely( sys == NULL ))
440         return VLC_ENOMEM;
441
442     sys->fd = -1;
443
444     sys->level = 100;
445     sys->mute = false;
446     sys->device = var_InheritString (aout, "oss-audio-device");
447
448     aout->sys = sys;
449     aout->start = Start;
450     aout->stop = Stop;
451     aout->volume_set = VolumeSet;
452     aout->mute_set = MuteSet;
453     aout->device_enum = DevicesEnum;
454     aout->device_select = DeviceSelect;
455     return VLC_SUCCESS;
456 }
457
458 static void Close (vlc_object_t *obj)
459 {
460     audio_output_t *aout = (audio_output_t *)obj;
461     aout_sys_t *sys = aout->sys;
462
463     free (sys->device);
464     free (sys);
465 }