Mercurial > mplayer.hg
annotate libao2/ao_alsa.c @ 37095:6cce7a87e800
Cosmetic: Adjust indent.
author | ib |
---|---|
date | Wed, 07 May 2014 14:16:02 +0000 |
parents | 86893b300a45 |
children |
rev | line source |
---|---|
12465 | 1 /* |
28343 | 2 * ALSA 0.9.x-1.x audio output driver |
3 * | |
4 * Copyright (C) 2004 Alex Beregszaszi | |
5 * | |
6 * modified for real ALSA 0.9.0 support by Zsolt Barat <joy@streamminister.de> | |
7 * additional AC-3 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 * This file is part of MPlayer. | |
13 * | |
14 * MPlayer is free software; you can redistribute it and/or modify | |
15 * it under the terms of the GNU General Public License as published by | |
16 * the Free Software Foundation; either version 2 of the License, or | |
17 * (at your option) any later version. | |
18 * | |
19 * MPlayer is distributed in the hope that it will be useful, | |
20 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
21 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
22 * GNU General Public License for more details. | |
23 * | |
24 * You should have received a copy of the GNU General Public License along | |
25 * with MPlayer; if not, write to the Free Software Foundation, Inc., | |
26 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. | |
27 */ | |
12465 | 28 |
29 #include <errno.h> | |
30 #include <sys/time.h> | |
31 #include <stdlib.h> | |
17691
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
32 #include <stdarg.h> |
22166
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
33 #include <ctype.h> |
12465 | 34 #include <math.h> |
35 #include <string.h> | |
27281
e47193172586
Our ALSA code needs alloca, so check for it in configure and include alloca.h
reimar
parents:
26757
diff
changeset
|
36 #include <alloca.h> |
34259
a9bfa5a9fcac
ao_alsa: drop check for sys/asoundlib.h and ALSA version
diego
parents:
34103
diff
changeset
|
37 #define ALSA_PCM_NEW_HW_PARAMS_API |
a9bfa5a9fcac
ao_alsa: drop check for sys/asoundlib.h and ALSA version
diego
parents:
34103
diff
changeset
|
38 #define ALSA_PCM_NEW_SW_PARAMS_API |
a9bfa5a9fcac
ao_alsa: drop check for sys/asoundlib.h and ALSA version
diego
parents:
34103
diff
changeset
|
39 #include <alsa/asoundlib.h> |
12465 | 40 |
14123 | 41 #include "config.h" |
14328 | 42 #include "subopt-helper.h" |
14123 | 43 #include "mixer.h" |
44 #include "mp_msg.h" | |
20764 | 45 #include "help_mp.h" |
12465 | 46 #include "audio_out.h" |
47 #include "audio_out_internal.h" | |
14245 | 48 #include "libaf/af_format.h" |
12465 | 49 |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
50 static const ao_info_t info = |
12465 | 51 { |
52 "ALSA-0.9.x-1.x audio output", | |
53 "alsa", | |
54 "Alex Beregszaszi, Zsolt Barat <joy@streamminister.de>", | |
31834
64ba1daa147a
various spelling fixes, found by the Debian QA tool 'lintian'
siretart
parents:
30410
diff
changeset
|
55 "under development" |
12465 | 56 }; |
57 | |
58 LIBAO_EXTERN(alsa) | |
59 | |
60 static snd_pcm_t *alsa_handler; | |
61 static snd_pcm_format_t alsa_format; | |
62 static snd_pcm_hw_params_t *alsa_hwparams; | |
63 static snd_pcm_sw_params_t *alsa_swparams; | |
64 | |
17619
9b619133f11a
Using non-blocking writes makes sense when the program wants to do other
cladisch
parents:
17618
diff
changeset
|
65 static size_t bytes_per_sample; |
12465 | 66 |
30399 | 67 static int alsa_can_pause; |
34754 | 68 static int prepause_space; |
12465 | 69 |
12747 | 70 #define ALSA_DEVICE_SIZE 256 |
12465 | 71 |
17691
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
72 static void alsa_error_handler(const char *file, int line, const char *function, |
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
73 int err, const char *format, ...) |
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
74 { |
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
75 char tmp[0xc00]; |
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
76 va_list va; |
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
77 |
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
78 va_start(va, format); |
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
79 vsnprintf(tmp, sizeof tmp, format, va); |
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
80 va_end(va); |
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
81 |
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
82 if (err) |
20764 | 83 mp_msg(MSGT_AO, MSGL_ERR, "[AO_ALSA] alsa-lib: %s:%i:(%s) %s: %s\n", |
17691
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
84 file, line, function, tmp, snd_strerror(err)); |
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
85 else |
20764 | 86 mp_msg(MSGT_AO, MSGL_ERR, "[AO_ALSA] alsa-lib: %s:%i:(%s) %s\n", |
17691
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
87 file, line, function, tmp); |
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
88 } |
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
89 |
12465 | 90 /* to set/get/query special features/parameters */ |
91 static int control(int cmd, void *arg) | |
92 { | |
93 switch(cmd) { | |
94 case AOCONTROL_QUERY_FORMAT: | |
95 return CONTROL_TRUE; | |
96 case AOCONTROL_GET_VOLUME: | |
97 case AOCONTROL_SET_VOLUME: | |
98 { | |
99 ao_control_vol_t *vol = (ao_control_vol_t *)arg; | |
100 | |
101 int err; | |
102 snd_mixer_t *handle; | |
103 snd_mixer_elem_t *elem; | |
104 snd_mixer_selem_id_t *sid; | |
105 | |
30400 | 106 char *mix_name = "PCM"; |
107 char *card = "default"; | |
108 int mix_index = 0; | |
12465 | 109 |
110 long pmin, pmax; | |
111 long get_vol, set_vol; | |
12805
0b154063a3ca
fixes provided by reimar d«Órfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
112 float f_multi; |
12465 | 113 |
35443
f045a1d92c06
Make AF_FORMAT_IS_IEC61937 include AF_FORMAT_IS_AC3.
reimar
parents:
35200
diff
changeset
|
114 if(AF_FORMAT_IS_IEC61937(ao_data.format)) |
28118
4455edd2be89
100l, reorder check for AC3 format to avoid a possible memleak
reimar
parents:
27706
diff
changeset
|
115 return CONTROL_TRUE; |
4455edd2be89
100l, reorder check for AC3 format to avoid a possible memleak
reimar
parents:
27706
diff
changeset
|
116 |
13434 | 117 if(mixer_channel) { |
118 char *test_mix_index; | |
119 | |
120 mix_name = strdup(mixer_channel); | |
17097 | 121 if ((test_mix_index = strchr(mix_name, ','))){ |
13434 | 122 *test_mix_index = 0; |
123 test_mix_index++; | |
124 mix_index = strtol(test_mix_index, &test_mix_index, 0); | |
125 | |
126 if (*test_mix_index){ | |
127 mp_msg(MSGT_AO,MSGL_ERR, | |
20764 | 128 MSGTR_AO_ALSA_InvalidMixerIndexDefaultingToZero); |
13434 | 129 mix_index = 0 ; |
130 } | |
131 } | |
132 } | |
12747 | 133 if(mixer_device) card = mixer_device; |
12465 | 134 |
135 //allocate simple id | |
136 snd_mixer_selem_id_alloca(&sid); | |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
137 |
12465 | 138 //sets simple-mixer index and name |
13434 | 139 snd_mixer_selem_id_set_index(sid, mix_index); |
12465 | 140 snd_mixer_selem_id_set_name(sid, mix_name); |
141 | |
13434 | 142 if (mixer_channel) { |
143 free(mix_name); | |
144 mix_name = NULL; | |
145 } | |
146 | |
12465 | 147 if ((err = snd_mixer_open(&handle, 0)) < 0) { |
20764 | 148 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_MixerOpenError, snd_strerror(err)); |
12465 | 149 return CONTROL_ERROR; |
150 } | |
151 | |
152 if ((err = snd_mixer_attach(handle, card)) < 0) { | |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
153 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_MixerAttachError, |
12465 | 154 card, snd_strerror(err)); |
155 snd_mixer_close(handle); | |
156 return CONTROL_ERROR; | |
157 } | |
158 | |
159 if ((err = snd_mixer_selem_register(handle, NULL, NULL)) < 0) { | |
20764 | 160 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_MixerRegisterError, snd_strerror(err)); |
12465 | 161 snd_mixer_close(handle); |
162 return CONTROL_ERROR; | |
163 } | |
164 err = snd_mixer_load(handle); | |
165 if (err < 0) { | |
20764 | 166 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_MixerLoadError, snd_strerror(err)); |
12465 | 167 snd_mixer_close(handle); |
168 return CONTROL_ERROR; | |
169 } | |
170 | |
171 elem = snd_mixer_find_selem(handle, sid); | |
37089
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
172 if (!elem && !mixer_channel) { |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
173 // if nothing specified just try picking the first |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
174 elem = snd_mixer_first_elem(handle); |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
175 if (elem) |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
176 mp_msg(MSGT_AO, MSGL_V, "[AO_ALSA] Auto-selected element %s\n", snd_mixer_selem_get_name(elem)); |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
177 } |
12465 | 178 if (!elem) { |
20764 | 179 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToFindSimpleControl, |
12465 | 180 snd_mixer_selem_id_get_name(sid), snd_mixer_selem_id_get_index(sid)); |
37089
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
181 mp_msg(MSGT_AO, MSGL_INFO, "[AO_ALSA] Available elements for device:\n"); |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
182 for (elem = snd_mixer_first_elem(handle); elem; elem = snd_mixer_elem_next(elem)) |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
183 mp_msg(MSGT_AO, MSGL_INFO, "[AO_ALSA] %s\n", snd_mixer_selem_get_name(elem)); |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
184 mp_msg(MSGT_AO, MSGL_INFO, "\n"); |
12465 | 185 snd_mixer_close(handle); |
186 return CONTROL_ERROR; | |
187 } | |
188 | |
37089
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
189 if (!snd_mixer_selem_has_playback_volume(elem) && |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
190 !snd_mixer_selem_has_playback_volume_joined(elem)) |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
191 { |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
192 if (!snd_mixer_selem_has_playback_switch(elem)) |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
193 return CONTROL_FALSE; |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
194 if (cmd == AOCONTROL_GET_VOLUME) { |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
195 int v; |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
196 snd_mixer_selem_get_playback_switch(elem, SND_MIXER_SCHN_FRONT_LEFT, &v); |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
197 vol->left = v ? 100 : 0; |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
198 snd_mixer_selem_get_playback_switch(elem, SND_MIXER_SCHN_FRONT_RIGHT, &v); |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
199 vol->right = v ? 100 : 0; |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
200 return CONTROL_TRUE; |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
201 } |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
202 // special case: only mute supported |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
203 if (vol->left != 0 && vol->left != 100 || |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
204 vol->right != 0 && vol->right != 100) |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
205 return CONTROL_FALSE; |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
206 if (snd_mixer_selem_has_playback_switch_joined(elem)) { |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
207 if (vol->left != vol->right) |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
208 return CONTROL_FALSE; |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
209 } else { |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
210 snd_mixer_selem_set_playback_switch(elem, SND_MIXER_SCHN_FRONT_RIGHT, vol->right != 0); |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
211 } |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
212 snd_mixer_selem_set_playback_switch(elem, SND_MIXER_SCHN_FRONT_LEFT, vol->left != 0); |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
213 return CONTROL_TRUE; |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
214 } |
86893b300a45
alsa: Better volume control handling, especially for S/PDIF.
reimar
parents:
36493
diff
changeset
|
215 |
12465 | 216 snd_mixer_selem_get_playback_volume_range(elem,&pmin,&pmax); |
12811
d5f8efddac6c
volume calc fixes for mixer, by reimar d«Óffinger, 10l reverse by me
joyping
parents:
12805
diff
changeset
|
217 f_multi = (100 / (float)(pmax - pmin)); |
12465 | 218 |
219 if (cmd == AOCONTROL_SET_VOLUME) { | |
220 | |
12811
d5f8efddac6c
volume calc fixes for mixer, by reimar d«Óffinger, 10l reverse by me
joyping
parents:
12805
diff
changeset
|
221 set_vol = vol->left / f_multi + pmin + 0.5; |
12465 | 222 |
223 //setting channels | |
27706 | 224 if ((err = snd_mixer_selem_set_playback_volume(elem, SND_MIXER_SCHN_FRONT_LEFT, set_vol)) < 0) { |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
225 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_ErrorSettingLeftChannel, |
12465 | 226 snd_strerror(err)); |
29436
4537ed655f0b
Do not leak the mixer handle if setting of a volume fails.
cladisch
parents:
29401
diff
changeset
|
227 snd_mixer_close(handle); |
12465 | 228 return CONTROL_ERROR; |
229 } | |
12805
0b154063a3ca
fixes provided by reimar d«Órfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
230 mp_msg(MSGT_AO,MSGL_DBG2,"left=%li, ", set_vol); |
0b154063a3ca
fixes provided by reimar d«Órfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
231 |
12811
d5f8efddac6c
volume calc fixes for mixer, by reimar d«Óffinger, 10l reverse by me
joyping
parents:
12805
diff
changeset
|
232 set_vol = vol->right / f_multi + pmin + 0.5; |
12805
0b154063a3ca
fixes provided by reimar d«Órfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
233 |
27706 | 234 if ((err = snd_mixer_selem_set_playback_volume(elem, SND_MIXER_SCHN_FRONT_RIGHT, set_vol)) < 0) { |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
235 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_ErrorSettingRightChannel, |
12465 | 236 snd_strerror(err)); |
29436
4537ed655f0b
Do not leak the mixer handle if setting of a volume fails.
cladisch
parents:
29401
diff
changeset
|
237 snd_mixer_close(handle); |
12465 | 238 return CONTROL_ERROR; |
239 } | |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
240 mp_msg(MSGT_AO,MSGL_DBG2,"right=%li, pmin=%li, pmax=%li, mult=%f\n", |
12805
0b154063a3ca
fixes provided by reimar d«Órfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
241 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
|
242 |
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
|
243 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
|
244 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
|
245 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
|
246 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
|
247 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
|
248 } 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
|
249 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
|
250 } |
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
|
251 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
|
252 } |
12465 | 253 } |
254 else { | |
27706 | 255 snd_mixer_selem_get_playback_volume(elem, SND_MIXER_SCHN_FRONT_LEFT, &get_vol); |
12811
d5f8efddac6c
volume calc fixes for mixer, by reimar d«Óffinger, 10l reverse by me
joyping
parents:
12805
diff
changeset
|
256 vol->left = (get_vol - pmin) * f_multi; |
27706 | 257 snd_mixer_selem_get_playback_volume(elem, SND_MIXER_SCHN_FRONT_RIGHT, &get_vol); |
12811
d5f8efddac6c
volume calc fixes for mixer, by reimar d«Óffinger, 10l reverse by me
joyping
parents:
12805
diff
changeset
|
258 vol->right = (get_vol - pmin) * f_multi; |
12465 | 259 |
12805
0b154063a3ca
fixes provided by reimar d«Órfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
260 mp_msg(MSGT_AO,MSGL_DBG2,"left=%f, right=%f\n",vol->left,vol->right); |
12465 | 261 } |
262 snd_mixer_close(handle); | |
263 return CONTROL_OK; | |
264 } | |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
265 |
12465 | 266 } //end switch |
26757
0fdf04b07ecb
cosmetics: Remove pointless parentheses from return statements.
diego
parents:
24590
diff
changeset
|
267 return CONTROL_UNKNOWN; |
12465 | 268 } |
269 | |
14328 | 270 static void parse_device (char *dest, const char *src, int len) |
12805
0b154063a3ca
fixes provided by reimar d«Órfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
271 { |
0b154063a3ca
fixes provided by reimar d«Órfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
272 char *tmp; |
14328 | 273 memmove(dest, src, len); |
274 dest[len] = 0; | |
35474
e53785e5205b
Do not use strrchr when strchr works just as well.
reimar
parents:
35443
diff
changeset
|
275 while ((tmp = strchr(dest, '.'))) |
12805
0b154063a3ca
fixes provided by reimar d«Órfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
276 tmp[0] = ','; |
35474
e53785e5205b
Do not use strrchr when strchr works just as well.
reimar
parents:
35443
diff
changeset
|
277 while ((tmp = strchr(dest, '='))) |
12805
0b154063a3ca
fixes provided by reimar d«Órfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
278 tmp[0] = ':'; |
0b154063a3ca
fixes provided by reimar d«Órfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
279 } |
0b154063a3ca
fixes provided by reimar d«Órfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
280 |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
281 static void print_help (void) |
12805
0b154063a3ca
fixes provided by reimar d«Órfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
282 { |
0b154063a3ca
fixes provided by reimar d«Órfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
283 mp_msg (MSGT_AO, MSGL_FATAL, |
20764 | 284 MSGTR_AO_ALSA_CommandlineHelp); |
12805
0b154063a3ca
fixes provided by reimar d«Órfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
285 } |
12465 | 286 |
30122
1772a5171ac7
Fix function declarations to avoid casting function pointers.
reimar
parents:
29826
diff
changeset
|
287 static int str_maxlen(void *strp) { |
1772a5171ac7
Fix function declarations to avoid casting function pointers.
reimar
parents:
29826
diff
changeset
|
288 strarg_t *str = strp; |
30123
0f5f75b4a015
Simplify range-checking functions for subopt parsing.
reimar
parents:
30122
diff
changeset
|
289 return str->len <= ALSA_DEVICE_SIZE; |
14328 | 290 } |
291 | |
22166
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
292 static int try_open_device(const char *device, int open_mode, int try_ac3) |
19889
d4bb39d65f87
When the hardware sample format is AC3, do not force using an hardcoded
cladisch
parents:
19887
diff
changeset
|
293 { |
22166
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
294 int err, len; |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
295 char *ac3_device, *args; |
19889
d4bb39d65f87
When the hardware sample format is AC3, do not force using an hardcoded
cladisch
parents:
19887
diff
changeset
|
296 |
22166
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
297 if (try_ac3) { |
36493 | 298 /* to set the non-audio bit, use AES0=6 |
299 * 6 == IEC958_AES0_NONAUDIO | IEC958_AES0_PRO_EMPHASIS_NONE */ | |
22166
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
300 len = strlen(device); |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
301 ac3_device = malloc(len + 7 + 1); |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
302 if (!ac3_device) |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
303 return -ENOMEM; |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
304 strcpy(ac3_device, device); |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
305 args = strchr(ac3_device, ':'); |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
306 if (!args) { |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
307 /* no existing parameters: add it behind device name */ |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
308 strcat(ac3_device, ":AES0=6"); |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
309 } else { |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
310 do |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
311 ++args; |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
312 while (isspace(*args)); |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
313 if (*args == '\0') { |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
314 /* ":" but no parameters */ |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
315 strcat(ac3_device, "AES0=6"); |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
316 } else if (*args != '{') { |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
317 /* a simple list of parameters: add it at the end of the list */ |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
318 strcat(ac3_device, ",AES0=6"); |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
319 } else { |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
320 /* parameters in config syntax: add it inside the { } block */ |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
321 do |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
322 --len; |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
323 while (len > 0 && isspace(ac3_device[len])); |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
324 if (ac3_device[len] == '}') |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
325 strcpy(ac3_device + len, " AES0=6}"); |
19889
d4bb39d65f87
When the hardware sample format is AC3, do not force using an hardcoded
cladisch
parents:
19887
diff
changeset
|
326 } |
d4bb39d65f87
When the hardware sample format is AC3, do not force using an hardcoded
cladisch
parents:
19887
diff
changeset
|
327 } |
22166
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
328 err = snd_pcm_open(&alsa_handler, ac3_device, SND_PCM_STREAM_PLAYBACK, |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
329 open_mode); |
68dceb30bbcf
When setting the non-audio bit for hwac3 output, just try to set the
cladisch
parents:
20764
diff
changeset
|
330 free(ac3_device); |
33347
f60b1fd07712
Simplify code and avoid an incorrect "may be used initialized"
reimar
parents:
31834
diff
changeset
|
331 if (err >= 0) |
f60b1fd07712
Simplify code and avoid an incorrect "may be used initialized"
reimar
parents:
31834
diff
changeset
|
332 return err; |
19889
d4bb39d65f87
When the hardware sample format is AC3, do not force using an hardcoded
cladisch
parents:
19887
diff
changeset
|
333 } |
33347
f60b1fd07712
Simplify code and avoid an incorrect "may be used initialized"
reimar
parents:
31834
diff
changeset
|
334 return snd_pcm_open(&alsa_handler, device, SND_PCM_STREAM_PLAYBACK, open_mode); |
19889
d4bb39d65f87
When the hardware sample format is AC3, do not force using an hardcoded
cladisch
parents:
19887
diff
changeset
|
335 } |
d4bb39d65f87
When the hardware sample format is AC3, do not force using an hardcoded
cladisch
parents:
19887
diff
changeset
|
336 |
12465 | 337 /* |
338 open & setup audio device | |
339 return: 1=success 0=fail | |
340 */ | |
341 static int init(int rate_hz, int channels, int format, int flags) | |
342 { | |
30398 | 343 unsigned int alsa_buffer_time = 500000; /* 0.5 s */ |
344 unsigned int alsa_fragcount = 16; | |
12465 | 345 int err; |
14328 | 346 int block; |
347 strarg_t device; | |
29507
fc8416cffdcd
Use a buffer of about half a second, instead of sizing it to have
cladisch
parents:
29436
diff
changeset
|
348 snd_pcm_uframes_t chunk_size; |
12465 | 349 snd_pcm_uframes_t bufsize; |
17620
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
350 snd_pcm_uframes_t boundary; |
29586
2eff450157cd
The suboption parser now takes a const options list, so mark them all const.
reimar
parents:
29507
diff
changeset
|
351 const opt_t subopts[] = { |
14328 | 352 {"block", OPT_ARG_BOOL, &block, NULL}, |
30122
1772a5171ac7
Fix function declarations to avoid casting function pointers.
reimar
parents:
29826
diff
changeset
|
353 {"device", OPT_ARG_STR, &device, str_maxlen}, |
14328 | 354 {NULL} |
355 }; | |
356 | |
12747 | 357 char alsa_device[ALSA_DEVICE_SIZE + 1]; |
358 // make sure alsa_device is null-terminated even when using strncpy etc. | |
359 memset(alsa_device, 0, ALSA_DEVICE_SIZE + 1); | |
12465 | 360 |
14249 | 361 mp_msg(MSGT_AO,MSGL_V,"alsa-init: requested format: %d Hz, %d channels, %x\n", rate_hz, |
362 channels, format); | |
12465 | 363 alsa_handler = NULL; |
17690
9ca95aee8449
Show the actual ALSA version instead of the version mplayer was compiled
cladisch
parents:
17621
diff
changeset
|
364 #if SND_LIB_VERSION >= 0x010005 |
9ca95aee8449
Show the actual ALSA version instead of the version mplayer was compiled
cladisch
parents:
17621
diff
changeset
|
365 mp_msg(MSGT_AO,MSGL_V,"alsa-init: using ALSA %s\n", snd_asoundlib_version()); |
9ca95aee8449
Show the actual ALSA version instead of the version mplayer was compiled
cladisch
parents:
17621
diff
changeset
|
366 #else |
12465 | 367 mp_msg(MSGT_AO,MSGL_V,"alsa-init: compiled for ALSA-%s\n", SND_LIB_VERSION_STR); |
17690
9ca95aee8449
Show the actual ALSA version instead of the version mplayer was compiled
cladisch
parents:
17621
diff
changeset
|
368 #endif |
17691
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
369 |
b73103a82416
Output error messages from the ALSA library through mp_msg() instead of
cladisch
parents:
17690
diff
changeset
|
370 snd_lib_error_set_handler(alsa_error_handler); |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
371 |
12465 | 372 ao_data.samplerate = rate_hz; |
373 ao_data.format = format; | |
374 ao_data.channels = channels; | |
375 | |
376 switch (format) | |
377 { | |
14245 | 378 case AF_FORMAT_S8: |
12465 | 379 alsa_format = SND_PCM_FORMAT_S8; |
380 break; | |
14245 | 381 case AF_FORMAT_U8: |
12465 | 382 alsa_format = SND_PCM_FORMAT_U8; |
383 break; | |
14245 | 384 case AF_FORMAT_U16_LE: |
12465 | 385 alsa_format = SND_PCM_FORMAT_U16_LE; |
386 break; | |
14245 | 387 case AF_FORMAT_U16_BE: |
12465 | 388 alsa_format = SND_PCM_FORMAT_U16_BE; |
389 break; | |
30241
02b9c1a452e1
Add support for distinguishing between little- and big-endian SPDIF AC3
reimar
parents:
30237
diff
changeset
|
390 case AF_FORMAT_AC3_LE: |
14245 | 391 case AF_FORMAT_S16_LE: |
34103 | 392 case AF_FORMAT_IEC61937_LE: |
12465 | 393 alsa_format = SND_PCM_FORMAT_S16_LE; |
394 break; | |
30241
02b9c1a452e1
Add support for distinguishing between little- and big-endian SPDIF AC3
reimar
parents:
30237
diff
changeset
|
395 case AF_FORMAT_AC3_BE: |
14245 | 396 case AF_FORMAT_S16_BE: |
34103 | 397 case AF_FORMAT_IEC61937_BE: |
12465 | 398 alsa_format = SND_PCM_FORMAT_S16_BE; |
399 break; | |
17571
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
400 case AF_FORMAT_U32_LE: |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
401 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
|
402 break; |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
403 case AF_FORMAT_U32_BE: |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
404 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
|
405 break; |
14245 | 406 case AF_FORMAT_S32_LE: |
12465 | 407 alsa_format = SND_PCM_FORMAT_S32_LE; |
408 break; | |
14245 | 409 case AF_FORMAT_S32_BE: |
12465 | 410 alsa_format = SND_PCM_FORMAT_S32_BE; |
411 break; | |
29386 | 412 case AF_FORMAT_U24_LE: |
29392
cd3ec59296a0
Use correct ALSA sample format for 24-bit samples packed in three bytes.
cladisch
parents:
29386
diff
changeset
|
413 alsa_format = SND_PCM_FORMAT_U24_3LE; |
29386 | 414 break; |
415 case AF_FORMAT_U24_BE: | |
29392
cd3ec59296a0
Use correct ALSA sample format for 24-bit samples packed in three bytes.
cladisch
parents:
29386
diff
changeset
|
416 alsa_format = SND_PCM_FORMAT_U24_3BE; |
29386 | 417 break; |
418 case AF_FORMAT_S24_LE: | |
29392
cd3ec59296a0
Use correct ALSA sample format for 24-bit samples packed in three bytes.
cladisch
parents:
29386
diff
changeset
|
419 alsa_format = SND_PCM_FORMAT_S24_3LE; |
29386 | 420 break; |
421 case AF_FORMAT_S24_BE: | |
29392
cd3ec59296a0
Use correct ALSA sample format for 24-bit samples packed in three bytes.
cladisch
parents:
29386
diff
changeset
|
422 alsa_format = SND_PCM_FORMAT_S24_3BE; |
29386 | 423 break; |
14245 | 424 case AF_FORMAT_FLOAT_LE: |
12570 | 425 alsa_format = SND_PCM_FORMAT_FLOAT_LE; |
426 break; | |
17571
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
427 case AF_FORMAT_FLOAT_BE: |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
428 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
|
429 break; |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
430 case AF_FORMAT_MU_LAW: |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
431 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
|
432 break; |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
433 case AF_FORMAT_A_LAW: |
e476a1d38087
This adds support for more sample formats (U32, float BE, mu/A-law).
cladisch
parents:
17570
diff
changeset
|
434 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
|
435 break; |
12465 | 436 |
437 default: | |
14251 | 438 alsa_format = SND_PCM_FORMAT_MPEG; //? default should be -1 |
12465 | 439 break; |
440 } | |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
441 |
12805
0b154063a3ca
fixes provided by reimar d«Órfinger. mixer, subdevice parsing, alsa#help,
joyping
parents:
12747
diff
changeset
|
442 //subdevice parsing |
14328 | 443 // set defaults |
444 block = 1; | |
12465 | 445 /* switch for spdif |
446 * sets opening sequence for SPDIF | |
447 * sets also the playback and other switches 'on the fly' | |
448 * while opening the abstract alias for the spdif subdevice | |
449 * 'iec958' | |
450 */ | |
35443
f045a1d92c06
Make AF_FORMAT_IS_IEC61937 include AF_FORMAT_IS_AC3.
reimar
parents:
35200
diff
changeset
|
451 if (AF_FORMAT_IS_IEC61937(format)) { |
19889
d4bb39d65f87
When the hardware sample format is AC3, do not force using an hardcoded
cladisch
parents:
19887
diff
changeset
|
452 device.str = "iec958"; |
34103 | 453 mp_msg(MSGT_AO,MSGL_V,"alsa-spdif-init: playing AC3/iec61937/iec958, %i channels\n", channels); |
12465 | 454 } |
13661
07dc40f25068
Only use S/PDIF output when no other alsa device is set, allows to use
reimar
parents:
13434
diff
changeset
|
455 else |
14328 | 456 /* in any case for multichannel playback we should select |
457 * appropriate device | |
458 */ | |
459 switch (channels) { | |
460 case 1: | |
461 case 2: | |
462 device.str = "default"; | |
463 mp_msg(MSGT_AO,MSGL_V,"alsa-init: setup for 1/2 channel(s)\n"); | |
464 break; | |
465 case 4: | |
466 if (alsa_format == SND_PCM_FORMAT_FLOAT_LE) | |
467 // hack - use the converter plugin | |
468 device.str = "plug:surround40"; | |
469 else | |
470 device.str = "surround40"; | |
471 mp_msg(MSGT_AO,MSGL_V,"alsa-init: device set to surround40\n"); | |
472 break; | |
473 case 6: | |
474 if (alsa_format == SND_PCM_FORMAT_FLOAT_LE) | |
475 device.str = "plug:surround51"; | |
476 else | |
477 device.str = "surround51"; | |
478 mp_msg(MSGT_AO,MSGL_V,"alsa-init: device set to surround51\n"); | |
479 break; | |
29826 | 480 case 8: |
481 if (alsa_format == SND_PCM_FORMAT_FLOAT_LE) | |
482 device.str = "plug:surround71"; | |
483 else | |
484 device.str = "surround71"; | |
485 mp_msg(MSGT_AO,MSGL_V,"alsa-init: device set to surround71\n"); | |
486 break; | |
14328 | 487 default: |
488 device.str = "default"; | |
20764 | 489 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_ChannelsNotSupported,channels); |
14328 | 490 } |
491 device.len = strlen(device.str); | |
492 if (subopt_parse(ao_subdevice, subopts) != 0) { | |
493 print_help(); | |
494 return 0; | |
495 } | |
496 parse_device(alsa_device, device.str, device.len); | |
12465 | 497 |
20185 | 498 mp_msg(MSGT_AO,MSGL_V,"alsa-init: using device %s\n", alsa_device); |
12465 | 499 |
500 if (!alsa_handler) { | |
30402 | 501 int open_mode = block ? 0 : SND_PCM_NONBLOCK; |
35443
f045a1d92c06
Make AF_FORMAT_IS_IEC61937 include AF_FORMAT_IS_AC3.
reimar
parents:
35200
diff
changeset
|
502 int isac3 = AF_FORMAT_IS_IEC61937(format); |
12465 | 503 //modes = 0, SND_PCM_NONBLOCK, SND_PCM_ASYNC |
34763
6834d78ac904
Clarify the used opening modes by improving the verbose status messages.
ib
parents:
34754
diff
changeset
|
504 mp_msg(MSGT_AO,MSGL_V,"alsa-init: opening device in %sblocking mode\n", block ? "" : "non-"); |
30234 | 505 if ((err = try_open_device(alsa_device, open_mode, isac3)) < 0) |
12465 | 506 { |
30401 | 507 if (err != -EBUSY && !block) { |
20764 | 508 mp_msg(MSGT_AO,MSGL_INFO,MSGTR_AO_ALSA_OpenInNonblockModeFailed); |
30234 | 509 if ((err = try_open_device(alsa_device, 0, isac3)) < 0) { |
20764 | 510 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_PlaybackOpenError, snd_strerror(err)); |
26757
0fdf04b07ecb
cosmetics: Remove pointless parentheses from return statements.
diego
parents:
24590
diff
changeset
|
511 return 0; |
12465 | 512 } |
513 } else { | |
20764 | 514 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_PlaybackOpenError, snd_strerror(err)); |
26757
0fdf04b07ecb
cosmetics: Remove pointless parentheses from return statements.
diego
parents:
24590
diff
changeset
|
515 return 0; |
12465 | 516 } |
517 } | |
518 | |
17619
9b619133f11a
Using non-blocking writes makes sense when the program wants to do other
cladisch
parents:
17618
diff
changeset
|
519 if ((err = snd_pcm_nonblock(alsa_handler, 0)) < 0) { |
20764 | 520 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_ErrorSetBlockMode, snd_strerror(err)); |
12465 | 521 } else { |
34763
6834d78ac904
Clarify the used opening modes by improving the verbose status messages.
ib
parents:
34754
diff
changeset
|
522 mp_msg(MSGT_AO,MSGL_V,"alsa-init: device reopened in blocking mode\n"); |
12465 | 523 } |
524 | |
525 snd_pcm_hw_params_alloca(&alsa_hwparams); | |
526 snd_pcm_sw_params_alloca(&alsa_swparams); | |
527 | |
528 // setting hw-parameters | |
529 if ((err = snd_pcm_hw_params_any(alsa_handler, alsa_hwparams)) < 0) | |
530 { | |
20764 | 531 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToGetInitialParameters, |
12465 | 532 snd_strerror(err)); |
26757
0fdf04b07ecb
cosmetics: Remove pointless parentheses from return statements.
diego
parents:
24590
diff
changeset
|
533 return 0; |
12465 | 534 } |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
535 |
17616
92431bc3d014
This patch removes mmap support because it doesn't have any benefit.
cladisch
parents:
17575
diff
changeset
|
536 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
|
537 SND_PCM_ACCESS_RW_INTERLEAVED); |
12465 | 538 if (err < 0) { |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
539 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToSetAccessType, |
12465 | 540 snd_strerror(err)); |
26757
0fdf04b07ecb
cosmetics: Remove pointless parentheses from return statements.
diego
parents:
24590
diff
changeset
|
541 return 0; |
12465 | 542 } |
543 | |
544 /* workaround for nonsupported formats | |
545 sets default format to S16_LE if the given formats aren't supported */ | |
546 if ((err = snd_pcm_hw_params_test_format(alsa_handler, alsa_hwparams, | |
547 alsa_format)) < 0) | |
548 { | |
549 mp_msg(MSGT_AO,MSGL_INFO, | |
20764 | 550 MSGTR_AO_ALSA_FormatNotSupportedByHardware, af_fmt2str_short(format)); |
12465 | 551 alsa_format = SND_PCM_FORMAT_S16_LE; |
30241
02b9c1a452e1
Add support for distinguishing between little- and big-endian SPDIF AC3
reimar
parents:
30237
diff
changeset
|
552 if (AF_FORMAT_IS_AC3(ao_data.format)) |
02b9c1a452e1
Add support for distinguishing between little- and big-endian SPDIF AC3
reimar
parents:
30237
diff
changeset
|
553 ao_data.format = AF_FORMAT_AC3_LE; |
34103 | 554 else if (AF_FORMAT_IS_IEC61937(ao_data.format)) |
555 ao_data.format = AF_FORMAT_IEC61937_LE; | |
30241
02b9c1a452e1
Add support for distinguishing between little- and big-endian SPDIF AC3
reimar
parents:
30237
diff
changeset
|
556 else |
14245 | 557 ao_data.format = AF_FORMAT_S16_LE; |
12465 | 558 } |
559 | |
560 if ((err = snd_pcm_hw_params_set_format(alsa_handler, alsa_hwparams, | |
561 alsa_format)) < 0) | |
562 { | |
20764 | 563 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToSetFormat, |
12465 | 564 snd_strerror(err)); |
26757
0fdf04b07ecb
cosmetics: Remove pointless parentheses from return statements.
diego
parents:
24590
diff
changeset
|
565 return 0; |
12465 | 566 } |
567 | |
16308
41278ab73e9b
set the nearest number of channels, return(0) upon errors
henry
parents:
14849
diff
changeset
|
568 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
|
569 &ao_data.channels)) < 0) |
12465 | 570 { |
20764 | 571 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToSetChannels, |
12465 | 572 snd_strerror(err)); |
26757
0fdf04b07ecb
cosmetics: Remove pointless parentheses from return statements.
diego
parents:
24590
diff
changeset
|
573 return 0; |
12465 | 574 } |
575 | |
17849
ebebe97331af
To avoid a bug in ALSA's rate plugin that causes spurious overruns, try
cladisch
parents:
17848
diff
changeset
|
576 /* workaround for buggy rate plugin (should be fixed in ALSA 1.0.11) |
30410
67b70534caeb
Explain why we still disable the ALSA resampler even though it is probably
reimar
parents:
30402
diff
changeset
|
577 prefer our own resampler, since that allows users to choose the resampler, |
67b70534caeb
Explain why we still disable the ALSA resampler even though it is probably
reimar
parents:
30402
diff
changeset
|
578 even per file if desired */ |
17849
ebebe97331af
To avoid a bug in ALSA's rate plugin that causes spurious overruns, try
cladisch
parents:
17848
diff
changeset
|
579 #if SND_LIB_VERSION >= 0x010009 |
ebebe97331af
To avoid a bug in ALSA's rate plugin that causes spurious overruns, try
cladisch
parents:
17848
diff
changeset
|
580 if ((err = snd_pcm_hw_params_set_rate_resample(alsa_handler, alsa_hwparams, |
ebebe97331af
To avoid a bug in ALSA's rate plugin that causes spurious overruns, try
cladisch
parents:
17848
diff
changeset
|
581 0)) < 0) |
ebebe97331af
To avoid a bug in ALSA's rate plugin that causes spurious overruns, try
cladisch
parents:
17848
diff
changeset
|
582 { |
20764 | 583 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToDisableResampling, |
17849
ebebe97331af
To avoid a bug in ALSA's rate plugin that causes spurious overruns, try
cladisch
parents:
17848
diff
changeset
|
584 snd_strerror(err)); |
26757
0fdf04b07ecb
cosmetics: Remove pointless parentheses from return statements.
diego
parents:
24590
diff
changeset
|
585 return 0; |
17849
ebebe97331af
To avoid a bug in ALSA's rate plugin that causes spurious overruns, try
cladisch
parents:
17848
diff
changeset
|
586 } |
ebebe97331af
To avoid a bug in ALSA's rate plugin that causes spurious overruns, try
cladisch
parents:
17848
diff
changeset
|
587 #endif |
ebebe97331af
To avoid a bug in ALSA's rate plugin that causes spurious overruns, try
cladisch
parents:
17848
diff
changeset
|
588 |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
589 if ((err = snd_pcm_hw_params_set_rate_near(alsa_handler, alsa_hwparams, |
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
590 &ao_data.samplerate, NULL)) < 0) |
12465 | 591 { |
20764 | 592 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToSetSamplerate2, |
12465 | 593 snd_strerror(err)); |
26757
0fdf04b07ecb
cosmetics: Remove pointless parentheses from return statements.
diego
parents:
24590
diff
changeset
|
594 return 0; |
12465 | 595 } |
596 | |
30237
9584cf67cea0
Use af_fmt2bits, it should give more sensible values than snd_pcm_format_physical_width,
reimar
parents:
30234
diff
changeset
|
597 bytes_per_sample = af_fmt2bits(ao_data.format) / 8; |
17570
401521ec0d61
This replaces the hardcoded numbers for the sample format widths with a
cladisch
parents:
17566
diff
changeset
|
598 bytes_per_sample *= ao_data.channels; |
401521ec0d61
This replaces the hardcoded numbers for the sample format widths with a
cladisch
parents:
17566
diff
changeset
|
599 ao_data.bps = ao_data.samplerate * bytes_per_sample; |
16309 | 600 |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
601 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
|
602 &alsa_buffer_time, NULL)) < 0) |
12465 | 603 { |
20764 | 604 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToSetBufferTimeNear, |
12465 | 605 snd_strerror(err)); |
26757
0fdf04b07ecb
cosmetics: Remove pointless parentheses from return statements.
diego
parents:
24590
diff
changeset
|
606 return 0; |
29507
fc8416cffdcd
Use a buffer of about half a second, instead of sizing it to have
cladisch
parents:
29436
diff
changeset
|
607 } |
12465 | 608 |
609 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
|
610 &alsa_fragcount, NULL)) < 0) { |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
611 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToSetPeriods, |
12465 | 612 snd_strerror(err)); |
19887
1259d6add8e6
When one of the PCM configuration function in init() fails, abort
cladisch
parents:
18009
diff
changeset
|
613 return 0; |
12465 | 614 } |
615 | |
616 /* finally install hardware parameters */ | |
617 if ((err = snd_pcm_hw_params(alsa_handler, alsa_hwparams)) < 0) | |
618 { | |
20764 | 619 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToSetHwParameters, |
12465 | 620 snd_strerror(err)); |
19887
1259d6add8e6
When one of the PCM configuration function in init() fails, abort
cladisch
parents:
18009
diff
changeset
|
621 return 0; |
12465 | 622 } |
623 // end setting hw-params | |
624 | |
625 | |
626 // gets buffersize for control | |
627 if ((err = snd_pcm_hw_params_get_buffer_size(alsa_hwparams, &bufsize)) < 0) | |
628 { | |
20764 | 629 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToGetBufferSize, snd_strerror(err)); |
19887
1259d6add8e6
When one of the PCM configuration function in init() fails, abort
cladisch
parents:
18009
diff
changeset
|
630 return 0; |
12465 | 631 } |
632 else { | |
633 ao_data.buffersize = bufsize * bytes_per_sample; | |
634 mp_msg(MSGT_AO,MSGL_V,"alsa-init: got buffersize=%i\n", ao_data.buffersize); | |
635 } | |
636 | |
17620
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
637 if ((err = snd_pcm_hw_params_get_period_size(alsa_hwparams, &chunk_size, NULL)) < 0) { |
20764 | 638 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToGetPeriodSize, snd_strerror(err)); |
19887
1259d6add8e6
When one of the PCM configuration function in init() fails, abort
cladisch
parents:
18009
diff
changeset
|
639 return 0; |
17620
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
640 } else { |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
641 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
|
642 } |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
643 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
|
644 |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
645 /* setting software parameters */ |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
646 if ((err = snd_pcm_sw_params_current(alsa_handler, alsa_swparams)) < 0) { |
20764 | 647 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToGetSwParameters, |
17620
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
648 snd_strerror(err)); |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
649 return 0; |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
650 } |
18009
fb7888812f13
Add workarounds for old prerelease versions of alsa-lib 0.9.0 that did
cladisch
parents:
17849
diff
changeset
|
651 #if SND_LIB_VERSION >= 0x000901 |
17620
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
652 if ((err = snd_pcm_sw_params_get_boundary(alsa_swparams, &boundary)) < 0) { |
20764 | 653 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToGetBoundary, |
17620
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
654 snd_strerror(err)); |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
655 return 0; |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
656 } |
18009
fb7888812f13
Add workarounds for old prerelease versions of alsa-lib 0.9.0 that did
cladisch
parents:
17849
diff
changeset
|
657 #else |
fb7888812f13
Add workarounds for old prerelease versions of alsa-lib 0.9.0 that did
cladisch
parents:
17849
diff
changeset
|
658 boundary = 0x7fffffff; |
fb7888812f13
Add workarounds for old prerelease versions of alsa-lib 0.9.0 that did
cladisch
parents:
17849
diff
changeset
|
659 #endif |
17620
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
660 /* 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
|
661 if ((err = snd_pcm_sw_params_set_start_threshold(alsa_handler, alsa_swparams, chunk_size)) < 0) { |
20764 | 662 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToSetStartThreshold, |
17620
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
663 snd_strerror(err)); |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
664 return 0; |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
665 } |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
666 /* disable underrun reporting */ |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
667 if ((err = snd_pcm_sw_params_set_stop_threshold(alsa_handler, alsa_swparams, boundary)) < 0) { |
20764 | 668 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToSetStopThreshold, |
17620
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
669 snd_strerror(err)); |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
670 return 0; |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
671 } |
18009
fb7888812f13
Add workarounds for old prerelease versions of alsa-lib 0.9.0 that did
cladisch
parents:
17849
diff
changeset
|
672 #if SND_LIB_VERSION >= 0x000901 |
17620
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
673 /* play silence when there is an underrun */ |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
674 if ((err = snd_pcm_sw_params_set_silence_size(alsa_handler, alsa_swparams, boundary)) < 0) { |
20764 | 675 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToSetSilenceSize, |
17620
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
676 snd_strerror(err)); |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
677 return 0; |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
678 } |
18009
fb7888812f13
Add workarounds for old prerelease versions of alsa-lib 0.9.0 that did
cladisch
parents:
17849
diff
changeset
|
679 #endif |
17620
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
680 if ((err = snd_pcm_sw_params(alsa_handler, alsa_swparams)) < 0) { |
20764 | 681 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_UnableToGetSwParameters, |
17620
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
682 snd_strerror(err)); |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
683 return 0; |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
684 } |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
685 /* end setting sw-params */ |
dd4db8c43d92
This changes the software parameters to be more compatible with the
cladisch
parents:
17619
diff
changeset
|
686 |
20185 | 687 mp_msg(MSGT_AO,MSGL_V,"alsa: %d Hz/%d channels/%d bpf/%d bytes buffer/%s\n", |
27682
25245b3e962f
Fix "format '%d' expects type 'int', but argument 6 has type 'size_t'" warning.
ranma
parents:
27680
diff
changeset
|
688 ao_data.samplerate, ao_data.channels, (int)bytes_per_sample, ao_data.buffersize, |
12465 | 689 snd_pcm_format_description(alsa_format)); |
690 | |
691 } // end switch alsa_handler (spdif) | |
692 alsa_can_pause = snd_pcm_hw_params_can_pause(alsa_hwparams); | |
26757
0fdf04b07ecb
cosmetics: Remove pointless parentheses from return statements.
diego
parents:
24590
diff
changeset
|
693 return 1; |
12465 | 694 } // end init |
695 | |
696 | |
697 /* close audio device */ | |
698 static void uninit(int immed) | |
699 { | |
700 | |
701 if (alsa_handler) { | |
702 int err; | |
703 | |
14849
d313f591d1a4
aos should respect the immed uninit flag (quit immediatly vs waiting till file
reimar
parents:
14612
diff
changeset
|
704 if (!immed) |
d313f591d1a4
aos should respect the immed uninit flag (quit immediatly vs waiting till file
reimar
parents:
14612
diff
changeset
|
705 snd_pcm_drain(alsa_handler); |
d313f591d1a4
aos should respect the immed uninit flag (quit immediatly vs waiting till file
reimar
parents:
14612
diff
changeset
|
706 |
12465 | 707 if ((err = snd_pcm_close(alsa_handler)) < 0) |
708 { | |
20764 | 709 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_PcmCloseError, snd_strerror(err)); |
12465 | 710 return; |
711 } | |
712 else { | |
713 alsa_handler = NULL; | |
20185 | 714 mp_msg(MSGT_AO,MSGL_V,"alsa-uninit: pcm closed\n"); |
12465 | 715 } |
716 } | |
717 else { | |
20764 | 718 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_NoHandlerDefined); |
12465 | 719 } |
720 } | |
721 | |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
722 static void audio_pause(void) |
12465 | 723 { |
724 int err; | |
725 | |
726 if (alsa_can_pause) { | |
727 if ((err = snd_pcm_pause(alsa_handler, 1)) < 0) | |
728 { | |
20764 | 729 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_PcmPauseError, snd_strerror(err)); |
12465 | 730 return; |
731 } | |
732 mp_msg(MSGT_AO,MSGL_V,"alsa-pause: pause supported by hardware\n"); | |
733 } else { | |
34754 | 734 prepause_space = get_space(); |
12465 | 735 if ((err = snd_pcm_drop(alsa_handler)) < 0) |
736 { | |
20764 | 737 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_PcmDropError, snd_strerror(err)); |
12465 | 738 return; |
739 } | |
740 } | |
741 } | |
742 | |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
743 static void audio_resume(void) |
12465 | 744 { |
745 int err; | |
746 | |
27680
a529f3763afa
Make alsa resume after suspend to disk (would say 'file descriptor is in bad state' before this change)
ranma
parents:
27281
diff
changeset
|
747 if (snd_pcm_state(alsa_handler) == SND_PCM_STATE_SUSPENDED) { |
a529f3763afa
Make alsa resume after suspend to disk (would say 'file descriptor is in bad state' before this change)
ranma
parents:
27281
diff
changeset
|
748 mp_msg(MSGT_AO,MSGL_INFO,MSGTR_AO_ALSA_PcmInSuspendModeTryingResume); |
a529f3763afa
Make alsa resume after suspend to disk (would say 'file descriptor is in bad state' before this change)
ranma
parents:
27281
diff
changeset
|
749 while ((err = snd_pcm_resume(alsa_handler)) == -EAGAIN) sleep(1); |
a529f3763afa
Make alsa resume after suspend to disk (would say 'file descriptor is in bad state' before this change)
ranma
parents:
27281
diff
changeset
|
750 } |
12465 | 751 if (alsa_can_pause) { |
752 if ((err = snd_pcm_pause(alsa_handler, 0)) < 0) | |
753 { | |
20764 | 754 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_PcmResumeError, snd_strerror(err)); |
12465 | 755 return; |
756 } | |
757 mp_msg(MSGT_AO,MSGL_V,"alsa-resume: resume supported by hardware\n"); | |
758 } else { | |
759 if ((err = snd_pcm_prepare(alsa_handler)) < 0) | |
760 { | |
20764 | 761 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_PcmPrepareError, snd_strerror(err)); |
12465 | 762 return; |
763 } | |
34754 | 764 mp_ao_resume_refill(&audio_out_alsa, prepause_space); |
12465 | 765 } |
766 } | |
767 | |
768 /* stop playing and empty buffers (for seeking/pause) */ | |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
769 static void reset(void) |
12465 | 770 { |
771 int err; | |
772 | |
773 if ((err = snd_pcm_drop(alsa_handler)) < 0) | |
774 { | |
20764 | 775 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_PcmPrepareError, snd_strerror(err)); |
12465 | 776 return; |
777 } | |
778 if ((err = snd_pcm_prepare(alsa_handler)) < 0) | |
779 { | |
20764 | 780 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_PcmPrepareError, snd_strerror(err)); |
12465 | 781 return; |
782 } | |
783 return; | |
784 } | |
785 | |
786 /* | |
787 plays 'len' bytes of 'data' | |
788 returns: number of bytes played | |
789 modified last at 29.06.02 by jp | |
790 thanxs for marius <marius@rospot.com> for giving us the light ;) | |
791 */ | |
792 | |
17617
adfab82139c0
After removing play_mmap(), the play() function just unconditionally
cladisch
parents:
17616
diff
changeset
|
793 static int play(void* data, int len, int flags) |
12465 | 794 { |
29706
54f54530a4d3
Make the ao_alsa play function always process a multiple of ao_data.outburst
reimar
parents:
29586
diff
changeset
|
795 int num_frames; |
12465 | 796 snd_pcm_sframes_t res = 0; |
29706
54f54530a4d3
Make the ao_alsa play function always process a multiple of ao_data.outburst
reimar
parents:
29586
diff
changeset
|
797 if (!(flags & AOPLAY_FINAL_CHUNK)) |
54f54530a4d3
Make the ao_alsa play function always process a multiple of ao_data.outburst
reimar
parents:
29586
diff
changeset
|
798 len = len / ao_data.outburst * ao_data.outburst; |
54f54530a4d3
Make the ao_alsa play function always process a multiple of ao_data.outburst
reimar
parents:
29586
diff
changeset
|
799 num_frames = len / bytes_per_sample; |
12465 | 800 |
801 //mp_msg(MSGT_AO,MSGL_ERR,"alsa-play: frames=%i, len=%i\n",num_frames,len); | |
802 | |
803 if (!alsa_handler) { | |
20764 | 804 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_DeviceConfigurationError); |
12465 | 805 return 0; |
806 } | |
807 | |
17621
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
808 if (num_frames == 0) |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
809 return 0; |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
810 |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
811 do { |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
812 res = snd_pcm_writei(alsa_handler, data, num_frames); |
12465 | 813 |
17621
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
814 if (res == -EINTR) { |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
815 /* nothing to do */ |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
816 res = 0; |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
817 } |
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
818 else if (res == -ESTRPIPE) { /* suspend */ |
20764 | 819 mp_msg(MSGT_AO,MSGL_INFO,MSGTR_AO_ALSA_PcmInSuspendModeTryingResume); |
12465 | 820 while ((res = snd_pcm_resume(alsa_handler)) == -EAGAIN) |
821 sleep(1); | |
822 } | |
17621
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
823 if (res < 0) { |
20764 | 824 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_WriteError, snd_strerror(res)); |
825 mp_msg(MSGT_AO,MSGL_INFO,MSGTR_AO_ALSA_TryingToResetSoundcard); | |
12465 | 826 if ((res = snd_pcm_prepare(alsa_handler)) < 0) { |
20764 | 827 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_PcmPrepareError, snd_strerror(res)); |
12465 | 828 break; |
829 } | |
35200 | 830 res = 0; |
12465 | 831 } |
17621
d9c518932302
Fix the error handling in the play() function: add a handler for EINTR,
cladisch
parents:
17620
diff
changeset
|
832 } while (res == 0); |
12465 | 833 |
35200 | 834 return res < 0 ? 0 : res * bytes_per_sample; |
12465 | 835 } |
836 | |
837 /* how many byes are free in the buffer */ | |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
838 static int get_space(void) |
12465 | 839 { |
840 snd_pcm_status_t *status; | |
841 int ret; | |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
842 |
12747 | 843 snd_pcm_status_alloca(&status); |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
844 |
12465 | 845 if ((ret = snd_pcm_status(alsa_handler, status)) < 0) |
846 { | |
20764 | 847 mp_msg(MSGT_AO,MSGL_ERR,MSGTR_AO_ALSA_CannotGetPcmStatus, snd_strerror(ret)); |
26757
0fdf04b07ecb
cosmetics: Remove pointless parentheses from return statements.
diego
parents:
24590
diff
changeset
|
848 return 0; |
12465 | 849 } |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
850 |
17572
580dc69d69bf
Fix get_space(): we don't need to differentiate between the various PCM
cladisch
parents:
17571
diff
changeset
|
851 ret = snd_pcm_status_get_avail(status) * bytes_per_sample; |
24590
2c238fa777ff
ao_alsa: Fix get_space() return values larger than buffersize
uau
parents:
22166
diff
changeset
|
852 if (ret > ao_data.buffersize) // Buffer underrun? |
2c238fa777ff
ao_alsa: Fix get_space() return values larger than buffersize
uau
parents:
22166
diff
changeset
|
853 ret = ao_data.buffersize; |
26757
0fdf04b07ecb
cosmetics: Remove pointless parentheses from return statements.
diego
parents:
24590
diff
changeset
|
854 return ret; |
12465 | 855 } |
856 | |
857 /* delay in seconds between first and last sample in buffer */ | |
17566
f580a7755ac5
Patch by Stefan Huehner / stefan % huehner ! org \
rathann
parents:
17366
diff
changeset
|
858 static float get_delay(void) |
12465 | 859 { |
860 if (alsa_handler) { | |
17573
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
861 snd_pcm_sframes_t delay; |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
862 |
17573
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
863 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
|
864 return 0; |
29263
0f1b5b68af32
whitespace cosmetics: Remove all trailing whitespace.
diego
parents:
28823
diff
changeset
|
865 |
17573
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
866 if (delay < 0) { |
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
867 /* 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
|
868 #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
|
869 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
|
870 #endif |
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
871 delay = 0; |
12465 | 872 } |
17573
8921544f4114
Simplify get_delay(): we don't need to get the complete PCM status but
cladisch
parents:
17572
diff
changeset
|
873 return (float)delay / (float)ao_data.samplerate; |
12465 | 874 } else { |
26757
0fdf04b07ecb
cosmetics: Remove pointless parentheses from return statements.
diego
parents:
24590
diff
changeset
|
875 return 0; |
12465 | 876 } |
877 } |