Mercurial > mplayer.hg
annotate libao2/ao_alsa9.c @ 1306:7ce37211e454
yuv2rgb_mmx crashes with ffdivx codec, when we play back avi files that have
a frame width that is not an exact multiple of 8.
Testcase: 405.avi (356x240). Playing on an MMX capable x86 system using the
x11 video-out driver results in a segfault.
The MMX routines convert image data in quantities of 8 pixels in each loop,
and the inner loop was not terminated in case there are only 1-7 pixels left,
producing too much RGB output.
For now, just ignore the last few pixels on each row, to avoid the segfaults.
(Gives a black vertical border on the right, if you play a video with
width%8 != 0) A possible future enhancement would be, to add a second loop
to convert the last width%8 pixels to RGB using a byte loop.
author | jkeil |
---|---|
date | Thu, 12 Jul 2001 15:23:26 +0000 |
parents | bf18175dfad7 |
children | 3c390543302c |
rev | line source |
---|---|
1050 | 1 /* |
2 ao_alsa9 - ALSA-0.9.x output plugin for MPlayer | |
3 | |
4 (C) Alex Beregszaszi <alex@naxine.org> | |
1180 | 5 |
6 Don't use this buggy driver, ALSA-0.9.x emulates OSS very well... | |
1050 | 7 */ |
8 | |
9 #include <errno.h> | |
10 #include <sys/asoundlib.h> | |
11 | |
12 #include "../config.h" | |
13 | |
14 #include "audio_out.h" | |
15 #include "audio_out_internal.h" | |
1058 | 16 #include "afmt.h" |
1050 | 17 |
18 extern int verbose; | |
19 | |
20 static ao_info_t info = | |
21 { | |
22 "ALSA-0.9.x audio output", | |
23 "alsa9", | |
24 "Alex Beregszaszi <alex@naxine.org>", | |
25 "under developement" | |
26 }; | |
27 | |
28 LIBAO_EXTERN(alsa9) | |
29 | |
30 /* global variables: | |
31 ao_samplerate | |
32 ao_channels | |
33 ao_format | |
34 ao_bps | |
35 ao_outburst | |
36 ao_buffersize | |
37 */ | |
38 | |
39 static snd_pcm_t *alsa_handler; | |
40 static snd_pcm_format_t alsa_format; | |
41 static snd_pcm_hw_params_t *alsa_hwparams; | |
42 static snd_pcm_sw_params_t *alsa_swparams; | |
43 static char *alsa_device; | |
44 #define ALSA_DEVICE_SIZE 48 | |
45 | |
1128 | 46 static int alsa_fragsize = 8192; /* 4096 */ |
47 static int alsa_fragcount = 8; | |
48 | |
1050 | 49 /* to set/get/query special features/parameters */ |
50 static int control(int cmd, int arg) | |
51 { | |
52 switch(cmd) | |
53 { | |
54 case AOCONTROL_GET_DEVICE: | |
1146
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
55 return((char *)alsa_device); /* egy kicsit brutalis, dehat :) */ |
1050 | 56 case AOCONTROL_SET_DEVICE: |
1146
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
57 { |
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
58 int ret; |
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
59 |
1050 | 60 strncpy(alsa_device, (char *)arg, ALSA_DEVICE_SIZE); |
1146
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
61 uninit(); |
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
62 ret = init(ao_samplerate, ao_channels, ao_format, 0); |
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
63 if (ret == 0) |
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
64 return(CONTROL_ERROR); |
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
65 else |
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
66 return(CONTROL_OK); |
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
67 } |
1050 | 68 } |
69 return(CONTROL_UNKNOWN); | |
70 } | |
71 | |
1128 | 72 #undef start |
73 #define buffersize | |
74 #undef buffertime | |
75 #define set_period | |
76 #undef sw_params | |
77 #undef set_start_mode | |
78 | |
1050 | 79 /* |
80 open & setup audio device | |
81 return: 1=success 0=fail | |
82 */ | |
83 static int init(int rate_hz, int channels, int format, int flags) | |
84 { | |
85 int err; | |
86 int cards = -1; | |
87 snd_pcm_info_t *alsa_info; | |
1180 | 88 |
89 printf("alsa-init: Don't use this buggy driver, ALSA-0.9.x emulates OSS very well...\n"); | |
1050 | 90 |
91 printf("alsa-init: requested format: %d Hz, %d channels, %s\n", rate_hz, | |
92 channels, audio_out_format_name(format)); | |
93 | |
94 alsa_handler = NULL; | |
95 | |
96 if (verbose) | |
97 printf("alsa-init: compiled for ALSA-%s (%d)\n", SND_LIB_VERSION_STR, | |
98 SND_LIB_VERSION); | |
99 | |
100 if ((err = snd_card_next(&cards)) < 0 || cards < 0) | |
101 { | |
102 printf("alsa-init: no soundcards found: %s\n", snd_strerror(err)); | |
103 return(0); | |
104 } | |
105 | |
1128 | 106 ao_samplerate = rate_hz; |
107 ao_bps = channels; /* really this is bytes per frame so bad varname */ | |
1050 | 108 ao_format = format; |
1128 | 109 ao_channels = channels; |
1050 | 110 ao_outburst = OUTBURST; |
111 ao_buffersize = 16384; | |
112 | |
113 switch (format) | |
114 { | |
115 case AFMT_S8: | |
116 alsa_format = SND_PCM_FORMAT_S8; | |
117 break; | |
118 case AFMT_U8: | |
119 alsa_format = SND_PCM_FORMAT_U8; | |
120 break; | |
121 case AFMT_U16_LE: | |
122 alsa_format = SND_PCM_FORMAT_U16_LE; | |
123 break; | |
124 case AFMT_U16_BE: | |
125 alsa_format = SND_PCM_FORMAT_U16_BE; | |
126 break; | |
127 case AFMT_S16_LE: | |
128 alsa_format = SND_PCM_FORMAT_S16_LE; | |
129 break; | |
130 case AFMT_S16_BE: | |
131 alsa_format = SND_PCM_FORMAT_S16_BE; | |
132 break; | |
133 default: | |
134 alsa_format = SND_PCM_FORMAT_MPEG; | |
135 break; | |
136 } | |
137 | |
138 switch(alsa_format) | |
139 { | |
140 case SND_PCM_FORMAT_S16_LE: | |
141 case SND_PCM_FORMAT_U16_LE: | |
142 ao_bps *= 2; | |
143 break; | |
144 case -1: | |
145 printf("alsa-init: invalid format (%s) requested - output disabled\n", | |
146 audio_out_format_name(format)); | |
147 return(0); | |
1115
a16b569f2702
-Wall style cleanups, TEST IT, it can be working by others
al3x
parents:
1058
diff
changeset
|
148 default: |
a16b569f2702
-Wall style cleanups, TEST IT, it can be working by others
al3x
parents:
1058
diff
changeset
|
149 break; |
1050 | 150 } |
1128 | 151 |
1050 | 152 if ((err = snd_pcm_info_malloc(&alsa_info)) < 0) |
153 { | |
154 printf("alsa-init: memory allocation error: %s\n", snd_strerror(err)); | |
155 return(0); | |
156 } | |
157 | |
1207 | 158 if (ao_subdevice != NULL) |
159 alsa_device = ao_subdevice; | |
160 | |
1146
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
161 if (alsa_device == NULL) |
1050 | 162 { |
1146
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
163 if ((alsa_device = malloc(ALSA_DEVICE_SIZE)) == NULL) |
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
164 { |
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
165 printf("alsa-init: memory allocation error: %s\n", strerror(errno)); |
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
166 return(0); |
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
167 } |
1050 | 168 |
1146
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
169 snprintf(alsa_device, ALSA_DEVICE_SIZE, "hw:%d,%d", |
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
170 snd_pcm_info_get_device(alsa_info), |
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
171 snd_pcm_info_get_subdevice(alsa_info)); |
1050 | 172 |
1146
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
173 snd_pcm_info_free(alsa_info); |
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
174 } |
1050 | 175 |
176 printf("alsa-init: %d soundcard%s found, using: %s\n", cards+1, | |
177 (cards >= 0) ? "" : "s", alsa_device); | |
178 | |
179 if ((err = snd_pcm_open(&alsa_handler, alsa_device, SND_PCM_STREAM_PLAYBACK, | |
180 0)) < 0) | |
181 { | |
182 printf("alsa-init: playback open error: %s\n", snd_strerror(err)); | |
183 return(0); | |
184 } | |
185 | |
186 snd_pcm_hw_params_malloc(&alsa_hwparams); | |
187 snd_pcm_sw_params_malloc(&alsa_swparams); | |
188 | |
189 if ((err = snd_pcm_hw_params_any(alsa_handler, alsa_hwparams)) < 0) | |
190 { | |
191 printf("alsa-init: unable to get initial parameters: %s\n", | |
192 snd_strerror(err)); | |
193 return(0); | |
194 } | |
195 | |
196 if ((err = snd_pcm_hw_params_set_access(alsa_handler, alsa_hwparams, | |
197 SND_PCM_ACCESS_RW_INTERLEAVED)) < 0) | |
198 { | |
199 printf("alsa-init: unable to set access type: %s\n", | |
200 snd_strerror(err)); | |
201 return(0); | |
202 } | |
203 | |
204 if ((err = snd_pcm_hw_params_set_format(alsa_handler, alsa_hwparams, | |
205 alsa_format)) < 0) | |
206 { | |
207 printf("alsa-init: unable to set format: %s\n", | |
208 snd_strerror(err)); | |
209 return(0); | |
210 } | |
211 | |
212 if ((err = snd_pcm_hw_params_set_channels(alsa_handler, alsa_hwparams, | |
213 ao_channels)) < 0) | |
214 { | |
215 printf("alsa-init: unable to set channels: %s\n", | |
216 snd_strerror(err)); | |
217 return(0); | |
218 } | |
219 | |
1129 | 220 if ((err = snd_pcm_hw_params_set_rate(alsa_handler, alsa_hwparams, |
1050 | 221 ao_samplerate, 0)) < 0) |
222 { | |
1128 | 223 printf("alsa-init: unable to set samplerate: %s\n", |
1050 | 224 snd_strerror(err)); |
225 return(0); | |
226 } | |
227 | |
1128 | 228 #ifdef set_period |
1115
a16b569f2702
-Wall style cleanups, TEST IT, it can be working by others
al3x
parents:
1058
diff
changeset
|
229 { |
1128 | 230 if ((err = snd_pcm_hw_params_set_period_size(alsa_handler, alsa_hwparams, alsa_fragsize / 4, 0)) < 0) |
231 { | |
232 printf("alsa-init: unable to set periodsize: %s\n", | |
233 snd_strerror(err)); | |
234 return(0); | |
235 } | |
236 if ((err = snd_pcm_hw_params_set_periods(alsa_handler, alsa_hwparams, alsa_fragcount, 0)) < 0) | |
237 { | |
238 printf("alsa-init: unable to set periods: %s\n", | |
239 snd_strerror(err)); | |
240 return(0); | |
241 } | |
1115
a16b569f2702
-Wall style cleanups, TEST IT, it can be working by others
al3x
parents:
1058
diff
changeset
|
242 } |
1128 | 243 #endif |
1050 | 244 #ifdef buffersize |
245 if ((err = snd_pcm_hw_params_get_buffer_size(alsa_hwparams)) < 0) | |
246 { | |
247 printf("alsa-init: unable to get buffer size: %s\n", | |
248 snd_strerror(err)); | |
249 return(0); | |
250 } else | |
1128 | 251 { |
1050 | 252 ao_buffersize = err; |
1146
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
253 if (verbose) |
a84610bb5476
hacked aocontrol_set_device working, and cleaned up a bit
al3x
parents:
1129
diff
changeset
|
254 printf("alsa-init: got buffersize %i\n", ao_buffersize); |
1128 | 255 } |
1050 | 256 #endif |
257 | |
258 #ifdef buffertime | |
259 { | |
260 int alsa_buffer_time = 60; | |
261 | |
262 if ((err = snd_pcm_hw_params_set_buffer_time_near(alsa_handler, alsa_hwparams, | |
263 alsa_buffer_time, 0)) < 0) | |
264 { | |
265 printf("alsa-init: unable to set buffer time near: %s\n", | |
266 snd_strerror(err)); | |
267 return(0); | |
268 } else | |
269 alsa_buffer_time = err; | |
270 | |
271 if ((err = snd_pcm_hw_params_set_period_time_near(alsa_handler, alsa_hwparams, | |
272 alsa_buffer_time/ao_bps, 0)) < 0) | |
273 { | |
274 printf("alsa-init: unable to set period time: %s\n", | |
275 snd_strerror(err)); | |
276 return(0); | |
277 } | |
1128 | 278 printf("alsa-init: buffer_time: %d, period_time :%d\n", |
279 alsa_buffer_time, err); | |
1050 | 280 } |
281 #endif | |
282 | |
283 if ((err = snd_pcm_hw_params(alsa_handler, alsa_hwparams)) < 0) | |
284 { | |
285 printf("alsa-init: unable to set parameters: %s\n", | |
286 snd_strerror(err)); | |
287 return(0); | |
288 } | |
1128 | 289 |
290 #ifdef sw_params | |
1050 | 291 if ((err = snd_pcm_sw_params_current(alsa_handler, alsa_swparams)) < 0) |
292 { | |
293 printf("alsa-init: unable to get parameters: %s\n", | |
294 snd_strerror(err)); | |
295 return(0); | |
296 } | |
297 | |
1128 | 298 #ifdef set_start_mode |
1050 | 299 if ((err = snd_pcm_sw_params_set_start_mode(alsa_handler, alsa_swparams, |
300 SND_PCM_START_DATA)) < 0) | |
301 { | |
302 printf("alsa-init: unable to set start mode: %s\n", | |
303 snd_strerror(err)); | |
304 return(0); | |
305 } | |
1128 | 306 #endif |
1050 | 307 |
308 if ((err = snd_pcm_sw_params(alsa_handler, alsa_swparams)) < 0) | |
309 { | |
310 printf("alsa-init: unable to set parameters: %s\n", | |
311 snd_strerror(err)); | |
312 return(0); | |
313 } | |
314 | |
1128 | 315 // snd_pcm_sw_params_default(alsa_handler, alsa_swparams); |
316 #endif | |
1050 | 317 if ((err = snd_pcm_prepare(alsa_handler)) < 0) |
318 { | |
319 printf("alsa-init: pcm prepare error: %s\n", snd_strerror(err)); | |
320 return(0); | |
321 } | |
322 | |
1115
a16b569f2702
-Wall style cleanups, TEST IT, it can be working by others
al3x
parents:
1058
diff
changeset
|
323 #ifdef start |
1050 | 324 if ((err = snd_pcm_start(alsa_handler)) < 0) |
325 { | |
326 printf("alsa-init: pcm start error: %s\n", snd_strerror(err)); | |
327 if (err != -EPIPE) | |
328 return(0); | |
329 if ((err = snd_pcm_start(alsa_handler)) < 0) | |
330 { | |
331 printf("alsa-init: pcm start error: %s\n", snd_strerror(err)); | |
332 return(0); | |
333 } | |
334 } | |
1115
a16b569f2702
-Wall style cleanups, TEST IT, it can be working by others
al3x
parents:
1058
diff
changeset
|
335 #endif |
1128 | 336 printf("AUDIO: %d Hz/%d channels/%d bpf/%d bytes buffer/%s\n", |
337 ao_samplerate, ao_channels, ao_bps, ao_buffersize, | |
1050 | 338 snd_pcm_format_description(alsa_format)); |
339 return(1); | |
340 } | |
341 | |
342 /* close audio device */ | |
343 static void uninit() | |
344 { | |
345 int err; | |
346 | |
347 if (alsa_device != NULL) | |
348 free(alsa_device); | |
349 | |
350 snd_pcm_hw_params_free(alsa_hwparams); | |
351 snd_pcm_sw_params_free(alsa_swparams); | |
352 | |
353 if ((err = snd_pcm_drain(alsa_handler)) < 0) | |
354 { | |
355 printf("alsa-uninit: pcm drain error: %s\n", snd_strerror(err)); | |
356 return; | |
357 } | |
358 | |
1129 | 359 #ifdef start |
1050 | 360 if ((err = snd_pcm_reset(alsa_handler)) < 0) |
361 { | |
362 printf("alsa-uninit: pcm reset error: %s\n", snd_strerror(err)); | |
363 return; | |
364 } | |
1129 | 365 #endif |
1050 | 366 |
367 if ((err = snd_pcm_close(alsa_handler)) < 0) | |
368 { | |
369 printf("alsa-uninit: pcm close error: %s\n", snd_strerror(err)); | |
370 return; | |
371 } | |
372 } | |
373 | |
374 static void audio_pause() | |
375 { | |
376 int err; | |
377 | |
378 if ((err = snd_pcm_drain(alsa_handler)) < 0) | |
379 { | |
380 printf("alsa-pause: pcm drain error: %s\n", snd_strerror(err)); | |
381 return; | |
382 } | |
383 | |
1129 | 384 #ifdef reset |
1050 | 385 if ((err = snd_pcm_reset(alsa_handler)) < 0) |
386 { | |
387 printf("alsa-pause: pcm reset error: %s\n", snd_strerror(err)); | |
388 return; | |
389 } | |
1129 | 390 #endif |
1050 | 391 } |
392 | |
393 static void audio_resume() | |
394 { | |
395 int err; | |
396 | |
397 if ((err = snd_pcm_prepare(alsa_handler)) < 0) | |
398 { | |
399 printf("alsa-resume: pcm prepare error: %s\n", snd_strerror(err)); | |
400 return; | |
401 } | |
402 | |
1129 | 403 #ifdef start |
1050 | 404 if ((err = snd_pcm_start(alsa_handler)) < 0) |
405 { | |
406 printf("alsa-resume: pcm start error: %s\n", snd_strerror(err)); | |
407 return; | |
408 } | |
1129 | 409 #endif |
1050 | 410 } |
411 | |
412 /* stop playing and empty buffers (for seeking/pause) */ | |
413 static void reset() | |
414 { | |
415 int err; | |
416 | |
417 if ((err = snd_pcm_drain(alsa_handler)) < 0) | |
418 { | |
419 printf("alsa-reset: pcm drain error: %s\n", snd_strerror(err)); | |
420 return; | |
421 } | |
422 | |
1129 | 423 #ifdef start |
1050 | 424 if ((err = snd_pcm_reset(alsa_handler)) < 0) |
425 { | |
426 printf("alsa-reset: pcm reset error: %s\n", snd_strerror(err)); | |
427 return; | |
428 } | |
1129 | 429 #endif |
1050 | 430 |
431 if ((err = snd_pcm_prepare(alsa_handler)) < 0) | |
432 { | |
433 printf("alsa-reset: pcm prepare error: %s\n", snd_strerror(err)); | |
434 return; | |
435 } | |
436 | |
1129 | 437 #ifdef start |
1050 | 438 if ((err = snd_pcm_start(alsa_handler)) < 0) |
439 { | |
440 printf("alsa-reset: pcm start error: %s\n", snd_strerror(err)); | |
441 return; | |
442 } | |
1129 | 443 #endif |
1050 | 444 } |
445 | |
446 /* | |
447 plays 'len' bytes of 'data' | |
448 returns: number of bytes played | |
449 */ | |
450 static int play(void* data, int len, int flags) | |
451 { | |
1115
a16b569f2702
-Wall style cleanups, TEST IT, it can be working by others
al3x
parents:
1058
diff
changeset
|
452 int got_len; |
a16b569f2702
-Wall style cleanups, TEST IT, it can be working by others
al3x
parents:
1058
diff
changeset
|
453 |
1128 | 454 if ((got_len = snd_pcm_writei(alsa_handler, data, (len/ao_bps))) != (len/ao_bps)) |
1050 | 455 { |
1115
a16b569f2702
-Wall style cleanups, TEST IT, it can be working by others
al3x
parents:
1058
diff
changeset
|
456 if (got_len == -EPIPE) /* underrun? */ |
1050 | 457 { |
458 printf("alsa-play: alsa underrun, resetting stream\n"); | |
1128 | 459 if ((got_len = snd_pcm_prepare(alsa_handler)) < 0) |
1050 | 460 { |
1128 | 461 printf("alsa-play: playback prepare error: %s\n", snd_strerror(got_len)); |
1050 | 462 return(0); |
463 } | |
1128 | 464 if ((got_len = snd_pcm_writei(alsa_handler, data, (len/ao_bps))) != (len/ao_bps)) |
1050 | 465 { |
466 printf("alsa-play: write error after reset: %s - giving up\n", | |
1128 | 467 snd_strerror(got_len)); |
1050 | 468 return(0); |
469 } | |
470 return(len); /* 2nd write was ok */ | |
471 } | |
472 } | |
473 return(len); | |
474 } | |
475 | |
476 /* how many byes are free in the buffer */ | |
477 static int get_space() | |
478 { | |
479 snd_pcm_status_t *status; | |
480 int ret; | |
481 | |
482 if ((ret = snd_pcm_status_malloc(&status)) < 0) | |
483 { | |
484 printf("alsa-space: memory allocation error: %s\n", snd_strerror(ret)); | |
485 return(0); | |
486 } | |
487 | |
488 if ((ret = snd_pcm_status(alsa_handler, status)) < 0) | |
489 { | |
490 printf("alsa-space: cannot get pcm status: %s\n", snd_strerror(ret)); | |
491 return(0); | |
492 } | |
493 | |
494 switch(snd_pcm_status_get_state(status)) | |
495 { | |
496 case SND_PCM_STATE_OPEN: | |
497 case SND_PCM_STATE_PREPARED: | |
498 case SND_PCM_STATE_RUNNING: | |
499 ret = snd_pcm_status_get_avail(status) * ao_bps; | |
500 break; | |
501 default: | |
502 ret = 0; | |
503 } | |
504 | |
505 snd_pcm_status_free(status); | |
1129 | 506 |
507 if (ret < 0) | |
508 ret = 0; | |
1050 | 509 return(ret); |
510 } | |
511 | |
512 /* how many unplayed bytes are in the buffer */ | |
513 static int get_delay() | |
514 { | |
515 snd_pcm_status_t *status; | |
516 int ret; | |
517 | |
518 if ((ret = snd_pcm_status_malloc(&status)) < 0) | |
519 { | |
520 printf("alsa-delay: memory allocation error: %s\n", snd_strerror(ret)); | |
521 return(0); | |
522 } | |
523 | |
524 if ((ret = snd_pcm_status(alsa_handler, status)) < 0) | |
525 { | |
526 printf("alsa-delay: cannot get pcm status: %s\n", snd_strerror(ret)); | |
527 return(0); | |
528 } | |
529 | |
530 switch(snd_pcm_status_get_state(status)) | |
531 { | |
532 case SND_PCM_STATE_OPEN: | |
533 case SND_PCM_STATE_PREPARED: | |
534 case SND_PCM_STATE_RUNNING: | |
535 ret = snd_pcm_status_get_delay(status) * ao_bps; | |
536 break; | |
537 default: | |
538 ret = 0; | |
539 } | |
540 | |
541 snd_pcm_status_free(status); | |
1129 | 542 |
543 if (ret < 0) | |
544 ret = 0; | |
1050 | 545 return(ret); |
546 } |