[AUDIO-BRINGUP]
[reactos.git] / dll / directx / dsound / tests / ds3d8.c
1 /*
2 * Tests the panning and 3D functions of DirectSound
3 *
4 * Part of this test involves playing test tones. But this only makes
5 * sense if someone is going to carefully listen to it, and would only
6 * bother everyone else.
7 * So this is only done if the test is being run in interactive mode.
8 *
9 * Copyright (c) 2002-2004 Francois Gouget
10 *
11 * This library is free software; you can redistribute it and/or
12 * modify it under the terms of the GNU Lesser General Public
13 * License as published by the Free Software Foundation; either
14 * version 2.1 of the License, or (at your option) any later version.
15 *
16 * This library is distributed in the hope that it will be useful,
17 * but WITHOUT ANY WARRANTY; without even the implied warranty of
18 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
19 * Lesser General Public License for more details.
20 *
21 * You should have received a copy of the GNU Lesser General Public
22 * License along with this library; if not, write to the Free Software
23 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
24 */
25
26 #define NONAMELESSSTRUCT
27 #define NONAMELESSUNION
28 #include <windows.h>
29
30 #include <math.h>
31
32 #include "wine/test.h"
33 #include "dsound.h"
34 #include "dxerr8.h"
35
36 #include "dsound_test.h"
37
38 static HRESULT (WINAPI *pDirectSoundCreate8)(LPCGUID,LPDIRECTSOUND8*,LPUNKNOWN)=NULL;
39
40 typedef struct {
41 char* wave;
42 DWORD wave_len;
43
44 LPDIRECTSOUNDBUFFER dsbo;
45 LPWAVEFORMATEX wfx;
46 DWORD buffer_size;
47 DWORD written;
48 DWORD played;
49 DWORD offset;
50 } play_state_t;
51
52 static int buffer_refill8(play_state_t* state, DWORD size)
53 {
54 LPVOID ptr1,ptr2;
55 DWORD len1,len2;
56 HRESULT rc;
57
58 if (size>state->wave_len-state->written)
59 size=state->wave_len-state->written;
60
61 rc=IDirectSoundBuffer_Lock(state->dsbo,state->offset,size,
62 &ptr1,&len1,&ptr2,&len2,0);
63 ok(rc==DS_OK,"IDirectSoundBuffer_Lock() failed: %s\n",
64 DXGetErrorString8(rc));
65 if (rc!=DS_OK)
66 return -1;
67
68 memcpy(ptr1,state->wave+state->written,len1);
69 state->written+=len1;
70 if (ptr2!=NULL) {
71 memcpy(ptr2,state->wave+state->written,len2);
72 state->written+=len2;
73 }
74 state->offset=state->written % state->buffer_size;
75 rc=IDirectSoundBuffer_Unlock(state->dsbo,ptr1,len1,ptr2,len2);
76 ok(rc==DS_OK,"IDirectSoundBuffer_Unlock() failed: %s\n",
77 DXGetErrorString8(rc));
78 if (rc!=DS_OK)
79 return -1;
80 return size;
81 }
82
83 static int buffer_silence8(play_state_t* state, DWORD size)
84 {
85 LPVOID ptr1,ptr2;
86 DWORD len1,len2;
87 HRESULT rc;
88 BYTE s;
89
90 rc=IDirectSoundBuffer_Lock(state->dsbo,state->offset,size,
91 &ptr1,&len1,&ptr2,&len2,0);
92 ok(rc==DS_OK,"IDirectSoundBuffer_Lock() failed: %s\n",
93 DXGetErrorString8(rc));
94 if (rc!=DS_OK)
95 return -1;
96
97 s=(state->wfx->wBitsPerSample==8?0x80:0);
98 memset(ptr1,s,len1);
99 if (ptr2!=NULL) {
100 memset(ptr2,s,len2);
101 }
102 state->offset=(state->offset+size) % state->buffer_size;
103 rc=IDirectSoundBuffer_Unlock(state->dsbo,ptr1,len1,ptr2,len2);
104 ok(rc==DS_OK,"IDirectSoundBuffer_Unlock() failed: %s\n",
105 DXGetErrorString8(rc));
106 if (rc!=DS_OK)
107 return -1;
108 return size;
109 }
110
111 static int buffer_service8(play_state_t* state)
112 {
113 DWORD last_play_pos,play_pos,buf_free;
114 HRESULT rc;
115
116 rc=IDirectSoundBuffer_GetCurrentPosition(state->dsbo,&play_pos,NULL);
117 ok(rc==DS_OK,"IDirectSoundBuffer_GetCurrentPosition() failed: %s\n",
118 DXGetErrorString8(rc));
119 if (rc!=DS_OK) {
120 goto STOP;
121 }
122
123 /* Update the amount played */
124 last_play_pos=state->played % state->buffer_size;
125 if (play_pos<last_play_pos)
126 state->played+=state->buffer_size-last_play_pos+play_pos;
127 else
128 state->played+=play_pos-last_play_pos;
129
130 if (winetest_debug > 1)
131 trace("buf size=%ld last_play_pos=%ld play_pos=%ld played=%ld / %ld\n",
132 state->buffer_size,last_play_pos,play_pos,state->played,
133 state->wave_len);
134
135 if (state->played>state->wave_len)
136 {
137 /* Everything has been played */
138 goto STOP;
139 }
140
141 /* Refill the buffer */
142 if (state->offset<=play_pos)
143 buf_free=play_pos-state->offset;
144 else
145 buf_free=state->buffer_size-state->offset+play_pos;
146
147 if (winetest_debug > 1)
148 trace("offset=%ld free=%ld written=%ld / %ld\n",
149 state->offset,buf_free,state->written,state->wave_len);
150 if (buf_free==0)
151 return 1;
152
153 if (state->written<state->wave_len)
154 {
155 int w=buffer_refill8(state,buf_free);
156 if (w==-1)
157 goto STOP;
158 buf_free-=w;
159 if (state->written==state->wave_len && winetest_debug > 1)
160 trace("last sound byte at %ld\n",
161 (state->written % state->buffer_size));
162 }
163
164 if (buf_free>0) {
165 /* Fill with silence */
166 if (winetest_debug > 1)
167 trace("writing %ld bytes of silence\n",buf_free);
168 if (buffer_silence8(state,buf_free)==-1)
169 goto STOP;
170 }
171 return 1;
172
173 STOP:
174 if (winetest_debug > 1)
175 trace("stopping playback\n");
176 rc=IDirectSoundBuffer_Stop(state->dsbo);
177 ok(rc==DS_OK,"IDirectSoundBuffer_Stop() failed: %s\n",
178 DXGetErrorString8(rc));
179 return 0;
180 }
181
182 void test_buffer8(LPDIRECTSOUND8 dso, LPDIRECTSOUNDBUFFER dsbo,
183 BOOL is_primary, BOOL set_volume, LONG volume,
184 BOOL set_pan, LONG pan, BOOL play, double duration,
185 BOOL buffer3d, LPDIRECTSOUND3DLISTENER listener,
186 BOOL move_listener, BOOL move_sound)
187 {
188 HRESULT rc;
189 DSBCAPS dsbcaps;
190 WAVEFORMATEX wfx,wfx2;
191 DWORD size,status,freq;
192 int ref;
193
194 /* DSOUND: Error: Invalid caps pointer */
195 rc=IDirectSoundBuffer_GetCaps(dsbo,0);
196 ok(rc==DSERR_INVALIDPARAM,"IDirectSoundBuffer_GetCaps() should have "
197 "returned DSERR_INVALIDPARAM, returned: %s\n",DXGetErrorString8(rc));
198
199 ZeroMemory(&dsbcaps, sizeof(dsbcaps));
200
201 /* DSOUND: Error: Invalid caps pointer */
202 rc=IDirectSoundBuffer_GetCaps(dsbo,&dsbcaps);
203 ok(rc==DSERR_INVALIDPARAM,"IDirectSoundBuffer_GetCaps() should have "
204 "returned DSERR_INVALIDPARAM, returned: %s\n",DXGetErrorString8(rc));
205
206 dsbcaps.dwSize=sizeof(dsbcaps);
207 rc=IDirectSoundBuffer_GetCaps(dsbo,&dsbcaps);
208 ok(rc==DS_OK,"IDirectSoundBuffer_GetCaps() failed: %s\n",
209 DXGetErrorString8(rc));
210 if (rc==DS_OK && winetest_debug > 1) {
211 trace(" Caps: flags=0x%08lx size=%ld\n",dsbcaps.dwFlags,
212 dsbcaps.dwBufferBytes);
213 }
214
215 /* Query the format size. Note that it may not match sizeof(wfx) */
216 size=0;
217 rc=IDirectSoundBuffer_GetFormat(dsbo,NULL,0,&size);
218 ok(rc==DS_OK && size!=0,"IDirectSoundBuffer_GetFormat() should have "
219 "returned the needed size: rc=%s size=%ld\n",DXGetErrorString8(rc),size);
220
221 rc=IDirectSoundBuffer_GetFormat(dsbo,&wfx,sizeof(wfx),NULL);
222 ok(rc==DS_OK,"IDirectSoundBuffer_GetFormat() failed: %s\n",
223 DXGetErrorString8(rc));
224 if (rc==DS_OK && winetest_debug > 1) {
225 trace(" Format: %s tag=0x%04x %ldx%dx%d avg.B/s=%ld align=%d\n",
226 is_primary ? "Primary" : "Secondary",
227 wfx.wFormatTag,wfx.nSamplesPerSec,wfx.wBitsPerSample,
228 wfx.nChannels,wfx.nAvgBytesPerSec,wfx.nBlockAlign);
229 }
230
231 /* DSOUND: Error: Invalid frequency buffer */
232 rc=IDirectSoundBuffer_GetFrequency(dsbo,0);
233 ok(rc==DSERR_INVALIDPARAM,"IDirectSoundBuffer_GetFrequency() should have "
234 "returned DSERR_INVALIDPARAM, returned: %s\n",DXGetErrorString8(rc));
235
236 /* DSOUND: Error: Primary buffers don't support CTRLFREQUENCY */
237 rc=IDirectSoundBuffer_GetFrequency(dsbo,&freq);
238 ok((rc==DS_OK && !is_primary) || (rc==DSERR_CONTROLUNAVAIL&&is_primary) ||
239 (rc==DSERR_CONTROLUNAVAIL&&!(dsbcaps.dwFlags&DSBCAPS_CTRLFREQUENCY)),
240 "IDirectSoundBuffer_GetFrequency() failed: %s\n",DXGetErrorString8(rc));
241 if (rc==DS_OK) {
242 ok(freq==wfx.nSamplesPerSec,"The frequency returned by GetFrequency "
243 "%ld does not match the format %ld\n",freq,wfx.nSamplesPerSec);
244 }
245
246 /* DSOUND: Error: Invalid status pointer */
247 rc=IDirectSoundBuffer_GetStatus(dsbo,0);
248 ok(rc==DSERR_INVALIDPARAM,"IDirectSoundBuffer_GetStatus() should have "
249 "returned DSERR_INVALIDPARAM, returned: %s\n",DXGetErrorString8(rc));
250
251 rc=IDirectSoundBuffer_GetStatus(dsbo,&status);
252 ok(rc==DS_OK,"IDirectSoundBuffer_GetStatus() failed: %s\n",
253 DXGetErrorString8(rc));
254 ok(status==0,"status=0x%lx instead of 0\n",status);
255
256 if (is_primary) {
257 /* We must call SetCooperativeLevel to be allowed to call SetFormat */
258 /* DSOUND: Setting DirectSound cooperative level to DSSCL_PRIORITY */
259 rc=IDirectSound8_SetCooperativeLevel(dso,get_hwnd(),DSSCL_PRIORITY);
260 ok(rc==DS_OK,"IDirectSound8_SetCooperativeLevel(DSSCL_PRIORITY) "
261 "failed: %s\n",DXGetErrorString8(rc));
262 if (rc!=DS_OK)
263 return;
264
265 /* DSOUND: Error: Invalid format pointer */
266 rc=IDirectSoundBuffer_SetFormat(dsbo,0);
267 ok(rc==DSERR_INVALIDPARAM,"IDirectSoundBuffer_SetFormat() should have "
268 "returned DSERR_INVALIDPARAM, returned: %s\n",DXGetErrorString8(rc));
269
270 init_format(&wfx2,WAVE_FORMAT_PCM,11025,16,2);
271 rc=IDirectSoundBuffer_SetFormat(dsbo,&wfx2);
272 ok(rc==DS_OK,"IDirectSoundBuffer_SetFormat() failed: %s\n",
273 DXGetErrorString8(rc));
274
275 /* There is no garantee that SetFormat will actually change the
276 * format to what we asked for. It depends on what the soundcard
277 * supports. So we must re-query the format.
278 */
279 rc=IDirectSoundBuffer_GetFormat(dsbo,&wfx,sizeof(wfx),NULL);
280 ok(rc==DS_OK,"IDirectSoundBuffer_GetFormat() failed: %s\n",
281 DXGetErrorString8(rc));
282 if (rc==DS_OK &&
283 (wfx.wFormatTag!=wfx2.wFormatTag ||
284 wfx.nSamplesPerSec!=wfx2.nSamplesPerSec ||
285 wfx.wBitsPerSample!=wfx2.wBitsPerSample ||
286 wfx.nChannels!=wfx2.nChannels)) {
287 trace("Requested format tag=0x%04x %ldx%dx%d avg.B/s=%ld align=%d\n",
288 wfx2.wFormatTag,wfx2.nSamplesPerSec,wfx2.wBitsPerSample,
289 wfx2.nChannels,wfx2.nAvgBytesPerSec,wfx2.nBlockAlign);
290 trace("Got tag=0x%04x %ldx%dx%d avg.B/s=%ld align=%d\n",
291 wfx.wFormatTag,wfx.nSamplesPerSec,wfx.wBitsPerSample,
292 wfx.nChannels,wfx.nAvgBytesPerSec,wfx.nBlockAlign);
293 }
294
295 /* Set the CooperativeLevel back to normal */
296 /* DSOUND: Setting DirectSound cooperative level to DSSCL_NORMAL */
297 rc=IDirectSound8_SetCooperativeLevel(dso,get_hwnd(),DSSCL_NORMAL);
298 ok(rc==DS_OK,"IDirectSound8_SetCooperativeLevel(DSSCL_NORMAL) "
299 "failed: %s\n",DXGetErrorString8(rc));
300 }
301
302 if (play) {
303 play_state_t state;
304 DS3DLISTENER listener_param;
305 LPDIRECTSOUND3DBUFFER buffer=NULL;
306 DS3DBUFFER buffer_param;
307 DWORD start_time,now;
308
309 if (winetest_interactive) {
310 trace(" Playing %g second 440Hz tone at %ldx%dx%d\n", duration,
311 wfx.nSamplesPerSec, wfx.wBitsPerSample,wfx.nChannels);
312 }
313
314 if (is_primary) {
315 /* We must call SetCooperativeLevel to be allowed to call Lock */
316 /* DSOUND: Setting DirectSound cooperative level to
317 * DSSCL_WRITEPRIMARY */
318 rc=IDirectSound8_SetCooperativeLevel(dso,get_hwnd(),
319 DSSCL_WRITEPRIMARY);
320 ok(rc==DS_OK,
321 "IDirectSound8_SetCooperativeLevel(DSSCL_WRITEPRIMARY) failed: "
322 "%s\n",DXGetErrorString8(rc));
323 if (rc!=DS_OK)
324 return;
325 }
326 if (buffer3d) {
327 LPDIRECTSOUNDBUFFER temp_buffer;
328
329 rc=IDirectSoundBuffer_QueryInterface(dsbo,&IID_IDirectSound3DBuffer,
330 (LPVOID *)&buffer);
331 ok(rc==DS_OK,"IDirectSoundBuffer_QueryInterface() failed: %s\n",
332 DXGetErrorString8(rc));
333 if (rc!=DS_OK)
334 return;
335
336 /* check the COM interface */
337 rc=IDirectSoundBuffer_QueryInterface(dsbo, &IID_IDirectSoundBuffer,
338 (LPVOID *)&temp_buffer);
339 ok(rc==DS_OK && temp_buffer!=NULL,
340 "IDirectSoundBuffer_QueryInterface() failed: %s\n",
341 DXGetErrorString8(rc));
342 ok(temp_buffer==dsbo,"COM interface broken: %p != %p\n",
343 temp_buffer,dsbo);
344 ref=IDirectSoundBuffer_Release(temp_buffer);
345 ok(ref==1,"IDirectSoundBuffer_Release() has %d references, "
346 "should have 1\n",ref);
347
348 temp_buffer=NULL;
349 rc=IDirectSound3DBuffer_QueryInterface(dsbo, &IID_IDirectSoundBuffer,
350 (LPVOID *)&temp_buffer);
351 ok(rc==DS_OK && temp_buffer!=NULL,
352 "IDirectSound3DBuffer_QueryInterface() failed: %s\n",
353 DXGetErrorString8(rc));
354 ok(temp_buffer==dsbo,"COM interface broken: %p != %p\n",
355 temp_buffer,dsbo);
356 ref=IDirectSoundBuffer_Release(temp_buffer);
357 ok(ref==1,"IDirectSoundBuffer_Release() has %d references, "
358 "should have 1\n",ref);
359
360 #if 0
361 /* FIXME: this works on windows */
362 ref=IDirectSoundBuffer_Release(dsbo);
363 ok(ref==0,"IDirectSoundBuffer_Release() has %d references, "
364 "should have 0\n",ref);
365
366 rc=IDirectSound3DBuffer_QueryInterface(buffer,
367 &IID_IDirectSoundBuffer,
368 (LPVOID *)&dsbo);
369 ok(rc==DS_OK && dsbo!=NULL,"IDirectSound3DBuffer_QueryInterface() "
370 "failed: %s\n",DXGetErrorString8(rc),
371 #endif
372
373 /* DSOUND: Error: Invalid buffer */
374 rc=IDirectSound3DBuffer_GetAllParameters(buffer,0);
375 ok(rc==DSERR_INVALIDPARAM,"IDirectSound3DBuffer_GetAllParameters() "
376 "failed: %s\n",DXGetErrorString8(rc));
377
378 ZeroMemory(&buffer_param, sizeof(buffer_param));
379
380 /* DSOUND: Error: Invalid buffer */
381 rc=IDirectSound3DBuffer_GetAllParameters(buffer,&buffer_param);
382 ok(rc==DSERR_INVALIDPARAM,"IDirectSound3DBuffer_GetAllParameters() "
383 "failed: %s\n",DXGetErrorString8(rc));
384
385 buffer_param.dwSize=sizeof(buffer_param);
386 rc=IDirectSound3DBuffer_GetAllParameters(buffer,&buffer_param);
387 ok(rc==DS_OK,"IDirectSound3DBuffer_GetAllParameters() failed: %s\n",
388 DXGetErrorString8(rc));
389 }
390 if (set_volume) {
391 if (dsbcaps.dwFlags & DSBCAPS_CTRLVOLUME) {
392 LONG val;
393 rc=IDirectSoundBuffer_GetVolume(dsbo,&val);
394 ok(rc==DS_OK,"IDirectSoundBuffer_GetVolume() failed: %s\n",
395 DXGetErrorString8(rc));
396
397 rc=IDirectSoundBuffer_SetVolume(dsbo,volume);
398 ok(rc==DS_OK,"IDirectSoundBuffer_SetVolume() failed: %s\n",
399 DXGetErrorString8(rc));
400 } else {
401 /* DSOUND: Error: Buffer does not have CTRLVOLUME */
402 rc=IDirectSoundBuffer_GetVolume(dsbo,&volume);
403 ok(rc==DSERR_CONTROLUNAVAIL,"IDirectSoundBuffer_GetVolume() "
404 "should have returned DSERR_CONTROLUNAVAIL, returned: %s\n",
405 DXGetErrorString8(rc));
406 }
407 }
408
409 if (set_pan) {
410 if (dsbcaps.dwFlags & DSBCAPS_CTRLPAN) {
411 LONG val;
412 rc=IDirectSoundBuffer_GetPan(dsbo,&val);
413 ok(rc==DS_OK,"IDirectSoundBuffer_GetPan() failed: %s\n",
414 DXGetErrorString8(rc));
415
416 rc=IDirectSoundBuffer_SetPan(dsbo,pan);
417 ok(rc==DS_OK,"IDirectSoundBuffer_SetPan() failed: %s\n",
418 DXGetErrorString8(rc));
419 } else {
420 /* DSOUND: Error: Buffer does not have CTRLPAN */
421 rc=IDirectSoundBuffer_GetPan(dsbo,&pan);
422 ok(rc==DSERR_CONTROLUNAVAIL,"IDirectSoundBuffer_GetPan() "
423 "should have returned DSERR_CONTROLUNAVAIL, returned: %s\n",
424 DXGetErrorString8(rc));
425 }
426 }
427
428 state.wave=wave_generate_la(&wfx,duration,&state.wave_len);
429
430 state.dsbo=dsbo;
431 state.wfx=&wfx;
432 state.buffer_size=dsbcaps.dwBufferBytes;
433 state.played=state.written=state.offset=0;
434 buffer_refill8(&state,state.buffer_size);
435
436 rc=IDirectSoundBuffer_Play(dsbo,0,0,DSBPLAY_LOOPING);
437 ok(rc==DS_OK,"IDirectSoundBuffer_Play() failed: %s\n",
438 DXGetErrorString8(rc));
439
440 rc=IDirectSoundBuffer_GetStatus(dsbo,&status);
441 ok(rc==DS_OK,"IDirectSoundBuffer_GetStatus() failed: %s\n",
442 DXGetErrorString8(rc));
443 ok(status==(DSBSTATUS_PLAYING|DSBSTATUS_LOOPING),
444 "GetStatus: bad status: %lx\n",status);
445
446 if (listener) {
447 ZeroMemory(&listener_param,sizeof(listener_param));
448 listener_param.dwSize=sizeof(listener_param);
449 rc=IDirectSound3DListener_GetAllParameters(listener,&listener_param);
450 ok(rc==DS_OK,"IDirectSound3dListener_GetAllParameters() "
451 "failed: %s\n",DXGetErrorString8(rc));
452 if (move_listener) {
453 listener_param.vPosition.x = -5.0;
454 listener_param.vVelocity.x = 10.0/duration;
455 }
456 rc=IDirectSound3DListener_SetAllParameters(listener,
457 &listener_param,
458 DS3D_IMMEDIATE);
459 ok(rc==DS_OK,"IDirectSound3dListener_SetPosition() failed: %s\n",
460 DXGetErrorString8(rc));
461 }
462 if (buffer3d) {
463 if (move_sound) {
464 buffer_param.vPosition.x = 100.0;
465 buffer_param.vVelocity.x = -200.0/duration;
466 }
467 buffer_param.flMinDistance = 10;
468 rc=IDirectSound3DBuffer_SetAllParameters(buffer,&buffer_param,
469 DS3D_IMMEDIATE);
470 ok(rc==DS_OK,"IDirectSound3dBuffer_SetPosition() failed: %s\n",
471 DXGetErrorString8(rc));
472 }
473
474 start_time=GetTickCount();
475 while (buffer_service8(&state)) {
476 WaitForSingleObject(GetCurrentProcess(),TIME_SLICE);
477 now=GetTickCount();
478 if (listener && move_listener) {
479 listener_param.vPosition.x = -5.0+10.0*(now-start_time)/
480 1000/duration;
481 if (winetest_debug>2)
482 trace("listener position=%g\n",listener_param.vPosition.x);
483 rc=IDirectSound3DListener_SetPosition(listener,
484 listener_param.vPosition.x,listener_param.vPosition.y,
485 listener_param.vPosition.z,DS3D_IMMEDIATE);
486 ok(rc==DS_OK,"IDirectSound3dListener_SetPosition() failed: "
487 "%s\n",DXGetErrorString8(rc));
488 }
489 if (buffer3d && move_sound) {
490 buffer_param.vPosition.x = 100-200.0*(now-start_time)/
491 1000/duration;
492 if (winetest_debug>2)
493 trace("sound position=%g\n",buffer_param.vPosition.x);
494 rc=IDirectSound3DBuffer_SetPosition(buffer,
495 buffer_param.vPosition.x,buffer_param.vPosition.y,
496 buffer_param.vPosition.z,DS3D_IMMEDIATE);
497 ok(rc==DS_OK,"IDirectSound3dBuffer_SetPosition() failed: %s\n",
498 DXGetErrorString8(rc));
499 }
500 }
501 /* Check the sound duration was within 10% of the expected value */
502 now=GetTickCount();
503 ok(fabs(1000*duration-now+start_time)<=100*duration,
504 "The sound played for %ld ms instead of %g ms\n",
505 now-start_time,1000*duration);
506
507 free(state.wave);
508 if (is_primary) {
509 /* Set the CooperativeLevel back to normal */
510 /* DSOUND: Setting DirectSound cooperative level to DSSCL_NORMAL */
511 rc=IDirectSound8_SetCooperativeLevel(dso,get_hwnd(),DSSCL_NORMAL);
512 ok(rc==DS_OK,"IDirectSound8_SetCooperativeLevel(DSSCL_NORMAL) "
513 "failed: %s\n",DXGetErrorString8(rc));
514 }
515 if (buffer3d) {
516 ref=IDirectSound3DBuffer_Release(buffer);
517 ok(ref==0,"IDirectSound3DBuffer_Release() has %d references, "
518 "should have 0\n",ref);
519 }
520 }
521 }
522
523 static HRESULT test_secondary8(LPGUID lpGuid, int play,
524 int has_3d, int has_3dbuffer,
525 int has_listener, int has_duplicate,
526 int move_listener, int move_sound)
527 {
528 HRESULT rc;
529 LPDIRECTSOUND8 dso=NULL;
530 LPDIRECTSOUNDBUFFER primary=NULL,secondary=NULL;
531 LPDIRECTSOUND3DLISTENER listener=NULL;
532 DSBUFFERDESC bufdesc;
533 WAVEFORMATEX wfx, wfx1;
534 int ref;
535
536 /* Create the DirectSound object */
537 rc=pDirectSoundCreate8(lpGuid,&dso,NULL);
538 ok(rc==DS_OK||rc==DSERR_NODRIVER,"DirectSoundCreate8() failed: %s\n",
539 DXGetErrorString8(rc));
540 if (rc!=DS_OK)
541 return rc;
542
543 /* We must call SetCooperativeLevel before creating primary buffer */
544 /* DSOUND: Setting DirectSound cooperative level to DSSCL_PRIORITY */
545 rc=IDirectSound8_SetCooperativeLevel(dso,get_hwnd(),DSSCL_PRIORITY);
546 ok(rc==DS_OK,"IDirectSound8_SetCooperativeLevel(DSSCL_PRIORITY) failed: "
547 "%s\n",DXGetErrorString8(rc));
548 if (rc!=DS_OK)
549 goto EXIT;
550
551 ZeroMemory(&bufdesc, sizeof(bufdesc));
552 bufdesc.dwSize=sizeof(bufdesc);
553 bufdesc.dwFlags=DSBCAPS_PRIMARYBUFFER;
554 if (has_3d)
555 bufdesc.dwFlags|=DSBCAPS_CTRL3D;
556 else
557 bufdesc.dwFlags|=(DSBCAPS_CTRLVOLUME|DSBCAPS_CTRLPAN);
558 rc=IDirectSound8_CreateSoundBuffer(dso,&bufdesc,&primary,NULL);
559 ok((rc==DS_OK && primary!=NULL) || (rc == DSERR_CONTROLUNAVAIL),
560 "IDirectSound8_CreateSoundBuffer() failed to create a %sprimary buffer: "
561 "%s\n",has_3d?"3D ":"", DXGetErrorString8(rc));
562 if (rc == DSERR_CONTROLUNAVAIL)
563 trace(" No Primary\n");
564 else if (rc==DS_OK && primary!=NULL) {
565 rc=IDirectSoundBuffer_GetFormat(primary,&wfx1,sizeof(wfx1),NULL);
566 ok(rc==DS_OK,"IDirectSoundBuffer8_Getformat() failed: %s\n",
567 DXGetErrorString8(rc));
568 if (rc!=DS_OK)
569 goto EXIT1;
570
571 if (has_listener) {
572 rc=IDirectSoundBuffer_QueryInterface(primary,
573 &IID_IDirectSound3DListener,
574 (void **)&listener);
575 ok(rc==DS_OK && listener!=NULL,
576 "IDirectSoundBuffer_QueryInterface() failed to get a 3D "
577 "listener %s\n",DXGetErrorString8(rc));
578 ref=IDirectSoundBuffer_Release(primary);
579 ok(ref==0,"IDirectSoundBuffer_Release() primary has %d references, "
580 "should have 0\n",ref);
581 if (rc==DS_OK && listener!=NULL) {
582 DS3DLISTENER listener_param;
583 ZeroMemory(&listener_param,sizeof(listener_param));
584 /* DSOUND: Error: Invalid buffer */
585 rc=IDirectSound3DListener_GetAllParameters(listener,0);
586 ok(rc==DSERR_INVALIDPARAM,
587 "IDirectSound3dListener_GetAllParameters() should have "
588 "returned DSERR_INVALIDPARAM, returned: %s\n",
589 DXGetErrorString8(rc));
590
591 /* DSOUND: Error: Invalid buffer */
592 rc=IDirectSound3DListener_GetAllParameters(listener,
593 &listener_param);
594 ok(rc==DSERR_INVALIDPARAM,
595 "IDirectSound3dListener_GetAllParameters() should have "
596 "returned DSERR_INVALIDPARAM, returned: %s\n",
597 DXGetErrorString8(rc));
598
599 listener_param.dwSize=sizeof(listener_param);
600 rc=IDirectSound3DListener_GetAllParameters(listener,
601 &listener_param);
602 ok(rc==DS_OK,"IDirectSound3dListener_GetAllParameters() "
603 "failed: %s\n",DXGetErrorString8(rc));
604 }
605 else
606 goto EXIT;
607 }
608
609 init_format(&wfx,WAVE_FORMAT_PCM,22050,16,2);
610 secondary=NULL;
611 ZeroMemory(&bufdesc, sizeof(bufdesc));
612 bufdesc.dwSize=sizeof(bufdesc);
613 bufdesc.dwFlags=DSBCAPS_GETCURRENTPOSITION2;
614 if (has_3d)
615 bufdesc.dwFlags|=DSBCAPS_CTRL3D;
616 else
617 bufdesc.dwFlags|=
618 (DSBCAPS_CTRLFREQUENCY|DSBCAPS_CTRLVOLUME|DSBCAPS_CTRLPAN);
619 bufdesc.dwBufferBytes=align(wfx.nAvgBytesPerSec*BUFFER_LEN/1000,
620 wfx.nBlockAlign);
621 bufdesc.lpwfxFormat=&wfx;
622 if (has_3d) {
623 /* a stereo 3D buffer should fail */
624 rc=IDirectSound8_CreateSoundBuffer(dso,&bufdesc,&secondary,NULL);
625 ok(rc==DSERR_INVALIDPARAM,
626 "IDirectSound8_CreateSoundBuffer(secondary) should have "
627 "returned DSERR_INVALIDPARAM, returned %s\n",
628 DXGetErrorString8(rc));
629 if (secondary)
630 ref=IDirectSoundBuffer_Release(secondary);
631 init_format(&wfx,WAVE_FORMAT_PCM,22050,16,1);
632 }
633
634 if (winetest_interactive) {
635 trace(" Testing a %s%ssecondary buffer %s%s%s%sat %ldx%dx%d "
636 "with a primary buffer at %ldx%dx%d\n",
637 has_3dbuffer?"3D ":"",
638 has_duplicate?"duplicated ":"",
639 listener!=NULL||move_sound?"with ":"",
640 move_listener?"moving ":"",
641 listener!=NULL?"listener ":"",
642 listener&&move_sound?"and moving sound ":move_sound?
643 "moving sound ":"",
644 wfx.nSamplesPerSec,wfx.wBitsPerSample,wfx.nChannels,
645 wfx1.nSamplesPerSec,wfx1.wBitsPerSample,wfx1.nChannels);
646 }
647 rc=IDirectSound8_CreateSoundBuffer(dso,&bufdesc,&secondary,NULL);
648 ok(rc==DS_OK && secondary!=NULL,"IDirectSound8_CreateSoundBuffer() "
649 "failed to create a %s%ssecondary buffer %s%s%s%sat %ldx%dx%d (%s): %s\n",
650 has_3dbuffer?"3D ":"", has_duplicate?"duplicated ":"",
651 listener!=NULL||move_sound?"with ":"", move_listener?"moving ":"",
652 listener!=NULL?"listener ":"",
653 listener&&move_sound?"and moving sound ":move_sound?
654 "moving sound ":"",
655 wfx.nSamplesPerSec,wfx.wBitsPerSample,wfx.nChannels,
656 getDSBCAPS(bufdesc.dwFlags),DXGetErrorString8(rc));
657 if (rc==DS_OK && secondary!=NULL) {
658 if (!has_3d) {
659 LONG refvol,vol,refpan,pan;
660
661 /* Check the initial secondary buffer's volume and pan */
662 rc=IDirectSoundBuffer_GetVolume(secondary,&vol);
663 ok(rc==DS_OK,"IDirectSoundBuffer_GetVolume(secondary) failed: "
664 "%s\n",DXGetErrorString8(rc));
665 ok(vol==0,"wrong volume for a new secondary buffer: %ld\n",vol);
666 rc=IDirectSoundBuffer_GetPan(secondary,&pan);
667 ok(rc==DS_OK,"IDirectSoundBuffer_GetPan(secondary) failed: "
668 "%s\n",DXGetErrorString8(rc));
669 ok(pan==0,"wrong pan for a new secondary buffer: %ld\n",pan);
670
671 /* Check that changing the secondary buffer's volume and pan
672 * does not impact the primary buffer's volume and pan
673 */
674 rc=IDirectSoundBuffer_GetVolume(primary,&refvol);
675 ok(rc==DS_OK,"IDirectSoundBuffer_GetVolume(primary) failed: "
676 "%s\n",DXGetErrorString8(rc));
677 rc=IDirectSoundBuffer_GetPan(primary,&refpan);
678 ok(rc==DS_OK,"IDirectSoundBuffer_GetPan(primary) failed: "
679 "%s\n",DXGetErrorString8(rc));
680
681 rc=IDirectSoundBuffer_SetVolume(secondary,-1000);
682 ok(rc==DS_OK,"IDirectSoundBuffer_SetVolume(secondary) failed: "
683 "%s\n",DXGetErrorString8(rc));
684 rc=IDirectSoundBuffer_GetVolume(secondary,&vol);
685 ok(rc==DS_OK,"IDirectSoundBuffer_SetVolume(secondary) failed: "
686 "%s\n",DXGetErrorString8(rc));
687 ok(vol==-1000,"secondary: wrong volume %ld instead of -1000\n",
688 vol);
689 rc=IDirectSoundBuffer_SetPan(secondary,-1000);
690 ok(rc==DS_OK,"IDirectSoundBuffer_SetPan(secondary) failed: "
691 "%s\n",DXGetErrorString8(rc));
692 rc=IDirectSoundBuffer_GetPan(secondary,&pan);
693 ok(rc==DS_OK,"IDirectSoundBuffer_SetPan(secondary) failed: "
694 "%s\n",DXGetErrorString8(rc));
695 ok(pan==-1000,"secondary: wrong pan %ld instead of -1000\n",
696 pan);
697
698 rc=IDirectSoundBuffer_GetVolume(primary,&vol);
699 ok(rc==DS_OK,"IDirectSoundBuffer_`GetVolume(primary) failed: i"
700 "%s\n",DXGetErrorString8(rc));
701 ok(vol==refvol,"The primary volume changed from %ld to %ld\n",
702 refvol,vol);
703 rc=IDirectSoundBuffer_GetPan(primary,&pan);
704 ok(rc==DS_OK,"IDirectSoundBuffer_GetPan(primary) failed: "
705 "%s\n",DXGetErrorString8(rc));
706 ok(pan==refpan,"The primary pan changed from %ld to %ld\n",
707 refpan,pan);
708
709 rc=IDirectSoundBuffer_SetVolume(secondary,0);
710 ok(rc==DS_OK,"IDirectSoundBuffer_SetVolume(secondary) failed: "
711 "%s\n",DXGetErrorString8(rc));
712 rc=IDirectSoundBuffer_SetPan(secondary,0);
713 ok(rc==DS_OK,"IDirectSoundBuffer_SetPan(secondary) failed: "
714 "%s\n",DXGetErrorString8(rc));
715 }
716 if (has_duplicate) {
717 LPDIRECTSOUNDBUFFER duplicated=NULL;
718
719 /* DSOUND: Error: Invalid source buffer */
720 rc=IDirectSound8_DuplicateSoundBuffer(dso,0,0);
721 ok(rc==DSERR_INVALIDPARAM,
722 "IDirectSound8_DuplicateSoundBuffer() should have returned "
723 "DSERR_INVALIDPARAM, returned: %s\n",DXGetErrorString8(rc));
724
725 /* DSOUND: Error: Invalid dest buffer */
726 rc=IDirectSound8_DuplicateSoundBuffer(dso,secondary,0);
727 ok(rc==DSERR_INVALIDPARAM,
728 "IDirectSound8_DuplicateSoundBuffer() should have returned "
729 "DSERR_INVALIDPARAM, returned: %s\n",DXGetErrorString8(rc));
730
731 /* DSOUND: Error: Invalid source buffer */
732 rc=IDirectSound8_DuplicateSoundBuffer(dso,0,&duplicated);
733 ok(rc==DSERR_INVALIDPARAM,
734 "IDirectSound8_DuplicateSoundBuffer() should have returned "
735 "DSERR_INVALIDPARAM, returned: %s\n",DXGetErrorString8(rc));
736
737 duplicated=NULL;
738 rc=IDirectSound8_DuplicateSoundBuffer(dso,secondary,
739 &duplicated);
740 ok(rc==DS_OK && duplicated!=NULL,
741 "IDirectSound8_DuplicateSoundBuffer() failed to duplicate "
742 "a secondary buffer: %s\n",DXGetErrorString8(rc));
743
744 if (rc==DS_OK && duplicated!=NULL) {
745 ref=IDirectSoundBuffer_Release(secondary);
746 ok(ref==0,"IDirectSoundBuffer_Release() secondary has %d "
747 "references, should have 0\n",ref);
748 secondary=duplicated;
749 }
750 }
751
752 if (rc==DS_OK && secondary!=NULL) {
753 double duration;
754 duration=(move_listener || move_sound?4.0:1.0);
755 test_buffer8(dso,secondary,0,FALSE,0,FALSE,0,
756 winetest_interactive,duration,has_3dbuffer,
757 listener,move_listener,move_sound);
758 ref=IDirectSoundBuffer_Release(secondary);
759 ok(ref==0,"IDirectSoundBuffer_Release() %s has %d references, "
760 "should have 0\n",has_duplicate?"duplicated":"secondary",
761 ref);
762 }
763 }
764 }
765 EXIT1:
766 if (has_listener) {
767 ref=IDirectSound3DListener_Release(listener);
768 ok(ref==0,"IDirectSound3dListener_Release() listener has %d "
769 "references, should have 0\n",ref);
770 } else {
771 ref=IDirectSoundBuffer_Release(primary);
772 ok(ref==0,"IDirectSoundBuffer_Release() primary has %d references, "
773 "should have 0\n",ref);
774 }
775
776 /* Set the CooperativeLevel back to normal */
777 /* DSOUND: Setting DirectSound cooperative level to DSSCL_NORMAL */
778 rc=IDirectSound8_SetCooperativeLevel(dso,get_hwnd(),DSSCL_NORMAL);
779 ok(rc==DS_OK,"IDirectSound8_SetCooperativeLevel(DSSCL_NORMAL) failed: "
780 "%s\n",DXGetErrorString8(rc));
781
782 EXIT:
783 ref=IDirectSound8_Release(dso);
784 ok(ref==0,"IDirectSound8_Release() has %d references, should have 0\n",ref);
785 if (ref!=0)
786 return DSERR_GENERIC;
787
788 return rc;
789 }
790
791 static HRESULT test_for_driver8(LPGUID lpGuid)
792 {
793 HRESULT rc;
794 LPDIRECTSOUND8 dso=NULL;
795 int ref;
796
797 /* Create the DirectSound object */
798 rc=pDirectSoundCreate8(lpGuid,&dso,NULL);
799 ok(rc==DS_OK||rc==DSERR_NODRIVER||rc==DSERR_ALLOCATED||rc==E_FAIL,
800 "DirectSoundCreate8() failed: %s\n",DXGetErrorString8(rc));
801 if (rc!=DS_OK)
802 return rc;
803
804 ref=IDirectSound8_Release(dso);
805 ok(ref==0,"IDirectSound8_Release() has %d references, should have 0\n",ref);
806 if (ref!=0)
807 return DSERR_GENERIC;
808
809 return rc;
810 }
811
812 static HRESULT test_primary8(LPGUID lpGuid)
813 {
814 HRESULT rc;
815 LPDIRECTSOUND8 dso=NULL;
816 LPDIRECTSOUNDBUFFER primary=NULL;
817 DSBUFFERDESC bufdesc;
818 DSCAPS dscaps;
819 int ref, i;
820
821 /* Create the DirectSound object */
822 rc=pDirectSoundCreate8(lpGuid,&dso,NULL);
823 ok(rc==DS_OK||rc==DSERR_NODRIVER,"DirectSoundCreate8() failed: %s\n",
824 DXGetErrorString8(rc));
825 if (rc!=DS_OK)
826 return rc;
827
828 /* Get the device capabilities */
829 ZeroMemory(&dscaps, sizeof(dscaps));
830 dscaps.dwSize=sizeof(dscaps);
831 rc=IDirectSound8_GetCaps(dso,&dscaps);
832 ok(rc==DS_OK,"IDirectSound8_GetCaps() failed: %s\n",DXGetErrorString8(rc));
833 if (rc!=DS_OK)
834 goto EXIT;
835
836 /* We must call SetCooperativeLevel before calling CreateSoundBuffer */
837 /* DSOUND: Setting DirectSound cooperative level to DSSCL_PRIORITY */
838 rc=IDirectSound8_SetCooperativeLevel(dso,get_hwnd(),DSSCL_PRIORITY);
839 ok(rc==DS_OK,"IDirectSound8_SetCooperativeLevel(DSSCL_PRIORITY) failed: "
840 "%s\n",DXGetErrorString8(rc));
841 if (rc!=DS_OK)
842 goto EXIT;
843
844 /* Testing the primary buffer */
845 primary=NULL;
846 ZeroMemory(&bufdesc, sizeof(bufdesc));
847 bufdesc.dwSize=sizeof(bufdesc);
848 bufdesc.dwFlags=DSBCAPS_PRIMARYBUFFER|DSBCAPS_CTRLVOLUME|DSBCAPS_CTRLPAN;
849 rc=IDirectSound8_CreateSoundBuffer(dso,&bufdesc,&primary,NULL);
850 ok((rc==DS_OK && primary!=NULL) || (rc == DSERR_CONTROLUNAVAIL),
851 "IDirectSound8_CreateSoundBuffer() failed to create a primary buffer: "
852 "%s\n",DXGetErrorString8(rc));
853 if (rc == DSERR_CONTROLUNAVAIL)
854 trace(" No Primary\n");
855 else if (rc==DS_OK && primary!=NULL) {
856 test_buffer8(dso,primary,1,TRUE,0,TRUE,0,winetest_interactive &&
857 !(dscaps.dwFlags & DSCAPS_EMULDRIVER),1.0,0,NULL,0,0);
858 if (winetest_interactive) {
859 LONG volume,pan;
860
861 volume = DSBVOLUME_MAX;
862 for (i = 0; i < 6; i++) {
863 test_buffer8(dso,primary,1,TRUE,volume,TRUE,0,
864 winetest_interactive &&
865 !(dscaps.dwFlags & DSCAPS_EMULDRIVER),
866 1.0,0,NULL,0,0);
867 volume -= ((DSBVOLUME_MAX-DSBVOLUME_MIN) / 40);
868 }
869
870 pan = DSBPAN_LEFT;
871 for (i = 0; i < 7; i++) {
872 test_buffer8(dso,primary,1,TRUE,0,TRUE,pan,
873 winetest_interactive &&
874 !(dscaps.dwFlags & DSCAPS_EMULDRIVER),1.0,0,0,0,0);
875 pan += ((DSBPAN_RIGHT-DSBPAN_LEFT) / 6);
876 }
877 }
878 ref=IDirectSoundBuffer_Release(primary);
879 ok(ref==0,"IDirectSoundBuffer_Release() primary has %d references, "
880 "should have 0\n",ref);
881 }
882
883 /* Set the CooperativeLevel back to normal */
884 /* DSOUND: Setting DirectSound cooperative level to DSSCL_NORMAL */
885 rc=IDirectSound8_SetCooperativeLevel(dso,get_hwnd(),DSSCL_NORMAL);
886 ok(rc==DS_OK,"IDirectSound8_SetCooperativeLevel(DSSCL_NORMAL) failed: "
887 "%s\n",DXGetErrorString8(rc));
888
889 EXIT:
890 ref=IDirectSound8_Release(dso);
891 ok(ref==0,"IDirectSound8_Release() has %d references, should have 0\n",ref);
892 if (ref!=0)
893 return DSERR_GENERIC;
894
895 return rc;
896 }
897
898 static HRESULT test_primary_3d8(LPGUID lpGuid)
899 {
900 HRESULT rc;
901 LPDIRECTSOUND8 dso=NULL;
902 LPDIRECTSOUNDBUFFER primary=NULL;
903 DSBUFFERDESC bufdesc;
904 DSCAPS dscaps;
905 int ref;
906
907 /* Create the DirectSound object */
908 rc=pDirectSoundCreate8(lpGuid,&dso,NULL);
909 ok(rc==DS_OK||rc==DSERR_NODRIVER,"DirectSoundCreate8() failed: %s\n",
910 DXGetErrorString8(rc));
911 if (rc!=DS_OK)
912 return rc;
913
914 /* Get the device capabilities */
915 ZeroMemory(&dscaps, sizeof(dscaps));
916 dscaps.dwSize=sizeof(dscaps);
917 rc=IDirectSound8_GetCaps(dso,&dscaps);
918 ok(rc==DS_OK,"IDirectSound8_GetCaps failed: %s\n",DXGetErrorString8(rc));
919 if (rc!=DS_OK)
920 goto EXIT;
921
922 /* We must call SetCooperativeLevel before calling CreateSoundBuffer */
923 /* DSOUND: Setting DirectSound cooperative level to DSSCL_PRIORITY */
924 rc=IDirectSound8_SetCooperativeLevel(dso,get_hwnd(),DSSCL_PRIORITY);
925 ok(rc==DS_OK,"IDirectSound8_SetCooperativeLevel(DSSCL_PRIORITY) failed: "
926 "%s\n",DXGetErrorString8(rc));
927 if (rc!=DS_OK)
928 goto EXIT;
929
930 primary=NULL;
931 ZeroMemory(&bufdesc, sizeof(bufdesc));
932 bufdesc.dwSize=sizeof(bufdesc);
933 bufdesc.dwFlags=DSBCAPS_PRIMARYBUFFER;
934 rc=IDirectSound8_CreateSoundBuffer(dso,&bufdesc,&primary,NULL);
935 ok(rc==DS_OK && primary!=NULL,"IDirectSound8_CreateSoundBuffer() failed "
936 "to create a primary buffer: %s\n",DXGetErrorString8(rc));
937 if (rc==DS_OK && primary!=NULL) {
938 ref=IDirectSoundBuffer_Release(primary);
939 ok(ref==0,"IDirectSoundBuffer_Release() primary has %d references, "
940 "should have 0\n",ref);
941 primary=NULL;
942 ZeroMemory(&bufdesc, sizeof(bufdesc));
943 bufdesc.dwSize=sizeof(bufdesc);
944 bufdesc.dwFlags=DSBCAPS_PRIMARYBUFFER|DSBCAPS_CTRL3D;
945 rc=IDirectSound8_CreateSoundBuffer(dso,&bufdesc,&primary,NULL);
946 ok(rc==DS_OK && primary!=NULL,"IDirectSound8_CreateSoundBuffer() "
947 "failed to create a 3D primary buffer: %s\n",DXGetErrorString8(rc));
948 if (rc==DS_OK && primary!=NULL) {
949 test_buffer8(dso,primary,1,FALSE,0,FALSE,0,
950 winetest_interactive &&
951 !(dscaps.dwFlags & DSCAPS_EMULDRIVER),1.0,0,0,0,0);
952 ref=IDirectSoundBuffer_Release(primary);
953 ok(ref==0,"IDirectSoundBuffer_Release() primary has %d references, "
954 "should have 0\n",ref);
955 }
956 }
957 /* Set the CooperativeLevel back to normal */
958 /* DSOUND: Setting DirectSound cooperative level to DSSCL_NORMAL */
959 rc=IDirectSound8_SetCooperativeLevel(dso,get_hwnd(),DSSCL_NORMAL);
960 ok(rc==DS_OK,"IDirectSound8_SetCooperativeLevel(DSSCL_NORMAL) failed: "
961 "%s\n",DXGetErrorString8(rc));
962
963 EXIT:
964 ref=IDirectSound8_Release(dso);
965 ok(ref==0,"IDirectSound8_Release() has %d references, should have 0\n",ref);
966 if (ref!=0)
967 return DSERR_GENERIC;
968
969 return rc;
970 }
971
972 static HRESULT test_primary_3d_with_listener8(LPGUID lpGuid)
973 {
974 HRESULT rc;
975 LPDIRECTSOUND8 dso=NULL;
976 LPDIRECTSOUNDBUFFER primary=NULL;
977 DSBUFFERDESC bufdesc;
978 DSCAPS dscaps;
979 int ref;
980
981 /* Create the DirectSound object */
982 rc=pDirectSoundCreate8(lpGuid,&dso,NULL);
983 ok(rc==DS_OK||rc==DSERR_NODRIVER,"DirectSoundCreate8() failed: %s\n",
984 DXGetErrorString8(rc));
985 if (rc!=DS_OK)
986 return rc;
987
988 /* Get the device capabilities */
989 ZeroMemory(&dscaps, sizeof(dscaps));
990 dscaps.dwSize=sizeof(dscaps);
991 rc=IDirectSound8_GetCaps(dso,&dscaps);
992 ok(rc==DS_OK,"IDirectSound8_GetCaps() failed: %s\n",DXGetErrorString8(rc));
993 if (rc!=DS_OK)
994 goto EXIT;
995
996 /* We must call SetCooperativeLevel before calling CreateSoundBuffer */
997 /* DSOUND: Setting DirectSound cooperative level to DSSCL_PRIORITY */
998 rc=IDirectSound8_SetCooperativeLevel(dso,get_hwnd(),DSSCL_PRIORITY);
999 ok(rc==DS_OK,"IDirectSound8_SetCooperativeLevel(DSSCL_PRIORITY) failed: "
1000 "%s\n",DXGetErrorString8(rc));
1001 if (rc!=DS_OK)
1002 goto EXIT;
1003 primary=NULL;
1004 ZeroMemory(&bufdesc, sizeof(bufdesc));
1005 bufdesc.dwSize=sizeof(bufdesc);
1006 bufdesc.dwFlags=DSBCAPS_PRIMARYBUFFER|DSBCAPS_CTRL3D;
1007 rc=IDirectSound8_CreateSoundBuffer(dso,&bufdesc,&primary,NULL);
1008 ok(rc==DS_OK && primary!=NULL,"IDirectSound8_CreateSoundBuffer() failed "
1009 "to create a 3D primary buffer %s\n",DXGetErrorString8(rc));
1010 if (rc==DS_OK && primary!=NULL) {
1011 LPDIRECTSOUND3DLISTENER listener=NULL;
1012 rc=IDirectSoundBuffer_QueryInterface(primary,
1013 &IID_IDirectSound3DListener,
1014 (void **)&listener);
1015 ok(rc==DS_OK && listener!=NULL,"IDirectSoundBuffer_QueryInterface() "
1016 "failed to get a 3D listener: %s\n",DXGetErrorString8(rc));
1017 if (rc==DS_OK && listener!=NULL) {
1018 LPDIRECTSOUNDBUFFER temp_buffer=NULL;
1019
1020 /* Checking the COM interface */
1021 rc=IDirectSoundBuffer_QueryInterface(primary,
1022 &IID_IDirectSoundBuffer,
1023 (LPVOID *)&temp_buffer);
1024 ok(rc==DS_OK && temp_buffer!=NULL,
1025 "IDirectSoundBuffer_QueryInterface() failed: %s\n",
1026 DXGetErrorString8(rc));
1027 ok(temp_buffer==primary,"COM interface broken: %p != %p\n",temp_buffer,primary);
1028 if (rc==DS_OK && temp_buffer!=NULL) {
1029 ref=IDirectSoundBuffer_Release(temp_buffer);
1030 ok(ref==1,"IDirectSoundBuffer_Release() has %d references, "
1031 "should have 1\n",ref);
1032
1033 temp_buffer=NULL;
1034 rc=IDirectSound3DListener_QueryInterface(listener,
1035 &IID_IDirectSoundBuffer,(LPVOID *)&temp_buffer);
1036 ok(rc==DS_OK && temp_buffer!=NULL,
1037 "IDirectSoundBuffer_QueryInterface() failed: %s\n",
1038 DXGetErrorString8(rc));
1039 ok(temp_buffer==primary,"COM interface broken: %p != %p\n",temp_buffer,primary);
1040 ref=IDirectSoundBuffer_Release(temp_buffer);
1041 ok(ref==1,"IDirectSoundBuffer_Release() has %d references, "
1042 "should have 1\n",ref);
1043
1044 /* Testing the buffer */
1045 test_buffer8(dso,primary,1,FALSE,0,FALSE,0,
1046 winetest_interactive &&
1047 !(dscaps.dwFlags & DSCAPS_EMULDRIVER),
1048 1.0,0,listener,0,0);
1049 }
1050
1051 /* Testing the reference counting */
1052 ref=IDirectSound3DListener_Release(listener);
1053 ok(ref==0,"IDirectSound3DListener_Release() listener has %d "
1054 "references, should have 0\n",ref);
1055 }
1056
1057 /* Testing the reference counting */
1058 ref=IDirectSoundBuffer_Release(primary);
1059 ok(ref==0,"IDirectSoundBuffer_Release() primary has %d references, "
1060 "should have 0\n",ref);
1061 }
1062
1063 EXIT:
1064 ref=IDirectSound8_Release(dso);
1065 ok(ref==0,"IDirectSound8_Release() has %d references, should have 0\n",ref);
1066 if (ref!=0)
1067 return DSERR_GENERIC;
1068
1069 return rc;
1070 }
1071
1072 static BOOL WINAPI dsenum_callback(LPGUID lpGuid, LPCSTR lpcstrDescription,
1073 LPCSTR lpcstrModule, LPVOID lpContext)
1074 {
1075 HRESULT rc;
1076 trace("*** Testing %s - %s ***\n",lpcstrDescription,lpcstrModule);
1077
1078 rc = test_for_driver8(lpGuid);
1079 if (rc == DSERR_NODRIVER) {
1080 trace(" No Driver\n");
1081 return 1;
1082 } else if (rc == DSERR_ALLOCATED) {
1083 trace(" Already In Use\n");
1084 return 1;
1085 } else if (rc == E_FAIL) {
1086 trace(" No Device\n");
1087 return 1;
1088 }
1089
1090 trace(" Testing the primary buffer\n");
1091 test_primary8(lpGuid);
1092
1093 trace(" Testing 3D primary buffer\n");
1094 test_primary_3d8(lpGuid);
1095
1096 trace(" Testing 3D primary buffer with listener\n");
1097 test_primary_3d_with_listener8(lpGuid);
1098
1099 /* Testing secondary buffers */
1100 test_secondary8(lpGuid,winetest_interactive,0,0,0,0,0,0);
1101 test_secondary8(lpGuid,winetest_interactive,0,0,0,1,0,0);
1102
1103 /* Testing 3D secondary buffers */
1104 test_secondary8(lpGuid,winetest_interactive,1,0,0,0,0,0);
1105 test_secondary8(lpGuid,winetest_interactive,1,1,0,0,0,0);
1106 test_secondary8(lpGuid,winetest_interactive,1,1,0,1,0,0);
1107 test_secondary8(lpGuid,winetest_interactive,1,0,1,0,0,0);
1108 test_secondary8(lpGuid,winetest_interactive,1,0,1,1,0,0);
1109 test_secondary8(lpGuid,winetest_interactive,1,1,1,0,0,0);
1110 test_secondary8(lpGuid,winetest_interactive,1,1,1,1,0,0);
1111 test_secondary8(lpGuid,winetest_interactive,1,1,1,0,1,0);
1112 test_secondary8(lpGuid,winetest_interactive,1,1,1,0,0,1);
1113 test_secondary8(lpGuid,winetest_interactive,1,1,1,0,1,1);
1114
1115 return 1;
1116 }
1117
1118 static void ds3d8_tests(void)
1119 {
1120 HRESULT rc;
1121 rc=DirectSoundEnumerateA(&dsenum_callback,NULL);
1122 ok(rc==DS_OK,"DirectSoundEnumerateA() failed: %s\n",DXGetErrorString8(rc));
1123 }
1124
1125 START_TEST(ds3d8)
1126 {
1127 HMODULE hDsound;
1128
1129 CoInitialize(NULL);
1130
1131 hDsound = LoadLibraryA("dsound.dll");
1132 if (!hDsound) {
1133 trace("dsound.dll not found\n");
1134 return;
1135 }
1136
1137 trace("DLL Version: %s\n", get_file_version("dsound.dll"));
1138
1139 pDirectSoundCreate8 = (void*)GetProcAddress(hDsound, "DirectSoundCreate8");
1140 if (!pDirectSoundCreate8) {
1141 trace("ds3d8 test skipped\n");
1142 return;
1143 }
1144
1145 ds3d8_tests();
1146
1147 CoUninitialize();
1148 }