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