/*++ Copyright (C) Microsoft Corporation Module Name: devdrvpg.cpp Abstract: This module implements CDeviceDriverPage -- device driver property page Author: William Hsieh (williamh) created Revision History: --*/ // devdrvpg.cpp : implementation file // #include "devmgr.h" #include "devdrvpg.h" #include "cdriver.h" #include "tswizard.h" #include "devrmdlg.h" // // help topic ids // const DWORD g_a106HelpIDs[]= { IDC_DEVDRV_ICON, IDH_DISABLEHELP, // Driver: "" (Static) IDC_DEVDRV_DESC, IDH_DISABLEHELP, // Driver: "" (Static) IDC_DEVDRV_TITLE_DRIVERPROVIDER, idh_devmgr_driver_provider_main, IDC_DEVDRV_DRIVERPROVIDER, idh_devmgr_driver_provider_main, IDC_DEVDRV_TITLE_DRIVERDATE, idh_devmgr_driver_date_main, IDC_DEVDRV_DRIVERDATE, idh_devmgr_driver_date_main, IDC_DEVDRV_TITLE_DRIVERVERSION, idh_devmgr_driver_version_main, IDC_DEVDRV_DRIVERVERSION, idh_devmgr_driver_version_main, IDC_DEVDRV_TITLE_DRIVERSIGNER, idh_devmgr_digital_signer, IDC_DEVDRV_DRIVERSIGNER, idh_devmgr_digital_signer, IDC_DEVDRV_DETAILS, idh_devmgr_devdrv_details, // Driver: "Driver Details" (Button) IDC_DEVDRV_DETAILS_TEXT, idh_devmgr_devdrv_details, // Driver: "Driver Details" (Button) IDC_DEVDRV_UNINSTALL, idh_devmgr_devdrv_uninstall, // Driver: "Uninstall" (Button) IDC_DEVDRV_UNINSTALL_TEXT, idh_devmgr_devdrv_uninstall, // Driver: "Uninstall" (Button) IDC_DEVDRV_CHANGEDRIVER, idh_devmgr_driver_change_driver, // Driver: "&Change Driver..." (Button) IDC_DEVDRV_CHANGEDRIVER_TEXT, idh_devmgr_driver_change_driver, // Driver: "&Change Driver..." (Button) IDC_DEVDRV_ROLLBACK, idh_devmgr_rollback_button, // Driver: "Roll Back Driver..." (Button) IDC_DEVDRV_ROLLBACK_TEXT, idh_devmgr_rollback_button, // Driver: "Roll Back Driver..." (Button) 0, 0 }; CDeviceDriverPage::~CDeviceDriverPage() { if (m_pDriver) { delete m_pDriver; } } BOOL CDeviceDriverPage::OnCommand( WPARAM wParam, LPARAM lParam ) { UNREFERENCED_PARAMETER(lParam); if (BN_CLICKED == HIWORD(wParam)) { switch (LOWORD(wParam)) { case IDC_DEVDRV_DETAILS: { // // We first need to call CDriver::BuildDriverList to build up a list // of drivers for this device. This can take some time so we will put // up the busy cursor. // HCURSOR hCursorOld = SetCursor(LoadCursor(NULL, IDC_WAIT)); CDriver* pDriver; pDriver = m_pDevice->CreateDriver(); if (pDriver) { pDriver->BuildDriverList(); SetCursor(hCursorOld); // // Show driver file details // if (pDriver->GetCount() > 0) { if (pDriver) { CDriverFilesDlg DriverFilesDlg(m_pDevice, pDriver); DriverFilesDlg.DoModal(m_hDlg, (LPARAM)&DriverFilesDlg); } } else { // // No driver files are loaded for this device // String strNoDrivers; strNoDrivers.LoadString(g_hInstance, IDS_DEVDRV_NODRIVERFILE); MessageBox(m_hDlg, strNoDrivers, m_pDevice->GetDisplayName(), MB_OK); return FALSE; } delete pDriver; pDriver = NULL; } break; } case IDC_DEVDRV_UNINSTALL: { BOOL Refresh = (m_pDevice->IsPhantom() || m_pDevice->HasProblem() || !m_pDevice->IsStarted()); if (UninstallDrivers(m_pDevice, m_hDlg)) { // // Enable refresh since we disabled it in the beginning. // // We only need to force a refresh here if the device that // was removed was a Phantom device. This is because Phantom // devices don't have kernel mode devnodes and so they won't // generate a WM_DEVICECHANGE like live devnodes will. // if (Refresh) { m_pDevice->m_pMachine->ScheduleRefresh(); } ::DestroyWindow(GetParent(m_hDlg)); } break; } case IDC_DEVDRV_CHANGEDRIVER: { BOOL fChanged; DWORD Reboot = 0; if (UpdateDriver(m_pDevice, m_hDlg, &fChanged, &Reboot) && fChanged) { // // ISSUE: JasonC 2/7/00 // // A refresh on m_pDevice->m_pMachine is necessary here. // Since we are running on a different thread and each // property page may have cached the HDEVINFO and the // SP_DEVINFO_DATA, refresh on the CMachine object can not // be done here. The problem is worsen by the fact that // user can go back to the device tree and work on the tree // while this property sheet is still up. // It would be nice if MMC would support modal dialog boxes! // m_pDevice->PropertyChanged(); m_pDevice->GetClass()->PropertyChanged(); m_pDevice->m_pMachine->DiTurnOnDiFlags(*m_pDevice, DI_PROPERTIES_CHANGE); UpdateControls(); PropSheet_SetTitle(GetParent(m_hDlg), PSH_PROPTITLE, m_pDevice->GetDisplayName()); PropSheet_CancelToClose(GetParent(m_hDlg)); if (Reboot & (DI_NEEDRESTART | DI_NEEDREBOOT)) { m_pDevice->m_pMachine->DiTurnOnDiFlags(*m_pDevice, DI_NEEDREBOOT); } } break; } case IDC_DEVDRV_ROLLBACK: { BOOL fChanged; DWORD Reboot = 0; if (RollbackDriver(m_pDevice, m_hDlg, &fChanged, &Reboot) && fChanged) { // // ISSUE: JasonC 2/7/00 // // A refresh on m_pDevice->m_pMachine is necessary here. // Since we are running on a different thread and each // property page may have cached the HDEVINFO and the // SP_DEVINFO_DATA, refresh on the CMachine object can not // be done here. The problem is worsen by the fact that // user can go back to the device tree and work on the tree // while this property sheet is still up. // It would be nice if MMC would support modal dialog boxes! // m_pDevice->PropertyChanged(); m_pDevice->GetClass()->PropertyChanged(); m_pDevice->m_pMachine->DiTurnOnDiFlags(*m_pDevice, DI_PROPERTIES_CHANGE); UpdateControls(); PropSheet_SetTitle(GetParent(m_hDlg), PSH_PROPTITLE, m_pDevice->GetDisplayName()); PropSheet_CancelToClose(GetParent(m_hDlg)); if (Reboot & (DI_NEEDRESTART | DI_NEEDREBOOT)) { m_pDevice->m_pMachine->DiTurnOnDiFlags(*m_pDevice, DI_NEEDREBOOT); } } break; } default: break; } } return FALSE; } // // This function uninstalls the drivers for the given device // INPUT: // pDevice -- the object represent the device // hDlg -- the property page window handle // // OUTPUT: // TRUE -- function succeeded. // FALSE -- function failed. // BOOL CDeviceDriverPage::UninstallDrivers( CDevice* pDevice, HWND hDlg ) { BOOL Return = FALSE; CMachine *pMachine; if (pDevice->m_pMachine->m_ParentMachine) { pMachine = pDevice->m_pMachine->m_ParentMachine; } else { pMachine = pDevice->m_pMachine; } if(!pDevice->m_pMachine->IsLocal() || !g_IsAdmin) { // // Must be an admin and on the local machine to remove a device. // return FALSE; } BOOL Refresh = (pDevice->IsPhantom() || pDevice->HasProblem() || !pDevice->IsStarted()); pMachine->EnableRefresh(FALSE); CRemoveDevDlg TheDlg(pDevice); if (IDOK == TheDlg.DoModal(hDlg, (LPARAM) &TheDlg)) { DWORD DiFlags; DiFlags = pDevice->m_pMachine->DiGetFlags(*pDevice); // // First try and send a MMCPropertyChangeNotify message to our // CComponent so that it can prompt for a reboot inside the // device manager thread instead of our thread. If this is not // done then the property sheet will hang around after device // manager has gone away...which will cause a "hung app" dialog // to appear. // // Note: In this case we pass NULL as the Window handle since // uninstalling will cause the property sheet to go away. // CNotifyRebootRequest* pNRR = new CNotifyRebootRequest(NULL, DiFlags, IDS_REMOVEDEV_RESTART); if (pNRR) { if (!m_pDevice->m_psd.PropertyChangeNotify(reinterpret_cast(pNRR))) { // // There isn't a CComponent around, so this is just a property // sheet running outside of MMC. // pNRR->Release(); PromptForRestart(hDlg, DiFlags, IDS_REMOVEDEV_RESTART); } } else { // // We couldn't allocate memory to create our CNotifyRebootRequest // instance, so just prompt for a reboot in this thread. // PromptForRestart(hDlg, DiFlags, IDS_REMOVEDEV_RESTART); } if (Refresh) { pMachine->ScheduleRefresh(); } Return = TRUE; } pMachine->EnableRefresh(TRUE); return Return; } BOOL CDeviceDriverPage::LaunchTroubleShooter( CDevice* pDevice, HWND hDlg, BOOL *pfChanged ) { BOOL fChanged = FALSE; DWORD Status, Problem = 0; CProblemAgent* ProblemAgent; if (pDevice->GetStatus(&Status, &Problem) || pDevice->IsPhantom()) { // // if the device is a phantom device, use the CM_PROB_DEVICE_NOT_THERE // if (pDevice->IsPhantom()) { Problem = CM_PROB_DEVICE_NOT_THERE; Status = DN_HAS_PROBLEM; } // // if the device is not started and no problem is assigned to it // fake the problem number to be failed start. // if (!(Status & DN_STARTED) && !Problem && pDevice->IsRAW()) { Problem = CM_PROB_FAILED_START; } } ProblemAgent = new CProblemAgent(pDevice, Problem, FALSE); if (ProblemAgent) { fChanged = ProblemAgent->FixIt(GetParent(hDlg)); delete ProblemAgent; } if (pfChanged) { *pfChanged = fChanged; } return TRUE; } // // This function updates drivers for the given device // INPUT: // pDevice -- the object represent the device // hDlg -- the property page window handle // pfChanged -- optional buffer to receive if driver changes // have occured. // OUTPUT: // TRUE -- function succeeded. // FALSE -- function failed. // BOOL CDeviceDriverPage::RollbackDriver( CDevice* pDevice, HWND hDlg, BOOL *pfChanged, DWORD *pdwReboot ) { HCURSOR hCursorOld; BOOL RollbackSuccessful = FALSE; DWORD RollbackError = ERROR_CANCELLED; DWORD Status = 0, Problem = 0; // // Verify that the process has Admin credentials and that we are running on the local // machine. // if (!pDevice || !pDevice->m_pMachine->IsLocal() || !g_IsAdmin) { ASSERT(FALSE); return FALSE; } hCursorOld = SetCursor(LoadCursor(NULL, IDC_WAIT)); // // If the device has the DN_WILL_BE_REMOVED flag set and the user is // attempting to roll back the driver then we will prompt them for a // reboot and include text in the prompt that explains this device // is in the process of being removed. // if (pDevice->GetStatus(&Status, &Problem) && (Status & DN_WILL_BE_REMOVED)) { // // First try and send a MMCPropertyChangeNotify message to our // CComponent so that it can prompt for a reboot inside the // device manager thread instead of our thread. If this is not // done then the property sheet will hang around after device // manager has gone away...which will cause a "hung app" dialog // to appear. // CNotifyRebootRequest* pNRR = new CNotifyRebootRequest(m_hDlg, DI_NEEDRESTART, IDS_WILL_BE_REMOVED_NO_ROLLBACK_DRIVER); if (pNRR) { if (!m_pDevice->m_psd.PropertyChangeNotify(reinterpret_cast(pNRR))) { // // There isn't a CComponent around, so this is just a property // sheet running outside of MMC. // pNRR->Release(); PromptForRestart(m_hDlg, DI_NEEDRESTART, IDS_WILL_BE_REMOVED_NO_ROLLBACK_DRIVER); } } else { // // We couldn't allocate memory to create our CNotifyRebootRequest // instance, so just prompt for a reboot in this thread. // PromptForRestart(m_hDlg, DI_NEEDRESTART, IDS_WILL_BE_REMOVED_NO_ROLLBACK_DRIVER); } return FALSE; } // // First check to see if there are any drivers to Rollback // CSafeRegistry regRollback; TCHAR RollbackSubkeyName[MAX_PATH + 1]; TCHAR ReinstallString[MAX_PATH]; BOOL bFoundMatch = FALSE; int index = 0; ReinstallString[0] = TEXT('\0'); RollbackSubkeyName[0] = TEXT('\0'); if (regRollback.Open(HKEY_LOCAL_MACHINE, TEXT("SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Reinstall"))) { DWORD SubkeySize = ARRAYLEN(RollbackSubkeyName); while (!bFoundMatch && regRollback.EnumerateSubkey(index, RollbackSubkeyName, &SubkeySize)) { CSafeRegistry regRollbackSubkey; if (regRollbackSubkey.Open(regRollback, RollbackSubkeyName)) { DWORD regType, cbSize; LPTSTR DeviceInstanceIds; cbSize = 0; if (regRollbackSubkey.GetValue(TEXT("DeviceInstanceIds"), ®Type, NULL, &cbSize)) { // // Allocate memory to hold the DeviceInstanceIds // DeviceInstanceIds = (LPTSTR)LocalAlloc(LPTR, cbSize); if (DeviceInstanceIds) { ZeroMemory(DeviceInstanceIds, cbSize); if (regRollbackSubkey.GetValue(TEXT("DeviceInstanceIds"), ®Type, (PBYTE)DeviceInstanceIds, &cbSize)) { // // Compare the list of DeviceInstanceIds in this registry key with this // devices DeviceInstanceId // for (LPTSTR p = DeviceInstanceIds; *p; p += (lstrlen(p) + 1)) { if (pDevice->GetDeviceID() && !lstrcmpi(p, pDevice->GetDeviceID())) { bFoundMatch = TRUE; cbSize = sizeof(ReinstallString); regRollbackSubkey.GetValue(TEXT("ReinstallString"), ®Type, (PBYTE)ReinstallString, &cbSize); break; } } } LocalFree(DeviceInstanceIds); } } } SubkeySize = ARRAYLEN(RollbackSubkeyName); index++; } } if (bFoundMatch) { // // Check the ReinstallString path to verify that a backup directory actually exists. // We first need to strip the INF name off of the end of the path. // PTSTR p; // // Assume that the directory does NOT exist // bFoundMatch = FALSE; if (ReinstallString[0] != TEXT('\0')) { p = StrRChr(ReinstallString, NULL, TEXT('\\')); if (p) { *p = 0; WIN32_FIND_DATA findData; HANDLE FindHandle; UINT OldMode; OldMode = SetErrorMode(SEM_FAILCRITICALERRORS); FindHandle = FindFirstFile(ReinstallString, &findData); if(FindHandle != INVALID_HANDLE_VALUE) { FindClose(FindHandle); bFoundMatch = TRUE; } else { // // The directory does not exist. Make sure we clean out the registry key // regRollback.DeleteSubkey(RollbackSubkeyName); } SetErrorMode(OldMode); } } } if (bFoundMatch) { // // We found a match, lets ask the user if they want to rollback the drivers // String strYesRollback; strYesRollback.LoadString(g_hInstance, IDS_DEVDRV_YESROLLBACK); if (MessageBox(hDlg, strYesRollback, pDevice->GetDisplayName(), MB_YESNO) == IDYES) { RollbackSuccessful = pDevice->m_pMachine->RollbackDriver(hDlg, RollbackSubkeyName, 0x3, pdwReboot); if (!RollbackSuccessful) { RollbackError = GetLastError(); } } } else { // // We could not find a drivers backup for this device. Lets ask the user if they want // to start the troubleshooter. // String strNoRollback; strNoRollback.LoadString(g_hInstance, IDS_DEVDRV_NOROLLBACK); if (MessageBox(hDlg, strNoRollback, pDevice->GetDisplayName(), MB_YESNO) == IDYES) { LaunchTroubleShooter(pDevice, hDlg, pfChanged); } } if (hCursorOld) { SetCursor(hCursorOld); } // // We will assume that something changed when we called InstallDevInst() // unless it returned FALSE and GetLastError() == ERROR_CANCELLED // if (pfChanged) { *pfChanged = TRUE; if (!bFoundMatch || (!RollbackSuccessful && (ERROR_CANCELLED == RollbackError))) { *pfChanged = FALSE; } } return TRUE; } // // This function updates drivers for the given device // INPUT: // pDevice -- the object represent the device // hDlg -- the property page window handle // pfChanged -- optional buffer to receive if driver changes // have occured. // OUTPUT: // TRUE -- function succeeded. // FALSE -- function failed. // BOOL CDeviceDriverPage::UpdateDriver( CDevice* pDevice, HWND hDlg, BOOL *pfChanged, DWORD *pdwReboot ) { BOOL Installed = FALSE; DWORD InstallError = ERROR_SUCCESS; DWORD Status = 0, Problem = 0; // // Must be an admin and on the local machine to update a device. // if (!pDevice || !pDevice->m_pMachine->IsLocal() || !g_IsAdmin) { ASSERT(FALSE); return FALSE; } // // If the device has the DN_WILL_BE_REMOVED flag set and the user is // attempting to update the driver then we will prompt them for a // reboot and include text in the prompt that explains this device // is in the process of being removed. // if (pDevice->GetStatus(&Status, &Problem) && (Status & DN_WILL_BE_REMOVED)) { // // First try and send a MMCPropertyChangeNotify message to our // CComponent so that it can prompt for a reboot inside the // device manager thread instead of our thread. If this is not // done then the property sheet will hang around after device // manager has gone away...which will cause a "hung app" dialog // to appear. // CNotifyRebootRequest* pNRR = new CNotifyRebootRequest(m_hDlg, DI_NEEDRESTART, IDS_WILL_BE_REMOVED_NO_UPDATE_DRIVER); if (pNRR) { if (!m_pDevice->m_psd.PropertyChangeNotify(reinterpret_cast(pNRR))) { // // There isn't a CComponent around, so this is just a property // sheet running outside of MMC. // pNRR->Release(); PromptForRestart(m_hDlg, DI_NEEDRESTART, IDS_WILL_BE_REMOVED_NO_UPDATE_DRIVER); } } else { // // We couldn't allocate memory to create our CNotifyRebootRequest // instance, so just prompt for a reboot in this thread. // PromptForRestart(m_hDlg, DI_NEEDRESTART, IDS_WILL_BE_REMOVED_NO_UPDATE_DRIVER); } return FALSE; } Installed = pDevice->m_pMachine->InstallDevInst(hDlg, pDevice->GetDeviceID(), TRUE, pdwReboot); if (!Installed) { InstallError = GetLastError(); } // // We will assume that something changed when we called InstallDevInst() // unless it returned FALSE and GetLastError() == ERROR_CANCELLED // if (pfChanged) { *pfChanged = TRUE; if (!Installed && (ERROR_CANCELLED == InstallError)) { *pfChanged = FALSE; } } return TRUE; } void CDeviceDriverPage::InitializeDriver() { if (m_pDriver) { delete m_pDriver; m_pDriver = NULL; } m_pDriver = m_pDevice->CreateDriver(); } void CDeviceDriverPage::UpdateControls( LPARAM lParam ) { if (lParam) { m_pDevice = (CDevice*) lParam; } try { // // Calling PropertyChanged() will update the display name for the device. We need // to do this in case a 3rd party property sheet did something that could change // the device's display name. // m_pDevice->PropertyChanged(); // // If we are not running locally then don't bother showing the driver // details since we can't get a list of the drivers and we can't get // any information about the driver. // if (!m_pDevice->m_pMachine->IsLocal()) { ::EnableWindow(GetControl(IDC_DEVDRV_DETAILS), FALSE); ::EnableWindow(GetControl(IDC_DEVDRV_DETAILS_TEXT), FALSE); } // // can not change driver on remote machine or the user // has no Administrator privilege. // if (!m_pDevice->m_pMachine->IsLocal() || !g_IsAdmin) { ::EnableWindow(GetControl(IDC_DEVDRV_CHANGEDRIVER), FALSE); ::EnableWindow(GetControl(IDC_DEVDRV_CHANGEDRIVER_TEXT), FALSE); ::EnableWindow(GetControl(IDC_DEVDRV_ROLLBACK), FALSE); ::EnableWindow(GetControl(IDC_DEVDRV_ROLLBACK_TEXT), FALSE); ::EnableWindow(GetControl(IDC_DEVDRV_UNINSTALL), FALSE); ::EnableWindow(GetControl(IDC_DEVDRV_UNINSTALL_TEXT), FALSE); } // // Hide the uninstall button if the device cannot be uninstalled // else if (!m_pDevice->IsUninstallable()) { ::EnableWindow(GetControl(IDC_DEVDRV_UNINSTALL), FALSE); ::EnableWindow(GetControl(IDC_DEVDRV_UNINSTALL_TEXT), FALSE); } HICON hIconOld; m_IDCicon = IDC_DEVDRV_ICON; // Save for cleanup in OnDestroy. hIconOld = (HICON)SendDlgItemMessage(m_hDlg, IDC_DEVDRV_ICON, STM_SETICON, (WPARAM)(m_pDevice->LoadClassIcon()), 0 ); if (hIconOld) { DestroyIcon(hIconOld); } InitializeDriver(); SetDlgItemText(m_hDlg, IDC_DEVDRV_DESC, m_pDevice->GetDisplayName()); String strDriverProvider, strDriverDate, strDriverVersion, strDriverSigner; m_pDevice->GetProviderString(strDriverProvider); SetDlgItemText(m_hDlg, IDC_DEVDRV_DRIVERPROVIDER, strDriverProvider); m_pDevice->GetDriverDateString(strDriverDate); SetDlgItemText(m_hDlg, IDC_DEVDRV_DRIVERDATE, strDriverDate); m_pDevice->GetDriverVersionString(strDriverVersion); SetDlgItemText(m_hDlg, IDC_DEVDRV_DRIVERVERSION, strDriverVersion); if (m_pDriver) { m_pDriver->GetDriverSignerString(strDriverSigner); } // // If we could not get a digital signature string or then just display // Unknown for the Digital Signer. // if (strDriverSigner.IsEmpty()) { strDriverSigner.LoadString(g_hInstance, IDS_UNKNOWN); } SetDlgItemText(m_hDlg, IDC_DEVDRV_DRIVERSIGNER, strDriverSigner); AddToolTips(m_hDlg, IDC_DEVDRV_DRIVERSIGNER, (LPTSTR)strDriverSigner, &m_hwndDigitalSignerTip); } catch (CMemoryException* e) { e->Delete(); // report memory error MsgBoxParam(m_hDlg, 0, 0, 0); } } BOOL CDeviceDriverPage::OnHelp( LPHELPINFO pHelpInfo ) { WinHelp((HWND)pHelpInfo->hItemHandle, DEVMGR_HELP_FILE_NAME, HELP_WM_HELP, (ULONG_PTR)g_a106HelpIDs); return FALSE; } BOOL CDeviceDriverPage::OnContextMenu( HWND hWnd, WORD xPos, WORD yPos ) { UNREFERENCED_PARAMETER(xPos); UNREFERENCED_PARAMETER(yPos); WinHelp(hWnd, DEVMGR_HELP_FILE_NAME, HELP_CONTEXTMENU, (ULONG_PTR)g_a106HelpIDs); return FALSE; } ////////////////////////////////////////////////////////////////////////////////////////// // // Driver Files // const DWORD g_a110HelpIDs[]= { IDC_DRIVERFILES_ICON, IDH_DISABLEHELP, // Driver: "" (Icon) IDC_DRIVERFILES_DESC, IDH_DISABLEHELP, // Driver: "" (Static) IDC_DRIVERFILES_FILES, IDH_DISABLEHELP, // Driver: "Provider:" (Static) IDC_DRIVERFILES_FILELIST, idh_devmgr_driver_driver_files, // Driver: "" (ListBox) IDC_DRIVERFILES_TITLE_PROVIDER, idh_devmgr_driver_provider, // Driver: "Provider:" (Static) IDC_DRIVERFILES_PROVIDER, idh_devmgr_driver_provider, // Driver: "" (Static) IDC_DRIVERFILES_TITLE_COPYRIGHT,idh_devmgr_driver_copyright, // Driver: "Copyright:" (Static) IDC_DRIVERFILES_COPYRIGHT, idh_devmgr_driver_copyright, // Driver: "" (Static) IDC_DRIVERFILES_TITLE_DIGITALSIGNER, IDH_DISABLEHELP, IDC_DRIVERFILES_DIGITALSIGNER, IDH_DISABLEHELP, IDC_DRIVERFILES_TITLE_VERSION, idh_devmgr_driver_file_version, // Driver: "Version:" (Static) IDC_DRIVERFILES_VERSION, idh_devmgr_driver_file_version, // Driver: "File Version:" (Static) 0, 0 }; BOOL CDriverFilesDlg::OnInitDialog() { int SignedIndex = 0, BlankIndex = 0, DriverBlockIndex = 0, CertIndex = 0; try { HICON hIcon; hIcon = (HICON)SendDlgItemMessage(m_hDlg, IDC_DRIVERFILES_ICON, STM_SETICON, (WPARAM)(m_pDevice->LoadClassIcon()), 0); if (hIcon) { DestroyIcon(hIcon); } SetDlgItemText(m_hDlg, IDC_DRIVERFILES_DESC, m_pDevice->GetDisplayName()); // // Create the ImageList that contains the signed and blank images. // // NOTE: On BiDi builds we need to set the ILC_MIRROR flag so that the // signed/unsigned icons are not mirrored. // m_ImageList = ImageList_Create(GetSystemMetrics(SM_CXSMICON), GetSystemMetrics(SM_CYSMICON), ILC_MASK | ((GetWindowLong(GetParent(m_hDlg), GWL_EXSTYLE) & WS_EX_LAYOUTRTL) ? ILC_MIRROR : 0), 1, 1 ); if (m_ImageList) { ImageList_SetBkColor(m_ImageList, GetSysColor(COLOR_WINDOW)); if ((hIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_BLANK))) != NULL) { BlankIndex = ImageList_AddIcon(m_ImageList, hIcon); DestroyIcon(hIcon); } if ((hIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_SIGNED))) != NULL) { SignedIndex = ImageList_AddIcon(m_ImageList, hIcon); DestroyIcon(hIcon); } if ((hIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_CERT))) != NULL) { CertIndex = ImageList_AddIcon(m_ImageList, hIcon); DestroyIcon(hIcon); } if ((hIcon = LoadIcon(g_hInstance, MAKEINTRESOURCE(IDI_DRIVERBLOCK))) != NULL) { DriverBlockIndex = ImageList_AddIcon(m_ImageList, hIcon); DestroyIcon(hIcon); } } // //Initialize the list of drivers // LV_COLUMN lvcCol; LV_ITEM lviItem; CDriverFile* pDrvFile; PVOID Context; HWND hwndFileList = GetDlgItem(m_hDlg, IDC_DRIVERFILES_FILELIST); // // Insert a single column for this list. // lvcCol.mask = LVCF_FMT | LVCF_WIDTH; lvcCol.fmt = LVCFMT_LEFT; lvcCol.iSubItem = 0; ListView_InsertColumn(hwndFileList, 0, (LV_COLUMN FAR *)&lvcCol); ListView_SetExtendedListViewStyle(hwndFileList, LVS_EX_FULLROWSELECT | LVS_EX_LABELTIP); if (m_ImageList) { ListView_SetImageList(hwndFileList, m_ImageList, LVSIL_SMALL); } ListView_DeleteAllItems(hwndFileList); if (m_pDriver && m_pDriver->GetFirstDriverFile(&pDrvFile, Context)) { do { ASSERT(pDrvFile); LPCTSTR pFullPathName; pFullPathName = pDrvFile->GetFullPathName(); if (pFullPathName) { lviItem.mask = LVIF_TEXT | LVIF_PARAM | LVIF_IMAGE; lviItem.iSubItem = 0; lviItem.lParam = (LPARAM)pDrvFile; lviItem.iItem = ListView_GetItemCount(hwndFileList); lviItem.pszText = (LPTSTR)pFullPathName; if (m_ImageList) { if (pDrvFile->IsDriverBlocked()) { // // The driver is blocked // lviItem.iImage = DriverBlockIndex; } else if (pDrvFile->GetWin32Error() == NO_ERROR) { // // The driver is WHQL signed. // lviItem.iImage = SignedIndex; } else if ((pDrvFile->GetWin32Error() == ERROR_AUTHENTICODE_TRUSTED_PUBLISHER) || (pDrvFile->GetWin32Error() == ERROR_AUTHENTICODE_TRUST_NOT_ESTABLISHED)) { // // The driver is Authenticode signed. // lviItem.iImage = CertIndex; } else { // // The driver is NOT blocked and NOT signed. // lviItem.iImage = BlankIndex; } } ListView_InsertItem(hwndFileList, &lviItem); } } while (m_pDriver->GetNextDriverFile(&pDrvFile, Context)); if (ListView_GetItemCount(hwndFileList) >= 1) { ListView_SetItemState(hwndFileList, 0, LVIS_SELECTED | LVIS_FOCUSED, LVIS_SELECTED | LVIS_FOCUSED ); ListView_EnsureVisible(hwndFileList, 0, FALSE); ListView_SetColumnWidth(hwndFileList, 0, LVSCW_AUTOSIZE_USEHEADER); ShowCurDriverFileDetail(); } else { // // nothing on the driver file list, disable it // ::EnableWindow(GetControl(IDC_DRIVERFILES_FILELIST), FALSE); } } } catch (CMemoryException* e) { e->Delete(); return FALSE; } return TRUE; } void CDriverFilesDlg::OnCommand( WPARAM wParam, LPARAM lParam ) { UNREFERENCED_PARAMETER(lParam); if (BN_CLICKED == HIWORD(wParam)) { if (IDOK == LOWORD(wParam)) { EndDialog(m_hDlg, IDOK); } else if (IDCANCEL == LOWORD(wParam)) { EndDialog(m_hDlg, IDCANCEL); } } } BOOL CDriverFilesDlg::OnNotify( LPNMHDR pnmhdr ) { switch (pnmhdr->code) { case LVN_ITEMCHANGED: ShowCurDriverFileDetail(); break; case NM_RETURN: case NM_CLICK: if (pnmhdr->idFrom == IDS_DRIVERFILES_BLOCKDRIVERLINK) { LaunchHelpForBlockedDriver(); } break; } return FALSE; } BOOL CDriverFilesDlg::OnDestroy() { HICON hIcon; hIcon = (HICON)SendDlgItemMessage(m_hDlg, IDC_DRIVERFILES_ICON, STM_GETICON, 0, 0); if (hIcon) { DestroyIcon(hIcon); } if (m_ImageList) { ImageList_Destroy(m_ImageList); } return FALSE; } BOOL CDriverFilesDlg::OnHelp( LPHELPINFO pHelpInfo ) { WinHelp((HWND)pHelpInfo->hItemHandle, DEVMGR_HELP_FILE_NAME, HELP_WM_HELP, (ULONG_PTR)g_a110HelpIDs); return FALSE; } BOOL CDriverFilesDlg::OnContextMenu( HWND hWnd, WORD xPos, WORD yPos ) { UNREFERENCED_PARAMETER(xPos); UNREFERENCED_PARAMETER(yPos); WinHelp(hWnd, DEVMGR_HELP_FILE_NAME, HELP_CONTEXTMENU, (ULONG_PTR)g_a110HelpIDs); return FALSE; } void CDriverFilesDlg::ShowCurDriverFileDetail() { HWND hwndFileList = GetDlgItem(m_hDlg, IDC_DRIVERFILES_FILELIST); LVITEM lvItem; LPCTSTR pString; lvItem.mask = LVIF_PARAM; lvItem.iSubItem = 0; lvItem.iItem = ListView_GetNextItem(hwndFileList, -1, LVNI_SELECTED ); if (lvItem.iItem != -1) { try { ListView_GetItem(hwndFileList, &lvItem); CDriverFile* pDrvFile = (CDriverFile*)lvItem.lParam; ASSERT(pDrvFile); TCHAR TempString[LINE_LEN]; LPCTSTR pFullPathName; pFullPathName = pDrvFile->GetFullPathName(); if (!pFullPathName || (pDrvFile->GetAttributes() == 0xFFFFFFFF)) { // // This is the case when the file is not present on the system // LoadResourceString(IDS_NOT_PRESENT, TempString, ARRAYLEN(TempString)); SetDlgItemText(m_hDlg, IDC_DRIVERFILES_VERSION, TempString); SetDlgItemText(m_hDlg, IDC_DRIVERFILES_PROVIDER, TempString); SetDlgItemText(m_hDlg, IDC_DRIVERFILES_COPYRIGHT, TempString); SetDlgItemText(m_hDlg, IDC_DRIVERFILES_DIGITALSIGNER, TempString); ShowWindow(GetControl(IDS_DRIVERFILES_BLOCKDRIVERLINK), FALSE); } else { if (!pDrvFile->HasVersionInfo()) { // // This is the case when the file is present but it does not have // version information. // LoadResourceString(IDS_UNKNOWN, TempString, ARRAYLEN(TempString)); SetDlgItemText(m_hDlg, IDC_DRIVERFILES_VERSION, TempString); SetDlgItemText(m_hDlg, IDC_DRIVERFILES_PROVIDER, TempString); SetDlgItemText(m_hDlg, IDC_DRIVERFILES_COPYRIGHT, TempString); } else { // // Show the file version information. // TempString[0] = _T('\0'); pString = pDrvFile->GetVersion(); if (!pString && _T('\0') == TempString[0]) { LoadResourceString(IDS_NOT_AVAILABLE, TempString, ARRAYLEN(TempString)); pString = TempString; } SetDlgItemText(m_hDlg, IDC_DRIVERFILES_VERSION, (LPTSTR)pString); pString = pDrvFile->GetProvider(); if (!pString && _T('\0') == TempString[0]) { LoadResourceString(IDS_NOT_AVAILABLE, TempString, ARRAYLEN(TempString)); pString = TempString; } SetDlgItemText(m_hDlg, IDC_DRIVERFILES_PROVIDER, (LPTSTR)pString); pString = pDrvFile->GetCopyright(); if (!pString && _T('\0') == TempString[0]) { LoadResourceString(IDS_NOT_AVAILABLE, TempString, ARRAYLEN(TempString)); pString = TempString; } SetDlgItemText(m_hDlg, IDC_DRIVERFILES_COPYRIGHT, (LPTSTR)pString); } // // Show the digital signer if the file is signed. // pString = pDrvFile->GetInfDigitalSigner(); if (!pString) { TempString[0] = _T('\0'); LoadResourceString(((pDrvFile->GetWin32Error() != 0) && (pDrvFile->GetWin32Error() != 0xFFFFFFFF) && (pDrvFile->GetWin32Error() != ERROR_DRIVER_BLOCKED)) ? IDS_NO_DIGITALSIGNATURE : IDS_NOT_AVAILABLE, TempString, ARRAYLEN(TempString)); pString = TempString; } SetDlgItemText(m_hDlg, IDC_DRIVERFILES_DIGITALSIGNER, (LPTSTR)pString); // // Show the block driver link if this driver is blocked and it // has a block driver html help ID. // ShowWindow(GetControl(IDS_DRIVERFILES_BLOCKDRIVERLINK), pDrvFile->GetBlockedDriverHtmlHelpID() ? TRUE : FALSE); } } catch (CMemoryException* e) { e->Delete(); // report memory error MsgBoxParam(m_hDlg, 0, 0, 0); } } else { // no selection SetDlgItemText(m_hDlg, IDC_DRIVERFILES_VERSION, TEXT("")); SetDlgItemText(m_hDlg, IDC_DRIVERFILES_PROVIDER, TEXT("")); SetDlgItemText(m_hDlg, IDC_DRIVERFILES_COPYRIGHT, TEXT("")); SetDlgItemText(m_hDlg, IDC_DRIVERFILES_DIGITALSIGNER, TEXT("")); } } void CDriverFilesDlg::LaunchHelpForBlockedDriver() { HWND hwndFileList = GetDlgItem(m_hDlg, IDC_DRIVERFILES_FILELIST); LVITEM lvItem; LPCTSTR pHtmlHelpID; String strHcpLink; lvItem.mask = LVIF_PARAM; lvItem.iSubItem = 0; lvItem.iItem = ListView_GetNextItem(hwndFileList, -1, LVNI_SELECTED ); if (lvItem.iItem != -1) { try { ListView_GetItem(hwndFileList, &lvItem); CDriverFile* pDrvFile = (CDriverFile*)lvItem.lParam; ASSERT(pDrvFile); if ((pHtmlHelpID = pDrvFile->GetBlockedDriverHtmlHelpID()) != NULL) { strHcpLink.Format(TEXT("HELPCTR.EXE -url %s"), pHtmlHelpID); ShellExecute(m_hDlg, TEXT("open"), TEXT("HELPCTR.EXE"), (LPTSTR)strHcpLink, NULL, SW_SHOWNORMAL); } } catch (CMemoryException* e) { e->Delete(); // report memory error MsgBoxParam(m_hDlg, 0, 0, 0); } } }