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.

955 lines
24 KiB

  1. /*
  2. Module Name:
  3. infutil.c
  4. Abstract:
  5. This module implements utility routines to parse net card INF files.
  6. Author:
  7. Andy Herron Mar 24 1998
  8. Revision History:
  9. */
  10. #include "binl.h"
  11. #pragma hdrstop
  12. #include "netinfp.h"
  13. const WCHAR NetInfHexToCharTable[17] = L"0123456789ABCDEF";
  14. ULONG
  15. NetInfCloseNetcardInfo (
  16. PNETCARD_INF_BLOCK pNetCards
  17. )
  18. /*++
  19. Routine Description:
  20. This function just dereferences the block for the 'alive' reference.
  21. This may cause it to be deleted.
  22. Arguments:
  23. pNetCards - A pointer to NETCARD_INF_BLOCK block allocated. Contains all
  24. the persistant info required for the netcards.
  25. Return Value:
  26. Windows Error.
  27. --*/
  28. {
  29. BinlAssert( pNetCards->ReferenceCount > 0 );
  30. DereferenceNetcardInfo( pNetCards );
  31. return ERROR_SUCCESS;
  32. }
  33. VOID
  34. DereferenceNetcardInfo (
  35. PNETCARD_INF_BLOCK pNetCards
  36. )
  37. /*++
  38. Routine Description:
  39. This function frees all memory, handles, etc that is stored in the
  40. NETCARD_INF_BLOCK passed in. Note that all NETCARD_RESPONSE_DATABASE
  41. entries are simply dereferenced, not freed here. This is because we don't
  42. want to require that all threads are done with these records before we
  43. close down the pNetCards block.
  44. Arguments:
  45. pNetCards - A pointer to NETCARD_INF_BLOCK block allocated. Contains all
  46. the persistant info required for the netcards.
  47. Return Value:
  48. Windows Error.
  49. --*/
  50. {
  51. ULONG i;
  52. EnterCriticalSection( &NetInfLock );
  53. pNetCards->ReferenceCount--;
  54. if (pNetCards->ReferenceCount > 0) {
  55. LeaveCriticalSection( &NetInfLock );
  56. return;
  57. }
  58. // only remove it from the global list if it was put on the list. It
  59. // might not be on the list if it's not called within BINL (i.e. RISETUP
  60. // is just enumerating files).
  61. if (pNetCards->InfBlockEntry.Flink != NULL) {
  62. RemoveEntryList( &pNetCards->InfBlockEntry );
  63. }
  64. LeaveCriticalSection( &NetInfLock );
  65. EnterCriticalSection( &pNetCards->Lock );
  66. //
  67. // No thread after this should call FindNetcardInfo to search the table,
  68. // since the caller just closed it.
  69. //
  70. // Free all entries allocated for this block. We just dereference in case
  71. // any thread is using a specific entry.
  72. //
  73. for (i = 0; i < NETCARD_HASH_TABLE_SIZE; i++) {
  74. PLIST_ENTRY listHead = &pNetCards->NetCardEntries[i];
  75. while (IsListEmpty( listHead ) == FALSE) {
  76. PNETCARD_RESPONSE_DATABASE pInfEntry;
  77. PLIST_ENTRY listEntry = RemoveHeadList( listHead );
  78. pInfEntry = (PNETCARD_RESPONSE_DATABASE) CONTAINING_RECORD(
  79. listEntry,
  80. NETCARD_RESPONSE_DATABASE,
  81. NetCardEntry );
  82. NetInfDereferenceNetcardEntry( pInfEntry );
  83. }
  84. }
  85. LeaveCriticalSection( &pNetCards->Lock );
  86. DeleteCriticalSection( &pNetCards->Lock );
  87. BinlFreeMemory( pNetCards );
  88. return;
  89. }
  90. VOID
  91. NetInfDereferenceNetcardEntry (
  92. PNETCARD_RESPONSE_DATABASE pInfEntry
  93. )
  94. /*++
  95. Routine Description:
  96. This function frees all memory, handles, etc that is stored in the
  97. NETCARD_INF_BLOCK passed in. Note that all NETCARD_RESPONSE_DATABASE
  98. entries are simply dereferenced, not freed here. This is because we don't
  99. want to require that all threads are done with these records before we
  100. close down the pNetCards block.
  101. Arguments:
  102. pNetCards - A pointer to NETCARD_INF_BLOCK block allocated. Contains all
  103. the persistant info required for the netcards.
  104. Return Value:
  105. Windows Error.
  106. --*/
  107. {
  108. LONG result;
  109. result = InterlockedDecrement( &pInfEntry->ReferenceCount );
  110. if (result > 0) {
  111. return;
  112. }
  113. BinlAssert( result == 0 );
  114. //
  115. // Time to free this one. It should've already been pulled from the list.
  116. //
  117. //
  118. // free the list of registry parameters we have stored off for it.
  119. //
  120. while (! IsListEmpty( &pInfEntry->Registry )) {
  121. PNETCARD_REGISTRY_PARAMETERS regParam;
  122. PLIST_ENTRY listEntry = RemoveHeadList( &pInfEntry->Registry );
  123. regParam = (PNETCARD_REGISTRY_PARAMETERS) CONTAINING_RECORD(
  124. listEntry,
  125. NETCARD_REGISTRY_PARAMETERS,
  126. RegistryListEntry );
  127. if (regParam->Parameter.Buffer) {
  128. BinlFreeMemory( regParam->Parameter.Buffer );
  129. }
  130. if (regParam->Value.Buffer) {
  131. BinlFreeMemory( regParam->Value.Buffer );
  132. }
  133. BinlFreeMemory( regParam );
  134. }
  135. //
  136. // free the list of registry parameters we have stored off for it.
  137. //
  138. while (! IsListEmpty( &pInfEntry->FileCopyList )) {
  139. PNETCARD_FILECOPY_PARAMETERS fileEntry;
  140. PLIST_ENTRY listEntry = RemoveHeadList( &pInfEntry->FileCopyList );
  141. fileEntry = (PNETCARD_FILECOPY_PARAMETERS) CONTAINING_RECORD(
  142. listEntry,
  143. NETCARD_FILECOPY_PARAMETERS,
  144. FileCopyListEntry );
  145. if (fileEntry->SourceFile.Buffer) {
  146. BinlFreeMemory( fileEntry->SourceFile.Buffer );
  147. }
  148. if (fileEntry->DestFile.Buffer) {
  149. BinlFreeMemory( fileEntry->DestFile.Buffer );
  150. }
  151. BinlFreeMemory( fileEntry );
  152. }
  153. if (pInfEntry->DriverName != NULL) {
  154. BinlFreeMemory( pInfEntry->DriverName );
  155. }
  156. if (pInfEntry->ServiceName != NULL) {
  157. BinlFreeMemory( pInfEntry->ServiceName );
  158. }
  159. //
  160. // if the section name is the same as the extended section name, then
  161. // they will be the same pointer. let's not free it twice.
  162. //
  163. if (pInfEntry->SectionNameExt != NULL &&
  164. pInfEntry->SectionNameExt != pInfEntry->SectionName) {
  165. BinlFreeMemory( pInfEntry->SectionNameExt );
  166. }
  167. if (pInfEntry->SectionName != NULL) {
  168. BinlFreeMemory( pInfEntry->SectionName );
  169. }
  170. if (pInfEntry->HardwareId != NULL) {
  171. BinlFreeMemory( pInfEntry->HardwareId );
  172. }
  173. if (pInfEntry->DriverDescription != NULL) {
  174. BinlFreeMemory( pInfEntry->DriverDescription );
  175. }
  176. BinlFreeMemory( pInfEntry );
  177. return;
  178. }
  179. ULONG
  180. FindNetcardInfo (
  181. PNETCARD_INF_BLOCK pNetCards,
  182. ULONG CardInfoVersion,
  183. NET_CARD_INFO UNALIGNED * CardIdentity,
  184. PNETCARD_RESPONSE_DATABASE *pInfEntry
  185. )
  186. /*++
  187. Routine Description:
  188. This function searches the drivers we've found and returns a pointer to
  189. an entry that most closely matches the client's request.
  190. Arguments:
  191. pNetCards - A pointer to NETCARD_INF_BLOCK block allocated. Contains all
  192. the persistant info required for the netcards.
  193. The NetCards structure has been referenced by the caller of this
  194. API and won't be going away from under us!
  195. CardInfoVersion - Version of the structure passed by the client.
  196. CardIdentity - has the values the app is looking for. we try our best to
  197. find one that matches.
  198. pInfEntry - the entry that was found if successful. NULL if in error.
  199. Return Value:
  200. ERROR_SUCCESS, ERROR_NOT_ENOUGH_MEMORY, or ERROR_NOT_SUPPORTED
  201. --*/
  202. {
  203. ULONG err;
  204. PWCHAR listOfPossibleCardIdentifiers = NULL;
  205. LIST_ENTRY listEntry;
  206. PWCHAR searchString;
  207. PNETCARD_RESPONSE_DATABASE pEntry = NULL;
  208. *pInfEntry = NULL;
  209. if (CardInfoVersion != OSCPKT_NETCARD_REQUEST_VERSION) {
  210. BinlPrint(( DEBUG_NETINF, "Not supporting CardInfoVersion %u\n", CardInfoVersion ));
  211. return ERROR_NOT_SUPPORTED;
  212. }
  213. err = CreateListOfCardIdentifiers( CardIdentity,
  214. &listOfPossibleCardIdentifiers );
  215. if (err != ERROR_SUCCESS) {
  216. goto exitFind;
  217. }
  218. BinlAssert( listOfPossibleCardIdentifiers != NULL);
  219. //
  220. // The search strings are ordered from most specific to least specific
  221. // so we have to search for them ordered top to bottom.
  222. //
  223. searchString = listOfPossibleCardIdentifiers;
  224. while (*searchString != L'\0') {
  225. PLIST_ENTRY listEntry;
  226. PLIST_ENTRY listHead;
  227. ULONG hwLength = lstrlenW( searchString );
  228. ULONG hashValue;
  229. COMPUTE_STRING_HASH( searchString, &hashValue );
  230. listHead = &pNetCards->NetCardEntries[HASH_TO_INF_INDEX(hashValue)];
  231. listEntry = listHead->Flink;
  232. while ( listEntry != listHead ) {
  233. pEntry = (PNETCARD_RESPONSE_DATABASE) CONTAINING_RECORD(
  234. listEntry,
  235. NETCARD_RESPONSE_DATABASE,
  236. NetCardEntry );
  237. err = CompareStringW( LOCALE_SYSTEM_DEFAULT,
  238. 0,
  239. searchString,
  240. hwLength,
  241. pEntry->HardwareId,
  242. -1
  243. );
  244. if (err == 2) {
  245. break; // a match was found.
  246. }
  247. pEntry = NULL;
  248. if (err == 3) {
  249. break; // it's greater therefore entry isn't present
  250. }
  251. listEntry = listEntry->Flink;
  252. }
  253. if (pEntry != NULL) {
  254. // we found one that matches it. reference it and return
  255. InterlockedIncrement( &pEntry->ReferenceCount );
  256. err = ERROR_SUCCESS;
  257. *pInfEntry = pEntry;
  258. break;
  259. }
  260. searchString += lstrlenW( searchString ) + 1; // point to next after null
  261. }
  262. exitFind:
  263. if (pEntry == NULL) {
  264. err = ERROR_NOT_SUPPORTED;
  265. } else {
  266. BinlAssert( err == ERROR_SUCCESS );
  267. }
  268. if (listOfPossibleCardIdentifiers) {
  269. BinlFreeMemory( listOfPossibleCardIdentifiers );
  270. }
  271. return err;
  272. }
  273. ULONG
  274. CreateListOfCardIdentifiers (
  275. NET_CARD_INFO UNALIGNED * CardIdentity,
  276. PWCHAR *CardIdentifiers
  277. )
  278. /*++
  279. Routine Description:
  280. This function creates the list of card identifiers for the given card.
  281. It generates a buffer that looks like this :
  282. "PCI\VEN_8086&DEV_1229&SUBSYS_00018086&REV_05"
  283. "PCI\VEN_8086&DEV_1229&SUBSYS_00018086"
  284. "PCI\VEN_8086&DEV_1229&REV_05"
  285. "PCI\VEN_8086&DEV_1229"
  286. empty string
  287. Note that if we support more than just PCI, we'll have to change this
  288. function.
  289. Arguments:
  290. CardIdentity - Holds the values we're looking for that identify this card.
  291. CardIdentifiers - where we put the resulant strings.
  292. Return Value:
  293. Windows Error.
  294. --*/
  295. {
  296. ULONG err = ERROR_SUCCESS;
  297. ULONG spaceRequired;
  298. PWCHAR nextField;
  299. if (CardIdentity->NicType == NETINF_BUS_TYPE_PCI) {
  300. UCHAR ch;
  301. WCHAR vendorBuff[5];
  302. WCHAR deviceBuff[5];
  303. WCHAR subsysBuff[9];
  304. WCHAR revBuff[3];
  305. // "PCI\VEN_8086&DEV_1229&SUBSYS_00018086&REV_05"
  306. // "PCI\VEN_8086&DEV_1229&SUBSYS_00018086"
  307. // "PCI\VEN_8086&DEV_1229&REV_05"
  308. // "PCI\VEN_8086&DEV_1229"
  309. spaceRequired = ((( sizeof( L"PCI\\1234&1234&" ) - 1 ) +
  310. ( sizeof( NETINF_VENDOR_STRING ) - 1 ) +
  311. ( sizeof( NETINF_DEVICE_STRING ) - 1 ) ) * 4 +
  312. (( sizeof( L"12&" ) - 1 ) +
  313. ( sizeof( NETINF_REVISION_STRING ) - 1 ) ) * 2 +
  314. (( sizeof( L"12345678&" ) - 1 ) +
  315. ( sizeof( NETINF_IOSUBS_STRING ) - 1 )) * 2 );
  316. spaceRequired += sizeof(WCHAR); // allocate 1 more for trailing null
  317. *CardIdentifiers = BinlAllocateMemory( spaceRequired );
  318. if (*CardIdentifiers == NULL) {
  319. return ERROR_NOT_ENOUGH_MEMORY;
  320. }
  321. nextField = *CardIdentifiers;
  322. //
  323. // Convert the numeric values to their char equivalents
  324. //
  325. ConvertHexToBuffer( &vendorBuff[0], CardIdentity->pci.Vendor_ID );
  326. vendorBuff[4] = '\0';
  327. ConvertHexToBuffer( &deviceBuff[0], CardIdentity->pci.Dev_ID );
  328. deviceBuff[4] = '\0';
  329. revBuff[0] = NetInfHexToCharTable[ ( CardIdentity->pci.Rev & 0xF0 ) >> 4 ];
  330. revBuff[1] = NetInfHexToCharTable[ ( CardIdentity->pci.Rev & 0x0F ) ];
  331. revBuff[2] = '\0';
  332. ConvertHexToBuffer( &subsysBuff[0], HIWORD( CardIdentity->pci.Subsys_ID ) );
  333. ConvertHexToBuffer( &subsysBuff[4], LOWORD( CardIdentity->pci.Subsys_ID ) );
  334. subsysBuff[8] = '\0';
  335. //
  336. // Now create the strings in most specific to least specific order
  337. //
  338. // "PCI\VEN_8086&DEV_1229&SUBSYS_00018086&REV_05"
  339. lstrcpyW( nextField, L"PCI\\" );
  340. lstrcatW( nextField, NETINF_VENDOR_STRING );
  341. lstrcatW( nextField, vendorBuff );
  342. lstrcatW( nextField, L"&" );
  343. lstrcatW( nextField, NETINF_DEVICE_STRING );
  344. lstrcatW( nextField, deviceBuff );
  345. lstrcatW( nextField, L"&" );
  346. lstrcatW( nextField, NETINF_IOSUBS_STRING );
  347. lstrcatW( nextField, subsysBuff );
  348. lstrcatW( nextField, L"&" );
  349. lstrcatW( nextField, NETINF_REVISION_STRING );
  350. lstrcatW( nextField, revBuff );
  351. nextField += lstrlenW( nextField ) + 1; // point to next after null
  352. // "PCI\VEN_8086&DEV_1229&SUBSYS_00018086"
  353. lstrcpyW( nextField, L"PCI\\" );
  354. lstrcatW( nextField, NETINF_VENDOR_STRING );
  355. lstrcatW( nextField, vendorBuff );
  356. lstrcatW( nextField, L"&" );
  357. lstrcatW( nextField, NETINF_DEVICE_STRING );
  358. lstrcatW( nextField, deviceBuff );
  359. lstrcatW( nextField, L"&" );
  360. lstrcatW( nextField, NETINF_IOSUBS_STRING );
  361. lstrcatW( nextField, subsysBuff );
  362. nextField += lstrlenW( nextField ) + 1; // point to next after null
  363. // "PCI\VEN_8086&DEV_1229&REV_05"
  364. lstrcpyW( nextField, L"PCI\\" );
  365. lstrcatW( nextField, NETINF_VENDOR_STRING );
  366. lstrcatW( nextField, vendorBuff );
  367. lstrcatW( nextField, L"&" );
  368. lstrcatW( nextField, NETINF_DEVICE_STRING );
  369. lstrcatW( nextField, deviceBuff );
  370. lstrcatW( nextField, L"&" );
  371. lstrcatW( nextField, NETINF_REVISION_STRING );
  372. lstrcatW( nextField, revBuff );
  373. nextField += lstrlenW( nextField ) + 1; // point to next after null
  374. // "PCI\VEN_8086&DEV_1229"
  375. lstrcpyW( nextField, L"PCI\\" );
  376. lstrcatW( nextField, NETINF_VENDOR_STRING );
  377. lstrcatW( nextField, vendorBuff );
  378. lstrcatW( nextField, L"&" );
  379. lstrcatW( nextField, NETINF_DEVICE_STRING );
  380. lstrcatW( nextField, deviceBuff );
  381. nextField += lstrlenW( nextField ) + 1; // point to next after null
  382. //
  383. // to mark the end of the multi-sz, stick on another null terminator
  384. //
  385. *(nextField++) = L'\0';
  386. } else {
  387. *CardIdentifiers = NULL;
  388. err = ERROR_NOT_SUPPORTED;
  389. }
  390. return err;
  391. }
  392. VOID
  393. ConvertHexToBuffer(
  394. PWCHAR Buff,
  395. USHORT Value
  396. )
  397. {
  398. UCHAR ch;
  399. ch = HIBYTE( Value );
  400. *(Buff+0) = NetInfHexToCharTable[ ( ch & 0xF0 ) >> 4 ];
  401. *(Buff+1) = NetInfHexToCharTable[ ( ch & 0x0F ) ];
  402. ch = LOBYTE( Value );
  403. *(Buff+2) = NetInfHexToCharTable[ ( ch & 0xF0 ) >> 4 ];
  404. *(Buff+3) = NetInfHexToCharTable[ ( ch & 0x0F ) ];
  405. return;
  406. }
  407. ULONG
  408. GetSetupLineWideText (
  409. PINFCONTEXT InfContext,
  410. HINF InfHandle,
  411. PWCHAR Section,
  412. PWCHAR Key,
  413. PWCHAR *String,
  414. PULONG SizeOfAllocation OPTIONAL
  415. )
  416. {
  417. ULONG sizeRequired = 0;
  418. if (SetupGetLineTextW( InfContext,
  419. InfHandle,
  420. Section,
  421. Key,
  422. NULL,
  423. 0,
  424. &sizeRequired) == FALSE) {
  425. return GetLastError();
  426. }
  427. if (*String == NULL ||
  428. SizeOfAllocation == NULL ||
  429. *SizeOfAllocation < sizeRequired) {
  430. if (*String != NULL) {
  431. BinlFreeMemory( *String );
  432. *String = NULL;
  433. }
  434. *String = (PWCHAR) BinlAllocateMemory( sizeRequired * sizeof(WCHAR));
  435. if (*String == NULL) {
  436. return ERROR_NOT_ENOUGH_MEMORY;
  437. }
  438. if (SizeOfAllocation != NULL) {
  439. *SizeOfAllocation = sizeRequired;
  440. }
  441. }
  442. if (SetupGetLineTextW( InfContext,
  443. InfHandle,
  444. Section,
  445. Key,
  446. *String,
  447. sizeRequired,
  448. NULL) == FALSE) {
  449. return GetLastError();
  450. }
  451. return ERROR_SUCCESS;
  452. }
  453. ULONG
  454. GetSetupWideTextField (
  455. PINFCONTEXT InfContext,
  456. DWORD FieldIndex,
  457. PWCHAR *String,
  458. PULONG SizeOfAllocation
  459. )
  460. {
  461. ULONG sizeRequired = 0;
  462. if (SetupGetStringFieldW( InfContext,
  463. FieldIndex,
  464. NULL,
  465. 0,
  466. &sizeRequired) == FALSE) {
  467. return GetLastError();
  468. }
  469. if (*String == NULL ||
  470. SizeOfAllocation == NULL ||
  471. *SizeOfAllocation < sizeRequired) {
  472. if (*String != NULL) {
  473. BinlFreeMemory( *String );
  474. *String = NULL;
  475. }
  476. *String = (PWCHAR) BinlAllocateMemory( sizeRequired * sizeof(WCHAR));
  477. if (*String == NULL) {
  478. return ERROR_NOT_ENOUGH_MEMORY;
  479. }
  480. if (SizeOfAllocation != NULL) {
  481. *SizeOfAllocation = sizeRequired;
  482. }
  483. }
  484. if (SetupGetStringFieldW( InfContext,
  485. FieldIndex,
  486. *String,
  487. sizeRequired,
  488. NULL) == FALSE) {
  489. return GetLastError();
  490. }
  491. return ERROR_SUCCESS;
  492. }
  493. ULONG
  494. CheckHwDescription (
  495. PWCHAR HardwareID
  496. )
  497. /*++
  498. Routine Description:
  499. This function parses the HardwareID field of the driver's detail record
  500. and determines a) if we can support it and b) fills in all the values.
  501. Note that if we support more than just PCI, we'll have to change this
  502. function.
  503. Arguments:
  504. HardwareID - uppercased hardware id representing the driver's configuration
  505. Identifiers - the fields we need to fill in.
  506. Return Value:
  507. ERROR_NOT_SUPPORTED, ERROR_SUCCESS, or ERROR_INVALID_PARAMETER
  508. --*/
  509. {
  510. ULONG err = ERROR_SUCCESS;
  511. PWCHAR hwPointer = HardwareID;
  512. USHORT busType;
  513. USHORT Vendor; // Vendor_ID to check
  514. USHORT Device; // Dev_ID to check
  515. ULONG Subsystem; // Subsys_ID to check
  516. USHORT Revision; // Revision to check
  517. BOOLEAN RevPresent; // Revision present
  518. BOOLEAN SubPresent; // Subsystem present
  519. //
  520. // for now, PCI is the only one supported.
  521. //
  522. if (IsSubString( L"PCI\\", hwPointer, FALSE )) {
  523. hwPointer += (sizeof( "pci\\" ) - 1);
  524. busType = NETINF_BUS_TYPE_PCI;
  525. } else {
  526. return ERROR_NOT_SUPPORTED;
  527. }
  528. //
  529. // we parse the HardwareID as it's passed to us in the
  530. // SP_DRVINFO_DETAIL_DATA structure. It is of the form :
  531. //
  532. // pci\ven_8086&dev_1229&rev_01&subsys_00018086
  533. //
  534. // where the vendor will always be present and the device, revision,
  535. // and subsystem may or may not be present.
  536. //
  537. if (busType == NETINF_BUS_TYPE_PCI) {
  538. while (*hwPointer != L'\0') {
  539. if (IsSubString( NETINF_VENDOR_STRING, hwPointer, FALSE )) {
  540. hwPointer += ((sizeof(NETINF_VENDOR_STRING)/sizeof(WCHAR)) - 1);
  541. err = GetHexValueFromHw( &hwPointer, NULL, &Vendor );
  542. if (err != ERROR_SUCCESS) {
  543. goto cardSyntaxError;
  544. }
  545. } else if (IsSubString( NETINF_DEVICE_STRING, hwPointer, FALSE )) {
  546. hwPointer += ((sizeof(NETINF_DEVICE_STRING)/sizeof(WCHAR)) - 1);
  547. err = GetHexValueFromHw( &hwPointer, NULL, &Device );
  548. if (err != ERROR_SUCCESS) {
  549. goto cardSyntaxError;
  550. }
  551. } else if (IsSubString( NETINF_REVISION_STRING, hwPointer, FALSE )) {
  552. hwPointer += ((sizeof(NETINF_REVISION_STRING)/sizeof(WCHAR)) - 1);
  553. err = GetHexValueFromHw( &hwPointer, NULL, &Revision );
  554. if (err != ERROR_SUCCESS) {
  555. goto cardSyntaxError;
  556. }
  557. RevPresent = TRUE;
  558. } else if (IsSubString( NETINF_IOSUBS_STRING, hwPointer, FALSE )) {
  559. hwPointer += ((sizeof(NETINF_IOSUBS_STRING)/sizeof(WCHAR)) - 1);
  560. err = GetHexValueFromHw( &hwPointer, &Subsystem, NULL);
  561. if (err != ERROR_SUCCESS) {
  562. goto cardSyntaxError;
  563. }
  564. SubPresent = TRUE;
  565. } else {
  566. //
  567. // we hit something else. hmmm.. bail on this one.
  568. //
  569. goto cardSyntaxError;
  570. }
  571. }
  572. } else {
  573. // we should never get here unless we start supporting ISAPNP,
  574. // PCMCIA, etc
  575. cardSyntaxError:
  576. BinlPrint(( DEBUG_NETINF, "Not supporting INF hw string of %S\n", HardwareID ));
  577. err = ERROR_NOT_SUPPORTED;
  578. }
  579. if ((err == ERROR_SUCCESS) &&
  580. ((Vendor == 0) ||
  581. (Device == 0))) {
  582. // both vendor and device are required for it to be valid.
  583. err = ERROR_NOT_SUPPORTED;
  584. }
  585. return err;
  586. }
  587. ULONG
  588. GetHexValueFromHw (
  589. PWCHAR *String, // this is updated.
  590. PULONG longValue,
  591. PUSHORT shortValue
  592. )
  593. /*++
  594. Routine Description:
  595. This function parses a hex integer out of the HardwareID field of
  596. the driver's detail record. The string is of the form :
  597. pci\ven_8086&dev_1229&rev_01&subsys_00018086
  598. so this routine needs to convert the hex chars to a value.
  599. Arguments:
  600. String - the input string that we manipulate by moving it to the end of
  601. the integer (we also move it past the '&' if there is one present.
  602. longValue - integer that we fill in if it's present
  603. shortValue - ushort that we fill in if it's present
  604. Return Value:
  605. ERROR_INVALID_PARAMETER or ERROR_SUCCESS
  606. --*/
  607. {
  608. PWCHAR targetString = *String;
  609. ULONG value = 0;
  610. UCHAR ch;
  611. UCHAR hexChar;
  612. ULONG length = 0;
  613. ULONG maxLength = ( (shortValue != NULL) ?
  614. (sizeof(USHORT) * 2) :
  615. (sizeof(ULONG) * 2) );
  616. ch = LOBYTE( *targetString );
  617. while ((length++ < maxLength) && (ch != '\0') && (ch != '&')) {
  618. //
  619. // convert from the ascii char to it's hex representation
  620. //
  621. if (ch >= '0' && ch <= '9') {
  622. hexChar = ch - '0';
  623. } else if (ch >= 'A' && ch <= 'F') {
  624. hexChar = ch - 'A' + 10;
  625. } else if (ch >= 'a' && ch <= 'f') {
  626. hexChar = ch - 'a' + 10;
  627. } else {
  628. break;
  629. }
  630. value = ( value << 4 ) | hexChar;
  631. targetString++; // on to the next character
  632. ch = LOBYTE( *targetString );
  633. }
  634. if ((ch != '\0') && (ch != '&')) {
  635. return ERROR_INVALID_PARAMETER;
  636. }
  637. // skip all trailing ampersands... we allow more than one just to be
  638. // generous
  639. while (*targetString == L'&') {
  640. targetString++;
  641. }
  642. *String = targetString;
  643. if (longValue) {
  644. *longValue = value;
  645. }
  646. if (shortValue) {
  647. *shortValue = LOWORD( value );
  648. }
  649. return ERROR_SUCCESS;
  650. }
  651. BOOLEAN
  652. IsSubString (
  653. PWCHAR subString,
  654. PWCHAR target,
  655. BOOLEAN ignoreCase
  656. )
  657. //
  658. // our local version of memicmp so as not to pull in full c runtimes.
  659. //
  660. {
  661. LONG subStringLength = lstrlenW( subString );
  662. if (lstrlenW( target ) < subStringLength) {
  663. return FALSE;
  664. }
  665. return (CompareStringW( LOCALE_SYSTEM_DEFAULT,
  666. ignoreCase ? NORM_IGNORECASE : 0,
  667. subString,
  668. subStringLength,
  669. target,
  670. subStringLength // note use same length
  671. ) == 2);
  672. }
  673. // infutil.c eof