diff options
author | Russell King <rmk+kernel@arm.linux.org.uk> | 2012-01-20 17:51:07 -0500 |
---|---|---|
committer | Russell King <rmk+kernel@arm.linux.org.uk> | 2012-02-03 12:37:48 -0500 |
commit | 30816ac0495cb4f33fc8d748f64ac3cc880cb3c1 (patch) | |
tree | 689f82c11f0a1097bbd0e715178f94aa345fcebf /drivers/mfd/mcp-sa11x0.c | |
parent | 0af5e4c36e70cfd4ae96d3704a425c414f530f2a (diff) |
MFD: mcp-core: sanitize host creation/removal
host_unregister() gives us no chance between removing the device
and the mcp data structure being freed to access the data inbetween,
which drivers may need to do if they need to iounmap() pointers in
their private data structures.
Therefore, re-jig the interfaces, which are now, on creation:
mcp = mcp_host_alloc()
if (mcp) {
ret = mcp_host_add(mcp, data);
if (!ret)
mcp_host_free(mcp);
}
and on removal:
mcp_host_del(mcp);
... access mcp ...
mcp_host_free(mcp);
The free does the final put_device() on the struct device as one would
expect.
Acked-by: Jochen Friedrich <jochen@scram.de>
Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>
Diffstat (limited to 'drivers/mfd/mcp-sa11x0.c')
-rw-r--r-- | drivers/mfd/mcp-sa11x0.c | 6 |
1 files changed, 4 insertions, 2 deletions
diff --git a/drivers/mfd/mcp-sa11x0.c b/drivers/mfd/mcp-sa11x0.c index 02c53a0766c4..33cadc0ec121 100644 --- a/drivers/mfd/mcp-sa11x0.c +++ b/drivers/mfd/mcp-sa11x0.c | |||
@@ -195,10 +195,11 @@ static int mcp_sa11x0_probe(struct platform_device *pdev) | |||
195 | mcp->rw_timeout = (64 * 3 * 1000000 + mcp->sclk_rate - 1) / | 195 | mcp->rw_timeout = (64 * 3 * 1000000 + mcp->sclk_rate - 1) / |
196 | mcp->sclk_rate; | 196 | mcp->sclk_rate; |
197 | 197 | ||
198 | ret = mcp_host_register(mcp); | 198 | ret = mcp_host_add(mcp); |
199 | if (ret == 0) | 199 | if (ret == 0) |
200 | goto out; | 200 | goto out; |
201 | 201 | ||
202 | mcp_host_free(mcp); | ||
202 | release: | 203 | release: |
203 | release_mem_region(0x80060000, 0x60); | 204 | release_mem_region(0x80060000, 0x60); |
204 | platform_set_drvdata(pdev, NULL); | 205 | platform_set_drvdata(pdev, NULL); |
@@ -212,7 +213,8 @@ static int mcp_sa11x0_remove(struct platform_device *dev) | |||
212 | struct mcp *mcp = platform_get_drvdata(dev); | 213 | struct mcp *mcp = platform_get_drvdata(dev); |
213 | 214 | ||
214 | platform_set_drvdata(dev, NULL); | 215 | platform_set_drvdata(dev, NULL); |
215 | mcp_host_unregister(mcp); | 216 | mcp_host_del(mcp); |
217 | mcp_host_free(mcp); | ||
216 | release_mem_region(0x80060000, 0x60); | 218 | release_mem_region(0x80060000, 0x60); |
217 | 219 | ||
218 | return 0; | 220 | return 0; |