|
|
/****************************************************************************
* * wdmaud32.c * * 32-bit specific interfaces for WDMAUD * * Copyright (C) Microsoft Corporation, 1997 - 1999 All Rights Reserved. * * History * 5-12-97 - Noel Cross (NoelC) * ***************************************************************************/
#include "wdmdrv.h"
HANDLE ghCallbacks = NULL; PCALLBACKS gpCallbacks = NULL;
HANDLE ghDevice = NULL;
//
// gpszDeviceInterfacePath is a pointer to the device interface string
// that represents wdmaud.sys to this driver.
//
LPWSTR gpszDeviceInterfacePath = NULL;
BOOL wdmaudCritSecInit; CRITICAL_SECTION wdmaudCritSec; static TCHAR gszCallbacks[] = TEXT("Global\\WDMAUD_Callbacks");
extern HANDLE mixercallbackevent; extern HANDLE mixerhardwarecallbackevent; extern HANDLE mixercallbackthread;
DWORD waveThread(LPDEVICEINFO DeviceInfo); DWORD midThread(LPDEVICEINFO DeviceInfo);
DWORD sndTranslateStatus();
typedef struct _SETUPAPIDLINFO { HINSTANCE hInstSetupAPI; BOOL (WINAPI *pfnDestroyDeviceInfoList)(HDEVINFO); BOOL (WINAPI *pfnGetDeviceInterfaceDetailW)(HDEVINFO, PSP_DEVICE_INTERFACE_DATA, PSP_DEVICE_INTERFACE_DETAIL_DATA_W, DWORD, PDWORD, PSP_DEVINFO_DATA); BOOL (WINAPI *pfnEnumDeviceInterfaces)(HDEVINFO, PSP_DEVINFO_DATA, CONST GUID*, DWORD, PSP_DEVICE_INTERFACE_DATA); HDEVINFO (WINAPI *pfnGetClassDevsW)(CONST GUID*, PCWSTR, HWND, DWORD); } SETUPAPIDLINFO;
SETUPAPIDLINFO saInfo = {NULL, NULL, NULL, NULL};
/****************************************************************************
Dynalinking setupapi
****************************************************************************/
BOOL Init_SetupAPI(); BOOL End_SetupAPI();
BOOL dl_SetupDiDestroyDeviceInfoList( HDEVINFO DeviceInfoSet );
BOOL dl_SetupDiGetDeviceInterfaceDetail( HDEVINFO DeviceInfoSet, PSP_DEVICE_INTERFACE_DATA DeviceInterfaceData, PSP_DEVICE_INTERFACE_DETAIL_DATA_W DeviceInterfaceDetailData, DWORD DeviceInterfaceDetailDataSize, PDWORD RequiredSize, PSP_DEVINFO_DATA DeviceInfoData );
BOOL dl_SetupDiEnumDeviceInterfaces( HDEVINFO DeviceInfoSet, PSP_DEVINFO_DATA DeviceInfoData, CONST GUID *InterfaceClassGuid, DWORD MemberIndex, PSP_DEVICE_INTERFACE_DATA DeviceInterfaceData );
HDEVINFO dl_SetupDiGetClassDevs( CONST GUID *ClassGuid, PCWSTR Enumerator, HWND hwndParent, DWORD Flags );
/****************************************************************************
Setting up SetupAPI Dynalink...
****************************************************************************/
BOOL Init_SetupAPI( void ) { if(NULL != saInfo.hInstSetupAPI) { return TRUE; }
DPF(DL_TRACE|FA_SETUP, ("Loading SetupAPI!!!") ); saInfo.hInstSetupAPI = LoadLibrary(TEXT("setupapi.dll"));
if(NULL == saInfo.hInstSetupAPI) { return FALSE; }
saInfo.pfnDestroyDeviceInfoList = (LPVOID)GetProcAddress(saInfo.hInstSetupAPI, "SetupDiDestroyDeviceInfoList"); saInfo.pfnGetDeviceInterfaceDetailW = (LPVOID)GetProcAddress(saInfo.hInstSetupAPI, "SetupDiGetDeviceInterfaceDetailW"); saInfo.pfnEnumDeviceInterfaces = (LPVOID)GetProcAddress(saInfo.hInstSetupAPI, "SetupDiEnumDeviceInterfaces"); saInfo.pfnGetClassDevsW = (LPVOID)GetProcAddress(saInfo.hInstSetupAPI, "SetupDiGetClassDevsW");
if ((NULL == saInfo.pfnDestroyDeviceInfoList) || (NULL == saInfo.pfnGetDeviceInterfaceDetailW) || (NULL == saInfo.pfnEnumDeviceInterfaces) || (NULL == saInfo.pfnGetClassDevsW)) { FreeLibrary(saInfo.hInstSetupAPI); ZeroMemory(&saInfo, sizeof(saInfo)); return FALSE; }
return TRUE; }
BOOL End_SetupAPI( void ) { HINSTANCE hInst;
hInst = saInfo.hInstSetupAPI;
if(NULL == hInst) { DPF(DL_WARNING|FA_SETUP, ("SetupAPI not dynalinked") ); return FALSE; }
ZeroMemory(&saInfo, sizeof(saInfo)); FreeLibrary(hInst);
return TRUE; }
BOOL dl_SetupDiDestroyDeviceInfoList( HDEVINFO DeviceInfoSet ) { if (NULL == saInfo.hInstSetupAPI) { DPF(DL_WARNING|FA_SETUP, ("SetupAPI not dynalinked") ); return FALSE; }
return (saInfo.pfnDestroyDeviceInfoList)(DeviceInfoSet); }
BOOL dl_SetupDiGetDeviceInterfaceDetail( HDEVINFO DeviceInfoSet, PSP_DEVICE_INTERFACE_DATA DeviceInterfaceData, PSP_DEVICE_INTERFACE_DETAIL_DATA_W DeviceInterfaceDetailData, DWORD DeviceInterfaceDetailDataSize, PDWORD RequiredSize, PSP_DEVINFO_DATA DeviceInfoData ) { if (NULL == saInfo.hInstSetupAPI) { DPF(DL_WARNING|FA_SETUP, ("SetupAPI not dynalinked") ); return FALSE; }
return (saInfo.pfnGetDeviceInterfaceDetailW)(DeviceInfoSet, DeviceInterfaceData, DeviceInterfaceDetailData, DeviceInterfaceDetailDataSize, RequiredSize, DeviceInfoData); }
BOOL dl_SetupDiEnumDeviceInterfaces( HDEVINFO DeviceInfoSet, PSP_DEVINFO_DATA DeviceInfoData, CONST GUID *InterfaceClassGuid, DWORD MemberIndex, PSP_DEVICE_INTERFACE_DATA DeviceInterfaceData ) { if (NULL == saInfo.hInstSetupAPI) { DPF(DL_WARNING|FA_SETUP, ("SetupAPI not dynalinked") ); return FALSE; }
return (saInfo.pfnEnumDeviceInterfaces)(DeviceInfoSet, DeviceInfoData, InterfaceClassGuid, MemberIndex, DeviceInterfaceData); }
HDEVINFO dl_SetupDiGetClassDevs( CONST GUID *ClassGuid, PCWSTR Enumerator, HWND hwndParent, DWORD Flags ) { if (NULL == saInfo.hInstSetupAPI) { DPF(DL_WARNING|FA_SETUP, ("SetupAPI not dynalinked") ); return FALSE; }
return (saInfo.pfnGetClassDevsW)(ClassGuid, Enumerator, hwndParent, Flags); }
PSECURITY_DESCRIPTOR BuildSecurityDescriptor( DWORD AccessMask ) { PSECURITY_DESCRIPTOR pSd; PSID pSidSystem; PSID pSidEveryone; PACL pDacl; ULONG cbDacl; BOOL fSuccess; BOOL f;
SID_IDENTIFIER_AUTHORITY AuthorityNt = SECURITY_NT_AUTHORITY; SID_IDENTIFIER_AUTHORITY AuthorityWorld = SECURITY_WORLD_SID_AUTHORITY;
fSuccess = FALSE;
pSd = HeapAlloc(GetProcessHeap(), 0, SECURITY_DESCRIPTOR_MIN_LENGTH); if (pSd) { if (InitializeSecurityDescriptor(pSd, SECURITY_DESCRIPTOR_REVISION)) { if (AllocateAndInitializeSid(&AuthorityNt, 1, SECURITY_LOCAL_SYSTEM_RID, 0, 0, 0, 0, 0, 0, 0, &pSidSystem)) { DPFASSERT(IsValidSid(pSidSystem)); if (AllocateAndInitializeSid(&AuthorityWorld, 1, SECURITY_WORLD_RID, 0, 0, 0, 0, 0, 0, 0, &pSidEveryone)) { DPFASSERT(IsValidSid(pSidEveryone)); cbDacl = sizeof(ACL) + 2 * (sizeof(ACCESS_ALLOWED_ACE) - sizeof(DWORD)) + GetLengthSid(pSidSystem) + GetLengthSid(pSidEveryone);
pDacl = HeapAlloc(GetProcessHeap(), 0, cbDacl); if (pDacl) { if (InitializeAcl(pDacl, cbDacl, ACL_REVISION)) { if (AddAccessAllowedAce(pDacl, ACL_REVISION, GENERIC_ALL, pSidSystem)) { if (AddAccessAllowedAce(pDacl, ACL_REVISION, AccessMask, pSidEveryone)) { if (SetSecurityDescriptorDacl(pSd, TRUE, pDacl, FALSE)) { fSuccess = TRUE; } else { DPF(DL_WARNING|FA_SETUP, ("BuildSD: SetSecurityDescriptorDacl failed")); } } else { DPF(DL_WARNING|FA_SETUP, ("BuildSD: AddAccessAlloweAce for Everyone failed")); } } else { DPF(DL_WARNING|FA_SETUP, ("BuildSD: AddAccessAllowedAce for System failed")); } } else { DPF(DL_WARNING|FA_SETUP, ("BuildSD: InitializeAcl failed")); }
if (!fSuccess) { f = HeapFree(GetProcessHeap(), 0, pDacl); DPFASSERT(f); } } FreeSid(pSidEveryone); } else { DPF(DL_WARNING|FA_SETUP, ("BuildSD: AllocateAndInitizeSid failed for Everyone")); } FreeSid(pSidSystem); } else { DPF(DL_WARNING|FA_SETUP, ("BuildSD: AllocateAndInitizeSid failed for System")); } } else { DPF(DL_WARNING|FA_SETUP, ("BuildSD: InitializeSecurityDescriptor failed")); }
if (!fSuccess) { f = HeapFree(GetProcessHeap(), 0, pSd); DPFASSERT(f); } }
return fSuccess ? pSd : NULL; }
void DestroySecurityDescriptor( PSECURITY_DESCRIPTOR pSd ) { PACL pDacl; BOOL fDaclPresent; BOOL fDaclDefaulted; BOOL f;
if (GetSecurityDescriptorDacl(pSd, &fDaclPresent, &pDacl, &fDaclDefaulted)) { if (fDaclPresent) { f = HeapFree(GetProcessHeap(), 0, pDacl); DPFASSERT(f); } } else { DPF(DL_WARNING|FA_SETUP, ("DestroySD: GetSecurityDescriptorDacl failed")); }
f = HeapFree(GetProcessHeap(), 0, pSd); DPFASSERT(f);
return; }
/**************************************************************************
@doc EXTERNAL
@api BOOL | DllEntryPoint | This procedure is called whenever a process attaches or detaches from the DLL.
@rdesc The return value is TRUE if the initialisation completed ok, FALSE if not.
**************************************************************************/ BOOL WINAPI DllEntryPoint( HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved ) { BOOL bRet;
if (fdwReason == DLL_PROCESS_ATTACH) { DisableThreadLibraryCalls(hinstDLL);
bRet = LibMain((HANDLE)hinstDLL, 0, NULL); } else { if (fdwReason == DLL_PROCESS_DETACH) { DPF(DL_TRACE|FA_ALL, ("Ending") ); DrvEnd(); }
bRet = TRUE; }
return bRet; }
/**************************************************************************
@doc EXTERNAL
@api BOOL | DrvInit | Driver initialization takes place here.
@rdesc The return value is TRUE if the initialisation completed ok, FALSE if not.
**************************************************************************/
BOOL DrvInit( ) { if (NULL == ghDevice) { ghDevice = wdmaOpenKernelDevice();
if(INVALID_HANDLE_VALUE == ghDevice) { ghDevice = NULL; return 0L; } }
if(NULL == gpCallbacks) { if(NULL == (gpCallbacks = wdmaGetCallbacks())) { gpCallbacks = wdmaCreateCallbacks(); } }
try { wdmaudCritSecInit = FALSE; InitializeCriticalSection(&wdmaudCritSec); } except(EXCEPTION_EXECUTE_HANDLER) { return 0L; }
wdmaudCritSecInit = TRUE;
return ( 1L ) ; }
PCALLBACKS wdmaGetCallbacks( ) { PCALLBACKS pCallbacks = NULL;
if(NULL == gpCallbacks) { ghCallbacks = OpenFileMapping(FILE_MAP_READ|FILE_MAP_WRITE, FALSE, gszCallbacks); if(NULL != ghCallbacks) { pCallbacks = MapViewOfFile(ghCallbacks, FILE_MAP_READ|FILE_MAP_WRITE, 0, 0, sizeof(CALLBACKS)); if(NULL == pCallbacks) { CloseHandle(ghCallbacks); ghCallbacks = NULL; } } } return (pCallbacks); }
PCALLBACKS wdmaCreateCallbacks( ) { SECURITY_ATTRIBUTES saCallbacks; PSECURITY_DESCRIPTOR pSdCallbacks; PCALLBACKS pCallbacks = NULL;
pSdCallbacks = BuildSecurityDescriptor(FILE_MAP_READ|FILE_MAP_WRITE);
if(NULL == pSdCallbacks) { return (NULL); }
saCallbacks.nLength = sizeof(SECURITY_ATTRIBUTES); saCallbacks.lpSecurityDescriptor = pSdCallbacks; saCallbacks.bInheritHandle = FALSE;
ghCallbacks = CreateFileMapping(GetCurrentProcess(), &saCallbacks, PAGE_READWRITE, 0, sizeof(CALLBACKS), gszCallbacks);
DestroySecurityDescriptor(pSdCallbacks);
if(NULL == ghCallbacks) { return (NULL); }
pCallbacks = (PCALLBACKS) MapViewOfFile(ghCallbacks, FILE_MAP_READ|FILE_MAP_WRITE, 0, 0, sizeof(CALLBACKS)); if(NULL == pCallbacks) { CloseHandle(ghCallbacks); ghCallbacks = NULL; return (NULL); }
pCallbacks->GlobalIndex = 0; return (pCallbacks); }
/*
This routine is the one that grovels setup API looking for the device path for wdmaud.sys. Once it is found, we allocate a global block of memory to store it in. When we call CreateFile, we use this string. */ LPWSTR wdmaGetGlobalDeviceInterfaceViaSetupAPI( ) { LPWSTR pszInterfacePath = NULL; HDEVINFO hDeviceInfoSet = NULL; SP_DEVICE_INTERFACE_DATA DeviceInterfaceData; PSP_DEVICE_INTERFACE_DETAIL_DATA pDeviceInterfaceDetailData = NULL; BOOL fResult; DWORD dwSize; GUID guidWDMAUD = KSCATEGORY_WDMAUD;
//
// Because setupapi is such a pig, we must dynaload it in order to keep it from slowing
// down all processes.
//
if (!Init_SetupAPI()) return NULL;
//
// Open the device information set
//
hDeviceInfoSet = dl_SetupDiGetClassDevs(&guidWDMAUD, NULL, NULL, DIGCF_PRESENT | DIGCF_INTERFACEDEVICE);
if((!hDeviceInfoSet) || (INVALID_HANDLE_VALUE == hDeviceInfoSet)) { DPF(DL_WARNING|FA_SETUP, ("Can't open device info set (%lu)", GetLastError()) ); fResult = FALSE; } else { fResult = TRUE; }
if (fResult) { //
// Get the first interface in the set
//
DeviceInterfaceData.cbSize = sizeof(DeviceInterfaceData); fResult = dl_SetupDiEnumDeviceInterfaces(hDeviceInfoSet, NULL, &guidWDMAUD, 0, &DeviceInterfaceData); if(!fResult) { DPF(DL_WARNING|FA_SETUP, ("No interfaces matching KSCATEGORY_WDMAUD exist") ); } }
//
// Get the interface's path
//
if (fResult) { fResult = dl_SetupDiGetDeviceInterfaceDetail(hDeviceInfoSet, &DeviceInterfaceData, NULL, 0, &dwSize, NULL); //
// because SetupApi reverses their logic here
//
if(fResult || ERROR_INSUFFICIENT_BUFFER != GetLastError()) { DPF(DL_WARNING|FA_SETUP, ("Can't get interface detail size (%lu)", GetLastError())); fResult = FALSE; } else { fResult = TRUE; }
if (fResult) { pDeviceInterfaceDetailData = (PSP_DEVICE_INTERFACE_DETAIL_DATA) GlobalAllocPtr( GPTR, dwSize ); if (NULL == pDeviceInterfaceDetailData) { fResult = FALSE; } }
if (fResult) { pDeviceInterfaceDetailData->cbSize = sizeof(*pDeviceInterfaceDetailData); fResult = dl_SetupDiGetDeviceInterfaceDetail(hDeviceInfoSet, &DeviceInterfaceData, pDeviceInterfaceDetailData, dwSize, NULL, NULL); if (!fResult) { GlobalFreePtr(pDeviceInterfaceDetailData); DPF(DL_WARNING|FA_SETUP, ("Can't get device interface detail (%lu)", GetLastError()) ); } }
if (fResult) { //
// Here, we have the Device Interface name. Let's allocate a block
// of memory to hold it and save it for later.
//
DPFASSERT(NULL == gpszDeviceInterfacePath); gpszDeviceInterfacePath = (LPWSTR) GlobalAllocPtr( GPTR, sizeof(WCHAR)*(lstrlenW(pDeviceInterfaceDetailData->DevicePath) + 1)); if (NULL == gpszDeviceInterfacePath) { fResult = FALSE; } else { //
// We now store the device interface name.
//
lstrcpyW(gpszDeviceInterfacePath, pDeviceInterfaceDetailData->DevicePath); }
GlobalFreePtr(pDeviceInterfaceDetailData); } }
if((hDeviceInfoSet) && (INVALID_HANDLE_VALUE != hDeviceInfoSet)) { dl_SetupDiDestroyDeviceInfoList(hDeviceInfoSet); }
End_SetupAPI();
return gpszDeviceInterfacePath; }
/*
We should be able to remove wdmaGetDeviceInterface and call wdmaGetGlobalDeviceInterfaceViaSetupAPI directly. wdmaGetDeviceInterface only gets the cached string. */ HANDLE wdmaOpenKernelDevice( ) { HANDLE hDevice = INVALID_HANDLE_VALUE;
if( NULL == gpszDeviceInterfacePath ) { wdmaGetGlobalDeviceInterfaceViaSetupAPI(); } if (gpszDeviceInterfacePath) { // Open the interface
hDevice = CreateFile(gpszDeviceInterfacePath, GENERIC_READ | GENERIC_WRITE, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OVERLAPPED, NULL );
if((!hDevice) || (INVALID_HANDLE_VALUE == hDevice)) { DPF(DL_WARNING|FA_SETUP, ("CreateFile failed to open %S with error %lu", gpszDeviceInterfacePath, GetLastError()) ); } } else { DPF(DL_WARNING|FA_SETUP, ("wdmaOpenKernelDevice failed with NULL pathname" )); }
return hDevice; }
/**************************************************************************
@doc EXTERNAL
@api void | DrvEnd | Driver cleanup takes place here.
@rdesc The return value is TRUE if the initialisation completed ok, FALSE if not.
**************************************************************************/ VOID DrvEnd() { if (gpszDeviceInterfacePath) { GlobalFreePtr(gpszDeviceInterfacePath); gpszDeviceInterfacePath = NULL; }
if (NULL != ghDevice) { CloseHandle(ghDevice); ghDevice = NULL; } if (NULL != gpCallbacks) { UnmapViewOfFile(gpCallbacks); gpCallbacks = NULL; }
if (NULL != ghCallbacks) { CloseHandle(ghCallbacks); ghCallbacks = NULL; }
if (wdmaudCritSecInit) { wdmaudCritSecInit=FALSE; DeleteCriticalSection(&wdmaudCritSec); }
if( NULL != mixercallbackevent ) { DPF(DL_WARNING|FA_ALL,("freeing mixercallbackevent") ); CloseHandle(mixercallbackevent); mixercallbackevent=NULL; }
if( NULL != mixerhardwarecallbackevent ) { DPF(DL_WARNING|FA_ALL,("freeing mixerhardwarecallbackevent") ); CloseHandle(mixerhardwarecallbackevent); mixerhardwarecallbackevent=NULL; }
if( NULL != mixercallbackthread ) { DPF(DL_WARNING|FA_ALL,("freeing mixercallbackthread") ); CloseHandle(mixercallbackthread); mixercallbackthread=NULL; }
return; }
/****************************************************************************
* @doc INTERNAL * * @api DWORD | wdmaudGetDevCaps | This function returns the device capabilities * of a WDM driver. * * @parm DWORD | id | Device id * * @parm UINT | DeviceType | type of device * * @parm LPBYTE | lpCaps | Far pointer to a WAVEOUTCAPS structure to * receive the information. * * @parm DWORD | dwSize | Size of the WAVEOUTCAPS structure. * * @rdesc MMSYS.. return code ***************************************************************************/ MMRESULT FAR wdmaudGetDevCaps ( LPDEVICEINFO DeviceInfo, MDEVICECAPSEX FAR* pdc ) { if (pdc->cbSize == 0) return MMSYSERR_NOERROR;
//
// Make sure that we don't take the critical section
// in wdmaudIoControl
//
DeviceInfo->OpenDone = 0;
//
// Inject a tag into the devcaps to signify that it is
// Unicode
//
((LPWAVEOUTCAPS)pdc->pCaps)->wMid = UNICODE_TAG;
return wdmaudIoControl(DeviceInfo, pdc->cbSize, pdc->pCaps, IOCTL_WDMAUD_GET_CAPABILITIES); }
/**************************************************************************
@doc EXTERNAL
@api void | wdmaudIoControl | Proxies requests for information to and from wdmaud.sys. This routine is synchronous.
@rdesc The return value is TRUE if the initialisation completed ok, FALSE if not.
**************************************************************************/ /*
Note: wdmaudIoControl calls wdmaud.sys through the DeviceIoControl routine. Take note that if wdmaud.sys returns an error, like STATUS_INVALID_PARAMETER or STATUS_INSUFFICIENT_RESOURCES the output buffer will not get filled! DeviceIoControl will only fill that buffer on STATUS_SUCCESS.
Why is this important to know? Well, wdmaud.sys takes advantage of this in order to return specific error codes. In other words, in order for wdmaud.sys to return MIXERR_INVALCONTROL it returns STATUS_SUCCESS with the mmr value of the DeviceInfo structure set to MIXERR_INVALCONTROL.
*/ MMRESULT FAR wdmaudIoControl ( LPDEVICEINFO DeviceInfo, DWORD dwSize, PVOID pData, ULONG IoCode ) { BOOL fResult; MMRESULT mmr; OVERLAPPED ov; ULONG cbDeviceInfo; ULONG cbReturned;
if (NULL == ghDevice) { MMRRETURN( MMSYSERR_NOTENABLED ); }
RtlZeroMemory( &ov, sizeof( OVERLAPPED ) ); if (NULL == (ov.hEvent = CreateEvent( NULL, TRUE, FALSE, NULL ))) MMRRETURN( MMSYSERR_NOMEM );
//
// Only take the critical section if there is an open
// wave handle. This is to ensure that the lpWaveQueue
// is not modified during the ioctl. Without this
// protection the copy at the end of the DeviceIoControl
// could copy an old DeviceInfo that is not in sync
// with the current DeviceInfo.
//
if ( (DeviceInfo->DeviceType != MixerDevice) && (DeviceInfo->DeviceType != AuxDevice) && (DeviceInfo->OpenDone == 1) ) CRITENTER ;
//
// Wrap the data buffer around the device context.
//
DeviceInfo->DataBuffer = pData; DeviceInfo->DataBufferSize = dwSize;
//
// Since we are not letting the OS do the user-to-kernel
// space mapping, we will have to do the mapping ourselves
// for writes to data buffers in wdmaud.sys.
//
cbDeviceInfo = sizeof(*DeviceInfo) + (lstrlenW(DeviceInfo->wstrDeviceInterface) * sizeof(WCHAR));
fResult = DeviceIoControl( ghDevice, IoCode, DeviceInfo, cbDeviceInfo, DeviceInfo, sizeof(*DeviceInfo), &cbReturned, &ov ); if (!fResult) { if (ERROR_IO_PENDING == GetLastError()) { WaitForSingleObject( ov.hEvent, INFINITE ); }
mmr = sndTranslateStatus(); } else { mmr = MMSYSERR_NOERROR; }
if ( (DeviceInfo->DeviceType != MixerDevice) && (DeviceInfo->DeviceType != AuxDevice) && (DeviceInfo->OpenDone == 1) ) CRITLEAVE ;
CloseHandle( ov.hEvent );
MMRRETURN( mmr ); }
/****************************************************************************
* @doc INTERNAL * * @api void | sndTranslateStatus | This function translates an NT status * code into a multimedia error code as far as possible. * * @parm NTSTATUS | Status | The NT base operating system return status. * * @rdesc The multimedia error code. ***************************************************************************/ DWORD sndTranslateStatus() { #if DBG
UINT n; switch (n=GetLastError()) { #else
switch (GetLastError()) { #endif
case NO_ERROR: case ERROR_IO_PENDING: return MMSYSERR_NOERROR;
case ERROR_BUSY: return MMSYSERR_ALLOCATED;
case ERROR_NOT_SUPPORTED: case ERROR_INVALID_FUNCTION: return MMSYSERR_NOTSUPPORTED;
case ERROR_NOT_ENOUGH_MEMORY: case ERROR_NO_SYSTEM_RESOURCES: return MMSYSERR_NOMEM;
case ERROR_ACCESS_DENIED: return MMSYSERR_BADDEVICEID;
case ERROR_INSUFFICIENT_BUFFER: case ERROR_INVALID_PARAMETER: case ERROR_INVALID_USER_BUFFER: return MMSYSERR_INVALPARAM;
case ERROR_NOT_READY: case ERROR_GEN_FAILURE: return MMSYSERR_ERROR;
case ERROR_FILE_NOT_FOUND: return MMSYSERR_NODRIVER;
default: DPF(DL_WARNING|FA_DEVICEIO, ("sndTranslateStatus: LastError = %d", n)); return MMSYSERR_ERROR; } }
/****************************************************************************
* @doc INTERNAL * * @api MMRESULT | wdmaudSubmitWaveHeader | Pass a new buffer to the Auxiliary * thread for a wave device. * * @parm LPWAVEALLOC | DeviceInfo | The data associated with the logical wave * device. * * @parm LPWAVEHDR | pHdr | Pointer to a wave buffer * * @rdesc A MMSYS... type return code for the application. * * @comm The buffer flags are set and the buffer is passed to the auxiliary * device task for processing. ***************************************************************************/ MMRESULT wdmaudSubmitWaveHeader ( LPDEVICEINFO DeviceInfo, LPWAVEHDR pHdr ) { LPDEVICEINFO WaveHeaderDeviceInfo; PWAVEPREPAREDATA pWavePrepareData; ULONG cbRead; ULONG cbWritten; ULONG cbDeviceInfo; BOOL fResult; MMRESULT mmr;
if (NULL == ghDevice) { MMRRETURN( MMSYSERR_NOTENABLED ); }
WaveHeaderDeviceInfo = GlobalAllocDeviceInfo(DeviceInfo->wstrDeviceInterface); if (!WaveHeaderDeviceInfo) { MMRRETURN( MMSYSERR_NOMEM ); }
//
// Catch the case when an application doesn't prepare headers correctly
//
if (!pHdr->reserved) { //
// This should never happen! wdmaudSubmitWaveHeader is called from
// waveWrite which is called from handling the WIDM_ADDBUFFER and
// WODM_WRITE messages. On both of these messages, we check that
// the header has been prepared!
//
DPF(DL_ERROR|FA_SYNC,("Unprepared header!") ); GlobalFreeDeviceInfo( WaveHeaderDeviceInfo ); return MMSYSERR_INVALPARAM; } //
// Free later in the callback routine
//
pWavePrepareData = (PWAVEPREPAREDATA)pHdr->reserved; pWavePrepareData->pdi = WaveHeaderDeviceInfo;
cbDeviceInfo = sizeof(*WaveHeaderDeviceInfo) + (lstrlenW(WaveHeaderDeviceInfo->wstrDeviceInterface) * sizeof(WCHAR)); //
// Fill the wave header's deviceinfo structure
//
WaveHeaderDeviceInfo->DeviceType = DeviceInfo->DeviceType; WaveHeaderDeviceInfo->DeviceNumber = DeviceInfo->DeviceNumber; WaveHeaderDeviceInfo->DeviceHandle = DeviceInfo->DeviceHandle; WaveHeaderDeviceInfo->DataBuffer = pHdr; WaveHeaderDeviceInfo->DataBufferSize = sizeof( WAVEHDR );
if (WaveInDevice == DeviceInfo->DeviceType) { fResult = DeviceIoControl(ghDevice, IOCTL_WDMAUD_WAVE_IN_READ_PIN, WaveHeaderDeviceInfo, cbDeviceInfo, WaveHeaderDeviceInfo, sizeof(*WaveHeaderDeviceInfo), &cbWritten, pWavePrepareData->pOverlapped);
} else // WaveOutDevice
{ fResult = DeviceIoControl(ghDevice, IOCTL_WDMAUD_WAVE_OUT_WRITE_PIN, WaveHeaderDeviceInfo, cbDeviceInfo, WaveHeaderDeviceInfo, sizeof(*WaveHeaderDeviceInfo), &cbRead, pWavePrepareData->pOverlapped); }
mmr = sndTranslateStatus();
if (MMSYSERR_NOERROR == mmr) { mmr = wdmaudCreateCompletionThread ( DeviceInfo ); }
return mmr; }
/****************************************************************************
* @doc INTERNAL * * @api DWORD | wdmaudSubmitMidiOutHeader | Synchronously process a midi output * buffer. * * @rdesc A MMSYS... type return code for the application. ***************************************************************************/ MMRESULT FAR wdmaudSubmitMidiOutHeader ( LPDEVICEINFO DeviceInfo, LPMIDIHDR pHdr ) { BOOL fResult; MMRESULT mmr; OVERLAPPED ov; ULONG cbReturned; ULONG cbDeviceInfo;
if (NULL == ghDevice) { MMRRETURN( MMSYSERR_NOTENABLED ); }
RtlZeroMemory( &ov, sizeof( OVERLAPPED ) ); if (NULL == (ov.hEvent = CreateEvent( NULL, TRUE, FALSE, NULL ))) return FALSE;
cbDeviceInfo = sizeof(*DeviceInfo) + (lstrlenW(DeviceInfo->wstrDeviceInterface) * sizeof(WCHAR)); //
// Wrap the data buffer around the device context.
//
DeviceInfo->DataBuffer = pHdr; DeviceInfo->DataBufferSize = sizeof( MIDIHDR );
//
// Since we are not letting the OS do the user-to-kernel
// space mapping, we will have to do the mapping ourselves
// for writes to data buffers in wdmaud.sys.
//
fResult = DeviceIoControl( ghDevice, IOCTL_WDMAUD_MIDI_OUT_WRITE_LONGDATA, DeviceInfo, cbDeviceInfo, DeviceInfo, sizeof(*DeviceInfo), &cbReturned, &ov ); if (!fResult) { if (ERROR_IO_PENDING == GetLastError()) { WaitForSingleObject( ov.hEvent, INFINITE ); mmr = MMSYSERR_NOERROR; } else { mmr = sndTranslateStatus(); } } else { mmr = MMSYSERR_NOERROR; }
CloseHandle( ov.hEvent );
MMRRETURN( mmr ); }
/****************************************************************************
* @doc INTERNAL * * @api MMRESULT | wdmaudGetMidiData | Pass a buffer down to * wdmaud.sys to be filled in with KSMUSICFORMAT data. * * @parm LPDEVICEINFO | DeviceInfo | The data associated with the logical * device. * * @rdesc A MMSYS... type return code for the application. ***************************************************************************/ MMRESULT wdmaudGetMidiData ( LPDEVICEINFO DeviceInfo, LPMIDIDATALISTENTRY pOldMidiDataListEntry ) { LPDEVICEINFO MidiDataDeviceInfo; ULONG cbWritten; ULONG cbDeviceInfo; LPMIDIDATALISTENTRY pMidiDataListEntry; LPMIDIDATALISTENTRY pTemp; MMRESULT mmr;
if (NULL == ghDevice) { MMRRETURN( MMSYSERR_NOTENABLED ); }
//
// Don't need to allocate another buffer and create another
// event if we can reuse the old one
//
if (pOldMidiDataListEntry) { //
// Make sure to pull it off the front of the queue
// before adding again
//
CRITENTER ; DeviceInfo->DeviceState->lpMidiDataQueue = DeviceInfo->DeviceState->lpMidiDataQueue->lpNext; CRITLEAVE ;
pMidiDataListEntry = pOldMidiDataListEntry; // RtlZeroMemory( &pMidiDataListEntry->MidiData, sizeof(MIDIDATA) );
// ResetEvent( ((LPOVERLAPPED)(pMidiDataListEntry->pOverlapped))->hEvent );
} else { //
// Allocate a buffer to receive the music data
//
pMidiDataListEntry = (LPMIDIDATALISTENTRY) GlobalAllocPtr( GPTR, sizeof(MIDIDATALISTENTRY)); if (NULL == pMidiDataListEntry) { MMRRETURN( MMSYSERR_NOMEM ); } #ifdef DEBUG
pMidiDataListEntry->dwSig=MIDIDATALISTENTRY_SIGNATURE; #endif
pMidiDataListEntry->MidiDataDeviceInfo = GlobalAllocDeviceInfo(DeviceInfo->wstrDeviceInterface); if (!pMidiDataListEntry->MidiDataDeviceInfo) { GlobalFreePtr(pMidiDataListEntry); MMRRETURN( MMSYSERR_NOMEM ); }
//
// Initialize music data structure
//
pMidiDataListEntry->pOverlapped = (LPOVERLAPPED)HeapAlloc( GetProcessHeap(), 0, sizeof( OVERLAPPED )); if (NULL == pMidiDataListEntry->pOverlapped) { GlobalFreePtr(pMidiDataListEntry->MidiDataDeviceInfo ); GlobalFreePtr(pMidiDataListEntry); MMRRETURN( MMSYSERR_NOMEM ); }
RtlZeroMemory( pMidiDataListEntry->pOverlapped, sizeof( OVERLAPPED ) );
if (NULL == ( ((LPOVERLAPPED)(pMidiDataListEntry->pOverlapped))->hEvent = CreateEvent( NULL, FALSE, FALSE, NULL ))) { HeapFree( GetProcessHeap(), 0, pMidiDataListEntry->pOverlapped); GlobalFreePtr(pMidiDataListEntry->MidiDataDeviceInfo ); GlobalFreePtr(pMidiDataListEntry); MMRRETURN( MMSYSERR_NOMEM ); } }
//
// Cauterize the next pointer for new and old list entries
//
pMidiDataListEntry->lpNext = NULL;
//
// Add music data structure to a queue
//
CRITENTER ;
if (!DeviceInfo->DeviceState->lpMidiDataQueue) { DeviceInfo->DeviceState->lpMidiDataQueue = pMidiDataListEntry; pTemp = NULL; #ifdef UNDER_NT
if( (DeviceInfo->DeviceState->hevtQueue) && (DeviceInfo->DeviceState->hevtQueue != (HANDLE)FOURTYTHREE) && (DeviceInfo->DeviceState->hevtQueue != (HANDLE)FOURTYTWO) ) { DPF(DL_TRACE|FA_MIDI,("SetEvent on hevtQueue") ); SetEvent( DeviceInfo->DeviceState->hevtQueue ); } #endif
} else { for (pTemp = DeviceInfo->DeviceState->lpMidiDataQueue; pTemp->lpNext != NULL; pTemp = pTemp->lpNext);
pTemp->lpNext = pMidiDataListEntry; }
CRITLEAVE ;
DPF(DL_TRACE|FA_MIDI, ("MidiData submitted: pMidiDataListEntry = 0x%08lx", pMidiDataListEntry) );
MidiDataDeviceInfo = pMidiDataListEntry->MidiDataDeviceInfo;
cbDeviceInfo = sizeof(*MidiDataDeviceInfo) + (lstrlenW(MidiDataDeviceInfo->wstrDeviceInterface) * sizeof(WCHAR));
//
// Wrap the data buffer around the device context.
//
MidiDataDeviceInfo->DeviceType = DeviceInfo->DeviceType; MidiDataDeviceInfo->DeviceNumber = DeviceInfo->DeviceNumber; MidiDataDeviceInfo->DataBuffer = &pMidiDataListEntry->MidiData; MidiDataDeviceInfo->DataBufferSize = sizeof( MIDIDATA );
//
// Send this buffer down to wdmaud.sys to fill in data
//
DeviceIoControl(ghDevice, IOCTL_WDMAUD_MIDI_IN_READ_PIN, MidiDataDeviceInfo, cbDeviceInfo, MidiDataDeviceInfo, sizeof(*MidiDataDeviceInfo), &cbWritten, pMidiDataListEntry->pOverlapped);
mmr = sndTranslateStatus();
//
// Make sure that the completion thread is running
//
if (MMSYSERR_NOERROR == mmr) { mmr = wdmaudCreateCompletionThread ( DeviceInfo ); } else { // Unlink...
CloseHandle( ((LPOVERLAPPED)(pMidiDataListEntry->pOverlapped))->hEvent ); HeapFree( GetProcessHeap(), 0, pMidiDataListEntry->pOverlapped); GlobalFreePtr( MidiDataDeviceInfo ); GlobalFreePtr( pMidiDataListEntry );
if (pTemp) { pTemp->lpNext = NULL; } else { DeviceInfo->DeviceState->lpMidiDataQueue = NULL; } }
MMRRETURN( mmr ); }
/****************************************************************************
* @doc INTERNAL * * @api MMRESULT | wdmaudCreateCompletionThread | * * @rdesc A MMSYS... type return code for the application. * ***************************************************************************/
MMRESULT wdmaudCreateCompletionThread ( LPDEVICEINFO DeviceInfo ) { PTHREAD_START_ROUTINE fpThreadRoutine;
DPFASSERT(DeviceInfo->DeviceType == WaveOutDevice || DeviceInfo->DeviceType == WaveInDevice || DeviceInfo->DeviceType == MidiInDevice);
//
// Thread already created so...forget about it.
//
if (DeviceInfo->DeviceState->fThreadRunning) { ISVALIDDEVICESTATE(DeviceInfo->DeviceState,TRUE); return MMSYSERR_NOERROR; }
//
// Pick which thread routine we want to create
//
if (WaveInDevice == DeviceInfo->DeviceType || WaveOutDevice == DeviceInfo->DeviceType) { fpThreadRoutine = (PTHREAD_START_ROUTINE)waveThread; } else if (MidiInDevice == DeviceInfo->DeviceType) { fpThreadRoutine = (PTHREAD_START_ROUTINE)midThread; } else { MMRRETURN( MMSYSERR_ERROR ); }
//
// Is there a problem with hThread? Well, here is where it gets set
// to a non-zero value. Basically, during this creation process, we
// look to see if there is already a work item scheduled on this thread. if
// not, we create one and schedule it.
//
// But, between the point where we check this value and the point where it
// gets set
//
if (NULL == DeviceInfo->DeviceState->hThread) { #ifdef DEBUG
if( (DeviceInfo->DeviceState->hevtQueue != NULL) && (DeviceInfo->DeviceState->hevtQueue != (HANDLE)FOURTYTHREE) && (DeviceInfo->DeviceState->hevtQueue != (HANDLE)FOURTYTWO) ) { DPF(DL_ERROR|FA_ALL,("hevtQueue getting overwritten! %08X",DeviceInfo) ); } #endif
DeviceInfo->DeviceState->hevtQueue = CreateEvent( NULL, // no security
FALSE, // auto reset
FALSE, // initially not signalled
NULL ); // unnamed
#ifdef DEBUG
if( (DeviceInfo->DeviceState->hevtExitThread != NULL) && (DeviceInfo->DeviceState->hevtExitThread != (HANDLE)FOURTYEIGHT) ) { DPF(DL_ERROR|FA_ALL,("hevtExitThread getting overwritten %08X",DeviceInfo) ); } #endif
DeviceInfo->DeviceState->hevtExitThread = CreateEvent( NULL, // no security
FALSE, // auto reset
FALSE, // initially not signalled
NULL ); // unnamed
DPFASSERT(NULL == DeviceInfo->DeviceState->hThread);
DPF(DL_TRACE|FA_SYNC,("Creating Completion Thread") );
DeviceInfo->DeviceState->hThread = CreateThread( NULL, // no security
0, // default stack
(PTHREAD_START_ROUTINE) fpThreadRoutine, (PVOID) DeviceInfo, // parameter
0, // default create flags
&DeviceInfo->DeviceState->dwThreadId ); // container for
// thread id
//
// TODO: I need to wait for the thread to actually start
// before I can move on
//
if (DeviceInfo->DeviceState->hThread) SetThreadPriority(DeviceInfo->DeviceState->hThread, THREAD_PRIORITY_TIME_CRITICAL);
}
if (NULL == DeviceInfo->DeviceState->hThread) { if (DeviceInfo->DeviceState->hevtQueue) { CloseHandle( DeviceInfo->DeviceState->hevtQueue ); DeviceInfo->DeviceState->hevtQueue = NULL; CloseHandle( DeviceInfo->DeviceState->hevtExitThread ); DeviceInfo->DeviceState->hevtExitThread = NULL; } MMRRETURN( MMSYSERR_ERROR ); }
InterlockedExchange( (LPLONG)&DeviceInfo->DeviceState->fThreadRunning, TRUE );
return MMSYSERR_NOERROR; }
/****************************************************************************
* @doc INTERNAL * * @api MMRESULT | wdmaudDestroyCompletionThread | * * @rdesc A MMSYS... type return code for the application. * ***************************************************************************/
MMRESULT wdmaudDestroyCompletionThread ( LPDEVICEINFO DeviceInfo ) { MMRESULT mmr;
if( (mmr=IsValidDeviceInfo(DeviceInfo)) != MMSYSERR_NOERROR ) { MMRRETURN( mmr ); }
CRITENTER; if( DeviceInfo->DeviceState->hThread ) { ISVALIDDEVICESTATE(DeviceInfo->DeviceState,FALSE); InterlockedExchange( (LPLONG)&DeviceInfo->DeviceState->fExit, TRUE ); //
// If the thread handling the completion notifications, waveThread and
// midThread have completed, then hevtQueue will be invalid. We don't
// want to call SetEvent with invalid info. Also, if the thread has
// completed, then we know that hevtExitThread will have been signaled and
// fThreadRunning will be FALSE.
//
if( DeviceInfo->DeviceState->fThreadRunning ) { ISVALIDDEVICESTATE(DeviceInfo->DeviceState,TRUE); SetEvent( DeviceInfo->DeviceState->hevtQueue ); }
CRITLEAVE; //
// Ok, here we're going to wait until that routine below, waveThread
// completes and signals us.
//
DPF(DL_TRACE|FA_SYNC, ("DestroyThread: Waiting for thread to go away") ); WaitForSingleObject( DeviceInfo->DeviceState->hevtExitThread, INFINITE ); DPF(DL_TRACE|FA_SYNC, ("DestroyThread: Done waiting for thread to go away") );
CRITENTER; CloseHandle( DeviceInfo->DeviceState->hThread ); DeviceInfo->DeviceState->hThread = NULL;
CloseHandle( DeviceInfo->DeviceState->hevtExitThread ); DeviceInfo->DeviceState->hevtExitThread = (HANDLE)FOURTYEIGHT; //NULL;
}
InterlockedExchange( (LPLONG)&DeviceInfo->DeviceState->fExit, FALSE );
ISVALIDDEVICEINFO(DeviceInfo); ISVALIDDEVICESTATE(DeviceInfo->DeviceState,FALSE); CRITLEAVE;
return MMSYSERR_NOERROR; }
/****************************************************************************
* @doc INTERNAL * * @api DWORD | waveThread | * ***************************************************************************/
DWORD waveThread ( LPDEVICEINFO DeviceInfo ) { BOOL fDone; LPWAVEHDR pWaveHdr; MMRESULT mmr;
//
// Keep looping until all notifications are posted...
//
fDone = FALSE; while (!fDone ) { fDone = FALSE; CRITENTER ;
ISVALIDDEVICEINFO(DeviceInfo); ISVALIDDEVICESTATE(DeviceInfo->DeviceState,TRUE);
if(pWaveHdr = DeviceInfo->DeviceState->lpWaveQueue) { PWAVEPREPAREDATA pWavePrepareData; HANDLE hEvent;
if( (mmr=IsValidWaveHeader(pWaveHdr)) == MMSYSERR_NOERROR ) { pWavePrepareData = (PWAVEPREPAREDATA)pWaveHdr->reserved;
if( (mmr=IsValidPrepareWaveHeader(pWavePrepareData)) == MMSYSERR_NOERROR ) { hEvent = pWavePrepareData->pOverlapped->hEvent;
CRITLEAVE ; WaitForSingleObject( hEvent, INFINITE ); CRITENTER ;
//
// Validate that our data is still intact
//
if( ( (mmr=IsValidDeviceInfo(DeviceInfo)) ==MMSYSERR_NOERROR ) && ( (mmr=IsValidDeviceState(DeviceInfo->DeviceState,TRUE)) == MMSYSERR_NOERROR ) ) { DPF(DL_TRACE|FA_WAVE, ("Calling waveCompleteHeader") );
waveCompleteHeader(DeviceInfo); } else { //
// Problem: Major structures have changed. How can we complete
// this header? The only thing I can think of here is to
// terminate the thread.
//
goto Terminate_waveThread; } } else { //
// Problem: reserved field that contains the Prepare data info
// is corrupt, thus we will not have a valid hEvent to wait on.
// remove this header and go on to the next.
//
DeviceInfo->DeviceState->lpWaveQueue = DeviceInfo->DeviceState->lpWaveQueue->lpNext; } } else { //
// Problem: Our header is corrupt. We can't possibly wait on this
// because we'll never get signaled! thus we will not have a valid
// hEvent to wait on. Remove this header and go on to the next.
//
DeviceInfo->DeviceState->lpWaveQueue = DeviceInfo->DeviceState->lpWaveQueue->lpNext; } CRITLEAVE ; } else { // fDone = TRUE;
if (DeviceInfo->DeviceState->fRunning) { wdmaudIoControl(DeviceInfo, 0, NULL, DeviceInfo->DeviceType == WaveOutDevice ? IOCTL_WDMAUD_WAVE_OUT_PAUSE : IOCTL_WDMAUD_WAVE_IN_STOP); InterlockedExchange( (LPLONG)&DeviceInfo->DeviceState->fRunning, FALSE ); }
CRITLEAVE ;
WaitForSingleObject( DeviceInfo->DeviceState->hevtQueue, INFINITE );
//
// We could have been here for two reasons 1) the thread got starved
// ie. the header list went empty or 2) we're done with the headers.
// Only when we're done do we really want to exit this thread.
//
if( DeviceInfo->DeviceState->fExit ) { fDone = TRUE; } } }
CRITENTER; ISVALIDDEVICEINFO(DeviceInfo); ISVALIDDEVICESTATE(DeviceInfo->DeviceState,TRUE);
CloseHandle( DeviceInfo->DeviceState->hevtQueue ); DeviceInfo->DeviceState->hevtQueue = (HANDLE)FOURTYTWO; // WAS NULL
InterlockedExchange( (LPLONG)&DeviceInfo->DeviceState->fThreadRunning, FALSE ); SetEvent( DeviceInfo->DeviceState->hevtExitThread );
DPF(DL_TRACE|FA_WAVE, ("waveThread: Closing") );
Terminate_waveThread: CRITLEAVE; return 0; }
/****************************************************************************
* @doc INTERNAL * * @api DWORD | midThread | * ***************************************************************************/
DWORD midThread ( LPDEVICEINFO DeviceInfo ) { BOOL fDone; LPMIDIDATALISTENTRY pMidiDataListEntry; int i; MMRESULT mmr;
DPF(DL_TRACE|FA_MIDI, ("Entering") );
//
// Keep looping until all notifications are posted...
//
fDone = FALSE; while (!fDone) { CRITENTER ;
ISVALIDDEVICEINFO(DeviceInfo); ISVALIDDEVICESTATE(DeviceInfo->DeviceState,TRUE);
if (pMidiDataListEntry = DeviceInfo->DeviceState->lpMidiDataQueue) { HANDLE hEvent;
if( (mmr=IsValidMidiDataListEntry(pMidiDataListEntry)) == MMSYSERR_NOERROR) { hEvent = ((LPOVERLAPPED)(pMidiDataListEntry->pOverlapped))->hEvent;
DPF(DL_TRACE|FA_MIDI, ("Waiting on pMidiDataListEntry = 0x%08lx", pMidiDataListEntry) );
CRITLEAVE ; WaitForSingleObject( hEvent, INFINITE ); CRITENTER ;
DPF(DL_TRACE|FA_MIDI, ("Completed pMidiDataListEntry = 0x%08lx", pMidiDataListEntry) );
if( ((mmr=IsValidDeviceInfo(DeviceInfo)) == MMSYSERR_NOERROR) && ((mmr=IsValidDeviceState(DeviceInfo->DeviceState,TRUE)) == MMSYSERR_NOERROR ) ) { //
// Parse and callback clients
//
wdmaudParseMidiData(DeviceInfo, pMidiDataListEntry);
if (DeviceInfo->DeviceState->fExit || !DeviceInfo->DeviceState->fRunning) { //
// Unlink from queue and free memory
//
wdmaudFreeMidiData(DeviceInfo, pMidiDataListEntry); } else { //
// Reuse this buffer to read Midi data
//
wdmaudGetMidiData(DeviceInfo, pMidiDataListEntry); } } else { //
// Problem: Our major structure is bad. There is nothing that
// we can do, exit and hope for the best.
//
goto Terminate_midThread; } } else { //
// Problem: the pMidiDataListEntry is invalid. We can't use it
// so we simply move on to the next one and hope for the best.
//
DeviceInfo->DeviceState->lpMidiDataQueue = DeviceInfo->DeviceState->lpMidiDataQueue->lpNext; } CRITLEAVE ; } else {
fDone = TRUE;
CRITLEAVE ;
DPF(DL_TRACE|FA_MIDI, ("Waiting for signal to kill thread") ); WaitForSingleObject( DeviceInfo->DeviceState->hevtQueue, INFINITE ); DPF(DL_TRACE|FA_MIDI, ("Done waiting for signal to kill thread") ); } }
CRITENTER; ISVALIDDEVICEINFO(DeviceInfo); ISVALIDDEVICESTATE(DeviceInfo->DeviceState,TRUE);
CloseHandle( DeviceInfo->DeviceState->hevtQueue ); DeviceInfo->DeviceState->hevtQueue = (HANDLE)FOURTYTHREE; //NULL;
InterlockedExchange( (LPLONG)&DeviceInfo->DeviceState->fThreadRunning, FALSE ); SetEvent( DeviceInfo->DeviceState->hevtExitThread );
DPF(DL_TRACE|FA_MIDI, ("Closing") );
Terminate_midThread: CRITLEAVE; return 0; }
BOOL IsMidiDataDiscontinuous ( PKSSTREAM_HEADER pHeader ) { DPFASSERT(pHeader);
//
// Check the OptionFlags for the end of the midi stream
//
return (pHeader->OptionsFlags & KSSTREAM_HEADER_OPTIONSF_DATADISCONTINUITY); }
ULONG GetStreamHeaderSize ( PKSSTREAM_HEADER pHeader ) { DPFASSERT(pHeader);
//
// Check the OptionFlags for the end of the midi stream
//
return (pHeader->DataUsed); }
BOOL IsSysExData ( LPBYTE MusicData ) { DPFASSERT(MusicData);
return ( IS_SYSEX(*MusicData) || IS_EOX(*MusicData) || IS_DATA_BYTE(*MusicData) ); }
BOOL IsEndofSysEx ( LPBYTE MusicData ) { DPFASSERT(MusicData);
return IS_EOX(*(MusicData)); }
void wdmaudParseSysExData ( LPDEVICEINFO DeviceInfo, LPMIDIDATA pMidiData, BOOL MidiDataDiscontinuous ) { BOOL fCompleteSysEx = FALSE; LPMIDIHDR pMidiInHdr; PKSMUSICFORMAT MusicFormat; ULONG MusicDataLeft; LPBYTE MusicData; ULONG RunningTimeMs; ULONG DataCopySize; ULONG HeaderFreeSpace; ULONG MusicFormatDataLeft; ULONG MusicFormatDataPosition = 0;
//
// Easier to use locals
//
MusicFormat = (PKSMUSICFORMAT)&pMidiData->MusicFormat; MusicData = (LPBYTE)pMidiData->MusicData; MusicDataLeft = pMidiData->StreamHeader.DataUsed; RunningTimeMs = 0;
if ( MidiDataDiscontinuous || IsEndofSysEx(MusicData + MusicFormat->ByteCount - 1) ) { fCompleteSysEx = TRUE; }
while (MusicDataLeft || MidiDataDiscontinuous) { //
// update the running time for this Music Format header
//
if (MusicFormat->ByteCount == 0) { RunningTimeMs = DeviceInfo->DeviceState->LastTimeMs; } else { RunningTimeMs += MusicFormat->TimeDeltaMs; DeviceInfo->DeviceState->LastTimeMs = RunningTimeMs; }
//
// Get the next header from the queue
//
pMidiInHdr = DeviceInfo->DeviceState->lpMidiInQueue;
while (pMidiInHdr && MusicFormatDataPosition <= MusicFormat->ByteCount) {
HeaderFreeSpace = pMidiInHdr->dwBufferLength - pMidiInHdr->dwBytesRecorded;
MusicFormatDataLeft = MusicFormat->ByteCount - MusicFormatDataPosition;
//
// Compute the size of the copy
//
DataCopySize = min(HeaderFreeSpace,MusicFormatDataLeft);
//
// Fill this, baby
//
if (DataCopySize) { RtlCopyMemory(pMidiInHdr->lpData + pMidiInHdr->dwBytesRecorded, MusicData + MusicFormatDataPosition, DataCopySize); }
//
// update the number of bytes recorded
//
pMidiInHdr->dwBytesRecorded += DataCopySize; MusicFormatDataPosition += DataCopySize;
DPF(DL_TRACE|FA_RECORD, ("Record SysEx: %d(%d) Data=0x%08lx", DataCopySize, pMidiInHdr->dwBytesRecorded, *MusicData) );
//
// If the buffer is full or end-of-sysex byte is received,
// the buffer is marked as 'done' and it's owner is called back.
//
if ( (fCompleteSysEx && pMidiInHdr->dwBytesRecorded && (MusicFormatDataPosition == MusicFormat->ByteCount) ) // copied whole SysEx
|| (pMidiInHdr->dwBufferLength == pMidiInHdr->dwBytesRecorded) ) // filled entire buffer
{
if (MidiDataDiscontinuous) { midiInCompleteHeader(DeviceInfo, RunningTimeMs, MIM_LONGERROR); } else { midiInCompleteHeader(DeviceInfo, RunningTimeMs, MIM_LONGDATA); }
//
// Grab the next header to fill, if it exists
//
pMidiInHdr = DeviceInfo->DeviceState->lpMidiInQueue; }
//
// Break out of loop when all of the data is copied
//
if (MusicFormatDataPosition == MusicFormat->ByteCount) { break; }
//
// in the middle of a sysex and we still
// have room left in the header
//
} // while we have more headers and data to copy
//
// don't continue messin' with this irp
//
if (MidiDataDiscontinuous) { break; }
MusicDataLeft -= sizeof(KSMUSICFORMAT) + ((MusicFormat->ByteCount + 3) & ~3); MusicFormat = (PKSMUSICFORMAT)(MusicData + ((MusicFormat->ByteCount + 3) & ~3)); MusicData = (LPBYTE)(MusicFormat + 1);
} // while IrpDataLeft
return; }
void wdmaudParseShortMidiData ( LPDEVICEINFO DeviceInfo, LPMIDIDATA pMidiData, BOOL MidiDataDiscontinuous ) { BOOL fCompleteSysEx = FALSE; LPMIDIHDR pMidiInHdr; PKSMUSICFORMAT MusicFormat; ULONG MusicDataLeft; LPBYTE MusicData; ULONG RunningTimeMs; ULONG DataCopySize; ULONG HeaderFreeSpace; ULONG MusicFormatDataLeft; ULONG MusicFormatDataPosition = 0;
//
// Easier to use locals
//
MusicFormat = (PKSMUSICFORMAT)&pMidiData->MusicFormat; MusicData = (LPBYTE)pMidiData->MusicData; MusicDataLeft = pMidiData->StreamHeader.DataUsed; RunningTimeMs = 0;
while (MusicDataLeft || MidiDataDiscontinuous) { //
// update the running time for this Music Format header
//
if (MusicFormat->ByteCount == 0) { RunningTimeMs = DeviceInfo->DeviceState->LastTimeMs; } else { RunningTimeMs += MusicFormat->TimeDeltaMs; DeviceInfo->DeviceState->LastTimeMs = RunningTimeMs; }
//
// Non-used bytes should be zero'ed out
//
midiCallback(DeviceInfo, MIM_DATA, *((LPDWORD)MusicData), RunningTimeMs);
//
// don't continue messin' with this irp
//
if (MidiDataDiscontinuous) { break; }
MusicDataLeft -= sizeof(KSMUSICFORMAT) + ((MusicFormat->ByteCount + 3) & ~3); MusicFormat = (PKSMUSICFORMAT)(MusicData + ((MusicFormat->ByteCount + 3) & ~3)); MusicData = (LPBYTE)(MusicFormat + 1);
} // while IrpDataLeft
return; }
/****************************************************************************
* @doc INTERNAL * * @api VOID | wdmaudParseMidiData | This routine takes the MIDI data retrieved * from kernel mode and calls back the application with the long or short * messages packed in the buffer. * * @parm LPDEVICEINFO | DeviceInfo | The data associated with the logical midi * device. * * @comm The buffer flags are set and the buffer is passed to the auxiliary * device task for processing. ****************************************************************************/ void wdmaudParseMidiData ( LPDEVICEINFO DeviceInfo, LPMIDIDATALISTENTRY pMidiDataListEntry ) { BOOL MidiDataDiscontinuous; ULONG DataRemaining; ULONG BytesUsed; MMRESULT mmr;
if( (mmr=IsValidMidiDataListEntry(pMidiDataListEntry)) == MMSYSERR_NOERROR ) { DataRemaining = GetStreamHeaderSize(&pMidiDataListEntry->MidiData.StreamHeader);
MidiDataDiscontinuous = IsMidiDataDiscontinuous(&pMidiDataListEntry->MidiData.StreamHeader);
if ( IsSysExData((LPBYTE)pMidiDataListEntry->MidiData.MusicData) ) { wdmaudParseSysExData(DeviceInfo, &pMidiDataListEntry->MidiData, MidiDataDiscontinuous); } else { // Must be short messages
wdmaudParseShortMidiData(DeviceInfo, &pMidiDataListEntry->MidiData, MidiDataDiscontinuous); } } }
/****************************************************************************
* @doc INTERNAL * * @api VOID | wdmaudFreeMidiData | This routine unlinks and free the MIDI * data structure pointed to on input. * * @parm LPDEVICEINFO | DeviceInfo | The data associated with the logical midi * device. * * @parm LPMIDIDATA | pMidiData | The data buffer to be cleaned up * ****************************************************************************/ void wdmaudFreeMidiData ( LPDEVICEINFO DeviceInfo, LPMIDIDATALISTENTRY pMidiDataListEntry ) { //
// Advance the head of the queue
//
DeviceInfo->DeviceState->lpMidiDataQueue = DeviceInfo->DeviceState->lpMidiDataQueue->lpNext;
//
// Free all associated data members
//
CloseHandle( ((LPOVERLAPPED)(pMidiDataListEntry->pOverlapped))->hEvent ); HeapFree( GetProcessHeap(), 0, pMidiDataListEntry->pOverlapped ); GlobalFreeDeviceInfo( pMidiDataListEntry->MidiDataDeviceInfo ); GlobalFreePtr( pMidiDataListEntry );
}
/****************************************************************************
* @doc INTERNAL * * @api MMRESULT | wdmaudFreeMidiQ | * ***************************************************************************/ MMRESULT wdmaudFreeMidiQ ( LPDEVICEINFO DeviceInfo ) { LPMIDIHDR pHdr; LPMIDIHDR pTemp;
DPF(DL_TRACE|FA_MIDI, ("entering") );
CRITENTER ;
//
// Grab the head of the MIDI In queue and iterate through
// completing the headers
//
pHdr = DeviceInfo->DeviceState->lpMidiInQueue;
DeviceInfo->DeviceState->lpMidiInQueue = NULL ; // mark the queue as empty
while (pHdr) { pTemp = pHdr->lpNext;
pHdr->dwFlags &= ~MHDR_INQUEUE ; pHdr->dwFlags |= MHDR_DONE ; pHdr->dwBytesRecorded = 0;
//
// Invoke the callback function
//
midiCallback(DeviceInfo, MIM_LONGDATA, (DWORD_PTR)pHdr, DeviceInfo->DeviceState->LastTimeMs); // NOTE: This is not precise, but there is no way to
// know what the kernel time is without defining
// a new interface just for this.
pHdr = pTemp; }
CRITLEAVE ;
return MMSYSERR_NOERROR; }
|