[dm-devel] [PATCH 6/7] dm core: disable interrupt when taking map_lock

Christof Schmitt christof.schmitt at de.ibm.com
Fri Apr 24 11:50:03 UTC 2009


On Fri, Apr 24, 2009 at 05:15:44PM +0900, Kiyoshi Ueda wrote:
> This patch disables interrupt when taking map_lock to prevent
> needless lockdep warnings in request-based dm.
> 
> request-based dm takes map_lock after taking queue_lock with
> disabling interrupt:
>   spin_lock_irqsave(queue_lock)
>   q->request_fn() == dm_request_fn()
>     => dm_get_table()
>          => read_lock(map_lock)
> while queue_lock could be taken in interrupt context.
> 
> So lockdep warns that a deadlock can happen:
>   write_lock(map_lock)
>   <interrupt>
>   spin_lock_irqsave(queue_lock)
>   q->request_fn() == dm_request_fn()
>     => dm_get_table()
>          => read_lock(map_lock)
> 
> Currently there is no such code path in request-based dm where
> q->request_fn() is called from interrupt context, so no such deadlock
> happens.
> But such warning messages confuse users, so prevent them by disabling
> interrupt when taking map_lock.
> 
> 
> Signed-off-by: Kiyoshi Ueda <k-ueda at ct.jp.nec.com>
> Signed-off-by: Jun'ichi Nomura <j-nomura at ce.jp.nec.com>
> Cc: Christof Schmitt <christof.schmitt at de.ibm.com>
> Cc: Alasdair G Kergon <agk at redhat.com>
> ---
>  drivers/md/dm.c |   15 +++++++++------
>  1 file changed, 9 insertions(+), 6 deletions(-)
> 
> Index: 2.6.30-rc3/drivers/md/dm.c
> ===================================================================
> --- 2.6.30-rc3.orig/drivers/md/dm.c
> +++ 2.6.30-rc3/drivers/md/dm.c
> @@ -510,12 +510,13 @@ static void queue_io(struct mapped_devic
>  struct dm_table *dm_get_table(struct mapped_device *md)
>  {
>  	struct dm_table *t;
> +	unsigned long flags;
> 
> -	read_lock(&md->map_lock);
> +	read_lock_irqsave(&md->map_lock, flags);
>  	t = md->map;
>  	if (t)
>  		dm_table_get(t);
> -	read_unlock(&md->map_lock);
> +	read_unlock_irqrestore(&md->map_lock, flags);
> 
>  	return t;
>  }
> @@ -1940,6 +1941,7 @@ static int __bind(struct mapped_device *
>  {
>  	struct request_queue *q = md->queue;
>  	sector_t size;
> +	unsigned long flags;
> 
>  	size = dm_table_get_size(t);
> 
> @@ -1971,10 +1973,10 @@ static int __bind(struct mapped_device *
> 
>  	__bind_mempools(md, t);
> 
> -	write_lock(&md->map_lock);
> +	write_lock_irqsave(&md->map_lock, flags);
>  	md->map = t;
>  	dm_table_set_restrictions(t, q);
> -	write_unlock(&md->map_lock);
> +	write_unlock_irqrestore(&md->map_lock, flags);
> 
>  	return 0;
>  }
> @@ -1982,14 +1984,15 @@ static int __bind(struct mapped_device *
>  static void __unbind(struct mapped_device *md)
>  {
>  	struct dm_table *map = md->map;
> +	unsigned long flags;
> 
>  	if (!map)
>  		return;
> 
>  	dm_table_event_callback(map, NULL, NULL);
> -	write_lock(&md->map_lock);
> +	write_lock_irqsave(&md->map_lock, flags);
>  	md->map = NULL;
> -	write_unlock(&md->map_lock);
> +	write_unlock_irqrestore(&md->map_lock, flags);
>  	dm_table_destroy(map);
>  }

Acked-by: Christof Schmitt <christof.schmitt at de.ibm.com>




More information about the dm-devel mailing list