提交 14bac5ac 编写于 作者: N Nick Piggin 提交者: 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: NNick Piggin <npiggin@suse.de>
Cc: Jared Hulbert <jaredeh@gmail.com>
Acked-by: NCarsten Otte <cotte@freenet.de>
Cc: Hugh Dickins <hugh@veritas.com>
Signed-off-by: NAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: NLinus Torvalds <torvalds@linux-foundation.org>
上级 538f8ea6
...@@ -248,15 +248,16 @@ static int xip_file_fault(struct vm_area_struct *vma, struct vm_fault *vmf) ...@@ -248,15 +248,16 @@ static int xip_file_fault(struct vm_area_struct *vma, struct vm_fault *vmf)
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% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册