mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
4701 lines
143 KiB
4701 lines
143 KiB
/*++
|
|
|
|
Copyright (c) 1995-2000 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
devres.c
|
|
|
|
Abstract:
|
|
|
|
Routines for displaying resource dialogs.
|
|
|
|
Author:
|
|
|
|
Paula Tomlinson (paulat) 7-Feb-1996
|
|
|
|
Revision History:
|
|
|
|
Jamie Hunter (jamiehun) 19-Mar-1998
|
|
|
|
--*/
|
|
|
|
#include "precomp.h"
|
|
#pragma hdrstop
|
|
|
|
//
|
|
// Private Prototypes
|
|
//
|
|
|
|
|
|
//
|
|
// Global Data
|
|
//
|
|
|
|
static INTERFACE_TYPE ResourcePickerReadOnlyInterfaces[] = {
|
|
//
|
|
// List of interface-types that we don't want user to edit properties of
|
|
//
|
|
|
|
PCIBus,
|
|
|
|
//
|
|
// End of list
|
|
//
|
|
InterfaceTypeUndefined
|
|
};
|
|
|
|
static const BOOL ResTypeEditable[ResType_MAX+1] = {
|
|
//
|
|
// lists resource types that are shown and are editable
|
|
FALSE, // ResType_None
|
|
TRUE, // ResType_Mem
|
|
TRUE, // ResType_IO
|
|
TRUE, // ResType_DMA
|
|
TRUE, // ResType_IRQ
|
|
FALSE, // ResType_DoNotUse
|
|
FALSE // ResType_BusNumber
|
|
};
|
|
|
|
#if (ResType_MAX+1) != 7
|
|
#error Fix SetupAPI devres.c, ResType_MAX has changed
|
|
#endif
|
|
|
|
//
|
|
// HELP ID's
|
|
//
|
|
static const DWORD DevResHelpIDs[]=
|
|
{
|
|
IDC_DEVRES_ICON, IDH_NOHELP, // "Low (%d)" (Static)
|
|
IDC_DEVRES_DEVDESC, IDH_NOHELP,
|
|
IDC_DEVRES_SETTINGSTATE, IDH_DEVMGR_RESOURCES_SETTINGS,
|
|
IDC_DEVRES_SETTINGSLIST, IDH_DEVMGR_RESOURCES_SETTINGS,
|
|
IDC_DEVRES_LCTEXT, IDH_DEVMGR_RESOURCES_BASEDON,
|
|
IDC_DEVRES_LOGCONFIGLIST, IDH_DEVMGR_RESOURCES_BASEDON,
|
|
IDC_DEVRES_CHANGE, IDH_DEVMGR_RESOURCES_CHANGE,
|
|
IDC_DEVRES_USESYSSETTINGS, IDH_DEVMGR_RESOURCES_AUTO,
|
|
IDC_DEVRES_CONFLICTDEVTEXT, IDH_DEVMGR_RESOURCES_CONFLICTS,
|
|
IDC_DEVRES_CONFLICTINFOLIST, IDH_DEVMGR_RESOURCES_CONFLICTS,
|
|
IDC_DEVRES_MFPARENT, IDH_DEVMGR_RESOURCES_PARENT,
|
|
IDC_DEVRES_MFPARENT_DESC, IDH_DEVMGR_RESOURCES_PARENT,
|
|
IDC_DEVRES_MAKEFORCED, IDH_DEVMGR_RESOURCES_SETMANUALLY,
|
|
0, 0
|
|
};
|
|
|
|
//
|
|
// HACKHACK (jamiehun)
|
|
// after we've changed UI from a MakeForced, we post this message to get back control of keyboard
|
|
//
|
|
|
|
#define WM_USER_FOCUS (WM_USER+101)
|
|
|
|
|
|
//
|
|
// API to obtain a resource-picker page
|
|
//
|
|
|
|
HPROPSHEETPAGE
|
|
GetResourceSelectionPage(
|
|
IN HDEVINFO DeviceInfoSet,
|
|
IN PSP_DEVINFO_DATA DeviceInfoData
|
|
)
|
|
{
|
|
LPDMPROP_DATA pdmData;
|
|
PROPSHEETPAGE PropPage;
|
|
|
|
//
|
|
// private data
|
|
// anything we "do" here must be "undone" in pResourcePickerPropPageCallback
|
|
//
|
|
pdmData = (LPDMPROP_DATA)MyMalloc(sizeof(DMPROP_DATA));
|
|
if (pdmData == NULL) {
|
|
return NULL;
|
|
}
|
|
ZeroMemory(pdmData,sizeof(DMPROP_DATA));
|
|
|
|
pdmData->hDevInfo = DeviceInfoSet;
|
|
pdmData->lpdi = DeviceInfoData;
|
|
|
|
//
|
|
// validate expectations
|
|
//
|
|
MYASSERT(pdmData->hDevInfo != NULL);
|
|
MYASSERT(pdmData->lpdi != NULL);
|
|
MYASSERT(pdmData->lpdi->DevInst != 0);
|
|
|
|
ZeroMemory(&PropPage,sizeof(PropPage));
|
|
|
|
//
|
|
// create the Resources Property Page
|
|
//
|
|
PropPage.dwSize = sizeof(PROPSHEETPAGE);
|
|
PropPage.dwFlags = PSP_DEFAULT | PSP_USECALLBACK;
|
|
PropPage.hInstance = MyDllModuleHandle;
|
|
PropPage.pszTemplate = MAKEINTRESOURCE(IDD_DEF_DEVRESOURCE_PROP);
|
|
PropPage.pszIcon = NULL;
|
|
PropPage.pszTitle = NULL;
|
|
PropPage.pfnDlgProc = pResourcePickerDlgProc;
|
|
PropPage.lParam = (LPARAM)pdmData;
|
|
PropPage.pfnCallback = pResourcePickerPropPageCallback;
|
|
#ifdef _UNICODE
|
|
PropPage.dwFlags |= PSP_USEFUSIONCONTEXT;
|
|
PropPage.hActCtx = NULL;
|
|
#endif
|
|
|
|
return CreatePropertySheetPage(&PropPage);
|
|
|
|
} // GetResourceSelectionPage
|
|
|
|
|
|
//
|
|
// CreatePropertySheetPage - callback function
|
|
//
|
|
UINT CALLBACK pResourcePickerPropPageCallback(
|
|
HWND hwnd,
|
|
UINT uMsg,
|
|
LPPROPSHEETPAGE ppsp
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Callback to handle cleanup of the property sheet
|
|
|
|
Arguments:
|
|
|
|
Standard PropSheetPageProc arguments.
|
|
|
|
Return Value:
|
|
|
|
Standard PropSheetPageProc return.
|
|
|
|
--*/
|
|
{
|
|
switch (uMsg) {
|
|
//case PSPCB_ADDREF:
|
|
// break;
|
|
|
|
case PSPCB_CREATE:
|
|
break;
|
|
|
|
case PSPCB_RELEASE:
|
|
//
|
|
// release the memory we've previously allocated, outside of the actual dialog
|
|
//
|
|
if (ppsp->lParam != 0) {
|
|
LPDMPROP_DATA pdmData = (LPDMPROP_DATA)(ppsp->lParam);
|
|
|
|
MyFree(pdmData);
|
|
}
|
|
break;
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
//
|
|
// Main dialog proceedure
|
|
//
|
|
|
|
|
|
INT_PTR
|
|
CALLBACK
|
|
pResourcePickerDlgProc(
|
|
HWND hDlg,
|
|
UINT message,
|
|
WPARAM wParam,
|
|
LPARAM lParam
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine provides the dialog box procedure for the main resource
|
|
picker property page. MEMPHIS COMPATIBLE.
|
|
|
|
Arguments:
|
|
|
|
Standard dialog box procedure arguments.
|
|
|
|
Return Value:
|
|
|
|
Standard dialog box procedure return.
|
|
|
|
--*/
|
|
|
|
{
|
|
LPDMPROP_DATA lpdmpd = NULL;
|
|
|
|
if (message == WM_INITDIALOG) {
|
|
lpdmpd = (LPDMPROP_DATA)((LPPROPSHEETPAGE)lParam)->lParam;
|
|
SetWindowLongPtr(hDlg, DWLP_USER, (LPARAM)lpdmpd);
|
|
} else {
|
|
lpdmpd = (LPDMPROP_DATA)GetWindowLongPtr(hDlg, DWLP_USER);
|
|
}
|
|
|
|
switch (message) {
|
|
|
|
//
|
|
// initialize
|
|
//
|
|
case WM_INITDIALOG: {
|
|
|
|
HICON hIcon = NULL;
|
|
int iIcon = 0, iIndex = 0;
|
|
ULONG ulSize;
|
|
PDEVICE_INFO_SET pDeviceInfoSet;
|
|
HMACHINE hMachine;
|
|
|
|
lpdmpd->himlResourceImages = NULL;
|
|
lpdmpd->CurrentLC = 0;
|
|
lpdmpd->CurrentLCType = 0;
|
|
lpdmpd->MatchingLC = 0;
|
|
lpdmpd->MatchingLCType = 0;
|
|
lpdmpd->SelectedLC = 0;
|
|
lpdmpd->SelectedLCType = 0;
|
|
lpdmpd->hDlg = hDlg;
|
|
lpdmpd->dwFlags = 0;
|
|
|
|
hMachine = pGetMachine(lpdmpd);
|
|
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_CHANGESSAVED; // Nothing to save yet
|
|
|
|
//
|
|
// NOTE: On Windows95, since lc info is in memory, they first
|
|
// call CM_Setup_DevNode with CM_SETUP_WRITE_LOG_CONFS flag so
|
|
// that in-memory lc data is flushed to the registry at this
|
|
// point.
|
|
//
|
|
|
|
//
|
|
// Init the Resource's image list.
|
|
//
|
|
lpdmpd->himlResourceImages = ImageList_Create(GetSystemMetrics(SM_CXSMICON),
|
|
GetSystemMetrics(SM_CYSMICON),
|
|
ILC_MASK, // | ILC_SHARED,
|
|
1,
|
|
1);
|
|
//
|
|
// add icons to image list
|
|
//
|
|
for (iIcon = IDI_RESOURCEFIRST;iIcon < IDI_RESOURCELAST;++iIcon) {
|
|
//
|
|
// resource icon
|
|
//
|
|
hIcon = LoadIcon(MyDllModuleHandle, MAKEINTRESOURCE(iIcon));
|
|
iIndex = ImageList_AddIcon(lpdmpd->himlResourceImages, hIcon);
|
|
}
|
|
|
|
for (iIcon = IDI_RESOURCEOVERLAYFIRST;iIcon <= IDI_RESOURCEOVERLAYLAST;++iIcon) {
|
|
//
|
|
// overlay icon
|
|
//
|
|
hIcon = LoadIcon(MyDllModuleHandle, MAKEINTRESOURCE(iIcon));
|
|
iIndex = ImageList_AddIcon(lpdmpd->himlResourceImages, hIcon);
|
|
|
|
//
|
|
// Tag this icon as an overlay icon (the first index is an
|
|
// index into the image list (specifies the icon), the
|
|
// second index is just an index to assign to each mask
|
|
// (starting with 1).
|
|
//
|
|
ImageList_SetOverlayImage(lpdmpd->himlResourceImages,
|
|
iIndex,
|
|
iIcon-IDI_RESOURCEOVERLAYFIRST+1);
|
|
}
|
|
|
|
if(pInitDevResourceDlg(lpdmpd)) {
|
|
lpdmpd->dwFlags &= ~DMPROP_FLAG_CHANGESSAVED; // need to save (prob because there was no config)
|
|
}
|
|
|
|
if (!(lpdmpd->dwFlags & DMPROP_FLAG_NO_RESOURCES)) {
|
|
pShowConflicts(lpdmpd);
|
|
}
|
|
if (GuiSetupInProgress) {
|
|
//
|
|
// occasionally legacy devices cause resource-picker popup during setup
|
|
// we do this here instead of create prop sheet, since I don't trust
|
|
// people to cleanup on fail. At least here is less risky
|
|
// clean this up in WM_DESTROY
|
|
//
|
|
lpdmpd->hDialogEvent = CreateEvent(NULL,TRUE,FALSE,SETUP_HAS_OPEN_DIALOG_EVENT);
|
|
if (lpdmpd->hDialogEvent) {
|
|
SetEvent(lpdmpd->hDialogEvent);
|
|
}
|
|
} else {
|
|
lpdmpd->hDialogEvent = NULL;
|
|
}
|
|
break;
|
|
}
|
|
|
|
|
|
//
|
|
// cleanup
|
|
//
|
|
case WM_DESTROY: {
|
|
|
|
HICON hIcon;
|
|
LOG_CONF LogConf;
|
|
LONG nItems, n;
|
|
HWND hList = GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST);
|
|
int Count, i;
|
|
|
|
//
|
|
// Clean up the ICON resource usage
|
|
//
|
|
if ((hIcon = (HICON)LOWORD(SendDlgItemMessage(hDlg,
|
|
IDC_DEVRES_ICON, STM_GETICON, 0, 0L)))) {
|
|
DestroyIcon(hIcon);
|
|
}
|
|
|
|
//
|
|
// free the LC handles that were saved in the combobox data
|
|
//
|
|
nItems = (LONG)SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST,
|
|
CB_GETCOUNT, 0, 0L);
|
|
|
|
for (n = 0; n < nItems ; n++) {
|
|
LogConf = (LOG_CONF)SendDlgItemMessage(hDlg,
|
|
IDC_DEVRES_LOGCONFIGLIST,
|
|
CB_GETITEMDATA, n, 0L);
|
|
CM_Free_Log_Conf_Handle(LogConf);
|
|
}
|
|
|
|
if (lpdmpd->CurrentLC != 0) {
|
|
CM_Free_Log_Conf_Handle(lpdmpd->CurrentLC);
|
|
}
|
|
|
|
ListView_DeleteAllItems(hList); // this will destroy all data
|
|
|
|
if (lpdmpd->himlResourceImages) {
|
|
ImageList_Destroy(lpdmpd->himlResourceImages);
|
|
}
|
|
|
|
if (lpdmpd->hDialogEvent) {
|
|
//
|
|
// we were holding up setup, now let setup proceed
|
|
//
|
|
ResetEvent(lpdmpd->hDialogEvent);
|
|
CloseHandle(lpdmpd->hDialogEvent);
|
|
lpdmpd->hDialogEvent = NULL;
|
|
}
|
|
// MyFree(lpdmpd); - do this in pResourcePickerPropPageCallback instead
|
|
break;
|
|
}
|
|
|
|
case WM_COMMAND:
|
|
//
|
|
// old-style controls
|
|
//
|
|
|
|
switch(LOWORD(wParam)) {
|
|
case IDC_DEVRES_USESYSSETTINGS: {
|
|
//
|
|
// consider resource settings to have changed
|
|
//
|
|
lpdmpd->dwFlags &= ~DMPROP_FLAG_CHANGESSAVED;
|
|
PropSheet_Changed(GetParent(hDlg), hDlg);
|
|
|
|
if (IsDlgButtonChecked(hDlg, (int)wParam)) {
|
|
//
|
|
// Revert back to allocated display, if any
|
|
//
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_USESYSSETTINGS;
|
|
pSelectLogConf(lpdmpd,(LOG_CONF)0,ALLOC_LOG_CONF,TRUE);
|
|
} else {
|
|
//
|
|
// Allow editing
|
|
//
|
|
lpdmpd->dwFlags &= ~DMPROP_FLAG_USESYSSETTINGS;
|
|
}
|
|
pShowUpdateEdit(lpdmpd); // update controls
|
|
|
|
break;
|
|
}
|
|
|
|
case IDC_DEVRES_LOGCONFIGLIST: {
|
|
//
|
|
// drop-down list action
|
|
//
|
|
switch (HIWORD(wParam)) {
|
|
case CBN_SELENDOK: {
|
|
ULONG ulIndex = 0;
|
|
int iItem;
|
|
LOG_CONF SelLC;
|
|
HWND hwndLC = GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST);
|
|
|
|
//
|
|
// If there is not a Log Config selected, then bail
|
|
//
|
|
iItem = (int)SendMessage(hwndLC, CB_GETCURSEL, 0, 0);
|
|
if(iItem != CB_ERR) {
|
|
SelLC = (LOG_CONF)SendMessage(hwndLC,CB_GETITEMDATA, (WPARAM)iItem,(LPARAM)0);
|
|
} else {
|
|
SelLC = (LOG_CONF)0;
|
|
}
|
|
if(SelLC != lpdmpd->SelectedLC) {
|
|
pSelectLogConf(lpdmpd,SelLC,lpdmpd->ConfigListLCType,FALSE);
|
|
}
|
|
//
|
|
// I prob don't need this here, but I'm playing safe!
|
|
//
|
|
lpdmpd->dwFlags &= ~DMPROP_FLAG_CHANGESSAVED;
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case IDC_DEVRES_CHANGE: {
|
|
//
|
|
// change selected setting
|
|
//
|
|
pChangeCurrentResSetting(lpdmpd);
|
|
break;
|
|
}
|
|
|
|
case IDC_DEVRES_MAKEFORCED: {
|
|
//
|
|
// possibly allow editing (after we've shown message)
|
|
// when we get here, always show a configuration
|
|
//
|
|
|
|
if(lpdmpd->dwFlags & DMPROP_FLAG_FORCEDONLY) {
|
|
lpdmpd->dwFlags &= ~DMPROP_FLAG_CHANGESSAVED; // need to save
|
|
}
|
|
pShowViewAllEdit(lpdmpd);
|
|
//
|
|
// select in the first available config to edit
|
|
//
|
|
pSelectLogConf(lpdmpd,(LOG_CONF)0,ALLOC_LOG_CONF,TRUE);
|
|
|
|
//
|
|
// ensure we have reasonable focus for accessability
|
|
//
|
|
PostMessage(hDlg,WM_USER_FOCUS,IDC_DEVRES_SETTINGSLIST,0);
|
|
break;
|
|
}
|
|
|
|
default:
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case WM_USER_FOCUS:
|
|
//
|
|
// change focus to DlgItem wParam
|
|
//
|
|
SetFocus(GetDlgItem(hDlg,(int)wParam));
|
|
return TRUE;
|
|
|
|
case WM_NOTIFY: {
|
|
//
|
|
// new controls & property codes
|
|
//
|
|
NMHDR * pHdr = (NMHDR*)lParam;
|
|
|
|
switch (pHdr->code) {
|
|
|
|
case PSN_SETACTIVE: {
|
|
|
|
HICON hIcon = NULL, hOldIcon = NULL;
|
|
TCHAR szString[MAX_PATH];
|
|
ULONG ulSize = 0;
|
|
HMACHINE hMachine;
|
|
|
|
hMachine = pGetMachine(lpdmpd);
|
|
|
|
|
|
//
|
|
// Set the ICON and device description
|
|
//
|
|
if (SetupDiLoadClassIcon(&lpdmpd->lpdi->ClassGuid, &hIcon, NULL)) {
|
|
|
|
if ((hOldIcon = (HICON)LOWORD(SendDlgItemMessage(hDlg, IDC_DEVRES_ICON,
|
|
STM_SETICON,
|
|
(WPARAM)hIcon, 0L)))) {
|
|
DestroyIcon(hOldIcon);
|
|
}
|
|
}
|
|
|
|
//
|
|
// First try to get the device's friendly name, then fall back to its description,
|
|
// and finally, use the "Unknown Device" description.
|
|
//
|
|
ulSize = MAX_PATH * sizeof(TCHAR);
|
|
if (CM_Get_DevInst_Registry_Property_Ex(lpdmpd->lpdi->DevInst,
|
|
CM_DRP_FRIENDLYNAME,
|
|
NULL, (LPBYTE)szString,
|
|
&ulSize, 0,hMachine) != CR_SUCCESS) {
|
|
|
|
ulSize = MAX_PATH * sizeof(TCHAR);
|
|
if (CM_Get_DevInst_Registry_Property_Ex(lpdmpd->lpdi->DevInst,
|
|
CM_DRP_DEVICEDESC,
|
|
NULL, (LPBYTE)szString,
|
|
&ulSize, 0,hMachine) != CR_SUCCESS) {
|
|
|
|
LoadString(MyDllModuleHandle, IDS_DEVNAME_UNK, szString, MAX_PATH);
|
|
}
|
|
}
|
|
SetDlgItemText(hDlg, IDC_DEVRES_DEVDESC, szString);
|
|
break;
|
|
}
|
|
|
|
case PSN_APPLY: {
|
|
BOOL bRet = FALSE;
|
|
//
|
|
// If there were Changes and they haven't been saved,
|
|
// then save them.
|
|
// consider some special cases as "haven't been saved"
|
|
//
|
|
if((lpdmpd->CurrentLC == 0) && (lpdmpd->dwFlags&DMPROP_FLAG_FIXEDCONFIG)) {
|
|
lpdmpd->dwFlags &= ~DMPROP_FLAG_CHANGESSAVED;
|
|
}
|
|
|
|
switch(pOkToSave(lpdmpd)) {
|
|
case IDNO:
|
|
//
|
|
// proceed without saving
|
|
//
|
|
bRet = TRUE;
|
|
break;
|
|
case IDCANCEL:
|
|
//
|
|
// don't proceed
|
|
//
|
|
bRet = FALSE;
|
|
break;
|
|
case IDYES:
|
|
//
|
|
// proceed and save
|
|
//
|
|
bRet = pSaveDevResSettings(lpdmpd);
|
|
#if 0
|
|
if (bRet) {
|
|
if ((lpdmpd->lpdi)->Flags & DI_NEEDREBOOT) {
|
|
PropSheet_RebootSystem(GetParent(hDlg));
|
|
} else if ((lpdmpd->lpdi)->Flags & DI_NEEDRESTART) {
|
|
PropSheet_RestartWindows(GetParent(hDlg));
|
|
}
|
|
#endif
|
|
if (bRet) {
|
|
//
|
|
// This page doesn't support roll-back, if we saved
|
|
// something then we're committed, disable the cancel
|
|
// botton.
|
|
//
|
|
PropSheet_CancelToClose(GetParent(hDlg));
|
|
}
|
|
break;
|
|
default:
|
|
MYASSERT(FALSE /* pOkToSave returned invalid value */);
|
|
bRet = FALSE;
|
|
break;
|
|
}
|
|
|
|
SetWindowLongPtr(hDlg, DWLP_MSGRESULT, bRet ? PSNRET_NOERROR : PSNRET_INVALID_NOCHANGEPAGE);
|
|
return TRUE;
|
|
}
|
|
|
|
case LVN_DELETEALLITEMS:
|
|
if (pHdr->idFrom == IDC_DEVRES_SETTINGSLIST) {
|
|
return FALSE; // we want LVN_DELETEITEM messages
|
|
}
|
|
break;
|
|
|
|
case LVN_DELETEITEM: {
|
|
LPNMLISTVIEW pListView = (LPNMLISTVIEW)pHdr;
|
|
if (pHdr->idFrom == IDC_DEVRES_SETTINGSLIST) {
|
|
PITEMDATA pItemData = (PITEMDATA)(LPVOID)(pListView->lParam);
|
|
//
|
|
// when an item is deleted, destroy associated data
|
|
//
|
|
if (pItemData->MatchingResDes) {
|
|
CM_Free_Res_Des_Handle(pItemData->MatchingResDes);
|
|
}
|
|
MyFree(pItemData);
|
|
}
|
|
break;
|
|
}
|
|
EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_CHANGE), FALSE);
|
|
break;
|
|
|
|
case LVN_ITEMCHANGED:
|
|
//
|
|
// If the item change is comming from the resource
|
|
// list, and there is a logconfig to be edited:
|
|
//
|
|
if (pHdr->idFrom == IDC_DEVRES_SETTINGSLIST) {
|
|
//
|
|
// see if we should enable resource change
|
|
//
|
|
pCheckEnableResourceChange(lpdmpd);
|
|
}
|
|
break;
|
|
|
|
case NM_DBLCLK:
|
|
//
|
|
// If the double click is from the SETTINGS list
|
|
// AND the DEVRES_CHANGE button is enabled, then
|
|
// allow the change.
|
|
//
|
|
if (pHdr->idFrom == IDC_DEVRES_SETTINGSLIST) {
|
|
//
|
|
// this routine should check that we can change settings
|
|
//
|
|
pChangeCurrentResSetting(lpdmpd);
|
|
}
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case WM_SYSCOLORCHANGE: {
|
|
|
|
HWND hChildWnd = GetWindow(hDlg, GW_CHILD);
|
|
|
|
while (hChildWnd != NULL) {
|
|
SendMessage(hChildWnd, WM_SYSCOLORCHANGE, wParam, lParam);
|
|
hChildWnd = GetWindow(hChildWnd, GW_HWNDNEXT);
|
|
}
|
|
break;
|
|
}
|
|
|
|
case WM_HELP: // F1
|
|
WinHelp(((LPHELPINFO)lParam)->hItemHandle, DEVRES_HELP, HELP_WM_HELP, (ULONG_PTR)DevResHelpIDs);
|
|
break;
|
|
|
|
case WM_CONTEXTMENU: // right mouse click
|
|
WinHelp((HWND)wParam, DEVRES_HELP, HELP_CONTEXTMENU, (ULONG_PTR)DevResHelpIDs);
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
//
|
|
// Helper functions
|
|
//
|
|
|
|
HMACHINE
|
|
pGetMachine(
|
|
LPDMPROP_DATA lpdmpd
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Retrieve Machine Handle
|
|
|
|
Arguments:
|
|
|
|
lpdmpd - Property Data
|
|
|
|
Return Value:
|
|
|
|
handle
|
|
|
|
--*/
|
|
{
|
|
HMACHINE hMachine;
|
|
PDEVICE_INFO_SET pDeviceInfoSet;
|
|
|
|
if(!(pDeviceInfoSet = AccessDeviceInfoSet(lpdmpd->hDevInfo))) {
|
|
return NULL;
|
|
}
|
|
hMachine = pDeviceInfoSet->hMachine;
|
|
UnlockDeviceInfoSet(pDeviceInfoSet);
|
|
return hMachine;
|
|
}
|
|
|
|
BOOL
|
|
pInitDevResourceDlg(
|
|
LPDMPROP_DATA lpdmpd
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine intializes the main resource picker property page.
|
|
MEMPHIS COMPATIBLE.
|
|
|
|
Arguments:
|
|
|
|
|
|
Return Value:
|
|
|
|
TRUE if "not saved"
|
|
|
|
--*/
|
|
|
|
{
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
CONFIGRET Status = CR_SUCCESS;
|
|
BOOL bHasCurrent = FALSE;
|
|
BOOL bShowCurrent = FALSE;
|
|
BOOL bHasForced = FALSE;
|
|
BOOL bNoForcedConfig = FALSE;
|
|
BOOL bNeedsForcedConfig = FALSE;
|
|
BOOL bHasConfigList = FALSE;
|
|
LV_COLUMN LvCol;
|
|
HWND hWndList = NULL;
|
|
TCHAR szString[MAX_PATH], szTemp[MAX_PATH], szConfigType[MAX_PATH],
|
|
szConfig[MAX_PATH];
|
|
ULONG ulIndex = 0, ulSize = 0, DevStatus = 0, DevProblem = 0;
|
|
DWORD BusType = (DWORD)(-1);
|
|
LOG_CONF LogConf;
|
|
DWORD dwPriority = 0;
|
|
WORD wItem;
|
|
ULONG ConfigFlags;
|
|
HMACHINE hMachine = NULL;
|
|
PDEVICE_INFO_SET pDeviceInfoSet;
|
|
int iIndex;
|
|
BOOL bHasPrivs = FALSE;
|
|
//
|
|
// Set initial control states
|
|
//
|
|
pHideAllControls(lpdmpd);
|
|
|
|
//
|
|
// determine priv token
|
|
// security checks are visual only
|
|
// real security checks are done in umpnpmgr
|
|
//
|
|
|
|
bHasPrivs = pSetupDoesUserHavePrivilege(SE_LOAD_DRIVER_NAME);
|
|
|
|
hMachine = pGetMachine(lpdmpd);
|
|
|
|
//
|
|
// retrieves current configuration, if any
|
|
//
|
|
bHasCurrent = pGetCurrentConfig(lpdmpd);
|
|
|
|
//
|
|
// We sometimes get called to show this page even if the device
|
|
// doesn't consume any resources. Check for that case and if so, just
|
|
// display an informational message and disable everything else.
|
|
//
|
|
|
|
if (!pDevRequiresResources(lpdmpd->lpdi->DevInst,hMachine)) {
|
|
|
|
//
|
|
// This device has no resources
|
|
//
|
|
pShowViewNoResources(lpdmpd);
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_NO_RESOURCES;
|
|
goto Final;
|
|
}
|
|
|
|
//
|
|
// Initialize the ListView control
|
|
//
|
|
hWndList = GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST);
|
|
LvCol.mask = LVCF_TEXT;
|
|
|
|
if (LoadString(MyDllModuleHandle, IDS_RESOURCETYPE, szString, MAX_PATH)) {
|
|
LvCol.pszText = (LPTSTR)szString;
|
|
ListView_InsertColumn(hWndList, 0, (LV_COLUMN FAR *)&LvCol);
|
|
}
|
|
|
|
if (LoadString(MyDllModuleHandle, IDS_RESOURCESETTING, szString, MAX_PATH)) {
|
|
LvCol.pszText = (LPTSTR)szString;
|
|
ListView_InsertColumn(hWndList, 1, (LV_COLUMN FAR *)&LvCol);
|
|
}
|
|
|
|
ListView_SetImageList(hWndList,lpdmpd->himlResourceImages, LVSIL_SMALL);
|
|
//
|
|
// Get DevStatus & DevProblem here, we may use this info further down
|
|
//
|
|
if (CM_Get_DevNode_Status_Ex(&DevStatus, &DevProblem, lpdmpd->lpdi->DevInst,
|
|
0,hMachine) != CR_SUCCESS) {
|
|
//
|
|
// we should never get here, show this as a problem
|
|
//
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_HASPROBLEM;
|
|
} else if (DevStatus & DN_HAS_PROBLEM) {
|
|
//
|
|
// cache problem flag away
|
|
//
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_HASPROBLEM;
|
|
} else if (DevStatus & DN_PRIVATE_PROBLEM) {
|
|
//
|
|
// driver indicates problem
|
|
// for now, do same as above
|
|
//
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_HASPROBLEM;
|
|
}
|
|
|
|
if (bIsMultiFunctionChild(lpdmpd->lpdi,hMachine)) {
|
|
//
|
|
// If this is a MultiFunction Child, disable all change controls, put up
|
|
// special text, and show the alloc config
|
|
//
|
|
pShowViewMFReadOnly(lpdmpd,FALSE);
|
|
goto Final;
|
|
}
|
|
|
|
//
|
|
// begin with read-only view, assuming settings are system
|
|
//
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_USESYSSETTINGS;
|
|
|
|
if (CM_Get_First_Log_Conf_Ex(NULL,
|
|
lpdmpd->lpdi->DevInst,
|
|
FORCED_LOG_CONF,
|
|
hMachine) == CR_SUCCESS) {
|
|
//
|
|
// the user currently has a forced config
|
|
//
|
|
lpdmpd->dwFlags &= ~DMPROP_FLAG_USESYSSETTINGS;
|
|
bHasForced = TRUE;
|
|
}
|
|
|
|
bShowCurrent = pShowViewReadOnly(lpdmpd,bHasPrivs);
|
|
if (!bHasPrivs || hMachine) {
|
|
//
|
|
// if we don't have enough priv's
|
|
// or we're displaying resources of a remote machine
|
|
// bottle out here
|
|
// we'll either be displaying current resources
|
|
// or displaying a problem
|
|
//
|
|
goto Final;
|
|
}
|
|
if(!bHasForced) {
|
|
//
|
|
// Check bus we're using
|
|
// to see if it's one of the read-only displays
|
|
//
|
|
ulSize = sizeof(BusType);
|
|
if (CM_Get_DevInst_Registry_Property_Ex(lpdmpd->lpdi->DevInst,
|
|
CM_DRP_LEGACYBUSTYPE,
|
|
NULL, (LPBYTE)&BusType,
|
|
&ulSize, 0,hMachine) != CR_SUCCESS) {
|
|
BusType = (DWORD)InterfaceTypeUndefined;
|
|
}
|
|
|
|
if (BusType != (DWORD)InterfaceTypeUndefined) {
|
|
int InterfaceItem;
|
|
|
|
for(InterfaceItem = 0; ResourcePickerReadOnlyInterfaces[InterfaceItem] != InterfaceTypeUndefined; InterfaceItem++) {
|
|
if (BusType == (DWORD)ResourcePickerReadOnlyInterfaces[InterfaceItem]) {
|
|
//
|
|
// Bus is one that we do not allow forced configs
|
|
// we can skip all the funky code below
|
|
//
|
|
// this is a good thing for 64-bit PCI
|
|
//
|
|
goto Final;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Retrieve alternate configurations for this device
|
|
//
|
|
if (bHasCurrent) {
|
|
//
|
|
// Current config (if any) is indicated with zero handle
|
|
//
|
|
LoadString(MyDllModuleHandle, IDS_CURRENTCONFIG, szString, MAX_PATH);
|
|
|
|
iIndex = (int)SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST,
|
|
CB_ADDSTRING, (WPARAM)0, (LPARAM)(LPSTR)szString);
|
|
SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST, CB_SETITEMDATA,(WPARAM)iIndex, (LPARAM)0);
|
|
SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST, CB_SETCURSEL,(WPARAM)0, (LPARAM)0);
|
|
}
|
|
//
|
|
// now fill in alternate configurations
|
|
// override preferred over basic (in that order)
|
|
// don't use Filtered, filtered might remove configs that require reboot
|
|
// but they are fine to have here
|
|
//
|
|
if(CM_Get_First_Log_Conf_Ex(&LogConf,lpdmpd->lpdi->DevInst,OVERRIDE_LOG_CONF,hMachine) == CR_SUCCESS) {
|
|
lpdmpd->ConfigListLCType = OVERRIDE_LOG_CONF;
|
|
LoadString(MyDllModuleHandle, IDS_OVERRIDECONFIG, szConfigType, MAX_PATH);
|
|
bHasConfigList = TRUE;
|
|
} else if(CM_Get_First_Log_Conf_Ex(&LogConf,lpdmpd->lpdi->DevInst,BASIC_LOG_CONF,hMachine) == CR_SUCCESS) {
|
|
lpdmpd->ConfigListLCType = BASIC_LOG_CONF;
|
|
LoadString(MyDllModuleHandle, IDS_BASICCONFIG, szConfigType, MAX_PATH);
|
|
bHasConfigList = TRUE;
|
|
} else {
|
|
//
|
|
// If there are no alternate configs, we cannot allow a forced config
|
|
//
|
|
bNoForcedConfig = TRUE; // cannot force
|
|
bHasConfigList = FALSE;
|
|
lpdmpd->ConfigListLCType = BASIC_LOG_CONF;
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_SINGLE_CONFIG;
|
|
}
|
|
if(bHasConfigList) {
|
|
|
|
ulIndex = 0;
|
|
if (!pConfigHasNoAlternates(lpdmpd,LogConf)) {
|
|
//
|
|
// first configuration has more than one alternative
|
|
//
|
|
lpdmpd->dwFlags &= ~DMPROP_FLAG_SINGLE_CONFIG;
|
|
} else {
|
|
//
|
|
// begin with the assumption there is a single fixed 'basic' config
|
|
// we will generally be proved wrong
|
|
//
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_SINGLE_CONFIG;
|
|
}
|
|
|
|
while (Status == CR_SUCCESS) {
|
|
//
|
|
// Add this config to the Combobox
|
|
//
|
|
wsprintf(szTemp, TEXT("%s %04u"), szConfigType, ulIndex);
|
|
|
|
wItem = (WORD)SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST,
|
|
CB_ADDSTRING, 0,
|
|
(LPARAM)(LPSTR)szTemp);
|
|
|
|
//
|
|
// Save the log config handle as the item data in the combobox
|
|
//
|
|
SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST, CB_SETITEMDATA,
|
|
wItem, (LPARAM)LogConf);
|
|
|
|
//
|
|
// Get the next config
|
|
//
|
|
Status = CM_Get_Next_Log_Conf_Ex(&LogConf, LogConf, 0,hMachine);
|
|
ulIndex++;
|
|
}
|
|
|
|
if (ulIndex > 1) {
|
|
//
|
|
// there is more than one config
|
|
//
|
|
lpdmpd->dwFlags &= ~DMPROP_FLAG_SINGLE_CONFIG;
|
|
}
|
|
|
|
if (lpdmpd->dwFlags & DMPROP_FLAG_SINGLE_CONFIG) {
|
|
bNoForcedConfig = TRUE;
|
|
}
|
|
|
|
if (bHasCurrent) {
|
|
//
|
|
// try to find a matching LC now, and if we could find one,
|
|
// re-load current display (this applies editable ranges to the resources)
|
|
//
|
|
if(pFindMatchingAllocConfig(lpdmpd)) {
|
|
pLoadCurrentConfig(lpdmpd,TRUE);
|
|
}
|
|
}
|
|
} else {
|
|
}
|
|
//
|
|
// Get ConfigFlags here, we may use this info further down
|
|
//
|
|
ulSize = sizeof(ConfigFlags);
|
|
if (CM_Get_DevInst_Registry_Property_Ex(lpdmpd->lpdi->DevInst,
|
|
CM_DRP_CONFIGFLAGS,
|
|
NULL, (LPBYTE)&ConfigFlags,
|
|
&ulSize, 0,hMachine) != CR_SUCCESS) {
|
|
ConfigFlags = 0;
|
|
}
|
|
if (ConfigFlags & CONFIGFLAG_NEEDS_FORCED_CONFIG) {
|
|
//
|
|
// registry says that we need a forced config
|
|
// registry can only say this to us once
|
|
//
|
|
bNeedsForcedConfig = TRUE;
|
|
ConfigFlags &= ~CONFIGFLAG_NEEDS_FORCED_CONFIG;
|
|
CM_Set_DevInst_Registry_Property_Ex(lpdmpd->lpdi->DevInst,
|
|
CM_DRP_CONFIGFLAGS,
|
|
(LPBYTE)&ConfigFlags,
|
|
sizeof(ConfigFlags),
|
|
0,
|
|
hMachine);
|
|
}
|
|
|
|
//
|
|
// determine if it can be software-config'd or not
|
|
// we need to do this prior to any initial display
|
|
//
|
|
dwPriority = pGetMinLCPriority(lpdmpd->lpdi->DevInst, lpdmpd->ConfigListLCType,hMachine);
|
|
if (dwPriority < LCPRI_HARDRECONFIG) {
|
|
//
|
|
// doesn't need to be manually configured
|
|
//
|
|
lpdmpd->dwFlags &= ~DMPROP_FLAG_FORCEDONLY;
|
|
} else {
|
|
//
|
|
// this cannot be software config'd
|
|
// FORCEDONLY & bNoForcedConfig is a quandry, shouldn't happen
|
|
//
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_FORCEDONLY;
|
|
if(!bHasConfigList) {
|
|
MYASSERT(bHasConfigList);
|
|
} else {
|
|
MYASSERT(!bNoForcedConfig);
|
|
bNoForcedConfig = FALSE;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Try to determine initial display
|
|
//
|
|
// we've already covered pShowViewNoResources (no actual or potential configs)
|
|
// and pShowViewMFReadOnly (it's a multi-function device)
|
|
//
|
|
// we're currently showing as pShowViewReadOnly
|
|
//
|
|
// some cases....
|
|
// (1) show forced config, don't allow auto-config (config flags say requires forced)
|
|
// (2) show forced config, allow auto-config
|
|
// (3) don't show any config, but maybe show a forced-config button
|
|
// (4) auto-config, don't allow forced config
|
|
// (5) show auto-config, allow forced-config
|
|
//
|
|
if (bNeedsForcedConfig) {
|
|
if (!bHasConfigList) {
|
|
MYASSERT(bHasConfigList);
|
|
bNeedsForcedConfig = FALSE;
|
|
} else {
|
|
MYASSERT(!bNoForcedConfig);
|
|
bNoForcedConfig = FALSE;
|
|
if (bHasForced) {
|
|
//
|
|
// already got one, but we'll go through the motions
|
|
// we'll show what we have, allow user to change it
|
|
// but we wont needlessly save it
|
|
//
|
|
bNeedsForcedConfig = FALSE;
|
|
}
|
|
//
|
|
// caller said that device must have forced config, so go immediately there
|
|
// = case (1) unless we've otherwise said we cannot have a forced config
|
|
//
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_FORCEDONLY;
|
|
pSelectLogConf(lpdmpd,(LOG_CONF)0,ALLOC_LOG_CONF,TRUE);
|
|
pShowViewAllEdit(lpdmpd);
|
|
goto Final;
|
|
}
|
|
}
|
|
if ((!bShowCurrent) || (lpdmpd->dwFlags & DMPROP_FLAG_HASPROBLEM)) {
|
|
//
|
|
// determine between pShowViewNoAlloc and pShowViewNeedForced
|
|
//
|
|
if (bNoForcedConfig) {
|
|
//
|
|
// there is a problem - device doesn't currently have a current configuration
|
|
// but we don't have the option of letting them set forced config
|
|
// so this ends up display only (tough-luck scenario)
|
|
// if there are current resources, show them
|
|
//
|
|
pShowViewReadOnly(lpdmpd,FALSE);
|
|
} else {
|
|
//
|
|
// we show the problem and we give the user
|
|
// an option to force config
|
|
//
|
|
pShowViewNeedForced(lpdmpd);
|
|
}
|
|
goto Final;
|
|
}
|
|
if (!bHasConfigList) {
|
|
//
|
|
// If we have a current config, but no basic configs, we just display what we have
|
|
// and don't give option to edit
|
|
//
|
|
pShowViewReadOnly(lpdmpd,FALSE);
|
|
goto Final;
|
|
}
|
|
if ((lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS) && bNoForcedConfig) {
|
|
//
|
|
// we can't force a bNoForcedConfig item - display only
|
|
//
|
|
pShowViewReadOnly(lpdmpd,FALSE);
|
|
goto Final;
|
|
}
|
|
//
|
|
// we already have and will be displaying a current config
|
|
//
|
|
pShowViewAllEdit(lpdmpd);
|
|
bNeedsForcedConfig = (BOOL)!bHasCurrent; // rarely, if ever, will this return TRUE
|
|
|
|
Final:
|
|
|
|
return bNeedsForcedConfig;
|
|
|
|
} // InitDevResourceDlg
|
|
|
|
PITEMDATA
|
|
pGetResourceToChange(
|
|
IN LPDMPROP_DATA lpdmpd,
|
|
OUT int *pCur
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Gets resource to change
|
|
NULL if we cannot change resource
|
|
|
|
Arguments:
|
|
|
|
lpdmpd = dialog data
|
|
pCur = (out) index
|
|
|
|
Return Value:
|
|
|
|
PITEMDATA saved for selected resource
|
|
|
|
--*/
|
|
{
|
|
HWND hList = GetDlgItem(lpdmpd->hDlg, IDC_DEVRES_SETTINGSLIST);
|
|
PITEMDATA pItemData = NULL;
|
|
int iCur;
|
|
|
|
//
|
|
// first check the obvious
|
|
//
|
|
if (lpdmpd->dwFlags & DMPROP_FLAG_VIEWONLYRES) {
|
|
//
|
|
// no editing allowed
|
|
//
|
|
return NULL;
|
|
}
|
|
if (lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS) {
|
|
//
|
|
// showing system settings
|
|
//
|
|
return NULL;
|
|
}
|
|
|
|
|
|
//
|
|
// Check if there is a selected item.
|
|
// If yes, then activate the change button
|
|
// if the LC allows editing.
|
|
//
|
|
iCur = (int)ListView_GetNextItem(hList,-1, LVNI_SELECTED);
|
|
if (iCur == LB_ERR) {
|
|
//
|
|
// no selection
|
|
//
|
|
return NULL;
|
|
}
|
|
pItemData = (PITEMDATA)pGetListViewItemData(hList, iCur, 0);
|
|
if (pItemData == NULL) {
|
|
//
|
|
// shouldn't happen
|
|
//
|
|
MYASSERT(pItemData);
|
|
return NULL;
|
|
}
|
|
if (pItemData->bFixed) {
|
|
//
|
|
// this is an un-editable setting
|
|
//
|
|
return NULL;
|
|
}
|
|
if (pItemData->MatchingResDes == (RES_DES)0) {
|
|
//
|
|
// should be caught by bFixed
|
|
//
|
|
MYASSERT(pItemData->MatchingResDes != (RES_DES)0);
|
|
return NULL;
|
|
}
|
|
//
|
|
// we're happy
|
|
//
|
|
if (pCur) {
|
|
*pCur = iCur;
|
|
}
|
|
return pItemData;
|
|
}
|
|
|
|
VOID
|
|
pCheckEnableResourceChange(
|
|
LPDMPROP_DATA lpdmpd
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
enables/disable change button
|
|
|
|
Arguments:
|
|
|
|
|
|
Return Value:
|
|
|
|
none
|
|
|
|
--*/
|
|
{
|
|
#if 0 // this seems to confuse people
|
|
EnableWindow(GetDlgItem(lpdmpd->hDlg, IDC_DEVRES_CHANGE),
|
|
pGetResourceToChange(lpdmpd,NULL)!=NULL);
|
|
#endif // 0
|
|
|
|
//
|
|
// show this button enabled if we are in EDIT mode
|
|
//
|
|
EnableWindow(GetDlgItem(lpdmpd->hDlg, IDC_DEVRES_CHANGE),
|
|
(lpdmpd->dwFlags & DMPROP_FLAG_VIEWONLYRES)==0 &&
|
|
(lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS)==0);
|
|
}
|
|
|
|
BOOL
|
|
pDevHasConfig(
|
|
DEVINST DevInst,
|
|
ULONG ulConfigType,
|
|
HMACHINE hMachine
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine determines whether a log config of the specified type
|
|
exists for this device instance.
|
|
MEMPHIS COMPATIBLE.
|
|
|
|
Arguments:
|
|
|
|
DevInst Device instance to query log configs for.
|
|
|
|
ulConfigType Specifies the type of log conf to check for the presense of.
|
|
|
|
Return Value:
|
|
|
|
TRUE if the device has a config of that type and FALSE if it does not.
|
|
|
|
--*/
|
|
|
|
{
|
|
BOOL bRet = (CM_Get_First_Log_Conf_Ex(NULL, DevInst, ulConfigType,hMachine) == CR_SUCCESS);
|
|
return bRet;
|
|
|
|
} // DevHasConfig
|
|
|
|
DWORD
|
|
pGetMinLCPriority(
|
|
IN DEVINST DevInst,
|
|
IN ULONG ulConfigType,
|
|
IN HMACHINE hMachine
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine returns the minimum priority value of all log confs of the
|
|
specified type for this device. MEMPHIS COMPATIBLE.
|
|
|
|
Arguments:
|
|
|
|
DevInst Device instance to query log configs for.
|
|
|
|
ulConfigType Specifies the type of log conf.
|
|
|
|
Return Value:
|
|
|
|
Returns the minimum priority value found or LCPRI_LASTSOFTCONFIG if no priorities
|
|
are found.
|
|
|
|
--*/
|
|
|
|
{
|
|
CONFIGRET Status = CR_SUCCESS;
|
|
ULONG priority, minPriority = MAX_LCPRI;
|
|
LOG_CONF LogConf, tempLC;
|
|
BOOL FoundOneLogConfWithPriority = FALSE;
|
|
|
|
//
|
|
// Walk through each log conf of this type for this device and
|
|
// save the smallest value.
|
|
//
|
|
|
|
Status = CM_Get_First_Log_Conf_Ex(&LogConf, DevInst, ulConfigType,hMachine);
|
|
while (Status == CR_SUCCESS) {
|
|
|
|
if (CM_Get_Log_Conf_Priority_Ex(LogConf, &priority, 0,hMachine) == CR_SUCCESS) {
|
|
FoundOneLogConfWithPriority = TRUE;
|
|
minPriority = min(minPriority, priority);
|
|
}
|
|
|
|
tempLC = LogConf;
|
|
Status = CM_Get_Next_Log_Conf_Ex(&LogConf, LogConf, 0,hMachine);
|
|
CM_Free_Log_Conf_Handle(tempLC);
|
|
}
|
|
|
|
if(FoundOneLogConfWithPriority) {
|
|
return minPriority;
|
|
} else {
|
|
//
|
|
// None of the LogConfigs had an associated priority. This is common on
|
|
// NT, because the bus drivers don't specify ConfigMgr-style priorities
|
|
// when responding to IRQ_MN_QUERY_RESOURCE_REQUIREMENTS. Since these
|
|
// cases are all PnP bus drivers, however, it is most correct to specify
|
|
// these LogConfigs as soft-settable.
|
|
//
|
|
return LCPRI_LASTSOFTCONFIG;
|
|
}
|
|
|
|
} // GetMinLCPriority
|
|
|
|
BOOL
|
|
pDevRequiresResources(
|
|
DEVINST DevInst,
|
|
HMACHINE hMachine
|
|
)
|
|
{
|
|
if (CM_Get_First_Log_Conf_Ex(NULL, DevInst, BASIC_LOG_CONF,hMachine) == CR_SUCCESS) {
|
|
return TRUE;
|
|
}
|
|
|
|
if (CM_Get_First_Log_Conf_Ex(NULL, DevInst, FILTERED_LOG_CONF,hMachine) == CR_SUCCESS) {
|
|
return TRUE;
|
|
}
|
|
|
|
if (CM_Get_First_Log_Conf_Ex(NULL, DevInst, OVERRIDE_LOG_CONF,hMachine) == CR_SUCCESS) {
|
|
return TRUE;
|
|
}
|
|
|
|
if (CM_Get_First_Log_Conf_Ex(NULL, DevInst, FORCED_LOG_CONF,hMachine) == CR_SUCCESS) {
|
|
return TRUE;
|
|
}
|
|
|
|
if (CM_Get_First_Log_Conf_Ex(NULL, DevInst, BOOT_LOG_CONF,hMachine) == CR_SUCCESS) {
|
|
return TRUE;
|
|
}
|
|
|
|
if (CM_Get_First_Log_Conf_Ex(NULL, DevInst, ALLOC_LOG_CONF,hMachine) == CR_SUCCESS) {
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
} // DevRequiresResources
|
|
|
|
BOOL
|
|
pGetCurrentConfig(
|
|
IN OUT LPDMPROP_DATA lpdmpd
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine determines the current known configuration
|
|
current configs are either forced, alloc or boot configs.
|
|
|
|
Arguments:
|
|
|
|
lpdmpd Property data.
|
|
|
|
Return Value:
|
|
|
|
TRUE if we set the current config
|
|
|
|
--*/
|
|
|
|
{
|
|
PDEVICE_INFO_SET pDeviceInfoSet;
|
|
HMACHINE hMachine;
|
|
ULONG Status;
|
|
ULONG Problem;
|
|
|
|
MYASSERT(lpdmpd!=NULL);
|
|
MYASSERT(lpdmpd->lpdi!=NULL);
|
|
MYASSERT(lpdmpd->CurrentLC==0);
|
|
MYASSERT(lpdmpd->lpdi->DevInst!=0);
|
|
|
|
if (lpdmpd==NULL ||
|
|
lpdmpd->lpdi==NULL ||
|
|
lpdmpd->lpdi->DevInst==0) {
|
|
return FALSE;
|
|
}
|
|
|
|
lpdmpd->dwFlags &= ~DMPROP_FLAG_DISPLAY_MASK;
|
|
|
|
if(!(pDeviceInfoSet = AccessDeviceInfoSet(lpdmpd->hDevInfo))) {
|
|
return FALSE;
|
|
}
|
|
|
|
hMachine = pDeviceInfoSet->hMachine;
|
|
|
|
UnlockDeviceInfoSet (pDeviceInfoSet);
|
|
|
|
if (CM_Get_DevNode_Status_Ex(&Status, &Problem, lpdmpd->lpdi->DevInst,
|
|
0,hMachine) != CR_SUCCESS) {
|
|
Problem = 0;
|
|
Status = 0;
|
|
} else if((Status & DN_HAS_PROBLEM)==0) {
|
|
//
|
|
// If this device is running, does this devinst have a ALLOC log config?
|
|
//
|
|
if (CM_Get_First_Log_Conf_Ex(&lpdmpd->CurrentLC,
|
|
lpdmpd->lpdi->DevInst,
|
|
ALLOC_LOG_CONF,
|
|
hMachine) == CR_SUCCESS) {
|
|
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_DISPLAY_ALLOC;
|
|
lpdmpd->CurrentLCType = ALLOC_LOG_CONF;
|
|
return TRUE;
|
|
}
|
|
}
|
|
//
|
|
// If no config so far, does it have a FORCED log config?
|
|
//
|
|
|
|
if (CM_Get_First_Log_Conf_Ex(&lpdmpd->CurrentLC,
|
|
lpdmpd->lpdi->DevInst,
|
|
FORCED_LOG_CONF,
|
|
hMachine) == CR_SUCCESS) {
|
|
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_DISPLAY_FORCED;
|
|
lpdmpd->CurrentLCType = FORCED_LOG_CONF;
|
|
return TRUE;
|
|
}
|
|
|
|
//
|
|
// if there's a hardware-disabled problem, boot-config isn't valid
|
|
//
|
|
if(((Status & DN_HAS_PROBLEM)==0) || (Problem != CM_PROB_HARDWARE_DISABLED)) {
|
|
//
|
|
// Does it have a BOOT log config?
|
|
//
|
|
if (CM_Get_First_Log_Conf_Ex(&lpdmpd->CurrentLC,
|
|
lpdmpd->lpdi->DevInst,
|
|
BOOT_LOG_CONF,
|
|
hMachine) == CR_SUCCESS) {
|
|
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_DISPLAY_BOOT;
|
|
lpdmpd->CurrentLCType = BOOT_LOG_CONF;
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
void
|
|
pGetHdrValues(
|
|
IN LPBYTE pData,
|
|
IN RESOURCEID ResType,
|
|
OUT PULONG64 pulValue,
|
|
OUT PULONG64 pulLen,
|
|
OUT PULONG64 pulEnd,
|
|
OUT PULONG pulFlags
|
|
)
|
|
{
|
|
switch (ResType) {
|
|
|
|
case ResType_Mem: {
|
|
|
|
PMEM_RESOURCE pMemData = (PMEM_RESOURCE)pData;
|
|
|
|
*pulValue = pMemData->MEM_Header.MD_Alloc_Base;
|
|
*pulLen = (pMemData->MEM_Header.MD_Alloc_End -
|
|
pMemData->MEM_Header.MD_Alloc_Base + 1);
|
|
*pulEnd = pMemData->MEM_Header.MD_Alloc_End;
|
|
*pulFlags = pMemData->MEM_Header.MD_Flags;
|
|
break;
|
|
}
|
|
|
|
case ResType_IO: {
|
|
|
|
PIO_RESOURCE pIoData = (PIO_RESOURCE)pData;
|
|
|
|
*pulValue = pIoData->IO_Header.IOD_Alloc_Base;
|
|
*pulLen = (pIoData->IO_Header.IOD_Alloc_End -
|
|
pIoData->IO_Header.IOD_Alloc_Base + 1);
|
|
*pulEnd = pIoData->IO_Header.IOD_Alloc_End;
|
|
*pulFlags = pIoData->IO_Header.IOD_DesFlags;
|
|
break;
|
|
}
|
|
|
|
case ResType_DMA: {
|
|
|
|
PDMA_RESOURCE pDmaData = (PDMA_RESOURCE)pData;
|
|
|
|
*pulValue = pDmaData->DMA_Header.DD_Alloc_Chan;
|
|
*pulLen = 1;
|
|
*pulEnd = *pulValue;
|
|
*pulFlags = pDmaData->DMA_Header.DD_Flags;
|
|
break;
|
|
}
|
|
|
|
case ResType_IRQ: {
|
|
|
|
DEVRES_PIRQ_RESOURCE pIrqData = (DEVRES_PIRQ_RESOURCE)pData;
|
|
|
|
*pulValue = pIrqData->IRQ_Header.IRQD_Alloc_Num;
|
|
*pulLen = 1;
|
|
*pulEnd = *pulValue;
|
|
*pulFlags = pIrqData->IRQ_Header.IRQD_Flags;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(*pulEnd < *pulValue) {
|
|
//
|
|
// filter out bad/zero-length range
|
|
//
|
|
*pulLen = 0;
|
|
}
|
|
|
|
return;
|
|
|
|
} // GetHdrValues
|
|
|
|
void
|
|
pGetRangeValues(
|
|
IN LPBYTE pData,
|
|
IN RESOURCEID ResType,
|
|
IN ULONG ulIndex,
|
|
OUT PULONG64 pulValue, OPTIONAL
|
|
OUT PULONG64 pulLen, OPTIONAL
|
|
OUT PULONG64 pulEnd, OPTIONAL
|
|
OUT PULONG64 pulAlign, OPTIONAL
|
|
OUT PULONG pulFlags OPTIONAL
|
|
)
|
|
{
|
|
//
|
|
// keep local copies
|
|
// we transfer to parameters at end
|
|
//
|
|
ULONG64 ulValue;
|
|
ULONG64 ulLen;
|
|
ULONG64 ulEnd;
|
|
ULONG64 ulAlign;
|
|
ULONG ulFlags;
|
|
|
|
switch (ResType) {
|
|
|
|
case ResType_Mem: {
|
|
|
|
PMEM_RESOURCE pMemData = (PMEM_RESOURCE)pData;
|
|
|
|
ulValue = pMemData->MEM_Data[ulIndex].MR_Min;
|
|
ulLen = pMemData->MEM_Data[ulIndex].MR_nBytes;
|
|
ulEnd = pMemData->MEM_Data[ulIndex].MR_Max;
|
|
ulFlags = pMemData->MEM_Data[ulIndex].MR_Flags;
|
|
ulAlign = pMemData->MEM_Data[ulIndex].MR_Align;
|
|
break;
|
|
}
|
|
|
|
case ResType_IO: {
|
|
|
|
PIO_RESOURCE pIoData = (PIO_RESOURCE)pData;
|
|
|
|
ulValue = pIoData->IO_Data[ulIndex].IOR_Min;
|
|
ulLen = pIoData->IO_Data[ulIndex].IOR_nPorts;
|
|
ulEnd = pIoData->IO_Data[ulIndex].IOR_Max;
|
|
ulFlags = pIoData->IO_Data[ulIndex].IOR_RangeFlags;
|
|
ulAlign = pIoData->IO_Data[ulIndex].IOR_Align;
|
|
break;
|
|
}
|
|
|
|
case ResType_DMA: {
|
|
|
|
PDMA_RESOURCE pDmaData = (PDMA_RESOURCE)pData;
|
|
|
|
ulValue = pDmaData->DMA_Data[ulIndex].DR_Min;
|
|
ulLen = 1;
|
|
ulEnd = ulValue;
|
|
ulFlags = pDmaData->DMA_Data[ulIndex].DR_Flags;
|
|
ulAlign = 1;
|
|
break;
|
|
}
|
|
|
|
case ResType_IRQ: {
|
|
|
|
DEVRES_PIRQ_RESOURCE pIrqData = (DEVRES_PIRQ_RESOURCE)pData;
|
|
|
|
ulValue = pIrqData->IRQ_Data[ulIndex].IRQR_Min;
|
|
ulLen = 1;
|
|
ulEnd = ulValue;
|
|
ulFlags = pIrqData->IRQ_Data[ulIndex].IRQR_Flags;
|
|
ulAlign = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(ulEnd < ulValue) {
|
|
//
|
|
// filter out bad/zero-length range
|
|
//
|
|
ulLen = 0;
|
|
}
|
|
|
|
pAlignValues(&ulValue, ulValue, ulLen, ulEnd, ulAlign,1);
|
|
|
|
//
|
|
// copy return parameters
|
|
//
|
|
if (pulValue) {
|
|
*pulValue = ulValue;
|
|
}
|
|
if (pulLen) {
|
|
*pulLen = ulLen;
|
|
}
|
|
if (pulEnd) {
|
|
*pulEnd = ulEnd;
|
|
}
|
|
if (pulAlign) {
|
|
*pulAlign = ulAlign;
|
|
}
|
|
if (pulFlags) {
|
|
*pulFlags = ulFlags;
|
|
}
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
BOOL
|
|
pAlignValues(
|
|
IN OUT PULONG64 pulValue,
|
|
IN ULONG64 ulStart,
|
|
IN ULONG64 ulLen,
|
|
IN ULONG64 ulEnd,
|
|
IN ULONG64 ulAlignment,
|
|
IN int Increment
|
|
)
|
|
{
|
|
ULONG64 NtAlign = ~ulAlignment + 1; // convert from mask to modulus
|
|
ULONG64 Value;
|
|
ULONG64 Upp;
|
|
ULONG64 Remainder;
|
|
|
|
Value = *pulValue;
|
|
|
|
if (NtAlign == 0) {
|
|
return FALSE; // bogus alignment value
|
|
}
|
|
|
|
if (NtAlign != 1 && Increment != 0) {
|
|
//
|
|
// see if we are aligned
|
|
//
|
|
|
|
Remainder = Value % NtAlign;
|
|
|
|
if (Remainder != 0) {
|
|
//
|
|
// need to re-align
|
|
//
|
|
if (Increment>0) {
|
|
//
|
|
// Return the first valid aligned value greater than this value
|
|
//
|
|
Value += NtAlign - Remainder;
|
|
|
|
if (Value <= *pulValue) {
|
|
//
|
|
// overflow detected
|
|
//
|
|
return FALSE;
|
|
}
|
|
|
|
} else {
|
|
//
|
|
// Return the first valid aligned value less than this value
|
|
//
|
|
Value -= Remainder;
|
|
//
|
|
// we never overflow going down, since zero is a common denominator
|
|
// of alignment
|
|
//
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
//
|
|
// now check boundaries
|
|
//
|
|
|
|
if (Value < ulStart) {
|
|
return FALSE;
|
|
}
|
|
|
|
Upp = Value+ulLen-1;
|
|
if (Upp < Value) {
|
|
//
|
|
// catch overflow error
|
|
//
|
|
return FALSE;
|
|
}
|
|
if (Upp > ulEnd) {
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// set newly aligned value
|
|
//
|
|
|
|
*pulValue = Value;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
void
|
|
pFormatResString(
|
|
LPDMPROP_DATA lpdmpd,
|
|
LPTSTR lpszString,
|
|
ULONG64 ulVal,
|
|
ULONG64 ulLen,
|
|
RESOURCEID ResType
|
|
)
|
|
{
|
|
if (ulLen == 0) {
|
|
wsprintf(lpszString, szNoValue);
|
|
} else if ((ResType == ResType_DMA) || (ResType == ResType_IRQ)) {
|
|
wsprintf(lpszString, szOneDecNoConflict, (UINT)ulVal);
|
|
} else if (ResType == ResType_IO) {
|
|
wsprintf(lpszString, szTwoWordHexNoConflict, (ULONG)ulVal,
|
|
(ULONG)(ulVal + ulLen - 1));
|
|
} else if (lpdmpd && (lpdmpd->dwFlags & DMPROP_FLAG_64BIT_RANGE)) {
|
|
wsprintf(lpszString, szTwo64bitHexNoConflict, (ULONG64)ulVal,
|
|
(ULONG64)(ulVal + ulLen - 1));
|
|
} else {
|
|
wsprintf(lpszString, szTwoDWordHexNoConflict, (ULONG)ulVal,
|
|
(ULONG)(ulVal + ulLen - 1));
|
|
}
|
|
|
|
}
|
|
|
|
BOOL
|
|
pUnFormatResString(
|
|
LPTSTR lpszString,
|
|
PULONG64 pulVal,
|
|
PULONG64 pulEnd,
|
|
RESOURCEID ridResType
|
|
)
|
|
{
|
|
BOOL bRet = FALSE;
|
|
LPTSTR lpszTemp = NULL;
|
|
LPTSTR lpszTemp2 = NULL;
|
|
LPTSTR lpszCopy;
|
|
|
|
// ISSUE-2000/02/03 Fix pUnFormatResString bugs
|
|
//
|
|
// - extend this to handling DWORDLONG values
|
|
// - use correct Prev/Next functions for parsing string
|
|
//
|
|
|
|
//
|
|
// Allocate space for, and make a copy of the input string
|
|
//
|
|
lpszCopy = MyMalloc((lstrlen(lpszString)+1) * sizeof(TCHAR));
|
|
|
|
if (lpszCopy == NULL) {
|
|
return FALSE;
|
|
}
|
|
|
|
lstrcpy(lpszCopy, lpszString);
|
|
|
|
//
|
|
// Locate the dash if there is one, and convert the white space prev to
|
|
// the dash to a NULL. (ie 0200 - 0400 while be 0200)
|
|
//
|
|
lpszTemp = lpszCopy;
|
|
while ((*lpszTemp != '-') && (*lpszTemp != '\0')) {
|
|
lpszTemp++; // AnsiNext?
|
|
}
|
|
|
|
if (*lpszTemp != '\0') {
|
|
lpszTemp2 = lpszTemp-1;
|
|
++lpszTemp;
|
|
}
|
|
|
|
//
|
|
// Search back to set the NULL for the Value
|
|
//
|
|
if (lpszTemp2 != NULL) {
|
|
while ((*lpszTemp2 == ' ') || (*lpszTemp2 == '\t'))
|
|
lpszTemp2--; // AnsiPrev?
|
|
*(lpszTemp2+1)= '\0';
|
|
}
|
|
|
|
//
|
|
// Convert the first entry
|
|
//
|
|
if (pConvertEditText(lpszCopy, pulVal, ridResType)) {
|
|
//
|
|
// If there is a second entry, convert it, otherwise assume a length
|
|
// of one.
|
|
//
|
|
if (*lpszTemp != '\0') {
|
|
if (pConvertEditText(lpszTemp, pulEnd,ridResType)) {
|
|
bRet = TRUE;
|
|
}
|
|
} else {
|
|
*pulEnd = *pulVal;
|
|
bRet = TRUE;
|
|
}
|
|
}
|
|
|
|
MyFree(lpszCopy);
|
|
return bRet;
|
|
|
|
}
|
|
|
|
BOOL
|
|
pConvertEditText(
|
|
LPTSTR lpszConvert,
|
|
PULONG64 pulVal,
|
|
RESOURCEID ridResType
|
|
)
|
|
{
|
|
LPTSTR lpConvert;
|
|
|
|
if ((ridResType == ResType_Mem) || (ridResType == ResType_IO)) {
|
|
*pulVal = _tcstoul(lpszConvert, &lpConvert, (WORD)16);
|
|
} else {
|
|
*pulVal = _tcstoul(lpszConvert, &lpConvert, (WORD)10);
|
|
}
|
|
|
|
if (lpConvert == lpszConvert+lstrlen(lpszConvert)) {
|
|
return TRUE;
|
|
} else {
|
|
return FALSE;
|
|
}
|
|
|
|
} // ConvertEditText
|
|
|
|
void
|
|
pWarnResSettingNotEditable(
|
|
HWND hDlg,
|
|
WORD idWarning
|
|
)
|
|
{
|
|
TCHAR szTitle[MAX_PATH];
|
|
TCHAR szMessage[MAX_PATH * 2];
|
|
|
|
//
|
|
// Give some warning Messages. If there is no logconfig,
|
|
// then we cannot edit any settings, if there is, then
|
|
// just the setting they are choosing is not editable.
|
|
//
|
|
LoadString(MyDllModuleHandle, IDS_DEVRES_NOMODIFYTITLE, szTitle, MAX_PATH);
|
|
LoadString(MyDllModuleHandle, idWarning, szMessage, MAX_PATH * 2);
|
|
MessageBox(hDlg, szMessage, szTitle, MB_OK | MB_TASKMODAL | MB_ICONEXCLAMATION);
|
|
|
|
} // WarnResSettingsNotEditable
|
|
|
|
int
|
|
pWarnNoSave(
|
|
HWND hDlg,
|
|
WORD idWarning
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Warn that the settings will not be saved
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
IDCANCEL = don't proceed
|
|
IDOK/IDYES/IDNO = proceed without saving
|
|
|
|
--*/
|
|
{
|
|
TCHAR szTitle[MAX_PATH];
|
|
TCHAR szMessage[MAX_PATH * 2];
|
|
int res;
|
|
|
|
//
|
|
// Give a warning message of why we can't save settings
|
|
//
|
|
LoadString(MyDllModuleHandle, IDS_MAKE_FORCED_TITLE, szTitle, MAX_PATH);
|
|
LoadString(MyDllModuleHandle, idWarning, szMessage, MAX_PATH * 2);
|
|
|
|
//res = MessageBox(hDlg, szMessage, szTitle, MB_OKCANCEL | MB_TASKMODAL | MB_ICONEXCLAMATION);
|
|
//return res;
|
|
res = MessageBox(hDlg, szMessage, szTitle, MB_OK | MB_TASKMODAL | MB_ICONEXCLAMATION);
|
|
return IDCANCEL;
|
|
}
|
|
|
|
LPVOID
|
|
pGetListViewItemData(
|
|
HWND hList,
|
|
int iItem,
|
|
int iSubItem
|
|
)
|
|
{
|
|
LV_ITEM lviItem;
|
|
|
|
lviItem.mask = LVIF_PARAM;
|
|
lviItem.iItem = iItem;
|
|
lviItem.iSubItem = iSubItem;
|
|
|
|
if (ListView_GetItem(hList, &lviItem)) {
|
|
return (LPVOID)lviItem.lParam;
|
|
} else {
|
|
return NULL;
|
|
}
|
|
|
|
} // GetListViewItemData
|
|
|
|
BOOL
|
|
pSaveDevResSettings(
|
|
LPDMPROP_DATA lpdmpd
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine saves the resources based on the users selections.
|
|
MEMPHIS COMPATIBLE.
|
|
|
|
Arguments:
|
|
|
|
lpdmpd Property data.
|
|
|
|
Return Value:
|
|
|
|
Returns TRUE if the function succeeded and FALSE if it failed.
|
|
|
|
--*/
|
|
|
|
{
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
HWND hList = GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST);
|
|
|
|
CONFIGRET Status = CR_SUCCESS;
|
|
LOG_CONF ForcedLogConf;
|
|
RES_DES ResDes, ResDesTemp, ResDes1;
|
|
RESOURCEID ResType;
|
|
ULONG ulSize = 0, ulCount = 0, i = 0, iCur = 0;
|
|
LPBYTE pData = NULL;
|
|
PITEMDATA pItemData = NULL;
|
|
BOOL bRet = TRUE;
|
|
SP_PROPCHANGE_PARAMS PropChangeParams;
|
|
HMACHINE hMachine = pGetMachine(lpdmpd);
|
|
|
|
if ((lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS)!=0) {
|
|
|
|
//-------------------------------------------------------------------
|
|
// If the user checked the "Use Automatic Settings" checkbox, then
|
|
// delete any Boot/Forced configs, otherwise write the current settings
|
|
// as a forced config.
|
|
//-------------------------------------------------------------------
|
|
|
|
if (CM_Get_First_Log_Conf_Ex(&ForcedLogConf, lpdmpd->lpdi->DevInst,
|
|
FORCED_LOG_CONF,hMachine) == CR_SUCCESS) {
|
|
CM_Free_Log_Conf_Ex(ForcedLogConf, 0,hMachine);
|
|
CM_Free_Log_Conf_Handle(ForcedLogConf);
|
|
}
|
|
|
|
// Let the helper modules (class installer/co-installers) get in on the act...
|
|
//
|
|
PropChangeParams.ClassInstallHeader.cbSize = sizeof(SP_CLASSINSTALL_HEADER);
|
|
PropChangeParams.ClassInstallHeader.InstallFunction = DIF_PROPERTYCHANGE;
|
|
|
|
PropChangeParams.StateChange = DICS_PROPCHANGE;
|
|
PropChangeParams.Scope = DICS_FLAG_GLOBAL;
|
|
// no need to set PropChangeParams.HwProfile, since this is a global property change.
|
|
|
|
DoInstallActionWithParams(DIF_PROPERTYCHANGE,
|
|
lpdmpd->hDevInfo,
|
|
lpdmpd->lpdi,
|
|
(PSP_CLASSINSTALL_HEADER)&PropChangeParams,
|
|
sizeof(PropChangeParams),
|
|
INSTALLACTION_CALL_CI
|
|
);
|
|
} else {
|
|
|
|
//-------------------------------------------------------------------
|
|
// The Use Automatic Settings is not selected.
|
|
//-------------------------------------------------------------------
|
|
|
|
bRet = pSaveCustomResSettings(lpdmpd,hMachine);
|
|
}
|
|
|
|
return bRet;
|
|
}
|
|
|
|
BOOL
|
|
pSaveCustomResSettings(
|
|
LPDMPROP_DATA lpdmpd,
|
|
IN HMACHINE hMachine
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine saves custom (user edited) resources. MEMPHIS COMPATIBLE but
|
|
extracted from Memphis version of SaveDevResSetting().
|
|
|
|
Arguments:
|
|
|
|
lpdmpd Property data.
|
|
|
|
Return Value:
|
|
|
|
Returns TRUE if the function succeeded and FALSE if it failed.
|
|
|
|
--*/
|
|
|
|
{
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
|
|
TCHAR szWarn[MAX_MSG_LEN];
|
|
TCHAR szTitle[MAX_MSG_LEN];
|
|
TCHAR szTemp[MAX_MSG_LEN];
|
|
DWORD dwPriority, dwLCPri;
|
|
LOG_CONF ForcedLogConf;
|
|
RES_DES ResDes;
|
|
HWND hList = GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST);
|
|
PITEMDATA pItemData = NULL;
|
|
LONG iCur;
|
|
BOOL bRet = FALSE;
|
|
SP_PROPCHANGE_PARAMS PropChangeParams;
|
|
DWORD HardReconfigFlag;
|
|
SP_DEVINSTALL_PARAMS DevInstallParams;
|
|
PRESDES_ENTRY pResList = NULL, pResDesEntry = NULL, pTemp = NULL;
|
|
PITEMDATA_LISTNODE ItemDataList = NULL, ItemDataListEntry, ItemDataListEnd = NULL;
|
|
PGENERIC_RESOURCE pGenRes;
|
|
ULONG i, ulFlags;
|
|
ULONG64 ulValue, ulLen, ulEnd;
|
|
LOG_CONF LogConf;
|
|
ULONG ulSize;
|
|
ULONG ulConfigFlags;
|
|
HCURSOR hOldCursor;
|
|
BOOL UsingMatch = FALSE;
|
|
|
|
LogConf = lpdmpd->SelectedLC;
|
|
if (LogConf == 0) {
|
|
LogConf = lpdmpd->MatchingLC;
|
|
UsingMatch = TRUE;
|
|
}
|
|
if (LogConf == 0) {
|
|
LogConf = lpdmpd->CurrentLC;
|
|
UsingMatch = FALSE;
|
|
}
|
|
if (LogConf == 0) {
|
|
//MYASSERT(FALSE);
|
|
return FALSE;
|
|
}
|
|
//
|
|
// form the "warning - do you want to continue" message
|
|
//
|
|
if(!LoadString(MyDllModuleHandle, IDS_MAKE_FORCED_TITLE, szTitle, MAX_MSG_LEN)) {
|
|
szTitle[0]=TEXT('\0');
|
|
}
|
|
if(!LoadString(MyDllModuleHandle, IDS_FORCEDCONFIG_WARN1, szWarn, MAX_MSG_LEN)) {
|
|
szWarn[0]=TEXT('\0');
|
|
}
|
|
if(LoadString(MyDllModuleHandle, IDS_FORCEDCONFIG_WARN2, szTemp, MAX_MSG_LEN)) {
|
|
lstrcat(szWarn, szTemp);
|
|
}
|
|
if(LoadString(MyDllModuleHandle, IDS_FORCEDCONFIG_WARN3, szTemp, MAX_MSG_LEN)) {
|
|
lstrcat(szWarn, szTemp);
|
|
}
|
|
if(LoadString(MyDllModuleHandle, IDS_FORCEDCONFIG_WARN4, szTemp, MAX_MSG_LEN)) {
|
|
lstrcat(szWarn, szTemp);
|
|
}
|
|
|
|
//
|
|
// If the LCPRI is soft configurable, and the user chooses YES to the
|
|
// warning, then save the new config. If the LCPRI is not soft
|
|
// configurable, just save with no warning
|
|
//
|
|
dwLCPri = pGetMinLCPriority(lpdmpd->lpdi->DevInst, lpdmpd->ConfigListLCType,hMachine);
|
|
|
|
if (((dwLCPri >= LCPRI_DESIRED) && (dwLCPri <= LCPRI_LASTSOFTCONFIG)) &&
|
|
(MessageBox(hDlg, szWarn, szTitle, MB_YESNO|MB_ICONEXCLAMATION) == IDNO)) {
|
|
//
|
|
// user doesn't want to change anything
|
|
//
|
|
bRet = FALSE;
|
|
|
|
} else {
|
|
//
|
|
// We're still using the selected basic LC, but use the range index
|
|
// embedded in the listview control
|
|
// ISSUE-2000/02/03-JamieHun Selected Basic LC, check if user overrode
|
|
// Need to check the value to see if a user overrode it (is this possible?)
|
|
//
|
|
bRet = TRUE;
|
|
|
|
if (CM_Get_First_Log_Conf_Ex(&ForcedLogConf, lpdmpd->lpdi->DevInst,
|
|
FORCED_LOG_CONF,hMachine) == CR_SUCCESS) {
|
|
CM_Free_Log_Conf_Ex(ForcedLogConf, 0,hMachine);
|
|
CM_Free_Log_Conf_Handle(ForcedLogConf);
|
|
}
|
|
|
|
//
|
|
// Save the current choices as the forced config
|
|
//
|
|
CM_Add_Empty_Log_Conf_Ex(&ForcedLogConf, lpdmpd->lpdi->DevInst, LCPRI_FORCECONFIG,
|
|
FORCED_LOG_CONF | PRIORITY_EQUAL_FIRST,hMachine);
|
|
|
|
pGetResDesDataList(LogConf, &pResList, FALSE,hMachine);
|
|
pResDesEntry = pResList;
|
|
|
|
if (UsingMatch && (lpdmpd->dwFlags & DMPROP_FLAG_MATCH_OUT_OF_ORDER)) {
|
|
//
|
|
// The resource descriptors are out-of-order. Maintain the original ordering.
|
|
//
|
|
// First, build up a linked list of the data in the listview resource items.
|
|
//
|
|
iCur = (int)ListView_GetNextItem(hList, -1, LVNI_ALL);
|
|
|
|
while (iCur != -1) {
|
|
|
|
pItemData = (PITEMDATA)pGetListViewItemData(hList, iCur, 0);
|
|
if (pItemData) {
|
|
//
|
|
// Allocate an item data list node for this data.
|
|
//
|
|
ItemDataListEntry = MyMalloc(sizeof(ITEMDATA_LISTNODE));
|
|
if (!ItemDataListEntry) {
|
|
bRet = FALSE;
|
|
goto clean0;
|
|
}
|
|
|
|
ItemDataListEntry->ItemData = pItemData;
|
|
ItemDataListEntry->Next = NULL;
|
|
|
|
//
|
|
// Append this new item to the end of our list.
|
|
//
|
|
if (ItemDataListEnd) {
|
|
ItemDataListEnd->Next = ItemDataListEntry;
|
|
} else {
|
|
ItemDataList = ItemDataListEntry;
|
|
}
|
|
ItemDataListEnd = ItemDataListEntry;
|
|
}
|
|
|
|
iCur = (int)ListView_GetNextItem(hList, iCur, LVNI_ALL);
|
|
}
|
|
|
|
//
|
|
// Now loop through each resdes entry, writing each one out. For each one, check
|
|
// to see if it has a corresponding entry in our listview item data list.
|
|
//
|
|
while (pResDesEntry) {
|
|
pGenRes = (PGENERIC_RESOURCE)pResDesEntry->ResDesData;
|
|
|
|
for(ItemDataListEntry = ItemDataList, ItemDataListEnd = NULL;
|
|
ItemDataListEntry;
|
|
ItemDataListEnd = ItemDataListEntry, ItemDataListEntry = ItemDataListEntry->Next)
|
|
{
|
|
if(pResDesEntry->ResDesType == ItemDataListEntry->ItemData->ResType) {
|
|
|
|
for (i = 0; i < pGenRes->GENERIC_Header.GENERIC_Count; i++) {
|
|
|
|
pGetRangeValues(pResDesEntry->ResDesData, pResDesEntry->ResDesType, i,
|
|
&ulValue, &ulLen, &ulEnd, NULL, &ulFlags);
|
|
|
|
if ((ItemDataListEntry->ItemData->ulLen == ulLen) &&
|
|
(ItemDataListEntry->ItemData->ulValue >= ulValue) &&
|
|
(ItemDataListEntry->ItemData->ulEnd <= ulEnd)) {
|
|
//
|
|
// We found the matching resource descriptor. Write this out.
|
|
//
|
|
pWriteValuesToForced(ForcedLogConf,
|
|
ItemDataListEntry->ItemData->ResType,
|
|
ItemDataListEntry->ItemData->RangeCount,
|
|
ItemDataListEntry->ItemData->MatchingResDes,
|
|
ItemDataListEntry->ItemData->ulValue,
|
|
ItemDataListEntry->ItemData->ulLen,
|
|
ItemDataListEntry->ItemData->ulEnd,
|
|
hMachine );
|
|
//
|
|
// Remove this item from our list.
|
|
//
|
|
if (ItemDataListEnd) {
|
|
ItemDataListEnd->Next = ItemDataListEntry->Next;
|
|
} else {
|
|
ItemDataList = ItemDataListEntry->Next;
|
|
}
|
|
MyFree(ItemDataListEntry);
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(i < pGenRes->GENERIC_Header.GENERIC_Count) {
|
|
//
|
|
// Then we broke out of the loop early, which means we found a match
|
|
// already.
|
|
//
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// If we didn't find a match, then go ahead and write out the non-arbitrated
|
|
// resdes.
|
|
//
|
|
if (!ItemDataListEntry) {
|
|
pWriteResDesRangeToForced(ForcedLogConf,
|
|
pResDesEntry->ResDesType,
|
|
0,
|
|
0,
|
|
pResDesEntry->ResDesData,
|
|
hMachine);
|
|
}
|
|
|
|
pResDesEntry = (PRESDES_ENTRY)pResDesEntry->Next;
|
|
}
|
|
|
|
} else {
|
|
|
|
iCur = (int)ListView_GetNextItem(hList, -1, LVNI_ALL);
|
|
|
|
while (iCur != -1) {
|
|
|
|
pItemData = (PITEMDATA)pGetListViewItemData(hList, iCur, 0);
|
|
|
|
if (pItemData) {
|
|
|
|
// retrieve values
|
|
|
|
while (pResDesEntry &&
|
|
(pItemData->ResType != pResDesEntry->ResDesType)) {
|
|
//
|
|
// write out any preceding non arbitrated resources
|
|
//
|
|
pWriteResDesRangeToForced(ForcedLogConf,
|
|
pResDesEntry->ResDesType,
|
|
0,
|
|
0,
|
|
pResDesEntry->ResDesData,
|
|
hMachine);
|
|
|
|
pResDesEntry = (PRESDES_ENTRY)pResDesEntry->Next;
|
|
}
|
|
if (pGetMatchingResDes(pItemData->ulValue,
|
|
pItemData->ulLen,
|
|
pItemData->ulEnd,
|
|
pItemData->ResType,
|
|
LogConf,
|
|
&ResDes,
|
|
hMachine)) {
|
|
//
|
|
// Write the first range as the chosen forced resource
|
|
//
|
|
pWriteValuesToForced(ForcedLogConf, pItemData->ResType,
|
|
pItemData->RangeCount, ResDes,
|
|
pItemData->ulValue,
|
|
pItemData->ulLen,
|
|
pItemData->ulEnd,
|
|
hMachine);
|
|
}
|
|
}
|
|
|
|
if (pResDesEntry) {
|
|
pResDesEntry = (PRESDES_ENTRY)pResDesEntry->Next;
|
|
} else {
|
|
MYASSERT(pResDesEntry);
|
|
}
|
|
iCur = (int)ListView_GetNextItem(hList, iCur, LVNI_ALL);
|
|
}
|
|
|
|
while (pResDesEntry) {
|
|
//
|
|
// write out any subsequent non arbitrated resources
|
|
//
|
|
pWriteResDesRangeToForced(ForcedLogConf,
|
|
pResDesEntry->ResDesType,
|
|
0,
|
|
0,
|
|
pResDesEntry->ResDesData,
|
|
hMachine);
|
|
|
|
pResDesEntry = (PRESDES_ENTRY)pResDesEntry->Next;
|
|
}
|
|
}
|
|
|
|
CM_Free_Log_Conf_Handle(ForcedLogConf);
|
|
|
|
//
|
|
// consider clearing problem flags
|
|
//
|
|
ulSize = sizeof(ulConfigFlags);
|
|
if (CM_Get_DevInst_Registry_Property_Ex(lpdmpd->lpdi->DevInst,
|
|
CM_DRP_CONFIGFLAGS,
|
|
NULL, (LPBYTE)&ulConfigFlags,
|
|
&ulSize, 0,hMachine) == CR_SUCCESS) {
|
|
if ((ulConfigFlags & CONFIGFLAG_PARTIAL_LOG_CONF) != 0) {
|
|
//
|
|
// have flag(s) to change
|
|
// CONFIGFLAG_PARTIAL_LOG_CONF should be cleared - we should have written a complete config now
|
|
//
|
|
ulConfigFlags &= ~ (CONFIGFLAG_PARTIAL_LOG_CONF);
|
|
CM_Set_DevInst_Registry_Property_Ex(lpdmpd->lpdi->DevInst,
|
|
CM_DRP_CONFIGFLAGS,
|
|
(LPBYTE)&ulConfigFlags,
|
|
sizeof(ulConfigFlags),
|
|
0,
|
|
hMachine);
|
|
}
|
|
|
|
}
|
|
|
|
//
|
|
// Give the class installer/co-installers a crack at the propchange process.
|
|
//
|
|
PropChangeParams.ClassInstallHeader.cbSize = sizeof(SP_CLASSINSTALL_HEADER);
|
|
PropChangeParams.ClassInstallHeader.InstallFunction = DIF_PROPERTYCHANGE;
|
|
|
|
PropChangeParams.StateChange = DICS_PROPCHANGE;
|
|
PropChangeParams.Scope = DICS_FLAG_GLOBAL;
|
|
// no need to set PropChangeParams.HwProfile, since this is a global property change.
|
|
|
|
DoInstallActionWithParams(DIF_PROPERTYCHANGE,
|
|
lpdmpd->hDevInfo,
|
|
lpdmpd->lpdi,
|
|
(PSP_CLASSINSTALL_HEADER)&PropChangeParams,
|
|
sizeof(PropChangeParams),
|
|
INSTALLACTION_CALL_CI | INSTALLACTION_NO_DEFAULT
|
|
);
|
|
|
|
//
|
|
// Check the Priority of this LC. If it is greater
|
|
// than LCPRI_LASTSOFTCONFIG, then we need to reboot
|
|
// otherwise try the dynamic changestate route.
|
|
//
|
|
|
|
if (CM_Get_Log_Conf_Priority_Ex(LogConf, &dwPriority, 0,hMachine) != CR_SUCCESS) {
|
|
dwPriority = LCPRI_LASTSOFTCONFIG;
|
|
}
|
|
|
|
if (dwPriority <= LCPRI_LASTSOFTCONFIG) {
|
|
//
|
|
// Do the default action for SoftConfigable devices, which
|
|
// will attempt to restart the device with the new config
|
|
// This could take a while so use an hourglass
|
|
//
|
|
hOldCursor = SetCursor(LoadCursor(NULL, IDC_WAIT));
|
|
DoInstallActionWithParams(DIF_PROPERTYCHANGE,
|
|
lpdmpd->hDevInfo,
|
|
lpdmpd->lpdi,
|
|
(PSP_CLASSINSTALL_HEADER)&PropChangeParams,
|
|
sizeof(PropChangeParams),
|
|
0 // don't call class-installer, just do default action
|
|
);
|
|
SetCursor(hOldCursor);
|
|
HardReconfigFlag = 0;
|
|
|
|
} else if((dwPriority > LCPRI_LASTSOFTCONFIG) && (dwPriority <= LCPRI_RESTART)) {
|
|
HardReconfigFlag = DI_NEEDRESTART;
|
|
} else {
|
|
HardReconfigFlag = DI_NEEDREBOOT;
|
|
}
|
|
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_CHANGESSAVED;
|
|
|
|
//
|
|
// Properties have changed, so set flags to indicate if restart/reboot is required,
|
|
// and to tell DevMgr to re-init the UI.
|
|
//
|
|
DevInstallParams.cbSize = sizeof(SP_DEVINSTALL_PARAMS);
|
|
if(SetupDiGetDeviceInstallParams(lpdmpd->hDevInfo,
|
|
lpdmpd->lpdi,
|
|
&DevInstallParams)) {
|
|
|
|
DevInstallParams.Flags |= (HardReconfigFlag | DI_PROPERTIES_CHANGE);
|
|
|
|
SetupDiSetDeviceInstallParams(lpdmpd->hDevInfo,
|
|
lpdmpd->lpdi,
|
|
&DevInstallParams
|
|
);
|
|
}
|
|
|
|
//
|
|
// If we need to reboot, then set a problem on the device that indicates this (in case
|
|
// the user doesn't listen to us, we want to flag this devnode so that the user will see
|
|
// that this devnode needs a reboot if they go into DevMgr, etc.)
|
|
//
|
|
if(HardReconfigFlag) {
|
|
PDEVICE_INFO_SET pDeviceInfoSet;
|
|
PDEVINFO_ELEM DevInfoElem;
|
|
|
|
if(!(pDeviceInfoSet = AccessDeviceInfoSet(lpdmpd->hDevInfo))) {
|
|
//
|
|
// We'd better be able to access this device information set!
|
|
// In case we couldn't don't bother trying to set the needs-reboot problem,
|
|
// because the whole mess is invalid!
|
|
//
|
|
MYASSERT(pDeviceInfoSet);
|
|
} else {
|
|
|
|
try {
|
|
DevInfoElem = FindAssociatedDevInfoElem(pDeviceInfoSet, lpdmpd->lpdi, NULL);
|
|
//
|
|
// We'd better be able to find this element!
|
|
//
|
|
MYASSERT(DevInfoElem);
|
|
//
|
|
// In case we can't find it, don't try to set any problem on the devnode.
|
|
//
|
|
if(DevInfoElem) {
|
|
|
|
SetDevnodeNeedsRebootProblem(DevInfoElem,
|
|
pDeviceInfoSet,
|
|
MSG_LOG_REBOOT_DEVRES
|
|
);
|
|
}
|
|
} finally {
|
|
UnlockDeviceInfoSet(pDeviceInfoSet);
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
clean0:
|
|
|
|
while (ItemDataList) {
|
|
ItemDataListEntry = ItemDataList->Next;
|
|
MyFree(ItemDataList);
|
|
ItemDataList = ItemDataListEntry;
|
|
}
|
|
|
|
pDeleteResDesDataList(pResList);
|
|
|
|
return bRet;
|
|
|
|
} // SaveCustomResSettings
|
|
|
|
BOOL
|
|
pWriteResDesRangeToForced(
|
|
IN LOG_CONF ForcedLogConf,
|
|
IN RESOURCEID ResType,
|
|
IN ULONG RangeIndex,
|
|
IN RES_DES RD, OPTIONAL
|
|
IN LPBYTE ResDesData, OPTIONAL
|
|
IN HMACHINE hMachine OPTIONAL
|
|
)
|
|
{
|
|
RES_DES ResDes;
|
|
ULONG ulSize;
|
|
LPBYTE pData = NULL;
|
|
BOOL Success = FALSE;
|
|
|
|
|
|
if ((RD == 0) && (ResDesData == NULL)) {
|
|
return FALSE; // pass in data or handle!
|
|
}
|
|
|
|
if (!ResDesData) {
|
|
|
|
if (CM_Get_Res_Des_Data_Size_Ex(&ulSize, RD, DEVRES_WIDTH_FLAGS,hMachine) != CR_SUCCESS) {
|
|
return FALSE;
|
|
}
|
|
|
|
pData = MyMalloc(ulSize);
|
|
if (pData == NULL) {
|
|
return FALSE;
|
|
}
|
|
|
|
if (CM_Get_Res_Des_Data_Ex(RD, pData, ulSize, DEVRES_WIDTH_FLAGS,hMachine) != CR_SUCCESS) {
|
|
MyFree(pData);
|
|
return FALSE;
|
|
}
|
|
} else {
|
|
pData = ResDesData;
|
|
}
|
|
|
|
//
|
|
// convert the first range data into hdr data
|
|
//
|
|
switch (ResType) {
|
|
|
|
case ResType_Mem: {
|
|
|
|
PMEM_RESOURCE pMemData = (PMEM_RESOURCE)pData;
|
|
PMEM_RESOURCE pForced = (PMEM_RESOURCE)MyMalloc(sizeof(MEM_RESOURCE));
|
|
if (!pForced) {
|
|
break;
|
|
}
|
|
|
|
pForced->MEM_Header.MD_Count = 0;
|
|
pForced->MEM_Header.MD_Type = MType_Range;
|
|
pForced->MEM_Header.MD_Alloc_Base = pMemData->MEM_Data[RangeIndex].MR_Min;
|
|
pForced->MEM_Header.MD_Alloc_End = pMemData->MEM_Data[RangeIndex].MR_Min +
|
|
pMemData->MEM_Data[RangeIndex].MR_nBytes - 1;
|
|
pForced->MEM_Header.MD_Flags = pMemData->MEM_Data[RangeIndex].MR_Flags;
|
|
pForced->MEM_Header.MD_Reserved = 0;
|
|
|
|
Success = CM_Add_Res_Des_Ex(NULL,
|
|
ForcedLogConf,
|
|
ResType_Mem,
|
|
pForced,
|
|
sizeof(MEM_RESOURCE),
|
|
DEVRES_WIDTH_FLAGS,
|
|
hMachine) == CR_SUCCESS;
|
|
MyFree(pForced);
|
|
break;
|
|
}
|
|
|
|
case ResType_IO: {
|
|
|
|
PIO_RESOURCE pIoData = (PIO_RESOURCE)pData;
|
|
PIO_RESOURCE pForced = (PIO_RESOURCE)MyMalloc(sizeof(IO_RESOURCE));
|
|
if (!pForced) {
|
|
break;
|
|
}
|
|
|
|
pForced->IO_Header.IOD_Count = 0;
|
|
pForced->IO_Header.IOD_Type = IOType_Range;
|
|
pForced->IO_Header.IOD_Alloc_Base = pIoData->IO_Data[RangeIndex].IOR_Min;
|
|
pForced->IO_Header.IOD_Alloc_End = pIoData->IO_Data[RangeIndex].IOR_Min +
|
|
pIoData->IO_Data[RangeIndex].IOR_nPorts - 1;
|
|
pForced->IO_Header.IOD_DesFlags = pIoData->IO_Data[RangeIndex].IOR_RangeFlags;
|
|
|
|
Success = CM_Add_Res_Des_Ex(NULL,
|
|
ForcedLogConf,
|
|
ResType_IO,
|
|
pForced,
|
|
sizeof(IO_RESOURCE),
|
|
DEVRES_WIDTH_FLAGS,
|
|
hMachine) == CR_SUCCESS;
|
|
MyFree(pForced);
|
|
break;
|
|
}
|
|
|
|
case ResType_DMA: {
|
|
|
|
PDMA_RESOURCE pDmaData = (PDMA_RESOURCE)pData;
|
|
PDMA_RESOURCE pForced = (PDMA_RESOURCE)MyMalloc(sizeof(DMA_RESOURCE));
|
|
if (!pForced) {
|
|
break;
|
|
}
|
|
|
|
pForced->DMA_Header.DD_Count = 0;
|
|
pForced->DMA_Header.DD_Type = DType_Range;
|
|
pForced->DMA_Header.DD_Flags = pDmaData->DMA_Data[RangeIndex].DR_Flags;
|
|
pForced->DMA_Header.DD_Alloc_Chan = pDmaData->DMA_Data[RangeIndex].DR_Min;
|
|
|
|
Success = CM_Add_Res_Des_Ex(NULL,
|
|
ForcedLogConf,
|
|
ResType_DMA,
|
|
pForced,
|
|
sizeof(DMA_RESOURCE),
|
|
DEVRES_WIDTH_FLAGS,
|
|
hMachine) == CR_SUCCESS;
|
|
MyFree(pForced);
|
|
break;
|
|
}
|
|
|
|
case ResType_IRQ: {
|
|
|
|
DEVRES_PIRQ_RESOURCE pIrqData = (DEVRES_PIRQ_RESOURCE)pData;
|
|
DEVRES_PIRQ_RESOURCE pForced = (DEVRES_PIRQ_RESOURCE)MyMalloc(sizeof(DEVRES_IRQ_RESOURCE));
|
|
if (!pForced) {
|
|
break;
|
|
}
|
|
|
|
pForced->IRQ_Header.IRQD_Count = 0;
|
|
pForced->IRQ_Header.IRQD_Type = IRQType_Range;
|
|
pForced->IRQ_Header.IRQD_Flags = pIrqData->IRQ_Data[RangeIndex].IRQR_Flags;
|
|
pForced->IRQ_Header.IRQD_Alloc_Num = pIrqData->IRQ_Data[RangeIndex].IRQR_Min;
|
|
pForced->IRQ_Header.IRQD_Affinity = (DEVRES_AFFINITY)(-1);
|
|
|
|
Success = CM_Add_Res_Des_Ex(NULL,
|
|
ForcedLogConf,
|
|
ResType_IRQ,
|
|
pForced,
|
|
sizeof(DEVRES_IRQ_RESOURCE),
|
|
DEVRES_WIDTH_FLAGS,
|
|
hMachine) == CR_SUCCESS;
|
|
MyFree(pForced);
|
|
break;
|
|
}
|
|
|
|
case ResType_BusNumber: {
|
|
|
|
PBUSNUMBER_RESOURCE pBusData = (PBUSNUMBER_RESOURCE)pData;
|
|
PBUSNUMBER_RESOURCE pForced = (PBUSNUMBER_RESOURCE)MyMalloc(sizeof(BUSNUMBER_RESOURCE));
|
|
if (!pForced) {
|
|
break;
|
|
}
|
|
|
|
pForced->BusNumber_Header.BUSD_Count = 0;
|
|
pForced->BusNumber_Header.BUSD_Type = BusNumberType_Range;
|
|
pForced->BusNumber_Header.BUSD_Flags = pBusData->BusNumber_Data[RangeIndex].BUSR_Flags;
|
|
pForced->BusNumber_Header.BUSD_Alloc_Base = pBusData->BusNumber_Data[RangeIndex].BUSR_Min;
|
|
pForced->BusNumber_Header.BUSD_Alloc_End = pBusData->BusNumber_Data[RangeIndex].BUSR_Min +
|
|
pBusData->BusNumber_Data[RangeIndex].BUSR_nBusNumbers;
|
|
|
|
Success = CM_Add_Res_Des_Ex(NULL,
|
|
ForcedLogConf,
|
|
ResType_BusNumber,
|
|
pForced,
|
|
sizeof(BUSNUMBER_RESOURCE),
|
|
DEVRES_WIDTH_FLAGS,
|
|
hMachine) == CR_SUCCESS;
|
|
MyFree(pForced);
|
|
break;
|
|
}
|
|
|
|
case ResType_DevicePrivate: {
|
|
|
|
PDEVPRIVATE_RESOURCE pPrvData = (PDEVPRIVATE_RESOURCE)pData;
|
|
PDEVPRIVATE_RESOURCE pForced = (PDEVPRIVATE_RESOURCE)MyMalloc(sizeof(DEVPRIVATE_RESOURCE));
|
|
if (!pForced) {
|
|
break;
|
|
}
|
|
|
|
pForced->PRV_Header.PD_Count = 0;
|
|
pForced->PRV_Header.PD_Type = PType_Range;
|
|
pForced->PRV_Header.PD_Data1 = pPrvData->PRV_Data[RangeIndex].PR_Data1;
|
|
pForced->PRV_Header.PD_Data2 = pPrvData->PRV_Data[RangeIndex].PR_Data2;
|
|
pForced->PRV_Header.PD_Data3 = pPrvData->PRV_Data[RangeIndex].PR_Data3;
|
|
pForced->PRV_Header.PD_Flags = 0;
|
|
|
|
Success = CM_Add_Res_Des_Ex(NULL,
|
|
ForcedLogConf,
|
|
ResType_DevicePrivate,
|
|
pForced,
|
|
sizeof(DEVPRIVATE_RESOURCE),
|
|
DEVRES_WIDTH_FLAGS,
|
|
hMachine) == CR_SUCCESS;
|
|
MyFree(pForced);
|
|
break;
|
|
}
|
|
|
|
case ResType_PcCardConfig: {
|
|
|
|
Success = CM_Add_Res_Des_Ex(NULL,
|
|
ForcedLogConf,
|
|
ResType_PcCardConfig,
|
|
pData,
|
|
sizeof(PCCARD_RESOURCE),
|
|
DEVRES_WIDTH_FLAGS,
|
|
hMachine);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (pData != ResDesData) {
|
|
MyFree(pData);
|
|
}
|
|
|
|
return Success;
|
|
|
|
} // WriteResDesRangeToForced
|
|
|
|
BOOL
|
|
pWriteValuesToForced(
|
|
IN LOG_CONF ForcedLogConf,
|
|
IN RESOURCEID ResType,
|
|
IN ULONG RangeIndex,
|
|
IN RES_DES RD,
|
|
IN ULONG64 ulValue,
|
|
IN ULONG64 ulLen,
|
|
IN ULONG64 ulEnd,
|
|
IN HMACHINE hMachine
|
|
)
|
|
{
|
|
RES_DES ResDes;
|
|
ULONG ulSize;
|
|
LPBYTE pData = NULL;
|
|
BOOL Success = FALSE;
|
|
|
|
|
|
if (CM_Get_Res_Des_Data_Size_Ex(&ulSize, RD, DEVRES_WIDTH_FLAGS,hMachine) != CR_SUCCESS) {
|
|
return FALSE;
|
|
}
|
|
|
|
pData = MyMalloc(ulSize);
|
|
if (pData == NULL) {
|
|
return FALSE;
|
|
}
|
|
|
|
if (CM_Get_Res_Des_Data_Ex(RD, pData, ulSize, DEVRES_WIDTH_FLAGS,hMachine) != CR_SUCCESS) {
|
|
MyFree(pData);
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// convert the first range data into hdr data
|
|
//
|
|
switch (ResType) {
|
|
|
|
case ResType_Mem: {
|
|
|
|
PMEM_RESOURCE pMemData = (PMEM_RESOURCE)pData;
|
|
PMEM_RESOURCE pForced = (PMEM_RESOURCE)MyMalloc(sizeof(MEM_RESOURCE));
|
|
if (!pForced) {
|
|
break;
|
|
}
|
|
|
|
pForced->MEM_Header.MD_Count = 0;
|
|
pForced->MEM_Header.MD_Type = MType_Range;
|
|
pForced->MEM_Header.MD_Alloc_Base = ulValue;
|
|
pForced->MEM_Header.MD_Alloc_End = ulEnd;
|
|
pForced->MEM_Header.MD_Flags = pMemData->MEM_Data[RangeIndex].MR_Flags;
|
|
pForced->MEM_Header.MD_Reserved = 0;
|
|
|
|
Success = CM_Add_Res_Des_Ex(NULL,
|
|
ForcedLogConf,
|
|
ResType_Mem,
|
|
pForced,
|
|
sizeof(MEM_RESOURCE),
|
|
DEVRES_WIDTH_FLAGS,
|
|
hMachine) == CR_SUCCESS;
|
|
MyFree(pForced);
|
|
break;
|
|
}
|
|
|
|
case ResType_IO: {
|
|
|
|
PIO_RESOURCE pIoData = (PIO_RESOURCE)pData;
|
|
PIO_RESOURCE pForced = (PIO_RESOURCE)MyMalloc(sizeof(IO_RESOURCE));
|
|
if (!pForced) {
|
|
break;
|
|
}
|
|
|
|
pForced->IO_Header.IOD_Count = 0;
|
|
pForced->IO_Header.IOD_Type = IOType_Range;
|
|
pForced->IO_Header.IOD_Alloc_Base = ulValue;
|
|
pForced->IO_Header.IOD_Alloc_End = ulEnd;
|
|
pForced->IO_Header.IOD_DesFlags = pIoData->IO_Data[RangeIndex].IOR_RangeFlags;
|
|
|
|
Success = CM_Add_Res_Des_Ex(NULL,
|
|
ForcedLogConf,
|
|
ResType_IO,
|
|
pForced,
|
|
sizeof(IO_RESOURCE),
|
|
DEVRES_WIDTH_FLAGS,
|
|
hMachine) == CR_SUCCESS;
|
|
MyFree(pForced);
|
|
break;
|
|
}
|
|
|
|
case ResType_DMA: {
|
|
|
|
PDMA_RESOURCE pDmaData = (PDMA_RESOURCE)pData;
|
|
PDMA_RESOURCE pForced = (PDMA_RESOURCE)MyMalloc(sizeof(DMA_RESOURCE));
|
|
if (!pForced) {
|
|
break;
|
|
}
|
|
|
|
pForced->DMA_Header.DD_Count = 0;
|
|
pForced->DMA_Header.DD_Type = DType_Range;
|
|
pForced->DMA_Header.DD_Flags = pDmaData->DMA_Data[RangeIndex].DR_Flags;
|
|
pForced->DMA_Header.DD_Alloc_Chan = (ULONG)ulValue;
|
|
|
|
Success = CM_Add_Res_Des_Ex(NULL,
|
|
ForcedLogConf,
|
|
ResType_DMA,
|
|
pForced,
|
|
sizeof(DMA_RESOURCE),
|
|
DEVRES_WIDTH_FLAGS,
|
|
hMachine) == CR_SUCCESS;
|
|
MyFree(pForced);
|
|
break;
|
|
}
|
|
|
|
case ResType_IRQ: {
|
|
|
|
DEVRES_PIRQ_RESOURCE pIrqData = (DEVRES_PIRQ_RESOURCE)pData;
|
|
DEVRES_PIRQ_RESOURCE pForced = (DEVRES_PIRQ_RESOURCE)MyMalloc(sizeof(DEVRES_IRQ_RESOURCE));
|
|
if (!pForced) {
|
|
break;
|
|
}
|
|
|
|
pForced->IRQ_Header.IRQD_Count = 0;
|
|
pForced->IRQ_Header.IRQD_Type = IRQType_Range;
|
|
pForced->IRQ_Header.IRQD_Flags = pIrqData->IRQ_Data[RangeIndex].IRQR_Flags;
|
|
pForced->IRQ_Header.IRQD_Alloc_Num = (ULONG)ulValue;
|
|
pForced->IRQ_Header.IRQD_Affinity = (DEVRES_AFFINITY)(-1);
|
|
|
|
Success = CM_Add_Res_Des_Ex(NULL,
|
|
ForcedLogConf,
|
|
ResType_IRQ,
|
|
pForced,
|
|
sizeof(DEVRES_IRQ_RESOURCE),
|
|
DEVRES_WIDTH_FLAGS,
|
|
hMachine) == CR_SUCCESS;
|
|
MyFree(pForced);
|
|
break;
|
|
}
|
|
|
|
case ResType_BusNumber: {
|
|
|
|
PBUSNUMBER_RESOURCE pBusData = (PBUSNUMBER_RESOURCE)pData;
|
|
PBUSNUMBER_RESOURCE pForced = (PBUSNUMBER_RESOURCE)MyMalloc(sizeof(BUSNUMBER_RESOURCE));
|
|
if (!pForced) {
|
|
break;
|
|
}
|
|
|
|
pForced->BusNumber_Header.BUSD_Count = 0;
|
|
pForced->BusNumber_Header.BUSD_Type = BusNumberType_Range;
|
|
pForced->BusNumber_Header.BUSD_Flags = pBusData->BusNumber_Data[RangeIndex].BUSR_Flags;
|
|
pForced->BusNumber_Header.BUSD_Alloc_Base = (ULONG)ulValue;
|
|
pForced->BusNumber_Header.BUSD_Alloc_End = (ULONG)ulEnd;
|
|
|
|
Success = CM_Add_Res_Des_Ex(NULL,
|
|
ForcedLogConf,
|
|
ResType_BusNumber,
|
|
pForced,
|
|
sizeof(BUSNUMBER_RESOURCE),
|
|
DEVRES_WIDTH_FLAGS,
|
|
hMachine) == CR_SUCCESS;
|
|
MyFree(pForced);
|
|
break;
|
|
}
|
|
|
|
case ResType_DevicePrivate: {
|
|
break;
|
|
}
|
|
|
|
case ResType_PcCardConfig: {
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (pData) {
|
|
MyFree(pData);
|
|
}
|
|
return Success;
|
|
|
|
} // WriteValuesToForced
|
|
|
|
BOOL
|
|
MakeResourceData(
|
|
OUT LPBYTE *ppResourceData,
|
|
OUT PULONG pulSize,
|
|
IN RESOURCEID ResType,
|
|
IN ULONG64 ulValue,
|
|
IN ULONG64 ulLen,
|
|
IN ULONG ulFlags
|
|
)
|
|
{
|
|
BOOL bStatus = TRUE;
|
|
|
|
try {
|
|
|
|
switch (ResType) {
|
|
|
|
case ResType_Mem: {
|
|
|
|
PMEM_RESOURCE p;
|
|
|
|
*pulSize = sizeof(MEM_RESOURCE);
|
|
if (ppResourceData) {
|
|
*ppResourceData = MyMalloc(*pulSize);
|
|
p = (PMEM_RESOURCE)(*ppResourceData);
|
|
if (!p) {
|
|
bStatus = FALSE;
|
|
break;
|
|
}
|
|
|
|
p->MEM_Header.MD_Count = 0;
|
|
p->MEM_Header.MD_Type = MType_Range;
|
|
p->MEM_Header.MD_Alloc_Base = ulValue;
|
|
p->MEM_Header.MD_Alloc_End = ulValue + ulLen - 1;
|
|
p->MEM_Header.MD_Flags = ulFlags;
|
|
p->MEM_Header.MD_Reserved = 0;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case ResType_IO: {
|
|
|
|
PIO_RESOURCE p;
|
|
|
|
*pulSize = sizeof(IO_RESOURCE);
|
|
if (ppResourceData) {
|
|
*ppResourceData = MyMalloc(*pulSize);
|
|
p = (PIO_RESOURCE)(*ppResourceData);
|
|
if (!p) {
|
|
bStatus = FALSE;
|
|
break;
|
|
}
|
|
|
|
p->IO_Header.IOD_Count = 0;
|
|
p->IO_Header.IOD_Type = IOType_Range;
|
|
p->IO_Header.IOD_Alloc_Base = ulValue;
|
|
p->IO_Header.IOD_Alloc_End = ulValue + ulLen - 1;
|
|
p->IO_Header.IOD_DesFlags = ulFlags;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case ResType_DMA: {
|
|
|
|
PDMA_RESOURCE p;
|
|
|
|
*pulSize = sizeof(DMA_RESOURCE);
|
|
if (ppResourceData) {
|
|
*ppResourceData = MyMalloc(*pulSize);
|
|
p = (PDMA_RESOURCE)(*ppResourceData);
|
|
if (!p) {
|
|
bStatus = FALSE;
|
|
break;
|
|
}
|
|
|
|
p->DMA_Header.DD_Count = 0;
|
|
p->DMA_Header.DD_Type = DType_Range;
|
|
p->DMA_Header.DD_Flags = ulFlags;
|
|
p->DMA_Header.DD_Alloc_Chan = (ULONG)ulValue;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case ResType_IRQ: {
|
|
|
|
DEVRES_PIRQ_RESOURCE p;
|
|
|
|
*pulSize = sizeof(DEVRES_IRQ_RESOURCE);
|
|
if (ppResourceData) {
|
|
*ppResourceData = MyMalloc(*pulSize);
|
|
p = (DEVRES_PIRQ_RESOURCE)(*ppResourceData);
|
|
if (!p) {
|
|
bStatus = FALSE;
|
|
break;
|
|
}
|
|
|
|
p->IRQ_Header.IRQD_Count = 0;
|
|
p->IRQ_Header.IRQD_Type = IRQType_Range;
|
|
p->IRQ_Header.IRQD_Flags = ulFlags;
|
|
p->IRQ_Header.IRQD_Alloc_Num = (ULONG)ulValue;
|
|
p->IRQ_Header.IRQD_Affinity = (DEVRES_AFFINITY)(-1); // for any processor
|
|
}
|
|
break;
|
|
}
|
|
|
|
default:
|
|
//
|
|
// ResTypeEditable or ResType_MAX may be wrong if this ASSERT's
|
|
//
|
|
MYASSERT(FALSE);
|
|
bStatus = FALSE;
|
|
}
|
|
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
bStatus = FALSE;
|
|
}
|
|
|
|
return bStatus;
|
|
|
|
} // MakeResourceData
|
|
|
|
|
|
BOOL
|
|
pShowWindow(
|
|
IN HWND hWnd,
|
|
IN int nShow
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
A variation of ShowWindow that enables/disables window
|
|
|
|
Arguments:
|
|
|
|
(See ShowWindow)
|
|
hWnd - handle of window to show
|
|
nShow - typically SW_HIDE or SW_SHOW
|
|
|
|
Return Value:
|
|
|
|
success status of ShowWindow
|
|
|
|
--*/
|
|
{
|
|
EnableWindow(hWnd,nShow!=SW_HIDE);
|
|
return ShowWindow(hWnd,nShow);
|
|
}
|
|
|
|
|
|
BOOL
|
|
pEnableWindow(
|
|
IN HWND hWnd,
|
|
IN BOOL Enable
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
A variation of EnableWindow that only enables a window if it is visible
|
|
|
|
Arguments:
|
|
|
|
(See EnableWindow)
|
|
hWnd - handle of window to enable/disable
|
|
Enable - TRUE enables window (if window visible) FALSE disables window
|
|
|
|
Return Value:
|
|
|
|
success status of EnableWindow
|
|
|
|
--*/
|
|
{
|
|
//
|
|
// I had to use GetWindowLong, as IsWindowVisible also checks parent flag
|
|
// and parent is hidden until dialog is initialized
|
|
//
|
|
if((GetWindowLong(hWnd,GWL_STYLE) & WS_VISIBLE) == FALSE) {
|
|
Enable = FALSE;
|
|
}
|
|
return EnableWindow(hWnd,Enable);
|
|
}
|
|
|
|
BOOL
|
|
pGetResDesDataList(
|
|
IN LOG_CONF LogConf,
|
|
IN OUT PRESDES_ENTRY *pResList,
|
|
IN BOOL bArbitratedOnly,
|
|
IN HMACHINE hMachine
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Creates a list of resource descriptors for further processing
|
|
|
|
Arguments:
|
|
|
|
LogConf - log config of interest
|
|
pResList - list out
|
|
bArbitratedOnly - filter out non-arbitrated resources
|
|
hMachine - machine that LogConf is on
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
BOOL bStatus = TRUE;
|
|
CONFIGRET Status = CR_SUCCESS;
|
|
PRESDES_ENTRY pHead = NULL, pEntry = NULL, pPrevious = NULL, pTemp = NULL;
|
|
RES_DES ResDes;
|
|
RESOURCEID ResType;
|
|
ULONG ulSize;
|
|
LPBYTE pData = NULL;
|
|
|
|
//
|
|
// Retrieve each res des in this log conf
|
|
//
|
|
|
|
Status = CM_Get_Next_Res_Des_Ex(&ResDes, LogConf, ResType_All, &ResType, 0,hMachine);
|
|
|
|
while (Status == CR_SUCCESS) {
|
|
|
|
if (bArbitratedOnly && (ResType <= ResType_None || ResType > ResType_MAX)) {
|
|
goto NextResDes;
|
|
}
|
|
if (bArbitratedOnly && ResTypeEditable[ResType] == FALSE) {
|
|
goto NextResDes;
|
|
}
|
|
|
|
if (CM_Get_Res_Des_Data_Size_Ex(&ulSize, ResDes, DEVRES_WIDTH_FLAGS,hMachine) != CR_SUCCESS) {
|
|
CM_Free_Res_Des_Handle(ResDes);
|
|
bStatus = FALSE;
|
|
goto Clean0;
|
|
}
|
|
|
|
if (ulSize>0) {
|
|
pData = MyMalloc(ulSize);
|
|
if (pData == NULL) {
|
|
CM_Free_Res_Des_Handle(ResDes);
|
|
bStatus = FALSE;
|
|
goto Clean0;
|
|
}
|
|
|
|
if (CM_Get_Res_Des_Data_Ex(ResDes, pData, ulSize, DEVRES_WIDTH_FLAGS,hMachine) != CR_SUCCESS) {
|
|
CM_Free_Res_Des_Handle(ResDes);
|
|
MyFree(pData);
|
|
bStatus = FALSE;
|
|
goto Clean0;
|
|
}
|
|
} else {
|
|
pData = NULL;
|
|
}
|
|
|
|
//
|
|
// Allocate a node for this res des and attach it to the list
|
|
//
|
|
|
|
pEntry = MyMalloc(sizeof(RESDES_ENTRY));
|
|
if (pEntry == NULL) {
|
|
CM_Free_Res_Des_Handle(ResDes);
|
|
MyFree(pData);
|
|
bStatus = FALSE;
|
|
goto Clean0;
|
|
}
|
|
|
|
pEntry->ResDesData = pData;
|
|
pEntry->ResDesType = ResType;
|
|
pEntry->ResDesDataSize = ulSize;
|
|
pEntry->ResDesHandle = ResDes;
|
|
pEntry->Next = NULL;
|
|
pEntry->CrossLink = NULL;
|
|
|
|
if (!pHead) {
|
|
pHead = pEntry; // first entry
|
|
}
|
|
|
|
if (pPrevious) {
|
|
pPrevious->Next = pEntry; // attach to previous entry
|
|
}
|
|
|
|
pPrevious = pEntry;
|
|
|
|
//
|
|
// Get next res des in LogConf
|
|
//
|
|
NextResDes:
|
|
|
|
Status = CM_Get_Next_Res_Des_Ex(&ResDes, ResDes, ResType_All, &ResType, 0,hMachine);
|
|
}
|
|
|
|
bStatus = TRUE;
|
|
|
|
Clean0:
|
|
|
|
if (!bStatus) {
|
|
pDeleteResDesDataList(pHead);
|
|
} else {
|
|
*pResList = pHead;
|
|
}
|
|
|
|
return bStatus;
|
|
|
|
} // GetResDesDataList
|
|
|
|
VOID
|
|
pDeleteResDesDataList(
|
|
IN PRESDES_ENTRY pResList
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Deletes memory used by RESDES list
|
|
|
|
Arguments:
|
|
|
|
pResList - list returned by GetResDesDataList
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
PRESDES_ENTRY pTemp;
|
|
while (pResList) {
|
|
pTemp = pResList;
|
|
pResList = (PRESDES_ENTRY)pResList->Next;
|
|
if (pTemp->ResDesData) {
|
|
MyFree(pTemp->ResDesData);
|
|
}
|
|
if (pTemp->ResDesHandle) {
|
|
CM_Free_Res_Des_Handle(pTemp->ResDesHandle);
|
|
}
|
|
MyFree(pTemp);
|
|
}
|
|
}
|
|
|
|
VOID
|
|
pHideAllControls(
|
|
IN LPDMPROP_DATA lpdmpd
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Hide (and disable) all controls - start off with a clean slate
|
|
Only Icon & device description will be visible
|
|
|
|
Arguments:
|
|
|
|
hDlg = dialog handle of controls
|
|
lpdmpd = property data
|
|
|
|
Return Value:
|
|
|
|
none
|
|
|
|
--*/
|
|
{
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_SETTINGSTATE), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_RESOURCES_TEXT), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NOALLOCTEXT), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LCTEXT), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_CHANGE_TEXT ), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_USESYSSETTINGS), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CHANGE), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MAKEFORCED), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MFPARENT), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MFPARENT_DESC), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTDEVTEXT), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTINFOLIST), SW_HIDE);
|
|
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_VIEWONLYRES;
|
|
}
|
|
|
|
VOID
|
|
pShowViewNoResources(
|
|
IN LPDMPROP_DATA lpdmpd
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Show page indicating this device has no resources
|
|
|
|
Arguments:
|
|
|
|
hDlg = dialog handle of controls
|
|
lpdmpd = property data
|
|
|
|
Return Value:
|
|
|
|
none
|
|
|
|
--*/
|
|
{
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
TCHAR szString[MAX_PATH];
|
|
|
|
pHideAllControls(lpdmpd); // all hidden and disabled
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_RESOURCES_TEXT), SW_SHOW); // show and enable text
|
|
LoadString(MyDllModuleHandle, IDS_DEVRES_NO_RESOURCES, szString, MAX_PATH);
|
|
SetDlgItemText(hDlg, IDC_DEVRES_NO_RESOURCES_TEXT, szString);
|
|
}
|
|
|
|
BOOL
|
|
pShowViewMFReadOnly(
|
|
IN LPDMPROP_DATA lpdmpd,
|
|
IN BOOL HideIfProb
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Show page apropriate for multifunction card that cannot be edited
|
|
Resource settings are visible
|
|
|
|
Arguments:
|
|
|
|
hDlg = dialog handle of controls
|
|
lpdmpd = property data
|
|
|
|
Return Value:
|
|
|
|
none
|
|
|
|
--*/
|
|
{
|
|
TCHAR szString[MAX_PATH];
|
|
DEVNODE dnParent;
|
|
ULONG ulSize;
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
HMACHINE hMachine = pGetMachine(lpdmpd);
|
|
|
|
pHideAllControls(lpdmpd); // all hidden and disabled
|
|
//pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LCTEXT), SW_SHOW); // show config information
|
|
//pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST), SW_SHOW); // show
|
|
//pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTDEVTEXT), SW_SHOW); // show conflict information space
|
|
//pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTINFOLIST), SW_SHOW);
|
|
//
|
|
// indicate we cannot change as it's multi-function
|
|
//
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_CHANGE_TEXT), SW_SHOW);
|
|
if (LoadString(MyDllModuleHandle, IDS_DEVRES_NO_CHANGE_MF, szString, MAX_PATH)) {
|
|
SetDlgItemText(hDlg, IDC_DEVRES_NO_CHANGE_TEXT, szString);
|
|
}
|
|
//
|
|
// for parent description
|
|
//
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MFPARENT), SW_SHOW);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MFPARENT_DESC), SW_SHOW);
|
|
//
|
|
// Get the Parent's Description.
|
|
//
|
|
LoadString(MyDllModuleHandle, IDS_DEVNAME_UNK, szString, MAX_PATH);
|
|
|
|
if (lpdmpd->lpdi->DevInst) {
|
|
|
|
if (CM_Get_Parent_Ex(&dnParent, lpdmpd->lpdi->DevInst, 0,hMachine)
|
|
== CR_SUCCESS) {
|
|
|
|
|
|
//
|
|
// First, try to retrieve friendly name, then fall back to device description.
|
|
//
|
|
ulSize = MAX_PATH * sizeof(TCHAR);
|
|
if(CM_Get_DevNode_Registry_Property_Ex(dnParent, CM_DRP_FRIENDLYNAME,
|
|
NULL, szString, &ulSize, 0,hMachine) != CR_SUCCESS) {
|
|
|
|
ulSize = MAX_PATH * sizeof(TCHAR);
|
|
CM_Get_DevNode_Registry_Property_Ex(dnParent, CM_DRP_DEVICEDESC,
|
|
NULL, szString, &ulSize, 0,hMachine);
|
|
}
|
|
}
|
|
}
|
|
|
|
SetDlgItemText(hDlg, IDC_DEVRES_MFPARENT_DESC, szString);
|
|
|
|
//
|
|
// load and display current config (if any)
|
|
// return FALSE if no current config
|
|
//
|
|
return pLoadCurrentConfig(lpdmpd,HideIfProb);
|
|
}
|
|
|
|
BOOL
|
|
pShowViewReadOnly(
|
|
IN LPDMPROP_DATA lpdmpd,
|
|
IN BOOL HideIfProb
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Show page of resources, don't allow editing, don't show editing controls
|
|
|
|
Arguments:
|
|
|
|
hDlg = dialog handle of controls
|
|
lpdmpd = property data
|
|
|
|
Return Value:
|
|
|
|
none
|
|
|
|
--*/
|
|
{
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
|
|
pHideAllControls(lpdmpd); // all hidden and disabled
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LCTEXT), SW_SHOW); // show
|
|
EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_LCTEXT), FALSE);
|
|
ShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST), SW_SHOW); // shown disabled
|
|
EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST), FALSE);
|
|
ShowWindow(GetDlgItem(hDlg, IDC_DEVRES_USESYSSETTINGS), SW_SHOW); // shown disabled
|
|
EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_USESYSSETTINGS), FALSE);
|
|
ShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CHANGE), SW_SHOW); // shown disabled
|
|
EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_CHANGE), FALSE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTDEVTEXT), SW_SHOW); // show conflict information space
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTINFOLIST), SW_SHOW);
|
|
|
|
//
|
|
// will indicate if we're showing system settings or forced settings
|
|
//
|
|
CheckDlgButton(hDlg, IDC_DEVRES_USESYSSETTINGS, (lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS ) ? TRUE : FALSE);
|
|
|
|
//
|
|
// load and display current config (if any)
|
|
// return FALSE if no current config
|
|
//
|
|
return pLoadCurrentConfig(lpdmpd,HideIfProb);
|
|
}
|
|
|
|
VOID
|
|
pShowViewNoAlloc(
|
|
IN LPDMPROP_DATA lpdmpd
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Modify the middle part of the control to indicate there is a problem (and there isn't much we can do about it)
|
|
|
|
Arguments:
|
|
|
|
hDlg = dialog handle of controls
|
|
lpdmpd = property data
|
|
|
|
Return Value:
|
|
|
|
none
|
|
|
|
--*/
|
|
{
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
|
|
//
|
|
// hide all middle controls
|
|
//
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LCTEXT), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST), SW_HIDE);
|
|
//pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_CHANGE_TEXT ), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_USESYSSETTINGS), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CHANGE), SW_HIDE);
|
|
//pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MAKEFORCED), SW_HIDE);
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_VIEWONLYRES;
|
|
|
|
//pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_CHANGE_TEXT), SW_SHOW); // this may say why
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MAKEFORCED), SW_HIDE);
|
|
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTDEVTEXT), SW_HIDE); // no alloc, so hide this header & textbox
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTINFOLIST), SW_HIDE);
|
|
}
|
|
|
|
VOID
|
|
pShowViewNeedForced(
|
|
IN LPDMPROP_DATA lpdmpd
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Modify the middle part of the control to indicate a forced config is required
|
|
|
|
Arguments:
|
|
|
|
hDlg = dialog handle of controls
|
|
lpdmpd = property data
|
|
|
|
Return Value:
|
|
|
|
none
|
|
|
|
--*/
|
|
{
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
|
|
pShowViewNoAlloc(lpdmpd);
|
|
//
|
|
// show what we need for make forced config
|
|
//
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_CHANGE_TEXT), SW_SHOW); // this may say why
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MAKEFORCED), SW_SHOW);
|
|
}
|
|
|
|
VOID
|
|
pShowViewAllEdit(
|
|
IN LPDMPROP_DATA lpdmpd
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Allow editing
|
|
|
|
Arguments:
|
|
|
|
lpdmpd = property data
|
|
|
|
Return Value:
|
|
|
|
none
|
|
|
|
--*/
|
|
{
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
|
|
//
|
|
// show middle controls for editing
|
|
//
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LCTEXT), SW_SHOW);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST), SW_SHOW);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_CHANGE_TEXT ), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_USESYSSETTINGS), SW_SHOW);
|
|
ShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CHANGE), SW_SHOW); // shown, but disabled
|
|
EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_CHANGE), FALSE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MAKEFORCED), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTDEVTEXT), SW_SHOW); // show conflict information space
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTINFOLIST), SW_SHOW);
|
|
|
|
pShowUpdateEdit(lpdmpd);
|
|
}
|
|
|
|
VOID
|
|
pShowUpdateEdit(
|
|
IN LPDMPROP_DATA lpdmpd
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Allow editing
|
|
|
|
Arguments:
|
|
|
|
lpdmpd = property data
|
|
|
|
Return Value:
|
|
|
|
none
|
|
|
|
--*/
|
|
{
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
|
|
//
|
|
// modify editing status - we can edit
|
|
//
|
|
lpdmpd->dwFlags &= ~DMPROP_FLAG_VIEWONLYRES;
|
|
|
|
if(lpdmpd->dwFlags & DMPROP_FLAG_FORCEDONLY) {
|
|
//
|
|
// in this case, we will never be able to use system settings
|
|
//
|
|
lpdmpd->dwFlags &= ~ DMPROP_FLAG_USESYSSETTINGS;
|
|
EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_USESYSSETTINGS), FALSE);
|
|
}
|
|
//
|
|
// indicate if it's system settings or not
|
|
//
|
|
CheckDlgButton(hDlg, IDC_DEVRES_USESYSSETTINGS,
|
|
(lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS)?TRUE:FALSE);
|
|
//
|
|
// we can change logconfiglist if it's not system settings
|
|
//
|
|
EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_LCTEXT), (lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS)?FALSE:TRUE);
|
|
EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST), (lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS)?FALSE:TRUE);
|
|
//
|
|
// change "Change Settings" button
|
|
//
|
|
pCheckEnableResourceChange(lpdmpd);
|
|
}
|
|
|
|
BOOL
|
|
pLoadCurrentConfig(
|
|
IN LPDMPROP_DATA lpdmpd,
|
|
BOOL HideIfProb
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Modify the top part, to show current configuration, if any
|
|
|
|
Arguments:
|
|
|
|
hDlg = dialog handle of controls
|
|
lpdmpd = property data
|
|
|
|
Return Value:
|
|
|
|
TRUE if we're showing current config
|
|
|
|
--*/
|
|
{
|
|
TCHAR szMessage[MAX_PATH];
|
|
ULONG Problem;
|
|
ULONG Status;
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
HMACHINE hMachine = pGetMachine(lpdmpd);
|
|
BOOL DoLoadConfig = FALSE;
|
|
|
|
lpdmpd->SelectedLC = 0;
|
|
lpdmpd->SelectedLCType = lpdmpd->CurrentLCType;
|
|
|
|
if (lpdmpd->CurrentLC != 0) {
|
|
DoLoadConfig = TRUE;
|
|
}
|
|
if(HideIfProb && (lpdmpd->dwFlags & DMPROP_FLAG_HASPROBLEM)) {
|
|
//
|
|
// if there's a problem and HideIfProb is TRUE, don't bother showing current config
|
|
//
|
|
DoLoadConfig = FALSE;
|
|
}
|
|
if (DoLoadConfig) {
|
|
//
|
|
// load in current configuration
|
|
//
|
|
pLoadConfig(lpdmpd,lpdmpd->CurrentLC,lpdmpd->CurrentLCType);
|
|
return TRUE;
|
|
}
|
|
//
|
|
// case where there is no suitable configuration
|
|
//
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_RESOURCES_TEXT), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LCTEXT), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_USESYSSETTINGS), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CHANGE), SW_HIDE);
|
|
pShowViewNoAlloc(lpdmpd);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_SETTINGSTATE), SW_SHOW);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NOALLOCTEXT), SW_SHOW);
|
|
|
|
//
|
|
// explain why there is a problem
|
|
// goes into NOALLOCTEXT
|
|
//
|
|
LoadString(MyDllModuleHandle, IDS_DEVRES_NOALLOC_PROBLEM, szMessage, MAX_PATH);
|
|
|
|
//
|
|
// consider being more descriptive
|
|
//
|
|
if ((lpdmpd->lpdi->DevInst==0)
|
|
|| (CM_Get_DevNode_Status_Ex(&Status, &Problem, lpdmpd->lpdi->DevInst,
|
|
0,hMachine) != CR_SUCCESS)) {
|
|
Status = 0;
|
|
Problem = 0;
|
|
}
|
|
|
|
if ((Status & DN_HAS_PROBLEM)!=0) {
|
|
|
|
switch (Problem) {
|
|
case CM_PROB_DISABLED:
|
|
case CM_PROB_HARDWARE_DISABLED:
|
|
LoadString(MyDllModuleHandle, IDS_DEVRES_NOALLOC_DISABLED, szMessage, MAX_PATH);
|
|
break;
|
|
|
|
case CM_PROB_NORMAL_CONFLICT:
|
|
LoadString(MyDllModuleHandle, IDS_DEVRES_NORMAL_CONFLICT, szMessage, MAX_PATH);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
SetDlgItemText(hDlg, IDC_DEVRES_NOALLOCTEXT, szMessage);
|
|
|
|
return FALSE; // display in NoAlloc state
|
|
}
|
|
|
|
BOOL
|
|
pConfigHasNoAlternates(
|
|
LPDMPROP_DATA lpdmpd,
|
|
LOG_CONF testLC
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
A Basic config could be restrictive "these are the set of resources to use"
|
|
This determines if the basic config passed is such a config
|
|
|
|
Arguments:
|
|
|
|
testLC = basic config to test
|
|
|
|
Return Value:
|
|
|
|
TRUE if it's a singular config
|
|
|
|
--*/
|
|
{
|
|
HMACHINE hMachine = NULL;
|
|
PRESDES_ENTRY pConfigValues = NULL;
|
|
PRESDES_ENTRY pValue = NULL;
|
|
BOOL bSuccess = TRUE;
|
|
ULONG64 ulValue = 0, ulLen = 0, ulEnd = 0;
|
|
ULONG ulFlags = 0;
|
|
PGENERIC_RESOURCE pGenRes = NULL;
|
|
|
|
hMachine = pGetMachine(lpdmpd);
|
|
pGetResDesDataList(testLC, &pConfigValues, TRUE, hMachine); // arbitratable resources
|
|
for(pValue = pConfigValues;pValue;pValue = pValue->Next) {
|
|
//
|
|
// is this a singular value?
|
|
//
|
|
pGenRes = (PGENERIC_RESOURCE)(pValue->ResDesData);
|
|
if(pGenRes->GENERIC_Header.GENERIC_Count != 1) {
|
|
//
|
|
// more than one entry - not singular
|
|
//
|
|
bSuccess = FALSE;
|
|
break;
|
|
}
|
|
pGetRangeValues(pValue->ResDesData, pValue->ResDesType, 0, &ulValue, &ulLen, &ulEnd, NULL, &ulFlags);
|
|
if (ulValue+(ulLen-1) != ulEnd) {
|
|
//
|
|
// not singular
|
|
//
|
|
bSuccess = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
pDeleteResDesDataList(pConfigValues);
|
|
|
|
return bSuccess;
|
|
}
|
|
|
|
BOOL
|
|
pLoadConfig(
|
|
LPDMPROP_DATA lpdmpd,
|
|
LOG_CONF forceLC,
|
|
ULONG forceLCType
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Display a configuration
|
|
|
|
Arguments:
|
|
|
|
hDlg = dialog handle of controls
|
|
lpdmpd = property data
|
|
forceLC = LogConf to display
|
|
forceLCType = type for LogConf
|
|
|
|
Return Value:
|
|
|
|
TRUE if config loaded
|
|
|
|
--*/
|
|
{
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
CONFIGRET Status = CR_SUCCESS;
|
|
HWND hWndList;
|
|
LV_ITEM lviItem;
|
|
TCHAR szTemp[MAX_PATH];
|
|
int iNewItem = 0;
|
|
ULONG ulSize,ulFlags;
|
|
ULONG64 ulValue, ulLen, ulEnd, ulAlign;
|
|
ULONG64 ulMaxMem = 0;
|
|
ULONG ulRange;
|
|
LPBYTE pData = NULL;
|
|
RES_DES ResDes;
|
|
RESOURCEID ResType;
|
|
PITEMDATA pItemData = NULL;
|
|
HMACHINE hMachine = NULL;
|
|
PDEVICE_INFO_SET pDeviceInfoSet;
|
|
BOOL RetCode = FALSE;
|
|
PRESDES_ENTRY pKnownValues = NULL;
|
|
PRESDES_ENTRY pShowValues = NULL;
|
|
PRESDES_ENTRY pShowEntry = NULL;
|
|
BOOL bFixedConfig = FALSE;
|
|
BOOL bNoMatch;
|
|
BOOL bFixed;
|
|
ULONG MatchLevel = NO_LC_MATCH;
|
|
|
|
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_SETTINGSTATE), SW_SHOW);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_RESOURCES_TEXT), SW_HIDE);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST), SW_SHOW);
|
|
pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NOALLOCTEXT), SW_HIDE);
|
|
|
|
hWndList = GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST);
|
|
SendMessage(hWndList, WM_SETREDRAW, (WPARAM)FALSE, 0);
|
|
ListView_DeleteAllItems(hWndList);
|
|
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_FIXEDCONFIG; // until we determine there is at least one setting we can edit
|
|
|
|
|
|
if (forceLC == 0) {
|
|
forceLC = lpdmpd->CurrentLC;
|
|
forceLCType = lpdmpd->CurrentLCType;
|
|
}
|
|
if (forceLC == 0) {
|
|
MYASSERT(FALSE);
|
|
goto Final;
|
|
}
|
|
hMachine = pGetMachine(lpdmpd);
|
|
|
|
//
|
|
// setup values that will remain the same each time I add an item
|
|
//
|
|
lviItem.mask = LVIF_TEXT | LVIF_PARAM | LVIF_IMAGE;
|
|
lviItem.pszText = szTemp; // reuse the szTemp buffer
|
|
lviItem.iSubItem = 0;
|
|
lviItem.iImage = IDI_RESOURCE - IDI_RESOURCEFIRST;
|
|
|
|
pGetResDesDataList(forceLC, &pShowValues, TRUE, hMachine); // editable resources
|
|
if (forceLCType == BOOT_LOG_CONF || forceLCType == FORCED_LOG_CONF || forceLCType == ALLOC_LOG_CONF) {
|
|
bFixedConfig = TRUE;
|
|
if (forceLC == lpdmpd->CurrentLC && lpdmpd->MatchingLC != 0) {
|
|
//
|
|
// we're displaying CurrentLC, use flags & resdes's from matching LC where possible
|
|
//
|
|
if (pGetResDesDataList(lpdmpd->MatchingLC, &pKnownValues, TRUE, hMachine)) {
|
|
//
|
|
// match-up currentLC with some matching LC, so we can use flags/ranges from matching LC
|
|
//
|
|
MatchLevel = pMergeResDesDataLists(pShowValues,pKnownValues,NULL);
|
|
}
|
|
}
|
|
} else if (lpdmpd->CurrentLC != 0) {
|
|
//
|
|
// the config we're displaying may allow ranges of values
|
|
// we're going to try and match up what we are displaying to current config
|
|
//
|
|
if (pGetResDesDataList(lpdmpd->CurrentLC, &pKnownValues, TRUE, hMachine)) {
|
|
//
|
|
// try and use current values where possible
|
|
//
|
|
MatchLevel = pMergeResDesDataLists(pKnownValues,pShowValues,NULL);
|
|
}
|
|
}
|
|
|
|
pShowEntry = pShowValues;
|
|
|
|
while (pShowEntry) {
|
|
bNoMatch = FALSE;
|
|
bFixed = FALSE;
|
|
ResDes = (RES_DES)0;
|
|
ResType = pShowEntry->ResDesType;
|
|
ulRange = 0;
|
|
|
|
if (bFixedConfig) {
|
|
//
|
|
// we've got a current config
|
|
//
|
|
pGetHdrValues(pShowEntry->ResDesData, pShowEntry->ResDesType, &ulValue, &ulLen, &ulEnd, &ulFlags);
|
|
if((ResType ==ResType_Mem) && (ulEnd > ulMaxMem)) {
|
|
//
|
|
// base our memory display on the fixed-config only
|
|
//
|
|
ulMaxMem = ulEnd;
|
|
}
|
|
if (pShowEntry->CrossLink) {
|
|
//
|
|
// use range's res-des
|
|
//
|
|
ResDes = pShowEntry->CrossLink->ResDesHandle;
|
|
pShowEntry->CrossLink->ResDesHandle = (RES_DES)0;
|
|
//
|
|
// allow adjustment based on nearest basic config
|
|
//
|
|
pGetMatchingRange(ulValue,ulLen,pShowEntry->CrossLink->ResDesData, pShowEntry->CrossLink->ResDesType,&ulRange,&bFixed,NULL);
|
|
} else {
|
|
//
|
|
// no range res-des
|
|
//
|
|
ResDes = (RES_DES)0;
|
|
//
|
|
// indicate that this is a non-adjustable value
|
|
//
|
|
bFixed = TRUE;
|
|
}
|
|
} else {
|
|
//
|
|
// we've got resource-ranges
|
|
//
|
|
if (pShowEntry->CrossLink) {
|
|
//
|
|
// take current settings from what we merged in
|
|
//
|
|
pGetHdrValues(pShowEntry->CrossLink->ResDesData, pShowEntry->CrossLink->ResDesType, &ulValue, &ulLen, &ulEnd, &ulFlags);
|
|
} else {
|
|
//
|
|
// just take first range
|
|
//
|
|
pGetRangeValues(pShowEntry->ResDesData, pShowEntry->ResDesType, 0, &ulValue, &ulLen, &ulEnd, &ulAlign, &ulFlags);
|
|
}
|
|
//
|
|
// range check if this is a memory resource - find highest memory value
|
|
//
|
|
if(ResType ==ResType_Mem) {
|
|
ULONG64 HighVal;
|
|
ULONG64 HighLen;
|
|
if(LocateClosestValue(pShowEntry->ResDesData, pShowEntry->ResDesType,(ULONG64)(-1),ulLen, 0 ,&HighVal,&HighLen,NULL)) {
|
|
HighVal += HighLen-1;
|
|
if(HighVal > ulMaxMem) {
|
|
ulMaxMem = ulEnd;
|
|
}
|
|
}
|
|
}
|
|
|
|
pGetMatchingRange(ulValue,ulLen,pShowEntry->ResDesData, pShowEntry->ResDesType,&ulRange,&bFixed,&ulFlags);
|
|
//
|
|
// use res-des from range
|
|
//
|
|
ResDes = pShowEntry->ResDesHandle;
|
|
pShowEntry->ResDesHandle = (RES_DES)0;
|
|
|
|
if (pShowEntry->CrossLink == NULL && bFixed == FALSE) {
|
|
//
|
|
// unknown value
|
|
//
|
|
bNoMatch = TRUE;
|
|
}
|
|
}
|
|
|
|
if (ulLen>0) {
|
|
//
|
|
// Write first column text field (uses szTemp, lParam is res type)
|
|
//
|
|
LoadString(MyDllModuleHandle, IDS_RESOURCE_BASE + ResType, szTemp, MAX_PATH);
|
|
ulRange = 0;
|
|
|
|
pItemData = (PITEMDATA)MyMalloc(sizeof(ITEMDATA));
|
|
if (pItemData != NULL) {
|
|
pItemData->ResType = ResType;
|
|
pItemData->MatchingResDes = ResDes;
|
|
pItemData->RangeCount = ulRange;
|
|
pItemData->ulValue = ulValue; // selected value
|
|
pItemData->ulLen = ulLen;
|
|
pItemData->ulEnd = ulValue + ulLen - 1;
|
|
pItemData->ulFlags = ulFlags;
|
|
pItemData->bValid = !bNoMatch; // if no chosen value
|
|
pItemData->bFixed = bFixed;
|
|
}
|
|
if (bFixed == FALSE) {
|
|
//
|
|
// we have at least one editable value
|
|
//
|
|
lpdmpd->dwFlags &= ~DMPROP_FLAG_FIXEDCONFIG;
|
|
}
|
|
|
|
lviItem.iItem = iNewItem;
|
|
lviItem.lParam = (LPARAM)pItemData;
|
|
ListView_InsertItem(hWndList, &lviItem);
|
|
|
|
//
|
|
// Write second column text field (uses szTemp, lParam is res handle)
|
|
//
|
|
if (bNoMatch) {
|
|
pFormatResString(lpdmpd,szTemp, 0, 0, ResType);
|
|
} else {
|
|
pFormatResString(lpdmpd,szTemp, ulValue, ulLen, ResType);
|
|
}
|
|
ListView_SetItemText(hWndList, iNewItem, 1, szTemp);
|
|
|
|
++iNewItem;
|
|
}
|
|
pShowEntry = pShowEntry->Next;
|
|
}
|
|
|
|
if(ulMaxMem > 0xFFFFFFFF) {
|
|
//
|
|
// an excuse to require 64-bit address range
|
|
//
|
|
lpdmpd->dwFlags |= DMPROP_FLAG_64BIT_RANGE;
|
|
}
|
|
|
|
SendMessage(hWndList, WM_SETREDRAW, (WPARAM)TRUE, 0);
|
|
|
|
RetCode = TRUE;
|
|
|
|
Final:
|
|
|
|
pDeleteResDesDataList(pKnownValues);
|
|
pDeleteResDesDataList(pShowValues);
|
|
|
|
//
|
|
// initialize listview headings here
|
|
//
|
|
ListView_SetColumnWidth(hWndList, 0, LVSCW_AUTOSIZE_USEHEADER);
|
|
ListView_SetColumnWidth(hWndList, 1, LVSCW_AUTOSIZE_USEHEADER);
|
|
//
|
|
// change "Change Settings" button
|
|
//
|
|
pCheckEnableResourceChange(lpdmpd);
|
|
|
|
return RetCode;
|
|
}
|
|
|
|
BOOL
|
|
bIsMultiFunctionChild(
|
|
PSP_DEVINFO_DATA lpdi,
|
|
HMACHINE hMachine
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Returns flag indicating if this is a child of a
|
|
multifunction device
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
TRUE if MF child
|
|
|
|
--*/
|
|
{
|
|
ULONG Status;
|
|
ULONG ProblemNumber;
|
|
|
|
if (lpdi->DevInst) {
|
|
|
|
if (CM_Get_DevNode_Status_Ex(&Status, &ProblemNumber,
|
|
lpdi->DevInst, 0,hMachine) == CR_SUCCESS) {
|
|
//
|
|
// If the passed in dev is not an MF child, then it is the top
|
|
// level MF_Parent
|
|
//
|
|
if (Status & DN_MF_CHILD) {
|
|
return TRUE;
|
|
} else {
|
|
return FALSE;
|
|
}
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
VOID
|
|
pSelectLogConf(
|
|
LPDMPROP_DATA lpdmpd,
|
|
LOG_CONF forceLC,
|
|
ULONG forceLCType,
|
|
BOOL Always
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Selects a LogConf, showing the config in the LC control
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
TRUE if MF child
|
|
|
|
--*/
|
|
{
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
int count;
|
|
int i;
|
|
LOG_CONF LogConf;
|
|
|
|
if (Always == FALSE && forceLC == lpdmpd->SelectedLC) {
|
|
//
|
|
// selection remains the same
|
|
//
|
|
return;
|
|
}
|
|
|
|
count = (int)SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST,CB_GETCOUNT, (WPARAM)0, (LPARAM)0);
|
|
if (count == 0) {
|
|
MYASSERT(FALSE/*shouldn't get here*/);
|
|
pLoadCurrentConfig(lpdmpd,FALSE);
|
|
return;
|
|
}
|
|
|
|
if (forceLC == 0 && lpdmpd->CurrentLC == 0) {
|
|
//
|
|
// no currentLC, so select first default
|
|
//
|
|
forceLC = (LOG_CONF)SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST,
|
|
CB_GETITEMDATA, (WPARAM)0, (LPARAM)0);
|
|
if (forceLC == (LOG_CONF)0) {
|
|
MYASSERT(FALSE/*shouldn't get here*/);
|
|
pLoadCurrentConfig(lpdmpd,FALSE);
|
|
return;
|
|
}
|
|
forceLCType = lpdmpd->ConfigListLCType;
|
|
}
|
|
|
|
for (i=0;i<count;i++) {
|
|
LogConf = (LOG_CONF)SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST,
|
|
CB_GETITEMDATA, (WPARAM)i, (LPARAM)0);
|
|
if (LogConf == forceLC) {
|
|
//
|
|
// set these first so we don't recurse around
|
|
//
|
|
lpdmpd->SelectedLC = forceLC;
|
|
lpdmpd->SelectedLCType = forceLCType;
|
|
//
|
|
// change dialog to reflect new selection
|
|
//
|
|
SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST,CB_SETCURSEL, (WPARAM)i, (LPARAM)0);
|
|
pLoadConfig(lpdmpd,forceLC,forceLCType);
|
|
pShowConflicts(lpdmpd);
|
|
return;
|
|
}
|
|
}
|
|
SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST,CB_SETCURSEL, (WPARAM)(-1), (LPARAM)0);
|
|
pLoadConfig(lpdmpd,forceLC,forceLCType);
|
|
pShowConflicts(lpdmpd);
|
|
}
|
|
|
|
|
|
VOID
|
|
pChangeCurrentResSetting(
|
|
IN LPDMPROP_DATA lpdmpd
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Brings up edit dialog to change currently selected resource
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
none
|
|
|
|
--*/
|
|
{
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
RESOURCEEDITINFO rei;
|
|
HWND hList = GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST);
|
|
int iCur;
|
|
PITEMDATA pItemData = NULL;
|
|
LV_ITEM lviItem;
|
|
GENERIC_RESOURCE GenResInfo;
|
|
PDEVICE_INFO_SET pDeviceInfoSet;
|
|
BOOL changed = FALSE;
|
|
TCHAR szTemp[MAX_PATH];
|
|
|
|
pItemData = pGetResourceToChange(lpdmpd,&iCur);
|
|
if (pItemData == NULL) {
|
|
//
|
|
// we cannot edit this resource for some reason, give the user a hint
|
|
// and maybe I'll get less ear-ache "I cannot change the settings"
|
|
//
|
|
if ((lpdmpd->dwFlags & DMPROP_FLAG_VIEWONLYRES)!=0 ||
|
|
(lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS)!=0) {
|
|
//
|
|
// editing not allowed - prob double-clicked on settings
|
|
//
|
|
return;
|
|
}
|
|
if (lpdmpd->dwFlags & DMPROP_FLAG_FIXEDCONFIG) {
|
|
pWarnResSettingNotEditable(hDlg, IDS_DEVRES_NOMODIFYALL);
|
|
} else {
|
|
//
|
|
// see if user needs to select a resource
|
|
//
|
|
iCur = (int)ListView_GetNextItem(hList,-1, LVNI_SELECTED);
|
|
if (iCur == LB_ERR) {
|
|
//
|
|
// no selection
|
|
//
|
|
pWarnResSettingNotEditable(hDlg, IDS_DEVRES_NOMODIFYSELECT);
|
|
} else {
|
|
//
|
|
// resource is just not editable
|
|
//
|
|
pWarnResSettingNotEditable(hDlg, IDS_DEVRES_NOMODIFYSINGLE);
|
|
}
|
|
}
|
|
goto clean0;
|
|
}
|
|
|
|
ZeroMemory(&rei,sizeof(rei));
|
|
rei.hMachine = pGetMachine(lpdmpd);
|
|
rei.KnownLC = lpdmpd->CurrentLC;
|
|
rei.MatchingBasicLC = lpdmpd->MatchingLC;
|
|
rei.SelectedBasicLC = lpdmpd->SelectedLC;
|
|
rei.lpdi = lpdmpd->lpdi;
|
|
rei.dwPropFlags = lpdmpd->dwFlags;
|
|
rei.bShareable = FALSE;
|
|
rei.ridResType = pItemData->ResType;
|
|
rei.ResDes = pItemData->MatchingResDes;
|
|
rei.ulCurrentVal = pItemData->ulValue;
|
|
rei.ulCurrentLen = pItemData->ulLen;
|
|
rei.ulCurrentEnd = pItemData->ulEnd;
|
|
rei.ulCurrentFlags = pItemData->ulFlags;
|
|
rei.ulRangeCount = pItemData->RangeCount;
|
|
rei.pData = NULL;
|
|
|
|
if (DialogBoxParam(MyDllModuleHandle,
|
|
MAKEINTRESOURCE(IDD_EDIT_RESOURCE),
|
|
hDlg,
|
|
EditResourceDlgProc,
|
|
(LPARAM)(PRESOURCEEDITINFO)&rei) != IDOK) {
|
|
goto clean0;
|
|
}
|
|
//
|
|
// Update The Current Resource settings to Future
|
|
// Settings, and update the Conflict list.
|
|
//
|
|
pItemData->ulValue = rei.ulCurrentVal;
|
|
pItemData->ulLen = rei.ulCurrentLen;
|
|
pItemData->ulEnd = rei.ulCurrentEnd;
|
|
pItemData->ulFlags = rei.ulCurrentFlags;
|
|
pItemData->RangeCount = rei.ulRangeCount;
|
|
pItemData->bValid = TRUE; // indicate that user has explicitly changed this value
|
|
|
|
pFormatResString(lpdmpd,szTemp,
|
|
rei.ulCurrentVal,
|
|
rei.ulCurrentLen,
|
|
rei.ridResType);
|
|
|
|
ListView_SetItemText(hList, iCur, 1, szTemp);
|
|
pShowConflicts(lpdmpd);
|
|
|
|
//
|
|
// clear the flag for saving changes
|
|
//
|
|
lpdmpd->dwFlags &= ~DMPROP_FLAG_CHANGESSAVED;
|
|
PropSheet_Changed(GetParent(hDlg), hDlg);
|
|
|
|
clean0:
|
|
;
|
|
}
|
|
|
|
VOID
|
|
pShowConflicts(
|
|
IN LPDMPROP_DATA lpdmpd
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Selects a LogConf, showing the config in the LC control
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
TRUE if MF child
|
|
|
|
--*/
|
|
{
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
CONFIGRET Status = CR_SUCCESS;
|
|
LPVOID vaArray[4];
|
|
TCHAR szTemp[MAX_PATH+4], szBuffer[MAX_PATH+16], szSetting[MAX_PATH];
|
|
TCHAR szFormat[MAX_PATH], szItemFormat[MAX_PATH];
|
|
TCHAR szUnavailable[MAX_PATH];
|
|
LPTSTR pszConflictList = NULL, pszConflictList2 = NULL;
|
|
ULONG ulSize = 0, ulLength, ulBufferLen, ulNewLength;
|
|
ULONG ulStartOffset = 0;
|
|
int Count = 0, i = 0;
|
|
PITEMDATA pItemData = NULL;
|
|
LPBYTE pResourceData = NULL;
|
|
HWND hwndResList = GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST);
|
|
HMACHINE hMachine;
|
|
ULONG ConflictCount = 0;
|
|
ULONG ConflictIndex = 0;
|
|
CONFLICT_LIST ConflictList = 0;
|
|
PDEVICE_INFO_SET pDeviceInfoSet;
|
|
CONFLICT_DETAILS ConflictDetails;
|
|
BOOL ReservedResource = FALSE;
|
|
BOOL AnyReportedResources = FALSE;
|
|
BOOL AnyBadResources = FALSE;
|
|
PCONFLICT_EXCEPTIONS pConflictExceptions = NULL;
|
|
//
|
|
// number of resources listed
|
|
//
|
|
Count = ListView_GetItemCount(hwndResList);
|
|
if (Count <= 0) {
|
|
goto Clean0;
|
|
}
|
|
|
|
//
|
|
// initial buffer that holds the strings
|
|
// with all the conflict info in them
|
|
//
|
|
ulBufferLen = 2048;
|
|
ulLength = 0;
|
|
|
|
pszConflictList = MyMalloc(ulBufferLen * sizeof(TCHAR));
|
|
if (pszConflictList == NULL) {
|
|
goto Clean0;
|
|
}
|
|
pszConflictList[0] = 0;
|
|
|
|
//
|
|
// obtain machine
|
|
//
|
|
if(!(pDeviceInfoSet = AccessDeviceInfoSet(lpdmpd->hDevInfo))) {
|
|
goto Clean0;
|
|
}
|
|
hMachine = pDeviceInfoSet->hMachine;
|
|
UnlockDeviceInfoSet (pDeviceInfoSet);
|
|
|
|
//
|
|
// do these once - these format strings use %1!s! type formats (FormatMessage)
|
|
//
|
|
LoadString(MyDllModuleHandle, IDS_CONFLICT_FMT, szFormat, MAX_PATH);
|
|
LoadString(MyDllModuleHandle, IDS_CONFLICT_UNAVAILABLE, szUnavailable, MAX_PATH);
|
|
|
|
//
|
|
// for every listed resource
|
|
//
|
|
|
|
for (i = 0; i < Count; i++) {
|
|
|
|
ConflictList = 0;
|
|
ConflictCount = 0;
|
|
|
|
//
|
|
// get the resource we're about to test
|
|
//
|
|
pItemData = (PITEMDATA)pGetListViewItemData(hwndResList, i, 0);
|
|
if (pItemData == NULL || pItemData->bValid == FALSE) {
|
|
//
|
|
// for whatever reason, we don't want to show conflict information on this resource
|
|
//
|
|
ListView_SetItemState(hwndResList, i,
|
|
INDEXTOOVERLAYMASK(0),
|
|
LVIS_OVERLAYMASK);
|
|
goto NextResource;
|
|
}
|
|
|
|
//
|
|
// this is set to indicate conflict not reported, but is reserved
|
|
//
|
|
ReservedResource = FALSE;
|
|
|
|
//
|
|
// need resource-data for determining conflict
|
|
//
|
|
if (MakeResourceData(&pResourceData, &ulSize,
|
|
pItemData->ResType,
|
|
pItemData->ulValue,
|
|
pItemData->ulLen,
|
|
pItemData->ulFlags)) {
|
|
|
|
Status = CM_Query_Resource_Conflict_List(&ConflictList,
|
|
lpdmpd->lpdi->DevInst,
|
|
pItemData->ResType,
|
|
pResourceData,
|
|
ulSize,
|
|
DEVRES_WIDTH_FLAGS,
|
|
hMachine);
|
|
|
|
if (Status != CR_SUCCESS) {
|
|
//
|
|
// on the unlikely event of an error, remember an error occurred
|
|
//
|
|
ConflictList = 0;
|
|
ConflictCount = 0;
|
|
AnyBadResources = TRUE;
|
|
} else {
|
|
//
|
|
// find out how many things conflicted
|
|
//
|
|
Status = CM_Get_Resource_Conflict_Count(ConflictList,&ConflictCount);
|
|
if (Status != CR_SUCCESS) {
|
|
MYASSERT(Status == CR_SUCCESS);
|
|
ConflictCount = 0;
|
|
AnyBadResources = TRUE;
|
|
}
|
|
}
|
|
if(ConflictCount && (lpdmpd->dwFlags & DMPROP_FLAG_SINGLE_CONFIG) && !(lpdmpd->dwFlags & DMPROP_FLAG_HASPROBLEM)) {
|
|
//
|
|
// NTRAID#166214-2000/08/19-JamieHun Conflict Supression Hack
|
|
//
|
|
// rules are
|
|
// (1) device doesn't have a problem
|
|
// (2) device can only have one configuration (ie, there's no basic config, or the basic config is singular)
|
|
// (3) it has a ResourcePickerExceptions string, and that string indicates that the exception is allowed for the specific conflict
|
|
|
|
if(pConflictExceptions==NULL) {
|
|
pConflictExceptions = pLoadConflictExceptions(lpdmpd);
|
|
}
|
|
|
|
if (pConflictExceptions) {
|
|
|
|
BOOL muted = TRUE;
|
|
//
|
|
// count from 0 (first conflict) through to ConflictCount (excl)
|
|
//
|
|
for(ConflictIndex = 0; ConflictIndex < ConflictCount; ConflictIndex ++) {
|
|
//
|
|
// obtain details for this conflict
|
|
//
|
|
ZeroMemory(&ConflictDetails,sizeof(ConflictDetails));
|
|
ConflictDetails.CD_ulSize = sizeof(ConflictDetails);
|
|
ConflictDetails.CD_ulMask = CM_CDMASK_DEVINST | CM_CDMASK_DESCRIPTION | CM_CDMASK_FLAGS;
|
|
Status = CM_Get_Resource_Conflict_Details(ConflictList,ConflictIndex,&ConflictDetails);
|
|
if (Status == CR_SUCCESS) {
|
|
if (!pIsConflictException(lpdmpd,pConflictExceptions,ConflictDetails.CD_dnDevInst,ConflictDetails.CD_szDescription,pItemData->ResType,pItemData->ulValue,pItemData->ulLen)) {
|
|
muted = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if(muted) {
|
|
ConflictCount = 0;
|
|
}
|
|
}
|
|
}
|
|
if (ConflictCount || ReservedResource) {
|
|
ulStartOffset = ulLength; // record start in case we decide to backtrack
|
|
AnyReportedResources = TRUE; // say we reported at least one problem
|
|
|
|
TreatAsReserved:
|
|
|
|
ulLength = ulStartOffset;
|
|
pszConflictList[ulLength] = 0;
|
|
//
|
|
// we're going to mark the resource as a problem
|
|
//
|
|
|
|
ListView_GetItemText(hwndResList, i, 1, szSetting, MAX_PATH);
|
|
|
|
|
|
switch (pItemData->ResType) {
|
|
case ResType_Mem:
|
|
LoadString(MyDllModuleHandle, IDS_MEMORY_FULL, szBuffer, MAX_PATH);
|
|
break;
|
|
case ResType_IO:
|
|
LoadString(MyDllModuleHandle, IDS_IO_FULL, szBuffer, MAX_PATH);
|
|
break;
|
|
case ResType_DMA:
|
|
LoadString(MyDllModuleHandle, IDS_DMA_FULL, szBuffer, MAX_PATH);
|
|
break;
|
|
case ResType_IRQ:
|
|
LoadString(MyDllModuleHandle, IDS_IRQ_FULL, szBuffer, MAX_PATH);
|
|
break;
|
|
default:
|
|
MYASSERT(FALSE);
|
|
}
|
|
|
|
if ( ReservedResource == FALSE) {
|
|
|
|
//
|
|
// count from 0 (header) 1 (first conflict) through to ConflictCount
|
|
//
|
|
for(ConflictIndex = 0; ConflictIndex <= ConflictCount; ConflictIndex ++) {
|
|
if (ConflictIndex == 0) {
|
|
//
|
|
// first pass through, do header message
|
|
//
|
|
vaArray[0] = szBuffer;
|
|
vaArray[1] = szSetting;
|
|
FormatMessage(FORMAT_MESSAGE_FROM_STRING | FORMAT_MESSAGE_ARGUMENT_ARRAY ,
|
|
szFormat,
|
|
0,0,
|
|
szTemp,MAX_PATH,
|
|
(va_list*)vaArray); // FORMAT_MESSAGE_ARGUMENT_ARRAY
|
|
|
|
} else {
|
|
|
|
//
|
|
// obtain details for this conflict
|
|
//
|
|
ZeroMemory(&ConflictDetails,sizeof(ConflictDetails));
|
|
ConflictDetails.CD_ulSize = sizeof(ConflictDetails);
|
|
ConflictDetails.CD_ulMask = CM_CDMASK_DEVINST | CM_CDMASK_DESCRIPTION | CM_CDMASK_FLAGS;
|
|
|
|
Status = CM_Get_Resource_Conflict_Details(ConflictList,ConflictIndex-1,&ConflictDetails);
|
|
if (Status == CR_SUCCESS) {
|
|
if ((ConflictDetails.CD_ulFlags & CM_CDFLAGS_RESERVED) != 0) {
|
|
//
|
|
// treat as reserved - backtrack
|
|
//
|
|
ReservedResource = TRUE;
|
|
goto TreatAsReserved;
|
|
} else {
|
|
if (ConflictDetails.CD_szDescription[0] == 0) {
|
|
//
|
|
// treat as reserved - backtrack
|
|
//
|
|
ReservedResource = TRUE;
|
|
goto TreatAsReserved;
|
|
}
|
|
wsprintf(szBuffer,TEXT(" %s\r\n"),ConflictDetails.CD_szDescription);
|
|
}
|
|
} else {
|
|
//
|
|
// treat as reserved
|
|
//
|
|
ReservedResource = TRUE;
|
|
goto TreatAsReserved;
|
|
}
|
|
lstrcpyn(szTemp,szBuffer,MAX_PATH);
|
|
}
|
|
|
|
ulNewLength = ulLength + lstrlen(szTemp); // excluding terminating NUL
|
|
|
|
if ((ulNewLength+1) < ulBufferLen) {
|
|
//
|
|
// need to allocate more space - we'll double it and add some more every time
|
|
//
|
|
pszConflictList2 = MyRealloc(pszConflictList,(ulBufferLen+ulNewLength+1) * sizeof(TCHAR));
|
|
if (pszConflictList2 != NULL) {
|
|
//
|
|
// succeeded in resizing buffer
|
|
//
|
|
pszConflictList = pszConflictList2;
|
|
ulBufferLen = ulBufferLen+ulNewLength+1;
|
|
}
|
|
}
|
|
if ((ulNewLength+1) < ulBufferLen) {
|
|
lstrcpy(pszConflictList + ulLength , szTemp);
|
|
ulLength = ulNewLength;
|
|
}
|
|
|
|
}
|
|
} else {
|
|
//
|
|
// there is some other problem with resource
|
|
//
|
|
|
|
vaArray[0] = szBuffer;
|
|
vaArray[1] = szSetting;
|
|
FormatMessage(FORMAT_MESSAGE_FROM_STRING | FORMAT_MESSAGE_ARGUMENT_ARRAY ,
|
|
szUnavailable,
|
|
0,0,
|
|
szTemp,MAX_PATH,
|
|
(va_list*)vaArray); // FORMAT_MESSAGE_ARGUMENT_ARRAY
|
|
|
|
ulNewLength = ulLength + lstrlen(szTemp); // excluding terminating NUL
|
|
|
|
if ((ulNewLength+1) < ulBufferLen) {
|
|
//
|
|
// need to allocate more space - we'll double it and add some more every time
|
|
//
|
|
pszConflictList2 = MyRealloc(pszConflictList,(ulBufferLen+ulNewLength+1) * sizeof(TCHAR));
|
|
if (pszConflictList2 != NULL) {
|
|
//
|
|
// succeeded in resizing buffer
|
|
//
|
|
pszConflictList = pszConflictList2;
|
|
ulBufferLen = ulBufferLen+ulNewLength+1;
|
|
}
|
|
}
|
|
if ((ulNewLength+1) < ulBufferLen) {
|
|
lstrcpy(pszConflictList + ulLength , szTemp);
|
|
ulLength = ulNewLength;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Set the Conflict Overlay for this resource.
|
|
//
|
|
ListView_SetItemState(hwndResList, i,
|
|
INDEXTOOVERLAYMASK(IDI_CONFLICT - IDI_RESOURCEOVERLAYFIRST + 1),
|
|
LVIS_OVERLAYMASK);
|
|
|
|
} else {
|
|
//
|
|
// resource is (aparently) working fine
|
|
//
|
|
ListView_SetItemState(hwndResList, i,
|
|
INDEXTOOVERLAYMASK(0),
|
|
LVIS_OVERLAYMASK);
|
|
}
|
|
|
|
if (ConflictList) {
|
|
CM_Free_Resource_Conflict_Handle(ConflictList);
|
|
}
|
|
|
|
if (pResourceData != NULL) {
|
|
MyFree(pResourceData);
|
|
}
|
|
} else {
|
|
//
|
|
// couldn't make the resource descriptor
|
|
AnyBadResources = TRUE;
|
|
}
|
|
|
|
NextResource:
|
|
;
|
|
}
|
|
|
|
|
|
Clean0:
|
|
;
|
|
|
|
//
|
|
// If there were any conflicts, put the list in the multiline edit box.
|
|
//
|
|
if (AnyReportedResources) {
|
|
SetDlgItemText(hDlg, IDC_DEVRES_CONFLICTINFOLIST, pszConflictList);
|
|
} else if (AnyBadResources) {
|
|
//
|
|
// this would most likely occur on
|
|
// (1) running this on 95/98 (shouldn't happen)
|
|
// (2) using new setupapi on old cfgmgr32
|
|
//
|
|
LoadString(MyDllModuleHandle, IDS_CONFLICT_GENERALERROR, szBuffer, MAX_PATH);
|
|
SetDlgItemText(hDlg, IDC_DEVRES_CONFLICTINFOLIST, szBuffer);
|
|
} else {
|
|
LoadString(MyDllModuleHandle, IDS_DEVRES_NOCONFLICTDEVS, szBuffer, MAX_PATH);
|
|
SetDlgItemText(hDlg, IDC_DEVRES_CONFLICTINFOLIST, szBuffer);
|
|
}
|
|
if(pszConflictList != NULL) {
|
|
MyFree(pszConflictList);
|
|
}
|
|
if (pConflictExceptions != NULL) {
|
|
pFreeConflictExceptions(pConflictExceptions);
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
int
|
|
pOkToSave(
|
|
IN LPDMPROP_DATA lpdmpd
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Check to see if there's something the user hasn't done
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
IDYES = save settings
|
|
IDNO = don't save settings
|
|
IDCANCEL = don't exit
|
|
|
|
--*/
|
|
{
|
|
HWND hDlg = lpdmpd->hDlg;
|
|
HWND hList = GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST);
|
|
int iCur;
|
|
int nRes;
|
|
PITEMDATA pItemData;
|
|
|
|
if (lpdmpd->dwFlags & DMPROP_FLAG_NO_RESOURCES) {
|
|
//
|
|
// no changes - because there are no resources
|
|
//
|
|
return IDNO;
|
|
}
|
|
if (lpdmpd->dwFlags & DMPROP_FLAG_CHANGESSAVED) {
|
|
//
|
|
// no changes
|
|
//
|
|
return IDNO;
|
|
}
|
|
if (lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS) {
|
|
//
|
|
// always ok to "use sys settings"
|
|
//
|
|
return IDYES;
|
|
}
|
|
//
|
|
// user is forcing a config - let's see if all settings are valid
|
|
//
|
|
//
|
|
// The resource descriptors are out-of-order. Maintain the original ordering.
|
|
//
|
|
// First, build up a linked list of the data in the listview resource items.
|
|
//
|
|
iCur = (int)ListView_GetNextItem(hList, -1, LVNI_ALL);
|
|
|
|
while (iCur >= 0) {
|
|
|
|
pItemData = (PITEMDATA)pGetListViewItemData(hList, iCur, 0);
|
|
if (pItemData) {
|
|
if (pItemData->bValid == FALSE) {
|
|
//
|
|
// we've got an invalid entry - can't save
|
|
//
|
|
nRes = pWarnNoSave(hDlg,IDS_FORCEDCONFIG_PARTIAL);
|
|
if (nRes != IDOK) {
|
|
return IDCANCEL;
|
|
}
|
|
return IDNO;
|
|
}
|
|
}
|
|
|
|
iCur = (int)ListView_GetNextItem(hList, iCur, LVNI_ALL);
|
|
}
|
|
|
|
//
|
|
// everything checks out
|
|
//
|
|
|
|
return IDYES;
|
|
}
|
|
|