2 * Copyright 2010 Maarten Lankhorst for CodeWeavers
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19 /* This test is for audio capture specific mechanisms
21 * - IAudioClient with eCapture and IAudioCaptureClient
26 #include "wine/test.h"
36 #include "mmdeviceapi.h"
37 #include "audioclient.h"
39 #define NULL_PTR_ERR MAKE_HRESULT(SEVERITY_ERROR, FACILITY_WIN32, RPC_X_NULL_REF_POINTER)
41 static IMMDevice
*dev
= NULL
;
43 static void test_uninitialized(IAudioClient
*ac
)
49 HANDLE handle
= CreateEventW(NULL
, FALSE
, FALSE
, NULL
);
52 hr
= IAudioClient_GetBufferSize(ac
, &num
);
53 ok(hr
== AUDCLNT_E_NOT_INITIALIZED
, "Uninitialized GetBufferSize call returns %08x\n", hr
);
55 hr
= IAudioClient_GetStreamLatency(ac
, &t1
);
56 ok(hr
== AUDCLNT_E_NOT_INITIALIZED
, "Uninitialized GetStreamLatency call returns %08x\n", hr
);
58 hr
= IAudioClient_GetCurrentPadding(ac
, &num
);
59 ok(hr
== AUDCLNT_E_NOT_INITIALIZED
, "Uninitialized GetCurrentPadding call returns %08x\n", hr
);
61 hr
= IAudioClient_Start(ac
);
62 ok(hr
== AUDCLNT_E_NOT_INITIALIZED
, "Uninitialized Start call returns %08x\n", hr
);
64 hr
= IAudioClient_Stop(ac
);
65 ok(hr
== AUDCLNT_E_NOT_INITIALIZED
, "Uninitialized Stop call returns %08x\n", hr
);
67 hr
= IAudioClient_Reset(ac
);
68 ok(hr
== AUDCLNT_E_NOT_INITIALIZED
, "Uninitialized Reset call returns %08x\n", hr
);
70 hr
= IAudioClient_SetEventHandle(ac
, handle
);
71 ok(hr
== AUDCLNT_E_NOT_INITIALIZED
, "Uninitialized SetEventHandle call returns %08x\n", hr
);
73 hr
= IAudioClient_GetService(ac
, &IID_IAudioStreamVolume
, (void**)&unk
);
74 ok(hr
== AUDCLNT_E_NOT_INITIALIZED
, "Uninitialized GetService call returns %08x\n", hr
);
79 static void test_capture(IAudioClient
*ac
, HANDLE handle
, WAVEFORMATEX
*wfx
)
81 IAudioCaptureClient
*acc
;
83 UINT32 frames
, next
, pad
, sum
= 0;
87 REFERENCE_TIME period
;
89 hr
= IAudioClient_GetService(ac
, &IID_IAudioCaptureClient
, (void**)&acc
);
90 ok(hr
== S_OK
, "IAudioClient_GetService(IID_IAudioCaptureClient) returns %08x\n", hr
);
95 data
= (void*)0xdeadf00d;
97 pos
= qpc
= 0xdeadbeef;
98 hr
= IAudioCaptureClient_GetBuffer(acc
, &data
, &frames
, &flags
, &pos
, &qpc
);
99 ok(hr
== AUDCLNT_S_BUFFER_EMPTY
, "Initial IAudioCaptureClient_GetBuffer returns %08x\n", hr
);
101 /* should be empty right after start. Otherwise consume one packet */
103 hr
= IAudioCaptureClient_ReleaseBuffer(acc
, frames
);
104 ok(hr
== S_OK
, "Releasing buffer returns %08x\n", hr
);
108 data
= (void*)0xdeadf00d;
110 pos
= qpc
= 0xdeadbeef;
111 hr
= IAudioCaptureClient_GetBuffer(acc
, &data
, &frames
, &flags
, &pos
, &qpc
);
112 ok(hr
== AUDCLNT_S_BUFFER_EMPTY
, "Initial IAudioCaptureClient_GetBuffer returns %08x\n", hr
);
115 if(hr
== AUDCLNT_S_BUFFER_EMPTY
){
116 ok(!frames
, "frames changed to %u\n", frames
);
117 ok(data
== (void*)0xdeadf00d, "data changed to %p\n", data
);
118 ok(flags
== 0xabadcafe, "flags changed to %x\n", flags
);
119 ok(pos
== 0xdeadbeef, "position changed to %u\n", (UINT
)pos
);
120 ok(qpc
== 0xdeadbeef, "timer changed to %u\n", (UINT
)qpc
);
122 /* GetNextPacketSize yields 0 if no data is yet available
123 * it is not constantly period_size * SamplesPerSec */
124 hr
= IAudioCaptureClient_GetNextPacketSize(acc
, &next
);
125 ok(hr
== S_OK
, "IAudioCaptureClient_GetNextPacketSize returns %08x\n", hr
);
126 ok(!next
, "GetNextPacketSize %u\n", next
);
129 hr
= IAudioCaptureClient_ReleaseBuffer(acc
, frames
);
130 ok(hr
== S_OK
, "Releasing buffer returns %08x\n", hr
);
133 ok(ResetEvent(handle
), "ResetEvent\n");
135 hr
= IAudioCaptureClient_GetNextPacketSize(acc
, &next
);
136 ok(hr
== S_OK
, "IAudioCaptureClient_GetNextPacketSize returns %08x\n", hr
);
138 hr
= IAudioClient_GetCurrentPadding(ac
, &pad
);
139 ok(hr
== S_OK
, "GetCurrentPadding call returns %08x\n", hr
);
140 ok(next
== pad
, "GetNextPacketSize %u vs. GCP %u\n", next
, pad
);
141 /* later GCP will grow, while GNPS is 0 or period size */
143 hr
= IAudioCaptureClient_GetNextPacketSize(acc
, NULL
);
144 ok(hr
== E_POINTER
, "IAudioCaptureClient_GetNextPacketSize(NULL) returns %08x\n", hr
);
146 data
= (void*)0xdeadf00d;
149 hr
= IAudioCaptureClient_GetBuffer(acc
, &data
, NULL
, NULL
, NULL
, NULL
);
150 ok(hr
== E_POINTER
, "IAudioCaptureClient_GetBuffer(data, NULL, NULL) returns %08x\n", hr
);
152 hr
= IAudioCaptureClient_GetBuffer(acc
, NULL
, &frames
, NULL
, NULL
, NULL
);
153 ok(hr
== E_POINTER
, "IAudioCaptureClient_GetBuffer(NULL, &frames, NULL) returns %08x\n", hr
);
155 hr
= IAudioCaptureClient_GetBuffer(acc
, NULL
, NULL
, &flags
, NULL
, NULL
);
156 ok(hr
== E_POINTER
, "IAudioCaptureClient_GetBuffer(NULL, NULL, &flags) returns %08x\n", hr
);
158 hr
= IAudioCaptureClient_GetBuffer(acc
, &data
, &frames
, NULL
, NULL
, NULL
);
159 ok(hr
== E_POINTER
, "IAudioCaptureClient_GetBuffer(&ata, &frames, NULL) returns %08x\n", hr
);
160 ok((DWORD_PTR
)data
== 0xdeadf00d, "data is reset to %p\n", data
);
161 ok(frames
== 0xdeadbeef, "frames is reset to %08x\n", frames
);
162 ok(flags
== 0xabadcafe, "flags is reset to %08x\n", flags
);
164 hr
= IAudioClient_GetDevicePeriod(ac
, &period
, NULL
);
165 ok(hr
== S_OK
, "GetDevicePeriod failed: %08x\n", hr
);
166 period
= MulDiv(period
, wfx
->nSamplesPerSec
, 10000000); /* as in render.c */
168 ok(WaitForSingleObject(handle
, 1000) == WAIT_OBJECT_0
, "Waiting on event handle failed!\n");
170 data
= (void*)0xdeadf00d;
171 hr
= IAudioCaptureClient_GetBuffer(acc
, &data
, &frames
, &flags
, &pos
, &qpc
);
172 ok(hr
== S_OK
|| hr
== AUDCLNT_S_BUFFER_EMPTY
, "Valid IAudioCaptureClient_GetBuffer returns %08x\n", hr
);
174 ok(frames
, "Amount of frames locked is 0!\n");
175 /* broken: some w7 machines return pad == 0 and DATA_DISCONTINUITY here,
176 * AUDCLNT_S_BUFFER_EMPTY above, yet pos == 1-2 * period rather than 0 */
177 ok(pos
== sum
|| broken(pos
== period
|| pos
== 2*period
),
178 "Position %u expected %u\n", (UINT
)pos
, sum
);
180 }else if (hr
== AUDCLNT_S_BUFFER_EMPTY
){
181 ok(!frames
, "Amount of frames locked with empty buffer is %u!\n", frames
);
182 ok(data
== (void*)0xdeadf00d, "No data changed to %p\n", data
);
185 trace("Wait'ed position %d pad %u flags %x, amount of frames locked: %u\n",
186 hr
==S_OK
? (UINT
)pos
: -1, pad
, flags
, frames
);
188 hr
= IAudioCaptureClient_GetNextPacketSize(acc
, &next
);
189 ok(hr
== S_OK
, "IAudioCaptureClient_GetNextPacketSize returns %08x\n", hr
);
190 ok(next
== frames
, "GetNextPacketSize %u vs. GetBuffer %u\n", next
, frames
);
192 hr
= IAudioCaptureClient_ReleaseBuffer(acc
, frames
);
193 ok(hr
== S_OK
, "Releasing buffer returns %08x\n", hr
);
195 hr
= IAudioCaptureClient_ReleaseBuffer(acc
, 0);
196 ok(hr
== S_OK
, "Releasing 0 returns %08x\n", hr
);
198 hr
= IAudioCaptureClient_GetNextPacketSize(acc
, &next
);
199 ok(hr
== S_OK
, "IAudioCaptureClient_GetNextPacketSize returns %08x\n", hr
);
202 hr
= IAudioCaptureClient_ReleaseBuffer(acc
, frames
);
203 ok(hr
== AUDCLNT_E_OUT_OF_ORDER
, "Releasing buffer twice returns %08x\n", hr
);
207 Sleep(350); /* for sure there's data now */
209 hr
= IAudioClient_GetCurrentPadding(ac
, &pad
);
210 ok(hr
== S_OK
, "GetCurrentPadding call returns %08x\n", hr
);
212 /** GetNextPacketSize
213 * returns either 0 or one period worth of frames
214 * whereas GetCurrentPadding grows when input is not consumed. */
215 hr
= IAudioCaptureClient_GetNextPacketSize(acc
, &next
);
216 ok(hr
== S_OK
, "IAudioCaptureClient_GetNextPacketSize returns %08x\n", hr
);
217 ok(next
< pad
, "GetNextPacketSize %u vs. GCP %u\n", next
, pad
);
219 hr
= IAudioCaptureClient_GetBuffer(acc
, &data
, &frames
, &flags
, &pos
, &qpc
);
220 ok(hr
== S_OK
, "Valid IAudioCaptureClient_GetBuffer returns %08x\n", hr
);
221 ok(next
== frames
, "GetNextPacketSize %u vs. GetBuffer %u\n", next
, frames
);
224 UINT32 frames2
= frames
;
226 ok(frames
, "Amount of frames locked is 0!\n");
227 ok(pos
== sum
, "Position %u expected %u\n", (UINT
)pos
, sum
);
229 hr
= IAudioCaptureClient_ReleaseBuffer(acc
, 0);
230 ok(hr
== S_OK
, "Releasing 0 returns %08x\n", hr
);
232 /* GCP did not decrement, no data consumed */
233 hr
= IAudioClient_GetCurrentPadding(ac
, &frames
);
234 ok(hr
== S_OK
, "GetCurrentPadding call returns %08x\n", hr
);
235 ok(frames
== pad
|| frames
== pad
+ next
/* concurrent feeder */,
236 "GCP %u past ReleaseBuffer(0) initially %u\n", frames
, pad
);
238 /* should re-get the same data */
239 hr
= IAudioCaptureClient_GetBuffer(acc
, &data
, &frames
, &flags
, &pos2
, &qpc2
);
240 ok(hr
== S_OK
, "Valid IAudioCaptureClient_GetBuffer returns %08x\n", hr
);
241 ok(frames2
== frames
, "GetBuffer after ReleaseBuffer(0) %u/%u\n", frames2
, frames
);
242 ok(pos2
== pos
, "Position after ReleaseBuffer(0) %u/%u\n", (UINT
)pos2
, (UINT
)pos
);
243 todo_wine
ok(qpc2
== qpc
, "HPC after ReleaseBuffer(0) %u vs. %u\n", (UINT
)qpc2
, (UINT
)qpc
);
246 /* trace after the GCP test because log output to MS-DOS console disturbs timing */
247 trace("Sleep.1 position %d pad %u flags %x, amount of frames locked: %u\n",
248 hr
==S_OK
? (UINT
)pos
: -1, pad
, flags
, frames
);
251 UINT32 frames2
= 0xabadcafe;
252 BYTE
*data2
= (void*)0xdeadf00d;
255 ok(pos
== sum
, "Position %u expected %u\n", (UINT
)pos
, sum
);
257 pos
= qpc
= 0xdeadbeef;
258 hr
= IAudioCaptureClient_GetBuffer(acc
, &data2
, &frames2
, &flags
, &pos
, &qpc
);
259 ok(hr
== AUDCLNT_E_OUT_OF_ORDER
, "Out of order IAudioCaptureClient_GetBuffer returns %08x\n", hr
);
260 ok(frames2
== 0xabadcafe, "Out of order frames changed to %x\n", frames2
);
261 ok(data2
== (void*)0xdeadf00d, "Out of order data changed to %p\n", data2
);
262 ok(flags
== 0xabadcafe, "Out of order flags changed to %x\n", flags
);
263 ok(pos
== 0xdeadbeef, "Out of order position changed to %x\n", (UINT
)pos
);
264 ok(qpc
== 0xdeadbeef, "Out of order timer changed to %x\n", (UINT
)qpc
);
266 hr
= IAudioCaptureClient_ReleaseBuffer(acc
, frames
+1);
267 ok(hr
== AUDCLNT_E_INVALID_SIZE
, "Releasing buffer+1 returns %08x\n", hr
);
269 hr
= IAudioCaptureClient_ReleaseBuffer(acc
, 1);
270 ok(hr
== AUDCLNT_E_INVALID_SIZE
, "Releasing 1 returns %08x\n", hr
);
272 hr
= IAudioClient_Reset(ac
);
273 ok(hr
== AUDCLNT_E_NOT_STOPPED
, "Reset failed: %08x\n", hr
);
276 hr
= IAudioCaptureClient_ReleaseBuffer(acc
, frames
);
277 ok(hr
== S_OK
, "Releasing buffer returns %08x\n", hr
);
281 hr
= IAudioCaptureClient_ReleaseBuffer(acc
, frames
);
282 ok(hr
== AUDCLNT_E_OUT_OF_ORDER
, "Releasing buffer twice returns %08x\n", hr
);
286 ok(next
== frames
, "GetNextPacketSize %u vs. GetDevicePeriod %u\n", next
, frames
);
288 /* GetBufferSize is not a multiple of the period size! */
289 hr
= IAudioClient_GetBufferSize(ac
, &next
);
290 ok(hr
== S_OK
, "GetBufferSize failed: %08x\n", hr
);
291 trace("GetBufferSize %u period size %u\n", next
, frames
);
293 Sleep(400); /* overrun */
295 hr
= IAudioClient_GetCurrentPadding(ac
, &pad
);
296 ok(hr
== S_OK
, "GetCurrentPadding call returns %08x\n", hr
);
298 hr
= IAudioCaptureClient_GetBuffer(acc
, &data
, &frames
, &flags
, &pos
, &qpc
);
299 ok(hr
== S_OK
, "Valid IAudioCaptureClient_GetBuffer returns %08x\n", hr
);
301 trace("Overrun position %d pad %u flags %x, amount of frames locked: %u\n",
302 hr
==S_OK
? (UINT
)pos
: -1, pad
, flags
, frames
);
305 /* The discontinuity is reported here, but is this an old or new packet? */
306 todo_wine
ok(flags
& AUDCLNT_BUFFERFLAGS_DATA_DISCONTINUITY
, "expect DISCONTINUITY %x\n", flags
);
307 ok(pad
== next
, "GCP %u vs. BufferSize %u\n", (UINT32
)pad
, next
);
309 /* Native's position is one period further than what we read.
310 * Perhaps that's precisely the meaning of DATA_DISCONTINUITY:
311 * signal when the position jump left a gap. */
312 todo_wine
ok(pos
== sum
+ frames
, "Position %u gap %d\n",
313 (UINT
)pos
, (UINT
)pos
- sum
);
314 if(flags
& AUDCLNT_BUFFERFLAGS_DATA_DISCONTINUITY
)
318 hr
= IAudioCaptureClient_ReleaseBuffer(acc
, frames
);
319 ok(hr
== S_OK
, "Releasing buffer returns %08x\n", hr
);
322 hr
= IAudioClient_GetCurrentPadding(ac
, &pad
);
323 ok(hr
== S_OK
, "GetCurrentPadding call returns %08x\n", hr
);
325 hr
= IAudioCaptureClient_GetBuffer(acc
, &data
, &frames
, &flags
, &pos
, &qpc
);
326 ok(hr
== S_OK
, "Valid IAudioCaptureClient_GetBuffer returns %08x\n", hr
);
328 trace("Cont'ed position %d pad %u flags %x, amount of frames locked: %u\n",
329 hr
==S_OK
? (UINT
)pos
: -1, pad
, flags
, frames
);
332 ok(pos
== sum
, "Position %u expected %u\n", (UINT
)pos
, sum
);
333 ok(!flags
, "flags %u\n", flags
);
335 hr
= IAudioCaptureClient_ReleaseBuffer(acc
, frames
);
336 ok(hr
== S_OK
, "Releasing buffer returns %08x\n", hr
);
340 hr
= IAudioClient_Stop(ac
);
341 ok(hr
== S_OK
, "Stop on a started stream returns %08x\n", hr
);
343 hr
= IAudioClient_Start(ac
);
344 ok(hr
== S_OK
, "Start on a stopped stream returns %08x\n", hr
);
346 hr
= IAudioCaptureClient_GetBuffer(acc
, &data
, &frames
, &flags
, &pos
, &qpc
);
347 ok(hr
== S_OK
, "Valid IAudioCaptureClient_GetBuffer returns %08x\n", hr
);
349 hr
= IAudioClient_GetCurrentPadding(ac
, &pad
);
350 ok(hr
== S_OK
, "GetCurrentPadding call returns %08x\n", hr
);
352 trace("Restart position %d pad %u flags %x, amount of frames locked: %u\n",
353 hr
==S_OK
? (UINT
)pos
: -1, pad
, flags
, frames
);
354 ok(pad
> sum
, "restarted GCP %u\n", pad
); /* GCP is still near buffer size */
357 ok(pos
== sum
, "Position %u expected %u\n", (UINT
)pos
, sum
);
358 ok(!flags
, "flags %u\n", flags
);
360 hr
= IAudioCaptureClient_ReleaseBuffer(acc
, frames
);
361 ok(hr
== S_OK
, "Releasing buffer returns %08x\n", hr
);
365 hr
= IAudioClient_Stop(ac
);
366 ok(hr
== S_OK
, "Stop on a started stream returns %08x\n", hr
);
368 hr
= IAudioClient_Reset(ac
);
369 ok(hr
== S_OK
, "Reset on a stopped stream returns %08x\n", hr
);
372 hr
= IAudioClient_Start(ac
);
373 ok(hr
== S_OK
, "Start on a stopped stream returns %08x\n", hr
);
375 hr
= IAudioClient_GetCurrentPadding(ac
, &pad
);
376 ok(hr
== S_OK
, "GetCurrentPadding call returns %08x\n", hr
);
379 hr
= IAudioCaptureClient_GetBuffer(acc
, &data
, &frames
, &flags
, &pos
, &qpc
);
380 ok(hr
== AUDCLNT_S_BUFFER_EMPTY
|| /*PulseAudio*/hr
== S_OK
,
381 "Initial IAudioCaptureClient_GetBuffer returns %08x\n", hr
);
383 trace("Reset position %d pad %u flags %x, amount of frames locked: %u\n",
384 hr
==S_OK
? (UINT
)pos
: -1, pad
, flags
, frames
);
387 /* Only PulseAudio goes here; despite snd_pcm_drop it manages
388 * to fill GetBufferSize with a single snd_pcm_read */
389 trace("Test marked todo: only PulseAudio gets here\n");
390 todo_wine
ok(flags
& AUDCLNT_BUFFERFLAGS_DATA_DISCONTINUITY
, "expect DISCONTINUITY %x\n", flags
);
391 /* Reset zeroes padding, not the position */
392 ok(pos
>= sum
, "Position %u last %u\n", (UINT
)pos
, sum
);
393 /*sum = pos; check after next GetBuffer */
395 hr
= IAudioCaptureClient_ReleaseBuffer(acc
, frames
);
396 ok(hr
== S_OK
, "Releasing buffer returns %08x\n", hr
);
399 else if(hr
== AUDCLNT_S_BUFFER_EMPTY
){
400 ok(!pad
, "resetted GCP %u\n", pad
);
404 hr
= IAudioClient_GetCurrentPadding(ac
, &pad
);
405 ok(hr
== S_OK
, "GetCurrentPadding call returns %08x\n", hr
);
407 hr
= IAudioCaptureClient_GetBuffer(acc
, &data
, &frames
, &flags
, &pos
, &qpc
);
408 ok(hr
== S_OK
, "Valid IAudioCaptureClient_GetBuffer returns %08x\n", hr
);
409 trace("Running position %d pad %u flags %x, amount of frames locked: %u\n",
410 hr
==S_OK
? (UINT
)pos
: -1, pad
, flags
, frames
);
413 /* Some w7 machines signal DATA_DISCONTINUITY here following the
414 * previous AUDCLNT_S_BUFFER_EMPTY, others not. What logic? */
415 ok(pos
>= sum
, "Position %u gap %d\n", (UINT
)pos
, (UINT
)pos
- sum
);
416 IAudioCaptureClient_ReleaseBuffer(acc
, frames
);
419 hr
= IAudioClient_Stop(ac
);
420 ok(hr
== S_OK
, "Stop failed: %08x\n", hr
);
422 ok(ResetEvent(handle
), "ResetEvent\n");
424 /* Still receiving events! */
425 r
= WaitForSingleObject(handle
, 20);
426 todo_wine
ok(r
== WAIT_OBJECT_0
, "Wait(event) after Stop gave %x\n", r
);
428 hr
= IAudioClient_Reset(ac
);
429 ok(hr
== S_OK
, "Reset failed: %08x\n", hr
);
431 ok(ResetEvent(handle
), "ResetEvent\n");
433 r
= WaitForSingleObject(handle
, 120);
434 todo_wine
ok(r
== WAIT_OBJECT_0
, "Wait(event) after Reset gave %x\n", r
);
436 hr
= IAudioClient_SetEventHandle(ac
, NULL
);
437 ok(hr
== E_INVALIDARG
, "SetEventHandle(NULL) returns %08x\n", hr
);
439 r
= WaitForSingleObject(handle
, 70);
440 todo_wine
ok(r
== WAIT_OBJECT_0
, "Wait(NULL event) gave %x\n", r
);
442 hr
= IAudioClient_Start(ac
);
443 ok(hr
== S_OK
, "Start failed: %08x\n", hr
);
445 IAudioCaptureClient_Release(acc
);
448 static void test_audioclient(void)
454 WAVEFORMATEX
*pwfx
, *pwfx2
;
455 REFERENCE_TIME t1
, t2
;
458 hr
= IMMDevice_Activate(dev
, &IID_IAudioClient
, CLSCTX_INPROC_SERVER
,
460 ok(hr
== S_OK
, "Activation failed with %08x\n", hr
);
464 handle
= CreateEventW(NULL
, FALSE
, FALSE
, NULL
);
466 hr
= IAudioClient_QueryInterface(ac
, &IID_IUnknown
, NULL
);
467 ok(hr
== E_POINTER
, "QueryInterface(NULL) returned %08x\n", hr
);
469 unk
= (void*)(LONG_PTR
)0x12345678;
470 hr
= IAudioClient_QueryInterface(ac
, &IID_NULL
, (void**)&unk
);
471 ok(hr
== E_NOINTERFACE
, "QueryInterface(IID_NULL) returned %08x\n", hr
);
472 ok(!unk
, "QueryInterface(IID_NULL) returned non-null pointer %p\n", unk
);
474 hr
= IAudioClient_QueryInterface(ac
, &IID_IUnknown
, (void**)&unk
);
475 ok(hr
== S_OK
, "QueryInterface(IID_IUnknown) returned %08x\n", hr
);
478 ref
= IUnknown_Release(unk
);
479 ok(ref
== 1, "Released count is %u\n", ref
);
482 hr
= IAudioClient_QueryInterface(ac
, &IID_IAudioClient
, (void**)&unk
);
483 ok(hr
== S_OK
, "QueryInterface(IID_IAudioClient) returned %08x\n", hr
);
486 ref
= IUnknown_Release(unk
);
487 ok(ref
== 1, "Released count is %u\n", ref
);
490 hr
= IAudioClient_GetDevicePeriod(ac
, NULL
, NULL
);
491 ok(hr
== E_POINTER
, "Invalid GetDevicePeriod call returns %08x\n", hr
);
493 hr
= IAudioClient_GetDevicePeriod(ac
, &t1
, NULL
);
494 ok(hr
== S_OK
, "Valid GetDevicePeriod call returns %08x\n", hr
);
496 hr
= IAudioClient_GetDevicePeriod(ac
, NULL
, &t2
);
497 ok(hr
== S_OK
, "Valid GetDevicePeriod call returns %08x\n", hr
);
499 hr
= IAudioClient_GetDevicePeriod(ac
, &t1
, &t2
);
500 ok(hr
== S_OK
, "Valid GetDevicePeriod call returns %08x\n", hr
);
501 trace("Returned periods: %u.%04u ms %u.%04u ms\n",
502 (UINT
)(t1
/10000), (UINT
)(t1
% 10000),
503 (UINT
)(t2
/10000), (UINT
)(t2
% 10000));
505 hr
= IAudioClient_GetMixFormat(ac
, NULL
);
506 ok(hr
== E_POINTER
, "GetMixFormat returns %08x\n", hr
);
508 hr
= IAudioClient_GetMixFormat(ac
, &pwfx
);
509 ok(hr
== S_OK
, "Valid GetMixFormat returns %08x\n", hr
);
513 trace("pwfx: %p\n", pwfx
);
514 trace("Tag: %04x\n", pwfx
->wFormatTag
);
515 trace("bits: %u\n", pwfx
->wBitsPerSample
);
516 trace("chan: %u\n", pwfx
->nChannels
);
517 trace("rate: %u\n", pwfx
->nSamplesPerSec
);
518 trace("align: %u\n", pwfx
->nBlockAlign
);
519 trace("extra: %u\n", pwfx
->cbSize
);
520 ok(pwfx
->wFormatTag
== WAVE_FORMAT_EXTENSIBLE
, "wFormatTag is %x\n", pwfx
->wFormatTag
);
521 if (pwfx
->wFormatTag
== WAVE_FORMAT_EXTENSIBLE
)
523 WAVEFORMATEXTENSIBLE
*pwfxe
= (void*)pwfx
;
524 trace("Res: %u\n", pwfxe
->Samples
.wReserved
);
525 trace("Mask: %x\n", pwfxe
->dwChannelMask
);
527 IsEqualGUID(&pwfxe
->SubFormat
, &KSDATAFORMAT_SUBTYPE_PCM
)?"PCM":
528 (IsEqualGUID(&pwfxe
->SubFormat
,
529 &KSDATAFORMAT_SUBTYPE_IEEE_FLOAT
)?"FLOAT":"Other"));
532 hr
= IAudioClient_IsFormatSupported(ac
, AUDCLNT_SHAREMODE_SHARED
, pwfx
, &pwfx2
);
533 ok(hr
== S_OK
, "Valid IsFormatSupported(Shared) call returns %08x\n", hr
);
534 ok(pwfx2
== NULL
, "pwfx2 is non-null\n");
535 CoTaskMemFree(pwfx2
);
537 hr
= IAudioClient_IsFormatSupported(ac
, AUDCLNT_SHAREMODE_SHARED
, NULL
, NULL
);
538 ok(hr
== E_POINTER
, "IsFormatSupported(NULL) call returns %08x\n", hr
);
540 hr
= IAudioClient_IsFormatSupported(ac
, AUDCLNT_SHAREMODE_SHARED
, pwfx
, NULL
);
541 ok(hr
== E_POINTER
, "IsFormatSupported(Shared,NULL) call returns %08x\n", hr
);
543 hr
= IAudioClient_IsFormatSupported(ac
, AUDCLNT_SHAREMODE_EXCLUSIVE
, pwfx
, NULL
);
544 ok(hr
== S_OK
|| hr
== AUDCLNT_E_UNSUPPORTED_FORMAT
, "IsFormatSupported(Exclusive) call returns %08x\n", hr
);
546 hr
= IAudioClient_IsFormatSupported(ac
, AUDCLNT_SHAREMODE_EXCLUSIVE
, pwfx
, &pwfx2
);
547 ok(hr
== S_OK
|| hr
== AUDCLNT_E_UNSUPPORTED_FORMAT
, "IsFormatSupported(Exclusive) call returns %08x\n", hr
);
548 ok(pwfx2
== NULL
, "pwfx2 non-null on exclusive IsFormatSupported\n");
550 hr
= IAudioClient_IsFormatSupported(ac
, 0xffffffff, pwfx
, NULL
);
551 ok(hr
== E_INVALIDARG
/*w32*/ ||
552 broken(hr
== AUDCLNT_E_UNSUPPORTED_FORMAT
/*w64 response from exclusive mode driver */),
553 "IsFormatSupported(0xffffffff) call returns %08x\n", hr
);
556 test_uninitialized(ac
);
558 hr
= IAudioClient_Initialize(ac
, 3, 0, 5000000, 0, pwfx
, NULL
);
559 ok(hr
== AUDCLNT_E_NOT_INITIALIZED
, "Initialize with invalid sharemode returns %08x\n", hr
);
561 hr
= IAudioClient_Initialize(ac
, AUDCLNT_SHAREMODE_SHARED
, 0xffffffff, 5000000, 0, pwfx
, NULL
);
562 ok(hr
== E_INVALIDARG
|| hr
== AUDCLNT_E_INVALID_STREAM_FLAG
, "Initialize with invalid flags returns %08x\n", hr
);
564 /* A period != 0 is ignored and the call succeeds.
565 * Since we can only initialize successfully once, skip those tests.
567 hr
= IAudioClient_Initialize(ac
, AUDCLNT_SHAREMODE_SHARED
, 0, 5000000, 0, NULL
, NULL
);
568 ok(hr
== E_POINTER
, "Initialize with null format returns %08x\n", hr
);
570 hr
= IAudioClient_Initialize(ac
, AUDCLNT_SHAREMODE_SHARED
, AUDCLNT_STREAMFLAGS_EVENTCALLBACK
, 4987654, 0, pwfx
, NULL
);
571 ok(hr
== S_OK
, "Valid Initialize returns %08x\n", hr
);
575 skip("Cannot initialize %08x, remainder of tests is useless\n", hr
);
580 hr
= IAudioClient_GetStreamLatency(ac
, NULL
);
581 ok(hr
== E_POINTER
, "GetStreamLatency(NULL) call returns %08x\n", hr
);
583 hr
= IAudioClient_GetStreamLatency(ac
, &t1
);
584 ok(hr
== S_OK
, "Valid GetStreamLatency call returns %08x\n", hr
);
585 trace("Returned latency: %u.%04u ms\n",
586 (UINT
)(t1
/10000), (UINT
)(t1
% 10000));
588 hr
= IAudioClient_Initialize(ac
, AUDCLNT_SHAREMODE_SHARED
, 0, 5000000, 0, pwfx
, NULL
);
589 ok(hr
== AUDCLNT_E_ALREADY_INITIALIZED
, "Calling Initialize twice returns %08x\n", hr
);
591 hr
= IAudioClient_SetEventHandle(ac
, NULL
);
592 ok(hr
== E_INVALIDARG
, "SetEventHandle(NULL) returns %08x\n", hr
);
594 hr
= IAudioClient_Start(ac
);
595 ok(hr
== AUDCLNT_E_EVENTHANDLE_NOT_SET
, "Start before SetEventHandle returns %08x\n", hr
);
597 hr
= IAudioClient_SetEventHandle(ac
, handle
);
598 ok(hr
== S_OK
, "SetEventHandle returns %08x\n", hr
);
600 hr
= IAudioClient_Reset(ac
);
601 ok(hr
== S_OK
, "Reset on a resetted stream returns %08x\n", hr
);
603 hr
= IAudioClient_Stop(ac
);
604 ok(hr
== S_FALSE
, "Stop on a stopped stream returns %08x\n", hr
);
606 hr
= IAudioClient_Start(ac
);
607 ok(hr
== S_OK
, "Start on a stopped stream returns %08x\n", hr
);
609 test_capture(ac
, handle
, pwfx
);
611 IAudioClient_Release(ac
);
616 static void test_streamvolume(void)
619 IAudioStreamVolume
*asv
;
625 hr
= IMMDevice_Activate(dev
, &IID_IAudioClient
, CLSCTX_INPROC_SERVER
,
627 ok(hr
== S_OK
, "Activation failed with %08x\n", hr
);
631 hr
= IAudioClient_GetMixFormat(ac
, &fmt
);
632 ok(hr
== S_OK
, "GetMixFormat failed: %08x\n", hr
);
634 hr
= IAudioClient_Initialize(ac
, AUDCLNT_SHAREMODE_SHARED
, 0, 5000000,
636 ok(hr
== S_OK
, "Initialize failed: %08x\n", hr
);
640 hr
= IAudioClient_GetService(ac
, &IID_IAudioStreamVolume
, (void**)&asv
);
641 ok(hr
== S_OK
, "GetService failed: %08x\n", hr
);
643 hr
= IAudioStreamVolume_GetChannelCount(asv
, NULL
);
644 ok(hr
== E_POINTER
, "GetChannelCount gave wrong error: %08x\n", hr
);
646 hr
= IAudioStreamVolume_GetChannelCount(asv
, &chans
);
647 ok(hr
== S_OK
, "GetChannelCount failed: %08x\n", hr
);
648 ok(chans
== fmt
->nChannels
, "GetChannelCount gave wrong number of channels: %d\n", chans
);
650 hr
= IAudioStreamVolume_GetChannelVolume(asv
, fmt
->nChannels
, NULL
);
651 ok(hr
== E_POINTER
, "GetChannelCount gave wrong error: %08x\n", hr
);
653 hr
= IAudioStreamVolume_GetChannelVolume(asv
, fmt
->nChannels
, &vol
);
654 ok(hr
== E_INVALIDARG
, "GetChannelCount gave wrong error: %08x\n", hr
);
656 hr
= IAudioStreamVolume_GetChannelVolume(asv
, 0, NULL
);
657 ok(hr
== E_POINTER
, "GetChannelCount gave wrong error: %08x\n", hr
);
659 hr
= IAudioStreamVolume_GetChannelVolume(asv
, 0, &vol
);
660 ok(hr
== S_OK
, "GetChannelCount failed: %08x\n", hr
);
661 ok(vol
== 1.f
, "Channel volume was not 1: %f\n", vol
);
663 hr
= IAudioStreamVolume_SetChannelVolume(asv
, fmt
->nChannels
, -1.f
);
664 ok(hr
== E_INVALIDARG
, "SetChannelVolume gave wrong error: %08x\n", hr
);
666 hr
= IAudioStreamVolume_SetChannelVolume(asv
, 0, -1.f
);
667 ok(hr
== E_INVALIDARG
, "SetChannelVolume gave wrong error: %08x\n", hr
);
669 hr
= IAudioStreamVolume_SetChannelVolume(asv
, 0, 2.f
);
670 ok(hr
== E_INVALIDARG
, "SetChannelVolume gave wrong error: %08x\n", hr
);
672 hr
= IAudioStreamVolume_SetChannelVolume(asv
, 0, 0.2f
);
673 ok(hr
== S_OK
, "SetChannelVolume failed: %08x\n", hr
);
675 hr
= IAudioStreamVolume_GetChannelVolume(asv
, 0, &vol
);
676 ok(hr
== S_OK
, "GetChannelCount failed: %08x\n", hr
);
677 ok(fabsf(vol
- 0.2f
) < 0.05f
, "Channel volume wasn't 0.2: %f\n", vol
);
679 hr
= IAudioStreamVolume_GetAllVolumes(asv
, 0, NULL
);
680 ok(hr
== E_POINTER
, "GetAllVolumes gave wrong error: %08x\n", hr
);
682 hr
= IAudioStreamVolume_GetAllVolumes(asv
, fmt
->nChannels
, NULL
);
683 ok(hr
== E_POINTER
, "GetAllVolumes gave wrong error: %08x\n", hr
);
685 vols
= HeapAlloc(GetProcessHeap(), 0, fmt
->nChannels
* sizeof(float));
686 ok(vols
!= NULL
, "HeapAlloc failed\n");
688 hr
= IAudioStreamVolume_GetAllVolumes(asv
, fmt
->nChannels
- 1, vols
);
689 ok(hr
== E_INVALIDARG
, "GetAllVolumes gave wrong error: %08x\n", hr
);
691 hr
= IAudioStreamVolume_GetAllVolumes(asv
, fmt
->nChannels
, vols
);
692 ok(hr
== S_OK
, "GetAllVolumes failed: %08x\n", hr
);
693 ok(fabsf(vols
[0] - 0.2f
) < 0.05f
, "Channel 0 volume wasn't 0.2: %f\n", vol
);
694 for(i
= 1; i
< fmt
->nChannels
; ++i
)
695 ok(vols
[i
] == 1.f
, "Channel %d volume is not 1: %f\n", i
, vols
[i
]);
697 hr
= IAudioStreamVolume_SetAllVolumes(asv
, 0, NULL
);
698 ok(hr
== E_POINTER
, "SetAllVolumes gave wrong error: %08x\n", hr
);
700 hr
= IAudioStreamVolume_SetAllVolumes(asv
, fmt
->nChannels
, NULL
);
701 ok(hr
== E_POINTER
, "SetAllVolumes gave wrong error: %08x\n", hr
);
703 hr
= IAudioStreamVolume_SetAllVolumes(asv
, fmt
->nChannels
- 1, vols
);
704 ok(hr
== E_INVALIDARG
, "SetAllVolumes gave wrong error: %08x\n", hr
);
706 hr
= IAudioStreamVolume_SetAllVolumes(asv
, fmt
->nChannels
, vols
);
707 ok(hr
== S_OK
, "SetAllVolumes failed: %08x\n", hr
);
709 HeapFree(GetProcessHeap(), 0, vols
);
710 IAudioStreamVolume_Release(asv
);
711 IAudioClient_Release(ac
);
715 static void test_channelvolume(void)
718 IChannelAudioVolume
*acv
;
724 hr
= IMMDevice_Activate(dev
, &IID_IAudioClient
, CLSCTX_INPROC_SERVER
,
726 ok(hr
== S_OK
, "Activation failed with %08x\n", hr
);
730 hr
= IAudioClient_GetMixFormat(ac
, &fmt
);
731 ok(hr
== S_OK
, "GetMixFormat failed: %08x\n", hr
);
733 hr
= IAudioClient_Initialize(ac
, AUDCLNT_SHAREMODE_SHARED
,
734 AUDCLNT_STREAMFLAGS_NOPERSIST
, 5000000, 0, fmt
, NULL
);
735 ok(hr
== S_OK
, "Initialize failed: %08x\n", hr
);
737 hr
= IAudioClient_GetService(ac
, &IID_IChannelAudioVolume
, (void**)&acv
);
738 ok(hr
== S_OK
, "GetService failed: %08x\n", hr
);
742 hr
= IChannelAudioVolume_GetChannelCount(acv
, NULL
);
743 ok(hr
== NULL_PTR_ERR
, "GetChannelCount gave wrong error: %08x\n", hr
);
745 hr
= IChannelAudioVolume_GetChannelCount(acv
, &chans
);
746 ok(hr
== S_OK
, "GetChannelCount failed: %08x\n", hr
);
747 ok(chans
== fmt
->nChannels
, "GetChannelCount gave wrong number of channels: %d\n", chans
);
749 hr
= IChannelAudioVolume_GetChannelVolume(acv
, fmt
->nChannels
, NULL
);
750 ok(hr
== NULL_PTR_ERR
, "GetChannelCount gave wrong error: %08x\n", hr
);
752 hr
= IChannelAudioVolume_GetChannelVolume(acv
, fmt
->nChannels
, &vol
);
753 ok(hr
== E_INVALIDARG
, "GetChannelCount gave wrong error: %08x\n", hr
);
755 hr
= IChannelAudioVolume_GetChannelVolume(acv
, 0, NULL
);
756 ok(hr
== NULL_PTR_ERR
, "GetChannelCount gave wrong error: %08x\n", hr
);
758 hr
= IChannelAudioVolume_GetChannelVolume(acv
, 0, &vol
);
759 ok(hr
== S_OK
, "GetChannelCount failed: %08x\n", hr
);
760 ok(vol
== 1.f
, "Channel volume was not 1: %f\n", vol
);
762 hr
= IChannelAudioVolume_SetChannelVolume(acv
, fmt
->nChannels
, -1.f
, NULL
);
763 ok(hr
== E_INVALIDARG
, "SetChannelVolume gave wrong error: %08x\n", hr
);
765 hr
= IChannelAudioVolume_SetChannelVolume(acv
, 0, -1.f
, NULL
);
766 ok(hr
== E_INVALIDARG
, "SetChannelVolume gave wrong error: %08x\n", hr
);
768 hr
= IChannelAudioVolume_SetChannelVolume(acv
, 0, 2.f
, NULL
);
769 ok(hr
== E_INVALIDARG
, "SetChannelVolume gave wrong error: %08x\n", hr
);
771 hr
= IChannelAudioVolume_SetChannelVolume(acv
, 0, 0.2f
, NULL
);
772 ok(hr
== S_OK
, "SetChannelVolume failed: %08x\n", hr
);
774 hr
= IChannelAudioVolume_GetChannelVolume(acv
, 0, &vol
);
775 ok(hr
== S_OK
, "GetChannelCount failed: %08x\n", hr
);
776 ok(fabsf(vol
- 0.2f
) < 0.05f
, "Channel volume wasn't 0.2: %f\n", vol
);
778 hr
= IChannelAudioVolume_GetAllVolumes(acv
, 0, NULL
);
779 ok(hr
== NULL_PTR_ERR
, "GetAllVolumes gave wrong error: %08x\n", hr
);
781 hr
= IChannelAudioVolume_GetAllVolumes(acv
, fmt
->nChannels
, NULL
);
782 ok(hr
== NULL_PTR_ERR
, "GetAllVolumes gave wrong error: %08x\n", hr
);
784 vols
= HeapAlloc(GetProcessHeap(), 0, fmt
->nChannels
* sizeof(float));
785 ok(vols
!= NULL
, "HeapAlloc failed\n");
787 hr
= IChannelAudioVolume_GetAllVolumes(acv
, fmt
->nChannels
- 1, vols
);
788 ok(hr
== E_INVALIDARG
, "GetAllVolumes gave wrong error: %08x\n", hr
);
790 hr
= IChannelAudioVolume_GetAllVolumes(acv
, fmt
->nChannels
, vols
);
791 ok(hr
== S_OK
, "GetAllVolumes failed: %08x\n", hr
);
792 ok(fabsf(vols
[0] - 0.2f
) < 0.05f
, "Channel 0 volume wasn't 0.2: %f\n", vol
);
793 for(i
= 1; i
< fmt
->nChannels
; ++i
)
794 ok(vols
[i
] == 1.f
, "Channel %d volume is not 1: %f\n", i
, vols
[i
]);
796 hr
= IChannelAudioVolume_SetAllVolumes(acv
, 0, NULL
, NULL
);
797 ok(hr
== NULL_PTR_ERR
, "SetAllVolumes gave wrong error: %08x\n", hr
);
799 hr
= IChannelAudioVolume_SetAllVolumes(acv
, fmt
->nChannels
, NULL
, NULL
);
800 ok(hr
== NULL_PTR_ERR
, "SetAllVolumes gave wrong error: %08x\n", hr
);
802 hr
= IChannelAudioVolume_SetAllVolumes(acv
, fmt
->nChannels
- 1, vols
, NULL
);
803 ok(hr
== E_INVALIDARG
, "SetAllVolumes gave wrong error: %08x\n", hr
);
805 hr
= IChannelAudioVolume_SetAllVolumes(acv
, fmt
->nChannels
, vols
, NULL
);
806 ok(hr
== S_OK
, "SetAllVolumes failed: %08x\n", hr
);
808 hr
= IChannelAudioVolume_SetChannelVolume(acv
, 0, 1.0f
, NULL
);
809 ok(hr
== S_OK
, "SetChannelVolume failed: %08x\n", hr
);
811 HeapFree(GetProcessHeap(), 0, vols
);
812 IChannelAudioVolume_Release(acv
);
813 IAudioClient_Release(ac
);
817 static void test_simplevolume(void)
820 ISimpleAudioVolume
*sav
;
826 hr
= IMMDevice_Activate(dev
, &IID_IAudioClient
, CLSCTX_INPROC_SERVER
,
828 ok(hr
== S_OK
, "Activation failed with %08x\n", hr
);
832 hr
= IAudioClient_GetMixFormat(ac
, &fmt
);
833 ok(hr
== S_OK
, "GetMixFormat failed: %08x\n", hr
);
835 hr
= IAudioClient_Initialize(ac
, AUDCLNT_SHAREMODE_SHARED
,
836 AUDCLNT_STREAMFLAGS_NOPERSIST
, 5000000, 0, fmt
, NULL
);
837 ok(hr
== S_OK
, "Initialize failed: %08x\n", hr
);
839 hr
= IAudioClient_GetService(ac
, &IID_ISimpleAudioVolume
, (void**)&sav
);
840 ok(hr
== S_OK
, "GetService failed: %08x\n", hr
);
844 hr
= ISimpleAudioVolume_GetMasterVolume(sav
, NULL
);
845 ok(hr
== NULL_PTR_ERR
, "GetMasterVolume gave wrong error: %08x\n", hr
);
847 hr
= ISimpleAudioVolume_GetMasterVolume(sav
, &vol
);
848 ok(hr
== S_OK
, "GetMasterVolume failed: %08x\n", hr
);
849 ok(vol
== 1.f
, "Master volume wasn't 1: %f\n", vol
);
851 hr
= ISimpleAudioVolume_SetMasterVolume(sav
, -1.f
, NULL
);
852 ok(hr
== E_INVALIDARG
, "SetMasterVolume gave wrong error: %08x\n", hr
);
854 hr
= ISimpleAudioVolume_SetMasterVolume(sav
, 2.f
, NULL
);
855 ok(hr
== E_INVALIDARG
, "SetMasterVolume gave wrong error: %08x\n", hr
);
857 hr
= ISimpleAudioVolume_SetMasterVolume(sav
, 0.2f
, NULL
);
858 ok(hr
== S_OK
, "SetMasterVolume failed: %08x\n", hr
);
860 hr
= ISimpleAudioVolume_GetMasterVolume(sav
, &vol
);
861 ok(hr
== S_OK
, "GetMasterVolume failed: %08x\n", hr
);
862 ok(fabsf(vol
- 0.2f
) < 0.05f
, "Master volume wasn't 0.2: %f\n", vol
);
864 hr
= ISimpleAudioVolume_GetMute(sav
, NULL
);
865 ok(hr
== NULL_PTR_ERR
, "GetMute gave wrong error: %08x\n", hr
);
868 hr
= ISimpleAudioVolume_GetMute(sav
, &mute
);
869 ok(hr
== S_OK
, "GetMute failed: %08x\n", hr
);
870 ok(mute
== FALSE
, "Session is already muted\n");
872 hr
= ISimpleAudioVolume_SetMute(sav
, TRUE
, NULL
);
873 ok(hr
== S_OK
, "SetMute failed: %08x\n", hr
);
876 hr
= ISimpleAudioVolume_GetMute(sav
, &mute
);
877 ok(hr
== S_OK
, "GetMute failed: %08x\n", hr
);
878 ok(mute
== TRUE
, "Session should have been muted\n");
880 hr
= ISimpleAudioVolume_GetMasterVolume(sav
, &vol
);
881 ok(hr
== S_OK
, "GetMasterVolume failed: %08x\n", hr
);
882 ok(fabsf(vol
- 0.2f
) < 0.05f
, "Master volume wasn't 0.2: %f\n", vol
);
884 hr
= ISimpleAudioVolume_SetMasterVolume(sav
, 1.f
, NULL
);
885 ok(hr
== S_OK
, "SetMasterVolume failed: %08x\n", hr
);
888 hr
= ISimpleAudioVolume_GetMute(sav
, &mute
);
889 ok(hr
== S_OK
, "GetMute failed: %08x\n", hr
);
890 ok(mute
== TRUE
, "Session should have been muted\n");
892 hr
= ISimpleAudioVolume_SetMute(sav
, FALSE
, NULL
);
893 ok(hr
== S_OK
, "SetMute failed: %08x\n", hr
);
895 ISimpleAudioVolume_Release(sav
);
896 IAudioClient_Release(ac
);
900 static void test_volume_dependence(void)
902 IAudioClient
*ac
, *ac2
;
903 ISimpleAudioVolume
*sav
;
904 IChannelAudioVolume
*cav
;
905 IAudioStreamVolume
*asv
;
912 hr
= CoCreateGuid(&session
);
913 ok(hr
== S_OK
, "CoCreateGuid failed: %08x\n", hr
);
915 hr
= IMMDevice_Activate(dev
, &IID_IAudioClient
, CLSCTX_INPROC_SERVER
,
917 ok(hr
== S_OK
, "Activation failed with %08x\n", hr
);
919 hr
= IAudioClient_GetMixFormat(ac
, &fmt
);
920 ok(hr
== S_OK
, "GetMixFormat failed: %08x\n", hr
);
922 hr
= IAudioClient_Initialize(ac
, AUDCLNT_SHAREMODE_SHARED
,
923 AUDCLNT_STREAMFLAGS_NOPERSIST
, 5000000, 0, fmt
, &session
);
924 ok(hr
== S_OK
, "Initialize failed: %08x\n", hr
);
926 hr
= IAudioClient_GetService(ac
, &IID_ISimpleAudioVolume
, (void**)&sav
);
927 ok(hr
== S_OK
, "GetService (SimpleAudioVolume) failed: %08x\n", hr
);
929 hr
= IAudioClient_GetService(ac
, &IID_IChannelAudioVolume
, (void**)&cav
);
930 ok(hr
== S_OK
, "GetService (ChannelAudioVolme) failed: %08x\n", hr
);
932 hr
= IAudioClient_GetService(ac
, &IID_IAudioStreamVolume
, (void**)&asv
);
933 ok(hr
== S_OK
, "GetService (AudioStreamVolume) failed: %08x\n", hr
);
937 hr
= IAudioStreamVolume_SetChannelVolume(asv
, 0, 0.2f
);
938 ok(hr
== S_OK
, "ASV_SetChannelVolume failed: %08x\n", hr
);
940 hr
= IChannelAudioVolume_SetChannelVolume(cav
, 0, 0.4f
, NULL
);
941 ok(hr
== S_OK
, "CAV_SetChannelVolume failed: %08x\n", hr
);
943 hr
= ISimpleAudioVolume_SetMasterVolume(sav
, 0.6f
, NULL
);
944 ok(hr
== S_OK
, "SAV_SetMasterVolume failed: %08x\n", hr
);
946 hr
= IAudioStreamVolume_GetChannelVolume(asv
, 0, &vol
);
947 ok(hr
== S_OK
, "ASV_GetChannelVolume failed: %08x\n", hr
);
948 ok(fabsf(vol
- 0.2) < 0.05f
, "ASV_GetChannelVolume gave wrong volume: %f\n", vol
);
950 hr
= IChannelAudioVolume_GetChannelVolume(cav
, 0, &vol
);
951 ok(hr
== S_OK
, "CAV_GetChannelVolume failed: %08x\n", hr
);
952 ok(fabsf(vol
- 0.4) < 0.05f
, "CAV_GetChannelVolume gave wrong volume: %f\n", vol
);
954 hr
= ISimpleAudioVolume_GetMasterVolume(sav
, &vol
);
955 ok(hr
== S_OK
, "SAV_GetMasterVolume failed: %08x\n", hr
);
956 ok(fabsf(vol
- 0.6) < 0.05f
, "SAV_GetMasterVolume gave wrong volume: %f\n", vol
);
958 hr
= IMMDevice_Activate(dev
, &IID_IAudioClient
, CLSCTX_INPROC_SERVER
,
961 IChannelAudioVolume
*cav2
;
962 IAudioStreamVolume
*asv2
;
964 hr
= IAudioClient_Initialize(ac2
, AUDCLNT_SHAREMODE_SHARED
,
965 AUDCLNT_STREAMFLAGS_NOPERSIST
, 5000000, 0, fmt
, &session
);
966 ok(hr
== S_OK
, "Initialize failed: %08x\n", hr
);
968 hr
= IAudioClient_GetService(ac2
, &IID_IChannelAudioVolume
, (void**)&cav2
);
969 ok(hr
== S_OK
, "GetService failed: %08x\n", hr
);
971 hr
= IAudioClient_GetService(ac2
, &IID_IAudioStreamVolume
, (void**)&asv2
);
972 ok(hr
== S_OK
, "GetService failed: %08x\n", hr
);
974 hr
= IChannelAudioVolume_GetChannelVolume(cav2
, 0, &vol
);
975 ok(hr
== S_OK
, "CAV_GetChannelVolume failed: %08x\n", hr
);
976 ok(fabsf(vol
- 0.4) < 0.05f
, "CAV_GetChannelVolume gave wrong volume: %f\n", vol
);
978 hr
= IAudioStreamVolume_GetChannelVolume(asv2
, 0, &vol
);
979 ok(hr
== S_OK
, "ASV_GetChannelVolume failed: %08x\n", hr
);
980 ok(vol
== 1.f
, "ASV_GetChannelVolume gave wrong volume: %f\n", vol
);
982 hr
= IChannelAudioVolume_GetChannelCount(cav2
, &nch
);
983 ok(hr
== S_OK
, "GetChannelCount failed: %08x\n", hr
);
984 ok(nch
== fmt
->nChannels
, "Got wrong channel count, expected %u: %u\n", fmt
->nChannels
, nch
);
986 hr
= IAudioStreamVolume_GetChannelCount(asv2
, &nch
);
987 ok(hr
== S_OK
, "GetChannelCount failed: %08x\n", hr
);
988 ok(nch
== fmt
->nChannels
, "Got wrong channel count, expected %u: %u\n", fmt
->nChannels
, nch
);
990 IAudioStreamVolume_Release(asv2
);
991 IChannelAudioVolume_Release(cav2
);
992 IAudioClient_Release(ac2
);
994 skip("Unable to open the same device twice. Skipping session volume control tests\n");
996 hr
= IChannelAudioVolume_SetChannelVolume(cav
, 0, 1.f
, NULL
);
997 ok(hr
== S_OK
, "CAV_SetChannelVolume failed: %08x\n", hr
);
999 hr
= ISimpleAudioVolume_SetMasterVolume(sav
, 1.f
, NULL
);
1000 ok(hr
== S_OK
, "SAV_SetMasterVolume failed: %08x\n", hr
);
1003 ISimpleAudioVolume_Release(sav
);
1004 IChannelAudioVolume_Release(cav
);
1005 IAudioStreamVolume_Release(asv
);
1006 IAudioClient_Release(ac
);
1012 IMMDeviceEnumerator
*mme
= NULL
;
1014 CoInitializeEx(NULL
, COINIT_MULTITHREADED
);
1015 hr
= CoCreateInstance(&CLSID_MMDeviceEnumerator
, NULL
, CLSCTX_INPROC_SERVER
, &IID_IMMDeviceEnumerator
, (void**)&mme
);
1018 skip("mmdevapi not available: 0x%08x\n", hr
);
1022 hr
= IMMDeviceEnumerator_GetDefaultAudioEndpoint(mme
, eCapture
, eMultimedia
, &dev
);
1023 ok(hr
== S_OK
|| hr
== E_NOTFOUND
, "GetDefaultAudioEndpoint failed: 0x%08x\n", hr
);
1024 if (hr
!= S_OK
|| !dev
)
1026 if (hr
== E_NOTFOUND
)
1027 skip("No sound card available\n");
1029 skip("GetDefaultAudioEndpoint returns 0x%08x\n", hr
);
1034 test_streamvolume();
1035 test_channelvolume();
1036 test_simplevolume();
1037 test_volume_dependence();
1039 IMMDevice_Release(dev
);
1043 IMMDeviceEnumerator_Release(mme
);