2 * QEMU Timer based audio emulation
4 * Copyright (c) 2004-2005 Vassili Karpov (malc)
6 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 * of this software and associated documentation files (the "Software"), to deal
8 * in the Software without restriction, including without limitation the rights
9 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10 * copies of the Software, and to permit persons to whom the Software is
11 * furnished to do so, subject to the following conditions:
13 * The above copyright notice and this permission notice shall be included in
14 * all copies or substantial portions of the Software.
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
24 #include "qemu/osdep.h"
25 #include "qemu-common.h"
26 #include "qemu/host-utils.h"
28 #include "qemu/timer.h"
30 #define AUDIO_CAP "noaudio"
31 #include "audio_int.h"
33 typedef struct NoVoiceOut
{
38 typedef struct NoVoiceIn
{
43 static int no_run_out (HWVoiceOut
*hw
, int live
)
45 NoVoiceOut
*no
= (NoVoiceOut
*) hw
;
51 now
= qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL
);
52 ticks
= now
- no
->old_ticks
;
53 bytes
= muldiv64(ticks
, hw
->info
.bytes_per_second
, NANOSECONDS_PER_SECOND
);
54 bytes
= audio_MIN(bytes
, INT_MAX
);
55 samples
= bytes
>> hw
->info
.shift
;
58 decr
= audio_MIN (live
, samples
);
59 hw
->rpos
= (hw
->rpos
+ decr
) % hw
->samples
;
63 static int no_write (SWVoiceOut
*sw
, void *buf
, int len
)
65 return audio_pcm_sw_write(sw
, buf
, len
);
68 static int no_init_out(HWVoiceOut
*hw
, struct audsettings
*as
, void *drv_opaque
)
70 audio_pcm_init_info (&hw
->info
, as
);
75 static void no_fini_out (HWVoiceOut
*hw
)
80 static int no_ctl_out (HWVoiceOut
*hw
, int cmd
, ...)
87 static int no_init_in(HWVoiceIn
*hw
, struct audsettings
*as
, void *drv_opaque
)
89 audio_pcm_init_info (&hw
->info
, as
);
94 static void no_fini_in (HWVoiceIn
*hw
)
99 static int no_run_in (HWVoiceIn
*hw
)
101 NoVoiceIn
*no
= (NoVoiceIn
*) hw
;
102 int live
= audio_pcm_hw_get_live_in (hw
);
103 int dead
= hw
->samples
- live
;
107 int64_t now
= qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL
);
108 int64_t ticks
= now
- no
->old_ticks
;
110 muldiv64(ticks
, hw
->info
.bytes_per_second
, NANOSECONDS_PER_SECOND
);
113 bytes
= audio_MIN (bytes
, INT_MAX
);
114 samples
= bytes
>> hw
->info
.shift
;
115 samples
= audio_MIN (samples
, dead
);
120 static int no_read (SWVoiceIn
*sw
, void *buf
, int size
)
122 /* use custom code here instead of audio_pcm_sw_read() to avoid
123 * useless resampling/mixing */
124 int samples
= size
>> sw
->info
.shift
;
125 int total
= sw
->hw
->total_samples_captured
- sw
->total_hw_samples_acquired
;
126 int to_clear
= audio_MIN (samples
, total
);
127 sw
->total_hw_samples_acquired
+= total
;
128 audio_pcm_info_clear_buf (&sw
->info
, buf
, to_clear
);
129 return to_clear
<< sw
->info
.shift
;
132 static int no_ctl_in (HWVoiceIn
*hw
, int cmd
, ...)
139 static void *no_audio_init (void)
141 return &no_audio_init
;
144 static void no_audio_fini (void *opaque
)
149 static struct audio_pcm_ops no_pcm_ops
= {
150 .init_out
= no_init_out
,
151 .fini_out
= no_fini_out
,
152 .run_out
= no_run_out
,
154 .ctl_out
= no_ctl_out
,
156 .init_in
= no_init_in
,
157 .fini_in
= no_fini_in
,
163 struct audio_driver no_audio_driver
= {
165 .descr
= "Timer based audio emulation",
167 .init
= no_audio_init
,
168 .fini
= no_audio_fini
,
169 .pcm_ops
= &no_pcm_ops
,
171 .max_voices_out
= INT_MAX
,
172 .max_voices_in
= INT_MAX
,
173 .voice_size_out
= sizeof (NoVoiceOut
),
174 .voice_size_in
= sizeof (NoVoiceIn
)