2 * QEMU PC speaker emulation
4 * Copyright (c) 2006 Joachim Henke
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
25 #include "qemu/osdep.h"
26 #include "hw/isa/isa.h"
27 #include "hw/audio/soundhw.h"
28 #include "audio/audio.h"
29 #include "qemu/module.h"
30 #include "qemu/timer.h"
31 #include "qemu/error-report.h"
32 #include "hw/timer/i8254.h"
33 #include "migration/vmstate.h"
34 #include "hw/audio/pcspk.h"
35 #include "qapi/error.h"
37 #define PCSPK_BUF_LEN 1792
38 #define PCSPK_SAMPLE_RATE 32000
39 #define PCSPK_MAX_FREQ (PCSPK_SAMPLE_RATE >> 1)
40 #define PCSPK_MIN_COUNT DIV_ROUND_UP(PIT_FREQ, PCSPK_MAX_FREQ)
42 #define PC_SPEAKER(obj) OBJECT_CHECK(PCSpkState, (obj), TYPE_PC_SPEAKER)
49 uint8_t sample_buf
[PCSPK_BUF_LEN
];
53 unsigned int pit_count
;
55 unsigned int play_pos
;
57 uint8_t dummy_refresh_clock
;
61 static const char *s_spk
= "pcspk";
62 static PCSpkState
*pcspk_state
;
64 static inline void generate_samples(PCSpkState
*s
)
69 const uint32_t m
= PCSPK_SAMPLE_RATE
* s
->pit_count
;
70 const uint32_t n
= ((uint64_t)PIT_FREQ
<< 32) / m
;
72 /* multiple of wavelength for gapless looping */
73 s
->samples
= (QEMU_ALIGN_DOWN(PCSPK_BUF_LEN
* PIT_FREQ
, m
) / (PIT_FREQ
>> 1) + 1) >> 1;
74 for (i
= 0; i
< s
->samples
; ++i
)
75 s
->sample_buf
[i
] = (64 & (n
* i
>> 25)) - 32;
77 s
->samples
= PCSPK_BUF_LEN
;
78 for (i
= 0; i
< PCSPK_BUF_LEN
; ++i
)
79 s
->sample_buf
[i
] = 128; /* silence */
83 static void pcspk_callback(void *opaque
, int free
)
85 PCSpkState
*s
= opaque
;
89 pit_get_channel_info(s
->pit
, 2, &ch
);
96 /* avoid frequencies that are not reproducible with sample rate */
97 if (n
< PCSPK_MIN_COUNT
)
100 if (s
->pit_count
!= n
) {
107 n
= MIN(s
->samples
- s
->play_pos
, (unsigned int)free
);
108 n
= AUD_write(s
->voice
, &s
->sample_buf
[s
->play_pos
], n
);
111 s
->play_pos
= (s
->play_pos
+ n
) % s
->samples
;
116 static int pcspk_audio_init(PCSpkState
*s
)
118 struct audsettings as
= {PCSPK_SAMPLE_RATE
, 1, AUDIO_FORMAT_U8
, 0};
121 /* already initialized */
125 AUD_register_card(s_spk
, &s
->card
);
127 s
->voice
= AUD_open_out(&s
->card
, s
->voice
, s_spk
, s
, pcspk_callback
, &as
);
129 AUD_log(s_spk
, "Could not open voice\n");
136 static uint64_t pcspk_io_read(void *opaque
, hwaddr addr
,
139 PCSpkState
*s
= opaque
;
142 pit_get_channel_info(s
->pit
, 2, &ch
);
144 s
->dummy_refresh_clock
^= (1 << 4);
146 return ch
.gate
| (s
->data_on
<< 1) | s
->dummy_refresh_clock
|
150 static void pcspk_io_write(void *opaque
, hwaddr addr
, uint64_t val
,
153 PCSpkState
*s
= opaque
;
154 const int gate
= val
& 1;
156 s
->data_on
= (val
>> 1) & 1;
157 pit_set_gate(s
->pit
, 2, gate
);
159 if (gate
) /* restart */
161 AUD_set_active_out(s
->voice
, gate
& s
->data_on
);
165 static const MemoryRegionOps pcspk_io_ops
= {
166 .read
= pcspk_io_read
,
167 .write
= pcspk_io_write
,
169 .min_access_size
= 1,
170 .max_access_size
= 1,
174 static void pcspk_initfn(Object
*obj
)
176 PCSpkState
*s
= PC_SPEAKER(obj
);
178 memory_region_init_io(&s
->ioport
, OBJECT(s
), &pcspk_io_ops
, s
, "pcspk", 1);
180 object_property_add_link(obj
, "pit", TYPE_PIT_COMMON
,
182 qdev_prop_allow_set_link_before_realize
,
186 static void pcspk_realizefn(DeviceState
*dev
, Error
**errp
)
188 ISADevice
*isadev
= ISA_DEVICE(dev
);
189 PCSpkState
*s
= PC_SPEAKER(dev
);
191 isa_register_ioport(isadev
, &s
->ioport
, s
->iobase
);
200 static bool migrate_needed(void *opaque
)
202 PCSpkState
*s
= opaque
;
207 static const VMStateDescription vmstate_spk
= {
210 .minimum_version_id
= 1,
211 .minimum_version_id_old
= 1,
212 .needed
= migrate_needed
,
213 .fields
= (VMStateField
[]) {
214 VMSTATE_UINT8(data_on
, PCSpkState
),
215 VMSTATE_UINT8(dummy_refresh_clock
, PCSpkState
),
216 VMSTATE_END_OF_LIST()
220 static Property pcspk_properties
[] = {
221 DEFINE_AUDIO_PROPERTIES(PCSpkState
, card
),
222 DEFINE_PROP_UINT32("iobase", PCSpkState
, iobase
, 0x61),
223 DEFINE_PROP_BOOL("migrate", PCSpkState
, migrate
, true),
224 DEFINE_PROP_END_OF_LIST(),
227 static void pcspk_class_initfn(ObjectClass
*klass
, void *data
)
229 DeviceClass
*dc
= DEVICE_CLASS(klass
);
231 dc
->realize
= pcspk_realizefn
;
232 set_bit(DEVICE_CATEGORY_SOUND
, dc
->categories
);
233 dc
->vmsd
= &vmstate_spk
;
234 device_class_set_props(dc
, pcspk_properties
);
235 /* Reason: realize sets global pcspk_state */
236 /* Reason: pit object link */
237 dc
->user_creatable
= false;
240 static const TypeInfo pcspk_info
= {
241 .name
= TYPE_PC_SPEAKER
,
242 .parent
= TYPE_ISA_DEVICE
,
243 .instance_size
= sizeof(PCSpkState
),
244 .instance_init
= pcspk_initfn
,
245 .class_init
= pcspk_class_initfn
,
248 static int pcspk_audio_init_soundhw(ISABus
*bus
)
250 PCSpkState
*s
= pcspk_state
;
252 warn_report("'-soundhw pcspk' is deprecated, "
253 "please set a backend using '-machine pcspk-audiodev=<name>' instead");
254 return pcspk_audio_init(s
);
257 static void pcspk_register(void)
259 type_register_static(&pcspk_info
);
260 isa_register_soundhw("pcspk", "PC speaker", pcspk_audio_init_soundhw
);
262 type_init(pcspk_register
)