The patch titled drivers: catch and return remap_vmalloc_range errors has been added to the -mm tree. Its filename is drivers-catch-and-return-remap_vmalloc_range-errors.patch See http://www.zip.com.au/~akpm/linux/patches/stuff/added-to-mm.txt to find out what to do about this From: Nick Piggin <npiggin@xxxxxxx> I noticed that on remap_xxx_range failure, drivers typically returned random things like -EAGAIN. Also, perfmon didn't handle insert_vm_area failures properly, which would result in bad things happening with mappings outside vmas (leaks, overwritten areas, etc). So these change userspace APIs slightly, which could be this way for a good reason. Signed-off-by: Nick Piggin <npiggin@xxxxxxx> Signed-off-by: Andrew Morton <akpm@xxxxxxxx> --- arch/ia64/kernel/perfmon.c | 32 +++++++++-------- drivers/media/video/cpia.c | 4 +- drivers/media/video/em28xx/em28xx-video.c | 26 +++++++------ drivers/media/video/et61x251/et61x251_core.c | 27 ++++++-------- drivers/media/video/meye.c | 18 ++++----- drivers/media/video/ov511.c | 8 +--- drivers/media/video/pwc/pwc-if.c | 5 -- drivers/media/video/se401.c | 20 +++++----- drivers/media/video/sn9c102/sn9c102_core.c | 27 ++++++-------- drivers/media/video/stv680.c | 20 +++++----- drivers/media/video/usbvideo/usbvideo.c | 5 -- drivers/media/video/usbvideo/vicam.c | 5 -- drivers/media/video/w9968cf.c | 6 +-- drivers/media/video/zc0301/zc0301_core.c | 27 ++++++-------- 14 files changed, 111 insertions(+), 119 deletions(-) diff -puN arch/ia64/kernel/perfmon.c~drivers-catch-and-return-remap_vmalloc_range-errors arch/ia64/kernel/perfmon.c --- devel/arch/ia64/kernel/perfmon.c~drivers-catch-and-return-remap_vmalloc_range-errors 2006-04-21 01:06:37.000000000 -0700 +++ devel-akpm/arch/ia64/kernel/perfmon.c 2006-04-21 01:06:37.000000000 -0700 @@ -2199,6 +2199,7 @@ pfm_smpl_buffer_alloc(struct task_struct struct vm_area_struct *vma = NULL; unsigned long size; void *smpl_buf; + int ret = -ENOMEM; /* @@ -2217,7 +2218,7 @@ pfm_smpl_buffer_alloc(struct task_struct * return -ENOMEM; */ if (size > task->signal->rlim[RLIMIT_MEMLOCK].rlim_cur) - return -ENOMEM; + goto out; /* * We do the easy to undo allocations first. @@ -2227,7 +2228,7 @@ pfm_smpl_buffer_alloc(struct task_struct smpl_buf = vmalloc_user(size); if (smpl_buf == NULL) { DPRINT(("Can't allocate sampling buffer\n")); - return -ENOMEM; + goto out; } DPRINT(("smpl_buf @%p\n", smpl_buf)); @@ -2267,7 +2268,6 @@ pfm_smpl_buffer_alloc(struct task_struct vma->vm_start = pfm_get_unmapped_area(NULL, 0, size, 0, MAP_PRIVATE|MAP_ANONYMOUS, 0); if (vma->vm_start == 0UL) { DPRINT(("Cannot find unmapped area for size %ld\n", size)); - up_write(&task->mm->mmap_sem); goto error; } vma->vm_end = vma->vm_start + size; @@ -2275,23 +2275,24 @@ pfm_smpl_buffer_alloc(struct task_struct DPRINT(("aligned size=%ld, hdr=%p mapped @0x%lx\n", size, ctx->ctx_smpl_hdr, vma->vm_start)); - /* can only be applied to current task, need to have the mm semaphore held when called */ - if (remap_vmalloc_range(vma, smpl_buf, 0)) { - DPRINT(("Can't remap buffer\n")); - up_write(&task->mm->mmap_sem); - goto error; - } - /* * now insert the vma in the vm list for the process, must be * done with mmap lock held */ - insert_vm_struct(mm, vma); + if ((ret = insert_vm_struct(mm, vma)) { + DPRINT(("Can't insert vma\n")); + goto error; + } + + /* can only be applied to current task, need to have the mm semaphore held when called */ + if ((ret = remap_vmalloc_range(vma, smpl_buf, 0))) { + DPRINT(("Can't remap buffer\n")); + goto error; + } mm->total_vm += size >> PAGE_SHIFT; vm_stat_account(vma->vm_mm, vma->vm_flags, vma->vm_file, vma_pages(vma)); - up_write(&task->mm->mmap_sem); /* * keep track of user level virtual address @@ -2299,14 +2300,17 @@ pfm_smpl_buffer_alloc(struct task_struct ctx->ctx_smpl_vaddr = (void *)vma->vm_start; *(unsigned long *)user_vaddr = vma->vm_start; + up_write(&task->mm->mmap_sem); + return 0; error: + up_write(&task->mm->mmap_sem); kmem_cache_free(vm_area_cachep, vma); error_kmem: vfree(smpl_buf); - - return -ENOMEM; +out: + return ret; } /* diff -puN drivers/media/video/cpia.c~drivers-catch-and-return-remap_vmalloc_range-errors drivers/media/video/cpia.c --- devel/drivers/media/video/cpia.c~drivers-catch-and-return-remap_vmalloc_range-errors 2006-04-21 01:06:37.000000000 -0700 +++ devel-akpm/drivers/media/video/cpia.c 2006-04-21 01:06:37.000000000 -0700 @@ -3734,9 +3734,9 @@ static int cpia_mmap(struct file *file, } } - if (remap_vmalloc_range(vma, cam->frame_buf, 0)) { + if ((retval = remap_vmalloc_range(vma, cam->frame_buf, 0))) { mutex_unlock(&cam->busy_lock); - return -EAGAIN; + return retval; } DBG("cpia_mmap: %ld\n", size); diff -puN drivers/media/video/em28xx/em28xx-video.c~drivers-catch-and-return-remap_vmalloc_range-errors drivers/media/video/em28xx/em28xx-video.c --- devel/drivers/media/video/em28xx/em28xx-video.c~drivers-catch-and-return-remap_vmalloc_range-errors 2006-04-21 01:06:37.000000000 -0700 +++ devel-akpm/drivers/media/video/em28xx/em28xx-video.c 2006-04-21 01:06:37.000000000 -0700 @@ -589,6 +589,7 @@ static int em28xx_v4l2_mmap(struct file { unsigned long size = vma->vm_end - vma->vm_start; u32 i; + int ret; struct em28xx *dev = filp->private_data; @@ -597,21 +598,21 @@ static int em28xx_v4l2_mmap(struct file if (dev->state & DEV_DISCONNECTED) { em28xx_videodbg("mmap: device not present\n"); - mutex_unlock(&dev->fileop_lock); - return -ENODEV; + ret = -ENODEV; + goto out; } if (dev->state & DEV_MISCONFIGURED) { em28xx_videodbg ("mmap: Device is misconfigured; close and " "open it again\n"); - mutex_unlock(&dev->fileop_lock); - return -EIO; + ret = -EIO; + goto out; } if (dev->io != IO_MMAP || !(vma->vm_flags & VM_WRITE) || size != PAGE_ALIGN(dev->frame[0].buf.length)) { - mutex_unlock(&dev->fileop_lock); - return -EINVAL; + ret = -EINVAL; + goto out; } for (i = 0; i < dev->num_frames; i++) { @@ -620,22 +621,23 @@ static int em28xx_v4l2_mmap(struct file } if (i == dev->num_frames) { em28xx_videodbg("mmap: user supplied mapping address is out of range\n"); - mutex_unlock(&dev->fileop_lock); - return -EINVAL; + ret = -EINVAL; + goto out; } - if (remap_vmalloc_range(vma, dev->frame[i].bufmem, 0)) { + if ((ret = remap_vmalloc_range(vma, dev->frame[i].bufmem, 0))) { em28xx_videodbg("mmap: remap_vmalloc_range failed\n"); - mutex_unlock(&dev->fileop_lock); - return -EAGAIN; + goto out; } vma->vm_ops = &em28xx_vm_ops; vma->vm_private_data = &dev->frame[i]; em28xx_vm_open(vma); + +out: mutex_unlock(&dev->fileop_lock); - return 0; + return ret; } /* diff -puN drivers/media/video/et61x251/et61x251_core.c~drivers-catch-and-return-remap_vmalloc_range-errors drivers/media/video/et61x251/et61x251_core.c --- devel/drivers/media/video/et61x251/et61x251_core.c~drivers-catch-and-return-remap_vmalloc_range-errors 2006-04-21 01:06:37.000000000 -0700 +++ devel-akpm/drivers/media/video/et61x251/et61x251_core.c 2006-04-21 01:06:37.000000000 -0700 @@ -1467,27 +1467,28 @@ static int et61x251_mmap(struct file* fi struct et61x251_device* cam = video_get_drvdata(video_devdata(filp)); unsigned long size = vma->vm_end - vma->vm_start; u32 i; + int ret; if (mutex_lock_interruptible(&cam->fileop_mutex)) return -ERESTARTSYS; if (cam->state & DEV_DISCONNECTED) { DBG(1, "Device not present"); - mutex_unlock(&cam->fileop_mutex); - return -ENODEV; + ret = -ENODEV; + goto out; } if (cam->state & DEV_MISCONFIGURED) { DBG(1, "The camera is misconfigured. Close and open it " "again."); - mutex_unlock(&cam->fileop_mutex); - return -EIO; + ret = -EIO; + goto out; } if (cam->io != IO_MMAP || !(vma->vm_flags & VM_WRITE) || size != PAGE_ALIGN(cam->frame[0].buf.length)) { - mutex_unlock(&cam->fileop_mutex); - return -EINVAL; + ret = -EINVAL; + goto out; } for (i = 0; i < cam->nbuffers; i++) { @@ -1495,23 +1496,21 @@ static int et61x251_mmap(struct file* fi break; } if (i == cam->nbuffers) { - mutex_unlock(&cam->fileop_mutex); - return -EINVAL; + ret = -EINVAL; + goto out; } - if (remap_vmalloc_range(vma, cam->frame[i].bufmem, 0)) { - mutex_unlock(&cam->fileop_mutex); - return -EAGAIN; - } + if ((ret = remap_vmalloc_range(vma, cam->frame[i].bufmem, 0))) + goto out; vma->vm_ops = &et61x251_vm_ops; vma->vm_private_data = &cam->frame[i]; et61x251_vm_open(vma); +out: mutex_unlock(&cam->fileop_mutex); - - return 0; + return ret; } /*****************************************************************************/ diff -puN drivers/media/video/meye.c~drivers-catch-and-return-remap_vmalloc_range-errors drivers/media/video/meye.c --- devel/drivers/media/video/meye.c~drivers-catch-and-return-remap_vmalloc_range-errors 2006-04-21 01:06:37.000000000 -0700 +++ devel-akpm/drivers/media/video/meye.c 2006-04-21 01:06:37.000000000 -0700 @@ -1663,11 +1663,12 @@ static struct vm_operations_struct meye_ static int meye_mmap(struct file *file, struct vm_area_struct *vma) { unsigned long size = vma->vm_end - vma->vm_start; + int ret; mutex_lock(&meye.lock); if (size > gbuffers * gbufsize) { /* XXX: should be size + vm_pgoff? */ - mutex_unlock(&meye.lock); - return -EINVAL; + ret = -EINVAL; + goto out; } if (!meye.grab_fbuffer) { int i; @@ -1676,24 +1677,23 @@ static int meye_mmap(struct file *file, meye.grab_fbuffer = vmalloc_32_user(gbuffers*gbufsize); if (!meye.grab_fbuffer) { printk(KERN_ERR "meye: v4l framebuffer allocation failed\n"); - mutex_unlock(&meye.lock); - return -ENOMEM; + ret = -ENOMEM; + goto out; } for (i = 0; i < gbuffers; i++) meye.vma_use_count[i] = 0; } - if (remap_vmalloc_range(vma, meye.grab_fbuffer, vma->vm_pgoff)) { - mutex_unlock(&meye.lock); - return -EAGAIN; - } + if ((ret = remap_vmalloc_range(vma, meye.grab_fbuffer, vma->vm_pgoff))) + goto out; vma->vm_ops = &meye_vm_ops; vma->vm_private_data = (void *) (offset / gbufsize); meye_vm_open(vma); +out: mutex_unlock(&meye.lock); - return 0; + return ret; } static struct file_operations meye_fops = { diff -puN drivers/media/video/ov511.c~drivers-catch-and-return-remap_vmalloc_range-errors drivers/media/video/ov511.c --- devel/drivers/media/video/ov511.c~drivers-catch-and-return-remap_vmalloc_range-errors 2006-04-21 01:06:37.000000000 -0700 +++ devel-akpm/drivers/media/video/ov511.c 2006-04-21 01:06:37.000000000 -0700 @@ -4574,6 +4574,7 @@ ov51x_v4l1_mmap(struct file *file, struc struct video_device *vdev = file->private_data; unsigned long size = vma->vm_end - vma->vm_start; struct usb_ov511 *ov = video_get_drvdata(vdev); + int ret; if (ov->dev == NULL) return -EIO; @@ -4588,13 +4589,10 @@ ov51x_v4l1_mmap(struct file *file, struc if (mutex_lock_interruptible(&ov->lock)) return -EINTR; - if (remap_vmalloc_range(vma, ov->fbuf, 0)) { - mutex_unlock(&ov->lock); - return -EAGAIN; - } + ret = remap_vmalloc_range(vma, ov->fbuf, 0); mutex_unlock(&ov->lock); - return 0; + return ret; } static struct file_operations ov511_fops = { diff -puN drivers/media/video/pwc/pwc-if.c~drivers-catch-and-return-remap_vmalloc_range-errors drivers/media/video/pwc/pwc-if.c --- devel/drivers/media/video/pwc/pwc-if.c~drivers-catch-and-return-remap_vmalloc_range-errors 2006-04-21 01:06:37.000000000 -0700 +++ devel-akpm/drivers/media/video/pwc/pwc-if.c 2006-04-21 01:06:37.000000000 -0700 @@ -1567,10 +1567,7 @@ static int pwc_video_mmap(struct file *f vma->vm_start, vma->vm_end - vma->vm_start); pdev = vdev->priv; - if (remap_vmalloc_range(vma, pdev->image_data, 0)) - return -EAGAIN; - - return 0; + return remap_vmalloc_range(vma, pdev->image_data, 0); } /***************************************************************************/ diff -puN drivers/media/video/se401.c~drivers-catch-and-return-remap_vmalloc_range-errors drivers/media/video/se401.c --- devel/drivers/media/video/se401.c~drivers-catch-and-return-remap_vmalloc_range-errors 2006-04-21 01:06:37.000000000 -0700 +++ devel-akpm/drivers/media/video/se401.c 2006-04-21 01:06:37.000000000 -0700 @@ -1110,24 +1110,24 @@ static int se401_mmap(struct file *file, struct video_device *dev = file->private_data; struct usb_se401 *se401 = (struct usb_se401 *)dev; unsigned long size = vma->vm_end-vma->vm_start; + int ret; mutex_lock(&se401->lock); if (se401->dev == NULL) { - mutex_unlock(&se401->lock); - return -EIO; + ret = -EIO; + goto out; } if (size > (((SE401_NUMFRAMES * se401->maxframesize) + PAGE_SIZE - 1) & ~(PAGE_SIZE - 1))) { - mutex_unlock(&se401->lock); - return -EINVAL; - } - if (remap_vmalloc_range(vma, se401->fbuf, 0)) { - mutex_unlock(&se401->lock); - return -EAGAIN; + ret = -EINVAL; + goto out; } - mutex_unlock(&se401->lock); - return 0; + ret = remap_vmalloc_range(vma, se401->fbuf, 0); + +out: + mutex_unlock(&se401->lock); + return ret; } static struct file_operations se401_fops = { diff -puN drivers/media/video/sn9c102/sn9c102_core.c~drivers-catch-and-return-remap_vmalloc_range-errors drivers/media/video/sn9c102/sn9c102_core.c --- devel/drivers/media/video/sn9c102/sn9c102_core.c~drivers-catch-and-return-remap_vmalloc_range-errors 2006-04-21 01:06:37.000000000 -0700 +++ devel-akpm/drivers/media/video/sn9c102/sn9c102_core.c 2006-04-21 01:06:37.000000000 -0700 @@ -1730,27 +1730,28 @@ static int sn9c102_mmap(struct file* fil struct sn9c102_device* cam = video_get_drvdata(video_devdata(filp)); unsigned long size = vma->vm_end - vma->vm_start; u32 i; + int ret; if (mutex_lock_interruptible(&cam->fileop_mutex)) return -ERESTARTSYS; if (cam->state & DEV_DISCONNECTED) { DBG(1, "Device not present"); - mutex_unlock(&cam->fileop_mutex); - return -ENODEV; + ret = -ENODEV; + goto out; } if (cam->state & DEV_MISCONFIGURED) { DBG(1, "The camera is misconfigured. Close and open it " "again."); - mutex_unlock(&cam->fileop_mutex); - return -EIO; + ret = -EIO; + goto out; } if (cam->io != IO_MMAP || !(vma->vm_flags & VM_WRITE) || size != PAGE_ALIGN(cam->frame[0].buf.length)) { - mutex_unlock(&cam->fileop_mutex); - return -EINVAL; + ret = -EINVAL; + goto out; } for (i = 0; i < cam->nbuffers; i++) { @@ -1758,23 +1759,21 @@ static int sn9c102_mmap(struct file* fil break; } if (i == cam->nbuffers) { - mutex_unlock(&cam->fileop_mutex); - return -EINVAL; + ret = -EINVAL; + goto out; } - if (remap_vmalloc_range(vma, cam->frame[i].bufmem, 0)) { - mutex_unlock(&cam->fileop_mutex); - return -EAGAIN; - } + if ((ret = remap_vmalloc_range(vma, cam->frame[i].bufmem, 0))) + goto out; vma->vm_ops = &sn9c102_vm_ops; vma->vm_private_data = &cam->frame[i]; sn9c102_vm_open(vma); +out: mutex_unlock(&cam->fileop_mutex); - - return 0; + return ret; } /*****************************************************************************/ diff -puN drivers/media/video/stv680.c~drivers-catch-and-return-remap_vmalloc_range-errors drivers/media/video/stv680.c --- devel/drivers/media/video/stv680.c~drivers-catch-and-return-remap_vmalloc_range-errors 2006-04-21 01:06:37.000000000 -0700 +++ devel-akpm/drivers/media/video/stv680.c 2006-04-21 01:06:37.000000000 -0700 @@ -1200,25 +1200,25 @@ static int stv680_mmap (struct file *fil struct video_device *dev = file->private_data; struct usb_stv *stv680 = video_get_drvdata(dev); unsigned long size = vma->vm_end-vma->vm_start; + int ret; mutex_lock(&stv680->lock); if (stv680->udev == NULL) { - mutex_unlock(&stv680->lock); - return -EIO; + ret = -EIO; + goto out; } if (size > (((STV680_NUMFRAMES * stv680->maxframesize) + PAGE_SIZE - 1) & ~(PAGE_SIZE - 1))) { - mutex_unlock(&stv680->lock); - return -EINVAL; - } - if (remap_vmalloc_range(vma, stv680->fbuf, 0)) { - mutex_unlock(&stv680->lock); - return -EAGAIN; + ret = -EINVAL; + goto out; } - mutex_unlock(&stv680->lock); - return 0; + ret = remap_vmalloc_range(vma, stv680->fbuf, 0); + +out: + mutex_unlock(&stv680->lock); + return ret; } static ssize_t stv680_read (struct file *file, char __user *buf, diff -puN drivers/media/video/usbvideo/usbvideo.c~drivers-catch-and-return-remap_vmalloc_range-errors drivers/media/video/usbvideo/usbvideo.c --- devel/drivers/media/video/usbvideo/usbvideo.c~drivers-catch-and-return-remap_vmalloc_range-errors 2006-04-21 01:06:37.000000000 -0700 +++ devel-akpm/drivers/media/video/usbvideo/usbvideo.c 2006-04-21 01:06:37.000000000 -0700 @@ -1036,10 +1036,7 @@ static int usbvideo_v4l_mmap(struct file if (size > (((USBVIDEO_NUMFRAMES * uvd->max_frame_size) + PAGE_SIZE - 1) & ~(PAGE_SIZE - 1))) return -EINVAL; - if (remap_vmalloc_range(vma, uvd->fbuf, 0)) - return -EAGAIN; - - return 0; + return remap_vmalloc_range(vma, uvd->fbuf, 0); } /* diff -puN drivers/media/video/usbvideo/vicam.c~drivers-catch-and-return-remap_vmalloc_range-errors drivers/media/video/usbvideo/vicam.c --- devel/drivers/media/video/usbvideo/vicam.c~drivers-catch-and-return-remap_vmalloc_range-errors 2006-04-21 01:06:37.000000000 -0700 +++ devel-akpm/drivers/media/video/usbvideo/vicam.c 2006-04-21 01:06:37.000000000 -0700 @@ -1002,10 +1002,7 @@ vicam_mmap(struct file *file, struct vm_ if (size > VICAM_FRAMES*VICAM_MAX_FRAME_SIZE) return -EINVAL; - if (remap_vmalloc_range(vma, cam->framebuf, 0)) - return -EAGAIN; - - return 0; + return remap_vmalloc_range(vma, cam->framebuf, 0); } #if defined(CONFIG_VIDEO_PROC_FS) diff -puN drivers/media/video/w9968cf.c~drivers-catch-and-return-remap_vmalloc_range-errors drivers/media/video/w9968cf.c --- devel/drivers/media/video/w9968cf.c~drivers-catch-and-return-remap_vmalloc_range-errors 2006-04-21 01:06:37.000000000 -0700 +++ devel-akpm/drivers/media/video/w9968cf.c 2006-04-21 01:06:37.000000000 -0700 @@ -2816,6 +2816,7 @@ static int w9968cf_mmap(struct file* fil video_get_drvdata(video_devdata(filp)); unsigned long vsize = vma->vm_end - vma->vm_start, psize = cam->nbuffers * cam->frame[0].size; + int ret; if (cam->disconnected) { DBG(2, "Device not present") @@ -2832,11 +2833,10 @@ static int w9968cf_mmap(struct file* fil if (vsize > psize - (vma->vm_pgoff << PAGE_SHIFT)) return -EINVAL; - if (remap_vmalloc_range(vma, cam->frame[0].buffer, vma->vm_pgoff)) - return -EAGAIN; + ret = remap_vmalloc_range(vma, cam->frame[0].buffer, vma->vm_pgoff); DBG(5, "mmap method successfully called") - return 0; + return ret; } diff -puN drivers/media/video/zc0301/zc0301_core.c~drivers-catch-and-return-remap_vmalloc_range-errors drivers/media/video/zc0301/zc0301_core.c --- devel/drivers/media/video/zc0301/zc0301_core.c~drivers-catch-and-return-remap_vmalloc_range-errors 2006-04-21 01:06:37.000000000 -0700 +++ devel-akpm/drivers/media/video/zc0301/zc0301_core.c 2006-04-21 01:06:37.000000000 -0700 @@ -931,27 +931,28 @@ static int zc0301_mmap(struct file* filp struct zc0301_device* cam = video_get_drvdata(video_devdata(filp)); unsigned long size = vma->vm_end - vma->vm_start; u32 i; + int ret; if (mutex_lock_interruptible(&cam->fileop_mutex)) return -ERESTARTSYS; if (cam->state & DEV_DISCONNECTED) { DBG(1, "Device not present"); - mutex_unlock(&cam->fileop_mutex); - return -ENODEV; + ret = -ENODEV; + goto out; } if (cam->state & DEV_MISCONFIGURED) { DBG(1, "The camera is misconfigured. Close and open it " "again."); - mutex_unlock(&cam->fileop_mutex); - return -EIO; + ret = -EIO; + goto out; } if (cam->io != IO_MMAP || !(vma->vm_flags & VM_WRITE) || size != PAGE_ALIGN(cam->frame[0].buf.length)) { - mutex_unlock(&cam->fileop_mutex); - return -EINVAL; + ret = -EINVAL; + goto out; } for (i = 0; i < cam->nbuffers; i++) { @@ -959,23 +960,21 @@ static int zc0301_mmap(struct file* filp break; } if (i == cam->nbuffers) { - mutex_unlock(&cam->fileop_mutex); - return -EINVAL; + ret = -EINVAL; + goto out; } - if (remap_vmalloc_range(vma, cam->frame[i].bufmem, 0)) { - mutex_unlock(&cam->fileop_mutex); - return -EAGAIN; - } + if ((ret = remap_vmalloc_range(vma, cam->frame[i].bufmem, 0))) + goto out; vma->vm_ops = &zc0301_vm_ops; vma->vm_private_data = &cam->frame[i]; zc0301_vm_open(vma); +out: mutex_unlock(&cam->fileop_mutex); - - return 0; + return ret; } /*****************************************************************************/ _ Patches currently in -mm which might be from npiggin@xxxxxxx are mm-vm_bug_on.patch mm-remap_vmalloc_range.patch mm-remove-vmalloc_to_pfn.patch mm-remove-rvmalloc.patch mm-extra-remap_vmalloc_range-check.patch drivers-leave-vm_flags-alone.patch drivers-catch-and-return-remap_vmalloc_range-errors.patch reiser4-releasepage-fix.patch - To unsubscribe from this list: send the line "unsubscribe mm-commits" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html