1 /* 2 * COPYRIGHT: See COPYING in the top level directory 3 * PROJECT: ReactOS Kernel Streaming 4 * FILE: drivers/wdm/audio/legacy/wdmaud/mmixer.c 5 * PURPOSE: WDM Legacy Mixer 6 * PROGRAMMER: Johannes Anderwald 7 */ 8 9 #include "wdmaud.h" 10 11 #include <mmixer.h> 12 13 #define NDEBUG 14 #include <debug.h> 15 16 PVOID Alloc(ULONG NumBytes); 17 MIXER_STATUS Close(HANDLE hDevice); 18 VOID Free(PVOID Block); 19 VOID Copy(PVOID Src, PVOID Dst, ULONG NumBytes); 20 MIXER_STATUS Open(IN LPWSTR DevicePath, OUT PHANDLE hDevice); 21 MIXER_STATUS Control(IN HANDLE hMixer, IN ULONG dwIoControlCode, IN PVOID lpInBuffer, IN ULONG nInBufferSize, OUT PVOID lpOutBuffer, ULONG nOutBufferSize, PULONG lpBytesReturned); 22 MIXER_STATUS Enum(IN PVOID EnumContext, IN ULONG DeviceIndex, OUT LPWSTR * DeviceName, OUT PHANDLE OutHandle, OUT PHANDLE OutKey); 23 MIXER_STATUS OpenKey(IN HANDLE hKey, IN LPWSTR SubKey, IN ULONG DesiredAccess, OUT PHANDLE OutKey); 24 MIXER_STATUS CloseKey(IN HANDLE hKey); 25 MIXER_STATUS QueryKeyValue(IN HANDLE hKey, IN LPWSTR KeyName, OUT PVOID * ResultBuffer, OUT PULONG ResultLength, OUT PULONG KeyType); 26 PVOID AllocEventData(IN ULONG ExtraSize); 27 VOID FreeEventData(IN PVOID EventData); 28 29 MIXER_CONTEXT MixerContext = 30 { 31 sizeof(MIXER_CONTEXT), 32 NULL, 33 Alloc, 34 Control, 35 Free, 36 Open, 37 Close, 38 Copy, 39 OpenKey, 40 QueryKeyValue, 41 CloseKey, 42 AllocEventData, 43 FreeEventData 44 }; 45 46 GUID CategoryGuid = {STATIC_KSCATEGORY_AUDIO}; 47 48 MIXER_STATUS 49 QueryKeyValue( 50 IN HANDLE hKey, 51 IN LPWSTR lpKeyName, 52 OUT PVOID * ResultBuffer, 53 OUT PULONG ResultLength, 54 OUT PULONG KeyType) 55 { 56 NTSTATUS Status; 57 UNICODE_STRING KeyName; 58 ULONG Length; 59 PKEY_VALUE_PARTIAL_INFORMATION PartialInformation; 60 61 /* initialize key name */ 62 RtlInitUnicodeString(&KeyName, lpKeyName); 63 64 /* now query MatchingDeviceId key */ 65 Status = ZwQueryValueKey(hKey, &KeyName, KeyValuePartialInformation, NULL, 0, &Length); 66 67 /* check for success */ 68 if (Status != STATUS_BUFFER_TOO_SMALL) 69 return MM_STATUS_UNSUCCESSFUL; 70 71 /* allocate a buffer for key data */ 72 PartialInformation = AllocateItem(NonPagedPool, Length); 73 74 if (!PartialInformation) 75 return MM_STATUS_NO_MEMORY; 76 77 78 /* now query MatchingDeviceId key */ 79 Status = ZwQueryValueKey(hKey, &KeyName, KeyValuePartialInformation, PartialInformation, Length, &Length); 80 81 /* check for success */ 82 if (!NT_SUCCESS(Status)) 83 { 84 FreeItem(PartialInformation); 85 return MM_STATUS_UNSUCCESSFUL; 86 } 87 88 if (KeyType) 89 { 90 /* return key type */ 91 *KeyType = PartialInformation->Type; 92 } 93 94 if (ResultLength) 95 { 96 /* return data length */ 97 *ResultLength = PartialInformation->DataLength; 98 } 99 100 *ResultBuffer = AllocateItem(NonPagedPool, PartialInformation->DataLength); 101 if (!*ResultBuffer) 102 { 103 /* not enough memory */ 104 FreeItem(PartialInformation); 105 return MM_STATUS_NO_MEMORY; 106 } 107 108 /* copy key value */ 109 RtlMoveMemory(*ResultBuffer, PartialInformation->Data, PartialInformation->DataLength); 110 111 /* free key info */ 112 FreeItem(PartialInformation); 113 114 return MM_STATUS_SUCCESS; 115 } 116 117 MIXER_STATUS 118 OpenKey( 119 IN HANDLE hKey, 120 IN LPWSTR lpSubKeyName, 121 IN ULONG DesiredAccess, 122 OUT PHANDLE OutKey) 123 { 124 OBJECT_ATTRIBUTES ObjectAttributes; 125 UNICODE_STRING SubKeyName; 126 NTSTATUS Status; 127 128 /* initialize sub key name */ 129 RtlInitUnicodeString(&SubKeyName, lpSubKeyName); 130 131 /* initialize key attributes */ 132 InitializeObjectAttributes(&ObjectAttributes, &SubKeyName, OBJ_CASE_INSENSITIVE | OBJ_OPENIF, hKey, NULL); 133 134 /* open the key */ 135 Status = ZwOpenKey(OutKey, DesiredAccess, &ObjectAttributes); 136 137 if (NT_SUCCESS(Status)) 138 return MM_STATUS_SUCCESS; 139 else 140 return MM_STATUS_UNSUCCESSFUL; 141 } 142 143 MIXER_STATUS 144 CloseKey( 145 IN HANDLE hKey) 146 { 147 if (ZwClose(hKey) == STATUS_SUCCESS) 148 return MM_STATUS_SUCCESS; 149 else 150 return MM_STATUS_UNSUCCESSFUL; 151 } 152 153 154 PVOID Alloc(ULONG NumBytes) 155 { 156 return AllocateItem(NonPagedPool, NumBytes); 157 } 158 159 MIXER_STATUS 160 Close(HANDLE hDevice) 161 { 162 if (ZwClose(hDevice) == STATUS_SUCCESS) 163 return MM_STATUS_SUCCESS; 164 else 165 return MM_STATUS_UNSUCCESSFUL; 166 } 167 168 VOID 169 Free(PVOID Block) 170 { 171 FreeItem(Block); 172 } 173 174 VOID 175 Copy(PVOID Src, PVOID Dst, ULONG NumBytes) 176 { 177 RtlMoveMemory(Src, Dst, NumBytes); 178 } 179 180 MIXER_STATUS 181 Open( 182 IN LPWSTR DevicePath, 183 OUT PHANDLE hDevice) 184 { 185 if (WdmAudOpenSysAudioDevice(DevicePath, hDevice) == STATUS_SUCCESS) 186 return MM_STATUS_SUCCESS; 187 else 188 return MM_STATUS_UNSUCCESSFUL; 189 } 190 191 MIXER_STATUS 192 Control( 193 IN HANDLE hMixer, 194 IN ULONG dwIoControlCode, 195 IN PVOID lpInBuffer, 196 IN ULONG nInBufferSize, 197 OUT PVOID lpOutBuffer, 198 ULONG nOutBufferSize, 199 PULONG lpBytesReturned) 200 { 201 NTSTATUS Status; 202 PFILE_OBJECT FileObject; 203 204 /* get file object */ 205 Status = ObReferenceObjectByHandle(hMixer, GENERIC_READ | GENERIC_WRITE, *IoFileObjectType, KernelMode, (PVOID*)&FileObject, NULL); 206 if (!NT_SUCCESS(Status)) 207 { 208 DPRINT("failed to reference %p with %lx\n", hMixer, Status); 209 return MM_STATUS_UNSUCCESSFUL; 210 } 211 212 /* perform request */ 213 Status = KsSynchronousIoControlDevice(FileObject, KernelMode, dwIoControlCode, lpInBuffer, nInBufferSize, lpOutBuffer, nOutBufferSize, lpBytesReturned); 214 215 /* release object reference */ 216 ObDereferenceObject(FileObject); 217 218 if (Status == STATUS_MORE_ENTRIES || Status == STATUS_BUFFER_OVERFLOW || Status == STATUS_BUFFER_TOO_SMALL) 219 { 220 /* more data is available */ 221 return MM_STATUS_MORE_ENTRIES; 222 } 223 else if (Status == STATUS_SUCCESS) 224 { 225 /* operation succeeded */ 226 return MM_STATUS_SUCCESS; 227 } 228 else 229 { 230 DPRINT("Failed with %lx\n", Status); 231 return MM_STATUS_UNSUCCESSFUL; 232 } 233 } 234 235 MIXER_STATUS 236 Enum( 237 IN PVOID EnumContext, 238 IN ULONG DeviceIndex, 239 OUT LPWSTR * DeviceName, 240 OUT PHANDLE OutHandle, 241 OUT PHANDLE OutKey) 242 { 243 PDEVICE_OBJECT DeviceObject; 244 ULONG DeviceCount; 245 NTSTATUS Status; 246 UNICODE_STRING KeyName; 247 248 /* get enumeration context */ 249 DeviceObject = (PDEVICE_OBJECT)EnumContext; 250 251 /* get device count */ 252 DeviceCount = GetSysAudioDeviceCount(DeviceObject); 253 254 if (DeviceIndex >= DeviceCount) 255 { 256 /* no more devices */ 257 return MM_STATUS_NO_MORE_DEVICES; 258 } 259 260 /* get device name */ 261 Status = GetSysAudioDevicePnpName(DeviceObject, DeviceIndex, DeviceName); 262 263 if (!NT_SUCCESS(Status)) 264 { 265 /* failed to retrieve device name */ 266 return MM_STATUS_UNSUCCESSFUL; 267 } 268 269 /* initialize key name */ 270 RtlInitUnicodeString(&KeyName, *DeviceName); 271 272 /* open device interface key */ 273 Status = IoOpenDeviceInterfaceRegistryKey(&KeyName, GENERIC_READ | GENERIC_WRITE, OutKey); 274 275 if (!NT_SUCCESS(Status)) 276 { 277 *OutKey = NULL; 278 } 279 280 #if 0 281 if (!NT_SUCCESS(Status)) 282 { 283 /* failed to open key */ 284 DPRINT("IoOpenDeviceInterfaceRegistryKey failed with %lx\n", Status); 285 FreeItem(*DeviceName); 286 return MM_STATUS_UNSUCCESSFUL; 287 } 288 #endif 289 290 /* open device handle */ 291 Status = OpenDevice(*DeviceName, OutHandle, NULL); 292 if (!NT_SUCCESS(Status)) 293 { 294 /* failed to open device */ 295 return MM_STATUS_UNSUCCESSFUL; 296 } 297 298 return MM_STATUS_SUCCESS; 299 } 300 301 PVOID 302 AllocEventData( 303 IN ULONG ExtraSize) 304 { 305 PKSEVENTDATA Data = (PKSEVENTDATA)AllocateItem(NonPagedPool, sizeof(KSEVENTDATA) + ExtraSize); 306 if (!Data) 307 return NULL; 308 309 Data->EventObject.Event = AllocateItem(NonPagedPool, sizeof(KEVENT)); 310 if (!Data->EventHandle.Event) 311 { 312 FreeItem(Data); 313 return NULL; 314 } 315 316 KeInitializeEvent(Data->EventObject.Event, NotificationEvent, FALSE); 317 318 Data->NotificationType = KSEVENTF_EVENT_HANDLE; 319 return Data; 320 } 321 322 VOID 323 FreeEventData(IN PVOID EventData) 324 { 325 PKSEVENTDATA Data = (PKSEVENTDATA)EventData; 326 327 FreeItem(Data->EventHandle.Event); 328 FreeItem(Data); 329 } 330 331 VOID 332 CALLBACK 333 EventCallback( 334 IN PVOID MixerEventContext, 335 IN HANDLE hMixer, 336 IN ULONG NotificationType, 337 IN ULONG Value) 338 { 339 PWDMAUD_CLIENT ClientInfo; 340 PEVENT_ENTRY Entry; 341 ULONG Index; 342 343 /* get client context */ 344 ClientInfo = (PWDMAUD_CLIENT)MixerEventContext; 345 346 /* now search for the mixer which originated the request */ 347 for(Index = 0; Index < ClientInfo->NumPins; Index++) 348 { 349 if (ClientInfo->hPins[Index].Handle == hMixer && ClientInfo->hPins[Index].Type == MIXER_DEVICE_TYPE) 350 { 351 if (ClientInfo->hPins[Index].NotifyEvent) 352 { 353 /* allocate event entry */ 354 Entry = AllocateItem(NonPagedPool, sizeof(EVENT_ENTRY)); 355 if (!Entry) 356 { 357 /* no memory */ 358 break; 359 } 360 361 /* setup event entry */ 362 Entry->NotificationType = NotificationType; 363 Entry->Value = Value; 364 Entry->hMixer = hMixer; 365 366 /* insert entry */ 367 InsertTailList(&ClientInfo->MixerEventList, &Entry->Entry); 368 369 /* now notify the client */ 370 KeSetEvent(ClientInfo->hPins[Index].NotifyEvent, 0, FALSE); 371 } 372 /* done */ 373 break; 374 } 375 } 376 } 377 378 379 NTSTATUS 380 WdmAudMixerInitialize( 381 IN PDEVICE_OBJECT DeviceObject) 382 { 383 MIXER_STATUS Status; 384 385 /* initialize the mixer library */ 386 Status = MMixerInitialize(&MixerContext, Enum, (PVOID)DeviceObject); 387 388 if (Status != MM_STATUS_SUCCESS) 389 { 390 /* failed to initialize mmixer library */ 391 DPRINT("MMixerInitialize failed with %lx\n", Status); 392 } 393 394 return Status; 395 } 396 397 NTSTATUS 398 WdmAudMixerCapabilities( 399 IN PDEVICE_OBJECT DeviceObject, 400 IN PWDMAUD_DEVICE_INFO DeviceInfo, 401 IN PWDMAUD_CLIENT ClientInfo, 402 IN PWDMAUD_DEVICE_EXTENSION DeviceExtension) 403 { 404 if (MMixerGetCapabilities(&MixerContext, DeviceInfo->DeviceIndex, &DeviceInfo->u.MixCaps) == MM_STATUS_SUCCESS) 405 return STATUS_SUCCESS; 406 407 return STATUS_INVALID_PARAMETER; 408 } 409 410 NTSTATUS 411 WdmAudControlOpenMixer( 412 IN PDEVICE_OBJECT DeviceObject, 413 IN PIRP Irp, 414 IN PWDMAUD_DEVICE_INFO DeviceInfo, 415 IN PWDMAUD_CLIENT ClientInfo) 416 { 417 HANDLE hMixer; 418 PWDMAUD_HANDLE Handles; 419 //PWDMAUD_DEVICE_EXTENSION DeviceExtension; 420 NTSTATUS Status; 421 PKEVENT EventObject = NULL; 422 423 DPRINT("WdmAudControlOpenMixer\n"); 424 425 //DeviceExtension = (PWDMAUD_DEVICE_EXTENSION)DeviceObject->DeviceExtension; 426 427 if (DeviceInfo->u.hNotifyEvent) 428 { 429 Status = ObReferenceObjectByHandle(DeviceInfo->u.hNotifyEvent, EVENT_MODIFY_STATE, *ExEventObjectType, UserMode, (LPVOID*)&EventObject, NULL); 430 431 if (!NT_SUCCESS(Status)) 432 { 433 DPRINT1("Invalid notify event passed %p from client %p\n", DeviceInfo->u.hNotifyEvent, ClientInfo); 434 DbgBreakPoint(); 435 return SetIrpIoStatus(Irp, STATUS_UNSUCCESSFUL, 0); 436 } 437 } 438 439 if (MMixerOpen(&MixerContext, DeviceInfo->DeviceIndex, ClientInfo, EventCallback, &hMixer) != MM_STATUS_SUCCESS) 440 { 441 ObDereferenceObject(EventObject); 442 DPRINT1("Failed to open mixer\n"); 443 return SetIrpIoStatus(Irp, STATUS_UNSUCCESSFUL, 0); 444 } 445 446 447 Handles = AllocateItem(NonPagedPool, sizeof(WDMAUD_HANDLE) * (ClientInfo->NumPins+1)); 448 449 if (Handles) 450 { 451 if (ClientInfo->NumPins) 452 { 453 RtlMoveMemory(Handles, ClientInfo->hPins, sizeof(WDMAUD_HANDLE) * ClientInfo->NumPins); 454 FreeItem(ClientInfo->hPins); 455 } 456 457 ClientInfo->hPins = Handles; 458 ClientInfo->hPins[ClientInfo->NumPins].Handle = hMixer; 459 ClientInfo->hPins[ClientInfo->NumPins].Type = MIXER_DEVICE_TYPE; 460 ClientInfo->hPins[ClientInfo->NumPins].NotifyEvent = EventObject; 461 ClientInfo->NumPins++; 462 } 463 else 464 { 465 ObDereferenceObject(EventObject); 466 return SetIrpIoStatus(Irp, STATUS_UNSUCCESSFUL, sizeof(WDMAUD_DEVICE_INFO)); 467 } 468 469 DeviceInfo->hDevice = hMixer; 470 471 return SetIrpIoStatus(Irp, STATUS_SUCCESS, sizeof(WDMAUD_DEVICE_INFO)); 472 } 473 474 NTSTATUS 475 WdmAudControlCloseMixer( 476 IN PDEVICE_OBJECT DeviceObject, 477 IN PIRP Irp, 478 IN PWDMAUD_DEVICE_INFO DeviceInfo, 479 IN PWDMAUD_CLIENT ClientInfo, 480 IN ULONG Index) 481 { 482 /* Remove event associated to this client */ 483 if (MMixerClose(&MixerContext, DeviceInfo->DeviceIndex, ClientInfo, EventCallback) != MM_STATUS_SUCCESS) 484 { 485 DPRINT1("Failed to close mixer\n"); 486 return SetIrpIoStatus(Irp, STATUS_UNSUCCESSFUL, sizeof(WDMAUD_DEVICE_INFO)); 487 } 488 489 /* Dereference event */ 490 if (ClientInfo->hPins[Index].NotifyEvent) 491 { 492 ObDereferenceObject(ClientInfo->hPins[Index].NotifyEvent); 493 ClientInfo->hPins[Index].NotifyEvent = NULL; 494 } 495 496 /* FIXME: do we need to free ClientInfo->hPins ? */ 497 return SetIrpIoStatus(Irp, STATUS_SUCCESS, sizeof(WDMAUD_DEVICE_INFO)); 498 } 499 500 VOID 501 WdmAudCloseAllMixers( 502 IN PDEVICE_OBJECT DeviceObject, 503 IN PWDMAUD_CLIENT ClientInfo, 504 IN ULONG Index) 505 { 506 ULONG DeviceCount, DeviceIndex; 507 508 /* Get all mixers */ 509 DeviceCount = GetSysAudioDeviceCount(DeviceObject); 510 511 /* Close every mixer attached to the device */ 512 for (DeviceIndex = 0; DeviceIndex < DeviceCount; DeviceIndex++) 513 { 514 if (MMixerClose(&MixerContext, DeviceIndex, ClientInfo, EventCallback) != MM_STATUS_SUCCESS) 515 { 516 DPRINT1("Failed to close mixer for device %lu\n", DeviceIndex); 517 } 518 } 519 520 /* Dereference event */ 521 if (ClientInfo->hPins[Index].NotifyEvent) 522 { 523 ObDereferenceObject(ClientInfo->hPins[Index].NotifyEvent); 524 ClientInfo->hPins[Index].NotifyEvent = NULL; 525 } 526 } 527 528 NTSTATUS 529 NTAPI 530 WdmAudGetControlDetails( 531 IN PDEVICE_OBJECT DeviceObject, 532 IN PIRP Irp, 533 IN PWDMAUD_DEVICE_INFO DeviceInfo, 534 IN PWDMAUD_CLIENT ClientInfo) 535 { 536 MIXER_STATUS Status; 537 538 /* clear hmixer type flag */ 539 DeviceInfo->Flags &= ~MIXER_OBJECTF_HMIXER; 540 541 /* query mmixer library */ 542 Status = MMixerGetControlDetails(&MixerContext, DeviceInfo->hDevice, DeviceInfo->DeviceIndex, DeviceInfo->Flags, &DeviceInfo->u.MixDetails); 543 544 if (Status == MM_STATUS_SUCCESS) 545 return SetIrpIoStatus(Irp, STATUS_SUCCESS, sizeof(WDMAUD_DEVICE_INFO)); 546 else 547 return SetIrpIoStatus(Irp, STATUS_UNSUCCESSFUL, sizeof(WDMAUD_DEVICE_INFO)); 548 } 549 550 NTSTATUS 551 NTAPI 552 WdmAudGetLineInfo( 553 IN PDEVICE_OBJECT DeviceObject, 554 IN PIRP Irp, 555 IN PWDMAUD_DEVICE_INFO DeviceInfo, 556 IN PWDMAUD_CLIENT ClientInfo) 557 { 558 MIXER_STATUS Status; 559 560 /* clear hmixer type flag */ 561 DeviceInfo->Flags &= ~MIXER_OBJECTF_HMIXER; 562 563 /* query mixer library */ 564 Status = MMixerGetLineInfo(&MixerContext, DeviceInfo->hDevice, DeviceInfo->DeviceIndex, DeviceInfo->Flags, &DeviceInfo->u.MixLine); 565 566 if (Status == MM_STATUS_SUCCESS) 567 return SetIrpIoStatus(Irp, STATUS_SUCCESS, sizeof(WDMAUD_DEVICE_INFO)); 568 else 569 return SetIrpIoStatus(Irp, STATUS_UNSUCCESSFUL, sizeof(WDMAUD_DEVICE_INFO)); 570 } 571 572 NTSTATUS 573 NTAPI 574 WdmAudGetLineControls( 575 IN PDEVICE_OBJECT DeviceObject, 576 IN PIRP Irp, 577 IN PWDMAUD_DEVICE_INFO DeviceInfo, 578 IN PWDMAUD_CLIENT ClientInfo) 579 { 580 MIXER_STATUS Status; 581 582 /* clear hmixer type flag */ 583 DeviceInfo->Flags &= ~MIXER_OBJECTF_HMIXER; 584 585 /* query mixer library */ 586 Status = MMixerGetLineControls(&MixerContext, DeviceInfo->hDevice, DeviceInfo->DeviceIndex, DeviceInfo->Flags, &DeviceInfo->u.MixControls); 587 588 if (Status == MM_STATUS_SUCCESS) 589 return SetIrpIoStatus(Irp, STATUS_SUCCESS, sizeof(WDMAUD_DEVICE_INFO)); 590 else 591 return SetIrpIoStatus(Irp, STATUS_UNSUCCESSFUL, sizeof(WDMAUD_DEVICE_INFO)); 592 593 594 } 595 596 NTSTATUS 597 NTAPI 598 WdmAudSetControlDetails( 599 IN PDEVICE_OBJECT DeviceObject, 600 IN PIRP Irp, 601 IN PWDMAUD_DEVICE_INFO DeviceInfo, 602 IN PWDMAUD_CLIENT ClientInfo) 603 { 604 MIXER_STATUS Status; 605 606 /* clear hmixer type flag */ 607 DeviceInfo->Flags &= ~MIXER_OBJECTF_HMIXER; 608 609 /* query mixer library */ 610 Status = MMixerSetControlDetails(&MixerContext, DeviceInfo->hDevice, DeviceInfo->DeviceIndex, DeviceInfo->Flags, &DeviceInfo->u.MixDetails); 611 612 if (Status == MM_STATUS_SUCCESS) 613 return SetIrpIoStatus(Irp, STATUS_SUCCESS, sizeof(WDMAUD_DEVICE_INFO)); 614 else 615 return SetIrpIoStatus(Irp, STATUS_UNSUCCESSFUL, sizeof(WDMAUD_DEVICE_INFO)); 616 } 617 618 NTSTATUS 619 NTAPI 620 WdmAudGetMixerEvent( 621 IN PDEVICE_OBJECT DeviceObject, 622 IN PIRP Irp, 623 IN PWDMAUD_DEVICE_INFO DeviceInfo, 624 IN PWDMAUD_CLIENT ClientInfo) 625 { 626 PLIST_ENTRY Entry; 627 PEVENT_ENTRY EventEntry; 628 629 /* enumerate event list and check if there is a new event */ 630 Entry = ClientInfo->MixerEventList.Flink; 631 632 while(Entry != &ClientInfo->MixerEventList) 633 { 634 /* grab event entry */ 635 EventEntry = (PEVENT_ENTRY)CONTAINING_RECORD(Entry, EVENT_ENTRY, Entry); 636 637 if (EventEntry->hMixer == DeviceInfo->hDevice) 638 { 639 /* found an entry */ 640 DeviceInfo->u.MixerEvent.hMixer = EventEntry->hMixer; 641 DeviceInfo->u.MixerEvent.NotificationType = EventEntry->NotificationType; 642 DeviceInfo->u.MixerEvent.Value = EventEntry->Value; 643 644 /* remove entry from list */ 645 RemoveEntryList(&EventEntry->Entry); 646 647 /* free event entry */ 648 FreeItem(EventEntry); 649 650 /* done */ 651 return SetIrpIoStatus(Irp, STATUS_SUCCESS, sizeof(WDMAUD_DEVICE_INFO)); 652 } 653 654 /* move to next */ 655 Entry = Entry->Flink; 656 } 657 658 /* no event entry available */ 659 return SetIrpIoStatus(Irp, STATUS_UNSUCCESSFUL, sizeof(WDMAUD_DEVICE_INFO)); 660 } 661 662 ULONG 663 WdmAudGetMixerDeviceCount() 664 { 665 return MMixerGetCount(&MixerContext); 666 } 667 668 ULONG 669 WdmAudGetWaveInDeviceCount() 670 { 671 return MMixerGetWaveInCount(&MixerContext); 672 } 673 674 ULONG 675 WdmAudGetWaveOutDeviceCount() 676 { 677 return MMixerGetWaveOutCount(&MixerContext); 678 } 679 680 ULONG 681 WdmAudGetMidiInDeviceCount() 682 { 683 return MMixerGetMidiInCount(&MixerContext); 684 } 685 686 ULONG 687 WdmAudGetMidiOutDeviceCount() 688 { 689 return MMixerGetWaveOutCount(&MixerContext); 690 } 691 692 NTSTATUS 693 WdmAudGetPnpNameByIndexAndType( 694 IN ULONG DeviceIndex, 695 IN SOUND_DEVICE_TYPE DeviceType, 696 OUT LPWSTR *DevicePath) 697 { 698 if (DeviceType == WAVE_IN_DEVICE_TYPE || DeviceType == WAVE_OUT_DEVICE_TYPE) 699 { 700 if (MMixerGetWaveDevicePath(&MixerContext, DeviceType == WAVE_IN_DEVICE_TYPE, DeviceIndex, DevicePath) == MM_STATUS_SUCCESS) 701 return STATUS_SUCCESS; 702 else 703 return STATUS_UNSUCCESSFUL; 704 } 705 else if (DeviceType == MIDI_IN_DEVICE_TYPE || DeviceType == MIDI_OUT_DEVICE_TYPE) 706 { 707 if (MMixerGetMidiDevicePath(&MixerContext, DeviceType == MIDI_IN_DEVICE_TYPE, DeviceIndex, DevicePath) == MM_STATUS_SUCCESS) 708 return STATUS_SUCCESS; 709 else 710 return STATUS_UNSUCCESSFUL; 711 } 712 else if (DeviceType == MIXER_DEVICE_TYPE) 713 { 714 UNIMPLEMENTED; 715 } 716 717 return STATUS_UNSUCCESSFUL; 718 } 719 720 NTSTATUS 721 WdmAudWaveCapabilities( 722 IN PDEVICE_OBJECT DeviceObject, 723 IN PWDMAUD_DEVICE_INFO DeviceInfo, 724 IN PWDMAUD_CLIENT ClientInfo, 725 IN PWDMAUD_DEVICE_EXTENSION DeviceExtension) 726 { 727 MIXER_STATUS Status = MM_STATUS_UNSUCCESSFUL; 728 729 if (DeviceInfo->DeviceType == WAVE_IN_DEVICE_TYPE) 730 { 731 /* get capabilities */ 732 Status = MMixerWaveInCapabilities(&MixerContext, DeviceInfo->DeviceIndex, &DeviceInfo->u.WaveInCaps); 733 } 734 else if (DeviceInfo->DeviceType == WAVE_OUT_DEVICE_TYPE) 735 { 736 /* get capabilities */ 737 Status = MMixerWaveOutCapabilities(&MixerContext, DeviceInfo->DeviceIndex, &DeviceInfo->u.WaveOutCaps); 738 } 739 740 if (Status == MM_STATUS_SUCCESS) 741 return STATUS_SUCCESS; 742 else 743 return Status; 744 } 745 746 NTSTATUS 747 WdmAudMidiCapabilities( 748 IN PDEVICE_OBJECT DeviceObject, 749 IN PWDMAUD_DEVICE_INFO DeviceInfo, 750 IN PWDMAUD_CLIENT ClientInfo, 751 IN PWDMAUD_DEVICE_EXTENSION DeviceExtension) 752 { 753 MIXER_STATUS Status = MM_STATUS_UNSUCCESSFUL; 754 755 if (DeviceInfo->DeviceType == MIDI_IN_DEVICE_TYPE) 756 { 757 /* get capabilities */ 758 Status = MMixerMidiInCapabilities(&MixerContext, DeviceInfo->DeviceIndex, &DeviceInfo->u.MidiInCaps); 759 } 760 else if (DeviceInfo->DeviceType == WAVE_OUT_DEVICE_TYPE) 761 { 762 /* get capabilities */ 763 Status = MMixerMidiOutCapabilities(&MixerContext, DeviceInfo->DeviceIndex, &DeviceInfo->u.MidiOutCaps); 764 } 765 766 if (Status == MM_STATUS_SUCCESS) 767 return STATUS_SUCCESS; 768 else 769 return STATUS_UNSUCCESSFUL; 770 } 771 772 773 MIXER_STATUS 774 CreatePinCallback( 775 IN PVOID Ctx, 776 IN ULONG VirtualDeviceId, 777 IN ULONG PinId, 778 IN HANDLE hFilter, 779 IN PKSPIN_CONNECT PinConnect, 780 IN ACCESS_MASK DesiredAccess, 781 OUT PHANDLE PinHandle) 782 { 783 ULONG BytesReturned; 784 SYSAUDIO_INSTANCE_INFO InstanceInfo; 785 NTSTATUS Status; 786 ULONG FreeIndex; 787 PPIN_CREATE_CONTEXT Context = (PPIN_CREATE_CONTEXT)Ctx; 788 789 /* setup property request */ 790 InstanceInfo.Property.Set = KSPROPSETID_Sysaudio; 791 InstanceInfo.Property.Id = KSPROPERTY_SYSAUDIO_INSTANCE_INFO; 792 InstanceInfo.Property.Flags = KSPROPERTY_TYPE_SET; 793 InstanceInfo.Flags = 0; 794 InstanceInfo.DeviceNumber = VirtualDeviceId; 795 796 /* attach to virtual device */ 797 Status = KsSynchronousIoControlDevice(Context->DeviceExtension->FileObject, KernelMode, IOCTL_KS_PROPERTY, (PVOID)&InstanceInfo, sizeof(SYSAUDIO_INSTANCE_INFO), NULL, 0, &BytesReturned); 798 799 if (!NT_SUCCESS(Status)) 800 return MM_STATUS_UNSUCCESSFUL; 801 802 /* close existing pin */ 803 FreeIndex = ClosePin(Context->ClientInfo, VirtualDeviceId, PinId, Context->DeviceType); 804 805 /* now create the pin */ 806 Status = KsCreatePin(Context->DeviceExtension->hSysAudio, PinConnect, DesiredAccess, PinHandle); 807 808 /* check for success */ 809 if (!NT_SUCCESS(Status)) 810 return MM_STATUS_UNSUCCESSFUL; 811 812 /* store the handle */ 813 Status = InsertPinHandle(Context->ClientInfo, VirtualDeviceId, PinId, Context->DeviceType, *PinHandle, FreeIndex); 814 if (!NT_SUCCESS(Status)) 815 { 816 /* failed to insert handle */ 817 ZwClose(*PinHandle); 818 return MM_STATUS_UNSUCCESSFUL; 819 } 820 821 return MM_STATUS_SUCCESS; 822 } 823 824 NTSTATUS 825 WdmAudControlOpenWave( 826 IN PDEVICE_OBJECT DeviceObject, 827 IN PIRP Irp, 828 IN PWDMAUD_DEVICE_INFO DeviceInfo, 829 IN PWDMAUD_CLIENT ClientInfo) 830 { 831 MIXER_STATUS Status; 832 PIN_CREATE_CONTEXT Context; 833 834 Context.ClientInfo = ClientInfo; 835 Context.DeviceExtension = (PWDMAUD_DEVICE_EXTENSION)DeviceObject->DeviceExtension; 836 Context.DeviceType = DeviceInfo->DeviceType; 837 838 Status = MMixerOpenWave(&MixerContext, DeviceInfo->DeviceIndex, DeviceInfo->DeviceType == WAVE_IN_DEVICE_TYPE, &DeviceInfo->u.WaveFormatEx, CreatePinCallback, &Context, &DeviceInfo->hDevice); 839 840 if (Status == MM_STATUS_SUCCESS) 841 return SetIrpIoStatus(Irp, STATUS_SUCCESS, sizeof(WDMAUD_DEVICE_INFO)); 842 else 843 return SetIrpIoStatus(Irp, STATUS_NOT_SUPPORTED, sizeof(WDMAUD_DEVICE_INFO)); 844 } 845 846 NTSTATUS 847 WdmAudControlOpenMidi( 848 IN PDEVICE_OBJECT DeviceObject, 849 IN PIRP Irp, 850 IN PWDMAUD_DEVICE_INFO DeviceInfo, 851 IN PWDMAUD_CLIENT ClientInfo) 852 { 853 MIXER_STATUS Status; 854 PIN_CREATE_CONTEXT Context; 855 856 Context.ClientInfo = ClientInfo; 857 Context.DeviceExtension = (PWDMAUD_DEVICE_EXTENSION)DeviceObject->DeviceExtension; 858 Context.DeviceType = DeviceInfo->DeviceType; 859 860 Status = MMixerOpenMidi(&MixerContext, DeviceInfo->DeviceIndex, DeviceInfo->DeviceType == MIDI_IN_DEVICE_TYPE, CreatePinCallback, &Context, &DeviceInfo->hDevice); 861 862 if (Status == MM_STATUS_SUCCESS) 863 return SetIrpIoStatus(Irp, STATUS_SUCCESS, sizeof(WDMAUD_DEVICE_INFO)); 864 else 865 return SetIrpIoStatus(Irp, STATUS_NOT_SUPPORTED, sizeof(WDMAUD_DEVICE_INFO)); 866 } 867