1 /* 2 * COPYRIGHT: See COPYING in the top level directory 3 * PROJECT: ReactOS Kernel Streaming 4 * FILE: drivers/ksfilter/ks/swenum.c 5 * PURPOSE: KS Software BUS functions 6 * PROGRAMMER: Johannes Anderwald 7 */ 8 9 #include "precomp.h" 10 11 #include <stdio.h> 12 #include <swenum.h> 13 14 #define NDEBUG 15 #include <debug.h> 16 17 LONG KsDeviceCount = 0; 18 19 typedef NTSTATUS (NTAPI *PKSP_BUS_ENUM_CALLBACK)( 20 IN PHANDLE hKey, 21 IN PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension, 22 IN PBUS_DEVICE_ENTRY DummyEntry, 23 IN LPWSTR RootName, 24 IN LPWSTR DirectoryName); 25 26 NTSTATUS 27 KspCreatePDO( 28 IN PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension, 29 IN PBUS_DEVICE_ENTRY DeviceEntry, 30 OUT PDEVICE_OBJECT * OutDeviceObject) 31 { 32 PDEVICE_OBJECT DeviceObject; 33 WCHAR Buffer[50]; 34 ULONG CurDeviceId; 35 UNICODE_STRING DeviceName; 36 NTSTATUS Status; 37 PCOMMON_DEVICE_EXTENSION DeviceExtension; 38 39 /* increment device count */ 40 CurDeviceId = InterlockedIncrement(&KsDeviceCount); 41 42 /* generate new device id */ 43 swprintf(Buffer, L"\\Device\\KSENUM%08x", CurDeviceId); 44 45 /* initialize new device name */ 46 RtlInitUnicodeString(&DeviceName, Buffer); 47 48 /* create new device object */ 49 Status = IoCreateDevice(BusDeviceExtension->BusDeviceObject->DriverObject, sizeof(PVOID), &DeviceName, FILE_DEVICE_UNKNOWN, 0, FALSE, &DeviceObject); 50 51 /* check for success */ 52 if (!NT_SUCCESS(Status)) 53 { 54 /* failed to create pdo */ 55 return Status; 56 } 57 58 /* now allocate device extension */ 59 DeviceExtension = (PCOMMON_DEVICE_EXTENSION)AllocateItem(NonPagedPool, sizeof(COMMON_DEVICE_EXTENSION)); 60 if (!DeviceExtension) 61 { 62 /* no memory */ 63 IoDeleteDevice(DeviceObject); 64 return STATUS_INSUFFICIENT_RESOURCES; 65 } 66 67 /* store device extension */ 68 *((PVOID*)DeviceObject->DeviceExtension) = DeviceExtension; 69 70 /* initialize device extension */ 71 DeviceExtension->IsBus = FALSE; 72 DeviceExtension->DeviceObject = DeviceObject; 73 DeviceExtension->DeviceEntry = DeviceEntry; 74 DeviceExtension->BusDeviceExtension = BusDeviceExtension; 75 76 /* not started yet*/ 77 DeviceEntry->DeviceState = NotStarted; 78 79 /* get current time */ 80 KeQueryTickCount(&DeviceEntry->TimeCreated); 81 82 /* setup flags */ 83 DeviceObject->Flags |= DO_POWER_PAGABLE; 84 DeviceObject->Flags &= ~ DO_DEVICE_INITIALIZING; 85 /* TODO: fire time when expired */ 86 87 *OutDeviceObject = DeviceObject; 88 89 return STATUS_SUCCESS; 90 } 91 92 NTSTATUS 93 KspRegisterDeviceAssociation( 94 IN PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension, 95 IN PBUS_DEVICE_ENTRY DeviceEntry, 96 IN OUT PBUS_INSTANCE_ENTRY BusInstanceEntry) 97 { 98 NTSTATUS Status; 99 UNICODE_STRING ReferenceString; 100 101 /* initialize reference string */ 102 RtlInitUnicodeString(&ReferenceString, DeviceEntry->DeviceName); 103 104 /* register device interface */ 105 Status = IoRegisterDeviceInterface(BusDeviceExtension->PhysicalDeviceObject, &BusInstanceEntry->InterfaceGuid, &ReferenceString, &BusInstanceEntry->SymbolicLink); 106 107 /* check for success */ 108 if (!NT_SUCCESS(Status)) 109 { 110 /* failed */ 111 return Status; 112 } 113 114 /* now enable the interface */ 115 Status = IoSetDeviceInterfaceState(&BusInstanceEntry->SymbolicLink, TRUE); 116 117 /* check for success */ 118 if (!NT_SUCCESS(Status)) 119 { 120 /* failed, free memory */ 121 FreeItem(BusInstanceEntry->SymbolicLink.Buffer); 122 return Status; 123 } 124 125 DPRINT("Registered DeviceInterface %wZ\n", &BusInstanceEntry->SymbolicLink); 126 127 128 /* done */ 129 return Status; 130 } 131 132 VOID 133 KspRemoveDeviceAssociations( 134 IN PBUS_DEVICE_ENTRY DeviceEntry) 135 { 136 PLIST_ENTRY Entry; 137 PBUS_INSTANCE_ENTRY CurEntry; 138 139 /* remove all entries */ 140 Entry = DeviceEntry->DeviceInterfaceList.Flink; 141 142 while(Entry != &DeviceEntry->DeviceInterfaceList) 143 { 144 /* get offset */ 145 CurEntry = (PBUS_INSTANCE_ENTRY)CONTAINING_RECORD(Entry, BUS_INSTANCE_ENTRY, Entry); 146 147 /* sanity check */ 148 ASSERT(CurEntry->SymbolicLink.Buffer); 149 150 /* de-register interface */ 151 IoSetDeviceInterfaceState(&CurEntry->SymbolicLink, FALSE); 152 153 /* free symbolic link buffer */ 154 FreeItem(CurEntry->SymbolicLink.Buffer); 155 156 /* remove entry from list */ 157 RemoveEntryList(Entry); 158 159 /* move to next entry */ 160 Entry = Entry->Flink; 161 162 /* free entry */ 163 FreeItem(CurEntry); 164 } 165 } 166 167 NTSTATUS 168 KspEnumerateBusRegistryKeys( 169 IN HANDLE hKey, 170 IN LPWSTR ReferenceString, 171 IN PKSP_BUS_ENUM_CALLBACK Callback, 172 IN PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension, 173 IN PBUS_DEVICE_ENTRY DeviceEntry) 174 { 175 UNICODE_STRING String; 176 OBJECT_ATTRIBUTES ObjectAttributes; 177 HANDLE hNewKey; 178 NTSTATUS Status; 179 ULONG ResultLength, Index, KeyInfoLength; 180 KEY_FULL_INFORMATION KeyInformation; 181 PKEY_BASIC_INFORMATION KeyInfo; 182 183 /* initialize key name */ 184 RtlInitUnicodeString(&String, ReferenceString); 185 186 /* initialize object attributes */ 187 InitializeObjectAttributes(&ObjectAttributes, &String, OBJ_KERNEL_HANDLE | OBJ_CASE_INSENSITIVE, hKey, NULL); 188 189 /* open new key */ 190 Status = ZwOpenKey(&hNewKey, GENERIC_READ, &ObjectAttributes); 191 192 /* check for success */ 193 if (!NT_SUCCESS(Status)) 194 { 195 /* failed to open key */ 196 197 return Status; 198 } 199 200 /* query key stats */ 201 Status = ZwQueryKey(hNewKey, KeyFullInformation, &KeyInformation, sizeof(KeyInformation), &ResultLength); 202 203 if (!NT_SUCCESS(Status)) 204 { 205 /* close key */ 206 ZwClose(hNewKey); 207 208 /* done */ 209 return Status; 210 } 211 212 /* calculate key info length */ 213 KeyInfoLength = KeyInformation.MaxNameLen + sizeof(KEY_BASIC_INFORMATION) + 1 * sizeof(WCHAR); 214 215 /* allocate buffer */ 216 KeyInfo = (PKEY_BASIC_INFORMATION)AllocateItem(NonPagedPool, KeyInfoLength); 217 if (!KeyInfo) 218 { 219 220 /* no memory */ 221 ZwClose(hNewKey); 222 223 /* done */ 224 return STATUS_INSUFFICIENT_RESOURCES; 225 } 226 227 /* enumerate all keys */ 228 for(Index = 0; Index < KeyInformation.SubKeys; Index++) 229 { 230 231 /* query sub key */ 232 Status = ZwEnumerateKey(hNewKey, Index, KeyBasicInformation, (PVOID)KeyInfo, KeyInfoLength, &ResultLength); 233 234 /* check for success */ 235 if (NT_SUCCESS(Status)) 236 { 237 /* perform callback */ 238 Status = Callback(hNewKey, BusDeviceExtension, DeviceEntry, KeyInfo->Name, ReferenceString); 239 240 /* should enumeration stop */ 241 if (!NT_SUCCESS(Status)) 242 break; 243 } 244 } 245 246 /* free info buffer */ 247 FreeItem(KeyInfo); 248 249 /* close key */ 250 ZwClose(hNewKey); 251 252 /* done */ 253 return Status; 254 } 255 256 NTSTATUS 257 NTAPI 258 KspCreateDeviceAssociation( 259 IN PHANDLE hKey, 260 IN PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension, 261 IN PBUS_DEVICE_ENTRY DeviceEntry, 262 IN LPWSTR InterfaceString, 263 IN LPWSTR ReferenceString) 264 { 265 GUID InterfaceGUID; 266 NTSTATUS Status; 267 PLIST_ENTRY Entry; 268 PBUS_INSTANCE_ENTRY CurEntry; 269 UNICODE_STRING DeviceName; 270 271 /* initialize interface string */ 272 RtlInitUnicodeString(&DeviceName, InterfaceString); 273 274 /* first convert device name to guid */ 275 RtlGUIDFromString(&DeviceName, &InterfaceGUID); 276 277 /* check if the device is already present */ 278 Entry = DeviceEntry->DeviceInterfaceList.Flink; 279 DPRINT("KspCreateDeviceAssociation ReferenceString %S\n", ReferenceString); 280 DPRINT("KspCreateDeviceAssociation InterfaceString %S\n", InterfaceString); 281 282 while(Entry != &DeviceEntry->DeviceInterfaceList) 283 { 284 /* get offset */ 285 CurEntry = (PBUS_INSTANCE_ENTRY)CONTAINING_RECORD(Entry, BUS_INSTANCE_ENTRY, Entry); 286 287 if (IsEqualGUIDAligned(&CurEntry->InterfaceGuid, &InterfaceGUID)) 288 { 289 /* entry already exists */ 290 return STATUS_SUCCESS; 291 } 292 293 /* move to next entry */ 294 Entry = Entry->Flink; 295 } 296 297 /* time to allocate new entry */ 298 CurEntry = (PBUS_INSTANCE_ENTRY)AllocateItem(NonPagedPool, sizeof(BUS_INSTANCE_ENTRY)); 299 300 if (!CurEntry) 301 { 302 /* no memory */ 303 return STATUS_INSUFFICIENT_RESOURCES; 304 } 305 306 /* store guid */ 307 RtlMoveMemory(&CurEntry->InterfaceGuid, &InterfaceGUID, sizeof(GUID)); 308 309 /* now register the association */ 310 Status = KspRegisterDeviceAssociation(BusDeviceExtension, DeviceEntry, CurEntry); 311 312 /* check for success */ 313 if (NT_SUCCESS(Status)) 314 { 315 /* store entry */ 316 InsertTailList(&DeviceEntry->DeviceInterfaceList, &CurEntry->Entry); 317 } 318 else 319 { 320 /* failed to associated device */ 321 FreeItem(CurEntry); 322 } 323 324 /* done */ 325 return Status; 326 } 327 328 NTSTATUS 329 NTAPI 330 KspCreateDeviceReference( 331 IN PHANDLE hKey, 332 IN PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension, 333 IN PBUS_DEVICE_ENTRY DummyEntry, 334 IN LPWSTR InterfaceId, 335 IN LPWSTR DeviceId) 336 { 337 LPWSTR DeviceName; 338 SIZE_T Length; 339 PLIST_ENTRY Entry; 340 PBUS_DEVICE_ENTRY DeviceEntry = NULL; /* GCC warning */ 341 BOOLEAN ItemExists = FALSE; 342 UNICODE_STRING String; 343 NTSTATUS Status; 344 KIRQL OldLevel; 345 346 /* first construct device name & reference guid */ 347 Length = wcslen(DeviceId) + wcslen(InterfaceId); 348 349 /* append '&' and null byte */ 350 Length += 2; 351 352 /* allocate device name */ 353 DeviceName = AllocateItem(NonPagedPool, Length * sizeof(WCHAR)); 354 355 if (!DeviceName) 356 { 357 /* not enough memory */ 358 return STATUS_INSUFFICIENT_RESOURCES; 359 } 360 361 /* construct device name */ 362 wcscpy(DeviceName, DeviceId); 363 wcscat(DeviceName, L"&"); 364 wcscat(DeviceName, InterfaceId); 365 366 /* scan list and check if it is already present */ 367 Entry = BusDeviceExtension->Common.Entry.Flink; 368 369 while(Entry != &BusDeviceExtension->Common.Entry) 370 { 371 /* get real offset */ 372 DeviceEntry = (PBUS_DEVICE_ENTRY)CONTAINING_RECORD(Entry, BUS_DEVICE_ENTRY, Entry); 373 374 /* check if name matches */ 375 if (!wcsicmp(DeviceEntry->DeviceName, DeviceName)) 376 { 377 /* item already exists */ 378 ItemExists = TRUE; 379 break; 380 } 381 382 /* move to next entry */ 383 Entry = Entry->Flink; 384 } 385 386 if (!ItemExists) 387 { 388 /* allocate new device entry */ 389 DeviceEntry = AllocateItem(NonPagedPool, sizeof(BUS_DEVICE_ENTRY)); 390 if (!DeviceEntry) 391 { 392 /* no memory */ 393 FreeItem(DeviceName); 394 return STATUS_INSUFFICIENT_RESOURCES; 395 } 396 397 /* initialize device entry */ 398 InitializeListHead(&DeviceEntry->DeviceInterfaceList); 399 InitializeListHead(&DeviceEntry->IrpPendingList); 400 401 /* copy device guid */ 402 RtlInitUnicodeString(&String, DeviceId); 403 RtlGUIDFromString(&String, &DeviceEntry->DeviceGuid); 404 405 /* copy device names */ 406 DeviceEntry->DeviceName = DeviceName; 407 DeviceEntry->Instance = (DeviceName + wcslen(DeviceId) + 1); 408 409 /* copy name */ 410 DeviceEntry->BusId = AllocateItem(NonPagedPool, (wcslen(DeviceId) + 1) * sizeof(WCHAR)); 411 if (!DeviceEntry->BusId) 412 { 413 /* no memory */ 414 FreeItem(DeviceName); 415 FreeItem(DeviceEntry); 416 return STATUS_INSUFFICIENT_RESOURCES; 417 } 418 wcscpy(DeviceEntry->BusId, DeviceId); 419 } 420 421 /* now enumerate the interfaces */ 422 Status = KspEnumerateBusRegistryKeys(hKey, InterfaceId, KspCreateDeviceAssociation, BusDeviceExtension, DeviceEntry); 423 424 /* check if list is empty */ 425 if (IsListEmpty(&DeviceEntry->DeviceInterfaceList)) 426 { 427 /* invalid device settings */ 428 FreeItem(DeviceEntry->BusId); 429 FreeItem(DeviceEntry->DeviceName); 430 FreeItem(DeviceEntry); 431 432 ASSERT(ItemExists == FALSE); 433 434 return STATUS_INVALID_DEVICE_STATE; 435 } 436 437 /* check if enumeration failed */ 438 if (!NT_SUCCESS(Status)) 439 { 440 /* failed */ 441 KspRemoveDeviceAssociations(DeviceEntry); 442 FreeItem(DeviceEntry->BusId); 443 FreeItem(DeviceEntry->DeviceName); 444 FreeItem(DeviceEntry); 445 446 ASSERT(ItemExists == FALSE); 447 448 /* done */ 449 return Status; 450 } 451 452 if (!ItemExists) 453 { 454 /* acquire lock */ 455 KeAcquireSpinLock(&BusDeviceExtension->Lock, &OldLevel); 456 457 /* successfully initialized entry */ 458 InsertTailList(&BusDeviceExtension->Common.Entry, &DeviceEntry->Entry); 459 460 /* release lock */ 461 KeReleaseSpinLock(&BusDeviceExtension->Lock, OldLevel); 462 } 463 464 /* done */ 465 return Status; 466 } 467 468 NTSTATUS 469 NTAPI 470 KspCreateDeviceReferenceTrampoline( 471 IN PHANDLE hKey, 472 IN PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension, 473 IN PBUS_DEVICE_ENTRY DummyEntry, 474 IN LPWSTR DeviceCategory, 475 IN LPWSTR ReferenceString) 476 { 477 return KspEnumerateBusRegistryKeys(hKey, DeviceCategory, KspCreateDeviceReference, BusDeviceExtension, DummyEntry); 478 } 479 480 481 NTSTATUS 482 KspOpenBusRegistryKey( 483 IN PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension, 484 OUT PHANDLE hKey) 485 { 486 OBJECT_ATTRIBUTES ObjectAttributes; 487 488 /* initialize object attributes */ 489 InitializeObjectAttributes(&ObjectAttributes, &BusDeviceExtension->ServicePath, OBJ_KERNEL_HANDLE | OBJ_CASE_INSENSITIVE, NULL, NULL); 490 491 return ZwCreateKey(hKey, GENERIC_READ | GENERIC_WRITE, &ObjectAttributes, 0, NULL, 0, NULL); 492 } 493 494 NTSTATUS 495 KspScanBus( 496 IN PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension) 497 { 498 HANDLE hKey; 499 NTSTATUS Status; 500 501 /* first open key */ 502 Status = KspOpenBusRegistryKey(BusDeviceExtension, &hKey); 503 504 /* check for success */ 505 if (!NT_SUCCESS(Status)) 506 { 507 /* no success */ 508 509 return Status; 510 } 511 512 /* TODO clear reference marks */ 513 514 /* construct device entries */ 515 Status = KspEnumerateBusRegistryKeys(hKey, NULL, KspCreateDeviceReferenceTrampoline, BusDeviceExtension, NULL); 516 517 /* TODO: delete unreferenced devices */ 518 519 /* close handle */ 520 ZwClose(hKey); 521 522 /* done */ 523 return Status; 524 } 525 526 527 NTSTATUS 528 NTAPI 529 KspBusQueryReferenceString( 530 IN PVOID Context, 531 IN OUT PWCHAR *String) 532 { 533 LPWSTR Name; 534 SIZE_T Length; 535 PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension = (PBUS_ENUM_DEVICE_EXTENSION)Context; 536 537 /* sanity checks */ 538 ASSERT(BusDeviceExtension); 539 ASSERT(BusDeviceExtension->BusIdentifier); 540 541 /* calculate length */ 542 Length = wcslen(BusDeviceExtension->BusIdentifier) + 1; 543 544 /* allocate buffer */ 545 Name = AllocateItem(PagedPool, Length * sizeof(WCHAR)); 546 547 if (!Name) 548 { 549 /* failed to allocate buffer */ 550 return STATUS_INSUFFICIENT_RESOURCES; 551 } 552 553 /* copy buffer */ 554 wcscpy(Name, BusDeviceExtension->BusIdentifier); 555 556 /* store result */ 557 *String = Name; 558 559 /* done */ 560 return STATUS_SUCCESS; 561 } 562 563 VOID 564 NTAPI 565 KspBusDeviceReference( 566 IN PVOID Context) 567 { 568 PCOMMON_DEVICE_EXTENSION ChildDeviceExtension = (PCOMMON_DEVICE_EXTENSION)Context; 569 570 /* reference count */ 571 InterlockedIncrement((PLONG)&ChildDeviceExtension->DeviceReferenceCount); 572 } 573 574 VOID 575 NTAPI 576 KspBusDeviceDereference( 577 IN PVOID Context) 578 { 579 PCOMMON_DEVICE_EXTENSION ChildDeviceExtension = (PCOMMON_DEVICE_EXTENSION)Context; 580 581 /* reference count */ 582 InterlockedDecrement((PLONG)&ChildDeviceExtension->DeviceReferenceCount); 583 } 584 585 VOID 586 NTAPI 587 KspBusReferenceDeviceObject( 588 IN PVOID Context) 589 { 590 PCOMMON_DEVICE_EXTENSION ChildDeviceExtension = (PCOMMON_DEVICE_EXTENSION)Context; 591 592 /* reference count */ 593 InterlockedIncrement((PLONG)&ChildDeviceExtension->DeviceObjectReferenceCount); 594 } 595 596 VOID 597 NTAPI 598 KspBusDereferenceDeviceObject( 599 IN PVOID Context) 600 { 601 PCOMMON_DEVICE_EXTENSION ChildDeviceExtension = (PCOMMON_DEVICE_EXTENSION)Context; 602 603 /* reference count */ 604 InterlockedDecrement((PLONG)&ChildDeviceExtension->DeviceObjectReferenceCount); 605 } 606 607 NTSTATUS 608 KspQueryBusDeviceInterface( 609 IN PCOMMON_DEVICE_EXTENSION ChildDeviceExtension, 610 IN PIRP Irp) 611 { 612 PBUS_INTERFACE_SWENUM Interface; 613 PIO_STACK_LOCATION IoStack; 614 615 /* get current irp stack location */ 616 IoStack = IoGetCurrentIrpStackLocation(Irp); 617 618 /* sanity checks */ 619 ASSERT(IoStack->Parameters.QueryInterface.Size == sizeof(BUS_INTERFACE_SWENUM)); 620 ASSERT(IoStack->Parameters.QueryInterface.Interface); 621 622 /* fill in interface */ 623 Interface = (PBUS_INTERFACE_SWENUM)IoStack->Parameters.QueryInterface.Interface; 624 Interface->Interface.Size = sizeof(BUS_INTERFACE_SWENUM); 625 Interface->Interface.Version = BUS_INTERFACE_SWENUM_VERSION; 626 Interface->Interface.Context = ChildDeviceExtension; 627 Interface->Interface.InterfaceReference = KspBusDeviceReference; 628 Interface->Interface.InterfaceDereference = KspBusDeviceDereference; 629 Interface->ReferenceDeviceObject = KspBusReferenceDeviceObject; 630 Interface->DereferenceDeviceObject = KspBusDereferenceDeviceObject; 631 Interface->QueryReferenceString = KspBusQueryReferenceString; 632 633 return STATUS_SUCCESS; 634 } 635 636 NTSTATUS 637 KspEnableBusDeviceInterface( 638 PBUS_DEVICE_ENTRY DeviceEntry, 639 BOOLEAN bEnable) 640 { 641 PLIST_ENTRY Entry; 642 PBUS_INSTANCE_ENTRY InstanceEntry; 643 NTSTATUS Status = STATUS_SUCCESS; 644 645 /* enable now all interfaces */ 646 Entry = DeviceEntry->DeviceInterfaceList.Flink; 647 648 while(Entry != &DeviceEntry->DeviceInterfaceList) 649 { 650 /* get bus instance entry */ 651 InstanceEntry = (PBUS_INSTANCE_ENTRY)CONTAINING_RECORD(Entry, BUS_INSTANCE_ENTRY, Entry); 652 DPRINT("Enabling %u %wZ Irql %u\n", bEnable, &InstanceEntry->SymbolicLink, KeGetCurrentIrql()); 653 654 /* set interface state */ 655 Status = IoSetDeviceInterfaceState(&InstanceEntry->SymbolicLink, bEnable); 656 657 if (!NT_SUCCESS(Status)) 658 { 659 /* failed to set interface */ 660 break; 661 } 662 663 /* move to next entry */ 664 Entry = Entry->Flink; 665 } 666 667 /* done */ 668 return Status; 669 } 670 671 NTSTATUS 672 KspDoReparseForIrp( 673 PIRP Irp, 674 PBUS_DEVICE_ENTRY DeviceEntry) 675 { 676 SIZE_T Length; 677 LPWSTR Buffer; 678 PIO_STACK_LOCATION IoStack; 679 680 /* get stack location */ 681 IoStack = IoGetCurrentIrpStackLocation(Irp); 682 683 /* sanity checks */ 684 ASSERT(DeviceEntry->PDODeviceName); 685 ASSERT(DeviceEntry->Instance); 686 ASSERT(IoStack->FileObject); 687 ASSERT(IoStack->FileObject->FileName.Buffer); 688 689 /* calculate length */ 690 Length = wcslen(DeviceEntry->PDODeviceName); 691 Length += wcslen(DeviceEntry->Instance); 692 693 /* zero byte and '\\' */ 694 Length += 2; 695 696 /* allocate buffer */ 697 Buffer = ExAllocatePoolWithTag(NonPagedPool, Length * sizeof(WCHAR), 'mNoI'); 698 if (!Buffer) 699 { 700 /* no resources */ 701 return STATUS_INSUFFICIENT_RESOURCES; 702 } 703 704 /* construct buffer */ 705 swprintf(Buffer, L"%s\\%s", DeviceEntry->PDODeviceName, DeviceEntry->Instance); 706 707 /* free old buffer*/ 708 ExFreePoolWithTag(IoStack->FileObject->FileName.Buffer, 'mNoI'); 709 710 /* store new file name */ 711 RtlInitUnicodeString(&IoStack->FileObject->FileName, Buffer); 712 713 /* done */ 714 return STATUS_REPARSE; 715 } 716 717 VOID 718 KspCompletePendingIrps( 719 IN PBUS_DEVICE_ENTRY DeviceEntry, 720 IN OUT NTSTATUS ResultCode) 721 { 722 PLIST_ENTRY Entry; 723 PIRP Irp; 724 NTSTATUS Status; 725 726 /* go through list */ 727 while(!IsListEmpty(&DeviceEntry->IrpPendingList)) 728 { 729 /* get first entry */ 730 Entry = RemoveHeadList(&DeviceEntry->IrpPendingList); 731 732 /* get irp */ 733 Irp = (PIRP)CONTAINING_RECORD(Entry, IRP, Tail.Overlay.ListEntry); 734 735 if (ResultCode == STATUS_REPARSE) 736 { 737 /* construct reparse information */ 738 Status = KspDoReparseForIrp(Irp, DeviceEntry); 739 } 740 else 741 { 742 /* use default code */ 743 Status = ResultCode; 744 } 745 746 /* store result code */ 747 Irp->IoStatus.Status = Status; 748 749 DPRINT("Completing IRP %p Status %x\n", Irp, Status); 750 751 /* complete the request */ 752 CompleteRequest(Irp, IO_NO_INCREMENT); 753 } 754 755 } 756 757 758 759 NTSTATUS 760 KspStartBusDevice( 761 IN PDEVICE_OBJECT DeviceObject, 762 IN PCOMMON_DEVICE_EXTENSION ChildDeviceExtension, 763 IN PIRP Irp) 764 { 765 WCHAR PDOName[256]; 766 NTSTATUS Status; 767 ULONG ResultLength; 768 LPWSTR Name; 769 ULONG NameLength; 770 PBUS_DEVICE_ENTRY DeviceEntry; 771 772 /* FIXME handle pending remove */ 773 774 /* get full device name */ 775 Status = IoGetDeviceProperty(DeviceObject, DevicePropertyPhysicalDeviceObjectName, sizeof(PDOName), PDOName, &ResultLength); 776 777 if (!NT_SUCCESS(Status)) 778 { 779 /* failed to get device name */ 780 return Status; 781 } 782 783 /* allocate device name buffer */ 784 NameLength = ResultLength + sizeof(UNICODE_NULL); 785 Name = AllocateItem(NonPagedPool, NameLength); 786 if (!Name) 787 { 788 /* no memory */ 789 return STATUS_INSUFFICIENT_RESOURCES; 790 } 791 792 /* copy name */ 793 NT_VERIFY(NT_SUCCESS(RtlStringCbCopyW(Name, NameLength, PDOName))); 794 795 /* TODO: time stamp creation time */ 796 797 /* get device entry */ 798 DeviceEntry = (PBUS_DEVICE_ENTRY)ChildDeviceExtension->DeviceEntry; 799 800 /* sanity check */ 801 ASSERT(DeviceEntry); 802 803 /* store device name */ 804 DeviceEntry->PDODeviceName = Name; 805 806 /* mark device as started */ 807 DeviceEntry->DeviceState = Started; 808 809 /* reference start time */ 810 KeQueryTickCount(&DeviceEntry->TimeCreated); 811 812 DPRINT1("KspStartBusDevice Name %S DeviceName %S Instance %S Started\n", Name, DeviceEntry->DeviceName, DeviceEntry->Instance); 813 814 /* enable device classes */ 815 //KspEnableBusDeviceInterface(DeviceEntry, TRUE); 816 817 /* done */ 818 return STATUS_SUCCESS; 819 } 820 821 NTSTATUS 822 KspQueryBusDeviceCapabilities( 823 IN PCOMMON_DEVICE_EXTENSION ChildDeviceExtension, 824 IN PIRP Irp) 825 { 826 PDEVICE_CAPABILITIES Capabilities; 827 PIO_STACK_LOCATION IoStack; 828 829 /* get stack location */ 830 IoStack = IoGetCurrentIrpStackLocation(Irp); 831 832 /* get capabilities */ 833 Capabilities = IoStack->Parameters.DeviceCapabilities.Capabilities; 834 835 RtlZeroMemory(Capabilities, sizeof(DEVICE_CAPABILITIES)); 836 837 /* setup capabilities */ 838 Capabilities->UniqueID = TRUE; 839 Capabilities->SilentInstall = TRUE; 840 Capabilities->SurpriseRemovalOK = TRUE; 841 Capabilities->Address = 0; 842 Capabilities->UINumber = 0; 843 Capabilities->SystemWake = PowerSystemWorking; /* FIXME common device extension */ 844 Capabilities->DeviceWake = PowerDeviceD0; 845 846 /* done */ 847 return STATUS_SUCCESS; 848 } 849 850 NTSTATUS 851 KspQueryBusInformation( 852 IN PCOMMON_DEVICE_EXTENSION ChildDeviceExtension, 853 IN PIRP Irp) 854 { 855 PPNP_BUS_INFORMATION BusInformation; 856 857 /* allocate bus information */ 858 BusInformation = (PPNP_BUS_INFORMATION)AllocateItem(PagedPool, sizeof(PNP_BUS_INFORMATION)); 859 860 if (!BusInformation) 861 { 862 /* no memory */ 863 return STATUS_INSUFFICIENT_RESOURCES; 864 } 865 866 /* return info */ 867 BusInformation->BusNumber = 0; 868 BusInformation->LegacyBusType = InterfaceTypeUndefined; 869 RtlMoveMemory(&BusInformation->BusTypeGuid, &KSMEDIUMSETID_Standard, sizeof(GUID)); 870 871 /* store result */ 872 Irp->IoStatus.Information = (ULONG_PTR)BusInformation; 873 874 /* done */ 875 return STATUS_SUCCESS; 876 } 877 878 NTSTATUS 879 KspQueryBusDevicePnpState( 880 IN PCOMMON_DEVICE_EXTENSION ChildDeviceExtension, 881 IN PIRP Irp) 882 { 883 /* set device flags */ 884 Irp->IoStatus.Information = PNP_DEVICE_DONT_DISPLAY_IN_UI | PNP_DEVICE_NOT_DISABLEABLE; 885 886 /* done */ 887 return STATUS_SUCCESS; 888 } 889 890 NTSTATUS 891 KspQueryId( 892 IN PCOMMON_DEVICE_EXTENSION ChildDeviceExtension, 893 IN PIRP Irp) 894 { 895 PIO_STACK_LOCATION IoStack; 896 PBUS_DEVICE_ENTRY DeviceEntry; 897 PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension; 898 LPWSTR Name; 899 SIZE_T Length; 900 901 /* get current irp stack location */ 902 IoStack = IoGetCurrentIrpStackLocation(Irp); 903 904 if (IoStack->Parameters.QueryId.IdType == BusQueryInstanceID) 905 { 906 /* get device entry */ 907 DeviceEntry = (PBUS_DEVICE_ENTRY) ChildDeviceExtension->DeviceEntry; 908 909 /* sanity check */ 910 ASSERT(DeviceEntry); 911 ASSERT(DeviceEntry->Instance); 912 913 /* calculate length */ 914 Length = wcslen(DeviceEntry->Instance) + 2; 915 916 /* allocate buffer */ 917 Name = AllocateItem(PagedPool, Length * sizeof(WCHAR)); 918 919 if (!Name) 920 { 921 /* failed to allocate buffer */ 922 return STATUS_INSUFFICIENT_RESOURCES; 923 } 924 925 /* copy buffer */ 926 wcscpy(Name, DeviceEntry->Instance); 927 928 /* store result */ 929 Irp->IoStatus.Information = (ULONG_PTR)Name; 930 931 /* done */ 932 return STATUS_SUCCESS; 933 } 934 else if (IoStack->Parameters.QueryId.IdType == BusQueryDeviceID || 935 IoStack->Parameters.QueryId.IdType == BusQueryHardwareIDs) 936 { 937 /* get device entry */ 938 DeviceEntry = (PBUS_DEVICE_ENTRY) ChildDeviceExtension->DeviceEntry; 939 940 /* get bus device extension */ 941 BusDeviceExtension = (PBUS_ENUM_DEVICE_EXTENSION) ChildDeviceExtension->BusDeviceExtension; 942 943 /* sanity check */ 944 ASSERT(DeviceEntry); 945 ASSERT(DeviceEntry->BusId); 946 ASSERT(BusDeviceExtension); 947 ASSERT(BusDeviceExtension->BusIdentifier); 948 949 /* calculate length */ 950 Length = wcslen(BusDeviceExtension->BusIdentifier); 951 Length += wcslen(DeviceEntry->BusId); 952 953 /* extra length for '\\' and 2 zero bytes */ 954 Length += 4; 955 956 /* allocate buffer */ 957 Name = AllocateItem(PagedPool, Length * sizeof(WCHAR)); 958 if (!Name) 959 { 960 /* failed to allocate buffer */ 961 return STATUS_INSUFFICIENT_RESOURCES; 962 } 963 964 /* construct id */ 965 wcscpy(Name, BusDeviceExtension->BusIdentifier); 966 wcscat(Name, L"\\"); 967 wcscat(Name, DeviceEntry->BusId); 968 //swprintf(Name, L"%s\\%s", BusDeviceExtension->BusIdentifier, DeviceEntry->BusId); 969 970 /* store result */ 971 Irp->IoStatus.Information = (ULONG_PTR)Name; 972 973 /* done */ 974 return STATUS_SUCCESS; 975 } 976 else 977 { 978 /* other ids are not supported */ 979 //DPRINT1("Not Supported ID Type %x\n", IoStack->Parameters.QueryId.IdType); 980 return Irp->IoStatus.Status; 981 } 982 } 983 984 NTSTATUS 985 KspInstallInterface( 986 IN PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension, 987 IN PSWENUM_INSTALL_INTERFACE InstallInterface) 988 { 989 SIZE_T Length, Index; 990 UNICODE_STRING DeviceString, InterfaceString, ReferenceString; 991 HANDLE hKey, hDeviceKey, hInterfaceKey, hReferenceKey; 992 NTSTATUS Status; 993 OBJECT_ATTRIBUTES ObjectAttributes; 994 995 /* sanity check */ 996 ASSERT(InstallInterface); 997 998 /* calculate length */ 999 Length = wcslen(InstallInterface->ReferenceString); 1000 1001 /* check for invalid characters */ 1002 for(Index = 0; Index < Length; Index++) 1003 { 1004 if (InstallInterface->ReferenceString[Index] <= L' ' || 1005 InstallInterface->ReferenceString[Index] > L'~' || 1006 InstallInterface->ReferenceString[Index] == L',' || 1007 InstallInterface->ReferenceString[Index] == L'\\' || 1008 InstallInterface->ReferenceString[Index] == L'/') 1009 { 1010 /* invalid character */ 1011 return STATUS_INVALID_PARAMETER; 1012 } 1013 } 1014 1015 /* open bus key */ 1016 Status = KspOpenBusRegistryKey(BusDeviceExtension, &hKey); 1017 if (NT_SUCCESS(Status)) 1018 { 1019 /* convert device guid to string */ 1020 Status = RtlStringFromGUID(&InstallInterface->DeviceId, &DeviceString); 1021 if (NT_SUCCESS(Status)) 1022 { 1023 /* initialize object attributes */ 1024 InitializeObjectAttributes(&ObjectAttributes, &DeviceString, OBJ_KERNEL_HANDLE | OBJ_CASE_INSENSITIVE, hKey, NULL); 1025 1026 /* construct device key */ 1027 Status = ZwCreateKey(&hDeviceKey, GENERIC_WRITE, &ObjectAttributes, 0, NULL, 0, NULL); 1028 if (NT_SUCCESS(Status)) 1029 { 1030 /* convert interface guid to string */ 1031 Status = RtlStringFromGUID(&InstallInterface->InterfaceId, &InterfaceString); 1032 if (NT_SUCCESS(Status)) 1033 { 1034 /* initialize object attributes */ 1035 InitializeObjectAttributes(&ObjectAttributes, &InterfaceString, OBJ_KERNEL_HANDLE | OBJ_CASE_INSENSITIVE, hDeviceKey, NULL); 1036 1037 /* construct device key */ 1038 Status = ZwCreateKey(&hInterfaceKey, GENERIC_WRITE, &ObjectAttributes, 0, NULL, 0, NULL); 1039 if (NT_SUCCESS(Status)) 1040 { 1041 /* initialize reference string */ 1042 RtlInitUnicodeString(&ReferenceString, InstallInterface->ReferenceString); 1043 1044 /* initialize object attributes */ 1045 InitializeObjectAttributes(&ObjectAttributes, &ReferenceString, OBJ_KERNEL_HANDLE | OBJ_CASE_INSENSITIVE, hInterfaceKey, NULL); 1046 1047 /* construct device key */ 1048 Status = ZwCreateKey(&hReferenceKey, GENERIC_WRITE, &ObjectAttributes, 0, NULL, 0, NULL); 1049 if (NT_SUCCESS(Status)) 1050 { 1051 /* close key */ 1052 ZwClose(hReferenceKey); 1053 } 1054 } 1055 /* free interface string */ 1056 RtlFreeUnicodeString(&InterfaceString); 1057 1058 /* close reference key */ 1059 ZwClose(hInterfaceKey); 1060 } 1061 /* close device key */ 1062 ZwClose(hDeviceKey); 1063 } 1064 /* free device string */ 1065 RtlFreeUnicodeString(&DeviceString); 1066 } 1067 /* close bus key */ 1068 ZwClose(hKey); 1069 } 1070 1071 /* done */ 1072 return Status; 1073 } 1074 1075 VOID 1076 NTAPI 1077 KspInstallBusEnumInterface( 1078 IN PVOID Ctx) 1079 { 1080 PIO_STACK_LOCATION IoStack; 1081 NTSTATUS Status; 1082 PLIST_ENTRY Entry; 1083 PBUS_DEVICE_ENTRY DeviceEntry; 1084 PSWENUM_INSTALL_INTERFACE InstallInterface; 1085 KIRQL OldLevel; 1086 PBUS_INSTALL_ENUM_CONTEXT Context = (PBUS_INSTALL_ENUM_CONTEXT)Ctx; 1087 1088 /* get current irp stack location */ 1089 IoStack = IoGetCurrentIrpStackLocation(Context->Irp); 1090 1091 /* get install request */ 1092 InstallInterface = (PSWENUM_INSTALL_INTERFACE)Context->Irp->AssociatedIrp.SystemBuffer; 1093 1094 /* sanity check */ 1095 ASSERT(InstallInterface); 1096 ASSERT(Context->BusDeviceExtension); 1097 1098 if (IoStack->Parameters.DeviceIoControl.InputBufferLength < sizeof(SWENUM_INSTALL_INTERFACE)) 1099 { 1100 /* buffer too small */ 1101 Context->Status = STATUS_INVALID_PARAMETER; 1102 1103 /* signal completion */ 1104 KeSetEvent(&Context->Event, 0, FALSE); 1105 1106 /* done */ 1107 return; 1108 } 1109 1110 /* FIXME locks */ 1111 1112 /* now install the interface */ 1113 Status = KspInstallInterface(Context->BusDeviceExtension, InstallInterface); 1114 if (!NT_SUCCESS(Status)) 1115 { 1116 /* failed to install interface */ 1117 Context->Status = Status; 1118 1119 /* signal completion */ 1120 KeSetEvent(&Context->Event, 0, FALSE); 1121 1122 /* done */ 1123 return; 1124 } 1125 1126 /* now scan the bus */ 1127 Status = KspScanBus(Context->BusDeviceExtension); 1128 // FIXME: We may need to check for success here, and properly fail... 1129 ASSERT(NT_SUCCESS(Status)); 1130 1131 /* acquire device entry lock */ 1132 KeAcquireSpinLock(&Context->BusDeviceExtension->Lock, &OldLevel); 1133 1134 /* now iterate all device entries */ 1135 ASSERT(!IsListEmpty(&Context->BusDeviceExtension->Common.Entry)); 1136 Entry = Context->BusDeviceExtension->Common.Entry.Flink; 1137 while(Entry != &Context->BusDeviceExtension->Common.Entry) 1138 { 1139 /* get device entry */ 1140 DeviceEntry = (PBUS_DEVICE_ENTRY)CONTAINING_RECORD(Entry, BUS_DEVICE_ENTRY, Entry); 1141 if (IsEqualGUIDAligned(&DeviceEntry->DeviceGuid, &InstallInterface->DeviceId)) 1142 { 1143 if (!DeviceEntry->PDO) 1144 { 1145 /* release device entry lock */ 1146 KeReleaseSpinLock(&Context->BusDeviceExtension->Lock, OldLevel); 1147 1148 /* create pdo */ 1149 Status = KspCreatePDO(Context->BusDeviceExtension, DeviceEntry, &DeviceEntry->PDO); 1150 1151 /* acquire device entry lock */ 1152 KeAcquireSpinLock(&Context->BusDeviceExtension->Lock, &OldLevel); 1153 1154 /* done */ 1155 break; 1156 } 1157 } 1158 1159 /* move to next entry */ 1160 Entry = Entry->Flink; 1161 } 1162 1163 /* release device entry lock */ 1164 KeReleaseSpinLock(&Context->BusDeviceExtension->Lock, OldLevel); 1165 1166 /* signal that bus driver relations has changed */ 1167 IoInvalidateDeviceRelations(Context->BusDeviceExtension->PhysicalDeviceObject, BusRelations); 1168 1169 /* update status */ 1170 Context->Status = Status; 1171 1172 /* signal completion */ 1173 KeSetEvent(&Context->Event, 0, FALSE); 1174 } 1175 1176 1177 VOID 1178 NTAPI 1179 KspBusWorkerRoutine( 1180 IN PVOID Parameter) 1181 { 1182 PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension; 1183 PBUS_DEVICE_ENTRY DeviceEntry; 1184 PLIST_ENTRY Entry; 1185 LARGE_INTEGER Time, Diff; 1186 BOOLEAN DoInvalidate = FALSE; 1187 KIRQL OldLevel; 1188 1189 /* get device extension */ 1190 BusDeviceExtension = (PBUS_ENUM_DEVICE_EXTENSION)Parameter; 1191 1192 /* acquire lock */ 1193 KeAcquireSpinLock(&BusDeviceExtension->Lock, &OldLevel); 1194 1195 /* get current time */ 1196 KeQueryTickCount(&Time); 1197 1198 /* enumerate all device entries */ 1199 Entry = BusDeviceExtension->Common.Entry.Flink; 1200 while(Entry != &BusDeviceExtension->Common.Entry) 1201 { 1202 /* get offset to device entry */ 1203 DeviceEntry = (PBUS_DEVICE_ENTRY)CONTAINING_RECORD(Entry, BUS_DEVICE_ENTRY, Entry); 1204 1205 /* sanity check */ 1206 ASSERT(DeviceEntry); 1207 1208 //DPRINT1("DeviceEntry %p PDO %p State %x\n", DeviceEntry, DeviceEntry->PDO, DeviceEntry->DeviceState); 1209 1210 if (DeviceEntry->PDO) 1211 { 1212 if (DeviceEntry->DeviceState == NotStarted) 1213 { 1214 Diff.QuadPart = (Time.QuadPart - DeviceEntry->TimeCreated.QuadPart) * KeQueryTimeIncrement(); 1215 1216 /* wait for 15 sec */ 1217 if (Diff.QuadPart > Int32x32To64(15000, 10000)) 1218 { 1219 /* release spin lock */ 1220 KeReleaseSpinLock(&BusDeviceExtension->Lock, OldLevel); 1221 1222 DPRINT1("DeviceID %S Instance %S TimeCreated %I64u Now %I64u Diff %I64u hung\n", 1223 DeviceEntry->DeviceName, 1224 DeviceEntry->Instance, 1225 DeviceEntry->TimeCreated.QuadPart * KeQueryTimeIncrement(), 1226 Time.QuadPart * KeQueryTimeIncrement(), 1227 Diff.QuadPart); 1228 1229 /* deactivate interfaces */ 1230 //KspEnableBusDeviceInterface(DeviceEntry, FALSE); 1231 1232 /* re-acquire lock */ 1233 KeAcquireSpinLock(&BusDeviceExtension->Lock, &OldLevel); 1234 1235 /* pending remove device object */ 1236 DeviceEntry->DeviceState = StopPending; 1237 1238 /* perform invalidation */ 1239 DoInvalidate = TRUE; 1240 } 1241 } 1242 else if (DeviceEntry->DeviceState == Started) 1243 { 1244 /* release spin lock */ 1245 KeReleaseSpinLock(&BusDeviceExtension->Lock, OldLevel); 1246 1247 /* found pending irps */ 1248 KspCompletePendingIrps(DeviceEntry, STATUS_REPARSE); 1249 1250 /* re-acquire lock */ 1251 KeAcquireSpinLock(&BusDeviceExtension->Lock, &OldLevel); 1252 } 1253 } 1254 1255 1256 /* move to next */ 1257 Entry = Entry->Flink; 1258 } 1259 1260 /* release lock */ 1261 KeReleaseSpinLock(&BusDeviceExtension->Lock, OldLevel); 1262 1263 if (DoInvalidate) 1264 { 1265 /* invalidate device relations */ 1266 IoInvalidateDeviceRelations(BusDeviceExtension->PhysicalDeviceObject, BusRelations); 1267 } 1268 1269 Time.QuadPart = Int32x32To64(5000, -10000); 1270 KeSetTimer(&BusDeviceExtension->Timer, Time, &BusDeviceExtension->Dpc); 1271 } 1272 1273 VOID 1274 NTAPI 1275 KspBusDpcRoutine( 1276 IN PKDPC Dpc, 1277 IN PVOID DeferredContext OPTIONAL, 1278 IN PVOID SystemArgument1 OPTIONAL, 1279 IN PVOID SystemArgument2 OPTIONAL) 1280 { 1281 /* get device extension */ 1282 PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension = (PBUS_ENUM_DEVICE_EXTENSION)DeferredContext; 1283 1284 /* queue the item */ 1285 ExQueueWorkItem(&BusDeviceExtension->WorkItem, DelayedWorkQueue); 1286 } 1287 1288 VOID 1289 NTAPI 1290 KspRemoveBusInterface( 1291 PVOID Ctx) 1292 { 1293 PBUS_INSTALL_ENUM_CONTEXT Context =(PBUS_INSTALL_ENUM_CONTEXT)Ctx; 1294 1295 /* TODO 1296 * get SWENUM_INSTALL_INTERFACE struct 1297 * open device key and delete the keys 1298 */ 1299 1300 UNIMPLEMENTED; 1301 1302 /* set status */ 1303 Context->Status = STATUS_NOT_IMPLEMENTED; 1304 1305 1306 /* signal completion */ 1307 KeSetEvent(&Context->Event, IO_NO_INCREMENT, FALSE); 1308 } 1309 1310 NTSTATUS 1311 KspQueryBusRelations( 1312 IN PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension, 1313 IN PIRP Irp) 1314 { 1315 PDEVICE_RELATIONS DeviceRelations; 1316 PLIST_ENTRY Entry; 1317 PBUS_DEVICE_ENTRY DeviceEntry; 1318 ULONG Count = 0, Length; 1319 KIRQL OldLevel; 1320 1321 /* acquire lock */ 1322 KeAcquireSpinLock(&BusDeviceExtension->Lock, &OldLevel); 1323 1324 /* first scan all device entries */ 1325 Entry = BusDeviceExtension->Common.Entry.Flink; 1326 1327 while(Entry != &BusDeviceExtension->Common.Entry) 1328 { 1329 /* get offset to device entry */ 1330 DeviceEntry = (PBUS_DEVICE_ENTRY)CONTAINING_RECORD(Entry, BUS_DEVICE_ENTRY, Entry); 1331 1332 /* is there a pdo yet */ 1333 if (DeviceEntry->PDO && (DeviceEntry->DeviceState == NotStarted || DeviceEntry->DeviceState == Started)) 1334 { 1335 /* increment count */ 1336 Count++; 1337 } 1338 1339 /* move to next entry */ 1340 Entry = Entry->Flink; 1341 } 1342 1343 /* calculate length */ 1344 Length = sizeof(DEVICE_RELATIONS) + (Count > 1 ? sizeof(PDEVICE_OBJECT) * (Count-1) : 0); 1345 1346 /* allocate device relations */ 1347 DeviceRelations = (PDEVICE_RELATIONS)AllocateItem(NonPagedPool, Length); 1348 1349 if (!DeviceRelations) 1350 { 1351 /* not enough memory */ 1352 KeReleaseSpinLock(&BusDeviceExtension->Lock, OldLevel); 1353 return STATUS_INSUFFICIENT_RESOURCES; 1354 } 1355 1356 /* rescan device entries */ 1357 Entry = BusDeviceExtension->Common.Entry.Flink; 1358 1359 while(Entry != &BusDeviceExtension->Common.Entry) 1360 { 1361 /* get offset to device entry */ 1362 DeviceEntry = (PBUS_DEVICE_ENTRY)CONTAINING_RECORD(Entry, BUS_DEVICE_ENTRY, Entry); 1363 1364 /* is there a pdo yet */ 1365 if (DeviceEntry->PDO && (DeviceEntry->DeviceState == NotStarted || DeviceEntry->DeviceState == Started)) 1366 { 1367 /* store pdo */ 1368 DeviceRelations->Objects[DeviceRelations->Count] = DeviceEntry->PDO; 1369 1370 /* reference device object */ 1371 ObReferenceObject(DeviceEntry->PDO); 1372 1373 /* increment pdo count */ 1374 DeviceRelations->Count++; 1375 } 1376 1377 /* move to next entry */ 1378 Entry = Entry->Flink; 1379 } 1380 1381 /* release lock */ 1382 KeReleaseSpinLock(&BusDeviceExtension->Lock, OldLevel); 1383 1384 /* FIXME handle existing device relations */ 1385 ASSERT(Irp->IoStatus.Information == 0); 1386 1387 /* store device relations */ 1388 Irp->IoStatus.Information = (ULONG_PTR)DeviceRelations; 1389 1390 /* done */ 1391 return STATUS_SUCCESS; 1392 } 1393 1394 //------------------------------------------------------------------------------------ 1395 1396 /* 1397 @implemented 1398 */ 1399 1400 KSDDKAPI 1401 NTSTATUS 1402 NTAPI 1403 KsGetBusEnumIdentifier( 1404 IN PIRP Irp) 1405 { 1406 PDEV_EXTENSION DeviceExtension; 1407 PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension; 1408 PIO_STACK_LOCATION IoStack; 1409 SIZE_T Length; 1410 NTSTATUS Status; 1411 LPWSTR Buffer; 1412 1413 DPRINT("KsGetBusEnumIdentifier\n"); 1414 1415 /* get stack location */ 1416 IoStack = IoGetCurrentIrpStackLocation(Irp); 1417 1418 /* sanity checks */ 1419 ASSERT(IoStack->DeviceObject); 1420 ASSERT(IoStack->DeviceObject->DeviceExtension); 1421 1422 /* get device extension */ 1423 DeviceExtension = (PDEV_EXTENSION)IoStack->DeviceObject->DeviceExtension; 1424 1425 /* get bus device extension */ 1426 BusDeviceExtension = (PBUS_ENUM_DEVICE_EXTENSION)DeviceExtension->Ext; 1427 1428 /* sanity checks */ 1429 ASSERT(BusDeviceExtension); 1430 ASSERT(BusDeviceExtension->Common.IsBus); 1431 1432 if (!BusDeviceExtension) 1433 { 1434 /* invalid parameter */ 1435 return STATUS_INVALID_PARAMETER; 1436 } 1437 1438 /* get length */ 1439 Length = (wcslen(BusDeviceExtension->BusIdentifier)+1) * sizeof(WCHAR); 1440 1441 /* is there an output buffer provided */ 1442 if (IoStack->Parameters.DeviceIoControl.InputBufferLength) 1443 { 1444 if (Length > IoStack->Parameters.DeviceIoControl.InputBufferLength) 1445 { 1446 /* buffer is too small */ 1447 return STATUS_BUFFER_TOO_SMALL; 1448 } 1449 1450 /* now allocate buffer */ 1451 Buffer = AllocateItem(NonPagedPool, Length); 1452 if (!Buffer) 1453 { 1454 /* no memory */ 1455 Status = STATUS_INSUFFICIENT_RESOURCES; 1456 } 1457 else 1458 { 1459 /* copy bus identifier */ 1460 wcscpy(Buffer, BusDeviceExtension->BusIdentifier); 1461 1462 /* store buffer */ 1463 Irp->AssociatedIrp.SystemBuffer = Buffer; 1464 1465 /* set flag that buffer gets copied back */ 1466 Irp->Flags |= IRP_BUFFERED_IO | IRP_DEALLOCATE_BUFFER | IRP_INPUT_OPERATION; 1467 1468 /* done */ 1469 Status = STATUS_SUCCESS; 1470 } 1471 } 1472 else 1473 { 1474 /* no buffer provided */ 1475 Status = STATUS_BUFFER_OVERFLOW; 1476 } 1477 1478 /* done */ 1479 Irp->IoStatus.Status = Status; 1480 return Status; 1481 } 1482 1483 /* 1484 @implemented 1485 */ 1486 KSDDKAPI 1487 NTSTATUS 1488 NTAPI 1489 KsGetBusEnumParentFDOFromChildPDO( 1490 IN PDEVICE_OBJECT DeviceObject, 1491 OUT PDEVICE_OBJECT *FunctionalDeviceObject) 1492 { 1493 PDEV_EXTENSION DeviceExtension; 1494 1495 DPRINT("KsGetBusEnumParentFDOFromChildPDO\n"); 1496 1497 /* get device extension */ 1498 DeviceExtension = (PDEV_EXTENSION)DeviceObject->DeviceExtension; 1499 1500 /* check if this is child pdo */ 1501 if (DeviceExtension->Ext->IsBus == FALSE) 1502 { 1503 /* return bus device object */ 1504 *FunctionalDeviceObject = DeviceExtension->Ext->BusDeviceExtension->BusDeviceObject; 1505 1506 /* done */ 1507 return STATUS_SUCCESS; 1508 } 1509 1510 /* invalid parameter */ 1511 return STATUS_INVALID_PARAMETER; 1512 } 1513 1514 1515 /* 1516 @implemented 1517 */ 1518 KSDDKAPI 1519 NTSTATUS 1520 NTAPI 1521 KsCreateBusEnumObject( 1522 IN PWCHAR BusIdentifier, 1523 IN PDEVICE_OBJECT BusDeviceObject, 1524 IN PDEVICE_OBJECT PhysicalDeviceObject, 1525 IN PDEVICE_OBJECT PnpDeviceObject OPTIONAL, 1526 IN REFGUID InterfaceGuid OPTIONAL, 1527 IN PWCHAR ServiceRelativePath OPTIONAL) 1528 { 1529 SIZE_T Length; 1530 NTSTATUS Status = STATUS_SUCCESS; 1531 UNICODE_STRING ServiceKeyPath = RTL_CONSTANT_STRING(L"\\REGISTRY\\MACHINE\\SYSTEM\\CurrentControlSet\\Services\\"); 1532 PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension; 1533 PDEV_EXTENSION DeviceExtension; 1534 PBUS_DEVICE_ENTRY DeviceEntry; 1535 PLIST_ENTRY Entry; 1536 KIRQL OldLevel; 1537 1538 DPRINT1("KsCreateBusEnumObject %S BusDeviceObject %p\n", ServiceRelativePath, BusDeviceObject); 1539 1540 /* calculate sizeof bus enum device extension */ 1541 Length = wcslen(BusIdentifier) * sizeof(WCHAR); 1542 Length += sizeof(BUS_ENUM_DEVICE_EXTENSION); 1543 1544 BusDeviceExtension = AllocateItem(NonPagedPool, Length); 1545 if (!BusDeviceExtension) 1546 { 1547 /* not enough memory */ 1548 1549 return STATUS_INSUFFICIENT_RESOURCES; 1550 } 1551 1552 /* get device extension */ 1553 DeviceExtension = (PDEV_EXTENSION)BusDeviceObject->DeviceExtension; 1554 1555 /* store bus device extension */ 1556 DeviceExtension->Ext = (PCOMMON_DEVICE_EXTENSION)BusDeviceExtension; 1557 1558 DPRINT("DeviceExtension %p BusDeviceExtension %p\n", DeviceExtension, DeviceExtension->Ext); 1559 1560 1561 /* zero device extension */ 1562 RtlZeroMemory(BusDeviceExtension, sizeof(BUS_ENUM_DEVICE_EXTENSION)); 1563 1564 /* initialize bus device extension */ 1565 wcscpy(BusDeviceExtension->BusIdentifier, BusIdentifier); 1566 1567 /* allocate service path string */ 1568 Length = ServiceKeyPath.MaximumLength; 1569 Length += BusDeviceObject->DriverObject->DriverExtension->ServiceKeyName.MaximumLength; 1570 1571 if (ServiceRelativePath) 1572 { 1573 /* relative path for devices */ 1574 Length += (wcslen(ServiceRelativePath) + 2) * sizeof(WCHAR); 1575 } 1576 1577 BusDeviceExtension->ServicePath.Length = 0; 1578 BusDeviceExtension->ServicePath.MaximumLength = (USHORT)Length; 1579 BusDeviceExtension->ServicePath.Buffer = AllocateItem(NonPagedPool, Length); 1580 1581 if (!BusDeviceExtension->ServicePath.Buffer) 1582 { 1583 /* not enough memory */ 1584 FreeItem(BusDeviceExtension); 1585 1586 return STATUS_INSUFFICIENT_RESOURCES; 1587 } 1588 1589 RtlAppendUnicodeStringToString(&BusDeviceExtension->ServicePath, &ServiceKeyPath); 1590 RtlAppendUnicodeStringToString(&BusDeviceExtension->ServicePath, &BusDeviceObject->DriverObject->DriverExtension->ServiceKeyName); 1591 1592 if (ServiceRelativePath) 1593 { 1594 RtlAppendUnicodeToString(&BusDeviceExtension->ServicePath, L"\\"); 1595 RtlAppendUnicodeToString(&BusDeviceExtension->ServicePath, ServiceRelativePath); 1596 } 1597 1598 if (InterfaceGuid) 1599 { 1600 /* register an device interface */ 1601 Status = IoRegisterDeviceInterface(PhysicalDeviceObject, InterfaceGuid, NULL, &BusDeviceExtension->DeviceInterfaceLink); 1602 1603 /* check for success */ 1604 if (!NT_SUCCESS(Status)) 1605 { 1606 DPRINT1("IoRegisterDeviceInterface failed Status %lx\n", Status); 1607 FreeItem(BusDeviceExtension->ServicePath.Buffer); 1608 FreeItem(BusDeviceExtension); 1609 return Status; 1610 } 1611 1612 /* now enable device interface */ 1613 Status = IoSetDeviceInterfaceState(&BusDeviceExtension->DeviceInterfaceLink, TRUE); 1614 1615 if (!NT_SUCCESS(Status)) 1616 { 1617 DPRINT1("IoSetDeviceInterfaceState failed Status %lx\n", Status); 1618 FreeItem(BusDeviceExtension->ServicePath.Buffer); 1619 FreeItem(BusDeviceExtension); 1620 return Status; 1621 } 1622 } 1623 1624 /* initialize common device extension */ 1625 BusDeviceExtension->Common.BusDeviceExtension = NULL; 1626 BusDeviceExtension->Common.DeviceObjectReferenceCount = 1; 1627 BusDeviceExtension->Common.DeviceReferenceCount = 1; 1628 BusDeviceExtension->Common.IsBus = TRUE; 1629 InitializeListHead(&BusDeviceExtension->Common.Entry); 1630 1631 /* store device objects */ 1632 BusDeviceExtension->BusDeviceObject = BusDeviceObject; 1633 BusDeviceExtension->PhysicalDeviceObject = PhysicalDeviceObject; 1634 1635 /* initialize lock */ 1636 KeInitializeSpinLock(&BusDeviceExtension->Lock); 1637 1638 /* initialize timer */ 1639 KeInitializeTimer(&BusDeviceExtension->Timer); 1640 1641 /* initialize dpc */ 1642 KeInitializeDpc(&BusDeviceExtension->Dpc, KspBusDpcRoutine, (PVOID)BusDeviceExtension); 1643 1644 /* initialize event */ 1645 KeInitializeEvent(&BusDeviceExtension->Event, SynchronizationEvent, FALSE); 1646 1647 /* initialize work item */ 1648 ExInitializeWorkItem(&BusDeviceExtension->WorkItem, KspBusWorkerRoutine, (PVOID)BusDeviceExtension); 1649 1650 if (!PnpDeviceObject) 1651 { 1652 /* attach device */ 1653 BusDeviceExtension->PnpDeviceObject = IoAttachDeviceToDeviceStack(BusDeviceObject, PhysicalDeviceObject); 1654 1655 if (!BusDeviceExtension->PnpDeviceObject) 1656 { 1657 /* failed to attach device */ 1658 DPRINT1("IoAttachDeviceToDeviceStack failed with %x\n", Status); 1659 if (BusDeviceExtension->DeviceInterfaceLink.Buffer) 1660 { 1661 IoSetDeviceInterfaceState(&BusDeviceExtension->DeviceInterfaceLink, FALSE); 1662 RtlFreeUnicodeString(&BusDeviceExtension->DeviceInterfaceLink); 1663 } 1664 1665 /* free device extension */ 1666 FreeItem(BusDeviceExtension->ServicePath.Buffer); 1667 FreeItem(BusDeviceExtension); 1668 1669 return STATUS_DEVICE_REMOVED; 1670 } 1671 1672 /* mark device as attached */ 1673 BusDeviceExtension->DeviceAttached = TRUE; 1674 } 1675 else 1676 { 1677 /* directly attach */ 1678 BusDeviceExtension->PnpDeviceObject = PnpDeviceObject; 1679 } 1680 1681 /* now scan the bus */ 1682 Status = KspScanBus(BusDeviceExtension); 1683 1684 /* check for success */ 1685 if (!NT_SUCCESS(Status)) 1686 { 1687 /* failed to scan bus */ 1688 if (BusDeviceExtension->DeviceInterfaceLink.Buffer) 1689 { 1690 IoSetDeviceInterfaceState(&BusDeviceExtension->DeviceInterfaceLink, FALSE); 1691 RtlFreeUnicodeString(&BusDeviceExtension->DeviceInterfaceLink); 1692 } 1693 1694 if (BusDeviceExtension->DeviceAttached) 1695 { 1696 /* detach device */ 1697 IoDetachDevice(BusDeviceExtension->PnpDeviceObject); 1698 } 1699 1700 /* free device extension */ 1701 FreeItem(BusDeviceExtension->ServicePath.Buffer); 1702 FreeItem(BusDeviceExtension); 1703 1704 return Status; 1705 } 1706 1707 /* acquire device entry lock */ 1708 KeAcquireSpinLock(&BusDeviceExtension->Lock, &OldLevel); 1709 1710 /* now iterate all device entries */ 1711 Entry = BusDeviceExtension->Common.Entry.Flink; 1712 while(Entry != &BusDeviceExtension->Common.Entry) 1713 { 1714 /* get device entry */ 1715 DeviceEntry = (PBUS_DEVICE_ENTRY)CONTAINING_RECORD(Entry, BUS_DEVICE_ENTRY, Entry); 1716 if (!DeviceEntry->PDO) 1717 { 1718 /* release device entry lock */ 1719 KeReleaseSpinLock(&BusDeviceExtension->Lock, OldLevel); 1720 1721 /* create pdo */ 1722 Status = KspCreatePDO(BusDeviceExtension, DeviceEntry, &DeviceEntry->PDO); 1723 1724 /* acquire device entry lock */ 1725 KeAcquireSpinLock(&BusDeviceExtension->Lock, &OldLevel); 1726 1727 /* done */ 1728 break; 1729 } 1730 /* move to next entry */ 1731 Entry = Entry->Flink; 1732 } 1733 1734 /* release device entry lock */ 1735 KeReleaseSpinLock(&BusDeviceExtension->Lock, OldLevel); 1736 1737 1738 /* invalidate device relations */ 1739 IoInvalidateDeviceRelations(BusDeviceExtension->PhysicalDeviceObject, BusRelations); 1740 DPRINT("KsCreateBusEnumObject Status %x\n", Status); 1741 /* done */ 1742 return Status; 1743 } 1744 1745 /* 1746 @implemented 1747 */ 1748 KSDDKAPI 1749 NTSTATUS 1750 NTAPI 1751 KsGetBusEnumPnpDeviceObject( 1752 IN PDEVICE_OBJECT DeviceObject, 1753 IN PDEVICE_OBJECT *PnpDeviceObject) 1754 { 1755 PDEV_EXTENSION DeviceExtension; 1756 PCOMMON_DEVICE_EXTENSION CommonDeviceExtension; 1757 PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension; 1758 1759 DPRINT("KsGetBusEnumPnpDeviceObject\n"); 1760 1761 if (!DeviceObject->DeviceExtension) 1762 { 1763 /* invalid parameter */ 1764 return STATUS_INVALID_PARAMETER; 1765 } 1766 1767 /* get device extension */ 1768 DeviceExtension = (PDEV_EXTENSION)DeviceObject->DeviceExtension; 1769 1770 /* get common device extension */ 1771 CommonDeviceExtension = DeviceExtension->Ext; 1772 1773 if (!CommonDeviceExtension) 1774 { 1775 /* invalid parameter */ 1776 return STATUS_INVALID_PARAMETER; 1777 } 1778 1779 if (!CommonDeviceExtension->IsBus) 1780 { 1781 /* getting pnp device object is only supported for software bus device object */ 1782 return STATUS_INVALID_PARAMETER; 1783 } 1784 1785 /* sanity checks */ 1786 ASSERT(CommonDeviceExtension); 1787 ASSERT(CommonDeviceExtension->IsBus); 1788 1789 /* cast to bus device extension */ 1790 BusDeviceExtension = (PBUS_ENUM_DEVICE_EXTENSION)CommonDeviceExtension; 1791 1792 /* store result */ 1793 *PnpDeviceObject = BusDeviceExtension->PnpDeviceObject; 1794 1795 /* done */ 1796 return STATUS_SUCCESS; 1797 } 1798 1799 /* 1800 @implemented 1801 */ 1802 KSDDKAPI 1803 NTSTATUS 1804 NTAPI 1805 KsInstallBusEnumInterface( 1806 PIRP Irp) 1807 { 1808 BUS_INSTALL_ENUM_CONTEXT Context; 1809 KPROCESSOR_MODE Mode; 1810 LUID luid; 1811 PIO_STACK_LOCATION IoStack; 1812 PDEV_EXTENSION DeviceExtension; 1813 PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension; 1814 1815 DPRINT("KsInstallBusEnumInterface\n"); 1816 1817 /* get current irp stack location */ 1818 IoStack = IoGetCurrentIrpStackLocation(Irp); 1819 1820 /* get previous mode */ 1821 Mode = ExGetPreviousMode(); 1822 1823 /* convert to luid */ 1824 luid = RtlConvertUlongToLuid(SE_LOAD_DRIVER_PRIVILEGE); 1825 1826 /* perform access check */ 1827 if (!SeSinglePrivilegeCheck(luid, Mode)) 1828 { 1829 /* FIXME insufficient privileges */ 1830 //return STATUS_PRIVILEGE_NOT_HELD; 1831 } 1832 1833 /* get device extension */ 1834 DeviceExtension = (PDEV_EXTENSION)IoStack->DeviceObject->DeviceExtension; 1835 1836 /* get bus device extension */ 1837 BusDeviceExtension = (PBUS_ENUM_DEVICE_EXTENSION)DeviceExtension->Ext; 1838 1839 1840 /* initialize context */ 1841 Context.Irp = Irp; 1842 KeInitializeEvent(&Context.Event, NotificationEvent, FALSE); 1843 Context.BusDeviceExtension = BusDeviceExtension; 1844 ExInitializeWorkItem(&Context.WorkItem, KspInstallBusEnumInterface, (PVOID)&Context); 1845 1846 /* queue the work item */ 1847 ExQueueWorkItem(&Context.WorkItem, DelayedWorkQueue); 1848 /* wait for completion */ 1849 KeWaitForSingleObject(&Context.Event, Executive, KernelMode, FALSE, NULL); 1850 1851 /* store result */ 1852 Irp->IoStatus.Status = Context.Status; 1853 1854 /* done */ 1855 return Context.Status; 1856 } 1857 1858 /* 1859 @implemented 1860 */ 1861 KSDDKAPI 1862 NTSTATUS 1863 NTAPI 1864 KsIsBusEnumChildDevice( 1865 IN PDEVICE_OBJECT DeviceObject, 1866 OUT PBOOLEAN ChildDevice) 1867 { 1868 PDEV_EXTENSION DeviceExtension; 1869 PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension; 1870 1871 DPRINT("KsIsBusEnumChildDevice %p\n", DeviceObject); 1872 1873 /* get device extension */ 1874 DeviceExtension = (PDEV_EXTENSION)DeviceObject->DeviceExtension; 1875 1876 /* get bus device extension */ 1877 BusDeviceExtension = (PBUS_ENUM_DEVICE_EXTENSION)DeviceExtension->Ext; 1878 1879 if (!BusDeviceExtension) 1880 { 1881 /* not a bus device */ 1882 return STATUS_INVALID_PARAMETER; 1883 } 1884 1885 /* store result */ 1886 *ChildDevice = (BusDeviceExtension->Common.IsBus == FALSE); 1887 1888 return STATUS_SUCCESS; 1889 } 1890 1891 /* 1892 @implemented 1893 */ 1894 KSDDKAPI 1895 NTSTATUS 1896 NTAPI 1897 KsServiceBusEnumCreateRequest( 1898 IN PDEVICE_OBJECT DeviceObject, 1899 IN OUT PIRP Irp) 1900 { 1901 PLIST_ENTRY Entry; 1902 PBUS_DEVICE_ENTRY DeviceEntry = NULL; /* fix gcc */ 1903 PIO_STACK_LOCATION IoStack; 1904 BOOLEAN ItemExists = FALSE; 1905 PDEV_EXTENSION DeviceExtension; 1906 PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension; 1907 //PCOMMON_DEVICE_EXTENSION ChildDeviceExtension; 1908 NTSTATUS Status; 1909 LARGE_INTEGER Time; 1910 1911 /* FIXME: locks */ 1912 1913 /* get device extension */ 1914 DeviceExtension = (PDEV_EXTENSION)DeviceObject->DeviceExtension; 1915 1916 /* get bus device extension */ 1917 BusDeviceExtension = (PBUS_ENUM_DEVICE_EXTENSION)DeviceExtension->Ext; 1918 1919 /* get current irp stack location */ 1920 IoStack = IoGetCurrentIrpStackLocation(Irp); 1921 1922 /* sanity checks */ 1923 ASSERT(IoStack->FileObject); 1924 if (IoStack->FileObject->FileName.Buffer == NULL) 1925 { 1926 DPRINT("KsServiceBusEnumCreateRequest PNP Hack\n"); 1927 Irp->IoStatus.Status = STATUS_SUCCESS; 1928 return STATUS_SUCCESS; 1929 } 1930 1931 ASSERT(IoStack->FileObject->FileName.Buffer); 1932 1933 DPRINT1("KsServiceBusEnumCreateRequest IRP %p Name %wZ\n", Irp, &IoStack->FileObject->FileName); 1934 1935 /* scan list and check if it is already present */ 1936 Entry = BusDeviceExtension->Common.Entry.Flink; 1937 1938 while(Entry != &BusDeviceExtension->Common.Entry) 1939 { 1940 /* get real offset */ 1941 DeviceEntry = (PBUS_DEVICE_ENTRY)CONTAINING_RECORD(Entry, BUS_DEVICE_ENTRY, Entry); 1942 1943 /* check if name matches */ 1944 if (!wcsicmp(DeviceEntry->DeviceName, IoStack->FileObject->FileName.Buffer + 1)) 1945 { 1946 /* item already exists */ 1947 ItemExists = TRUE; 1948 break; 1949 } 1950 1951 /* move to next entry */ 1952 Entry = Entry->Flink; 1953 } 1954 1955 if (!ItemExists) 1956 { 1957 /* interface not registered */ 1958 DPRINT1("Interface %wZ not registered\n", &IoStack->FileObject->FileName); 1959 return STATUS_OBJECT_NAME_NOT_FOUND; 1960 } 1961 1962 /* is there a pdo yet */ 1963 if (DeviceEntry->PDO) 1964 { 1965 if (DeviceEntry->DeviceState == Started) 1966 { 1967 /* issue reparse */ 1968 Status = KspDoReparseForIrp(Irp, DeviceEntry); 1969 DPRINT("REPARSE Irp %p '%wZ'\n", Irp, &IoStack->FileObject->FileName); 1970 1971 Irp->IoStatus.Status = Status; 1972 Irp->IoStatus.Information = IO_REPARSE; 1973 return Status; 1974 } 1975 1976 /* delay processing until pnp is finished with enumeration */ 1977 IoMarkIrpPending(Irp); 1978 1979 /* insert into irp pending list */ 1980 InsertTailList(&DeviceEntry->IrpPendingList, &Irp->Tail.Overlay.ListEntry); 1981 1982 Time.QuadPart = Int32x32To64(1500, -10000); 1983 DbgPrint("PENDING Irp %p %wZ DeviceState %d\n", Irp, &IoStack->FileObject->FileName, DeviceEntry->DeviceState); 1984 1985 1986 /* set timer */ 1987 KeSetTimer(&BusDeviceExtension->Timer, Time, &BusDeviceExtension->Dpc); 1988 1989 /* done for now */ 1990 return STATUS_PENDING; 1991 1992 } 1993 else 1994 { 1995 /* time to create PDO */ 1996 Status = KspCreatePDO(BusDeviceExtension, DeviceEntry, &DeviceEntry->PDO); 1997 1998 if (!NT_SUCCESS(Status)) 1999 { 2000 /* failed to create PDO */ 2001 DPRINT1("KsServiceBusEnumCreateRequest failed to create PDO with %x\n", Status); 2002 return Status; 2003 } 2004 DPRINT("PENDING CREATE Irp %p %wZ\n", Irp, &IoStack->FileObject->FileName); 2005 2006 /* delay processing until pnp is finished with enumeration */ 2007 IoMarkIrpPending(Irp); 2008 2009 /* insert into irp pending list */ 2010 InsertTailList(&DeviceEntry->IrpPendingList, &Irp->Tail.Overlay.ListEntry); 2011 2012 /* invalidate device relations */ 2013 IoInvalidateDeviceRelations(BusDeviceExtension->PhysicalDeviceObject, BusRelations); 2014 2015 /* done for now */ 2016 return STATUS_PENDING; 2017 } 2018 } 2019 2020 /* 2021 @implemented 2022 */ 2023 KSDDKAPI 2024 NTSTATUS 2025 NTAPI 2026 KsServiceBusEnumPnpRequest( 2027 IN PDEVICE_OBJECT DeviceObject, 2028 IN OUT PIRP Irp) 2029 { 2030 PDEV_EXTENSION DeviceExtension; 2031 PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension; 2032 PCOMMON_DEVICE_EXTENSION ChildDeviceExtension; 2033 PIO_STACK_LOCATION IoStack; 2034 NTSTATUS Status; 2035 LARGE_INTEGER Time; 2036 PDEVICE_RELATIONS DeviceRelation; 2037 PBUS_DEVICE_ENTRY DeviceEntry; 2038 2039 /* get device extension */ 2040 DeviceExtension = (PDEV_EXTENSION)DeviceObject->DeviceExtension; 2041 2042 /* get bus device extension */ 2043 BusDeviceExtension = (PBUS_ENUM_DEVICE_EXTENSION)DeviceExtension->Ext; 2044 2045 /* get current irp stack location */ 2046 IoStack = IoGetCurrentIrpStackLocation(Irp); 2047 2048 if (BusDeviceExtension->Common.IsBus) 2049 { 2050 if (IoStack->MinorFunction == IRP_MN_START_DEVICE) 2051 { 2052 /* no op for bus driver */ 2053 Status = STATUS_SUCCESS; 2054 } 2055 else if (IoStack->MinorFunction == IRP_MN_QUERY_DEVICE_RELATIONS) 2056 { 2057 /* handle bus device relations */ 2058 ASSERT(IoStack->Parameters.QueryDeviceRelations.Type == BusRelations); 2059 2060 Status = KspQueryBusRelations(BusDeviceExtension, Irp); 2061 } 2062 else 2063 { 2064 /* get default status */ 2065 Status = Irp->IoStatus.Status; 2066 } 2067 } 2068 else 2069 { 2070 /* get child device extension */ 2071 ChildDeviceExtension = DeviceExtension->Ext; 2072 2073 /* get bus device extension */ 2074 BusDeviceExtension = ChildDeviceExtension->BusDeviceExtension; 2075 2076 if (IoStack->MinorFunction == IRP_MN_QUERY_ID) 2077 { 2078 /* query id */ 2079 Status = KspQueryId(ChildDeviceExtension, Irp); 2080 } 2081 else if (IoStack->MinorFunction == IRP_MN_REMOVE_DEVICE) 2082 { 2083 ASSERT(ChildDeviceExtension->DeviceEntry->DeviceState != Started || ChildDeviceExtension->DeviceEntry->DeviceState == NotStarted); 2084 ASSERT(ChildDeviceExtension->DeviceEntry->PDO == DeviceObject); 2085 2086 /* backup device entry */ 2087 DeviceEntry = ChildDeviceExtension->DeviceEntry; 2088 2089 /* free device extension */ 2090 FreeItem(ChildDeviceExtension); 2091 2092 /* clear PDO reference */ 2093 DeviceEntry->PDO = NULL; 2094 2095 /* delete the device */ 2096 IoDeleteDevice(DeviceObject); 2097 2098 if (DeviceEntry->PDODeviceName) 2099 { 2100 /* delete pdo device name */ 2101 FreeItem(DeviceEntry->PDODeviceName); 2102 2103 /* set to null */ 2104 DeviceEntry->PDODeviceName = NULL; 2105 } 2106 2107 /* set state no notstarted */ 2108 DeviceEntry->DeviceState = NotStarted; 2109 2110 /* complete pending irps */ 2111 KspCompletePendingIrps(DeviceEntry, STATUS_DEVICE_REMOVED); 2112 2113 /* done */ 2114 Status = STATUS_SUCCESS; 2115 } 2116 else if (IoStack->MinorFunction == IRP_MN_QUERY_BUS_INFORMATION) 2117 { 2118 /* query bus information */ 2119 Status = KspQueryBusInformation(ChildDeviceExtension, Irp); 2120 } 2121 else if (IoStack->MinorFunction == IRP_MN_QUERY_RESOURCES) 2122 { 2123 /* no op */ 2124 Status = STATUS_SUCCESS; 2125 } 2126 else if (IoStack->MinorFunction == IRP_MN_QUERY_RESOURCE_REQUIREMENTS) 2127 { 2128 /* no op */ 2129 Status = STATUS_SUCCESS; 2130 } 2131 else if (IoStack->MinorFunction == IRP_MN_START_DEVICE) 2132 { 2133 /* start bus */ 2134 Status = KspStartBusDevice(DeviceObject, ChildDeviceExtension, Irp); 2135 if (NT_SUCCESS(Status)) 2136 { 2137 /* complete pending irps*/ 2138 KspCompletePendingIrps(ChildDeviceExtension->DeviceEntry, STATUS_REPARSE); 2139 } 2140 2141 /* set time out */ 2142 Time.QuadPart = Int32x32To64(1500, -10000); 2143 2144 /* sanity check */ 2145 ASSERT(BusDeviceExtension); 2146 2147 /* set timer */ 2148 KeSetTimer(&BusDeviceExtension->Timer, Time, &BusDeviceExtension->Dpc); 2149 } 2150 else if (IoStack->MinorFunction == IRP_MN_QUERY_CAPABILITIES) 2151 { 2152 /* query capabilities */ 2153 Status = KspQueryBusDeviceCapabilities(ChildDeviceExtension, Irp); 2154 } 2155 else if (IoStack->MinorFunction == IRP_MN_QUERY_PNP_DEVICE_STATE) 2156 { 2157 /* query pnp state */ 2158 Status = KspQueryBusDevicePnpState(ChildDeviceExtension, Irp); 2159 } 2160 else if (IoStack->MinorFunction == IRP_MN_QUERY_INTERFACE) 2161 { 2162 /* query interface */ 2163 Status = KspQueryBusDeviceInterface(ChildDeviceExtension, Irp); 2164 } 2165 else if (IoStack->MinorFunction == IRP_MN_QUERY_DEVICE_RELATIONS && IoStack->Parameters.QueryDeviceRelations.Type == TargetDeviceRelation) 2166 { 2167 /* handle target device relations */ 2168 ASSERT(IoStack->Parameters.QueryDeviceRelations.Type == TargetDeviceRelation); 2169 ASSERT(Irp->IoStatus.Information == 0); 2170 2171 /* allocate device relation */ 2172 DeviceRelation = AllocateItem(PagedPool, sizeof(DEVICE_RELATIONS)); 2173 if (DeviceRelation) 2174 { 2175 DeviceRelation->Count = 1; 2176 DeviceRelation->Objects[0] = DeviceObject; 2177 2178 /* reference self */ 2179 ObReferenceObject(DeviceObject); 2180 2181 /* store result */ 2182 Irp->IoStatus.Information = (ULONG_PTR)DeviceRelation; 2183 2184 /* done */ 2185 Status = STATUS_SUCCESS; 2186 } 2187 else 2188 { 2189 /* no memory */ 2190 Status = STATUS_INSUFFICIENT_RESOURCES; 2191 } 2192 } 2193 else 2194 { 2195 /* get default status */ 2196 Status = Irp->IoStatus.Status; 2197 } 2198 } 2199 2200 DPRINT("KsServiceBusEnumPnpRequest %p Bus %u Function %x Status %x\n", DeviceObject, BusDeviceExtension->Common.IsBus, IoStack->MinorFunction, Status); 2201 Irp->IoStatus.Status = Status; 2202 return Status; 2203 } 2204 2205 /* 2206 @implemented 2207 */ 2208 KSDDKAPI 2209 NTSTATUS 2210 NTAPI 2211 KsRemoveBusEnumInterface( 2212 IN PIRP Irp) 2213 { 2214 KPROCESSOR_MODE Mode; 2215 LUID luid; 2216 BUS_INSTALL_ENUM_CONTEXT Ctx; 2217 PDEV_EXTENSION DeviceExtension; 2218 PBUS_ENUM_DEVICE_EXTENSION BusDeviceExtension; 2219 PIO_STACK_LOCATION IoStack; 2220 2221 DPRINT("KsRemoveBusEnumInterface\n"); 2222 2223 /* get io stack location */ 2224 IoStack = IoGetCurrentIrpStackLocation(Irp); 2225 2226 /* get device extension */ 2227 DeviceExtension = (PDEV_EXTENSION)IoStack->DeviceObject->DeviceExtension; 2228 2229 /* get bus device extension */ 2230 BusDeviceExtension = DeviceExtension->Ext->BusDeviceExtension; 2231 2232 /* get previous mode */ 2233 Mode = ExGetPreviousMode(); 2234 2235 /* convert to luid */ 2236 luid = RtlConvertUlongToLuid(SE_LOAD_DRIVER_PRIVILEGE); 2237 2238 /* perform access check */ 2239 if (!SeSinglePrivilegeCheck(luid, Mode)) 2240 { 2241 /* insufficient privileges */ 2242 return STATUS_PRIVILEGE_NOT_HELD; 2243 } 2244 2245 /* initialize context */ 2246 KeInitializeEvent(&Ctx.Event, NotificationEvent, FALSE); 2247 Ctx.Irp = Irp; 2248 Ctx.BusDeviceExtension = BusDeviceExtension; 2249 ExInitializeWorkItem(&Ctx.WorkItem, KspRemoveBusInterface, (PVOID)&Ctx); 2250 2251 /* now queue the work item */ 2252 ExQueueWorkItem(&Ctx.WorkItem, DelayedWorkQueue); 2253 2254 /* wait for completion */ 2255 KeWaitForSingleObject(&Ctx.Event, Executive, KernelMode, FALSE, NULL); 2256 2257 /* return result */ 2258 return Ctx.Status; 2259 } 2260