Mercurial > mplayer.hg
annotate libao2/ao_alsa.c @ 17607:156305f44337
10l
author | alex |
---|---|
date | Sun, 12 Feb 2006 14:50:05 +0000 |
parents | 9972b744fd98 |
children | 92431bc3d014 |
rev | line source |
---|---|
12465 | 1 /* |
2 ao_alsa9/1.x - ALSA-0.9.x-1.x output plugin for MPlayer | |
3 | |
4 (C) Alex Beregszaszi | |
5 | |
6 modified for real alsa-0.9.0-support by Zsolt Barat <joy@streamminister.de> | |
7 additional AC3 passthrough support by Andy Lo A Foe <andy@alsaplayer.org> | |
8 08/22/2002 iec958-init rewritten and merged with common init, zsolt | |
9 04/13/2004 merged with ao_alsa1.x, fixes provided by Jindrich Makovicka | |
10 04/25/2004 printfs converted to mp_msg, Zsolt. | |
11 | |
12 Any bugreports regarding to this driver are welcome. | |
13 */ | |
14 | |
15 #include <errno.h> | |
16 #include <sys/time.h> | |
17 #include <stdlib.h> | |
18 #include <math.h> | |
19 #include <string.h> | |
20 #include <sys/poll.h> | |
21 | |
14123 | 22 #include "config.h" |
14328 | 23 #include "subopt-helper.h" |
14123 | 24 #include "mixer.h" |
25 #include "mp_msg.h" | |
12465 | 26 |
27 #define ALSA_PCM_NEW_HW_PARAMS_API | |
28 #define ALSA_PCM_NEW_SW_PARAMS_API | |
29 | |
30 #if HAVE_SYS_ASOUNDLIB_H | |
31 #include <sys/asoundlib.h> | |
32 #elif HAVE_ALSA_ASOUNDLIB_H | |
33 #include <alsa/asoundlib.h> | |
34 #else | |
35 #error "asoundlib.h is not in sys/ or alsa/ - please bugreport" | |
36 #endif | |
37 | |
38 | |
39 #include "audio_out.h" | |
40 #include "audio_out_internal.h" | |
14245 | 41 #include "libaf/af_format.h" |
12465 | 42 |
43 static ao_info_t info = | |
44 { | |
45 "ALSA-0.9.x-1.x audio output", | |
46 "alsa", | |
47 "Alex Beregszaszi, Zsolt Barat <joy@streamminister.de>", | |
48 "under developement" | |
49 }; | |
50 | |
51 LIBAO_EXTERN(alsa) | |
52 | |
53 static snd_pcm_t *alsa_handler; | |
54 static snd_pcm_format_t alsa_format; | |
55 static snd_pcm_hw_params_t *alsa_hwparams; | |
56 static snd_pcm_sw_params_t *alsa_swparams; | |
57 | |
58 /* 16 sets buffersize to 16 * chunksize is as default 1024 | |
59 * which seems to be good avarge for most situations | |
60 * so buffersize is 16384 frames by default */ | |
61 static int alsa_fragcount = 16; | |
17575
9972b744fd98
Small fixes: make all global variables static, remove some unused
cladisch
parents:
17574
diff
changeset
|
62 static snd_pcm_uframes_t chunk_size = 1024; |
12465 | 63 |
64 static size_t bits_per_sample, bytes_per_sample, bits_per_frame; | |
65 static size_t chunk_bytes; | |
66 | |
17575
9972b744fd98
Small fixes: make all global variables static, remove some unused
cladisch
parents:
17574
diff
changeset
|
67 static int ao_mmap = 0; |
9972b744fd98
Small fixes: make all global variables static, remove some unused
cladisch
parents:
17574
diff
changeset
|
68 static int ao_noblock = 0; |
9972b744fd98
Small fixes: make all global variables static, remove some unused
cladisch
parents:
17574
diff
changeset
|
69 static int first = 1; |
12465 | 70 |
71 static int open_mode; | |
72 static int set_block_mode; | |
73 static int alsa_can_pause = 0; | |
74 | |
12747 | 75 #define ALSA_DEVICE_SIZE 256 |
12465 | 76 |
77 #undef BUFFERTIME | |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
78 #define SET_CHUNKSIZE |
12465 | 79 #undef USE_POLL |
80 | |
81 /* to set/get/query special features/parameters */ | |
82 static int control(int cmd, void *arg) | |
83 { | |
84 switch(cmd) { | |
85 case AOCONTROL_QUERY_FORMAT: | |
86 return CONTROL_TRUE; | |
87 #ifndef WORDS_BIGENDIAN | |
88 case AOCONTROL_GET_VOLUME: | |
89 case AOCONTROL_SET_VOLUME: | |
90 { | |
91 ao_control_vol_t *vol = (ao_control_vol_t *)arg; | |
92 | |
93 int err; | |
94 snd_mixer_t *handle; | |
95 snd_mixer_elem_t *elem; | |
96 snd_mixer_selem_id_t *sid; | |
97 | |
12747 | 98 static char *mix_name = "PCM"; |
99 static char *card = "default"; | |
13434 | 100 static int mix_index = 0; |
12465 | 101 |
102 long pmin, pmax; | |
103 long get_vol, set_vol; | |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
104 float f_multi; |
12465 | 105 |
13434 | 106 if(mixer_channel) { |
107 char *test_mix_index; | |
108 | |
109 mix_name = strdup(mixer_channel); | |
17097 | 110 if ((test_mix_index = strchr(mix_name, ','))){ |
13434 | 111 *test_mix_index = 0; |
112 test_mix_index++; | |
113 mix_index = strtol(test_mix_index, &test_mix_index, 0); | |
114 | |
115 if (*test_mix_index){ | |
116 mp_msg(MSGT_AO,MSGL_ERR, | |
117 "alsa-control: invalid mixer index. Defaulting to 0\n"); | |
118 mix_index = 0 ; | |
119 } | |
120 } | |
121 } | |
12747 | 122 if(mixer_device) card = mixer_device; |
12465 | 123 |
14245 | 124 if(ao_data.format == AF_FORMAT_AC3) |
12465 | 125 return CONTROL_TRUE; |
126 | |
127 //allocate simple id | |
128 snd_mixer_selem_id_alloca(&sid); | |
129 | |
130 //sets simple-mixer index and name | |
13434 | 131 snd_mixer_selem_id_set_index(sid, mix_index); |
12465 | 132 snd_mixer_selem_id_set_name(sid, mix_name); |
133 | |
13434 | 134 if (mixer_channel) { |
135 free(mix_name); | |
136 mix_name = NULL; | |
137 } | |
138 | |
12465 | 139 if ((err = snd_mixer_open(&handle, 0)) < 0) { |
140 mp_msg(MSGT_AO,MSGL_ERR,"alsa-control: mixer open error: %s\n", snd_strerror(err)); | |
141 return CONTROL_ERROR; | |
142 } | |
143 | |
144 if ((err = snd_mixer_attach(handle, card)) < 0) { | |
145 mp_msg(MSGT_AO,MSGL_ERR,"alsa-control: mixer attach %s error: %s\n", | |
146 card, snd_strerror(err)); | |
147 snd_mixer_close(handle); | |
148 return CONTROL_ERROR; | |
149 } | |
150 | |
151 if ((err = snd_mixer_selem_register(handle, NULL, NULL)) < 0) { | |
152 mp_msg(MSGT_AO,MSGL_ERR,"alsa-control: mixer register error: %s\n", snd_strerror(err)); | |
153 snd_mixer_close(handle); | |
154 return CONTROL_ERROR; | |
155 } | |
156 err = snd_mixer_load(handle); | |
157 if (err < 0) { | |
158 mp_msg(MSGT_AO,MSGL_ERR,"alsa-control: mixer load error: %s\n", snd_strerror(err)); | |
159 snd_mixer_close(handle); | |
160 return CONTROL_ERROR; | |
161 } | |
162 | |
163 elem = snd_mixer_find_selem(handle, sid); | |
164 if (!elem) { | |
165 mp_msg(MSGT_AO,MSGL_ERR,"alsa-control: unable to find simple control '%s',%i\n", | |
166 snd_mixer_selem_id_get_name(sid), snd_mixer_selem_id_get_index(sid)); | |
167 snd_mixer_close(handle); | |
168 return CONTROL_ERROR; | |
169 } | |
170 | |
171 snd_mixer_selem_get_playback_volume_range(elem,&pmin,&pmax); | |
12811
d5f8efddac6c
volume calc fixes for mixer, by reimar dffinger, 10l reverse by me
joyping
parents:
12805
diff
changeset
|
172 f_multi = (100 / (float)(pmax - pmin)); |
12465 | 173 |
174 if (cmd == AOCONTROL_SET_VOLUME) { | |
175 | |
12811
d5f8efddac6c
volume calc fixes for mixer, by reimar dffinger, 10l reverse by me
joyping
parents:
12805
diff
changeset
|
176 set_vol = vol->left / f_multi + pmin + 0.5; |
12465 | 177 |
178 //setting channels | |
179 if ((err = snd_mixer_selem_set_playback_volume(elem, 0, set_vol)) < 0) { | |
180 mp_msg(MSGT_AO,MSGL_ERR,"alsa-control: error setting left channel, %s\n", | |
181 snd_strerror(err)); | |
182 return CONTROL_ERROR; | |
183 } | |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
184 mp_msg(MSGT_AO,MSGL_DBG2,"left=%li, ", set_vol); |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
185 |
12811
d5f8efddac6c
volume calc fixes for mixer, by reimar dffinger, 10l reverse by me
joyping
parents:
12805
diff
changeset
|
186 set_vol = vol->right / f_multi + pmin + 0.5; |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
187 |
12465 | 188 if ((err = snd_mixer_selem_set_playback_volume(elem, 1, set_vol)) < 0) { |
189 mp_msg(MSGT_AO,MSGL_ERR,"alsa-control: error setting right channel, %s\n", | |
190 snd_strerror(err)); | |
191 return CONTROL_ERROR; | |
192 } | |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
193 mp_msg(MSGT_AO,MSGL_DBG2,"right=%li, pmin=%li, pmax=%li, mult=%f\n", |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
194 set_vol, pmin, pmax, f_multi); |
17194
cd527e59d128
use snd_mixer_selem_set_playback_switch when muting ALSA, patch by Matthias Lederhofer <matled -at- gmx dot net>
wanderer
parents:
17097
diff
changeset
|
195 |
cd527e59d128
use snd_mixer_selem_set_playback_switch when muting ALSA, patch by Matthias Lederhofer <matled -at- gmx dot net>
wanderer
parents:
17097
diff
changeset
|
196 if (snd_mixer_selem_has_playback_switch(elem)) { |
cd527e59d128
use snd_mixer_selem_set_playback_switch when muting ALSA, patch by Matthias Lederhofer <matled -at- gmx dot net>
wanderer
parents:
17097
diff
changeset
|
197 int lmute = (vol->left == 0.0); |
cd527e59d128
use snd_mixer_selem_set_playback_switch when muting ALSA, patch by Matthias Lederhofer <matled -at- gmx dot net>
wanderer
parents:
17097
diff
changeset
|
198 int rmute = (vol->right == 0.0); |
cd527e59d128
use snd_mixer_selem_set_playback_switch when muting ALSA, patch by Matthias Lederhofer <matled -at- gmx dot net>
wanderer
parents:
17097
diff
changeset
|
199 if (snd_mixer_selem_has_playback_switch_joined(elem)) { |
cd527e59d128
use snd_mixer_selem_set_playback_switch when muting ALSA, patch by Matthias Lederhofer <matled -at- gmx dot net>
wanderer
parents:
17097
diff
changeset
|
200 lmute = rmute = lmute && rmute; |
cd527e59d128
use snd_mixer_selem_set_playback_switch when muting ALSA, patch by Matthias Lederhofer <matled -at- gmx dot net>
wanderer
parents:
17097
diff
changeset
|
201 } else { |
cd527e59d128
use snd_mixer_selem_set_playback_switch when muting ALSA, patch by Matthias Lederhofer <matled -at- gmx dot net>
wanderer
parents:
17097
diff
changeset
|
202 snd_mixer_selem_set_playback_switch(elem, SND_MIXER_SCHN_FRONT_RIGHT, !rmute); |
cd527e59d128
use snd_mixer_selem_set_playback_switch when muting ALSA, patch by Matthias Lederhofer <matled -at- gmx dot net>
wanderer
parents:
17097
diff
changeset
|
203 } |
cd527e59d128
use snd_mixer_selem_set_playback_switch when muting ALSA, patch by Matthias Lederhofer <matled -at- gmx dot net>
wanderer
parents:
17097
diff
changeset
|
204 snd_mixer_selem_set_playback_switch(elem, SND_MIXER_SCHN_FRONT_LEFT, !lmute); |
cd527e59d128
use snd_mixer_selem_set_playback_switch when muting ALSA, patch by Matthias Lederhofer <matled -at- gmx dot net>
wanderer
parents:
17097
diff
changeset
|
205 } |
12465 | 206 } |
207 else { | |
208 snd_mixer_selem_get_playback_volume(elem, 0, &get_vol); | |
12811
d5f8efddac6c
volume calc fixes for mixer, by reimar dffinger, 10l reverse by me
joyping
parents:
12805
diff
changeset
|
209 vol->left = (get_vol - pmin) * f_multi; |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
210 snd_mixer_selem_get_playback_volume(elem, 1, &get_vol); |
12811
d5f8efddac6c
volume calc fixes for mixer, by reimar dffinger, 10l reverse by me
joyping
parents:
12805
diff
changeset
|
211 vol->right = (get_vol - pmin) * f_multi; |
12465 | 212 |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
213 mp_msg(MSGT_AO,MSGL_DBG2,"left=%f, right=%f\n",vol->left,vol->right); |
12465 | 214 } |
215 snd_mixer_close(handle); | |
216 return CONTROL_OK; | |
217 } | |
218 #endif | |
219 | |
220 } //end switch | |
221 return(CONTROL_UNKNOWN); | |
222 } | |
223 | |
14328 | 224 static void parse_device (char *dest, const char *src, int len) |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
225 { |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
226 char *tmp; |
14328 | 227 memmove(dest, src, len); |
228 dest[len] = 0; | |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
229 while ((tmp = strrchr(dest, '.'))) |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
230 tmp[0] = ','; |
12919
aba44b58dea7
Use = instead if # in ALSA device name, as # irritates our config-parser.
reimar
parents:
12819
diff
changeset
|
231 while ((tmp = strrchr(dest, '='))) |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
232 tmp[0] = ':'; |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
233 } |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
234 |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
235 static void print_help (void) |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
236 { |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
237 mp_msg (MSGT_AO, MSGL_FATAL, |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
238 "\n-ao alsa commandline help:\n" |
12919
aba44b58dea7
Use = instead if # in ALSA device name, as # irritates our config-parser.
reimar
parents:
12819
diff
changeset
|
239 "Example: mplayer -ao alsa:mmap:device=hw=0.3\n" |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
240 " sets mmap-mode and first card fourth device\n" |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
241 "\nOptions:\n" |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
242 " mmap\n" |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
243 " Set memory-mapped mode, experimental\n" |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
244 " noblock\n" |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
245 " Sets non-blocking mode\n" |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
246 " device=<device-name>\n" |
12919
aba44b58dea7
Use = instead if # in ALSA device name, as # irritates our config-parser.
reimar
parents:
12819
diff
changeset
|
247 " Sets device (change , to . and : to =)\n"); |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
248 } |
12465 | 249 |
14328 | 250 static int str_maxlen(strarg_t *str) { |
251 if (str->len > ALSA_DEVICE_SIZE) | |
252 return 0; | |
253 return 1; | |
254 } | |
255 | |
12465 | 256 /* |
257 open & setup audio device | |
258 return: 1=success 0=fail | |
259 */ | |
260 static int init(int rate_hz, int channels, int format, int flags) | |
261 { | |
262 int err; | |
263 int cards = -1; | |
264 char *str_block_mode; | |
14328 | 265 int block; |
266 strarg_t device; | |
12465 | 267 snd_pcm_uframes_t bufsize; |
14328 | 268 opt_t subopts[] = { |
269 {"mmap", OPT_ARG_BOOL, &ao_mmap, NULL}, | |
270 {"block", OPT_ARG_BOOL, &block, NULL}, | |
271 {"device", OPT_ARG_STR, &device, (opt_test_f)str_maxlen}, | |
272 {NULL} | |
273 }; | |
274 | |
12747 | 275 char alsa_device[ALSA_DEVICE_SIZE + 1]; |
276 // make sure alsa_device is null-terminated even when using strncpy etc. | |
277 memset(alsa_device, 0, ALSA_DEVICE_SIZE + 1); | |
12465 | 278 |
14249 | 279 mp_msg(MSGT_AO,MSGL_V,"alsa-init: requested format: %d Hz, %d channels, %x\n", rate_hz, |
280 channels, format); | |
12465 | 281 alsa_handler = NULL; |
282 mp_msg(MSGT_AO,MSGL_V,"alsa-init: compiled for ALSA-%s\n", SND_LIB_VERSION_STR); | |
283 | |
284 if ((err = snd_card_next(&cards)) < 0 || cards < 0) | |
285 { | |
286 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: no soundcards found: %s\n", snd_strerror(err)); | |
287 return(0); | |
288 } | |
289 | |
290 ao_data.samplerate = rate_hz; | |
291 ao_data.format = format; | |
292 ao_data.channels = channels; | |
293 ao_data.outburst = OUTBURST; | |
294 | |
295 switch (format) | |
296 { | |
14245 | 297 case AF_FORMAT_S8: |
12465 | 298 alsa_format = SND_PCM_FORMAT_S8; |
299 break; | |
14245 | 300 case AF_FORMAT_U8: |
12465 | 301 alsa_format = SND_PCM_FORMAT_U8; |
302 break; | |
14245 | 303 case AF_FORMAT_U16_LE: |
12465 | 304 alsa_format = SND_PCM_FORMAT_U16_LE; |
305 break; | |
14245 | 306 case AF_FORMAT_U16_BE: |
12465 | 307 alsa_format = SND_PCM_FORMAT_U16_BE; |
308 break; | |
309 #ifndef WORDS_BIGENDIAN | |
14245 | 310 case AF_FORMAT_AC3: |
12465 | 311 #endif |
14245 | 312 case AF_FORMAT_S16_LE: |
12465 | 313 alsa_format = SND_PCM_FORMAT_S16_LE; |
314 break; | |
315 #ifdef WORDS_BIGENDIAN | |
14245 | 316 case AF_FORMAT_AC3: |
12465 | 317 #endif |
14245 | 318 case AF_FORMAT_S16_BE: |
12465 | 319 alsa_format = SND_PCM_FORMAT_S16_BE; |
320 break; | |
17571
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
321 case AF_FORMAT_U32_LE: |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
322 alsa_format = SND_PCM_FORMAT_U32_LE; |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
323 break; |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
324 case AF_FORMAT_U32_BE: |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
325 alsa_format = SND_PCM_FORMAT_U32_BE; |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
326 break; |
14245 | 327 case AF_FORMAT_S32_LE: |
12465 | 328 alsa_format = SND_PCM_FORMAT_S32_LE; |
329 break; | |
14245 | 330 case AF_FORMAT_S32_BE: |
12465 | 331 alsa_format = SND_PCM_FORMAT_S32_BE; |
332 break; | |
14245 | 333 case AF_FORMAT_FLOAT_LE: |
12570 | 334 alsa_format = SND_PCM_FORMAT_FLOAT_LE; |
335 break; | |
17571
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
336 case AF_FORMAT_FLOAT_BE: |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
337 alsa_format = SND_PCM_FORMAT_FLOAT_BE; |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
338 break; |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
339 case AF_FORMAT_MU_LAW: |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
340 alsa_format = SND_PCM_FORMAT_MU_LAW; |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
341 break; |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
342 case AF_FORMAT_A_LAW: |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
343 alsa_format = SND_PCM_FORMAT_A_LAW; |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
344 break; |
12465 | 345 |
346 default: | |
14251 | 347 alsa_format = SND_PCM_FORMAT_MPEG; //? default should be -1 |
12465 | 348 break; |
349 } | |
350 | |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
351 //subdevice parsing |
14328 | 352 // set defaults |
353 ao_mmap = 0; | |
354 block = 1; | |
12465 | 355 /* switch for spdif |
356 * sets opening sequence for SPDIF | |
357 * sets also the playback and other switches 'on the fly' | |
358 * while opening the abstract alias for the spdif subdevice | |
359 * 'iec958' | |
360 */ | |
14245 | 361 if (format == AF_FORMAT_AC3) { |
12465 | 362 unsigned char s[4]; |
363 | |
364 s[0] = IEC958_AES0_NONAUDIO | | |
365 IEC958_AES0_CON_EMPHASIS_NONE; | |
366 s[1] = IEC958_AES1_CON_ORIGINAL | | |
367 IEC958_AES1_CON_PCM_CODER; | |
368 s[2] = 0; | |
369 s[3] = IEC958_AES3_CON_FS_48000; | |
370 | |
12747 | 371 snprintf(alsa_device, ALSA_DEVICE_SIZE, |
14612 | 372 "iec958:{CARD 0 AES0 0x%02x AES1 0x%02x AES2 0x%02x AES3 0x%02x}", |
12747 | 373 s[0], s[1], s[2], s[3]); |
14328 | 374 device.str = alsa_device; |
12465 | 375 |
12747 | 376 mp_msg(MSGT_AO,MSGL_V,"alsa-spdif-init: playing AC3, %i channels\n", channels); |
12465 | 377 } |
13661
07dc40f25068
Only use S/PDIF output when no other alsa device is set, allows to use
reimar
parents:
13434
diff
changeset
|
378 else |
14328 | 379 /* in any case for multichannel playback we should select |
380 * appropriate device | |
381 */ | |
382 switch (channels) { | |
383 case 1: | |
384 case 2: | |
385 device.str = "default"; | |
386 mp_msg(MSGT_AO,MSGL_V,"alsa-init: setup for 1/2 channel(s)\n"); | |
387 break; | |
388 case 4: | |
389 if (alsa_format == SND_PCM_FORMAT_FLOAT_LE) | |
390 // hack - use the converter plugin | |
391 device.str = "plug:surround40"; | |
392 else | |
393 device.str = "surround40"; | |
394 mp_msg(MSGT_AO,MSGL_V,"alsa-init: device set to surround40\n"); | |
395 break; | |
396 case 6: | |
397 if (alsa_format == SND_PCM_FORMAT_FLOAT_LE) | |
398 device.str = "plug:surround51"; | |
399 else | |
400 device.str = "surround51"; | |
401 mp_msg(MSGT_AO,MSGL_V,"alsa-init: device set to surround51\n"); | |
402 break; | |
403 default: | |
404 device.str = "default"; | |
405 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: %d channels are not supported\n",channels); | |
406 } | |
407 device.len = strlen(device.str); | |
408 if (subopt_parse(ao_subdevice, subopts) != 0) { | |
409 print_help(); | |
410 return 0; | |
411 } | |
412 ao_noblock = !block; | |
413 parse_device(alsa_device, device.str, device.len); | |
12465 | 414 |
415 mp_msg(MSGT_AO,MSGL_INFO,"alsa-init: %d soundcard%s found, using: %s\n", cards+1,(cards >= 0) ? "" : "s", alsa_device); | |
416 | |
417 //setting modes for block or nonblock-mode | |
418 if (ao_noblock) { | |
419 open_mode = SND_PCM_NONBLOCK; | |
420 set_block_mode = 1; | |
421 str_block_mode = "nonblock-mode"; | |
422 } | |
423 else { | |
424 open_mode = 0; | |
425 set_block_mode = 0; | |
426 str_block_mode = "block-mode"; | |
427 } | |
428 | |
429 //sets buff/chunksize if its set manually | |
430 if (ao_data.buffersize) { | |
431 switch (ao_data.buffersize) | |
432 { | |
433 case 1: | |
434 alsa_fragcount = 16; | |
435 chunk_size = 512; | |
436 mp_msg(MSGT_AO,MSGL_V,"alsa-init: buffersize set manually to 8192\n"); | |
437 mp_msg(MSGT_AO,MSGL_V,"alsa-init: chunksize set manually to 512\n"); | |
438 break; | |
439 case 2: | |
440 alsa_fragcount = 8; | |
441 chunk_size = 1024; | |
442 mp_msg(MSGT_AO,MSGL_V,"alsa-init: buffersize set manually to 8192\n"); | |
443 mp_msg(MSGT_AO,MSGL_V,"alsa-init: chunksize set manually to 1024\n"); | |
444 break; | |
445 case 3: | |
446 alsa_fragcount = 32; | |
447 chunk_size = 512; | |
448 mp_msg(MSGT_AO,MSGL_V,"alsa-init: buffersize set manually to 16384\n"); | |
449 mp_msg(MSGT_AO,MSGL_V,"alsa-init: chunksize set manually to 512\n"); | |
450 break; | |
451 case 4: | |
452 alsa_fragcount = 16; | |
453 chunk_size = 1024; | |
454 mp_msg(MSGT_AO,MSGL_V,"alsa-init: buffersize set manually to 16384\n"); | |
455 mp_msg(MSGT_AO,MSGL_V,"alsa-init: chunksize set manually to 1024\n"); | |
456 break; | |
457 default: | |
458 alsa_fragcount = 16; | |
459 if (ao_mmap) | |
460 chunk_size = 512; | |
461 else | |
462 chunk_size = 1024; | |
463 break; | |
464 } | |
465 } | |
466 | |
467 if (!alsa_handler) { | |
468 //modes = 0, SND_PCM_NONBLOCK, SND_PCM_ASYNC | |
469 if ((err = snd_pcm_open(&alsa_handler, alsa_device, SND_PCM_STREAM_PLAYBACK, open_mode)) < 0) | |
470 { | |
471 if (err != -EBUSY && ao_noblock) { | |
472 mp_msg(MSGT_AO,MSGL_INFO,"alsa-init: open in nonblock-mode failed, trying to open in block-mode\n"); | |
473 if ((err = snd_pcm_open(&alsa_handler, alsa_device, SND_PCM_STREAM_PLAYBACK, 0)) < 0) { | |
474 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: playback open error: %s\n", snd_strerror(err)); | |
475 return(0); | |
476 } else { | |
477 set_block_mode = 0; | |
478 str_block_mode = "block-mode"; | |
479 } | |
480 } else { | |
481 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: playback open error: %s\n", snd_strerror(err)); | |
482 return(0); | |
483 } | |
484 } | |
485 | |
486 if ((err = snd_pcm_nonblock(alsa_handler, set_block_mode)) < 0) { | |
487 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: error set block-mode %s\n", snd_strerror(err)); | |
488 } else { | |
489 mp_msg(MSGT_AO,MSGL_V,"alsa-init: pcm opend in %s\n", str_block_mode); | |
490 } | |
491 | |
492 snd_pcm_hw_params_alloca(&alsa_hwparams); | |
493 snd_pcm_sw_params_alloca(&alsa_swparams); | |
494 | |
495 // setting hw-parameters | |
496 if ((err = snd_pcm_hw_params_any(alsa_handler, alsa_hwparams)) < 0) | |
497 { | |
498 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to get initial parameters: %s\n", | |
499 snd_strerror(err)); | |
500 return(0); | |
501 } | |
502 | |
503 if (ao_mmap) { | |
504 snd_pcm_access_mask_t *mask = alloca(snd_pcm_access_mask_sizeof()); | |
505 snd_pcm_access_mask_none(mask); | |
506 snd_pcm_access_mask_set(mask, SND_PCM_ACCESS_MMAP_INTERLEAVED); | |
507 snd_pcm_access_mask_set(mask, SND_PCM_ACCESS_MMAP_NONINTERLEAVED); | |
508 snd_pcm_access_mask_set(mask, SND_PCM_ACCESS_MMAP_COMPLEX); | |
509 err = snd_pcm_hw_params_set_access_mask(alsa_handler, alsa_hwparams, mask); | |
510 mp_msg(MSGT_AO,MSGL_INFO,"alsa-init: mmap set\n"); | |
511 } else { | |
512 err = snd_pcm_hw_params_set_access(alsa_handler, alsa_hwparams, | |
513 SND_PCM_ACCESS_RW_INTERLEAVED); | |
514 } | |
515 if (err < 0) { | |
516 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set access type: %s\n", | |
517 snd_strerror(err)); | |
518 return (0); | |
519 } | |
520 | |
521 /* workaround for nonsupported formats | |
522 sets default format to S16_LE if the given formats aren't supported */ | |
523 if ((err = snd_pcm_hw_params_test_format(alsa_handler, alsa_hwparams, | |
524 alsa_format)) < 0) | |
525 { | |
526 mp_msg(MSGT_AO,MSGL_INFO, | |
14264 | 527 "alsa-init: format %s are not supported by hardware, trying default\n", af_fmt2str_short(format)); |
12465 | 528 alsa_format = SND_PCM_FORMAT_S16_LE; |
14245 | 529 ao_data.format = AF_FORMAT_S16_LE; |
12465 | 530 } |
531 | |
532 if ((err = snd_pcm_hw_params_set_format(alsa_handler, alsa_hwparams, | |
533 alsa_format)) < 0) | |
534 { | |
535 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set format: %s\n", | |
536 snd_strerror(err)); | |
16308
41278ab73e9b
set the nearest number of channels, return(0) upon errors
henry
parents:
14849
diff
changeset
|
537 return(0); |
12465 | 538 } |
539 | |
16308
41278ab73e9b
set the nearest number of channels, return(0) upon errors
henry
parents:
14849
diff
changeset
|
540 if ((err = snd_pcm_hw_params_set_channels_near(alsa_handler, alsa_hwparams, |
41278ab73e9b
set the nearest number of channels, return(0) upon errors
henry
parents:
14849
diff
changeset
|
541 &ao_data.channels)) < 0) |
12465 | 542 { |
543 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set channels: %s\n", | |
544 snd_strerror(err)); | |
16308
41278ab73e9b
set the nearest number of channels, return(0) upon errors
henry
parents:
14849
diff
changeset
|
545 return(0); |
12465 | 546 } |
547 | |
548 if ((err = snd_pcm_hw_params_set_rate_near(alsa_handler, alsa_hwparams, | |
17575
9972b744fd98
Small fixes: make all global variables static, remove some unused
cladisch
parents:
17574
diff
changeset
|
549 &ao_data.samplerate, NULL)) < 0) |
12465 | 550 { |
551 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set samplerate-2: %s\n", | |
552 snd_strerror(err)); | |
553 return(0); | |
554 } | |
555 | |
17570
401521ec0d61
This replaces the hardcoded numbers for the sample format widths with a
cladisch
parents:
17566
diff
changeset
|
556 bytes_per_sample = snd_pcm_format_physical_width(alsa_format) / 8; |
401521ec0d61
This replaces the hardcoded numbers for the sample format widths with a
cladisch
parents:
17566
diff
changeset
|
557 bytes_per_sample *= ao_data.channels; |
401521ec0d61
This replaces the hardcoded numbers for the sample format widths with a
cladisch
parents:
17566
diff
changeset
|
558 ao_data.bps = ao_data.samplerate * bytes_per_sample; |
16309 | 559 |
12465 | 560 #ifdef BUFFERTIME |
561 { | |
562 int alsa_buffer_time = 500000; /* original 60 */ | |
563 int alsa_period_time; | |
564 alsa_period_time = alsa_buffer_time/4; | |
565 if ((err = snd_pcm_hw_params_set_buffer_time_near(alsa_handler, alsa_hwparams, | |
17575
9972b744fd98
Small fixes: make all global variables static, remove some unused
cladisch
parents:
17574
diff
changeset
|
566 &alsa_buffer_time, NULL)) < 0) |
12465 | 567 { |
568 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set buffer time near: %s\n", | |
569 snd_strerror(err)); | |
570 return(0); | |
571 } else | |
572 alsa_buffer_time = err; | |
573 | |
574 if ((err = snd_pcm_hw_params_set_period_time_near(alsa_handler, alsa_hwparams, | |
17575
9972b744fd98
Small fixes: make all global variables static, remove some unused
cladisch
parents:
17574
diff
changeset
|
575 &alsa_period_time, NULL)) < 0) |
12465 | 576 /* original: alsa_buffer_time/ao_data.bps */ |
577 { | |
578 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set period time: %s\n", | |
579 snd_strerror(err)); | |
580 } | |
581 mp_msg(MSGT_AO,MSGL_INFO,"alsa-init: buffer_time: %d, period_time :%d\n", | |
582 alsa_buffer_time, err); | |
583 } | |
584 #endif//end SET_BUFFERTIME | |
585 | |
586 #ifdef SET_CHUNKSIZE | |
587 { | |
588 //set chunksize | |
589 if ((err = snd_pcm_hw_params_set_period_size_near(alsa_handler, alsa_hwparams, | |
17575
9972b744fd98
Small fixes: make all global variables static, remove some unused
cladisch
parents:
17574
diff
changeset
|
590 &chunk_size, NULL)) < 0) |
12465 | 591 { |
17366 | 592 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set periodsize(%ld): %s\n", |
12465 | 593 chunk_size, snd_strerror(err)); |
594 } | |
595 else { | |
17366 | 596 mp_msg(MSGT_AO,MSGL_V,"alsa-init: chunksize set to %li\n", chunk_size); |
12465 | 597 } |
598 if ((err = snd_pcm_hw_params_set_periods_near(alsa_handler, alsa_hwparams, | |
17575
9972b744fd98
Small fixes: make all global variables static, remove some unused
cladisch
parents:
17574
diff
changeset
|
599 &alsa_fragcount, NULL)) < 0) { |
12465 | 600 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set periods: %s\n", |
601 snd_strerror(err)); | |
602 } | |
603 else { | |
604 mp_msg(MSGT_AO,MSGL_V,"alsa-init: fragcount=%i\n", alsa_fragcount); | |
605 } | |
606 } | |
607 #endif//end SET_CHUNKSIZE | |
608 | |
609 /* finally install hardware parameters */ | |
610 if ((err = snd_pcm_hw_params(alsa_handler, alsa_hwparams)) < 0) | |
611 { | |
612 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set hw-parameters: %s\n", | |
613 snd_strerror(err)); | |
614 } | |
615 // end setting hw-params | |
616 | |
617 | |
618 // gets buffersize for control | |
619 if ((err = snd_pcm_hw_params_get_buffer_size(alsa_hwparams, &bufsize)) < 0) | |
620 { | |
621 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to get buffersize: %s\n", snd_strerror(err)); | |
622 } | |
623 else { | |
624 ao_data.buffersize = bufsize * bytes_per_sample; | |
625 mp_msg(MSGT_AO,MSGL_V,"alsa-init: got buffersize=%i\n", ao_data.buffersize); | |
626 } | |
627 | |
628 // setting sw-params (only avail-min) if noblocking mode was choosed | |
629 if (ao_noblock) | |
630 { | |
631 | |
632 if ((err = snd_pcm_sw_params_current(alsa_handler, alsa_swparams)) < 0) | |
633 { | |
634 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to get parameters: %s\n", | |
635 snd_strerror(err)); | |
636 | |
637 } | |
638 | |
639 //set min available frames to consider pcm ready (4) | |
640 //increased for nonblock-mode should be set dynamically later | |
641 if ((err = snd_pcm_sw_params_set_avail_min(alsa_handler, alsa_swparams, 4)) < 0) | |
642 { | |
643 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set avail_min %s\n", | |
644 snd_strerror(err)); | |
645 } | |
646 | |
647 if ((err = snd_pcm_sw_params(alsa_handler, alsa_swparams)) < 0) | |
648 { | |
649 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to install sw-params\n"); | |
650 } | |
651 | |
652 bits_per_sample = snd_pcm_format_physical_width(alsa_format); | |
16309 | 653 bits_per_frame = bits_per_sample * ao_data.channels; |
12465 | 654 chunk_bytes = chunk_size * bits_per_frame / 8; |
655 | |
656 mp_msg(MSGT_AO,MSGL_V,"alsa-init: bits per sample (bps)=%i, bits per frame (bpf)=%i, chunk_bytes=%i\n",bits_per_sample,bits_per_frame,chunk_bytes);} | |
657 //end swparams | |
658 | |
659 mp_msg(MSGT_AO,MSGL_INFO,"alsa: %d Hz/%d channels/%d bpf/%d bytes buffer/%s\n", | |
660 ao_data.samplerate, ao_data.channels, bytes_per_sample, ao_data.buffersize, | |
661 snd_pcm_format_description(alsa_format)); | |
662 | |
663 } // end switch alsa_handler (spdif) | |
664 alsa_can_pause = snd_pcm_hw_params_can_pause(alsa_hwparams); | |
665 return(1); | |
666 } // end init | |
667 | |
668 | |
669 /* close audio device */ | |
670 static void uninit(int immed) | |
671 { | |
672 | |
673 if (alsa_handler) { | |
674 int err; | |
675 | |
14849
d313f591d1a4
aos should respect the immed uninit flag (quit immediatly vs waiting till file
reimar
parents:
14612
diff
changeset
|
676 if (!immed) |
d313f591d1a4
aos should respect the immed uninit flag (quit immediatly vs waiting till file
reimar
parents:
14612
diff
changeset
|
677 snd_pcm_drain(alsa_handler); |
d313f591d1a4
aos should respect the immed uninit flag (quit immediatly vs waiting till file
reimar
parents:
14612
diff
changeset
|
678 |
12465 | 679 if ((err = snd_pcm_close(alsa_handler)) < 0) |
680 { | |
681 mp_msg(MSGT_AO,MSGL_ERR,"alsa-uninit: pcm close error: %s\n", snd_strerror(err)); | |
682 return; | |
683 } | |
684 else { | |
685 alsa_handler = NULL; | |
686 mp_msg(MSGT_AO,MSGL_INFO,"alsa-uninit: pcm closed\n"); | |
687 } | |
688 } | |
689 else { | |
690 mp_msg(MSGT_AO,MSGL_ERR,"alsa-uninit: no handler defined!\n"); | |
691 } | |
692 } | |
693 | |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
694 static void audio_pause(void) |
12465 | 695 { |
696 int err; | |
697 | |
698 if (alsa_can_pause) { | |
699 if ((err = snd_pcm_pause(alsa_handler, 1)) < 0) | |
700 { | |
701 mp_msg(MSGT_AO,MSGL_ERR,"alsa-pause: pcm pause error: %s\n", snd_strerror(err)); | |
702 return; | |
703 } | |
704 mp_msg(MSGT_AO,MSGL_V,"alsa-pause: pause supported by hardware\n"); | |
705 } else { | |
706 if ((err = snd_pcm_drop(alsa_handler)) < 0) | |
707 { | |
708 mp_msg(MSGT_AO,MSGL_ERR,"alsa-pause: pcm drop error: %s\n", snd_strerror(err)); | |
709 return; | |
710 } | |
711 } | |
712 } | |
713 | |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
714 static void audio_resume(void) |
12465 | 715 { |
716 int err; | |
717 | |
718 if (alsa_can_pause) { | |
719 if ((err = snd_pcm_pause(alsa_handler, 0)) < 0) | |
720 { | |
721 mp_msg(MSGT_AO,MSGL_ERR,"alsa-resume: pcm resume error: %s\n", snd_strerror(err)); | |
722 return; | |
723 } | |
724 mp_msg(MSGT_AO,MSGL_V,"alsa-resume: resume supported by hardware\n"); | |
725 } else { | |
726 if ((err = snd_pcm_prepare(alsa_handler)) < 0) | |
727 { | |
728 mp_msg(MSGT_AO,MSGL_ERR,"alsa-resume: pcm prepare error: %s\n", snd_strerror(err)); | |
729 return; | |
730 } | |
731 } | |
732 } | |
733 | |
734 /* stop playing and empty buffers (for seeking/pause) */ | |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
735 static void reset(void) |
12465 | 736 { |
737 int err; | |
738 | |
739 if ((err = snd_pcm_drop(alsa_handler)) < 0) | |
740 { | |
741 mp_msg(MSGT_AO,MSGL_ERR,"alsa-reset: pcm drop error: %s\n", snd_strerror(err)); | |
742 return; | |
743 } | |
744 if ((err = snd_pcm_prepare(alsa_handler)) < 0) | |
745 { | |
746 mp_msg(MSGT_AO,MSGL_ERR,"alsa-reset: pcm prepare error: %s\n", snd_strerror(err)); | |
747 return; | |
748 } | |
749 return; | |
750 } | |
751 | |
752 #ifdef USE_POLL | |
753 static int wait_for_poll(snd_pcm_t *handle, struct pollfd *ufds, unsigned int count) | |
754 { | |
755 unsigned short revents; | |
756 | |
757 while (1) { | |
758 poll(ufds, count, -1); | |
759 snd_pcm_poll_descriptors_revents(handle, ufds, count, &revents); | |
760 if (revents & POLLERR) | |
761 return -EIO; | |
762 if (revents & POLLOUT) | |
763 return 0; | |
764 } | |
765 } | |
766 #endif | |
767 | |
768 #ifndef timersub | |
769 #define timersub(a, b, result) \ | |
770 do { \ | |
771 (result)->tv_sec = (a)->tv_sec - (b)->tv_sec; \ | |
772 (result)->tv_usec = (a)->tv_usec - (b)->tv_usec; \ | |
773 if ((result)->tv_usec < 0) { \ | |
774 --(result)->tv_sec; \ | |
775 (result)->tv_usec += 1000000; \ | |
776 } \ | |
777 } while (0) | |
778 #endif | |
779 | |
780 /* I/O error handler */ | |
781 static int xrun(u_char *str_mode) | |
782 { | |
783 int err; | |
784 snd_pcm_status_t *status; | |
785 | |
786 snd_pcm_status_alloca(&status); | |
787 | |
788 if ((err = snd_pcm_status(alsa_handler, status))<0) { | |
789 mp_msg(MSGT_AO,MSGL_ERR,"status error: %s", snd_strerror(err)); | |
790 return(0); | |
791 } | |
792 | |
793 if (snd_pcm_status_get_state(status) == SND_PCM_STATE_XRUN) { | |
794 struct timeval now, diff, tstamp; | |
795 gettimeofday(&now, 0); | |
796 snd_pcm_status_get_trigger_tstamp(status, &tstamp); | |
797 timersub(&now, &tstamp, &diff); | |
798 mp_msg(MSGT_AO,MSGL_INFO,"alsa-%s: xrun of at least %.3f msecs. resetting stream\n", | |
799 str_mode, | |
800 diff.tv_sec * 1000 + diff.tv_usec / 1000.0); | |
801 } | |
802 | |
803 if ((err = snd_pcm_prepare(alsa_handler))<0) { | |
804 mp_msg(MSGT_AO,MSGL_ERR,"xrun: prepare error: %s", snd_strerror(err)); | |
805 return(0); | |
806 } | |
807 | |
808 return(1); /* ok, data should be accepted again */ | |
809 } | |
810 | |
811 static int play_normal(void* data, int len); | |
812 static int play_mmap(void* data, int len); | |
813 | |
814 static int play(void* data, int len, int flags) | |
815 { | |
816 int result; | |
817 if (ao_mmap) | |
818 result = play_mmap(data, len); | |
819 else | |
820 result = play_normal(data, len); | |
821 | |
822 return result; | |
823 } | |
824 | |
825 /* | |
826 plays 'len' bytes of 'data' | |
827 returns: number of bytes played | |
828 modified last at 29.06.02 by jp | |
829 thanxs for marius <marius@rospot.com> for giving us the light ;) | |
830 */ | |
831 | |
832 static int play_normal(void* data, int len) | |
833 { | |
834 | |
835 //bytes_per_sample is always 4 for 2 chn S16_LE | |
836 int num_frames = len / bytes_per_sample; | |
837 char *output_samples = (char *)data; | |
838 snd_pcm_sframes_t res = 0; | |
839 | |
840 //mp_msg(MSGT_AO,MSGL_ERR,"alsa-play: frames=%i, len=%i\n",num_frames,len); | |
841 | |
842 if (!alsa_handler) { | |
843 mp_msg(MSGT_AO,MSGL_ERR,"alsa-play: device configuration error"); | |
844 return 0; | |
845 } | |
846 | |
847 while (num_frames > 0) { | |
848 | |
849 res = snd_pcm_writei(alsa_handler, (void *)output_samples, num_frames); | |
850 | |
851 if (res == -EAGAIN) { | |
852 snd_pcm_wait(alsa_handler, 1000); | |
853 } | |
854 else if (res == -EPIPE) { /* underrun */ | |
855 if (xrun("play") <= 0) { | |
856 mp_msg(MSGT_AO,MSGL_ERR,"alsa-play: xrun reset error"); | |
857 return(0); | |
858 } | |
859 } | |
860 else if (res == -ESTRPIPE) { /* suspend */ | |
861 mp_msg(MSGT_AO,MSGL_INFO,"alsa-play: pcm in suspend mode. trying to resume\n"); | |
862 while ((res = snd_pcm_resume(alsa_handler)) == -EAGAIN) | |
863 sleep(1); | |
864 } | |
865 else if (res < 0) { | |
866 mp_msg(MSGT_AO,MSGL_INFO,"alsa-play: unknown status, trying to reset soundcard\n"); | |
867 if ((res = snd_pcm_prepare(alsa_handler)) < 0) { | |
868 mp_msg(MSGT_AO,MSGL_ERR,"alsa-play: snd prepare error"); | |
869 return(0); | |
870 break; | |
871 } | |
872 } | |
873 | |
874 if (res > 0) { | |
875 | |
876 /* output_samples += ao_data.channels * res; */ | |
877 output_samples += res * bytes_per_sample; | |
878 | |
879 num_frames -= res; | |
880 } | |
881 | |
882 } //end while | |
883 | |
884 if (res < 0) { | |
885 mp_msg(MSGT_AO,MSGL_ERR,"alsa-play: write error %s", snd_strerror(res)); | |
886 return 0; | |
887 } | |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
888 return len - len % bytes_per_sample; |
12465 | 889 } |
890 | |
891 /* mmap-mode mainly based on descriptions by Joshua Haberman <joshua@haberman.com> | |
892 * 'An overview of the ALSA API' http://people.debian.org/~joshua/x66.html | |
893 * and some help by Paul Davis <pbd@op.net> */ | |
894 | |
895 static int play_mmap(void* data, int len) | |
896 { | |
897 snd_pcm_sframes_t commitres, frames_available; | |
898 snd_pcm_uframes_t frames_transmit, size, offset; | |
899 const snd_pcm_channel_area_t *area; | |
900 void *outbuffer; | |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
901 int result; |
12465 | 902 |
903 #ifdef USE_POLL //seems not really be needed | |
904 struct pollfd *ufds; | |
905 int count; | |
906 | |
907 count = snd_pcm_poll_descriptors_count (alsa_handler); | |
908 ufds = malloc(sizeof(struct pollfd) * count); | |
909 snd_pcm_poll_descriptors(alsa_handler, ufds, count); | |
910 | |
911 //first wait_for_poll | |
912 if (err = (wait_for_poll(alsa_handler, ufds, count) < 0)) { | |
913 if (snd_pcm_state(alsa_handler) == SND_PCM_STATE_XRUN || | |
914 snd_pcm_state(alsa_handler) == SND_PCM_STATE_SUSPENDED) { | |
915 xrun("play"); | |
916 } | |
917 } | |
918 #endif | |
919 | |
920 outbuffer = alloca(ao_data.buffersize); | |
921 | |
922 //don't trust get_space() ;) | |
923 frames_available = snd_pcm_avail_update(alsa_handler) * bytes_per_sample; | |
924 if (frames_available < 0) | |
925 xrun("play"); | |
926 | |
927 if (frames_available < 4) { | |
928 if (first) { | |
929 first = 0; | |
930 snd_pcm_start(alsa_handler); | |
931 } | |
932 else { //FIXME should break and return 0? | |
933 snd_pcm_wait(alsa_handler, -1); | |
934 first = 1; | |
935 } | |
936 } | |
937 | |
938 /* len is simply the available bufferspace got by get_space() | |
939 * but real avail_buffer in frames is ab/bytes_per_sample */ | |
940 size = len / bytes_per_sample; | |
941 | |
942 //mp_msg(MSGT_AO,MSGL_V,"len: %i size %i, f_avail %i, bps %i ...\n", len, size, frames_available, bytes_per_sample); | |
943 | |
944 frames_transmit = size; | |
945 | |
946 /* prepare areas and set sw-pointers | |
947 * frames_transmit returns the real available buffer-size | |
948 * sometimes != frames_available cause of ringbuffer 'emulation' */ | |
949 snd_pcm_mmap_begin(alsa_handler, &area, &offset, &frames_transmit); | |
950 | |
951 /* this is specific to interleaved streams (or non-interleaved | |
952 * streams with only one channel) */ | |
953 outbuffer = ((char *) area->addr + (area->first + area->step * offset) / 8); //8 | |
954 | |
955 //write data | |
956 memcpy(outbuffer, data, (frames_transmit * bytes_per_sample)); | |
957 | |
958 commitres = snd_pcm_mmap_commit(alsa_handler, offset, frames_transmit); | |
959 | |
960 if (commitres < 0 || commitres != frames_transmit) { | |
961 if (snd_pcm_state(alsa_handler) == SND_PCM_STATE_XRUN || | |
962 snd_pcm_state(alsa_handler) == SND_PCM_STATE_SUSPENDED) { | |
963 xrun("play"); | |
964 } | |
965 } | |
966 | |
967 //mp_msg(MSGT_AO,MSGL_V,"mmap ft: %i, cres: %i\n", frames_transmit, commitres); | |
968 | |
969 /* err = snd_pcm_area_copy(&area, offset, &data, offset, len, alsa_format); */ | |
970 /* if (err < 0) { */ | |
971 /* mp_msg(MSGT_AO,MSGL_ERR,"area-copy-error\n"); */ | |
972 /* return 0; */ | |
973 /* } */ | |
974 | |
975 | |
976 //calculate written frames! | |
977 result = commitres * bytes_per_sample; | |
978 | |
979 | |
980 /* if (verbose) { */ | |
981 /* if (len == result) */ | |
982 /* mp_msg(MSGT_AO,MSGL_V,"result: %i, frames written: %i ...\n", result, frames_transmit); */ | |
983 /* else */ | |
984 /* mp_msg(MSGT_AO,MSGL_V,"result: %i, frames written: %i, result != len ...\n", result, frames_transmit); */ | |
985 /* } */ | |
986 | |
987 //mplayer doesn't like -result | |
988 if (result < 0) | |
989 result = 0; | |
990 | |
991 #ifdef USE_POLL | |
992 free(ufds); | |
993 #endif | |
994 | |
995 return result; | |
996 } | |
997 | |
998 /* how many byes are free in the buffer */ | |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
999 static int get_space(void) |
12465 | 1000 { |
1001 snd_pcm_status_t *status; | |
1002 int ret; | |
1003 | |
12747 | 1004 snd_pcm_status_alloca(&status); |
12465 | 1005 |
1006 if ((ret = snd_pcm_status(alsa_handler, status)) < 0) | |
1007 { | |
1008 mp_msg(MSGT_AO,MSGL_ERR,"alsa-space: cannot get pcm status: %s\n", snd_strerror(ret)); | |
1009 return(0); | |
1010 } | |
1011 | |
17572
580dc69d69bf
Fix get_space(): we don't need to differentiate between the various PCM
cladisch
parents:
17571
diff
changeset
|
1012 ret = snd_pcm_status_get_avail(status) * bytes_per_sample; |
580dc69d69bf
Fix get_space(): we don't need to differentiate between the various PCM
cladisch
parents:
17571
diff
changeset
|
1013 if (ret > MAX_OUTBURST) |
580dc69d69bf
Fix get_space(): we don't need to differentiate between the various PCM
cladisch
parents:
17571
diff
changeset
|
1014 ret = MAX_OUTBURST; |
12465 | 1015 return(ret); |
1016 } | |
1017 | |
1018 /* delay in seconds between first and last sample in buffer */ | |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
1019 static float get_delay(void) |
12465 | 1020 { |
1021 if (alsa_handler) { | |
17573
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
1022 snd_pcm_sframes_t delay; |
12465 | 1023 |
17573
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
1024 if (snd_pcm_delay(alsa_handler, &delay) < 0) |
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
1025 return 0; |
12465 | 1026 |
17573
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
1027 if (delay < 0) { |
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
1028 /* underrun - move the application pointer forward to catch up */ |
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
1029 #if SND_LIB_VERSION >= 0x000901 /* snd_pcm_forward() exists since 0.9.0rc8 */ |
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
1030 snd_pcm_forward(alsa_handler, -delay); |
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
1031 #endif |
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
1032 delay = 0; |
12465 | 1033 } |
17573
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
1034 return (float)delay / (float)ao_data.samplerate; |
12465 | 1035 } else { |
1036 return(0); | |
1037 } | |
1038 } |