windows-nt/Source/XPSP1/NT/multimedia/directx/dplay/dvoice/dxvutils/dscrecd.cpp
2020-09-26 16:20:57 +08:00

284 lines
7.9 KiB
C++

/*==========================================================================
*
* Copyright (C) 1999 Microsoft Corporation. All Rights Reserved.
*
* File: dscrecd.cpp
* Content:
* This file contains the DirectSoundCapture implementation of the
* CAudioRecordDevice abstraction.
*
* History:
* Date By Reason
* ==== == ======
* 11/04/99 rodtoll Created
* 11/12/99 rodtoll Modified abstraction for new waveIN support.
* Now abstracted types look almost like dsoundcap objects
* 11/22/99 rodtoll Added code to map from GUID to waveIN device
* ID for non-millenium systems.
* 11/23/99 rodtoll Updated to use waveIn device ID or DSound 7.1 when they are avail
* 12/01/99 rodtoll Bug #115783 - Will always adjust volume of default device
* Now uses new CMixerLine class for adjusting volumes/selecting mic
* rodtoll New algorithm to map from GUIDs to device IDs if DSound 7.1 is not
* available. Will map device correctly on DX7, will guess for other
* DX versions < 7. However, default device is assumed to be waveIN ID #0.
* 12/08/99 rodtoll Bug #121054 - DirectX 7.1 support.
* - Added hwndOwner param for capture focus support
* 04/21/2000 rodtoll Bug #32952 - Does not run on Win95 GOLD w/o IE4 -- modified
* to allow reads of REG_BINARY when expecting REG_DWORD
* 06/09/00 rmt Updates to split CLSID and allow whistler compat and support external create funcs
* 06/28/2000 rodtoll Prefix Bug #38022
* 08/03/2000 rodtoll Bug #41457 - DPVOICE: need way to discover which specific dsound call failed when returning DVERR_SOUNDINITFAILURE
* 08/28/2000 masonb Voice Merge: Changed ccomutil.h to comutil.h
* 09/13/2000 rodtoll Bug #44806 - When volume control not avail, dropping to DX7 levels instead of disabling volume control
*
***************************************************************************/
#include "dxvutilspch.h"
#undef DPF_SUBCOMP
#define DPF_SUBCOMP DN_SUBCOMP_VOICE
// This function is responsible for mapping from the Device's GUID to the
// waveIN ID.
//
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordDevice::FindDeviceID"
HRESULT CDirectSoundCaptureRecordDevice::FindDeviceID()
{
HRESULT hr;
DWORD dwDeviceID = 0;
hr = DV_MapGUIDToWaveID( TRUE, m_guidDevice, &dwDeviceID );
// If we were going to use the hack for enum of devices
if( FAILED( hr ) )
{
DPFX(DPFPREP, DVF_ERRORLEVEL, "Unable to find waveIN ID, mapping to ID 0 hr=0x%x", hr );
m_uiWaveDeviceID = 0;
}
else
{
m_uiWaveDeviceID = dwDeviceID;
}
return hr;
}
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordDevice::CDirectSoundCaptureRecordDevice"
CDirectSoundCaptureRecordDevice::CDirectSoundCaptureRecordDevice(
): CAudioRecordDevice(), m_lpdscDirectSound(NULL), m_uiWaveDeviceID(0), m_guidDevice(GUID_NULL)
{
}
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordDevice::CDirectSoundCaptureRecordDevice"
CDirectSoundCaptureRecordDevice::~CDirectSoundCaptureRecordDevice()
{
if( m_lpdscDirectSound != NULL )
{
m_lpdscDirectSound->Release();
}
}
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordDevice::Initialize"
HRESULT CDirectSoundCaptureRecordDevice::Initialize( LPDIRECTSOUNDCAPTURE lpdscDirectSound, const GUID &guidDevice )
{
HRESULT hr;
if( m_lpdscDirectSound != NULL )
{
DPFX(DPFPREP, DVF_ERRORLEVEL, "Already initialized" );
return DVERR_INITIALIZED;
}
hr = lpdscDirectSound->QueryInterface( IID_IDirectSoundCapture, (void **) &m_lpdscDirectSound );
if( FAILED( hr ) )
{
DPFX(DPFPREP, DVF_ERRORLEVEL, "DirectSoundCapture Object passed failed. 0x%x Creating internal", hr );
m_lpdscDirectSound = NULL;
return hr;
}
m_guidDevice = guidDevice;
hr = FindDeviceID();
if( FAILED( hr ) )
{
DPFX(DPFPREP, DVF_ERRORLEVEL, "Unable to find waveIn ID for device hr=0x%x", hr );
m_lpdscDirectSound->Release();
return hr;
}
return hr;
}
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordDevice::Initialize"
HRESULT CDirectSoundCaptureRecordDevice::Initialize( const GUID &guidDevice )
{
HRESULT hr;
if( m_lpdscDirectSound != NULL )
{
DPFX(DPFPREP, DVF_ERRORLEVEL, "Already initialized" );
return DVERR_INITIALIZED;
}
hr = COM_CoCreateInstance( CLSID_DirectSoundCapture, NULL, CLSCTX_INPROC_SERVER , IID_IDirectSoundCapture, (void **) &m_lpdscDirectSound );
DSERTRACK_Update( "DSCD:CoCreateInstance()", hr );
if( FAILED( hr ) )
{
DPFX(DPFPREP, DVF_ERRORLEVEL, "Unable to load directsoundcapture hr=0x%x", hr );
goto INITIALIZE_ERROR;
}
hr = m_lpdscDirectSound->Initialize( &guidDevice );
DSERTRACK_Update( "DSCD:Initialize()", hr );
if( FAILED( hr ) )
{
DPFX(DPFPREP, DVF_ERRORLEVEL, "Unable to initialize directsoundcapture hr=0x%x", hr );
goto INITIALIZE_ERROR;
}
m_guidDevice = guidDevice;
hr = FindDeviceID();
if( FAILED( hr ) )
{
DPFX(DPFPREP, DVF_ERRORLEVEL, "Unable to find waveIn ID for device hr=0x%x", hr );
return hr;
}
return DV_OK;
INITIALIZE_ERROR:
if( m_lpdscDirectSound != NULL )
{
m_lpdscDirectSound->Release();
m_lpdscDirectSound = NULL;
}
return hr;
}
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordDevice::CreateBuffer"
HRESULT CDirectSoundCaptureRecordDevice::CreateBuffer( LPDSCBUFFERDESC lpdsBufferDesc, HWND hwndOwner, DWORD dwFrameSize, CAudioRecordBuffer **lplpacBuffer )
{
HRESULT hr;
LPDIRECTSOUNDCAPTUREBUFFER lpdscBuffer;
lpdsBufferDesc->dwFlags |= DSCBCAPS_CTRLVOLUME;
hr = m_lpdscDirectSound->CreateCaptureBuffer( lpdsBufferDesc, &lpdscBuffer, NULL );
DSERTRACK_Update( "DSCD::CreateCaptureBuffer()", hr );
// Ask for volume control, if we can't get it, do the old create
if( hr == DSERR_INVALIDPARAM || hr == DSERR_CONTROLUNAVAIL )
{
DPFX(DPFPREP, DVF_ERRORLEVEL, "New caps are not available, attempting old create hr=0x%x", hr );
// Turn off the new caps -- (for non-Millenium systems).
lpdsBufferDesc->dwFlags &= ~(DSCBCAPS_CTRLVOLUME);
hr = m_lpdscDirectSound->CreateCaptureBuffer( lpdsBufferDesc, &lpdscBuffer, NULL );
}
if( FAILED( hr ) )
{
DPFX(DPFPREP, DVF_ERRORLEVEL, "Failed to create the capture buffer hr=0x%x", hr );
return hr;
}
*lplpacBuffer = new CDirectSoundCaptureRecordBuffer( lpdscBuffer, hwndOwner, m_guidDevice, m_uiWaveDeviceID, lpdsBufferDesc );
lpdscBuffer->Release();
if( *lplpacBuffer == NULL )
{
DPFX(DPFPREP, DVF_ERRORLEVEL, "Out of memory" );
return DVERR_OUTOFMEMORY;
}
return DV_OK;
}
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordDevice::GetCaptureDevice"
LPDIRECTSOUNDCAPTURE CDirectSoundCaptureRecordDevice::GetCaptureDevice()
{
return m_lpdscDirectSound;
}
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordDevice::GetMixerQuality"
HRESULT CDirectSoundCaptureRecordDevice::GetMixerQuality( DIRECTSOUNDMIXER_SRCQUALITY *psrcQuality )
{
HRESULT hr;
LPKSPROPERTYSET pPropertySet = NULL;
hr = DirectSoundPrivateCreate( &pPropertySet );
if( FAILED( hr ) )
{
DPFX(DPFPREP, DVF_WARNINGLEVEL, "Unable to get int to get mixer quality hr=0x%x", hr );
return hr;
}
hr = PrvGetMixerSrcQuality( pPropertySet, m_guidDevice, psrcQuality );
if( FAILED( hr ) )
{
DPFX(DPFPREP, DVF_WARNINGLEVEL, "Unable to retrieve mixer quality hr=0x%x", hr );
}
pPropertySet->Release();
return hr;
}
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordDevice::SetMixerQuality"
HRESULT CDirectSoundCaptureRecordDevice::SetMixerQuality( const DIRECTSOUNDMIXER_SRCQUALITY srcQuality )
{
HRESULT hr;
LPKSPROPERTYSET pPropertySet = NULL;
hr = DirectSoundPrivateCreate( &pPropertySet );
if( FAILED( hr ) )
{
DPFX(DPFPREP, DVF_WARNINGLEVEL, "Unable to get int to set mixer quality hr=0x%x", hr );
return hr;
}
hr = PrvSetMixerSrcQuality( pPropertySet, m_guidDevice, srcQuality );
if( FAILED( hr ) )
{
DPFX(DPFPREP, DVF_WARNINGLEVEL, "Unable to set mixer quality hr=0x%x", hr );
}
pPropertySet->Release();
return hr;
}