Dynamic Enumeration

Dynamic enumeration is a driver's ability to detect and report changes to the number and type of devices that are connected to the system while the system is running.

Bus drivers must use dynamic enumeration if the number or types of devices that are connected to the parent device depend on a system's configuration. Some of these devices might be always connected to the system, and some might be plugged in and unplugged while the system is running.

For example, the number and type of devices that are plugged into a system's PCI bus are system-dependent, but they are permanent unless a user turns off power, opens the case, and adds or removes a device by using a screwdriver. On the other hand, a user can add or remove USB devices by plugging in or unplugging a cable while the system is running.

Dynamic Child Lists

The framework enables drivers to support dynamic enumeration by providing framework child-list objects. Each child-list object represents a list of child devices that are connected to a parent device. The bus driver for the parent device must identify the parent's child devices, add them to the parent device's child list, and create a physical device object (PDO) for each child.

Each time a driver creates a framework device object that represents an FDO for a device, the framework creates an empty, default child list for the device. Your driver can obtain a handle to a device's default child list by calling WdfFdoGetDefaultChildList. Typically, if you are writing a bus driver that enumerates a device's children, your driver can add children to the default child list. If you need to create additional child lists, your driver can call WdfChildListCreate.

Before a driver can use a child list, it must configure the child-list object by initializing a WDF_CHILD_LIST_CONFIG structure and passing the structure to either WdfFdoInitSetDefaultChildListConfig, for the default child list, or to WdfChildListCreate, for additional child lists.

Dynamic Child Descriptions

Each time a bus driver identifies a child device, it must add the child device's description to a child list. A child description consists of a required identification description and an optional address description.

Identification Description An identification description is a structure that contains information that uniquely identifies each device that the driver enumerates. The driver defines this structure, but its first member must be a WDF_CHILD_IDENTIFICATION_DESCRIPTION_HEADER structure.

Typically, an identification description contains a device's device identification strings, possibly a serial number, and information about the device's location on the bus, such as a slot number.

The driver can provide the following set of callback functions, which allow the framework to manipulate the information in an identification description:

Typically, you will need to provide these callback functions if your driver's identification description structures contain pointers to dynamically allocated buffers. For more information about the purpose of these callback functions, see their reference pages.

Address Description An address description is a structure that contains information that the driver requires so that it can access the device on its bus, if the information can change while the device is plugged in. The driver defines this structure, but its first member must be a WDF_CHILD_ADDRESS_DESCRIPTION_HEADER structure.

Address descriptions are optional. If a device's address information cannot change between the time the device is plugged in and the time it is unplugged, all of the device's address information can be stored in an identification description. For example, USB controllers assign addresses to devices when the devices are plugged in, and these addresses do not change.

On the other hand, some buses use addressing information that can change. For example, the IEEE 1394 bus uses a "generation count," which is the number of bus resets that have occurred. Each asynchronous I/O request to an IEEE 1394 device must include the generation count. Because this address information can change, your driver must store it in an address description.

The driver can provide the following set of callback functions to manipulate the information in an address description:

Typically, you will need to provide these callback functions if your driver's address description structures contain pointers to dynamically allocated buffers. For more information about the purpose of these callback functions, see their reference pages.

Adding Devices to a Dynamic Child List

When the framework calls a bus driver's EvtDriverDeviceAdd callback function, the callback function must call WdfDeviceCreate to create an FDO for the parent device, which is typically a bus adapter. For more information about creating an FDO, see Creating Device Objects in a Function Driver. The driver must then enumerate the parent device's children and add the children to a child list.

Optionally, the driver can call WdfDeviceSetBusInformationForChildren to provide the framework with information about the bus. Doing so is recommended because it makes it easier for child devices and apps to identify the bus.

To add children to a child list, the driver must call WdfChildListAddOrUpdateChildDescriptionAsPresent for each child device that it finds. This call informs the framework that a driver has discovered a child device that is connected to a parent device. When your driver calls WdfChildListAddOrUpdateChildDescriptionAsPresent, it supplies an identification description and, optionally, an address description.

