mirror of
https://github.com/reactos/reactos.git
synced 2024-12-28 18:15:11 +00:00
197 lines
4.9 KiB
C
197 lines
4.9 KiB
C
/*
|
|
* PROJECT: ReactOS
|
|
* LICENSE: GPL - See COPYING in the top level directory
|
|
* PURPOSE: Audio Service Plug and Play
|
|
* COPYRIGHT: Copyright 2007 Andrew Greenwood
|
|
*/
|
|
|
|
#include "audiosrv.h"
|
|
|
|
#include <winreg.h>
|
|
#include <winuser.h>
|
|
#include <mmsystem.h>
|
|
#include <setupapi.h>
|
|
#include <ks.h>
|
|
#include <ksmedia.h>
|
|
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
static HDEVNOTIFY device_notification_handle = NULL;
|
|
|
|
/*
|
|
Finds all devices within the KSCATEGORY_AUDIO category and puts them
|
|
in the shared device list.
|
|
*/
|
|
|
|
BOOL
|
|
ProcessExistingDevices(VOID)
|
|
{
|
|
SP_DEVICE_INTERFACE_DATA interface_data;
|
|
SP_DEVINFO_DATA device_data;
|
|
PSP_DEVICE_INTERFACE_DETAIL_DATA_W detail_data;
|
|
HDEVINFO dev_info;
|
|
DWORD length;
|
|
int index = 0;
|
|
|
|
const GUID category_guid = {STATIC_KSCATEGORY_AUDIO};
|
|
|
|
dev_info = SetupDiGetClassDevsExW(&category_guid,
|
|
NULL,
|
|
NULL,
|
|
DIGCF_PRESENT | DIGCF_DEVICEINTERFACE,
|
|
NULL,
|
|
NULL,
|
|
NULL);
|
|
|
|
interface_data.cbSize = sizeof(interface_data);
|
|
interface_data.Reserved = 0;
|
|
|
|
/* Enumerate the devices within the category */
|
|
index = 0;
|
|
|
|
length = sizeof(SP_DEVICE_INTERFACE_DETAIL_DATA)
|
|
+ (MAX_PATH * sizeof(WCHAR));
|
|
|
|
detail_data =
|
|
(PSP_DEVICE_INTERFACE_DETAIL_DATA_W)HeapAlloc(GetProcessHeap(),
|
|
0,
|
|
length);
|
|
|
|
if ( ! detail_data )
|
|
{
|
|
DPRINT("failed to allocate detail_data\n");
|
|
return TRUE;
|
|
}
|
|
|
|
while (
|
|
SetupDiEnumDeviceInterfaces(dev_info,
|
|
NULL,
|
|
&category_guid,
|
|
index,
|
|
&interface_data) )
|
|
{
|
|
PnP_AudioDevice* list_node;
|
|
|
|
ZeroMemory(detail_data, length);
|
|
|
|
/* NOTE: We don't actually use device_data... */
|
|
detail_data->cbSize = sizeof(SP_DEVICE_INTERFACE_DETAIL_DATA_W);
|
|
device_data.cbSize = sizeof(device_data);
|
|
device_data.Reserved = 0;
|
|
SetupDiGetDeviceInterfaceDetailW(dev_info,
|
|
&interface_data,
|
|
detail_data,
|
|
length,
|
|
NULL,
|
|
&device_data);
|
|
|
|
list_node = CreateDeviceDescriptor(detail_data->DevicePath, TRUE);
|
|
AppendAudioDeviceToList(list_node);
|
|
DestroyDeviceDescriptor(list_node);
|
|
|
|
/* TODO: Cleanup the device we enumerated? */
|
|
|
|
index ++;
|
|
};
|
|
|
|
HeapFree(GetProcessHeap(), 0, detail_data);
|
|
|
|
SetupDiDestroyDeviceInfoList(dev_info);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
/*
|
|
Add new devices to the list as they arrive.
|
|
*/
|
|
|
|
DWORD
|
|
ProcessDeviceArrival(DEV_BROADCAST_DEVICEINTERFACE* device)
|
|
{
|
|
PnP_AudioDevice* list_node;
|
|
list_node = CreateDeviceDescriptor(device->dbcc_name, TRUE);
|
|
AppendAudioDeviceToList(list_node);
|
|
DestroyDeviceDescriptor(list_node);
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
|
|
/*
|
|
Request notification of device additions/removals.
|
|
*/
|
|
|
|
BOOL
|
|
RegisterForDeviceNotifications(VOID)
|
|
{
|
|
DEV_BROADCAST_DEVICEINTERFACE notification_filter;
|
|
|
|
const GUID wdmaud_guid = {STATIC_KSCATEGORY_AUDIO};
|
|
|
|
ZeroMemory(¬ification_filter, sizeof(notification_filter));
|
|
notification_filter.dbcc_size = sizeof(DEV_BROADCAST_DEVICEINTERFACE);
|
|
notification_filter.dbcc_devicetype = DBT_DEVTYP_DEVICEINTERFACE;
|
|
notification_filter.dbcc_classguid = wdmaud_guid;
|
|
|
|
device_notification_handle =
|
|
RegisterDeviceNotificationW((HANDLE) service_status_handle,
|
|
¬ification_filter,
|
|
DEVICE_NOTIFY_SERVICE_HANDLE
|
|
/* |
|
|
DEVICE_NOTIFY_ALL_INTERFACE_CLASSES*/);
|
|
if (!device_notification_handle)
|
|
{
|
|
DPRINT("failed with error %d\n", GetLastError());
|
|
}
|
|
|
|
return ( device_notification_handle != NULL );
|
|
}
|
|
|
|
|
|
/*
|
|
When we're not interested in device notifications any more, this gets
|
|
called.
|
|
*/
|
|
|
|
VOID
|
|
UnregisterDeviceNotifications(VOID)
|
|
{
|
|
/* TODO -- NOT IMPLEMENTED! */
|
|
|
|
if (device_notification_handle)
|
|
{
|
|
/* TODO */
|
|
device_notification_handle = NULL;
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
Device events from the main service handler get passed to this.
|
|
*/
|
|
|
|
DWORD
|
|
HandleDeviceEvent(
|
|
DWORD dwEventType,
|
|
LPVOID lpEventData)
|
|
{
|
|
switch (dwEventType)
|
|
{
|
|
case DBT_DEVICEARRIVAL:
|
|
{
|
|
DEV_BROADCAST_DEVICEINTERFACE* incoming_device =
|
|
(DEV_BROADCAST_DEVICEINTERFACE*)lpEventData;
|
|
|
|
return ProcessDeviceArrival(incoming_device);
|
|
}
|
|
|
|
default :
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
return NO_ERROR;
|
|
}
|