Source code of Windows XP (NT5)
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.

1339 lines
39 KiB

  1. #include "pch.h"
  2. #pragma hdrstop
  3. #include <ncxbase.h>
  4. #include "hwres.h"
  5. #include "kkcwinf.h"
  6. #include "ncreg.h"
  7. extern const WCHAR c_szAfIoAddr[];
  8. extern const WCHAR c_szAfIrq[];
  9. extern const WCHAR c_szAfDma[];
  10. extern const WCHAR c_szAfMem[];
  11. extern const WCHAR c_szBusType[];
  12. //+---------------------------------------------------------------------------
  13. //
  14. // Member: CHwRes::CHwRes
  15. //
  16. // Purpose: Class constructor. (Variable initialization)
  17. //
  18. // Author: t-nabilr 07 Apr 1997
  19. //
  20. // Notes: Doesn't do much. Just some member variable initialization.
  21. // Bulk of the init work is done in HrInit().
  22. //
  23. CHwRes::CHwRes()
  24. : m_DevNode(NULL),
  25. m_pnccItem(NULL),
  26. m_fInitialized(FALSE),
  27. m_fHrInitCalled(FALSE),
  28. m_fDirty(FALSE)
  29. {
  30. m_vAfDma.InitNotPresent(VALUETYPE_DWORD);
  31. m_vAfIrq.InitNotPresent(VALUETYPE_DWORD);
  32. m_vAfMem.InitNotPresent(VALUETYPE_DWORD);
  33. m_vAfMemEnd.InitNotPresent(VALUETYPE_DWORD);
  34. m_vAfIo.InitNotPresent(VALUETYPE_DWORD);
  35. m_vAfIoEnd.InitNotPresent(VALUETYPE_DWORD);
  36. }
  37. //+---------------------------------------------------------------------------
  38. //
  39. // Member: CHwRes::HrInit
  40. //
  41. // Purpose: Initialize class
  42. //
  43. // Arguments:
  44. // hInst [in] Handle to our instance.
  45. // pnccItem [in] Our INetCfgComponent.
  46. //
  47. // Returns: S_OK - init successfull;
  48. // HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) - no config
  49. // info for device; E_FAIL - other failure (device not found,etc)
  50. //
  51. // Author: t-nabilr 07 Apr 1997
  52. //
  53. // Notes: - should initialize all non-ui stuff
  54. // - initialize configuration and resource lists.
  55. //
  56. HRESULT CHwRes::HrInit(const DEVNODE& devnode)
  57. {
  58. HRESULT hr = E_FAIL;
  59. m_fHrInitCalled = TRUE;
  60. // use the config manager to get devnode
  61. // Since the devnode might have a problem, e.g. resources not set
  62. // correctly, we need to retrieve possible phantoms
  63. //
  64. CONFIGRET crRet = ERROR_SUCCESS;
  65. HKEY hkey;
  66. // We only do work on Isa adapters so we need to get the bustype
  67. // value from the driver (software) key
  68. crRet = CM_Open_DevNode_Key(devnode, KEY_READ, 0,
  69. RegDisposition_OpenExisting, &hkey, CM_REGISTRY_SOFTWARE);
  70. if (CR_SUCCESS == crRet)
  71. {
  72. // Get BusType
  73. ULONG ulBusType;
  74. hr = HrRegQueryStringAsUlong(hkey, c_szBusType, c_nBase10,
  75. &ulBusType);
  76. RegCloseKey(hkey);
  77. // If Isa, than we can continue
  78. if (SUCCEEDED(hr) && (Isa == ulBusType))
  79. {
  80. m_DevNode = devnode;
  81. // read configuration current config info
  82. hr = HrInitConfigList();
  83. }
  84. else
  85. {
  86. hr = S_FALSE;
  87. }
  88. }
  89. if (S_OK == hr)
  90. {
  91. m_fInitialized = TRUE;
  92. }
  93. TraceError("CHwRes::HrInit",
  94. (HRESULT_FROM_WIN32(ERROR_FILE_EXISTS) == hr ||
  95. S_FALSE == hr) ? S_OK : hr);
  96. return hr;
  97. }
  98. //+---------------------------------------------------------------------------
  99. //
  100. // Member: CHwRes::GetNextElement
  101. //
  102. // Purpose: Gets the next (or prev) element in a resource list
  103. //
  104. // Arguments:
  105. // pResource [in] resource list to traverse.
  106. // ppeList [out] the "next" element is returned.
  107. // fNext [in] TRUE - moving fwd in list; FALSE - move bkwrds.
  108. //
  109. // Author: t-nabilr 07 Apr 1997
  110. //
  111. // Notes:
  112. //
  113. VOID CHwRes::GetNextElement(PRESOURCE pResource, PVOID *ppeList, BOOL fNext)
  114. {
  115. UINT size = 0;
  116. AssertSz(m_fInitialized,
  117. "GetNextElement called before CHwRes is HrInit'ed");
  118. Assert(pResource != NULL);
  119. // get the vector size (dependant on resource type)
  120. switch (pResource->ResourceType) {
  121. case ResType_IRQ:
  122. size = pResource->pIRQList->size();
  123. break;
  124. case ResType_DMA:
  125. size = pResource->pDMAList->size();
  126. break;
  127. case ResType_IO:
  128. size = pResource->pIOList->size();
  129. break;
  130. case ResType_Mem:
  131. size = pResource->pMEMList->size();
  132. break;
  133. default:
  134. Assert(FALSE);
  135. break;
  136. }
  137. // increment/decrement current position within vector
  138. if (fNext)
  139. {
  140. pResource->pos++;
  141. }
  142. else
  143. {
  144. pResource->pos--;
  145. }
  146. // Check for wrapping
  147. if ((int)(pResource->pos) < 0)
  148. {
  149. Assert(pResource->pos == -1);
  150. Assert(!fNext); // we're going backwards
  151. pResource->pos = size-1;
  152. }
  153. else if (pResource->pos >= size)
  154. {
  155. Assert(pResource->pos == size);
  156. Assert(fNext);
  157. pResource->pos = 0;
  158. }
  159. // return the current vector element (dependant on res type)
  160. // REVIEW: do we ever use the element that's gathered below?
  161. switch (pResource->ResourceType) {
  162. case ResType_IRQ:
  163. *ppeList = (*pResource->pIRQList)[pResource->pos];
  164. break;
  165. case ResType_DMA:
  166. *ppeList = (*pResource->pDMAList)[pResource->pos];
  167. break;
  168. case ResType_IO:
  169. *ppeList = (*pResource->pIOList)[pResource->pos];
  170. break;
  171. case ResType_Mem:
  172. *ppeList = (*pResource->pMEMList)[pResource->pos];
  173. break;
  174. default:
  175. Assert(FALSE);
  176. break;
  177. }
  178. }
  179. //+---------------------------------------------------------------------------
  180. //
  181. // Member: CHwRes::HrInitConfigList
  182. //
  183. // Purpose: Initialize m_ConfigList (the internal vector of
  184. // configurations.)
  185. //
  186. // Returns: S_OK - init successful;
  187. // HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) if no device configs
  188. // found; E_FAIL otherwise (invalid device, etc.)
  189. //
  190. // Author: t-nabilr 07 Apr 1997
  191. //
  192. // Notes: "Lists" are actually implemented as STL vectors.
  193. //
  194. HRESULT CHwRes::HrInitConfigList() {
  195. HRESULT hr = S_OK;
  196. PCONFIGURATION pConfiguration;
  197. CONFIGRET ConfigRet;
  198. LOG_CONF lcCurrent, lcNext;
  199. UINT iBasicConfig;
  200. BOOL fFoundConfig = FALSE;
  201. Assert(NULL != m_DevNode);
  202. // erase all elements
  203. m_ConfigList.erase(m_ConfigList.begin(), m_ConfigList.end());
  204. // Boot Configuration
  205. if (CM_Get_First_Log_Conf(&lcCurrent, m_DevNode, BOOT_LOG_CONF)
  206. == CR_SUCCESS)
  207. {
  208. TraceTag(ttidNetComm, "Boot config already exists");
  209. hr = HRESULT_FROM_WIN32(ERROR_FILE_EXISTS);
  210. }
  211. if (SUCCEEDED(hr))
  212. {
  213. // Basic Configurations (may be more than one)
  214. iBasicConfig = 0;
  215. ConfigRet = CM_Get_First_Log_Conf(&lcCurrent, m_DevNode, BASIC_LOG_CONF);
  216. #ifdef ENABLETRACE
  217. if (CR_SUCCESS != ConfigRet)
  218. {
  219. TraceTag(ttidNetComm, "CM_Get_First_Log_conf returned %lX", ConfigRet);
  220. }
  221. #endif // ENABLETRACE
  222. while (CR_SUCCESS == ConfigRet)
  223. {
  224. pConfiguration = new CONFIGURATION;
  225. if (pConfiguration == NULL)
  226. {
  227. TraceError("pConfiguration == NULL", E_FAIL);
  228. goto error;
  229. }
  230. pConfiguration->LogConf = lcCurrent;
  231. pConfiguration->fBoot = FALSE;
  232. pConfiguration->fAlloc = FALSE;
  233. if (!FInitResourceList(pConfiguration))
  234. {
  235. hr = E_FAIL;
  236. TraceError("CHwRes::FInitResourceList",hr);
  237. goto error;
  238. }
  239. m_ConfigList.push_back(pConfiguration);
  240. // Move on to the next basic config
  241. iBasicConfig++;
  242. ConfigRet = CM_Get_Next_Log_Conf(&lcNext, lcCurrent, 0);
  243. lcCurrent = lcNext;
  244. fFoundConfig = TRUE;
  245. }
  246. if (!fFoundConfig)
  247. {
  248. TraceTag(ttidNetComm, "No basic configs");
  249. // if no config entries found, return ERROR_FILE_NOT_FOUND.
  250. hr = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
  251. }
  252. }
  253. error:
  254. // ERROR_FILE_NOT_FOUND is an okay error message.
  255. TraceError("CHwRes::HrInitConfigList",
  256. (HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND) == hr ||
  257. HRESULT_FROM_WIN32(ERROR_FILE_EXISTS) == hr ) ? S_OK : hr);
  258. return hr;
  259. }
  260. //+---------------------------------------------------------------------------
  261. //
  262. // Member: CHwRes::FInitResourceList
  263. //
  264. // Purpose: Initializes the resource list for a given logical config.
  265. //
  266. // Arguments:
  267. // pConfiguration [in] configuration who's resource list is to
  268. // be initialized.
  269. //
  270. // Returns: TRUE if init sucessful; FALSE otherwise.
  271. //
  272. // Author: t-nabilr 07 Apr 1997
  273. //
  274. // Notes: Requires pConfiguration->LogConf to be initialized.
  275. //
  276. BOOL CHwRes::FInitResourceList(PCONFIGURATION pConfiguration)
  277. {
  278. RES_DES rdCurrent;
  279. RES_DES rdNext;
  280. RESOURCEID ResType;
  281. UINT iResource;
  282. #define RESOURCE_BUFFER_SIZE 4096
  283. BYTE Buffer[RESOURCE_BUFFER_SIZE];
  284. ULONG ulDataSize;
  285. RESOURCE * pRes;
  286. CONFIGRET cr;
  287. Assert(pConfiguration->LogConf);
  288. rdCurrent = (RES_DES) pConfiguration->LogConf;
  289. iResource = 0;
  290. while ((cr = CM_Get_Next_Res_Des(&rdNext, rdCurrent,
  291. ResType_All, &ResType, 0)) == CR_SUCCESS)
  292. {
  293. rdCurrent = rdNext;
  294. // Only process this resource if the ignore bit is not set
  295. if (ResType_Ignored_Bit != ResType)
  296. {
  297. pRes = &(pConfiguration->aResource[iResource]);
  298. pRes->ResDes = rdCurrent;
  299. pRes->ResourceType = ResType;
  300. cr = CM_Get_Res_Des_Data_Size(&ulDataSize, rdCurrent, 0);
  301. if (CR_SUCCESS != cr)
  302. {
  303. TraceTag (ttidDefault, "CM_Get_Res_Des_Data_Size returned 0x%08x", cr);
  304. goto error;
  305. }
  306. AssertSz (ulDataSize, "CM_Get_Res_Des_Data_Size returned 0!");
  307. AssertSz (ulDataSize <= sizeof(Buffer), "CHwRes::FInitResourceList: buffer is too small.");;
  308. cr = CM_Get_Res_Des_Data(rdCurrent, Buffer, sizeof(Buffer), 0);
  309. if (CR_SUCCESS != cr)
  310. {
  311. TraceTag (ttidDefault, "CM_Get_Res_Des_Data returned 0x%08x", cr);
  312. goto error;
  313. }
  314. // Depending on the ResType, we have to initialize our resource list.
  315. switch (ResType)
  316. {
  317. case ResType_Mem:
  318. InitMEMList(&(pRes->pMEMList), (PMEM_RESOURCE)Buffer);
  319. break;
  320. case ResType_IO:
  321. InitIOList(&(pRes->pIOList), (PIO_RESOURCE)Buffer);
  322. break;
  323. case ResType_DMA:
  324. InitDMAList(&(pRes->pDMAList), (PDMA_RESOURCE)Buffer);
  325. break;
  326. case ResType_IRQ:
  327. InitIRQList(&(pRes->pIRQList), (PIRQ_RESOURCE)Buffer);
  328. break;
  329. default:
  330. AssertSz (ResType_None != ResType, "ResType_None hit caught in CHwRes::FInitResourceList.");
  331. break;
  332. }
  333. // set the list position to the first element;
  334. // applied_pos will get copied to pos when dialog box is created.
  335. pRes->applied_pos = 0;
  336. iResource++;
  337. pConfiguration->cResource = iResource;
  338. if (iResource >= c_nMaxResCtls)
  339. {
  340. break; // we're done.
  341. }
  342. }
  343. } //while
  344. if ((CR_SUCCESS != cr) && (CR_NO_MORE_RES_DES != cr))
  345. {
  346. TraceTag (ttidDefault, "CM_Get_Next_Res_Des returned 0x%08x", cr);
  347. goto error;
  348. }
  349. return TRUE;
  350. error:
  351. return FALSE;
  352. }
  353. //+---------------------------------------------------------------------------
  354. //
  355. // Member: CHwRes::InitIRQList
  356. //
  357. // Purpose: Initialize an IRQ resource vector given a config manager
  358. // resource structure.
  359. //
  360. // Arguments:
  361. // ppIRQList [out] returns IRQ_LIST that will be created.
  362. // pIRQResource [in] IRQ_RESOURCE structure from config manager.
  363. //
  364. // Author: t-nabilr 07 Apr 1997
  365. //
  366. // Notes:
  367. //
  368. VOID CHwRes::InitIRQList(PIRQ_LIST* ppIRQList, PIRQ_RESOURCE pIRQResource) {
  369. PIRQ_DES pIRQHeader;
  370. PIRQ_RANGE pIRQData;
  371. PIRQ_LIST_ELEMENT pIRQListElement;
  372. pIRQHeader = &(pIRQResource->IRQ_Header);
  373. pIRQData = &(pIRQResource->IRQ_Data[0]);
  374. // Create a new list.
  375. *ppIRQList = new IRQ_LIST;
  376. if (*ppIRQList == NULL)
  377. {
  378. return;
  379. }
  380. ULONG iData;
  381. ULONG iIRQ;
  382. for (iData = 0; iData < pIRQHeader->IRQD_Count; iData++)
  383. {
  384. for (iIRQ = pIRQData[iData].IRQR_Min;
  385. iIRQ <= pIRQData[iData].IRQR_Max;
  386. iIRQ++)
  387. {
  388. // For each IRQ that falls within the given range,
  389. // create new IRQ List Element, populate its fields and insert
  390. // it into the m_IRQList.
  391. pIRQListElement = new IRQ_LIST_ELEMENT;
  392. if (pIRQListElement == NULL)
  393. {
  394. continue;
  395. }
  396. pIRQListElement->dwIRQ = iIRQ;
  397. pIRQListElement->fConflict = FALSE;
  398. pIRQListElement->fAllocated = FALSE;
  399. (*ppIRQList)->push_back(pIRQListElement);
  400. }
  401. }
  402. }
  403. //+---------------------------------------------------------------------------
  404. //
  405. // Member: CHwRes::InitDMAList
  406. //
  407. // Purpose: Initialize a DMA resource vector given a config manager
  408. // resource structure.
  409. //
  410. // Arguments:
  411. // ppDMAList [out] returns DMA_LIST that will be created.
  412. // pDMAResource [in] DMA_RESOURCE structure from config manager.
  413. //
  414. // Author: t-nabilr 07 Apr 1997
  415. //
  416. // Notes:
  417. //
  418. VOID CHwRes::InitDMAList(PDMA_LIST* ppDMAList, PDMA_RESOURCE pDMAResource) {
  419. PDMA_DES pDMAHeader;
  420. PDMA_RANGE pDMAData;
  421. PDMA_LIST_ELEMENT peDMAList;
  422. pDMAHeader = &(pDMAResource->DMA_Header);
  423. pDMAData = &(pDMAResource->DMA_Data[0]);
  424. // Create a new list.
  425. *ppDMAList = new DMA_LIST;
  426. if (*ppDMAList == NULL)
  427. {
  428. return;
  429. }
  430. ULONG iData; // index of DMA_Range structure we're looking at.
  431. ULONG iDMA; // current DMA number we're adding to the list.
  432. // Go through all the DMA_Range structures, and all DMAs in the
  433. // specified range to the list.
  434. for (iData = 0; iData < pDMAHeader->DD_Count; iData++)
  435. {
  436. for (iDMA = pDMAData[iData].DR_Min;
  437. iDMA <= pDMAData[iData].DR_Max;
  438. iDMA++)
  439. {
  440. // For each DMA that falls within the given range,
  441. // create new DMA List Element, populate its fields and insert
  442. // it into the DMAList.
  443. peDMAList = new DMA_LIST_ELEMENT;
  444. if (peDMAList == NULL)
  445. {
  446. continue;
  447. }
  448. peDMAList->dwDMA = iDMA;
  449. peDMAList->fConflict = FALSE;
  450. peDMAList->fAllocated = FALSE;
  451. (*ppDMAList)->push_back(peDMAList);
  452. }
  453. }
  454. }
  455. //+---------------------------------------------------------------------------
  456. //
  457. // Member: CHwRes::InitMEMList
  458. //
  459. // Purpose: Initialize a MEM resource vector given a config manager
  460. // resource structure.
  461. //
  462. // Arguments:
  463. // ppMEMList [out] returns MEM_LIST that will be created.
  464. // pMEMResource [in] MEM_RESOURCE structure from config manager.
  465. //
  466. // Author: t-nabilr 07 Apr 1997
  467. //
  468. // Notes:
  469. //
  470. VOID CHwRes::InitMEMList(PMEM_LIST* ppMEMList, PMEM_RESOURCE pMEMResource)
  471. {
  472. PMEM_DES pMEMHeader;
  473. PMEM_RANGE pMEMData;
  474. PMEM_LIST_ELEMENT peMEMList;
  475. // For easy access
  476. pMEMHeader = &(pMEMResource->MEM_Header);
  477. pMEMData = pMEMResource->MEM_Data;
  478. // Create a new list.
  479. *ppMEMList = new MEM_LIST;
  480. if (*ppMEMList == NULL)
  481. {
  482. return;
  483. }
  484. ULONG iData; // index of MEM_Range structure we're looking at.
  485. DWORDLONG MEMBase; // current MEM Base we're adding to the list.
  486. ULONG cMEMBytes; // number of bytes required.
  487. DWORDLONG MEMAlign;
  488. // Go through all the MEM_Range structures, and all MEMs in the
  489. // specified range to the list.
  490. for (iData = 0; iData < pMEMHeader->MD_Count; iData++)
  491. {
  492. MEMAlign = pMEMData[iData].MR_Align;
  493. cMEMBytes = pMEMData[iData].MR_nBytes;
  494. // do sanity checks
  495. if (0 == MEMAlign)
  496. {
  497. TraceTag(ttidNetComm, "CHwRes::InitMEMList() - Bogus alignment "
  498. "field while processing info from Config Manager.");
  499. break;
  500. }
  501. if (0 == cMEMBytes)
  502. {
  503. TraceTag(ttidNetComm, "CHwRes::InitMEMList() - Bogus membytes "
  504. "field while processing info from Config Manager.");
  505. break;
  506. }
  507. for (MEMBase = pMEMData[iData].MR_Min;
  508. MEMBase+cMEMBytes-1 <= pMEMData[iData].MR_Max;
  509. MEMBase += ~MEMAlign + 1)
  510. {
  511. // For each MEM that falls within the given range,
  512. // create new MEM List Element, populate its fields and insert
  513. // it into the MEMList.
  514. peMEMList = new MEM_LIST_ELEMENT;
  515. if (peMEMList == NULL)
  516. {
  517. continue;
  518. }
  519. peMEMList->dwMEM_Base = MEMBase;
  520. peMEMList->dwMEM_End = MEMBase + cMEMBytes - 1;
  521. peMEMList->fConflict = FALSE;
  522. peMEMList->fAllocated = FALSE;
  523. (*ppMEMList)->push_back(peMEMList);
  524. // Check for wrapping.
  525. if (MEMBase >= MEMBase + ~MEMAlign + 1)
  526. {
  527. TraceTag(ttidError, "Memory base is greater than Memory "
  528. "end!!!");
  529. break;
  530. }
  531. }
  532. }
  533. }
  534. //+---------------------------------------------------------------------------
  535. //
  536. // Member: CHwRes::InitIOList
  537. //
  538. // Purpose: Initialize an IO resource vector given a config manager
  539. // resource structure.
  540. //
  541. // Arguments:
  542. // ppIOList [out] returns IO_LIST that will be created.
  543. // pIOResource [in] IO_RESOURCE structure from config manager.
  544. //
  545. // Author: t-nabilr 07 Apr 1997
  546. //
  547. // Notes:
  548. //
  549. VOID CHwRes::InitIOList(PIO_LIST* ppIOList, PIO_RESOURCE pIOResource)
  550. {
  551. PIO_DES pIOHeader;
  552. PIO_RANGE pIOData;
  553. PIO_LIST_ELEMENT peIOList;
  554. // For easy access
  555. pIOHeader = &(pIOResource->IO_Header);
  556. pIOData = pIOResource->IO_Data;
  557. // Create a new list.
  558. *ppIOList = new IO_LIST;
  559. if (*ppIOList == NULL)
  560. {
  561. return;
  562. }
  563. ULONG iData; // index of IO_Range structure we're looking at.
  564. DWORDLONG IOBase; // current IO Base we're adding to the list.
  565. ULONG cIOBytes; // number of bytes required.
  566. DWORDLONG IOAlign;
  567. // Go through all the IO_Range structures, and all IOs in the
  568. // specified range to the list.
  569. for (iData = 0; iData < pIOHeader->IOD_Count; iData++)
  570. {
  571. IOAlign = pIOData[iData].IOR_Align;
  572. cIOBytes = pIOData[iData].IOR_nPorts;
  573. // Perform sanity checks.
  574. if (0 == IOAlign)
  575. {
  576. TraceTag(ttidError, "CHwRes::InitIOList - Bogus alignment field "
  577. "while processing data from Config Manager.");
  578. break;
  579. }
  580. if (0 == cIOBytes)
  581. {
  582. TraceTag(ttidError, "CHwRes::InitIOList - Bogus IObytes field "
  583. "while processing data from Config Manager.");
  584. break;
  585. }
  586. for (IOBase = pIOData[iData].IOR_Min;
  587. IOBase+cIOBytes-1 <= pIOData[iData].IOR_Max;
  588. IOBase += ~IOAlign + 1)
  589. {
  590. // For each IO that falls within the given range,
  591. // create new IO List Element, populate its fields and insert
  592. // it into the IOList.
  593. peIOList = new IO_LIST_ELEMENT;
  594. if (peIOList == NULL)
  595. {
  596. continue;
  597. }
  598. peIOList->dwIO_Base = IOBase;
  599. peIOList->dwIO_End = IOBase + cIOBytes-1;
  600. peIOList->fConflict = FALSE;
  601. peIOList->fAllocated = FALSE;
  602. (*ppIOList)->push_back(peIOList);
  603. // Check for wrapping.
  604. if (IOBase >= IOBase + ~IOAlign+1)
  605. {
  606. TraceTag(ttidError, "IO base is greater than IO end!!!");
  607. break;
  608. }
  609. }
  610. }
  611. }
  612. //+---------------------------------------------------------------------------
  613. //
  614. // Member: CHwRes::UseAnswerFile
  615. //
  616. // Purpose: Reads in settings from answerfile and puts them into m_vAf*
  617. // member variables.
  618. //
  619. // Arguments:
  620. // szAnswerFile [in] Path to answerfile.
  621. // szSection [in] Section to read within answerfile
  622. //
  623. // Author: t-nabilr 07 Apr 1997
  624. //
  625. // Notes:
  626. //
  627. VOID CHwRes::UseAnswerFile(const WCHAR * const szAnswerFile, const WCHAR * const szSection) {
  628. CWInfFile AnswerFile;
  629. PCWInfSection pSection;
  630. DWORD dw;
  631. AssertSz(m_fInitialized,
  632. "UseAnswerFile called before CHwRes class HrInit'ed");
  633. // initialize answer file class
  634. if (AnswerFile.Init() == FALSE)
  635. {
  636. AssertSz(FALSE,"CHwRes::UseAnswerFile - Failed to initialize CWInfFile");
  637. return;
  638. }
  639. // Open the answerfile and find the desired section.
  640. AnswerFile.Open(szAnswerFile);
  641. pSection = AnswerFile.FindSection(szSection);
  642. // If the answer file section specified is missing
  643. // we should skip trying to read
  644. //
  645. if (pSection)
  646. {
  647. // Get the hardware resource keys
  648. if (pSection->GetIntValue(c_szAfIoAddr, &dw))
  649. {
  650. // set this only if the value isn't obviously wrong (i.e. <= 0)
  651. if (dw > 0)
  652. {
  653. m_vAfIo.SetDword(dw);
  654. m_vAfIo.SetPresent(TRUE);
  655. }
  656. }
  657. if (pSection->GetIntValue(c_szAfIrq, &dw))
  658. {
  659. // set this only if the value isn't obviously wrong (i.e. <= 0)
  660. if (dw > 0)
  661. {
  662. m_vAfIrq.SetDword(dw);
  663. m_vAfIrq.SetPresent(TRUE);
  664. }
  665. }
  666. if (pSection->GetIntValue(c_szAfDma, &dw))
  667. {
  668. // set this only if the value isn't obviously wrong (i.e. <= 0)
  669. if (dw > 0)
  670. {
  671. m_vAfDma.SetDword(dw);
  672. m_vAfDma.SetPresent(TRUE);
  673. }
  674. }
  675. if (pSection->GetIntValue(c_szAfMem, &dw))
  676. {
  677. // set this only if the value isn't obviously wrong (i.e. <= 0)
  678. if (dw > 0)
  679. {
  680. m_vAfMem.SetDword(dw);
  681. m_vAfMem.SetPresent(TRUE);
  682. }
  683. }
  684. }
  685. }
  686. //+---------------------------------------------------------------------------
  687. //
  688. // Member: CHwRes::FValidateAnswerfileSettings
  689. //
  690. // Purpose:
  691. // Ensures that the hw resource settings read in from the answerfile
  692. // are valid. It will, optionally, raise UI if the properties
  693. // are invalid.
  694. //
  695. // Arguments:
  696. // fDisplayUI [in] TRUE, if an error UI is to be displayed if the
  697. // answerfile settings are invalid
  698. //
  699. // Returns: HRESULT. S_OK if the answerfile settings are valid, S_FALSE if there
  700. // are no resources to set, an error code otherwise
  701. //
  702. // Author: t-nabilr 07 Apr 1997
  703. //
  704. // Notes:
  705. // Will set the m_vAfMemEnd and m_vAfIoEnd to correspond to
  706. // m_vAfMem and m_vAfIo.
  707. //
  708. HRESULT
  709. CHwRes::HrValidateAnswerfileSettings(BOOL fDisplayUI)
  710. {
  711. HRESULT hr = S_OK;
  712. AssertSz(m_fInitialized, "FValidateAnswerfileSettings called before "
  713. "CHwRes class HrInit'ed");
  714. // override current resource settings
  715. if (!m_vAfDma.IsPresent() &&
  716. !m_vAfIrq.IsPresent() &&
  717. !m_vAfIo.IsPresent() &&
  718. !m_vAfMem.IsPresent())
  719. {
  720. // no resources found...
  721. TraceTag(ttidNetComm, "No Hardware Resources found in answerfile.");
  722. hr = S_FALSE;
  723. }
  724. else
  725. {
  726. if (!FValidateAnswerfileResources())
  727. {
  728. hr = HRESULT_FROM_WIN32(ERROR_INVALID_PARAMETER);
  729. TraceError("Error setting adapter resources from "
  730. "answerfile.", hr);
  731. }
  732. else
  733. {
  734. // m_vAfMemEnd and m_vAfIoEnd were set by
  735. // FValidateAnswerfileResources()
  736. Assert(FImplies(m_vAfMem.IsPresent(), m_vAfMemEnd.IsPresent()));
  737. Assert(FImplies(m_vAfIo.IsPresent(), m_vAfIoEnd.IsPresent()));
  738. }
  739. }
  740. TraceError("CHwRes::HrValidateAnswerfileSettings",
  741. (S_FALSE == hr) ? S_OK : hr);
  742. return hr;
  743. }
  744. //+---------------------------------------------------------------------------
  745. //
  746. // Member: CHwRes::FCommitAnswerfileSettings
  747. //
  748. // Purpose:
  749. // Commits (to the config manager) the hw resource settings read
  750. // in from the Answerfile.
  751. //
  752. // Returns: FALSE if there were problems writing the BootConfig entry.
  753. //
  754. // Author: t-nabilr 07 Apr 1997
  755. //
  756. // Notes:
  757. //
  758. BOOL CHwRes::FCommitAnswerfileSettings()
  759. {
  760. AssertSz(m_fInitialized, "FCommitAnswerfileSettings called before "
  761. "CHwRes class HrInit'ed");
  762. Assert(FImplies(m_vAfIo.IsPresent(), m_vAfIoEnd.IsPresent()));
  763. Assert(FImplies(m_vAfMem.IsPresent(), m_vAfMemEnd.IsPresent()));
  764. // write out forced config entry to the config manager
  765. BOOL f;
  766. f = FCreateBootConfig(&m_vAfMem, &m_vAfMemEnd,
  767. &m_vAfIo, &m_vAfIo,
  768. &m_vAfDma,
  769. &m_vAfIrq);
  770. return f;
  771. }
  772. //+---------------------------------------------------------------------------
  773. //
  774. // Member: CHwRes::FValidateAnswerfileResources
  775. //
  776. // Purpose: Validates the resource requirements read in from Answerfile.
  777. //
  778. // Arguments:
  779. // nID [out] ID of thingy.
  780. // fInstall [in] TRUE if installing, FALSE otherwise.
  781. // ppv [in,out] Old value is freed and this returns new value.
  782. //
  783. // Returns: TRUE, if resource requirement are valid.
  784. //
  785. // Author: t-nabilr 07 Apr 1997
  786. //
  787. // Notes:
  788. // Implementation note: We have a set of resource requirements from
  789. // the answerfile (stored in m_vAf*). This method iterates through
  790. // every logical config (execept alloc or boot ones) and tests to see
  791. // if the resource requirements are valid in the logical config.
  792. // If they are, then we can use the resource requirements. If they're
  793. // not all valid in any logical config, then we return FALSE.
  794. //
  795. BOOL CHwRes::FValidateAnswerfileResources()
  796. {
  797. DWORD dwMemEnd;
  798. DWORD dwIoEnd;
  799. BOOL fResourceValid;
  800. AssertSz(m_fInitialized, "FValidateAnswerfileResources called before "
  801. "CHwRes class HrInit'ed");
  802. // Configuration List should be initialized
  803. Assert(0 != m_ConfigList.size());
  804. PRESOURCE pResource;
  805. for (size_t iConfig = 0; iConfig < m_ConfigList.size(); iConfig++)
  806. {
  807. // we only want Basic configurations, so skip alloc or boot.
  808. if (m_ConfigList[iConfig]->fBoot || m_ConfigList[iConfig]->fAlloc)
  809. {
  810. continue;
  811. }
  812. fResourceValid = TRUE;
  813. if (m_vAfDma.IsPresent())
  814. {
  815. if (!FValidateDMA(m_ConfigList[iConfig], m_vAfDma.GetDword()))
  816. {
  817. fResourceValid = FALSE;
  818. }
  819. }
  820. if (m_vAfIrq.IsPresent())
  821. {
  822. if (!FValidateIRQ(m_ConfigList[iConfig], m_vAfIrq.GetDword()))
  823. {
  824. fResourceValid = FALSE;
  825. }
  826. }
  827. if (m_vAfIo.IsPresent())
  828. {
  829. if (!FGetIOEndPortGivenBasePort(m_ConfigList[iConfig],
  830. m_vAfIo.GetDword(), &dwIoEnd))
  831. {
  832. m_vAfIoEnd.SetPresent(FALSE);
  833. fResourceValid = FALSE;
  834. }
  835. else
  836. {
  837. m_vAfIoEnd.SetDword(dwIoEnd);
  838. m_vAfIoEnd.SetPresent(TRUE);
  839. }
  840. }
  841. if (m_vAfMem.IsPresent())
  842. {
  843. if (!FGetMEMEndGivenBase(m_ConfigList[iConfig],
  844. m_vAfMem.GetDword(), &dwMemEnd))
  845. {
  846. m_vAfMemEnd.SetPresent(FALSE);
  847. fResourceValid = FALSE;
  848. }
  849. else
  850. {
  851. m_vAfMemEnd.SetDword(dwMemEnd);
  852. m_vAfMemEnd.SetPresent(TRUE);
  853. }
  854. }
  855. if (fResourceValid) break; // found valid one.
  856. }
  857. // something has to be present (otherwise don't call this function!)
  858. Assert(m_vAfIo.IsPresent() || m_vAfIrq.IsPresent() ||
  859. m_vAfDma.IsPresent() || m_vAfMem.IsPresent());
  860. return fResourceValid;
  861. }
  862. //+---------------------------------------------------------------------------
  863. //
  864. // Member: CHwRes::FCreateBootConfig
  865. //
  866. // Purpose: Create and insert a Boot Config entry into the Config
  867. // Manager.
  868. //
  869. // Arguments:
  870. // pvMem [in] Memory range base
  871. // pvMemEnd [in] Memory range end
  872. // pvIo [in] Io range base
  873. // pvIoEnd [in] Io range end
  874. // pvDma [in] Dma resource required.
  875. // pvIrq [in] Irq resource required.
  876. //
  877. // Returns: TRUE if creation of forced config was successful.
  878. //
  879. // Author: t-nabilr 07 Apr 1997
  880. //
  881. // Notes:
  882. //
  883. BOOL CHwRes::FCreateBootConfig(
  884. CValue * pvMem,
  885. CValue * pvMemEnd,
  886. CValue * pvIo,
  887. CValue * pvIoEnd,
  888. CValue * pvDma,
  889. CValue * pvIrq)
  890. {
  891. DMA_RESOURCE DMARes;
  892. IO_RESOURCE IORes;
  893. MEM_RESOURCE MEMRes;
  894. IRQ_RESOURCE IRQRes;
  895. LOG_CONF lcLogConf = NULL;
  896. AssertSz(pvMem && pvMemEnd && pvIo && pvIoEnd && pvDma && pvIrq,
  897. "One of the pointer parameters passed to CHwRes::FCreate"
  898. "BootConfig() is null.");
  899. AssertSz(m_fInitialized, "FCreateBootConfig called before "
  900. "CHwRes class HrInit'ed");
  901. TraceTag(ttidNetComm, "In FCreateBootConfig");
  902. // Create a boot config
  903. //
  904. if (CM_Add_Empty_Log_Conf(&lcLogConf, m_DevNode, LCPRI_BOOTCONFIG,
  905. BOOT_LOG_CONF)
  906. != CR_SUCCESS)
  907. {
  908. TraceTag(ttidNetComm, "Unable to create BOOT_LOG_CONF");
  909. return FALSE;
  910. }
  911. if (pvDma->IsPresent())
  912. {
  913. Assert(pvDma->GetDword() > 0);
  914. // fill out DMAResource structure's header
  915. ZeroMemory(&DMARes, sizeof(DMARes));
  916. DMARes.DMA_Header.DD_Count = 0;
  917. DMARes.DMA_Header.DD_Type = DType_Range;
  918. DMARes.DMA_Header.DD_Flags = 0;
  919. DMARes.DMA_Header.DD_Alloc_Chan = pvDma->GetDword();
  920. // add to boot conf
  921. CM_Add_Res_Des(NULL, lcLogConf, ResType_DMA, &DMARes,
  922. sizeof(DMARes), 0);
  923. TraceTag(ttidNetComm, "added Dma resource %lX", pvDma->GetDword());
  924. }
  925. if (pvIrq->IsPresent())
  926. {
  927. Assert(pvIrq->GetDword() > 0);
  928. // IRQResource structure
  929. ZeroMemory(&IRQRes, sizeof(IRQRes));
  930. IRQRes.IRQ_Header.IRQD_Count = 0;
  931. IRQRes.IRQ_Header.IRQD_Type = IRQType_Range;
  932. IRQRes.IRQ_Header.IRQD_Flags |= fIRQD_Edge;
  933. IRQRes.IRQ_Header.IRQD_Alloc_Num = pvIrq->GetDword();
  934. IRQRes.IRQ_Header.IRQD_Affinity = 0;
  935. // add to boot conf
  936. CM_Add_Res_Des(NULL, lcLogConf, ResType_IRQ, &IRQRes,
  937. sizeof(IRQRes), 0);
  938. TraceTag(ttidNetComm, "added IRQ resource %lX", pvIrq->GetDword());
  939. }
  940. if (pvIo->IsPresent()) {
  941. Assert(pvIo->GetDword() > 0);
  942. Assert(pvIoEnd->GetDword() > 0);
  943. ZeroMemory(&IORes, sizeof(IORes));
  944. IORes.IO_Header.IOD_Count = 0;
  945. IORes.IO_Header.IOD_Type = IOType_Range;
  946. IORes.IO_Header.IOD_Alloc_Base = pvIo->GetDword();
  947. IORes.IO_Header.IOD_Alloc_End = pvIoEnd->GetDword();
  948. IORes.IO_Header.IOD_DesFlags = fIOD_10_BIT_DECODE;
  949. // add to boot conf
  950. CM_Add_Res_Des(NULL, lcLogConf, ResType_IO, &IORes, sizeof(IORes), 0);
  951. TraceTag(ttidNetComm, "added IO resource %lX-%lX", pvIo->GetDword(),
  952. pvIoEnd->GetDword());
  953. }
  954. if (pvMem->IsPresent()) {
  955. Assert(pvMem->GetDword() > 0);
  956. Assert(pvMemEnd->GetDword() > 0);
  957. ZeroMemory(&MEMRes, sizeof(MEMRes));
  958. MEMRes.MEM_Header.MD_Count = 0;
  959. MEMRes.MEM_Header.MD_Type = MType_Range;
  960. MEMRes.MEM_Header.MD_Alloc_Base = pvMem->GetDword();
  961. MEMRes.MEM_Header.MD_Alloc_End = pvMemEnd->GetDword();
  962. MEMRes.MEM_Header.MD_Flags = 0;
  963. // add to boot conf
  964. CM_Add_Res_Des(NULL, lcLogConf, ResType_Mem, &MEMRes,
  965. sizeof(MEMRes), 0);
  966. TraceTag(ttidNetComm, "added Memory resource %lX - %lX",
  967. pvMem->GetDword(), pvMemEnd->GetDword());
  968. }
  969. CM_Free_Log_Conf_Handle(lcLogConf);
  970. return TRUE;
  971. }
  972. //+---------------------------------------------------------------------------
  973. //
  974. // Member: CHwRes::FValidateIRQ
  975. //
  976. // Purpose: Validates that the IRQ given is valid in the given config.
  977. //
  978. // Arguments:
  979. // pConfig [in] Config to use.
  980. // dwIRQ [in] irq setting to validate.
  981. //
  982. // Returns: TRUE if irq setting is valid.
  983. //
  984. // Author: t-nabilr 07 Apr 1997
  985. //
  986. // Notes:
  987. //
  988. BOOL CHwRes::FValidateIRQ(PCONFIGURATION pConfig, ULONG dwIRQ)
  989. {
  990. PIRQ_LIST pIRQList;
  991. AssertSz(m_fInitialized, "FValidateIRQ called before "
  992. "CHwRes class HrInit'ed");
  993. Assert(pConfig != NULL);
  994. Assert(dwIRQ > 0);
  995. // For each IRQ resource in the given config
  996. // go through list of valid IRQ looking for given one
  997. // if found, return TRUE
  998. for (size_t iRes = 0; iRes < pConfig->cResource; iRes++)
  999. {
  1000. if (pConfig->aResource[iRes].ResourceType != ResType_IRQ)
  1001. continue;
  1002. pIRQList = pConfig->aResource[iRes].pIRQList; // for easy access
  1003. for (size_t iIRQ = 0; iIRQ < pIRQList->size(); iIRQ++)
  1004. {
  1005. if ((*pIRQList)[iIRQ]->dwIRQ == dwIRQ)
  1006. {
  1007. return TRUE; // found it.
  1008. }
  1009. }
  1010. }
  1011. TraceTag(ttidNetComm, "IRQ %lX is not valid for this device", dwIRQ);
  1012. return FALSE;
  1013. }
  1014. //+---------------------------------------------------------------------------
  1015. //
  1016. // Member: CHwRes::FValidateDMA
  1017. //
  1018. // Purpose: Validate that given DMA is valid in given config.
  1019. //
  1020. // Arguments:
  1021. // pConfig [in] configuration to use
  1022. // dwDMA [in] dma setting to validate
  1023. //
  1024. // Returns: TRUE if dma setting is valid
  1025. //
  1026. // Author: t-nabilr 07 Apr 1997
  1027. //
  1028. // Notes:
  1029. //
  1030. BOOL CHwRes::FValidateDMA(PCONFIGURATION pConfig, ULONG dwDMA)
  1031. {
  1032. PDMA_LIST pDMAList;
  1033. AssertSz(m_fInitialized, "FValidateDMA called before "
  1034. "CHwRes class HrInit'ed");
  1035. Assert(pConfig != NULL);
  1036. Assert(dwDMA > 0);
  1037. // For each dma resource in the given config
  1038. // go through list of valid dma looking for given one
  1039. // if found, return TRUE
  1040. for (size_t iRes = 0; iRes < pConfig->cResource; iRes++)
  1041. {
  1042. if (pConfig->aResource[iRes].ResourceType != ResType_DMA)
  1043. continue;
  1044. pDMAList = pConfig->aResource[iRes].pDMAList; // for easy access
  1045. for (size_t iDMA = 0; iDMA < pDMAList->size(); iDMA++)
  1046. {
  1047. if ((*pDMAList)[iDMA]->dwDMA == dwDMA)
  1048. {
  1049. return TRUE;
  1050. }
  1051. }
  1052. }
  1053. TraceTag(ttidNetComm, "DMA %lX is not valid for this device", dwDMA);
  1054. return FALSE;
  1055. }
  1056. //+---------------------------------------------------------------------------
  1057. //
  1058. // Member: CHwRes::FGetIOEndPortGivenBasePort
  1059. //
  1060. // Purpose: Get an IO Range given only the BasePort
  1061. //
  1062. // Arguments:
  1063. // pConfig [in] configuration to use.
  1064. // dwBase [in] Io base
  1065. // pdwEnd [out] Io end is returned
  1066. //
  1067. // Returns: TRUE if Io base is valid in given config.
  1068. //
  1069. // Author: t-nabilr 07 Apr 1997
  1070. //
  1071. // Notes:
  1072. //
  1073. BOOL CHwRes::FGetIOEndPortGivenBasePort(PCONFIGURATION pConfig, DWORD dwBase,
  1074. DWORD * pdwEnd)
  1075. {
  1076. PIO_LIST pIOList;
  1077. AssertSz(m_fInitialized, "FGetIOEndPortGivenBasePort called before "
  1078. "CHwRes class HrInit'ed");
  1079. Assert(pConfig != NULL);
  1080. Assert(dwBase > 0);
  1081. // For each resource in the given configuration
  1082. // if it's an IO Resource
  1083. // Go through the list of valid IO resources looking for
  1084. // a matching base port
  1085. // if found, then set the corresponding end port, return TRUE;
  1086. for (size_t iRes = 0; iRes < pConfig->cResource; iRes++)
  1087. {
  1088. // ensure we're looking at an IO type
  1089. if (pConfig->aResource[iRes].ResourceType != ResType_IO)
  1090. continue;
  1091. pIOList = pConfig->aResource[iRes].pIOList; // for easy access
  1092. // go through all IO Elements in this list
  1093. for (size_t iIO = 0; iIO < pIOList->size(); iIO++)
  1094. {
  1095. if ((*pIOList)[iIO]->dwIO_Base == dwBase)
  1096. {
  1097. // found matching IO base port
  1098. *pdwEnd = (*pIOList)[iIO]->dwIO_End;
  1099. return TRUE;
  1100. }
  1101. }
  1102. }
  1103. TraceTag(ttidNetComm, "IO %lX is not valid for this device", dwBase);
  1104. return FALSE; // not found
  1105. }
  1106. //+---------------------------------------------------------------------------
  1107. //
  1108. // Member: CHwRes::FGetMEMEndGivenBase
  1109. //
  1110. // Purpose: Get a MEM range given the Mem base and config.
  1111. //
  1112. // Arguments:
  1113. // pConfig [in] configuration to use
  1114. // dwBase [in] mem base
  1115. // pdwEnd [out] mem end is returned.
  1116. //
  1117. // Returns: TRUE if the dwBase is a valid mem setting.
  1118. //
  1119. // Author: t-nabilr 07 Apr 1997
  1120. //
  1121. // Notes:
  1122. //
  1123. BOOL CHwRes::FGetMEMEndGivenBase(PCONFIGURATION pConfig, DWORD dwBase,
  1124. DWORD * pdwEnd)
  1125. {
  1126. PMEM_LIST pMEMList;
  1127. AssertSz(m_fInitialized, "FGetMEMEndGivenBase called before "
  1128. "CHwRes class HrInit'ed");
  1129. Assert(pConfig != NULL);
  1130. Assert(dwBase > 0);
  1131. // For each resource in the given configuration
  1132. // if it's an MEM Resource
  1133. // Go through the list of valid MEM resources looking for
  1134. // a matching base port
  1135. // if found, then set the corresponding end port,return TRUE;
  1136. for (size_t iRes = 0; iRes < pConfig->cResource; iRes++)
  1137. {
  1138. // ensure we're looking at an MEM type
  1139. if (pConfig->aResource[iRes].ResourceType != ResType_Mem)
  1140. continue;
  1141. pMEMList = pConfig->aResource[iRes].pMEMList; // for easy access
  1142. // go through all MEM Elements in this list
  1143. for (size_t iMEM = 0; iMEM < pMEMList->size(); iMEM++)
  1144. {
  1145. if ((*pMEMList)[iMEM]->dwMEM_Base == dwBase)
  1146. {
  1147. // found matching MEM base addr
  1148. *pdwEnd = (*pMEMList)[iMEM]->dwMEM_End;
  1149. return TRUE;
  1150. }
  1151. }
  1152. }
  1153. TraceTag(ttidNetComm, "Memory %lX is not valid for this device", dwBase);
  1154. return FALSE; // not found
  1155. }
  1156. //$REVIEW (t-pkoch) this function isn't yet in our custom STL...
  1157. // it can be removed later (when it causes errors)
  1158. template<class T> void os_release(vector<T> & v)
  1159. {
  1160. for(vector<T>::iterator iterDelete = v.begin() ; iterDelete != v.end() ;
  1161. ++iterDelete)
  1162. delete *iterDelete;
  1163. }
  1164. CHwRes::~CHwRes()
  1165. {
  1166. AssertSz(m_fHrInitCalled, "CHwRes destructor called before "
  1167. "CHwRes::HrInit() called");
  1168. vector<CONFIGURATION *>::iterator ppConfig;
  1169. RESOURCE * pRes;
  1170. // Delete everything from m_ConfigList.
  1171. for (ppConfig = m_ConfigList.begin(); ppConfig != m_ConfigList.end();
  1172. ppConfig++)
  1173. {
  1174. for (size_t iRes = 0; iRes < (*ppConfig)->cResource; iRes++)
  1175. {
  1176. pRes = &((*ppConfig)->aResource[iRes]);
  1177. switch(pRes->ResourceType)
  1178. {
  1179. case ResType_IRQ:
  1180. os_release(*(pRes->pIRQList));
  1181. delete pRes->pIRQList;
  1182. break;
  1183. case ResType_DMA:
  1184. os_release(*(pRes->pDMAList));
  1185. delete pRes->pDMAList;
  1186. break;
  1187. case ResType_IO:
  1188. os_release(*(pRes->pIOList));
  1189. delete pRes->pIOList;
  1190. break;
  1191. case ResType_Mem:
  1192. os_release(*(pRes->pMEMList));
  1193. delete pRes->pMEMList;
  1194. break;
  1195. }
  1196. }
  1197. delete *ppConfig;
  1198. }
  1199. ReleaseObj(m_pnccItem);
  1200. }