After the driver calls WdfChildListAddOrUpdateChildDescriptionAsPresent to report a new device, the framework informs the PnP manager that the new device exists. The PnP manager then builds a device stack and driver stack for the new device. As part of this process, the framework calls the bus driver's EvtChildListCreateDevice callback function. This callback function must call WdfDeviceCreate to create a PDO for the new device.

Typically, several child devices are connected to a parent, so the bus driver will need to call WdfChildListAddOrUpdateChildDescriptionAsPresent several times. The most efficient way to do this is the following:

  1. Call WdfChildListBeginScan.

  2. Call WdfChildListAddOrUpdateChildDescriptionAsPresent for each child device.

  3. Call WdfChildListEndScan.

If you surround your driver's dynamic enumeration with calls to WdfChildListBeginScan and WdfChildListEndScan, the framework stores all of the changes to the child list, and notifies the PnP manager of the changes when the driver calls WdfChildListEndScan. At some later time, the framework calls the bus driver's EvtChildListCreateDevice callback function for each device in the child list. This callback function calls WdfDeviceCreate to create a PDO for each new device.

When your driver calls WdfChildListBeginScan, the framework marks all previously reported devices as no longer being present. Therefore, the driver must call WdfChildListAddOrUpdateChildDescriptionAsPresent for all children that the driver can detect, not just newly discovered children. To add a single child to a child list, the driver can make a single call to WdfChildListUpdateAllChildDescriptionsAsPresent without first calling WdfChildListBeginScan.

Updating a Dynamic Child List

There are two common ways to update the information in a dynamic child list:

  1. When a parent device receives an interrupt that indicates the arrival or removal of a child, the driver's EvtInterruptDpc callback function calls WdfChildListAddOrUpdateChildDescriptionAsPresent if a device has been plugged in or WdfChildListUpdateChildDescriptionAsMissing if a device has been unplugged.

  2. The driver can provide an EvtChildListScanForChildren callback function, which the framework calls each time the parent device enters its working (D0) state. This callback function should enumerate all child devices by calling WdfChildListBeginScan, WdfChildListAddOrUpdateChildDescriptionAsPresent (or WdfChildListUpdateAllChildDescriptionsAsPresent), and WdfChildListEndScan.

You can use one or both of these techniques in your driver.

Traversing a Dynamic Child List

If you want your driver to examine the contents of a child list, it can traverse the list by using one of the following techniques:

  • To obtain the contents of each child device description, one at a time, the driver can:

    1. Call WdfChildListBeginIteration.
    2. Call WdfChildListRetrieveNextDevice, as many times as necessary.
    3. Call WdfChildListEndIteration.

    When calling WdfChildListBeginIteration, the driver specifies a WDF_RETRIEVE_CHILD_FLAGS-typed flag that indicates whether the framework should retrieve all device descriptions or only a subset. When WdfChildListRetrieveNextDevice finds a match, it retrieves the child device's identification and address descriptions, plus a handle to its device object.

  • If you need to obtain the address description that is currently contained in a child device description, your driver can call WdfChildListRetrieveAddressDescription, specifying an identification description. The framework traverses the child list until it finds a child device with a matching identification description, and then it retrieves the address description.

  • If you need to obtain a handle to the framework device object that is associated with a particular child device, your driver can call WdfChildListRetrievePdo. The framework traverses the child list until it finds a child device with a matching identification description, and then it returns a device object handle. Be sure to wrap the call with WdfChildListBeginIteration and WdfChildListEndIteration to protect the caller from sudden PnP removal of the PDO on another thread.

Accessing a PDO's Identification and Address Descriptions

Your driver can call the following methods to access a PDO's identification description or address description:

Handling re-enumeration requests

Framework-based bus drivers that support dynamic enumeration can receive a request to reenumerate a particular child device through the REENUMERATE_SELF_INTERFACE_STANDARD interface. For more info, see Handling Enumeration Requests