400 lines
12 KiB
C
400 lines
12 KiB
C
/* -*- tab-width: 8; c-basic-offset: 4 -*- */
|
|
|
|
/*
|
|
* MSACM32 library
|
|
*
|
|
* Copyright 1998 Patrik Stridvall
|
|
* 1999 Eric Pouech
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
#include <string.h>
|
|
|
|
#include "winbase.h"
|
|
#include "windef.h"
|
|
#include "wingdi.h"
|
|
#include "winuser.h"
|
|
#include "winerror.h"
|
|
#include "winreg.h"
|
|
#include "mmsystem.h"
|
|
#include "msacm.h"
|
|
#include "msacmdrv.h"
|
|
#include "wineacm.h"
|
|
#include "wine/debug.h"
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(msacm);
|
|
|
|
/**********************************************************************/
|
|
|
|
HANDLE MSACM_hHeap = (HANDLE) NULL;
|
|
PWINE_ACMDRIVERID MSACM_pFirstACMDriverID = NULL;
|
|
PWINE_ACMDRIVERID MSACM_pLastACMDriverID = NULL;
|
|
|
|
#if 0
|
|
/***********************************************************************
|
|
* MSACM_DumpCache
|
|
*/
|
|
static void MSACM_DumpCache(PWINE_ACMDRIVERID padid)
|
|
{
|
|
unsigned i;
|
|
|
|
TRACE("cFilterTags=%lu cFormatTags=%lu fdwSupport=%08lx\n",
|
|
padid->cFilterTags, padid->cFormatTags, padid->fdwSupport);
|
|
for (i = 0; i < padid->cache->cFormatTags; i++) {
|
|
TRACE("\tdwFormatTag=%lu cbwfx=%lu\n",
|
|
padid->aFormatTag[i].dwFormatTag, padid->aFormatTag[i].cbwfx);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/***********************************************************************
|
|
* MSACM_FindFormatTagInCache [internal]
|
|
*
|
|
* Returns TRUE is the format tag fmtTag is present in the cache.
|
|
* If so, idx is set to its index.
|
|
*/
|
|
BOOL MSACM_FindFormatTagInCache(WINE_ACMDRIVERID* padid, DWORD fmtTag, LPDWORD idx)
|
|
{
|
|
unsigned i;
|
|
|
|
for (i = 0; i < padid->cFormatTags; i++) {
|
|
if (padid->aFormatTag[i].dwFormatTag == fmtTag) {
|
|
if (idx) *idx = i;
|
|
return TRUE;
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MSACM_FillCache
|
|
*/
|
|
static BOOL MSACM_FillCache(PWINE_ACMDRIVERID padid)
|
|
{
|
|
HACMDRIVER had = 0;
|
|
int ntag;
|
|
ACMDRIVERDETAILSW add;
|
|
ACMFORMATTAGDETAILSW aftd;
|
|
|
|
if (acmDriverOpen(&had, (HACMDRIVERID)padid, 0) != 0)
|
|
return FALSE;
|
|
|
|
padid->aFormatTag = NULL;
|
|
add.cbStruct = sizeof(add);
|
|
if (MSACM_Message(had, ACMDM_DRIVER_DETAILS, (LPARAM)&add, 0))
|
|
goto errCleanUp;
|
|
|
|
if (add.cFormatTags > 0) {
|
|
padid->aFormatTag = HeapAlloc(MSACM_hHeap, HEAP_ZERO_MEMORY,
|
|
add.cFormatTags * sizeof(padid->aFormatTag[0]));
|
|
if (!padid->aFormatTag) goto errCleanUp;
|
|
}
|
|
|
|
padid->cFormatTags = add.cFormatTags;
|
|
padid->cFilterTags = add.cFilterTags;
|
|
padid->fdwSupport = add.fdwSupport;
|
|
|
|
aftd.cbStruct = sizeof(aftd);
|
|
|
|
for (ntag = 0; ntag < add.cFormatTags; ntag++) {
|
|
aftd.dwFormatTagIndex = ntag;
|
|
if (MSACM_Message(had, ACMDM_FORMATTAG_DETAILS, (LPARAM)&aftd, ACM_FORMATTAGDETAILSF_INDEX)) {
|
|
TRACE("IIOs (%s)\n", padid->pszDriverAlias);
|
|
goto errCleanUp;
|
|
}
|
|
padid->aFormatTag[ntag].dwFormatTag = aftd.dwFormatTag;
|
|
padid->aFormatTag[ntag].cbwfx = aftd.cbFormatSize;
|
|
}
|
|
|
|
acmDriverClose(had, 0);
|
|
|
|
return TRUE;
|
|
|
|
errCleanUp:
|
|
if (had) acmDriverClose(had, 0);
|
|
HeapFree(MSACM_hHeap, 0, padid->aFormatTag);
|
|
padid->aFormatTag = NULL;
|
|
return FALSE;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MSACM_GetRegistryKey
|
|
*/
|
|
static LPSTR MSACM_GetRegistryKey(const WINE_ACMDRIVERID* padid)
|
|
{
|
|
static const char* baseKey = "Software\\Microsoft\\AudioCompressionManager\\DriverCache\\";
|
|
LPSTR ret;
|
|
int len;
|
|
|
|
if (!padid->pszDriverAlias) {
|
|
ERR("No alias needed for registry entry\n");
|
|
return NULL;
|
|
}
|
|
len = strlen(baseKey);
|
|
ret = HeapAlloc(MSACM_hHeap, 0, len + strlen(padid->pszDriverAlias) + 1);
|
|
if (!ret) return NULL;
|
|
|
|
strcpy(ret, baseKey);
|
|
strcpy(ret + len, padid->pszDriverAlias);
|
|
CharLowerA(ret + len);
|
|
return ret;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MSACM_ReadCache
|
|
*/
|
|
static BOOL MSACM_ReadCache(PWINE_ACMDRIVERID padid)
|
|
{
|
|
LPSTR key = MSACM_GetRegistryKey(padid);
|
|
HKEY hKey;
|
|
DWORD type, size;
|
|
|
|
if (!key) return FALSE;
|
|
|
|
padid->aFormatTag = NULL;
|
|
|
|
if (RegCreateKeyA(HKEY_LOCAL_MACHINE, key, &hKey))
|
|
goto errCleanUp;
|
|
|
|
size = sizeof(padid->cFormatTags);
|
|
if (RegQueryValueExA(hKey, "cFormatTags", 0, &type, (void*)&padid->cFormatTags, &size))
|
|
goto errCleanUp;
|
|
size = sizeof(padid->cFilterTags);
|
|
if (RegQueryValueExA(hKey, "cFilterTags", 0, &type, (void*)&padid->cFilterTags, &size))
|
|
goto errCleanUp;
|
|
size = sizeof(padid->fdwSupport);
|
|
if (RegQueryValueExA(hKey, "fdwSupport", 0, &type, (void*)&padid->fdwSupport, &size))
|
|
goto errCleanUp;
|
|
|
|
if (padid->cFormatTags > 0) {
|
|
size = padid->cFormatTags * sizeof(padid->aFormatTag[0]);
|
|
padid->aFormatTag = HeapAlloc(MSACM_hHeap, HEAP_ZERO_MEMORY, size);
|
|
if (!padid->aFormatTag) goto errCleanUp;
|
|
if (RegQueryValueExA(hKey, "aFormatTagCache", 0, &type, (void*)padid->aFormatTag, &size))
|
|
goto errCleanUp;
|
|
}
|
|
HeapFree(MSACM_hHeap, 0, key);
|
|
return TRUE;
|
|
|
|
errCleanUp:
|
|
HeapFree(MSACM_hHeap, 0, key);
|
|
HeapFree(MSACM_hHeap, 0, padid->aFormatTag);
|
|
padid->aFormatTag = NULL;
|
|
RegCloseKey(hKey);
|
|
return FALSE;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MSACM_WriteCache
|
|
*/
|
|
static BOOL MSACM_WriteCache(PWINE_ACMDRIVERID padid)
|
|
{
|
|
LPSTR key = MSACM_GetRegistryKey(padid);
|
|
HKEY hKey;
|
|
|
|
if (!key) return FALSE;
|
|
|
|
if (RegCreateKeyA(HKEY_LOCAL_MACHINE, key, &hKey))
|
|
goto errCleanUp;
|
|
|
|
if (RegSetValueExA(hKey, "cFormatTags", 0, REG_DWORD, (void*)&padid->cFormatTags, sizeof(DWORD)))
|
|
goto errCleanUp;
|
|
if (RegSetValueExA(hKey, "cFilterTags", 0, REG_DWORD, (void*)&padid->cFilterTags, sizeof(DWORD)))
|
|
goto errCleanUp;
|
|
if (RegSetValueExA(hKey, "fdwSupport", 0, REG_DWORD, (void*)&padid->fdwSupport, sizeof(DWORD)))
|
|
goto errCleanUp;
|
|
if (RegSetValueExA(hKey, "aFormatTagCache", 0, REG_BINARY,
|
|
(void*)padid->aFormatTag,
|
|
padid->cFormatTags * sizeof(padid->aFormatTag[0])))
|
|
goto errCleanUp;
|
|
HeapFree(MSACM_hHeap, 0, key);
|
|
return TRUE;
|
|
|
|
errCleanUp:
|
|
HeapFree(MSACM_hHeap, 0, key);
|
|
return FALSE;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MSACM_RegisterDriver()
|
|
*/
|
|
PWINE_ACMDRIVERID MSACM_RegisterDriver(LPSTR pszDriverAlias, LPSTR pszFileName,
|
|
HINSTANCE hinstModule)
|
|
{
|
|
PWINE_ACMDRIVERID padid;
|
|
|
|
TRACE("('%s', '%s', 0x%08x)\n", pszDriverAlias, pszFileName, hinstModule);
|
|
|
|
padid = (PWINE_ACMDRIVERID) HeapAlloc(MSACM_hHeap, 0, sizeof(WINE_ACMDRIVERID));
|
|
padid->obj.dwType = WINE_ACMOBJ_DRIVERID;
|
|
padid->obj.pACMDriverID = padid;
|
|
padid->pszDriverAlias = NULL;
|
|
if (pszDriverAlias)
|
|
{
|
|
padid->pszDriverAlias = HeapAlloc( MSACM_hHeap, 0, strlen(pszDriverAlias)+1 );
|
|
strcpy( padid->pszDriverAlias, pszDriverAlias );
|
|
}
|
|
padid->pszFileName = NULL;
|
|
if (pszFileName)
|
|
{
|
|
padid->pszFileName = HeapAlloc( MSACM_hHeap, 0, strlen(pszFileName)+1 );
|
|
strcpy( padid->pszFileName, pszFileName );
|
|
}
|
|
padid->hInstModule = hinstModule;
|
|
|
|
padid->pACMDriverList = NULL;
|
|
padid->pNextACMDriverID = NULL;
|
|
padid->pPrevACMDriverID = MSACM_pLastACMDriverID;
|
|
if (MSACM_pLastACMDriverID)
|
|
MSACM_pLastACMDriverID->pNextACMDriverID = padid;
|
|
MSACM_pLastACMDriverID = padid;
|
|
if (!MSACM_pFirstACMDriverID)
|
|
MSACM_pFirstACMDriverID = padid;
|
|
/* disable the driver if we cannot load the cache */
|
|
if (!MSACM_ReadCache(padid) && !MSACM_FillCache(padid)) {
|
|
WARN("Couldn't load cache for ACM driver (%s)\n", pszFileName);
|
|
MSACM_UnregisterDriver(padid);
|
|
return NULL;
|
|
}
|
|
return padid;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MSACM_RegisterAllDrivers()
|
|
*/
|
|
void MSACM_RegisterAllDrivers(void)
|
|
{
|
|
LPSTR pszBuffer;
|
|
DWORD dwBufferLength;
|
|
|
|
/* FIXME
|
|
* What if the user edits system.ini while the program is running?
|
|
* Does Windows handle that?
|
|
*/
|
|
if (MSACM_pFirstACMDriverID)
|
|
return;
|
|
|
|
/* FIXME: Does not work! How do I determine the section length? */
|
|
dwBufferLength = 1024;
|
|
/* EPP GetPrivateProfileSectionA("drivers32", NULL, 0, "system.ini"); */
|
|
|
|
pszBuffer = (LPSTR) HeapAlloc(MSACM_hHeap, 0, dwBufferLength);
|
|
if (GetPrivateProfileSectionA("drivers32", pszBuffer, dwBufferLength, "system.ini")) {
|
|
char* s = pszBuffer;
|
|
while (*s) {
|
|
if (!strncasecmp("MSACM.", s, 6)) {
|
|
char *s2 = s;
|
|
while (*s2 != '\0' && *s2 != '=') s2++;
|
|
if (*s2) {
|
|
*s2 = '\0';
|
|
MSACM_RegisterDriver(s, s2 + 1, 0);
|
|
*s2 = '=';
|
|
}
|
|
}
|
|
s += strlen(s) + 1; /* Either next char or \0 */
|
|
}
|
|
}
|
|
|
|
HeapFree(MSACM_hHeap, 0, pszBuffer);
|
|
|
|
MSACM_RegisterDriver("msacm32.dll", "msacm32.dll", 0);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MSACM_UnregisterDriver()
|
|
*/
|
|
PWINE_ACMDRIVERID MSACM_UnregisterDriver(PWINE_ACMDRIVERID p)
|
|
{
|
|
PWINE_ACMDRIVERID pNextACMDriverID;
|
|
|
|
while (p->pACMDriverList)
|
|
acmDriverClose((HACMDRIVER) p->pACMDriverList, 0);
|
|
|
|
if (p->pszDriverAlias)
|
|
HeapFree(MSACM_hHeap, 0, p->pszDriverAlias);
|
|
if (p->pszFileName)
|
|
HeapFree(MSACM_hHeap, 0, p->pszFileName);
|
|
HeapFree(MSACM_hHeap, 0, p->aFormatTag);
|
|
|
|
if (p == MSACM_pFirstACMDriverID)
|
|
MSACM_pFirstACMDriverID = p->pNextACMDriverID;
|
|
if (p == MSACM_pLastACMDriverID)
|
|
MSACM_pLastACMDriverID = p->pPrevACMDriverID;
|
|
|
|
if (p->pPrevACMDriverID)
|
|
p->pPrevACMDriverID->pNextACMDriverID = p->pNextACMDriverID;
|
|
if (p->pNextACMDriverID)
|
|
p->pNextACMDriverID->pPrevACMDriverID = p->pPrevACMDriverID;
|
|
|
|
pNextACMDriverID = p->pNextACMDriverID;
|
|
|
|
HeapFree(MSACM_hHeap, 0, p);
|
|
|
|
return pNextACMDriverID;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MSACM_UnregisterAllDrivers()
|
|
*/
|
|
void MSACM_UnregisterAllDrivers(void)
|
|
{
|
|
PWINE_ACMDRIVERID p = MSACM_pFirstACMDriverID;
|
|
|
|
while (p) {
|
|
MSACM_WriteCache(p);
|
|
p = MSACM_UnregisterDriver(p);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MSACM_GetObj()
|
|
*/
|
|
PWINE_ACMOBJ MSACM_GetObj(HACMOBJ hObj, DWORD type)
|
|
{
|
|
PWINE_ACMOBJ pao = (PWINE_ACMOBJ)hObj;
|
|
|
|
if (pao == NULL || IsBadReadPtr(pao, sizeof(WINE_ACMOBJ)) ||
|
|
((type != WINE_ACMOBJ_DONTCARE) && (type != pao->dwType)))
|
|
return NULL;
|
|
return pao;
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MSACM_GetDriverID()
|
|
*/
|
|
PWINE_ACMDRIVERID MSACM_GetDriverID(HACMDRIVERID hDriverID)
|
|
{
|
|
return (PWINE_ACMDRIVERID)MSACM_GetObj((HACMOBJ)hDriverID, WINE_ACMOBJ_DRIVERID);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MSACM_GetDriver()
|
|
*/
|
|
PWINE_ACMDRIVER MSACM_GetDriver(HACMDRIVER hDriver)
|
|
{
|
|
return (PWINE_ACMDRIVER)MSACM_GetObj((HACMOBJ)hDriver, WINE_ACMOBJ_DRIVER);
|
|
}
|
|
|
|
/***********************************************************************
|
|
* MSACM_Message()
|
|
*/
|
|
MMRESULT MSACM_Message(HACMDRIVER had, UINT uMsg, LPARAM lParam1, LPARAM lParam2)
|
|
{
|
|
PWINE_ACMDRIVER pad = MSACM_GetDriver(had);
|
|
|
|
return pad ? SendDriverMessage(pad->hDrvr, uMsg, lParam1, lParam2) : MMSYSERR_INVALHANDLE;
|
|
}
|