Windows2003-3790/multimedia/directx/dxdiag/sndinfo.cpp
2020-09-30 16:53:55 +02:00

745 lines
27 KiB
C++

/****************************************************************************
*
* File: sndinfo.cpp
* Project: DxDiag (DirectX Diagnostic Tool)
* Author: Mike Anderson (manders@microsoft.com)
* Purpose: Gather information about sound devices on this machine
*
* (C) Copyright 1998-1999 Microsoft Corp. All rights reserved.
*
****************************************************************************/
#define DIRECTSOUND_VERSION 0x0600
#include <tchar.h>
#include <Windows.h>
#include <mmsystem.h>
#include <d3dtypes.h>
#include <dsound.h>
#include <stdio.h>
#include "mmddk.h" // for DRV_QUERYDEVNODE
#include "dsprv.h"
#include "dsprvobj.h"
#include "reginfo.h"
#include "sysinfo.h" // for BIsPlatformNT
#include "dispinfo.h"
#include "sndinfo.h"
#include "fileinfo.h" // for GetFileVersion, FileIsSigned
#include "resource.h"
// This function is defined in sndinfo7.cpp:
HRESULT GetRegKey(LPKSPROPERTYSET pKSPS7, REFGUID guidDeviceID, TCHAR* pszRegKey);
typedef HRESULT (WINAPI* LPDIRECTSOUNDENUMERATE)(LPDSENUMCALLBACK lpDSEnumCallback,
LPVOID lpContext);
typedef HRESULT (WINAPI* LPDIRECTSOUNDCREATE)(LPGUID lpGUID, LPDIRECTSOUND* ppDS,
LPUNKNOWN pUnkOuter);
static BOOL CALLBACK DSEnumCallback(LPGUID pGuid, TCHAR* pszDescription,
TCHAR* pszModule, LPVOID lpContext);
static VOID GetRegSoundInfo9x(SoundInfo* pSoundInfo);
static VOID GetRegSoundInfoNT(SoundInfo* pSoundInfo);
static HRESULT GetDirectSoundInfo(LPDIRECTSOUNDCREATE pDSCreate, SoundInfo* pSoundInfo);
static HRESULT CheckRegistry(RegError** ppRegErrorFirst);
static LPKSPROPERTYSET s_pKSPS = NULL;
static DWORD s_dwWaveIDDefault = 0;
/****************************************************************************
*
* GetBasicSoundInfo
*
****************************************************************************/
HRESULT GetBasicSoundInfo(SoundInfo** ppSoundInfoFirst)
{
HRESULT hr = S_OK;
TCHAR szPath[MAX_PATH];
HINSTANCE hInstDSound = NULL;
LPDIRECTSOUNDENUMERATE pdse;
lstrcpy( szPath, TEXT("") );
// Find which waveout device is the default, the one that would
// be used by DirectSoundCreate(NULL). If the following code
// fails, assume it's device 0.
DWORD dwParam2 = 0;
waveOutMessage( (HWAVEOUT)IntToPtr(WAVE_MAPPER), DRVM_MAPPER_PREFERRED_GET, (DWORD_PTR) &s_dwWaveIDDefault, (DWORD_PTR) &dwParam2 );
if( s_dwWaveIDDefault == -1 )
s_dwWaveIDDefault = 0;
GetSystemDirectory(szPath, MAX_PATH);
lstrcat(szPath, TEXT("\\dsound.dll"));
hInstDSound = LoadLibrary(szPath);
if (hInstDSound == NULL)
goto LEnd;
// Get Private DirectSound object:
if (FAILED(hr = DirectSoundPrivateCreate(&s_pKSPS)))
{
// note: no error. This will always fail on Win95.
}
// Get DirectSoundEnumerate and call it:
pdse = (LPDIRECTSOUNDENUMERATE)GetProcAddress(hInstDSound,
#ifdef UNICODE
"DirectSoundEnumerateW"
#else
"DirectSoundEnumerateA"
#endif
);
if (pdse == NULL)
goto LEnd;
if (FAILED(hr = pdse((LPDSENUMCALLBACK)DSEnumCallback, ppSoundInfoFirst)))
goto LEnd;
LEnd:
if (s_pKSPS != NULL)
{
s_pKSPS->Release();
s_pKSPS = NULL;
}
if (hInstDSound != NULL)
FreeLibrary(hInstDSound);
return hr;
}
/****************************************************************************
*
* DSEnumCallback
*
****************************************************************************/
BOOL CALLBACK DSEnumCallback(LPGUID pGuid, TCHAR* pszDescription,
TCHAR* pszModule, LPVOID lpContext)
{
SoundInfo** ppSoundInfoFirst = (SoundInfo**)lpContext;
SoundInfo* pSoundInfoNew;
PDSPROPERTY_DIRECTSOUNDDEVICE_DESCRIPTION_DATA pdsdData = NULL;
if (pGuid == NULL)
return TRUE; // skip this guy
// We decided to only report on the default, primary DSound device for
// now. Remove the next few lines to show a page per device.
// Use private interface to get bonus information:
if (*ppSoundInfoFirst != NULL)
return FALSE; // We got our device, so stop enumerating
if (s_pKSPS != NULL)
{
if (FAILED(PrvGetDeviceDescription(s_pKSPS, *pGuid, &pdsdData)))
return TRUE; // error on this device--keep enumerating
if (pdsdData->WaveDeviceId != s_dwWaveIDDefault)
return TRUE; // not default device--keep enumerating
}
// Note: on Win95, where s_pKSPS is NULL, we can't get the device ID,
// so we don't know which DSound device is the default one. So just
// use the first one that comes up in the enumeration after the one
// with NULL pGuid.
pSoundInfoNew = new SoundInfo;
if (pSoundInfoNew == NULL)
return FALSE;
ZeroMemory(pSoundInfoNew, sizeof(SoundInfo));
if (*ppSoundInfoFirst == NULL)
{
*ppSoundInfoFirst = pSoundInfoNew;
}
else
{
SoundInfo* pSoundInfo;
for (pSoundInfo = *ppSoundInfoFirst;
pSoundInfo->m_pSoundInfoNext != NULL;
pSoundInfo = pSoundInfo->m_pSoundInfoNext)
{
}
pSoundInfo->m_pSoundInfoNext = pSoundInfoNew;
}
pSoundInfoNew->m_guid = *pGuid;
_tcsncpy(pSoundInfoNew->m_szDescription, pszDescription, 200);
pSoundInfoNew->m_szDescription[199] = 0;
if (s_pKSPS == NULL)
{
// Without the DSound private interface, we can't use it to get
// waveout device ID or devnode. Assume device ID is 0, and use
// waveOutMessage(DRV_QUERYDEVNODE) to get dev node.
waveOutMessage((HWAVEOUT)0, DRV_QUERYDEVNODE, (DWORD_PTR)&pSoundInfoNew->m_dwDevnode, 0);
pSoundInfoNew->m_lwAccelerationLevel = -1;
}
else
{
pSoundInfoNew->m_dwDevnode = pdsdData->Devnode;
if (pdsdData->Type == 0)
LoadString(NULL, IDS_EMULATED, pSoundInfoNew->m_szType, 100);
else if (pdsdData->Type == 1)
LoadString(NULL, IDS_VXD, pSoundInfoNew->m_szType, 100);
else if (pdsdData->Type == 2)
LoadString(NULL, IDS_WDM, pSoundInfoNew->m_szType, 100);
DIRECTSOUNDBASICACCELERATION_LEVEL accelLevel;
if (FAILED(PrvGetBasicAcceleration(s_pKSPS, *pGuid, &accelLevel)))
pSoundInfoNew->m_lwAccelerationLevel = -1;
else
pSoundInfoNew->m_lwAccelerationLevel = (LONG)accelLevel;
// This will only work on DX7 and beyond
GetRegKey(s_pKSPS, *pGuid, pSoundInfoNew->m_szRegKey);
}
WAVEOUTCAPS waveoutcaps;
LONG devID;
if (pdsdData == NULL)
devID = 0;
else
devID = pdsdData->WaveDeviceId;
if (MMSYSERR_NOERROR == waveOutGetDevCaps(devID, &waveoutcaps, sizeof(waveoutcaps)))
{
// May want to use mmreg.h to add strings for manufacturer/product names here
wsprintf(pSoundInfoNew->m_szManufacturerID, TEXT("%d"), waveoutcaps.wMid);
wsprintf(pSoundInfoNew->m_szProductID, TEXT("%d"), waveoutcaps.wPid);
}
// Sometimes, pszModule is the full path. Sometimes it's just the leaf.
// Sometimes, it's something inbetween. Separate the leaf, and look
// in a few different places.
TCHAR* pszLeaf;
pszLeaf = _tcsrchr(pszModule, TEXT('\\'));
if (pszLeaf == NULL)
{
lstrcpy(pSoundInfoNew->m_szDriverName, pszModule);
}
else
{
lstrcpy(pSoundInfoNew->m_szDriverName, (pszLeaf + 1));
}
// Try just module string
int nLenMod = lstrlen(pszModule);
_tcsncpy(pSoundInfoNew->m_szDriverPath, pszModule, 500);
pSoundInfoNew->m_szDriverPath[499]=0;
if (pszLeaf == NULL || GetFileAttributes(pSoundInfoNew->m_szDriverPath) == 0xFFFFFFFF)
{
// Try windows dir + module string
if( GetWindowsDirectory(pSoundInfoNew->m_szDriverPath, MAX_PATH) != 0 )
{
lstrcat(pSoundInfoNew->m_szDriverPath, TEXT("\\"));
if( lstrlen(pSoundInfoNew->m_szDriverPath) + nLenMod < 500 )
lstrcat(pSoundInfoNew->m_szDriverPath, pszModule);
if (GetFileAttributes(pSoundInfoNew->m_szDriverPath) == 0xFFFFFFFF)
{
// Try system dir + module string
if( GetSystemDirectory(pSoundInfoNew->m_szDriverPath, MAX_PATH) != 0 )
{
lstrcat(pSoundInfoNew->m_szDriverPath, TEXT("\\"));
if( lstrlen(pSoundInfoNew->m_szDriverPath) + nLenMod < 500 )
lstrcat(pSoundInfoNew->m_szDriverPath, pszModule);
if (GetFileAttributes(pSoundInfoNew->m_szDriverPath) == 0xFFFFFFFF)
{
// Try windows dir + \system32\drivers\ + module string
if( GetWindowsDirectory(pSoundInfoNew->m_szDriverPath, MAX_PATH) != 0 )
{
lstrcat(pSoundInfoNew->m_szDriverPath, TEXT("\\System32\\Drivers\\"));
if( lstrlen(pSoundInfoNew->m_szDriverPath) + nLenMod < 500 )
lstrcat(pSoundInfoNew->m_szDriverPath, pszModule);
}
}
}
}
}
}
PrvReleaseDeviceDescription( pdsdData );
return TRUE;
}
/****************************************************************************
*
* GetExtraSoundInfo
*
****************************************************************************/
HRESULT GetExtraSoundInfo(SoundInfo* pSoundInfoFirst)
{
SoundInfo* pSoundInfo;
BOOL bNT = BIsPlatformNT();
for (pSoundInfo = pSoundInfoFirst; pSoundInfo != NULL;
pSoundInfo = pSoundInfo->m_pSoundInfoNext)
{
CheckRegistry(&pSoundInfo->m_pRegErrorFirst);
if (bNT)
GetRegSoundInfoNT(pSoundInfo);
else
GetRegSoundInfo9x(pSoundInfo);
// Bug 18245: Try to distinguish between the various IBM MWave cards
if (_tcsstr(pSoundInfo->m_szDeviceID, TEXT("MWAVEAUDIO_0460")) != NULL)
lstrcat(pSoundInfo->m_szDescription, TEXT(" (Stingray)"));
else if (_tcsstr(pSoundInfo->m_szDeviceID, TEXT("MWAVEAUDIO_0465")) != NULL)
lstrcat(pSoundInfo->m_szDescription, TEXT(" (Marlin)"));
else
{
TCHAR szBoard[100];
lstrcpy( szBoard, TEXT("") );
GetPrivateProfileString(TEXT("Mwave,Board"), TEXT("board"), TEXT(""),
szBoard, 100, TEXT("MWave.ini"));
if (lstrcmp(szBoard, TEXT("MWAT-046")) == 0)
lstrcat(pSoundInfo->m_szDescription, TEXT(" (Dolphin)"));
else if (lstrcmp(szBoard, TEXT("MWAT-043")) == 0)
lstrcat(pSoundInfo->m_szDescription, TEXT(" (Whale)"));
}
// Sometimes, like when a sound driver is emulated, the driver
// will be reported as something like "WaveOut 0". In this case,
// just blank out the file-related fields.
if (_tcsstr(pSoundInfo->m_szDriverName, TEXT(".")) == NULL)
{
lstrcpy(pSoundInfo->m_szDriverName, TEXT(""));
lstrcpy(pSoundInfo->m_szDriverPath, TEXT(""));
}
else
{
GetFileVersion(pSoundInfo->m_szDriverPath, pSoundInfo->m_szDriverVersion,
pSoundInfo->m_szDriverAttributes, pSoundInfo->m_szDriverLanguageLocal, pSoundInfo->m_szDriverLanguage,
&pSoundInfo->m_bDriverBeta, &pSoundInfo->m_bDriverDebug);
FileIsSigned(pSoundInfo->m_szDriverPath, &pSoundInfo->m_bDriverSigned, &pSoundInfo->m_bDriverSignedValid);
GetFileDateAndSize(pSoundInfo->m_szDriverPath,
pSoundInfo->m_szDriverDateLocal, pSoundInfo->m_szDriverDate, &pSoundInfo->m_numBytes);
}
}
return S_OK;
}
/****************************************************************************
*
* GetRegSoundInfo9x
*
****************************************************************************/
VOID GetRegSoundInfo9x(SoundInfo* pSoundInfo)
{
HKEY hkey = NULL;
DWORD iKey = 0;
TCHAR szSubKey[200];
DWORD dwSubKeySize;
TCHAR szClass[100];
DWORD dwClassSize;
HKEY hkeySub = NULL;
DWORD dwDevnode;
DWORD cb;
DWORD dwType;
HKEY hkeyOther = NULL;
// We have the DevNode, so find the device in the registry with the
// matching DevNode and gather more info there.
if (ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE, TEXT("System\\CurrentControlSet\\control\\MediaResources\\wave"),
0, KEY_READ, &hkey))
{
while (TRUE)
{
dwSubKeySize = sizeof(szSubKey);
dwClassSize = sizeof(szClass);
if (ERROR_SUCCESS != RegEnumKeyEx(hkey, iKey, szSubKey, &dwSubKeySize, NULL, szClass, &dwClassSize, NULL))
break;
if (ERROR_SUCCESS == RegOpenKeyEx(hkey, szSubKey, 0, KEY_READ, &hkeySub))
{
cb = sizeof(dwDevnode);
if (ERROR_SUCCESS == RegQueryValueEx(hkeySub, TEXT("DevNode"), NULL, &dwType, (BYTE*)&dwDevnode, &cb))
{
if (dwDevnode == pSoundInfo->m_dwDevnode)
{
// Found match...gather yummy info
cb = sizeof(pSoundInfo->m_szDeviceID);
RegQueryValueEx(hkeySub, TEXT("DeviceID"), NULL, &dwType, (BYTE*)pSoundInfo->m_szDeviceID, &cb);
// Occasionally the driver name that DirectSoundEnumerate spits out
// is garbage (as with my Crystal SoundFusion). If that's the case,
// use the driver name listed here instead.
if (lstrlen(pSoundInfo->m_szDriverName) < 4)
{
cb = sizeof(pSoundInfo->m_szDriverName);
RegQueryValueEx(hkeySub, TEXT("Driver"), NULL, &dwType, (BYTE*)pSoundInfo->m_szDriverName, &cb);
GetSystemDirectory(pSoundInfo->m_szDriverPath, MAX_PATH);
lstrcat(pSoundInfo->m_szDriverPath, TEXT("\\"));
if( lstrlen(pSoundInfo->m_szDriverPath) + lstrlen(pSoundInfo->m_szDriverName) < 500 )
lstrcat(pSoundInfo->m_szDriverPath, pSoundInfo->m_szDriverName);
}
TCHAR szOtherKey[300];
cb = sizeof(szOtherKey);
RegQueryValueEx(hkeySub, TEXT("SOFTWAREKEY"), NULL, &dwType, (BYTE*)szOtherKey, &cb);
if (lstrlen(szOtherKey) > 0)
{
if (ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE, szOtherKey, 0, KEY_READ, &hkeyOther))
{
cb = sizeof(pSoundInfo->m_szOtherDrivers);
RegQueryValueEx(hkeyOther, TEXT("Driver"), NULL, &dwType, (BYTE*)pSoundInfo->m_szOtherDrivers, &cb);
cb = sizeof(pSoundInfo->m_szProvider);
RegQueryValueEx(hkeyOther, TEXT("ProviderName"), NULL, &dwType, (BYTE*)pSoundInfo->m_szProvider, &cb);
RegCloseKey(hkeyOther);
}
}
}
}
RegCloseKey(hkeySub);
}
iKey++;
}
RegCloseKey(hkey);
}
}
/****************************************************************************
*
* GetRegSoundInfoNT
*
****************************************************************************/
VOID GetRegSoundInfoNT(SoundInfo* pSoundInfo)
{
TCHAR szFullKey[300];
HKEY hkey;
DWORD cbData;
DWORD dwType;
TCHAR szDriverKey[300];
TCHAR szOtherFullKey[300];
lstrcpy(szFullKey, TEXT("System\\CurrentControlSet\\Enum\\"));
lstrcat(szFullKey, pSoundInfo->m_szRegKey);
if (ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE, szFullKey, 0, KEY_READ, &hkey))
{
cbData = sizeof(pSoundInfo->m_szDeviceID);
RegQueryValueEx(hkey, TEXT("HardwareID"), 0, &dwType, (LPBYTE)pSoundInfo->m_szDeviceID, &cbData);
cbData = sizeof(szDriverKey);
RegQueryValueEx(hkey, TEXT("Driver"), 0, &dwType, (LPBYTE)szDriverKey, &cbData);
RegCloseKey(hkey);
}
lstrcpy(szOtherFullKey, TEXT("System\\CurrentControlSet\\Control\\Class\\"));
lstrcat(szOtherFullKey, szDriverKey);
if (ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE, szOtherFullKey, 0, KEY_READ, &hkey))
{
cbData = sizeof(pSoundInfo->m_szProvider);
RegQueryValueEx(hkey, TEXT("ProviderName"), 0, &dwType, (LPBYTE)pSoundInfo->m_szProvider, &cbData);
RegCloseKey(hkey);
}
}
/****************************************************************************
*
* GetDSSoundInfo
*
****************************************************************************/
HRESULT GetDSSoundInfo(SoundInfo* pSoundInfoFirst)
{
HRESULT hr;
HRESULT hrRet = S_OK;
SoundInfo* pSoundInfo;
TCHAR szPath[MAX_PATH];
HINSTANCE hInstDSound;
LPDIRECTSOUNDCREATE pDSCreate;
GetSystemDirectory(szPath, MAX_PATH);
lstrcat(szPath, TEXT("\\dsound.dll"));
hInstDSound = LoadLibrary(szPath);
if (hInstDSound == NULL)
return E_FAIL;
pDSCreate = (LPDIRECTSOUNDCREATE)GetProcAddress(hInstDSound, "DirectSoundCreate");
if (pDSCreate == NULL)
{
FreeLibrary(hInstDSound);
return E_FAIL;
}
for (pSoundInfo = pSoundInfoFirst; pSoundInfo != NULL;
pSoundInfo = pSoundInfo->m_pSoundInfoNext)
{
if (FAILED(hr = GetDirectSoundInfo(pDSCreate, pSoundInfo)))
hrRet = hr; // but keep going
}
FreeLibrary(hInstDSound);
return hrRet;
}
/****************************************************************************
*
* IsDriverWDM
*
****************************************************************************/
BOOL IsDriverWDM( TCHAR* szDriverName )
{
if( _tcsstr( szDriverName, TEXT(".sys") ) == NULL )
return FALSE;
else
return TRUE;
}
/****************************************************************************
*
* GetDirectSoundInfo
*
****************************************************************************/
HRESULT GetDirectSoundInfo(LPDIRECTSOUNDCREATE pDSCreate, SoundInfo* pSoundInfo)
{
HRESULT hr;
LPDIRECTSOUND pds = NULL;
GUID* pGUID;
DSCAPS dscaps;
// Right now, this function only calls DSCreate/GetCaps to determine if
// the driver is signed. If we have already determined that it is by
// other means, don't bother with this test.
if (pSoundInfo->m_bDriverSigned)
return S_OK;
// Bug 29918: If this is a WDM driver, then don't call GetCaps() since
// on DX7.1+ GetCaps() will always return DSCAPS_CERTIFIED on WDM drivers
if( IsDriverWDM( pSoundInfo->m_szDriverName ) )
return S_OK;
if (pSoundInfo->m_guid == GUID_NULL)
pGUID = NULL;
else
pGUID = &pSoundInfo->m_guid;
if (FAILED(hr = pDSCreate(pGUID, &pds, NULL)))
goto LFail;
dscaps.dwSize = sizeof(dscaps);
if (FAILED(hr = pds->GetCaps(&dscaps)))
goto LFail;
pSoundInfo->m_bDriverSignedValid = TRUE;
if (dscaps.dwFlags & DSCAPS_CERTIFIED)
pSoundInfo->m_bDriverSigned = TRUE;
pds->Release();
return S_OK;
LFail:
if (pds != NULL)
pds->Release();
return hr;
}
/****************************************************************************
*
* ChangeAccelerationLevel
*
****************************************************************************/
HRESULT ChangeAccelerationLevel(SoundInfo* pSoundInfo, LONG lwLevel)
{
HRESULT hr = S_OK;
DIRECTSOUNDBASICACCELERATION_LEVEL level = (DIRECTSOUNDBASICACCELERATION_LEVEL)lwLevel;
LPKSPROPERTYSET pksps = NULL;
TCHAR szPath[MAX_PATH];
HINSTANCE hInstDSound = NULL;
GetSystemDirectory(szPath, MAX_PATH);
lstrcat(szPath, TEXT("\\dsound.dll"));
hInstDSound = LoadLibrary(szPath);
if (hInstDSound == NULL)
{
hr = DDERR_NOTFOUND;
goto LEnd;
}
if (FAILED(hr = DirectSoundPrivateCreate(&pksps)))
goto LEnd;
if (FAILED(hr = PrvSetBasicAcceleration(pksps, pSoundInfo->m_guid, level)))
goto LEnd;
LEnd:
if (pksps != NULL)
pksps->Release();
if (hInstDSound != NULL)
FreeLibrary(hInstDSound);
pSoundInfo->m_lwAccelerationLevel = lwLevel;
return hr;
}
/****************************************************************************
*
* CheckRegistry
*
****************************************************************************/
HRESULT CheckRegistry(RegError** ppRegErrorFirst)
{
HRESULT hr;
HKEY HKCR = HKEY_CLASSES_ROOT;
TCHAR szVersion[100];
HKEY hkey;
DWORD cbData;
ULONG ulType;
DWORD dwMajor = 0;
DWORD dwMinor = 0;
DWORD dwRevision = 0;
DWORD dwBuild = 0;
if (ERROR_SUCCESS == RegOpenKeyEx(HKEY_LOCAL_MACHINE, TEXT("Software\\Microsoft\\DirectX"),
0, KEY_READ, &hkey))
{
cbData = 100;
RegQueryValueEx(hkey, TEXT("Version"), 0, &ulType, (LPBYTE)szVersion, &cbData);
RegCloseKey(hkey);
if (lstrlen(szVersion) > 6 &&
lstrlen(szVersion) < 20)
{
if( _stscanf(szVersion, TEXT("%d.%d.%d.%d"), &dwMajor, &dwMinor, &dwRevision, &dwBuild) != 4 )
{
dwMajor = 0;
dwMinor = 0;
dwRevision = 0;
dwBuild = 0;
}
}
}
// No registry checking on DX versions before DX7
if (dwMinor < 7)
return S_OK;
// From dsound.inf:
if (FAILED(hr = CheckRegString(ppRegErrorFirst, HKCR, TEXT("DirectSound"), TEXT(""), TEXT("*"))))
return hr;
if (FAILED(hr = CheckRegString(ppRegErrorFirst, HKCR, TEXT("DirectSound\\CLSID"), TEXT(""), TEXT("{47D4D946-62E8-11cf-93BC-444553540000}"))))
return hr;
if (FAILED(hr = CheckRegString(ppRegErrorFirst, HKCR, TEXT("CLSID\\{47D4D946-62E8-11cf-93BC-444553540000}"), TEXT(""), TEXT("*"))))
return hr;
if (FAILED(hr = CheckRegString(ppRegErrorFirst, HKCR, TEXT("CLSID\\{47D4D946-62E8-11cf-93BC-444553540000}\\InprocServer32"), TEXT(""), TEXT("dsound.dll"), CRF_LEAF)))
return hr;
if (FAILED(hr = CheckRegString(ppRegErrorFirst, HKCR, TEXT("CLSID\\{47D4D946-62E8-11cf-93BC-444553540000}\\InprocServer32"), TEXT("ThreadingModel"), TEXT("Both"))))
return hr;
if (FAILED(hr = CheckRegString(ppRegErrorFirst, HKCR, TEXT("DirectSoundCapture"), TEXT(""), TEXT("*"))))
return hr;
if (FAILED(hr = CheckRegString(ppRegErrorFirst, HKCR, TEXT("DirectSoundCapture\\CLSID"), TEXT(""), TEXT("{B0210780-89CD-11d0-AF08-00A0C925CD16}"))))
return hr;
if (FAILED(hr = CheckRegString(ppRegErrorFirst, HKCR, TEXT("CLSID\\{B0210780-89CD-11d0-AF08-00A0C925CD16}"), TEXT(""), TEXT("*"))))
return hr;
if (FAILED(hr = CheckRegString(ppRegErrorFirst, HKCR, TEXT("CLSID\\{B0210780-89CD-11d0-AF08-00A0C925CD16}\\InprocServer32"), TEXT(""), TEXT("dsound.dll"), CRF_LEAF)))
return hr;
if (FAILED(hr = CheckRegString(ppRegErrorFirst, HKCR, TEXT("CLSID\\{B0210780-89CD-11d0-AF08-00A0C925CD16}\\InprocServer32"), TEXT("ThreadingModel"), TEXT("Both"))))
return hr;
return S_OK;
}
/****************************************************************************
*
* DestroySoundInfo
*
****************************************************************************/
VOID DestroySoundInfo(SoundInfo* pSoundInfoFirst)
{
SoundInfo* pSoundInfo;
SoundInfo* pSoundInfoNext;
for (pSoundInfo = pSoundInfoFirst; pSoundInfo != NULL;
pSoundInfo = pSoundInfoNext)
{
DestroyReg( &pSoundInfo->m_pRegErrorFirst );
pSoundInfoNext = pSoundInfo->m_pSoundInfoNext;
delete pSoundInfo;
}
}
/****************************************************************************
*
* DiagnoseSound
*
****************************************************************************/
VOID DiagnoseSound(SoundInfo* pSoundInfoFirst)
{
SoundInfo* pSoundInfo;
TCHAR sz[500];
TCHAR szFmt[500];
for (pSoundInfo = pSoundInfoFirst; pSoundInfo != NULL;
pSoundInfo = pSoundInfo->m_pSoundInfoNext)
{
_tcscpy( pSoundInfo->m_szNotes, TEXT("") );
_tcscpy( pSoundInfo->m_szNotesEnglish, TEXT("") );
// Report any problems:
BOOL bProblem = FALSE;
if ( pSoundInfo->m_bDriverSignedValid &&
!pSoundInfo->m_bDriverSigned &&
lstrlen(pSoundInfo->m_szDriverName) > 0)
{
LoadString(NULL, IDS_UNSIGNEDDRIVERFMT1, szFmt, 300);
wsprintf(sz, szFmt, pSoundInfo->m_szDriverName);
_tcscat( pSoundInfo->m_szNotes, sz );
LoadString(NULL, IDS_UNSIGNEDDRIVERFMT1_ENGLISH, szFmt, 300);
wsprintf(sz, szFmt, pSoundInfo->m_szDriverName);
_tcscat( pSoundInfo->m_szNotesEnglish, sz );
bProblem = TRUE;
}
if (pSoundInfo->m_pRegErrorFirst != NULL)
{
LoadString(NULL, IDS_REGISTRYPROBLEM, sz, 500);
_tcscat( pSoundInfo->m_szNotes, sz );
LoadString(NULL, IDS_REGISTRYPROBLEM_ENGLISH, sz, 500);
_tcscat( pSoundInfo->m_szNotesEnglish, sz );
bProblem = TRUE;
}
// Report any DSound test results:
if (pSoundInfo->m_testResultSnd.m_bStarted &&
!pSoundInfo->m_testResultSnd.m_bCancelled)
{
LoadString(NULL, IDS_DSRESULTS, sz, 500);
_tcscat( pSoundInfo->m_szNotes, sz );
_tcscat( pSoundInfo->m_szNotes, pSoundInfo->m_testResultSnd.m_szDescription );
_tcscat( pSoundInfo->m_szNotes, TEXT("\r\n") );
LoadString(NULL, IDS_DSRESULTS_ENGLISH, sz, 500);
_tcscat( pSoundInfo->m_szNotesEnglish, sz );
_tcscat( pSoundInfo->m_szNotesEnglish, pSoundInfo->m_testResultSnd.m_szDescriptionEnglish );
_tcscat( pSoundInfo->m_szNotesEnglish, TEXT("\r\n") );
bProblem = TRUE;
}
else
{
LoadString(NULL, IDS_DSINSTRUCTIONS, sz, 500);
_tcscat( pSoundInfo->m_szNotes, sz );
LoadString(NULL, IDS_DSINSTRUCTIONS_ENGLISH, sz, 500);
_tcscat( pSoundInfo->m_szNotesEnglish, sz );
}
if (!bProblem)
{
LoadString(NULL, IDS_NOPROBLEM, sz, 500);
_tcscat( pSoundInfo->m_szNotes, sz );
LoadString(NULL, IDS_NOPROBLEM_ENGLISH, sz, 500);
_tcscat( pSoundInfo->m_szNotesEnglish, sz );
}
}
}