19454b2d8SWarner Losh#- 29f7f340aSWarner Losh# Copyright (c) 1998-2004 Doug Rabson 3b1bf6610SDoug Rabson# All rights reserved. 4b1bf6610SDoug Rabson# 5b1bf6610SDoug Rabson# Redistribution and use in source and binary forms, with or without 6b1bf6610SDoug Rabson# modification, are permitted provided that the following conditions 7b1bf6610SDoug Rabson# are met: 8b1bf6610SDoug Rabson# 1. Redistributions of source code must retain the above copyright 9b1bf6610SDoug Rabson# notice, this list of conditions and the following disclaimer. 10b1bf6610SDoug Rabson# 2. Redistributions in binary form must reproduce the above copyright 11b1bf6610SDoug Rabson# notice, this list of conditions and the following disclaimer in the 12b1bf6610SDoug Rabson# documentation and/or other materials provided with the distribution. 13b1bf6610SDoug Rabson# 14b1bf6610SDoug Rabson# THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 15b1bf6610SDoug Rabson# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 16b1bf6610SDoug Rabson# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 17b1bf6610SDoug Rabson# ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 18b1bf6610SDoug Rabson# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 19b1bf6610SDoug Rabson# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 20b1bf6610SDoug Rabson# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 21b1bf6610SDoug Rabson# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 22b1bf6610SDoug Rabson# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 23b1bf6610SDoug Rabson# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 24b1bf6610SDoug Rabson# SUCH DAMAGE. 25b1bf6610SDoug Rabson# 26b1bf6610SDoug Rabson# 27b1bf6610SDoug Rabson 28f7b77691SDoug Rabson#include <sys/bus.h> 29f7b77691SDoug Rabson 3067f8f14aSDoug Rabson/** 3167f8f14aSDoug Rabson * @defgroup DEVICE device - KObj methods for all device drivers 3267f8f14aSDoug Rabson * @brief A basic set of methods required for all device drivers. 3367f8f14aSDoug Rabson * 3467f8f14aSDoug Rabson * The device interface is used to match devices to drivers during 3567f8f14aSDoug Rabson * autoconfiguration and provides methods to allow drivers to handle 3667f8f14aSDoug Rabson * system-wide events such as suspend, resume or shutdown. 3767f8f14aSDoug Rabson * @{ 3867f8f14aSDoug Rabson */ 397a8ecb9eSNicolas SouchuINTERFACE device; 40b1bf6610SDoug Rabson 41*a21a2da5SColin Percival# Needed for timestamping device probe/attach calls 42*a21a2da5SColin PercivalHEADER { 43*a21a2da5SColin Percival #include <sys/tslog.h> 44*a21a2da5SColin Percival} 45*a21a2da5SColin Percival 46b1bf6610SDoug Rabson# 478b2970bbSDoug Rabson# Default implementations of some methods. 488b2970bbSDoug Rabson# 498b2970bbSDoug RabsonCODE { 508b2970bbSDoug Rabson static int null_shutdown(device_t dev) 518b2970bbSDoug Rabson { 528b2970bbSDoug Rabson return 0; 538b2970bbSDoug Rabson } 548b2970bbSDoug Rabson 558b2970bbSDoug Rabson static int null_suspend(device_t dev) 568b2970bbSDoug Rabson { 578b2970bbSDoug Rabson return 0; 588b2970bbSDoug Rabson } 598b2970bbSDoug Rabson 608b2970bbSDoug Rabson static int null_resume(device_t dev) 618b2970bbSDoug Rabson { 628b2970bbSDoug Rabson return 0; 638b2970bbSDoug Rabson } 64d0d4cc63SWarner Losh 65d0d4cc63SWarner Losh static int null_quiesce(device_t dev) 66d0d4cc63SWarner Losh { 677a635f0dSJohn Baldwin return 0; 68d0d4cc63SWarner Losh } 694c7070dbSScott Long 704c7070dbSScott Long static void * null_register(device_t dev) 714c7070dbSScott Long { 724c7070dbSScott Long return NULL; 734c7070dbSScott Long } 748b2970bbSDoug Rabson}; 758b2970bbSDoug Rabson 7667f8f14aSDoug Rabson/** 7767f8f14aSDoug Rabson * @brief Probe to see if a device matches a driver. 7867f8f14aSDoug Rabson * 7967f8f14aSDoug Rabson * Users should not call this method directly. Normally, this 8067f8f14aSDoug Rabson * is called via device_probe_and_attach() to select a driver 8167f8f14aSDoug Rabson * calling the DEVICE_PROBE() of all candidate drivers and attach 8267f8f14aSDoug Rabson * the winning driver (if any) to the device. 8367f8f14aSDoug Rabson * 8467f8f14aSDoug Rabson * This function is used to match devices to device drivers. 8567f8f14aSDoug Rabson * Typically, the driver will examine the device to see if 8667f8f14aSDoug Rabson * it is suitable for this driver. This might include checking 8767f8f14aSDoug Rabson * the values of various device instance variables or reading 8867f8f14aSDoug Rabson * hardware registers. 8967f8f14aSDoug Rabson * 9067f8f14aSDoug Rabson * In some cases, there may be more than one driver available 9167f8f14aSDoug Rabson * which can be used for a device (for instance there might 9267f8f14aSDoug Rabson * be a generic driver which works for a set of many types of 9367f8f14aSDoug Rabson * device and a more specific driver which works for a subset 9467f8f14aSDoug Rabson * of devices). Because of this, a driver should not assume 9567f8f14aSDoug Rabson * that it will be the driver that attaches to the device even 9667f8f14aSDoug Rabson * if it returns a success status from DEVICE_PROBE(). In particular, 9767f8f14aSDoug Rabson * a driver must free any resources which it allocated during 9867f8f14aSDoug Rabson * the probe before returning. The return value of DEVICE_PROBE() 9967f8f14aSDoug Rabson * is used to elect which driver is used - the driver which returns 10067f8f14aSDoug Rabson * the largest non-error value wins the election and attaches to 101af21235aSJohn Baldwin * the device. Common non-error values are described in the 102af21235aSJohn Baldwin * DEVICE_PROBE(9) manual page. 10367f8f14aSDoug Rabson * 10467f8f14aSDoug Rabson * If a driver matches the hardware, it should set the device 10567f8f14aSDoug Rabson * description string using device_set_desc() or 106af21235aSJohn Baldwin * device_set_desc_copy(). This string is used to generate an 107af21235aSJohn Baldwin * informative message when DEVICE_ATTACH() is called. 10867f8f14aSDoug Rabson * 10967f8f14aSDoug Rabson * As a special case, if a driver returns zero, the driver election 11067f8f14aSDoug Rabson * is cut short and that driver will attach to the device 111af21235aSJohn Baldwin * immediately. This should rarely be used. 11267f8f14aSDoug Rabson * 113af21235aSJohn Baldwin * For example, a probe method for a PCI device driver might look 11467f8f14aSDoug Rabson * like this: 11567f8f14aSDoug Rabson * 11667f8f14aSDoug Rabson * @code 117af21235aSJohn Baldwin * int 118af21235aSJohn Baldwin * foo_probe(device_t dev) 11967f8f14aSDoug Rabson * { 12067f8f14aSDoug Rabson * if (pci_get_vendor(dev) == FOOVENDOR && 12167f8f14aSDoug Rabson * pci_get_device(dev) == FOODEVICE) { 12267f8f14aSDoug Rabson * device_set_desc(dev, "Foo device"); 123af21235aSJohn Baldwin * return (BUS_PROBE_DEFAULT); 12467f8f14aSDoug Rabson * } 12567f8f14aSDoug Rabson * return (ENXIO); 12667f8f14aSDoug Rabson * } 12767f8f14aSDoug Rabson * @endcode 12867f8f14aSDoug Rabson * 12967f8f14aSDoug Rabson * To include this method in a device driver, use a line like this 13067f8f14aSDoug Rabson * in the driver's method list: 13167f8f14aSDoug Rabson * 13267f8f14aSDoug Rabson * @code 13367f8f14aSDoug Rabson * KOBJMETHOD(device_probe, foo_probe) 13467f8f14aSDoug Rabson * @endcode 13567f8f14aSDoug Rabson * 13667f8f14aSDoug Rabson * @param dev the device to probe 13767f8f14aSDoug Rabson * 138af21235aSJohn Baldwin * @retval 0 if this is the only possible driver for this 139af21235aSJohn Baldwin * device 14067f8f14aSDoug Rabson * @retval negative if the driver can match this device - the 14167f8f14aSDoug Rabson * least negative value is used to select the 14267f8f14aSDoug Rabson * driver 14367f8f14aSDoug Rabson * @retval ENXIO if the driver does not match the device 14467f8f14aSDoug Rabson * @retval positive if some kind of error was detected during 14567f8f14aSDoug Rabson * the probe, a regular unix error code should 14667f8f14aSDoug Rabson * be returned to indicate the type of error 14767f8f14aSDoug Rabson * @see DEVICE_ATTACH(), pci_get_vendor(), pci_get_device() 14867f8f14aSDoug Rabson */ 149*a21a2da5SColin PercivalPROLOG { 150*a21a2da5SColin Percival TSENTER2(device_get_name(dev)); 151*a21a2da5SColin Percival} 152*a21a2da5SColin PercivalEPILOG { 153*a21a2da5SColin Percival TSEXIT2(device_get_name(dev)); 154*a21a2da5SColin Percival} 155b1bf6610SDoug RabsonMETHOD int probe { 156b1bf6610SDoug Rabson device_t dev; 157b1bf6610SDoug Rabson}; 158b1bf6610SDoug Rabson 15967f8f14aSDoug Rabson/** 1609f7f340aSWarner Losh * @brief Allow a device driver to detect devices not otherwise enumerated. 16167f8f14aSDoug Rabson * 1629f7f340aSWarner Losh * The DEVICE_IDENTIFY() method is used by some drivers (e.g. the ISA 1639f7f340aSWarner Losh * bus driver) to help populate the bus device with a useful set of 1649f7f340aSWarner Losh * child devices, normally by calling the BUS_ADD_CHILD() method of 1659f7f340aSWarner Losh * the parent device. For instance, the ISA bus driver uses several 1669f7f340aSWarner Losh * special drivers, including the isahint driver and the pnp driver to 1679f7f340aSWarner Losh * create child devices based on configuration hints and PnP bus 16867f8f14aSDoug Rabson * probes respectively. 16967f8f14aSDoug Rabson * 1709f7f340aSWarner Losh * Many bus drivers which support true plug-and-play do not need to 1719f7f340aSWarner Losh * use this method at all since child devices can be discovered 1729f7f340aSWarner Losh * automatically without help from child drivers. 17367f8f14aSDoug Rabson * 17467f8f14aSDoug Rabson * To include this method in a device driver, use a line like this 17567f8f14aSDoug Rabson * in the driver's method list: 17667f8f14aSDoug Rabson * 17767f8f14aSDoug Rabson * @code 17867f8f14aSDoug Rabson * KOBJMETHOD(device_identify, foo_identify) 17967f8f14aSDoug Rabson * @endcode 18067f8f14aSDoug Rabson * 18167f8f14aSDoug Rabson * @param driver the driver whose identify method is being called 18267f8f14aSDoug Rabson * @param parent the parent device to use when adding new children 18367f8f14aSDoug Rabson */ 1846c2e3ddeSDoug RabsonSTATICMETHOD void identify { 1856c2e3ddeSDoug Rabson driver_t *driver; 1866c2e3ddeSDoug Rabson device_t parent; 1876c2e3ddeSDoug Rabson}; 1886c2e3ddeSDoug Rabson 18967f8f14aSDoug Rabson/** 19067f8f14aSDoug Rabson * @brief Attach a device to a device driver 19167f8f14aSDoug Rabson * 19267f8f14aSDoug Rabson * Normally only called via device_probe_and_attach(), this is called 19367f8f14aSDoug Rabson * when a driver has succeeded in probing against a device. 19467f8f14aSDoug Rabson * This method should initialise the hardware and allocate other 19567f8f14aSDoug Rabson * system resources (e.g. devfs entries) as required. 19667f8f14aSDoug Rabson * 19767f8f14aSDoug Rabson * To include this method in a device driver, use a line like this 19867f8f14aSDoug Rabson * in the driver's method list: 19967f8f14aSDoug Rabson * 20067f8f14aSDoug Rabson * @code 20167f8f14aSDoug Rabson * KOBJMETHOD(device_attach, foo_attach) 20267f8f14aSDoug Rabson * @endcode 20367f8f14aSDoug Rabson * 20467f8f14aSDoug Rabson * @param dev the device to probe 20567f8f14aSDoug Rabson * 20667f8f14aSDoug Rabson * @retval 0 success 20767f8f14aSDoug Rabson * @retval non-zero if some kind of error was detected during 20867f8f14aSDoug Rabson * the attach, a regular unix error code should 20967f8f14aSDoug Rabson * be returned to indicate the type of error 21067f8f14aSDoug Rabson * @see DEVICE_PROBE() 21167f8f14aSDoug Rabson */ 212*a21a2da5SColin PercivalPROLOG { 213*a21a2da5SColin Percival TSENTER2(device_get_name(dev)); 214*a21a2da5SColin Percival} 215*a21a2da5SColin PercivalEPILOG { 216*a21a2da5SColin Percival TSEXIT2(device_get_name(dev)); 217*a21a2da5SColin Percival} 218b1bf6610SDoug RabsonMETHOD int attach { 219b1bf6610SDoug Rabson device_t dev; 220b1bf6610SDoug Rabson}; 221b1bf6610SDoug Rabson 22267f8f14aSDoug Rabson/** 22367f8f14aSDoug Rabson * @brief Detach a driver from a device. 22467f8f14aSDoug Rabson * 22567f8f14aSDoug Rabson * This can be called if the user is replacing the 22667f8f14aSDoug Rabson * driver software or if a device is about to be physically removed 22767f8f14aSDoug Rabson * from the system (e.g. for removable hardware such as USB or PCCARD). 22867f8f14aSDoug Rabson * 22967f8f14aSDoug Rabson * To include this method in a device driver, use a line like this 23067f8f14aSDoug Rabson * in the driver's method list: 23167f8f14aSDoug Rabson * 23267f8f14aSDoug Rabson * @code 23367f8f14aSDoug Rabson * KOBJMETHOD(device_detach, foo_detach) 23467f8f14aSDoug Rabson * @endcode 23567f8f14aSDoug Rabson * 23667f8f14aSDoug Rabson * @param dev the device to detach 23767f8f14aSDoug Rabson * 23867f8f14aSDoug Rabson * @retval 0 success 23967f8f14aSDoug Rabson * @retval non-zero the detach could not be performed, e.g. if the 24067f8f14aSDoug Rabson * driver does not support detaching. 24167f8f14aSDoug Rabson * 24267f8f14aSDoug Rabson * @see DEVICE_ATTACH() 24367f8f14aSDoug Rabson */ 244b1bf6610SDoug RabsonMETHOD int detach { 245b1bf6610SDoug Rabson device_t dev; 246b1bf6610SDoug Rabson}; 247b1bf6610SDoug Rabson 24867f8f14aSDoug Rabson/** 24967f8f14aSDoug Rabson * @brief Called during system shutdown. 25067f8f14aSDoug Rabson * 25167f8f14aSDoug Rabson * This method allows drivers to detect when the system is being shut down. 25267f8f14aSDoug Rabson * Some drivers need to use this to place their hardware in a consistent 25367f8f14aSDoug Rabson * state before rebooting the computer. 25467f8f14aSDoug Rabson * 25567f8f14aSDoug Rabson * To include this method in a device driver, use a line like this 25667f8f14aSDoug Rabson * in the driver's method list: 25767f8f14aSDoug Rabson * 25867f8f14aSDoug Rabson * @code 25967f8f14aSDoug Rabson * KOBJMETHOD(device_shutdown, foo_shutdown) 26067f8f14aSDoug Rabson * @endcode 26167f8f14aSDoug Rabson */ 262b1bf6610SDoug RabsonMETHOD int shutdown { 263b1bf6610SDoug Rabson device_t dev; 2648b2970bbSDoug Rabson} DEFAULT null_shutdown; 26514177d72SGarrett Wollman 26667f8f14aSDoug Rabson/** 2679f7f340aSWarner Losh * @brief This is called by the power-management subsystem when a 2689f7f340aSWarner Losh * suspend has been requested by the user or by some automatic 2699f7f340aSWarner Losh * mechanism. 27067f8f14aSDoug Rabson * 2719f7f340aSWarner Losh * This gives drivers a chance to veto the suspend or save their 2729f7f340aSWarner Losh * configuration before power is removed. 27367f8f14aSDoug Rabson * 2749f7f340aSWarner Losh * To include this method in a device driver, use a line like this in 2759f7f340aSWarner Losh * the driver's method list: 27667f8f14aSDoug Rabson * 27767f8f14aSDoug Rabson * @code 27867f8f14aSDoug Rabson * KOBJMETHOD(device_suspend, foo_suspend) 27967f8f14aSDoug Rabson * @endcode 28067f8f14aSDoug Rabson * 28167f8f14aSDoug Rabson * @param dev the device being suspended 28267f8f14aSDoug Rabson * 28367f8f14aSDoug Rabson * @retval 0 success 2849f7f340aSWarner Losh * @retval non-zero an error occurred while attempting to prepare the 2859f7f340aSWarner Losh * device for suspension 28667f8f14aSDoug Rabson * 28767f8f14aSDoug Rabson * @see DEVICE_RESUME() 28867f8f14aSDoug Rabson */ 28914177d72SGarrett WollmanMETHOD int suspend { 29014177d72SGarrett Wollman device_t dev; 2918b2970bbSDoug Rabson} DEFAULT null_suspend; 29214177d72SGarrett Wollman 29367f8f14aSDoug Rabson/** 29467f8f14aSDoug Rabson * @brief This is called when the system resumes after a suspend. 29567f8f14aSDoug Rabson * 29667f8f14aSDoug Rabson * To include this method in a device driver, use a line like this 29767f8f14aSDoug Rabson * in the driver's method list: 29867f8f14aSDoug Rabson * 29967f8f14aSDoug Rabson * @code 30067f8f14aSDoug Rabson * KOBJMETHOD(device_resume, foo_resume) 30167f8f14aSDoug Rabson * @endcode 30267f8f14aSDoug Rabson * 30367f8f14aSDoug Rabson * @param dev the device being resumed 30467f8f14aSDoug Rabson * 30567f8f14aSDoug Rabson * @retval 0 success 3069f7f340aSWarner Losh * @retval non-zero an error occurred while attempting to restore the 3079f7f340aSWarner Losh * device from suspension 30867f8f14aSDoug Rabson * 30967f8f14aSDoug Rabson * @see DEVICE_SUSPEND() 31067f8f14aSDoug Rabson */ 31114177d72SGarrett WollmanMETHOD int resume { 31214177d72SGarrett Wollman device_t dev; 3138b2970bbSDoug Rabson} DEFAULT null_resume; 314d0d4cc63SWarner Losh 315d0d4cc63SWarner Losh/** 316d0d4cc63SWarner Losh * @brief This is called when the driver is asked to quiesce itself. 317d0d4cc63SWarner Losh * 318d0d4cc63SWarner Losh * The driver should arrange for the orderly shutdown of this device. 319d0d4cc63SWarner Losh * All further access to the device should be curtailed. Soon there 320d0d4cc63SWarner Losh * will be a request to detach, but there won't necessarily be one. 321d0d4cc63SWarner Losh * 322d0d4cc63SWarner Losh * To include this method in a device driver, use a line like this 323d0d4cc63SWarner Losh * in the driver's method list: 324d0d4cc63SWarner Losh * 325d0d4cc63SWarner Losh * @code 326d0d4cc63SWarner Losh * KOBJMETHOD(device_quiesce, foo_quiesce) 327d0d4cc63SWarner Losh * @endcode 328d0d4cc63SWarner Losh * 329d0d4cc63SWarner Losh * @param dev the device being quiesced 330d0d4cc63SWarner Losh * 331d0d4cc63SWarner Losh * @retval 0 success 332d0d4cc63SWarner Losh * @retval non-zero an error occurred while attempting to quiesce the 333d0d4cc63SWarner Losh * device 334d0d4cc63SWarner Losh * 335d0d4cc63SWarner Losh * @see DEVICE_DETACH() 336d0d4cc63SWarner Losh */ 337d0d4cc63SWarner LoshMETHOD int quiesce { 338d0d4cc63SWarner Losh device_t dev; 339d0d4cc63SWarner Losh} DEFAULT null_quiesce; 3404c7070dbSScott Long 3414c7070dbSScott Long/** 3424c7070dbSScott Long * @brief This is called when the driver is asked to register handlers. 3434c7070dbSScott Long * 3444c7070dbSScott Long * 3454c7070dbSScott Long * To include this method in a device driver, use a line like this 3464c7070dbSScott Long * in the driver's method list: 3474c7070dbSScott Long * 3484c7070dbSScott Long * @code 3494c7070dbSScott Long * KOBJMETHOD(device_register, foo_register) 3504c7070dbSScott Long * @endcode 3514c7070dbSScott Long * 3524c7070dbSScott Long * @param dev the device for which handlers are being registered 3534c7070dbSScott Long * 3544c7070dbSScott Long * @retval NULL method not implemented 3554c7070dbSScott Long * @retval non-NULL a pointer to implementation specific static driver state 3564c7070dbSScott Long * 3574c7070dbSScott Long */ 3584c7070dbSScott LongMETHOD void * register { 3594c7070dbSScott Long device_t dev; 3604c7070dbSScott Long} DEFAULT null_register; 361