diff options
author | Charles Keepax <ckeepax@opensource.wolfsonmicro.com> | 2012-11-09 11:15:28 -0500 |
---|---|---|
committer | Samuel Ortiz <sameo@linux.intel.com> | 2012-11-20 06:21:12 -0500 |
commit | b9fbb62eb61452d728c39b2e5020739c575aac53 (patch) | |
tree | 0cba6c18d652ea5f43972245ea07ef30a639a91f | |
parent | d511b9c3626c83874824b1118f8a2e84ef63a7d3 (diff) |
mfd: Only unregister platform devices allocated by the mfd core
mfd_remove_devices would iterate over all devices sharing a parent with
an mfd device regardless of whether they were allocated by the mfd core
or not. This especially caused problems when the device structure was
not contained within a platform_device, because to_platform_device is
used on each device pointer.
This patch defines a device_type for mfd devices and checks this is
present from mfd_remove_devices_fn before processing the device.
Cc: stable@vger.kernel.org
Signed-off-by: Charles Keepax <ckeepax@opensource.wolfsonmicro.com>
Tested-by: Peter Tyser <ptyser@xes-inc.com>
Reviewed-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
Signed-off-by: Samuel Ortiz <sameo@linux.intel.com>
-rw-r--r-- | drivers/mfd/mfd-core.c | 15 |
1 files changed, 13 insertions, 2 deletions
diff --git a/drivers/mfd/mfd-core.c b/drivers/mfd/mfd-core.c index f8b77711ad2d..7604f4e5df40 100644 --- a/drivers/mfd/mfd-core.c +++ b/drivers/mfd/mfd-core.c | |||
@@ -21,6 +21,10 @@ | |||
21 | #include <linux/irqdomain.h> | 21 | #include <linux/irqdomain.h> |
22 | #include <linux/of.h> | 22 | #include <linux/of.h> |
23 | 23 | ||
24 | static struct device_type mfd_dev_type = { | ||
25 | .name = "mfd_device", | ||
26 | }; | ||
27 | |||
24 | int mfd_cell_enable(struct platform_device *pdev) | 28 | int mfd_cell_enable(struct platform_device *pdev) |
25 | { | 29 | { |
26 | const struct mfd_cell *cell = mfd_get_cell(pdev); | 30 | const struct mfd_cell *cell = mfd_get_cell(pdev); |
@@ -91,6 +95,7 @@ static int mfd_add_device(struct device *parent, int id, | |||
91 | goto fail_device; | 95 | goto fail_device; |
92 | 96 | ||
93 | pdev->dev.parent = parent; | 97 | pdev->dev.parent = parent; |
98 | pdev->dev.type = &mfd_dev_type; | ||
94 | 99 | ||
95 | if (parent->of_node && cell->of_compatible) { | 100 | if (parent->of_node && cell->of_compatible) { |
96 | for_each_child_of_node(parent->of_node, np) { | 101 | for_each_child_of_node(parent->of_node, np) { |
@@ -204,10 +209,16 @@ EXPORT_SYMBOL(mfd_add_devices); | |||
204 | 209 | ||
205 | static int mfd_remove_devices_fn(struct device *dev, void *c) | 210 | static int mfd_remove_devices_fn(struct device *dev, void *c) |
206 | { | 211 | { |
207 | struct platform_device *pdev = to_platform_device(dev); | 212 | struct platform_device *pdev; |
208 | const struct mfd_cell *cell = mfd_get_cell(pdev); | 213 | const struct mfd_cell *cell; |
209 | atomic_t **usage_count = c; | 214 | atomic_t **usage_count = c; |
210 | 215 | ||
216 | if (dev->type != &mfd_dev_type) | ||
217 | return 0; | ||
218 | |||
219 | pdev = to_platform_device(dev); | ||
220 | cell = mfd_get_cell(pdev); | ||
221 | |||
211 | /* find the base address of usage_count pointers (for freeing) */ | 222 | /* find the base address of usage_count pointers (for freeing) */ |
212 | if (!*usage_count || (cell->usage_count < *usage_count)) | 223 | if (!*usage_count || (cell->usage_count < *usage_count)) |
213 | *usage_count = cell->usage_count; | 224 | *usage_count = cell->usage_count; |