xref: /freebsd/sys/kern/device_if.m (revision 031beb4e239bfce798af17f5fe8dba8bcaf13d99)
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