|
|
/*==========================================================================
* * Copyright (C) 1999 Microsoft Corporation. All Rights Reserved. * * File: dscrecb.cpp * Content: * This module contains the implementation of the * CDirectSoundCaptureRecordBuffer. * * History: * Date By Reason * ==== == ====== * 11/04/99 rodtoll Created * 11/22/99 rodtoll Added code to allow specification of wave device ID * 11/23/99 rodtoll Updated to use waveIn device ID or DSound 7.1 when they are avail * rodtoll Updated with new Microphone select member * 12/08/99 rodtoll Bug #115783 - Will always adjust volume of default device * Now uses new CMixerLine class for adjusting volumes/selecting mic * 12/08/99 rodtoll Bug #121054 - DirectX 7.1 support. * - Added hwndOwner param for capture focus support * - Added lpfLostFocus param to GetCurrentPosition so upper * layers can detect lost focus. * 01/28/2000 rodtoll Bug #130465: Record Mute/Unmute must call YieldFocus() / ClaimFocus() * 02/10/2000 rodtoll Removed more capture focus code * 04/19/2000 rodtoll Re-enabled capture focus behaviour * 08/03/2000 rodtoll Bug #41457 - DPVOICE: need way to discover which specific dsound call failed when returning DVERR_SOUNDINITFAILURE * 04/22/2001 rodtoll Fixed bug in lockup simulation code (TESTING PURPOSES ONLY) * ***************************************************************************/
#include "dxvutilspch.h"
#undef DPF_SUBCOMP
#define DPF_SUBCOMP DN_SUBCOMP_VOICE
#define DSC_STARTUP_LATENCY 1
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordBuffer::CDirectSoundCaptureRecordBuffer"
CDirectSoundCaptureRecordBuffer::CDirectSoundCaptureRecordBuffer( LPDIRECTSOUNDCAPTUREBUFFER lpdscBuffer, HWND hwndOwner, const GUID &guidDevice, UINT uiWaveDeviceID, LPDSCBUFFERDESC lpdsBufferDesc ): CAudioRecordBuffer(), m_lpwfxRecordFormat(NULL), m_lpdscBuffer7(NULL), m_guidDevice(guidDevice), m_uiWaveDeviceID(uiWaveDeviceID), #ifdef LOCKUP_SIMULATION
m_dwNumSinceLastLockup(0), m_dwLastPosition(0), #endif
m_hwndOwner(hwndOwner), m_fUseCaptureFocus( FALSE ) { HRESULT hr; DWORD dwSize;
if( lpdsBufferDesc->dwFlags & DSCBCAPS_FOCUSAWARE ) { DPFX(DPFPREP, 1, "Enabling focus" ); m_fUseCaptureFocus = TRUE; } else { DPFX(DPFPREP, 1, "Disabling focus" ); m_fUseCaptureFocus = FALSE; } hr = lpdscBuffer->QueryInterface( IID_IDirectSoundCaptureBuffer, (void **) &m_lpdscBuffer );
if( FAILED( hr ) ) { DPFX(DPFPREP, DVF_ERRORLEVEL, "Unable to get dsound buffer interface" ); m_lpdscBuffer = NULL; }
hr = lpdscBuffer->QueryInterface( IID_IDirectSoundCaptureBuffer7_1, (void **) &m_lpdscBuffer7 );
if( FAILED( hr ) ) { DPFX(DPFPREP, DVF_WARNINGLEVEL, "Could not retrieve new interface hr=0x%x.", hr );
hr = m_mixerLine.Initialize( m_uiWaveDeviceID );
if( FAILED( hr ) ) { DPFX(DPFPREP, DVF_ERRORLEVEL, "Unable to acquire volume controls hr=0x%x", hr ); DNASSERT( FALSE ); } } else { if( m_fUseCaptureFocus ) { hr = m_lpdscBuffer7->SetFocusHWND( hwndOwner );
if( FAILED( hr ) ) { DPFX(DPFPREP, DVF_ERRORLEVEL, "Unable to set the focus window hr = 0x%x", hr ); DNASSERT( FALSE ); } } }
hr = lpdscBuffer->GetFormat( NULL, 0, &dwSize );
m_lpwfxRecordFormat = (LPWAVEFORMATEX) new BYTE[dwSize];
if( m_lpwfxRecordFormat == NULL ) { DPFX(DPFPREP, DVF_ERRORLEVEL, "Unable to allocate space for record format" ); DNASSERT( FALSE ); }
hr = lpdscBuffer->GetFormat( m_lpwfxRecordFormat, dwSize, NULL );
if( FAILED( hr ) ) { delete [] ((LPBYTE) m_lpwfxRecordFormat); m_lpwfxRecordFormat = NULL; }
}
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordBuffer::CDirectSoundCaptureRecordBuffer"
CDirectSoundCaptureRecordBuffer::~CDirectSoundCaptureRecordBuffer() { if( m_lpdscBuffer7 != NULL ) { m_lpdscBuffer7->Release(); } if( m_lpdscBuffer != NULL ) { m_lpdscBuffer->Release(); }
if( m_lpwfxRecordFormat != NULL ) { delete [] ((LPBYTE) m_lpwfxRecordFormat); } }
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordBuffer::Lock"
HRESULT CDirectSoundCaptureRecordBuffer::Lock( DWORD dwWriteCursor, DWORD dwWriteBytes, LPVOID *lplpvBuffer1, LPDWORD lpdwSize1, LPVOID *lplpvBuffer2, LPDWORD lpdwSize2, DWORD dwFlags ) { HRESULT hr; if( m_lpdscBuffer == NULL ) { DPFX(DPFPREP, DVF_ERRORLEVEL, "No DirectSoundCapture Buffer Available" ); return DVERR_NOTINITIALIZED; }
hr = m_lpdscBuffer->Lock( dwWriteCursor, dwWriteBytes, lplpvBuffer1, lpdwSize1, lplpvBuffer2, lpdwSize2, dwFlags );
DSERTRACK_Update( "DSCB::Lock()", hr );
return hr; }
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordBuffer::UnLock"
HRESULT CDirectSoundCaptureRecordBuffer::UnLock( LPVOID lpvBuffer1, DWORD dwSize1, LPVOID lpvBuffer2, DWORD dwSize2 ) { HRESULT hr; if( m_lpdscBuffer == NULL ) { DPFX(DPFPREP, DVF_ERRORLEVEL, "No DirectSoundCapture Buffer Available" ); return DVERR_NOTINITIALIZED; }
hr = m_lpdscBuffer->Unlock( lpvBuffer1, dwSize1, lpvBuffer2, dwSize2 );
return hr; }
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordBuffer::SetVolume"
HRESULT CDirectSoundCaptureRecordBuffer::SetVolume( LONG lVolume ) { if( m_lpdscBuffer == NULL ) { DPFX(DPFPREP, DVF_ERRORLEVEL, "No DirectSoundCapture Buffer Available" ); return DVERR_NOTINITIALIZED; }
HRESULT hr;
if( m_lpdscBuffer7 != NULL ) { hr = m_lpdscBuffer7->SetVolume( lVolume ); } else { hr = m_mixerLine.SetMasterRecordVolume( lVolume ); }
// This is only a warning condition
if( FAILED( hr ) ) { DPFX(DPFPREP, DVF_INFOLEVEL, "Could not set master volume hr=0x%x", hr ); }
if( m_lpdscBuffer7 != NULL ) { hr = m_lpdscBuffer7->SetMicVolume( lVolume ); } else { hr = m_mixerLine.SetMicrophoneVolume( lVolume ); }
if( FAILED( hr ) ) { DPFX(DPFPREP, DVF_INFOLEVEL, "Unable to set Microphone volume hr=0x%x", hr ); return hr; } return DV_OK; }
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordBuffer::GetVolume"
HRESULT CDirectSoundCaptureRecordBuffer::GetVolume( LPLONG lplVolume ) { if( m_lpdscBuffer == NULL ) { DPFX(DPFPREP, DVF_ERRORLEVEL, "No DirectSoundCapture Buffer Available" ); return DVERR_NOTINITIALIZED; }
HRESULT hr; if( m_lpdscBuffer7 != NULL ) { hr = m_lpdscBuffer7->GetMicVolume( lplVolume ); } else { hr = m_mixerLine.GetMicrophoneVolume( lplVolume ); }
if( FAILED( hr ) ) { DPFX(DPFPREP, DVF_WARNINGLEVEL, "Unable to get Microphone volume hr=0x%x", hr );
if( m_lpdscBuffer7 != NULL ) { hr = m_lpdscBuffer7->GetVolume( lplVolume ); } else { hr = m_mixerLine.GetMasterRecordVolume( lplVolume ); }
// This is only a warning condition
if( FAILED( hr ) ) { DPFX(DPFPREP, DVF_WARNINGLEVEL, "Could not get master record volume hr=0x%x", hr ); return hr; }
return hr; }
return hr; }
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordBuffer::GetCurrentPosition"
HRESULT CDirectSoundCaptureRecordBuffer::GetCurrentPosition( LPDWORD lpdwPosition, LPBOOL lpfLostFocus ) { if( m_lpdscBuffer == NULL ) { DPFX(DPFPREP, DVF_ERRORLEVEL, "No DirectSoundCapture Buffer Available" ); return DVERR_NOTINITIALIZED; }
#ifdef LOCKUP_SIMULATION
m_dwNumSinceLastLockup++;
if( m_dwNumSinceLastLockup > LOCKUP_NUM_CALLS_BEFORE_LOCKUP ) { *lpdwPosition = m_dwLastPosition; *lpfLostFocus = FALSE; return DV_OK; } #endif
HRESULT hr;
if( m_lpdscBuffer7 != NULL && m_fUseCaptureFocus ) { DWORD dwStatus; hr = m_lpdscBuffer7->GetStatus( &dwStatus );
if( FAILED( hr ) ) { DPFX(DPFPREP, DVF_ERRORLEVEL, "Error getting status hr=0x%x", hr ); *lpfLostFocus = FALSE; } else { *lpfLostFocus = (dwStatus & DSCBSTATUS_LOSTFOCUS) ? TRUE : FALSE; } } else { *lpfLostFocus = FALSE; }
hr = m_lpdscBuffer->GetCurrentPosition( NULL, lpdwPosition );
#ifdef LOCKUP_SIMULATION
m_dwLastPosition = *lpdwPosition; #endif
DSERTRACK_Update( "DSCB::GetCurrentPosition()", hr );
return hr;
}
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordBuffer::Play"
HRESULT CDirectSoundCaptureRecordBuffer::Record( BOOL fLooping ) { if( m_lpdscBuffer == NULL ) { DPFX(DPFPREP, DVF_ERRORLEVEL, "No DirectSound Buffer Available" ); return DVERR_NOTINITIALIZED; }
HRESULT hr;
hr = m_lpdscBuffer->Start( (fLooping) ? DSCBSTART_LOOPING : 0 );
#ifdef LOCKUP_SIMULATION
#ifdef LOCKUP_STOPFAIL
if( m_dwNumSinceLastLockup > LOCKUP_NUM_CALLS_BEFORE_LOCKUP ) { hr = DSERR_GENERIC; } #else
m_dwNumSinceLastLockup = 0; #endif
#endif
DSERTRACK_Update( "DSCB::Start()", hr );
return hr; }
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordBuffer::Stop"
HRESULT CDirectSoundCaptureRecordBuffer::Stop() { if( m_lpdscBuffer == NULL ) { DPFX(DPFPREP, DVF_ERRORLEVEL, "No DirectSound Buffer Available" ); return DVERR_NOTINITIALIZED; }
HRESULT hr;
hr = m_lpdscBuffer->Stop( );
#ifdef LOCKUP_SIMULATION
#ifdef LOCKUP_STOPFAIL
if( m_dwNumSinceLastLockup > LOCKUP_NUM_CALLS_BEFORE_LOCKUP ) { hr = DSERR_GENERIC; } #endif
#endif
DSERTRACK_Update( "DSCB::Stop()", hr );
return hr; }
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordBuffer::GetStartupLatency"
DWORD CDirectSoundCaptureRecordBuffer::GetStartupLatency() { return DSC_STARTUP_LATENCY; }
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordBuffer::GetRecordFormat"
LPWAVEFORMATEX CDirectSoundCaptureRecordBuffer::GetRecordFormat() { return m_lpwfxRecordFormat; }
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordBuffer::SelectMicrophone"
HRESULT CDirectSoundCaptureRecordBuffer::SelectMicrophone( BOOL fSelect ) { if( m_lpdscBuffer7 != NULL ) { return m_lpdscBuffer7->EnableMic( fSelect ); } else { return m_mixerLine.EnableMicrophone( fSelect ); } }
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordBuffer::ClaimFocus"
HRESULT CDirectSoundCaptureRecordBuffer::ClaimFocus( ) { if( m_lpdscBuffer7 != NULL && m_fUseCaptureFocus) { return m_lpdscBuffer7->ClaimFocus(); } else { return DVERR_NOTSUPPORTED; } }
#undef DPF_MODNAME
#define DPF_MODNAME "CDirectSoundCaptureRecordBuffer::YieldFocus"
HRESULT CDirectSoundCaptureRecordBuffer::YieldFocus( ) { if( m_lpdscBuffer7 != NULL && m_fUseCaptureFocus ) { return m_lpdscBuffer7->YieldFocus(); } else { return DVERR_NOTSUPPORTED; } }
|