3 * Copyright 1998 Marcus Meissner
4 * Copyright 1998 Rob Riggs
5 * Copyright 2000-2002 TransGaming Technologies, Inc.
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21 * Most thread locking is complete. There may be a few race
22 * conditions still lurking.
25 * Implement SetCooperativeLevel properly (need to address focus issues)
26 * Implement DirectSound3DBuffers (stubs in place)
27 * Use hardware 3D support if available
28 * Add critical section locking inside Release and AddRef methods
29 * Handle static buffers - put those in hardware, non-static not in hardware
30 * Hardware DuplicateSoundBuffer
31 * Proper volume calculation for 3d buffers
32 * Remove DS_HEL_FRAGS and use mixer fragment length for it
43 #include <mmdeviceapi.h>
44 #include <devpropdef.h>
46 #include "dsound_private.h"
47 #include "eax-presets.h"
49 #ifndef DECLSPEC_EXPORT
51 #define DECLSPEC_EXPORT __declspec(dllexport)
53 #define DECLSPEC_EXPORT
62 typedef struct DeviceList
{
66 static DeviceList PlaybackDevices
= { NULL
, 0 };
67 static DeviceList CaptureDevices
= { NULL
, 0 };
69 const WCHAR aldriver_name
[] = L
"dsoal-aldrv.dll";
72 const EAXREVERBPROPERTIES EnvironmentDefaults
[EAX_ENVIRONMENT_UNDEFINED
] = {
73 REVERB_PRESET_GENERIC
,
74 REVERB_PRESET_PADDEDCELL
,
76 REVERB_PRESET_BATHROOM
,
77 REVERB_PRESET_LIVINGROOM
,
78 REVERB_PRESET_STONEROOM
,
79 REVERB_PRESET_AUDITORIUM
,
80 REVERB_PRESET_CONCERTHALL
,
84 REVERB_PRESET_CARPETEDHALLWAY
,
85 REVERB_PRESET_HALLWAY
,
86 REVERB_PRESET_STONECORRIDOR
,
90 REVERB_PRESET_MOUNTAINS
,
93 REVERB_PRESET_PARKINGLOT
,
94 REVERB_PRESET_SEWERPIPE
,
95 REVERB_PRESET_UNDERWATER
,
96 REVERB_PRESET_DRUGGED
,
98 REVERB_PRESET_PSYCHOTIC
101 static CRITICAL_SECTION_DEBUG openal_crst_debug
=
104 { &openal_crst_debug
.ProcessLocksList
,
105 &openal_crst_debug
.ProcessLocksList
},
106 0, 0, { (DWORD_PTR
)(__FILE__
": openal_crst_debug") }
108 CRITICAL_SECTION openal_crst
= { &openal_crst_debug
, -1, 0, 0, 0, 0 };
110 int openal_loaded
= 0;
111 static HANDLE openal_handle
= NULL
;
112 LPALCCREATECONTEXT palcCreateContext
= NULL
;
113 LPALCMAKECONTEXTCURRENT palcMakeContextCurrent
= NULL
;
114 LPALCPROCESSCONTEXT palcProcessContext
= NULL
;
115 LPALCSUSPENDCONTEXT palcSuspendContext
= NULL
;
116 LPALCDESTROYCONTEXT palcDestroyContext
= NULL
;
117 LPALCGETCURRENTCONTEXT palcGetCurrentContext
= NULL
;
118 LPALCGETCONTEXTSDEVICE palcGetContextsDevice
= NULL
;
119 LPALCOPENDEVICE palcOpenDevice
= NULL
;
120 LPALCCLOSEDEVICE palcCloseDevice
= NULL
;
121 LPALCGETERROR palcGetError
= NULL
;
122 LPALCISEXTENSIONPRESENT palcIsExtensionPresent
= NULL
;
123 LPALCGETPROCADDRESS palcGetProcAddress
= NULL
;
124 LPALCGETENUMVALUE palcGetEnumValue
= NULL
;
125 LPALCGETSTRING palcGetString
= NULL
;
126 LPALCGETINTEGERV palcGetIntegerv
= NULL
;
127 LPALCCAPTUREOPENDEVICE palcCaptureOpenDevice
= NULL
;
128 LPALCCAPTURECLOSEDEVICE palcCaptureCloseDevice
= NULL
;
129 LPALCCAPTURESTART palcCaptureStart
= NULL
;
130 LPALCCAPTURESTOP palcCaptureStop
= NULL
;
131 LPALCCAPTURESAMPLES palcCaptureSamples
= NULL
;
132 LPALENABLE palEnable
= NULL
;
133 LPALDISABLE palDisable
= NULL
;
134 LPALISENABLED palIsEnabled
= NULL
;
135 LPALGETSTRING palGetString
= NULL
;
136 LPALGETBOOLEANV palGetBooleanv
= NULL
;
137 LPALGETINTEGERV palGetIntegerv
= NULL
;
138 LPALGETFLOATV palGetFloatv
= NULL
;
139 LPALGETDOUBLEV palGetDoublev
= NULL
;
140 LPALGETBOOLEAN palGetBoolean
= NULL
;
141 LPALGETINTEGER palGetInteger
= NULL
;
142 LPALGETFLOAT palGetFloat
= NULL
;
143 LPALGETDOUBLE palGetDouble
= NULL
;
144 LPALGETERROR palGetError
= NULL
;
145 LPALISEXTENSIONPRESENT palIsExtensionPresent
= NULL
;
146 LPALGETPROCADDRESS palGetProcAddress
= NULL
;
147 LPALGETENUMVALUE palGetEnumValue
= NULL
;
148 LPALLISTENERF palListenerf
= NULL
;
149 LPALLISTENER3F palListener3f
= NULL
;
150 LPALLISTENERFV palListenerfv
= NULL
;
151 LPALLISTENERI palListeneri
= NULL
;
152 LPALLISTENER3I palListener3i
= NULL
;
153 LPALLISTENERIV palListeneriv
= NULL
;
154 LPALGETLISTENERF palGetListenerf
= NULL
;
155 LPALGETLISTENER3F palGetListener3f
= NULL
;
156 LPALGETLISTENERFV palGetListenerfv
= NULL
;
157 LPALGETLISTENERI palGetListeneri
= NULL
;
158 LPALGETLISTENER3I palGetListener3i
= NULL
;
159 LPALGETLISTENERIV palGetListeneriv
= NULL
;
160 LPALGENSOURCES palGenSources
= NULL
;
161 LPALDELETESOURCES palDeleteSources
= NULL
;
162 LPALISSOURCE palIsSource
= NULL
;
163 LPALSOURCEF palSourcef
= NULL
;
164 LPALSOURCE3F palSource3f
= NULL
;
165 LPALSOURCEFV palSourcefv
= NULL
;
166 LPALSOURCEI palSourcei
= NULL
;
167 LPALSOURCE3I palSource3i
= NULL
;
168 LPALSOURCEIV palSourceiv
= NULL
;
169 LPALGETSOURCEF palGetSourcef
= NULL
;
170 LPALGETSOURCE3F palGetSource3f
= NULL
;
171 LPALGETSOURCEFV palGetSourcefv
= NULL
;
172 LPALGETSOURCEI palGetSourcei
= NULL
;
173 LPALGETSOURCE3I palGetSource3i
= NULL
;
174 LPALGETSOURCEIV palGetSourceiv
= NULL
;
175 LPALSOURCEPLAYV palSourcePlayv
= NULL
;
176 LPALSOURCESTOPV palSourceStopv
= NULL
;
177 LPALSOURCEREWINDV palSourceRewindv
= NULL
;
178 LPALSOURCEPAUSEV palSourcePausev
= NULL
;
179 LPALSOURCEPLAY palSourcePlay
= NULL
;
180 LPALSOURCESTOP palSourceStop
= NULL
;
181 LPALSOURCEREWIND palSourceRewind
= NULL
;
182 LPALSOURCEPAUSE palSourcePause
= NULL
;
183 LPALSOURCEQUEUEBUFFERS palSourceQueueBuffers
= NULL
;
184 LPALSOURCEUNQUEUEBUFFERS palSourceUnqueueBuffers
= NULL
;
185 LPALGENBUFFERS palGenBuffers
= NULL
;
186 LPALDELETEBUFFERS palDeleteBuffers
= NULL
;
187 LPALISBUFFER palIsBuffer
= NULL
;
188 LPALBUFFERF palBufferf
= NULL
;
189 LPALBUFFER3F palBuffer3f
= NULL
;
190 LPALBUFFERFV palBufferfv
= NULL
;
191 LPALBUFFERI palBufferi
= NULL
;
192 LPALBUFFER3I palBuffer3i
= NULL
;
193 LPALBUFFERIV palBufferiv
= NULL
;
194 LPALGETBUFFERF palGetBufferf
= NULL
;
195 LPALGETBUFFER3F palGetBuffer3f
= NULL
;
196 LPALGETBUFFERFV palGetBufferfv
= NULL
;
197 LPALGETBUFFERI palGetBufferi
= NULL
;
198 LPALGETBUFFER3I palGetBuffer3i
= NULL
;
199 LPALGETBUFFERIV palGetBufferiv
= NULL
;
200 LPALBUFFERDATA palBufferData
= NULL
;
201 LPALDOPPLERFACTOR palDopplerFactor
= NULL
;
202 LPALDOPPLERVELOCITY palDopplerVelocity
= NULL
;
203 LPALDISTANCEMODEL palDistanceModel
= NULL
;
204 LPALSPEEDOFSOUND palSpeedOfSound
= NULL
;
206 LPALGENFILTERS palGenFilters
= NULL
;
207 LPALDELETEFILTERS palDeleteFilters
= NULL
;
208 LPALFILTERI palFilteri
= NULL
;
209 LPALFILTERF palFilterf
= NULL
;
210 LPALGENEFFECTS palGenEffects
= NULL
;
211 LPALDELETEEFFECTS palDeleteEffects
= NULL
;
212 LPALEFFECTI palEffecti
= NULL
;
213 LPALEFFECTF palEffectf
= NULL
;
214 LPALEFFECTFV palEffectfv
= NULL
;
215 LPALGENAUXILIARYEFFECTSLOTS palGenAuxiliaryEffectSlots
= NULL
;
216 LPALDELETEAUXILIARYEFFECTSLOTS palDeleteAuxiliaryEffectSlots
= NULL
;
217 LPALAUXILIARYEFFECTSLOTI palAuxiliaryEffectSloti
= NULL
;
218 LPALAUXILIARYEFFECTSLOTF palAuxiliaryEffectSlotf
= NULL
;
219 LPALDEFERUPDATESSOFT palDeferUpdatesSOFT
= NULL
;
220 LPALPROCESSUPDATESSOFT palProcessUpdatesSOFT
= NULL
;
221 LPALBUFFERSTORAGESOFT palBufferStorageSOFT
= NULL
;
222 LPALMAPBUFFERSOFT palMapBufferSOFT
= NULL
;
223 LPALUNMAPBUFFERSOFT palUnmapBufferSOFT
= NULL
;
224 LPALFLUSHMAPPEDBUFFERSOFT palFlushMappedBufferSOFT
= NULL
;
226 LPALCMAKECONTEXTCURRENT set_context
;
227 LPALCGETCURRENTCONTEXT get_context
;
231 static void AL_APIENTRY
wrap_DeferUpdates(void)
232 { alcSuspendContext(alcGetCurrentContext()); }
233 static void AL_APIENTRY
wrap_ProcessUpdates(void)
234 { alcProcessContext(alcGetCurrentContext()); }
236 static void EnterALSectionTLS(ALCcontext
*ctx
);
237 static void LeaveALSectionTLS(void);
238 static void EnterALSectionGlob(ALCcontext
*ctx
);
239 static void LeaveALSectionGlob(void);
242 void (*EnterALSection
)(ALCcontext
*ctx
) = EnterALSectionGlob
;
243 void (*LeaveALSection
)(void) = LeaveALSectionGlob
;
246 static BOOL
load_libopenal(void)
251 str
= getenv("DSOAL_LOGLEVEL");
253 LogLevel
= atoi(str
);
255 openal_handle
= LoadLibraryW(aldriver_name
);
258 ERR("Couldn't load %ls: %lu\n", aldriver_name
, GetLastError());
262 #define LOAD_FUNCPTR(f) do { \
263 union { void *ptr; FARPROC *proc; } func = { &p##f }; \
264 if((*func.proc = GetProcAddress(openal_handle, #f)) == NULL) \
266 ERR("Couldn't lookup %s in %ls\n", #f, aldriver_name); \
271 LOAD_FUNCPTR(alcCreateContext
);
272 LOAD_FUNCPTR(alcMakeContextCurrent
);
273 LOAD_FUNCPTR(alcProcessContext
);
274 LOAD_FUNCPTR(alcSuspendContext
);
275 LOAD_FUNCPTR(alcDestroyContext
);
276 LOAD_FUNCPTR(alcGetCurrentContext
);
277 LOAD_FUNCPTR(alcGetContextsDevice
);
278 LOAD_FUNCPTR(alcOpenDevice
);
279 LOAD_FUNCPTR(alcCloseDevice
);
280 LOAD_FUNCPTR(alcGetError
);
281 LOAD_FUNCPTR(alcIsExtensionPresent
);
282 LOAD_FUNCPTR(alcGetProcAddress
);
283 LOAD_FUNCPTR(alcGetEnumValue
);
284 LOAD_FUNCPTR(alcGetString
);
285 LOAD_FUNCPTR(alcGetIntegerv
);
286 LOAD_FUNCPTR(alcCaptureOpenDevice
);
287 LOAD_FUNCPTR(alcCaptureCloseDevice
);
288 LOAD_FUNCPTR(alcCaptureStart
);
289 LOAD_FUNCPTR(alcCaptureStop
);
290 LOAD_FUNCPTR(alcCaptureSamples
);
291 LOAD_FUNCPTR(alEnable
);
292 LOAD_FUNCPTR(alDisable
);
293 LOAD_FUNCPTR(alIsEnabled
);
294 LOAD_FUNCPTR(alGetString
);
295 LOAD_FUNCPTR(alGetBooleanv
);
296 LOAD_FUNCPTR(alGetIntegerv
);
297 LOAD_FUNCPTR(alGetFloatv
);
298 LOAD_FUNCPTR(alGetDoublev
);
299 LOAD_FUNCPTR(alGetBoolean
);
300 LOAD_FUNCPTR(alGetInteger
);
301 LOAD_FUNCPTR(alGetFloat
);
302 LOAD_FUNCPTR(alGetDouble
);
303 LOAD_FUNCPTR(alGetError
);
304 LOAD_FUNCPTR(alIsExtensionPresent
);
305 LOAD_FUNCPTR(alGetProcAddress
);
306 LOAD_FUNCPTR(alGetEnumValue
);
307 LOAD_FUNCPTR(alListenerf
);
308 LOAD_FUNCPTR(alListener3f
);
309 LOAD_FUNCPTR(alListenerfv
);
310 LOAD_FUNCPTR(alListeneri
);
311 LOAD_FUNCPTR(alListener3i
);
312 LOAD_FUNCPTR(alListeneriv
);
313 LOAD_FUNCPTR(alGetListenerf
);
314 LOAD_FUNCPTR(alGetListener3f
);
315 LOAD_FUNCPTR(alGetListenerfv
);
316 LOAD_FUNCPTR(alGetListeneri
);
317 LOAD_FUNCPTR(alGetListener3i
);
318 LOAD_FUNCPTR(alGetListeneriv
);
319 LOAD_FUNCPTR(alGenSources
);
320 LOAD_FUNCPTR(alDeleteSources
);
321 LOAD_FUNCPTR(alIsSource
);
322 LOAD_FUNCPTR(alSourcef
);
323 LOAD_FUNCPTR(alSource3f
);
324 LOAD_FUNCPTR(alSourcefv
);
325 LOAD_FUNCPTR(alSourcei
);
326 LOAD_FUNCPTR(alSource3i
);
327 LOAD_FUNCPTR(alSourceiv
);
328 LOAD_FUNCPTR(alGetSourcef
);
329 LOAD_FUNCPTR(alGetSource3f
);
330 LOAD_FUNCPTR(alGetSourcefv
);
331 LOAD_FUNCPTR(alGetSourcei
);
332 LOAD_FUNCPTR(alGetSource3i
);
333 LOAD_FUNCPTR(alGetSourceiv
);
334 LOAD_FUNCPTR(alSourcePlayv
);
335 LOAD_FUNCPTR(alSourceStopv
);
336 LOAD_FUNCPTR(alSourceRewindv
);
337 LOAD_FUNCPTR(alSourcePausev
);
338 LOAD_FUNCPTR(alSourcePlay
);
339 LOAD_FUNCPTR(alSourceStop
);
340 LOAD_FUNCPTR(alSourceRewind
);
341 LOAD_FUNCPTR(alSourcePause
);
342 LOAD_FUNCPTR(alSourceQueueBuffers
);
343 LOAD_FUNCPTR(alSourceUnqueueBuffers
);
344 LOAD_FUNCPTR(alGenBuffers
);
345 LOAD_FUNCPTR(alDeleteBuffers
);
346 LOAD_FUNCPTR(alIsBuffer
);
347 LOAD_FUNCPTR(alBufferf
);
348 LOAD_FUNCPTR(alBuffer3f
);
349 LOAD_FUNCPTR(alBufferfv
);
350 LOAD_FUNCPTR(alBufferi
);
351 LOAD_FUNCPTR(alBuffer3i
);
352 LOAD_FUNCPTR(alBufferiv
);
353 LOAD_FUNCPTR(alGetBufferf
);
354 LOAD_FUNCPTR(alGetBuffer3f
);
355 LOAD_FUNCPTR(alGetBufferfv
);
356 LOAD_FUNCPTR(alGetBufferi
);
357 LOAD_FUNCPTR(alGetBuffer3i
);
358 LOAD_FUNCPTR(alGetBufferiv
);
359 LOAD_FUNCPTR(alBufferData
);
360 LOAD_FUNCPTR(alDopplerFactor
);
361 LOAD_FUNCPTR(alDopplerVelocity
);
362 LOAD_FUNCPTR(alDistanceModel
);
363 LOAD_FUNCPTR(alSpeedOfSound
);
367 WARN("Unloading %ls\n", aldriver_name
);
368 if (openal_handle
!= NULL
)
369 FreeLibrary(openal_handle
);
370 openal_handle
= NULL
;
375 TRACE("Loaded %ls\n", aldriver_name
);
377 #define LOAD_FUNCPTR(f) p##f = alcGetProcAddress(NULL, #f)
378 LOAD_FUNCPTR(alGenFilters
);
379 LOAD_FUNCPTR(alDeleteFilters
);
380 LOAD_FUNCPTR(alFilteri
);
381 LOAD_FUNCPTR(alFilterf
);
382 LOAD_FUNCPTR(alGenEffects
);
383 LOAD_FUNCPTR(alDeleteEffects
);
384 LOAD_FUNCPTR(alEffecti
);
385 LOAD_FUNCPTR(alEffectf
);
386 LOAD_FUNCPTR(alEffectfv
);
387 LOAD_FUNCPTR(alGenAuxiliaryEffectSlots
);
388 LOAD_FUNCPTR(alDeleteAuxiliaryEffectSlots
);
389 LOAD_FUNCPTR(alAuxiliaryEffectSloti
);
390 LOAD_FUNCPTR(alAuxiliaryEffectSlotf
);
391 LOAD_FUNCPTR(alDeferUpdatesSOFT
);
392 LOAD_FUNCPTR(alProcessUpdatesSOFT
);
393 LOAD_FUNCPTR(alBufferStorageSOFT
);
394 LOAD_FUNCPTR(alMapBufferSOFT
);
395 LOAD_FUNCPTR(alUnmapBufferSOFT
);
396 LOAD_FUNCPTR(alFlushMappedBufferSOFT
);
398 if(!palDeferUpdatesSOFT
|| !palProcessUpdatesSOFT
)
400 palDeferUpdatesSOFT
= wrap_DeferUpdates
;
401 palProcessUpdatesSOFT
= wrap_ProcessUpdates
;
404 local_contexts
= alcIsExtensionPresent(NULL
, "ALC_EXT_thread_local_context");
407 TRACE("Found ALC_EXT_thread_local_context\n");
409 set_context
= alcGetProcAddress(NULL
, "alcSetThreadContext");
410 get_context
= alcGetProcAddress(NULL
, "alcGetThreadContext");
411 if(!set_context
|| !get_context
)
413 ERR("TLS advertised but functions not found, disabling thread local contexts\n");
419 set_context
= alcMakeContextCurrent
;
420 get_context
= alcGetCurrentContext
;
424 EnterALSection
= EnterALSectionTLS
;
425 LeaveALSection
= LeaveALSectionTLS
;
432 static void EnterALSectionTLS(ALCcontext
*ctx
)
434 if(LIKELY(ctx
== TlsGetValue(TlsThreadPtr
)))
437 if(LIKELY(set_context(ctx
) != ALC_FALSE
))
438 TlsSetValue(TlsThreadPtr
, ctx
);
441 ERR("Couldn't set current context!!\n");
442 checkALCError(alcGetContextsDevice(ctx
));
445 static void LeaveALSectionTLS(void)
449 static void EnterALSectionGlob(ALCcontext
*ctx
)
451 EnterCriticalSection(&openal_crst
);
452 if(UNLIKELY(alcMakeContextCurrent(ctx
) == ALC_FALSE
))
454 ERR("Couldn't set current context!!\n");
455 checkALCError(alcGetContextsDevice(ctx
));
458 static void LeaveALSectionGlob(void)
460 LeaveCriticalSection(&openal_crst
);
464 static const char *get_device_id(LPCGUID pGuid
)
466 if(IsEqualGUID(&DSDEVID_DefaultPlayback
, pGuid
))
467 return "DSDEVID_DefaultPlayback";
468 if(IsEqualGUID(&DSDEVID_DefaultVoicePlayback
, pGuid
))
469 return "DSDEVID_DefaultVoicePlayback";
470 if(IsEqualGUID(&DSDEVID_DefaultCapture
, pGuid
))
471 return "DSDEVID_DefaultCapture";
472 if(IsEqualGUID(&DSDEVID_DefaultVoiceCapture
, pGuid
))
473 return "DSDEVID_DefaultVoiceCapture";
474 return debugstr_guid(pGuid
);
477 static HRESULT
get_mmdevenum(IMMDeviceEnumerator
**devenum
)
481 init_hr
= CoInitialize(NULL
);
483 hr
= CoCreateInstance(&CLSID_MMDeviceEnumerator
, NULL
,
484 CLSCTX_INPROC_SERVER
, &IID_IMMDeviceEnumerator
, (void**)devenum
);
487 if(SUCCEEDED(init_hr
))
490 ERR("CoCreateInstance failed: %08lx\n", hr
);
497 static void release_mmdevenum(IMMDeviceEnumerator
*devenum
, HRESULT init_hr
)
499 IMMDeviceEnumerator_Release(devenum
);
500 if(SUCCEEDED(init_hr
))
504 static HRESULT
get_mmdevice_guid(IMMDevice
*device
, IPropertyStore
*ps
, GUID
*guid
)
510 IPropertyStore_AddRef(ps
);
513 hr
= IMMDevice_OpenPropertyStore(device
, STGM_READ
, &ps
);
516 WARN("OpenPropertyStore failed: %08lx\n", hr
);
521 PropVariantInit(&pv
);
523 hr
= IPropertyStore_GetValue(ps
, &PKEY_AudioEndpoint_GUID
, &pv
);
526 IPropertyStore_Release(ps
);
527 WARN("GetValue(GUID) failed: %08lx\n", hr
);
531 CLSIDFromString(pv
.pwszVal
, guid
);
533 PropVariantClear(&pv
);
534 IPropertyStore_Release(ps
);
540 static BOOL
send_device(IMMDevice
*device
, EDataFlow flow
, DeviceList
*devlist
, PRVTENUMCALLBACK cb
, void *user
)
549 PropVariantInit(&pv
);
551 hr
= IMMDevice_OpenPropertyStore(device
, STGM_READ
, &ps
);
554 WARN("OpenPropertyStore failed: %08lx\n", hr
);
558 hr
= get_mmdevice_guid(device
, ps
, &guid
);
559 if(FAILED(hr
) || (devlist
->Count
> 0 && IsEqualGUID(&devlist
->Guids
[0], &guid
)))
561 IPropertyStore_Release(ps
);
565 hr
= IPropertyStore_GetValue(ps
, (const PROPERTYKEY
*)&DEVPKEY_Device_FriendlyName
, &pv
);
568 IPropertyStore_Release(ps
);
569 WARN("GetValue(FriendlyName) failed: %08lx\n", hr
);
573 dev_count
= devlist
->Count
++;
574 devlist
->Guids
[dev_count
] = guid
;
579 TRACE("Calling back with %s - %ls\n", debugstr_guid(&devlist
->Guids
[dev_count
]),
581 keep_going
= cb(flow
, &devlist
->Guids
[dev_count
], pv
.pwszVal
, aldriver_name
, user
);
584 PropVariantClear(&pv
);
585 IPropertyStore_Release(ps
);
590 HRESULT
get_mmdevice(EDataFlow flow
, const GUID
*tgt
, IMMDevice
**device
)
592 IMMDeviceEnumerator
*devenum
;
593 IMMDeviceCollection
*coll
;
597 init_hr
= get_mmdevenum(&devenum
);
598 if(!devenum
) return init_hr
;
600 hr
= IMMDeviceEnumerator_EnumAudioEndpoints(devenum
, flow
, DEVICE_STATE_ACTIVE
, &coll
);
603 WARN("EnumAudioEndpoints failed: %08lx\n", hr
);
604 release_mmdevenum(devenum
, init_hr
);
608 hr
= IMMDeviceCollection_GetCount(coll
, &count
);
611 IMMDeviceCollection_Release(coll
);
612 release_mmdevenum(devenum
, init_hr
);
613 WARN("GetCount failed: %08lx\n", hr
);
617 for(i
= 0; i
< count
;++i
)
621 hr
= IMMDeviceCollection_Item(coll
, i
, device
);
622 if(FAILED(hr
)) continue;
624 hr
= get_mmdevice_guid(*device
, NULL
, &guid
);
627 IMMDevice_Release(*device
);
631 if(IsEqualGUID(&guid
, tgt
))
633 IMMDeviceCollection_Release(coll
);
634 release_mmdevenum(devenum
, init_hr
);
638 IMMDevice_Release(*device
);
641 WARN("No device with GUID %s found!\n", debugstr_guid(tgt
));
643 IMMDeviceCollection_Release(coll
);
644 release_mmdevenum(devenum
, init_hr
);
646 return DSERR_INVALIDPARAM
;
649 /* S_FALSE means the callback returned FALSE at some point
650 * S_OK means the callback always returned TRUE */
651 HRESULT
enumerate_mmdevices(EDataFlow flow
, PRVTENUMCALLBACK cb
, void *user
)
653 static const WCHAR primary_desc
[] = L
"Primary Sound Driver";
655 IMMDeviceEnumerator
*devenum
;
656 IMMDeviceCollection
*coll
;
663 init_hr
= get_mmdevenum(&devenum
);
664 if(!devenum
) return init_hr
;
666 hr
= IMMDeviceEnumerator_EnumAudioEndpoints(devenum
, flow
, DEVICE_STATE_ACTIVE
, &coll
);
669 release_mmdevenum(devenum
, init_hr
);
670 WARN("EnumAudioEndpoints failed: %08lx\n", hr
);
674 hr
= IMMDeviceCollection_GetCount(coll
, &count
);
677 IMMDeviceCollection_Release(coll
);
678 release_mmdevenum(devenum
, init_hr
);
679 WARN("GetCount failed: %08lx\n", hr
);
685 IMMDeviceCollection_Release(coll
);
686 release_mmdevenum(devenum
, init_hr
);
690 devlist
= (flow
==eCapture
) ? &CaptureDevices
: &PlaybackDevices
;
692 HeapFree(GetProcessHeap(), 0, devlist
->Guids
);
694 devlist
->Guids
= HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
,
695 sizeof(devlist
->Guids
[0])*count
);
697 TRACE("Calling back with NULL (%ls)\n", primary_desc
);
698 keep_going
= cb(flow
, NULL
, primary_desc
, L
"", user
);
700 /* always send the default device first */
701 hr
= IMMDeviceEnumerator_GetDefaultAudioEndpoint(devenum
, flow
, eMultimedia
, &device
);
704 if(!keep_going
) cb
= NULL
;
705 keep_going
= send_device(device
, flow
, devlist
, cb
, user
);
706 IMMDevice_Release(device
);
709 for(i
= 0;i
< count
;++i
)
714 hr
= IMMDeviceCollection_Item(coll
, i
, &device
);
717 WARN("Item failed: %08lx\n", hr
);
721 keep_going
= send_device(device
, flow
, devlist
, cb
, user
);
723 IMMDevice_Release(device
);
725 IMMDeviceCollection_Release(coll
);
727 release_mmdevenum(devenum
, init_hr
);
729 return keep_going
? S_OK
: S_FALSE
;
732 /***************************************************************************
733 * GetDeviceID [DSOUND.9]
735 * Retrieves unique identifier of default device specified
738 * pGuidSrc [I] Address of device GUID.
739 * pGuidDest [O] Address to receive unique device GUID.
743 * Failure: DSERR_INVALIDPARAM
746 * pGuidSrc is a valid device GUID or DSDEVID_DefaultPlayback,
747 * DSDEVID_DefaultCapture, DSDEVID_DefaultVoicePlayback, or
748 * DSDEVID_DefaultVoiceCapture.
749 * Returns pGuidSrc if pGuidSrc is a valid device or the device
750 * GUID for the specified constants.
752 DECLSPEC_EXPORT HRESULT WINAPI
GetDeviceID(LPCGUID pGuidSrc
, LPGUID pGuidDest
)
754 IMMDeviceEnumerator
*devenum
;
760 TRACE("(%s, %p)\n", get_device_id(pGuidSrc
), pGuidDest
);
762 if(!pGuidSrc
|| !pGuidDest
)
763 return DSERR_INVALIDPARAM
;
766 if(IsEqualGUID(&DSDEVID_DefaultPlayback
, pGuidSrc
))
768 else if(IsEqualGUID(&DSDEVID_DefaultVoicePlayback
, pGuidSrc
))
769 role
= eCommunications
;
773 if(IsEqualGUID(&DSDEVID_DefaultCapture
, pGuidSrc
))
775 else if(IsEqualGUID(&DSDEVID_DefaultVoiceCapture
, pGuidSrc
))
776 role
= eCommunications
;
779 *pGuidDest
= *pGuidSrc
;
784 init_hr
= get_mmdevenum(&devenum
);
785 if(!devenum
) return init_hr
;
787 hr
= IMMDeviceEnumerator_GetDefaultAudioEndpoint(devenum
, flow
, role
, &device
);
790 WARN("GetDefaultAudioEndpoint failed: %08lx\n", hr
);
791 release_mmdevenum(devenum
, init_hr
);
792 return DSERR_NODRIVER
;
795 hr
= get_mmdevice_guid(device
, NULL
, pGuidDest
);
796 IMMDevice_Release(device
);
798 release_mmdevenum(devenum
, init_hr
);
804 struct morecontextW
{
805 LPDSENUMCALLBACKW callW
;
809 static BOOL CALLBACK
w_callback(EDataFlow flow
, LPGUID guid
, LPCWSTR descW
, LPCWSTR modW
, LPVOID data
)
811 struct morecontextW
*context
= data
;
814 return context
->callW(guid
, descW
, modW
, context
->data
);
817 struct morecontextA
{
818 LPDSENUMCALLBACKA callA
;
822 static BOOL CALLBACK
w_to_a_callback(EDataFlow flow
, LPGUID guid
, LPCWSTR descW
, LPCWSTR modW
, LPVOID data
)
824 struct morecontextA
*context
= data
;
830 dlen
= WideCharToMultiByte(CP_ACP
, 0, descW
, -1, NULL
, 0, NULL
, NULL
);
831 mlen
= WideCharToMultiByte(CP_ACP
, 0, modW
, -1, NULL
, 0, NULL
, NULL
);
832 if(dlen
< 0 || mlen
< 0) return FALSE
;
834 descA
= HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY
, dlen
+mlen
+2);
835 if(!descA
) return FALSE
;
836 modA
= descA
+ dlen
+1;
838 WideCharToMultiByte(CP_ACP
, 0, descW
, -1, descA
, dlen
, NULL
, NULL
);
839 WideCharToMultiByte(CP_ACP
, 0, modW
, -1, modA
, mlen
, NULL
, NULL
);
841 ret
= context
->callA(guid
, descA
, modA
, context
->data
);
843 HeapFree(GetProcessHeap(), 0, descA
);
847 /***************************************************************************
848 * DirectSoundEnumerateA [DSOUND.2]
850 * Enumerate all DirectSound drivers installed in the system
853 * lpDSEnumCallback [I] Address of callback function.
854 * lpContext [I] Address of user defined context passed to callback function.
858 * Failure: DSERR_INVALIDPARAM
860 DECLSPEC_EXPORT HRESULT WINAPI
DirectSoundEnumerateA(
861 LPDSENUMCALLBACKA lpDSEnumCallback
,
864 struct morecontextA ctx
;
867 TRACE("(%p, %p)\n", lpDSEnumCallback
, lpContext
);
869 if(lpDSEnumCallback
== NULL
)
871 WARN("invalid parameter: lpDSEnumCallback == NULL\n");
872 return DSERR_INVALIDPARAM
;
875 ctx
.callA
= lpDSEnumCallback
;
876 ctx
.data
= lpContext
;
878 hr
= enumerate_mmdevices(eRender
, w_to_a_callback
, &ctx
);
879 return SUCCEEDED(hr
) ? DS_OK
: hr
;
883 /***************************************************************************
884 * DirectSoundEnumerateW [DSOUND.3]
886 * Enumerate all DirectSound drivers installed in the system
889 * lpDSEnumCallback [I] Address of callback function.
890 * lpContext [I] Address of user defined context passed to callback function.
894 * Failure: DSERR_INVALIDPARAM
896 DECLSPEC_EXPORT HRESULT WINAPI
DirectSoundEnumerateW(
897 LPDSENUMCALLBACKW lpDSEnumCallback
,
900 struct morecontextW ctx
;
903 TRACE("(%p, %p)\n", lpDSEnumCallback
, lpContext
);
905 if(lpDSEnumCallback
== NULL
)
907 WARN("invalid parameter: lpDSEnumCallback == NULL\n");
908 return DSERR_INVALIDPARAM
;
911 ctx
.callW
= lpDSEnumCallback
;
912 ctx
.data
= lpContext
;
914 hr
= enumerate_mmdevices(eRender
, w_callback
, &ctx
);
915 return SUCCEEDED(hr
) ? DS_OK
: hr
;
918 /***************************************************************************
919 * DirectSoundCaptureEnumerateA [DSOUND.7]
921 * Enumerate all DirectSound drivers installed in the system.
924 * lpDSEnumCallback [I] Address of callback function.
925 * lpContext [I] Address of user defined context passed to callback function.
929 * Failure: DSERR_INVALIDPARAM
931 DECLSPEC_EXPORT HRESULT WINAPI
DirectSoundCaptureEnumerateA(
932 LPDSENUMCALLBACKA lpDSEnumCallback
,
935 struct morecontextA ctx
;
938 TRACE("(%p, %p)\n", lpDSEnumCallback
, lpContext
);
940 if(lpDSEnumCallback
== NULL
)
942 WARN("invalid parameter: lpDSEnumCallback == NULL\n");
943 return DSERR_INVALIDPARAM
;
946 ctx
.callA
= lpDSEnumCallback
;
947 ctx
.data
= lpContext
;
949 hr
= enumerate_mmdevices(eCapture
, w_to_a_callback
, &ctx
);
950 return SUCCEEDED(hr
) ? DS_OK
: hr
;
953 /***************************************************************************
954 * DirectSoundCaptureEnumerateW [DSOUND.8]
956 * Enumerate all DirectSound drivers installed in the system.
959 * lpDSEnumCallback [I] Address of callback function.
960 * lpContext [I] Address of user defined context passed to callback function.
964 * Failure: DSERR_INVALIDPARAM
966 DECLSPEC_EXPORT HRESULT WINAPI
DirectSoundCaptureEnumerateW(
967 LPDSENUMCALLBACKW lpDSEnumCallback
,
970 struct morecontextW ctx
;
973 TRACE("(%p, %p)\n", lpDSEnumCallback
, lpContext
);
975 if(lpDSEnumCallback
== NULL
)
977 WARN("invalid parameter: lpDSEnumCallback == NULL\n");
978 return DSERR_INVALIDPARAM
;
981 ctx
.callW
= lpDSEnumCallback
;
982 ctx
.data
= lpContext
;
984 hr
= enumerate_mmdevices(eCapture
, w_callback
, &ctx
);
985 return SUCCEEDED(hr
) ? DS_OK
: hr
;
988 /*******************************************************************************
989 * DirectSoundCreate (DSOUND.1)
991 * Creates and initializes a DirectSound interface.
994 * lpcGUID [I] Address of the GUID that identifies the sound device.
995 * ppDS [O] Address of a variable to receive the interface pointer.
996 * pUnkOuter [I] Must be NULL.
1000 * Failure: DSERR_ALLOCATED, DSERR_INVALIDPARAM, DSERR_NOAGGREGATION,
1001 * DSERR_NODRIVER, DSERR_OUTOFMEMORY
1003 DECLSPEC_EXPORT HRESULT WINAPI
1004 DirectSoundCreate(LPCGUID lpcGUID
, IDirectSound
**ppDS
, IUnknown
*pUnkOuter
)
1009 TRACE("(%s, %p, %p)\n", debugstr_guid(lpcGUID
), ppDS
, pUnkOuter
);
1012 WARN("invalid parameter: ppDS == NULL\n");
1013 return DSERR_INVALIDPARAM
;
1017 if (pUnkOuter
!= NULL
) {
1018 WARN("invalid parameter: pUnkOuter != NULL\n");
1019 return DSERR_INVALIDPARAM
;
1022 hr
= DSOUND_Create(&IID_IDirectSound
, &pDS
);
1026 hr
= IDirectSound_Initialize(*ppDS
, lpcGUID
);
1029 IDirectSound_Release(*ppDS
);
1037 /*******************************************************************************
1038 * DirectSoundCreate8 (DSOUND.11)
1040 * Creates and initializes a DirectSound8 interface.
1043 * lpcGUID [I] Address of the GUID that identifies the sound device.
1044 * ppDS [O] Address of a variable to receive the interface pointer.
1045 * pUnkOuter [I] Must be NULL.
1049 * Failure: DSERR_ALLOCATED, DSERR_INVALIDPARAM, DSERR_NOAGGREGATION,
1050 * DSERR_NODRIVER, DSERR_OUTOFMEMORY
1052 DECLSPEC_EXPORT HRESULT WINAPI
1053 DirectSoundCreate8(LPCGUID lpcGUID
, IDirectSound8
**ppDS
, IUnknown
*pUnkOuter
)
1058 TRACE("(%s, %p, %p)\n", debugstr_guid(lpcGUID
), ppDS
, pUnkOuter
);
1061 WARN("invalid parameter: ppDS == NULL\n");
1062 return DSERR_INVALIDPARAM
;
1066 if (pUnkOuter
!= NULL
) {
1067 WARN("invalid parameter: pUnkOuter != NULL\n");
1068 return DSERR_INVALIDPARAM
;
1071 hr
= DSOUND_Create8(&IID_IDirectSound8
, &pDS
);
1075 hr
= IDirectSound8_Initialize(*ppDS
, lpcGUID
);
1078 IDirectSound8_Release(*ppDS
);
1086 /***************************************************************************
1087 * DirectSoundCaptureCreate [DSOUND.6]
1089 * Create and initialize a DirectSoundCapture interface.
1092 * lpcGUID [I] Address of the GUID that identifies the sound capture device.
1093 * lplpDSC [O] Address of a variable to receive the interface pointer.
1094 * pUnkOuter [I] Must be NULL.
1098 * Failure: DSERR_NOAGGREGATION, DSERR_ALLOCATED, DSERR_INVALIDPARAM,
1102 * lpcGUID must be one of the values returned from DirectSoundCaptureEnumerate
1103 * or NULL for the default device or DSDEVID_DefaultCapture or
1104 * DSDEVID_DefaultVoiceCapture.
1106 * DSERR_ALLOCATED is returned for sound devices that do not support full duplex.
1108 DECLSPEC_EXPORT HRESULT WINAPI
1109 DirectSoundCaptureCreate(LPCGUID lpcGUID
, IDirectSoundCapture
**ppDSC
, IUnknown
*pUnkOuter
)
1114 TRACE("(%s, %p, %p)\n", debugstr_guid(lpcGUID
), ppDSC
, pUnkOuter
);
1118 WARN("invalid parameter: pUnkOuter != NULL\n");
1119 return DSERR_NOAGGREGATION
;
1124 WARN("invalid parameter: ppDSC == NULL\n");
1125 return DSERR_INVALIDPARAM
;
1129 hr
= DSOUND_CaptureCreate(&IID_IDirectSoundCapture
, &pDSC
);
1133 hr
= IDirectSoundCapture_Initialize(*ppDSC
, lpcGUID
);
1136 IDirectSoundCapture_Release(*ppDSC
);
1144 /***************************************************************************
1145 * DirectSoundCaptureCreate8 [DSOUND.12]
1147 * Create and initialize a DirectSoundCapture interface.
1150 * lpcGUID [I] Address of the GUID that identifies the sound capture device.
1151 * lplpDSC [O] Address of a variable to receive the interface pointer.
1152 * pUnkOuter [I] Must be NULL.
1156 * Failure: DSERR_NOAGGREGATION, DSERR_ALLOCATED, DSERR_INVALIDPARAM,
1160 * lpcGUID must be one of the values returned from DirectSoundCaptureEnumerate
1161 * or NULL for the default device or DSDEVID_DefaultCapture or
1162 * DSDEVID_DefaultVoiceCapture.
1164 * DSERR_ALLOCATED is returned for sound devices that do not support full duplex.
1166 DECLSPEC_EXPORT HRESULT WINAPI
1167 DirectSoundCaptureCreate8(LPCGUID lpcGUID
, IDirectSoundCapture8
**ppDSC8
, IUnknown
*pUnkOuter
)
1172 TRACE("(%s, %p, %p)\n", debugstr_guid(lpcGUID
), ppDSC8
, pUnkOuter
);
1176 WARN("invalid parameter: pUnkOuter != NULL\n");
1177 return DSERR_NOAGGREGATION
;
1182 WARN("invalid parameter: ppDSC8 == NULL\n");
1183 return DSERR_INVALIDPARAM
;
1187 hr
= DSOUND_CaptureCreate8(&IID_IDirectSoundCapture
, &pDSC8
);
1191 hr
= IDirectSoundCapture_Initialize(*ppDSC8
, lpcGUID
);
1194 IDirectSoundCapture_Release(*ppDSC8
);
1202 /*******************************************************************************
1203 * DirectSound ClassFactory
1206 typedef HRESULT (*FnCreateInstance
)(REFIID riid
, LPVOID
*ppobj
);
1209 IClassFactory IClassFactory_iface
;
1212 FnCreateInstance pfnCreateInstance
;
1213 } IClassFactoryImpl
;
1215 static inline IClassFactoryImpl
*impl_from_IClassFactory(IClassFactory
*iface
)
1217 return CONTAINING_RECORD(iface
, IClassFactoryImpl
, IClassFactory_iface
);
1220 static HRESULT WINAPI
DSCF_QueryInterface(LPCLASSFACTORY iface
, REFIID riid
, LPVOID
*ppobj
)
1222 IClassFactoryImpl
*This
= impl_from_IClassFactory(iface
);
1223 TRACE("(%p, %s, %p)\n", This
, debugstr_guid(riid
), ppobj
);
1226 if (IsEqualIID(riid
, &IID_IUnknown
) ||
1227 IsEqualIID(riid
, &IID_IClassFactory
))
1230 IUnknown_AddRef(iface
);
1234 return E_NOINTERFACE
;
1237 static ULONG WINAPI
DSCF_AddRef(LPCLASSFACTORY iface
)
1239 IClassFactoryImpl
*This
= impl_from_IClassFactory(iface
);
1240 ULONG ref
= InterlockedIncrement(&(This
->ref
));
1241 TRACE("(%p) ref %lu\n", iface
, ref
);
1245 static ULONG WINAPI
DSCF_Release(LPCLASSFACTORY iface
)
1247 IClassFactoryImpl
*This
= impl_from_IClassFactory(iface
);
1248 ULONG ref
= InterlockedDecrement(&(This
->ref
));
1249 TRACE("(%p) ref %lu\n", iface
, ref
);
1250 /* static class, won't be freed */
1254 static HRESULT WINAPI
DSCF_CreateInstance(
1255 LPCLASSFACTORY iface
,
1260 IClassFactoryImpl
*This
= impl_from_IClassFactory(iface
);
1261 TRACE("(%p, %p, %s, %p)\n", This
, pOuter
, debugstr_guid(riid
), ppobj
);
1264 return CLASS_E_NOAGGREGATION
;
1266 if (ppobj
== NULL
) {
1267 WARN("invalid parameter\n");
1268 return DSERR_INVALIDPARAM
;
1271 return This
->pfnCreateInstance(riid
, ppobj
);
1274 static HRESULT WINAPI
DSCF_LockServer(LPCLASSFACTORY iface
, BOOL dolock
)
1276 IClassFactoryImpl
*This
= impl_from_IClassFactory(iface
);
1277 FIXME("(%p, %d) stub!\n", This
, dolock
);
1281 static const IClassFactoryVtbl DSCF_Vtbl
= {
1282 DSCF_QueryInterface
,
1285 DSCF_CreateInstance
,
1289 static IClassFactoryImpl DSOUND_CF
[] = {
1290 { {&DSCF_Vtbl
}, 1, &CLSID_DirectSound
, DSOUND_Create
},
1291 { {&DSCF_Vtbl
}, 1, &CLSID_DirectSound8
, DSOUND_Create8
},
1292 { {&DSCF_Vtbl
}, 1, &CLSID_DirectSoundCapture
, DSOUND_CaptureCreate
},
1293 { {&DSCF_Vtbl
}, 1, &CLSID_DirectSoundCapture8
, DSOUND_CaptureCreate8
},
1294 { {&DSCF_Vtbl
}, 1, &CLSID_DirectSoundFullDuplex
, DSOUND_FullDuplexCreate
},
1295 { {&DSCF_Vtbl
}, 1, &CLSID_DirectSoundPrivate
, IKsPrivatePropertySetImpl_Create
},
1296 { {NULL
}, 0, NULL
, NULL
}
1299 /*******************************************************************************
1300 * DllGetClassObject [DSOUND.@]
1301 * Retrieves class object from a DLL object
1304 * Docs say returns STDAPI
1307 * rclsid [I] CLSID for the class object
1308 * riid [I] Reference to identifier of interface for class object
1309 * ppv [O] Address of variable to receive interface pointer for riid
1313 * Failure: CLASS_E_CLASSNOTAVAILABLE, E_OUTOFMEMORY, E_INVALIDARG,
1316 DECLSPEC_EXPORT HRESULT WINAPI
DllGetClassObject(REFCLSID rclsid
, REFIID riid
, LPVOID
*ppv
)
1319 TRACE("(%s, %s, %p)\n", debugstr_guid(rclsid
), debugstr_guid(riid
), ppv
);
1322 WARN("invalid parameter\n");
1323 return E_INVALIDARG
;
1328 if (!IsEqualIID(riid
, &IID_IClassFactory
) &&
1329 !IsEqualIID(riid
, &IID_IUnknown
)) {
1330 WARN("no interface for %s\n", debugstr_guid(riid
));
1331 return E_NOINTERFACE
;
1334 while (NULL
!= DSOUND_CF
[i
].rclsid
) {
1335 if (IsEqualGUID(rclsid
, DSOUND_CF
[i
].rclsid
)) {
1336 DSCF_AddRef(&DSOUND_CF
[i
].IClassFactory_iface
);
1337 *ppv
= &DSOUND_CF
[i
].IClassFactory_iface
;
1343 WARN("No class found for %s\n", debugstr_guid(rclsid
));
1344 return CLASS_E_CLASSNOTAVAILABLE
;
1348 /*******************************************************************************
1349 * DllCanUnloadNow [DSOUND.4]
1350 * Determines whether the DLL is in use.
1356 DECLSPEC_EXPORT HRESULT WINAPI
DllCanUnloadNow(void)
1358 FIXME("(void): stub\n");
1362 /***********************************************************************
1363 * DllMain (DSOUND.init)
1365 DECLSPEC_EXPORT BOOL WINAPI
DllMain(HINSTANCE hInstDLL
, DWORD fdwReason
, LPVOID lpvReserved
)
1369 TRACE("(%p, %lu, %p)\n", hInstDLL
, fdwReason
, lpvReserved
);
1373 case DLL_PROCESS_ATTACH
:
1375 if((wstr
=_wgetenv(L
"DSOAL_LOGFILE")) != NULL
&& wstr
[0] != 0)
1377 FILE *f
= _wfopen(wstr
, L
"wt");
1378 if(!f
) ERR("Failed to open log file %ls\n", wstr
);
1382 if(!load_libopenal())
1384 TlsThreadPtr
= TlsAlloc();
1385 /* Increase refcount on dsound by 1 */
1386 GetModuleHandleExW(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS
, (LPCWSTR
)hInstDLL
, &hInstDLL
);
1389 case DLL_THREAD_ATTACH
:
1392 case DLL_THREAD_DETACH
:
1397 case DLL_PROCESS_DETACH
:
1398 HeapFree(GetProcessHeap(), 0, PlaybackDevices
.Guids
);
1399 PlaybackDevices
.Guids
= NULL
;
1400 PlaybackDevices
.Count
= 0;
1401 HeapFree(GetProcessHeap(), 0, CaptureDevices
.Guids
);
1402 CaptureDevices
.Guids
= NULL
;
1403 CaptureDevices
.Count
= 0;
1406 FreeLibrary(openal_handle
);
1407 TlsFree(TlsThreadPtr
);
1408 if(LogFile
!= stderr
)
1417 /***********************************************************************
1418 * DllRegisterServer (DSOUND.@)
1420 HRESULT WINAPI
DllRegisterServer(void)
1422 return __wine_register_resources(instance
);
1425 /***********************************************************************
1426 * DllUnregisterServer (DSOUND.@)
1428 HRESULT WINAPI
DllUnregisterServer(void)
1430 return __wine_unregister_resources(instance
);