Mercurial > mplayer.hg
annotate libao2/ao_alsa.c @ 17689:351c4f61140f
synced with 1.101
author | gabrov |
---|---|
date | Sun, 26 Feb 2006 12:39:17 +0000 |
parents | d9c518932302 |
children | 9ca95aee8449 |
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 | |
14123 | 21 #include "config.h" |
14328 | 22 #include "subopt-helper.h" |
14123 | 23 #include "mixer.h" |
24 #include "mp_msg.h" | |
12465 | 25 |
26 #define ALSA_PCM_NEW_HW_PARAMS_API | |
27 #define ALSA_PCM_NEW_SW_PARAMS_API | |
28 | |
29 #if HAVE_SYS_ASOUNDLIB_H | |
30 #include <sys/asoundlib.h> | |
31 #elif HAVE_ALSA_ASOUNDLIB_H | |
32 #include <alsa/asoundlib.h> | |
33 #else | |
34 #error "asoundlib.h is not in sys/ or alsa/ - please bugreport" | |
35 #endif | |
36 | |
37 | |
38 #include "audio_out.h" | |
39 #include "audio_out_internal.h" | |
14245 | 40 #include "libaf/af_format.h" |
12465 | 41 |
42 static ao_info_t info = | |
43 { | |
44 "ALSA-0.9.x-1.x audio output", | |
45 "alsa", | |
46 "Alex Beregszaszi, Zsolt Barat <joy@streamminister.de>", | |
47 "under developement" | |
48 }; | |
49 | |
50 LIBAO_EXTERN(alsa) | |
51 | |
52 static snd_pcm_t *alsa_handler; | |
53 static snd_pcm_format_t alsa_format; | |
54 static snd_pcm_hw_params_t *alsa_hwparams; | |
55 static snd_pcm_sw_params_t *alsa_swparams; | |
56 | |
57 /* 16 sets buffersize to 16 * chunksize is as default 1024 | |
58 * which seems to be good avarge for most situations | |
59 * so buffersize is 16384 frames by default */ | |
60 static int alsa_fragcount = 16; | |
17575
9972b744fd98
Small fixes: make all global variables static, remove some unused
cladisch
parents:
17574
diff
changeset
|
61 static snd_pcm_uframes_t chunk_size = 1024; |
12465 | 62 |
17619
9b619133f11a
Using non-blocking writes makes sense when the program wants to do other
cladisch
parents:
17618
diff
changeset
|
63 static size_t bytes_per_sample; |
12465 | 64 |
17575
9972b744fd98
Small fixes: make all global variables static, remove some unused
cladisch
parents:
17574
diff
changeset
|
65 static int ao_noblock = 0; |
12465 | 66 |
67 static int open_mode; | |
68 static int alsa_can_pause = 0; | |
69 | |
12747 | 70 #define ALSA_DEVICE_SIZE 256 |
12465 | 71 |
72 #undef BUFFERTIME | |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
73 #define SET_CHUNKSIZE |
12465 | 74 |
75 /* to set/get/query special features/parameters */ | |
76 static int control(int cmd, void *arg) | |
77 { | |
78 switch(cmd) { | |
79 case AOCONTROL_QUERY_FORMAT: | |
80 return CONTROL_TRUE; | |
81 #ifndef WORDS_BIGENDIAN | |
82 case AOCONTROL_GET_VOLUME: | |
83 case AOCONTROL_SET_VOLUME: | |
84 { | |
85 ao_control_vol_t *vol = (ao_control_vol_t *)arg; | |
86 | |
87 int err; | |
88 snd_mixer_t *handle; | |
89 snd_mixer_elem_t *elem; | |
90 snd_mixer_selem_id_t *sid; | |
91 | |
12747 | 92 static char *mix_name = "PCM"; |
93 static char *card = "default"; | |
13434 | 94 static int mix_index = 0; |
12465 | 95 |
96 long pmin, pmax; | |
97 long get_vol, set_vol; | |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
98 float f_multi; |
12465 | 99 |
13434 | 100 if(mixer_channel) { |
101 char *test_mix_index; | |
102 | |
103 mix_name = strdup(mixer_channel); | |
17097 | 104 if ((test_mix_index = strchr(mix_name, ','))){ |
13434 | 105 *test_mix_index = 0; |
106 test_mix_index++; | |
107 mix_index = strtol(test_mix_index, &test_mix_index, 0); | |
108 | |
109 if (*test_mix_index){ | |
110 mp_msg(MSGT_AO,MSGL_ERR, | |
111 "alsa-control: invalid mixer index. Defaulting to 0\n"); | |
112 mix_index = 0 ; | |
113 } | |
114 } | |
115 } | |
12747 | 116 if(mixer_device) card = mixer_device; |
12465 | 117 |
14245 | 118 if(ao_data.format == AF_FORMAT_AC3) |
12465 | 119 return CONTROL_TRUE; |
120 | |
121 //allocate simple id | |
122 snd_mixer_selem_id_alloca(&sid); | |
123 | |
124 //sets simple-mixer index and name | |
13434 | 125 snd_mixer_selem_id_set_index(sid, mix_index); |
12465 | 126 snd_mixer_selem_id_set_name(sid, mix_name); |
127 | |
13434 | 128 if (mixer_channel) { |
129 free(mix_name); | |
130 mix_name = NULL; | |
131 } | |
132 | |
12465 | 133 if ((err = snd_mixer_open(&handle, 0)) < 0) { |
134 mp_msg(MSGT_AO,MSGL_ERR,"alsa-control: mixer open error: %s\n", snd_strerror(err)); | |
135 return CONTROL_ERROR; | |
136 } | |
137 | |
138 if ((err = snd_mixer_attach(handle, card)) < 0) { | |
139 mp_msg(MSGT_AO,MSGL_ERR,"alsa-control: mixer attach %s error: %s\n", | |
140 card, snd_strerror(err)); | |
141 snd_mixer_close(handle); | |
142 return CONTROL_ERROR; | |
143 } | |
144 | |
145 if ((err = snd_mixer_selem_register(handle, NULL, NULL)) < 0) { | |
146 mp_msg(MSGT_AO,MSGL_ERR,"alsa-control: mixer register error: %s\n", snd_strerror(err)); | |
147 snd_mixer_close(handle); | |
148 return CONTROL_ERROR; | |
149 } | |
150 err = snd_mixer_load(handle); | |
151 if (err < 0) { | |
152 mp_msg(MSGT_AO,MSGL_ERR,"alsa-control: mixer load error: %s\n", snd_strerror(err)); | |
153 snd_mixer_close(handle); | |
154 return CONTROL_ERROR; | |
155 } | |
156 | |
157 elem = snd_mixer_find_selem(handle, sid); | |
158 if (!elem) { | |
159 mp_msg(MSGT_AO,MSGL_ERR,"alsa-control: unable to find simple control '%s',%i\n", | |
160 snd_mixer_selem_id_get_name(sid), snd_mixer_selem_id_get_index(sid)); | |
161 snd_mixer_close(handle); | |
162 return CONTROL_ERROR; | |
163 } | |
164 | |
165 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
|
166 f_multi = (100 / (float)(pmax - pmin)); |
12465 | 167 |
168 if (cmd == AOCONTROL_SET_VOLUME) { | |
169 | |
12811
d5f8efddac6c
volume calc fixes for mixer, by reimar dffinger, 10l reverse by me
joyping
parents:
12805
diff
changeset
|
170 set_vol = vol->left / f_multi + pmin + 0.5; |
12465 | 171 |
172 //setting channels | |
173 if ((err = snd_mixer_selem_set_playback_volume(elem, 0, set_vol)) < 0) { | |
174 mp_msg(MSGT_AO,MSGL_ERR,"alsa-control: error setting left channel, %s\n", | |
175 snd_strerror(err)); | |
176 return CONTROL_ERROR; | |
177 } | |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
178 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
|
179 |
12811
d5f8efddac6c
volume calc fixes for mixer, by reimar dffinger, 10l reverse by me
joyping
parents:
12805
diff
changeset
|
180 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
|
181 |
12465 | 182 if ((err = snd_mixer_selem_set_playback_volume(elem, 1, set_vol)) < 0) { |
183 mp_msg(MSGT_AO,MSGL_ERR,"alsa-control: error setting right channel, %s\n", | |
184 snd_strerror(err)); | |
185 return CONTROL_ERROR; | |
186 } | |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
187 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
|
188 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
|
189 |
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
|
190 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
|
191 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
|
192 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
|
193 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
|
194 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
|
195 } 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
|
196 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
|
197 } |
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 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
|
199 } |
12465 | 200 } |
201 else { | |
202 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
|
203 vol->left = (get_vol - pmin) * f_multi; |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
204 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
|
205 vol->right = (get_vol - pmin) * f_multi; |
12465 | 206 |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
207 mp_msg(MSGT_AO,MSGL_DBG2,"left=%f, right=%f\n",vol->left,vol->right); |
12465 | 208 } |
209 snd_mixer_close(handle); | |
210 return CONTROL_OK; | |
211 } | |
212 #endif | |
213 | |
214 } //end switch | |
215 return(CONTROL_UNKNOWN); | |
216 } | |
217 | |
14328 | 218 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
|
219 { |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
220 char *tmp; |
14328 | 221 memmove(dest, src, len); |
222 dest[len] = 0; | |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
223 while ((tmp = strrchr(dest, '.'))) |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
224 tmp[0] = ','; |
12919
aba44b58dea7
Use = instead if # in ALSA device name, as # irritates our config-parser.
reimar
parents:
12819
diff
changeset
|
225 while ((tmp = strrchr(dest, '='))) |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
226 tmp[0] = ':'; |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
227 } |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
228 |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
229 static void print_help (void) |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
230 { |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
231 mp_msg (MSGT_AO, MSGL_FATAL, |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
232 "\n-ao alsa commandline help:\n" |
17616
92431bc3d014
This patch removes mmap support because it doesn't have any benefit.
cladisch
parents:
17575
diff
changeset
|
233 "Example: mplayer -ao alsa:device=hw=0.3\n" |
92431bc3d014
This patch removes mmap support because it doesn't have any benefit.
cladisch
parents:
17575
diff
changeset
|
234 " sets first card fourth hardware device\n" |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
235 "\nOptions:\n" |
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
236 " noblock\n" |
17619
9b619133f11a
Using non-blocking writes makes sense when the program wants to do other
cladisch
parents:
17618
diff
changeset
|
237 " Opens device in non-blocking mode\n" |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
238 " device=<device-name>\n" |
12919
aba44b58dea7
Use = instead if # in ALSA device name, as # irritates our config-parser.
reimar
parents:
12819
diff
changeset
|
239 " Sets device (change , to . and : to =)\n"); |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
240 } |
12465 | 241 |
14328 | 242 static int str_maxlen(strarg_t *str) { |
243 if (str->len > ALSA_DEVICE_SIZE) | |
244 return 0; | |
245 return 1; | |
246 } | |
247 | |
12465 | 248 /* |
249 open & setup audio device | |
250 return: 1=success 0=fail | |
251 */ | |
252 static int init(int rate_hz, int channels, int format, int flags) | |
253 { | |
254 int err; | |
255 int cards = -1; | |
14328 | 256 int block; |
257 strarg_t device; | |
12465 | 258 snd_pcm_uframes_t bufsize; |
17620
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
259 snd_pcm_uframes_t boundary; |
14328 | 260 opt_t subopts[] = { |
261 {"block", OPT_ARG_BOOL, &block, NULL}, | |
262 {"device", OPT_ARG_STR, &device, (opt_test_f)str_maxlen}, | |
263 {NULL} | |
264 }; | |
265 | |
12747 | 266 char alsa_device[ALSA_DEVICE_SIZE + 1]; |
267 // make sure alsa_device is null-terminated even when using strncpy etc. | |
268 memset(alsa_device, 0, ALSA_DEVICE_SIZE + 1); | |
12465 | 269 |
14249 | 270 mp_msg(MSGT_AO,MSGL_V,"alsa-init: requested format: %d Hz, %d channels, %x\n", rate_hz, |
271 channels, format); | |
12465 | 272 alsa_handler = NULL; |
273 mp_msg(MSGT_AO,MSGL_V,"alsa-init: compiled for ALSA-%s\n", SND_LIB_VERSION_STR); | |
274 | |
275 if ((err = snd_card_next(&cards)) < 0 || cards < 0) | |
276 { | |
277 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: no soundcards found: %s\n", snd_strerror(err)); | |
278 return(0); | |
279 } | |
280 | |
281 ao_data.samplerate = rate_hz; | |
282 ao_data.format = format; | |
283 ao_data.channels = channels; | |
284 | |
285 switch (format) | |
286 { | |
14245 | 287 case AF_FORMAT_S8: |
12465 | 288 alsa_format = SND_PCM_FORMAT_S8; |
289 break; | |
14245 | 290 case AF_FORMAT_U8: |
12465 | 291 alsa_format = SND_PCM_FORMAT_U8; |
292 break; | |
14245 | 293 case AF_FORMAT_U16_LE: |
12465 | 294 alsa_format = SND_PCM_FORMAT_U16_LE; |
295 break; | |
14245 | 296 case AF_FORMAT_U16_BE: |
12465 | 297 alsa_format = SND_PCM_FORMAT_U16_BE; |
298 break; | |
299 #ifndef WORDS_BIGENDIAN | |
14245 | 300 case AF_FORMAT_AC3: |
12465 | 301 #endif |
14245 | 302 case AF_FORMAT_S16_LE: |
12465 | 303 alsa_format = SND_PCM_FORMAT_S16_LE; |
304 break; | |
305 #ifdef WORDS_BIGENDIAN | |
14245 | 306 case AF_FORMAT_AC3: |
12465 | 307 #endif |
14245 | 308 case AF_FORMAT_S16_BE: |
12465 | 309 alsa_format = SND_PCM_FORMAT_S16_BE; |
310 break; | |
17571
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
311 case AF_FORMAT_U32_LE: |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
312 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
|
313 break; |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
314 case AF_FORMAT_U32_BE: |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
315 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
|
316 break; |
14245 | 317 case AF_FORMAT_S32_LE: |
12465 | 318 alsa_format = SND_PCM_FORMAT_S32_LE; |
319 break; | |
14245 | 320 case AF_FORMAT_S32_BE: |
12465 | 321 alsa_format = SND_PCM_FORMAT_S32_BE; |
322 break; | |
14245 | 323 case AF_FORMAT_FLOAT_LE: |
12570 | 324 alsa_format = SND_PCM_FORMAT_FLOAT_LE; |
325 break; | |
17571
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
326 case AF_FORMAT_FLOAT_BE: |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
327 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
|
328 break; |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
329 case AF_FORMAT_MU_LAW: |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
330 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
|
331 break; |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
332 case AF_FORMAT_A_LAW: |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
333 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
|
334 break; |
12465 | 335 |
336 default: | |
14251 | 337 alsa_format = SND_PCM_FORMAT_MPEG; //? default should be -1 |
12465 | 338 break; |
339 } | |
340 | |
12805
0b154063a3ca
fixes provided by reimar drfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
341 //subdevice parsing |
14328 | 342 // set defaults |
343 block = 1; | |
12465 | 344 /* switch for spdif |
345 * sets opening sequence for SPDIF | |
346 * sets also the playback and other switches 'on the fly' | |
347 * while opening the abstract alias for the spdif subdevice | |
348 * 'iec958' | |
349 */ | |
14245 | 350 if (format == AF_FORMAT_AC3) { |
12465 | 351 unsigned char s[4]; |
352 | |
353 s[0] = IEC958_AES0_NONAUDIO | | |
354 IEC958_AES0_CON_EMPHASIS_NONE; | |
355 s[1] = IEC958_AES1_CON_ORIGINAL | | |
356 IEC958_AES1_CON_PCM_CODER; | |
357 s[2] = 0; | |
358 s[3] = IEC958_AES3_CON_FS_48000; | |
359 | |
12747 | 360 snprintf(alsa_device, ALSA_DEVICE_SIZE, |
14612 | 361 "iec958:{CARD 0 AES0 0x%02x AES1 0x%02x AES2 0x%02x AES3 0x%02x}", |
12747 | 362 s[0], s[1], s[2], s[3]); |
14328 | 363 device.str = alsa_device; |
12465 | 364 |
12747 | 365 mp_msg(MSGT_AO,MSGL_V,"alsa-spdif-init: playing AC3, %i channels\n", channels); |
12465 | 366 } |
13661
07dc40f25068
Only use S/PDIF output when no other alsa device is set, allows to use
reimar
parents:
13434
diff
changeset
|
367 else |
14328 | 368 /* in any case for multichannel playback we should select |
369 * appropriate device | |
370 */ | |
371 switch (channels) { | |
372 case 1: | |
373 case 2: | |
374 device.str = "default"; | |
375 mp_msg(MSGT_AO,MSGL_V,"alsa-init: setup for 1/2 channel(s)\n"); | |
376 break; | |
377 case 4: | |
378 if (alsa_format == SND_PCM_FORMAT_FLOAT_LE) | |
379 // hack - use the converter plugin | |
380 device.str = "plug:surround40"; | |
381 else | |
382 device.str = "surround40"; | |
383 mp_msg(MSGT_AO,MSGL_V,"alsa-init: device set to surround40\n"); | |
384 break; | |
385 case 6: | |
386 if (alsa_format == SND_PCM_FORMAT_FLOAT_LE) | |
387 device.str = "plug:surround51"; | |
388 else | |
389 device.str = "surround51"; | |
390 mp_msg(MSGT_AO,MSGL_V,"alsa-init: device set to surround51\n"); | |
391 break; | |
392 default: | |
393 device.str = "default"; | |
394 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: %d channels are not supported\n",channels); | |
395 } | |
396 device.len = strlen(device.str); | |
397 if (subopt_parse(ao_subdevice, subopts) != 0) { | |
398 print_help(); | |
399 return 0; | |
400 } | |
401 ao_noblock = !block; | |
402 parse_device(alsa_device, device.str, device.len); | |
12465 | 403 |
404 mp_msg(MSGT_AO,MSGL_INFO,"alsa-init: %d soundcard%s found, using: %s\n", cards+1,(cards >= 0) ? "" : "s", alsa_device); | |
405 | |
406 //setting modes for block or nonblock-mode | |
407 if (ao_noblock) { | |
408 open_mode = SND_PCM_NONBLOCK; | |
409 } | |
410 else { | |
411 open_mode = 0; | |
412 } | |
413 | |
414 //sets buff/chunksize if its set manually | |
415 if (ao_data.buffersize) { | |
416 switch (ao_data.buffersize) | |
417 { | |
418 case 1: | |
419 alsa_fragcount = 16; | |
420 chunk_size = 512; | |
421 mp_msg(MSGT_AO,MSGL_V,"alsa-init: buffersize set manually to 8192\n"); | |
422 mp_msg(MSGT_AO,MSGL_V,"alsa-init: chunksize set manually to 512\n"); | |
423 break; | |
424 case 2: | |
425 alsa_fragcount = 8; | |
426 chunk_size = 1024; | |
427 mp_msg(MSGT_AO,MSGL_V,"alsa-init: buffersize set manually to 8192\n"); | |
428 mp_msg(MSGT_AO,MSGL_V,"alsa-init: chunksize set manually to 1024\n"); | |
429 break; | |
430 case 3: | |
431 alsa_fragcount = 32; | |
432 chunk_size = 512; | |
433 mp_msg(MSGT_AO,MSGL_V,"alsa-init: buffersize set manually to 16384\n"); | |
434 mp_msg(MSGT_AO,MSGL_V,"alsa-init: chunksize set manually to 512\n"); | |
435 break; | |
436 case 4: | |
437 alsa_fragcount = 16; | |
438 chunk_size = 1024; | |
439 mp_msg(MSGT_AO,MSGL_V,"alsa-init: buffersize set manually to 16384\n"); | |
440 mp_msg(MSGT_AO,MSGL_V,"alsa-init: chunksize set manually to 1024\n"); | |
441 break; | |
442 default: | |
443 alsa_fragcount = 16; | |
17616
92431bc3d014
This patch removes mmap support because it doesn't have any benefit.
cladisch
parents:
17575
diff
changeset
|
444 chunk_size = 1024; |
12465 | 445 break; |
446 } | |
447 } | |
448 | |
449 if (!alsa_handler) { | |
450 //modes = 0, SND_PCM_NONBLOCK, SND_PCM_ASYNC | |
451 if ((err = snd_pcm_open(&alsa_handler, alsa_device, SND_PCM_STREAM_PLAYBACK, open_mode)) < 0) | |
452 { | |
453 if (err != -EBUSY && ao_noblock) { | |
454 mp_msg(MSGT_AO,MSGL_INFO,"alsa-init: open in nonblock-mode failed, trying to open in block-mode\n"); | |
455 if ((err = snd_pcm_open(&alsa_handler, alsa_device, SND_PCM_STREAM_PLAYBACK, 0)) < 0) { | |
456 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: playback open error: %s\n", snd_strerror(err)); | |
457 return(0); | |
458 } | |
459 } else { | |
460 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: playback open error: %s\n", snd_strerror(err)); | |
461 return(0); | |
462 } | |
463 } | |
464 | |
17619
9b619133f11a
Using non-blocking writes makes sense when the program wants to do other
cladisch
parents:
17618
diff
changeset
|
465 if ((err = snd_pcm_nonblock(alsa_handler, 0)) < 0) { |
12465 | 466 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: error set block-mode %s\n", snd_strerror(err)); |
467 } else { | |
17619
9b619133f11a
Using non-blocking writes makes sense when the program wants to do other
cladisch
parents:
17618
diff
changeset
|
468 mp_msg(MSGT_AO,MSGL_V,"alsa-init: pcm opend in blocking mode\n"); |
12465 | 469 } |
470 | |
471 snd_pcm_hw_params_alloca(&alsa_hwparams); | |
472 snd_pcm_sw_params_alloca(&alsa_swparams); | |
473 | |
474 // setting hw-parameters | |
475 if ((err = snd_pcm_hw_params_any(alsa_handler, alsa_hwparams)) < 0) | |
476 { | |
477 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to get initial parameters: %s\n", | |
478 snd_strerror(err)); | |
479 return(0); | |
480 } | |
481 | |
17616
92431bc3d014
This patch removes mmap support because it doesn't have any benefit.
cladisch
parents:
17575
diff
changeset
|
482 err = snd_pcm_hw_params_set_access(alsa_handler, alsa_hwparams, |
92431bc3d014
This patch removes mmap support because it doesn't have any benefit.
cladisch
parents:
17575
diff
changeset
|
483 SND_PCM_ACCESS_RW_INTERLEAVED); |
12465 | 484 if (err < 0) { |
485 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set access type: %s\n", | |
486 snd_strerror(err)); | |
487 return (0); | |
488 } | |
489 | |
490 /* workaround for nonsupported formats | |
491 sets default format to S16_LE if the given formats aren't supported */ | |
492 if ((err = snd_pcm_hw_params_test_format(alsa_handler, alsa_hwparams, | |
493 alsa_format)) < 0) | |
494 { | |
495 mp_msg(MSGT_AO,MSGL_INFO, | |
14264 | 496 "alsa-init: format %s are not supported by hardware, trying default\n", af_fmt2str_short(format)); |
12465 | 497 alsa_format = SND_PCM_FORMAT_S16_LE; |
14245 | 498 ao_data.format = AF_FORMAT_S16_LE; |
12465 | 499 } |
500 | |
501 if ((err = snd_pcm_hw_params_set_format(alsa_handler, alsa_hwparams, | |
502 alsa_format)) < 0) | |
503 { | |
504 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set format: %s\n", | |
505 snd_strerror(err)); | |
16308
41278ab73e9b
set the nearest number of channels, return(0) upon errors
henry
parents:
14849
diff
changeset
|
506 return(0); |
12465 | 507 } |
508 | |
16308
41278ab73e9b
set the nearest number of channels, return(0) upon errors
henry
parents:
14849
diff
changeset
|
509 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
|
510 &ao_data.channels)) < 0) |
12465 | 511 { |
512 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set channels: %s\n", | |
513 snd_strerror(err)); | |
16308
41278ab73e9b
set the nearest number of channels, return(0) upon errors
henry
parents:
14849
diff
changeset
|
514 return(0); |
12465 | 515 } |
516 | |
517 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
|
518 &ao_data.samplerate, NULL)) < 0) |
12465 | 519 { |
520 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set samplerate-2: %s\n", | |
521 snd_strerror(err)); | |
522 return(0); | |
523 } | |
524 | |
17570
401521ec0d61
This replaces the hardcoded numbers for the sample format widths with a
cladisch
parents:
17566
diff
changeset
|
525 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
|
526 bytes_per_sample *= ao_data.channels; |
401521ec0d61
This replaces the hardcoded numbers for the sample format widths with a
cladisch
parents:
17566
diff
changeset
|
527 ao_data.bps = ao_data.samplerate * bytes_per_sample; |
16309 | 528 |
12465 | 529 #ifdef BUFFERTIME |
530 { | |
531 int alsa_buffer_time = 500000; /* original 60 */ | |
532 int alsa_period_time; | |
533 alsa_period_time = alsa_buffer_time/4; | |
534 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
|
535 &alsa_buffer_time, NULL)) < 0) |
12465 | 536 { |
537 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set buffer time near: %s\n", | |
538 snd_strerror(err)); | |
539 return(0); | |
540 } else | |
541 alsa_buffer_time = err; | |
542 | |
543 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
|
544 &alsa_period_time, NULL)) < 0) |
12465 | 545 /* original: alsa_buffer_time/ao_data.bps */ |
546 { | |
547 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set period time: %s\n", | |
548 snd_strerror(err)); | |
549 } | |
550 mp_msg(MSGT_AO,MSGL_INFO,"alsa-init: buffer_time: %d, period_time :%d\n", | |
551 alsa_buffer_time, err); | |
552 } | |
553 #endif//end SET_BUFFERTIME | |
554 | |
555 #ifdef SET_CHUNKSIZE | |
556 { | |
557 //set chunksize | |
558 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
|
559 &chunk_size, NULL)) < 0) |
12465 | 560 { |
17366 | 561 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set periodsize(%ld): %s\n", |
12465 | 562 chunk_size, snd_strerror(err)); |
563 } | |
564 else { | |
17366 | 565 mp_msg(MSGT_AO,MSGL_V,"alsa-init: chunksize set to %li\n", chunk_size); |
12465 | 566 } |
567 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
|
568 &alsa_fragcount, NULL)) < 0) { |
12465 | 569 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set periods: %s\n", |
570 snd_strerror(err)); | |
571 } | |
572 else { | |
573 mp_msg(MSGT_AO,MSGL_V,"alsa-init: fragcount=%i\n", alsa_fragcount); | |
574 } | |
575 } | |
576 #endif//end SET_CHUNKSIZE | |
577 | |
578 /* finally install hardware parameters */ | |
579 if ((err = snd_pcm_hw_params(alsa_handler, alsa_hwparams)) < 0) | |
580 { | |
581 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set hw-parameters: %s\n", | |
582 snd_strerror(err)); | |
583 } | |
584 // end setting hw-params | |
585 | |
586 | |
587 // gets buffersize for control | |
588 if ((err = snd_pcm_hw_params_get_buffer_size(alsa_hwparams, &bufsize)) < 0) | |
589 { | |
590 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to get buffersize: %s\n", snd_strerror(err)); | |
591 } | |
592 else { | |
593 ao_data.buffersize = bufsize * bytes_per_sample; | |
594 mp_msg(MSGT_AO,MSGL_V,"alsa-init: got buffersize=%i\n", ao_data.buffersize); | |
595 } | |
596 | |
17620
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
597 if ((err = snd_pcm_hw_params_get_period_size(alsa_hwparams, &chunk_size, NULL)) < 0) { |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
598 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to get period size: %s\n", snd_strerror(err)); |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
599 } else { |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
600 mp_msg(MSGT_AO,MSGL_V,"alsa-init: got period size %li\n", chunk_size); |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
601 } |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
602 ao_data.outburst = chunk_size * bytes_per_sample; |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
603 |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
604 /* setting software parameters */ |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
605 if ((err = snd_pcm_sw_params_current(alsa_handler, alsa_swparams)) < 0) { |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
606 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to get sw-parameters: %s\n", |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
607 snd_strerror(err)); |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
608 return 0; |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
609 } |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
610 if ((err = snd_pcm_sw_params_get_boundary(alsa_swparams, &boundary)) < 0) { |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
611 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to get boundary: %s\n", |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
612 snd_strerror(err)); |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
613 return 0; |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
614 } |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
615 /* start playing when one period has been written */ |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
616 if ((err = snd_pcm_sw_params_set_start_threshold(alsa_handler, alsa_swparams, chunk_size)) < 0) { |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
617 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set start threshold: %s\n", |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
618 snd_strerror(err)); |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
619 return 0; |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
620 } |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
621 /* disable underrun reporting */ |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
622 if ((err = snd_pcm_sw_params_set_stop_threshold(alsa_handler, alsa_swparams, boundary)) < 0) { |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
623 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set stop threshold: %s\n", |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
624 snd_strerror(err)); |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
625 return 0; |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
626 } |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
627 /* play silence when there is an underrun */ |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
628 if ((err = snd_pcm_sw_params_set_silence_size(alsa_handler, alsa_swparams, boundary)) < 0) { |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
629 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set silence size: %s\n", |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
630 snd_strerror(err)); |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
631 return 0; |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
632 } |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
633 if ((err = snd_pcm_sw_params(alsa_handler, alsa_swparams)) < 0) { |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
634 mp_msg(MSGT_AO,MSGL_ERR,"alsa-init: unable to set sw-parameters: %s\n", |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
635 snd_strerror(err)); |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
636 return 0; |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
637 } |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
638 /* end setting sw-params */ |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
639 |
12465 | 640 mp_msg(MSGT_AO,MSGL_INFO,"alsa: %d Hz/%d channels/%d bpf/%d bytes buffer/%s\n", |
641 ao_data.samplerate, ao_data.channels, bytes_per_sample, ao_data.buffersize, | |
642 snd_pcm_format_description(alsa_format)); | |
643 | |
644 } // end switch alsa_handler (spdif) | |
645 alsa_can_pause = snd_pcm_hw_params_can_pause(alsa_hwparams); | |
646 return(1); | |
647 } // end init | |
648 | |
649 | |
650 /* close audio device */ | |
651 static void uninit(int immed) | |
652 { | |
653 | |
654 if (alsa_handler) { | |
655 int err; | |
656 | |
14849
d313f591d1a4
aos should respect the immed uninit flag (quit immediatly vs waiting till file
reimar
parents:
14612
diff
changeset
|
657 if (!immed) |
d313f591d1a4
aos should respect the immed uninit flag (quit immediatly vs waiting till file
reimar
parents:
14612
diff
changeset
|
658 snd_pcm_drain(alsa_handler); |
d313f591d1a4
aos should respect the immed uninit flag (quit immediatly vs waiting till file
reimar
parents:
14612
diff
changeset
|
659 |
12465 | 660 if ((err = snd_pcm_close(alsa_handler)) < 0) |
661 { | |
662 mp_msg(MSGT_AO,MSGL_ERR,"alsa-uninit: pcm close error: %s\n", snd_strerror(err)); | |
663 return; | |
664 } | |
665 else { | |
666 alsa_handler = NULL; | |
667 mp_msg(MSGT_AO,MSGL_INFO,"alsa-uninit: pcm closed\n"); | |
668 } | |
669 } | |
670 else { | |
671 mp_msg(MSGT_AO,MSGL_ERR,"alsa-uninit: no handler defined!\n"); | |
672 } | |
673 } | |
674 | |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
675 static void audio_pause(void) |
12465 | 676 { |
677 int err; | |
678 | |
679 if (alsa_can_pause) { | |
680 if ((err = snd_pcm_pause(alsa_handler, 1)) < 0) | |
681 { | |
682 mp_msg(MSGT_AO,MSGL_ERR,"alsa-pause: pcm pause error: %s\n", snd_strerror(err)); | |
683 return; | |
684 } | |
685 mp_msg(MSGT_AO,MSGL_V,"alsa-pause: pause supported by hardware\n"); | |
686 } else { | |
687 if ((err = snd_pcm_drop(alsa_handler)) < 0) | |
688 { | |
689 mp_msg(MSGT_AO,MSGL_ERR,"alsa-pause: pcm drop error: %s\n", snd_strerror(err)); | |
690 return; | |
691 } | |
692 } | |
693 } | |
694 | |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
695 static void audio_resume(void) |
12465 | 696 { |
697 int err; | |
698 | |
699 if (alsa_can_pause) { | |
700 if ((err = snd_pcm_pause(alsa_handler, 0)) < 0) | |
701 { | |
702 mp_msg(MSGT_AO,MSGL_ERR,"alsa-resume: pcm resume error: %s\n", snd_strerror(err)); | |
703 return; | |
704 } | |
705 mp_msg(MSGT_AO,MSGL_V,"alsa-resume: resume supported by hardware\n"); | |
706 } else { | |
707 if ((err = snd_pcm_prepare(alsa_handler)) < 0) | |
708 { | |
709 mp_msg(MSGT_AO,MSGL_ERR,"alsa-resume: pcm prepare error: %s\n", snd_strerror(err)); | |
710 return; | |
711 } | |
712 } | |
713 } | |
714 | |
715 /* stop playing and empty buffers (for seeking/pause) */ | |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
716 static void reset(void) |
12465 | 717 { |
718 int err; | |
719 | |
720 if ((err = snd_pcm_drop(alsa_handler)) < 0) | |
721 { | |
722 mp_msg(MSGT_AO,MSGL_ERR,"alsa-reset: pcm drop error: %s\n", snd_strerror(err)); | |
723 return; | |
724 } | |
725 if ((err = snd_pcm_prepare(alsa_handler)) < 0) | |
726 { | |
727 mp_msg(MSGT_AO,MSGL_ERR,"alsa-reset: pcm prepare error: %s\n", snd_strerror(err)); | |
728 return; | |
729 } | |
730 return; | |
731 } | |
732 | |
733 /* | |
734 plays 'len' bytes of 'data' | |
735 returns: number of bytes played | |
736 modified last at 29.06.02 by jp | |
737 thanxs for marius <marius@rospot.com> for giving us the light ;) | |
738 */ | |
739 | |
17617
adfab82139c0
After removing play_mmap(), the play() function just unconditionally
cladisch
parents:
17616
diff
changeset
|
740 static int play(void* data, int len, int flags) |
12465 | 741 { |
742 int num_frames = len / bytes_per_sample; | |
743 snd_pcm_sframes_t res = 0; | |
744 | |
745 //mp_msg(MSGT_AO,MSGL_ERR,"alsa-play: frames=%i, len=%i\n",num_frames,len); | |
746 | |
747 if (!alsa_handler) { | |
748 mp_msg(MSGT_AO,MSGL_ERR,"alsa-play: device configuration error"); | |
749 return 0; | |
750 } | |
751 | |
17621
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
752 if (num_frames == 0) |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
753 return 0; |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
754 |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
755 do { |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
756 res = snd_pcm_writei(alsa_handler, data, num_frames); |
12465 | 757 |
17621
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
758 if (res == -EINTR) { |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
759 /* nothing to do */ |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
760 res = 0; |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
761 } |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
762 else if (res == -ESTRPIPE) { /* suspend */ |
12465 | 763 mp_msg(MSGT_AO,MSGL_INFO,"alsa-play: pcm in suspend mode. trying to resume\n"); |
764 while ((res = snd_pcm_resume(alsa_handler)) == -EAGAIN) | |
765 sleep(1); | |
766 } | |
17621
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
767 if (res < 0) { |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
768 mp_msg(MSGT_AO,MSGL_ERR,"alsa-play: write error: %s\n", snd_strerror(res)); |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
769 mp_msg(MSGT_AO,MSGL_INFO,"alsa-play: trying to reset soundcard\n"); |
12465 | 770 if ((res = snd_pcm_prepare(alsa_handler)) < 0) { |
17621
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
771 mp_msg(MSGT_AO,MSGL_ERR,"alsa-play: pcm prepare error: %s\n", snd_strerror(res)); |
12465 | 772 return(0); |
773 break; | |
774 } | |
775 } | |
17621
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
776 } while (res == 0); |
12465 | 777 |
17621
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
778 return res < 0 ? res : res * bytes_per_sample; |
12465 | 779 } |
780 | |
781 /* how many byes are free in the buffer */ | |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
782 static int get_space(void) |
12465 | 783 { |
784 snd_pcm_status_t *status; | |
785 int ret; | |
786 | |
12747 | 787 snd_pcm_status_alloca(&status); |
12465 | 788 |
789 if ((ret = snd_pcm_status(alsa_handler, status)) < 0) | |
790 { | |
791 mp_msg(MSGT_AO,MSGL_ERR,"alsa-space: cannot get pcm status: %s\n", snd_strerror(ret)); | |
792 return(0); | |
793 } | |
794 | |
17572
580dc69d69bf
Fix get_space(): we don't need to differentiate between the various PCM
cladisch
parents:
17571
diff
changeset
|
795 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
|
796 if (ret > MAX_OUTBURST) |
580dc69d69bf
Fix get_space(): we don't need to differentiate between the various PCM
cladisch
parents:
17571
diff
changeset
|
797 ret = MAX_OUTBURST; |
12465 | 798 return(ret); |
799 } | |
800 | |
801 /* delay in seconds between first and last sample in buffer */ | |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
802 static float get_delay(void) |
12465 | 803 { |
804 if (alsa_handler) { | |
17573
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
805 snd_pcm_sframes_t delay; |
12465 | 806 |
17573
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
807 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
|
808 return 0; |
12465 | 809 |
17573
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
810 if (delay < 0) { |
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
811 /* 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
|
812 #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
|
813 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
|
814 #endif |
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
815 delay = 0; |
12465 | 816 } |
17573
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
817 return (float)delay / (float)ao_data.samplerate; |
12465 | 818 } else { |
819 return(0); | |
820 } | |
821 } |