Commit 4489f161 authored by Mauro Carvalho Chehab's avatar Mauro Carvalho Chehab Committed by Greg Kroah-Hartman

docs: driver-model: convert docs to ReST and rename to *.rst

Convert the various documents at the driver-model, preparing
them to be part of the driver-api book.

The conversion is actually:
  - add blank lines and identation in order to identify paragraphs;
  - fix tables markups;
  - add some lists markups;
  - mark literal blocks;
  - adjust title markups.

At its new index.rst, let's add a :orphan: while this is not linked to
the main index.rst file, in order to avoid build warnings.
Signed-off-by: default avatarMauro Carvalho Chehab <mchehab+samsung@kernel.org>
Acked-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com> # ice
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 58cb346c
...@@ -399,7 +399,7 @@ symbol: ...@@ -399,7 +399,7 @@ symbol:
will pass the struct gpio_chip* for the chip to all IRQ callbacks, so the will pass the struct gpio_chip* for the chip to all IRQ callbacks, so the
callbacks need to embed the gpio_chip in its state container and obtain a callbacks need to embed the gpio_chip in its state container and obtain a
pointer to the container using container_of(). pointer to the container using container_of().
(See Documentation/driver-model/design-patterns.txt) (See Documentation/driver-model/design-patterns.rst)
- gpiochip_irqchip_add_nested(): adds a nested cascaded irqchip to a gpiochip, - gpiochip_irqchip_add_nested(): adds a nested cascaded irqchip to a gpiochip,
as discussed above regarding different types of cascaded irqchips. The as discussed above regarding different types of cascaded irqchips. The
......
==============
Driver Binding Driver Binding
==============
Driver binding is the process of associating a device with a device Driver binding is the process of associating a device with a device
driver that can control it. Bus drivers have typically handled this driver that can control it. Bus drivers have typically handled this
...@@ -25,7 +26,7 @@ device_register ...@@ -25,7 +26,7 @@ device_register
When a new device is added, the bus's list of drivers is iterated over When a new device is added, the bus's list of drivers is iterated over
to find one that supports it. In order to determine that, the device to find one that supports it. In order to determine that, the device
ID of the device must match one of the device IDs that the driver ID of the device must match one of the device IDs that the driver
supports. The format and semantics for comparing IDs is bus-specific. supports. The format and semantics for comparing IDs is bus-specific.
Instead of trying to derive a complex state machine and matching Instead of trying to derive a complex state machine and matching
algorithm, it is up to the bus driver to provide a callback to compare algorithm, it is up to the bus driver to provide a callback to compare
a device against the IDs of a driver. The bus returns 1 if a match was a device against the IDs of a driver. The bus returns 1 if a match was
...@@ -36,14 +37,14 @@ int match(struct device * dev, struct device_driver * drv); ...@@ -36,14 +37,14 @@ int match(struct device * dev, struct device_driver * drv);
If a match is found, the device's driver field is set to the driver If a match is found, the device's driver field is set to the driver
and the driver's probe callback is called. This gives the driver a and the driver's probe callback is called. This gives the driver a
chance to verify that it really does support the hardware, and that chance to verify that it really does support the hardware, and that
it's in a working state. it's in a working state.
Device Class Device Class
~~~~~~~~~~~~ ~~~~~~~~~~~~
Upon the successful completion of probe, the device is registered with Upon the successful completion of probe, the device is registered with
the class to which it belongs. Device drivers belong to one and only one the class to which it belongs. Device drivers belong to one and only one
class, and that is set in the driver's devclass field. class, and that is set in the driver's devclass field.
devclass_add_device is called to enumerate the device within the class devclass_add_device is called to enumerate the device within the class
and actually register it with the class, which happens with the and actually register it with the class, which happens with the
class's register_dev callback. class's register_dev callback.
...@@ -53,7 +54,7 @@ Driver ...@@ -53,7 +54,7 @@ Driver
~~~~~~ ~~~~~~
When a driver is attached to a device, the device is inserted into the When a driver is attached to a device, the device is inserted into the
driver's list of devices. driver's list of devices.
sysfs sysfs
...@@ -67,18 +68,18 @@ to the device's directory in the physical hierarchy. ...@@ -67,18 +68,18 @@ to the device's directory in the physical hierarchy.
A directory for the device is created in the class's directory. A A directory for the device is created in the class's directory. A
symlink is created in that directory that points to the device's symlink is created in that directory that points to the device's
physical location in the sysfs tree. physical location in the sysfs tree.
A symlink can be created (though this isn't done yet) in the device's A symlink can be created (though this isn't done yet) in the device's
physical directory to either its class directory, or the class's physical directory to either its class directory, or the class's
top-level directory. One can also be created to point to its driver's top-level directory. One can also be created to point to its driver's
directory also. directory also.
driver_register driver_register
~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~
The process is almost identical for when a new driver is added. The process is almost identical for when a new driver is added.
The bus's list of devices is iterated over to find a match. Devices The bus's list of devices is iterated over to find a match. Devices
that already have a driver are skipped. All the devices are iterated that already have a driver are skipped. All the devices are iterated
over, to bind as many devices as possible to the driver. over, to bind as many devices as possible to the driver.
...@@ -94,5 +95,4 @@ of the driver is decremented. All symlinks between the two are removed. ...@@ -94,5 +95,4 @@ of the driver is decremented. All symlinks between the two are removed.
When a driver is removed, the list of devices that it supports is When a driver is removed, the list of devices that it supports is
iterated over, and the driver's remove callback is called for each iterated over, and the driver's remove callback is called for each
one. The device is removed from that list and the symlinks removed. one. The device is removed from that list and the symlinks removed.
=========
Bus Types Bus Types
=========
Definition Definition
~~~~~~~~~~ ~~~~~~~~~~
...@@ -13,12 +14,12 @@ Declaration ...@@ -13,12 +14,12 @@ Declaration
Each bus type in the kernel (PCI, USB, etc) should declare one static Each bus type in the kernel (PCI, USB, etc) should declare one static
object of this type. They must initialize the name field, and may object of this type. They must initialize the name field, and may
optionally initialize the match callback. optionally initialize the match callback::
struct bus_type pci_bus_type = { struct bus_type pci_bus_type = {
.name = "pci", .name = "pci",
.match = pci_bus_match, .match = pci_bus_match,
}; };
The structure should be exported to drivers in a header file: The structure should be exported to drivers in a header file:
...@@ -30,8 +31,8 @@ Registration ...@@ -30,8 +31,8 @@ Registration
When a bus driver is initialized, it calls bus_register. This When a bus driver is initialized, it calls bus_register. This
initializes the rest of the fields in the bus object and inserts it initializes the rest of the fields in the bus object and inserts it
into a global list of bus types. Once the bus object is registered, into a global list of bus types. Once the bus object is registered,
the fields in it are usable by the bus driver. the fields in it are usable by the bus driver.
Callbacks Callbacks
...@@ -43,17 +44,17 @@ match(): Attaching Drivers to Devices ...@@ -43,17 +44,17 @@ match(): Attaching Drivers to Devices
The format of device ID structures and the semantics for comparing The format of device ID structures and the semantics for comparing
them are inherently bus-specific. Drivers typically declare an array them are inherently bus-specific. Drivers typically declare an array
of device IDs of devices they support that reside in a bus-specific of device IDs of devices they support that reside in a bus-specific
driver structure. driver structure.
The purpose of the match callback is to give the bus an opportunity to The purpose of the match callback is to give the bus an opportunity to
determine if a particular driver supports a particular device by determine if a particular driver supports a particular device by
comparing the device IDs the driver supports with the device ID of a comparing the device IDs the driver supports with the device ID of a
particular device, without sacrificing bus-specific functionality or particular device, without sacrificing bus-specific functionality or
type-safety. type-safety.
When a driver is registered with the bus, the bus's list of devices is When a driver is registered with the bus, the bus's list of devices is
iterated over, and the match callback is called for each device that iterated over, and the match callback is called for each device that
does not have a driver associated with it. does not have a driver associated with it.
...@@ -64,22 +65,23 @@ The lists of devices and drivers are intended to replace the local ...@@ -64,22 +65,23 @@ The lists of devices and drivers are intended to replace the local
lists that many buses keep. They are lists of struct devices and lists that many buses keep. They are lists of struct devices and
struct device_drivers, respectively. Bus drivers are free to use the struct device_drivers, respectively. Bus drivers are free to use the
lists as they please, but conversion to the bus-specific type may be lists as they please, but conversion to the bus-specific type may be
necessary. necessary.
The LDM core provides helper functions for iterating over each list. The LDM core provides helper functions for iterating over each list::
int bus_for_each_dev(struct bus_type * bus, struct device * start, void * data, int bus_for_each_dev(struct bus_type * bus, struct device * start,
int (*fn)(struct device *, void *)); void * data,
int (*fn)(struct device *, void *));
int bus_for_each_drv(struct bus_type * bus, struct device_driver * start, int bus_for_each_drv(struct bus_type * bus, struct device_driver * start,
void * data, int (*fn)(struct device_driver *, void *)); void * data, int (*fn)(struct device_driver *, void *));
These helpers iterate over the respective list, and call the callback These helpers iterate over the respective list, and call the callback
for each device or driver in the list. All list accesses are for each device or driver in the list. All list accesses are
synchronized by taking the bus's lock (read currently). The reference synchronized by taking the bus's lock (read currently). The reference
count on each object in the list is incremented before the callback is count on each object in the list is incremented before the callback is
called; it is decremented after the next object has been obtained. The called; it is decremented after the next object has been obtained. The
lock is not held when calling the callback. lock is not held when calling the callback.
sysfs sysfs
...@@ -87,14 +89,14 @@ sysfs ...@@ -87,14 +89,14 @@ sysfs
There is a top-level directory named 'bus'. There is a top-level directory named 'bus'.
Each bus gets a directory in the bus directory, along with two default Each bus gets a directory in the bus directory, along with two default
directories: directories::
/sys/bus/pci/ /sys/bus/pci/
|-- devices |-- devices
`-- drivers `-- drivers
Drivers registered with the bus get a directory in the bus's drivers Drivers registered with the bus get a directory in the bus's drivers
directory: directory::
/sys/bus/pci/ /sys/bus/pci/
|-- devices |-- devices
...@@ -106,7 +108,7 @@ directory: ...@@ -106,7 +108,7 @@ directory:
Each device that is discovered on a bus of that type gets a symlink in Each device that is discovered on a bus of that type gets a symlink in
the bus's devices directory to the device's directory in the physical the bus's devices directory to the device's directory in the physical
hierarchy: hierarchy::
/sys/bus/pci/ /sys/bus/pci/
|-- devices |-- devices
...@@ -118,26 +120,27 @@ hierarchy: ...@@ -118,26 +120,27 @@ hierarchy:
Exporting Attributes Exporting Attributes
~~~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~~~
struct bus_attribute {
::
struct bus_attribute {
struct attribute attr; struct attribute attr;
ssize_t (*show)(struct bus_type *, char * buf); ssize_t (*show)(struct bus_type *, char * buf);
ssize_t (*store)(struct bus_type *, const char * buf, size_t count); ssize_t (*store)(struct bus_type *, const char * buf, size_t count);
}; };
Bus drivers can export attributes using the BUS_ATTR_RW macro that works Bus drivers can export attributes using the BUS_ATTR_RW macro that works
similarly to the DEVICE_ATTR_RW macro for devices. For example, a similarly to the DEVICE_ATTR_RW macro for devices. For example, a
definition like this: definition like this::
static BUS_ATTR_RW(debug); static BUS_ATTR_RW(debug);
is equivalent to declaring: is equivalent to declaring::
static bus_attribute bus_attr_debug; static bus_attribute bus_attr_debug;
This can then be used to add and remove the attribute from the bus's This can then be used to add and remove the attribute from the bus's
sysfs directory using: sysfs directory using::
int bus_create_file(struct bus_type *, struct bus_attribute *);
void bus_remove_file(struct bus_type *, struct bus_attribute *);
int bus_create_file(struct bus_type *, struct bus_attribute *);
void bus_remove_file(struct bus_type *, struct bus_attribute *);
==============
Device Classes Device Classes
==============
Introduction Introduction
~~~~~~~~~~~~ ~~~~~~~~~~~~
...@@ -13,37 +13,37 @@ device. The following device classes have been identified: ...@@ -13,37 +13,37 @@ device. The following device classes have been identified:
Each device class defines a set of semantics and a programming interface Each device class defines a set of semantics and a programming interface
that devices of that class adhere to. Device drivers are the that devices of that class adhere to. Device drivers are the
implementation of that programming interface for a particular device on implementation of that programming interface for a particular device on
a particular bus. a particular bus.
Device classes are agnostic with respect to what bus a device resides Device classes are agnostic with respect to what bus a device resides
on. on.
Programming Interface Programming Interface
~~~~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~~~~
The device class structure looks like: The device class structure looks like::
typedef int (*devclass_add)(struct device *); typedef int (*devclass_add)(struct device *);
typedef void (*devclass_remove)(struct device *); typedef void (*devclass_remove)(struct device *);
See the kerneldoc for the struct class. See the kerneldoc for the struct class.
A typical device class definition would look like: A typical device class definition would look like::
struct device_class input_devclass = { struct device_class input_devclass = {
.name = "input", .name = "input",
.add_device = input_add_device, .add_device = input_add_device,
.remove_device = input_remove_device, .remove_device = input_remove_device,
}; };
Each device class structure should be exported in a header file so it Each device class structure should be exported in a header file so it
can be used by drivers, extensions and interfaces. can be used by drivers, extensions and interfaces.
Device classes are registered and unregistered with the core using: Device classes are registered and unregistered with the core using::
int devclass_register(struct device_class * cls); int devclass_register(struct device_class * cls);
void devclass_unregister(struct device_class * cls); void devclass_unregister(struct device_class * cls);
Devices Devices
...@@ -52,16 +52,16 @@ As devices are bound to drivers, they are added to the device class ...@@ -52,16 +52,16 @@ As devices are bound to drivers, they are added to the device class
that the driver belongs to. Before the driver model core, this would that the driver belongs to. Before the driver model core, this would
typically happen during the driver's probe() callback, once the device typically happen during the driver's probe() callback, once the device
has been initialized. It now happens after the probe() callback has been initialized. It now happens after the probe() callback
finishes from the core. finishes from the core.
The device is enumerated in the class. Each time a device is added to The device is enumerated in the class. Each time a device is added to
the class, the class's devnum field is incremented and assigned to the the class, the class's devnum field is incremented and assigned to the
device. The field is never decremented, so if the device is removed device. The field is never decremented, so if the device is removed
from the class and re-added, it will receive a different enumerated from the class and re-added, it will receive a different enumerated
value. value.
The class is allowed to create a class-specific structure for the The class is allowed to create a class-specific structure for the
device and store it in the device's class_data pointer. device and store it in the device's class_data pointer.
There is no list of devices in the device class. Each driver has a There is no list of devices in the device class. Each driver has a
list of devices that it supports. The device class has a list of list of devices that it supports. The device class has a list of
...@@ -73,15 +73,15 @@ Device Drivers ...@@ -73,15 +73,15 @@ Device Drivers
~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~
Device drivers are added to device classes when they are registered Device drivers are added to device classes when they are registered
with the core. A driver specifies the class it belongs to by setting with the core. A driver specifies the class it belongs to by setting
the struct device_driver::devclass field. the struct device_driver::devclass field.
sysfs directory structure sysfs directory structure
~~~~~~~~~~~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~
There is a top-level sysfs directory named 'class'. There is a top-level sysfs directory named 'class'.
Each class gets a directory in the class directory, along with two Each class gets a directory in the class directory, along with two
default subdirectories: default subdirectories::
class/ class/
`-- input `-- input
...@@ -89,8 +89,8 @@ default subdirectories: ...@@ -89,8 +89,8 @@ default subdirectories:
`-- drivers `-- drivers
Drivers registered with the class get a symlink in the drivers/ directory Drivers registered with the class get a symlink in the drivers/ directory
that points to the driver's directory (under its bus directory): that points to the driver's directory (under its bus directory)::
class/ class/
`-- input `-- input
...@@ -99,8 +99,8 @@ that points to the driver's directory (under its bus directory): ...@@ -99,8 +99,8 @@ that points to the driver's directory (under its bus directory):
`-- usb:usb_mouse -> ../../../bus/drivers/usb_mouse/ `-- usb:usb_mouse -> ../../../bus/drivers/usb_mouse/
Each device gets a symlink in the devices/ directory that points to the Each device gets a symlink in the devices/ directory that points to the
device's directory in the physical hierarchy: device's directory in the physical hierarchy::
class/ class/
`-- input `-- input
...@@ -111,37 +111,39 @@ device's directory in the physical hierarchy: ...@@ -111,37 +111,39 @@ device's directory in the physical hierarchy:
Exporting Attributes Exporting Attributes
~~~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~~~
struct devclass_attribute {
::
struct devclass_attribute {
struct attribute attr; struct attribute attr;
ssize_t (*show)(struct device_class *, char * buf, size_t count, loff_t off); ssize_t (*show)(struct device_class *, char * buf, size_t count, loff_t off);
ssize_t (*store)(struct device_class *, const char * buf, size_t count, loff_t off); ssize_t (*store)(struct device_class *, const char * buf, size_t count, loff_t off);
}; };
Class drivers can export attributes using the DEVCLASS_ATTR macro that works Class drivers can export attributes using the DEVCLASS_ATTR macro that works
similarly to the DEVICE_ATTR macro for devices. For example, a definition similarly to the DEVICE_ATTR macro for devices. For example, a definition
like this: like this::
static DEVCLASS_ATTR(debug,0644,show_debug,store_debug); static DEVCLASS_ATTR(debug,0644,show_debug,store_debug);
is equivalent to declaring: is equivalent to declaring::
static devclass_attribute devclass_attr_debug; static devclass_attribute devclass_attr_debug;
The bus driver can add and remove the attribute from the class's The bus driver can add and remove the attribute from the class's
sysfs directory using: sysfs directory using::
int devclass_create_file(struct device_class *, struct devclass_attribute *); int devclass_create_file(struct device_class *, struct devclass_attribute *);
void devclass_remove_file(struct device_class *, struct devclass_attribute *); void devclass_remove_file(struct device_class *, struct devclass_attribute *);
In the example above, the file will be named 'debug' in placed in the In the example above, the file will be named 'debug' in placed in the
class's directory in sysfs. class's directory in sysfs.
Interfaces Interfaces
~~~~~~~~~~ ~~~~~~~~~~
There may exist multiple mechanisms for accessing the same device of a There may exist multiple mechanisms for accessing the same device of a
particular class type. Device interfaces describe these mechanisms. particular class type. Device interfaces describe these mechanisms.
When a device is added to a device class, the core attempts to add it When a device is added to a device class, the core attempts to add it
to every interface that is registered with the device class. to every interface that is registered with the device class.
=============================
Device Driver Design Patterns Device Driver Design Patterns
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ =============================
This document describes a few common design patterns found in device drivers. This document describes a few common design patterns found in device drivers.
It is likely that subsystem maintainers will ask driver developers to It is likely that subsystem maintainers will ask driver developers to
...@@ -19,23 +19,23 @@ that the device the driver binds to will appear in several instances. This ...@@ -19,23 +19,23 @@ that the device the driver binds to will appear in several instances. This
means that the probe() function and all callbacks need to be reentrant. means that the probe() function and all callbacks need to be reentrant.
The most common way to achieve this is to use the state container design The most common way to achieve this is to use the state container design
pattern. It usually has this form: pattern. It usually has this form::
struct foo { struct foo {
spinlock_t lock; /* Example member */ spinlock_t lock; /* Example member */
(...) (...)
}; };
static int foo_probe(...) static int foo_probe(...)
{ {
struct foo *foo; struct foo *foo;
foo = devm_kzalloc(dev, sizeof(*foo), GFP_KERNEL); foo = devm_kzalloc(dev, sizeof(*foo), GFP_KERNEL);
if (!foo) if (!foo)
return -ENOMEM; return -ENOMEM;
spin_lock_init(&foo->lock); spin_lock_init(&foo->lock);
(...) (...)
} }
This will create an instance of struct foo in memory every time probe() is This will create an instance of struct foo in memory every time probe() is
called. This is our state container for this instance of the device driver. called. This is our state container for this instance of the device driver.
...@@ -43,21 +43,21 @@ Of course it is then necessary to always pass this instance of the ...@@ -43,21 +43,21 @@ Of course it is then necessary to always pass this instance of the
state around to all functions that need access to the state and its members. state around to all functions that need access to the state and its members.
For example, if the driver is registering an interrupt handler, you would For example, if the driver is registering an interrupt handler, you would
pass around a pointer to struct foo like this: pass around a pointer to struct foo like this::
static irqreturn_t foo_handler(int irq, void *arg) static irqreturn_t foo_handler(int irq, void *arg)
{ {
struct foo *foo = arg; struct foo *foo = arg;
(...) (...)
} }
static int foo_probe(...) static int foo_probe(...)
{ {
struct foo *foo; struct foo *foo;
(...) (...)
ret = request_irq(irq, foo_handler, 0, "foo", foo); ret = request_irq(irq, foo_handler, 0, "foo", foo);
} }
This way you always get a pointer back to the correct instance of foo in This way you always get a pointer back to the correct instance of foo in
your interrupt handler. your interrupt handler.
...@@ -66,38 +66,38 @@ your interrupt handler. ...@@ -66,38 +66,38 @@ your interrupt handler.
2. container_of() 2. container_of()
~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~
Continuing on the above example we add an offloaded work: Continuing on the above example we add an offloaded work::
struct foo { struct foo {
spinlock_t lock; spinlock_t lock;
struct workqueue_struct *wq; struct workqueue_struct *wq;
struct work_struct offload; struct work_struct offload;
(...) (...)
}; };
static void foo_work(struct work_struct *work) static void foo_work(struct work_struct *work)
{ {
struct foo *foo = container_of(work, struct foo, offload); struct foo *foo = container_of(work, struct foo, offload);
(...) (...)
} }
static irqreturn_t foo_handler(int irq, void *arg) static irqreturn_t foo_handler(int irq, void *arg)
{ {
struct foo *foo = arg; struct foo *foo = arg;
queue_work(foo->wq, &foo->offload); queue_work(foo->wq, &foo->offload);
(...) (...)
} }
static int foo_probe(...) static int foo_probe(...)
{ {
struct foo *foo; struct foo *foo;
foo->wq = create_singlethread_workqueue("foo-wq"); foo->wq = create_singlethread_workqueue("foo-wq");
INIT_WORK(&foo->offload, foo_work); INIT_WORK(&foo->offload, foo_work);
(...) (...)
} }
The design pattern is the same for an hrtimer or something similar that will The design pattern is the same for an hrtimer or something similar that will
return a single argument which is a pointer to a struct member in the return a single argument which is a pointer to a struct member in the
......
==========================
The Basic Device Structure The Basic Device Structure
~~~~~~~~~~~~~~~~~~~~~~~~~~ ==========================
See the kerneldoc for the struct device. See the kerneldoc for the struct device.
...@@ -8,9 +8,9 @@ See the kerneldoc for the struct device. ...@@ -8,9 +8,9 @@ See the kerneldoc for the struct device.
Programming Interface Programming Interface
~~~~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~~~~
The bus driver that discovers the device uses this to register the The bus driver that discovers the device uses this to register the
device with the core: device with the core::
int device_register(struct device * dev); int device_register(struct device * dev);
The bus should initialize the following fields: The bus should initialize the following fields:
...@@ -20,30 +20,33 @@ The bus should initialize the following fields: ...@@ -20,30 +20,33 @@ The bus should initialize the following fields:
- bus - bus
A device is removed from the core when its reference count goes to A device is removed from the core when its reference count goes to
0. The reference count can be adjusted using: 0. The reference count can be adjusted using::
struct device * get_device(struct device * dev); struct device * get_device(struct device * dev);
void put_device(struct device * dev); void put_device(struct device * dev);
get_device() will return a pointer to the struct device passed to it get_device() will return a pointer to the struct device passed to it
if the reference is not already 0 (if it's in the process of being if the reference is not already 0 (if it's in the process of being
removed already). removed already).
A driver can access the lock in the device structure using: A driver can access the lock in the device structure using::
void lock_device(struct device * dev); void lock_device(struct device * dev);
void unlock_device(struct device * dev); void unlock_device(struct device * dev);
Attributes Attributes
~~~~~~~~~~ ~~~~~~~~~~
struct device_attribute {
::
struct device_attribute {
struct attribute attr; struct attribute attr;
ssize_t (*show)(struct device *dev, struct device_attribute *attr, ssize_t (*show)(struct device *dev, struct device_attribute *attr,
char *buf); char *buf);
ssize_t (*store)(struct device *dev, struct device_attribute *attr, ssize_t (*store)(struct device *dev, struct device_attribute *attr,
const char *buf, size_t count); const char *buf, size_t count);
}; };
Attributes of devices can be exported by a device driver through sysfs. Attributes of devices can be exported by a device driver through sysfs.
...@@ -54,39 +57,39 @@ As explained in Documentation/kobject.txt, device attributes must be ...@@ -54,39 +57,39 @@ As explained in Documentation/kobject.txt, device attributes must be
created before the KOBJ_ADD uevent is generated. The only way to realize created before the KOBJ_ADD uevent is generated. The only way to realize
that is by defining an attribute group. that is by defining an attribute group.
Attributes are declared using a macro called DEVICE_ATTR: Attributes are declared using a macro called DEVICE_ATTR::
#define DEVICE_ATTR(name,mode,show,store) #define DEVICE_ATTR(name,mode,show,store)
Example: Example:::
static DEVICE_ATTR(type, 0444, show_type, NULL); static DEVICE_ATTR(type, 0444, show_type, NULL);
static DEVICE_ATTR(power, 0644, show_power, store_power); static DEVICE_ATTR(power, 0644, show_power, store_power);
This declares two structures of type struct device_attribute with respective This declares two structures of type struct device_attribute with respective
names 'dev_attr_type' and 'dev_attr_power'. These two attributes can be names 'dev_attr_type' and 'dev_attr_power'. These two attributes can be
organized as follows into a group: organized as follows into a group::
static struct attribute *dev_attrs[] = { static struct attribute *dev_attrs[] = {
&dev_attr_type.attr, &dev_attr_type.attr,
&dev_attr_power.attr, &dev_attr_power.attr,
NULL, NULL,
}; };
static struct attribute_group dev_attr_group = { static struct attribute_group dev_attr_group = {
.attrs = dev_attrs, .attrs = dev_attrs,
}; };
static const struct attribute_group *dev_attr_groups[] = { static const struct attribute_group *dev_attr_groups[] = {
&dev_attr_group, &dev_attr_group,
NULL, NULL,
}; };
This array of groups can then be associated with a device by setting the This array of groups can then be associated with a device by setting the
group pointer in struct device before device_register() is invoked: group pointer in struct device before device_register() is invoked::
dev->groups = dev_attr_groups; dev->groups = dev_attr_groups;
device_register(dev); device_register(dev);
The device_register() function will use the 'groups' pointer to create the The device_register() function will use the 'groups' pointer to create the
device attributes and the device_unregister() function will use this pointer device attributes and the device_unregister() function will use this pointer
......
================================
Devres - Managed Device Resource Devres - Managed Device Resource
================================ ================================
...@@ -5,17 +6,18 @@ Tejun Heo <teheo@suse.de> ...@@ -5,17 +6,18 @@ Tejun Heo <teheo@suse.de>
First draft 10 January 2007 First draft 10 January 2007
.. contents
1. Intro : Huh? Devres? 1. Intro : Huh? Devres?
2. Devres : Devres in a nutshell 2. Devres : Devres in a nutshell
3. Devres Group : Group devres'es and release them together 3. Devres Group : Group devres'es and release them together
4. Details : Life time rules, calling context, ... 4. Details : Life time rules, calling context, ...
5. Overhead : How much do we have to pay for this? 5. Overhead : How much do we have to pay for this?
6. List of managed interfaces : Currently implemented managed interfaces 6. List of managed interfaces: Currently implemented managed interfaces
1. Intro 1. Intro
-------- --------
devres came up while trying to convert libata to use iomap. Each devres came up while trying to convert libata to use iomap. Each
iomapped address should be kept and unmapped on driver detach. For iomapped address should be kept and unmapped on driver detach. For
...@@ -42,8 +44,8 @@ would leak resources or even cause oops when failure occurs. iomap ...@@ -42,8 +44,8 @@ would leak resources or even cause oops when failure occurs. iomap
adds more to this mix. So do msi and msix. adds more to this mix. So do msi and msix.
2. Devres 2. Devres
--------- ---------
devres is basically linked list of arbitrarily sized memory areas devres is basically linked list of arbitrarily sized memory areas
associated with a struct device. Each devres entry is associated with associated with a struct device. Each devres entry is associated with
...@@ -58,7 +60,7 @@ using dma_alloc_coherent(). The managed version is called ...@@ -58,7 +60,7 @@ using dma_alloc_coherent(). The managed version is called
dmam_alloc_coherent(). It is identical to dma_alloc_coherent() except dmam_alloc_coherent(). It is identical to dma_alloc_coherent() except
for the DMA memory allocated using it is managed and will be for the DMA memory allocated using it is managed and will be
automatically released on driver detach. Implementation looks like automatically released on driver detach. Implementation looks like
the following. the following::
struct dma_devres { struct dma_devres {
size_t size; size_t size;
...@@ -98,7 +100,7 @@ If a driver uses dmam_alloc_coherent(), the area is guaranteed to be ...@@ -98,7 +100,7 @@ If a driver uses dmam_alloc_coherent(), the area is guaranteed to be
freed whether initialization fails half-way or the device gets freed whether initialization fails half-way or the device gets
detached. If most resources are acquired using managed interface, a detached. If most resources are acquired using managed interface, a
driver can have much simpler init and exit code. Init path basically driver can have much simpler init and exit code. Init path basically
looks like the following. looks like the following::
my_init_one() my_init_one()
{ {
...@@ -119,7 +121,7 @@ looks like the following. ...@@ -119,7 +121,7 @@ looks like the following.
return register_to_upper_layer(d); return register_to_upper_layer(d);
} }
And exit path, And exit path::
my_remove_one() my_remove_one()
{ {
...@@ -140,13 +142,13 @@ on you. In some cases this may mean introducing checks that were not ...@@ -140,13 +142,13 @@ on you. In some cases this may mean introducing checks that were not
necessary before moving to the managed devm_* calls. necessary before moving to the managed devm_* calls.
3. Devres group 3. Devres group
--------------- ---------------
Devres entries can be grouped using devres group. When a group is Devres entries can be grouped using devres group. When a group is
released, all contained normal devres entries and properly nested released, all contained normal devres entries and properly nested
groups are released. One usage is to rollback series of acquired groups are released. One usage is to rollback series of acquired
resources on failure. For example, resources on failure. For example::
if (!devres_open_group(dev, NULL, GFP_KERNEL)) if (!devres_open_group(dev, NULL, GFP_KERNEL))
return -ENOMEM; return -ENOMEM;
...@@ -172,7 +174,7 @@ like above are usually useful in midlayer driver (e.g. libata core ...@@ -172,7 +174,7 @@ like above are usually useful in midlayer driver (e.g. libata core
layer) where interface function shouldn't have side effect on failure. layer) where interface function shouldn't have side effect on failure.
For LLDs, just returning error code suffices in most cases. For LLDs, just returning error code suffices in most cases.
Each group is identified by void *id. It can either be explicitly Each group is identified by `void *id`. It can either be explicitly
specified by @id argument to devres_open_group() or automatically specified by @id argument to devres_open_group() or automatically
created by passing NULL as @id as in the above example. In both created by passing NULL as @id as in the above example. In both
cases, devres_open_group() returns the group's id. The returned id cases, devres_open_group() returns the group's id. The returned id
...@@ -180,7 +182,7 @@ can be passed to other devres functions to select the target group. ...@@ -180,7 +182,7 @@ can be passed to other devres functions to select the target group.
If NULL is given to those functions, the latest open group is If NULL is given to those functions, the latest open group is
selected. selected.
For example, you can do something like the following. For example, you can do something like the following::
int my_midlayer_create_something() int my_midlayer_create_something()
{ {
...@@ -199,8 +201,8 @@ For example, you can do something like the following. ...@@ -199,8 +201,8 @@ For example, you can do something like the following.
} }
4. Details 4. Details
---------- ----------
Lifetime of a devres entry begins on devres allocation and finishes Lifetime of a devres entry begins on devres allocation and finishes
when it is released or destroyed (removed and freed) - no reference when it is released or destroyed (removed and freed) - no reference
...@@ -220,8 +222,8 @@ All devres interface functions can be called without context if the ...@@ -220,8 +222,8 @@ All devres interface functions can be called without context if the
right gfp mask is given. right gfp mask is given.
5. Overhead 5. Overhead
----------- -----------
Each devres bookkeeping info is allocated together with requested data Each devres bookkeeping info is allocated together with requested data
area. With debug option turned off, bookkeeping info occupies 16 area. With debug option turned off, bookkeeping info occupies 16
...@@ -237,8 +239,8 @@ and 400 bytes on 32bit machine after naive conversion (we can ...@@ -237,8 +239,8 @@ and 400 bytes on 32bit machine after naive conversion (we can
certainly invest a bit more effort into libata core layer). certainly invest a bit more effort into libata core layer).
6. List of managed interfaces 6. List of managed interfaces
----------------------------- -----------------------------
CLOCK CLOCK
devm_clk_get() devm_clk_get()
......
==============
Device Drivers Device Drivers
==============
See the kerneldoc for the struct device_driver. See the kerneldoc for the struct device_driver.
...@@ -26,50 +27,50 @@ Declaration ...@@ -26,50 +27,50 @@ Declaration
As stated above, struct device_driver objects are statically As stated above, struct device_driver objects are statically
allocated. Below is an example declaration of the eepro100 allocated. Below is an example declaration of the eepro100
driver. This declaration is hypothetical only; it relies on the driver driver. This declaration is hypothetical only; it relies on the driver
being converted completely to the new model. being converted completely to the new model::
static struct device_driver eepro100_driver = { static struct device_driver eepro100_driver = {
.name = "eepro100", .name = "eepro100",
.bus = &pci_bus_type, .bus = &pci_bus_type,
.probe = eepro100_probe, .probe = eepro100_probe,
.remove = eepro100_remove, .remove = eepro100_remove,
.suspend = eepro100_suspend, .suspend = eepro100_suspend,
.resume = eepro100_resume, .resume = eepro100_resume,
}; };
Most drivers will not be able to be converted completely to the new Most drivers will not be able to be converted completely to the new
model because the bus they belong to has a bus-specific structure with model because the bus they belong to has a bus-specific structure with
bus-specific fields that cannot be generalized. bus-specific fields that cannot be generalized.
The most common example of this are device ID structures. A driver The most common example of this are device ID structures. A driver
typically defines an array of device IDs that it supports. The format typically defines an array of device IDs that it supports. The format
of these structures and the semantics for comparing device IDs are of these structures and the semantics for comparing device IDs are
completely bus-specific. Defining them as bus-specific entities would completely bus-specific. Defining them as bus-specific entities would
sacrifice type-safety, so we keep bus-specific structures around. sacrifice type-safety, so we keep bus-specific structures around.
Bus-specific drivers should include a generic struct device_driver in Bus-specific drivers should include a generic struct device_driver in
the definition of the bus-specific driver. Like this: the definition of the bus-specific driver. Like this::
struct pci_driver { struct pci_driver {
const struct pci_device_id *id_table; const struct pci_device_id *id_table;
struct device_driver driver; struct device_driver driver;
}; };
A definition that included bus-specific fields would look like A definition that included bus-specific fields would look like
(using the eepro100 driver again): (using the eepro100 driver again)::
static struct pci_driver eepro100_driver = { static struct pci_driver eepro100_driver = {
.id_table = eepro100_pci_tbl, .id_table = eepro100_pci_tbl,
.driver = { .driver = {
.name = "eepro100", .name = "eepro100",
.bus = &pci_bus_type, .bus = &pci_bus_type,
.probe = eepro100_probe, .probe = eepro100_probe,
.remove = eepro100_remove, .remove = eepro100_remove,
.suspend = eepro100_suspend, .suspend = eepro100_suspend,
.resume = eepro100_resume, .resume = eepro100_resume,
}, },
}; };
Some may find the syntax of embedded struct initialization awkward or Some may find the syntax of embedded struct initialization awkward or
even a bit ugly. So far, it's the best way we've found to do what we want... even a bit ugly. So far, it's the best way we've found to do what we want...
...@@ -77,12 +78,14 @@ even a bit ugly. So far, it's the best way we've found to do what we want... ...@@ -77,12 +78,14 @@ even a bit ugly. So far, it's the best way we've found to do what we want...
Registration Registration
~~~~~~~~~~~~ ~~~~~~~~~~~~
int driver_register(struct device_driver * drv); ::
int driver_register(struct device_driver *drv);
The driver registers the structure on startup. For drivers that have The driver registers the structure on startup. For drivers that have
no bus-specific fields (i.e. don't have a bus-specific driver no bus-specific fields (i.e. don't have a bus-specific driver
structure), they would use driver_register and pass a pointer to their structure), they would use driver_register and pass a pointer to their
struct device_driver object. struct device_driver object.
Most drivers, however, will have a bus-specific structure and will Most drivers, however, will have a bus-specific structure and will
need to register with the bus using something like pci_driver_register. need to register with the bus using something like pci_driver_register.
...@@ -101,7 +104,7 @@ By defining wrapper functions, the transition to the new model can be ...@@ -101,7 +104,7 @@ By defining wrapper functions, the transition to the new model can be
made easier. Drivers can ignore the generic structure altogether and made easier. Drivers can ignore the generic structure altogether and
let the bus wrapper fill in the fields. For the callbacks, the bus can let the bus wrapper fill in the fields. For the callbacks, the bus can
define generic callbacks that forward the call to the bus-specific define generic callbacks that forward the call to the bus-specific
callbacks of the drivers. callbacks of the drivers.
This solution is intended to be only temporary. In order to get class This solution is intended to be only temporary. In order to get class
information in the driver, the drivers must be modified anyway. Since information in the driver, the drivers must be modified anyway. Since
...@@ -113,16 +116,16 @@ Access ...@@ -113,16 +116,16 @@ Access
~~~~~~ ~~~~~~
Once the object has been registered, it may access the common fields of Once the object has been registered, it may access the common fields of
the object, like the lock and the list of devices. the object, like the lock and the list of devices::
int driver_for_each_dev(struct device_driver * drv, void * data, int driver_for_each_dev(struct device_driver *drv, void *data,
int (*callback)(struct device * dev, void * data)); int (*callback)(struct device *dev, void *data));
The devices field is a list of all the devices that have been bound to The devices field is a list of all the devices that have been bound to
the driver. The LDM core provides a helper function to operate on all the driver. The LDM core provides a helper function to operate on all
the devices a driver controls. This helper locks the driver on each the devices a driver controls. This helper locks the driver on each
node access, and does proper reference counting on each device as it node access, and does proper reference counting on each device as it
accesses it. accesses it.
sysfs sysfs
...@@ -142,7 +145,9 @@ supports. ...@@ -142,7 +145,9 @@ supports.
Callbacks Callbacks
~~~~~~~~~ ~~~~~~~~~
int (*probe) (struct device * dev); ::
int (*probe) (struct device *dev);
The probe() entry is called in task context, with the bus's rwsem locked The probe() entry is called in task context, with the bus's rwsem locked
and the driver partially bound to the device. Drivers commonly use and the driver partially bound to the device. Drivers commonly use
...@@ -162,9 +167,9 @@ the driver to that device. ...@@ -162,9 +167,9 @@ the driver to that device.
A driver's probe() may return a negative errno value to indicate that A driver's probe() may return a negative errno value to indicate that
the driver did not bind to this device, in which case it should have the driver did not bind to this device, in which case it should have
released all resources it allocated. released all resources it allocated::
int (*remove) (struct device * dev); int (*remove) (struct device *dev);
remove is called to unbind a driver from a device. This may be remove is called to unbind a driver from a device. This may be
called if a device is physically removed from the system, if the called if a device is physically removed from the system, if the
...@@ -173,43 +178,46 @@ in other cases. ...@@ -173,43 +178,46 @@ in other cases.
It is up to the driver to determine if the device is present or It is up to the driver to determine if the device is present or
not. It should free any resources allocated specifically for the not. It should free any resources allocated specifically for the
device; i.e. anything in the device's driver_data field. device; i.e. anything in the device's driver_data field.
If the device is still present, it should quiesce the device and place If the device is still present, it should quiesce the device and place
it into a supported low-power state. it into a supported low-power state::
int (*suspend) (struct device * dev, pm_message_t state); int (*suspend) (struct device *dev, pm_message_t state);
suspend is called to put the device in a low power state. suspend is called to put the device in a low power state::
int (*resume) (struct device * dev); int (*resume) (struct device *dev);
Resume is used to bring a device back from a low power state. Resume is used to bring a device back from a low power state.
Attributes Attributes
~~~~~~~~~~ ~~~~~~~~~~
struct driver_attribute {
struct attribute attr;
ssize_t (*show)(struct device_driver *driver, char *buf);
ssize_t (*store)(struct device_driver *, const char * buf, size_t count);
};
Device drivers can export attributes via their sysfs directories. ::
struct driver_attribute {
struct attribute attr;
ssize_t (*show)(struct device_driver *driver, char *buf);
ssize_t (*store)(struct device_driver *, const char *buf, size_t count);
};
Device drivers can export attributes via their sysfs directories.
Drivers can declare attributes using a DRIVER_ATTR_RW and DRIVER_ATTR_RO Drivers can declare attributes using a DRIVER_ATTR_RW and DRIVER_ATTR_RO
macro that works identically to the DEVICE_ATTR_RW and DEVICE_ATTR_RO macro that works identically to the DEVICE_ATTR_RW and DEVICE_ATTR_RO
macros. macros.
Example: Example::
DRIVER_ATTR_RW(debug); DRIVER_ATTR_RW(debug);
This is equivalent to declaring: This is equivalent to declaring::
struct driver_attribute driver_attr_debug; struct driver_attribute driver_attr_debug;
This can then be used to add and remove the attribute from the This can then be used to add and remove the attribute from the
driver's directory using: driver's directory using::
int driver_create_file(struct device_driver *, const struct driver_attribute *); int driver_create_file(struct device_driver *, const struct driver_attribute *);
void driver_remove_file(struct device_driver *, const struct driver_attribute *); void driver_remove_file(struct device_driver *, const struct driver_attribute *);
:orphan:
============
Driver Model
============
.. toctree::
:maxdepth: 1
binding
bus
class
design-patterns
device
devres
driver
overview
platform
porting
.. only:: subproject and html
Indices
=======
* :ref:`genindex`
=============================
The Linux Kernel Device Model The Linux Kernel Device Model
=============================
Patrick Mochel <mochel@digitalimplant.org> Patrick Mochel <mochel@digitalimplant.org>
...@@ -41,14 +43,14 @@ data structure. These fields must still be accessed by the bus layers, ...@@ -41,14 +43,14 @@ data structure. These fields must still be accessed by the bus layers,
and sometimes by the device-specific drivers. and sometimes by the device-specific drivers.
Other bus layers are encouraged to do what has been done for the PCI layer. Other bus layers are encouraged to do what has been done for the PCI layer.
struct pci_dev now looks like this: struct pci_dev now looks like this::
struct pci_dev { struct pci_dev {
... ...
struct device dev; /* Generic device interface */ struct device dev; /* Generic device interface */
... ...
}; };
Note first that the struct device dev within the struct pci_dev is Note first that the struct device dev within the struct pci_dev is
statically allocated. This means only one allocation on device discovery. statically allocated. This means only one allocation on device discovery.
...@@ -80,26 +82,26 @@ easy. This has been accomplished by implementing a special purpose virtual ...@@ -80,26 +82,26 @@ easy. This has been accomplished by implementing a special purpose virtual
file system named sysfs. file system named sysfs.
Almost all mainstream Linux distros mount this filesystem automatically; you Almost all mainstream Linux distros mount this filesystem automatically; you
can see some variation of the following in the output of the "mount" command: can see some variation of the following in the output of the "mount" command::
$ mount $ mount
... ...
none on /sys type sysfs (rw,noexec,nosuid,nodev) none on /sys type sysfs (rw,noexec,nosuid,nodev)
... ...
$ $
The auto-mounting of sysfs is typically accomplished by an entry similar to The auto-mounting of sysfs is typically accomplished by an entry similar to
the following in the /etc/fstab file: the following in the /etc/fstab file::
none /sys sysfs defaults 0 0 none /sys sysfs defaults 0 0
or something similar in the /lib/init/fstab file on Debian-based systems: or something similar in the /lib/init/fstab file on Debian-based systems::
none /sys sysfs nodev,noexec,nosuid 0 0 none /sys sysfs nodev,noexec,nosuid 0 0
If sysfs is not automatically mounted, you can always do it manually with: If sysfs is not automatically mounted, you can always do it manually with::
# mount -t sysfs sysfs /sys # mount -t sysfs sysfs /sys
Whenever a device is inserted into the tree, a directory is created for it. Whenever a device is inserted into the tree, a directory is created for it.
This directory may be populated at each layer of discovery - the global layer, This directory may be populated at each layer of discovery - the global layer,
...@@ -108,7 +110,7 @@ the bus layer, or the device layer. ...@@ -108,7 +110,7 @@ the bus layer, or the device layer.
The global layer currently creates two files - 'name' and 'power'. The The global layer currently creates two files - 'name' and 'power'. The
former only reports the name of the device. The latter reports the former only reports the name of the device. The latter reports the
current power state of the device. It will also be used to set the current current power state of the device. It will also be used to set the current
power state. power state.
The bus layer may also create files for the devices it finds while probing the The bus layer may also create files for the devices it finds while probing the
bus. For example, the PCI layer currently creates 'irq' and 'resource' files bus. For example, the PCI layer currently creates 'irq' and 'resource' files
...@@ -118,6 +120,5 @@ A device-specific driver may also export files in its directory to expose ...@@ -118,6 +120,5 @@ A device-specific driver may also export files in its directory to expose
device-specific data or tunable interfaces. device-specific data or tunable interfaces.
More information about the sysfs directory layout can be found in More information about the sysfs directory layout can be found in
the other documents in this directory and in the file the other documents in this directory and in the file
Documentation/filesystems/sysfs.txt. Documentation/filesystems/sysfs.txt.
============================
Platform Devices and Drivers Platform Devices and Drivers
~~~~~~~~~~~~~~~~~~~~~~~~~~~~ ============================
See <linux/platform_device.h> for the driver model interface to the See <linux/platform_device.h> for the driver model interface to the
platform bus: platform_device, and platform_driver. This pseudo-bus platform bus: platform_device, and platform_driver. This pseudo-bus
is used to connect devices on busses with minimal infrastructure, is used to connect devices on busses with minimal infrastructure,
...@@ -19,15 +21,15 @@ be connected through a segment of some other kind of bus; but its ...@@ -19,15 +21,15 @@ be connected through a segment of some other kind of bus; but its
registers will still be directly addressable. registers will still be directly addressable.
Platform devices are given a name, used in driver binding, and a Platform devices are given a name, used in driver binding, and a
list of resources such as addresses and IRQs. list of resources such as addresses and IRQs::
struct platform_device { struct platform_device {
const char *name; const char *name;
u32 id; u32 id;
struct device dev; struct device dev;
u32 num_resources; u32 num_resources;
struct resource *resource; struct resource *resource;
}; };
Platform drivers Platform drivers
...@@ -35,9 +37,9 @@ Platform drivers ...@@ -35,9 +37,9 @@ Platform drivers
Platform drivers follow the standard driver model convention, where Platform drivers follow the standard driver model convention, where
discovery/enumeration is handled outside the drivers, and drivers discovery/enumeration is handled outside the drivers, and drivers
provide probe() and remove() methods. They support power management provide probe() and remove() methods. They support power management
and shutdown notifications using the standard conventions. and shutdown notifications using the standard conventions::
struct platform_driver { struct platform_driver {
int (*probe)(struct platform_device *); int (*probe)(struct platform_device *);
int (*remove)(struct platform_device *); int (*remove)(struct platform_device *);
void (*shutdown)(struct platform_device *); void (*shutdown)(struct platform_device *);
...@@ -46,25 +48,25 @@ struct platform_driver { ...@@ -46,25 +48,25 @@ struct platform_driver {
int (*resume_early)(struct platform_device *); int (*resume_early)(struct platform_device *);
int (*resume)(struct platform_device *); int (*resume)(struct platform_device *);
struct device_driver driver; struct device_driver driver;
}; };
Note that probe() should in general verify that the specified device hardware Note that probe() should in general verify that the specified device hardware
actually exists; sometimes platform setup code can't be sure. The probing actually exists; sometimes platform setup code can't be sure. The probing
can use device resources, including clocks, and device platform_data. can use device resources, including clocks, and device platform_data.
Platform drivers register themselves the normal way: Platform drivers register themselves the normal way::
int platform_driver_register(struct platform_driver *drv); int platform_driver_register(struct platform_driver *drv);
Or, in common situations where the device is known not to be hot-pluggable, Or, in common situations where the device is known not to be hot-pluggable,
the probe() routine can live in an init section to reduce the driver's the probe() routine can live in an init section to reduce the driver's
runtime memory footprint: runtime memory footprint::
int platform_driver_probe(struct platform_driver *drv, int platform_driver_probe(struct platform_driver *drv,
int (*probe)(struct platform_device *)) int (*probe)(struct platform_device *))
Kernel modules can be composed of several platform drivers. The platform core Kernel modules can be composed of several platform drivers. The platform core
provides helpers to register and unregister an array of drivers: provides helpers to register and unregister an array of drivers::
int __platform_register_drivers(struct platform_driver * const *drivers, int __platform_register_drivers(struct platform_driver * const *drivers,
unsigned int count, struct module *owner); unsigned int count, struct module *owner);
...@@ -73,7 +75,7 @@ provides helpers to register and unregister an array of drivers: ...@@ -73,7 +75,7 @@ provides helpers to register and unregister an array of drivers:
If one of the drivers fails to register, all drivers registered up to that If one of the drivers fails to register, all drivers registered up to that
point will be unregistered in reverse order. Note that there is a convenience point will be unregistered in reverse order. Note that there is a convenience
macro that passes THIS_MODULE as owner parameter: macro that passes THIS_MODULE as owner parameter::
#define platform_register_drivers(drivers, count) #define platform_register_drivers(drivers, count)
...@@ -81,7 +83,7 @@ macro that passes THIS_MODULE as owner parameter: ...@@ -81,7 +83,7 @@ macro that passes THIS_MODULE as owner parameter:
Device Enumeration Device Enumeration
~~~~~~~~~~~~~~~~~~ ~~~~~~~~~~~~~~~~~~
As a rule, platform specific (and often board-specific) setup code will As a rule, platform specific (and often board-specific) setup code will
register platform devices: register platform devices::
int platform_device_register(struct platform_device *pdev); int platform_device_register(struct platform_device *pdev);
...@@ -133,14 +135,14 @@ tend to already have "normal" modes, such as ones using device nodes that ...@@ -133,14 +135,14 @@ tend to already have "normal" modes, such as ones using device nodes that
were created by PNP or by platform device setup. were created by PNP or by platform device setup.
None the less, there are some APIs to support such legacy drivers. Avoid None the less, there are some APIs to support such legacy drivers. Avoid
using these calls except with such hotplug-deficient drivers. using these calls except with such hotplug-deficient drivers::
struct platform_device *platform_device_alloc( struct platform_device *platform_device_alloc(
const char *name, int id); const char *name, int id);
You can use platform_device_alloc() to dynamically allocate a device, which You can use platform_device_alloc() to dynamically allocate a device, which
you will then initialize with resources and platform_device_register(). you will then initialize with resources and platform_device_register().
A better solution is usually: A better solution is usually::
struct platform_device *platform_device_register_simple( struct platform_device *platform_device_register_simple(
const char *name, int id, const char *name, int id,
......
...@@ -103,7 +103,7 @@ id_table an array of NULL terminated EISA id strings, ...@@ -103,7 +103,7 @@ id_table an array of NULL terminated EISA id strings,
(driver_data). (driver_data).
driver a generic driver, such as described in driver a generic driver, such as described in
Documentation/driver-model/driver.txt. Only .name, Documentation/driver-model/driver.rst. Only .name,
.probe and .remove members are mandatory. .probe and .remove members are mandatory.
=============== ==================================================== =============== ====================================================
...@@ -152,7 +152,7 @@ state set of flags indicating the state of the device. Current ...@@ -152,7 +152,7 @@ state set of flags indicating the state of the device. Current
flags are EISA_CONFIG_ENABLED and EISA_CONFIG_FORCED. flags are EISA_CONFIG_ENABLED and EISA_CONFIG_FORCED.
res set of four 256 bytes I/O regions allocated to this device res set of four 256 bytes I/O regions allocated to this device
dma_mask DMA mask set from the parent device. dma_mask DMA mask set from the parent device.
dev generic device (see Documentation/driver-model/device.txt) dev generic device (see Documentation/driver-model/device.rst)
======== ============================================================ ======== ============================================================
You can get the 'struct eisa_device' from 'struct device' using the You can get the 'struct eisa_device' from 'struct device' using the
......
...@@ -89,7 +89,7 @@ increase the chances of your change being accepted. ...@@ -89,7 +89,7 @@ increase the chances of your change being accepted.
console. Excessive logging can seriously affect system performance. console. Excessive logging can seriously affect system performance.
* Use devres functions whenever possible to allocate resources. For rationale * Use devres functions whenever possible to allocate resources. For rationale
and supported functions, please see Documentation/driver-model/devres.txt. and supported functions, please see Documentation/driver-model/devres.rst.
If a function is not supported by devres, consider using devm_add_action(). If a function is not supported by devres, consider using devm_add_action().
* If the driver has a detect function, make sure it is silent. Debug messages * If the driver has a detect function, make sure it is silent. Debug messages
......
...@@ -5,7 +5,7 @@ ...@@ -5,7 +5,7 @@
* Copyright (c) 2002-3 Patrick Mochel * Copyright (c) 2002-3 Patrick Mochel
* Copyright (c) 2002-3 Open Source Development Labs * Copyright (c) 2002-3 Open Source Development Labs
* *
* Please see Documentation/driver-model/platform.txt for more * Please see Documentation/driver-model/platform.rst for more
* information. * information.
*/ */
......
...@@ -41,7 +41,7 @@ MODULE_PARM_DESC(mask, "GPIO channel mask."); ...@@ -41,7 +41,7 @@ MODULE_PARM_DESC(mask, "GPIO channel mask.");
/* /*
* FIXME: convert this singleton driver to use the state container * FIXME: convert this singleton driver to use the state container
* design pattern, see Documentation/driver-model/design-patterns.txt * design pattern, see Documentation/driver-model/design-patterns.rst
*/ */
static struct cs5535_gpio_chip { static struct cs5535_gpio_chip {
struct gpio_chip chip; struct gpio_chip chip;
......
...@@ -2237,7 +2237,7 @@ ice_probe(struct pci_dev *pdev, const struct pci_device_id __always_unused *ent) ...@@ -2237,7 +2237,7 @@ ice_probe(struct pci_dev *pdev, const struct pci_device_id __always_unused *ent)
struct ice_hw *hw; struct ice_hw *hw;
int err; int err;
/* this driver uses devres, see Documentation/driver-model/devres.txt */ /* this driver uses devres, see Documentation/driver-model/devres.rst */
err = pcim_enable_device(pdev); err = pcim_enable_device(pdev);
if (err) if (err)
return err; return err;
......
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
/// functions. Values allocated using the devm_functions are freed when /// functions. Values allocated using the devm_functions are freed when
/// the device is detached, and thus the use of the standard freeing /// the device is detached, and thus the use of the standard freeing
/// function would cause a double free. /// function would cause a double free.
/// See Documentation/driver-model/devres.txt for more information. /// See Documentation/driver-model/devres.rst for more information.
/// ///
/// A difficulty of detecting this problem is that the standard freeing /// A difficulty of detecting this problem is that the standard freeing
/// function might be called from a different function than the one /// function might be called from a different function than the one
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment