Blame view

mm/madvise.c 11.5 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
4
5
6
7
8
9
10
  /*
   *	linux/mm/madvise.c
   *
   * Copyright (C) 1999  Linus Torvalds
   * Copyright (C) 2002  Christoph Hellwig
   */
  
  #include <linux/mman.h>
  #include <linux/pagemap.h>
  #include <linux/syscalls.h>
05b743847   Prasanna Meda   [PATCH] madvise: ...
11
  #include <linux/mempolicy.h>
afcf938ee   Andi Kleen   HWPOISON: Add a m...
12
  #include <linux/page-isolation.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
13
  #include <linux/hugetlb.h>
e8edc6e03   Alexey Dobriyan   Detach sched.h fr...
14
  #include <linux/sched.h>
f8af4da3b   Hugh Dickins   ksm: the mm inter...
15
  #include <linux/ksm.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
16
17
  
  /*
0a27a14a6   Nick Piggin   mm: madvise avoid...
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
   * Any behaviour which results in changes to the vma->vm_flags needs to
   * take mmap_sem for writing. Others, which simply traverse vmas, need
   * to only take it for reading.
   */
  static int madvise_need_mmap_write(int behavior)
  {
  	switch (behavior) {
  	case MADV_REMOVE:
  	case MADV_WILLNEED:
  	case MADV_DONTNEED:
  		return 0;
  	default:
  		/* be safe, default to 1. list exceptions explicitly */
  		return 1;
  	}
  }
  
  /*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
36
37
38
   * We can potentially split a vm area into separate
   * areas, each area with its own behavior.
   */
05b743847   Prasanna Meda   [PATCH] madvise: ...
39
40
41
  static long madvise_behavior(struct vm_area_struct * vma,
  		     struct vm_area_struct **prev,
  		     unsigned long start, unsigned long end, int behavior)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
