You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
869 lines
27 KiB
869 lines
27 KiB
/****************************************************************************
|
|
*
|
|
* wavein.c
|
|
*
|
|
* WDM Audio support for Wave Input devices
|
|
*
|
|
* Copyright (C) Microsoft Corporation, 1997 - 1999 All Rights Reserved.
|
|
*
|
|
* History
|
|
* 5-12-97 - Noel Cross (NoelC)
|
|
*
|
|
***************************************************************************/
|
|
|
|
#include "wdmdrv.h"
|
|
|
|
#ifndef UNDER_NT
|
|
#pragma alloc_text(FIXCODE, waveCallback)
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
|
|
This function conforms to the standard Wave input driver message proc
|
|
(widMessage), which is documented in mmddk.d.
|
|
|
|
****************************************************************************/
|
|
DWORD FAR PASCAL _loadds widMessage
|
|
(
|
|
UINT id,
|
|
UINT msg,
|
|
DWORD_PTR dwUser,
|
|
DWORD_PTR dwParam1,
|
|
DWORD_PTR dwParam2
|
|
)
|
|
{
|
|
LPDEVICEINFO pInClient;
|
|
LPDEVICEINFO pDeviceInfo;
|
|
LPWAVEHDR lpWaveHdr;
|
|
MMRESULT mmr;
|
|
|
|
switch (msg)
|
|
{
|
|
case WIDM_INIT:
|
|
DPF(DL_TRACE|FA_WAVE, ("WIDM_INIT") );
|
|
return(wdmaudAddRemoveDevNode(WaveInDevice, (LPCWSTR)dwParam2, TRUE));
|
|
|
|
case DRVM_EXIT:
|
|
DPF(DL_TRACE|FA_WAVE, ("DRVM_EXIT WaveIn") );
|
|
return(wdmaudAddRemoveDevNode(WaveInDevice, (LPCWSTR)dwParam2, FALSE));
|
|
|
|
case WIDM_GETNUMDEVS:
|
|
DPF(DL_TRACE|FA_WAVE, ("WIDM_GETNUMDEVS") );
|
|
return wdmaudGetNumDevs(WaveInDevice, (LPCWSTR)dwParam1);
|
|
|
|
case WIDM_GETDEVCAPS:
|
|
DPF(DL_TRACE|FA_WAVE, ("WIDM_GETDEVCAPS") );
|
|
if (pDeviceInfo = GlobalAllocDeviceInfo((LPCWSTR)dwParam2))
|
|
{
|
|
pDeviceInfo->DeviceType = WaveInDevice;
|
|
pDeviceInfo->DeviceNumber = id;
|
|
mmr = wdmaudGetDevCaps(pDeviceInfo, (MDEVICECAPSEX FAR*)dwParam1);
|
|
GlobalFreeDeviceInfo(pDeviceInfo);
|
|
return mmr;
|
|
} else {
|
|
MMRRETURN( MMSYSERR_NOMEM );
|
|
}
|
|
|
|
case WIDM_PREFERRED:
|
|
DPF(DL_TRACE|FA_WAVE, ("WIDM_PREFERRED") );
|
|
return wdmaudSetPreferredDevice(
|
|
WaveInDevice,
|
|
id,
|
|
dwParam1,
|
|
dwParam2);
|
|
|
|
case WIDM_OPEN:
|
|
{
|
|
LPWAVEOPENDESC pwod = (LPWAVEOPENDESC)dwParam1;
|
|
|
|
if( (mmr=IsValidWaveOpenDesc(pwod)) != MMSYSERR_NOERROR )
|
|
{
|
|
MMRRETURN( mmr );
|
|
}
|
|
|
|
DPF(DL_TRACE|FA_WAVE, ("WIDM_OPEN") );
|
|
if (pDeviceInfo = GlobalAllocDeviceInfo((LPCWSTR)pwod->dnDevNode))
|
|
{
|
|
pDeviceInfo->DeviceType = WaveInDevice;
|
|
pDeviceInfo->DeviceNumber = id;
|
|
mmr = waveOpen(pDeviceInfo, dwUser, pwod, (DWORD)dwParam2);
|
|
GlobalFreeDeviceInfo(pDeviceInfo);
|
|
return mmr;
|
|
} else {
|
|
MMRRETURN( MMSYSERR_NOMEM );
|
|
}
|
|
}
|
|
|
|
case WIDM_CLOSE:
|
|
DPF(DL_TRACE|FA_WAVE, ("WIDM_CLOSE") );
|
|
pInClient = (LPDEVICEINFO)dwUser;
|
|
|
|
//
|
|
// At this point, we've committed to closing down this DeviceInfo.
|
|
// We mark the DeviceState as closing and hope for the best! If
|
|
// someone calls WIDM_ADDBUFFER while we're in this state, we've got
|
|
// problems!
|
|
//
|
|
if( ( (mmr=IsValidDeviceInfo(pInClient)) != MMSYSERR_NOERROR ) ||
|
|
( (mmr=IsValidDeviceState(pInClient->DeviceState,FALSE)) != MMSYSERR_NOERROR ) )
|
|
{
|
|
MMRRETURN( mmr );
|
|
}
|
|
|
|
mmr = wdmaudCloseDev(pInClient);
|
|
if (MMSYSERR_NOERROR == mmr)
|
|
{
|
|
waveCallback(pInClient, WIM_CLOSE, 0L);
|
|
|
|
ISVALIDDEVICEINFO(pInClient);
|
|
ISVALIDDEVICESTATE(pInClient->DeviceState,FALSE);
|
|
|
|
waveCleanUp(pInClient);
|
|
}
|
|
return mmr;
|
|
|
|
case WIDM_ADDBUFFER:
|
|
DPF(DL_TRACE|FA_WAVE, ("WIDM_ADDBUFFER") );
|
|
lpWaveHdr = (LPWAVEHDR)dwParam1;
|
|
pInClient = (LPDEVICEINFO)dwUser;
|
|
|
|
//
|
|
// Perform our asserts
|
|
//
|
|
if( ( (mmr=IsValidDeviceInfo(pInClient)) != MMSYSERR_NOERROR ) ||
|
|
( (mmr=IsValidDeviceState(pInClient->DeviceState,FALSE)) != MMSYSERR_NOERROR ) ||
|
|
( (mmr=IsValidWaveHeader(lpWaveHdr)) != MMSYSERR_NOERROR ) )
|
|
{
|
|
MMRRETURN( mmr );
|
|
}
|
|
|
|
// sanity check on the wavehdr
|
|
DPFASSERT(lpWaveHdr != NULL);
|
|
if (lpWaveHdr == NULL)
|
|
MMRRETURN( MMSYSERR_INVALPARAM );
|
|
|
|
// check if it's been prepared
|
|
DPFASSERT(lpWaveHdr->dwFlags & WHDR_PREPARED);
|
|
if (!(lpWaveHdr->dwFlags & WHDR_PREPARED))
|
|
MMRRETURN( WAVERR_UNPREPARED );
|
|
|
|
// if it is already in our Q, then we cannot do this
|
|
DPFASSERT(!(lpWaveHdr->dwFlags & WHDR_INQUEUE));
|
|
if ( lpWaveHdr->dwFlags & WHDR_INQUEUE )
|
|
MMRRETURN( WAVERR_STILLPLAYING );
|
|
//
|
|
// Put the request at the end of our queue.
|
|
//
|
|
return waveWrite(pInClient, lpWaveHdr);
|
|
|
|
case WIDM_STOP:
|
|
DPF(DL_TRACE|FA_WAVE, ("WIDM_STOP") );
|
|
pInClient = (LPDEVICEINFO)dwUser;
|
|
return wdmaudSetDeviceState(pInClient,
|
|
IOCTL_WDMAUD_WAVE_IN_STOP);
|
|
|
|
case WIDM_START:
|
|
DPF(DL_TRACE|FA_WAVE, ("WIDM_START") );
|
|
pInClient = (LPDEVICEINFO)dwUser;
|
|
return wdmaudSetDeviceState(pInClient,
|
|
IOCTL_WDMAUD_WAVE_IN_RECORD);
|
|
|
|
case WIDM_RESET:
|
|
DPF(DL_TRACE|FA_WAVE, ("WIDM_RESET") );
|
|
pInClient = (LPDEVICEINFO)dwUser;
|
|
return wdmaudSetDeviceState(pInClient,
|
|
IOCTL_WDMAUD_WAVE_IN_RESET);
|
|
|
|
case WIDM_GETPOS:
|
|
DPF(DL_TRACE|FA_WAVE, ("WIDM_GETPOS") );
|
|
pInClient = (LPDEVICEINFO)dwUser;
|
|
|
|
if( ( (mmr=IsValidDeviceInfo(pInClient)) != MMSYSERR_NOERROR) ||
|
|
( (mmr=IsValidDeviceState(pInClient->DeviceState,FALSE)) != MMSYSERR_NOERROR) )
|
|
{
|
|
MMRRETURN( mmr );
|
|
}
|
|
|
|
return wdmaudGetPos(pInClient,
|
|
(LPMMTIME)dwParam1,
|
|
(DWORD)dwParam2,
|
|
WaveInDevice);
|
|
|
|
#ifdef UNDER_NT
|
|
case WIDM_PREPARE:
|
|
DPF(DL_TRACE|FA_WAVE, ("WIDM_PREPARE") );
|
|
pInClient = (LPDEVICEINFO)dwUser;
|
|
|
|
if( ( (mmr=IsValidDeviceInfo(pInClient)) != MMSYSERR_NOERROR) ||
|
|
( (mmr=IsValidDeviceState(pInClient->DeviceState,FALSE)) != MMSYSERR_NOERROR) )
|
|
{
|
|
MMRRETURN( mmr );
|
|
}
|
|
|
|
return wdmaudPrepareWaveHeader(pInClient, (LPWAVEHDR)dwParam1);
|
|
|
|
case WIDM_UNPREPARE:
|
|
DPF(DL_TRACE|FA_WAVE, ("WIDM_UNPREPARE") );
|
|
pInClient = (LPDEVICEINFO)dwUser;
|
|
|
|
if( ( (mmr=IsValidDeviceInfo(pInClient)) != MMSYSERR_NOERROR) ||
|
|
( (mmr=IsValidDeviceState(pInClient->DeviceState,FALSE)) != MMSYSERR_NOERROR) )
|
|
{
|
|
MMRRETURN( mmr );
|
|
}
|
|
|
|
return wdmaudUnprepareWaveHeader(pInClient, (LPWAVEHDR)dwParam1);
|
|
#endif
|
|
|
|
default:
|
|
MMRRETURN( MMSYSERR_NOTSUPPORTED );
|
|
}
|
|
|
|
//
|
|
// Should not get here
|
|
//
|
|
|
|
DPFASSERT(0);
|
|
MMRRETURN( MMSYSERR_NOTSUPPORTED );
|
|
}
|
|
|
|
/****************************************************************************
|
|
* @doc INTERNAL
|
|
*
|
|
* @api void | waveCallback | This calls DriverCallback for a WAVEHDR.
|
|
*
|
|
* @parm LPDEVICEINFO | pWave | Pointer to wave device.
|
|
*
|
|
* @parm DWORD | msg | The message.
|
|
*
|
|
* @parm DWORD | dw1 | message DWORD (dw2 is always set to 0).
|
|
*
|
|
* @rdesc There is no return value.
|
|
***************************************************************************/
|
|
VOID FAR waveCallback
|
|
(
|
|
LPDEVICEINFO pWave,
|
|
UINT msg,
|
|
DWORD_PTR dw1
|
|
)
|
|
{
|
|
|
|
// invoke the callback function, if it exists. dwFlags contains
|
|
// wave driver specific flags in the LOWORD and generic driver
|
|
// flags in the HIWORD
|
|
|
|
if (pWave->dwCallback)
|
|
DriverCallback(pWave->dwCallback, // user's callback DWORD
|
|
HIWORD(pWave->dwFlags), // callback flags
|
|
(HDRVR)pWave->DeviceHandle, // handle to the wave device
|
|
msg, // the message
|
|
pWave->dwInstance, // user's instance data
|
|
dw1, // first DWORD
|
|
0L); // second DWORD
|
|
}
|
|
|
|
/****************************************************************************
|
|
* @doc INTERNAL
|
|
*
|
|
* @api MMRESULT | waveOpen | Open wave device and set up logical device data
|
|
*
|
|
* @parm LPDEVICEINFO | DeviceInfo | Specifies if it's a wave input or output
|
|
* device
|
|
*
|
|
* @parm DWORD | dwUser | Input parameter to wodMessage - pointer to
|
|
* application's handle (generated by this routine)
|
|
*
|
|
* @parm LPWAVEOPENDESC | pwod | pointer to WAVEOPENDESC. Was dwParam1
|
|
* parameter to wodMessage
|
|
*
|
|
* @parm DWORD | dwParam2 | Input parameter to wodMessage
|
|
*
|
|
* @rdesc wodMessage return code.
|
|
***************************************************************************/
|
|
MMRESULT waveOpen
|
|
(
|
|
LPDEVICEINFO DeviceInfo,
|
|
DWORD_PTR dwUser,
|
|
LPWAVEOPENDESC pwod,
|
|
DWORD dwParam2
|
|
)
|
|
{
|
|
LPDEVICEINFO pClient; // pointer to client information structure
|
|
MMRESULT mmr;
|
|
#ifndef UNDER_NT
|
|
DWORD dwCallback16;
|
|
#endif
|
|
|
|
//
|
|
// allocate my per-client structure
|
|
//
|
|
pClient = GlobalAllocDeviceInfo(DeviceInfo->wstrDeviceInterface);
|
|
if (NULL == pClient)
|
|
{
|
|
MMRRETURN( MMSYSERR_NOMEM );
|
|
}
|
|
|
|
pClient->DeviceState = (LPVOID) GlobalAllocPtr( GPTR, sizeof( DEVICESTATE ) );
|
|
if (NULL == pClient->DeviceState)
|
|
{
|
|
GlobalFreeDeviceInfo( pClient );
|
|
MMRRETURN( MMSYSERR_NOMEM );
|
|
}
|
|
|
|
//
|
|
// Handle the query case and return early
|
|
//
|
|
if (WAVE_FORMAT_QUERY & dwParam2)
|
|
{
|
|
pClient->DeviceType = DeviceInfo->DeviceType;
|
|
pClient->DeviceNumber = DeviceInfo->DeviceNumber;
|
|
pClient->dwFlags = dwParam2;
|
|
|
|
mmr = wdmaudOpenDev( pClient, (LPWAVEFORMATEX)pwod->lpFormat );
|
|
|
|
if (mmr == MMSYSERR_NOTSUPPORTED)
|
|
{
|
|
mmr = WAVERR_BADFORMAT;
|
|
}
|
|
|
|
GlobalFreePtr( pClient->DeviceState );
|
|
GlobalFreeDeviceInfo( pClient );
|
|
return mmr;
|
|
}
|
|
|
|
#ifdef UNDER_NT
|
|
//
|
|
// Allocate memory for our critical section
|
|
//
|
|
pClient->DeviceState->csQueue = (LPVOID) GlobalAllocPtr( GPTR, sizeof( CRITICAL_SECTION ) );
|
|
if (NULL == pClient->DeviceState->csQueue)
|
|
{
|
|
GlobalFreePtr( pClient->DeviceState );
|
|
GlobalFreeDeviceInfo( pClient );
|
|
MMRRETURN( MMSYSERR_NOMEM );
|
|
}
|
|
|
|
try
|
|
{
|
|
InitializeCriticalSection( (LPCRITICAL_SECTION)pClient->DeviceState->csQueue );
|
|
}
|
|
except(EXCEPTION_EXECUTE_HANDLER)
|
|
{
|
|
GlobalFreePtr( pClient->DeviceState->csQueue );
|
|
GlobalFreePtr( pClient->DeviceState );
|
|
GlobalFreeDeviceInfo( pClient );
|
|
MMRRETURN( MMSYSERR_NOMEM );
|
|
}
|
|
#endif
|
|
|
|
//
|
|
// fill out context data
|
|
//
|
|
pClient->DeviceNumber= DeviceInfo->DeviceNumber;
|
|
pClient->DeviceType = DeviceInfo->DeviceType;
|
|
pClient->dwInstance = pwod->dwInstance;
|
|
pClient->dwCallback = pwod->dwCallback;
|
|
pClient->dwFlags = dwParam2;
|
|
#ifdef UNDER_NT
|
|
pClient->DeviceHandle= (HANDLE32)pwod->hWave;
|
|
#else
|
|
pClient->DeviceHandle= (HANDLE32)MAKELONG(pwod->hWave,0);
|
|
_asm
|
|
{
|
|
mov ax, offset WaveDeviceCallback
|
|
mov word ptr [dwCallback16], ax
|
|
mov ax, seg WaveDeviceCallback
|
|
mov word ptr [dwCallback16+2], ax
|
|
}
|
|
pClient->dwCallback16= dwCallback16;
|
|
#endif
|
|
|
|
|
|
//
|
|
// initialize the device state
|
|
//
|
|
pClient->DeviceState->lpWaveQueue = NULL;
|
|
pClient->DeviceState->fRunning = FALSE;
|
|
pClient->DeviceState->fExit = FALSE;
|
|
if (pClient->DeviceType == WaveOutDevice)
|
|
pClient->DeviceState->fPaused = FALSE;
|
|
else
|
|
pClient->DeviceState->fPaused = TRUE;
|
|
#ifdef DEBUG
|
|
pClient->DeviceState->dwSig = DEVICESTATE_SIGNATURE;
|
|
#endif
|
|
//
|
|
// See if we can open our device
|
|
//
|
|
mmr = wdmaudOpenDev( pClient, (LPWAVEFORMATEX)pwod->lpFormat );
|
|
|
|
if (mmr != MMSYSERR_NOERROR)
|
|
{
|
|
if (mmr == MMSYSERR_NOTSUPPORTED)
|
|
{
|
|
mmr = WAVERR_BADFORMAT;
|
|
}
|
|
|
|
#ifdef UNDER_NT
|
|
DeleteCriticalSection( (LPCRITICAL_SECTION)pClient->DeviceState->csQueue );
|
|
GlobalFreePtr( pClient->DeviceState->csQueue );
|
|
//
|
|
// explicitly clear these values! We don't want to ever see these set
|
|
// again!
|
|
//
|
|
pClient->DeviceState->csQueue=NULL;
|
|
#endif
|
|
GlobalFreePtr( pClient->DeviceState );
|
|
pClient->DeviceState=NULL;
|
|
GlobalFreeDeviceInfo( pClient );
|
|
pClient=NULL;
|
|
|
|
MMRRETURN( mmr );
|
|
}
|
|
|
|
//
|
|
// Add instance to chain of devices
|
|
//
|
|
EnterCriticalSection(&wdmaudCritSec);
|
|
pClient->Next = pWaveDeviceList;
|
|
pWaveDeviceList = pClient;
|
|
LeaveCriticalSection(&wdmaudCritSec);
|
|
|
|
//
|
|
// give the client my driver dw
|
|
//
|
|
{
|
|
LPDEVICEINFO FAR *pUserHandle;
|
|
|
|
pUserHandle = (LPDEVICEINFO FAR *)dwUser;
|
|
*pUserHandle = pClient;
|
|
}
|
|
|
|
//
|
|
// sent client his OPEN callback message
|
|
//
|
|
waveCallback(pClient, DeviceInfo->DeviceType == WaveOutDevice ? WOM_OPEN : WIM_OPEN, 0L);
|
|
|
|
return MMSYSERR_NOERROR;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* @doc INTERNAL
|
|
*
|
|
* @api void | waveCleanUp | Free resources for a wave device
|
|
*
|
|
* @parm LPWAVEALLOC | pClient | Pointer to a WAVEALLOC structure describing
|
|
* resources to be freed.
|
|
*
|
|
* @rdesc There is no return value.
|
|
*
|
|
* @comm If the pointer to the resource is NULL then the resource has not
|
|
* been allocated.
|
|
***************************************************************************/
|
|
VOID waveCleanUp
|
|
(
|
|
LPDEVICEINFO pClient
|
|
)
|
|
{
|
|
LPDEVICEINFO FAR *ppCur ;
|
|
|
|
//
|
|
// remove from device chain
|
|
//
|
|
EnterCriticalSection(&wdmaudCritSec);
|
|
for (ppCur = &pWaveDeviceList;
|
|
*ppCur != NULL;
|
|
ppCur = &(*ppCur)->Next)
|
|
{
|
|
if (*ppCur == pClient)
|
|
{
|
|
*ppCur = (*ppCur)->Next;
|
|
break;
|
|
}
|
|
}
|
|
LeaveCriticalSection(&wdmaudCritSec);
|
|
|
|
#ifdef UNDER_NT
|
|
DeleteCriticalSection( (LPCRITICAL_SECTION)pClient->DeviceState->csQueue );
|
|
GlobalFreePtr( pClient->DeviceState->csQueue );
|
|
#endif
|
|
#ifdef DEBUG
|
|
//
|
|
// In debug, let's set all the values in the DEVICESTATE structure to bad
|
|
// values.
|
|
//
|
|
pClient->DeviceState->cSampleBits=0xDEADBEEF;
|
|
pClient->DeviceState->hThread=NULL;
|
|
pClient->DeviceState->dwThreadId=0xDEADBEEF;
|
|
pClient->DeviceState->lpWaveQueue=NULL;
|
|
pClient->DeviceState->csQueue=NULL;
|
|
pClient->DeviceState->hevtQueue=NULL;
|
|
pClient->DeviceState->hevtExitThread=NULL;
|
|
|
|
#endif
|
|
GlobalFreePtr( pClient->DeviceState );
|
|
pClient->DeviceState=NULL;
|
|
|
|
#ifdef DEBUG
|
|
//
|
|
// Now set all the values in the DEVICEINFO structure to bad values.
|
|
//
|
|
// pClient->Next=(LPDEVICEINFO)0xDEADBEEF;
|
|
pClient->DeviceNumber=-1;
|
|
pClient->DeviceType=0xDEADBEEF;
|
|
pClient->DeviceHandle=NULL;
|
|
pClient->dwInstance=(DWORD_PTR)NULL;
|
|
pClient->dwCallback=(DWORD_PTR)NULL;
|
|
pClient->dwCallback16=0xDEADBEEF;
|
|
pClient->dwFlags=0xDEADBEEF;
|
|
pClient->DataBuffer=NULL;
|
|
pClient->HardwareCallbackEventHandle=NULL;
|
|
pClient->dwCallbackType=0xDEADBEEF;
|
|
pClient->dwLineID=0xDEADBEEF;
|
|
pClient->dwFormat=0xDEADBEEF;
|
|
// pClient->DeviceState=(LPDEVICESTATE)0xDEADBEEF;
|
|
|
|
#endif
|
|
GlobalFreeDeviceInfo( pClient ) ;
|
|
pClient=NULL;
|
|
DPF(DL_TRACE|FA_WAVE,("DeviceState gone!") );
|
|
}
|
|
|
|
#ifdef UNDER_NT
|
|
/****************************************************************************
|
|
* @doc INTERNAL
|
|
*
|
|
* @api MMRESULT | wdmaudPrepareWaveHeader |
|
|
*
|
|
* @parm LPDEVICEINFO | 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.
|
|
*
|
|
***************************************************************************/
|
|
MMRESULT wdmaudPrepareWaveHeader
|
|
(
|
|
LPDEVICEINFO DeviceInfo,
|
|
LPWAVEHDR pHdr
|
|
)
|
|
{
|
|
PWAVEPREPAREDATA pWavePrepareData;
|
|
|
|
DPFASSERT(pHdr);
|
|
|
|
pHdr->lpNext = NULL;
|
|
pHdr->reserved = (DWORD_PTR)NULL;
|
|
|
|
//
|
|
// Allocate memory for the prepared header instance data
|
|
//
|
|
pWavePrepareData = (PWAVEPREPAREDATA) GlobalAllocPtr( GPTR, sizeof(*pWavePrepareData));
|
|
if (pWavePrepareData == NULL)
|
|
{
|
|
MMRRETURN( MMSYSERR_NOMEM );
|
|
}
|
|
|
|
//
|
|
// Allocate memory for our overlapped structure
|
|
//
|
|
pWavePrepareData->pOverlapped =
|
|
(LPOVERLAPPED)HeapAlloc( GetProcessHeap(), 0, sizeof( OVERLAPPED ));
|
|
if (NULL == pWavePrepareData->pOverlapped)
|
|
{
|
|
GlobalFreePtr( pWavePrepareData );
|
|
MMRRETURN( MMSYSERR_NOMEM );
|
|
}
|
|
|
|
RtlZeroMemory( pWavePrepareData->pOverlapped, sizeof( OVERLAPPED ) );
|
|
|
|
//
|
|
// Initialize the event once per preparation
|
|
//
|
|
if (NULL == (pWavePrepareData->pOverlapped->hEvent =
|
|
CreateEvent( NULL, FALSE, FALSE, NULL )))
|
|
{
|
|
HeapFree( GetProcessHeap(), 0, pWavePrepareData->pOverlapped);
|
|
GlobalFreePtr( pWavePrepareData );
|
|
MMRRETURN( MMSYSERR_NOMEM );
|
|
}
|
|
#ifdef DEBUG
|
|
pWavePrepareData->dwSig=WAVEPREPAREDATA_SIGNATURE;
|
|
#endif
|
|
|
|
//
|
|
// This next line adds this info to the main header. Only after this point
|
|
// will the info be used.
|
|
//
|
|
pHdr->reserved = (DWORD_PTR)pWavePrepareData;
|
|
|
|
// Still have WinMM prepare this header
|
|
return( MMSYSERR_NOTSUPPORTED );
|
|
}
|
|
|
|
/****************************************************************************
|
|
* @doc INTERNAL
|
|
*
|
|
* @api MMRESULT | wdmaudUnprepareWaveHeader |
|
|
*
|
|
* @parm LPDEVICEINFO | 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.
|
|
*
|
|
***************************************************************************/
|
|
MMRESULT wdmaudUnprepareWaveHeader
|
|
(
|
|
LPDEVICEINFO DeviceInfo,
|
|
LPWAVEHDR pHdr
|
|
)
|
|
{
|
|
MMRESULT mmr;
|
|
PWAVEPREPAREDATA pWavePrepareData;
|
|
|
|
if( ((mmr=IsValidWaveHeader(pHdr)) !=MMSYSERR_NOERROR ) ||
|
|
((mmr=IsValidPrepareWaveHeader((PWAVEPREPAREDATA)(pHdr->reserved))) != MMSYSERR_NOERROR ) )
|
|
{
|
|
MMRRETURN( mmr );
|
|
}
|
|
|
|
pWavePrepareData = (PWAVEPREPAREDATA)pHdr->reserved;
|
|
//
|
|
// This next line removes the WaveHeader from the list. It is no longer
|
|
// valid after this point!
|
|
//
|
|
pHdr->reserved = (DWORD_PTR)NULL;
|
|
|
|
CloseHandle( pWavePrepareData->pOverlapped->hEvent );
|
|
//
|
|
// When you free something, you should make sure that you trash it before
|
|
// the free. But, in this case, the only thing we use in the pOverlapped
|
|
// structure is the hEvent!
|
|
//
|
|
pWavePrepareData->pOverlapped->hEvent=NULL;
|
|
HeapFree( GetProcessHeap(), 0, pWavePrepareData->pOverlapped);
|
|
#ifdef DEBUG
|
|
pWavePrepareData->pOverlapped=NULL;
|
|
pWavePrepareData->dwSig=0;
|
|
#endif
|
|
GlobalFreePtr( pWavePrepareData );
|
|
|
|
// Still have WinMM prepare this header
|
|
return( MMSYSERR_NOTSUPPORTED );
|
|
}
|
|
|
|
#endif
|
|
/****************************************************************************
|
|
* @doc INTERNAL
|
|
*
|
|
* @api MMRESULT | waveWrite | This routine adds the header
|
|
* to the queue and then submits the buffer to the device
|
|
*
|
|
* @parm LPDEVICEINFO | 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 waveWrite
|
|
(
|
|
LPDEVICEINFO DeviceInfo,
|
|
LPWAVEHDR pHdr
|
|
)
|
|
{
|
|
MMRESULT mmr;
|
|
LPWAVEHDR pTemp;
|
|
|
|
//
|
|
// Mark this buffer because kmixer doesn't handle the dwFlags
|
|
//
|
|
pHdr->dwFlags |= WHDR_INQUEUE;
|
|
pHdr->dwFlags &= ~WHDR_DONE;
|
|
|
|
#ifndef UNDER_NT
|
|
//
|
|
// Store the context for this write in the header so that
|
|
// we know which client to send this back to on completion.
|
|
//
|
|
pHdr->reserved = (DWORD)DeviceInfo;
|
|
#endif
|
|
|
|
CRITENTER ;
|
|
DPF(DL_MAX|FA_WAVE, ("(ECS)") ); // Enter critical section
|
|
|
|
if (!DeviceInfo->DeviceState->lpWaveQueue)
|
|
{
|
|
DeviceInfo->DeviceState->lpWaveQueue = pHdr;
|
|
pTemp = NULL;
|
|
|
|
#ifdef UNDER_NT
|
|
if( (DeviceInfo->DeviceState->hevtQueue) &&
|
|
(DeviceInfo->DeviceState->hevtQueue != (HANDLE)FOURTYTHREE) &&
|
|
(DeviceInfo->DeviceState->hevtQueue != (HANDLE)FOURTYTWO) )
|
|
{
|
|
//
|
|
// If we get here, waveThread is waiting on hevtQueue because lpWaveQueue
|
|
// was empty and the completion thread exists == hevtQueue exists.
|
|
// So,we want to signal the thread to wake up so we can have this
|
|
// header serviced. Note: the difference between this call and the
|
|
// call made in wdmaudDestroyCompletionThread is that we don't set
|
|
// fExit to TRUE before making the call.
|
|
//
|
|
DPF(DL_MAX|FA_WAVE, ("Setting DeviceInfo->hevtQueue"));
|
|
SetEvent( DeviceInfo->DeviceState->hevtQueue );
|
|
}
|
|
#endif
|
|
}
|
|
else
|
|
{
|
|
for (pTemp = DeviceInfo->DeviceState->lpWaveQueue;
|
|
pTemp->lpNext != NULL;
|
|
pTemp = pTemp->lpNext);
|
|
|
|
pTemp->lpNext = pHdr;
|
|
}
|
|
|
|
DPF(DL_MAX|FA_WAVE, ("(LCS)") ); // Leave critical section
|
|
CRITLEAVE ;
|
|
|
|
//
|
|
// Call the 16 or 32-bit routine to send the buffer down
|
|
// to the kernel
|
|
//
|
|
mmr = wdmaudSubmitWaveHeader(DeviceInfo, pHdr);
|
|
if (mmr != MMSYSERR_NOERROR)
|
|
{
|
|
// Unlink...
|
|
if (pTemp)
|
|
{
|
|
pTemp->lpNext = NULL;
|
|
} else {
|
|
DeviceInfo->DeviceState->lpWaveQueue = NULL;
|
|
}
|
|
pHdr->dwFlags &= ~WHDR_INQUEUE;
|
|
DPF(DL_WARNING|FA_WAVE,("wdmaudSubmitWaveHeader failed mmr=%08X", mmr) );
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Kick start the device if it has been shutdown because of
|
|
// starvation. Also this allows waveOut to start when the
|
|
// first waveheader is submitted to the device.
|
|
//
|
|
if (!DeviceInfo->DeviceState->fRunning && !DeviceInfo->DeviceState->fPaused)
|
|
{
|
|
mmr = wdmaudSetDeviceState(DeviceInfo, (DeviceInfo->DeviceType == WaveOutDevice) ?
|
|
IOCTL_WDMAUD_WAVE_OUT_PLAY :
|
|
IOCTL_WDMAUD_WAVE_IN_RECORD);
|
|
if (mmr != MMSYSERR_NOERROR)
|
|
{
|
|
MMRESULT mmrError;
|
|
|
|
mmrError = wdmaudSetDeviceState(DeviceInfo, (DeviceInfo->DeviceType == WaveOutDevice) ?
|
|
IOCTL_WDMAUD_WAVE_OUT_RESET :
|
|
IOCTL_WDMAUD_WAVE_IN_RESET);
|
|
if (mmrError != MMSYSERR_NOERROR)
|
|
{
|
|
DPF(DL_WARNING|FA_WAVE, ("Couldn't reset device after error putting into run state"));
|
|
}
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
DPF(DL_MAX|FA_WAVE, ("DeviceInfo = x%08lx, fRunning = %d, fPaused = %d",
|
|
DeviceInfo,
|
|
DeviceInfo->DeviceState->fRunning,
|
|
DeviceInfo->DeviceState->fPaused) );
|
|
}
|
|
}
|
|
|
|
MMRRETURN( mmr );
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
* @doc INTERNAL
|
|
*
|
|
* @api VOID | waveCompleteHeader |
|
|
*
|
|
* @parm LPDEVICEINFO | DeviceInfo | The data associated with the logical wave
|
|
* device.
|
|
*
|
|
* @comm The buffer flags are set and the buffer is passed to the auxiliary
|
|
* device task for processing.
|
|
***************************************************************************/
|
|
VOID waveCompleteHeader
|
|
(
|
|
LPDEVICEINFO DeviceInfo
|
|
)
|
|
{
|
|
LPWAVEHDR pHdr;
|
|
MMRESULT mmr;
|
|
|
|
// NOTE: This routine is called from within the csQueue critical section!!!
|
|
//
|
|
// Only remove headers from the front of the queue so that order is maintained.
|
|
// Note that pHdr is the head, The DeviceInfo structure's DeviceState's
|
|
// lpWaveQueue pointer then gets updated to the next location.
|
|
//
|
|
|
|
//
|
|
// Never use bad data when handling this Completion!
|
|
//
|
|
if( (pHdr = DeviceInfo->DeviceState->lpWaveQueue) &&
|
|
( (mmr=IsValidWaveHeader(pHdr)) == MMSYSERR_NOERROR ) )
|
|
{
|
|
DeviceInfo->DeviceState->lpWaveQueue = DeviceInfo->DeviceState->lpWaveQueue->lpNext;
|
|
|
|
#ifdef UNDER_NT
|
|
//
|
|
// Free temporary DeviceInfo for the asynchronous i/o
|
|
//
|
|
{
|
|
PWAVEPREPAREDATA pWavePrepareData;
|
|
|
|
pWavePrepareData = (PWAVEPREPAREDATA)pHdr->reserved;
|
|
|
|
//
|
|
// Never attempt to free garbage!
|
|
//
|
|
if( (mmr=IsValidPrepareWaveHeader(pWavePrepareData)) == MMSYSERR_NOERROR )
|
|
GlobalFreePtr( pWavePrepareData->pdi );
|
|
}
|
|
// Invoke the callback function..
|
|
DPF(DL_TRACE|FA_WAVE, ("WaveHdr being returned: pHdr = 0x%08lx dwBytesRecorded = 0x%08lx",
|
|
pHdr, pHdr->dwBytesRecorded) );
|
|
|
|
pHdr->lpNext = NULL ;
|
|
pHdr->dwFlags &= ~WHDR_INQUEUE ;
|
|
pHdr->dwFlags |= WHDR_DONE ;
|
|
|
|
|
|
// NOTE: This callback is within the csQueue critical section !!!
|
|
|
|
waveCallback(DeviceInfo,
|
|
DeviceInfo->DeviceType == WaveOutDevice ? WOM_DONE : WIM_DATA,
|
|
(DWORD_PTR)pHdr);
|
|
|
|
DPF(DL_TRACE|FA_WAVE, ("Done") );
|
|
#else
|
|
pHdr->dwFlags &= ~WHDR_INQUEUE;
|
|
pHdr->dwFlags |= WHDR_DONE;
|
|
pHdr->lpNext = NULL;
|
|
|
|
waveCallback((LPDEVICEINFO)pHdr->reserved,
|
|
DeviceInfo->DeviceType == WaveOutDevice ? WOM_DONE : WIM_DATA,
|
|
(DWORD)pHdr);
|
|
#endif
|
|
|
|
}
|
|
}
|