1 /*
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.
5 *
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.
10 *
11 * Copyright (c) 2002-2004 Francois Gouget
12 * Copyright (c) 2007 Maarten Lankhorst
13 *
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.
18 *
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.
23 *
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
27 */
28
29 #include <windows.h>
30
31 #include "wine/test.h"
32 #include "dsound.h"
33 #include "dsconf.h"
34 #include "mmreg.h"
35 #include "initguid.h"
36 #include "ks.h"
37 #include "ksmedia.h"
38
39 #include "dsound_test.h"
40
41 DEFINE_GUID(GUID_NULL,0,0,0,0,0,0,0,0,0,0,0);
42
43 static HRESULT (WINAPI *pDirectSoundEnumerateA)(LPDSENUMCALLBACKA,LPVOID)=NULL;
44 static HRESULT (WINAPI *pDirectSoundCreate)(LPCGUID,LPDIRECTSOUND*,
45 LPUNKNOWN)=NULL;
46
47 static BOOL gotdx8;
48
49 static void IDirectSound_test(LPDIRECTSOUND dso, BOOL initialized,
50 LPCGUID lpGuid)
51 {
52 HRESULT rc;
53 DSCAPS dscaps;
54 int ref;
55 IUnknown * unknown;
56 IDirectSound * ds;
57 IDirectSound8 * ds8;
58 DWORD speaker_config, new_speaker_config;
59
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);
63 if (rc==DS_OK)
64 IDirectSound_Release(unknown);
65
66 rc=IDirectSound_QueryInterface(dso,&IID_IDirectSound,(LPVOID*)&ds);
67 ok(rc==DS_OK,"IDirectSound_QueryInterface(IID_IDirectSound) failed: %08x\n", rc);
68 if (rc==DS_OK)
69 IDirectSound_Release(ds);
70
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);
74 if (rc==DS_OK)
75 IDirectSound8_Release(ds8);
76
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);
82
83 rc=IDirectSound_GetCaps(dso,&dscaps);
84 ok(rc==DSERR_UNINITIALIZED,"IDirectSound_GetCaps() "
85 "should have returned DSERR_UNINITIALIZED, returned: %08x\n", rc);
86
87 rc=IDirectSound_Compact(dso);
88 ok(rc==DSERR_UNINITIALIZED,"IDirectSound_Compact() "
89 "should have returned DSERR_UNINITIALIZED, returned: %08x\n", rc);
90
91 rc=IDirectSound_GetSpeakerConfig(dso,&speaker_config);
92 ok(rc==DSERR_UNINITIALIZED,"IDirectSound_GetSpeakerConfig() "
93 "should have returned DSERR_UNINITIALIZED, returned: %08x\n", rc);
94
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");
100 goto EXIT;
101 } else if (rc==E_FAIL) {
102 trace(" No Device\n");
103 goto EXIT;
104 } else if (rc==DSERR_ALLOCATED) {
105 trace(" Already In Use\n");
106 goto EXIT;
107 }
108 }
109
110 rc=IDirectSound_Initialize(dso,lpGuid);
111 ok(rc==DSERR_ALREADYINITIALIZED, "IDirectSound_Initialize() "
112 "should have returned DSERR_ALREADYINITIALIZED: %08x\n", rc);
113
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);
118
119 ZeroMemory(&dscaps, sizeof(dscaps));
120
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);
125
126 dscaps.dwSize=sizeof(dscaps);
127
128 /* DSOUND: Running on a certified driver */
129 rc=IDirectSound_GetCaps(dso,&dscaps);
130 ok(rc==DS_OK,"IDirectSound_GetCaps() failed: %08x\n",rc);
131
132 rc=IDirectSound_Compact(dso);
133 ok(rc==DSERR_PRIOLEVELNEEDED,"IDirectSound_Compact() failed: %08x\n", rc);
134
135 rc=IDirectSound_SetCooperativeLevel(dso,get_hwnd(),DSSCL_PRIORITY);
136 ok(rc==DS_OK,"IDirectSound_SetCooperativeLevel() failed: %08x\n", rc);
137
138 rc=IDirectSound_Compact(dso);
139 ok(rc==DS_OK,"IDirectSound_Compact() failed: %08x\n",rc);
140
141 rc=IDirectSound_GetSpeakerConfig(dso,0);
142 ok(rc==DSERR_INVALIDPARAM,"IDirectSound_GetSpeakerConfig(NULL) "
143 "should have returned DSERR_INVALIDPARAM, returned: %08x\n", rc);
144
145 rc=IDirectSound_GetSpeakerConfig(dso,&speaker_config);
146 ok(rc==DS_OK,"IDirectSound_GetSpeakerConfig() failed: %08x\n", rc);
147
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);
152 if (rc==DS_OK) {
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);
159 }
160
161 EXIT:
162 ref=IDirectSound_Release(dso);
163 ok(ref==0,"IDirectSound_Release() has %d references, should have 0\n",ref);
164 }
165
166 static void IDirectSound_tests(void)
167 {
168 HRESULT rc;
169 LPDIRECTSOUND dso=NULL;
170 LPCLASSFACTORY cf=NULL;
171
172 trace("Testing IDirectSound\n");
173
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);
178
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);
183
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);
188 if (dso)
189 IDirectSound_test(dso, FALSE, NULL);
190
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);
196 if (dso)
197 IDirectSound_test(dso, FALSE, &DSDEVID_DefaultPlayback);
198
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);
204 if (dso)
205 IDirectSound_test(dso, FALSE, &DSDEVID_DefaultVoicePlayback);
206
207 /* try the COM class factory method of creation with a bad
208 * IID specified */
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);
214
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);
222
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);
227 if (rc==S_OK && dso)
228 IDirectSound_test(dso, TRUE, NULL);
229
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);
236
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);
243
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);
250 }
251
252 static HRESULT test_dsound(LPGUID lpGuid)
253 {
254 HRESULT rc;
255 LPDIRECTSOUND dso=NULL;
256 int ref;
257
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);
262
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);
267 if (rc!=DS_OK)
268 return rc;
269
270 /* Try the enumerated device */
271 IDirectSound_test(dso, TRUE, lpGuid);
272
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);
277 if (dso)
278 IDirectSound_test(dso, FALSE, lpGuid);
279
280 /* Create a DirectSound object */
281 rc=pDirectSoundCreate(lpGuid,&dso,NULL);
282 ok(rc==DS_OK,"DirectSoundCreate() failed: %08x\n",rc);
283 if (rc==DS_OK) {
284 LPDIRECTSOUND dso1=NULL;
285
286 /* Create a second DirectSound object */
287 rc=pDirectSoundCreate(lpGuid,&dso1,NULL);
288 ok(rc==DS_OK,"DirectSoundCreate() failed: %08x\n",rc);
289 if (rc==DS_OK) {
290 /* Release the second DirectSound object */
291 ref=IDirectSound_Release(dso1);
292 ok(ref==0,"IDirectSound_Release() has %d references, should have "
293 "\n",ref);
294 ok(dso!=dso1,"DirectSound objects should be unique: dso=%p,dso1=%p\n",dso,dso1);
295 }
296
297 /* Release the first DirectSound object */
298 ref=IDirectSound_Release(dso);
299 ok(ref==0,"IDirectSound_Release() has %d references, should have 0\n",
300 ref);
301 if (ref!=0)
302 return DSERR_GENERIC;
303 } else
304 return rc;
305
306 /* Create a DirectSound object */
307 rc=pDirectSoundCreate(lpGuid,&dso,NULL);
308 ok(rc==DS_OK,"DirectSoundCreate() failed: %08x\n",rc);
309 if (rc==DS_OK) {
310 LPDIRECTSOUNDBUFFER secondary;
311 DSBUFFERDESC bufdesc;
312 WAVEFORMATEX wfx;
313
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,
319 wfx.nBlockAlign);
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 "
324 "buffer %08x\n",rc);
325 if (rc==DS_OK && secondary!=NULL) {
326 LPDIRECTSOUND3DBUFFER buffer3d;
327 rc=IDirectSound_QueryInterface(secondary, &IID_IDirectSound3DBuffer,
328 (void **)&buffer3d);
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);
335 }
336 ref=IDirectSoundBuffer_AddRef(secondary);
337 ok(ref==2,"IDirectSoundBuffer_AddRef() has %d references, "
338 "should have 2\n",ref);
339 }
340 /* release with buffer */
341 ref=IDirectSound_Release(dso);
342 ok(ref==0,"IDirectSound_Release() has %d references, should have 0\n",
343 ref);
344 if (ref!=0)
345 return DSERR_GENERIC;
346 } else
347 return rc;
348
349 return DS_OK;
350 }
351
352 static HRESULT test_primary(LPGUID lpGuid)
353 {
354 HRESULT rc;
355 LPDIRECTSOUND dso=NULL;
356 LPDIRECTSOUNDBUFFER primary=NULL,second=NULL,third=NULL;
357 DSBUFFERDESC bufdesc;
358 DSCAPS dscaps;
359 WAVEFORMATEX wfx;
360 int ref;
361
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);
366 if (rc!=DS_OK)
367 return rc;
368
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);
374 if (rc!=DS_OK)
375 goto EXIT;
376
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);
381
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);
388
389 /* DSOUND: Error: Invalid size */
390 /* DSOUND: Error: Invalid buffer description */
391 primary=NULL;
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);
398
399 /* DSOUND: Error: DSBCAPS_PRIMARYBUFFER flag with non-NULL lpwfxFormat */
400 /* DSOUND: Error: Invalid buffer description pointer */
401 primary=NULL;
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);
410
411 /* DSOUND: Error: No DSBCAPS_PRIMARYBUFFER flag with NULL lpwfxFormat */
412 /* DSOUND: Error: Invalid buffer description pointer */
413 primary=NULL;
414 ZeroMemory(&bufdesc, sizeof(bufdesc));
415 bufdesc.dwSize=sizeof(bufdesc);
416 bufdesc.dwFlags=0;
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);
422
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);
427 if (rc!=DS_OK)
428 goto EXIT;
429
430 /* Testing the primary buffer */
431 primary=NULL;
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);
442
443 primary=NULL;
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) {
453 LONG vol;
454
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);
465
466 /* Try to duplicate a primary buffer */
467 /* DSOUND: Error: Can't duplicate primary buffers */
468 rc=IDirectSound_DuplicateSoundBuffer(dso,primary,&third);
469 /* rc=0x88780032 */
470 ok(rc!=DS_OK,"IDirectSound_DuplicateSoundBuffer() primary buffer "
471 "should have failed %08x\n",rc);
472
473 rc=IDirectSoundBuffer_GetVolume(primary,&vol);
474 ok(rc==DS_OK,"IDirectSoundBuffer_GetVolume() failed: %08x\n", rc);
475
476 if (winetest_interactive) {
477 trace("Playing a 5 seconds reference tone at the current "
478 "volume.\n");
479 if (rc==DS_OK)
480 trace("(the current volume is %d according to DirectSound)\n",
481 vol);
482 trace("All subsequent tones should be identical to this one.\n");
483 trace("Listen for stutter, changes in pitch, volume, etc.\n");
484 }
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);
487
488 ref=IDirectSoundBuffer_Release(primary);
489 ok(ref==0,"IDirectSoundBuffer_Release() primary has %d references, "
490 "should have 0\n",ref);
491 }
492
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);
497
498 EXIT:
499 ref=IDirectSound_Release(dso);
500 ok(ref==0,"IDirectSound_Release() has %d references, should have 0\n",ref);
501 if (ref!=0)
502 return DSERR_GENERIC;
503
504 return rc;
505 }
506
507 /*
508 * Test the primary buffer at different formats while keeping the
509 * secondary buffer at a constant format.
510 */
511 static HRESULT test_primary_secondary(LPGUID lpGuid)
512 {
513 HRESULT rc;
514 LPDIRECTSOUND dso=NULL;
515 LPDIRECTSOUNDBUFFER primary=NULL,secondary=NULL;
516 DSBUFFERDESC bufdesc;
517 DSCAPS dscaps;
518 WAVEFORMATEX wfx, wfx2;
519 int f,ref;
520
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);
525 if (rc!=DS_OK)
526 return rc;
527
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);
533 if (rc!=DS_OK)
534 goto EXIT;
535
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);
540 if (rc!=DS_OK)
541 goto EXIT;
542
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);
549
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
553 * SetFormat */
554 /* DSOUND: Setting DirectSound cooperative level to
555 * DSSCL_PRIORITY */
556 rc=IDirectSound_SetCooperativeLevel(dso,get_hwnd(),DSSCL_PRIORITY);
557 ok(rc==DS_OK,"IDirectSound_SetCooperativeLevel() failed: %08x\n", rc);
558 if (rc!=DS_OK)
559 goto EXIT;
560
561 init_format(&wfx,WAVE_FORMAT_PCM,formats[f][0],formats[f][1],
562 formats[f][2]);
563 wfx2=wfx;
564 rc=IDirectSoundBuffer_SetFormat(primary,&wfx);
565
566 if (wfx.wBitsPerSample <= 16)
567 ok(rc==DS_OK,"IDirectSoundBuffer_SetFormat(%s) failed: %08x\n",
568 format_string(&wfx), rc);
569 else
570 ok(rc==DS_OK || rc == E_INVALIDARG, "SetFormat (%s) failed: %08x\n",
571 format_string(&wfx), rc);
572
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.
576 */
577 rc=IDirectSoundBuffer_GetFormat(primary,&wfx,sizeof(wfx),NULL);
578 ok(rc==DS_OK,"IDirectSoundBuffer_GetFormat() failed: %08x\n", rc);
579 if (rc==DS_OK &&
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);
591 }
592
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);
597
598 init_format(&wfx2,WAVE_FORMAT_PCM,11025,16,2);
599
600 secondary=NULL;
601 ZeroMemory(&bufdesc, sizeof(bufdesc));
602 bufdesc.dwSize=sizeof(bufdesc);
603 bufdesc.dwFlags=DSBCAPS_GETCURRENTPOSITION2;
604 bufdesc.dwBufferBytes=align(wfx.nAvgBytesPerSec*BUFFER_LEN/1000,
605 wfx.nBlockAlign);
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);
612 }
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);
616
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);
620
621 ref=IDirectSoundBuffer_Release(secondary);
622 ok(ref==0,"IDirectSoundBuffer_Release() has %d references, "
623 "should have 0\n",ref);
624 }
625 }
626
627 ref=IDirectSoundBuffer_Release(primary);
628 ok(ref==0,"IDirectSoundBuffer_Release() primary has %d references, "
629 "should have 0\n",ref);
630 }
631
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);
636
637 EXIT:
638 ref=IDirectSound_Release(dso);
639 ok(ref==0,"IDirectSound_Release() has %d references, should have 0\n",ref);
640 if (ref!=0)
641 return DSERR_GENERIC;
642
643 return rc;
644 }
645
646 static HRESULT test_secondary(LPGUID lpGuid)
647 {
648 HRESULT rc;
649 LPDIRECTSOUND dso=NULL;
650 LPDIRECTSOUNDBUFFER primary=NULL,secondary=NULL;
651 DSBUFFERDESC bufdesc;
652 DSCAPS dscaps;
653 WAVEFORMATEX wfx, wfx1;
654 DWORD f;
655 int ref;
656
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);
661 if (rc!=DS_OK)
662 return rc;
663
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);
669 if (rc!=DS_OK)
670 goto EXIT;
671
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);
676 if (rc!=DS_OK)
677 goto EXIT;
678
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);
685
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);
689 if (rc!=DS_OK)
690 goto EXIT1;
691
692 for (f=0;f<NB_FORMATS;f++) {
693 WAVEFORMATEXTENSIBLE wfxe;
694 init_format(&wfx,WAVE_FORMAT_PCM,formats[f][0],formats[f][1],
695 formats[f][2]);
696 secondary=NULL;
697 ZeroMemory(&bufdesc, sizeof(bufdesc));
698 bufdesc.dwSize=sizeof(bufdesc);
699 bufdesc.dwFlags=DSBCAPS_GETCURRENTPOSITION2;
700 bufdesc.dwBufferBytes=align(wfx.nAvgBytesPerSec*BUFFER_LEN/1000,
701 wfx.nBlockAlign);
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);
707
708 secondary=NULL;
709 ZeroMemory(&bufdesc, sizeof(bufdesc));
710 bufdesc.dwSize=sizeof(bufdesc);
711 bufdesc.dwFlags=DSBCAPS_GETCURRENTPOSITION2;
712 bufdesc.dwBufferBytes=align(wfx.nAvgBytesPerSec*BUFFER_LEN/1000,
713 wfx.nBlockAlign);
714 bufdesc.lpwfxFormat=&wfx;
715 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&secondary,NULL);
716 if (gotdx8 || wfx.wBitsPerSample <= 16)
717 {
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);
724 else
725 ok(rc==DS_OK && secondary!=NULL,
726 "IDirectSound_CreateSoundBuffer() failed to create a secondary buffer %08x\n",rc);
727 }
728 else
729 ok(rc==E_INVALIDARG, "Creating %d bpp buffer on dx < 8 returned: %p %08x\n",
730 wfx.wBitsPerSample, secondary, rc);
731
732 if (!gotdx8)
733 {
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
737 */
738 goto no_wfe;
739 }
740
741 if (secondary)
742 IDirectSoundBuffer_Release(secondary);
743 secondary = NULL;
744
745 bufdesc.lpwfxFormat=(WAVEFORMATEX*)&wfxe;
746 wfxe.Format = wfx;
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);
752
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",
756 rc, secondary);
757 if (secondary)
758 {
759 IDirectSoundBuffer_Release(secondary);
760 secondary=NULL;
761 }
762
763 wfxe.Format.cbSize = sizeof(wfxe) - sizeof(wfx) + 1;
764
765 rc=IDirectSound_CreateSoundBuffer(dso,&bufdesc,&secondary,NULL);
766 ok(((rc==DSERR_CONTROLUNAVAIL || rc==DSERR_INVALIDCALL || rc==DSERR_INVALIDPARAM)
767 && !secondary)
768 || rc==DS_OK, /* 2003 / 2008 */
769 "IDirectSound_CreateSoundBuffer() returned: %08x %p\n",
770 rc, secondary);
771 if (secondary)
772 {
773 IDirectSoundBuffer_Release(secondary);
774 secondary=NULL;
775 }
776
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",
782 rc, secondary);
783 if (secondary)
784 {
785 IDirectSoundBuffer_Release(secondary);
786 secondary=NULL;
787 }
788 wfxe.SubFormat = KSDATAFORMAT_SUBTYPE_PCM;
789
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",
794 rc, secondary);
795 if (secondary)
796 {
797 IDirectSoundBuffer_Release(secondary);
798 secondary=NULL;
799 }
800 --wfxe.Samples.wValidBitsPerSample;
801
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",
806 rc, secondary);
807 if (secondary)
808 {
809 IDirectSoundBuffer_Release(secondary);
810 secondary=NULL;