aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/hwspinlock
diff options
context:
space:
mode:
authorOhad Ben-Cohen <ohad@wizery.com>2011-09-05 16:15:06 -0400
committerOhad Ben-Cohen <ohad@wizery.com>2011-09-21 12:45:32 -0400
commitc3c1250e93a7ab1327a9fc49d2a22405672f4204 (patch)
tree2dc99df2ab4657de87a66da7d0a047a95ff27711 /drivers/hwspinlock
parentc97f6dd0fe21dfd658c59c144a1b7fd5d8db04ac (diff)
hwspinlock/core/omap: fix id issues on multiple hwspinlock devices
hwspinlock devices provide system-wide hardware locks that are used by remote processors that have no other way to achieve synchronization. To achieve that, each physical lock must have a system-wide id number that is agreed upon, otherwise remote processors can't possibly assume they're using the same hardware lock. Usually boards have a single hwspinlock device, which provides several hwspinlocks, and in this case, they can be trivially numbered 0 to (num-of-locks - 1). In case boards have several hwspinlocks devices, a different base id should be used for each hwspinlock device (they can't all use 0 as a starting id!). While this is certainly not common, it's just plain wrong to just silently use 0 as a base id whenever the hwspinlock driver is probed. This patch provides a hwspinlock_pdata structure, that boards can use to set a different base id for each of the hwspinlock devices they may have, and demonstrates how to use it with the omap hwspinlock driver. While we're at it, make sure the hwspinlock core prints an explicit error message in case an hwspinlock is registered with an id number that already exists; this will help users catch such base id issues. Reported-by: Arnd Bergmann <arnd@arndb.de> Signed-off-by: Ohad Ben-Cohen <ohad@wizery.com> Acked-by: Tony Lindgren <tony@atomide.com>
Diffstat (limited to 'drivers/hwspinlock')
-rw-r--r--drivers/hwspinlock/hwspinlock_core.c2
-rw-r--r--drivers/hwspinlock/omap_hwspinlock.c6
2 files changed, 7 insertions, 1 deletions
diff --git a/drivers/hwspinlock/hwspinlock_core.c b/drivers/hwspinlock/hwspinlock_core.c
index af5175c5d5f4..4eb85b4a320e 100644
--- a/drivers/hwspinlock/hwspinlock_core.c
+++ b/drivers/hwspinlock/hwspinlock_core.c
@@ -282,6 +282,8 @@ int hwspin_lock_register(struct hwspinlock *hwlock)
282 spin_lock(&hwspinlock_tree_lock); 282 spin_lock(&hwspinlock_tree_lock);
283 283
284 ret = radix_tree_insert(&hwspinlock_tree, hwlock->id, hwlock); 284 ret = radix_tree_insert(&hwspinlock_tree, hwlock->id, hwlock);
285 if (ret == -EEXIST)
286 pr_err("hwspinlock id %d already exists!\n", hwlock->id);
285 if (ret) 287 if (ret)
286 goto out; 288 goto out;
287 289
diff --git a/drivers/hwspinlock/omap_hwspinlock.c b/drivers/hwspinlock/omap_hwspinlock.c
index d0583480fe33..2044d181e49d 100644
--- a/drivers/hwspinlock/omap_hwspinlock.c
+++ b/drivers/hwspinlock/omap_hwspinlock.c
@@ -94,12 +94,16 @@ static const struct hwspinlock_ops omap_hwspinlock_ops = {
94 94
95static int __devinit omap_hwspinlock_probe(struct platform_device *pdev) 95static int __devinit omap_hwspinlock_probe(struct platform_device *pdev)
96{ 96{
97 struct hwspinlock_pdata *pdata = pdev->dev.platform_data;
97 struct omap_hwspinlock *omap_lock; 98 struct omap_hwspinlock *omap_lock;
98 struct omap_hwspinlock_state *state; 99 struct omap_hwspinlock_state *state;
99 struct resource *res; 100 struct resource *res;
100 void __iomem *io_base; 101 void __iomem *io_base;
101 int i, ret; 102 int i, ret;
102 103
104 if (!pdata)
105 return -ENODEV;
106
103 res = platform_get_resource(pdev, IORESOURCE_MEM, 0); 107 res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
104 if (!res) 108 if (!res)
105 return -ENODEV; 109 return -ENODEV;
@@ -141,7 +145,7 @@ static int __devinit omap_hwspinlock_probe(struct platform_device *pdev)
141 omap_lock = &state->lock[i]; 145 omap_lock = &state->lock[i];
142 146
143 omap_lock->lock.dev = &pdev->dev; 147 omap_lock->lock.dev = &pdev->dev;
144 omap_lock->lock.id = i; 148 omap_lock->lock.id = pdata->base_id + i;
145 omap_lock->lock.ops = &omap_hwspinlock_ops; 149 omap_lock->lock.ops = &omap_hwspinlock_ops;
146 omap_lock->addr = io_base + LOCK_BASE_OFFSET + sizeof(u32) * i; 150 omap_lock->addr = io_base + LOCK_BASE_OFFSET + sizeof(u32) * i;
147 151