2 * Tests basic sound playback in DirectSound.
3 * In particular we test each standard Windows sound format to make sure
4 * we handle the sound card/driver quirks correctly.
6 * Part of this test involves playing test tones. But this only makes
7 * sense if someone is going to carefully listen to it, and would only
8 * bother everyone else.
9 * So this is only done if the test is being run in interactive mode.
11 * Copyright (c) 2002-2004 Francois Gouget
12 * Copyright (c) 2007 Maarten Lankhorst
14 * This library is free software; you can redistribute it and/or
15 * modify it under the terms of the GNU Lesser General Public
16 * License as published by the Free Software Foundation; either
17 * version 2.1 of the License, or (at your option) any later version.
19 * This library 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 GNU
22 * Lesser General Public License for more details.
24 * You should have received a copy of the GNU Lesser General Public
25 * License along with this library; if not, write to the Free Software
26 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
31 #include "wine/test.h"
39 #include "dsound_test.h"
41 DEFINE_GUID(GUID_NULL,0,0,0,0,0,0,0,0,0,0,0);
43 static HRESULT (WINAPI *pDirectSoundEnumerateA)(LPDSENUMCALLBACKA,LPVOID)=NULL;
44 static HRESULT (WINAPI *pDirectSoundCreate)(LPCGUID,LPDIRECTSOUND*,
49 static void IDirectSound_test(LPDIRECTSOUND dso, BOOL initialized,
58 DWORD speaker_config, new_speaker_config;
60 /* Try to Query for objects */
61 rc=IDirectSound_QueryInterface(dso,&IID_IUnknown,(LPVOID*)&unknown);
62 ok(rc==DS_OK,"IDirectSound_QueryInterface(IID_IUnknown) failed: %08x\n", rc);
64 IDirectSound_Release(unknown);
66 rc=IDirectSound_QueryInterface(dso,&IID_IDirectSound,(LPVOID*)&ds);
67 ok(rc==DS_OK,"IDirectSound_QueryInterface(IID_IDirectSound) failed: %08x\n", rc);
69 IDirectSound_Release(ds);
71 rc=IDirectSound_QueryInterface(dso,&IID_IDirectSound8,(LPVOID*)&ds8);
72 ok(rc==E_NOINTERFACE,"IDirectSound_QueryInterface(IID_IDirectSound8) "
73 "should have failed: %08x\n",rc);
75 IDirectSound8_Release(ds8);
77 if (initialized == FALSE) {
78 /* try uninitialized object */
79 rc=IDirectSound_GetCaps(dso,0);
80 ok(rc==DSERR_UNINITIALIZED,"IDirectSound_GetCaps(NULL) "
81 "should have returned DSERR_UNINITIALIZED, returned: %08x\n", rc);
83 rc=IDirectSound_GetCaps(dso,&dscaps);
84 ok(rc==DSERR_UNINITIALIZED,"IDirectSound_GetCaps() "
85 "should have returned DSERR_UNINITIALIZED, returned: %08x\n", rc);
87 rc=IDirectSound_Compact(dso);
88 ok(rc==DSERR_UNINITIALIZED,"IDirectSound_Compact() "
89 "should have returned DSERR_UNINITIALIZED, returned: %08x\n", rc);
91 rc=IDirectSound_GetSpeakerConfig(dso,&speaker_config);
92 ok(rc==DSERR_UNINITIALIZED,"IDirectSound_GetSpeakerConfig() "
93 "should have returned DSERR_UNINITIALIZED, returned: %08x\n", rc);
95 rc=IDirectSound_Initialize(dso,lpGuid);
96 ok(rc==DS_OK||rc==DSERR_NODRIVER||rc==DSERR_ALLOCATED||rc==E_FAIL,
97 "IDirectSound_Initialize() failed: %08x\n",rc);
98 if (rc==DSERR_NODRIVER) {
99 trace(" No Driver\n");
101 } else if (rc==E_FAIL) {
102 trace(" No Device\n");
104 } else if (rc==DSERR_ALLOCATED) {
105 trace(" Already In Use\n");
110 rc=IDirectSound_Initialize(dso,lpGuid);
111 ok(rc==DSERR_ALREADYINITIALIZED, "IDirectSound_Initialize() "
112 "should have returned DSERR_ALREADYINITIALIZED: %08x\n", rc);
114 /* DSOUND: Error: Invalid caps buffer */
115 rc=IDirectSound_GetCaps(dso,0);
116 ok(rc==DSERR_INVALIDPARAM,"IDirectSound_GetCaps(NULL) "
117 "should have returned DSERR_INVALIDPARAM, returned: %08x\n", rc);
119 ZeroMemory(&dscaps, sizeof(dscaps));
121 /* DSOUND: Error: Invalid caps buffer */
122 rc=IDirectSound_GetCaps(dso,&dscaps);
123 ok(rc==DSERR_INVALIDPARAM,"IDirectSound_GetCaps() "
124 "should have returned DSERR_INVALIDPARAM, returned: %08x\n", rc);
126 dscaps.dwSize=sizeof(dscaps);
128 /* DSOUND: Running on a certified driver */
129 rc=IDirectSound_GetCaps(dso,&dscaps);
130 ok(rc==DS_OK,"IDirectSound_GetCaps() failed: %08x\n",rc);
132 rc=IDirectSound_Compact(dso);
133 ok(rc==DSERR_PRIOLEVELNEEDED,"IDirectSound_Compact() failed: %08x\n", rc);
135 rc=IDirectSound_SetCooperativeLevel(dso,get_hwnd(),DSSCL_PRIORITY);
136 ok(rc==DS_OK,"IDirectSound_SetCooperativeLevel() failed: %08x\n", rc);
138 rc=IDirectSound_Compact(dso);
139 ok(rc==DS_OK,"IDirectSound_Compact() failed: %08x\n",rc);
141 rc=IDirectSound_GetSpeakerConfig(dso,0);
142 ok(rc==DSERR_INVALIDPARAM,"IDirectSound_GetSpeakerConfig(NULL) "
143 "should have returned DSERR_INVALIDPARAM, returned: %08x\n", rc);
145 rc=IDirectSound_GetSpeakerConfig(dso,&speaker_config);
146 ok(rc==DS_OK,"IDirectSound_GetSpeakerConfig() failed: %08x\n", rc);
148 speaker_config = DSSPEAKER_COMBINED(DSSPEAKER_STEREO,
149 DSSPEAKER_GEOMETRY_WIDE);
150 rc=IDirectSound_SetSpeakerConfig(dso,speaker_config);
151 ok(rc==DS_OK,"IDirectSound_SetSpeakerConfig() failed: %08x\n", rc);
153 rc=IDirectSound_GetSpeakerConfig(dso,&new_speaker_config);
154 ok(rc==DS_OK,"IDirectSound_GetSpeakerConfig() failed: %08x\n", rc);
155 if (rc==DS_OK && speaker_config!=new_speaker_config)
156 trace("IDirectSound_GetSpeakerConfig() failed to set speaker "
157 "config: expected 0x%08x, got 0x%08x\n",
158 speaker_config,new_speaker_config);
162 ref=IDirectSound_Release(dso);
163 ok(ref==0,"IDirectSound_Release() has %d references, should have 0\n",ref);
166 static void IDirectSound_tests(void)
169 LPDIRECTSOUND dso=NULL;
170 LPCLASSFACTORY cf=NULL;
172 trace("Testing IDirectSound\n");
174 rc=CoGetClassObject(&CLSID_DirectSound, CLSCTX_INPROC_SERVER, NULL,
175 &IID_IClassFactory, (void**)&cf);
176 ok(rc==S_OK,"CoGetClassObject(CLSID_DirectSound, IID_IClassFactory) "
177 "failed: %08x\n", rc);
179 rc=CoGetClassObject(&CLSID_DirectSound, CLSCTX_INPROC_SERVER, NULL,
180 &IID_IUnknown, (void**)&cf);
181 ok(rc==S_OK,"CoGetClassObject(CLSID_DirectSound, IID_IUnknown) "
182 "failed: %08x\n", rc);
184 /* try the COM class factory method of creation with no device specified */
185 rc=CoCreateInstance(&CLSID_DirectSound, NULL, CLSCTX_INPROC_SERVER,
186 &IID_IDirectSound, (void**)&dso);
187 ok(rc==S_OK,"CoCreateInstance(CLSID_DirectSound) failed: %08x\n", rc);
189 IDirectSound_test(dso, FALSE, NULL);
191 /* try the COM class factory method of creation with default playback
192 * device specified */
193 rc=CoCreateInstance(&CLSID_DirectSound, NULL, CLSCTX_INPROC_SERVER,
194 &IID_IDirectSound, (void**)&dso);
195 ok(rc==S_OK,"CoCreateInstance(CLSID_DirectSound) failed: %08x\n", rc);
197 IDirectSound_test(dso, FALSE, &DSDEVID_DefaultPlayback);
199 /* try the COM class factory method of creation with default voice
200 * playback device specified */
201 rc=CoCreateInstance(&CLSID_DirectSound, NULL, CLSCTX_INPROC_SERVER,
202 &IID_IDirectSound, (void**)&dso);
203 ok(rc==S_OK,"CoCreateInstance(CLSID_DirectSound) failed: %08x\n", rc);
205 IDirectSound_test(dso, FALSE, &DSDEVID_DefaultVoicePlayback);
207 /* try the COM class factory method of creation with a bad
209 rc=CoCreateInstance(&CLSID_DirectSound, NULL, CLSCTX_INPROC_SERVER,
210 &CLSID_DirectSoundPrivate, (void**)&dso);
211 ok(rc==E_NOINTERFACE,
212 "CoCreateInstance(CLSID_DirectSound,CLSID_DirectSoundPrivate) "
213 "should have failed: %08x\n",rc);
215 /* try the COM class factory method of creation with a bad
216 * GUID and IID specified */
217 rc=CoCreateInstance(&CLSID_DirectSoundPrivate, NULL, CLSCTX_INPROC_SERVER,
218 &IID_IDirectSound, (void**)&dso);
219 ok(rc==REGDB_E_CLASSNOTREG,
220 "CoCreateInstance(CLSID_DirectSoundPrivate,IID_IDirectSound) "
221 "should have failed: %08x\n",rc);
223 /* try with no device specified */
224 rc=pDirectSoundCreate(NULL,&dso,NULL);
225 ok(rc==DS_OK||rc==DSERR_NODRIVER||rc==DSERR_ALLOCATED||rc==E_FAIL,
226 "DirectSoundCreate(NULL) failed: %08x\n",rc);
228 IDirectSound_test(dso, TRUE, NULL);
230 /* try with default playback device specified */
231 rc=pDirectSoundCreate(&DSDEVID_DefaultPlayback,&dso,NULL);
232 ok(rc==DS_OK||rc==DSERR_NODRIVER||rc==DSERR_ALLOCATED||rc==E_FAIL,
233 "DirectSoundCreate(DSDEVID_DefaultPlayback) failed: %08x\n", rc);
234 if (rc==DS_OK && dso)
235 IDirectSound_test(dso, TRUE, NULL);
237 /* try with default voice playback device specified */
238 rc=pDirectSoundCreate(&DSDEVID_DefaultVoicePlayback,&dso,NULL);
239 ok(rc==DS_OK||rc==DSERR_NODRIVER||rc==DSERR_ALLOCATED||rc==E_FAIL,
240 "DirectSoundCreate(DSDEVID_DefaultVoicePlayback) failed: %08x\n", rc);
241 if (rc==DS_OK && dso)
242 IDirectSound_test(dso, TRUE, NULL);
244 /* try with a bad device specified */
245 rc=pDirectSoundCreate(&DSDEVID_DefaultVoiceCapture,&dso,NULL);
246 ok(rc==DSERR_NODRIVER,"DirectSoundCreate(DSDEVID_DefaultVoiceCapture) "
247 "should have failed: %08x\n",rc);
248 if (rc==DS_OK && dso)
249 IDirectSound_Release(dso);
252 static HRESULT test_dsound(LPGUID lpGuid)
255 LPDIRECTSOUND dso=NULL;
258 /* DSOUND: Error: Invalid interface buffer */
259 rc=pDirectSoundCreate(lpGuid,0,NULL);
260 ok(rc==DSERR_INVALIDPARAM,"DirectSoundCreate() should have returned "
261 "DSERR_INVALIDPARAM, returned: %08x\n",rc);
263 /* Create the DirectSound object */
264 rc=pDirectSoundCreate(lpGuid,&dso,NULL);
265 ok(rc==DS_OK||rc==DSERR_NODRIVER||rc==DSERR_ALLOCATED||rc==E_FAIL,
266 "DirectSoundCreate() failed: %08x\n",rc);
270 /* Try the enumerated device */
271 IDirectSound_test(dso, TRUE, lpGuid);
273 /* Try the COM class factory method of creation with enumerated device */
274 rc=CoCreateInstance(&CLSID_DirectSound, NULL, CLSCTX_INPROC_SERVER,
275 &IID_IDirectSound, (void**)&dso);
276 ok(rc==S_OK,"CoCreateInstance(CLSID_DirectSound) failed: %08x\n", rc);
278 IDirectSound_test(dso, FALSE, lpGuid);
280 /* Create a DirectSound object */
281 rc=pDirectSoundCreate(lpGuid,&dso,NULL);
282 ok(rc==DS_OK,"DirectSoundCreate() failed: %08x\n",rc);
284 LPDIRECTSOUND dso1=NULL;
286 /* Create a second DirectSound object */
287 rc=pDirectSoundCreate(lpGuid,&dso1,NULL);
288 ok(rc==DS_OK,"DirectSoundCreate() failed: %08x\n",rc);
290 /* Release the second DirectSound object */
291 ref=IDirectSound_Release(dso1);
292 ok(ref==0,"IDirectSound_Release() has %d references, should have "
294 ok(dso!=dso1,"DirectSound objects should be unique: dso=%p,dso1=%p\n",dso,dso1);
297 /* Release the first DirectSound object */
298 ref=IDirectSound_Release(dso);
299 ok(ref==0,"IDirectSound_Release() has %d references, should have 0\n",
302 return DSERR_GENERIC;
306 /* Create a DirectSound object */
307 rc=pDirectSoundCreate(lpGuid,&dso,NULL);
308 ok(rc==DS_OK,"DirectSoundCreate() failed: %08x\n",rc);
310 LPDIRECTSOUNDBUFFER secondary;
311 DSBUFFERDESC bufdesc;
314 init_format(&wfx,WAVE_FORMAT_PCM,11025,8,1);
315 ZeroMemory(&bufdesc, sizeof(bufdesc));
316 bufdesc.dwSize=sizeof(bufdesc);
317 bufdesc.dwFlags=DSBCAPS_GETCURRENTPOSITION2 | DSBCAPS_CTRL3D;
318 bufdesc.dwBufferBytes=align(wfx.nAvgBytesPerSec*BUFFER_LEN/1000,
320 bufdesc.lpwfxFormat=&wfx;
321 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&secondary,NULL);
322 ok(rc==DS_OK && secondary!=NULL,
323 "IDirectSound_CreateSoundBuffer() failed to create a secondary "
325 if (rc==DS_OK && secondary!=NULL) {
326 LPDIRECTSOUND3DBUFFER buffer3d;
327 rc=IDirectSound_QueryInterface(secondary, &IID_IDirectSound3DBuffer,
329 ok(rc==DS_OK && buffer3d!=NULL,"IDirectSound_QueryInterface() "
330 "failed: %08x\n",rc);
331 if (rc==DS_OK && buffer3d!=NULL) {
332 ref=IDirectSound3DBuffer_AddRef(buffer3d);
333 ok(ref==2,"IDirectSound3DBuffer_AddRef() has %d references, "
334 "should have 2\n",ref);
336 ref=IDirectSoundBuffer_AddRef(secondary);
337 ok(ref==2,"IDirectSoundBuffer_AddRef() has %d references, "
338 "should have 2\n",ref);
340 /* release with buffer */
341 ref=IDirectSound_Release(dso);
342 ok(ref==0,"IDirectSound_Release() has %d references, should have 0\n",
345 return DSERR_GENERIC;
352 static HRESULT test_primary(LPGUID lpGuid)
355 LPDIRECTSOUND dso=NULL;
356 LPDIRECTSOUNDBUFFER primary=NULL,second=NULL,third=NULL;
357 DSBUFFERDESC bufdesc;
362 /* Create the DirectSound object */
363 rc=pDirectSoundCreate(lpGuid,&dso,NULL);
364 ok(rc==DS_OK||rc==DSERR_NODRIVER||rc==DSERR_ALLOCATED,
365 "DirectSoundCreate() failed: %08x\n",rc);
369 /* Get the device capabilities */
370 ZeroMemory(&dscaps, sizeof(dscaps));
371 dscaps.dwSize=sizeof(dscaps);
372 rc=IDirectSound_GetCaps(dso,&dscaps);
373 ok(rc==DS_OK,"IDirectSound_GetCaps() failed: %08x\n",rc);
377 /* DSOUND: Error: Invalid buffer description pointer */
378 rc=IDirectSound_CreateSoundBuffer(dso,0,0,NULL);
379 ok(rc==DSERR_INVALIDPARAM,
380 "IDirectSound_CreateSoundBuffer() should have failed: %08x\n", rc);
382 /* DSOUND: Error: NULL pointer is invalid */
383 /* DSOUND: Error: Invalid buffer description pointer */
384 rc=IDirectSound_CreateSoundBuffer(dso,0,&primary,NULL);
385 ok(rc==DSERR_INVALIDPARAM && primary==0,
386 "IDirectSound_CreateSoundBuffer() should have failed: rc=%08x,"
387 "dsbo=%p\n",rc,primary);
389 /* DSOUND: Error: Invalid size */
390 /* DSOUND: Error: Invalid buffer description */
392 ZeroMemory(&bufdesc, sizeof(bufdesc));
393 bufdesc.dwSize=sizeof(bufdesc)-1;
394 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&primary,NULL);
395 ok(rc==DSERR_INVALIDPARAM && primary==0,
396 "IDirectSound_CreateSoundBuffer() should have failed: rc=%08x,"
397 "primary=%p\n",rc,primary);
399 /* DSOUND: Error: DSBCAPS_PRIMARYBUFFER flag with non-NULL lpwfxFormat */
400 /* DSOUND: Error: Invalid buffer description pointer */
402 ZeroMemory(&bufdesc, sizeof(bufdesc));
403 bufdesc.dwSize=sizeof(bufdesc);
404 bufdesc.dwFlags=DSBCAPS_PRIMARYBUFFER;
405 bufdesc.lpwfxFormat=&wfx;
406 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&primary,NULL);
407 ok(rc==DSERR_INVALIDPARAM && primary==0,
408 "IDirectSound_CreateSoundBuffer() should have failed: rc=%08x,"
409 "primary=%p\n",rc,primary);
411 /* DSOUND: Error: No DSBCAPS_PRIMARYBUFFER flag with NULL lpwfxFormat */
412 /* DSOUND: Error: Invalid buffer description pointer */
414 ZeroMemory(&bufdesc, sizeof(bufdesc));
415 bufdesc.dwSize=sizeof(bufdesc);
417 bufdesc.lpwfxFormat=NULL;
418 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&primary,NULL);
419 ok(rc==DSERR_INVALIDPARAM && primary==0,
420 "IDirectSound_CreateSoundBuffer() should have failed: rc=%08x,"
421 "primary=%p\n",rc,primary);
423 /* We must call SetCooperativeLevel before calling CreateSoundBuffer */
424 /* DSOUND: Setting DirectSound cooperative level to DSSCL_PRIORITY */
425 rc=IDirectSound_SetCooperativeLevel(dso,get_hwnd(),DSSCL_PRIORITY);
426 ok(rc==DS_OK,"IDirectSound_SetCooperativeLevel() failed: %08x\n", rc);
430 /* Testing the primary buffer */
432 ZeroMemory(&bufdesc, sizeof(bufdesc));
433 bufdesc.dwSize=sizeof(bufdesc);
434 bufdesc.dwFlags=DSBCAPS_PRIMARYBUFFER|DSBCAPS_CTRLVOLUME;
435 bufdesc.lpwfxFormat = &wfx;
436 init_format(&wfx,WAVE_FORMAT_PCM,11025,8,2);
437 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&primary,NULL);
438 ok(rc==DSERR_INVALIDPARAM,"IDirectSound_CreateSoundBuffer() should have "
439 "returned DSERR_INVALIDPARAM, returned: %08x\n", rc);
440 if (rc==DS_OK && primary!=NULL)
441 IDirectSoundBuffer_Release(primary);
444 ZeroMemory(&bufdesc, sizeof(bufdesc));
445 bufdesc.dwSize=sizeof(bufdesc);
446 bufdesc.dwFlags=DSBCAPS_PRIMARYBUFFER|DSBCAPS_CTRLVOLUME;
447 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&primary,NULL);
448 ok((rc==DS_OK && primary!=NULL) || (rc==DSERR_CONTROLUNAVAIL),
449 "IDirectSound_CreateSoundBuffer() failed to create a primary buffer: %08x\n",rc);
450 if (rc==DSERR_CONTROLUNAVAIL)
451 trace(" No Primary\n");
452 else if (rc==DS_OK && primary!=NULL) {
455 /* Try to create a second primary buffer */
456 /* DSOUND: Error: The primary buffer already exists.
457 * Any changes made to the buffer description will be ignored. */
458 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&second,NULL);
459 ok(rc==DS_OK && second==primary,
460 "IDirectSound_CreateSoundBuffer() should have returned original "
461 "primary buffer: %08x\n",rc);
462 ref=IDirectSoundBuffer_Release(second);
463 ok(ref==1,"IDirectSoundBuffer_Release() primary has %d references, "
464 "should have 1\n",ref);
466 /* Try to duplicate a primary buffer */
467 /* DSOUND: Error: Can't duplicate primary buffers */
468 rc=IDirectSound_DuplicateSoundBuffer(dso,primary,&third);
470 ok(rc!=DS_OK,"IDirectSound_DuplicateSoundBuffer() primary buffer "
471 "should have failed %08x\n",rc);
473 rc=IDirectSoundBuffer_GetVolume(primary,&vol);
474 ok(rc==DS_OK,"IDirectSoundBuffer_GetVolume() failed: %08x\n", rc);
476 if (winetest_interactive) {
477 trace("Playing a 5 seconds reference tone at the current "
480 trace("(the current volume is %d according to DirectSound)\n",
482 trace("All subsequent tones should be identical to this one.\n");
483 trace("Listen for stutter, changes in pitch, volume, etc.\n");
485 test_buffer(dso,&primary,1,FALSE,0,FALSE,0,winetest_interactive &&
486 !(dscaps.dwFlags & DSCAPS_EMULDRIVER),5.0,0,0,0,0,FALSE,0);
488 ref=IDirectSoundBuffer_Release(primary);
489 ok(ref==0,"IDirectSoundBuffer_Release() primary has %d references, "
490 "should have 0\n",ref);
493 /* Set the CooperativeLevel back to normal */
494 /* DSOUND: Setting DirectSound cooperative level to DSSCL_NORMAL */
495 rc=IDirectSound_SetCooperativeLevel(dso,get_hwnd(),DSSCL_NORMAL);
496 ok(rc==DS_OK,"IDirectSound_SetCooperativeLevel() failed: %08x\n", rc);
499 ref=IDirectSound_Release(dso);
500 ok(ref==0,"IDirectSound_Release() has %d references, should have 0\n",ref);
502 return DSERR_GENERIC;
508 * Test the primary buffer at different formats while keeping the
509 * secondary buffer at a constant format.
511 static HRESULT test_primary_secondary(LPGUID lpGuid)
514 LPDIRECTSOUND dso=NULL;
515 LPDIRECTSOUNDBUFFER primary=NULL,secondary=NULL;
516 DSBUFFERDESC bufdesc;
518 WAVEFORMATEX wfx, wfx2;
521 /* Create the DirectSound object */
522 rc=pDirectSoundCreate(lpGuid,&dso,NULL);
523 ok(rc==DS_OK||rc==DSERR_NODRIVER||rc==DSERR_ALLOCATED,
524 "DirectSoundCreate() failed: %08x\n",rc);
528 /* Get the device capabilities */
529 ZeroMemory(&dscaps, sizeof(dscaps));
530 dscaps.dwSize=sizeof(dscaps);
531 rc=IDirectSound_GetCaps(dso,&dscaps);
532 ok(rc==DS_OK,"IDirectSound_GetCaps() failed: %08x\n",rc);
536 /* We must call SetCooperativeLevel before creating primary buffer */
537 /* DSOUND: Setting DirectSound cooperative level to DSSCL_PRIORITY */
538 rc=IDirectSound_SetCooperativeLevel(dso,get_hwnd(),DSSCL_PRIORITY);
539 ok(rc==DS_OK,"IDirectSound_SetCooperativeLevel() failed: %08x\n", rc);
543 ZeroMemory(&bufdesc, sizeof(bufdesc));
544 bufdesc.dwSize=sizeof(bufdesc);
545 bufdesc.dwFlags=DSBCAPS_PRIMARYBUFFER;
546 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&primary,NULL);
547 ok(rc==DS_OK && primary!=NULL,
548 "IDirectSound_CreateSoundBuffer() failed to create a primary buffer %08x\n",rc);
550 if (rc==DS_OK && primary!=NULL) {
551 for (f=0;f<NB_FORMATS;f++) {
552 /* We must call SetCooperativeLevel to be allowed to call
554 /* DSOUND: Setting DirectSound cooperative level to
556 rc=IDirectSound_SetCooperativeLevel(dso,get_hwnd(),DSSCL_PRIORITY);
557 ok(rc==DS_OK,"IDirectSound_SetCooperativeLevel() failed: %08x\n", rc);
561 init_format(&wfx,WAVE_FORMAT_PCM,formats[f][0],formats[f][1],
564 rc=IDirectSoundBuffer_SetFormat(primary,&wfx);
566 if (wfx.wBitsPerSample <= 16)
567 ok(rc==DS_OK,"IDirectSoundBuffer_SetFormat(%s) failed: %08x\n",
568 format_string(&wfx), rc);
570 ok(rc==DS_OK || rc == E_INVALIDARG, "SetFormat (%s) failed: %08x\n",
571 format_string(&wfx), rc);
573 /* There is no guarantee that SetFormat will actually change the
574 * format to what we asked for. It depends on what the soundcard
575 * supports. So we must re-query the format.
577 rc=IDirectSoundBuffer_GetFormat(primary,&wfx,sizeof(wfx),NULL);
578 ok(rc==DS_OK,"IDirectSoundBuffer_GetFormat() failed: %08x\n", rc);
580 (wfx.wFormatTag!=wfx2.wFormatTag ||
581 wfx.nSamplesPerSec!=wfx2.nSamplesPerSec ||
582 wfx.wBitsPerSample!=wfx2.wBitsPerSample ||
583 wfx.nChannels!=wfx2.nChannels)) {
584 trace("Requested primary format tag=0x%04x %dx%dx%d "
585 "avg.B/s=%d align=%d\n",
586 wfx2.wFormatTag,wfx2.nSamplesPerSec,wfx2.wBitsPerSample,
587 wfx2.nChannels,wfx2.nAvgBytesPerSec,wfx2.nBlockAlign);
588 trace("Got tag=0x%04x %dx%dx%d avg.B/s=%d align=%d\n",
589 wfx.wFormatTag,wfx.nSamplesPerSec,wfx.wBitsPerSample,
590 wfx.nChannels,wfx.nAvgBytesPerSec,wfx.nBlockAlign);
593 /* Set the CooperativeLevel back to normal */
594 /* DSOUND: Setting DirectSound cooperative level to DSSCL_NORMAL */
595 rc=IDirectSound_SetCooperativeLevel(dso,get_hwnd(),DSSCL_NORMAL);
596 ok(rc==DS_OK,"IDirectSound_SetCooperativeLevel() failed: %08x\n", rc);
598 init_format(&wfx2,WAVE_FORMAT_PCM,11025,16,2);
601 ZeroMemory(&bufdesc, sizeof(bufdesc));
602 bufdesc.dwSize=sizeof(bufdesc);
603 bufdesc.dwFlags=DSBCAPS_GETCURRENTPOSITION2;
604 bufdesc.dwBufferBytes=align(wfx.nAvgBytesPerSec*BUFFER_LEN/1000,
606 bufdesc.lpwfxFormat=&wfx2;
607 if (winetest_interactive) {
608 trace(" Testing a primary buffer at %dx%dx%d with a "
609 "secondary buffer at %dx%dx%d\n",
610 wfx.nSamplesPerSec,wfx.wBitsPerSample,wfx.nChannels,
611 wfx2.nSamplesPerSec,wfx2.wBitsPerSample,wfx2.nChannels);
613 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&secondary,NULL);
614 ok(rc==DS_OK && secondary!=NULL,
615 "IDirectSound_CreateSoundBuffer() failed to create a secondary buffer %08x\n",rc);
617 if (rc==DS_OK && secondary!=NULL) {
618 test_buffer(dso,&secondary,0,FALSE,0,FALSE,0,
619 winetest_interactive,1.0,0,NULL,0,0,FALSE,0);
621 ref=IDirectSoundBuffer_Release(secondary);
622 ok(ref==0,"IDirectSoundBuffer_Release() has %d references, "
623 "should have 0\n",ref);
627 ref=IDirectSoundBuffer_Release(primary);
628 ok(ref==0,"IDirectSoundBuffer_Release() primary has %d references, "
629 "should have 0\n",ref);
632 /* Set the CooperativeLevel back to normal */
633 /* DSOUND: Setting DirectSound cooperative level to DSSCL_NORMAL */
634 rc=IDirectSound_SetCooperativeLevel(dso,get_hwnd(),DSSCL_NORMAL);
635 ok(rc==DS_OK,"IDirectSound_SetCooperativeLevel() failed: %08x\n", rc);
638 ref=IDirectSound_Release(dso);
639 ok(ref==0,"IDirectSound_Release() has %d references, should have 0\n",ref);
641 return DSERR_GENERIC;
646 static HRESULT test_secondary(LPGUID lpGuid)
649 LPDIRECTSOUND dso=NULL;
650 LPDIRECTSOUNDBUFFER primary=NULL,secondary=NULL;
651 DSBUFFERDESC bufdesc;
653 WAVEFORMATEX wfx, wfx1;
657 /* Create the DirectSound object */
658 rc=pDirectSoundCreate(lpGuid,&dso,NULL);
659 ok(rc==DS_OK||rc==DSERR_NODRIVER||rc==DSERR_ALLOCATED,
660 "DirectSoundCreate() failed: %08x\n",rc);
664 /* Get the device capabilities */
665 ZeroMemory(&dscaps, sizeof(dscaps));
666 dscaps.dwSize=sizeof(dscaps);
667 rc=IDirectSound_GetCaps(dso,&dscaps);
668 ok(rc==DS_OK,"IDirectSound_GetCaps() failed: %08x\n",rc);
672 /* We must call SetCooperativeLevel before creating primary buffer */
673 /* DSOUND: Setting DirectSound cooperative level to DSSCL_PRIORITY */
674 rc=IDirectSound_SetCooperativeLevel(dso,get_hwnd(),DSSCL_PRIORITY);
675 ok(rc==DS_OK,"IDirectSound_SetCooperativeLevel() failed: %08x\n", rc);
679 ZeroMemory(&bufdesc, sizeof(bufdesc));
680 bufdesc.dwSize=sizeof(bufdesc);
681 bufdesc.dwFlags=DSBCAPS_PRIMARYBUFFER;
682 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&primary,NULL);
683 ok(rc==DS_OK && primary!=NULL,
684 "IDirectSound_CreateSoundBuffer() failed to create a primary buffer %08x\n",rc);
686 if (rc==DS_OK && primary!=NULL) {
687 rc=IDirectSoundBuffer_GetFormat(primary,&wfx1,sizeof(wfx1),NULL);
688 ok(rc==DS_OK,"IDirectSoundBuffer8_Getformat() failed: %08x\n", rc);
692 for (f=0;f<NB_FORMATS;f++) {
693 WAVEFORMATEXTENSIBLE wfxe;
694 init_format(&wfx,WAVE_FORMAT_PCM,formats[f][0],formats[f][1],
697 ZeroMemory(&bufdesc, sizeof(bufdesc));
698 bufdesc.dwSize=sizeof(bufdesc);
699 bufdesc.dwFlags=DSBCAPS_GETCURRENTPOSITION2;
700 bufdesc.dwBufferBytes=align(wfx.nAvgBytesPerSec*BUFFER_LEN/1000,
702 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&secondary,NULL);
703 ok(rc==DSERR_INVALIDPARAM,"IDirectSound_CreateSoundBuffer() "
704 "should have returned DSERR_INVALIDPARAM, returned: %08x\n", rc);
705 if (rc==DS_OK && secondary!=NULL)
706 IDirectSoundBuffer_Release(secondary);
709 ZeroMemory(&bufdesc, sizeof(bufdesc));
710 bufdesc.dwSize=sizeof(bufdesc);
711 bufdesc.dwFlags=DSBCAPS_GETCURRENTPOSITION2;
712 bufdesc.dwBufferBytes=align(wfx.nAvgBytesPerSec*BUFFER_LEN/1000,
714 bufdesc.lpwfxFormat=&wfx;
715 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&secondary,NULL);
716 if (gotdx8 || wfx.wBitsPerSample <= 16)
718 if (wfx.wBitsPerSample > 16)
719 ok(((rc == DSERR_CONTROLUNAVAIL || rc == DSERR_INVALIDCALL || rc == DSERR_INVALIDPARAM /* 2003 */) && !secondary)
720 || rc == DS_OK, /* driver dependent? */
721 "IDirectSound_CreateSoundBuffer() "
722 "should have returned (DSERR_CONTROLUNAVAIL or DSERR_INVALIDCALL) "
723 "and NULL, returned: %08x %p\n", rc, secondary);
725 ok(rc==DS_OK && secondary!=NULL,
726 "IDirectSound_CreateSoundBuffer() failed to create a secondary buffer %08x\n",rc);
729 ok(rc==E_INVALIDARG, "Creating %d bpp buffer on dx < 8 returned: %p %08x\n",
730 wfx.wBitsPerSample, secondary, rc);
734 skip("Not doing the WAVE_FORMAT_EXTENSIBLE tests\n");
735 /* Apparently they succeed with bogus values,
736 * which means that older dsound doesn't look at them
742 IDirectSoundBuffer_Release(secondary);
745 bufdesc.lpwfxFormat=(WAVEFORMATEX*)&wfxe;
747 wfxe.Format.wFormatTag = WAVE_FORMAT_EXTENSIBLE;
748 wfxe.SubFormat = KSDATAFORMAT_SUBTYPE_PCM;
749 wfxe.Format.cbSize = 1;
750 wfxe.Samples.wValidBitsPerSample = wfx.wBitsPerSample;
751 wfxe.dwChannelMask = (wfx.nChannels == 1 ? KSAUDIO_SPEAKER_MONO : KSAUDIO_SPEAKER_STEREO);
753 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&secondary,NULL);
754 ok((rc==DSERR_INVALIDPARAM || rc==DSERR_INVALIDCALL /* 2003 */) && !secondary,
755 "IDirectSound_CreateSoundBuffer() returned: %08x %p\n",
759 IDirectSoundBuffer_Release(secondary);
763 wfxe.Format.cbSize = sizeof(wfxe) - sizeof(wfx) + 1;
765 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&secondary,NULL);
766 ok(((rc==DSERR_CONTROLUNAVAIL || rc==DSERR_INVALIDCALL || rc==DSERR_INVALIDPARAM)
768 || rc==DS_OK, /* 2003 / 2008 */
769 "IDirectSound_CreateSoundBuffer() returned: %08x %p\n",
773 IDirectSoundBuffer_Release(secondary);
777 wfxe.Format.cbSize = sizeof(wfxe) - sizeof(wfx);
778 wfxe.SubFormat = GUID_NULL;
779 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&secondary,NULL);
780 ok((rc==DSERR_INVALIDPARAM || rc==DSERR_INVALIDCALL) && !secondary,
781 "IDirectSound_CreateSoundBuffer() returned: %08x %p\n",
785 IDirectSoundBuffer_Release(secondary);
788 wfxe.SubFormat = KSDATAFORMAT_SUBTYPE_PCM;
790 ++wfxe.Samples.wValidBitsPerSample;
791 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&secondary,NULL);
792 ok(rc==DSERR_INVALIDPARAM && !secondary,
793 "IDirectSound_CreateSoundBuffer() returned: %08x %p\n",
797 IDirectSoundBuffer_Release(secondary);
800 --wfxe.Samples.wValidBitsPerSample;
802 wfxe.Samples.wValidBitsPerSample = 0;
803 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&secondary,NULL);
804 ok(rc==DS_OK && secondary,
805 "IDirectSound_CreateSoundBuffer() returned: %08x %p\n",
809 IDirectSoundBuffer_Release(secondary);
812 wfxe.Samples.wValidBitsPerSample = wfxe.Format.wBitsPerSample;
814 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&secondary,NULL);
815 ok(rc==DS_OK && secondary!=NULL,
816 "IDirectSound_CreateSoundBuffer() failed to create a secondary buffer %08x\n",rc);
819 if (rc==DS_OK && secondary!=NULL) {
820 if (winetest_interactive) {
821 trace(" Testing a secondary buffer at %dx%dx%d "
822 "with a primary buffer at %dx%dx%d\n",
823 wfx.nSamplesPerSec,wfx.wBitsPerSample,wfx.nChannels,
824 wfx1.nSamplesPerSec,wfx1.wBitsPerSample,wfx1.nChannels);
826 test_buffer(dso,&secondary,0,FALSE,0,FALSE,0,
827 winetest_interactive,1.0,0,NULL,0,0,FALSE,0);
829 ref=IDirectSoundBuffer_Release(secondary);
830 ok(ref==0,"IDirectSoundBuffer_Release() has %d references, "
831 "should have 0\n",ref);
835 ref=IDirectSoundBuffer_Release(primary);
836 ok(ref==0,"IDirectSoundBuffer_Release() primary has %d references, "
837 "should have 0\n",ref);
840 /* Set the CooperativeLevel back to normal */
841 /* DSOUND: Setting DirectSound cooperative level to DSSCL_NORMAL */
842 rc=IDirectSound_SetCooperativeLevel(dso,get_hwnd(),DSSCL_NORMAL);
843 ok(rc==DS_OK,"IDirectSound_SetCooperativeLevel() failed: %08x\n", rc);
846 ref=IDirectSound_Release(dso);
847 ok(ref==0,"IDirectSound_Release() has %d references, should have 0\n",ref);
849 return DSERR_GENERIC;
854 static HRESULT test_block_align(LPGUID lpGuid)
857 LPDIRECTSOUND dso=NULL;
858 LPDIRECTSOUNDBUFFER secondary=NULL;
859 DSBUFFERDESC bufdesc;
865 /* Create the DirectSound object */
866 rc=pDirectSoundCreate(lpGuid,&dso,NULL);
867 ok(rc==DS_OK||rc==DSERR_NODRIVER||rc==DSERR_ALLOCATED,
868 "DirectSoundCreate() failed: %08x\n",rc);
872 init_format(&wfx,WAVE_FORMAT_PCM,11025,16,2);
873 ZeroMemory(&bufdesc, sizeof(bufdesc));
874 bufdesc.dwSize=sizeof(bufdesc);
875 bufdesc.dwFlags=DSBCAPS_GETCURRENTPOSITION2;
876 bufdesc.dwBufferBytes=wfx.nAvgBytesPerSec + 1;
877 bufdesc.lpwfxFormat=&wfx;
878 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&secondary,NULL);
879 ok(rc==DS_OK,"IDirectSound_CreateSoundBuffer() "
880 "should have returned DS_OK, returned: %08x\n", rc);
882 if (rc==DS_OK && secondary!=NULL) {
883 ZeroMemory(&dsbcaps, sizeof(dsbcaps));
884 dsbcaps.dwSize = sizeof(dsbcaps);
885 rc=IDirectSoundBuffer_GetCaps(secondary,&dsbcaps);
886 ok(rc==DS_OK,"IDirectSoundBuffer_GetCaps() should have returned DS_OK, "
887 "returned: %08x\n", rc);
888 if (rc==DS_OK && wfx.nBlockAlign > 1)
890 ok(dsbcaps.dwBufferBytes==(wfx.nAvgBytesPerSec + wfx.nBlockAlign),
891 "Buffer size not a multiple of nBlockAlign: requested %d, "
892 "got %d, should be %d\n", bufdesc.dwBufferBytes,
893 dsbcaps.dwBufferBytes, wfx.nAvgBytesPerSec + wfx.nBlockAlign);
895 rc = IDirectSoundBuffer_SetCurrentPosition(secondary, 0);
896 ok(rc == DS_OK, "Could not set position to 0: %08x\n", rc);
897 rc = IDirectSoundBuffer_GetCurrentPosition(secondary, &pos, NULL);
898 ok(rc == DS_OK, "Could not get position: %08x\n", rc);
899 rc = IDirectSoundBuffer_SetCurrentPosition(secondary, 1);
900 ok(rc == DS_OK, "Could not set position to 1: %08x\n", rc);
901 rc = IDirectSoundBuffer_GetCurrentPosition(secondary, &pos2, NULL);
902 ok(rc == DS_OK, "Could not get new position: %08x\n", rc);
903 ok(pos == pos2, "Positions not the same! Old position: %d, new position: %d\n", pos, pos2);
905 ref=IDirectSoundBuffer_Release(secondary);
906 ok(ref==0,"IDirectSoundBuffer_Release() secondary has %d references, "
907 "should have 0\n",ref);
910 ref=IDirectSound_Release(dso);
911 ok(ref==0,"IDirectSound_Release() has %d references, should have 0\n",ref);
913 return DSERR_GENERIC;
921 } fmts[] = { { 8, 1 }, { 8, 2 }, { 16, 1 }, {16, 2 } };
923 static HRESULT test_frequency(LPGUID lpGuid)
926 LPDIRECTSOUND dso=NULL;
927 LPDIRECTSOUNDBUFFER primary=NULL,secondary=NULL;
928 DSBUFFERDESC bufdesc;
930 WAVEFORMATEX wfx, wfx1;
933 int rates[] = { 8000, 11025, 12000, 16000, 22050, 24000, 32000, 44100,
936 /* Create the DirectSound object */
937 rc=pDirectSoundCreate(lpGuid,&dso,NULL);
938 ok(rc==DS_OK||rc==DSERR_NODRIVER||rc==DSERR_ALLOCATED,
939 "DirectSoundCreate() failed: %08x\n",rc);
943 /* Get the device capabilities */
944 ZeroMemory(&dscaps, sizeof(dscaps));
945 dscaps.dwSize=sizeof(dscaps);
946 rc=IDirectSound_GetCaps(dso,&dscaps);
947 ok(rc==DS_OK,"IDirectSound_GetCaps() failed: %08x\n",rc);
951 /* We must call SetCooperativeLevel before creating primary buffer */
952 /* DSOUND: Setting DirectSound cooperative level to DSSCL_PRIORITY */
953 rc=IDirectSound_SetCooperativeLevel(dso,get_hwnd(),DSSCL_PRIORITY);
954 ok(rc==DS_OK,"IDirectSound_SetCooperativeLevel() failed: %08x\n", rc);
958 ZeroMemory(&bufdesc, sizeof(bufdesc));
959 bufdesc.dwSize=sizeof(bufdesc);
960 bufdesc.dwFlags=DSBCAPS_PRIMARYBUFFER;
961 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&primary,NULL);
962 ok(rc==DS_OK && primary!=NULL,
963 "IDirectSound_CreateSoundBuffer() failed to create a primary buffer %08x\n",rc);
965 if (rc==DS_OK && primary!=NULL) {
966 rc=IDirectSoundBuffer_GetFormat(primary,&wfx1,sizeof(wfx1),NULL);
967 ok(rc==DS_OK,"IDirectSoundBuffer8_Getformat() failed: %08x\n", rc);
971 for (f=0;f<sizeof(fmts)/sizeof(fmts[0]);f++) {
972 for (r=0;r<sizeof(rates)/sizeof(rates[0]);r++) {
973 init_format(&wfx,WAVE_FORMAT_PCM,11025,fmts[f].bits,
976 ZeroMemory(&bufdesc, sizeof(bufdesc));
977 bufdesc.dwSize=sizeof(bufdesc);
978 bufdesc.dwFlags=DSBCAPS_GETCURRENTPOSITION2|DSBCAPS_CTRLFREQUENCY;
979 bufdesc.dwBufferBytes=align((wfx.nAvgBytesPerSec*rates[r]/11025)*
980 BUFFER_LEN/1000,wfx.nBlockAlign);
981 bufdesc.lpwfxFormat=&wfx;
982 if (winetest_interactive) {
983 trace(" Testing a secondary buffer at %dx%dx%d "
984 "with a primary buffer at %dx%dx%d\n",
985 wfx.nSamplesPerSec,wfx.wBitsPerSample,wfx.nChannels,
986 wfx1.nSamplesPerSec,wfx1.wBitsPerSample,wfx1.nChannels);
988 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&secondary,NULL);
989 ok(rc==DS_OK && secondary!=NULL,
990 "IDirectSound_CreateSoundBuffer() failed to create a secondary buffer %08x\n",rc);
992 if (rc==DS_OK && secondary!=NULL) {
993 test_buffer(dso,&secondary,0,FALSE,0,FALSE,0,
994 winetest_interactive,1.0,0,NULL,0,0,TRUE,rates[r]);
996 ref=IDirectSoundBuffer_Release(secondary);
997 ok(ref==0,"IDirectSoundBuffer_Release() has %d references, "
998 "should have 0\n",ref);
1003 ref=IDirectSoundBuffer_Release(primary);
1004 ok(ref==0,"IDirectSoundBuffer_Release() primary has %d references, "
1005 "should have 0\n",ref);
1008 /* Set the CooperativeLevel back to normal */
1009 /* DSOUND: Setting DirectSound cooperative level to DSSCL_NORMAL */
1010 rc=IDirectSound_SetCooperativeLevel(dso,get_hwnd(),DSSCL_NORMAL);
1011 ok(rc==DS_OK,"IDirectSound_SetCooperativeLevel() failed: %08x\n", rc);
1014 ref=IDirectSound_Release(dso);
1015 ok(ref==0,"IDirectSound_Release() has %d references, should have 0\n",ref);
1017 return DSERR_GENERIC;
1022 static unsigned int number;
1024 static BOOL WINAPI dsenum_callback(LPGUID lpGuid, LPCSTR lpcstrDescription,
1025 LPCSTR lpcstrModule, LPVOID lpContext)
1028 trace("*** Testing %s - %s ***\n",lpcstrDescription,lpcstrModule);
1030 /* Don't test the primary device */
1033 ok (!lpcstrModule[0], "lpcstrModule(%s) != NULL\n", lpcstrModule);
1037 rc = test_dsound(lpGuid);
1038 if (rc == DSERR_NODRIVER)
1039 trace(" No Driver\n");
1040 else if (rc == DSERR_ALLOCATED)
1041 trace(" Already In Use\n");
1042 else if (rc == E_FAIL)
1043 trace(" No Device\n");
1045 test_block_align(lpGuid);
1046 test_primary(lpGuid);
1047 test_primary_secondary(lpGuid);
1048 test_secondary(lpGuid);
1049 test_frequency(lpGuid);
1055 static void dsound_tests(void)
1058 rc=pDirectSoundEnumerateA(&dsenum_callback,NULL);
1059 ok(rc==DS_OK,"DirectSoundEnumerateA() failed: %08x\n",rc);
1068 hDsound = LoadLibrary("dsound.dll");
1073 ok( FreeLibrary(hDsound), "FreeLibrary(1) returned %d\n", GetLastError());
1074 SetLastError(0xdeadbeef);
1075 ret = FreeLibrary(hDsound);
1077 broken(!ret && GetLastError() == ERROR_MOD_NOT_FOUND) || /* NT4 */
1078 broken(!ret && GetLastError() == ERROR_INVALID_HANDLE), /* Win9x */
1079 "FreeLibrary(2) returned %d\n", GetLastError());
1080 ok(!FreeLibrary(hDsound), "DirectSound DLL still loaded\n");
1083 hDsound = LoadLibrary("dsound.dll");
1087 pDirectSoundEnumerateA = (void*)GetProcAddress(hDsound,
1088 "DirectSoundEnumerateA");
1089 pDirectSoundCreate = (void*)GetProcAddress(hDsound,
1090 "DirectSoundCreate");
1092 gotdx8 = !!GetProcAddress(hDsound, "DirectSoundCreate8");
1094 IDirectSound_tests();
1097 FreeLibrary(hDsound);
1100 skip("dsound.dll not found!\n");