Commit fa6e49a2 authored by Linus Torvalds's avatar Linus Torvalds

Handle two threads both trying to expand their stack simultaneously.

We had all the locking right, but we didn't check whether one of the
threads now no longer needed to expand, so we could incorrectly _shrink_
the stack in the other thread instead (not only causing segfaults, but
since we didn't do a proper unmap, we'd possibly leak pages too).

So re-check the need for expand after getting the lock.

Noticed by Paul Starzetz.
parent 749e1c8b
...@@ -1475,7 +1475,6 @@ static int acct_stack_growth(struct vm_area_struct * vma, unsigned long size, un ...@@ -1475,7 +1475,6 @@ static int acct_stack_growth(struct vm_area_struct * vma, unsigned long size, un
int expand_stack(struct vm_area_struct * vma, unsigned long address) int expand_stack(struct vm_area_struct * vma, unsigned long address)
{ {
int error; int error;
unsigned long size, grow;
if (!(vma->vm_flags & VM_GROWSUP)) if (!(vma->vm_flags & VM_GROWSUP))
return -EFAULT; return -EFAULT;
...@@ -1495,12 +1494,19 @@ int expand_stack(struct vm_area_struct * vma, unsigned long address) ...@@ -1495,12 +1494,19 @@ int expand_stack(struct vm_area_struct * vma, unsigned long address)
*/ */
address += 4 + PAGE_SIZE - 1; address += 4 + PAGE_SIZE - 1;
address &= PAGE_MASK; address &= PAGE_MASK;
error = 0;
/* Somebody else might have raced and expanded it already */
if (address > vma->vm_end) {
unsigned long size, grow;
size = address - vma->vm_start; size = address - vma->vm_start;
grow = (address - vma->vm_end) >> PAGE_SHIFT; grow = (address - vma->vm_end) >> PAGE_SHIFT;
error = acct_stack_growth(vma, size, grow); error = acct_stack_growth(vma, size, grow);
if (!error) if (!error)
vma->vm_end = address; vma->vm_end = address;
}
anon_vma_unlock(vma); anon_vma_unlock(vma);
return error; return error;
} }
...@@ -1528,7 +1534,6 @@ find_extend_vma(struct mm_struct *mm, unsigned long addr) ...@@ -1528,7 +1534,6 @@ find_extend_vma(struct mm_struct *mm, unsigned long addr)
int expand_stack(struct vm_area_struct *vma, unsigned long address) int expand_stack(struct vm_area_struct *vma, unsigned long address)
{ {
int error; int error;
unsigned long size, grow;
/* /*
* We must make sure the anon_vma is allocated * We must make sure the anon_vma is allocated
...@@ -1544,6 +1549,12 @@ int expand_stack(struct vm_area_struct *vma, unsigned long address) ...@@ -1544,6 +1549,12 @@ int expand_stack(struct vm_area_struct *vma, unsigned long address)
* anon_vma lock to serialize against concurrent expand_stacks. * anon_vma lock to serialize against concurrent expand_stacks.
*/ */
address &= PAGE_MASK; address &= PAGE_MASK;
error = 0;
/* Somebody else might have raced and expanded it already */
if (address < vma->vm_start) {
unsigned long size, grow;
size = vma->vm_end - address; size = vma->vm_end - address;
grow = (vma->vm_start - address) >> PAGE_SHIFT; grow = (vma->vm_start - address) >> PAGE_SHIFT;
...@@ -1552,6 +1563,7 @@ int expand_stack(struct vm_area_struct *vma, unsigned long address) ...@@ -1552,6 +1563,7 @@ int expand_stack(struct vm_area_struct *vma, unsigned long address)
vma->vm_start = address; vma->vm_start = address;
vma->vm_pgoff -= grow; vma->vm_pgoff -= grow;
} }
}
anon_vma_unlock(vma); anon_vma_unlock(vma);
return error; return error;
} }
......
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