Commit 14bac5ac authored by Nick Piggin's avatar Nick Piggin Committed by Linus Torvalds

mm: xip/ext2 fix block allocation race

XIP can call into get_xip_mem concurrently with the same file,offset with
create=1.  This usually maps down to get_block, which expects the page
lock to prevent such a situation.  This causes ext2 to explode for one
reason or another.

Serialise those calls for the moment.  For common usages today, I suspect
get_xip_mem rarely is called to create new blocks.  In future as XIP
technologies evolve we might need to look at which operations require
scalability, and rework the locking to suit.
Signed-off-by: default avatarNick Piggin <npiggin@suse.de>
Cc: Jared Hulbert <jaredeh@gmail.com>
Acked-by: default avatarCarsten Otte <cotte@freenet.de>
Cc: Hugh Dickins <hugh@veritas.com>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent 538f8ea6
...@@ -248,15 +248,16 @@ again: ...@@ -248,15 +248,16 @@ again:
int err; int err;
/* maybe shared writable, allocate new block */ /* maybe shared writable, allocate new block */
mutex_lock(&xip_sparse_mutex);
error = mapping->a_ops->get_xip_mem(mapping, vmf->pgoff, 1, error = mapping->a_ops->get_xip_mem(mapping, vmf->pgoff, 1,
&xip_mem, &xip_pfn); &xip_mem, &xip_pfn);
mutex_unlock(&xip_sparse_mutex);
if (error) if (error)
return VM_FAULT_SIGBUS; return VM_FAULT_SIGBUS;
/* unmap sparse mappings at pgoff from all other vmas */ /* unmap sparse mappings at pgoff from all other vmas */
__xip_unmap(mapping, vmf->pgoff); __xip_unmap(mapping, vmf->pgoff);
found: found:
printk("%s insert %lx@%lx\n", current->comm, (unsigned long)vmf->virtual_address, xip_pfn);
err = vm_insert_mixed(vma, (unsigned long)vmf->virtual_address, err = vm_insert_mixed(vma, (unsigned long)vmf->virtual_address,
xip_pfn); xip_pfn);
if (err == -ENOMEM) if (err == -ENOMEM)
...@@ -340,8 +341,10 @@ __xip_file_write(struct file *filp, const char __user *buf, ...@@ -340,8 +341,10 @@ __xip_file_write(struct file *filp, const char __user *buf,
&xip_mem, &xip_pfn); &xip_mem, &xip_pfn);
if (status == -ENODATA) { if (status == -ENODATA) {
/* we allocate a new page unmap it */ /* we allocate a new page unmap it */
mutex_lock(&xip_sparse_mutex);
status = a_ops->get_xip_mem(mapping, index, 1, status = a_ops->get_xip_mem(mapping, index, 1,
&xip_mem, &xip_pfn); &xip_mem, &xip_pfn);
mutex_unlock(&xip_sparse_mutex);
if (!status) if (!status)
/* unmap page at pgoff from all other vmas */ /* unmap page at pgoff from all other vmas */
__xip_unmap(mapping, index); __xip_unmap(mapping, index);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment