VirtualBox

source: vbox/trunk/src/VBox/Devices/Audio/dsound_template.h@ 217

Last change on this file since 217 was 1, checked in by vboxsync, 55 years ago

import

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 6.9 KB
Line 
1/*
2 * QEMU DirectSound audio driver header
3 *
4 * Copyright (c) 2005 Vassili Karpov (malc)
5 *
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:
12 *
13 * The above copyright notice and this permission notice shall be included in
14 * all copies or substantial portions of the Software.
15 *
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
22 * THE SOFTWARE.
23 */
24#ifdef DSBTYPE_IN
25#define NAME "capture buffer"
26#define TYPE in
27#define IFACE IDirectSoundCaptureBuffer
28#define BUFPTR LPDIRECTSOUNDCAPTUREBUFFER
29#define FIELD dsound_capture_buffer
30#else
31#define NAME "playback buffer"
32#define TYPE out
33#define IFACE IDirectSoundBuffer
34#define BUFPTR LPDIRECTSOUNDBUFFER
35#define FIELD dsound_buffer
36#endif
37
38static int glue (dsound_unlock_, TYPE) (
39 BUFPTR buf,
40 LPVOID p1,
41 LPVOID p2,
42 DWORD blen1,
43 DWORD blen2
44 )
45{
46 HRESULT hr;
47
48 hr = glue (IFACE, _Unlock) (buf, p1, blen1, p2, blen2);
49 if (FAILED (hr)) {
50 dsound_logerr (hr, "Could not unlock " NAME "\n");
51 return -1;
52 }
53
54 return 0;
55}
56
57static int glue (dsound_lock_, TYPE) (
58 BUFPTR buf,
59 struct audio_pcm_info *info,
60 DWORD pos,
61 DWORD len,
62 LPVOID *p1p,
63 LPVOID *p2p,
64 DWORD *blen1p,
65 DWORD *blen2p,
66 int entire
67 )
68{
69 HRESULT hr;
70 int i;
71 LPVOID p1 = NULL, p2 = NULL;
72 DWORD blen1 = 0, blen2 = 0;
73 DWORD flag;
74
75#ifdef DSBTYPE_IN
76 flag = entire ? DSCBLOCK_ENTIREBUFFER : 0;
77#else
78 flag = entire ? DSBLOCK_ENTIREBUFFER : 0;
79#endif
80
81 for (i = 0; i < conf.lock_retries; ++i) {
82 hr = glue (IFACE, _Lock) (
83 buf,
84 pos,
85 len,
86 &p1,
87 &blen1,
88 &p2,
89 &blen2,
90 flag
91 );
92
93 if (FAILED (hr)) {
94#ifndef DSBTYPE_IN
95 if (hr == DSERR_BUFFERLOST) {
96 if (glue (dsound_restore_, TYPE) (buf)) {
97 dsound_logerr (hr, "Could not lock " NAME "\n");
98 goto fail;
99 }
100 continue;
101 }
102#endif
103 dsound_logerr (hr, "Could not lock " NAME "\n");
104 goto fail;
105 }
106
107 break;
108 }
109
110 if (i == conf.lock_retries) {
111 dolog ("%d attempts to lock " NAME " failed\n", i);
112 goto fail;
113 }
114
115 if ((p1 && (blen1 & info->align)) || (p2 && (blen2 & info->align))) {
116 dolog ("DirectSound returned misaligned buffer %ld %ld\n",
117 blen1, blen2);
118 glue (dsound_unlock_, TYPE) (buf, p1, p2, blen1, blen2);
119 goto fail;
120 }
121
122 if (!p1 && blen1) {
123 dolog ("warning: !p1 && blen1=%ld\n", blen1);
124 blen1 = 0;
125 }
126
127 if (!p2 && blen2) {
128 dolog ("warning: !p2 && blen2=%ld\n", blen2);
129 blen2 = 0;
130 }
131
132 *p1p = p1;
133 *p2p = p2;
134 *blen1p = blen1;
135 *blen2p = blen2;
136 return 0;
137
138 fail:
139 *p1p = NULL;
140 *p2p = NULL;
141 *blen1p = -1;
142 *blen2p = -1;
143 return -1;
144}
145
146#ifdef DSBTYPE_IN
147static void dsound_fini_in (HWVoiceIn *hw)
148#else
149static void dsound_fini_out (HWVoiceOut *hw)
150#endif
151{
152 HRESULT hr;
153#ifdef DSBTYPE_IN
154 DSoundVoiceIn *ds = (DSoundVoiceIn *) hw;
155#else
156 DSoundVoiceOut *ds = (DSoundVoiceOut *) hw;
157#endif
158
159 if (ds->FIELD) {
160 hr = glue (IFACE, _Stop) (ds->FIELD);
161 if (FAILED (hr)) {
162 dsound_logerr (hr, "Could not stop " NAME "\n");
163 }
164
165 hr = glue (IFACE, _Release) (ds->FIELD);
166 if (FAILED (hr)) {
167 dsound_logerr (hr, "Could not release " NAME "\n");
168 }
169 ds->FIELD = NULL;
170 }
171}
172
173#ifdef DSBTYPE_IN
174static int dsound_init_in (HWVoiceIn *hw, audsettings_t *as)
175#else
176static int dsound_init_out (HWVoiceOut *hw, audsettings_t *as)
177#endif
178{
179 int err;
180 HRESULT hr;
181 dsound *s = &glob_dsound;
182 WAVEFORMATEX wfx;
183 audsettings_t obt_as;
184#ifdef DSBTYPE_IN
185 const char *typ = "ADC";
186 DSoundVoiceIn *ds = (DSoundVoiceIn *) hw;
187 DSCBUFFERDESC bd;
188 DSCBCAPS bc;
189#else
190 const char *typ = "DAC";
191 DSoundVoiceOut *ds = (DSoundVoiceOut *) hw;
192 DSBUFFERDESC bd;
193 DSBCAPS bc;
194#endif
195
196 err = waveformat_from_audio_settings (&wfx, as);
197 if (err) {
198 return -1;
199 }
200
201 memset (&bd, 0, sizeof (bd));
202 bd.dwSize = sizeof (bd);
203 bd.lpwfxFormat = &wfx;
204#ifdef DSBTYPE_IN
205 bd.dwBufferBytes = conf.bufsize_in;
206 hr = IDirectSoundCapture_CreateCaptureBuffer (
207 s->dsound_capture,
208 &bd,
209 &ds->dsound_capture_buffer,
210 NULL
211 );
212#else
213 bd.dwFlags = DSBCAPS_STICKYFOCUS | DSBCAPS_GETCURRENTPOSITION2;
214 bd.dwBufferBytes = conf.bufsize_out;
215 hr = IDirectSound_CreateSoundBuffer (
216 s->dsound,
217 &bd,
218 &ds->dsound_buffer,
219 NULL
220 );
221#endif
222
223 if (FAILED (hr)) {
224 dsound_logerr2 (hr, typ, "Could not create " NAME "\n");
225 return -1;
226 }
227
228 hr = glue (IFACE, _GetFormat) (ds->FIELD, &wfx, sizeof (wfx), NULL);
229 if (FAILED (hr)) {
230 dsound_logerr2 (hr, typ, "Could not get " NAME " format\n");
231 goto fail0;
232 }
233
234#ifdef DEBUG_DSOUND
235 dolog (NAME "\n");
236 print_wave_format (&wfx);
237#endif
238
239 memset (&bc, 0, sizeof (bc));
240 bc.dwSize = sizeof (bc);
241
242 hr = glue (IFACE, _GetCaps) (ds->FIELD, &bc);
243 if (FAILED (hr)) {
244 dsound_logerr2 (hr, typ, "Could not get " NAME " format\n");
245 goto fail0;
246 }
247
248 err = waveformat_to_audio_settings (&wfx, &obt_as);
249 if (err) {
250 goto fail0;
251 }
252
253 ds->first_time = 1;
254 obt_as.endianness = 0;
255 audio_pcm_init_info (&hw->info, &obt_as);
256
257 if (bc.dwBufferBytes & hw->info.align) {
258 dolog (
259 "GetCaps returned misaligned buffer size %ld, alignment %d\n",
260 bc.dwBufferBytes, hw->info.align + 1
261 );
262 }
263 hw->samples = bc.dwBufferBytes >> hw->info.shift;
264
265#ifdef DEBUG_DSOUND
266 dolog ("caps %ld, desc %ld\n",
267 bc.dwBufferBytes, bd.dwBufferBytes);
268
269 dolog ("bufsize %d, freq %d, chan %d, bits %d, sign %d\n",
270 hw->samples << hw->info.shift,
271 hw->info.freq,
272 hw->info.nchannels,
273 hw->info.bits,
274 hw->info.sign);
275#endif
276 return 0;
277
278 fail0:
279 glue (dsound_fini_, TYPE) (hw);
280 return -1;
281}
282
283#undef NAME
284#undef TYPE
285#undef IFACE
286#undef BUFPTR
287#undef FIELD
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette