Leaked source code of windows server 2003
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.

4704 lines
148 KiB

  1. /*++
  2. Copyright (c) 1995-2000 Microsoft Corporation
  3. Module Name:
  4. devres.c
  5. Abstract:
  6. Routines for displaying resource dialogs.
  7. Author:
  8. Paula Tomlinson (paulat) 7-Feb-1996
  9. Revision History:
  10. Jamie Hunter (jamiehun) 19-Mar-1998
  11. --*/
  12. #include "precomp.h"
  13. #pragma hdrstop
  14. //
  15. // Private Prototypes
  16. //
  17. //
  18. // Global Data
  19. //
  20. static INTERFACE_TYPE ResourcePickerReadOnlyInterfaces[] = {
  21. //
  22. // List of interface-types that we don't want user to edit properties of
  23. //
  24. PCIBus,
  25. //
  26. // End of list
  27. //
  28. InterfaceTypeUndefined
  29. };
  30. static const BOOL ResTypeEditable[ResType_MAX+1] = {
  31. //
  32. // lists resource types that are shown and are editable
  33. FALSE, // ResType_None
  34. TRUE, // ResType_Mem
  35. TRUE, // ResType_IO
  36. TRUE, // ResType_DMA
  37. TRUE, // ResType_IRQ
  38. FALSE, // ResType_DoNotUse
  39. FALSE // ResType_BusNumber
  40. };
  41. #if (ResType_MAX+1) != 7
  42. #error Fix SetupAPI devres.c, ResType_MAX has changed
  43. #endif
  44. //
  45. // HELP ID's
  46. //
  47. static const DWORD DevResHelpIDs[]=
  48. {
  49. IDC_DEVRES_ICON, IDH_NOHELP, // "Low (%d)" (Static)
  50. IDC_DEVRES_DEVDESC, IDH_NOHELP,
  51. IDC_DEVRES_SETTINGSTATE, IDH_DEVMGR_RESOURCES_SETTINGS,
  52. IDC_DEVRES_SETTINGSLIST, IDH_DEVMGR_RESOURCES_SETTINGS,
  53. IDC_DEVRES_LCTEXT, IDH_DEVMGR_RESOURCES_BASEDON,
  54. IDC_DEVRES_LOGCONFIGLIST, IDH_DEVMGR_RESOURCES_BASEDON,
  55. IDC_DEVRES_CHANGE, IDH_DEVMGR_RESOURCES_CHANGE,
  56. IDC_DEVRES_USESYSSETTINGS, IDH_DEVMGR_RESOURCES_AUTO,
  57. IDC_DEVRES_CONFLICTDEVTEXT, IDH_DEVMGR_RESOURCES_CONFLICTS,
  58. IDC_DEVRES_CONFLICTINFOLIST, IDH_DEVMGR_RESOURCES_CONFLICTS,
  59. IDC_DEVRES_MFPARENT, IDH_DEVMGR_RESOURCES_PARENT,
  60. IDC_DEVRES_MFPARENT_DESC, IDH_DEVMGR_RESOURCES_PARENT,
  61. IDC_DEVRES_MAKEFORCED, IDH_DEVMGR_RESOURCES_SETMANUALLY,
  62. 0, 0
  63. };
  64. //
  65. // HACKHACK (jamiehun)
  66. // after we've changed UI from a MakeForced, we post this message to get back control of keyboard
  67. //
  68. #define WM_USER_FOCUS (WM_USER+101)
  69. //
  70. // API to obtain a resource-picker page
  71. //
  72. HPROPSHEETPAGE
  73. GetResourceSelectionPage(
  74. IN HDEVINFO DeviceInfoSet,
  75. IN PSP_DEVINFO_DATA DeviceInfoData
  76. )
  77. {
  78. LPDMPROP_DATA pdmData;
  79. PROPSHEETPAGE PropPage;
  80. //
  81. // private data
  82. // anything we "do" here must be "undone" in pResourcePickerPropPageCallback
  83. //
  84. pdmData = (LPDMPROP_DATA)MyMalloc(sizeof(DMPROP_DATA));
  85. if (pdmData == NULL) {
  86. return NULL;
  87. }
  88. ZeroMemory(pdmData,sizeof(DMPROP_DATA));
  89. pdmData->hDevInfo = DeviceInfoSet;
  90. pdmData->lpdi = DeviceInfoData;
  91. //
  92. // validate expectations
  93. //
  94. MYASSERT(pdmData->hDevInfo != NULL);
  95. MYASSERT(pdmData->lpdi != NULL);
  96. MYASSERT(pdmData->lpdi->DevInst != 0);
  97. ZeroMemory(&PropPage,sizeof(PropPage));
  98. //
  99. // create the Resources Property Page
  100. //
  101. PropPage.dwSize = sizeof(PROPSHEETPAGE);
  102. PropPage.dwFlags = PSP_DEFAULT | PSP_USECALLBACK;
  103. PropPage.hInstance = MyDllModuleHandle;
  104. PropPage.pszTemplate = MAKEINTRESOURCE(IDD_DEF_DEVRESOURCE_PROP);
  105. PropPage.pszIcon = NULL;
  106. PropPage.pszTitle = NULL;
  107. PropPage.pfnDlgProc = pResourcePickerDlgProc;
  108. PropPage.lParam = (LPARAM)pdmData;
  109. PropPage.pfnCallback = pResourcePickerPropPageCallback;
  110. #ifdef _UNICODE
  111. PropPage.dwFlags |= PSP_USEFUSIONCONTEXT;
  112. PropPage.hActCtx = NULL;
  113. #endif
  114. return CreatePropertySheetPage(&PropPage);
  115. } // GetResourceSelectionPage
  116. //
  117. // CreatePropertySheetPage - callback function
  118. //
  119. UINT CALLBACK pResourcePickerPropPageCallback(
  120. HWND hwnd,
  121. UINT uMsg,
  122. LPPROPSHEETPAGE ppsp
  123. )
  124. /*++
  125. Routine Description:
  126. Callback to handle cleanup of the property sheet
  127. Arguments:
  128. Standard PropSheetPageProc arguments.
  129. Return Value:
  130. Standard PropSheetPageProc return.
  131. --*/
  132. {
  133. switch (uMsg) {
  134. //case PSPCB_ADDREF:
  135. // break;
  136. case PSPCB_CREATE:
  137. break;
  138. case PSPCB_RELEASE:
  139. //
  140. // release the memory we've previously allocated, outside of the actual dialog
  141. //
  142. if (ppsp->lParam != 0) {
  143. LPDMPROP_DATA pdmData = (LPDMPROP_DATA)(ppsp->lParam);
  144. MyFree(pdmData);
  145. }
  146. break;
  147. }
  148. return TRUE;
  149. }
  150. //
  151. // Main dialog proceedure
  152. //
  153. INT_PTR
  154. CALLBACK
  155. pResourcePickerDlgProc(
  156. HWND hDlg,
  157. UINT message,
  158. WPARAM wParam,
  159. LPARAM lParam
  160. )
  161. /*++
  162. Routine Description:
  163. This routine provides the dialog box procedure for the main resource
  164. picker property page. MEMPHIS COMPATIBLE.
  165. Arguments:
  166. Standard dialog box procedure arguments.
  167. Return Value:
  168. Standard dialog box procedure return.
  169. --*/
  170. {
  171. LPDMPROP_DATA lpdmpd = NULL;
  172. if (message == WM_INITDIALOG) {
  173. lpdmpd = (LPDMPROP_DATA)((LPPROPSHEETPAGE)lParam)->lParam;
  174. SetWindowLongPtr(hDlg, DWLP_USER, (LPARAM)lpdmpd);
  175. } else {
  176. lpdmpd = (LPDMPROP_DATA)GetWindowLongPtr(hDlg, DWLP_USER);
  177. }
  178. switch (message) {
  179. //
  180. // initialize
  181. //
  182. case WM_INITDIALOG: {
  183. HICON hIcon = NULL;
  184. int iIcon = 0, iIndex = 0;
  185. ULONG ulSize;
  186. PDEVICE_INFO_SET pDeviceInfoSet;
  187. HMACHINE hMachine;
  188. lpdmpd->himlResourceImages = NULL;
  189. lpdmpd->CurrentLC = 0;
  190. lpdmpd->CurrentLCType = 0;
  191. lpdmpd->MatchingLC = 0;
  192. lpdmpd->MatchingLCType = 0;
  193. lpdmpd->SelectedLC = 0;
  194. lpdmpd->SelectedLCType = 0;
  195. lpdmpd->hDlg = hDlg;
  196. lpdmpd->dwFlags = 0;
  197. hMachine = pGetMachine(lpdmpd);
  198. lpdmpd->dwFlags |= DMPROP_FLAG_CHANGESSAVED; // Nothing to save yet
  199. //
  200. // NOTE: On Windows95, since lc info is in memory, they first
  201. // call CM_Setup_DevNode with CM_SETUP_WRITE_LOG_CONFS flag so
  202. // that in-memory lc data is flushed to the registry at this
  203. // point.
  204. //
  205. //
  206. // Init the Resource's image list.
  207. //
  208. lpdmpd->himlResourceImages = ImageList_Create(GetSystemMetrics(SM_CXSMICON),
  209. GetSystemMetrics(SM_CYSMICON),
  210. ILC_MASK, // | ILC_SHARED,
  211. 1,
  212. 1);
  213. //
  214. // add icons to image list
  215. //
  216. for (iIcon = IDI_RESOURCEFIRST;iIcon < IDI_RESOURCELAST;++iIcon) {
  217. //
  218. // resource icon
  219. //
  220. hIcon = LoadIcon(MyDllModuleHandle, MAKEINTRESOURCE(iIcon));
  221. iIndex = ImageList_AddIcon(lpdmpd->himlResourceImages, hIcon);
  222. }
  223. for (iIcon = IDI_RESOURCEOVERLAYFIRST;iIcon <= IDI_RESOURCEOVERLAYLAST;++iIcon) {
  224. //
  225. // overlay icon
  226. //
  227. hIcon = LoadIcon(MyDllModuleHandle, MAKEINTRESOURCE(iIcon));
  228. iIndex = ImageList_AddIcon(lpdmpd->himlResourceImages, hIcon);
  229. //
  230. // Tag this icon as an overlay icon (the first index is an
  231. // index into the image list (specifies the icon), the
  232. // second index is just an index to assign to each mask
  233. // (starting with 1).
  234. //
  235. ImageList_SetOverlayImage(lpdmpd->himlResourceImages,
  236. iIndex,
  237. iIcon-IDI_RESOURCEOVERLAYFIRST+1);
  238. }
  239. if(pInitDevResourceDlg(lpdmpd)) {
  240. lpdmpd->dwFlags &= ~DMPROP_FLAG_CHANGESSAVED; // need to save (prob because there was no config)
  241. }
  242. if (!(lpdmpd->dwFlags & DMPROP_FLAG_NO_RESOURCES)) {
  243. pShowConflicts(lpdmpd);
  244. }
  245. if (GuiSetupInProgress) {
  246. //
  247. // occasionally legacy devices cause resource-picker popup during setup
  248. // we do this here instead of create prop sheet, since I don't trust
  249. // people to cleanup on fail. At least here is less risky
  250. // clean this up in WM_DESTROY
  251. //
  252. lpdmpd->hDialogEvent = CreateEvent(NULL,TRUE,FALSE,SETUP_HAS_OPEN_DIALOG_EVENT);
  253. if (lpdmpd->hDialogEvent) {
  254. SetEvent(lpdmpd->hDialogEvent);
  255. }
  256. } else {
  257. lpdmpd->hDialogEvent = NULL;
  258. }
  259. break;
  260. }
  261. //
  262. // cleanup
  263. //
  264. case WM_DESTROY: {
  265. HICON hIcon;
  266. LOG_CONF LogConf;
  267. LONG nItems, n;
  268. HWND hList = GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST);
  269. int Count, i;
  270. //
  271. // Clean up the ICON resource usage
  272. //
  273. if ((hIcon = (HICON)LOWORD(SendDlgItemMessage(hDlg,
  274. IDC_DEVRES_ICON, STM_GETICON, 0, 0L)))) {
  275. DestroyIcon(hIcon);
  276. }
  277. //
  278. // free the LC handles that were saved in the combobox data
  279. //
  280. nItems = (LONG)SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST,
  281. CB_GETCOUNT, 0, 0L);
  282. for (n = 0; n < nItems ; n++) {
  283. LogConf = (LOG_CONF)SendDlgItemMessage(hDlg,
  284. IDC_DEVRES_LOGCONFIGLIST,
  285. CB_GETITEMDATA, n, 0L);
  286. CM_Free_Log_Conf_Handle(LogConf);
  287. }
  288. if (lpdmpd->CurrentLC != 0) {
  289. CM_Free_Log_Conf_Handle(lpdmpd->CurrentLC);
  290. }
  291. ListView_DeleteAllItems(hList); // this will destroy all data
  292. if (lpdmpd->himlResourceImages) {
  293. ImageList_Destroy(lpdmpd->himlResourceImages);
  294. }
  295. if (lpdmpd->hDialogEvent) {
  296. //
  297. // we were holding up setup, now let setup proceed
  298. //
  299. ResetEvent(lpdmpd->hDialogEvent);
  300. CloseHandle(lpdmpd->hDialogEvent);
  301. lpdmpd->hDialogEvent = NULL;
  302. }
  303. // MyFree(lpdmpd); - do this in pResourcePickerPropPageCallback instead
  304. break;
  305. }
  306. case WM_COMMAND:
  307. //
  308. // old-style controls
  309. //
  310. switch(LOWORD(wParam)) {
  311. case IDC_DEVRES_USESYSSETTINGS: {
  312. //
  313. // consider resource settings to have changed
  314. //
  315. lpdmpd->dwFlags &= ~DMPROP_FLAG_CHANGESSAVED;
  316. PropSheet_Changed(GetParent(hDlg), hDlg);
  317. if (IsDlgButtonChecked(hDlg, (int)wParam)) {
  318. //
  319. // Revert back to allocated display, if any
  320. //
  321. lpdmpd->dwFlags |= DMPROP_FLAG_USESYSSETTINGS;
  322. pSelectLogConf(lpdmpd,(LOG_CONF)0,ALLOC_LOG_CONF,TRUE);
  323. } else {
  324. //
  325. // Allow editing
  326. //
  327. lpdmpd->dwFlags &= ~DMPROP_FLAG_USESYSSETTINGS;
  328. }
  329. pShowUpdateEdit(lpdmpd); // update controls
  330. break;
  331. }
  332. case IDC_DEVRES_LOGCONFIGLIST: {
  333. //
  334. // drop-down list action
  335. //
  336. switch (HIWORD(wParam)) {
  337. case CBN_SELENDOK: {
  338. ULONG ulIndex = 0;
  339. int iItem;
  340. LOG_CONF SelLC;
  341. HWND hwndLC = GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST);
  342. //
  343. // If there is not a Log Config selected, then bail
  344. //
  345. iItem = (int)SendMessage(hwndLC, CB_GETCURSEL, 0, 0);
  346. if(iItem != CB_ERR) {
  347. SelLC = (LOG_CONF)SendMessage(hwndLC,CB_GETITEMDATA, (WPARAM)iItem,(LPARAM)0);
  348. } else {
  349. SelLC = (LOG_CONF)0;
  350. }
  351. if(SelLC != lpdmpd->SelectedLC) {
  352. pSelectLogConf(lpdmpd,SelLC,lpdmpd->ConfigListLCType,FALSE);
  353. }
  354. //
  355. // I prob don't need this here, but I'm playing safe!
  356. //
  357. lpdmpd->dwFlags &= ~DMPROP_FLAG_CHANGESSAVED;
  358. break;
  359. }
  360. }
  361. break;
  362. }
  363. case IDC_DEVRES_CHANGE: {
  364. //
  365. // change selected setting
  366. //
  367. pChangeCurrentResSetting(lpdmpd);
  368. break;
  369. }
  370. case IDC_DEVRES_MAKEFORCED: {
  371. //
  372. // possibly allow editing (after we've shown message)
  373. // when we get here, always show a configuration
  374. //
  375. if(lpdmpd->dwFlags & DMPROP_FLAG_FORCEDONLY) {
  376. lpdmpd->dwFlags &= ~DMPROP_FLAG_CHANGESSAVED; // need to save
  377. }
  378. pShowViewAllEdit(lpdmpd);
  379. //
  380. // select in the first available config to edit
  381. //
  382. pSelectLogConf(lpdmpd,(LOG_CONF)0,ALLOC_LOG_CONF,TRUE);
  383. //
  384. // ensure we have reasonable focus for accessability
  385. //
  386. PostMessage(hDlg,WM_USER_FOCUS,IDC_DEVRES_SETTINGSLIST,0);
  387. break;
  388. }
  389. default:
  390. break;
  391. }
  392. break;
  393. case WM_USER_FOCUS:
  394. //
  395. // change focus to DlgItem wParam
  396. //
  397. SetFocus(GetDlgItem(hDlg,(int)wParam));
  398. return TRUE;
  399. case WM_NOTIFY: {
  400. //
  401. // new controls & property codes
  402. //
  403. NMHDR * pHdr = (NMHDR*)lParam;
  404. switch (pHdr->code) {
  405. case PSN_SETACTIVE: {
  406. HICON hIcon = NULL, hOldIcon = NULL;
  407. TCHAR szString[MAX_PATH];
  408. ULONG ulSize = 0;
  409. HMACHINE hMachine;
  410. hMachine = pGetMachine(lpdmpd);
  411. //
  412. // Set the ICON and device description
  413. //
  414. if (SetupDiLoadClassIcon(&lpdmpd->lpdi->ClassGuid, &hIcon, NULL)) {
  415. if ((hOldIcon = (HICON)LOWORD(SendDlgItemMessage(hDlg, IDC_DEVRES_ICON,
  416. STM_SETICON,
  417. (WPARAM)hIcon, 0L)))) {
  418. DestroyIcon(hOldIcon);
  419. }
  420. }
  421. //
  422. // First try to get the device's friendly name, then fall back to its description,
  423. // and finally, use the "Unknown Device" description.
  424. //
  425. ulSize = MAX_PATH * sizeof(TCHAR);
  426. if (CM_Get_DevInst_Registry_Property_Ex(lpdmpd->lpdi->DevInst,
  427. CM_DRP_FRIENDLYNAME,
  428. NULL, (LPBYTE)szString,
  429. &ulSize, 0,hMachine) != CR_SUCCESS) {
  430. ulSize = MAX_PATH * sizeof(TCHAR);
  431. if (CM_Get_DevInst_Registry_Property_Ex(lpdmpd->lpdi->DevInst,
  432. CM_DRP_DEVICEDESC,
  433. NULL, (LPBYTE)szString,
  434. &ulSize, 0,hMachine) != CR_SUCCESS) {
  435. LoadString(MyDllModuleHandle, IDS_DEVNAME_UNK, szString, MAX_PATH);
  436. }
  437. }
  438. SetDlgItemText(hDlg, IDC_DEVRES_DEVDESC, szString);
  439. break;
  440. }
  441. case PSN_APPLY: {
  442. BOOL bRet = FALSE;
  443. //
  444. // If there were Changes and they haven't been saved,
  445. // then save them.
  446. // consider some special cases as "haven't been saved"
  447. //
  448. if((lpdmpd->CurrentLC == 0) && (lpdmpd->dwFlags&DMPROP_FLAG_FIXEDCONFIG)) {
  449. lpdmpd->dwFlags &= ~DMPROP_FLAG_CHANGESSAVED;
  450. }
  451. switch(pOkToSave(lpdmpd)) {
  452. case IDNO:
  453. //
  454. // proceed without saving
  455. //
  456. bRet = TRUE;
  457. break;
  458. case IDCANCEL:
  459. //
  460. // don't proceed
  461. //
  462. bRet = FALSE;
  463. break;
  464. case IDYES:
  465. //
  466. // proceed and save
  467. //
  468. bRet = pSaveDevResSettings(lpdmpd);
  469. #if 0
  470. if (bRet) {
  471. if ((lpdmpd->lpdi)->Flags & DI_NEEDREBOOT) {
  472. PropSheet_RebootSystem(GetParent(hDlg));
  473. } else if ((lpdmpd->lpdi)->Flags & DI_NEEDRESTART) {
  474. PropSheet_RestartWindows(GetParent(hDlg));
  475. }
  476. #endif
  477. if (bRet) {
  478. //
  479. // This page doesn't support roll-back, if we saved
  480. // something then we're committed, disable the cancel
  481. // botton.
  482. //
  483. PropSheet_CancelToClose(GetParent(hDlg));
  484. }
  485. break;
  486. default:
  487. MYASSERT(FALSE /* pOkToSave returned invalid value */);
  488. bRet = FALSE;
  489. break;
  490. }
  491. SetWindowLongPtr(hDlg, DWLP_MSGRESULT, bRet ? PSNRET_NOERROR : PSNRET_INVALID_NOCHANGEPAGE);
  492. return TRUE;
  493. }
  494. case LVN_DELETEALLITEMS:
  495. if (pHdr->idFrom == IDC_DEVRES_SETTINGSLIST) {
  496. return FALSE; // we want LVN_DELETEITEM messages
  497. }
  498. break;
  499. case LVN_DELETEITEM: {
  500. LPNMLISTVIEW pListView = (LPNMLISTVIEW)pHdr;
  501. if (pHdr->idFrom == IDC_DEVRES_SETTINGSLIST) {
  502. PITEMDATA pItemData = (PITEMDATA)(LPVOID)(pListView->lParam);
  503. //
  504. // when an item is deleted, destroy associated data
  505. //
  506. if (pItemData->MatchingResDes) {
  507. CM_Free_Res_Des_Handle(pItemData->MatchingResDes);
  508. }
  509. MyFree(pItemData);
  510. }
  511. break;
  512. }
  513. EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_CHANGE), FALSE);
  514. break;
  515. case LVN_ITEMCHANGED:
  516. //
  517. // If the item change is comming from the resource
  518. // list, and there is a logconfig to be edited:
  519. //
  520. if (pHdr->idFrom == IDC_DEVRES_SETTINGSLIST) {
  521. //
  522. // see if we should enable resource change
  523. //
  524. pCheckEnableResourceChange(lpdmpd);
  525. }
  526. break;
  527. case NM_DBLCLK:
  528. //
  529. // If the double click is from the SETTINGS list
  530. // AND the DEVRES_CHANGE button is enabled, then
  531. // allow the change.
  532. //
  533. if (pHdr->idFrom == IDC_DEVRES_SETTINGSLIST) {
  534. //
  535. // this routine should check that we can change settings
  536. //
  537. pChangeCurrentResSetting(lpdmpd);
  538. }
  539. break;
  540. }
  541. break;
  542. }
  543. case WM_SYSCOLORCHANGE: {
  544. HWND hChildWnd = GetWindow(hDlg, GW_CHILD);
  545. while (hChildWnd != NULL) {
  546. SendMessage(hChildWnd, WM_SYSCOLORCHANGE, wParam, lParam);
  547. hChildWnd = GetWindow(hChildWnd, GW_HWNDNEXT);
  548. }
  549. break;
  550. }
  551. case WM_HELP: // F1
  552. WinHelp(((LPHELPINFO)lParam)->hItemHandle, DEVRES_HELP, HELP_WM_HELP, (ULONG_PTR)DevResHelpIDs);
  553. break;
  554. case WM_CONTEXTMENU: // right mouse click
  555. WinHelp((HWND)wParam, DEVRES_HELP, HELP_CONTEXTMENU, (ULONG_PTR)DevResHelpIDs);
  556. break;
  557. }
  558. return FALSE;
  559. }
  560. //
  561. // Helper functions
  562. //
  563. HMACHINE
  564. pGetMachine(
  565. LPDMPROP_DATA lpdmpd
  566. )
  567. /*++
  568. Routine Description:
  569. Retrieve Machine Handle
  570. Arguments:
  571. lpdmpd - Property Data
  572. Return Value:
  573. handle
  574. --*/
  575. {
  576. HMACHINE hMachine;
  577. PDEVICE_INFO_SET pDeviceInfoSet;
  578. if(!(pDeviceInfoSet = AccessDeviceInfoSet(lpdmpd->hDevInfo))) {
  579. return NULL;
  580. }
  581. hMachine = pDeviceInfoSet->hMachine;
  582. UnlockDeviceInfoSet(pDeviceInfoSet);
  583. return hMachine;
  584. }
  585. BOOL
  586. pInitDevResourceDlg(
  587. LPDMPROP_DATA lpdmpd
  588. )
  589. /*++
  590. Routine Description:
  591. This routine intializes the main resource picker property page.
  592. MEMPHIS COMPATIBLE.
  593. Arguments:
  594. Return Value:
  595. TRUE if "not saved"
  596. --*/
  597. {
  598. HWND hDlg = lpdmpd->hDlg;
  599. CONFIGRET Status = CR_SUCCESS;
  600. BOOL bHasCurrent = FALSE;
  601. BOOL bShowCurrent = FALSE;
  602. BOOL bHasForced = FALSE;
  603. BOOL bNoForcedConfig = FALSE;
  604. BOOL bNeedsForcedConfig = FALSE;
  605. BOOL bHasConfigList = FALSE;
  606. LV_COLUMN LvCol;
  607. HWND hWndList = NULL;
  608. TCHAR szString[MAX_PATH], szTemp[MAX_PATH], szConfigType[MAX_PATH],
  609. szConfig[MAX_PATH];
  610. ULONG ulIndex = 0, ulSize = 0, DevStatus = 0, DevProblem = 0;
  611. DWORD BusType = (DWORD)(-1);
  612. LOG_CONF LogConf;
  613. DWORD dwPriority = 0;
  614. WORD wItem;
  615. ULONG ConfigFlags;
  616. HMACHINE hMachine = NULL;
  617. PDEVICE_INFO_SET pDeviceInfoSet;
  618. int iIndex;
  619. BOOL bHasPrivs = FALSE;
  620. //
  621. // Set initial control states
  622. //
  623. pHideAllControls(lpdmpd);
  624. //
  625. // determine priv token
  626. // security checks are visual only
  627. // real security checks are done in umpnpmgr
  628. //
  629. bHasPrivs = pSetupDoesUserHavePrivilege(SE_LOAD_DRIVER_NAME);
  630. hMachine = pGetMachine(lpdmpd);
  631. //
  632. // retrieves current configuration, if any
  633. //
  634. bHasCurrent = pGetCurrentConfig(lpdmpd);
  635. //
  636. // We sometimes get called to show this page even if the device
  637. // doesn't consume any resources. Check for that case and if so, just
  638. // display an informational message and disable everything else.
  639. //
  640. if (!pDevRequiresResources(lpdmpd->lpdi->DevInst,hMachine)) {
  641. //
  642. // This device has no resources
  643. //
  644. pShowViewNoResources(lpdmpd);
  645. lpdmpd->dwFlags |= DMPROP_FLAG_NO_RESOURCES;
  646. goto Final;
  647. }
  648. //
  649. // Initialize the ListView control
  650. //
  651. hWndList = GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST);
  652. LvCol.mask = LVCF_TEXT;
  653. if (LoadString(MyDllModuleHandle, IDS_RESOURCETYPE, szString, MAX_PATH)) {
  654. LvCol.pszText = (LPTSTR)szString;
  655. ListView_InsertColumn(hWndList, 0, (LV_COLUMN FAR *)&LvCol);
  656. }
  657. if (LoadString(MyDllModuleHandle, IDS_RESOURCESETTING, szString, MAX_PATH)) {
  658. LvCol.pszText = (LPTSTR)szString;
  659. ListView_InsertColumn(hWndList, 1, (LV_COLUMN FAR *)&LvCol);
  660. }
  661. ListView_SetImageList(hWndList,lpdmpd->himlResourceImages, LVSIL_SMALL);
  662. //
  663. // Get DevStatus & DevProblem here, we may use this info further down
  664. //
  665. if (CM_Get_DevNode_Status_Ex(&DevStatus, &DevProblem, lpdmpd->lpdi->DevInst,
  666. 0,hMachine) != CR_SUCCESS) {
  667. //
  668. // we should never get here, show this as a problem
  669. //
  670. lpdmpd->dwFlags |= DMPROP_FLAG_HASPROBLEM;
  671. } else if (DevStatus & DN_HAS_PROBLEM) {
  672. //
  673. // cache problem flag away
  674. //
  675. lpdmpd->dwFlags |= DMPROP_FLAG_HASPROBLEM;
  676. } else if (DevStatus & DN_PRIVATE_PROBLEM) {
  677. //
  678. // driver indicates problem
  679. // for now, do same as above
  680. //
  681. lpdmpd->dwFlags |= DMPROP_FLAG_HASPROBLEM;
  682. }
  683. if (bIsMultiFunctionChild(lpdmpd->lpdi,hMachine)) {
  684. //
  685. // If this is a MultiFunction Child, disable all change controls, put up
  686. // special text, and show the alloc config
  687. //
  688. pShowViewMFReadOnly(lpdmpd,FALSE);
  689. goto Final;
  690. }
  691. //
  692. // begin with read-only view, assuming settings are system
  693. //
  694. lpdmpd->dwFlags |= DMPROP_FLAG_USESYSSETTINGS;
  695. if (CM_Get_First_Log_Conf_Ex(NULL,
  696. lpdmpd->lpdi->DevInst,
  697. FORCED_LOG_CONF,
  698. hMachine) == CR_SUCCESS) {
  699. //
  700. // the user currently has a forced config
  701. //
  702. lpdmpd->dwFlags &= ~DMPROP_FLAG_USESYSSETTINGS;
  703. bHasForced = TRUE;
  704. }
  705. bShowCurrent = pShowViewReadOnly(lpdmpd,bHasPrivs);
  706. if (!bHasPrivs || hMachine) {
  707. //
  708. // if we don't have enough priv's
  709. // or we're displaying resources of a remote machine
  710. // bottle out here
  711. // we'll either be displaying current resources
  712. // or displaying a problem
  713. //
  714. goto Final;
  715. }
  716. if(!bHasForced) {
  717. //
  718. // Check bus we're using
  719. // to see if it's one of the read-only displays
  720. //
  721. ulSize = sizeof(BusType);
  722. if (CM_Get_DevInst_Registry_Property_Ex(lpdmpd->lpdi->DevInst,
  723. CM_DRP_LEGACYBUSTYPE,
  724. NULL, (LPBYTE)&BusType,
  725. &ulSize, 0,hMachine) != CR_SUCCESS) {
  726. BusType = (DWORD)InterfaceTypeUndefined;
  727. }
  728. if (BusType != (DWORD)InterfaceTypeUndefined) {
  729. int InterfaceItem;
  730. for(InterfaceItem = 0; ResourcePickerReadOnlyInterfaces[InterfaceItem] != InterfaceTypeUndefined; InterfaceItem++) {
  731. if (BusType == (DWORD)ResourcePickerReadOnlyInterfaces[InterfaceItem]) {
  732. //
  733. // Bus is one that we do not allow forced configs
  734. // we can skip all the funky code below
  735. //
  736. // this is a good thing for 64-bit PCI
  737. //
  738. goto Final;
  739. }
  740. }
  741. }
  742. }
  743. //
  744. // Retrieve alternate configurations for this device
  745. //
  746. if (bHasCurrent) {
  747. //
  748. // Current config (if any) is indicated with zero handle
  749. //
  750. LoadString(MyDllModuleHandle, IDS_CURRENTCONFIG, szString, MAX_PATH);
  751. iIndex = (int)SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST,
  752. CB_ADDSTRING, (WPARAM)0, (LPARAM)(LPSTR)szString);
  753. SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST, CB_SETITEMDATA,(WPARAM)iIndex, (LPARAM)0);
  754. SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST, CB_SETCURSEL,(WPARAM)0, (LPARAM)0);
  755. }
  756. //
  757. // now fill in alternate configurations
  758. // override preferred over basic (in that order)
  759. // don't use Filtered, filtered might remove configs that require reboot
  760. // but they are fine to have here
  761. //
  762. if(CM_Get_First_Log_Conf_Ex(&LogConf,lpdmpd->lpdi->DevInst,OVERRIDE_LOG_CONF,hMachine) == CR_SUCCESS) {
  763. lpdmpd->ConfigListLCType = OVERRIDE_LOG_CONF;
  764. LoadString(MyDllModuleHandle, IDS_OVERRIDECONFIG, szConfigType, MAX_PATH);
  765. bHasConfigList = TRUE;
  766. } else if(CM_Get_First_Log_Conf_Ex(&LogConf,lpdmpd->lpdi->DevInst,BASIC_LOG_CONF,hMachine) == CR_SUCCESS) {
  767. lpdmpd->ConfigListLCType = BASIC_LOG_CONF;
  768. LoadString(MyDllModuleHandle, IDS_BASICCONFIG, szConfigType, MAX_PATH);
  769. bHasConfigList = TRUE;
  770. } else {
  771. //
  772. // If there are no alternate configs, we cannot allow a forced config
  773. //
  774. bNoForcedConfig = TRUE; // cannot force
  775. bHasConfigList = FALSE;
  776. lpdmpd->ConfigListLCType = BASIC_LOG_CONF;
  777. lpdmpd->dwFlags |= DMPROP_FLAG_SINGLE_CONFIG;
  778. }
  779. if(bHasConfigList) {
  780. ulIndex = 0;
  781. if (!pConfigHasNoAlternates(lpdmpd,LogConf)) {
  782. //
  783. // first configuration has more than one alternative
  784. //
  785. lpdmpd->dwFlags &= ~DMPROP_FLAG_SINGLE_CONFIG;
  786. } else {
  787. //
  788. // begin with the assumption there is a single fixed 'basic' config
  789. // we will generally be proved wrong
  790. //
  791. lpdmpd->dwFlags |= DMPROP_FLAG_SINGLE_CONFIG;
  792. }
  793. while (Status == CR_SUCCESS) {
  794. //
  795. // Add this config to the Combobox
  796. //
  797. wsprintf(szTemp, TEXT("%s %04u"), szConfigType, ulIndex);
  798. wItem = (WORD)SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST,
  799. CB_ADDSTRING, 0,
  800. (LPARAM)(LPSTR)szTemp);
  801. //
  802. // Save the log config handle as the item data in the combobox
  803. //
  804. SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST, CB_SETITEMDATA,
  805. wItem, (LPARAM)LogConf);
  806. //
  807. // Get the next config
  808. //
  809. Status = CM_Get_Next_Log_Conf_Ex(&LogConf, LogConf, 0,hMachine);
  810. ulIndex++;
  811. }
  812. if (ulIndex > 1) {
  813. //
  814. // there is more than one config
  815. //
  816. lpdmpd->dwFlags &= ~DMPROP_FLAG_SINGLE_CONFIG;
  817. }
  818. if (lpdmpd->dwFlags & DMPROP_FLAG_SINGLE_CONFIG) {
  819. bNoForcedConfig = TRUE;
  820. }
  821. if (bHasCurrent) {
  822. //
  823. // try to find a matching LC now, and if we could find one,
  824. // re-load current display (this applies editable ranges to the resources)
  825. //
  826. if(pFindMatchingAllocConfig(lpdmpd)) {
  827. pLoadCurrentConfig(lpdmpd,TRUE);
  828. }
  829. }
  830. } else {
  831. }
  832. //
  833. // Get ConfigFlags here, we may use this info further down
  834. //
  835. ulSize = sizeof(ConfigFlags);
  836. if (CM_Get_DevInst_Registry_Property_Ex(lpdmpd->lpdi->DevInst,
  837. CM_DRP_CONFIGFLAGS,
  838. NULL, (LPBYTE)&ConfigFlags,
  839. &ulSize, 0,hMachine) != CR_SUCCESS) {
  840. ConfigFlags = 0;
  841. }
  842. if (ConfigFlags & CONFIGFLAG_NEEDS_FORCED_CONFIG) {
  843. //
  844. // registry says that we need a forced config
  845. // registry can only say this to us once
  846. //
  847. bNeedsForcedConfig = TRUE;
  848. ConfigFlags &= ~CONFIGFLAG_NEEDS_FORCED_CONFIG;
  849. CM_Set_DevInst_Registry_Property_Ex(lpdmpd->lpdi->DevInst,
  850. CM_DRP_CONFIGFLAGS,
  851. (LPBYTE)&ConfigFlags,
  852. sizeof(ConfigFlags),
  853. 0,
  854. hMachine);
  855. }
  856. //
  857. // determine if it can be software-config'd or not
  858. // we need to do this prior to any initial display
  859. //
  860. dwPriority = pGetMinLCPriority(lpdmpd->lpdi->DevInst, lpdmpd->ConfigListLCType,hMachine);
  861. if (dwPriority < LCPRI_HARDRECONFIG) {
  862. //
  863. // doesn't need to be manually configured
  864. //
  865. lpdmpd->dwFlags &= ~DMPROP_FLAG_FORCEDONLY;
  866. } else {
  867. //
  868. // this cannot be software config'd
  869. // FORCEDONLY & bNoForcedConfig is a quandry, shouldn't happen
  870. //
  871. lpdmpd->dwFlags |= DMPROP_FLAG_FORCEDONLY;
  872. if(!bHasConfigList) {
  873. MYASSERT(bHasConfigList);
  874. } else {
  875. MYASSERT(!bNoForcedConfig);
  876. bNoForcedConfig = FALSE;
  877. }
  878. }
  879. //
  880. // Try to determine initial display
  881. //
  882. // we've already covered pShowViewNoResources (no actual or potential configs)
  883. // and pShowViewMFReadOnly (it's a multi-function device)
  884. //
  885. // we're currently showing as pShowViewReadOnly
  886. //
  887. // some cases....
  888. // (1) show forced config, don't allow auto-config (config flags say requires forced)
  889. // (2) show forced config, allow auto-config
  890. // (3) don't show any config, but maybe show a forced-config button
  891. // (4) auto-config, don't allow forced config
  892. // (5) show auto-config, allow forced-config
  893. //
  894. if (bNeedsForcedConfig) {
  895. if (!bHasConfigList) {
  896. MYASSERT(bHasConfigList);
  897. bNeedsForcedConfig = FALSE;
  898. } else {
  899. MYASSERT(!bNoForcedConfig);
  900. bNoForcedConfig = FALSE;
  901. if (bHasForced) {
  902. //
  903. // already got one, but we'll go through the motions
  904. // we'll show what we have, allow user to change it
  905. // but we wont needlessly save it
  906. //
  907. bNeedsForcedConfig = FALSE;
  908. }
  909. //
  910. // caller said that device must have forced config, so go immediately there
  911. // = case (1) unless we've otherwise said we cannot have a forced config
  912. //
  913. lpdmpd->dwFlags |= DMPROP_FLAG_FORCEDONLY;
  914. pSelectLogConf(lpdmpd,(LOG_CONF)0,ALLOC_LOG_CONF,TRUE);
  915. pShowViewAllEdit(lpdmpd);
  916. goto Final;
  917. }
  918. }
  919. if ((!bShowCurrent) || (lpdmpd->dwFlags & DMPROP_FLAG_HASPROBLEM)) {
  920. //
  921. // determine between pShowViewNoAlloc and pShowViewNeedForced
  922. //
  923. if (bNoForcedConfig) {
  924. //
  925. // there is a problem - device doesn't currently have a current configuration
  926. // but we don't have the option of letting them set forced config
  927. // so this ends up display only (tough-luck scenario)
  928. // if there are current resources, show them
  929. //
  930. pShowViewReadOnly(lpdmpd,FALSE);
  931. } else {
  932. //
  933. // we show the problem and we give the user
  934. // an option to force config
  935. //
  936. pShowViewNeedForced(lpdmpd);
  937. }
  938. goto Final;
  939. }
  940. if (!bHasConfigList) {
  941. //
  942. // If we have a current config, but no basic configs, we just display what we have
  943. // and don't give option to edit
  944. //
  945. pShowViewReadOnly(lpdmpd,FALSE);
  946. goto Final;
  947. }
  948. if ((lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS) && bNoForcedConfig) {
  949. //
  950. // we can't force a bNoForcedConfig item - display only
  951. //
  952. pShowViewReadOnly(lpdmpd,FALSE);
  953. goto Final;
  954. }
  955. //
  956. // we already have and will be displaying a current config
  957. //
  958. pShowViewAllEdit(lpdmpd);
  959. bNeedsForcedConfig = (BOOL)!bHasCurrent; // rarely, if ever, will this return TRUE
  960. Final:
  961. return bNeedsForcedConfig;
  962. } // InitDevResourceDlg
  963. PITEMDATA
  964. pGetResourceToChange(
  965. IN LPDMPROP_DATA lpdmpd,
  966. OUT int *pCur
  967. )
  968. /*++
  969. Routine Description:
  970. Gets resource to change
  971. NULL if we cannot change resource
  972. Arguments:
  973. lpdmpd = dialog data
  974. pCur = (out) index
  975. Return Value:
  976. PITEMDATA saved for selected resource
  977. --*/
  978. {
  979. HWND hList = GetDlgItem(lpdmpd->hDlg, IDC_DEVRES_SETTINGSLIST);
  980. PITEMDATA pItemData = NULL;
  981. int iCur;
  982. //
  983. // first check the obvious
  984. //
  985. if (lpdmpd->dwFlags & DMPROP_FLAG_VIEWONLYRES) {
  986. //
  987. // no editing allowed
  988. //
  989. return NULL;
  990. }
  991. if (lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS) {
  992. //
  993. // showing system settings
  994. //
  995. return NULL;
  996. }
  997. //
  998. // Check if there is a selected item.
  999. // If yes, then activate the change button
  1000. // if the LC allows editing.
  1001. //
  1002. iCur = (int)ListView_GetNextItem(hList,-1, LVNI_SELECTED);
  1003. if (iCur == LB_ERR) {
  1004. //
  1005. // no selection
  1006. //
  1007. return NULL;
  1008. }
  1009. pItemData = (PITEMDATA)pGetListViewItemData(hList, iCur, 0);
  1010. if (pItemData == NULL) {
  1011. //
  1012. // shouldn't happen
  1013. //
  1014. MYASSERT(pItemData);
  1015. return NULL;
  1016. }
  1017. if (pItemData->bFixed) {
  1018. //
  1019. // this is an un-editable setting
  1020. //
  1021. return NULL;
  1022. }
  1023. if (pItemData->MatchingResDes == (RES_DES)0) {
  1024. //
  1025. // should be caught by bFixed
  1026. //
  1027. MYASSERT(pItemData->MatchingResDes != (RES_DES)0);
  1028. return NULL;
  1029. }
  1030. //
  1031. // we're happy
  1032. //
  1033. if (pCur) {
  1034. *pCur = iCur;
  1035. }
  1036. return pItemData;
  1037. }
  1038. VOID
  1039. pCheckEnableResourceChange(
  1040. LPDMPROP_DATA lpdmpd
  1041. )
  1042. /*++
  1043. Routine Description:
  1044. enables/disable change button
  1045. Arguments:
  1046. Return Value:
  1047. none
  1048. --*/
  1049. {
  1050. #if 0 // this seems to confuse people
  1051. EnableWindow(GetDlgItem(lpdmpd->hDlg, IDC_DEVRES_CHANGE),
  1052. pGetResourceToChange(lpdmpd,NULL)!=NULL);
  1053. #endif // 0
  1054. //
  1055. // show this button enabled if we are in EDIT mode
  1056. //
  1057. EnableWindow(GetDlgItem(lpdmpd->hDlg, IDC_DEVRES_CHANGE),
  1058. (lpdmpd->dwFlags & DMPROP_FLAG_VIEWONLYRES)==0 &&
  1059. (lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS)==0);
  1060. }
  1061. BOOL
  1062. pDevHasConfig(
  1063. DEVINST DevInst,
  1064. ULONG ulConfigType,
  1065. HMACHINE hMachine
  1066. )
  1067. /*++
  1068. Routine Description:
  1069. This routine determines whether a log config of the specified type
  1070. exists for this device instance.
  1071. MEMPHIS COMPATIBLE.
  1072. Arguments:
  1073. DevInst Device instance to query log configs for.
  1074. ulConfigType Specifies the type of log conf to check for the presense of.
  1075. Return Value:
  1076. TRUE if the device has a config of that type and FALSE if it does not.
  1077. --*/
  1078. {
  1079. BOOL bRet = (CM_Get_First_Log_Conf_Ex(NULL, DevInst, ulConfigType,hMachine) == CR_SUCCESS);
  1080. return bRet;
  1081. } // DevHasConfig
  1082. DWORD
  1083. pGetMinLCPriority(
  1084. IN DEVINST DevInst,
  1085. IN ULONG ulConfigType,
  1086. IN HMACHINE hMachine
  1087. )
  1088. /*++
  1089. Routine Description:
  1090. This routine returns the minimum priority value of all log confs of the
  1091. specified type for this device. MEMPHIS COMPATIBLE.
  1092. Arguments:
  1093. DevInst Device instance to query log configs for.
  1094. ulConfigType Specifies the type of log conf.
  1095. Return Value:
  1096. Returns the minimum priority value found or LCPRI_LASTSOFTCONFIG if no priorities
  1097. are found.
  1098. --*/
  1099. {
  1100. CONFIGRET Status = CR_SUCCESS;
  1101. ULONG priority, minPriority = MAX_LCPRI;
  1102. LOG_CONF LogConf, tempLC;
  1103. BOOL FoundOneLogConfWithPriority = FALSE;
  1104. //
  1105. // Walk through each log conf of this type for this device and
  1106. // save the smallest value.
  1107. //
  1108. Status = CM_Get_First_Log_Conf_Ex(&LogConf, DevInst, ulConfigType,hMachine);
  1109. while (Status == CR_SUCCESS) {
  1110. if (CM_Get_Log_Conf_Priority_Ex(LogConf, &priority, 0,hMachine) == CR_SUCCESS) {
  1111. FoundOneLogConfWithPriority = TRUE;
  1112. minPriority = min(minPriority, priority);
  1113. }
  1114. tempLC = LogConf;
  1115. Status = CM_Get_Next_Log_Conf_Ex(&LogConf, LogConf, 0,hMachine);
  1116. CM_Free_Log_Conf_Handle(tempLC);
  1117. }
  1118. if(FoundOneLogConfWithPriority) {
  1119. return minPriority;
  1120. } else {
  1121. //
  1122. // None of the LogConfigs had an associated priority. This is common on
  1123. // NT, because the bus drivers don't specify ConfigMgr-style priorities
  1124. // when responding to IRQ_MN_QUERY_RESOURCE_REQUIREMENTS. Since these
  1125. // cases are all PnP bus drivers, however, it is most correct to specify
  1126. // these LogConfigs as soft-settable.
  1127. //
  1128. return LCPRI_LASTSOFTCONFIG;
  1129. }
  1130. } // GetMinLCPriority
  1131. BOOL
  1132. pDevRequiresResources(
  1133. DEVINST DevInst,
  1134. HMACHINE hMachine
  1135. )
  1136. {
  1137. if (CM_Get_First_Log_Conf_Ex(NULL, DevInst, BASIC_LOG_CONF,hMachine) == CR_SUCCESS) {
  1138. return TRUE;
  1139. }
  1140. if (CM_Get_First_Log_Conf_Ex(NULL, DevInst, FILTERED_LOG_CONF,hMachine) == CR_SUCCESS) {
  1141. return TRUE;
  1142. }
  1143. if (CM_Get_First_Log_Conf_Ex(NULL, DevInst, OVERRIDE_LOG_CONF,hMachine) == CR_SUCCESS) {
  1144. return TRUE;
  1145. }
  1146. if (CM_Get_First_Log_Conf_Ex(NULL, DevInst, FORCED_LOG_CONF,hMachine) == CR_SUCCESS) {
  1147. return TRUE;
  1148. }
  1149. if (CM_Get_First_Log_Conf_Ex(NULL, DevInst, BOOT_LOG_CONF,hMachine) == CR_SUCCESS) {
  1150. return TRUE;
  1151. }
  1152. if (CM_Get_First_Log_Conf_Ex(NULL, DevInst, ALLOC_LOG_CONF,hMachine) == CR_SUCCESS) {
  1153. return TRUE;
  1154. }
  1155. return FALSE;
  1156. } // DevRequiresResources
  1157. BOOL
  1158. pGetCurrentConfig(
  1159. IN OUT LPDMPROP_DATA lpdmpd
  1160. )
  1161. /*++
  1162. Routine Description:
  1163. This routine determines the current known configuration
  1164. current configs are either forced, alloc or boot configs.
  1165. Arguments:
  1166. lpdmpd Property data.
  1167. Return Value:
  1168. TRUE if we set the current config
  1169. --*/
  1170. {
  1171. PDEVICE_INFO_SET pDeviceInfoSet;
  1172. HMACHINE hMachine;
  1173. ULONG Status;
  1174. ULONG Problem;
  1175. MYASSERT(lpdmpd!=NULL);
  1176. MYASSERT(lpdmpd->lpdi!=NULL);
  1177. MYASSERT(lpdmpd->CurrentLC==0);
  1178. MYASSERT(lpdmpd->lpdi->DevInst!=0);
  1179. if (lpdmpd==NULL ||
  1180. lpdmpd->lpdi==NULL ||
  1181. lpdmpd->lpdi->DevInst==0) {
  1182. return FALSE;
  1183. }
  1184. lpdmpd->dwFlags &= ~DMPROP_FLAG_DISPLAY_MASK;
  1185. if(!(pDeviceInfoSet = AccessDeviceInfoSet(lpdmpd->hDevInfo))) {
  1186. return FALSE;
  1187. }
  1188. hMachine = pDeviceInfoSet->hMachine;
  1189. UnlockDeviceInfoSet (pDeviceInfoSet);
  1190. if (CM_Get_DevNode_Status_Ex(&Status, &Problem, lpdmpd->lpdi->DevInst,
  1191. 0,hMachine) != CR_SUCCESS) {
  1192. Problem = 0;
  1193. Status = 0;
  1194. } else if((Status & DN_HAS_PROBLEM)==0) {
  1195. //
  1196. // If this device is running, does this devinst have a ALLOC log config?
  1197. //
  1198. if (CM_Get_First_Log_Conf_Ex(&lpdmpd->CurrentLC,
  1199. lpdmpd->lpdi->DevInst,
  1200. ALLOC_LOG_CONF,
  1201. hMachine) == CR_SUCCESS) {
  1202. lpdmpd->dwFlags |= DMPROP_FLAG_DISPLAY_ALLOC;
  1203. lpdmpd->CurrentLCType = ALLOC_LOG_CONF;
  1204. return TRUE;
  1205. }
  1206. }
  1207. //
  1208. // If no config so far, does it have a FORCED log config?
  1209. //
  1210. if (CM_Get_First_Log_Conf_Ex(&lpdmpd->CurrentLC,
  1211. lpdmpd->lpdi->DevInst,
  1212. FORCED_LOG_CONF,
  1213. hMachine) == CR_SUCCESS) {
  1214. lpdmpd->dwFlags |= DMPROP_FLAG_DISPLAY_FORCED;
  1215. lpdmpd->CurrentLCType = FORCED_LOG_CONF;
  1216. return TRUE;
  1217. }
  1218. //
  1219. // if there's a hardware-disabled problem, boot-config isn't valid
  1220. //
  1221. if(((Status & DN_HAS_PROBLEM)==0) || (Problem != CM_PROB_HARDWARE_DISABLED)) {
  1222. //
  1223. // Does it have a BOOT log config?
  1224. //
  1225. if (CM_Get_First_Log_Conf_Ex(&lpdmpd->CurrentLC,
  1226. lpdmpd->lpdi->DevInst,
  1227. BOOT_LOG_CONF,
  1228. hMachine) == CR_SUCCESS) {
  1229. lpdmpd->dwFlags |= DMPROP_FLAG_DISPLAY_BOOT;
  1230. lpdmpd->CurrentLCType = BOOT_LOG_CONF;
  1231. return TRUE;
  1232. }
  1233. }
  1234. return FALSE;
  1235. }
  1236. void
  1237. pGetHdrValues(
  1238. IN LPBYTE pData,
  1239. IN RESOURCEID ResType,
  1240. OUT PULONG64 pulValue,
  1241. OUT PULONG64 pulLen,
  1242. OUT PULONG64 pulEnd,
  1243. OUT PULONG pulFlags
  1244. )
  1245. {
  1246. switch (ResType) {
  1247. case ResType_Mem: {
  1248. PMEM_RESOURCE pMemData = (PMEM_RESOURCE)pData;
  1249. *pulValue = pMemData->MEM_Header.MD_Alloc_Base;
  1250. *pulLen = (pMemData->MEM_Header.MD_Alloc_End -
  1251. pMemData->MEM_Header.MD_Alloc_Base + 1);
  1252. *pulEnd = pMemData->MEM_Header.MD_Alloc_End;
  1253. *pulFlags = pMemData->MEM_Header.MD_Flags;
  1254. break;
  1255. }
  1256. case ResType_IO: {
  1257. PIO_RESOURCE pIoData = (PIO_RESOURCE)pData;
  1258. *pulValue = pIoData->IO_Header.IOD_Alloc_Base;
  1259. *pulLen = (pIoData->IO_Header.IOD_Alloc_End -
  1260. pIoData->IO_Header.IOD_Alloc_Base + 1);
  1261. *pulEnd = pIoData->IO_Header.IOD_Alloc_End;
  1262. *pulFlags = pIoData->IO_Header.IOD_DesFlags;
  1263. break;
  1264. }
  1265. case ResType_DMA: {
  1266. PDMA_RESOURCE pDmaData = (PDMA_RESOURCE)pData;
  1267. *pulValue = pDmaData->DMA_Header.DD_Alloc_Chan;
  1268. *pulLen = 1;
  1269. *pulEnd = *pulValue;
  1270. *pulFlags = pDmaData->DMA_Header.DD_Flags;
  1271. break;
  1272. }
  1273. case ResType_IRQ: {
  1274. DEVRES_PIRQ_RESOURCE pIrqData = (DEVRES_PIRQ_RESOURCE)pData;
  1275. *pulValue = pIrqData->IRQ_Header.IRQD_Alloc_Num;
  1276. *pulLen = 1;
  1277. *pulEnd = *pulValue;
  1278. *pulFlags = pIrqData->IRQ_Header.IRQD_Flags;
  1279. break;
  1280. }
  1281. }
  1282. if(*pulEnd < *pulValue) {
  1283. //
  1284. // filter out bad/zero-length range
  1285. //
  1286. *pulLen = 0;
  1287. }
  1288. return;
  1289. } // GetHdrValues
  1290. void
  1291. pGetRangeValues(
  1292. IN LPBYTE pData,
  1293. IN RESOURCEID ResType,
  1294. IN ULONG ulIndex,
  1295. OUT PULONG64 pulValue, OPTIONAL
  1296. OUT PULONG64 pulLen, OPTIONAL
  1297. OUT PULONG64 pulEnd, OPTIONAL
  1298. OUT PULONG64 pulAlign, OPTIONAL
  1299. OUT PULONG pulFlags OPTIONAL
  1300. )
  1301. {
  1302. //
  1303. // keep local copies
  1304. // we transfer to parameters at end
  1305. //
  1306. ULONG64 ulValue;
  1307. ULONG64 ulLen;
  1308. ULONG64 ulEnd;
  1309. ULONG64 ulAlign;
  1310. ULONG ulFlags;
  1311. switch (ResType) {
  1312. case ResType_Mem: {
  1313. PMEM_RESOURCE pMemData = (PMEM_RESOURCE)pData;
  1314. ulValue = pMemData->MEM_Data[ulIndex].MR_Min;
  1315. ulLen = pMemData->MEM_Data[ulIndex].MR_nBytes;
  1316. ulEnd = pMemData->MEM_Data[ulIndex].MR_Max;
  1317. ulFlags = pMemData->MEM_Data[ulIndex].MR_Flags;
  1318. ulAlign = pMemData->MEM_Data[ulIndex].MR_Align;
  1319. break;
  1320. }
  1321. case ResType_IO: {
  1322. PIO_RESOURCE pIoData = (PIO_RESOURCE)pData;
  1323. ulValue = pIoData->IO_Data[ulIndex].IOR_Min;
  1324. ulLen = pIoData->IO_Data[ulIndex].IOR_nPorts;
  1325. ulEnd = pIoData->IO_Data[ulIndex].IOR_Max;
  1326. ulFlags = pIoData->IO_Data[ulIndex].IOR_RangeFlags;
  1327. ulAlign = pIoData->IO_Data[ulIndex].IOR_Align;
  1328. break;
  1329. }
  1330. case ResType_DMA: {
  1331. PDMA_RESOURCE pDmaData = (PDMA_RESOURCE)pData;
  1332. ulValue = pDmaData->DMA_Data[ulIndex].DR_Min;
  1333. ulLen = 1;
  1334. ulEnd = ulValue;
  1335. ulFlags = pDmaData->DMA_Data[ulIndex].DR_Flags;
  1336. ulAlign = 1;
  1337. break;
  1338. }
  1339. case ResType_IRQ: {
  1340. DEVRES_PIRQ_RESOURCE pIrqData = (DEVRES_PIRQ_RESOURCE)pData;
  1341. ulValue = pIrqData->IRQ_Data[ulIndex].IRQR_Min;
  1342. ulLen = 1;
  1343. ulEnd = ulValue;
  1344. ulFlags = pIrqData->IRQ_Data[ulIndex].IRQR_Flags;
  1345. ulAlign = 1;
  1346. break;
  1347. }
  1348. }
  1349. if(ulEnd < ulValue) {
  1350. //
  1351. // filter out bad/zero-length range
  1352. //
  1353. ulLen = 0;
  1354. }
  1355. pAlignValues(&ulValue, ulValue, ulLen, ulEnd, ulAlign,1);
  1356. //
  1357. // copy return parameters
  1358. //
  1359. if (pulValue) {
  1360. *pulValue = ulValue;
  1361. }
  1362. if (pulLen) {
  1363. *pulLen = ulLen;
  1364. }
  1365. if (pulEnd) {
  1366. *pulEnd = ulEnd;
  1367. }
  1368. if (pulAlign) {
  1369. *pulAlign = ulAlign;
  1370. }
  1371. if (pulFlags) {
  1372. *pulFlags = ulFlags;
  1373. }
  1374. return;
  1375. }
  1376. BOOL
  1377. pAlignValues(
  1378. IN OUT PULONG64 pulValue,
  1379. IN ULONG64 ulStart,
  1380. IN ULONG64 ulLen,
  1381. IN ULONG64 ulEnd,
  1382. IN ULONG64 ulAlignment,
  1383. IN int Increment
  1384. )
  1385. {
  1386. ULONG64 NtAlign = ~ulAlignment + 1; // convert from mask to modulus
  1387. ULONG64 Value;
  1388. ULONG64 Upp;
  1389. ULONG64 Remainder;
  1390. Value = *pulValue;
  1391. if (NtAlign == 0) {
  1392. return FALSE; // bogus alignment value
  1393. }
  1394. if (NtAlign != 1 && Increment != 0) {
  1395. //
  1396. // see if we are aligned
  1397. //
  1398. Remainder = Value % NtAlign;
  1399. if (Remainder != 0) {
  1400. //
  1401. // need to re-align
  1402. //
  1403. if (Increment>0) {
  1404. //
  1405. // Return the first valid aligned value greater than this value
  1406. //
  1407. Value += NtAlign - Remainder;
  1408. if (Value <= *pulValue) {
  1409. //
  1410. // overflow detected
  1411. //
  1412. return FALSE;
  1413. }
  1414. } else {
  1415. //
  1416. // Return the first valid aligned value less than this value
  1417. //
  1418. Value -= Remainder;
  1419. //
  1420. // we never overflow going down, since zero is a common denominator
  1421. // of alignment
  1422. //
  1423. }
  1424. }
  1425. }
  1426. //
  1427. // now check boundaries
  1428. //
  1429. if (Value < ulStart) {
  1430. return FALSE;
  1431. }
  1432. Upp = Value+ulLen-1;
  1433. if (Upp < Value) {
  1434. //
  1435. // catch overflow error
  1436. //
  1437. return FALSE;
  1438. }
  1439. if (Upp > ulEnd) {
  1440. return FALSE;
  1441. }
  1442. //
  1443. // set newly aligned value
  1444. //
  1445. *pulValue = Value;
  1446. return TRUE;
  1447. }
  1448. void
  1449. pFormatResString(
  1450. LPDMPROP_DATA lpdmpd,
  1451. LPTSTR lpszString,
  1452. ULONG64 ulVal,
  1453. ULONG64 ulLen,
  1454. RESOURCEID ResType
  1455. )
  1456. {
  1457. if (ulLen == 0) {
  1458. wsprintf(lpszString, szNoValue);
  1459. } else if ((ResType == ResType_DMA) || (ResType == ResType_IRQ)) {
  1460. wsprintf(lpszString, szOneDecNoConflict, (UINT)ulVal);
  1461. } else if (ResType == ResType_IO) {
  1462. wsprintf(lpszString, szTwoWordHexNoConflict, (ULONG)ulVal,
  1463. (ULONG)(ulVal + ulLen - 1));
  1464. } else if ((ulVal+(ulLen-1)) >= 0x100000000) {
  1465. //
  1466. // NTRAID # 712013 - this needs to be improved
  1467. //
  1468. wsprintf(lpszString, szTwo64bitHexNoConflict, (ULONG64)ulVal,
  1469. (ULONG64)(ulVal + ulLen - 1));
  1470. } else {
  1471. wsprintf(lpszString, szTwoDWordHexNoConflict, (ULONG)ulVal,
  1472. (ULONG)(ulVal + ulLen - 1));
  1473. }
  1474. }
  1475. BOOL
  1476. pUnFormatResString(
  1477. LPTSTR lpszString,
  1478. PULONG64 pulVal,
  1479. PULONG64 pulEnd,
  1480. RESOURCEID ridResType
  1481. )
  1482. {
  1483. BOOL bRet = FALSE;
  1484. LPTSTR lpszTemp = NULL;
  1485. LPTSTR lpszTemp2 = NULL;
  1486. LPTSTR lpszCopy;
  1487. // ISSUE-2000/02/03 Fix pUnFormatResString bugs
  1488. //
  1489. // - extend this to handling DWORDLONG values
  1490. // - use correct Prev/Next functions for parsing string
  1491. //
  1492. //
  1493. // Allocate space for, and make a copy of the input string
  1494. //
  1495. lpszCopy = MyMalloc((lstrlen(lpszString)+1) * sizeof(TCHAR));
  1496. if (lpszCopy == NULL) {
  1497. return FALSE;
  1498. }
  1499. lstrcpy(lpszCopy, lpszString);
  1500. //
  1501. // Locate the dash if there is one, and convert the white space prev to
  1502. // the dash to a NULL. (ie 0200 - 0400 while be 0200)
  1503. //
  1504. lpszTemp = lpszCopy;
  1505. while ((*lpszTemp != '-') && (*lpszTemp != '\0')) {
  1506. lpszTemp++; // AnsiNext?
  1507. }
  1508. if (*lpszTemp != '\0') {
  1509. lpszTemp2 = lpszTemp-1;
  1510. ++lpszTemp;
  1511. }
  1512. //
  1513. // Search back to set the NULL for the Value
  1514. //
  1515. if (lpszTemp2 != NULL) {
  1516. while ((*lpszTemp2 == ' ') || (*lpszTemp2 == '\t'))
  1517. lpszTemp2--; // AnsiPrev?
  1518. *(lpszTemp2+1)= '\0';
  1519. }
  1520. //
  1521. // Convert the first entry
  1522. //
  1523. if (pConvertEditText(lpszCopy, pulVal, ridResType)) {
  1524. //
  1525. // If there is a second entry, convert it, otherwise assume a length
  1526. // of one.
  1527. //
  1528. if (*lpszTemp != '\0') {
  1529. if (pConvertEditText(lpszTemp, pulEnd,ridResType)) {
  1530. bRet = TRUE;
  1531. }
  1532. } else {
  1533. *pulEnd = *pulVal;
  1534. bRet = TRUE;
  1535. }
  1536. }
  1537. MyFree(lpszCopy);
  1538. return bRet;
  1539. }
  1540. BOOL
  1541. pConvertEditText(
  1542. LPTSTR lpszConvert,
  1543. PULONG64 pulVal,
  1544. RESOURCEID ridResType
  1545. )
  1546. {
  1547. LPTSTR lpConvert;
  1548. if ((ridResType == ResType_Mem) || (ridResType == ResType_IO)) {
  1549. *pulVal = _tcstoul(lpszConvert, &lpConvert, (WORD)16);
  1550. } else {
  1551. *pulVal = _tcstoul(lpszConvert, &lpConvert, (WORD)10);
  1552. }
  1553. if (lpConvert == lpszConvert+lstrlen(lpszConvert)) {
  1554. return TRUE;
  1555. } else {
  1556. return FALSE;
  1557. }
  1558. } // ConvertEditText
  1559. void
  1560. pWarnResSettingNotEditable(
  1561. HWND hDlg,
  1562. WORD idWarning
  1563. )
  1564. {
  1565. TCHAR szTitle[MAX_PATH];
  1566. TCHAR szMessage[MAX_PATH * 2];
  1567. //
  1568. // Give some warning Messages. If there is no logconfig,
  1569. // then we cannot edit any settings, if there is, then
  1570. // just the setting they are choosing is not editable.
  1571. //
  1572. LoadString(MyDllModuleHandle, IDS_DEVRES_NOMODIFYTITLE, szTitle, MAX_PATH);
  1573. LoadString(MyDllModuleHandle, idWarning, szMessage, MAX_PATH * 2);
  1574. MessageBox(hDlg, szMessage, szTitle, MB_OK | MB_TASKMODAL | MB_ICONEXCLAMATION);
  1575. } // WarnResSettingsNotEditable
  1576. int
  1577. pWarnNoSave(
  1578. HWND hDlg,
  1579. WORD idWarning
  1580. )
  1581. /*++
  1582. Routine Description:
  1583. Warn that the settings will not be saved
  1584. Arguments:
  1585. Return Value:
  1586. IDCANCEL = don't proceed
  1587. IDOK/IDYES/IDNO = proceed without saving
  1588. --*/
  1589. {
  1590. TCHAR szTitle[MAX_PATH];
  1591. TCHAR szMessage[MAX_PATH * 2];
  1592. int res;
  1593. //
  1594. // Give a warning message of why we can't save settings
  1595. //
  1596. LoadString(MyDllModuleHandle, IDS_MAKE_FORCED_TITLE, szTitle, MAX_PATH);
  1597. LoadString(MyDllModuleHandle, idWarning, szMessage, MAX_PATH * 2);
  1598. //res = MessageBox(hDlg, szMessage, szTitle, MB_OKCANCEL | MB_TASKMODAL | MB_ICONEXCLAMATION);
  1599. //return res;
  1600. res = MessageBox(hDlg, szMessage, szTitle, MB_OK | MB_TASKMODAL | MB_ICONEXCLAMATION);
  1601. return IDCANCEL;
  1602. }
  1603. LPVOID
  1604. pGetListViewItemData(
  1605. HWND hList,
  1606. int iItem,
  1607. int iSubItem
  1608. )
  1609. {
  1610. LV_ITEM lviItem;
  1611. lviItem.mask = LVIF_PARAM;
  1612. lviItem.iItem = iItem;
  1613. lviItem.iSubItem = iSubItem;
  1614. if (ListView_GetItem(hList, &lviItem)) {
  1615. return (LPVOID)lviItem.lParam;
  1616. } else {
  1617. return NULL;
  1618. }
  1619. } // GetListViewItemData
  1620. BOOL
  1621. pSaveDevResSettings(
  1622. LPDMPROP_DATA lpdmpd
  1623. )
  1624. /*++
  1625. Routine Description:
  1626. This routine saves the resources based on the users selections.
  1627. MEMPHIS COMPATIBLE.
  1628. Arguments:
  1629. lpdmpd Property data.
  1630. Return Value:
  1631. Returns TRUE if the function succeeded and FALSE if it failed.
  1632. --*/
  1633. {
  1634. HWND hDlg = lpdmpd->hDlg;
  1635. HWND hList = GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST);
  1636. CONFIGRET Status = CR_SUCCESS;
  1637. LOG_CONF ForcedLogConf;
  1638. RES_DES ResDes, ResDesTemp, ResDes1;
  1639. RESOURCEID ResType;
  1640. ULONG ulSize = 0, ulCount = 0, i = 0, iCur = 0;
  1641. LPBYTE pData = NULL;
  1642. PITEMDATA pItemData = NULL;
  1643. BOOL bRet = TRUE;
  1644. SP_PROPCHANGE_PARAMS PropChangeParams;
  1645. HMACHINE hMachine = pGetMachine(lpdmpd);
  1646. if ((lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS)!=0) {
  1647. //-------------------------------------------------------------------
  1648. // If the user checked the "Use Automatic Settings" checkbox, then
  1649. // delete any Boot/Forced configs, otherwise write the current settings
  1650. // as a forced config.
  1651. //-------------------------------------------------------------------
  1652. if (CM_Get_First_Log_Conf_Ex(&ForcedLogConf, lpdmpd->lpdi->DevInst,
  1653. FORCED_LOG_CONF,hMachine) == CR_SUCCESS) {
  1654. CM_Free_Log_Conf_Ex(ForcedLogConf, 0,hMachine);
  1655. CM_Free_Log_Conf_Handle(ForcedLogConf);
  1656. }
  1657. // Let the helper modules (class installer/co-installers) get in on the act...
  1658. //
  1659. PropChangeParams.ClassInstallHeader.cbSize = sizeof(SP_CLASSINSTALL_HEADER);
  1660. PropChangeParams.ClassInstallHeader.InstallFunction = DIF_PROPERTYCHANGE;
  1661. PropChangeParams.StateChange = DICS_PROPCHANGE;
  1662. PropChangeParams.Scope = DICS_FLAG_GLOBAL;
  1663. // no need to set PropChangeParams.HwProfile, since this is a global property change.
  1664. DoInstallActionWithParams(DIF_PROPERTYCHANGE,
  1665. lpdmpd->hDevInfo,
  1666. lpdmpd->lpdi,
  1667. (PSP_CLASSINSTALL_HEADER)&PropChangeParams,
  1668. sizeof(PropChangeParams),
  1669. INSTALLACTION_CALL_CI
  1670. );
  1671. } else {
  1672. //-------------------------------------------------------------------
  1673. // The Use Automatic Settings is not selected.
  1674. //-------------------------------------------------------------------
  1675. bRet = pSaveCustomResSettings(lpdmpd,hMachine);
  1676. }
  1677. return bRet;
  1678. }
  1679. BOOL
  1680. pSaveCustomResSettings(
  1681. LPDMPROP_DATA lpdmpd,
  1682. IN HMACHINE hMachine
  1683. )
  1684. /*++
  1685. Routine Description:
  1686. This routine saves custom (user edited) resources. MEMPHIS COMPATIBLE but
  1687. extracted from Memphis version of SaveDevResSetting().
  1688. Arguments:
  1689. lpdmpd Property data.
  1690. Return Value:
  1691. Returns TRUE if the function succeeded and FALSE if it failed.
  1692. --*/
  1693. {
  1694. HWND hDlg = lpdmpd->hDlg;
  1695. TCHAR szWarn[MAX_MSG_LEN];
  1696. TCHAR szTitle[MAX_MSG_LEN];
  1697. TCHAR szTemp[MAX_MSG_LEN];
  1698. DWORD dwPriority, dwLCPri;
  1699. LOG_CONF ForcedLogConf;
  1700. RES_DES ResDes;
  1701. HWND hList = GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST);
  1702. PITEMDATA pItemData = NULL;
  1703. LONG iCur;
  1704. BOOL bRet = FALSE;
  1705. SP_PROPCHANGE_PARAMS PropChangeParams;
  1706. DWORD HardReconfigFlag;
  1707. SP_DEVINSTALL_PARAMS DevInstallParams;
  1708. PRESDES_ENTRY pResList = NULL, pResDesEntry = NULL, pTemp = NULL;
  1709. PITEMDATA_LISTNODE ItemDataList = NULL, ItemDataListEntry, ItemDataListEnd = NULL;
  1710. PGENERIC_RESOURCE pGenRes;
  1711. ULONG i, ulFlags;
  1712. ULONG64 ulValue, ulLen, ulEnd;
  1713. LOG_CONF LogConf;
  1714. ULONG ulSize;
  1715. ULONG ulConfigFlags;
  1716. HCURSOR hOldCursor;
  1717. BOOL UsingMatch = FALSE;
  1718. LogConf = lpdmpd->SelectedLC;
  1719. if (LogConf == 0) {
  1720. LogConf = lpdmpd->MatchingLC;
  1721. UsingMatch = TRUE;
  1722. }
  1723. if (LogConf == 0) {
  1724. LogConf = lpdmpd->CurrentLC;
  1725. UsingMatch = FALSE;
  1726. }
  1727. if (LogConf == 0) {
  1728. //MYASSERT(FALSE);
  1729. return FALSE;
  1730. }
  1731. //
  1732. // form the "warning - do you want to continue" message
  1733. //
  1734. if(!LoadString(MyDllModuleHandle, IDS_MAKE_FORCED_TITLE, szTitle, MAX_MSG_LEN)) {
  1735. szTitle[0]=TEXT('\0');
  1736. }
  1737. if(!LoadString(MyDllModuleHandle, IDS_FORCEDCONFIG_WARN1, szWarn, MAX_MSG_LEN)) {
  1738. szWarn[0]=TEXT('\0');
  1739. }
  1740. if(LoadString(MyDllModuleHandle, IDS_FORCEDCONFIG_WARN2, szTemp, MAX_MSG_LEN)) {
  1741. lstrcat(szWarn, szTemp);
  1742. }
  1743. if(LoadString(MyDllModuleHandle, IDS_FORCEDCONFIG_WARN3, szTemp, MAX_MSG_LEN)) {
  1744. lstrcat(szWarn, szTemp);
  1745. }
  1746. if(LoadString(MyDllModuleHandle, IDS_FORCEDCONFIG_WARN4, szTemp, MAX_MSG_LEN)) {
  1747. lstrcat(szWarn, szTemp);
  1748. }
  1749. //
  1750. // If the LCPRI is soft configurable, and the user chooses YES to the
  1751. // warning, then save the new config. If the LCPRI is not soft
  1752. // configurable, just save with no warning
  1753. //
  1754. dwLCPri = pGetMinLCPriority(lpdmpd->lpdi->DevInst, lpdmpd->ConfigListLCType,hMachine);
  1755. if (((dwLCPri >= LCPRI_DESIRED) && (dwLCPri <= LCPRI_LASTSOFTCONFIG)) &&
  1756. (MessageBox(hDlg, szWarn, szTitle, MB_YESNO|MB_ICONEXCLAMATION) == IDNO)) {
  1757. //
  1758. // user doesn't want to change anything
  1759. //
  1760. bRet = FALSE;
  1761. } else {
  1762. //
  1763. // We're still using the selected basic LC, but use the range index
  1764. // embedded in the listview control
  1765. // ISSUE-2000/02/03-JamieHun Selected Basic LC, check if user overrode
  1766. // Need to check the value to see if a user overrode it (is this possible?)
  1767. //
  1768. bRet = TRUE;
  1769. if (CM_Get_First_Log_Conf_Ex(&ForcedLogConf, lpdmpd->lpdi->DevInst,
  1770. FORCED_LOG_CONF,hMachine) == CR_SUCCESS) {
  1771. CM_Free_Log_Conf_Ex(ForcedLogConf, 0,hMachine);
  1772. CM_Free_Log_Conf_Handle(ForcedLogConf);
  1773. }
  1774. //
  1775. // Save the current choices as the forced config
  1776. //
  1777. CM_Add_Empty_Log_Conf_Ex(&ForcedLogConf, lpdmpd->lpdi->DevInst, LCPRI_FORCECONFIG,
  1778. FORCED_LOG_CONF | PRIORITY_EQUAL_FIRST,hMachine);
  1779. pGetResDesDataList(LogConf, &pResList, FALSE,hMachine);
  1780. pResDesEntry = pResList;
  1781. if (UsingMatch && (lpdmpd->dwFlags & DMPROP_FLAG_MATCH_OUT_OF_ORDER)) {
  1782. //
  1783. // The resource descriptors are out-of-order. Maintain the original ordering.
  1784. //
  1785. // First, build up a linked list of the data in the listview resource items.
  1786. //
  1787. iCur = (int)ListView_GetNextItem(hList, -1, LVNI_ALL);
  1788. while (iCur != -1) {
  1789. pItemData = (PITEMDATA)pGetListViewItemData(hList, iCur, 0);
  1790. if (pItemData) {
  1791. //
  1792. // Allocate an item data list node for this data.
  1793. //
  1794. ItemDataListEntry = MyMalloc(sizeof(ITEMDATA_LISTNODE));
  1795. if (!ItemDataListEntry) {
  1796. bRet = FALSE;
  1797. goto clean0;
  1798. }
  1799. ItemDataListEntry->ItemData = pItemData;
  1800. ItemDataListEntry->Next = NULL;
  1801. //
  1802. // Append this new item to the end of our list.
  1803. //
  1804. if (ItemDataListEnd) {
  1805. ItemDataListEnd->Next = ItemDataListEntry;
  1806. } else {
  1807. ItemDataList = ItemDataListEntry;
  1808. }
  1809. ItemDataListEnd = ItemDataListEntry;
  1810. }
  1811. iCur = (int)ListView_GetNextItem(hList, iCur, LVNI_ALL);
  1812. }
  1813. //
  1814. // Now loop through each resdes entry, writing each one out. For each one, check
  1815. // to see if it has a corresponding entry in our listview item data list.
  1816. //
  1817. while (pResDesEntry) {
  1818. pGenRes = (PGENERIC_RESOURCE)pResDesEntry->ResDesData;
  1819. for(ItemDataListEntry = ItemDataList, ItemDataListEnd = NULL;
  1820. ItemDataListEntry;
  1821. ItemDataListEnd = ItemDataListEntry, ItemDataListEntry = ItemDataListEntry->Next)
  1822. {
  1823. if(pResDesEntry->ResDesType == ItemDataListEntry->ItemData->ResType) {
  1824. for (i = 0; i < pGenRes->GENERIC_Header.GENERIC_Count; i++) {
  1825. pGetRangeValues(pResDesEntry->ResDesData, pResDesEntry->ResDesType, i,
  1826. &ulValue, &ulLen, &ulEnd, NULL, &ulFlags);
  1827. if ((ItemDataListEntry->ItemData->ulLen == ulLen) &&
  1828. (ItemDataListEntry->ItemData->ulValue >= ulValue) &&
  1829. (ItemDataListEntry->ItemData->ulEnd <= ulEnd)) {
  1830. //
  1831. // We found the matching resource descriptor. Write this out.
  1832. //
  1833. pWriteValuesToForced(ForcedLogConf,
  1834. ItemDataListEntry->ItemData->ResType,
  1835. ItemDataListEntry->ItemData->RangeCount,
  1836. ItemDataListEntry->ItemData->MatchingResDes,
  1837. ItemDataListEntry->ItemData->ulValue,
  1838. ItemDataListEntry->ItemData->ulLen,
  1839. ItemDataListEntry->ItemData->ulEnd,
  1840. hMachine );
  1841. //
  1842. // Remove this item from our list.
  1843. //
  1844. if (ItemDataListEnd) {
  1845. ItemDataListEnd->Next = ItemDataListEntry->Next;
  1846. } else {
  1847. ItemDataList = ItemDataListEntry->Next;
  1848. }
  1849. MyFree(ItemDataListEntry);
  1850. break;
  1851. }
  1852. }
  1853. if(i < pGenRes->GENERIC_Header.GENERIC_Count) {
  1854. //
  1855. // Then we broke out of the loop early, which means we found a match
  1856. // already.
  1857. //
  1858. break;
  1859. }
  1860. }
  1861. }
  1862. //
  1863. // If we didn't find a match, then go ahead and write out the non-arbitrated
  1864. // resdes.
  1865. //
  1866. if (!ItemDataListEntry) {
  1867. pWriteResDesRangeToForced(ForcedLogConf,
  1868. pResDesEntry->ResDesType,
  1869. 0,
  1870. 0,
  1871. pResDesEntry->ResDesData,
  1872. hMachine);
  1873. }
  1874. pResDesEntry = (PRESDES_ENTRY)pResDesEntry->Next;
  1875. }
  1876. } else {
  1877. iCur = (int)ListView_GetNextItem(hList, -1, LVNI_ALL);
  1878. while (iCur != -1) {
  1879. pItemData = (PITEMDATA)pGetListViewItemData(hList, iCur, 0);
  1880. if (pItemData) {
  1881. // retrieve values
  1882. while (pResDesEntry &&
  1883. (pItemData->ResType != pResDesEntry->ResDesType)) {
  1884. //
  1885. // write out any preceding non arbitrated resources
  1886. //
  1887. pWriteResDesRangeToForced(ForcedLogConf,
  1888. pResDesEntry->ResDesType,
  1889. 0,
  1890. 0,
  1891. pResDesEntry->ResDesData,
  1892. hMachine);
  1893. pResDesEntry = (PRESDES_ENTRY)pResDesEntry->Next;
  1894. }
  1895. if (pGetMatchingResDes(pItemData->ulValue,
  1896. pItemData->ulLen,
  1897. pItemData->ulEnd,
  1898. pItemData->ResType,
  1899. LogConf,
  1900. &ResDes,
  1901. hMachine)) {
  1902. //
  1903. // Write the first range as the chosen forced resource
  1904. //
  1905. pWriteValuesToForced(ForcedLogConf, pItemData->ResType,
  1906. pItemData->RangeCount, ResDes,
  1907. pItemData->ulValue,
  1908. pItemData->ulLen,
  1909. pItemData->ulEnd,
  1910. hMachine);
  1911. }
  1912. }
  1913. if (pResDesEntry) {
  1914. pResDesEntry = (PRESDES_ENTRY)pResDesEntry->Next;
  1915. } else {
  1916. MYASSERT(pResDesEntry);
  1917. }
  1918. iCur = (int)ListView_GetNextItem(hList, iCur, LVNI_ALL);
  1919. }
  1920. while (pResDesEntry) {
  1921. //
  1922. // write out any subsequent non arbitrated resources
  1923. //
  1924. pWriteResDesRangeToForced(ForcedLogConf,
  1925. pResDesEntry->ResDesType,
  1926. 0,
  1927. 0,
  1928. pResDesEntry->ResDesData,
  1929. hMachine);
  1930. pResDesEntry = (PRESDES_ENTRY)pResDesEntry->Next;
  1931. }
  1932. }
  1933. CM_Free_Log_Conf_Handle(ForcedLogConf);
  1934. //
  1935. // consider clearing problem flags
  1936. //
  1937. ulSize = sizeof(ulConfigFlags);
  1938. if (CM_Get_DevInst_Registry_Property_Ex(lpdmpd->lpdi->DevInst,
  1939. CM_DRP_CONFIGFLAGS,
  1940. NULL, (LPBYTE)&ulConfigFlags,
  1941. &ulSize, 0,hMachine) == CR_SUCCESS) {
  1942. if ((ulConfigFlags & CONFIGFLAG_PARTIAL_LOG_CONF) != 0) {
  1943. //
  1944. // have flag(s) to change
  1945. // CONFIGFLAG_PARTIAL_LOG_CONF should be cleared - we should have written a complete config now
  1946. //
  1947. ulConfigFlags &= ~ (CONFIGFLAG_PARTIAL_LOG_CONF);
  1948. CM_Set_DevInst_Registry_Property_Ex(lpdmpd->lpdi->DevInst,
  1949. CM_DRP_CONFIGFLAGS,
  1950. (LPBYTE)&ulConfigFlags,
  1951. sizeof(ulConfigFlags),
  1952. 0,
  1953. hMachine);
  1954. }
  1955. }
  1956. //
  1957. // Give the class installer/co-installers a crack at the propchange process.
  1958. //
  1959. PropChangeParams.ClassInstallHeader.cbSize = sizeof(SP_CLASSINSTALL_HEADER);
  1960. PropChangeParams.ClassInstallHeader.InstallFunction = DIF_PROPERTYCHANGE;
  1961. PropChangeParams.StateChange = DICS_PROPCHANGE;
  1962. PropChangeParams.Scope = DICS_FLAG_GLOBAL;
  1963. // no need to set PropChangeParams.HwProfile, since this is a global property change.
  1964. DoInstallActionWithParams(DIF_PROPERTYCHANGE,
  1965. lpdmpd->hDevInfo,
  1966. lpdmpd->lpdi,
  1967. (PSP_CLASSINSTALL_HEADER)&PropChangeParams,
  1968. sizeof(PropChangeParams),
  1969. INSTALLACTION_CALL_CI | INSTALLACTION_NO_DEFAULT
  1970. );
  1971. //
  1972. // Check the Priority of this LC. If it is greater
  1973. // than LCPRI_LASTSOFTCONFIG, then we need to reboot
  1974. // otherwise try the dynamic changestate route.
  1975. //
  1976. if (CM_Get_Log_Conf_Priority_Ex(LogConf, &dwPriority, 0,hMachine) != CR_SUCCESS) {
  1977. dwPriority = LCPRI_LASTSOFTCONFIG;
  1978. }
  1979. if (dwPriority <= LCPRI_LASTSOFTCONFIG) {
  1980. //
  1981. // Do the default action for SoftConfigable devices, which
  1982. // will attempt to restart the device with the new config
  1983. // This could take a while so use an hourglass
  1984. //
  1985. hOldCursor = SetCursor(LoadCursor(NULL, IDC_WAIT));
  1986. DoInstallActionWithParams(DIF_PROPERTYCHANGE,
  1987. lpdmpd->hDevInfo,
  1988. lpdmpd->lpdi,
  1989. (PSP_CLASSINSTALL_HEADER)&PropChangeParams,
  1990. sizeof(PropChangeParams),
  1991. 0 // don't call class-installer, just do default action
  1992. );
  1993. SetCursor(hOldCursor);
  1994. HardReconfigFlag = 0;
  1995. } else if((dwPriority > LCPRI_LASTSOFTCONFIG) && (dwPriority <= LCPRI_RESTART)) {
  1996. HardReconfigFlag = DI_NEEDRESTART;
  1997. } else {
  1998. HardReconfigFlag = DI_NEEDREBOOT;
  1999. }
  2000. lpdmpd->dwFlags |= DMPROP_FLAG_CHANGESSAVED;
  2001. //
  2002. // Properties have changed, so set flags to indicate if restart/reboot is required,
  2003. // and to tell DevMgr to re-init the UI.
  2004. //
  2005. DevInstallParams.cbSize = sizeof(SP_DEVINSTALL_PARAMS);
  2006. if(SetupDiGetDeviceInstallParams(lpdmpd->hDevInfo,
  2007. lpdmpd->lpdi,
  2008. &DevInstallParams)) {
  2009. DevInstallParams.Flags |= (HardReconfigFlag | DI_PROPERTIES_CHANGE);
  2010. SetupDiSetDeviceInstallParams(lpdmpd->hDevInfo,
  2011. lpdmpd->lpdi,
  2012. &DevInstallParams
  2013. );
  2014. }
  2015. //
  2016. // If we need to reboot, then set a problem on the device that indicates this (in case
  2017. // the user doesn't listen to us, we want to flag this devnode so that the user will see
  2018. // that this devnode needs a reboot if they go into DevMgr, etc.)
  2019. //
  2020. if(HardReconfigFlag) {
  2021. PDEVICE_INFO_SET pDeviceInfoSet;
  2022. PDEVINFO_ELEM DevInfoElem;
  2023. if(!(pDeviceInfoSet = AccessDeviceInfoSet(lpdmpd->hDevInfo))) {
  2024. //
  2025. // We'd better be able to access this device information set!
  2026. // In case we couldn't don't bother trying to set the needs-reboot problem,
  2027. // because the whole mess is invalid!
  2028. //
  2029. MYASSERT(pDeviceInfoSet);
  2030. } else {
  2031. try {
  2032. DevInfoElem = FindAssociatedDevInfoElem(pDeviceInfoSet, lpdmpd->lpdi, NULL);
  2033. //
  2034. // We'd better be able to find this element!
  2035. //
  2036. MYASSERT(DevInfoElem);
  2037. //
  2038. // In case we can't find it, don't try to set any problem on the devnode.
  2039. //
  2040. if(DevInfoElem) {
  2041. SetDevnodeNeedsRebootProblem(DevInfoElem,
  2042. pDeviceInfoSet,
  2043. MSG_LOG_REBOOT_DEVRES
  2044. );
  2045. }
  2046. } finally {
  2047. UnlockDeviceInfoSet(pDeviceInfoSet);
  2048. }
  2049. }
  2050. }
  2051. }
  2052. clean0:
  2053. while (ItemDataList) {
  2054. ItemDataListEntry = ItemDataList->Next;
  2055. MyFree(ItemDataList);
  2056. ItemDataList = ItemDataListEntry;
  2057. }
  2058. pDeleteResDesDataList(pResList);
  2059. return bRet;
  2060. } // SaveCustomResSettings
  2061. BOOL
  2062. pWriteResDesRangeToForced(
  2063. IN LOG_CONF ForcedLogConf,
  2064. IN RESOURCEID ResType,
  2065. IN ULONG RangeIndex,
  2066. IN RES_DES RD, OPTIONAL
  2067. IN LPBYTE ResDesData, OPTIONAL
  2068. IN HMACHINE hMachine OPTIONAL
  2069. )
  2070. {
  2071. RES_DES ResDes;
  2072. ULONG ulSize;
  2073. LPBYTE pData = NULL;
  2074. BOOL Success = FALSE;
  2075. if ((RD == 0) && (ResDesData == NULL)) {
  2076. return FALSE; // pass in data or handle!
  2077. }
  2078. if (!ResDesData) {
  2079. if (CM_Get_Res_Des_Data_Size_Ex(&ulSize, RD, DEVRES_WIDTH_FLAGS,hMachine) != CR_SUCCESS) {
  2080. return FALSE;
  2081. }
  2082. pData = MyMalloc(ulSize);
  2083. if (pData == NULL) {
  2084. return FALSE;
  2085. }
  2086. if (CM_Get_Res_Des_Data_Ex(RD, pData, ulSize, DEVRES_WIDTH_FLAGS,hMachine) != CR_SUCCESS) {
  2087. MyFree(pData);
  2088. return FALSE;
  2089. }
  2090. } else {
  2091. pData = ResDesData;
  2092. }
  2093. //
  2094. // convert the first range data into hdr data
  2095. //
  2096. switch (ResType) {
  2097. case ResType_Mem: {
  2098. PMEM_RESOURCE pMemData = (PMEM_RESOURCE)pData;
  2099. PMEM_RESOURCE pForced = (PMEM_RESOURCE)MyMalloc(sizeof(MEM_RESOURCE));
  2100. if (!pForced) {
  2101. break;
  2102. }
  2103. pForced->MEM_Header.MD_Count = 0;
  2104. pForced->MEM_Header.MD_Type = MType_Range;
  2105. pForced->MEM_Header.MD_Alloc_Base = pMemData->MEM_Data[RangeIndex].MR_Min;
  2106. pForced->MEM_Header.MD_Alloc_End = pMemData->MEM_Data[RangeIndex].MR_Min +
  2107. pMemData->MEM_Data[RangeIndex].MR_nBytes - 1;
  2108. pForced->MEM_Header.MD_Flags = pMemData->MEM_Data[RangeIndex].MR_Flags;
  2109. pForced->MEM_Header.MD_Reserved = 0;
  2110. Success = CM_Add_Res_Des_Ex(NULL,
  2111. ForcedLogConf,
  2112. ResType_Mem,
  2113. pForced,
  2114. sizeof(MEM_RESOURCE),
  2115. DEVRES_WIDTH_FLAGS,
  2116. hMachine) == CR_SUCCESS;
  2117. MyFree(pForced);
  2118. break;
  2119. }
  2120. case ResType_IO: {
  2121. PIO_RESOURCE pIoData = (PIO_RESOURCE)pData;
  2122. PIO_RESOURCE pForced = (PIO_RESOURCE)MyMalloc(sizeof(IO_RESOURCE));
  2123. if (!pForced) {
  2124. break;
  2125. }
  2126. pForced->IO_Header.IOD_Count = 0;
  2127. pForced->IO_Header.IOD_Type = IOType_Range;
  2128. pForced->IO_Header.IOD_Alloc_Base = pIoData->IO_Data[RangeIndex].IOR_Min;
  2129. pForced->IO_Header.IOD_Alloc_End = pIoData->IO_Data[RangeIndex].IOR_Min +
  2130. pIoData->IO_Data[RangeIndex].IOR_nPorts - 1;
  2131. pForced->IO_Header.IOD_DesFlags = pIoData->IO_Data[RangeIndex].IOR_RangeFlags;
  2132. Success = CM_Add_Res_Des_Ex(NULL,
  2133. ForcedLogConf,
  2134. ResType_IO,
  2135. pForced,
  2136. sizeof(IO_RESOURCE),
  2137. DEVRES_WIDTH_FLAGS,
  2138. hMachine) == CR_SUCCESS;
  2139. MyFree(pForced);
  2140. break;
  2141. }
  2142. case ResType_DMA: {
  2143. PDMA_RESOURCE pDmaData = (PDMA_RESOURCE)pData;
  2144. PDMA_RESOURCE pForced = (PDMA_RESOURCE)MyMalloc(sizeof(DMA_RESOURCE));
  2145. if (!pForced) {
  2146. break;
  2147. }
  2148. pForced->DMA_Header.DD_Count = 0;
  2149. pForced->DMA_Header.DD_Type = DType_Range;
  2150. pForced->DMA_Header.DD_Flags = pDmaData->DMA_Data[RangeIndex].DR_Flags;
  2151. pForced->DMA_Header.DD_Alloc_Chan = pDmaData->DMA_Data[RangeIndex].DR_Min;
  2152. Success = CM_Add_Res_Des_Ex(NULL,
  2153. ForcedLogConf,
  2154. ResType_DMA,
  2155. pForced,
  2156. sizeof(DMA_RESOURCE),
  2157. DEVRES_WIDTH_FLAGS,
  2158. hMachine) == CR_SUCCESS;
  2159. MyFree(pForced);
  2160. break;
  2161. }
  2162. case ResType_IRQ: {
  2163. DEVRES_PIRQ_RESOURCE pIrqData = (DEVRES_PIRQ_RESOURCE)pData;
  2164. DEVRES_PIRQ_RESOURCE pForced = (DEVRES_PIRQ_RESOURCE)MyMalloc(sizeof(DEVRES_IRQ_RESOURCE));
  2165. if (!pForced) {
  2166. break;
  2167. }
  2168. pForced->IRQ_Header.IRQD_Count = 0;
  2169. pForced->IRQ_Header.IRQD_Type = IRQType_Range;
  2170. pForced->IRQ_Header.IRQD_Flags = pIrqData->IRQ_Data[RangeIndex].IRQR_Flags;
  2171. pForced->IRQ_Header.IRQD_Alloc_Num = pIrqData->IRQ_Data[RangeIndex].IRQR_Min;
  2172. pForced->IRQ_Header.IRQD_Affinity = (DEVRES_AFFINITY)(-1);
  2173. Success = CM_Add_Res_Des_Ex(NULL,
  2174. ForcedLogConf,
  2175. ResType_IRQ,
  2176. pForced,
  2177. sizeof(DEVRES_IRQ_RESOURCE),
  2178. DEVRES_WIDTH_FLAGS,
  2179. hMachine) == CR_SUCCESS;
  2180. MyFree(pForced);
  2181. break;
  2182. }
  2183. case ResType_BusNumber: {
  2184. PBUSNUMBER_RESOURCE pBusData = (PBUSNUMBER_RESOURCE)pData;
  2185. PBUSNUMBER_RESOURCE pForced = (PBUSNUMBER_RESOURCE)MyMalloc(sizeof(BUSNUMBER_RESOURCE));
  2186. if (!pForced) {
  2187. break;
  2188. }
  2189. pForced->BusNumber_Header.BUSD_Count = 0;
  2190. pForced->BusNumber_Header.BUSD_Type = BusNumberType_Range;
  2191. pForced->BusNumber_Header.BUSD_Flags = pBusData->BusNumber_Data[RangeIndex].BUSR_Flags;
  2192. pForced->BusNumber_Header.BUSD_Alloc_Base = pBusData->BusNumber_Data[RangeIndex].BUSR_Min;
  2193. pForced->BusNumber_Header.BUSD_Alloc_End = pBusData->BusNumber_Data[RangeIndex].BUSR_Min +
  2194. pBusData->BusNumber_Data[RangeIndex].BUSR_nBusNumbers;
  2195. Success = CM_Add_Res_Des_Ex(NULL,
  2196. ForcedLogConf,
  2197. ResType_BusNumber,
  2198. pForced,
  2199. sizeof(BUSNUMBER_RESOURCE),
  2200. DEVRES_WIDTH_FLAGS,
  2201. hMachine) == CR_SUCCESS;
  2202. MyFree(pForced);
  2203. break;
  2204. }
  2205. case ResType_DevicePrivate: {
  2206. PDEVPRIVATE_RESOURCE pPrvData = (PDEVPRIVATE_RESOURCE)pData;
  2207. PDEVPRIVATE_RESOURCE pForced = (PDEVPRIVATE_RESOURCE)MyMalloc(sizeof(DEVPRIVATE_RESOURCE));
  2208. if (!pForced) {
  2209. break;
  2210. }
  2211. pForced->PRV_Header.PD_Count = 0;
  2212. pForced->PRV_Header.PD_Type = PType_Range;
  2213. pForced->PRV_Header.PD_Data1 = pPrvData->PRV_Data[RangeIndex].PR_Data1;
  2214. pForced->PRV_Header.PD_Data2 = pPrvData->PRV_Data[RangeIndex].PR_Data2;
  2215. pForced->PRV_Header.PD_Data3 = pPrvData->PRV_Data[RangeIndex].PR_Data3;
  2216. pForced->PRV_Header.PD_Flags = 0;
  2217. Success = CM_Add_Res_Des_Ex(NULL,
  2218. ForcedLogConf,
  2219. ResType_DevicePrivate,
  2220. pForced,
  2221. sizeof(DEVPRIVATE_RESOURCE),
  2222. DEVRES_WIDTH_FLAGS,
  2223. hMachine) == CR_SUCCESS;
  2224. MyFree(pForced);
  2225. break;
  2226. }
  2227. case ResType_PcCardConfig: {
  2228. Success = CM_Add_Res_Des_Ex(NULL,
  2229. ForcedLogConf,
  2230. ResType_PcCardConfig,
  2231. pData,
  2232. sizeof(PCCARD_RESOURCE),
  2233. DEVRES_WIDTH_FLAGS,
  2234. hMachine);
  2235. break;
  2236. }
  2237. }
  2238. if (pData != ResDesData) {
  2239. MyFree(pData);
  2240. }
  2241. return Success;
  2242. } // WriteResDesRangeToForced
  2243. BOOL
  2244. pWriteValuesToForced(
  2245. IN LOG_CONF ForcedLogConf,
  2246. IN RESOURCEID ResType,
  2247. IN ULONG RangeIndex,
  2248. IN RES_DES RD,
  2249. IN ULONG64 ulValue,
  2250. IN ULONG64 ulLen,
  2251. IN ULONG64 ulEnd,
  2252. IN HMACHINE hMachine
  2253. )
  2254. {
  2255. RES_DES ResDes;
  2256. ULONG ulSize;
  2257. LPBYTE pData = NULL;
  2258. BOOL Success = FALSE;
  2259. if (CM_Get_Res_Des_Data_Size_Ex(&ulSize, RD, DEVRES_WIDTH_FLAGS,hMachine) != CR_SUCCESS) {
  2260. return FALSE;
  2261. }
  2262. pData = MyMalloc(ulSize);
  2263. if (pData == NULL) {
  2264. return FALSE;
  2265. }
  2266. if (CM_Get_Res_Des_Data_Ex(RD, pData, ulSize, DEVRES_WIDTH_FLAGS,hMachine) != CR_SUCCESS) {
  2267. MyFree(pData);
  2268. return FALSE;
  2269. }
  2270. //
  2271. // convert the first range data into hdr data
  2272. //
  2273. switch (ResType) {
  2274. case ResType_Mem: {
  2275. PMEM_RESOURCE pMemData = (PMEM_RESOURCE)pData;
  2276. PMEM_RESOURCE pForced = (PMEM_RESOURCE)MyMalloc(sizeof(MEM_RESOURCE));
  2277. if (!pForced) {
  2278. break;
  2279. }
  2280. pForced->MEM_Header.MD_Count = 0;
  2281. pForced->MEM_Header.MD_Type = MType_Range;
  2282. pForced->MEM_Header.MD_Alloc_Base = ulValue;
  2283. pForced->MEM_Header.MD_Alloc_End = ulEnd;
  2284. pForced->MEM_Header.MD_Flags = pMemData->MEM_Data[RangeIndex].MR_Flags;
  2285. pForced->MEM_Header.MD_Reserved = 0;
  2286. Success = CM_Add_Res_Des_Ex(NULL,
  2287. ForcedLogConf,
  2288. ResType_Mem,
  2289. pForced,
  2290. sizeof(MEM_RESOURCE),
  2291. DEVRES_WIDTH_FLAGS,
  2292. hMachine) == CR_SUCCESS;
  2293. MyFree(pForced);
  2294. break;
  2295. }
  2296. case ResType_IO: {
  2297. PIO_RESOURCE pIoData = (PIO_RESOURCE)pData;
  2298. PIO_RESOURCE pForced = (PIO_RESOURCE)MyMalloc(sizeof(IO_RESOURCE));
  2299. if (!pForced) {
  2300. break;
  2301. }
  2302. pForced->IO_Header.IOD_Count = 0;
  2303. pForced->IO_Header.IOD_Type = IOType_Range;
  2304. pForced->IO_Header.IOD_Alloc_Base = ulValue;
  2305. pForced->IO_Header.IOD_Alloc_End = ulEnd;
  2306. pForced->IO_Header.IOD_DesFlags = pIoData->IO_Data[RangeIndex].IOR_RangeFlags;
  2307. Success = CM_Add_Res_Des_Ex(NULL,
  2308. ForcedLogConf,
  2309. ResType_IO,
  2310. pForced,
  2311. sizeof(IO_RESOURCE),
  2312. DEVRES_WIDTH_FLAGS,
  2313. hMachine) == CR_SUCCESS;
  2314. MyFree(pForced);
  2315. break;
  2316. }
  2317. case ResType_DMA: {
  2318. PDMA_RESOURCE pDmaData = (PDMA_RESOURCE)pData;
  2319. PDMA_RESOURCE pForced = (PDMA_RESOURCE)MyMalloc(sizeof(DMA_RESOURCE));
  2320. if (!pForced) {
  2321. break;
  2322. }
  2323. pForced->DMA_Header.DD_Count = 0;
  2324. pForced->DMA_Header.DD_Type = DType_Range;
  2325. pForced->DMA_Header.DD_Flags = pDmaData->DMA_Data[RangeIndex].DR_Flags;
  2326. pForced->DMA_Header.DD_Alloc_Chan = (ULONG)ulValue;
  2327. Success = CM_Add_Res_Des_Ex(NULL,
  2328. ForcedLogConf,
  2329. ResType_DMA,
  2330. pForced,
  2331. sizeof(DMA_RESOURCE),
  2332. DEVRES_WIDTH_FLAGS,
  2333. hMachine) == CR_SUCCESS;
  2334. MyFree(pForced);
  2335. break;
  2336. }
  2337. case ResType_IRQ: {
  2338. DEVRES_PIRQ_RESOURCE pIrqData = (DEVRES_PIRQ_RESOURCE)pData;
  2339. DEVRES_PIRQ_RESOURCE pForced = (DEVRES_PIRQ_RESOURCE)MyMalloc(sizeof(DEVRES_IRQ_RESOURCE));
  2340. if (!pForced) {
  2341. break;
  2342. }
  2343. pForced->IRQ_Header.IRQD_Count = 0;
  2344. pForced->IRQ_Header.IRQD_Type = IRQType_Range;
  2345. pForced->IRQ_Header.IRQD_Flags = pIrqData->IRQ_Data[RangeIndex].IRQR_Flags;
  2346. pForced->IRQ_Header.IRQD_Alloc_Num = (ULONG)ulValue;
  2347. pForced->IRQ_Header.IRQD_Affinity = (DEVRES_AFFINITY)(-1);
  2348. Success = CM_Add_Res_Des_Ex(NULL,
  2349. ForcedLogConf,
  2350. ResType_IRQ,
  2351. pForced,
  2352. sizeof(DEVRES_IRQ_RESOURCE),
  2353. DEVRES_WIDTH_FLAGS,
  2354. hMachine) == CR_SUCCESS;
  2355. MyFree(pForced);
  2356. break;
  2357. }
  2358. case ResType_BusNumber: {
  2359. PBUSNUMBER_RESOURCE pBusData = (PBUSNUMBER_RESOURCE)pData;
  2360. PBUSNUMBER_RESOURCE pForced = (PBUSNUMBER_RESOURCE)MyMalloc(sizeof(BUSNUMBER_RESOURCE));
  2361. if (!pForced) {
  2362. break;
  2363. }
  2364. pForced->BusNumber_Header.BUSD_Count = 0;
  2365. pForced->BusNumber_Header.BUSD_Type = BusNumberType_Range;
  2366. pForced->BusNumber_Header.BUSD_Flags = pBusData->BusNumber_Data[RangeIndex].BUSR_Flags;
  2367. pForced->BusNumber_Header.BUSD_Alloc_Base = (ULONG)ulValue;
  2368. pForced->BusNumber_Header.BUSD_Alloc_End = (ULONG)ulEnd;
  2369. Success = CM_Add_Res_Des_Ex(NULL,
  2370. ForcedLogConf,
  2371. ResType_BusNumber,
  2372. pForced,
  2373. sizeof(BUSNUMBER_RESOURCE),
  2374. DEVRES_WIDTH_FLAGS,
  2375. hMachine) == CR_SUCCESS;
  2376. MyFree(pForced);
  2377. break;
  2378. }
  2379. case ResType_DevicePrivate: {
  2380. break;
  2381. }
  2382. case ResType_PcCardConfig: {
  2383. break;
  2384. }
  2385. }
  2386. if (pData) {
  2387. MyFree(pData);
  2388. }
  2389. return Success;
  2390. } // WriteValuesToForced
  2391. BOOL
  2392. MakeResourceData(
  2393. OUT LPBYTE *ppResourceData,
  2394. OUT PULONG pulSize,
  2395. IN RESOURCEID ResType,
  2396. IN ULONG64 ulValue,
  2397. IN ULONG64 ulLen,
  2398. IN ULONG ulFlags
  2399. )
  2400. {
  2401. BOOL bStatus = TRUE;
  2402. try {
  2403. switch (ResType) {
  2404. case ResType_Mem: {
  2405. PMEM_RESOURCE p;
  2406. *pulSize = sizeof(MEM_RESOURCE);
  2407. if (ppResourceData) {
  2408. *ppResourceData = MyMalloc(*pulSize);
  2409. p = (PMEM_RESOURCE)(*ppResourceData);
  2410. if (!p) {
  2411. bStatus = FALSE;
  2412. break;
  2413. }
  2414. p->MEM_Header.MD_Count = 0;
  2415. p->MEM_Header.MD_Type = MType_Range;
  2416. p->MEM_Header.MD_Alloc_Base = ulValue;
  2417. p->MEM_Header.MD_Alloc_End = ulValue + ulLen - 1;
  2418. p->MEM_Header.MD_Flags = ulFlags;
  2419. p->MEM_Header.MD_Reserved = 0;
  2420. }
  2421. break;
  2422. }
  2423. case ResType_IO: {
  2424. PIO_RESOURCE p;
  2425. *pulSize = sizeof(IO_RESOURCE);
  2426. if (ppResourceData) {
  2427. *ppResourceData = MyMalloc(*pulSize);
  2428. p = (PIO_RESOURCE)(*ppResourceData);
  2429. if (!p) {
  2430. bStatus = FALSE;
  2431. break;
  2432. }
  2433. p->IO_Header.IOD_Count = 0;
  2434. p->IO_Header.IOD_Type = IOType_Range;
  2435. p->IO_Header.IOD_Alloc_Base = ulValue;
  2436. p->IO_Header.IOD_Alloc_End = ulValue + ulLen - 1;
  2437. p->IO_Header.IOD_DesFlags = ulFlags;
  2438. }
  2439. break;
  2440. }
  2441. case ResType_DMA: {
  2442. PDMA_RESOURCE p;
  2443. *pulSize = sizeof(DMA_RESOURCE);
  2444. if (ppResourceData) {
  2445. *ppResourceData = MyMalloc(*pulSize);
  2446. p = (PDMA_RESOURCE)(*ppResourceData);
  2447. if (!p) {
  2448. bStatus = FALSE;
  2449. break;
  2450. }
  2451. p->DMA_Header.DD_Count = 0;
  2452. p->DMA_Header.DD_Type = DType_Range;
  2453. p->DMA_Header.DD_Flags = ulFlags;
  2454. p->DMA_Header.DD_Alloc_Chan = (ULONG)ulValue;
  2455. }
  2456. break;
  2457. }
  2458. case ResType_IRQ: {
  2459. DEVRES_PIRQ_RESOURCE p;
  2460. *pulSize = sizeof(DEVRES_IRQ_RESOURCE);
  2461. if (ppResourceData) {
  2462. *ppResourceData = MyMalloc(*pulSize);
  2463. p = (DEVRES_PIRQ_RESOURCE)(*ppResourceData);
  2464. if (!p) {
  2465. bStatus = FALSE;
  2466. break;
  2467. }
  2468. p->IRQ_Header.IRQD_Count = 0;
  2469. p->IRQ_Header.IRQD_Type = IRQType_Range;
  2470. p->IRQ_Header.IRQD_Flags = ulFlags;
  2471. p->IRQ_Header.IRQD_Alloc_Num = (ULONG)ulValue;
  2472. p->IRQ_Header.IRQD_Affinity = (DEVRES_AFFINITY)(-1); // for any processor
  2473. }
  2474. break;
  2475. }
  2476. default:
  2477. //
  2478. // ResTypeEditable or ResType_MAX may be wrong if this ASSERT's
  2479. //
  2480. MYASSERT(FALSE);
  2481. bStatus = FALSE;
  2482. }
  2483. } except(EXCEPTION_EXECUTE_HANDLER) {
  2484. bStatus = FALSE;
  2485. }
  2486. return bStatus;
  2487. } // MakeResourceData
  2488. BOOL
  2489. pShowWindow(
  2490. IN HWND hWnd,
  2491. IN int nShow
  2492. )
  2493. /*++
  2494. Routine Description:
  2495. A variation of ShowWindow that enables/disables window
  2496. Arguments:
  2497. (See ShowWindow)
  2498. hWnd - handle of window to show
  2499. nShow - typically SW_HIDE or SW_SHOW
  2500. Return Value:
  2501. success status of ShowWindow
  2502. --*/
  2503. {
  2504. EnableWindow(hWnd,nShow!=SW_HIDE);
  2505. return ShowWindow(hWnd,nShow);
  2506. }
  2507. BOOL
  2508. pEnableWindow(
  2509. IN HWND hWnd,
  2510. IN BOOL Enable
  2511. )
  2512. /*++
  2513. Routine Description:
  2514. A variation of EnableWindow that only enables a window if it is visible
  2515. Arguments:
  2516. (See EnableWindow)
  2517. hWnd - handle of window to enable/disable
  2518. Enable - TRUE enables window (if window visible) FALSE disables window
  2519. Return Value:
  2520. success status of EnableWindow
  2521. --*/
  2522. {
  2523. //
  2524. // I had to use GetWindowLong, as IsWindowVisible also checks parent flag
  2525. // and parent is hidden until dialog is initialized
  2526. //
  2527. if((GetWindowLong(hWnd,GWL_STYLE) & WS_VISIBLE) == FALSE) {
  2528. Enable = FALSE;
  2529. }
  2530. return EnableWindow(hWnd,Enable);
  2531. }
  2532. BOOL
  2533. pGetResDesDataList(
  2534. IN LOG_CONF LogConf,
  2535. IN OUT PRESDES_ENTRY *pResList,
  2536. IN BOOL bArbitratedOnly,
  2537. IN HMACHINE hMachine
  2538. )
  2539. /*++
  2540. Routine Description:
  2541. Creates a list of resource descriptors for further processing
  2542. Arguments:
  2543. LogConf - log config of interest
  2544. pResList - list out
  2545. bArbitratedOnly - filter out non-arbitrated resources
  2546. hMachine - machine that LogConf is on
  2547. Return Value:
  2548. None.
  2549. --*/
  2550. {
  2551. BOOL bStatus = TRUE;
  2552. CONFIGRET Status = CR_SUCCESS;
  2553. PRESDES_ENTRY pHead = NULL, pEntry = NULL, pPrevious = NULL, pTemp = NULL;
  2554. RES_DES ResDes;
  2555. RESOURCEID ResType;
  2556. ULONG ulSize;
  2557. LPBYTE pData = NULL;
  2558. //
  2559. // Retrieve each res des in this log conf
  2560. //
  2561. Status = CM_Get_Next_Res_Des_Ex(&ResDes, LogConf, ResType_All, &ResType, 0,hMachine);
  2562. while (Status == CR_SUCCESS) {
  2563. if (bArbitratedOnly && (ResType <= ResType_None || ResType > ResType_MAX)) {
  2564. goto NextResDes;
  2565. }
  2566. if (bArbitratedOnly && ResTypeEditable[ResType] == FALSE) {
  2567. goto NextResDes;
  2568. }
  2569. if (CM_Get_Res_Des_Data_Size_Ex(&ulSize, ResDes, DEVRES_WIDTH_FLAGS,hMachine) != CR_SUCCESS) {
  2570. CM_Free_Res_Des_Handle(ResDes);
  2571. bStatus = FALSE;
  2572. goto Clean0;
  2573. }
  2574. if (ulSize>0) {
  2575. pData = MyMalloc(ulSize);
  2576. if (pData == NULL) {
  2577. CM_Free_Res_Des_Handle(ResDes);
  2578. bStatus = FALSE;
  2579. goto Clean0;
  2580. }
  2581. if (CM_Get_Res_Des_Data_Ex(ResDes, pData, ulSize, DEVRES_WIDTH_FLAGS,hMachine) != CR_SUCCESS) {
  2582. CM_Free_Res_Des_Handle(ResDes);
  2583. MyFree(pData);
  2584. bStatus = FALSE;
  2585. goto Clean0;
  2586. }
  2587. } else {
  2588. pData = NULL;
  2589. }
  2590. //
  2591. // Allocate a node for this res des and attach it to the list
  2592. //
  2593. pEntry = MyMalloc(sizeof(RESDES_ENTRY));
  2594. if (pEntry == NULL) {
  2595. CM_Free_Res_Des_Handle(ResDes);
  2596. MyFree(pData);
  2597. bStatus = FALSE;
  2598. goto Clean0;
  2599. }
  2600. pEntry->ResDesData = pData;
  2601. pEntry->ResDesType = ResType;
  2602. pEntry->ResDesDataSize = ulSize;
  2603. pEntry->ResDesHandle = ResDes;
  2604. pEntry->Next = NULL;
  2605. pEntry->CrossLink = NULL;
  2606. if (!pHead) {
  2607. pHead = pEntry; // first entry
  2608. }
  2609. if (pPrevious) {
  2610. pPrevious->Next = pEntry; // attach to previous entry
  2611. }
  2612. pPrevious = pEntry;
  2613. //
  2614. // Get next res des in LogConf
  2615. //
  2616. NextResDes:
  2617. Status = CM_Get_Next_Res_Des_Ex(&ResDes, ResDes, ResType_All, &ResType, 0,hMachine);
  2618. }
  2619. bStatus = TRUE;
  2620. Clean0:
  2621. if (!bStatus) {
  2622. pDeleteResDesDataList(pHead);
  2623. } else {
  2624. *pResList = pHead;
  2625. }
  2626. return bStatus;
  2627. } // GetResDesDataList
  2628. VOID
  2629. pDeleteResDesDataList(
  2630. IN PRESDES_ENTRY pResList
  2631. )
  2632. /*++
  2633. Routine Description:
  2634. Deletes memory used by RESDES list
  2635. Arguments:
  2636. pResList - list returned by GetResDesDataList
  2637. Return Value:
  2638. None.
  2639. --*/
  2640. {
  2641. PRESDES_ENTRY pTemp;
  2642. while (pResList) {
  2643. pTemp = pResList;
  2644. pResList = (PRESDES_ENTRY)pResList->Next;
  2645. if (pTemp->ResDesData) {
  2646. MyFree(pTemp->ResDesData);
  2647. }
  2648. if (pTemp->ResDesHandle) {
  2649. CM_Free_Res_Des_Handle(pTemp->ResDesHandle);
  2650. }
  2651. MyFree(pTemp);
  2652. }
  2653. }
  2654. VOID
  2655. pHideAllControls(
  2656. IN LPDMPROP_DATA lpdmpd
  2657. )
  2658. /*++
  2659. Routine Description:
  2660. Hide (and disable) all controls - start off with a clean slate
  2661. Only Icon & device description will be visible
  2662. Arguments:
  2663. hDlg = dialog handle of controls
  2664. lpdmpd = property data
  2665. Return Value:
  2666. none
  2667. --*/
  2668. {
  2669. HWND hDlg = lpdmpd->hDlg;
  2670. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_SETTINGSTATE), SW_HIDE);
  2671. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_RESOURCES_TEXT), SW_HIDE);
  2672. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST), SW_HIDE);
  2673. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NOALLOCTEXT), SW_HIDE);
  2674. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LCTEXT), SW_HIDE);
  2675. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST), SW_HIDE);
  2676. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_CHANGE_TEXT ), SW_HIDE);
  2677. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_USESYSSETTINGS), SW_HIDE);
  2678. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CHANGE), SW_HIDE);
  2679. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MAKEFORCED), SW_HIDE);
  2680. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MFPARENT), SW_HIDE);
  2681. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MFPARENT_DESC), SW_HIDE);
  2682. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTDEVTEXT), SW_HIDE);
  2683. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTINFOLIST), SW_HIDE);
  2684. lpdmpd->dwFlags |= DMPROP_FLAG_VIEWONLYRES;
  2685. }
  2686. VOID
  2687. pShowViewNoResources(
  2688. IN LPDMPROP_DATA lpdmpd
  2689. )
  2690. /*++
  2691. Routine Description:
  2692. Show page indicating this device has no resources
  2693. Arguments:
  2694. hDlg = dialog handle of controls
  2695. lpdmpd = property data
  2696. Return Value:
  2697. none
  2698. --*/
  2699. {
  2700. HWND hDlg = lpdmpd->hDlg;
  2701. TCHAR szString[MAX_PATH];
  2702. pHideAllControls(lpdmpd); // all hidden and disabled
  2703. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_RESOURCES_TEXT), SW_SHOW); // show and enable text
  2704. LoadString(MyDllModuleHandle, IDS_DEVRES_NO_RESOURCES, szString, MAX_PATH);
  2705. SetDlgItemText(hDlg, IDC_DEVRES_NO_RESOURCES_TEXT, szString);
  2706. }
  2707. BOOL
  2708. pShowViewMFReadOnly(
  2709. IN LPDMPROP_DATA lpdmpd,
  2710. IN BOOL HideIfProb
  2711. )
  2712. /*++
  2713. Routine Description:
  2714. Show page apropriate for multifunction card that cannot be edited
  2715. Resource settings are visible
  2716. Arguments:
  2717. hDlg = dialog handle of controls
  2718. lpdmpd = property data
  2719. Return Value:
  2720. none
  2721. --*/
  2722. {
  2723. TCHAR szString[MAX_PATH];
  2724. DEVNODE dnParent;
  2725. ULONG ulSize;
  2726. HWND hDlg = lpdmpd->hDlg;
  2727. HMACHINE hMachine = pGetMachine(lpdmpd);
  2728. pHideAllControls(lpdmpd); // all hidden and disabled
  2729. //pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LCTEXT), SW_SHOW); // show config information
  2730. //pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST), SW_SHOW); // show
  2731. //pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTDEVTEXT), SW_SHOW); // show conflict information space
  2732. //pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTINFOLIST), SW_SHOW);
  2733. //
  2734. // indicate we cannot change as it's multi-function
  2735. //
  2736. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_CHANGE_TEXT), SW_SHOW);
  2737. if (LoadString(MyDllModuleHandle, IDS_DEVRES_NO_CHANGE_MF, szString, MAX_PATH)) {
  2738. SetDlgItemText(hDlg, IDC_DEVRES_NO_CHANGE_TEXT, szString);
  2739. }
  2740. //
  2741. // for parent description
  2742. //
  2743. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MFPARENT), SW_SHOW);
  2744. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MFPARENT_DESC), SW_SHOW);
  2745. //
  2746. // Get the Parent's Description.
  2747. //
  2748. LoadString(MyDllModuleHandle, IDS_DEVNAME_UNK, szString, MAX_PATH);
  2749. if (lpdmpd->lpdi->DevInst) {
  2750. if (CM_Get_Parent_Ex(&dnParent, lpdmpd->lpdi->DevInst, 0,hMachine)
  2751. == CR_SUCCESS) {
  2752. //
  2753. // First, try to retrieve friendly name, then fall back to device description.
  2754. //
  2755. ulSize = MAX_PATH * sizeof(TCHAR);
  2756. if(CM_Get_DevNode_Registry_Property_Ex(dnParent, CM_DRP_FRIENDLYNAME,
  2757. NULL, szString, &ulSize, 0,hMachine) != CR_SUCCESS) {
  2758. ulSize = MAX_PATH * sizeof(TCHAR);
  2759. CM_Get_DevNode_Registry_Property_Ex(dnParent, CM_DRP_DEVICEDESC,
  2760. NULL, szString, &ulSize, 0,hMachine);
  2761. }
  2762. }
  2763. }
  2764. SetDlgItemText(hDlg, IDC_DEVRES_MFPARENT_DESC, szString);
  2765. //
  2766. // load and display current config (if any)
  2767. // return FALSE if no current config
  2768. //
  2769. return pLoadCurrentConfig(lpdmpd,HideIfProb);
  2770. }
  2771. BOOL
  2772. pShowViewReadOnly(
  2773. IN LPDMPROP_DATA lpdmpd,
  2774. IN BOOL HideIfProb
  2775. )
  2776. /*++
  2777. Routine Description:
  2778. Show page of resources, don't allow editing, don't show editing controls
  2779. Arguments:
  2780. hDlg = dialog handle of controls
  2781. lpdmpd = property data
  2782. Return Value:
  2783. none
  2784. --*/
  2785. {
  2786. HWND hDlg = lpdmpd->hDlg;
  2787. pHideAllControls(lpdmpd); // all hidden and disabled
  2788. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LCTEXT), SW_SHOW); // show
  2789. EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_LCTEXT), FALSE);
  2790. ShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST), SW_SHOW); // shown disabled
  2791. EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST), FALSE);
  2792. ShowWindow(GetDlgItem(hDlg, IDC_DEVRES_USESYSSETTINGS), SW_SHOW); // shown disabled
  2793. EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_USESYSSETTINGS), FALSE);
  2794. ShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CHANGE), SW_SHOW); // shown disabled
  2795. EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_CHANGE), FALSE);
  2796. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTDEVTEXT), SW_SHOW); // show conflict information space
  2797. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTINFOLIST), SW_SHOW);
  2798. //
  2799. // will indicate if we're showing system settings or forced settings
  2800. //
  2801. CheckDlgButton(hDlg, IDC_DEVRES_USESYSSETTINGS, (lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS ) ? TRUE : FALSE);
  2802. //
  2803. // load and display current config (if any)
  2804. // return FALSE if no current config
  2805. //
  2806. return pLoadCurrentConfig(lpdmpd,HideIfProb);
  2807. }
  2808. VOID
  2809. pShowViewNoAlloc(
  2810. IN LPDMPROP_DATA lpdmpd
  2811. )
  2812. /*++
  2813. Routine Description:
  2814. Modify the middle part of the control to indicate there is a problem (and there isn't much we can do about it)
  2815. Arguments:
  2816. hDlg = dialog handle of controls
  2817. lpdmpd = property data
  2818. Return Value:
  2819. none
  2820. --*/
  2821. {
  2822. HWND hDlg = lpdmpd->hDlg;
  2823. //
  2824. // hide all middle controls
  2825. //
  2826. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LCTEXT), SW_HIDE);
  2827. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST), SW_HIDE);
  2828. //pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_CHANGE_TEXT ), SW_HIDE);
  2829. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_USESYSSETTINGS), SW_HIDE);
  2830. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CHANGE), SW_HIDE);
  2831. //pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MAKEFORCED), SW_HIDE);
  2832. lpdmpd->dwFlags |= DMPROP_FLAG_VIEWONLYRES;
  2833. //pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_CHANGE_TEXT), SW_SHOW); // this may say why
  2834. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MAKEFORCED), SW_HIDE);
  2835. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTDEVTEXT), SW_HIDE); // no alloc, so hide this header & textbox
  2836. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTINFOLIST), SW_HIDE);
  2837. }
  2838. VOID
  2839. pShowViewNeedForced(
  2840. IN LPDMPROP_DATA lpdmpd
  2841. )
  2842. /*++
  2843. Routine Description:
  2844. Modify the middle part of the control to indicate a forced config is required
  2845. Arguments:
  2846. hDlg = dialog handle of controls
  2847. lpdmpd = property data
  2848. Return Value:
  2849. none
  2850. --*/
  2851. {
  2852. HWND hDlg = lpdmpd->hDlg;
  2853. pShowViewNoAlloc(lpdmpd);
  2854. //
  2855. // show what we need for make forced config
  2856. //
  2857. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_CHANGE_TEXT), SW_SHOW); // this may say why
  2858. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MAKEFORCED), SW_SHOW);
  2859. }
  2860. VOID
  2861. pShowViewAllEdit(
  2862. IN LPDMPROP_DATA lpdmpd
  2863. )
  2864. /*++
  2865. Routine Description:
  2866. Allow editing
  2867. Arguments:
  2868. lpdmpd = property data
  2869. Return Value:
  2870. none
  2871. --*/
  2872. {
  2873. HWND hDlg = lpdmpd->hDlg;
  2874. //
  2875. // show middle controls for editing
  2876. //
  2877. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LCTEXT), SW_SHOW);
  2878. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST), SW_SHOW);
  2879. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_CHANGE_TEXT ), SW_HIDE);
  2880. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_USESYSSETTINGS), SW_SHOW);
  2881. ShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CHANGE), SW_SHOW); // shown, but disabled
  2882. EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_CHANGE), FALSE);
  2883. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_MAKEFORCED), SW_HIDE);
  2884. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTDEVTEXT), SW_SHOW); // show conflict information space
  2885. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CONFLICTINFOLIST), SW_SHOW);
  2886. pShowUpdateEdit(lpdmpd);
  2887. }
  2888. VOID
  2889. pShowUpdateEdit(
  2890. IN LPDMPROP_DATA lpdmpd
  2891. )
  2892. /*++
  2893. Routine Description:
  2894. Allow editing
  2895. Arguments:
  2896. lpdmpd = property data
  2897. Return Value:
  2898. none
  2899. --*/
  2900. {
  2901. HWND hDlg = lpdmpd->hDlg;
  2902. //
  2903. // modify editing status - we can edit
  2904. //
  2905. lpdmpd->dwFlags &= ~DMPROP_FLAG_VIEWONLYRES;
  2906. if(lpdmpd->dwFlags & DMPROP_FLAG_FORCEDONLY) {
  2907. //
  2908. // in this case, we will never be able to use system settings
  2909. //
  2910. lpdmpd->dwFlags &= ~ DMPROP_FLAG_USESYSSETTINGS;
  2911. EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_USESYSSETTINGS), FALSE);
  2912. }
  2913. //
  2914. // indicate if it's system settings or not
  2915. //
  2916. CheckDlgButton(hDlg, IDC_DEVRES_USESYSSETTINGS,
  2917. (lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS)?TRUE:FALSE);
  2918. //
  2919. // we can change logconfiglist if it's not system settings
  2920. //
  2921. EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_LCTEXT), (lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS)?FALSE:TRUE);
  2922. EnableWindow(GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST), (lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS)?FALSE:TRUE);
  2923. //
  2924. // change "Change Settings" button
  2925. //
  2926. pCheckEnableResourceChange(lpdmpd);
  2927. }
  2928. BOOL
  2929. pLoadCurrentConfig(
  2930. IN LPDMPROP_DATA lpdmpd,
  2931. BOOL HideIfProb
  2932. )
  2933. /*++
  2934. Routine Description:
  2935. Modify the top part, to show current configuration, if any
  2936. Arguments:
  2937. hDlg = dialog handle of controls
  2938. lpdmpd = property data
  2939. Return Value:
  2940. TRUE if we're showing current config
  2941. --*/
  2942. {
  2943. TCHAR szMessage[MAX_PATH];
  2944. ULONG Problem;
  2945. ULONG Status;
  2946. HWND hDlg = lpdmpd->hDlg;
  2947. HMACHINE hMachine = pGetMachine(lpdmpd);
  2948. BOOL DoLoadConfig = FALSE;
  2949. lpdmpd->SelectedLC = 0;
  2950. lpdmpd->SelectedLCType = lpdmpd->CurrentLCType;
  2951. if (lpdmpd->CurrentLC != 0) {
  2952. DoLoadConfig = TRUE;
  2953. }
  2954. if(HideIfProb && (lpdmpd->dwFlags & DMPROP_FLAG_HASPROBLEM)) {
  2955. //
  2956. // if there's a problem and HideIfProb is TRUE, don't bother showing current config
  2957. //
  2958. DoLoadConfig = FALSE;
  2959. }
  2960. if (DoLoadConfig) {
  2961. //
  2962. // load in current configuration
  2963. //
  2964. pLoadConfig(lpdmpd,lpdmpd->CurrentLC,lpdmpd->CurrentLCType);
  2965. return TRUE;
  2966. }
  2967. //
  2968. // case where there is no suitable configuration
  2969. //
  2970. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_RESOURCES_TEXT), SW_HIDE);
  2971. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST), SW_HIDE);
  2972. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LCTEXT), SW_HIDE);
  2973. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_LOGCONFIGLIST), SW_HIDE);
  2974. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_USESYSSETTINGS), SW_HIDE);
  2975. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_CHANGE), SW_HIDE);
  2976. pShowViewNoAlloc(lpdmpd);
  2977. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_SETTINGSTATE), SW_SHOW);
  2978. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NOALLOCTEXT), SW_SHOW);
  2979. //
  2980. // explain why there is a problem
  2981. // goes into NOALLOCTEXT
  2982. //
  2983. LoadString(MyDllModuleHandle, IDS_DEVRES_NOALLOC_PROBLEM, szMessage, MAX_PATH);
  2984. //
  2985. // consider being more descriptive
  2986. //
  2987. if ((lpdmpd->lpdi->DevInst==0)
  2988. || (CM_Get_DevNode_Status_Ex(&Status, &Problem, lpdmpd->lpdi->DevInst,
  2989. 0,hMachine) != CR_SUCCESS)) {
  2990. Status = 0;
  2991. Problem = 0;
  2992. }
  2993. if ((Status & DN_HAS_PROBLEM)!=0) {
  2994. switch (Problem) {
  2995. case CM_PROB_DISABLED:
  2996. case CM_PROB_HARDWARE_DISABLED:
  2997. LoadString(MyDllModuleHandle, IDS_DEVRES_NOALLOC_DISABLED, szMessage, MAX_PATH);
  2998. break;
  2999. case CM_PROB_NORMAL_CONFLICT:
  3000. LoadString(MyDllModuleHandle, IDS_DEVRES_NORMAL_CONFLICT, szMessage, MAX_PATH);
  3001. break;
  3002. default:
  3003. break;
  3004. }
  3005. }
  3006. SetDlgItemText(hDlg, IDC_DEVRES_NOALLOCTEXT, szMessage);
  3007. return FALSE; // display in NoAlloc state
  3008. }
  3009. BOOL
  3010. pConfigHasNoAlternates(
  3011. LPDMPROP_DATA lpdmpd,
  3012. LOG_CONF testLC
  3013. )
  3014. /*++
  3015. Routine Description:
  3016. A Basic config could be restrictive "these are the set of resources to use"
  3017. This determines if the basic config passed is such a config
  3018. Arguments:
  3019. testLC = basic config to test
  3020. Return Value:
  3021. TRUE if it's a singular config
  3022. --*/
  3023. {
  3024. HMACHINE hMachine = NULL;
  3025. PRESDES_ENTRY pConfigValues = NULL;
  3026. PRESDES_ENTRY pValue = NULL;
  3027. BOOL bSuccess = TRUE;
  3028. ULONG64 ulValue = 0, ulLen = 0, ulEnd = 0;
  3029. ULONG ulFlags = 0;
  3030. PGENERIC_RESOURCE pGenRes = NULL;
  3031. hMachine = pGetMachine(lpdmpd);
  3032. pGetResDesDataList(testLC, &pConfigValues, TRUE, hMachine); // arbitratable resources
  3033. for(pValue = pConfigValues;pValue;pValue = pValue->Next) {
  3034. //
  3035. // is this a singular value?
  3036. //
  3037. pGenRes = (PGENERIC_RESOURCE)(pValue->ResDesData);
  3038. if(pGenRes->GENERIC_Header.GENERIC_Count != 1) {
  3039. //
  3040. // more than one entry - not singular
  3041. //
  3042. bSuccess = FALSE;
  3043. break;
  3044. }
  3045. pGetRangeValues(pValue->ResDesData, pValue->ResDesType, 0, &ulValue, &ulLen, &ulEnd, NULL, &ulFlags);
  3046. if (ulValue+(ulLen-1) != ulEnd) {
  3047. //
  3048. // not singular
  3049. //
  3050. bSuccess = FALSE;
  3051. break;
  3052. }
  3053. }
  3054. pDeleteResDesDataList(pConfigValues);
  3055. return bSuccess;
  3056. }
  3057. BOOL
  3058. pLoadConfig(
  3059. LPDMPROP_DATA lpdmpd,
  3060. LOG_CONF forceLC,
  3061. ULONG forceLCType
  3062. )
  3063. /*++
  3064. Routine Description:
  3065. Display a configuration
  3066. Arguments:
  3067. hDlg = dialog handle of controls
  3068. lpdmpd = property data
  3069. forceLC = LogConf to display
  3070. forceLCType = type for LogConf
  3071. Return Value:
  3072. TRUE if config loaded
  3073. --*/
  3074. {
  3075. HWND hDlg = lpdmpd->hDlg;
  3076. CONFIGRET Status = CR_SUCCESS;
  3077. HWND hWndList;
  3078. LV_ITEM lviItem;
  3079. TCHAR szTemp[MAX_PATH];
  3080. int iNewItem = 0;
  3081. ULONG ulSize,ulFlags;
  3082. ULONG64 ulValue, ulLen, ulEnd, ulAlign;
  3083. ULONG64 ulMaxMem = 0;
  3084. ULONG ulRange;
  3085. LPBYTE pData = NULL;
  3086. RES_DES ResDes;
  3087. RESOURCEID ResType;
  3088. PITEMDATA pItemData = NULL;
  3089. HMACHINE hMachine = NULL;
  3090. PDEVICE_INFO_SET pDeviceInfoSet;
  3091. BOOL RetCode = FALSE;
  3092. PRESDES_ENTRY pKnownValues = NULL;
  3093. PRESDES_ENTRY pShowValues = NULL;
  3094. PRESDES_ENTRY pShowEntry = NULL;
  3095. BOOL bFixedConfig = FALSE;
  3096. BOOL bNoMatch;
  3097. BOOL bFixed;
  3098. ULONG MatchLevel = NO_LC_MATCH;
  3099. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_SETTINGSTATE), SW_SHOW);
  3100. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NO_RESOURCES_TEXT), SW_HIDE);
  3101. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST), SW_SHOW);
  3102. pShowWindow(GetDlgItem(hDlg, IDC_DEVRES_NOALLOCTEXT), SW_HIDE);
  3103. hWndList = GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST);
  3104. SendMessage(hWndList, WM_SETREDRAW, (WPARAM)FALSE, 0);
  3105. ListView_DeleteAllItems(hWndList);
  3106. lpdmpd->dwFlags |= DMPROP_FLAG_FIXEDCONFIG; // until we determine there is at least one setting we can edit
  3107. if (forceLC == 0) {
  3108. forceLC = lpdmpd->CurrentLC;
  3109. forceLCType = lpdmpd->CurrentLCType;
  3110. }
  3111. if (forceLC == 0) {
  3112. MYASSERT(FALSE);
  3113. goto Final;
  3114. }
  3115. hMachine = pGetMachine(lpdmpd);
  3116. //
  3117. // setup values that will remain the same each time I add an item
  3118. //
  3119. lviItem.mask = LVIF_TEXT | LVIF_PARAM | LVIF_IMAGE;
  3120. lviItem.pszText = szTemp; // reuse the szTemp buffer
  3121. lviItem.iSubItem = 0;
  3122. lviItem.iImage = IDI_RESOURCE - IDI_RESOURCEFIRST;
  3123. pGetResDesDataList(forceLC, &pShowValues, TRUE, hMachine); // editable resources
  3124. if (forceLCType == BOOT_LOG_CONF || forceLCType == FORCED_LOG_CONF || forceLCType == ALLOC_LOG_CONF) {
  3125. bFixedConfig = TRUE;
  3126. if (forceLC == lpdmpd->CurrentLC && lpdmpd->MatchingLC != 0) {
  3127. //
  3128. // we're displaying CurrentLC, use flags & resdes's from matching LC where possible
  3129. //
  3130. if (pGetResDesDataList(lpdmpd->MatchingLC, &pKnownValues, TRUE, hMachine)) {
  3131. //
  3132. // match-up currentLC with some matching LC, so we can use flags/ranges from matching LC
  3133. //
  3134. MatchLevel = pMergeResDesDataLists(pShowValues,pKnownValues,NULL);
  3135. }
  3136. }
  3137. } else if (lpdmpd->CurrentLC != 0) {
  3138. //
  3139. // the config we're displaying may allow ranges of values
  3140. // we're going to try and match up what we are displaying to current config
  3141. //
  3142. if (pGetResDesDataList(lpdmpd->CurrentLC, &pKnownValues, TRUE, hMachine)) {
  3143. //
  3144. // try and use current values where possible
  3145. //
  3146. MatchLevel = pMergeResDesDataLists(pKnownValues,pShowValues,NULL);
  3147. }
  3148. }
  3149. pShowEntry = pShowValues;
  3150. while (pShowEntry) {
  3151. bNoMatch = FALSE;
  3152. bFixed = FALSE;
  3153. ResDes = (RES_DES)0;
  3154. ResType = pShowEntry->ResDesType;
  3155. ulRange = 0;
  3156. if (bFixedConfig) {
  3157. //
  3158. // we've got a current config
  3159. //
  3160. pGetHdrValues(pShowEntry->ResDesData, pShowEntry->ResDesType, &ulValue, &ulLen, &ulEnd, &ulFlags);
  3161. if((ResType ==ResType_Mem) && (ulEnd > ulMaxMem)) {
  3162. //
  3163. // base our memory display on the fixed-config only
  3164. //
  3165. ulMaxMem = ulEnd;
  3166. }
  3167. if (pShowEntry->CrossLink) {
  3168. //
  3169. // use range's res-des
  3170. //
  3171. ResDes = pShowEntry->CrossLink->ResDesHandle;
  3172. pShowEntry->CrossLink->ResDesHandle = (RES_DES)0;
  3173. //
  3174. // allow adjustment based on nearest basic config
  3175. //
  3176. pGetMatchingRange(ulValue,ulLen,pShowEntry->CrossLink->ResDesData, pShowEntry->CrossLink->ResDesType,&ulRange,&bFixed,NULL);
  3177. } else {
  3178. //
  3179. // no range res-des
  3180. //
  3181. ResDes = (RES_DES)0;
  3182. //
  3183. // indicate that this is a non-adjustable value
  3184. //
  3185. bFixed = TRUE;
  3186. }
  3187. } else {
  3188. //
  3189. // we've got resource-ranges
  3190. //
  3191. if (pShowEntry->CrossLink) {
  3192. //
  3193. // take current settings from what we merged in
  3194. //
  3195. pGetHdrValues(pShowEntry->CrossLink->ResDesData, pShowEntry->CrossLink->ResDesType, &ulValue, &ulLen, &ulEnd, &ulFlags);
  3196. } else {
  3197. //
  3198. // just take first range
  3199. //
  3200. pGetRangeValues(pShowEntry->ResDesData, pShowEntry->ResDesType, 0, &ulValue, &ulLen, &ulEnd, &ulAlign, &ulFlags);
  3201. }
  3202. //
  3203. // range check if this is a memory resource - find highest memory value
  3204. //
  3205. if(ResType ==ResType_Mem) {
  3206. ULONG64 HighVal;
  3207. ULONG64 HighLen;
  3208. if(LocateClosestValue(pShowEntry->ResDesData, pShowEntry->ResDesType,(ULONG64)(-1),ulLen, 0 ,&HighVal,&HighLen,NULL)) {
  3209. HighVal += HighLen-1;
  3210. if(HighVal > ulMaxMem) {
  3211. ulMaxMem = ulEnd;
  3212. }
  3213. }
  3214. }
  3215. pGetMatchingRange(ulValue,ulLen,pShowEntry->ResDesData, pShowEntry->ResDesType,&ulRange,&bFixed,&ulFlags);
  3216. //
  3217. // use res-des from range
  3218. //
  3219. ResDes = pShowEntry->ResDesHandle;
  3220. pShowEntry->ResDesHandle = (RES_DES)0;
  3221. if (pShowEntry->CrossLink == NULL && bFixed == FALSE) {
  3222. //
  3223. // unknown value
  3224. //
  3225. bNoMatch = TRUE;
  3226. }
  3227. }
  3228. if (ulLen>0) {
  3229. //
  3230. // Write first column text field (uses szTemp, lParam is res type)
  3231. //
  3232. LoadString(MyDllModuleHandle, IDS_RESOURCE_BASE + ResType, szTemp, MAX_PATH);
  3233. ulRange = 0;
  3234. pItemData = (PITEMDATA)MyMalloc(sizeof(ITEMDATA));
  3235. if (pItemData != NULL) {
  3236. pItemData->ResType = ResType;
  3237. pItemData->MatchingResDes = ResDes;
  3238. pItemData->RangeCount = ulRange;
  3239. pItemData->ulValue = ulValue; // selected value
  3240. pItemData->ulLen = ulLen;
  3241. pItemData->ulEnd = ulValue + ulLen - 1;
  3242. pItemData->ulFlags = ulFlags;
  3243. pItemData->bValid = !bNoMatch; // if no chosen value
  3244. pItemData->bFixed = bFixed;
  3245. }
  3246. if (bFixed == FALSE) {
  3247. //
  3248. // we have at least one editable value
  3249. //
  3250. lpdmpd->dwFlags &= ~DMPROP_FLAG_FIXEDCONFIG;
  3251. }
  3252. lviItem.iItem = iNewItem;
  3253. lviItem.lParam = (LPARAM)pItemData;
  3254. ListView_InsertItem(hWndList, &lviItem);
  3255. //
  3256. // Write second column text field (uses szTemp, lParam is res handle)
  3257. //
  3258. if (bNoMatch) {
  3259. pFormatResString(lpdmpd,szTemp, 0, 0, ResType);
  3260. } else {
  3261. pFormatResString(lpdmpd,szTemp, ulValue, ulLen, ResType);
  3262. }
  3263. ListView_SetItemText(hWndList, iNewItem, 1, szTemp);
  3264. ++iNewItem;
  3265. }
  3266. pShowEntry = pShowEntry->Next;
  3267. }
  3268. if(ulMaxMem > 0xFFFFFFFF) {
  3269. //
  3270. // an excuse to require 64-bit address range
  3271. //
  3272. lpdmpd->dwFlags |= DMPROP_FLAG_64BIT_RANGE;
  3273. }
  3274. SendMessage(hWndList, WM_SETREDRAW, (WPARAM)TRUE, 0);
  3275. RetCode = TRUE;
  3276. Final:
  3277. pDeleteResDesDataList(pKnownValues);
  3278. pDeleteResDesDataList(pShowValues);
  3279. //
  3280. // initialize listview headings here
  3281. //
  3282. ListView_SetColumnWidth(hWndList, 0, LVSCW_AUTOSIZE_USEHEADER);
  3283. ListView_SetColumnWidth(hWndList, 1, LVSCW_AUTOSIZE_USEHEADER);
  3284. //
  3285. // change "Change Settings" button
  3286. //
  3287. pCheckEnableResourceChange(lpdmpd);
  3288. return RetCode;
  3289. }
  3290. BOOL
  3291. bIsMultiFunctionChild(
  3292. PSP_DEVINFO_DATA lpdi,
  3293. HMACHINE hMachine
  3294. )
  3295. /*++
  3296. Routine Description:
  3297. Returns flag indicating if this is a child of a
  3298. multifunction device
  3299. Arguments:
  3300. Return Value:
  3301. TRUE if MF child
  3302. --*/
  3303. {
  3304. ULONG Status;
  3305. ULONG ProblemNumber;
  3306. if (lpdi->DevInst) {
  3307. if (CM_Get_DevNode_Status_Ex(&Status, &ProblemNumber,
  3308. lpdi->DevInst, 0,hMachine) == CR_SUCCESS) {
  3309. //
  3310. // If the passed in dev is not an MF child, then it is the top
  3311. // level MF_Parent
  3312. //
  3313. if (Status & DN_MF_CHILD) {
  3314. return TRUE;
  3315. } else {
  3316. return FALSE;
  3317. }
  3318. }
  3319. }
  3320. return FALSE;
  3321. }
  3322. VOID
  3323. pSelectLogConf(
  3324. LPDMPROP_DATA lpdmpd,
  3325. LOG_CONF forceLC,
  3326. ULONG forceLCType,
  3327. BOOL Always
  3328. )
  3329. /*++
  3330. Routine Description:
  3331. Selects a LogConf, showing the config in the LC control
  3332. Arguments:
  3333. Return Value:
  3334. TRUE if MF child
  3335. --*/
  3336. {
  3337. HWND hDlg = lpdmpd->hDlg;
  3338. int count;
  3339. int i;
  3340. LOG_CONF LogConf;
  3341. if (Always == FALSE && forceLC == lpdmpd->SelectedLC) {
  3342. //
  3343. // selection remains the same
  3344. //
  3345. return;
  3346. }
  3347. count = (int)SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST,CB_GETCOUNT, (WPARAM)0, (LPARAM)0);
  3348. if (count == 0) {
  3349. MYASSERT(FALSE/*shouldn't get here*/);
  3350. pLoadCurrentConfig(lpdmpd,FALSE);
  3351. return;
  3352. }
  3353. if (forceLC == 0 && lpdmpd->CurrentLC == 0) {
  3354. //
  3355. // no currentLC, so select first default
  3356. //
  3357. forceLC = (LOG_CONF)SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST,
  3358. CB_GETITEMDATA, (WPARAM)0, (LPARAM)0);
  3359. if (forceLC == (LOG_CONF)0) {
  3360. MYASSERT(FALSE/*shouldn't get here*/);
  3361. pLoadCurrentConfig(lpdmpd,FALSE);
  3362. return;
  3363. }
  3364. forceLCType = lpdmpd->ConfigListLCType;
  3365. }
  3366. for (i=0;i<count;i++) {
  3367. LogConf = (LOG_CONF)SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST,
  3368. CB_GETITEMDATA, (WPARAM)i, (LPARAM)0);
  3369. if (LogConf == forceLC) {
  3370. //
  3371. // set these first so we don't recurse around
  3372. //
  3373. lpdmpd->SelectedLC = forceLC;
  3374. lpdmpd->SelectedLCType = forceLCType;
  3375. //
  3376. // change dialog to reflect new selection
  3377. //
  3378. SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST,CB_SETCURSEL, (WPARAM)i, (LPARAM)0);
  3379. pLoadConfig(lpdmpd,forceLC,forceLCType);
  3380. pShowConflicts(lpdmpd);
  3381. return;
  3382. }
  3383. }
  3384. SendDlgItemMessage(hDlg, IDC_DEVRES_LOGCONFIGLIST,CB_SETCURSEL, (WPARAM)(-1), (LPARAM)0);
  3385. pLoadConfig(lpdmpd,forceLC,forceLCType);
  3386. pShowConflicts(lpdmpd);
  3387. }
  3388. VOID
  3389. pChangeCurrentResSetting(
  3390. IN LPDMPROP_DATA lpdmpd
  3391. )
  3392. /*++
  3393. Routine Description:
  3394. Brings up edit dialog to change currently selected resource
  3395. Arguments:
  3396. Return Value:
  3397. none
  3398. --*/
  3399. {
  3400. HWND hDlg = lpdmpd->hDlg;
  3401. RESOURCEEDITINFO rei;
  3402. HWND hList = GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST);
  3403. int iCur;
  3404. PITEMDATA pItemData = NULL;
  3405. LV_ITEM lviItem;
  3406. GENERIC_RESOURCE GenResInfo;
  3407. PDEVICE_INFO_SET pDeviceInfoSet;
  3408. BOOL changed = FALSE;
  3409. TCHAR szTemp[MAX_PATH];
  3410. pItemData = pGetResourceToChange(lpdmpd,&iCur);
  3411. if (pItemData == NULL) {
  3412. //
  3413. // we cannot edit this resource for some reason, give the user a hint
  3414. // and maybe I'll get less ear-ache "I cannot change the settings"
  3415. //
  3416. if ((lpdmpd->dwFlags & DMPROP_FLAG_VIEWONLYRES)!=0 ||
  3417. (lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS)!=0) {
  3418. //
  3419. // editing not allowed - prob double-clicked on settings
  3420. //
  3421. return;
  3422. }
  3423. if (lpdmpd->dwFlags & DMPROP_FLAG_FIXEDCONFIG) {
  3424. pWarnResSettingNotEditable(hDlg, IDS_DEVRES_NOMODIFYALL);
  3425. } else {
  3426. //
  3427. // see if user needs to select a resource
  3428. //
  3429. iCur = (int)ListView_GetNextItem(hList,-1, LVNI_SELECTED);
  3430. if (iCur == LB_ERR) {
  3431. //
  3432. // no selection
  3433. //
  3434. pWarnResSettingNotEditable(hDlg, IDS_DEVRES_NOMODIFYSELECT);
  3435. } else {
  3436. //
  3437. // resource is just not editable
  3438. //
  3439. pWarnResSettingNotEditable(hDlg, IDS_DEVRES_NOMODIFYSINGLE);
  3440. }
  3441. }
  3442. goto clean0;
  3443. }
  3444. ZeroMemory(&rei,sizeof(rei));
  3445. rei.hMachine = pGetMachine(lpdmpd);
  3446. rei.KnownLC = lpdmpd->CurrentLC;
  3447. rei.MatchingBasicLC = lpdmpd->MatchingLC;
  3448. rei.SelectedBasicLC = lpdmpd->SelectedLC;
  3449. rei.lpdi = lpdmpd->lpdi;
  3450. rei.dwPropFlags = lpdmpd->dwFlags;
  3451. rei.bShareable = FALSE;
  3452. rei.ridResType = pItemData->ResType;
  3453. rei.ResDes = pItemData->MatchingResDes;
  3454. rei.ulCurrentVal = pItemData->ulValue;
  3455. rei.ulCurrentLen = pItemData->ulLen;
  3456. rei.ulCurrentEnd = pItemData->ulEnd;
  3457. rei.ulCurrentFlags = pItemData->ulFlags;
  3458. rei.ulRangeCount = pItemData->RangeCount;
  3459. rei.pData = NULL;
  3460. if (DialogBoxParam(MyDllModuleHandle,
  3461. MAKEINTRESOURCE(IDD_EDIT_RESOURCE),
  3462. hDlg,
  3463. EditResourceDlgProc,
  3464. (LPARAM)(PRESOURCEEDITINFO)&rei) != IDOK) {
  3465. goto clean0;
  3466. }
  3467. //
  3468. // Update The Current Resource settings to Future
  3469. // Settings, and update the Conflict list.
  3470. //
  3471. pItemData->ulValue = rei.ulCurrentVal;
  3472. pItemData->ulLen = rei.ulCurrentLen;
  3473. pItemData->ulEnd = rei.ulCurrentEnd;
  3474. pItemData->ulFlags = rei.ulCurrentFlags;
  3475. pItemData->RangeCount = rei.ulRangeCount;
  3476. pItemData->bValid = TRUE; // indicate that user has explicitly changed this value
  3477. pFormatResString(lpdmpd,szTemp,
  3478. rei.ulCurrentVal,
  3479. rei.ulCurrentLen,
  3480. rei.ridResType);
  3481. ListView_SetItemText(hList, iCur, 1, szTemp);
  3482. pShowConflicts(lpdmpd);
  3483. //
  3484. // clear the flag for saving changes
  3485. //
  3486. lpdmpd->dwFlags &= ~DMPROP_FLAG_CHANGESSAVED;
  3487. PropSheet_Changed(GetParent(hDlg), hDlg);
  3488. clean0:
  3489. ;
  3490. }
  3491. VOID
  3492. pShowConflicts(
  3493. IN LPDMPROP_DATA lpdmpd
  3494. )
  3495. /*++
  3496. Routine Description:
  3497. Selects a LogConf, showing the config in the LC control
  3498. Arguments:
  3499. Return Value:
  3500. TRUE if MF child
  3501. --*/
  3502. {
  3503. HWND hDlg = lpdmpd->hDlg;
  3504. CONFIGRET Status = CR_SUCCESS;
  3505. LPVOID vaArray[4];
  3506. TCHAR szTemp[MAX_PATH+4], szBuffer[MAX_PATH+16], szSetting[MAX_PATH];
  3507. TCHAR szFormat[MAX_PATH], szItemFormat[MAX_PATH];
  3508. TCHAR szUnavailable[MAX_PATH];
  3509. LPTSTR pszConflictList = NULL, pszConflictList2 = NULL;
  3510. ULONG ulSize = 0, ulLength, ulBufferLen, ulNewLength;
  3511. ULONG ulStartOffset = 0;
  3512. int Count = 0, i = 0;
  3513. PITEMDATA pItemData = NULL;
  3514. LPBYTE pResourceData = NULL;
  3515. HWND hwndResList = GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST);
  3516. HMACHINE hMachine;
  3517. ULONG ConflictCount = 0;
  3518. ULONG ConflictIndex = 0;
  3519. CONFLICT_LIST ConflictList = 0;
  3520. PDEVICE_INFO_SET pDeviceInfoSet;
  3521. CONFLICT_DETAILS ConflictDetails;
  3522. BOOL ReservedResource = FALSE;
  3523. BOOL AnyReportedResources = FALSE;
  3524. BOOL AnyBadResources = FALSE;
  3525. PCONFLICT_EXCEPTIONS pConflictExceptions = NULL;
  3526. //
  3527. // number of resources listed
  3528. //
  3529. Count = ListView_GetItemCount(hwndResList);
  3530. if (Count <= 0) {
  3531. goto Clean0;
  3532. }
  3533. //
  3534. // initial buffer that holds the strings
  3535. // with all the conflict info in them
  3536. //
  3537. ulBufferLen = 2048;
  3538. ulLength = 0;
  3539. pszConflictList = MyMalloc(ulBufferLen * sizeof(TCHAR));
  3540. if (pszConflictList == NULL) {
  3541. goto Clean0;
  3542. }
  3543. pszConflictList[0] = 0;
  3544. //
  3545. // obtain machine
  3546. //
  3547. if(!(pDeviceInfoSet = AccessDeviceInfoSet(lpdmpd->hDevInfo))) {
  3548. goto Clean0;
  3549. }
  3550. hMachine = pDeviceInfoSet->hMachine;
  3551. UnlockDeviceInfoSet (pDeviceInfoSet);
  3552. //
  3553. // do these once - these format strings use %1!s! type formats (FormatMessage)
  3554. //
  3555. LoadString(MyDllModuleHandle, IDS_CONFLICT_FMT, szFormat, MAX_PATH);
  3556. LoadString(MyDllModuleHandle, IDS_CONFLICT_UNAVAILABLE, szUnavailable, MAX_PATH);
  3557. //
  3558. // for every listed resource
  3559. //
  3560. for (i = 0; i < Count; i++) {
  3561. ConflictList = 0;
  3562. ConflictCount = 0;
  3563. //
  3564. // get the resource we're about to test
  3565. //
  3566. pItemData = (PITEMDATA)pGetListViewItemData(hwndResList, i, 0);
  3567. if (pItemData == NULL || pItemData->bValid == FALSE) {
  3568. //
  3569. // for whatever reason, we don't want to show conflict information on this resource
  3570. //
  3571. ListView_SetItemState(hwndResList, i,
  3572. INDEXTOOVERLAYMASK(0),
  3573. LVIS_OVERLAYMASK);
  3574. goto NextResource;
  3575. }
  3576. //
  3577. // this is set to indicate conflict not reported, but is reserved
  3578. //
  3579. ReservedResource = FALSE;
  3580. //
  3581. // need resource-data for determining conflict
  3582. //
  3583. if (MakeResourceData(&pResourceData, &ulSize,
  3584. pItemData->ResType,
  3585. pItemData->ulValue,
  3586. pItemData->ulLen,
  3587. pItemData->ulFlags)) {
  3588. Status = CM_Query_Resource_Conflict_List(&ConflictList,
  3589. lpdmpd->lpdi->DevInst,
  3590. pItemData->ResType,
  3591. pResourceData,
  3592. ulSize,
  3593. DEVRES_WIDTH_FLAGS,
  3594. hMachine);
  3595. if (Status != CR_SUCCESS) {
  3596. //
  3597. // on the unlikely event of an error, remember an error occurred
  3598. //
  3599. ConflictList = 0;
  3600. ConflictCount = 0;
  3601. AnyBadResources = TRUE;
  3602. } else {
  3603. //
  3604. // find out how many things conflicted
  3605. //
  3606. Status = CM_Get_Resource_Conflict_Count(ConflictList,&ConflictCount);
  3607. if (Status != CR_SUCCESS) {
  3608. MYASSERT(Status == CR_SUCCESS);
  3609. ConflictCount = 0;
  3610. AnyBadResources = TRUE;
  3611. }
  3612. }
  3613. if(ConflictCount && (lpdmpd->dwFlags & DMPROP_FLAG_SINGLE_CONFIG) && !(lpdmpd->dwFlags & DMPROP_FLAG_HASPROBLEM)) {
  3614. //
  3615. // NTRAID#166214-2000/08/19-JamieHun Conflict Supression Hack
  3616. //
  3617. // rules are
  3618. // (1) device doesn't have a problem
  3619. // (2) device can only have one configuration (ie, there's no basic config, or the basic config is singular)
  3620. // (3) it has a ResourcePickerExceptions string, and that string indicates that the exception is allowed for the specific conflict
  3621. if(pConflictExceptions==NULL) {
  3622. pConflictExceptions = pLoadConflictExceptions(lpdmpd);
  3623. }
  3624. if (pConflictExceptions) {
  3625. BOOL muted = TRUE;
  3626. //
  3627. // count from 0 (first conflict) through to ConflictCount (excl)
  3628. //
  3629. for(ConflictIndex = 0; ConflictIndex < ConflictCount; ConflictIndex ++) {
  3630. //
  3631. // obtain details for this conflict
  3632. //
  3633. ZeroMemory(&ConflictDetails,sizeof(ConflictDetails));
  3634. ConflictDetails.CD_ulSize = sizeof(ConflictDetails);
  3635. ConflictDetails.CD_ulMask = CM_CDMASK_DEVINST | CM_CDMASK_DESCRIPTION | CM_CDMASK_FLAGS;
  3636. Status = CM_Get_Resource_Conflict_Details(ConflictList,ConflictIndex,&ConflictDetails);
  3637. if (Status == CR_SUCCESS) {
  3638. if (!pIsConflictException(lpdmpd,pConflictExceptions,ConflictDetails.CD_dnDevInst,ConflictDetails.CD_szDescription,pItemData->ResType,pItemData->ulValue,pItemData->ulLen)) {
  3639. muted = FALSE;
  3640. break;
  3641. }
  3642. }
  3643. }
  3644. if(muted) {
  3645. ConflictCount = 0;
  3646. }
  3647. }
  3648. }
  3649. if (ConflictCount || ReservedResource) {
  3650. ulStartOffset = ulLength; // record start in case we decide to backtrack
  3651. AnyReportedResources = TRUE; // say we reported at least one problem
  3652. TreatAsReserved:
  3653. ulLength = ulStartOffset;
  3654. pszConflictList[ulLength] = 0;
  3655. //
  3656. // we're going to mark the resource as a problem
  3657. //
  3658. ListView_GetItemText(hwndResList, i, 1, szSetting, MAX_PATH);
  3659. switch (pItemData->ResType) {
  3660. case ResType_Mem:
  3661. LoadString(MyDllModuleHandle, IDS_MEMORY_FULL, szBuffer, MAX_PATH);
  3662. break;
  3663. case ResType_IO:
  3664. LoadString(MyDllModuleHandle, IDS_IO_FULL, szBuffer, MAX_PATH);
  3665. break;
  3666. case ResType_DMA:
  3667. LoadString(MyDllModuleHandle, IDS_DMA_FULL, szBuffer, MAX_PATH);
  3668. break;
  3669. case ResType_IRQ:
  3670. LoadString(MyDllModuleHandle, IDS_IRQ_FULL, szBuffer, MAX_PATH);
  3671. break;
  3672. default:
  3673. MYASSERT(FALSE);
  3674. }
  3675. if ( ReservedResource == FALSE) {
  3676. //
  3677. // count from 0 (header) 1 (first conflict) through to ConflictCount
  3678. //
  3679. for(ConflictIndex = 0; ConflictIndex <= ConflictCount; ConflictIndex ++) {
  3680. if (ConflictIndex == 0) {
  3681. //
  3682. // first pass through, do header message
  3683. //
  3684. vaArray[0] = szBuffer;
  3685. vaArray[1] = szSetting;
  3686. FormatMessage(FORMAT_MESSAGE_FROM_STRING | FORMAT_MESSAGE_ARGUMENT_ARRAY ,
  3687. szFormat,
  3688. 0,0,
  3689. szTemp,MAX_PATH,
  3690. (va_list*)vaArray); // FORMAT_MESSAGE_ARGUMENT_ARRAY
  3691. } else {
  3692. //
  3693. // obtain details for this conflict
  3694. //
  3695. ZeroMemory(&ConflictDetails,sizeof(ConflictDetails));
  3696. ConflictDetails.CD_ulSize = sizeof(ConflictDetails);
  3697. ConflictDetails.CD_ulMask = CM_CDMASK_DEVINST | CM_CDMASK_DESCRIPTION | CM_CDMASK_FLAGS;
  3698. Status = CM_Get_Resource_Conflict_Details(ConflictList,ConflictIndex-1,&ConflictDetails);
  3699. if (Status == CR_SUCCESS) {
  3700. if ((ConflictDetails.CD_ulFlags & CM_CDFLAGS_RESERVED) != 0) {
  3701. //
  3702. // treat as reserved - backtrack
  3703. //
  3704. ReservedResource = TRUE;
  3705. goto TreatAsReserved;
  3706. } else {
  3707. if (ConflictDetails.CD_szDescription[0] == 0) {
  3708. //
  3709. // treat as reserved - backtrack
  3710. //
  3711. ReservedResource = TRUE;
  3712. goto TreatAsReserved;
  3713. }
  3714. wsprintf(szBuffer,TEXT(" %s\r\n"),ConflictDetails.CD_szDescription);
  3715. }
  3716. } else {
  3717. //
  3718. // treat as reserved
  3719. //
  3720. ReservedResource = TRUE;
  3721. goto TreatAsReserved;
  3722. }
  3723. lstrcpyn(szTemp,szBuffer,MAX_PATH);
  3724. }
  3725. ulNewLength = ulLength + lstrlen(szTemp); // excluding terminating NUL
  3726. if ((ulNewLength+1) < ulBufferLen) {
  3727. //
  3728. // need to allocate more space - we'll double it and add some more every time
  3729. //
  3730. pszConflictList2 = MyRealloc(pszConflictList,(ulBufferLen+ulNewLength+1) * sizeof(TCHAR));
  3731. if (pszConflictList2 != NULL) {
  3732. //
  3733. // succeeded in resizing buffer
  3734. //
  3735. pszConflictList = pszConflictList2;
  3736. ulBufferLen = ulBufferLen+ulNewLength+1;
  3737. }
  3738. }
  3739. if ((ulNewLength+1) < ulBufferLen) {
  3740. lstrcpy(pszConflictList + ulLength , szTemp);
  3741. ulLength = ulNewLength;
  3742. }
  3743. }
  3744. } else {
  3745. //
  3746. // there is some other problem with resource
  3747. //
  3748. vaArray[0] = szBuffer;
  3749. vaArray[1] = szSetting;
  3750. FormatMessage(FORMAT_MESSAGE_FROM_STRING | FORMAT_MESSAGE_ARGUMENT_ARRAY ,
  3751. szUnavailable,
  3752. 0,0,
  3753. szTemp,MAX_PATH,
  3754. (va_list*)vaArray); // FORMAT_MESSAGE_ARGUMENT_ARRAY
  3755. ulNewLength = ulLength + lstrlen(szTemp); // excluding terminating NUL
  3756. if ((ulNewLength+1) < ulBufferLen) {
  3757. //
  3758. // need to allocate more space - we'll double it and add some more every time
  3759. //
  3760. pszConflictList2 = MyRealloc(pszConflictList,(ulBufferLen+ulNewLength+1) * sizeof(TCHAR));
  3761. if (pszConflictList2 != NULL) {
  3762. //
  3763. // succeeded in resizing buffer
  3764. //
  3765. pszConflictList = pszConflictList2;
  3766. ulBufferLen = ulBufferLen+ulNewLength+1;
  3767. }
  3768. }
  3769. if ((ulNewLength+1) < ulBufferLen) {
  3770. lstrcpy(pszConflictList + ulLength , szTemp);
  3771. ulLength = ulNewLength;
  3772. }
  3773. }
  3774. //
  3775. // Set the Conflict Overlay for this resource.
  3776. //
  3777. ListView_SetItemState(hwndResList, i,
  3778. INDEXTOOVERLAYMASK(IDI_CONFLICT - IDI_RESOURCEOVERLAYFIRST + 1),
  3779. LVIS_OVERLAYMASK);
  3780. } else {
  3781. //
  3782. // resource is (aparently) working fine
  3783. //
  3784. ListView_SetItemState(hwndResList, i,
  3785. INDEXTOOVERLAYMASK(0),
  3786. LVIS_OVERLAYMASK);
  3787. }
  3788. if (ConflictList) {
  3789. CM_Free_Resource_Conflict_Handle(ConflictList);
  3790. }
  3791. if (pResourceData != NULL) {
  3792. MyFree(pResourceData);
  3793. }
  3794. } else {
  3795. //
  3796. // couldn't make the resource descriptor
  3797. AnyBadResources = TRUE;
  3798. }
  3799. NextResource:
  3800. ;
  3801. }
  3802. Clean0:
  3803. ;
  3804. //
  3805. // If there were any conflicts, put the list in the multiline edit box.
  3806. //
  3807. if (AnyReportedResources) {
  3808. SetDlgItemText(hDlg, IDC_DEVRES_CONFLICTINFOLIST, pszConflictList);
  3809. } else if (AnyBadResources) {
  3810. //
  3811. // this would most likely occur on
  3812. // (1) running this on 95/98 (shouldn't happen)
  3813. // (2) using new setupapi on old cfgmgr32
  3814. //
  3815. LoadString(MyDllModuleHandle, IDS_CONFLICT_GENERALERROR, szBuffer, MAX_PATH);
  3816. SetDlgItemText(hDlg, IDC_DEVRES_CONFLICTINFOLIST, szBuffer);
  3817. } else {
  3818. LoadString(MyDllModuleHandle, IDS_DEVRES_NOCONFLICTDEVS, szBuffer, MAX_PATH);
  3819. SetDlgItemText(hDlg, IDC_DEVRES_CONFLICTINFOLIST, szBuffer);
  3820. }
  3821. if(pszConflictList != NULL) {
  3822. MyFree(pszConflictList);
  3823. }
  3824. if (pConflictExceptions != NULL) {
  3825. pFreeConflictExceptions(pConflictExceptions);
  3826. }
  3827. return;
  3828. }
  3829. int
  3830. pOkToSave(
  3831. IN LPDMPROP_DATA lpdmpd
  3832. )
  3833. /*++
  3834. Routine Description:
  3835. Check to see if there's something the user hasn't done
  3836. Arguments:
  3837. Return Value:
  3838. IDYES = save settings
  3839. IDNO = don't save settings
  3840. IDCANCEL = don't exit
  3841. --*/
  3842. {
  3843. HWND hDlg = lpdmpd->hDlg;
  3844. HWND hList = GetDlgItem(hDlg, IDC_DEVRES_SETTINGSLIST);
  3845. int iCur;
  3846. int nRes;
  3847. PITEMDATA pItemData;
  3848. if (lpdmpd->dwFlags & DMPROP_FLAG_NO_RESOURCES) {
  3849. //
  3850. // no changes - because there are no resources
  3851. //
  3852. return IDNO;
  3853. }
  3854. if (lpdmpd->dwFlags & DMPROP_FLAG_CHANGESSAVED) {
  3855. //
  3856. // no changes
  3857. //
  3858. return IDNO;
  3859. }
  3860. if (lpdmpd->dwFlags & DMPROP_FLAG_USESYSSETTINGS) {
  3861. //
  3862. // always ok to "use sys settings"
  3863. //
  3864. return IDYES;
  3865. }
  3866. //
  3867. // user is forcing a config - let's see if all settings are valid
  3868. //
  3869. //
  3870. // The resource descriptors are out-of-order. Maintain the original ordering.
  3871. //
  3872. // First, build up a linked list of the data in the listview resource items.
  3873. //
  3874. iCur = (int)ListView_GetNextItem(hList, -1, LVNI_ALL);
  3875. while (iCur >= 0) {
  3876. pItemData = (PITEMDATA)pGetListViewItemData(hList, iCur, 0);
  3877. if (pItemData) {
  3878. if (pItemData->bValid == FALSE) {
  3879. //
  3880. // we've got an invalid entry - can't save
  3881. //
  3882. nRes = pWarnNoSave(hDlg,IDS_FORCEDCONFIG_PARTIAL);
  3883. if (nRes != IDOK) {
  3884. return IDCANCEL;
  3885. }
  3886. return IDNO;
  3887. }
  3888. }
  3889. iCur = (int)ListView_GetNextItem(hList, iCur, LVNI_ALL);
  3890. }
  3891. //
  3892. // everything checks out
  3893. //
  3894. return IDYES;
  3895. }