123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148 |
- LINUX HOTPLUGGING
- In hotpluggable busses like USB (and Cardbus PCI), end-users plug devices
- into the bus with power on. In most cases, users expect the devices to become
- immediately usable. That means the system must do many things, including:
- - Find a driver that can handle the device. That may involve
- loading a kernel module; newer drivers can use module-init-tools
- to publish their device (and class) support to user utilities.
- - Bind a driver to that device. Bus frameworks do that using a
- device driver's probe() routine.
- - Tell other subsystems to configure the new device. Print
- queues may need to be enabled, networks brought up, disk
- partitions mounted, and so on. In some cases these will
- be driver-specific actions.
- This involves a mix of kernel mode and user mode actions. Making devices
- be immediately usable means that any user mode actions can't wait for an
- administrator to do them: the kernel must trigger them, either passively
- (triggering some monitoring daemon to invoke a helper program) or
- actively (calling such a user mode helper program directly).
- Those triggered actions must support a system's administrative policies;
- such programs are called "policy agents" here. Typically they involve
- shell scripts that dispatch to more familiar administration tools.
- Because some of those actions rely on information about drivers (metadata)
- that is currently available only when the drivers are dynamically linked,
- you get the best hotplugging when you configure a highly modular system.
- KERNEL HOTPLUG HELPER (/sbin/hotplug)
- There is a kernel parameter: /proc/sys/kernel/hotplug, which normally
- holds the pathname "/sbin/hotplug". That parameter names a program
- which the kernel may invoke at various times.
- The /sbin/hotplug program can be invoked by any subsystem as part of its
- reaction to a configuration change, from a thread in that subsystem.
- Only one parameter is required: the name of a subsystem being notified of
- some kernel event. That name is used as the first key for further event
- dispatch; any other argument and environment parameters are specified by
- the subsystem making that invocation.
- Hotplug software and other resources is available at:
- http://linux-hotplug.sourceforge.net
- Mailing list information is also available at that site.
- --------------------------------------------------------------------------
- USB POLICY AGENT
- The USB subsystem currently invokes /sbin/hotplug when USB devices
- are added or removed from system. The invocation is done by the kernel
- hub workqueue [hub_wq], or else as part of root hub initialization
- (done by init, modprobe, kapmd, etc). Its single command line parameter
- is the string "usb", and it passes these environment variables:
- ACTION ... "add", "remove"
- PRODUCT ... USB vendor, product, and version codes (hex)
- TYPE ... device class codes (decimal)
- INTERFACE ... interface 0 class codes (decimal)
- If "usbdevfs" is configured, DEVICE and DEVFS are also passed. DEVICE is
- the pathname of the device, and is useful for devices with multiple and/or
- alternate interfaces that complicate driver selection. By design, USB
- hotplugging is independent of "usbdevfs": you can do most essential parts
- of USB device setup without using that filesystem, and without running a
- user mode daemon to detect changes in system configuration.
- Currently available policy agent implementations can load drivers for
- modules, and can invoke driver-specific setup scripts. The newest ones
- leverage USB module-init-tools support. Later agents might unload drivers.
- USB MODUTILS SUPPORT
- Current versions of module-init-tools will create a "modules.usbmap" file
- which contains the entries from each driver's MODULE_DEVICE_TABLE. Such
- files can be used by various user mode policy agents to make sure all the
- right driver modules get loaded, either at boot time or later.
- See <linux/usb.h> for full information about such table entries; or look
- at existing drivers. Each table entry describes one or more criteria to
- be used when matching a driver to a device or class of devices. The
- specific criteria are identified by bits set in "match_flags", paired
- with field values. You can construct the criteria directly, or with
- macros such as these, and use driver_info to store more information.
- USB_DEVICE (vendorId, productId)
- ... matching devices with specified vendor and product ids
- USB_DEVICE_VER (vendorId, productId, lo, hi)
- ... like USB_DEVICE with lo <= productversion <= hi
- USB_INTERFACE_INFO (class, subclass, protocol)
- ... matching specified interface class info
- USB_DEVICE_INFO (class, subclass, protocol)
- ... matching specified device class info
- A short example, for a driver that supports several specific USB devices
- and their quirks, might have a MODULE_DEVICE_TABLE like this:
- static const struct usb_device_id mydriver_id_table[] = {
- { USB_DEVICE (0x9999, 0xaaaa), driver_info: QUIRK_X },
- { USB_DEVICE (0xbbbb, 0x8888), driver_info: QUIRK_Y|QUIRK_Z },
- ...
- { } /* end with an all-zeroes entry */
- };
- MODULE_DEVICE_TABLE(usb, mydriver_id_table);
- Most USB device drivers should pass these tables to the USB subsystem as
- well as to the module management subsystem. Not all, though: some driver
- frameworks connect using interfaces layered over USB, and so they won't
- need such a "struct usb_driver".
- Drivers that connect directly to the USB subsystem should be declared
- something like this:
- static struct usb_driver mydriver = {
- .name = "mydriver",
- .id_table = mydriver_id_table,
- .probe = my_probe,
- .disconnect = my_disconnect,
- /*
- if using the usb chardev framework:
- .minor = MY_USB_MINOR_START,
- .fops = my_file_ops,
- if exposing any operations through usbdevfs:
- .ioctl = my_ioctl,
- */
- };
- When the USB subsystem knows about a driver's device ID table, it's used when
- choosing drivers to probe(). The thread doing new device processing checks
- drivers' device ID entries from the MODULE_DEVICE_TABLE against interface and
- device descriptors for the device. It will only call probe() if there is a
- match, and the third argument to probe() will be the entry that matched.
- If you don't provide an id_table for your driver, then your driver may get
- probed for each new device; the third parameter to probe() will be null.
|