Leaked source code of windows server 2003
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.
 
 
 
 
 
 

3159 lines
100 KiB

/* *************************************************************************
** INTEL Corporation Proprietary Information
**
** This listing is supplied under the terms of a license
** agreement with INTEL Corporation and may not be copied
** nor disclosed except in accordance with the terms of
** that agreement.
**
** Copyright (c) 1996 Intel Corporation.
** All Rights Reserved.
**
** *************************************************************************
** codec.c
**
** Description:
** This file contains the ACM wrapper code for the G.723.1 compressor.
**
**
****************************************************************************
*/
#include <windows.h>
#include <windowsx.h>
#include <mmsystem.h>
//#include <mmddk.h>
#include <ctype.h>
#include <mmreg.h>
#include <msacm.h>
#include <msacmdrv.h>
#include "g723.h"
#include "debug.h"
//#include "cst_lbc.h"
#include "coder.h"
#include "decod.h"
#include "sdstuff.h"
#include "float.h"
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
#include "isrg.h"
#endif // } NO_DEBUGGING_OUTPUT
#endif
const UINT gauFormatTagIndexToTag[] =
{
WAVE_FORMAT_PCM,
WAVE_FORMAT_MSG723
};
#define ACM_DRIVER_MAX_FORMAT_TAGS SIZEOF_ARRAY(gauFormatTagIndexToTag)
#define ACM_DRIVER_MAX_FILTER_TAGS 0
//
// Required by config.c as well as codec.c.
//
#define ACM_DRIVER_MAX_BITSPERSAMPLE_PCM 1
#define ACM_DRIVER_MAX_BITSPERSAMPLE_G723 0
//extern void glblSDinitialize(CODDEF *CodStat);
//extern void prefilter(CODDEF *CodStat,int bufsize);
//extern void getParams(CODDEF *CodStat,int buffersize);
//extern int initializeSD(CODDEF *CodStat);
//extern int silenceDetect(CODDEF *CodStat);
//extern void execSDloop(CODDEF *CodStat, int *isFrameSilent);
//
// number of formats we enumerate per channel is number of sample rates
// times number of channels times number of types (bits per sample).
//
#define ACM_DRIVER_MAX_FORMATS_PCM 2
#define ACM_DRIVER_MAX_FORMATS_G723 2
//==========================================================================;
//
//
//
//
//==========================================================================;
static float SDThreashold = 5.0f; // silence converter threashold.
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
static WORD ghISRInst = 0;
#endif // } NO_DEBUGGING_OUTPUT
#endif
//--------------------------------------------------------------------------;
//
// int LoadStringCodec
//
// Description:
// This function should be used by all codecs to load resource strings
// which will be passed back to the ACM. It works correctly for all
// platforms, as follows:
//
// Win16: Compiled to LoadString to load ANSI strings.
//
// Win32: The 32-bit ACM always expects Unicode strings. Therefore,
// when UNICODE is defined, this function is compiled to
// LoadStringW to load a Unicode string. When UNICODE is
// not defined, this function loads an ANSI string, converts
// it to Unicode, and returns the Unicode string to the
// codec.
//
// Note that you may use LoadString for other strings (strings which
// will not be passed back to the ACM), because these strings will
// always be consistent with the definition of UNICODE.
//
// Arguments:
// Same as LoadString, except that it expects an LPSTR for Win16 and a
// LPWSTR for Win32.
//
// Return (int):
// Same as LoadString.
//
//--------------------------------------------------------------------------;
#ifndef _WIN32
#define LoadStringCodec LoadString
#else
#ifdef UNICODE
#define LoadStringCodec LoadStringW
#else
int FNGLOBAL LoadStringCodec
(
HINSTANCE hinst,
UINT uID,
LPWSTR lpwstr,
int cch)
{
LPSTR lpstr;
int iReturn;
lpstr = (LPSTR)GlobalAlloc(GPTR, cch);
if (NULL == lpstr)
{
return 0;
}
iReturn = LoadStringA(hinst, uID, lpstr, cch);
if (0 == iReturn)
{
if (0 != cch)
{
lpwstr[0] = '\0';
}
}
else
{
MultiByteToWideChar( GetACP(), 0, lpstr, cch, lpwstr, cch );
}
GlobalFree((HGLOBAL)lpstr);
return iReturn;
}
#endif // UNICODE
#endif // _WIN32
//--------------------------------------------------------------------------;
//
// BOOL pcmIsValidFormat
//
// Description:
// This function verifies that a wave format header is a valid PCM
// header that we can deal with. Right now we are limited to an 8k
// sampling rate. That should go to 8 and 11 shortly.
//
// Arguments:
// LPWAVEFORMATEX pwfx: Pointer to format header to verify.
//
// Return (BOOL):
// The return value is non-zero if the format header looks valid. A
// zero return means the header is not valid.
//
//--------------------------------------------------------------------------;
BOOL FNLOCAL pcmIsValidFormat
(
LPWAVEFORMATEX pwfx
)
{
int i;
if (NULL == pwfx)
return (FALSE);
if (WAVE_FORMAT_PCM != pwfx->wFormatTag)
{
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Bad pcm wave format tag: %d",
pwfx->wFormatTag);
#endif // } NO_DEBUGGING_OUTPUT
#endif
return (FALSE);
}
//
// verify nChannels member is within the allowed range
//
if ((pwfx->nChannels != G723_MAX_CHANNELS) )
{
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Bad pcm channels: %d",pwfx->nChannels);
#endif // } NO_DEBUGGING_OUTPUT
#endif
return (FALSE);
}
//
// only allow the bits per sample that we can encode and decode with
//
if (16 != pwfx->wBitsPerSample )
{
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Bad pcm bits per sample: %d",
pwfx->wBitsPerSample);
#endif // } NO_DEBUGGING_OUTPUT
#endif
return (FALSE);
}
//
// only allow supported sampling rates
//
for(i=0;i<ACM_DRIVER_MAX_FORMATS_PCM;i++)
if (pwfx->nSamplesPerSec == PCM_SAMPLING_RATE[i])
break;
if (i == ACM_DRIVER_MAX_FORMATS_PCM)
{
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Bad pcm sampling rate: %d",
pwfx->nSamplesPerSec);
#endif // } NO_DEBUGGING_OUTPUT
#endif
return(FALSE);
}
//
// now verify that the block alignment is correct..
//
if (PCM_BLOCKALIGNMENT(pwfx) != pwfx->nBlockAlign)
{
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Bad pcm block alignment: %d",
pwfx->nBlockAlign);
#endif // } NO_DEBUGGING_OUTPUT
#endif
return (FALSE);
}
//
// finally, verify that avg bytes per second is correct
//
if (PCM_AVGBYTESPERSEC(pwfx) != pwfx->nAvgBytesPerSec)
{
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Bad pcm avg bytes per sec: %d",
pwfx->nAvgBytesPerSec);
#endif // } NO_DEBUGGING_OUTPUT
#endif
return (FALSE);
}
return (TRUE);
}
//--------------------------------------------------------------------------;
//
// BOOL G723IsValidFormat
//
// Description:
// This function verifies that a wave format header is a valid
// G.723.1 header that this ACM driver can deal with.
//
// Arguments:
// LPWAVEFORMATEX pwfx: Pointer to format header to verify.
//
// Return (BOOL):
// The return value is non-zero if the format header looks valid. A
// zero return means the header is not valid.
//
//--------------------------------------------------------------------------;
BOOL FNLOCAL g723IsValidFormat
(
LPWAVEFORMATEX pwfx
)
{
// LPMSG723WAVEFORMAT pwfg723;
if (NULL == pwfx)
{
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Bad 723 format structure pointer");
#endif // } NO_DEBUGGING_OUTPUT
#endif
return (FALSE);
}
if (WAVE_FORMAT_MSG723 != pwfx->wFormatTag)
{
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Bad 723 format tag: %d",
pwfx->wFormatTag);
#endif // } NO_DEBUGGING_OUTPUT
#endif
return (FALSE);
}
//
// check wBitsPerSample
//
if (G723_BITS_PER_SAMPLE != pwfx->wBitsPerSample)
{
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Bad 723 bits per sample: %d",
pwfx->wBitsPerSample);
#endif // } NO_DEBUGGING_OUTPUT
#endif
return (FALSE);
}
//
// check channels
//
if (pwfx->nChannels != G723_MAX_CHANNELS)
{
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Bad 723 channels: %d",
pwfx->nChannels);
#endif // } NO_DEBUGGING_OUTPUT
#endif
return (FALSE);
}
//
// Check block alignment - must be an integral number of DWORDs for
// mono, or an even number of DWORDs for stereo.
//
if( 0 != pwfx->nBlockAlign % (sizeof(DWORD)) )
{
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Bad 723 block alignment: %d",
pwfx->nBlockAlign);
#endif // } NO_DEBUGGING_OUTPUT
#endif
return FALSE;
}
if (G723_WFX_EXTRA_BYTES != pwfx->cbSize)
{
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Bad 723 extra bytes: %d",
pwfx->cbSize);
#endif // } NO_DEBUGGING_OUTPUT
#endif
return (FALSE);
}
// pwfg723 = (LPMSG723WAVEFORMAT)pwfx;
return (TRUE);
}
//--------------------------------------------------------------------------;
//
// UINT g723BlockAlign
//
// Description:
// This function computes the standard block alignment that should
// be used given the WAVEFORMATEX structure.
//
// NOTE! It is _assumed_ that the format is a valid G723 format
// and that the following fields in the format structure are valid:
//
// nChannels
// nSamplesPerSec
//
// Arguments:
// LPWAVEFORMATEX pwfx: Pointer to format header.
//
// Return (UINT):
// The return value is the block alignment that should be placed in
// the pwfx->nBlockAlign field.
//
//--------------------------------------------------------------------------;
UINT FNLOCAL g723BlockAlign
(
LPMSG723WAVEFORMAT pwfg723
)
{
UINT uBlockAlign;
if((pwfg723->wConfigWord&RATE) == Rate63)
uBlockAlign = 24;
else uBlockAlign = 20;
return (uBlockAlign);
}
//--------------------------------------------------------------------------;
//
// UINT g723AvgBytesPerSec
//
// Description:
// This function computes the Average Bytes Per Second (decoded!)
// that should be used given the WAVEFORMATEX structure.
//
// NOTE! It is _assumed_ that the format is a valid g723 format
// and that the following fields in the format structure are valid:
//
// nSamplesPerSec
// nBlockAlign
//
// Arguments:
// LPWAVEFORMATEX pwfx: Pointer to format header.
//
// Return (DWORD):
// The return value is the average bytes per second that should be
// placed in the pwfx->nAvgBytesPerSec field.
//
//--------------------------------------------------------------------------;
DWORD FNLOCAL g723AvgBytesPerSec
(
LPWAVEFORMATEX pwfx
)
{
int i;
DWORD dwAvgBytesPerSec;
//
// compute bytes per second including header bytes
//
dwAvgBytesPerSec = (pwfx->nSamplesPerSec * pwfx->nBlockAlign)
/ G723_SAMPLES_PER_BLOCK_PCM[0];
for(i=1;i<ACM_DRIVER_MAX_FORMATS_PCM;i++)
if (pwfx->nSamplesPerSec == PCM_SAMPLING_RATE[i])
dwAvgBytesPerSec = (pwfx->nSamplesPerSec * pwfx->nBlockAlign)
/ G723_SAMPLES_PER_BLOCK_PCM[i];
return (dwAvgBytesPerSec);
}
//--------------------------------------------------------------------------;
//
// LRESULT acmdDriverOpen
//
// Description:
// This function is used to handle the DRV_OPEN message for the ACM
// driver. The driver is 'opened' for many reasons with the most common
// being in preperation for conversion work. It is very important that
// the driver be able to correctly handle multiple open driver
// instances.
//
// Read the comments for this function carefully!
//
// Note that multiple _streams_ can (and will) be opened on a single
// open _driver instance_. Do not store/create instance data that must
// be unique for each stream in this function. See the acmdStreamOpen
// function for information on conversion streams.
//
// Arguments:
// HDRVR hdrvr: Driver handle that will be returned to caller of the
// OpenDriver function. Normally, this will be the ACM--but this is
// not guaranteed. For example, if an ACM driver is implemented within
// a waveform driver, then the driver will be opened by both MMSYSTEM
// and the ACM.
//
// LPACMDRVOPENDESC paod: Open description defining how the ACM driver
// is being opened. This argument may be NULL--see the comments below
// for more information.
//
// Return (LRESULT):
// The return value is non-zero if the open is successful. A zero
// return signifies that the driver cannot be opened.
//
//--------------------------------------------------------------------------;
LRESULT FNLOCAL acmdDriverOpen
(
HDRVR hdrvr,
LPACMDRVOPENDESC paod
)
{
PDRIVERINSTANCE pdi;
//
// the [optional] open description that is passed to this driver can
// be from multiple 'managers.' for example, AVI looks for installable
// drivers that are tagged with 'vidc' and 'vcap'. we need to verify
// that we are being opened as an Audio Compression Manager driver.
//
// if paod is NULL, then the driver is being opened for some purpose
// other than converting (that is, there will be no stream open
// requests for this instance of being opened). the most common case
// of this is the Control Panel's Drivers option checking for config
// support (DRV_[QUERY]CONFIGURE).
//
// we want to succeed this open, but be able to know that this
// open instance is bogus for creating streams. for this purpose we
// leave most of the members of our instance structure that we
// allocate below as zero...
//
// for some reason, floating point exceptions will crash Win9x
// machines. They should be getting ignored. The only work-around
// is to call _fpreset to force the floating point control word to
// be reinitialized
_fpreset();
if (NULL != paod)
{
//
// refuse to open if we are not being opened as an ACM driver.
// note that we do NOT modify the value of paod->dwError in this
// case.
//
if (ACMDRIVERDETAILS_FCCTYPE_AUDIOCODEC != paod->fccType)
return (0L);
}
//
// we are being opened as an installable driver--we can allocate some
// instance data to be returned in dwId argument of the DriverProc;
// or simply return non-zero to succeed the open.
//
// this driver allocates a small instance structure. note that we
// rely on allocating the memory as zero-initialized!
//
pdi = (PDRIVERINSTANCE)LocalAlloc(LPTR, sizeof(*pdi));
if (NULL == pdi)
{
//
// if this open attempt was as an ACM driver, then return the
// reason we are failing in the open description structure..
//
if (NULL != paod)
{
paod->dwError = MMSYSERR_NOMEM;
}
return (0L);
}
#ifdef NOTPRODUCT
pdi->enabled = TRUE;
#else
pdi->enabled = FALSE;
#endif
pdi->hdrvr = hdrvr;
pdi->hinst = GetDriverModuleHandle(hdrvr); // Module handle.
if (NULL != paod)
{
pdi->fnDriverProc = NULL;
pdi->fccType = paod->fccType;
pdi->vdwACM = paod->dwVersion;
pdi->fdwOpen = paod->dwFlags;
paod->dwError = MMSYSERR_NOERROR;
}
#ifdef G723_USECONFIG
//
// Get config info for this driver. If we're not passed an
// ACMDRVOPENDESC structure then we'll assume we are being
// opened for configuration and will put off getting the config
// info until we receive the DRV_CONFIGURE message. Otherwise we
// get the config info now using the alias passed through the
// ACMDRVOPENDESC structure.
//
pdi->hkey = NULL; // This is important!
if (NULL != paod)
{
#if defined(_WIN32) && !defined(UNICODE)
//
// We must translate the UNICODE alias name to an ANSI version
// that we can use.
//
LPSTR lpstr;
int iLen;
//
// Calculate required length without calling UNICODE APIs or CRT.
//
iLen = WideCharToMultiByte( GetACP(), 0, paod->pszAliasName,-1,
NULL, 0, NULL, NULL );
lpstr = (LPSTR)GlobalAllocPtr( GPTR, iLen );
if (NULL != lpstr)
{
WideCharToMultiByte( GetACP(), 0, paod->pszAliasName, iLen,
lpstr, iLen, NULL, NULL );
}
acmdDriverConfigInit(pdi, lpstr); // Note: OK to pass lpstr==NULL
if (NULL != lpstr)
{
GlobalFreePtr( lpstr );
}
#else
acmdDriverConfigInit(pdi, paod->pszAliasName);
#endif // _WIN32 && !UNICODE
}
#else
//
// Actually, fdwConfig is not used - there is no configuration data.
//
pdi->fdwConfig = 0L;
#endif // G723_USECONFIG
//
// non-zero return is success for DRV_OPEN
//
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
ISRREGISTERMODULE(&ghISRInst,"G723ACM","G.723.1 ACM Driver");
TTDBG(ghISRInst,TT_TRACE,"Driver Opened");
#endif // } NO_DEBUGGING_OUTPUT
#endif
return ((LRESULT)(UINT)pdi);
}
//--------------------------------------------------------------------------;
//
// LRESULT acmdDriverClose
//
// Description:
// This function handles the DRV_CLOSE message for the ACM driver. The
// driver receives a DRV_CLOSE message for each succeeded DRV_OPEN
// message (see acmdDriverOpen). The driver will only receive a close
// message for _successful_ opens.
//
// Arguments:
// PDRIVERINSTANCE pdi: Pointer to private ACM driver instance structure.
// This structure is [optionally] allocated during the DRV_OPEN message
// which is handled by the acmdDriverOpen function.
//
// Return (LRESULT):
// The return value is non-zero if the open instance can be closed.
// A zero return signifies that the ACM driver instance could not be
// closed.
//
// NOTE! It is _strongly_ recommended that the driver never fail to
// close. Note that the ACM will never allow a driver instance to
// be closed if there are open streams. An ACM driver does not need
// to check for this case.
//
//--------------------------------------------------------------------------;
LRESULT FNLOCAL acmdDriverClose
(
PDRIVERINSTANCE pdi
)
{
#ifdef G723_USECONFIG
//
// Release the registry key, if we allocated one.
//
if( NULL != pdi->hkey )
{
(void)RegCloseKey( pdi->hkey );
}
#endif
//
// check to see if we allocated instance data. if we did not, then
// immediately succeed.
//
if (NULL != pdi)
{
//
// close down the driver instance. this driver simply needs
// to free the instance data structure... note that if this
// 'free' fails, then this ACM driver probably trashed its
// heap; assume we didn't do that.
//
LocalFree((HLOCAL)pdi);
}
return (1L);
} // acmdDriverClose()
//--------------------------------------------------------------------------;
//
// LRESULT acmdDriverConfigure
//
// Description:
// This function is called to handle the DRV_[QUERY]CONFIGURE messages.
// These messages are for 'configuration' support of the driver.
// Normally this will be for 'hardware'--that is, a dialog should be
// displayed to configure ports, IRQ's, memory mappings, etc if it
// needs to. However, a software only ACM driver may also require
// configuration for 'what is real time' or other quality vs time
// issues.
//
// The most common way that these messages are generated under Win 3.1
// and NT Product 1 is from the Control Panel's Drivers option. Other
// sources may generate these messages in future versions of Windows.
//
// Arguments:
// PDRIVERINSTANCE pdi: Pointer to private ACM driver instance structure.
// This structure is [optionally] allocated during the DRV_OPEN message
// which is handled by the acmdDriverOpen function.
//
// HWND hwnd: Handle to parent window to use when displaying the
// configuration dialog box. An ACM driver is _required_ to display a
// modal dialog box using this hwnd argument as the parent. This
// argument may be (HWND)-1 which tells the driver that it is only
// being queried for configuration support.
//
// LPDRVCONFIGINFO pdci: Pointer to optional DRVCONFIGINFO structure.
// If this argument is NULL, then the ACM driver should invent its own
// storage location.
//
// Return (LRESULT):
// If the driver is being 'queried' for configuration support (that is,
// hwnd == (HWND)-1), then non-zero should be returned specifying
// the driver does support a configuration dialog--or zero should be
// returned specifying that no configuration dialog is supported.
//
// If the driver is being called to display the configuration dialog
// (that is, hwnd != (HWND)-1), then one of the following values
// should be returned:
//
// DRVCNF_CANCEL (0x0000): specifies that the dialog was displayed
// and canceled by the user. this value should also be returned if
// no configuration information was modified.
//
// DRVCNF_OK (0x0001): specifies that the dialog was displayed and
// the user pressed OK. This value should be returned even if the
// user didn't change anything - otherwise, the driver may not
// install properly.
//
// DRVCNF_RESTART (0x0002): specifies that the dialog was displayed
// and some configuration information was changed that requires
// Windows to be restarted before the changes take affect. the driver
// should remain configured with current values until the driver
// has been 'rebooted'.
//
//--------------------------------------------------------------------------;
LRESULT FNLOCAL acmdDriverConfigure
(
PDRIVERINSTANCE pdi,
HWND hwnd,
LPDRVCONFIGINFO pdci
)
{
// int n;
//
// first check to see if we are only being queried for configuration
// support. if hwnd == (HWND)-1 then we are being queried and should
// return zero for 'not supported' and non-zero for 'supported'.
//
if ((HWND)-1 == hwnd)
{
#ifdef G723_USECONFIG
//
// this ACM driver supports a configuration dialog box, so
// return non-zero...
//
return (1L);
#else
return(0L);
#endif
}
//
// we are being asked to bring up our configuration dialog. if this
// driver supports a configuration dialog box, then after the dialog
// is dismissed we must return one of the following values:
//
// DRVCNF_CANCEL (0x0000): specifies that the dialog was displayed
// and canceled by the user. this value should also be returned if
// no configuration information was modified.
//
// DRVCNF_OK (0x0001): specifies that the dialog was displayed and
// the user pressed OK. This value should be returned even if the
// user didn't change anything - otherwise, the driver may not
// install properly.
//
// DRVCNF_RESTART (0x0002): specifies that the dialog was displayed
// and some configuration information was changed that requires
// Windows to be restarted before the changes take affect. the driver
// should remain configured with current values until the driver
// has been 'rebooted'.
//
#ifdef G723_USECONFIG
if (NULL == pdci)
{
//
// !!!
//
DPF(2,"acmdDriverConfigure returning CANCEL due to NULL==pdci.");
return (DRVCNF_CANCEL);
}
pdi->pdci = pdci;
//
// We may not have our config info yet if the driver has only been
// opened specifically for configuration. So, read our configuration
// using the alias passed in the DRVCONFIGINFO structure passed
// through the DRV_CONFIGURE message
//
#if (defined(_WIN32) && !defined(UNICODE))
{
//
// We must translate the UNICODE alias name to an ANSI version
// that we can use.
//
LPSTR lpstr;
int iLen;
//
// Calculate required length without calling UNICODE APIs or CRT.
//
iLen = WideCharToMultiByte( GetACP(), 0, pdci->lpszDCIAliasName, -1,
NULL, 0, NULL, NULL );
lpstr = (LPSTR)GlobalAllocPtr( GPTR, iLen );
if (NULL != lpstr)
{
WideCharToMultiByte( GetACP(), 0, pdci->lpszDCIAliasName, iLen,
lpstr, iLen, NULL, NULL );
}
acmdDriverConfigInit(pdi, lpstr); // Note: OK to pass lpstr==NULL
if (NULL != lpstr)
{
GlobalFreePtr( lpstr );
}
}
#else
acmdDriverConfigInit(pdi, pdci->lpszDCIAliasName);
#endif // _WIN32 && !UNICODE
n = DialogBoxParam(pdi->hinst,
IDD_CONFIG,
hwnd,
acmdDlgProcConfigure,
(LPARAM)(UINT)pdi);
pdi->pdci = NULL;
return ((LRESULT)n);
#else
return(DRVCNF_CANCEL);
#endif // G723_USECONFIG
}
//--------------------------------------------------------------------------;
//
// LRESULT acmdDriverDetails
//
// Description:
// This function handles the ACMDM_DRIVER_DETAILS message. The ACM
// driver is responsible for filling in the ACMDRIVERDETAILS structure
// with various information.
//
// NOTE! It is *VERY* important that you fill in your ACMDRIVERDETAILS
// structure correctly. The ACM and applications must be able to
// rely on this information.
//
// WARNING! The _reserved_ bits of any fields of the ACMDRIVERDETAILS
// structure are _exactly that_: RESERVED. Do NOT use any extra
// flag bits, etc. for custom information. The proper way to add
// custom capabilities to your ACM driver is this:
//
// o define a new message in the ACMDM_USER range.
//
// o an application that wishes to use one of these extra features
// should then:
//
// o open the driver with acmDriverOpen.
//
// o check for the proper wMid and wPid using acmDriverDetails.
//
// o send the 'user defined' message with acmDriverMessage
// to retrieve additional information, etc.
//
// o close the driver with acmDriverClose.
//
// Arguments:
// PDRIVERINSTANCE pdi: Pointer to private ACM driver instance structure.
// This structure is [optionally] allocated during the DRV_OPEN message
// which is handled by the acmdDriverOpen function.
//
// LPACMDRIVERDETAILS padd: Pointer to ACMDRIVERDETAILS structure to
// fill in for the caller. This structure may be larger or smaller than
// the current definition of ACMDRIVERDETAILS--cbStruct specifies the
// valid size.
//
// Return (LRESULT):
// The return value is zero (MMSYSERR_NOERROR) for success. Non-zero
// signifies that the driver details could not be retrieved.
//
// NOTE THAT THIS FUNCTION SHOULD NEVER FAIL! There are two possible
// error conditions:
//
// o if padd is NULL or an invalid pointer.
//
// o if cbStruct is less than four; in this case, there is not enough
// room to return the number of bytes filled in.
//
// Because these two error conditions are easily defined, the ACM
// will catch these errors. The driver does NOT need to check for these
// conditions.
//
//--------------------------------------------------------------------------;
LRESULT FNLOCAL acmdDriverDetails
(
PDRIVERINSTANCE pdi,
LPACMDRIVERDETAILS padd
)
{
ACMDRIVERDETAILS add;
DWORD cbStruct;
//
// it is easiest to fill in a temporary structure with valid info
// and then copy the requested number of bytes to the destination
// buffer.
//
cbStruct = min(padd->cbStruct, sizeof(ACMDRIVERDETAILS));
add.cbStruct = cbStruct;
//
// for the current implementation of an ACM driver, the fccType and
// fccComp members *MUST* always be ACMDRIVERDETAILS_FCCTYPE_AUDIOCODEC
// ('audc') and ACMDRIVERDETAILS_FCCCOMP_UNDEFINED (0) respectively.
//
add.fccType = ACMDRIVERDETAILS_FCCTYPE_AUDIOCODEC;
add.fccComp = ACMDRIVERDETAILS_FCCCOMP_UNDEFINED;
//
// the manufacturer id (wMid) and product id (wPid) must be filled
// in with your company's _registered_ identifier's. for more
// information on these identifier's and how to get them registered
// contact Microsoft and get the Multimedia Developer Registration Kit:
//
// Microsoft Corporation
// Multimedia Technology Group
// One Microsoft Way
// Redmond, WA 98052-6399
//
// Developer Services Phone: (800) 227-4679 x11771
//
// note that during the development phase or your ACM driver, you may
// use the reserved value of '0' for both wMid and wPid. however it
// is not acceptable to ship a driver with these values.
//
add.wMid = MM_MICROSOFT;
add.wPid = NETMEETING_MSG723_ACM_ID;
//
// the vdwACM and vdwDriver members contain version information for
// the driver.
//
// vdwACM: must contain the version of the *ACM* that the driver was
// _designed_ for. this is the _minimum_ version number of the ACM
// that the driver will work with. this value must be >= V2.00.000.
//
// vdwDriver: the version of this ACM driver.
//
// ACM driver versions are 32 bit numbers broken into three parts as
// follows (note these parts are displayed as decimal values):
//
// bits 24 - 31: 8 bit _major_ version number
// bits 16 - 23: 8 bit _minor_ version number
// bits 0 - 15: 16 bit build number
//
add.vdwACM = VERSION_MSACM;
add.vdwDriver = VERSION_ACM_DRIVER;
//
// the following flags are used to specify the type of conversion(s)
// that the ACM driver supports. note that a driver may support one or
// more of these flags in any combination.
//
// ACMDRIVERDETAILS_SUPPORTF_CODEC: this flag is set if the driver
// supports conversions from one format tag to another format tag. for
// example, if a converter compresses or decompresses WAVE_FORMAT_PCM
// and WAVE_FORMAT_IMA_ADPCM, then this bit should be set. this is
// true even if the data is not actually changed in size--for example
// a conversion from u-Law to A-Law will still set this bit because
// the format tags differ.
//
// ACMDRIVERDETAILS_SUPPORTF_CONVERTER: this flags is set if the
// driver supports conversions on the same format tag. as an example,
// the PCM converter that is built into the ACM sets this bit (and only
// this bit) because it converts only between PCM formats (bits, sample
// rate).
//
// ACMDRIVERDETAILS_SUPPORTF_FILTER: this flag is set if the driver
// supports transformations on a single format tag but does change
// the base characteristics of the format (bit depth, sample rate, etc
// will remain the same). for example, a driver that changed the
// 'volume' of PCM data or applied a low pass filter would set this bit.
//
add.fdwSupport = ACMDRIVERDETAILS_SUPPORTF_CODEC;
//
// the number of individual format tags this ACM driver supports. for
// example, if a driver uses the WAVE_FORMAT_IMA_ADPCM and
// WAVE_FORMAT_PCM format tags, then this value would be two. if the
// driver only supports filtering on WAVE_FORMAT_PCM, then this value
// would be one. if this driver supported WAVE_FORMAT_ALAW,
// WAVE_FORMAT_MULAW and WAVE_FORMAT_PCM, then this value would be
// three. etc, etc.
//
add.cFormatTags = ACM_DRIVER_MAX_FORMAT_TAGS;
//
// the number of individual filter tags this ACM driver supports. if
// a driver supports no filters (ACMDRIVERDETAILS_SUPPORTF_FILTER is
// NOT set in the fdwSupport member), then this value must be zero.
//
add.cFilterTags = ACM_DRIVER_MAX_FILTER_TAGS;
//
// the remaining members in the ACMDRIVERDETAILS structure are sometimes
// not needed. because of this we make a quick check to see if we
// should go through the effort of filling in these members.
//
if (FIELD_OFFSET(ACMDRIVERDETAILS, hicon) < cbStruct)
{
//
// fill in the hicon member will a handle to a custom icon for
// the ACM driver. this allows the driver to be represented by
// an application graphically (usually this will be a company
// logo or something). if a driver does not wish to have a custom
// icon displayed, then simply set this member to NULL and a
// generic icon will be displayed instead.
//
// See the MSFILTER sample for a codec which contains a custom icon.
//
add.hicon = NULL;
//
// the short name and long name are used to represent the driver
// in a unique description. the short name is intended for small
// display areas (for example, in a menu or combo box). the long
// name is intended for more descriptive displays (for example,
// in an 'about box').
//
// NOTE! an ACM driver should never place formatting characters
// of any sort in these strings (for example CR/LF's, etc). it
// is up to the application to format the text.
//
LoadStringCodec(pdi->hinst, IDS_ACM_DRIVER_SHORTNAME, add.szShortName, SIZEOFACMSTR(add.szShortName));
LoadStringCodec(pdi->hinst, IDS_ACM_DRIVER_LONGNAME, add.szLongName, SIZEOFACMSTR(add.szLongName));
//
// the last three members are intended for 'about box' information.
// these members are optional and may be zero length strings if
// the driver wishes.
//
// NOTE! an ACM driver should never place formatting characters
// of any sort in these strings (for example CR/LF's, etc). it
// is up to the application to format the text.
//
if (FIELD_OFFSET(ACMDRIVERDETAILS, szCopyright) < cbStruct)
{
LoadStringCodec(pdi->hinst, IDS_ACM_DRIVER_COPYRIGHT, add.szCopyright, SIZEOFACMSTR(add.szCopyright));
LoadStringCodec(pdi->hinst, IDS_ACM_DRIVER_LICENSING, add.szLicensing, SIZEOFACMSTR(add.szLicensing));
LoadStringCodec(pdi->hinst, IDS_ACM_DRIVER_FEATURES, add.szFeatures, SIZEOFACMSTR(add.szFeatures));
}
}
//
// now copy the correct number of bytes to the caller's buffer
//
_fmemcpy(padd, &add, (UINT)add.cbStruct);
//
// success!
//
return (MMSYSERR_NOERROR);
}
//--------------------------------------------------------------------------;
//
// LRESULT acmdDriverAbout
//
// Description:
// This function is called to handle the ACMDM_DRIVER_ABOUT message.
// An ACM driver has the option of displaying its own 'about box' or
// letting the ACM (or calling application) display one for it. This
// message is normally sent by the Control Panel's Sound Mapper
// option.
//
// It is recommended that an ACM driver allow a default about box
// be displayed for it--there should be no reason to bloat the size
// of a driver to simply display copyright, etc information when that
// information is contained in the ACMDRIVERDETAILS structure.
//
// Arguments:
// PDRIVERINSTANCE pdi: Pointer to private ACM driver instance structure.
// This structure is [optionally] allocated during the DRV_OPEN message
// which is handled by the acmdDriverOpen function.
//
// HWND hwnd: Handle to parent window to use when displaying the
// configuration dialog box. An ACM driver is _required_ to display a
// modal dialog box using this hwnd argument as the parent. This
// argument may be (HWND)-1 which tells the driver that it is only
// being queried for about box support.
//
// Return (LRESULT):
// The return value is MMSYSERR_NOTSUPPORTED if the ACM driver does
// not support a custom dialog box. In this case, the ACM or calling
// application will display a generic about box using the information
// contained in the ACMDRIVERDETAILS structure returned by the
// ACMDM_DRIVER_DETAILS message.
//
// If the driver chooses to display its own dialog box, then after
// the dialog is dismissed by the user, MMSYSERR_NOERROR should be
// returned.
//
// If the hwnd argument is equal to (HWND)-1, then no dialog should
// be displayed (the driver is only being queried for support). The
// driver must still return MMSYSERR_NOERROR (supported) or
// MMSYSERR_NOTSUPPORTED (no custom about box supported).
//
//--------------------------------------------------------------------------;
LRESULT FNLOCAL acmdDriverAbout
(
PDRIVERINSTANCE pdi,
HWND hwnd
)
{
//
// first check to see if we are only being queried for custom about
// box support. if hwnd == (HWND)-1 then we are being queried and
// should return MMSYSERR_NOTSUPPORTED for 'not supported' and
// MMSYSERR_NOERROR for 'supported'.
//
if ((HWND)-1 == hwnd)
{
//
// this ACM driver does NOT support a custom about box, so
// return MMSYSERR_NOTSUPPORTED...
//
return (MMSYSERR_NOTSUPPORTED);
}
//
// this driver does not support a custom dialog, so tell the ACM or
// calling application to display one for us. note that this is the
// _recommended_ method for consistency and simplicity of ACM drivers.
// why write code when you don't have to?
//
return (MMSYSERR_NOTSUPPORTED);
}
//--------------------------------------------------------------------------;
//
// LRESULT acmdFormatSuggest
//
// Description:
// This function handles the ACMDM_FORMAT_SUGGEST message. The purpose
// of this function is to provide a way for the ACM, a wave mapper or
// an application to quickly get a destination format that this driver
// can convert the source format to. The 'suggested' format should
// be as close to a common format as possible. This message is normally
// sent in response to an acmFormatSuggest function call.
//
// Another way to think about this message is: what format would this
// driver like to convert the source format to?
//
// The caller may place restrictions on the destination format that
// should be suggested. The padfs->fdwSuggest member contains the
// restriction bits passed by the caller--see the description for
// the return value for more information.
//
// Arguments:
// PDRIVERINSTANCE pdi: Pointer to private ACM driver instance structure.
// This structure is [optionally] allocated during the DRV_OPEN message
// which is handled by the acmdDriverOpen function.
//
// LPACMDRVFORMATSUGGEST padfs: Pointer to an ACMDRVFORMATSUGGEST
// structure that describes the source and destination (possibly with
// restrictions) for a conversion.
//
// Return (LRESULT):
// The return value is zero (MMSYSERR_NOERROR) if this function
// succeeds with no errors. The return value is a non-zero error code
// if the function fails.
//
// The driver should return MMSYSERR_NOTSUPPORTED if one or more of
// the destination restriction bits is not supported. It is strongly
// recommended that the driver support at least the following suggestion
// restriction bits:
//
// ACM_FORMATSUGGESTF_WFORMATTAG: The destination format tag must be
// the same as the wFormatTag member in the destination format header.
//
// ACM_FORMATSUGGESTF_NCHANNELS: The destination channel count must be
// the same as the nChannels member in the destination format header.
//
// ACM_FORMATSUGGESTF_NSAMPLESPERSEC: The destination samples per
// second must be the same as the nSamplesPerSec member in the
// destination format header.
//
// ACM_FORMATSUGGESTF_WBITSPERSAMPLE: The destination bits per sample
// must be the same as the wBitsPerSample member in the destination
// format header.
//
// If no destintation format can be suggested, then the driver should
// return ACMERR_NOTPOSSIBLE.
//
//--------------------------------------------------------------------------;
LRESULT FNLOCAL acmdFormatSuggest
(
PDRIVERINSTANCE pdi,
LPACMDRVFORMATSUGGEST padfs
)
{
#define ACMD_FORMAT_SUGGEST_SUPPORT (ACM_FORMATSUGGESTF_WFORMATTAG | \
ACM_FORMATSUGGESTF_NCHANNELS | \
ACM_FORMATSUGGESTF_NSAMPLESPERSEC |\
ACM_FORMATSUGGESTF_WBITSPERSAMPLE)
LPWAVEFORMATEX pwfxSrc;
LPWAVEFORMATEX pwfxDst;
LPMSG723WAVEFORMAT pwfg723;
DWORD fdwSuggest;
int i;
//
// grab the suggestion restriction bits and verify that we support
// the ones that are specified... an ACM driver must return the
// MMSYSERR_NOTSUPPORTED if the suggestion restriction bits specified
// are not supported.
//
fdwSuggest = (ACM_FORMATSUGGESTF_TYPEMASK & padfs->fdwSuggest);
if (~ACMD_FORMAT_SUGGEST_SUPPORT & fdwSuggest)
return (MMSYSERR_NOTSUPPORTED);
//
// get the source and destination formats in more convenient variables
//
pwfxSrc = padfs->pwfxSrc;
pwfxDst = padfs->pwfxDst;
//
//
//
//
switch (pwfxSrc->wFormatTag)
{
case WAVE_FORMAT_PCM:
//
// strictly verify that the source format is acceptable for
// this driver
//
if (!pcmIsValidFormat(pwfxSrc))
break;
//
// if the destination format tag is restricted, verify that
// it is within our capabilities...
//
// this driver is only able to encode to G.723.1
//
if (ACM_FORMATSUGGESTF_WFORMATTAG & fdwSuggest)
{
if (WAVE_FORMAT_MSG723 != pwfxDst->wFormatTag)
return (ACMERR_NOTPOSSIBLE);
}
else
{
pwfxDst->wFormatTag = WAVE_FORMAT_MSG723;
}
//
// if the destination channel count is restricted, verify that
// it is within our capabilities...
//
// this driver is not able to handle more than 1 channel
//
if (ACM_FORMATSUGGESTF_NCHANNELS & fdwSuggest)
{
if (pwfxSrc->nChannels != G723_MAX_CHANNELS)
return (ACMERR_NOTPOSSIBLE);
}
else
{
pwfxDst->nChannels = G723_MAX_CHANNELS;
}
//
// if the destination samples per second is restricted, verify
// that it is within our capabilities...
//
// G.723.1 is designed for 8000 Hz sampling rate
//
if (ACM_FORMATSUGGESTF_NSAMPLESPERSEC & fdwSuggest)
{
if (pwfxDst->nSamplesPerSec != 8000)
return (ACMERR_NOTPOSSIBLE);
}
else
{
pwfxDst->nSamplesPerSec = 8000;
}
//
// if the destination bits per sample is restricted, verify
// that it is within our capabilities...
//
if (ACM_FORMATSUGGESTF_WBITSPERSAMPLE & fdwSuggest)
{
if (G723_BITS_PER_SAMPLE != pwfxDst->wBitsPerSample)
return (ACMERR_NOTPOSSIBLE);
}
else
{
pwfxDst->wBitsPerSample = G723_BITS_PER_SAMPLE;
}
//
// at this point, we have filled in all fields except the
// following for our 'suggested' destination format:
//
// nAvgBytesPerSec
// nBlockAlign
// cbSize
//
// wSamplesPerBlock -> G723 extended information
//
pwfg723 = (LPMSG723WAVEFORMAT)pwfxDst;
pwfg723->wConfigWord = Rate63;
pwfxDst->nBlockAlign = g723BlockAlign(pwfg723);
pwfxDst->nAvgBytesPerSec = g723AvgBytesPerSec(pwfxDst);
pwfxDst->cbSize = G723_WFX_EXTRA_BYTES;
return (MMSYSERR_NOERROR);
case WAVE_FORMAT_MSG723:
//
// strictly verify that the source format is acceptable for
// this driver
//
if (!g723IsValidFormat(pwfxSrc))
return (ACMERR_NOTPOSSIBLE);
//
// if the destination format tag is restricted, verify that
// it is within our capabilities...
//
// this driver is only able to decode to PCM
//
if (ACM_FORMATSUGGESTF_WFORMATTAG & fdwSuggest)
{
if (WAVE_FORMAT_PCM != pwfxDst->wFormatTag)
return (ACMERR_NOTPOSSIBLE);
}
else
{
pwfxDst->wFormatTag = WAVE_FORMAT_PCM;
}
//
// if the destination channel count is restricted, verify that
// it is within our capabilities...
//
// this driver is not able to change the number of channels
//
if (ACM_FORMATSUGGESTF_NCHANNELS & fdwSuggest)
{
if (pwfxSrc->nChannels != G723_MAX_CHANNELS)
return (ACMERR_NOTPOSSIBLE);
}
else
{
pwfxDst->nChannels = G723_MAX_CHANNELS;
}
//
// if the destination samples per second is restricted, verify
// that it is within our capabilities...
//
// G.723.1 produces PCM at 8000 Hz sampling rate
//
if (ACM_FORMATSUGGESTF_NSAMPLESPERSEC & fdwSuggest)
{
for(i=0;i<ACM_DRIVER_MAX_FORMATS_PCM;i++)
if (pwfxDst->nSamplesPerSec == PCM_SAMPLING_RATE[i])
break;
if (i == ACM_DRIVER_MAX_FORMATS_PCM)
return (ACMERR_NOTPOSSIBLE);
pwfxDst->nSamplesPerSec = pwfxSrc->nSamplesPerSec;
}
else
{
//
// 11025 is the default since it is more common
// in the PC world than 8000. Moreover, this
// prevents applications like MS Sound Recorder
// from using a low quality SRC in certain cases.
//
pwfxDst->nSamplesPerSec = 11025;
}
//
// if the destination bits per sample is restricted, verify
// that it is within our capabilities...
//
// this driver is only able to decode to 16 bit
//
if (ACM_FORMATSUGGESTF_WBITSPERSAMPLE & fdwSuggest)
{
if (16 != pwfxDst->wBitsPerSample)
return (ACMERR_NOTPOSSIBLE);
}
else
{
pwfxDst->wBitsPerSample = 16;
}
//
// at this point, we have filled in all fields except the
// following for our 'suggested' destination format:
//
// nAvgBytesPerSec
// nBlockAlign
// cbSize !!! not used for PCM !!!
//
pwfxDst->nBlockAlign = PCM_BLOCKALIGNMENT(pwfxDst);
pwfxDst->nAvgBytesPerSec = pwfxDst->nSamplesPerSec *
pwfxDst->nBlockAlign;
// pwfxDst->cbSize = not used;
return (MMSYSERR_NOERROR);
}
//
// can't suggest anything because either the source format is foreign
// or the destination format has restrictions that this ACM driver
// cannot deal with.
//
return (ACMERR_NOTPOSSIBLE);
}
//--------------------------------------------------------------------------;
//
// LRESULT acmdFormatTagDetails
//
// Description:
// This function handles the ACMDM_FORMATTAG_DETAILS message. This
// message is normally sent in response to an acmFormatTagDetails or
// acmFormatTagEnum function call. The purpose of this function is
// to get details about a specific format tag supported by this ACM
// driver.
//
// Arguments:
// PDRIVERINSTANCE pdi: Pointer to private ACM driver instance structure.
// This structure is [optionally] allocated during the DRV_OPEN message
// which is handled by the acmdDriverOpen function.
//
// LPACMFORMATTAGDETAILS padft: Pointer to an ACMFORMATTAGDETAILS
// structure that describes what format tag to retrieve details for.
//
// DWORD fdwDetails: Flags defining what format tag to retrieve the
// details for.
//
// Return (LRESULT):
// The return value is zero (MMSYSERR_NOERROR) if this function
// succeeds with no errors. The return value is a non-zero error code
// if the function fails.
//
// The driver should return MMSYSERR_NOTSUPPORTED if the query type
// specified in fdwDetails is not supported. An ACM driver must
// support at least the following query types:
//
// ACM_FORMATTAGDETAILSF_INDEX: Indicates that a format tag index
// was given in the dwFormatTagIndex member of the ACMFORMATTAGDETAILS
// structure. The format tag and details must be returned in the
// structure specified by padft. The index ranges from zero to one less
// than the cFormatTags member returned in the ACMDRIVERDETAILS
// structure for this driver.
//
// ACM_FORMATTAGDETAILSF_FORMATTAG: Indicates that a format tag
// was given in the dwFormatTag member of the ACMFORMATTAGDETAILS
// structure. The format tag details must be returned in the structure
// specified by padft.
//
// ACM_FORMATTAGDETAILSF_LARGESTSIZE: Indicates that the details
// on the format tag with the largest format size in bytes must be
// returned. The dwFormatTag member will either be WAVE_FORMAT_UNKNOWN
// or the format tag to find the largest size for.
//
// If the details for the specified format tag cannot be retrieved
// from this driver, then ACMERR_NOTPOSSIBLE should be returned.
//
//--------------------------------------------------------------------------;
LRESULT FNLOCAL acmdFormatTagDetails
(
PDRIVERINSTANCE pdi,
LPACMFORMATTAGDETAILS padft,
DWORD fdwDetails
)
{
UINT uFormatTag;
switch (ACM_FORMATTAGDETAILSF_QUERYMASK & fdwDetails)
{
case ACM_FORMATTAGDETAILSF_INDEX:
//
// if the index is too large, then they are asking for a
// non-existant format. return error.
//
if (ACM_DRIVER_MAX_FORMAT_TAGS <= padft->dwFormatTagIndex)
return (ACMERR_NOTPOSSIBLE);
uFormatTag = gauFormatTagIndexToTag[(UINT)padft->dwFormatTagIndex];
break;
case ACM_FORMATTAGDETAILSF_LARGESTSIZE:
switch (padft->dwFormatTag)
{
case WAVE_FORMAT_UNKNOWN:
case WAVE_FORMAT_MSG723:
uFormatTag = WAVE_FORMAT_MSG723;
break;
case WAVE_FORMAT_PCM:
uFormatTag = WAVE_FORMAT_PCM;
break;
default:
return (ACMERR_NOTPOSSIBLE);
}
break;
case ACM_FORMATTAGDETAILSF_FORMATTAG:
switch (padft->dwFormatTag)
{
case WAVE_FORMAT_MSG723:
uFormatTag = WAVE_FORMAT_MSG723;
break;
case WAVE_FORMAT_PCM:
uFormatTag = WAVE_FORMAT_PCM;
break;
default:
return (ACMERR_NOTPOSSIBLE);
}
break;
//
// if this ACM driver does not understand a query type, then
// return 'not supported'
//
default:
return (MMSYSERR_NOTSUPPORTED);
}
//
//
//
//
switch (uFormatTag)
{
case WAVE_FORMAT_PCM:
padft->dwFormatTagIndex = 0;
padft->dwFormatTag = WAVE_FORMAT_PCM;
padft->cbFormatSize = sizeof(PCMWAVEFORMAT);
padft->fdwSupport = ACMDRIVERDETAILS_SUPPORTF_CODEC;
padft->cStandardFormats = ACM_DRIVER_MAX_FORMATS_PCM;
//
// the ACM is responsible for the PCM format tag name
//
padft->szFormatTag[0] = '\0';
break;
case WAVE_FORMAT_MSG723:
padft->dwFormatTagIndex = 1;
padft->dwFormatTag = WAVE_FORMAT_MSG723;
padft->cbFormatSize = sizeof(WAVEFORMATEX) +
G723_WFX_EXTRA_BYTES;
padft->fdwSupport = ACMDRIVERDETAILS_SUPPORTF_CODEC;
padft->cStandardFormats = ACM_DRIVER_MAX_FORMATS_G723;
LoadStringCodec(pdi->hinst,
IDS_ACM_DRIVER_TAG_NAME,
padft->szFormatTag,
SIZEOFACMSTR(padft->szFormatTag));
break;
default:
return (ACMERR_NOTPOSSIBLE);
}
//
// return only the requested info
//
// the ACM will guarantee that the ACMFORMATTAGDETAILS structure
// passed is at least large enough to hold the base information of
// the details structure
//
padft->cbStruct = min(padft->cbStruct, sizeof(*padft));
//
return (MMSYSERR_NOERROR);
} // acmdFormatTagDetails()
//--------------------------------------------------------------------------;
//
// LRESULT acmdFormatDetails
//
// Description:
// This function handles the ACMDM_FORMAT_DETAILS message. This
// message is normally sent in response to an acmFormatDetails or
// acmFormatEnum function call. The purpose of this function is
// to get details about a specific format for a specified format tag
// supported by this ACM driver.
//
// Note that an ACM driver can return a zero length string for the
// format name if it wishes to have the ACM create a format string
// for it. This is strongly recommended to simplify internationalizing
// the driver--the ACM will automatically take care of that. The
// following formula is used to format a string by the ACM:
//
// <nSamplesPerSec> kHz, <bit depth> bit, [Mono | Stereo | nChannels]
//
// <bit depth> = <nAvgBytesPerSec> * 8 / nSamplesPerSec / nChannels;
//
// Arguments:
// PDRIVERINSTANCE pdi: Pointer to private ACM driver instance structure.
// This structure is [optionally] allocated during the DRV_OPEN message
// which is handled by the acmdDriverOpen function.
//
// LPACMFORMATDETAILS padf: Pointer to an ACMFORMATDETAILS structure
// that describes what format (for a specified format tag) to retrieve
// details for.
//
// DWORD fdwDetails: Flags defining what format for a specified format
// tag to retrieve the details for.
//
// Return (LRESULT):
// The return value is zero (MMSYSERR_NOERROR) if this function
// succeeds with no errors. The return value is a non-zero error code
// if the function fails.
//
// The driver should return MMSYSERR_NOTSUPPORTED if the query type
// specified in fdwDetails is not supported. An ACM driver must
// support at least the following query types:
//
// ACM_FORMATDETAILSF_INDEX: Indicates that a format index for the
// format tag was given in the dwFormatIndex member of the
// ACMFORMATDETAILS structure. The format details must be returned in
// the structure specified by padf. The index ranges from zero to one
// less than the cStandardFormats member returned in the
// ACMFORMATTAGDETAILS structure for a format tag.
//
// ACM_FORMATDETAILSF_FORMAT: Indicates that a WAVEFORMATEX structure
// pointed to by pwfx of the ACMFORMATDETAILS structure was given and
// the remaining details should be returned. The dwFormatTag member
// of the ACMFORMATDETAILS will be initialized to the same format
// tag as the pwfx member specifies. This query type may be used to
// get a string description of an arbitrary format structure.
//
// If the details for the specified format cannot be retrieved
// from this driver, then ACMERR_NOTPOSSIBLE should be returned.
//
//--------------------------------------------------------------------------;
LRESULT FNLOCAL acmdFormatDetails
(
PDRIVERINSTANCE pdi,
LPACMFORMATDETAILS padf,
DWORD fdwDetails
)
{
LPWAVEFORMATEX pwfx;
LPMSG723WAVEFORMAT pwfg723;
switch (ACM_FORMATDETAILSF_QUERYMASK & fdwDetails)
{
//
// enumerate by index
//
// verify that the format tag is something we know about and
// return the details on the 'standard format' supported by
// this driver at the specified index...
//
case ACM_FORMATDETAILSF_INDEX:
pwfx = padf->pwfx;
switch (padf->dwFormatTag)
{
case WAVE_FORMAT_PCM:
if (padf->dwFormatIndex >= ACM_DRIVER_MAX_FORMATS_PCM)
return (ACMERR_NOTPOSSIBLE);
pwfx->nSamplesPerSec
= PCM_SAMPLING_RATE[padf->dwFormatIndex];
pwfx->wFormatTag = WAVE_FORMAT_PCM;
pwfx->nChannels = 1;
pwfx->wBitsPerSample = 16;
pwfx->nBlockAlign = PCM_BLOCKALIGNMENT(pwfx);
pwfx->nAvgBytesPerSec = pwfx->nSamplesPerSec * pwfx->nBlockAlign;
//
// note that the cbSize field is NOT valid for PCM
// formats
//
// pwfx->cbSize = 0;
break;
case WAVE_FORMAT_MSG723:
if (padf->dwFormatIndex >= ACM_DRIVER_MAX_FORMATS_G723)
return (ACMERR_NOTPOSSIBLE);
pwfx->wFormatTag = WAVE_FORMAT_MSG723;
pwfx->nSamplesPerSec
= G723_SAMPLING_RATE[padf->dwFormatIndex];
pwfx->nChannels = G723_MAX_CHANNELS;
pwfx->wBitsPerSample = G723_BITS_PER_SAMPLE;
pwfg723 = (LPMSG723WAVEFORMAT)pwfx;
if(padf->dwFormatIndex == 0)
{
pwfx->nBlockAlign = 24;
pwfx->nAvgBytesPerSec = 800;
pwfg723->wConfigWord = Rate63+POST_FILTER;
}
if(padf->dwFormatIndex == 1)
{
pwfx->nBlockAlign = 20;
pwfx->nAvgBytesPerSec = 666;
pwfg723->wConfigWord = Rate53+POST_FILTER;
}
if(padf->dwFormatIndex == 2)
{
pwfx->nBlockAlign = 24;
pwfx->nAvgBytesPerSec = 800;
pwfg723->wConfigWord=Rate63+POST_FILTER+SILENCE_ENABLE;
}
if(padf->dwFormatIndex == 3)
{
pwfx->nBlockAlign = 20;
pwfx->nAvgBytesPerSec = 666;
pwfg723->wConfigWord=Rate53+POST_FILTER+SILENCE_ENABLE;
}
pwfx->cbSize = G723_WFX_EXTRA_BYTES;
break;
default:
return (ACMERR_NOTPOSSIBLE);
}
//
// return details on specified format
//
// the caller normally uses this to verify that the format is
// supported and to retrieve a string description...
//
case ACM_FORMATDETAILSF_FORMAT:
pwfx = padf->pwfx;
switch (pwfx->wFormatTag)
{
case WAVE_FORMAT_PCM:
if (!pcmIsValidFormat(pwfx))
return (ACMERR_NOTPOSSIBLE);
break;
case WAVE_FORMAT_MSG723:
if (!g723IsValidFormat(pwfx))
return (ACMERR_NOTPOSSIBLE);
break;
default:
return (ACMERR_NOTPOSSIBLE);
}
break;
default:
return (MMSYSERR_NOTSUPPORTED);
}
padf->cbStruct = min(padf->cbStruct, sizeof(*padf));
padf->fdwSupport = ACMDRIVERDETAILS_SUPPORTF_CODEC;
pwfg723 = (LPMSG723WAVEFORMAT)pwfx;
if(padf->dwFormatTag == WAVE_FORMAT_MSG723)
{
if((pwfg723->wConfigWord&5) == 0)
LoadStringCodec(pdi->hinst, IDS_FORMAT_DETAILS_MONO_8KHZ_6400BIT_S,
padf->szFormat,SIZEOFACMSTR(padf->szFormat));
if((pwfg723->wConfigWord&5) == 4)
LoadStringCodec(pdi->hinst, IDS_FORMAT_DETAILS_MONO_8KHZ_6400BIT_SID,
padf->szFormat,SIZEOFACMSTR(padf->szFormat));
if((pwfg723->wConfigWord&5) == 1)
LoadStringCodec(pdi->hinst, IDS_FORMAT_DETAILS_MONO_8KHZ_5333BIT_S,
padf->szFormat,SIZEOFACMSTR(padf->szFormat));
if((pwfg723->wConfigWord&5) == 5)
LoadStringCodec(pdi->hinst, IDS_FORMAT_DETAILS_MONO_8KHZ_5333BIT_SID,
padf->szFormat,SIZEOFACMSTR(padf->szFormat));
}
else
padf->szFormat[0] = '\0';
//
//
//
return (MMSYSERR_NOERROR);
} // acmdFormatDetails()
//==========================================================================;
//
//
//
//
//==========================================================================;
//--------------------------------------------------------------------------;
//
// LRESULT acmdStreamOpen
//
// Description:
// This function handles the ACMDM_STREAM_OPEN message. This message
// is sent to initiate a new conversion stream. This is usually caused
// by an application calling acmStreamOpen. If this function is
// successful, then one or more ACMDM_STREAM_CONVERT messages will be
// sent to convert individual buffers (user calls acmStreamConvert).
//
// Note that an ACM driver will not receive open requests for ASYNC
// or FILTER operations unless the ACMDRIVERDETAILS_SUPPORTF_ASYNC
// or ACMDRIVERDETAILS_SUPPORTF_FILTER flags are set in the
// ACMDRIVERDETAILS structure. There is no need for the driver to
// check for these requests unless it sets those support bits.
//
// If the ACM_STREAMOPENF_QUERY flag is set in the padsi->fdwOpen
// member, then no resources should be allocated. Just verify that
// the conversion request is possible by this driver and return the
// appropriate error (either ACMERR_NOTPOSSIBLE or MMSYSERR_NOERROR).
// The driver will NOT receive an ACMDM_STREAM_CLOSE for queries.
//
// Arguments:
// PDRIVERINSTANCE pdi: Pointer to private ACM driver instance structure.
// This structure is [optionally] allocated during the DRV_OPEN message
// which is handled by the acmdDriverOpen function.
//
// LPACMDRVSTREAMINSTANCE padsi: Pointer to instance data for the
// conversion stream. This structure was allocated by the ACM and
// filled with the most common instance data needed for conversions.
// This structure will be passed back to all future stream messages
// if the open succeeds. The information in this structure will never
// change during the lifetime of the stream--so it is not necessary
// to re-verify the information referenced by this structure.
//
// Return (LRESULT):
// The return value is zero (MMSYSERR_NOERROR) if this function
// succeeds with no errors. The return value is a non-zero error code
// if the function fails.
//
// A driver should return ACMERR_NOTPOSSIBLE if the conversion cannot
// be performed due to incompatible source and destination formats.
//
// A driver should return MMSYSERR_NOTSUPPORTED if the conversion
// cannot be performed in real-time and the request does not specify
// the ACM_STREAMOPENF_NONREALTIME flag.
//
//--------------------------------------------------------------------------;
LRESULT FNLOCAL acmdStreamOpen
(
PDRIVERINSTANCE pdi,
LPACMDRVSTREAMINSTANCE padsi
)
{
LPWAVEFORMATEX pwfxSrc;
LPWAVEFORMATEX pwfxDst;
LPMSG723WAVEFORMAT pwfg723;
CODDEF *CodStat;
DECDEF *DecStat;
G723CODDEF *g723Inst;
INSTNCE *SD_Inst;
UINT psi;
int i;
#ifdef G723_USECONFIG
DWORD nConfigMaxRTEncodeSamplesPerSec;
DWORD nConfigMaxRTDecodeSamplesPerSec;
DWORD dw;
#endif
//
//
//
pwfxSrc = padsi->pwfxSrc;
pwfxDst = padsi->pwfxDst;
// fRealTime = (0 == (padsi->fdwOpen & ACM_STREAMOPENF_NONREALTIME));
//
// this driver first verifies that the source and destination formats
// are acceptable...
//
switch (pwfxSrc->wFormatTag)
{
case WAVE_FORMAT_PCM:
if (!pcmIsValidFormat(pwfxSrc))
return (ACMERR_NOTPOSSIBLE);
if (!g723IsValidFormat(pwfxDst))
return (ACMERR_NOTPOSSIBLE);
break;
case WAVE_FORMAT_MSG723:
if (!g723IsValidFormat(pwfxSrc))
return (ACMERR_NOTPOSSIBLE);
if (!pcmIsValidFormat(pwfxDst))
return (ACMERR_NOTPOSSIBLE);
break;
default:
return (ACMERR_NOTPOSSIBLE);
}
//
// for this driver, we must also verify that the nChannels
// member is the same between the source and destination
// formats.
//
if (pwfxSrc->nChannels != pwfxDst->nChannels)
return (MMSYSERR_NOTSUPPORTED);
//
// we have determined that the conversion requested is possible by
// this driver. now check if we are just being queried for support.
// if this is just a query, then do NOT allocate any instance data
// or create tables, etc. just succeed the call.
//
if (0 != (ACM_STREAMOPENF_QUERY & padsi->fdwOpen))
{
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Stream open query");
#endif // } NO_DEBUGGING_OUTPUT
#endif
return (MMSYSERR_NOERROR);
}
//
// we have decided that this driver can handle the conversion stream.
// so we want to do _AS MUCH WORK AS POSSIBLE_ right now to prepare
// for converting data. any resource allocation, table building, etc
// that can be dealt with at this time should be done.
//
// THIS IS VERY IMPORTANT! all ACMDM_STREAM_CONVERT messages need to
// be handled as quickly as possible.
//
// this driver allocates a small instance structure for each stream
//
//
switch (pwfxSrc->wFormatTag)
{
case WAVE_FORMAT_PCM:
g723Inst = LocalAlloc(LPTR,sizeof(G723CODDEF));
if (NULL == g723Inst)
return (MMSYSERR_NOMEM);
CodStat = &g723Inst->CodStat;
Init_Coder(CodStat);
CodStat->UseHp = True;
pwfg723 = (LPMSG723WAVEFORMAT)pwfxDst;
CodStat->WrkRate = pwfg723->wConfigWord&RATE;
CodStat->srccount = (int)0;
for(i=0;i<SRCSTATELEN;i++)
CodStat->srcstate[i] = (short)0;
/*****************************************************************
Silence detector Init
*****************************************************************/
/* This value enables silence detection, early exit, and the default
** squelch value.
*/
SD_Inst = &g723Inst->SD_Instance;
SD_Inst->SDFlags = 0x00000005;
// isFrameSilent = 0; // initialize to "not silent"
/*
** Inintialize first OFFSET samples of storebuff
*/
for(i=0;i<OFFSET;i++)SD_Inst->SDstate.Filt.storebuff[i] = 0.0f;
glblSDinitialize(SD_Inst);
psi = (DWORD)(UINT)g723Inst;
#ifndef NOTPRODUCT
if ((pwfg723->dwCodeword1 == G723MAGICWORD1)
&& (pwfg723->dwCodeword2 == G723MAGICWORD2))
{
pdi->enabled = TRUE;
}
#endif // NOTPRODUCT
break;
case WAVE_FORMAT_MSG723:
DecStat = LocalAlloc(LPTR,sizeof(DECDEF));
if (NULL == DecStat)
return (MMSYSERR_NOMEM);
Init_Decod(DecStat); //low rate high pass and post filters enabled
pwfg723 = (LPMSG723WAVEFORMAT)pwfxSrc;
DecStat->UsePf = pwfg723->wConfigWord & POST_FILTER;
#ifndef NOTPRODUCT
if ((pwfg723->dwCodeword1 == G723MAGICWORD1)
&& (pwfg723->dwCodeword2 == G723MAGICWORD2))
{
pdi->enabled = TRUE;
}
#endif // NOTPRODUCT
// DecStat->FrameSize = 20;
// if((pwfg723->wConfigWord & RATE) == 0)
// DecStat->FrameSize = 24;
DecStat->srccount = (int)0;
for(i=0;i<SRCSTATELEN;i++)
DecStat->srcstate[i] = (short)0;
DecStat->i = 0;
// DecStat->srcbuffend = DecStat->srcbuff;
psi = (DWORD)(UINT)DecStat;
break;
}
//
// fill out our instance structure
//
// this driver stores a pointer to the conversion function that will
// be used for each conversion on this stream. we also store a
// copy of the _current_ configuration of the driver instance we
// are opened on. this must not change during the life of the stream
// instance.
//
// this is also very important! if the user is able to configure how
// the driver performs conversions, the changes should only affect
// future open streams. all current open streams should behave as
// they were configured during the open.
//
// psi->fnConvert = fnConvert;
// psi->fdwConfig = pdi->fdwConfig;
//
// fill in our instance data--this will be passed back to all stream
// messages in the ACMDRVSTREAMINSTANCE structure. it is entirely
// up to the driver what gets stored (and maintained) in the
// fdwDriver and dwDriver members.
//
padsi->fdwDriver = 0L;
padsi->dwDriver = psi;
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Stream Open padsi=%lx psi=%lx",padsi,psi);
#endif // } NO_DEBUGGING_OUTPUT
#endif
return (MMSYSERR_NOERROR);
} // acmdStreamOpen()
//--------------------------------------------------------------------------;
//
// LRESULT acmdStreamClose
//
// Description:
// This function is called to handle the ACMDM_STREAM_CLOSE message.
// This message is sent when a conversion stream is no longer being
// used (the stream is being closed; usually by an application
// calling acmStreamClose).
//
// Arguments:
// LPACMDRVSTREAMINSTANCE padsi: Pointer to instance data for the
// conversion stream.
//
// Return (LRESULT):
// The return value is zero (MMSYSERR_NOERROR) if this function
// succeeds with no errors. The return value is a non-zero error code
// if the function fails.
//
//
// An asyncronous conversion stream may fail with ACMERR_BUSY if there
// are pending buffers. An application may call acmStreamReset to
// force all pending buffers to be posted.
//
//--------------------------------------------------------------------------;
LRESULT FNLOCAL acmdStreamClose
(
LPACMDRVSTREAMINSTANCE padsi
)
{
UINT psi;
#ifdef LOG_ENCODE_TIMINGS_ON // { LOG_ENCODE_TIMINGS_ON
CODDEF *CodStat;
G723CODDEF *g723Inst;
#endif // } LOG_ENCODE_TIMINGS_ON
#ifdef LOG_DECODE_TIMINGS_ON // { LOG_DECODE_TIMINGS_ON
DECDEF *DecStat;
#endif // } LOG_ENCODE_TIMINGS_ON
//
// the driver should clean up all privately allocated resources that
// were created for maintaining the stream instance. if no resources
// were allocated, then simply succeed.
//
// in the case of this driver, we need to free the stream instance
// structure that we allocated during acmdStreamOpen.
//
psi = (UINT)padsi->dwDriver;
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Stream Close padsi=%lx psi=%lx",padsi,psi);
#endif // } NO_DEBUGGING_OUTPUT
#endif
if (0 != psi)
{
#ifdef LOG_ENCODE_TIMINGS_ON // { LOG_ENCODE_TIMINGS_ON
g723Inst = (G723CODDEF *)psi;
CodStat = &g723Inst->CodStat;
OutputEncodeTimingStatistics("c:\\encode.txt", CodStat->EncTimingInfo, CodStat->dwStatFrameCount);
#endif // } LOG_ENCODE_TIMINGS_ON
#ifdef LOG_DECODE_TIMINGS_ON // { LOG_DECODE_TIMINGS_ON
DecStat = (DECDEF *)psi;
OutputDecodeTimingStatistics("c:\\decode.txt", DecStat->DecTimingInfo, DecStat->dwStatFrameCount);
#endif // } LOG_DECODE_TIMINGS_ON
//
// free the stream instance structure
//
LocalFree((HLOCAL)psi);
}
return (MMSYSERR_NOERROR);
} // acmdStreamClose()
//--------------------------------------------------------------------------;
//
// LRESULT acmdStreamSize
//
// Description:
// This function handles the ACMDM_STREAM_SIZE message. The purpose
// of this function is to provide the _largest size in bytes_ that
// the source or destination buffer needs to be given the input and
// output formats and the size in bytes of the source or destination
// data buffer.
//
// In other words: how big does my destination buffer need to be to
// hold the converted data? (ACM_STREAMSIZEF_SOURCE)
//
// Or: how big can my source buffer be given the destination buffer?
// (ACM_STREAMSIZEF_DESTINATION)
//
// Arguments:
// LPACMDRVSTREAMINSTANCE padsi: Pointer to instance data for the
// conversion stream. This structure was allocated by the ACM and
// filled with the most common instance data needed for conversions.
// The information in this structure is exactly the same as it was
// during the ACMDM_STREAM_OPEN message--so it is not necessary
// to re-verify the information referenced by this structure.
//
// LPACMDRVSTREAMSIZE padss: Specifies a pointer to the ACMDRVSTREAMSIZE
// structure that defines the conversion stream size query attributes.
//
// Return (LRESULT):
// The return value is zero (MMSYSERR_NOERROR) if this function
// succeeds with no errors. The return value is a non-zero error code
// if the function fails.
//
// An ACM driver should return MMSYSERR_NOTSUPPORTED if a query type
// is requested that the driver does not understand. Note that a driver
// must support both the ACM_STREAMSIZEF_DESTINATION and
// ACM_STREAMSIZEF_SOURCE queries.
//
// If the conversion would be 'out of range' given the input arguments,
// then ACMERR_NOTPOSSIBLE should be returned.
//
//--------------------------------------------------------------------------;
LRESULT FNLOCAL acmdStreamSize
(
LPACMDRVSTREAMINSTANCE padsi,
LPACMDRVSTREAMSIZE padss
)
{
PSTREAMINSTANCE psi;
LPWAVEFORMATEX pwfxSrc;
LPWAVEFORMATEX pwfxDst;
LPMSG723WAVEFORMAT pwfg723;
DWORD cb;
DWORD cBlocks;
DWORD cbBytesPerBlock;
int i;
pwfxSrc = padsi->pwfxSrc;
pwfxDst = padsi->pwfxDst;
psi = (PSTREAMINSTANCE)(UINT)padsi->dwDriver;
//
//
//
//
//
switch (ACM_STREAMSIZEF_QUERYMASK & padss->fdwSize)
{
case ACM_STREAMSIZEF_SOURCE:
cb = padss->cbSrcLength;
if (WAVE_FORMAT_MSG723 == pwfxSrc->wFormatTag)
{
//
// how many destination PCM bytes are needed to hold
// the decoded g723 data of padss->cbSrcLength bytes
//
// always round UP
//
cBlocks = cb / pwfxSrc->nBlockAlign;
if (0 == cBlocks)
{
return (ACMERR_NOTPOSSIBLE);
}
pwfg723 = (LPMSG723WAVEFORMAT)pwfxSrc;
for(i=0;i<ACM_DRIVER_MAX_FORMATS_PCM;i++)
if (pwfxDst->nSamplesPerSec == PCM_SAMPLING_RATE[i])
{
cbBytesPerBlock = G723_SAMPLES_PER_BLOCK_PCM[i]
* pwfxDst->nBlockAlign;
break;
}
if (i == ACM_DRIVER_MAX_FORMATS_PCM)
return (ACMERR_NOTPOSSIBLE);
if ((0xFFFFFFFFL / cbBytesPerBlock) < cBlocks)
{
return (ACMERR_NOTPOSSIBLE);
}
if (0 == (cb % pwfxSrc->nBlockAlign))
{
cb = cBlocks * cbBytesPerBlock;
}
else
{
cb = (cBlocks + 1) * cbBytesPerBlock;
}
//
// This ensures that there is enough room to adjust
// from 11000 to 11025 Hz sampling rate when necessary
//
if (G723_SAMPLES_PER_BLOCK_PCM[i] == 330)
cb += (int)(1.0 + cBlocks / 440.0) * cbBytesPerBlock;
}
else
{
//
// how many destination g723 bytes are needed to hold
// the encoded PCM data of padss->cbSrcLength bytes
//
// always round UP
//
pwfg723 = (LPMSG723WAVEFORMAT)pwfxDst;
for(i=0;i<ACM_DRIVER_MAX_FORMATS_PCM;i++)
if (pwfxSrc->nSamplesPerSec == PCM_SAMPLING_RATE[i])
{
cbBytesPerBlock = G723_SAMPLES_PER_BLOCK_PCM[i]
* pwfxSrc->nBlockAlign;
break;
}
if (i == ACM_DRIVER_MAX_FORMATS_PCM)
return (ACMERR_NOTPOSSIBLE);
cBlocks = cb / cbBytesPerBlock;
if (0 == (cb % cbBytesPerBlock))
{
cb = cBlocks * pwfxDst->nBlockAlign;
}
else
{
cb = (cBlocks + 1) * pwfxDst->nBlockAlign;
}
if (0L == cb)
{
return (ACMERR_NOTPOSSIBLE);
}
}
padss->cbDstLength = cb;
return (MMSYSERR_NOERROR);
case ACM_STREAMSIZEF_DESTINATION:
cb = padss->cbDstLength;
if (WAVE_FORMAT_MSG723 == pwfxDst->wFormatTag)
{
//
// how many source PCM bytes can be encoded into a
// destination buffer of padss->cbDstLength bytes
//
// always round DOWN
//
cBlocks = cb / pwfxDst->nBlockAlign;
if (0 == cBlocks)
{
return (ACMERR_NOTPOSSIBLE);
}
pwfg723 = (LPMSG723WAVEFORMAT)pwfxDst;
for(i=0;i<ACM_DRIVER_MAX_FORMATS_PCM;i++)
if (pwfxSrc->nSamplesPerSec == PCM_SAMPLING_RATE[i])
{
cbBytesPerBlock = G723_SAMPLES_PER_BLOCK_PCM[i]
* pwfxSrc->nBlockAlign;
break;
}
if (i == ACM_DRIVER_MAX_FORMATS_PCM)
return (ACMERR_NOTPOSSIBLE);
if ((0xFFFFFFFFL / cbBytesPerBlock) < cBlocks)
{
return (ACMERR_NOTPOSSIBLE);
}
cb = cBlocks * cbBytesPerBlock;
}
else
{
//
// how many source g723 bytes can be decoded into a
// destination buffer of padss->cbDstLength bytes
//
// always round DOWN
//
pwfg723 = (LPMSG723WAVEFORMAT)pwfxSrc;
for(i=0;i<ACM_DRIVER_MAX_FORMATS_PCM;i++)
if (pwfxDst->nSamplesPerSec == PCM_SAMPLING_RATE[i])
{
cbBytesPerBlock = G723_SAMPLES_PER_BLOCK_PCM[i]
* pwfxDst->nBlockAlign;
break;
}
if (i == ACM_DRIVER_MAX_FORMATS_PCM)
return (ACMERR_NOTPOSSIBLE);
cBlocks = cb / cbBytesPerBlock;
//
// This ensures that there is enough room to adjust
// from 11000 to 11025 Hz sampling rate when necessary
//
if (G723_SAMPLES_PER_BLOCK_PCM[i] == 330)
cBlocks -= (int)(1.0 + cBlocks / 440.0);
if (0 == cBlocks)
{
return (ACMERR_NOTPOSSIBLE);
}
cb = cBlocks * pwfxSrc->nBlockAlign;
}
padss->cbSrcLength = cb;
return (MMSYSERR_NOERROR);
}
//
//
//
return (MMSYSERR_NOTSUPPORTED);
} // acmdStreamSize()
//--------------------------------------------------------------------------;
//
// LRESULT acmdStreamConvert
//
// Description:
// This function handles the ACMDM_STREAM_CONVERT message.
//
// Arguments:
// PDRIVERINSTANCE pdi: Pointer to private ACM driver instance structure.
// This structure is [optionally] allocated during the DRV_OPEN message
// which is handled by the acmdDriverOpen function.
//
// LPACMDRVSTREAMINSTANCE padsi: Pointer to instance data for the
// conversion stream. This structure was allocated by the ACM and
// filled with the most common instance data needed for conversions.
// The information in this structure is exactly the same as it was
// during the ACMDM_STREAM_OPEN message--so it is not necessary
// to re-verify the information referenced by this structure.
//
// LPACMDRVSTREAMHEADER padsh: Pointer to stream header structure
// that defines the source data and destination buffer to convert.
//
// Return (LRESULT):
// The return value is zero (MMSYSERR_NOERROR) if this function
// succeeds with no errors. The return value is a non-zero error code
// if the function fails.
//
//--------------------------------------------------------------------------;
LRESULT FNLOCAL acmdStreamConvert
(
PDRIVERINSTANCE pdi,
LPACMDRVSTREAMINSTANCE padsi,
LPACMDRVSTREAMHEADER padsh
)
{
LPWAVEFORMATEX pwfxSrc;
LPWAVEFORMATEX pwfxDst;
LPMSG723WAVEFORMAT pwfg723;
UINT psi;
CODDEF *CodStat;
DECDEF *DecStat;
G723CODDEF *g723Inst;
INSTNCE *SD_Inst;
char *Dst,*Src;
short *wDst;
short *wSrc;
float Dbuf[240];
short Ebuf[330];
short Sbuf[240];
int i,k,m,cBlocks,temp,src_length,frame_size;
int Dst_length,isFrameSilent,silence,tmpbuf[24];
int pcm_format;
//__asm int 3
psi = padsi->dwDriver;
pwfxSrc = padsi->pwfxSrc;
pwfxDst = padsi->pwfxDst;
if (WAVE_FORMAT_PCM == pwfxSrc->wFormatTag) {
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Stream Encode padsi=%lx psi=%lx",padsi,psi);
#endif // } NO_DEBUGGING_OUTPUT
#endif
//
// Encode.
//
g723Inst = (G723CODDEF *)psi;
CodStat = &g723Inst->CodStat;
SD_Inst = &g723Inst->SD_Instance;
pwfg723 = (LPMSG723WAVEFORMAT)padsi->pwfxDst;
if(padsh->cbSrcLength == 0)
return (MMSYSERR_NOERROR);
// *** compute # of frames that we can encode.***
for(pcm_format=0;pcm_format<ACM_DRIVER_MAX_FORMATS_PCM;pcm_format++)
if (pwfxSrc->nSamplesPerSec == PCM_SAMPLING_RATE[pcm_format])
break;
if (pcm_format == ACM_DRIVER_MAX_FORMATS_PCM)
return(MMSYSERR_INVALPARAM);
cBlocks = padsh->cbSrcLength
/ (pwfxSrc->nBlockAlign
* G723_SAMPLES_PER_BLOCK_PCM[pcm_format]);
// check to see if there are enough extra samples to remove
// every 441st sample (to go from 11025 to 11000 Hz)
if (G723_SAMPLES_PER_BLOCK_PCM[pcm_format] == 330)
{
i = padsh->cbSrcLength / pwfxSrc->nBlockAlign; // input samples
k = cBlocks
* G723_SAMPLES_PER_BLOCK_PCM[pcm_format]; // used samples
if (i - k < k / 441) // if there aren't enough extra samples
cBlocks -= (1+k/(441*330)); // then decrement the number of blocks
}
if(cBlocks == 0)
{
padsh->cbDstLengthUsed = 0;
padsh->cbSrcLengthUsed = 0;
return (MMSYSERR_NOERROR);
}
wSrc = (short *)padsh->pbSrc;
Dst = (char *)padsh->pbDst;
frame_size = (CodStat->WrkRate == Rate63) ? 24 : 20;
silence = 0;
for(i=0;i<cBlocks;i++)
{
if (G723_SAMPLES_PER_BLOCK_PCM[pcm_format] == 330)
{
for (k=0; k<330; k++)
{
Ebuf[k] = *wSrc++;
// *** code to adjust from 11025 Hz to 11000 Hz ***
if (++ CodStat->srccount == 441)
{
Ebuf[k] = *wSrc++; // skip a sample
CodStat->srccount = 0;
}
}
convert11to8(Ebuf,Sbuf,CodStat->srcstate,330);
for (k=0; k<240; k++)
SD_Inst->SDstate.Filt.sbuff[k] = (float) Sbuf[k];
}
else
for (k=0; k<240; k++)
SD_Inst->SDstate.Filt.sbuff[k] = (float) *wSrc++;
if(pwfg723->wConfigWord & SILENCE_ENABLE)
{
prefilter(SD_Inst,SD_Inst->SDstate.Filt.sbuff,
SD_Inst->SDstate.Filt.storebuff,240);
getParams(SD_Inst, SD_Inst->SDstate.Filt.storebuff,240);
execSDloop(SD_Inst,&isFrameSilent,SDThreashold);
}
else isFrameSilent = 0;
if(isFrameSilent)
{
*Dst++ = 0x02;
*Dst++ = 0;
*Dst++ = 0;
*Dst++ = 0;
padsh->cbDstLengthUsed += 4;
silence++;
}
else
{
Coder(SD_Inst->SDstate.Filt.sbuff,(Word32*)(long HUGE_T*)Dst,CodStat,0,1,0);
Dst += frame_size;
padsh->cbDstLengthUsed += frame_size;
}
}
padsh->cbSrcLengthUsed = pwfxSrc->nBlockAlign
* (wSrc - (short *)padsh->pbSrc);
return (MMSYSERR_NOERROR);
}
else
{
// Decode ---------------------------------------------
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Stream Decode padsi=%lx psi=%lx",padsi,psi);
#endif // } NO_DEBUGGING_OUTPUT
#endif
for(pcm_format=0;pcm_format<ACM_DRIVER_MAX_FORMATS_PCM;pcm_format++)
if (pwfxDst->nSamplesPerSec == PCM_SAMPLING_RATE[pcm_format])
break;
if (pcm_format == ACM_DRIVER_MAX_FORMATS_PCM)
return(MMSYSERR_INVALPARAM);
DecStat = (DECDEF *)psi;
cBlocks = 0;
wDst = (short *)padsh->pbDst;
Dst_length = padsh->cbDstLength;
Src = (char *)padsh->pbSrc;
src_length = padsh->cbSrcLength;
padsh->cbSrcLengthUsed = 0;
while(src_length > 0)
{
switch(*Src&3)
{
case 0 :
DecStat->WrkRate = Rate63;
frame_size = 24;
break;
case 1 :
DecStat->WrkRate = Rate53;
frame_size = 20;
break;
case 2 :
// DecStat->WrkRate = Silent; // Uncomment for V5.0 of G.723.1
frame_size = 4; // this is an SID frame
break;
case 3 :
DecStat->WrkRate = Lost;
frame_size = 4;
break;
}
src_length-=frame_size;
if(src_length < 0) break; // just in case we are given a partial frame
memcpy(tmpbuf,(long HUGE_T *)Src,frame_size);
Decod(Dbuf,(Word32*)(long HUGE_T*)tmpbuf,0,DecStat);
Src += frame_size;
padsh->cbSrcLengthUsed += frame_size;
cBlocks++;
if (G723_SAMPLES_PER_BLOCK_PCM[pcm_format] == 330)
{
//
// Ouput Sampling rate is 11025 Hz
//
if (440 - DecStat->srccount <= 240) // need to insert a sample?
{
DecStat->srcbuff[DecStat->i++] = FLOATTOSHORT(Dbuf[0]);
if (++ DecStat->srccount == 440) DecStat->srccount = 0;
DecStat->srcbuff[DecStat->i++] = FLOATTOSHORT(0.25 * Dbuf[0]
+ 0.75 * Dbuf[1]);
if (++ DecStat->srccount == 440) DecStat->srccount = 0;
DecStat->srcbuff[DecStat->i++] = FLOATTOSHORT(0.50 * Dbuf[1]
+ 0.50 * Dbuf[2]);
if (++ DecStat->srccount == 440) DecStat->srccount = 0;
DecStat->srcbuff[DecStat->i++] = FLOATTOSHORT(0.75 * Dbuf[2]
+ 0.25 * Dbuf[3]);
if (++ DecStat->srccount == 440) DecStat->srccount = 0;
m = 3;
}
else m = 0;
for (k=m; k<240; k++)
{
DecStat->srcbuff[DecStat->i++] = FLOATTOSHORT(Dbuf[k]);
if (++ DecStat->srccount == 440)
DecStat->srccount = 0;
}
if (DecStat->i == 480)
{
//
// *** 240 extra samples have accumulated ***
//
convert8to11(DecStat->srcbuff,wDst,DecStat->srcstate,480);
Dst_length -= 1320; // two output frames (1320 bytes)
wDst += 660;
DecStat->i = 0;
if(Dst_length < 1320) break; // no more space in output buffer
}
else
{
//
// *** less than 240 extra samples have accumulated ***
//
convert8to11(DecStat->srcbuff,wDst,DecStat->srcstate,240);
Dst_length -= 660; // one output frame (660 bytes)
wDst += 330;
//
// move partial frame to the front of the SRC buffer
//
for(i=0;i<DecStat->i-240;i++)
DecStat->srcbuff[i] = DecStat->srcbuff[i+240];
DecStat->i -=240;
if(Dst_length < 1320) break; // no more space in output buffer
}
}
else
{
//
// Ouput Sampling rate is 8000 Hz
//
for (k=0; k<240; k++)
{
temp = FLOATTOSHORT(Dbuf[k]);
*wDst++ = temp;
}
Dst_length -= 480; // we just chewed up 480 bytes;
if(Dst_length < 480) break; // we can't do any more!
}
}
padsh->cbDstLengthUsed = pwfxDst->nBlockAlign * cBlocks
* G723_SAMPLES_PER_BLOCK_PCM[pcm_format];
}
return (MMSYSERR_NOERROR);
} // acmdStreamConvert()
//==========================================================================;
//
//
//
//
//==========================================================================;
//--------------------------------------------------------------------------;
//
// LRESULT DriverProc
//
// Description:
//
//
// Arguments:
// DWORD dwId: For most messages, dwId is the DWORD value that
// the driver returns in response to a DRV_OPEN message. Each time
// the driver is opened, through the OpenDriver API, the driver
// receives a DRV_OPEN message and can return an arbitrary, non-zero
// value. The installable driver interface saves this value and returns
// a unique driver handle to the application. Whenever the application
// sends a message to the driver using the driver handle, the interface
// routes the message to this entry point and passes the corresponding
// dwId. This mechanism allows the driver to use the same or different
// identifiers for multiple opens but ensures that driver handles are
// unique at the application interface layer.
//
// The following messages are not related to a particular open instance
// of the driver. For these messages, the dwId will always be zero.
//
// DRV_LOAD, DRV_FREE, DRV_ENABLE, DRV_DISABLE, DRV_OPEN
//
// HDRVR hdrvr: This is the handle returned to the application
// by the driver interface.
//
// UINT uMsg: The requested action to be performed. Message
// values below DRV_RESERVED are used for globally defined messages.
// Message values from DRV_RESERVED to DRV_USER are used for defined
// driver protocols. Messages above DRV_USER are used for driver
// specific messages.
//
// LPARAM lParam1: Data for this message. Defined separately for
// each message.
//
// LPARAM lParam2: Data for this message. Defined separately for
// each message.
//
// Return (LRESULT):
// Defined separately for each message.
//
//--------------------------------------------------------------------------;
EXTERN_C LRESULT FNEXPORT DriverProc
(
DWORD dwId,
HDRVR hdrvr,
UINT uMsg,
LPARAM lParam1,
LPARAM lParam2
)
{
LRESULT lr;
PDRIVERINSTANCE pdi;
int k;
//
// make pdi either NULL or a valid instance pointer. note that dwId
// is 0 for several of the DRV_* messages (ie DRV_LOAD, DRV_OPEN...)
// see acmdDriverOpen for information on what dwId is for other
// messages (instance data).
//
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"DriverProc uMsg=%x lParam1=%lx lParam2=%lx",uMsg,lParam1,lParam2);
#endif // } NO_DEBUGGING_OUTPUT
#endif
pdi = (PDRIVERINSTANCE)(UINT)dwId;
switch (uMsg)
{
//
// lParam1: Unused.
//
// lParam2: Unused.
//
case DRV_LOAD:
#ifdef _WIN32
DbgInitialize(TRUE);
#endif
return(1L);
//
// lParam1: Unused.
//
// lParam2: Unused.
//
case DRV_FREE:
return (1L);
//
// lParam1: Not used. Ignore this argument.
//
// lParam2: Pointer to ACMDRVOPENDESC (or NULL).
//
case DRV_OPEN:
lr = acmdDriverOpen(hdrvr, (LPACMDRVOPENDESC)lParam2);
return (lr);
//
// lParam1: Unused.
//
// lParam2: Unused.
//
case DRV_CLOSE:
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Driver Closed");
#endif // } NO_DEBUGGING_OUTPUT
#endif
lr = acmdDriverClose(pdi);
return (lr);
//
// lParam1: Unused.
//
// lParam2: Unused.
//
case DRV_INSTALL:
return ((LRESULT)DRVCNF_RESTART);
//
// lParam1: Unused.
//
// lParam2: Unused.
//
case DRV_REMOVE:
return ((LRESULT)DRVCNF_RESTART);
//
// lParam1: Not used.
//
// lParam2: Not used.
//
case DRV_QUERYCONFIGURE:
//
// set up lParam1 and lParam2 to values that can be used by
// acmdDriverConfigure to know that it is being 'queried'
// for configuration support.
//
lParam1 = -1L;
lParam2 = 0L;
//--fall through--//
//
// lParam1: Handle to parent window for the configuration dialog
// box.
//
// lParam2: Optional pointer to DRVCONFIGINFO structure.
//
case DRV_CONFIGURE:
lr = acmdDriverConfigure(pdi, (HWND)lParam1, (LPDRVCONFIGINFO)lParam2);
return (lr);
//
// lParam1: Pointer to ACMDRIVERDETAILS structure.
//
// lParam2: Size in bytes of ACMDRIVERDETAILS stucture passed.
//
case ACMDM_DRIVER_DETAILS:
lr = acmdDriverDetails(pdi, (LPACMDRIVERDETAILS)lParam1);
return (lr);
//
// lParam1: Handle to parent window to use if displaying your own
// about box.
//
// lParam2: Not used.
//
case ACMDM_DRIVER_ABOUT:
lr = acmdDriverAbout(pdi, (HWND)lParam1);
return (lr);
//--------------------------------------------------------------------------;
//--------------------------------------------------------------------------;
//
// lParam1: Pointer to ACMDRVFORMATSUGGEST structure.
//
// lParam2: Not used.
//
case ACMDM_FORMAT_SUGGEST:
lr = acmdFormatSuggest(pdi, (LPACMDRVFORMATSUGGEST)lParam1);
return (lr);
//
// lParam1: Pointer to FORMATTAGDETAILS structure.
//
// lParam2: fdwDetails
//
case ACMDM_FORMATTAG_DETAILS:
lr = acmdFormatTagDetails(pdi, (LPACMFORMATTAGDETAILS)lParam1, lParam2);
return (lr);
//
// lParam1: Pointer to FORMATDETAILS structure.
//
// lParam2: fdwDetails
//
case ACMDM_FORMAT_DETAILS:
lr = acmdFormatDetails(pdi, (LPACMFORMATDETAILS)lParam1, lParam2);
return (lr);
//--------------------------------------------------------------------------;
//--------------------------------------------------------------------------;
//
// lParam1: Pointer to ACMDRVSTREAMINSTANCE structure.
//
// lParam2: Not used.
//
case ACMDM_STREAM_OPEN:
lr = acmdStreamOpen(pdi, (LPACMDRVSTREAMINSTANCE)lParam1);
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Stream open result: %d",lr);
#endif // } NO_DEBUGGING_OUTPUT
#endif
return (lr);
//
// lParam1: Pointer to ACMDRVSTREAMINSTANCE structure.
//
// lParam2: Not Used.
//
case ACMDM_STREAM_CLOSE:
lr = acmdStreamClose((LPACMDRVSTREAMINSTANCE)lParam1);
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Stream close result: %d",lr);
#endif // } NO_DEBUGGING_OUTPUT
#endif
return (lr);
//
// lParam1: Pointer to ACMDRVSTREAMINSTANCE structure.
//
// lParam2: Pointer to ACMDRVSTREAMSIZE structure.
//
case ACMDM_STREAM_SIZE:
lr = acmdStreamSize((LPACMDRVSTREAMINSTANCE)lParam1,
(LPACMDRVSTREAMSIZE)lParam2);
return (lr);
//
// lParam1: Pointer to ACMDRVSTREAMINSTANCE structure.
//
// lParam2: Pointer to ACMDRVSTREAMHEADER structure.
//
case ACMDM_STREAM_CONVERT:
if (pdi->enabled)
{
lr = acmdStreamConvert(pdi, (LPACMDRVSTREAMINSTANCE)lParam1,
(LPACMDRVSTREAMHEADER)lParam2);
return (lr);
}
else return(MMSYSERR_NOTSUPPORTED);
//
// lParam1: Setting for Silence detector threashold.
//
// lParam2: Not used. Ignore this argument.
//
case DRV_USER:
k = (int)SDThreashold;
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Old Threashold = %d",k);
#endif // } NO_DEBUGGING_OUTPUT
#endif
SDThreashold = (float)lParam1;
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Setting Threashold to %ld",lParam1);
#endif // } NO_DEBUGGING_OUTPUT
#endif
return (lParam1);
#ifndef NOTPRODUCT
//
// DRV_USER+1
//
// lParam1: Pointer to ACMDRVSTREAMINSTANCE structure.
//
// lParam2: 0 = PCM stream, 1 = G.723.1 stream
//
case DRV_USER+1:
if (lParam1 == G723MAGICWORD1 && lParam2 == G723MAGICWORD2)
pdi->enabled = TRUE;
else
pdi->enabled = FALSE;
return (MMSYSERR_NOERROR);
#endif // NOTPRODUCT
}
//
// if we are executing the following code, then this ACM driver does not
// handle the message that was sent. there are two ranges of messages
// we need to deal with:
//
// o ACM specific driver messages: if an ACM driver does not answer a
// message sent in the ACM driver message range, then it must
// return MMSYSERR_NOTSUPPORTED. this applies to the 'user'
// range as well (for consistency).
//
// o other installable driver messages: if an ACM driver does not
// answer a message that is NOT in the ACM driver message range,
// then it must call DefDriverProc and return that result.
// the exception to this is ACM driver procedures installed as
// ACM_DRIVERADDF_FUNCTION through acmDriverAdd. in this case,
// the driver procedure should conform to the ACMDRIVERPROC
// prototype and also return zero instead of calling DefDriverProc.
//
#ifdef DEBUG
#ifndef NO_DEBUGGING_OUTPUT // { NO_DEBUGGING_OUTPUT
TTDBG(ghISRInst,TT_TRACE,"Unknown command %d / %d",uMsg,DRV_USER);
#endif // } NO_DEBUGGING_OUTPUT
#endif
if (uMsg >= ACMDM_USER)
return (MMSYSERR_NOTSUPPORTED);
else
return (DefDriverProc(dwId, hdrvr, uMsg, lParam1, lParam2));
} // DriverProc()