|
|
/*++
Copyright (C) Microsoft Corporation
Module Name:
CDriver.cpp
Abstract:
This module implements CDriver and CService classes
Author:
William Hsieh (williamh) created
Revision History:
--*/
#include "devmgr.h"
#include "cdriver.h"
const TCHAR* tszStringFileInfo = TEXT("StringFileInfo\\%04X%04X\\"); const TCHAR* tszFileVersion = TEXT("FileVersion"); const TCHAR* tszLegalCopyright = TEXT("LegalCopyright"); const TCHAR* tszCompanyName = TEXT("CompanyName"); const TCHAR* tszTranslation = TEXT("VarFileInfo\\Translation"); const TCHAR* tszStringFileInfoDefault = TEXT("StringFileInfo\\040904B0\\");
BOOL CDriver::Create( CDevice* pDevice ) { HKEY hKey; TCHAR InfName[MAX_PATH];
ASSERT(pDevice);
m_pDevice = pDevice; m_OnLocalMachine = pDevice->m_pMachine->IsLocal();
CMachine* pMachine = m_pDevice->m_pMachine;
ASSERT(pMachine);
//
// We can't get the driver list on remote machines
//
if (!m_OnLocalMachine) { return TRUE; }
m_hSDBDrvMain = SdbInitDatabase(SDB_DATABASE_MAIN_DRIVERS, NULL);
//
// Open drvice's driver registry key to get the InfPath
//
hKey = pMachine->DiOpenDevRegKey(*m_pDevice, DICS_FLAG_GLOBAL, 0, DIREG_DRV, KEY_READ);
if (INVALID_HANDLE_VALUE != hKey) {
DWORD regType; DWORD Len = sizeof(InfName); CSafeRegistry regDrv(hKey);
//
// Get the inf path from the driver key
//
if (regDrv.GetValue(REGSTR_VAL_INFPATH, ®Type, (PBYTE)InfName, &Len)) {
String strInfPath;
if (strInfPath.GetSystemWindowsDirectory()) {
//
// Tack on an extra back slash if one is needed
//
if (_T('\\') != strInfPath[strInfPath.GetLength() - 1]) { strInfPath += (LPCTSTR)TEXT("\\"); }
strInfPath += (LPCTSTR)TEXT("INF\\"); strInfPath += (LPCTSTR)InfName;
pMachine->GetInfDigitalSigner((LPCTSTR)strInfPath, m_DigitalSigner); } } }
return TRUE; }
BOOL CDriver::BuildDriverList( BOOL bFunctionAndFiltersOnly ) { SP_DRVINFO_DATA DrvInfoData; HSPFILEQ hFileQueue = INVALID_HANDLE_VALUE; SP_DEVINSTALL_PARAMS DevInstParams;
//
// If we already built up the list of driver files then we don't need
// to do it again.
//
if (m_DriverListBuilt) { return m_listDriverFile.GetCount(); }
ASSERT(m_pDevice);
if (!m_OnLocalMachine) { AddFunctionAndFilterDrivers(m_pDevice); return m_listDriverFile.GetCount(); }
CMachine* pMachine = m_pDevice->m_pMachine; ASSERT(pMachine);
hFileQueue = SetupOpenFileQueue();
//
// Only build up the list of files from the INF if bFunctionAndFiltersOnly
// is not TRUE.
//
if (!bFunctionAndFiltersOnly) { DevInstParams.cbSize = sizeof(DevInstParams);
pMachine->DiGetDeviceInstallParams(*m_pDevice, &DevInstParams);
//
// Set the DI_FLAGSEX_INSTALLEDDRIVER flag before calling SetupDiBuildDriverInfoList.
// This will have it only put the installed driver into the list.
//
DevInstParams.FlagsEx |= (DI_FLAGSEX_INSTALLEDDRIVER | DI_FLAGSEX_ALLOWEXCLUDEDDRVS);
if (pMachine->DiSetDeviceInstallParams(*m_pDevice, &DevInstParams) && pMachine->DiBuildDriverInfoList(*m_pDevice, SPDIT_CLASSDRIVER)) {
DrvInfoData.cbSize = sizeof(DrvInfoData);
//
// There should only be one driver in this list. If there isn't any
// drivers in this list then there must not be a driver currently
// installed on this device.
//
if (pMachine->DiEnumDriverInfo(*m_pDevice, SPDIT_CLASSDRIVER, 0, &DrvInfoData)) {
//
// Set this as the selected driver
//
if (pMachine->DiSetSelectedDriver(*m_pDevice, &DrvInfoData)) { //
// Get a list of all the files installed for this device
//
if (INVALID_HANDLE_VALUE != hFileQueue) {
DevInstParams.FileQueue = hFileQueue; DevInstParams.Flags |= DI_NOVCP;
if (pMachine->DiSetDeviceInstallParams(*m_pDevice, &DevInstParams) && pMachine->DiCallClassInstaller(DIF_INSTALLDEVICEFILES, *m_pDevice)) { //
// Dereference the file queue so that we can close it
//
DevInstParams.FileQueue = NULL; DevInstParams.Flags &= ~DI_NOVCP; pMachine->DiSetDeviceInstallParams(*m_pDevice, &DevInstParams); } } }
} else {
//
// We did not find a match...so just destroy it.
//
pMachine->DiDestroyDriverInfoList(*m_pDevice, SPDIT_CLASSDRIVER); } } }
//
// Add the funtion and device and class upper and lower filters, sometimes
// these aren't added via the INF file directly so this makes sure they
// show up in the list.
//
AddFunctionAndFilterDrivers(m_pDevice, hFileQueue);
if (hFileQueue != INVALID_HANDLE_VALUE) { //
// Scan the file queue.
//
DWORD ScanResult; SetupScanFileQueue(hFileQueue, SPQ_SCAN_USE_CALLBACK_SIGNERINFO, NULL, ScanQueueCallback, (PVOID)this, &ScanResult );
//
// Close the file queue
//
SetupCloseFileQueue(hFileQueue); }
m_DriverListBuilt = TRUE;
return m_listDriverFile.GetCount(); }
void CDriver::AddDriverFile( CDriverFile* pNewDrvFile ) { //
// Check to see if this driver already exists in the list.
//
POSITION pos = m_listDriverFile.GetHeadPosition();
while (NULL != pos) { CDriverFile* pDrvFile = m_listDriverFile.GetNext(pos);
if (lstrcmpi(pDrvFile->GetFullPathName(), pNewDrvFile->GetFullPathName()) == 0) { //
// This file already exists in the list so just return without
// adding it.
//
return; } }
m_listDriverFile.AddTail(pNewDrvFile); }
void CDriver::AddFunctionAndFilterDrivers( CDevice* pDevice, HSPFILEQ hFileQueue ) { TCHAR ServiceName[MAX_PATH]; ULONG BufferLen; HKEY hKey; DWORD regType;
//
// Get the function driver
//
if (pDevice->m_pMachine->DiGetDeviceRegistryProperty(*pDevice, SPDRP_SERVICE, NULL, (PBYTE)ServiceName, sizeof(ServiceName), NULL )) { CreateFromService(pDevice, ServiceName, hFileQueue); }
//
// Add the upper and lower device filters
//
for (int i = 0; i<2; i++) { BufferLen = 0; pDevice->m_pMachine->DiGetDeviceRegistryProperty( *pDevice, i ? SPDRP_LOWERFILTERS : SPDRP_UPPERFILTERS, NULL, NULL, BufferLen, &BufferLen );
if (BufferLen != 0) { PTSTR Buffer = new TCHAR[BufferLen+2];
if (Buffer) { ZeroMemory(Buffer, BufferLen+2);
if (pDevice->m_pMachine->DiGetDeviceRegistryProperty( *pDevice, i ? SPDRP_LOWERFILTERS : SPDRP_UPPERFILTERS, NULL, (PBYTE)Buffer, BufferLen, &BufferLen )) { for (PTSTR SingleItem = Buffer; *SingleItem; SingleItem += (lstrlen(SingleItem) + 1)) { CreateFromService(pDevice, SingleItem, hFileQueue); } }
delete [] Buffer; } } }
//
// Add the upper and lower class filters
//
GUID ClassGuid; pDevice->ClassGuid(ClassGuid); hKey = m_pDevice->m_pMachine->DiOpenClassRegKey(&ClassGuid, KEY_READ, DIOCR_INSTALLER);
if (INVALID_HANDLE_VALUE != hKey) {
CSafeRegistry regClass(hKey);
for (int i = 0; i<2; i++) { BufferLen = 0; regClass.GetValue(i ? REGSTR_VAL_LOWERFILTERS : REGSTR_VAL_UPPERFILTERS, ®Type, NULL, &BufferLen ); if (BufferLen != 0) { PTSTR Buffer = new TCHAR[BufferLen+2];
if (Buffer) { ZeroMemory(Buffer, BufferLen+2);
if (regClass.GetValue(i ? REGSTR_VAL_LOWERFILTERS : REGSTR_VAL_UPPERFILTERS, ®Type, (PBYTE)Buffer, &BufferLen )) { for (PTSTR SingleItem = Buffer; *SingleItem; SingleItem += (lstrlen(SingleItem) + 1)) { CreateFromService(pDevice, SingleItem, hFileQueue); } }
delete [] Buffer; } } } }
}
void CDriver::CreateFromService( CDevice* pDevice, PCTSTR ServiceName, HSPFILEQ hFileQueue ) { SC_HANDLE hscManager = NULL; SC_HANDLE hscService = NULL;
if (!ServiceName) { return; }
try { BOOL ComposePathNameFromServiceName = TRUE;
hscManager = OpenSCManager(m_OnLocalMachine ? NULL : pDevice->m_pMachine->GetMachineFullName(), NULL, GENERIC_READ);
if (NULL != hscManager) { hscService = OpenService(hscManager, ServiceName, GENERIC_READ); if (NULL != hscService) { DWORD BytesRequired;
// first, probe for buffer size
if (!QueryServiceConfig(hscService, NULL, 0, &BytesRequired) && ERROR_INSUFFICIENT_BUFFER == GetLastError()) { TCHAR FullPath[MAX_PATH]; BufferPtr<BYTE> BufPtr(BytesRequired); LPQUERY_SERVICE_CONFIG pqsc; pqsc = (LPQUERY_SERVICE_CONFIG)(PBYTE)BufPtr; DWORD Size;
if (QueryServiceConfig(hscService, pqsc, BytesRequired, &Size) && pqsc->lpBinaryPathName && (TEXT('\0') != pqsc->lpBinaryPathName[0])) { ComposePathNameFromServiceName = FALSE;
//
// Make sure we have a valid full path.
//
if (GetFullPathFromImagePath(pqsc->lpBinaryPathName, FullPath, ARRAYLEN(FullPath))) {
if (hFileQueue != INVALID_HANDLE_VALUE) { //
// Add the file to the queue.
//
TCHAR TargetPath[MAX_PATH]; StringCchCopy(TargetPath, ARRAYLEN(TargetPath), FullPath); PTSTR p = (PTSTR)StrRChr(TargetPath, NULL, TEXT('\\')); if (p) { *p = TEXT('\0'); }
SetupQueueCopy(hFileQueue, NULL, NULL, pSetupGetFileTitle(FullPath), NULL, NULL, TargetPath, NULL, 0 ); } else { //
// No file queue was passed in so just manually
// add this to our list of driver files.
//
SafePtr<CDriverFile> DrvFilePtr; CDriverFile* pDrvFile = new CDriverFile(); DrvFilePtr.Attach(pDrvFile);
//
// We will set the GetWin32Error to 0xFFFFFFFF which will
// cause the UI to say 'not available' for the
// signature.
//
if (pDrvFile->Create(FullPath, m_OnLocalMachine, 0xFFFFFFFF, NULL, m_hSDBDrvMain)) { AddDriverFile(pDrvFile); DrvFilePtr.Detach(); } } } } }
CloseServiceHandle(hscService); hscService = NULL; }
CloseServiceHandle(hscManager); hscManager = NULL; }
if (ComposePathNameFromServiceName) { String strFullPathName; strFullPathName.GetSystemDirectory(); strFullPathName += (LPCTSTR)TEXT("\\drivers\\"); strFullPathName += (LPCTSTR)ServiceName; strFullPathName += (LPCTSTR)TEXT(".sys");
if (hFileQueue != INVALID_HANDLE_VALUE) { //
// Add the file to the queue.
//
String strTargetPath; strTargetPath = strFullPathName; PTSTR p = (PTSTR)StrRChr((LPCTSTR)strTargetPath, NULL, TEXT('\\')); if (p) { *p = TEXT('\0'); }
SetupQueueCopy(hFileQueue, NULL, NULL, pSetupGetFileTitle((LPCTSTR)strFullPathName), NULL, NULL, strTargetPath, NULL, 0 ); } else { //
// No file queue was passed in so just manually
// add this to our list of driver files.
//
SafePtr<CDriverFile> DrvFilePtr; CDriverFile* pDrvFile = new CDriverFile(); DrvFilePtr.Attach(pDrvFile);
//
// We will set the GetWin32Error to 0xFFFFFFFF which will
// cause the UI to say 'not available' for the
// signature.
//
if (pDrvFile->Create((LPCTSTR)strFullPathName, m_OnLocalMachine, 0xFFFFFFFF, NULL, m_hSDBDrvMain)) { AddDriverFile(pDrvFile); DrvFilePtr.Detach(); } } } }
catch (CMemoryException* e) { UNREFERENCED_PARAMETER(e);
if (hscService) { CloseServiceHandle(hscService); }
if (hscManager) { CloseServiceHandle(hscManager); } throw; } }
CDriver::~CDriver() { if (!m_listDriverFile.IsEmpty()) { POSITION pos = m_listDriverFile.GetHeadPosition();
while (NULL != pos) { CDriverFile* pDrvFile = m_listDriverFile.GetNext(pos); delete pDrvFile; }
m_listDriverFile.RemoveAll(); }
if (m_hSDBDrvMain) { SdbReleaseDatabase(m_hSDBDrvMain); } }
//
// Can not throw a exception from this function because it is a callback
//
UINT CDriver::ScanQueueCallback( PVOID Context, UINT Notification, UINT_PTR Param1, UINT_PTR Param2 ) { UNREFERENCED_PARAMETER(Param2);
try { if (SPFILENOTIFY_QUEUESCAN_SIGNERINFO == Notification && Param1) { CDriver* pDriver = (CDriver*)Context;
if (pDriver) { SafePtr<CDriverFile> DrvFilePtr; CDriverFile* pDrvFile = new CDriverFile(); DrvFilePtr.Attach(pDrvFile);
//
// When creating the CDriver set the Win32Error to 0xFFFFFFFF
// if the user is loged in as a guest. This is because we
// cannot tell if a file is digitally signed if the user is
// a guest. If the user is not a guest then use the Win32Error
// returned from setupapi.
//
if (pDrvFile->Create((LPCTSTR)((PFILEPATHS_SIGNERINFO)Param1)->Target, pDriver->IsLocal(), pDriver->m_pDevice->m_pMachine->IsUserAGuest() ? 0xFFFFFFFF : ((PFILEPATHS_SIGNERINFO)Param1)->Win32Error, ((PFILEPATHS_SIGNERINFO)Param1)->DigitalSigner, pDriver->m_hSDBDrvMain )) { pDriver->AddDriverFile(pDrvFile); DrvFilePtr.Detach(); } } } }
catch (CMemoryException* e) { e->Delete(); return ERROR_NOT_ENOUGH_MEMORY; }
return NO_ERROR; }
BOOL CDriver::GetFirstDriverFile( CDriverFile** ppDrvFile, PVOID& Context ) { ASSERT(ppDrvFile);
if (!m_listDriverFile.IsEmpty()) { POSITION pos = m_listDriverFile.GetHeadPosition(); *ppDrvFile = m_listDriverFile.GetNext(pos); Context = pos; return TRUE; }
Context = NULL; *ppDrvFile = NULL;
return FALSE; }
BOOL CDriver::GetNextDriverFile( CDriverFile** ppDrvFile, PVOID& Context ) { ASSERT(ppDrvFile);
POSITION pos = (POSITION)Context;
if (NULL != pos) { *ppDrvFile = m_listDriverFile.GetNext(pos); Context = pos; return TRUE; }
*ppDrvFile = NULL; return FALSE; }
void CDriver::GetDriverSignerString( String& strDriverSigner ) { if (m_DigitalSigner.IsEmpty()) {
strDriverSigner.LoadString(g_hInstance, IDS_NO_DIGITALSIGNATURE);
} else {
strDriverSigner = m_DigitalSigner; } }
BOOL CDriver::GetFullPathFromImagePath( LPCTSTR ImagePath, LPTSTR FullPath, UINT FullPathLength ) { TCHAR OriginalCurrentDirectory[MAX_PATH]; LPTSTR pRelativeString; LPTSTR lpFilePart;
if (!ImagePath || (ImagePath[0] == TEXT('\0'))) { return FALSE; }
//
// If we aren't on a local machine then just return the file name and not
// the full path.
//
if (!m_OnLocalMachine) { if (SUCCEEDED(StringCchCopy(FullPath, FullPathLength, pSetupGetFileTitle(ImagePath)))) { return TRUE; } else { return FALSE; } }
//
// First check if the ImagePath happens to be a valid full path.
//
if (GetFileAttributes(ImagePath) != 0xFFFFFFFF) { ::GetFullPathName(ImagePath, FullPathLength, FullPath, &lpFilePart); return TRUE; }
pRelativeString = (LPTSTR)ImagePath;
//
// If the ImagePath starts with "\SystemRoot" or "%SystemRoot%" then
// remove those values.
//
if (StrCmpNI(ImagePath, TEXT("\\SystemRoot\\"), lstrlen(TEXT("\\SystemRoot\\"))) == 0) { pRelativeString += lstrlen(TEXT("\\SystemRoot\\")); } else if (StrCmpNI(ImagePath, TEXT("%SystemRoot%\\"), lstrlen(TEXT("%SystemRoot%\\"))) == 0) { pRelativeString += lstrlen(TEXT("%SystemRoot%\\")); }
//
// At this point pRelativeString should point to the image path relative to
// the windows directory.
//
if (!GetSystemWindowsDirectory(FullPath, FullPathLength)) { return FALSE; }
if (!GetCurrentDirectory(ARRAYLEN(OriginalCurrentDirectory), OriginalCurrentDirectory)) { OriginalCurrentDirectory[0] = TEXT('\0'); }
if (!SetCurrentDirectory(FullPath)) { return FALSE; }
::GetFullPathName(pRelativeString, FullPathLength, FullPath, &lpFilePart);
if (OriginalCurrentDirectory[0] != TEXT('\0')) { SetCurrentDirectory(OriginalCurrentDirectory); }
return TRUE; }
BOOL CDriverFile::Create( LPCTSTR ServiceName, BOOL LocalMachine, DWORD Win32Error, LPCTSTR DigitalSigner, HSDB hSDBDrvMain ) { if (!ServiceName || (TEXT('\0') == ServiceName[0])) { return FALSE; }
m_Win32Error = Win32Error;
if (DigitalSigner) { m_strDigitalSigner = DigitalSigner; }
//
// For remote machine, we can not verify if the driver file exits.
// we only show the driver name.
//
if (LocalMachine) { m_Attributes = GetFileAttributes(ServiceName);
if (0xFFFFFFFF != m_Attributes) { m_strFullPathName = ServiceName;
} else { //
// The driver is a service. Do not search for the current director --
// GetFullPathName is useless here.
// Search for Windows dir and System directory
//
String strBaseDir;
if (strBaseDir.GetSystemWindowsDirectory()) {
if (_T('\\') != strBaseDir[strBaseDir.GetLength() - 1]) { strBaseDir += (LPCTSTR)TEXT("\\"); }
strBaseDir += (LPCTSTR)pSetupGetFileTitle(ServiceName); m_Attributes = GetFileAttributes((LPCTSTR)strBaseDir);
if (0xFFFFFFFF == m_Attributes) { if (strBaseDir.GetSystemDirectory()) {
if (_T('\\') != strBaseDir[strBaseDir.GetLength() - 1]) { strBaseDir += (LPCTSTR)TEXT("\\"); }
strBaseDir += (LPCTSTR)pSetupGetFileTitle(ServiceName); m_Attributes = GetFileAttributes(strBaseDir); } }
//
// hopeless, we could find the path
//
if (0xFFFFFFFF == m_Attributes) { return FALSE; }
m_strFullPathName = strBaseDir;
} else {
return FALSE; } }
m_HasVersionInfo = GetVersionInfo(); }
else { m_strFullPathName = ServiceName;
//
//we do not have version info
//
m_HasVersionInfo = FALSE; }
if (!m_strFullPathName.IsEmpty() && hSDBDrvMain != NULL) { TAGREF tagref = TAGREF_NULL; HAPPHELPINFOCONTEXT hAppHelpInfoContext = NULL; SDBENTRYINFO entryinfo; DWORD cbSize;
tagref = SdbGetDatabaseMatch(hSDBDrvMain, (LPTSTR)m_strFullPathName, INVALID_HANDLE_VALUE, NULL, 0 );
if (tagref != TAGREF_NULL) { //
// This driver is in the database.
//
m_IsDriverBlocked = TRUE;
//
// Call SdbReadDriverInformation to get the database GUID and the
// driver GUID for this entry.
//
ZeroMemory(&entryinfo, sizeof(entryinfo)); if (SdbReadDriverInformation(hSDBDrvMain, tagref, &entryinfo)) { //
// Open up the App help information database and query for the
// html link.
//
hAppHelpInfoContext = SdbOpenApphelpInformation(&(entryinfo.guidDB), &(entryinfo.guidID));
if (hAppHelpInfoContext) {
cbSize = 0; PBYTE pBuffer = NULL;
cbSize = SdbQueryApphelpInformation(hAppHelpInfoContext, ApphelpHelpCenterURL, NULL, 0);
if (cbSize && ((pBuffer = new BYTE[cbSize]) != NULL)) {
cbSize = SdbQueryApphelpInformation(hAppHelpInfoContext, ApphelpHelpCenterURL, (LPVOID)pBuffer, cbSize);
if (cbSize) { m_strHtmlHelpID = (LPTSTR)pBuffer; }
delete [] pBuffer; }
SdbCloseApphelpInformation(hAppHelpInfoContext); } } } }
return TRUE; }
BOOL CDriverFile::GetVersionInfo() { DWORD Size, dwHandle;
Size = GetFileVersionInfoSize((LPTSTR)(LPCTSTR)m_strFullPathName, &dwHandle);
if (!Size) { return FALSE; }
BufferPtr<BYTE> BufPtr(Size); PVOID pVerInfo = BufPtr;
if (GetFileVersionInfo((LPTSTR)(LPCTSTR)m_strFullPathName, dwHandle, Size, pVerInfo)) { // get VarFileInfo\Translation
PVOID pBuffer; UINT Len; String strStringFileInfo;
if (!VerQueryValue(pVerInfo, (LPTSTR)tszTranslation, &pBuffer, &Len)) { strStringFileInfo = tszStringFileInfoDefault; }
else { strStringFileInfo.Format(tszStringFileInfo, *((WORD*)pBuffer), *(((WORD*)pBuffer) + 1)); }
String str; str = strStringFileInfo + tszFileVersion;
if (VerQueryValue(pVerInfo, (LPTSTR)(LPCTSTR)str, &pBuffer, &Len)) { m_strVersion = (LPTSTR)pBuffer; str = strStringFileInfo + tszLegalCopyright;
if (VerQueryValue(pVerInfo, (LPTSTR)(LPCTSTR)str, &pBuffer, &Len)) { m_strCopyright = (LPTSTR)pBuffer; str = strStringFileInfo + tszCompanyName;
if (VerQueryValue(pVerInfo, (LPTSTR)(LPCTSTR)str, &pBuffer, &Len)) { m_strProvider = (LPTSTR)pBuffer; } } } }
return TRUE; }
BOOL CDriverFile::operator ==( CDriverFile& OtherDrvFile ) { return \ m_HasVersionInfo == OtherDrvFile.HasVersionInfo() && (GetFullPathName() == OtherDrvFile.GetFullPathName() || (GetFullPathName() && OtherDrvFile.GetFullPathName() && !lstrcmpi(GetFullPathName(), OtherDrvFile.GetFullPathName()) ) ) && (GetProvider() == OtherDrvFile.GetProvider() || (GetProvider() && OtherDrvFile.GetProvider() && !lstrcmpi(GetProvider(), OtherDrvFile.GetProvider()) ) ) && (GetCopyright() == OtherDrvFile.GetCopyright() || (GetCopyright() && OtherDrvFile.GetCopyright() && !lstrcmpi(GetCopyright(), OtherDrvFile.GetCopyright()) ) ) && (GetVersion() == OtherDrvFile.GetVersion() || (GetVersion() && OtherDrvFile.GetVersion() && !lstrcmpi(GetVersion(), OtherDrvFile.GetVersion()) ) ); }
|