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) |