/*========================================================================== * * 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; }