diff options
| -rw-r--r-- | drivers/base/bus.c | 34 | ||||
| -rw-r--r-- | drivers/base/core.c | 17 | ||||
| -rw-r--r-- | drivers/base/driver.c | 15 | ||||
| -rw-r--r-- | include/linux/klist.h | 8 | ||||
| -rw-r--r-- | lib/klist.c | 18 |
5 files changed, 84 insertions, 8 deletions
diff --git a/drivers/base/bus.c b/drivers/base/bus.c index 17e96698410e..03204bfd17af 100644 --- a/drivers/base/bus.c +++ b/drivers/base/bus.c | |||
| @@ -568,6 +568,36 @@ static void bus_remove_attrs(struct bus_type * bus) | |||
| 568 | } | 568 | } |
| 569 | } | 569 | } |
| 570 | 570 | ||
| 571 | static void klist_devices_get(struct klist_node *n) | ||
| 572 | { | ||
| 573 | struct device *dev = container_of(n, struct device, knode_bus); | ||
| 574 | |||
| 575 | get_device(dev); | ||
| 576 | } | ||
| 577 | |||
| 578 | static void klist_devices_put(struct klist_node *n) | ||
| 579 | { | ||
| 580 | struct device *dev = container_of(n, struct device, knode_bus); | ||
| 581 | |||
| 582 | put_device(dev); | ||
| 583 | } | ||
| 584 | |||
| 585 | static void klist_drivers_get(struct klist_node *n) | ||
| 586 | { | ||
| 587 | struct device_driver *drv = container_of(n, struct device_driver, | ||
| 588 | knode_bus); | ||
| 589 | |||
| 590 | get_driver(drv); | ||
| 591 | } | ||
| 592 | |||
| 593 | static void klist_drivers_put(struct klist_node *n) | ||
| 594 | { | ||
| 595 | struct device_driver *drv = container_of(n, struct device_driver, | ||
| 596 | knode_bus); | ||
| 597 | |||
| 598 | put_driver(drv); | ||
| 599 | } | ||
| 600 | |||
| 571 | /** | 601 | /** |
| 572 | * bus_register - register a bus with the system. | 602 | * bus_register - register a bus with the system. |
| 573 | * @bus: bus. | 603 | * @bus: bus. |
| @@ -602,8 +632,8 @@ int bus_register(struct bus_type * bus) | |||
| 602 | if (retval) | 632 | if (retval) |
| 603 | goto bus_drivers_fail; | 633 | goto bus_drivers_fail; |
| 604 | 634 | ||
| 605 | klist_init(&bus->klist_devices); | 635 | klist_init(&bus->klist_devices, klist_devices_get, klist_devices_put); |
| 606 | klist_init(&bus->klist_drivers); | 636 | klist_init(&bus->klist_drivers, klist_drivers_get, klist_drivers_put); |
| 607 | bus_add_attrs(bus); | 637 | bus_add_attrs(bus); |
| 608 | 638 | ||
| 609 | pr_debug("bus type '%s' registered\n", bus->name); | 639 | pr_debug("bus type '%s' registered\n", bus->name); |
diff --git a/drivers/base/core.c b/drivers/base/core.c index c8a33df00761..6ab73f5c799a 100644 --- a/drivers/base/core.c +++ b/drivers/base/core.c | |||
| @@ -191,6 +191,20 @@ void device_remove_file(struct device * dev, struct device_attribute * attr) | |||
| 191 | } | 191 | } |
| 192 | } | 192 | } |
| 193 | 193 | ||
| 194 | static void klist_children_get(struct klist_node *n) | ||
| 195 | { | ||
| 196 | struct device *dev = container_of(n, struct device, knode_parent); | ||
| 197 | |||
| 198 | get_device(dev); | ||
| 199 | } | ||
| 200 | |||
| 201 | static void klist_children_put(struct klist_node *n) | ||
| 202 | { | ||
| 203 | struct device *dev = container_of(n, struct device, knode_parent); | ||
| 204 | |||
| 205 | put_device(dev); | ||
| 206 | } | ||
| 207 | |||
| 194 | 208 | ||
| 195 | /** | 209 | /** |
| 196 | * device_initialize - init device structure. | 210 | * device_initialize - init device structure. |
| @@ -207,7 +221,8 @@ void device_initialize(struct device *dev) | |||
| 207 | { | 221 | { |
| 208 | kobj_set_kset_s(dev, devices_subsys); | 222 | kobj_set_kset_s(dev, devices_subsys); |
| 209 | kobject_init(&dev->kobj); | 223 | kobject_init(&dev->kobj); |
| 210 | klist_init(&dev->klist_children); | 224 | klist_init(&dev->klist_children, klist_children_get, |
| 225 | klist_children_put); | ||
| 211 | INIT_LIST_HEAD(&dev->dma_pools); | 226 | INIT_LIST_HEAD(&dev->dma_pools); |
| 212 | init_MUTEX(&dev->sem); | 227 | init_MUTEX(&dev->sem); |
| 213 | } | 228 | } |
diff --git a/drivers/base/driver.c b/drivers/base/driver.c index 291c5954a3af..ef3fe513e398 100644 --- a/drivers/base/driver.c +++ b/drivers/base/driver.c | |||
| @@ -142,6 +142,19 @@ void put_driver(struct device_driver * drv) | |||
| 142 | kobject_put(&drv->kobj); | 142 | kobject_put(&drv->kobj); |
| 143 | } | 143 | } |
| 144 | 144 | ||
| 145 | static void klist_devices_get(struct klist_node *n) | ||
| 146 | { | ||
| 147 | struct device *dev = container_of(n, struct device, knode_driver); | ||
| 148 | |||
| 149 | get_device(dev); | ||
| 150 | } | ||
| 151 | |||
| 152 | static void klist_devices_put(struct klist_node *n) | ||
| 153 | { | ||
| 154 | struct device *dev = container_of(n, struct device, knode_driver); | ||
| 155 | |||
| 156 | put_device(dev); | ||
| 157 | } | ||
| 145 | 158 | ||
| 146 | /** | 159 | /** |
| 147 | * driver_register - register driver with bus | 160 | * driver_register - register driver with bus |
| @@ -157,7 +170,7 @@ void put_driver(struct device_driver * drv) | |||
| 157 | */ | 170 | */ |
| 158 | int driver_register(struct device_driver * drv) | 171 | int driver_register(struct device_driver * drv) |
| 159 | { | 172 | { |
| 160 | klist_init(&drv->klist_devices); | 173 | klist_init(&drv->klist_devices, klist_devices_get, klist_devices_put); |
| 161 | init_completion(&drv->unloaded); | 174 | init_completion(&drv->unloaded); |
| 162 | return bus_add_driver(drv); | 175 | return bus_add_driver(drv); |
| 163 | } | 176 | } |
diff --git a/include/linux/klist.h b/include/linux/klist.h index c4d1fae4dd89..74071254c9d3 100644 --- a/include/linux/klist.h +++ b/include/linux/klist.h | |||
| @@ -17,15 +17,17 @@ | |||
| 17 | #include <linux/kref.h> | 17 | #include <linux/kref.h> |
| 18 | #include <linux/list.h> | 18 | #include <linux/list.h> |
| 19 | 19 | ||
| 20 | 20 | struct klist_node; | |
| 21 | struct klist { | 21 | struct klist { |
| 22 | spinlock_t k_lock; | 22 | spinlock_t k_lock; |
| 23 | struct list_head k_list; | 23 | struct list_head k_list; |
| 24 | void (*get)(struct klist_node *); | ||
| 25 | void (*put)(struct klist_node *); | ||
| 24 | }; | 26 | }; |
| 25 | 27 | ||
| 26 | 28 | ||
| 27 | extern void klist_init(struct klist * k); | 29 | extern void klist_init(struct klist * k, void (*get)(struct klist_node *), |
| 28 | 30 | void (*put)(struct klist_node *)); | |
| 29 | 31 | ||
| 30 | struct klist_node { | 32 | struct klist_node { |
| 31 | struct klist * n_klist; | 33 | struct klist * n_klist; |
diff --git a/lib/klist.c b/lib/klist.c index a70c836c5c4c..bb2f3551d50a 100644 --- a/lib/klist.c +++ b/lib/klist.c | |||
| @@ -42,12 +42,23 @@ | |||
| 42 | /** | 42 | /** |
| 43 | * klist_init - Initialize a klist structure. | 43 | * klist_init - Initialize a klist structure. |
| 44 | * @k: The klist we're initializing. | 44 | * @k: The klist we're initializing. |
| 45 | * @get: The get function for the embedding object (NULL if none) | ||
| 46 | * @put: The put function for the embedding object (NULL if none) | ||
| 47 | * | ||
| 48 | * Initialises the klist structure. If the klist_node structures are | ||
| 49 | * going to be embedded in refcounted objects (necessary for safe | ||
| 50 | * deletion) then the get/put arguments are used to initialise | ||
| 51 | * functions that take and release references on the embedding | ||
| 52 | * objects. | ||
| 45 | */ | 53 | */ |
| 46 | 54 | ||
| 47 | void klist_init(struct klist * k) | 55 | void klist_init(struct klist * k, void (*get)(struct klist_node *), |
| 56 | void (*put)(struct klist_node *)) | ||
| 48 | { | 57 | { |
| 49 | INIT_LIST_HEAD(&k->k_list); | 58 | INIT_LIST_HEAD(&k->k_list); |
| 50 | spin_lock_init(&k->k_lock); | 59 | spin_lock_init(&k->k_lock); |
| 60 | k->get = get; | ||
| 61 | k->put = put; | ||
| 51 | } | 62 | } |
| 52 | 63 | ||
| 53 | EXPORT_SYMBOL_GPL(klist_init); | 64 | EXPORT_SYMBOL_GPL(klist_init); |
| @@ -74,6 +85,8 @@ static void klist_node_init(struct klist * k, struct klist_node * n) | |||
| 74 | init_completion(&n->n_removed); | 85 | init_completion(&n->n_removed); |
| 75 | kref_init(&n->n_ref); | 86 | kref_init(&n->n_ref); |
| 76 | n->n_klist = k; | 87 | n->n_klist = k; |
| 88 | if (k->get) | ||
| 89 | k->get(n); | ||
| 77 | } | 90 | } |
| 78 | 91 | ||
| 79 | 92 | ||
| @@ -110,9 +123,12 @@ EXPORT_SYMBOL_GPL(klist_add_tail); | |||
| 110 | static void klist_release(struct kref * kref) | 123 | static void klist_release(struct kref * kref) |
| 111 | { | 124 | { |
| 112 | struct klist_node * n = container_of(kref, struct klist_node, n_ref); | 125 | struct klist_node * n = container_of(kref, struct klist_node, n_ref); |
| 126 | void (*put)(struct klist_node *) = n->n_klist->put; | ||
| 113 | list_del(&n->n_node); | 127 | list_del(&n->n_node); |
| 114 | complete(&n->n_removed); | 128 | complete(&n->n_removed); |
| 115 | n->n_klist = NULL; | 129 | n->n_klist = NULL; |
| 130 | if (put) | ||
| 131 | put(n); | ||
| 116 | } | 132 | } |
| 117 | 133 | ||
| 118 | static int klist_dec_and_del(struct klist_node * n) | 134 | static int klist_dec_and_del(struct klist_node * n) |
