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.

907 lines
24 KiB

  1. /*++
  2. Copyright (c) 1996 Microsoft Corporation
  3. Module Name:
  4. id.c
  5. Abstract:
  6. This module contains the code to handle IRP_MN_QUERY_ID
  7. Authors:
  8. Ravisankar Pudipeddi (ravisp)
  9. Environment:
  10. Kernel mode only
  11. Notes:
  12. Revision History:
  13. Neil Sandlin (neilsa) - split off from pdopnp.c
  14. --*/
  15. #include "pch.h"
  16. //
  17. // Internal References
  18. //
  19. NTSTATUS
  20. PcmciaGenerateDeviceId(
  21. IN PSOCKET_DATA SocketData,
  22. IN ULONG FunctionNumber,
  23. OUT PUCHAR *DeviceId
  24. );
  25. BOOLEAN
  26. PcmciaCheckInstance(
  27. IN PUCHAR DeviceId,
  28. IN ULONG Instance
  29. );
  30. NTSTATUS
  31. PcmciaGetDeviceType(
  32. IN PDEVICE_OBJECT Pdo,
  33. IN ULONG FunctionNumber,
  34. OUT PUCHAR DeviceType
  35. );
  36. VOID
  37. PcmciaFilterIdString(
  38. IN PUCHAR pIn,
  39. OUT PUCHAR pOut,
  40. ULONG MaxLen
  41. );
  42. #ifdef ALLOC_PRAGMA
  43. #pragma alloc_text(PAGE, PcmciaGenerateDeviceId)
  44. #pragma alloc_text(PAGE, PcmciaGetDeviceId)
  45. #pragma alloc_text(PAGE, PcmciaGetHardwareIds)
  46. #pragma alloc_text(PAGE, PcmciaGetCompatibleIds)
  47. #pragma alloc_text(PAGE, PcmciaGetDeviceType)
  48. #pragma alloc_text(PAGE, PcmciaFilterIdString)
  49. #endif
  50. #define PCMCIA_MAX_DEVICE_TYPE_SUPPORTED 12
  51. const
  52. UCHAR *PcmciaCompatibleIds[PCMCIA_MAX_DEVICE_TYPE_SUPPORTED+1] = {
  53. "", // Unknown..
  54. "", // Memory card (RAM, ROM, EPROM, Flash)
  55. "", // Serial I/O port, includes modems
  56. "", // Parallel printer port
  57. "*PNP0600", // Disk driver (ATA)
  58. "", // Video interface
  59. "", // Local Area Network adapter
  60. "", // Auto Increment Mass Storage card
  61. "", // SCSI bridge card
  62. "", // Security card
  63. "*PNP0D00", // Multi-Function 3.0 PC Card
  64. "", // Flash memory card
  65. "*PNPC200", // Modem card (sync with PCCARD_TYPE_MODEM)
  66. };
  67. NTSTATUS
  68. PcmciaGenerateDeviceId(
  69. IN PSOCKET_DATA SocketData,
  70. IN ULONG FunctionNumber,
  71. OUT PUCHAR *DeviceId
  72. )
  73. /*++
  74. The device ID is created from tuple information on the PC Card
  75. The goal is to create a unique ID for each
  76. card. The device ID is created from the manufacturer name string,
  77. the product name string, and a 16-bit CRC of a set of tuples.
  78. The ID is created by concatenating the "PCMCIA" prefix, the manufacturer
  79. name string, the product name string and the 16-bit CRC for the card.
  80. PCMCIA\<manuf_name>-<prod_name>-<crc>
  81. If the CISTPL_VERS_1 tuple is not available, or the manufacturer name is
  82. NULL, the string "UNKNOWN_MANUFACTURER" will be included in its place.
  83. If this is for a child function within a multifunctionn card, the generated
  84. device id would be:
  85. PCMCIA\<manuf_name>-<prod_name>-DEV<function number>-<crc>
  86. This device id is compatible with win 9x device id's (excluding the instance
  87. number which is returned separtely by handling another IRP.
  88. Arguments:
  89. Pdo - Pointer to the physical device object for the pc-card
  90. FunctionNumber - Function number of the function in a multi-function card.
  91. If this is PCMCIA_MULTIFUNCTION_PARENT, then the requested device id
  92. is for the parent device - not for any individual function
  93. DeviceId - Pointer to the string in which device id is returned
  94. Return Value
  95. Status
  96. --*/
  97. {
  98. PUCHAR deviceId;
  99. PAGED_CODE();
  100. deviceId = ExAllocatePool(PagedPool, PCMCIA_MAXIMUM_DEVICE_ID_LENGTH);
  101. if (deviceId == NULL) {
  102. return STATUS_INSUFFICIENT_RESOURCES;
  103. }
  104. //
  105. // Generate the device id
  106. //
  107. if (*(SocketData->Mfg) == '\0' ) {
  108. //
  109. // No manufacturer name available
  110. //
  111. if (FunctionNumber == PCMCIA_MULTIFUNCTION_PARENT) {
  112. //
  113. // Device id for the pc-card
  114. //
  115. if (SocketData->Flags & SDF_JEDEC_ID) {
  116. //
  117. // Flash memory cards have the special device id
  118. //
  119. sprintf(deviceId, "%s\\%s-%04x",
  120. PCMCIA_ID_STRING,
  121. PCMCIA_MEMORY_ID_STRING,
  122. SocketData->JedecId);
  123. } else {
  124. sprintf(deviceId, "%s\\%s-%04X",
  125. PCMCIA_ID_STRING,
  126. PCMCIA_UNKNOWN_MANUFACTURER_STRING,
  127. SocketData->CisCrc);
  128. }
  129. } else {
  130. //
  131. // This is for the individual multifunction child
  132. //
  133. sprintf(deviceId, "%s\\%s-DEV%d-%04X",
  134. PCMCIA_ID_STRING,
  135. PCMCIA_UNKNOWN_MANUFACTURER_STRING,
  136. FunctionNumber,
  137. SocketData->CisCrc);
  138. }
  139. } else {
  140. UCHAR Mfg[MAX_MANFID_LENGTH];
  141. UCHAR Ident[MAX_IDENT_LENGTH];
  142. PcmciaFilterIdString(SocketData->Mfg, Mfg, MAX_MANFID_LENGTH);
  143. PcmciaFilterIdString(SocketData->Ident, Ident, MAX_IDENT_LENGTH);
  144. if (FunctionNumber == PCMCIA_MULTIFUNCTION_PARENT) {
  145. //
  146. // Device id for the pc-card
  147. //
  148. sprintf(deviceId, "%s\\%s-%s-%04X",
  149. PCMCIA_ID_STRING,
  150. Mfg,
  151. Ident,
  152. SocketData->CisCrc);
  153. } else {
  154. //
  155. // This is for the individual multifunction child
  156. //
  157. sprintf(deviceId, "%s\\%s-%s-DEV%d-%04X",
  158. PCMCIA_ID_STRING,
  159. Mfg,
  160. Ident,
  161. FunctionNumber,
  162. SocketData->CisCrc);
  163. }
  164. }
  165. *DeviceId = deviceId;
  166. if ((FunctionNumber == PCMCIA_MULTIFUNCTION_PARENT) &&
  167. (SocketData->PdoExtension != NULL) &&
  168. (SocketData->PdoExtension->DeviceId == NULL)) {
  169. //
  170. // Keep a copy of the device id
  171. //
  172. PPDO_EXTENSION pdoExtension = SocketData->PdoExtension;
  173. pdoExtension->DeviceId = ExAllocatePool(NonPagedPool, strlen(deviceId)+1);
  174. if (pdoExtension->DeviceId) {
  175. RtlCopyMemory(pdoExtension->DeviceId, deviceId, strlen(deviceId)+1);
  176. }
  177. }
  178. return STATUS_SUCCESS;
  179. }
  180. NTSTATUS
  181. PcmciaGetDeviceId(
  182. IN PDEVICE_OBJECT Pdo,
  183. IN ULONG FunctionNumber,
  184. OUT PUNICODE_STRING DeviceId
  185. )
  186. /*++
  187. Generated device id is returned for the supplied pc-card
  188. Arguments:
  189. Pdo - Pointer to the physical device object for the pc-card
  190. FunctionNumber - Function number of the function in a multi-function card.
  191. If this is PCMCIA_MULTIFUNCTION_PARENT, then the requested device id
  192. is for the parent device - not for any individual function
  193. DeviceId - Pointer to the unicode string in which device id is returned
  194. Return Value
  195. Status
  196. --*/
  197. {
  198. PPDO_EXTENSION pdoExtension=Pdo->DeviceExtension;
  199. PSOCKET_DATA socketData = pdoExtension->SocketData;
  200. ANSI_STRING ansiId;
  201. PUCHAR deviceId;
  202. NTSTATUS status;
  203. PAGED_CODE();
  204. ASSERT(DeviceId);
  205. if (!socketData) {
  206. ASSERT (socketData);
  207. return STATUS_DEVICE_NOT_READY;
  208. }
  209. status = PcmciaGenerateDeviceId(socketData,
  210. FunctionNumber,
  211. &deviceId);
  212. if (!NT_SUCCESS(status)) {
  213. return status;
  214. }
  215. DebugPrint((PCMCIA_DEBUG_INFO, "pdo %08x Device Id=%s\n", Pdo, deviceId));
  216. RtlInitAnsiString(&ansiId, deviceId);
  217. status = RtlAnsiStringToUnicodeString(DeviceId,
  218. &ansiId,
  219. TRUE);
  220. ExFreePool(deviceId);
  221. return status;
  222. }
  223. NTSTATUS
  224. PcmciaGetHardwareIds(
  225. IN PDEVICE_OBJECT Pdo,
  226. IN ULONG FunctionNumber,
  227. OUT PUNICODE_STRING HardwareIds
  228. )
  229. /*++
  230. Routine Description:
  231. This routine generates the hardware id's for the given PC-Card and returns them
  232. as a Unicode multi-string.
  233. Hardware ids for PC-Cards are:
  234. 1. The device id of the PC-Card
  235. 2. The device id of the PC-Card with the CRC replaced with the Manufacturer code and
  236. Manufacturer info fields obtained from the tuple information on the PC-Card
  237. These hardware id's are compatible with win 9x hardware ids
  238. Arguments:
  239. Pdo - Pointer to device object representing the PC-Card
  240. FunctionNumber - Function number of the function in a multi-function card.
  241. If this is PCMCIA_MULTIFUNCTION_PARENT, then the requested hardware id
  242. is for the parent device - not for any individual function
  243. HardwareIds - Pointer to the unicode string which contains the hardware id's as a multi-string
  244. Return value:
  245. --*/
  246. {
  247. PPDO_EXTENSION pdoExtension=Pdo->DeviceExtension;
  248. PSOCKET socket = pdoExtension->Socket;
  249. PSOCKET_DATA socketData = pdoExtension->SocketData;
  250. NTSTATUS status;
  251. PSTR strings[4] = {NULL};
  252. PUCHAR hwId, hwId2;
  253. UCHAR deviceType;
  254. UCHAR stringCount = 0;
  255. PAGED_CODE();
  256. if (!socketData) {
  257. ASSERT (socketData);
  258. return STATUS_DEVICE_NOT_READY;
  259. }
  260. //
  261. // get the device type for later use
  262. //
  263. status = PcmciaGetDeviceType(Pdo, FunctionNumber, &deviceType);
  264. if (!NT_SUCCESS(status)) {
  265. return status;
  266. }
  267. //
  268. // The first hardware id is identical to the device id
  269. // Generate the device id
  270. //
  271. status = PcmciaGenerateDeviceId(socketData,
  272. FunctionNumber,
  273. &strings[stringCount++]);
  274. if (!NT_SUCCESS(status)) {
  275. return status;
  276. }
  277. try {
  278. status = STATUS_INSUFFICIENT_RESOURCES;
  279. hwId = ExAllocatePool(PagedPool, PCMCIA_MAXIMUM_DEVICE_ID_LENGTH);
  280. if (!hwId) {
  281. leave;
  282. }
  283. strings[stringCount++] = hwId;
  284. //
  285. // The second hardware is the device id with the CRC replaced
  286. // with the manufacturer code and info
  287. //
  288. if (*(socketData->Mfg) == '\0' ) {
  289. //
  290. // No manufacturer name available
  291. //
  292. if (FunctionNumber == PCMCIA_MULTIFUNCTION_PARENT) {
  293. if (socketData->Flags & SDF_JEDEC_ID) {
  294. sprintf(hwId, "%s\\%s-%04x",
  295. PCMCIA_ID_STRING,
  296. PCMCIA_MEMORY_ID_STRING,
  297. socketData->JedecId);
  298. } else {
  299. sprintf(hwId, "%s\\%s-%04X-%04X",
  300. PCMCIA_ID_STRING,
  301. PCMCIA_UNKNOWN_MANUFACTURER_STRING,
  302. socketData->ManufacturerCode,
  303. socketData->ManufacturerInfo);
  304. }
  305. } else {
  306. sprintf(hwId, "%s\\%s-DEV%d-%04X-%04X", PCMCIA_ID_STRING,
  307. PCMCIA_UNKNOWN_MANUFACTURER_STRING,
  308. FunctionNumber,
  309. socketData->ManufacturerCode,
  310. socketData->ManufacturerInfo);
  311. }
  312. } else {
  313. UCHAR Mfg[MAX_MANFID_LENGTH];
  314. UCHAR Ident[MAX_IDENT_LENGTH];
  315. PcmciaFilterIdString(socketData->Mfg, Mfg, MAX_MANFID_LENGTH);
  316. PcmciaFilterIdString(socketData->Ident, Ident, MAX_IDENT_LENGTH);
  317. //
  318. // Here a mistake on Win2000 is forcing us to now generate two different
  319. // IDs. The intended and documented form at this point is to generate:
  320. //
  321. // PCMCIA\<mfg>-<ident>-<code>-<info>
  322. //
  323. // but Win2000 had a bug where this was generated instead:
  324. //
  325. // PCMCIA\<mfg>-<code>-<info>
  326. //
  327. // So now we generate both in case someone started using the bogus format.
  328. //
  329. hwId2 = ExAllocatePool(PagedPool, PCMCIA_MAXIMUM_DEVICE_ID_LENGTH);
  330. if (!hwId2) {
  331. leave;
  332. }
  333. strings[stringCount++] = hwId2;
  334. if (FunctionNumber == PCMCIA_MULTIFUNCTION_PARENT) {
  335. sprintf(hwId, "%s\\%s-%s-%04X-%04X", PCMCIA_ID_STRING,
  336. Mfg,
  337. Ident,
  338. socketData->ManufacturerCode,
  339. socketData->ManufacturerInfo);
  340. sprintf(hwId2, "%s\\%s-%04X-%04X", PCMCIA_ID_STRING,
  341. Mfg,
  342. socketData->ManufacturerCode,
  343. socketData->ManufacturerInfo);
  344. } else {
  345. sprintf(hwId, "%s\\%s-%s-DEV%d-%04X-%04X",
  346. PCMCIA_ID_STRING,
  347. Mfg,
  348. Ident,
  349. FunctionNumber,
  350. socketData->ManufacturerCode,
  351. socketData->ManufacturerInfo);
  352. sprintf(hwId2, "%s\\%s-DEV%d-%04X-%04X",
  353. PCMCIA_ID_STRING,
  354. Mfg,
  355. FunctionNumber,
  356. socketData->ManufacturerCode,
  357. socketData->ManufacturerInfo);
  358. }
  359. }
  360. if (deviceType == PCCARD_TYPE_ATA) {
  361. hwId = ExAllocatePool(PagedPool, PCMCIA_MAXIMUM_DEVICE_ID_LENGTH);
  362. if (!hwId) {
  363. leave;
  364. }
  365. strings[stringCount++] = hwId;
  366. sprintf(hwId, "%s\\%s",
  367. PCMCIA_ID_STRING,
  368. PcmciaCompatibleIds[PCCARD_TYPE_ATA]);
  369. }
  370. status = PcmciaStringsToMultiString(strings , stringCount, HardwareIds);
  371. } finally {
  372. while(stringCount != 0) {
  373. ExFreePool(strings[--stringCount]);
  374. }
  375. }
  376. return status;
  377. }
  378. NTSTATUS
  379. PcmciaGetCompatibleIds(
  380. IN PDEVICE_OBJECT Pdo,
  381. IN ULONG FunctionNumber,
  382. OUT PUNICODE_STRING CompatibleIds
  383. )
  384. /*++
  385. Routine Description:
  386. This routine returns the compatible ids for the given PC-Card.
  387. Compatible id's are generated based on the Function Id of the PC-Card
  388. obtained from the CISTPL_FUNCID in the CIS tuple info. on the PC-Card.
  389. A table lookup is done based on the CISTPL_FUNCID to obtain the compatible id
  390. This compatible id is identical to the win 9x generated compatible ids
  391. Arguments:
  392. Pdo - Pointer to the device object representing the PC-Card
  393. FunctionNumber - Function number of the function in a multi-function card.
  394. If this is PCMCIA_MULTIFUNCTION_PARENT, then the requested compatibleid
  395. is for the parent device - not for any individual function
  396. CompatibleIds - Pointer to the unicode string which would contain the compatible ids
  397. as a multi-string on return
  398. Return value:
  399. STATUS_SUCCESS
  400. Any other status - could not generate compatible ids
  401. --*/
  402. {
  403. UCHAR deviceType ;
  404. NTSTATUS status;
  405. PCSTR strings[1] = {""};
  406. PAGED_CODE();
  407. status = PcmciaGetDeviceType(Pdo, FunctionNumber, &deviceType);
  408. if (!NT_SUCCESS(status)) {
  409. return status;
  410. }
  411. if ((deviceType == PCCARD_TYPE_RESERVED) ||
  412. (deviceType > PCMCIA_MAX_DEVICE_TYPE_SUPPORTED)) {
  413. status = PcmciaStringsToMultiString(strings, 1, CompatibleIds);
  414. } else {
  415. status = PcmciaStringsToMultiString(&PcmciaCompatibleIds[deviceType], 1, CompatibleIds);
  416. }
  417. return status;
  418. }
  419. NTSTATUS
  420. PcmciaGetDeviceType(
  421. IN PDEVICE_OBJECT Pdo,
  422. IN ULONG FunctionNumber,
  423. OUT PUCHAR DeviceType
  424. )
  425. /*++
  426. Routine Description:
  427. This routine returns the device type for the given PC-Card.
  428. device type is obtained from the CISTPL_FUNCID in the CIS tuple info. on the PC-Card.
  429. Arguments:
  430. Pdo - Pointer to the device object representing the PC-Card
  431. FunctionNumber - Function number of the function in a multi-function card.
  432. If this is PCMCIA_MULTIFUNCTION_PARENT, then the requested compatibleid
  433. is for the parent device - not for any individual function
  434. Return value:
  435. device type
  436. --*/
  437. {
  438. UCHAR deviceType ;
  439. PPDO_EXTENSION pdoExtension;
  440. PAGED_CODE();
  441. pdoExtension = Pdo->DeviceExtension;
  442. if (IsDeviceMultifunction(pdoExtension)) {
  443. if (FunctionNumber == PCMCIA_MULTIFUNCTION_PARENT) {
  444. //
  445. // This is for the root multifunction pc-card
  446. //
  447. deviceType = PCCARD_TYPE_MULTIFUNCTION3;
  448. } else {
  449. //
  450. // This is for the individual multifunction child
  451. //
  452. PSOCKET_DATA socketData;
  453. ULONG index;
  454. for (socketData = pdoExtension->SocketData, index = 0; (socketData != NULL);
  455. socketData = socketData->Next,index++) {
  456. if (socketData->Function == FunctionNumber) {
  457. //
  458. // Found the child;
  459. //
  460. break;
  461. }
  462. }
  463. if (!socketData) {
  464. ASSERT (socketData);
  465. return STATUS_DEVICE_NOT_READY;
  466. }
  467. deviceType = socketData->DeviceType;
  468. }
  469. } else {
  470. //
  471. // This is a run-of-the mill single function card
  472. //
  473. deviceType = pdoExtension->SocketData->DeviceType;
  474. }
  475. *DeviceType = deviceType;
  476. return STATUS_SUCCESS;
  477. }
  478. NTSTATUS
  479. PcmciaStringsToMultiString(
  480. IN PCSTR * Strings,
  481. IN ULONG Count,
  482. IN PUNICODE_STRING MultiString
  483. )
  484. /*++
  485. Routine Description:
  486. This routine formats a set of supplied strings into a multi string format, terminating
  487. it with a double '\0' character
  488. Arguments:
  489. Strings - Pointer to an array of strings
  490. Count - Number of strings in the supplied array which are packed into the multi-string
  491. MultiString - Pointer to the Unicode string which packs the supplied string as a multi-string
  492. terminated by double NULL
  493. Return value:
  494. STATUS_SUCCESS
  495. STATUS_INSUFFICIENT_RESOURCES - Could not allocate memory for the multi-string
  496. --*/
  497. {
  498. ULONG i, multiStringLength=0;
  499. UNICODE_STRING tempMultiString;
  500. PCSTR * currentString;
  501. ANSI_STRING ansiString;
  502. NTSTATUS status;
  503. ASSERT (MultiString->Buffer == NULL);
  504. for (i = Count, currentString = Strings; i > 0;i--, currentString++) {
  505. RtlInitAnsiString(&ansiString, *currentString);
  506. multiStringLength += RtlAnsiStringToUnicodeSize(&ansiString);
  507. }
  508. ASSERT(multiStringLength != 0);
  509. multiStringLength += sizeof(WCHAR);
  510. MultiString->Buffer = ExAllocatePool(PagedPool, multiStringLength);
  511. if (MultiString->Buffer == NULL) {
  512. return STATUS_INSUFFICIENT_RESOURCES;
  513. }
  514. MultiString->MaximumLength = (USHORT) multiStringLength;
  515. MultiString->Length = (USHORT) multiStringLength;
  516. tempMultiString = *MultiString;
  517. for (i = Count, currentString = Strings; i > 0;i--, currentString++) {
  518. RtlInitAnsiString(&ansiString, *currentString);
  519. status = RtlAnsiStringToUnicodeString(&tempMultiString,
  520. &ansiString,
  521. FALSE);
  522. ASSERT(NT_SUCCESS(status));
  523. ((PSTR) tempMultiString.Buffer) += tempMultiString.Length + sizeof(WCHAR);
  524. };
  525. //
  526. // Add one more NULL to terminate the multi string
  527. //
  528. RtlZeroMemory(tempMultiString.Buffer, sizeof(WCHAR));
  529. return STATUS_SUCCESS;
  530. }
  531. NTSTATUS
  532. PcmciaGetInstanceId(
  533. IN PDEVICE_OBJECT Pdo,
  534. OUT PUNICODE_STRING InstanceId
  535. )
  536. /*++
  537. Routine Description:
  538. This routine generates a unique instance id (1 upwards) for the supplied
  539. PC-Card which is guaranteed not to clash with any other instance ids under
  540. the same pcmcia controller, for the same type of card.
  541. A new instance id is computed only if it was not already present for the PC-Card.
  542. Arguments:
  543. Pdo - Pointer to the device object representing the PC-Card
  544. InstanceId - Pointer to a unicode string which will contain the generated
  545. instance id.
  546. Memory for the unicode string allocated by this routine.
  547. Caller's responsibility to free it .
  548. Return value:
  549. STATUS_SUCCESS
  550. STATUS_UNSUCCESSFUL - Currently there's a cap on the maximum value of instance id - 999999
  551. This status returned only if more than 999999 PC-Cards exist under
  552. this PCMCIA controller!
  553. Any other status - Something failed in the string allocation/conversion
  554. --*/
  555. {
  556. PPDO_EXTENSION pdoExtension=Pdo->DeviceExtension;
  557. PSOCKET socket = pdoExtension->Socket;
  558. PSOCKET_DATA socketData = pdoExtension->SocketData;
  559. ULONG instance;
  560. NTSTATUS status;
  561. ANSI_STRING sizeString;
  562. ASSERT(InstanceId);
  563. if (!socketData) {
  564. return STATUS_DEVICE_NOT_READY;
  565. }
  566. //
  567. // Allocate memory for the unicode string
  568. // Maximum of 6 digits in the instance..
  569. //
  570. RtlInitAnsiString(&sizeString, "123456");
  571. status = RtlAnsiStringToUnicodeString(InstanceId, &sizeString, TRUE);
  572. if (!NT_SUCCESS(status)) {
  573. return status;
  574. }
  575. //
  576. // Don't recompute instance if it's already present
  577. //
  578. if (socketData->Instance) {
  579. status = RtlIntegerToUnicodeString(socketData->Instance, 10, InstanceId);
  580. } else {
  581. KIRQL OldIrql;
  582. //
  583. // Synchronize access to prevent two identical ids/instances
  584. //
  585. KeAcquireSpinLock(&PcmciaGlobalLock, &OldIrql);
  586. //
  587. // assume failure
  588. //
  589. status = STATUS_UNSUCCESSFUL;
  590. for (instance = 1; instance <= PCMCIA_MAX_INSTANCE; instance++) {
  591. if (PcmciaCheckInstance(pdoExtension->DeviceId,
  592. instance)) {
  593. socketData->Instance = instance;
  594. break;
  595. }
  596. }
  597. KeReleaseSpinLock(&PcmciaGlobalLock, OldIrql);
  598. if (socketData->Instance) {
  599. status = RtlIntegerToUnicodeString(instance, 10, InstanceId);
  600. }
  601. }
  602. if (!NT_SUCCESS(status)) {
  603. RtlFreeUnicodeString(InstanceId);
  604. }
  605. return status;
  606. }
  607. BOOLEAN
  608. PcmciaCheckInstance(
  609. IN PUCHAR DeviceId,
  610. IN ULONG Instance
  611. )
  612. /*++
  613. Routine Description:
  614. This routine checks to see if the supplied instance id clashes with any other PC-card
  615. with the same device id
  616. Arguments:
  617. SocketList - Pointer to the list of sockets on the PCMCIA controller
  618. DeviceId - Pointer to the device id of the PC-Card for which the Instance Id is being checked
  619. Instance - Instance Id which needs to be verified
  620. Return value:
  621. TRUE - Instance is unique for the given DeviceId and may be used
  622. FALSE - Instance clashes with another instance id for the same device id
  623. --*/
  624. {
  625. PPDO_EXTENSION pdoExtension;
  626. PFDO_EXTENSION fdoExtension;
  627. PSOCKET_DATA socketData;
  628. PDEVICE_OBJECT fdo, pdo;
  629. for (fdo = FdoList; fdo != NULL; fdo = fdoExtension->NextFdo) {
  630. fdoExtension = fdo->DeviceExtension;
  631. ASSERT (fdoExtension);
  632. if (!IsDeviceStarted(fdoExtension)) {
  633. continue;
  634. }
  635. for (pdo = fdoExtension->PdoList; pdo != NULL; pdo = pdoExtension->NextPdoInFdoChain) {
  636. pdoExtension = pdo->DeviceExtension;
  637. socketData = pdoExtension->SocketData;
  638. if (IsDevicePhysicallyRemoved(pdoExtension)) {
  639. //
  640. // going to be removed soon
  641. //
  642. continue;
  643. }
  644. if (!socketData) {
  645. //
  646. // socketData already cleaned up
  647. //
  648. continue;
  649. }
  650. //
  651. // If an instance has not
  652. // been assigned yet to this card, skip
  653. //
  654. if (socketData->Instance == 0) {
  655. continue;
  656. }
  657. //
  658. // If this socket's device id matches the given socket's device id
  659. // compare the instances: if equal, then this instance is not ok.
  660. //
  661. //
  662. if ((pdoExtension->DeviceId == NULL) || (DeviceId == NULL)) {
  663. continue;
  664. }
  665. if ((strcmp(pdoExtension->DeviceId, DeviceId)==0) &&
  666. (socketData->Instance == Instance)) {
  667. return FALSE;
  668. }
  669. }
  670. }
  671. //
  672. // Instance is ok and unique
  673. //
  674. return TRUE;
  675. }
  676. VOID
  677. PcmciaFilterIdString(
  678. IN PUCHAR pIn,
  679. OUT PUCHAR pOut,
  680. ULONG MaxLen
  681. )
  682. /*++
  683. Filters out characters that shouldn't appear in device id's
  684. Arguments:
  685. pIn - pointer to input string
  686. pOut - pointer to output string
  687. MaxLen - size of buffers
  688. Return Value
  689. none
  690. --*/
  691. {
  692. ULONG i;
  693. for (i=0; i < MaxLen; i++) {
  694. if (*pIn == 0) {
  695. *pOut = 0;
  696. break;
  697. }
  698. if (*pIn >= ' ' && *pIn < 0x7F) {
  699. *pOut++ = *pIn++;
  700. } else {
  701. pIn++;
  702. }
  703. }
  704. }