diff options
Diffstat (limited to 'mm/madvise.c')
-rw-r--r-- | mm/madvise.c | 19 |
1 files changed, 14 insertions, 5 deletions
diff --git a/mm/madvise.c b/mm/madvise.c index 77916e9fc52b..603c5257ed6e 100644 --- a/mm/madvise.c +++ b/mm/madvise.c | |||
@@ -159,9 +159,10 @@ static long madvise_remove(struct vm_area_struct *vma, | |||
159 | unsigned long start, unsigned long end) | 159 | unsigned long start, unsigned long end) |
160 | { | 160 | { |
161 | struct address_space *mapping; | 161 | struct address_space *mapping; |
162 | loff_t offset, endoff; | 162 | loff_t offset, endoff; |
163 | int error; | ||
163 | 164 | ||
164 | *prev = vma; | 165 | *prev = NULL; /* tell sys_madvise we drop mmap_sem */ |
165 | 166 | ||
166 | if (vma->vm_flags & (VM_LOCKED|VM_NONLINEAR|VM_HUGETLB)) | 167 | if (vma->vm_flags & (VM_LOCKED|VM_NONLINEAR|VM_HUGETLB)) |
167 | return -EINVAL; | 168 | return -EINVAL; |
@@ -180,7 +181,12 @@ static long madvise_remove(struct vm_area_struct *vma, | |||
180 | + ((loff_t)vma->vm_pgoff << PAGE_SHIFT); | 181 | + ((loff_t)vma->vm_pgoff << PAGE_SHIFT); |
181 | endoff = (loff_t)(end - vma->vm_start - 1) | 182 | endoff = (loff_t)(end - vma->vm_start - 1) |
182 | + ((loff_t)vma->vm_pgoff << PAGE_SHIFT); | 183 | + ((loff_t)vma->vm_pgoff << PAGE_SHIFT); |
183 | return vmtruncate_range(mapping->host, offset, endoff); | 184 | |
185 | /* vmtruncate_range needs to take i_mutex and i_alloc_sem */ | ||
186 | up_write(¤t->mm->mmap_sem); | ||
187 | error = vmtruncate_range(mapping->host, offset, endoff); | ||
188 | down_write(¤t->mm->mmap_sem); | ||
189 | return error; | ||
184 | } | 190 | } |
185 | 191 | ||
186 | static long | 192 | static long |
@@ -315,12 +321,15 @@ asmlinkage long sys_madvise(unsigned long start, size_t len_in, int behavior) | |||
315 | if (error) | 321 | if (error) |
316 | goto out; | 322 | goto out; |
317 | start = tmp; | 323 | start = tmp; |
318 | if (start < prev->vm_end) | 324 | if (prev && start < prev->vm_end) |
319 | start = prev->vm_end; | 325 | start = prev->vm_end; |
320 | error = unmapped_error; | 326 | error = unmapped_error; |
321 | if (start >= end) | 327 | if (start >= end) |
322 | goto out; | 328 | goto out; |
323 | vma = prev->vm_next; | 329 | if (prev) |
330 | vma = prev->vm_next; | ||
331 | else /* madvise_remove dropped mmap_sem */ | ||
332 | vma = find_vma(current->mm, start); | ||
324 | } | 333 | } |
325 | out: | 334 | out: |
326 | up_write(¤t->mm->mmap_sem); | 335 | up_write(¤t->mm->mmap_sem); |