Драйвер для нескольких устройств? (КМДФ/ВДФ)

Я только что написал USB-драйвер KMDF. Теперь хочу подключить к ПК несколько (не менее четырех) устройств. С чего начать? Я заметил, что когда я подключаю второе устройство к ПК, оно использует тот же экземпляр драйвера, что и для первого подключенного устройства. EvtDeviceAdd(...) запускается один раз для каждого устройства, и, поскольку у меня нет никакой обработки для нескольких устройств, все становится странно... Прямо сейчас мой EvtDeviceAdd выглядит так:

NTSTATUS EvtDeviceAdd(IN WDFDRIVER Driver, IN PWDFDEVICE_INIT DeviceInit) {
    WDF_PNPPOWER_EVENT_CALLBACKS        pnpPowerCallbacks;
    WDF_OBJECT_ATTRIBUTES               attributes;
    NTSTATUS                            status;
    WDFDEVICE                           device;
    WDF_DEVICE_PNP_CAPABILITIES         pnpCaps;
    WDF_IO_QUEUE_CONFIG                 ioQueueConfig;
    PDEVICE_CONTEXT                     pDevContext;
    WDFQUEUE                            queue;
    PWSTR                               driverRegistryPath;

    UNREFERENCED_PARAMETER(Driver);
    PAGED_CODE();

    DbgPrint("New device was added\n");

    WDF_PNPPOWER_EVENT_CALLBACKS_INIT(&pnpPowerCallbacks);
    pnpPowerCallbacks.EvtDevicePrepareHardware = EvtDevicePrepareHardware;
    WdfDeviceInitSetPnpPowerEventCallbacks(DeviceInit, &pnpPowerCallbacks);

    WdfDeviceInitSetIoType(DeviceInit, WdfDeviceIoBuffered);

    WDF_OBJECT_ATTRIBUTES_INIT_CONTEXT_TYPE(&attributes, DEVICE_CONTEXT);

    status = WdfDeviceCreate(&DeviceInit, &attributes, &device);
    if (!NT_SUCCESS(status)) {
        DbgPrint("WdfDeviceCreate failed with Status code %!STATUS!\n", status);
        return status;
    }

    pDevContext = GetDeviceContext(device);

    WDF_DEVICE_PNP_CAPABILITIES_INIT(&pnpCaps);
    pnpCaps.SurpriseRemovalOK = WdfTrue;

    WdfDeviceSetPnpCapabilities(device, &pnpCaps);

    WDF_IO_QUEUE_CONFIG_INIT_DEFAULT_QUEUE(&ioQueueConfig, WdfIoQueueDispatchParallel);

    ioQueueConfig.EvtIoRead = EvtIoRead;
    ioQueueConfig.EvtIoWrite = EvtIoWrite;
    ioQueueConfig.EvtIoDeviceControl = EvtIoDeviceControl;
    ioQueueConfig.PowerManaged = WdfTrue;

    status = WdfIoQueueCreate(device, &ioQueueConfig, WDF_NO_OBJECT_ATTRIBUTES, &queue);
    if (!NT_SUCCESS(status)) {
        DbgPrint("WdfIoQueueCreate failed  %!STATUS!\n", status);
        return status;
    }
    pDevContext->DeviceIOControlQueue = queue;

    status = WdfDeviceCreateDeviceInterface(device, (LPGUID) &GUID_DEVINTERFACE_MYDEVICE, NULL);

    if (!NT_SUCCESS(status)) {
        DbgPrint("WdfDeviceCreateDeviceInterface failed  %!STATUS!\n", status);
        return status;
    }
}

С чего начать? Есть хорошие примеры?


person user872661    schedule 15.09.2011    source источник


Ответы (1)


В памяти находится только один экземпляр драйвера для всех подключенных устройств (это синглтон). Обращения ОС к драйверу сопровождаются соответствующим контекстом устройства, и с этого момента устройства не должны мешать работе друг друга. Проблемы начинаются, если используются непостоянные глобальные/статические переменные. Поскольку пространство ядра является общим, такие переменные будут фактически общими и доступными со всех подключенных устройств. По этой причине глобальные/статические данные не должны зависеть от устройства и должны быть защищены, поскольку это общий ресурс. В WDK есть несколько примеров, демонстрирующих драйверы для нескольких устройств.

person SomeWittyUsername    schedule 22.10.2012