aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorLars-Peter Clausen <lars@metafoo.de>2013-05-24 04:29:22 -0400
committerMark Brown <broonie@opensource.wolfsonmicro.com>2013-05-25 11:16:02 -0400
commit92ab1aab59c61b3e05200b9aa0e05ab770059142 (patch)
tree53c798ac04a0ffabcdca31f33e69420fe864f1e7
parent81485f5220770c381ac076573642ac44f13723af (diff)
regmap: Make regmap-mmio usable from atomic contexts
regmap-mmio uses a spinlock with spin_lock() and spin_unlock() for locking. To be able to use the regmap API from different contexts (atomic vs non-atomic), without the risk of race conditions, we need to use spin_lock_irqsave() and spin_lock_irqrestore() instead. A new field, the spinlock_flags field, is added to regmap struct to store the flags between regmap_{,un}lock_spinlock(). The spinlock_flags field itself is also protected by the spinlock. Thanks to Stephen Warren for the suggestion of this particular solution. Signed-off-by: Lars-Peter Clausen <lars@metafoo.de> Reviewed-by: Stephen Warren <swarren@nvidia.com> Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
-rw-r--r--drivers/base/regmap/internal.h1
-rw-r--r--drivers/base/regmap/regmap.c7
2 files changed, 6 insertions, 2 deletions
diff --git a/drivers/base/regmap/internal.h b/drivers/base/regmap/internal.h
index b33a4ff67adf..ae23d8391aa0 100644
--- a/drivers/base/regmap/internal.h
+++ b/drivers/base/regmap/internal.h
@@ -52,6 +52,7 @@ struct regmap_async {
52struct regmap { 52struct regmap {
53 struct mutex mutex; 53 struct mutex mutex;
54 spinlock_t spinlock; 54 spinlock_t spinlock;
55 unsigned long spinlock_flags;
55 regmap_lock lock; 56 regmap_lock lock;
56 regmap_unlock unlock; 57 regmap_unlock unlock;
57 void *lock_arg; /* This is passed to lock/unlock functions */ 58 void *lock_arg; /* This is passed to lock/unlock functions */
diff --git a/drivers/base/regmap/regmap.c b/drivers/base/regmap/regmap.c
index 307f5a1c1fe8..1a01553189b3 100644
--- a/drivers/base/regmap/regmap.c
+++ b/drivers/base/regmap/regmap.c
@@ -302,13 +302,16 @@ static void regmap_unlock_mutex(void *__map)
302static void regmap_lock_spinlock(void *__map) 302static void regmap_lock_spinlock(void *__map)
303{ 303{
304 struct regmap *map = __map; 304 struct regmap *map = __map;
305 spin_lock(&map->spinlock); 305 unsigned long flags;
306
307 spin_lock_irqsave(&map->spinlock, flags);
308 map->spinlock_flags = flags;
306} 309}
307 310
308static void regmap_unlock_spinlock(void *__map) 311static void regmap_unlock_spinlock(void *__map)
309{ 312{
310 struct regmap *map = __map; 313 struct regmap *map = __map;
311 spin_unlock(&map->spinlock); 314 spin_unlock_irqrestore(&map->spinlock, map->spinlock_flags);
312} 315}
313 316
314static void dev_get_regmap_release(struct device *dev, void *res) 317static void dev_get_regmap_release(struct device *dev, void *res)