Tejun Heo | 9ac7849 | 2007-01-20 16:00:26 +0900 | [diff] [blame] | 1 | Devres - Managed Device Resource |
| 2 | ================================ |
| 3 | |
| 4 | Tejun Heo <teheo@suse.de> |
| 5 | |
| 6 | First draft 10 January 2007 |
| 7 | |
| 8 | |
| 9 | 1. Intro : Huh? Devres? |
| 10 | 2. Devres : Devres in a nutshell |
| 11 | 3. Devres Group : Group devres'es and release them together |
| 12 | 4. Details : Life time rules, calling context, ... |
| 13 | 5. Overhead : How much do we have to pay for this? |
| 14 | 6. List of managed interfaces : Currently implemented managed interfaces |
| 15 | |
| 16 | |
| 17 | 1. Intro |
| 18 | -------- |
| 19 | |
| 20 | devres came up while trying to convert libata to use iomap. Each |
| 21 | iomapped address should be kept and unmapped on driver detach. For |
| 22 | example, a plain SFF ATA controller (that is, good old PCI IDE) in |
| 23 | native mode makes use of 5 PCI BARs and all of them should be |
| 24 | maintained. |
| 25 | |
| 26 | As with many other device drivers, libata low level drivers have |
| 27 | sufficient bugs in ->remove and ->probe failure path. Well, yes, |
| 28 | that's probably because libata low level driver developers are lazy |
| 29 | bunch, but aren't all low level driver developers? After spending a |
| 30 | day fiddling with braindamaged hardware with no document or |
| 31 | braindamaged document, if it's finally working, well, it's working. |
| 32 | |
| 33 | For one reason or another, low level drivers don't receive as much |
| 34 | attention or testing as core code, and bugs on driver detach or |
Matt LaPlante | 01dd2fb | 2007-10-20 01:34:40 +0200 | [diff] [blame] | 35 | initialization failure don't happen often enough to be noticeable. |
Tejun Heo | 9ac7849 | 2007-01-20 16:00:26 +0900 | [diff] [blame] | 36 | Init failure path is worse because it's much less travelled while |
| 37 | needs to handle multiple entry points. |
| 38 | |
| 39 | So, many low level drivers end up leaking resources on driver detach |
| 40 | and having half broken failure path implementation in ->probe() which |
| 41 | would leak resources or even cause oops when failure occurs. iomap |
| 42 | adds more to this mix. So do msi and msix. |
| 43 | |
| 44 | |
| 45 | 2. Devres |
| 46 | --------- |
| 47 | |
| 48 | devres is basically linked list of arbitrarily sized memory areas |
| 49 | associated with a struct device. Each devres entry is associated with |
| 50 | a release function. A devres can be released in several ways. No |
| 51 | matter what, all devres entries are released on driver detach. On |
| 52 | release, the associated release function is invoked and then the |
| 53 | devres entry is freed. |
| 54 | |
| 55 | Managed interface is created for resources commonly used by device |
| 56 | drivers using devres. For example, coherent DMA memory is acquired |
| 57 | using dma_alloc_coherent(). The managed version is called |
| 58 | dmam_alloc_coherent(). It is identical to dma_alloc_coherent() except |
| 59 | for the DMA memory allocated using it is managed and will be |
| 60 | automatically released on driver detach. Implementation looks like |
| 61 | the following. |
| 62 | |
| 63 | struct dma_devres { |
| 64 | size_t size; |
| 65 | void *vaddr; |
| 66 | dma_addr_t dma_handle; |
| 67 | }; |
| 68 | |
| 69 | static void dmam_coherent_release(struct device *dev, void *res) |
| 70 | { |
| 71 | struct dma_devres *this = res; |
| 72 | |
| 73 | dma_free_coherent(dev, this->size, this->vaddr, this->dma_handle); |
| 74 | } |
| 75 | |
| 76 | dmam_alloc_coherent(dev, size, dma_handle, gfp) |
| 77 | { |
| 78 | struct dma_devres *dr; |
| 79 | void *vaddr; |
| 80 | |
| 81 | dr = devres_alloc(dmam_coherent_release, sizeof(*dr), gfp); |
| 82 | ... |
| 83 | |
| 84 | /* alloc DMA memory as usual */ |
| 85 | vaddr = dma_alloc_coherent(...); |
| 86 | ... |
| 87 | |
| 88 | /* record size, vaddr, dma_handle in dr */ |
| 89 | dr->vaddr = vaddr; |
| 90 | ... |
| 91 | |
| 92 | devres_add(dev, dr); |
| 93 | |
| 94 | return vaddr; |
| 95 | } |
| 96 | |
| 97 | If a driver uses dmam_alloc_coherent(), the area is guaranteed to be |
| 98 | freed whether initialization fails half-way or the device gets |
| 99 | detached. If most resources are acquired using managed interface, a |
| 100 | driver can have much simpler init and exit code. Init path basically |
| 101 | looks like the following. |
| 102 | |
| 103 | my_init_one() |
| 104 | { |
| 105 | struct mydev *d; |
| 106 | |
| 107 | d = devm_kzalloc(dev, sizeof(*d), GFP_KERNEL); |
| 108 | if (!d) |
| 109 | return -ENOMEM; |
| 110 | |
| 111 | d->ring = dmam_alloc_coherent(...); |
| 112 | if (!d->ring) |
| 113 | return -ENOMEM; |
| 114 | |
| 115 | if (check something) |
| 116 | return -EINVAL; |
| 117 | ... |
| 118 | |
| 119 | return register_to_upper_layer(d); |
| 120 | } |
| 121 | |
| 122 | And exit path, |
| 123 | |
| 124 | my_remove_one() |
| 125 | { |
| 126 | unregister_from_upper_layer(d); |
| 127 | shutdown_my_hardware(); |
| 128 | } |
| 129 | |
| 130 | As shown above, low level drivers can be simplified a lot by using |
| 131 | devres. Complexity is shifted from less maintained low level drivers |
| 132 | to better maintained higher layer. Also, as init failure path is |
| 133 | shared with exit path, both can get more testing. |
| 134 | |
| 135 | |
| 136 | 3. Devres group |
| 137 | --------------- |
| 138 | |
| 139 | Devres entries can be grouped using devres group. When a group is |
| 140 | released, all contained normal devres entries and properly nested |
| 141 | groups are released. One usage is to rollback series of acquired |
| 142 | resources on failure. For example, |
| 143 | |
| 144 | if (!devres_open_group(dev, NULL, GFP_KERNEL)) |
| 145 | return -ENOMEM; |
| 146 | |
| 147 | acquire A; |
| 148 | if (failed) |
| 149 | goto err; |
| 150 | |
| 151 | acquire B; |
| 152 | if (failed) |
| 153 | goto err; |
| 154 | ... |
| 155 | |
| 156 | devres_remove_group(dev, NULL); |
| 157 | return 0; |
| 158 | |
| 159 | err: |
| 160 | devres_release_group(dev, NULL); |
| 161 | return err_code; |
| 162 | |
Matt LaPlante | 01dd2fb | 2007-10-20 01:34:40 +0200 | [diff] [blame] | 163 | As resource acquisition failure usually means probe failure, constructs |
Tejun Heo | 9ac7849 | 2007-01-20 16:00:26 +0900 | [diff] [blame] | 164 | like above are usually useful in midlayer driver (e.g. libata core |
| 165 | layer) where interface function shouldn't have side effect on failure. |
| 166 | For LLDs, just returning error code suffices in most cases. |
| 167 | |
| 168 | Each group is identified by void *id. It can either be explicitly |
| 169 | specified by @id argument to devres_open_group() or automatically |
| 170 | created by passing NULL as @id as in the above example. In both |
| 171 | cases, devres_open_group() returns the group's id. The returned id |
| 172 | can be passed to other devres functions to select the target group. |
| 173 | If NULL is given to those functions, the latest open group is |
| 174 | selected. |
| 175 | |
| 176 | For example, you can do something like the following. |
| 177 | |
| 178 | int my_midlayer_create_something() |
| 179 | { |
| 180 | if (!devres_open_group(dev, my_midlayer_create_something, GFP_KERNEL)) |
| 181 | return -ENOMEM; |
| 182 | |
| 183 | ... |
| 184 | |
Rolf Eike Beer | 3265b54 | 2007-05-01 11:00:19 +0200 | [diff] [blame] | 185 | devres_close_group(dev, my_midlayer_create_something); |
Tejun Heo | 9ac7849 | 2007-01-20 16:00:26 +0900 | [diff] [blame] | 186 | return 0; |
| 187 | } |
| 188 | |
| 189 | void my_midlayer_destroy_something() |
| 190 | { |
Matt LaPlante | 19f5946 | 2009-04-27 15:06:31 +0200 | [diff] [blame] | 191 | devres_release_group(dev, my_midlayer_create_something); |
Tejun Heo | 9ac7849 | 2007-01-20 16:00:26 +0900 | [diff] [blame] | 192 | } |
| 193 | |
| 194 | |
| 195 | 4. Details |
| 196 | ---------- |
| 197 | |
| 198 | Lifetime of a devres entry begins on devres allocation and finishes |
| 199 | when it is released or destroyed (removed and freed) - no reference |
| 200 | counting. |
| 201 | |
| 202 | devres core guarantees atomicity to all basic devres operations and |
| 203 | has support for single-instance devres types (atomic |
| 204 | lookup-and-add-if-not-found). Other than that, synchronizing |
| 205 | concurrent accesses to allocated devres data is caller's |
| 206 | responsibility. This is usually non-issue because bus ops and |
| 207 | resource allocations already do the job. |
| 208 | |
| 209 | For an example of single-instance devres type, read pcim_iomap_table() |
Brandon Philips | 2c19c49 | 2007-07-17 22:09:34 -0700 | [diff] [blame] | 210 | in lib/devres.c. |
Tejun Heo | 9ac7849 | 2007-01-20 16:00:26 +0900 | [diff] [blame] | 211 | |
| 212 | All devres interface functions can be called without context if the |
| 213 | right gfp mask is given. |
| 214 | |
| 215 | |
| 216 | 5. Overhead |
| 217 | ----------- |
| 218 | |
| 219 | Each devres bookkeeping info is allocated together with requested data |
| 220 | area. With debug option turned off, bookkeeping info occupies 16 |
| 221 | bytes on 32bit machines and 24 bytes on 64bit (three pointers rounded |
| 222 | up to ull alignment). If singly linked list is used, it can be |
| 223 | reduced to two pointers (8 bytes on 32bit, 16 bytes on 64bit). |
| 224 | |
| 225 | Each devres group occupies 8 pointers. It can be reduced to 6 if |
| 226 | singly linked list is used. |
| 227 | |
| 228 | Memory space overhead on ahci controller with two ports is between 300 |
| 229 | and 400 bytes on 32bit machine after naive conversion (we can |
| 230 | certainly invest a bit more effort into libata core layer). |
| 231 | |
| 232 | |
| 233 | 6. List of managed interfaces |
| 234 | ----------------------------- |
| 235 | |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 236 | CLOCK |
| 237 | devm_clk_get() |
| 238 | devm_clk_put() |
Stephen Boyd | 4143804 | 2016-02-05 17:02:52 -0800 | [diff] [blame] | 239 | devm_clk_hw_register() |
Stephen Boyd | aa795c4 | 2017-09-01 16:16:40 -0700 | [diff] [blame] | 240 | devm_of_clk_add_hw_provider() |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 241 | |
| 242 | DMA |
| 243 | dmam_alloc_coherent() |
Christoph Hellwig | 63d36c9 | 2017-06-12 19:15:04 +0200 | [diff] [blame] | 244 | dmam_alloc_attrs() |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 245 | dmam_declare_coherent_memory() |
| 246 | dmam_free_coherent() |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 247 | dmam_pool_create() |
| 248 | dmam_pool_destroy() |
| 249 | |
| 250 | GPIO |
| 251 | devm_gpiod_get() |
| 252 | devm_gpiod_get_index() |
| 253 | devm_gpiod_get_index_optional() |
| 254 | devm_gpiod_get_optional() |
| 255 | devm_gpiod_put() |
Laxman Dewangan | 38115ea | 2016-02-22 15:00:08 +0530 | [diff] [blame] | 256 | devm_gpiochip_add_data() |
| 257 | devm_gpiochip_remove() |
Laxman Dewangan | 77ae582 | 2016-02-22 15:04:08 +0530 | [diff] [blame] | 258 | devm_gpio_request() |
| 259 | devm_gpio_request_one() |
| 260 | devm_gpio_free() |
Wolfram Sang | 543f43c | 2012-01-15 13:31:46 +0100 | [diff] [blame] | 261 | |
Oleksandr Kravchenko | 224b995 | 2013-07-23 09:39:00 +0100 | [diff] [blame] | 262 | IIO |
| 263 | devm_iio_device_alloc() |
| 264 | devm_iio_device_free() |
Sachin Kamat | 8caa07c | 2013-10-29 11:39:00 +0000 | [diff] [blame] | 265 | devm_iio_device_register() |
| 266 | devm_iio_device_unregister() |
Karol Wrona | 780103f | 2014-12-19 18:39:25 +0100 | [diff] [blame] | 267 | devm_iio_kfifo_allocate() |
| 268 | devm_iio_kfifo_free() |
Gregor Boirie | 70e4834 | 2016-09-02 20:47:55 +0200 | [diff] [blame] | 269 | devm_iio_triggered_buffer_setup() |
| 270 | devm_iio_triggered_buffer_cleanup() |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 271 | devm_iio_trigger_alloc() |
| 272 | devm_iio_trigger_free() |
Gregor Boirie | 9083325 | 2016-09-02 20:47:54 +0200 | [diff] [blame] | 273 | devm_iio_trigger_register() |
| 274 | devm_iio_trigger_unregister() |
Laxman Dewangan | e21a294 | 2016-04-06 16:01:08 +0530 | [diff] [blame] | 275 | devm_iio_channel_get() |
| 276 | devm_iio_channel_release() |
| 277 | devm_iio_channel_get_all() |
| 278 | devm_iio_channel_release_all() |
Oleksandr Kravchenko | 224b995 | 2013-07-23 09:39:00 +0100 | [diff] [blame] | 279 | |
Alexander Kurz | 2ea2dc8 | 2016-04-21 19:02:04 +0200 | [diff] [blame] | 280 | INPUT |
| 281 | devm_input_allocate_device() |
| 282 | |
Tejun Heo | 9ac7849 | 2007-01-20 16:00:26 +0900 | [diff] [blame] | 283 | IO region |
Tejun Heo | 9ac7849 | 2007-01-20 16:00:26 +0900 | [diff] [blame] | 284 | devm_release_mem_region() |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 285 | devm_release_region() |
Thierry Reding | 8d38821 | 2014-08-01 14:15:10 +0200 | [diff] [blame] | 286 | devm_release_resource() |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 287 | devm_request_mem_region() |
| 288 | devm_request_region() |
Thierry Reding | 8d38821 | 2014-08-01 14:15:10 +0200 | [diff] [blame] | 289 | devm_request_resource() |
Tejun Heo | 9ac7849 | 2007-01-20 16:00:26 +0900 | [diff] [blame] | 290 | |
| 291 | IOMAP |
| 292 | devm_ioport_map() |
| 293 | devm_ioport_unmap() |
| 294 | devm_ioremap() |
| 295 | devm_ioremap_nocache() |
Abhilash Kesavan | 3464452 | 2015-02-06 19:15:27 +0530 | [diff] [blame] | 296 | devm_ioremap_wc() |
Thierry Reding | 7509657 | 2013-01-21 11:08:54 +0100 | [diff] [blame] | 297 | devm_ioremap_resource() : checks resource, requests memory region, ioremaps |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 298 | devm_iounmap() |
Tejun Heo | 9ac7849 | 2007-01-20 16:00:26 +0900 | [diff] [blame] | 299 | pcim_iomap() |
Tejun Heo | 9ac7849 | 2007-01-20 16:00:26 +0900 | [diff] [blame] | 300 | pcim_iomap_regions() : do request_region() and iomap() on multiple BARs |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 301 | pcim_iomap_table() : array of mapped addresses indexed by BAR |
| 302 | pcim_iounmap() |
Stephen Boyd | 070b907 | 2012-01-16 19:39:58 -0800 | [diff] [blame] | 303 | |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 304 | IRQ |
| 305 | devm_free_irq() |
Tobias Klauser | ea05166 | 2014-08-14 10:05:03 +0200 | [diff] [blame] | 306 | devm_request_any_context_irq() |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 307 | devm_request_irq() |
Tobias Klauser | ea05166 | 2014-08-14 10:05:03 +0200 | [diff] [blame] | 308 | devm_request_threaded_irq() |
Bartosz Golaszewski | 2b5e773 | 2017-02-10 13:23:23 +0100 | [diff] [blame] | 309 | devm_irq_alloc_descs() |
| 310 | devm_irq_alloc_desc() |
| 311 | devm_irq_alloc_desc_at() |
| 312 | devm_irq_alloc_desc_from() |
| 313 | devm_irq_alloc_descs_from() |
Bartosz Golaszewski | 1c3e363 | 2017-05-31 18:06:59 +0200 | [diff] [blame] | 314 | devm_irq_alloc_generic_chip() |
Bartosz Golaszewski | 30fd8fc | 2017-05-31 18:07:00 +0200 | [diff] [blame] | 315 | devm_irq_setup_generic_chip() |
Bartosz Golaszewski | 44e72c7 | 2017-08-14 16:53:17 +0200 | [diff] [blame] | 316 | devm_irq_sim_init() |
Mark Brown | a8a97db | 2012-04-05 11:42:09 +0100 | [diff] [blame] | 317 | |
Bjorn Andersson | ca1bb4e | 2015-02-23 16:11:41 -0800 | [diff] [blame] | 318 | LED |
| 319 | devm_led_classdev_register() |
| 320 | devm_led_classdev_unregister() |
| 321 | |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 322 | MDIO |
| 323 | devm_mdiobus_alloc() |
| 324 | devm_mdiobus_alloc_size() |
| 325 | devm_mdiobus_free() |
| 326 | |
| 327 | MEM |
| 328 | devm_free_pages() |
| 329 | devm_get_free_pages() |
Geert Uytterhoeven | bef59c5 | 2014-08-20 15:26:35 +0200 | [diff] [blame] | 330 | devm_kasprintf() |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 331 | devm_kcalloc() |
| 332 | devm_kfree() |
| 333 | devm_kmalloc() |
| 334 | devm_kmalloc_array() |
| 335 | devm_kmemdup() |
Geert Uytterhoeven | 5427035 | 2014-08-20 15:26:34 +0200 | [diff] [blame] | 336 | devm_kstrdup() |
Geert Uytterhoeven | bef59c5 | 2014-08-20 15:26:35 +0200 | [diff] [blame] | 337 | devm_kvasprintf() |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 338 | devm_kzalloc() |
| 339 | |
Laxman Dewangan | 3698283 | 2016-04-08 00:12:56 +0530 | [diff] [blame] | 340 | MFD |
Peter Rosin | b594b10 | 2017-05-14 21:51:04 +0200 | [diff] [blame] | 341 | devm_mfd_add_devices() |
Laxman Dewangan | 3698283 | 2016-04-08 00:12:56 +0530 | [diff] [blame] | 342 | |
Peter Rosin | a3b02a9 | 2017-05-14 21:51:06 +0200 | [diff] [blame] | 343 | MUX |
| 344 | devm_mux_chip_alloc() |
| 345 | devm_mux_chip_register() |
| 346 | devm_mux_control_get() |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 347 | |
Madalin Bucur | ff86aae | 2016-11-15 10:41:01 +0200 | [diff] [blame] | 348 | PER-CPU MEM |
| 349 | devm_alloc_percpu() |
| 350 | devm_free_percpu() |
| 351 | |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 352 | PCI |
Lorenzo Pieralisi | 5c3f18c | 2017-06-28 15:13:53 -0500 | [diff] [blame] | 353 | devm_pci_alloc_host_bridge() : managed PCI host bridge allocation |
Lorenzo Pieralisi | 490cb6d | 2017-04-19 17:48:55 +0100 | [diff] [blame] | 354 | devm_pci_remap_cfgspace() : ioremap PCI configuration space |
| 355 | devm_pci_remap_cfg_resource() : ioremap PCI configuration space resource |
| 356 | pcim_enable_device() : after success, all PCI ops become managed |
| 357 | pcim_pin_device() : keep PCI device enabled after release |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 358 | |
| 359 | PHY |
| 360 | devm_usb_get_phy() |
| 361 | devm_usb_put_phy() |
Linus Torvalds | 3d1482f | 2012-05-21 16:58:23 -0700 | [diff] [blame] | 362 | |
Stephen Warren | 6d4ca1f | 2012-04-16 10:51:00 -0600 | [diff] [blame] | 363 | PINCTRL |
| 364 | devm_pinctrl_get() |
| 365 | devm_pinctrl_put() |
Laxman Dewangan | 1f8dd72 | 2016-02-24 14:14:35 +0530 | [diff] [blame] | 366 | devm_pinctrl_register() |
| 367 | devm_pinctrl_unregister() |
Alexandre Courbot | 6354316 | 2012-08-01 19:20:58 +0900 | [diff] [blame] | 368 | |
Bjorn Andersson | c1a9634 | 2016-08-03 22:04:05 -0700 | [diff] [blame] | 369 | POWER |
| 370 | devm_reboot_mode_register() |
| 371 | devm_reboot_mode_unregister() |
| 372 | |
Alexandre Courbot | 6354316 | 2012-08-01 19:20:58 +0900 | [diff] [blame] | 373 | PWM |
| 374 | devm_pwm_get() |
| 375 | devm_pwm_put() |
Marko Katic | 2202d4e | 2012-12-13 19:14:54 +0100 | [diff] [blame] | 376 | |
Geert Uytterhoeven | d8e1e01 | 2014-07-10 10:10:23 +0200 | [diff] [blame] | 377 | REGULATOR |
| 378 | devm_regulator_bulk_get() |
| 379 | devm_regulator_get() |
| 380 | devm_regulator_put() |
| 381 | devm_regulator_register() |
Andy Shevchenko | 0244d84 | 2013-08-21 14:27:05 +0300 | [diff] [blame] | 382 | |
Masahiro Yamada | 8d5b5d5 | 2016-05-01 19:36:57 +0900 | [diff] [blame] | 383 | RESET |
| 384 | devm_reset_control_get() |
| 385 | devm_reset_controller_register() |
| 386 | |
Andrey Smirnov | 2cb67d2 | 2017-12-20 22:51:15 -0800 | [diff] [blame] | 387 | SERDEV |
| 388 | devm_serdev_device_open() |
| 389 | |
Andy Shevchenko | 0244d84 | 2013-08-21 14:27:05 +0300 | [diff] [blame] | 390 | SLAVE DMA ENGINE |
| 391 | devm_acpi_dma_controller_register() |
Mark Brown | 666d5b4 | 2013-08-31 18:50:52 +0100 | [diff] [blame] | 392 | |
| 393 | SPI |
| 394 | devm_spi_register_master() |
Neil Armstrong | 83fbae5 | 2016-05-27 17:33:54 +0200 | [diff] [blame] | 395 | |
| 396 | WATCHDOG |
| 397 | devm_watchdog_register_device() |