Please ignore, something weird happened with my emailer.

Will repost in a minute.

Regards,

        Hans

On 01/03/2014 12:02 PM, Hans Verkuil wrote:
> From: Hans Verkuil <[email protected]>
> 
> This reverts commit a242f426108c284049a69710f871cc9f11b13e61.
> 
> That commit actually caused deadlocks, rather then fixing them.
> 
> If ext_lock is set to NULL (otherwise videobuf_queue_lock doesn't do
> anything), then you get this deadlock:
> 
> The driver's mmap function calls videobuf_mmap_mapper which calls
> videobuf_queue_lock on q. videobuf_mmap_mapper calls  __videobuf_mmap_mapper,
> __videobuf_mmap_mapper calls videobuf_vm_open and videobuf_vm_open
> calls videobuf_queue_lock on q (introduced by above patch): deadlocked.
> 
> This affects drivers using dma-contig and dma-vmalloc. Only dma-sg is
> not affected since it doesn't call videobuf_vm_open from 
> __videobuf_mmap_mapper.
> 
> Most drivers these days have a non-NULL ext_lock. Those that still use
> NULL there are all fairly obscure drivers, which is why this hasn't been
> seen earlier.
> 
> Since everything worked perfectly fine for many years I prefer to just
> revert this patch rather than trying to fix it. videobuf is quite fragile
> and I rather not touch it too much. Work is (slowly) progressing to move
> everything over to vb2 or at the very least use non-NULL ext_lock in
> videobuf.
> 
> Signed-off-by: Hans Verkuil <[email protected]>
> Cc: [email protected]      # for v3.11 and up
> Cc: Al Viro <[email protected]>
> Reported-by: Pete Eberlein <[email protected]>
> ---
>  drivers/media/v4l2-core/videobuf-dma-contig.c | 12 +++++-------
>  drivers/media/v4l2-core/videobuf-dma-sg.c     | 10 ++++------
>  drivers/media/v4l2-core/videobuf-vmalloc.c    | 10 ++++------
>  3 files changed, 13 insertions(+), 19 deletions(-)
> 
> diff --git a/drivers/media/v4l2-core/videobuf-dma-contig.c 
> b/drivers/media/v4l2-core/videobuf-dma-contig.c
> index 65411ad..7e6b209 100644
> --- a/drivers/media/v4l2-core/videobuf-dma-contig.c
> +++ b/drivers/media/v4l2-core/videobuf-dma-contig.c
> @@ -66,14 +66,11 @@ static void __videobuf_dc_free(struct device *dev,
>  static void videobuf_vm_open(struct vm_area_struct *vma)
>  {
>       struct videobuf_mapping *map = vma->vm_private_data;
> -     struct videobuf_queue *q = map->q;
>  
> -     dev_dbg(q->dev, "vm_open %p [count=%u,vma=%08lx-%08lx]\n",
> +     dev_dbg(map->q->dev, "vm_open %p [count=%u,vma=%08lx-%08lx]\n",
>               map, map->count, vma->vm_start, vma->vm_end);
>  
> -     videobuf_queue_lock(q);
>       map->count++;
> -     videobuf_queue_unlock(q);
>  }
>  
>  static void videobuf_vm_close(struct vm_area_struct *vma)
> @@ -85,11 +82,12 @@ static void videobuf_vm_close(struct vm_area_struct *vma)
>       dev_dbg(q->dev, "vm_close %p [count=%u,vma=%08lx-%08lx]\n",
>               map, map->count, vma->vm_start, vma->vm_end);
>  
> -     videobuf_queue_lock(q);
> -     if (!--map->count) {
> +     map->count--;
> +     if (0 == map->count) {
>               struct videobuf_dma_contig_memory *mem;
>  
>               dev_dbg(q->dev, "munmap %p q=%p\n", map, q);
> +             videobuf_queue_lock(q);
>  
>               /* We need first to cancel streams, before unmapping */
>               if (q->streaming)
> @@ -128,8 +126,8 @@ static void videobuf_vm_close(struct vm_area_struct *vma)
>  
>               kfree(map);
>  
> +             videobuf_queue_unlock(q);
>       }
> -     videobuf_queue_unlock(q);
>  }
>  
>  static const struct vm_operations_struct videobuf_vm_ops = {
> diff --git a/drivers/media/v4l2-core/videobuf-dma-sg.c 
> b/drivers/media/v4l2-core/videobuf-dma-sg.c
> index 9db674c..828e7c1 100644
> --- a/drivers/media/v4l2-core/videobuf-dma-sg.c
> +++ b/drivers/media/v4l2-core/videobuf-dma-sg.c
> @@ -338,14 +338,11 @@ EXPORT_SYMBOL_GPL(videobuf_dma_free);
>  static void videobuf_vm_open(struct vm_area_struct *vma)
>  {
>       struct videobuf_mapping *map = vma->vm_private_data;
> -     struct videobuf_queue *q = map->q;
>  
>       dprintk(2, "vm_open %p [count=%d,vma=%08lx-%08lx]\n", map,
>               map->count, vma->vm_start, vma->vm_end);
>  
> -     videobuf_queue_lock(q);
>       map->count++;
> -     videobuf_queue_unlock(q);
>  }
>  
>  static void videobuf_vm_close(struct vm_area_struct *vma)
> @@ -358,9 +355,10 @@ static void videobuf_vm_close(struct vm_area_struct *vma)
>       dprintk(2, "vm_close %p [count=%d,vma=%08lx-%08lx]\n", map,
>               map->count, vma->vm_start, vma->vm_end);
>  
> -     videobuf_queue_lock(q);
> -     if (!--map->count) {
> +     map->count--;
> +     if (0 == map->count) {
>               dprintk(1, "munmap %p q=%p\n", map, q);
> +             videobuf_queue_lock(q);
>               for (i = 0; i < VIDEO_MAX_FRAME; i++) {
>                       if (NULL == q->bufs[i])
>                               continue;
> @@ -376,9 +374,9 @@ static void videobuf_vm_close(struct vm_area_struct *vma)
>                       q->bufs[i]->baddr = 0;
>                       q->ops->buf_release(q, q->bufs[i]);
>               }
> +             videobuf_queue_unlock(q);
>               kfree(map);
>       }
> -     videobuf_queue_unlock(q);
>       return;
>  }
>  
> diff --git a/drivers/media/v4l2-core/videobuf-vmalloc.c 
> b/drivers/media/v4l2-core/videobuf-vmalloc.c
> index 1365c65..2ff7fcc 100644
> --- a/drivers/media/v4l2-core/videobuf-vmalloc.c
> +++ b/drivers/media/v4l2-core/videobuf-vmalloc.c
> @@ -54,14 +54,11 @@ MODULE_LICENSE("GPL");
>  static void videobuf_vm_open(struct vm_area_struct *vma)
>  {
>       struct videobuf_mapping *map = vma->vm_private_data;
> -     struct videobuf_queue *q = map->q;
>  
>       dprintk(2, "vm_open %p [count=%u,vma=%08lx-%08lx]\n", map,
>               map->count, vma->vm_start, vma->vm_end);
>  
> -     videobuf_queue_lock(q);
>       map->count++;
> -     videobuf_queue_unlock(q);
>  }
>  
>  static void videobuf_vm_close(struct vm_area_struct *vma)
> @@ -73,11 +70,12 @@ static void videobuf_vm_close(struct vm_area_struct *vma)
>       dprintk(2, "vm_close %p [count=%u,vma=%08lx-%08lx]\n", map,
>               map->count, vma->vm_start, vma->vm_end);
>  
> -     videobuf_queue_lock(q);
> -     if (!--map->count) {
> +     map->count--;
> +     if (0 == map->count) {
>               struct videobuf_vmalloc_memory *mem;
>  
>               dprintk(1, "munmap %p q=%p\n", map, q);
> +             videobuf_queue_lock(q);
>  
>               /* We need first to cancel streams, before unmapping */
>               if (q->streaming)
> @@ -116,8 +114,8 @@ static void videobuf_vm_close(struct vm_area_struct *vma)
>  
>               kfree(map);
>  
> +             videobuf_queue_unlock(q);
>       }
> -     videobuf_queue_unlock(q);
>  
>       return;
>  }
> 

--
To unsubscribe from this list: send the line "unsubscribe stable" in
the body of a message to [email protected]
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to