xref: /reactos/dll/win32/msacm32/internal.c (revision 4567e13e)
1 /* -*- tab-width: 8; c-basic-offset: 4 -*- */
2 
3 /*
4  *      MSACM32 library
5  *
6  *      Copyright 1998  Patrik Stridvall
7  *		  1999	Eric Pouech
8  *
9  * This library is free software; you can redistribute it and/or
10  * modify it under the terms of the GNU Lesser General Public
11  * License as published by the Free Software Foundation; either
12  * version 2.1 of the License, or (at your option) any later version.
13  *
14  * This library is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
17  * Lesser General Public License for more details.
18  *
19  * You should have received a copy of the GNU Lesser General Public
20  * License along with this library; if not, write to the Free Software
21  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
22  */
23 
24 #include <stdarg.h>
25 #include <string.h>
26 
27 #include "windef.h"
28 #include "winbase.h"
29 #include "wingdi.h"
30 #include "winuser.h"
31 #include "winerror.h"
32 #include "winreg.h"
33 #include "mmsystem.h"
34 #include "mmreg.h"
35 #include "msacm.h"
36 #include "msacmdrv.h"
37 #include "wineacm.h"
38 #include "wine/debug.h"
39 #include "wine/unicode.h"
40 
41 WINE_DEFAULT_DEBUG_CHANNEL(msacm);
42 
43 /**********************************************************************/
44 
45 HANDLE MSACM_hHeap = NULL;
46 PWINE_ACMDRIVERID MSACM_pFirstACMDriverID = NULL;
47 static PWINE_ACMDRIVERID MSACM_pLastACMDriverID;
48 
49 static DWORD MSACM_suspendBroadcastCount = 0;
50 static BOOL MSACM_pendingBroadcast = FALSE;
51 static PWINE_ACMNOTIFYWND MSACM_pFirstACMNotifyWnd = NULL;
52 static PWINE_ACMNOTIFYWND MSACM_pLastACMNotifyWnd = NULL;
53 
54 static void MSACM_ReorderDriversByPriority(void);
55 
56 /***********************************************************************
57  *           MSACM_RegisterDriverFromRegistry()
58  */
59 PWINE_ACMDRIVERID MSACM_RegisterDriverFromRegistry(LPCWSTR pszRegEntry)
60 {
61     static const WCHAR msacmW[] = {'M','S','A','C','M','.'};
62     static const WCHAR drvkey[] = {'S','o','f','t','w','a','r','e','\\',
63 				   'M','i','c','r','o','s','o','f','t','\\',
64 				   'W','i','n','d','o','w','s',' ','N','T','\\',
65 				   'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
66 				   'D','r','i','v','e','r','s','3','2','\0'};
67     WCHAR buf[2048];
68     DWORD bufLen, lRet;
69     HKEY hKey;
70     PWINE_ACMDRIVERID padid = NULL;
71 
72     /* The requested registry entry must have the format msacm.XXXXX in order to
73        be recognized in any future sessions of msacm
74      */
75     if (0 == strncmpiW(pszRegEntry, msacmW, ARRAY_SIZE(msacmW))) {
76         lRet = RegOpenKeyExW(HKEY_LOCAL_MACHINE, drvkey, 0, KEY_QUERY_VALUE, &hKey);
77         if (lRet != ERROR_SUCCESS) {
78             WARN("unable to open registry key - 0x%08x\n", lRet);
79         } else {
80             bufLen = sizeof(buf);
81             lRet = RegQueryValueExW(hKey, pszRegEntry, NULL, NULL, (LPBYTE)buf, &bufLen);
82             if (lRet != ERROR_SUCCESS) {
83                 WARN("unable to query requested subkey %s - 0x%08x\n", debugstr_w(pszRegEntry), lRet);
84             } else {
85                 MSACM_RegisterDriver(pszRegEntry, buf, 0);
86             }
87             RegCloseKey( hKey );
88         }
89     }
90     return padid;
91 }
92 
93 #if 0
94 /***********************************************************************
95  *           MSACM_DumpCache
96  */
97 static	void MSACM_DumpCache(PWINE_ACMDRIVERID padid)
98 {
99     unsigned 	i;
100 
101     TRACE("cFilterTags=%lu cFormatTags=%lu fdwSupport=%08lx\n",
102 	  padid->cFilterTags, padid->cFormatTags, padid->fdwSupport);
103     for (i = 0; i < padid->cache->cFormatTags; i++) {
104 	TRACE("\tdwFormatTag=%lu cbwfx=%lu\n",
105 	      padid->aFormatTag[i].dwFormatTag, padid->aFormatTag[i].cbwfx);
106     }
107 }
108 #endif
109 
110 /***********************************************************************
111  *           MSACM_FindFormatTagInCache 		[internal]
112  *
113  *	Returns TRUE is the format tag fmtTag is present in the cache.
114  *	If so, idx is set to its index.
115  */
116 BOOL MSACM_FindFormatTagInCache(const WINE_ACMDRIVERID* padid, DWORD fmtTag, LPDWORD idx)
117 {
118     unsigned 	i;
119 
120     for (i = 0; i < padid->cFormatTags; i++) {
121 	if (padid->aFormatTag[i].dwFormatTag == fmtTag) {
122 	    if (idx) *idx = i;
123 	    return TRUE;
124 	}
125     }
126     return FALSE;
127 }
128 
129 /***********************************************************************
130  *           MSACM_FillCache
131  */
132 static BOOL MSACM_FillCache(PWINE_ACMDRIVERID padid)
133 {
134     HACMDRIVER		        had = 0;
135     unsigned int		        ntag;
136     ACMDRIVERDETAILSW	        add;
137     ACMFORMATTAGDETAILSW        aftd;
138 
139     if (acmDriverOpen(&had, (HACMDRIVERID)padid, 0) != 0)
140 	return FALSE;
141 
142     padid->aFormatTag = NULL;
143     add.cbStruct = sizeof(add);
144     if (MSACM_Message(had, ACMDM_DRIVER_DETAILS, (LPARAM)&add,  0))
145 	goto errCleanUp;
146 
147     if (add.cFormatTags > 0) {
148 	padid->aFormatTag = HeapAlloc(MSACM_hHeap, HEAP_ZERO_MEMORY,
149 				      add.cFormatTags * sizeof(padid->aFormatTag[0]));
150 	if (!padid->aFormatTag) goto errCleanUp;
151     }
152 
153     padid->cFormatTags = add.cFormatTags;
154     padid->cFilterTags = add.cFilterTags;
155     padid->fdwSupport  = add.fdwSupport;
156 
157     aftd.cbStruct = sizeof(aftd);
158 
159     for (ntag = 0; ntag < add.cFormatTags; ntag++) {
160 	aftd.dwFormatTagIndex = ntag;
161 	if (MSACM_Message(had, ACMDM_FORMATTAG_DETAILS, (LPARAM)&aftd, ACM_FORMATTAGDETAILSF_INDEX)) {
162 	    TRACE("IIOs (%s)\n", debugstr_w(padid->pszDriverAlias));
163 	    goto errCleanUp;
164 	}
165 	padid->aFormatTag[ntag].dwFormatTag = aftd.dwFormatTag;
166 	padid->aFormatTag[ntag].cbwfx = aftd.cbFormatSize;
167     }
168 
169     acmDriverClose(had, 0);
170 
171     return TRUE;
172 
173 errCleanUp:
174     if (had) acmDriverClose(had, 0);
175     HeapFree(MSACM_hHeap, 0, padid->aFormatTag);
176     padid->aFormatTag = NULL;
177     return FALSE;
178 }
179 
180 /***********************************************************************
181  *           MSACM_GetRegistryKey
182  */
183 static	LPWSTR	MSACM_GetRegistryKey(const WINE_ACMDRIVERID* padid)
184 {
185     static const WCHAR	baseKey[] = {'S','o','f','t','w','a','r','e','\\','M','i','c','r','o','s','o','f','t','\\',
186                                      'A','u','d','i','o','C','o','m','p','r','e','s','s','i','o','n','M','a','n','a','g','e','r','\\',
187                                      'D','r','i','v','e','r','C','a','c','h','e','\\','\0'};
188     LPWSTR	ret;
189     int		len;
190 
191     if (!padid->pszDriverAlias) {
192 	ERR("No alias needed for registry entry\n");
193 	return NULL;
194     }
195     len = strlenW(baseKey);
196     ret = HeapAlloc(MSACM_hHeap, 0, (len + strlenW(padid->pszDriverAlias) + 1) * sizeof(WCHAR));
197     if (!ret) return NULL;
198 
199     strcpyW(ret, baseKey);
200     strcpyW(ret + len, padid->pszDriverAlias);
201     CharLowerW(ret + len);
202     return ret;
203 }
204 
205 /***********************************************************************
206  *           MSACM_ReadCache
207  */
208 static BOOL MSACM_ReadCache(PWINE_ACMDRIVERID padid)
209 {
210     LPWSTR	key = MSACM_GetRegistryKey(padid);
211     HKEY	hKey;
212     DWORD	type, size;
213 
214     if (!key) return FALSE;
215 
216     padid->aFormatTag = NULL;
217 
218     if (RegCreateKeyW(HKEY_LOCAL_MACHINE, key, &hKey))
219 	goto errCleanUp;
220 
221     size = sizeof(padid->cFormatTags);
222     if (RegQueryValueExA(hKey, "cFormatTags", 0, &type, (void*)&padid->cFormatTags, &size))
223 	goto errCleanUp;
224     size = sizeof(padid->cFilterTags);
225     if (RegQueryValueExA(hKey, "cFilterTags", 0, &type, (void*)&padid->cFilterTags, &size))
226 	goto errCleanUp;
227     size = sizeof(padid->fdwSupport);
228     if (RegQueryValueExA(hKey, "fdwSupport", 0, &type, (void*)&padid->fdwSupport, &size))
229 	goto errCleanUp;
230 
231     if (padid->cFormatTags > 0) {
232 	size = padid->cFormatTags * sizeof(padid->aFormatTag[0]);
233 	padid->aFormatTag = HeapAlloc(MSACM_hHeap, HEAP_ZERO_MEMORY, size);
234 	if (!padid->aFormatTag) goto errCleanUp;
235 	if (RegQueryValueExA(hKey, "aFormatTagCache", 0, &type, (void*)padid->aFormatTag, &size))
236 	    goto errCleanUp;
237     }
238     HeapFree(MSACM_hHeap, 0, key);
239     return TRUE;
240 
241  errCleanUp:
242     HeapFree(MSACM_hHeap, 0, key);
243     HeapFree(MSACM_hHeap, 0, padid->aFormatTag);
244     padid->aFormatTag = NULL;
245     RegCloseKey(hKey);
246     return FALSE;
247 }
248 
249 /***********************************************************************
250  *           MSACM_WriteCache
251  */
252 static	BOOL MSACM_WriteCache(const WINE_ACMDRIVERID *padid)
253 {
254     LPWSTR	key = MSACM_GetRegistryKey(padid);
255     HKEY	hKey;
256 
257     if (!key) return FALSE;
258 
259     if (RegCreateKeyW(HKEY_LOCAL_MACHINE, key, &hKey))
260 	goto errCleanUp;
261 
262     if (RegSetValueExA(hKey, "cFormatTags", 0, REG_DWORD, (const void*)&padid->cFormatTags, sizeof(DWORD)))
263 	goto errCleanUp;
264     if (RegSetValueExA(hKey, "cFilterTags", 0, REG_DWORD, (const void*)&padid->cFilterTags, sizeof(DWORD)))
265 	goto errCleanUp;
266     if (RegSetValueExA(hKey, "fdwSupport", 0, REG_DWORD, (const void*)&padid->fdwSupport, sizeof(DWORD)))
267 	goto errCleanUp;
268     if (RegSetValueExA(hKey, "aFormatTagCache", 0, REG_BINARY,
269 		       (void*)padid->aFormatTag,
270 		       padid->cFormatTags * sizeof(padid->aFormatTag[0])))
271 	goto errCleanUp;
272     HeapFree(MSACM_hHeap, 0, key);
273     return TRUE;
274 
275  errCleanUp:
276     HeapFree(MSACM_hHeap, 0, key);
277     return FALSE;
278 }
279 
280 /***********************************************************************
281  *           MSACM_RegisterDriver()
282  */
283 PWINE_ACMDRIVERID MSACM_RegisterDriver(LPCWSTR pszDriverAlias, LPCWSTR pszFileName,
284 				       PWINE_ACMLOCALDRIVER pLocalDriver)
285 {
286     PWINE_ACMDRIVERID	padid;
287 
288     TRACE("(%s, %s, %p)\n",
289           debugstr_w(pszDriverAlias), debugstr_w(pszFileName), pLocalDriver);
290 
291     padid = HeapAlloc(MSACM_hHeap, 0, sizeof(WINE_ACMDRIVERID));
292     if (!padid)
293         return NULL;
294     padid->obj.dwType = WINE_ACMOBJ_DRIVERID;
295     padid->obj.pACMDriverID = padid;
296     padid->pszDriverAlias = NULL;
297     if (pszDriverAlias)
298     {
299         padid->pszDriverAlias = HeapAlloc( MSACM_hHeap, 0, (strlenW(pszDriverAlias)+1) * sizeof(WCHAR) );
300         if (!padid->pszDriverAlias) {
301             HeapFree(MSACM_hHeap, 0, padid);
302             return NULL;
303         }
304         strcpyW( padid->pszDriverAlias, pszDriverAlias );
305     }
306     padid->pszFileName = NULL;
307     if (pszFileName)
308     {
309         padid->pszFileName = HeapAlloc( MSACM_hHeap, 0, (strlenW(pszFileName)+1) * sizeof(WCHAR) );
310         if (!padid->pszFileName) {
311             HeapFree(MSACM_hHeap, 0, padid->pszDriverAlias);
312             HeapFree(MSACM_hHeap, 0, padid);
313             return NULL;
314         }
315         strcpyW( padid->pszFileName, pszFileName );
316     }
317     padid->pLocalDriver = pLocalDriver;
318 
319     padid->pACMDriverList = NULL;
320 
321     if (pLocalDriver) {
322         padid->pPrevACMDriverID = NULL;
323         padid->pNextACMDriverID = MSACM_pFirstACMDriverID;
324         if (MSACM_pFirstACMDriverID)
325             MSACM_pFirstACMDriverID->pPrevACMDriverID = padid;
326         MSACM_pFirstACMDriverID = padid;
327         if (!MSACM_pLastACMDriverID)
328             MSACM_pLastACMDriverID = padid;
329     } else {
330         padid->pNextACMDriverID = NULL;
331         padid->pPrevACMDriverID = MSACM_pLastACMDriverID;
332         if (MSACM_pLastACMDriverID)
333 	    MSACM_pLastACMDriverID->pNextACMDriverID = padid;
334         MSACM_pLastACMDriverID = padid;
335         if (!MSACM_pFirstACMDriverID)
336 	    MSACM_pFirstACMDriverID = padid;
337     }
338     /* disable the driver if we cannot load the cache */
339     if ((!padid->pszDriverAlias || !MSACM_ReadCache(padid)) && !MSACM_FillCache(padid)) {
340 	WARN("Couldn't load cache for ACM driver (%s)\n", debugstr_w(pszFileName));
341 	MSACM_UnregisterDriver(padid);
342 	return NULL;
343     }
344 
345     if (pLocalDriver) padid->fdwSupport |= ACMDRIVERDETAILS_SUPPORTF_LOCAL;
346     return padid;
347 }
348 
349 /***********************************************************************
350  *           MSACM_RegisterAllDrivers()
351  */
352 void MSACM_RegisterAllDrivers(void)
353 {
354     static const WCHAR msacm32[] = {'m','s','a','c','m','3','2','.','d','l','l','\0'};
355     static const WCHAR msacmW[] = {'M','S','A','C','M','.'};
356     static const WCHAR drv32[] = {'d','r','i','v','e','r','s','3','2','\0'};
357     static const WCHAR sys[] = {'s','y','s','t','e','m','.','i','n','i','\0'};
358     static const WCHAR drvkey[] = {'S','o','f','t','w','a','r','e','\\',
359 				   'M','i','c','r','o','s','o','f','t','\\',
360 				   'W','i','n','d','o','w','s',' ','N','T','\\',
361 				   'C','u','r','r','e','n','t','V','e','r','s','i','o','n','\\',
362 				   'D','r','i','v','e','r','s','3','2','\0'};
363     DWORD i, cnt, bufLen, lRet, type;
364     WCHAR buf[2048], valname[64], *name, *s;
365     FILETIME lastWrite;
366     HKEY hKey;
367 
368     /* FIXME: What if the user edits system.ini while the program is running?
369      * Does Windows handle that?  */
370     if (MSACM_pFirstACMDriverID) return;
371 
372     lRet = RegOpenKeyExW(HKEY_LOCAL_MACHINE, drvkey, 0, KEY_QUERY_VALUE, &hKey);
373     if (lRet == ERROR_SUCCESS) {
374 	RegQueryInfoKeyW( hKey, 0, 0, 0, &cnt, 0, 0, 0, 0, 0, 0, 0);
375 	for (i = 0; i < cnt; i++) {
376 	    bufLen = ARRAY_SIZE(buf);
377 	    lRet = RegEnumKeyExW(hKey, i, buf, &bufLen, 0, 0, 0, &lastWrite);
378 	    if (lRet != ERROR_SUCCESS) continue;
379 	    if (strncmpiW(buf, msacmW, ARRAY_SIZE(msacmW))) continue;
380 	    if (!(name = strchrW(buf, '='))) continue;
381 	    *name = 0;
382 	    MSACM_RegisterDriver(buf, name + 1, 0);
383 	}
384 	i = 0;
385 	cnt = ARRAY_SIZE(valname);
386 	bufLen = sizeof(buf);
387 	while(RegEnumValueW(hKey, i, valname, &cnt, 0,
388 		    &type, (BYTE*)buf, &bufLen) == ERROR_SUCCESS){
389 	    if (!strncmpiW(valname, msacmW, ARRAY_SIZE(msacmW)))
390 		MSACM_RegisterDriver(valname, buf, 0);
391 	    ++i;
392 	}
393     	RegCloseKey( hKey );
394     }
395 
396     if (GetPrivateProfileSectionW(drv32, buf, ARRAY_SIZE(buf), sys))
397     {
398 	for(s = buf; *s;  s += strlenW(s) + 1)
399 	{
400 	    if (strncmpiW(s, msacmW, ARRAY_SIZE(msacmW))) continue;
401 	    if (!(name = strchrW(s, '='))) continue;
402 	    *name = 0;
403 	    MSACM_RegisterDriver(s, name + 1, 0);
404 	    *name = '=';
405 	}
406     }
407     MSACM_ReorderDriversByPriority();
408     MSACM_RegisterDriver(msacm32, msacm32, 0);
409 }
410 
411 /***********************************************************************
412  *           MSACM_RegisterNotificationWindow()
413  */
414 PWINE_ACMNOTIFYWND MSACM_RegisterNotificationWindow(HWND hNotifyWnd, DWORD dwNotifyMsg)
415 {
416     PWINE_ACMNOTIFYWND	panwnd;
417 
418     TRACE("(%p,0x%08x)\n", hNotifyWnd, dwNotifyMsg);
419 
420     panwnd = HeapAlloc(MSACM_hHeap, 0, sizeof(WINE_ACMNOTIFYWND));
421     panwnd->obj.dwType = WINE_ACMOBJ_NOTIFYWND;
422     panwnd->obj.pACMDriverID = 0;
423     panwnd->hNotifyWnd = hNotifyWnd;
424     panwnd->dwNotifyMsg = dwNotifyMsg;
425     panwnd->fdwSupport = 0;
426 
427     panwnd->pNextACMNotifyWnd = NULL;
428     panwnd->pPrevACMNotifyWnd = MSACM_pLastACMNotifyWnd;
429     if (MSACM_pLastACMNotifyWnd)
430         MSACM_pLastACMNotifyWnd->pNextACMNotifyWnd = panwnd;
431     MSACM_pLastACMNotifyWnd = panwnd;
432     if (!MSACM_pFirstACMNotifyWnd)
433         MSACM_pFirstACMNotifyWnd = panwnd;
434 
435     return panwnd;
436 }
437 
438 /***********************************************************************
439  *           MSACM_BroadcastNotification()
440  */
441 void MSACM_BroadcastNotification(void)
442 {
443     if (MSACM_suspendBroadcastCount <= 0) {
444         PWINE_ACMNOTIFYWND panwnd;
445 
446         for (panwnd = MSACM_pFirstACMNotifyWnd; panwnd; panwnd = panwnd->pNextACMNotifyWnd)
447         if (!(panwnd->fdwSupport & ACMDRIVERDETAILS_SUPPORTF_DISABLED))
448             SendMessageW(panwnd->hNotifyWnd, panwnd->dwNotifyMsg, 0, 0);
449     } else {
450         MSACM_pendingBroadcast = TRUE;
451     }
452 }
453 
454 /***********************************************************************
455  *           MSACM_DisableNotifications()
456  */
457 void MSACM_DisableNotifications(void)
458 {
459     MSACM_suspendBroadcastCount++;
460 }
461 
462 /***********************************************************************
463  *           MSACM_EnableNotifications()
464  */
465 void MSACM_EnableNotifications(void)
466 {
467     if (MSACM_suspendBroadcastCount > 0) {
468         MSACM_suspendBroadcastCount--;
469         if (MSACM_suspendBroadcastCount == 0 && MSACM_pendingBroadcast) {
470             MSACM_pendingBroadcast = FALSE;
471             MSACM_BroadcastNotification();
472         }
473     }
474 }
475 
476 /***********************************************************************
477  *           MSACM_UnRegisterNotificationWindow()
478  */
479 PWINE_ACMNOTIFYWND MSACM_UnRegisterNotificationWindow(const WINE_ACMNOTIFYWND *panwnd)
480 {
481     PWINE_ACMNOTIFYWND p;
482 
483     for (p = MSACM_pFirstACMNotifyWnd; p; p = p->pNextACMNotifyWnd) {
484         if (p == panwnd) {
485             PWINE_ACMNOTIFYWND pNext = p->pNextACMNotifyWnd;
486 
487             if (p->pPrevACMNotifyWnd) p->pPrevACMNotifyWnd->pNextACMNotifyWnd = p->pNextACMNotifyWnd;
488             if (p->pNextACMNotifyWnd) p->pNextACMNotifyWnd->pPrevACMNotifyWnd = p->pPrevACMNotifyWnd;
489             if (MSACM_pFirstACMNotifyWnd == p) MSACM_pFirstACMNotifyWnd = p->pNextACMNotifyWnd;
490             if (MSACM_pLastACMNotifyWnd == p) MSACM_pLastACMNotifyWnd = p->pPrevACMNotifyWnd;
491             HeapFree(MSACM_hHeap, 0, p);
492 
493             return pNext;
494         }
495     }
496     return NULL;
497 }
498 
499 /***********************************************************************
500  *           MSACM_RePositionDriver()
501  */
502 void MSACM_RePositionDriver(PWINE_ACMDRIVERID padid, DWORD dwPriority)
503 {
504     PWINE_ACMDRIVERID pTargetPosition = NULL;
505 
506     /* Remove selected driver from linked list */
507     if (MSACM_pFirstACMDriverID == padid) {
508         MSACM_pFirstACMDriverID = padid->pNextACMDriverID;
509     }
510     if (MSACM_pLastACMDriverID == padid) {
511         MSACM_pLastACMDriverID = padid->pPrevACMDriverID;
512     }
513     if (padid->pPrevACMDriverID != NULL) {
514         padid->pPrevACMDriverID->pNextACMDriverID = padid->pNextACMDriverID;
515     }
516     if (padid->pNextACMDriverID != NULL) {
517         padid->pNextACMDriverID->pPrevACMDriverID = padid->pPrevACMDriverID;
518     }
519 
520     /* Look up position where selected driver should be */
521     if (dwPriority == 1) {
522         pTargetPosition = padid->pPrevACMDriverID;
523         while (pTargetPosition->pPrevACMDriverID != NULL &&
524             !(pTargetPosition->pPrevACMDriverID->fdwSupport & ACMDRIVERDETAILS_SUPPORTF_LOCAL)) {
525             pTargetPosition = pTargetPosition->pPrevACMDriverID;
526         }
527     } else if (dwPriority == -1) {
528         pTargetPosition = padid->pNextACMDriverID;
529         while (pTargetPosition->pNextACMDriverID != NULL) {
530             pTargetPosition = pTargetPosition->pNextACMDriverID;
531         }
532     }
533 
534     /* Place selected driver in selected position */
535     padid->pPrevACMDriverID = pTargetPosition->pPrevACMDriverID;
536     padid->pNextACMDriverID = pTargetPosition;
537     if (padid->pPrevACMDriverID != NULL) {
538         padid->pPrevACMDriverID->pNextACMDriverID = padid;
539     } else {
540         MSACM_pFirstACMDriverID = padid;
541     }
542     if (padid->pNextACMDriverID != NULL) {
543         padid->pNextACMDriverID->pPrevACMDriverID = padid;
544     } else {
545         MSACM_pLastACMDriverID = padid;
546     }
547 }
548 
549 /***********************************************************************
550  *           MSACM_ReorderDriversByPriority()
551  * Reorders all drivers based on the priority list indicated by the registry key:
552  * HKCU\\Software\\Microsoft\\Multimedia\\Audio Compression Manager\\Priority v4.00
553  */
554 static void MSACM_ReorderDriversByPriority(void)
555 {
556     PWINE_ACMDRIVERID	padid;
557     unsigned int iNumDrivers;
558     PWINE_ACMDRIVERID * driverList = NULL;
559     HKEY hPriorityKey = NULL;
560 
561     TRACE("\n");
562 
563     /* Count drivers && alloc corresponding memory for list */
564     iNumDrivers = 0;
565     for (padid = MSACM_pFirstACMDriverID; padid; padid = padid->pNextACMDriverID) iNumDrivers++;
566     if (iNumDrivers > 1)
567     {
568         LONG lError;
569         static const WCHAR basePriorityKey[] = {
570             'S','o','f','t','w','a','r','e','\\',
571             'M','i','c','r','o','s','o','f','t','\\',
572             'M','u','l','t','i','m','e','d','i','a','\\',
573             'A','u','d','i','o',' ','C','o','m','p','r','e','s','s','i','o','n',' ','M','a','n','a','g','e','r','\\',
574             'P','r','i','o','r','i','t','y',' ','v','4','.','0','0','\0'
575         };
576         unsigned int i;
577         LONG lBufferLength;
578         WCHAR szBuffer[256];
579 
580         driverList = HeapAlloc(MSACM_hHeap, 0, iNumDrivers * sizeof(PWINE_ACMDRIVERID));
581         if (!driverList)
582         {
583             ERR("out of memory\n");
584             goto errCleanUp;
585         }
586 
587         lError = RegOpenKeyW(HKEY_CURRENT_USER, basePriorityKey, &hPriorityKey);
588         if (lError != ERROR_SUCCESS) {
589             TRACE("RegOpenKeyW failed, possibly key does not exist yet\n");
590             hPriorityKey = NULL;
591             goto errCleanUp;
592         }
593 
594         /* Copy drivers into list to simplify linked list modification */
595         for (i = 0, padid = MSACM_pFirstACMDriverID; padid; padid = padid->pNextACMDriverID, i++)
596         {
597             driverList[i] = padid;
598         }
599 
600         /* Query each of the priorities in turn. Alias key is in lowercase.
601             The general form of the priority record is the following:
602             "PriorityN" --> "1, msacm.driveralias"
603             where N is an integer, and the value is a string of the driver
604             alias, prefixed by "1, " for an enabled driver, or "0, " for a
605             disabled driver.
606             */
607         for (i = 0; i < iNumDrivers; i++)
608         {
609             static const WCHAR priorityTmpl[] = {'P','r','i','o','r','i','t','y','%','l','d','\0'};
610             WCHAR szSubKey[17];
611             unsigned int iTargetPosition;
612             unsigned int iCurrentPosition;
613             WCHAR * pAlias;
614             static const WCHAR sPrefix[] = {'m','s','a','c','m','.','\0'};
615 
616             /* Build expected entry name */
617             snprintfW(szSubKey, 17, priorityTmpl, i + 1);
618             lBufferLength = sizeof(szBuffer);
619             lError = RegQueryValueExW(hPriorityKey, szSubKey, NULL, NULL, (LPBYTE)szBuffer, (LPDWORD)&lBufferLength);
620             if (lError != ERROR_SUCCESS) continue;
621 
622             /* Recovered driver alias should be at this position */
623             iTargetPosition = i;
624 
625             /* Locate driver alias in driver list */
626             pAlias = strstrW(szBuffer, sPrefix);
627             if (pAlias == NULL) continue;
628 
629             for (iCurrentPosition = 0; iCurrentPosition < iNumDrivers; iCurrentPosition++) {
630                 if (strcmpiW(driverList[iCurrentPosition]->pszDriverAlias, pAlias) == 0)
631                     break;
632             }
633             if (iCurrentPosition < iNumDrivers && iTargetPosition != iCurrentPosition) {
634                 padid = driverList[iTargetPosition];
635                 driverList[iTargetPosition] = driverList[iCurrentPosition];
636                 driverList[iCurrentPosition] = padid;
637 
638                 /* Locate enabled status */
639                 if (szBuffer[0] == '1') {
640                     driverList[iTargetPosition]->fdwSupport &= ~ACMDRIVERDETAILS_SUPPORTF_DISABLED;
641                 } else if (szBuffer[0] == '0') {
642                     driverList[iTargetPosition]->fdwSupport |= ACMDRIVERDETAILS_SUPPORTF_DISABLED;
643                 }
644             }
645         }
646 
647         /* Re-assign pointers so that linked list traverses the ordered array */
648         for (i = 0; i < iNumDrivers; i++) {
649             driverList[i]->pPrevACMDriverID = (i > 0) ? driverList[i - 1] : NULL;
650             driverList[i]->pNextACMDriverID = (i < iNumDrivers - 1) ? driverList[i + 1] : NULL;
651         }
652         MSACM_pFirstACMDriverID = driverList[0];
653         MSACM_pLastACMDriverID = driverList[iNumDrivers - 1];
654     }
655 
656 errCleanUp:
657     if (hPriorityKey != NULL) RegCloseKey(hPriorityKey);
658     HeapFree(MSACM_hHeap, 0, driverList);
659 }
660 
661 /***********************************************************************
662  *           MSACM_WriteCurrentPriorities()
663  * Writes out current order of driver priorities to registry key:
664  * HKCU\\Software\\Microsoft\\Multimedia\\Audio Compression Manager\\Priority v4.00
665  */
666 void MSACM_WriteCurrentPriorities(void)
667 {
668     LONG lError;
669     HKEY hPriorityKey;
670     static const WCHAR basePriorityKey[] = {
671         'S','o','f','t','w','a','r','e','\\',
672         'M','i','c','r','o','s','o','f','t','\\',
673         'M','u','l','t','i','m','e','d','i','a','\\',
674         'A','u','d','i','o',' ','C','o','m','p','r','e','s','s','i','o','n',' ','M','a','n','a','g','e','r','\\',
675         'P','r','i','o','r','i','t','y',' ','v','4','.','0','0','\0'
676     };
677     PWINE_ACMDRIVERID padid;
678     DWORD dwPriorityCounter;
679     static const WCHAR priorityTmpl[] = {'P','r','i','o','r','i','t','y','%','l','d','\0'};
680     static const WCHAR valueTmpl[] = {'%','c',',',' ','%','s','\0'};
681     static const WCHAR converterAlias[] = {'I','n','t','e','r','n','a','l',' ','P','C','M',' ','C','o','n','v','e','r','t','e','r','\0'};
682     WCHAR szSubKey[17];
683     WCHAR szBuffer[256];
684 
685     /* Delete ACM priority key and create it anew */
686     lError = RegDeleteKeyW(HKEY_CURRENT_USER, basePriorityKey);
687     if (lError != ERROR_SUCCESS && lError != ERROR_FILE_NOT_FOUND) {
688         ERR("unable to remove current key %s (0x%08x) - priority changes won't persist past application end.\n",
689             debugstr_w(basePriorityKey), lError);
690         return;
691     }
692     lError = RegCreateKeyW(HKEY_CURRENT_USER, basePriorityKey, &hPriorityKey);
693     if (lError != ERROR_SUCCESS) {
694         ERR("unable to create key %s (0x%08x) - priority changes won't persist past application end.\n",
695             debugstr_w(basePriorityKey), lError);
696         return;
697     }
698 
699     /* Write current list of priorities */
700     for (dwPriorityCounter = 0, padid = MSACM_pFirstACMDriverID; padid; padid = padid->pNextACMDriverID) {
701         if (padid->fdwSupport & ACMDRIVERDETAILS_SUPPORTF_LOCAL) continue;
702         if (padid->pszDriverAlias == NULL) continue;    /* internal PCM converter is last */
703 
704         /* Build required value name */
705         dwPriorityCounter++;
706         snprintfW(szSubKey, 17, priorityTmpl, dwPriorityCounter);
707 
708         /* Value has a 1 in front for enabled drivers and 0 for disabled drivers */
709         snprintfW(szBuffer, 256, valueTmpl, (padid->fdwSupport & ACMDRIVERDETAILS_SUPPORTF_DISABLED) ? '0' : '1', padid->pszDriverAlias);
710         strlwrW(szBuffer);
711 
712         lError = RegSetValueExW(hPriorityKey, szSubKey, 0, REG_SZ, (BYTE *)szBuffer, (strlenW(szBuffer) + 1) * sizeof(WCHAR));
713         if (lError != ERROR_SUCCESS) {
714             ERR("unable to write value for %s under key %s (0x%08x)\n",
715                 debugstr_w(padid->pszDriverAlias), debugstr_w(basePriorityKey), lError);
716         }
717     }
718 
719     /* Build required value name */
720     dwPriorityCounter++;
721     snprintfW(szSubKey, 17, priorityTmpl, dwPriorityCounter);
722 
723     /* Value has a 1 in front for enabled drivers and 0 for disabled drivers */
724     snprintfW(szBuffer, 256, valueTmpl, '1', converterAlias);
725 
726     lError = RegSetValueExW(hPriorityKey, szSubKey, 0, REG_SZ, (BYTE *)szBuffer, (strlenW(szBuffer) + 1) * sizeof(WCHAR));
727     if (lError != ERROR_SUCCESS) {
728         ERR("unable to write value for %s under key %s (0x%08x)\n",
729             debugstr_w(converterAlias), debugstr_w(basePriorityKey), lError);
730     }
731     RegCloseKey(hPriorityKey);
732 }
733 
734 static PWINE_ACMLOCALDRIVER MSACM_pFirstACMLocalDriver;
735 static PWINE_ACMLOCALDRIVER MSACM_pLastACMLocalDriver;
736 
737 static PWINE_ACMLOCALDRIVER MSACM_UnregisterLocalDriver(PWINE_ACMLOCALDRIVER paldrv)
738 {
739     PWINE_ACMLOCALDRIVER pNextACMLocalDriver;
740 
741     if (paldrv->pACMInstList) {
742         ERR("local driver instances still present after closing all drivers - memory leak\n");
743         return NULL;
744     }
745 
746     if (paldrv == MSACM_pFirstACMLocalDriver)
747         MSACM_pFirstACMLocalDriver = paldrv->pNextACMLocalDrv;
748     if (paldrv == MSACM_pLastACMLocalDriver)
749         MSACM_pLastACMLocalDriver = paldrv->pPrevACMLocalDrv;
750 
751     if (paldrv->pPrevACMLocalDrv)
752         paldrv->pPrevACMLocalDrv->pNextACMLocalDrv = paldrv->pNextACMLocalDrv;
753     if (paldrv->pNextACMLocalDrv)
754         paldrv->pNextACMLocalDrv->pPrevACMLocalDrv = paldrv->pPrevACMLocalDrv;
755 
756     pNextACMLocalDriver = paldrv->pNextACMLocalDrv;
757 
758     HeapFree(MSACM_hHeap, 0, paldrv);
759 
760     return pNextACMLocalDriver;
761 }
762 
763 /***********************************************************************
764  *           MSACM_UnregisterDriver()
765  */
766 PWINE_ACMDRIVERID MSACM_UnregisterDriver(PWINE_ACMDRIVERID p)
767 {
768     PWINE_ACMDRIVERID pNextACMDriverID;
769 
770     while (p->pACMDriverList)
771 	acmDriverClose((HACMDRIVER) p->pACMDriverList, 0);
772 
773     HeapFree(MSACM_hHeap, 0, p->pszDriverAlias);
774     HeapFree(MSACM_hHeap, 0, p->pszFileName);
775     HeapFree(MSACM_hHeap, 0, p->aFormatTag);
776 
777     if (p == MSACM_pFirstACMDriverID)
778 	MSACM_pFirstACMDriverID = p->pNextACMDriverID;
779     if (p == MSACM_pLastACMDriverID)
780 	MSACM_pLastACMDriverID = p->pPrevACMDriverID;
781 
782     if (p->pPrevACMDriverID)
783 	p->pPrevACMDriverID->pNextACMDriverID = p->pNextACMDriverID;
784     if (p->pNextACMDriverID)
785 	p->pNextACMDriverID->pPrevACMDriverID = p->pPrevACMDriverID;
786 
787     pNextACMDriverID = p->pNextACMDriverID;
788 
789     if (p->pLocalDriver) MSACM_UnregisterLocalDriver(p->pLocalDriver);
790     HeapFree(MSACM_hHeap, 0, p);
791 
792     return pNextACMDriverID;
793 }
794 
795 /***********************************************************************
796  *           MSACM_UnregisterAllDrivers()
797  */
798 void MSACM_UnregisterAllDrivers(void)
799 {
800     PWINE_ACMNOTIFYWND panwnd = MSACM_pFirstACMNotifyWnd;
801     PWINE_ACMDRIVERID p = MSACM_pFirstACMDriverID;
802 
803     while (p) {
804 	MSACM_WriteCache(p);
805 	p = MSACM_UnregisterDriver(p);
806     }
807 
808     while (panwnd) {
809 	panwnd = MSACM_UnRegisterNotificationWindow(panwnd);
810     }
811 }
812 
813 /***********************************************************************
814  *           MSACM_GetObj()
815  */
816 PWINE_ACMOBJ MSACM_GetObj(HACMOBJ hObj, DWORD type)
817 {
818     PWINE_ACMOBJ	pao = (PWINE_ACMOBJ)hObj;
819 
820     if (pao == NULL || IsBadReadPtr(pao, sizeof(WINE_ACMOBJ)) ||
821 	((type != WINE_ACMOBJ_DONTCARE) && (type != pao->dwType)))
822 	return NULL;
823     return pao;
824 }
825 
826 /***********************************************************************
827  *           MSACM_GetDriverID()
828  */
829 PWINE_ACMDRIVERID MSACM_GetDriverID(HACMDRIVERID hDriverID)
830 {
831     return (PWINE_ACMDRIVERID)MSACM_GetObj((HACMOBJ)hDriverID, WINE_ACMOBJ_DRIVERID);
832 }
833 
834 /***********************************************************************
835  *           MSACM_GetDriver()
836  */
837 PWINE_ACMDRIVER MSACM_GetDriver(HACMDRIVER hDriver)
838 {
839     return (PWINE_ACMDRIVER)MSACM_GetObj((HACMOBJ)hDriver, WINE_ACMOBJ_DRIVER);
840 }
841 
842 /***********************************************************************
843  *           MSACM_GetNotifyWnd()
844  */
845 PWINE_ACMNOTIFYWND MSACM_GetNotifyWnd(HACMDRIVERID hDriver)
846 {
847     return (PWINE_ACMNOTIFYWND)MSACM_GetObj((HACMOBJ)hDriver, WINE_ACMOBJ_NOTIFYWND);
848 }
849 
850 /***********************************************************************
851  *           MSACM_GetLocalDriver()
852  */
853 /*
854 PWINE_ACMLOCALDRIVER MSACM_GetLocalDriver(HACMDRIVER hDriver)
855 {
856     return (PWINE_ACMLOCALDRIVER)MSACM_GetObj((HACMOBJ)hDriver, WINE_ACMOBJ_LOCALDRIVER);
857 }
858 */
859 #define MSACM_DRIVER_SendMessage(PDRVRINST, msg, lParam1, lParam2) \
860         (PDRVRINST)->pLocalDriver->lpDrvProc((PDRVRINST)->dwDriverID, (HDRVR)(PDRVRINST), msg, lParam1, lParam2)
861 
862 /***********************************************************************
863  *           MSACM_Message()
864  */
865 MMRESULT MSACM_Message(HACMDRIVER had, UINT uMsg, LPARAM lParam1, LPARAM lParam2)
866 {
867     PWINE_ACMDRIVER	pad = MSACM_GetDriver(had);
868 
869     if (!pad) return MMSYSERR_INVALHANDLE;
870     if (pad->hDrvr) return SendDriverMessage(pad->hDrvr, uMsg, lParam1, lParam2);
871     if (pad->pLocalDrvrInst) return MSACM_DRIVER_SendMessage(pad->pLocalDrvrInst, uMsg, lParam1, lParam2);
872 
873     return MMSYSERR_INVALHANDLE;
874 }
875 
876 PWINE_ACMLOCALDRIVER MSACM_RegisterLocalDriver(HMODULE hModule, DRIVERPROC lpDriverProc)
877 {
878     PWINE_ACMLOCALDRIVER paldrv;
879 
880     TRACE("(%p, %p)\n", hModule, lpDriverProc);
881     if (!hModule || !lpDriverProc) return NULL;
882 
883     /* look up previous instance of local driver module */
884     for (paldrv = MSACM_pFirstACMLocalDriver; paldrv; paldrv = paldrv->pNextACMLocalDrv)
885     {
886         if (paldrv->hModule == hModule && paldrv->lpDrvProc == lpDriverProc) return paldrv;
887     }
888 
889     paldrv = HeapAlloc(MSACM_hHeap, 0, sizeof(WINE_ACMLOCALDRIVER));
890     paldrv->obj.dwType = WINE_ACMOBJ_LOCALDRIVER;
891     paldrv->obj.pACMDriverID = 0;
892     paldrv->hModule = hModule;
893     paldrv->lpDrvProc = lpDriverProc;
894     paldrv->pACMInstList = NULL;
895 
896     paldrv->pNextACMLocalDrv = NULL;
897     paldrv->pPrevACMLocalDrv = MSACM_pLastACMLocalDriver;
898     if (MSACM_pLastACMLocalDriver)
899 	MSACM_pLastACMLocalDriver->pNextACMLocalDrv = paldrv;
900     MSACM_pLastACMLocalDriver = paldrv;
901     if (!MSACM_pFirstACMLocalDriver)
902 	MSACM_pFirstACMLocalDriver = paldrv;
903 
904     return paldrv;
905 }
906 
907 /**************************************************************************
908  *			MSACM_GetNumberOfModuleRefs		[internal]
909  *
910  * Returns the number of open drivers which share the same module.
911  * Inspired from implementation in dlls/winmm/driver.c
912  */
913 static unsigned MSACM_GetNumberOfModuleRefs(HMODULE hModule, DRIVERPROC lpDrvProc, WINE_ACMLOCALDRIVERINST ** found)
914 {
915     PWINE_ACMLOCALDRIVER lpDrv;
916     unsigned		count = 0;
917 
918     if (found) *found = NULL;
919     for (lpDrv = MSACM_pFirstACMLocalDriver; lpDrv; lpDrv = lpDrv->pNextACMLocalDrv)
920     {
921 	if (lpDrv->hModule == hModule && lpDrv->lpDrvProc == lpDrvProc)
922         {
923             PWINE_ACMLOCALDRIVERINST pInst = lpDrv->pACMInstList;
924 
925 	    while (pInst) {
926                 if (found && !*found) *found = pInst;
927 	        count++;
928 	        pInst = pInst->pNextACMInst;
929 	    }
930 	}
931     }
932     return count;
933 }
934 
935 /**************************************************************************
936  *				MSACM_RemoveFromList		[internal]
937  *
938  * Generates all the logic to handle driver closure / deletion
939  * Removes a driver struct to the list of open drivers.
940  */
941 static	BOOL	MSACM_RemoveFromList(PWINE_ACMLOCALDRIVERINST lpDrv)
942 {
943     PWINE_ACMLOCALDRIVER pDriverBase = lpDrv->pLocalDriver;
944     PWINE_ACMLOCALDRIVERINST pPrevInst;
945 
946     /* last of this driver in list ? */
947     if (MSACM_GetNumberOfModuleRefs(pDriverBase->hModule, pDriverBase->lpDrvProc, NULL) == 1) {
948         MSACM_DRIVER_SendMessage(lpDrv, DRV_DISABLE, 0L, 0L);
949         MSACM_DRIVER_SendMessage(lpDrv, DRV_FREE,    0L, 0L);
950     }
951 
952     pPrevInst = NULL;
953     if (pDriverBase->pACMInstList != lpDrv) {
954         pPrevInst = pDriverBase->pACMInstList;
955         while (pPrevInst && pPrevInst->pNextACMInst != lpDrv)
956             pPrevInst = pPrevInst->pNextACMInst;
957         if (!pPrevInst) {
958             ERR("requested to remove invalid instance %p\n", pPrevInst);
959             return FALSE;
960         }
961     }
962     if (!pPrevInst) {
963         /* first driver instance on list */
964         pDriverBase->pACMInstList = lpDrv->pNextACMInst;
965     } else {
966         pPrevInst->pNextACMInst = lpDrv->pNextACMInst;
967     }
968     return TRUE;
969 }
970 
971 /**************************************************************************
972  *				MSACM_AddToList		[internal]
973  *
974  * Adds a driver struct to the list of open drivers.
975  * Generates all the logic to handle driver creation / open.
976  */
977 static	BOOL	MSACM_AddToList(PWINE_ACMLOCALDRIVERINST lpNewDrv, LPARAM lParam2)
978 {
979     PWINE_ACMLOCALDRIVER pDriverBase = lpNewDrv->pLocalDriver;
980 
981     /* first of this driver in list ? */
982     if (MSACM_GetNumberOfModuleRefs(pDriverBase->hModule, pDriverBase->lpDrvProc, NULL) == 0) {
983         if (MSACM_DRIVER_SendMessage(lpNewDrv, DRV_LOAD, 0L, 0L) != DRV_SUCCESS) {
984             FIXME("DRV_LOAD failed on driver %p\n", lpNewDrv);
985             return FALSE;
986         }
987         /* returned value is not checked */
988         MSACM_DRIVER_SendMessage(lpNewDrv, DRV_ENABLE, 0L, 0L);
989     }
990 
991     lpNewDrv->pNextACMInst = NULL;
992     if (pDriverBase->pACMInstList == NULL) {
993 	pDriverBase->pACMInstList = lpNewDrv;
994     } else {
995         PWINE_ACMLOCALDRIVERINST lpDrvInst = pDriverBase->pACMInstList;
996 
997         while (lpDrvInst->pNextACMInst != NULL)
998             lpDrvInst = lpDrvInst->pNextACMInst;
999 
1000 	lpDrvInst->pNextACMInst = lpNewDrv;
1001     }
1002 
1003     /* Now just open a new instance of a driver on this module */
1004     lpNewDrv->dwDriverID = MSACM_DRIVER_SendMessage(lpNewDrv, DRV_OPEN, 0, lParam2);
1005 
1006     if (lpNewDrv->dwDriverID == 0) {
1007         FIXME("DRV_OPEN failed on driver %p\n", lpNewDrv);
1008         MSACM_RemoveFromList(lpNewDrv);
1009         return FALSE;
1010     }
1011     return TRUE;
1012 }
1013 
1014 PWINE_ACMLOCALDRIVERINST MSACM_OpenLocalDriver(PWINE_ACMLOCALDRIVER paldrv, LPARAM lParam2)
1015 {
1016     PWINE_ACMLOCALDRIVERINST pDrvInst;
1017 
1018     pDrvInst = HeapAlloc(MSACM_hHeap, 0, sizeof(WINE_ACMLOCALDRIVERINST));
1019     if (!pDrvInst)
1020         return NULL;
1021 
1022     pDrvInst->pLocalDriver = paldrv;
1023     pDrvInst->dwDriverID = 0;
1024     pDrvInst->pNextACMInst = NULL;
1025     pDrvInst->bSession = FALSE;
1026 
1027     /* Win32 installable drivers must support a two phase opening scheme:
1028      * + first open with NULL as lParam2 (session instance),
1029      * + then do a second open with the real non null lParam2)
1030      */
1031     if (MSACM_GetNumberOfModuleRefs(paldrv->hModule, paldrv->lpDrvProc, NULL) == 0 && lParam2)
1032     {
1033         PWINE_ACMLOCALDRIVERINST   ret;
1034 
1035         if (!MSACM_AddToList(pDrvInst, 0L))
1036         {
1037             ERR("load0 failed\n");
1038             goto exit;
1039         }
1040         ret = MSACM_OpenLocalDriver(paldrv, lParam2);
1041         if (!ret)
1042         {
1043             ERR("load1 failed\n");
1044             /* If MSACM_CloseLocalDriver returns TRUE,
1045              * then pDrvInst has been freed
1046              */
1047             if (!MSACM_CloseLocalDriver(pDrvInst))
1048                 goto exit;
1049 
1050             return NULL;
1051         }
1052         pDrvInst->bSession = TRUE;
1053         return ret;
1054     }
1055 
1056     if (!MSACM_AddToList(pDrvInst, lParam2))
1057     {
1058         ERR("load failed\n");
1059         goto exit;
1060     }
1061 
1062     TRACE("=> %p\n", pDrvInst);
1063     return pDrvInst;
1064 exit:
1065     HeapFree(MSACM_hHeap, 0, pDrvInst);
1066     return NULL;
1067 }
1068 
1069 LRESULT MSACM_CloseLocalDriver(PWINE_ACMLOCALDRIVERINST paldrv)
1070 {
1071     if (MSACM_RemoveFromList(paldrv)) {
1072         PWINE_ACMLOCALDRIVERINST lpDrv0;
1073         PWINE_ACMLOCALDRIVER pDriverBase = paldrv->pLocalDriver;
1074 
1075         MSACM_DRIVER_SendMessage(paldrv, DRV_CLOSE, 0, 0);
1076         paldrv->dwDriverID = 0;
1077 
1078         if (paldrv->bSession)
1079             ERR("should not directly close session instance (%p)\n", paldrv);
1080 
1081         /* if driver has an opened session instance, we have to close it too */
1082         if (MSACM_GetNumberOfModuleRefs(pDriverBase->hModule, pDriverBase->lpDrvProc, &lpDrv0) == 1 &&
1083                 lpDrv0->bSession)
1084         {
1085             MSACM_DRIVER_SendMessage(lpDrv0, DRV_CLOSE, 0L, 0L);
1086             lpDrv0->dwDriverID = 0;
1087             MSACM_RemoveFromList(lpDrv0);
1088             HeapFree(MSACM_hHeap, 0, lpDrv0);
1089         }
1090 
1091         HeapFree(MSACM_hHeap, 0, paldrv);
1092         return TRUE;
1093     }
1094     ERR("unable to close driver instance\n");
1095     return FALSE;
1096 }
1097