42
43
44
  {
  	struct mm_struct * mm = vma->vm_mm;
  	int error = 0;
05b743847   Prasanna Meda   [PATCH] madvise: ...
45
  	pgoff_t pgoff;
3866ea90d   Hugh Dickins   ksm: first tidy u...
46
  	unsigned long new_flags = vma->vm_flags;
e798c6e87   Prasanna Meda   [PATCH] madvise: ...
47
48
  
  	switch (behavior) {
f82256616   Michael S. Tsirkin   [PATCH] madvise M...
49
50
51
  	case MADV_NORMAL:
  		new_flags = new_flags & ~VM_RAND_READ & ~VM_SEQ_READ;
  		break;
e798c6e87   Prasanna Meda   [PATCH] madvise: ...
52
  	case MADV_SEQUENTIAL:
f82256616   Michael S. Tsirkin   [PATCH] madvise M...
53
  		new_flags = (new_flags & ~VM_RAND_READ) | VM_SEQ_READ;
e798c6e87   Prasanna Meda   [PATCH] madvise: ...
54
55
  		break;
  	case MADV_RANDOM:
f82256616   Michael S. Tsirkin   [PATCH] madvise M...
56
  		new_flags = (new_flags & ~VM_SEQ_READ) | VM_RAND_READ;
e798c6e87   Prasanna Meda   [PATCH] madvise: ...
57
  		break;
f82256616   Michael S. Tsirkin   [PATCH] madvise M...
58
59
60
61
  	case MADV_DONTFORK:
  		new_flags |= VM_DONTCOPY;
  		break;
  	case MADV_DOFORK:
3866ea90d   Hugh Dickins   ksm: first tidy u...
62
63
64
65
  		if (vma->vm_flags & VM_IO) {
  			error = -EINVAL;
  			goto out;
  		}
f82256616   Michael S. Tsirkin   [PATCH] madvise M...
66
  		new_flags &= ~VM_DONTCOPY;
e798c6e87   Prasanna Meda   [PATCH] madvise: ...
67
  		break;
f8af4da3b   Hugh Dickins   ksm: the mm inter...
68
69
70
71
72
73
  	case MADV_MERGEABLE:
  	case MADV_UNMERGEABLE:
  		error = ksm_madvise(vma, start, end, behavior, &new_flags);
  		if (error)
  			goto out;
  		break;
0af4e98b6   Andrea Arcangeli   thp: madvise(MADV...
74
  	case MADV_HUGEPAGE:
a664b2d85   Andrea Arcangeli   thp: madvise(MADV...
75
  	case MADV_NOHUGEPAGE:
60ab3244e   Andrea Arcangeli   thp: khugepaged: ...
76
  		error = hugepage_madvise(vma, &new_flags, behavior);
0af4e98b6   Andrea Arcangeli   thp: madvise(MADV...
77
78
79
  		if (error)
  			goto out;
  		break;
e798c6e87   Prasanna Meda   [PATCH] madvise: ...
80
  	}
05b743847   Prasanna Meda   [PATCH] madvise: ...
81
82
  	if (new_flags == vma->vm_flags) {
  		*prev = vma;
836d5ffd3   Hugh Dickins   [PATCH] mm: fix m...
83
  		goto out;
05b743847   Prasanna Meda   [PATCH] madvise: ...
84
85
86
87
88
89
90
91
92
93
94
  	}
  
  	pgoff = vma->vm_pgoff + ((start - vma->vm_start) >> PAGE_SHIFT);
  	*prev = vma_merge(mm, *prev, start, end, new_flags, vma->anon_vma,
  				vma->vm_file, pgoff, vma_policy(vma));
  	if (*prev) {
  		vma = *prev;
  		goto success;
  	}
  
  	*prev = vma;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
95
96
97
98
99
100
101
102
103
104
105
106
  
  	if (start != vma->vm_start) {
  		error = split_vma(mm, vma, start, 1);
  		if (error)
  			goto out;
  	}
  
  	if (end != vma->vm_end) {
  		error = split_vma(mm, vma, end, 0);
  		if (error)
  			goto out;
  	}
836d5ffd3   Hugh Dickins   [PATCH] mm: fix m...
107
  success:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
108
109
110
  	/*
  	 * vm_flags is protected by the mmap_sem held in write mode.
  	 */
e798c6e87   Prasanna Meda   [PATCH] madvise: ...
111
  	vma->vm_flags = new_flags;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
112
113
114
115
116
117
118
119
120
121
122
  
  out:
  	if (error == -ENOMEM)
  		error = -EAGAIN;
  	return error;
  }
  
  /*
   * Schedule all required I/O operations.  Do not wait for completion.
   */
  static long madvise_willneed(struct vm_area_struct * vma,
05b743847   Prasanna Meda   [PATCH] madvise: ...
123
  			     struct vm_area_struct ** prev,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
124
125
126
  			     unsigned long start, unsigned long end)
  {
  	struct file *file = vma->vm_file;
1bef40032   Suzuki   [PATCH] madvise: ...
127
128
  	if (!file)
  		return -EBADF;
70688e4dd   Nick Piggin   xip: support non-...
129
  	if (file->f_mapping->a_ops->get_xip_mem) {
fe77ba6f4   Carsten Otte   [PATCH] xip: madv...
130
131
132
  		/* no bad return value, but ignore advice */
  		return 0;
  	}
05b743847   Prasanna Meda   [PATCH] madvise: ...
133
  	*prev = vma;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
134
135
136
137
  	start = ((start - vma->vm_start) >> PAGE_SHIFT) + vma->vm_pgoff;
  	if (end > vma->vm_end)
  		end = vma->vm_end;
  	end = ((end - vma->vm_start) >> PAGE_SHIFT) + vma->vm_pgoff;
f7e839dd3   Wu Fengguang   readahead: move m...
138
  	force_page_cache_readahead(file->f_mapping, file, start, end - start);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
139
140
141
142
143
144
145
  	return 0;
  }
  
  /*
   * Application no longer needs these pages.  If the pages are dirty,
   * it's OK to just throw them away.  The app will be more careful about
   * data it wants to keep.  Be sure to free swap resources too.  The
7e6cbea39   Fernando Luis Vazquez Cao   madvise: update f...
146
   * zap_page_range call sets things up for shrink_active_list to actually free
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
147
148
   * these pages later if no one else has touched them in the meantime,
   * although we could add these pages to a global reuse list for
7e6cbea39   Fernando Luis Vazquez Cao   madvise: update f...
149
   * shrink_active_list to pick up before reclaiming other pages.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
150
151
152
153
154
155
156
157
158
159
160
161
   *
   * NB: This interface discards data rather than pushes it out to swap,
   * as some implementations do.  This has performance implications for
   * applications like large transactional databases which want to discard
   * pages in anonymous maps after committing to backing store the data
   * that was kept in them.  There is no reason to write this data out to
   * the swap area if the application is discarding it.
   *
   * An interface that causes the system to free clean pages and flush
   * dirty pages is already available as msync(MS_INVALIDATE).
   */
  static long madvise_dontneed(struct vm_area_struct * vma,
05b743847   Prasanna Meda   [PATCH] madvise: ...
162
  			     struct vm_area_struct ** prev,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
163
164
  			     unsigned long start, unsigned long end)
  {
05b743847   Prasanna Meda   [PATCH] madvise: ...
165
  	*prev = vma;
6aab341e0   Linus Torvalds   mm: re-architect ...
166
  	if (vma->vm_flags & (VM_LOCKED|VM_HUGETLB|VM_PFNMAP))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
167
168
169
170
171
172
173
174
175
176
177
178
  		return -EINVAL;
  
  	if (unlikely(vma->vm_flags & VM_NONLINEAR)) {
  		struct zap_details details = {
  			.nonlinear_vma = vma,
  			.last_index = ULONG_MAX,
  		};
  		zap_page_range(vma, start, end - start, &details);
  	} else
  		zap_page_range(vma, start, end - start, NULL);
  	return 0;
  }
f6b3ec238   Badari Pulavarty   [PATCH] madvise(M...
179
180
181
182
183
184
185
186
  /*
   * Application wants to free up the pages and associated backing store.
   * This is effectively punching a hole into the middle of a file.
   *
   * NOTE: Currently, only shmfs/tmpfs is supported for this operation.
   * Other filesystems return -ENOSYS.
   */
  static long madvise_remove(struct vm_area_struct *vma,
00e9fa2d6   Nick Piggin   [PATCH] mm: fix m...
187
  				struct vm_area_struct **prev,
f6b3ec238   Badari Pulavarty   [PATCH] madvise(M...
188
189
190
  				unsigned long start, unsigned long end)
  {
  	struct address_space *mapping;
90ed52ebe   Hugh Dickins   [PATCH] holepunch...
191
192
  	loff_t offset, endoff;
  	int error;
f6b3ec238   Badari Pulavarty   [PATCH] madvise(M...
193

90ed52ebe   Hugh Dickins   [PATCH] holepunch...
194
  	*prev = NULL;	/* tell sys_madvise we drop mmap_sem */
00e9fa2d6   Nick Piggin   [PATCH] mm: fix m...
195

f6b3ec238   Badari Pulavarty   [PATCH] madvise(M...
196
197
198
199
200
201
202
  	if (vma->vm_flags & (VM_LOCKED|VM_NONLINEAR|VM_HUGETLB))
  		return -EINVAL;
  
  	if (!vma->vm_file || !vma->vm_file->f_mapping
  		|| !vma->vm_file->f_mapping->host) {
  			return -EINVAL;
  	}
69cf0fac6   Hugh Dickins   [PATCH] Fix MADV_...
203
204
  	if ((vma->vm_flags & (VM_SHARED|VM_WRITE)) != (VM_SHARED|VM_WRITE))
  		return -EACCES;
f6b3ec238   Badari Pulavarty   [PATCH] madvise(M...
205
206
207
208
209
210
  	mapping = vma->vm_file->f_mapping;
  
  	offset = (loff_t)(start - vma->vm_start)
  			+ ((loff_t)vma->vm_pgoff << PAGE_SHIFT);
  	endoff = (loff_t)(end - vma->vm_start - 1)
  			+ ((loff_t)vma->vm_pgoff << PAGE_SHIFT);
90ed52ebe   Hugh Dickins   [PATCH] holepunch...
211

bd5fe6c5e   Christoph Hellwig   fs: kill i_alloc_sem
212
  	/* vmtruncate_range needs to take i_mutex */
0a27a14a6   Nick Piggin   mm: madvise avoid...
213
  	up_read(&current->mm->mmap_sem);
90ed52ebe   Hugh Dickins   [PATCH] holepunch...
214
  	error = vmtruncate_range(mapping->host, offset, endoff);
0a27a14a6   Nick Piggin   mm: madvise avoid...
215
  	down_read(&current->mm->mmap_sem);
90ed52ebe   Hugh Dickins   [PATCH] holepunch...
216
  	return error;
f6b3ec238   Badari Pulavarty   [PATCH] madvise(M...
217
  }
9893e49d6   Andi Kleen   HWPOISON: Add mad...
218
219
220
221
  #ifdef CONFIG_MEMORY_FAILURE
  /*
   * Error injection support for memory error handling.
   */
afcf938ee   Andi Kleen   HWPOISON: Add a m...
222
  static int madvise_hwpoison(int bhv, unsigned long start, unsigned long end)
9893e49d6   Andi Kleen   HWPOISON: Add mad...
223
224
225
226
227
228
229
  {
  	int ret = 0;
  
  	if (!capable(CAP_SYS_ADMIN))
  		return -EPERM;
  	for (; start < end; start += PAGE_SIZE) {
  		struct page *p;
d15f107d9   Andi Kleen   HWPOISON: Use get...
230
  		int ret = get_user_pages_fast(start, 1, 0, &p);
9893e49d6   Andi Kleen   HWPOISON: Add mad...
231
232
  		if (ret != 1)
  			return ret;
afcf938ee   Andi Kleen   HWPOISON: Add a m...
233
234
235
236
237
238
239
240
241
  		if (bhv == MADV_SOFT_OFFLINE) {
  			printk(KERN_INFO "Soft offlining page %lx at %lx
  ",
  				page_to_pfn(p), start);
  			ret = soft_offline_page(p, MF_COUNT_INCREASED);
  			if (ret)
  				break;
  			continue;
  		}
9893e49d6   Andi Kleen   HWPOISON: Add mad...
242
243
244
245
  		printk(KERN_INFO "Injecting memory failure for page %lx at %lx
  ",
  		       page_to_pfn(p), start);
  		/* Ignore return value for now */
82ba011b9   Andi Kleen   HWPOISON: Turn re...
246
  		__memory_failure(page_to_pfn(p), 0, MF_COUNT_INCREASED);
9893e49d6   Andi Kleen   HWPOISON: Add mad...
247
248
249
250
  	}
  	return ret;
  }
  #endif
165cd4023   suzuki   [PATCH] madvise()...
251
252
253
  static long
  madvise_vma(struct vm_area_struct *vma, struct vm_area_struct **prev,
  		unsigned long start, unsigned long end, int behavior)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
254
  {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
255
  	switch (behavior) {
f6b3ec238   Badari Pulavarty   [PATCH] madvise(M...
256
  	case MADV_REMOVE:
3866ea90d   Hugh Dickins   ksm: first tidy u...
257
  		return madvise_remove(vma, prev, start, end);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
258
  	case MADV_WILLNEED:
3866ea90d   Hugh Dickins   ksm: first tidy u...
259
  		return madvise_willneed(vma, prev, start, end);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
260
  	case MADV_DONTNEED:
3866ea90d   Hugh Dickins   ksm: first tidy u...
261
  		return madvise_dontneed(vma, prev, start, end);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
262
  	default:
3866ea90d   Hugh Dickins   ksm: first tidy u...
263
  		return madvise_behavior(vma, prev, start, end, behavior);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
264
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
265
  }
75927af8b   Nick Piggin   mm: madvise(): co...
266
267
268
269
270
271
272
273
274
275
276
277
  static int
  madvise_behavior_valid(int behavior)
  {
  	switch (behavior) {
  	case MADV_DOFORK:
  	case MADV_DONTFORK:
  	case MADV_NORMAL:
  	case MADV_SEQUENTIAL:
  	case MADV_RANDOM:
  	case MADV_REMOVE:
  	case MADV_WILLNEED:
  	case MADV_DONTNEED:
f8af4da3b   Hugh Dickins   ksm: the mm inter...
278
279
280
281
  #ifdef CONFIG_KSM
  	case MADV_MERGEABLE:
  	case MADV_UNMERGEABLE:
  #endif
0af4e98b6   Andrea Arcangeli   thp: madvise(MADV...
282
283
  #ifdef CONFIG_TRANSPARENT_HUGEPAGE
  	case MADV_HUGEPAGE:
a664b2d85   Andrea Arcangeli   thp: madvise(MADV...
284
  	case MADV_NOHUGEPAGE:
0af4e98b6   Andrea Arcangeli   thp: madvise(MADV...
285
  #endif
75927af8b   Nick Piggin   mm: madvise(): co...
286
287
288
289
290
291
  		return 1;
  
  	default:
  		return 0;
  	}
  }
3866ea90d   Hugh Dickins   ksm: first tidy u...
292

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
  /*
   * The madvise(2) system call.
   *
   * Applications can use madvise() to advise the kernel how it should
   * handle paging I/O in this VM area.  The idea is to help the kernel
   * use appropriate read-ahead and caching techniques.  The information
   * provided is advisory only, and can be safely disregarded by the
   * kernel without affecting the correct operation of the application.
   *
   * behavior values:
   *  MADV_NORMAL - the default behavior is to read clusters.  This
   *		results in some read-ahead and read-behind.
   *  MADV_RANDOM - the system should read the minimum amount of data
   *		on any access, since it is unlikely that the appli-
   *		cation will need more than what it asks for.
   *  MADV_SEQUENTIAL - pages in the given range will probably be accessed
   *		once, so they can be aggressively read ahead, and
   *		can be freed soon after they are accessed.
   *  MADV_WILLNEED - the application is notifying the system to read
   *		some pages ahead.
   *  MADV_DONTNEED - the application is finished with the given range,
   *		so the kernel can free resources associated with it.
f6b3ec238   Badari Pulavarty   [PATCH] madvise(M...
315
316
   *  MADV_REMOVE - the application wants to free up the given range of
   *		pages and associated backing store.
3866ea90d   Hugh Dickins   ksm: first tidy u...
317
318
319
   *  MADV_DONTFORK - omit this area from child's address space when forking:
   *		typically, to avoid COWing pages pinned by get_user_pages().
   *  MADV_DOFORK - cancel MADV_DONTFORK: no longer omit this area when forking.
f8af4da3b   Hugh Dickins   ksm: the mm inter...
320
321
322
   *  MADV_MERGEABLE - the application recommends that KSM try to merge pages in
   *		this area with pages of identical content from other such areas.
   *  MADV_UNMERGEABLE- cancel MADV_MERGEABLE: no longer merge pages with others.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
323
324
325
326
327
328
329
330
331
332
333
334
   *
   * return values:
   *  zero    - success
   *  -EINVAL - start + len < 0, start is not page-aligned,
   *		"behavior" is not a valid value, or application
   *		is attempting to release locked or shared pages.
   *  -ENOMEM - addresses in the specified range are not currently
   *		mapped, or are outside the AS of the process.
   *  -EIO    - an I/O error occurred while paging in data.
   *  -EBADF  - map exists, but area maps something that isn't a file.
   *  -EAGAIN - a kernel resource was temporarily unavailable.
   */
3480b2574   Heiko Carstens   [CVE-2009-0029] S...
335
  SYSCALL_DEFINE3(madvise, unsigned long, start, size_t, len_in, int, behavior)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
336
  {
05b743847   Prasanna Meda   [PATCH] madvise: ...
337
338
  	unsigned long end, tmp;
  	struct vm_area_struct * vma, *prev;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
339
340
  	int unmapped_error = 0;
  	int error = -EINVAL;
f79777932   Jason Baron   speed up madvise_...
341
  	int write;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
342
  	size_t len;
9893e49d6   Andi Kleen   HWPOISON: Add mad...
343
  #ifdef CONFIG_MEMORY_FAILURE
afcf938ee   Andi Kleen   HWPOISON: Add a m...
344
345
  	if (behavior == MADV_HWPOISON || behavior == MADV_SOFT_OFFLINE)
  		return madvise_hwpoison(behavior, start, start+len_in);
9893e49d6   Andi Kleen   HWPOISON: Add mad...
346
  #endif
75927af8b   Nick Piggin   mm: madvise(): co...
347
348
  	if (!madvise_behavior_valid(behavior))
  		return error;
f79777932   Jason Baron   speed up madvise_...
349
350
  	write = madvise_need_mmap_write(behavior);
  	if (write)
0a27a14a6   Nick Piggin   mm: madvise avoid...
351
352
353
  		down_write(&current->mm->mmap_sem);
  	else
  		down_read(&current->mm->mmap_sem);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
  
  	if (start & ~PAGE_MASK)
  		goto out;
  	len = (len_in + ~PAGE_MASK) & PAGE_MASK;
  
  	/* Check to see whether len was rounded up from small -ve to zero */
  	if (len_in && !len)
  		goto out;
  
  	end = start + len;
  	if (end < start)
  		goto out;
  
  	error = 0;
  	if (end == start)
  		goto out;
  
  	/*
  	 * If the interval [start,end) covers some unmapped address
  	 * ranges, just ignore them, but return -ENOMEM at the end.
05b743847   Prasanna Meda   [PATCH] madvise: ...
374
  	 * - different from the way of handling in mlock etc.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
375
  	 */
05b743847   Prasanna Meda   [PATCH] madvise: ...
376
  	vma = find_vma_prev(current->mm, start, &prev);
836d5ffd3   Hugh Dickins   [PATCH] mm: fix m...
377
378
  	if (vma && start > vma->vm_start)
  		prev = vma;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
379
380
381
382
383
  	for (;;) {
  		/* Still start < end. */
  		error = -ENOMEM;
  		if (!vma)
  			goto out;
05b743847   Prasanna Meda   [PATCH] madvise: ...
384
  		/* Here start < (end|vma->vm_end). */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
385
386
387
  		if (start < vma->vm_start) {
  			unmapped_error = -ENOMEM;
  			start = vma->vm_start;
05b743847   Prasanna Meda   [PATCH] madvise: ...
388
389
  			if (start >= end)
  				goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
390
  		}
05b743847   Prasanna Meda   [PATCH] madvise: ...
391
392
393
394
  		/* Here vma->vm_start <= start < (end|vma->vm_end) */
  		tmp = vma->vm_end;
  		if (end < tmp)
  			tmp = end;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
395

05b743847   Prasanna Meda   [PATCH] madvise: ...
396
397
  		/* Here vma->vm_start <= start < tmp <= (end|vma->vm_end). */
  		error = madvise_vma(vma, &prev, start, tmp, behavior);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
398
399
  		if (error)
  			goto out;
05b743847   Prasanna Meda   [PATCH] madvise: ...
400
  		start = tmp;
90ed52ebe   Hugh Dickins   [PATCH] holepunch...
401
  		if (prev && start < prev->vm_end)
05b743847   Prasanna Meda   [PATCH] madvise: ...
402
403
404
405
  			start = prev->vm_end;
  		error = unmapped_error;
  		if (start >= end)
  			goto out;
90ed52ebe   Hugh Dickins   [PATCH] holepunch...
406
407
408
409
  		if (prev)
  			vma = prev->vm_next;
  		else	/* madvise_remove dropped mmap_sem */
  			vma = find_vma(current->mm, start);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
410
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
411
  out:
f79777932   Jason Baron   speed up madvise_...
412
  	if (write)
0a27a14a6   Nick Piggin   mm: madvise avoid...
413
414
415
  		up_write(&current->mm->mmap_sem);
  	else
  		up_read(&current->mm->mmap_sem);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
416
417
  	return error;
  }