Blame view

mm/rmap.c 45.4 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
  /*
   * mm/rmap.c - physical to virtual reverse mappings
   *
   * Copyright 2001, Rik van Riel <riel@conectiva.com.br>
   * Released under the General Public License (GPL).
   *
   * Simple, low overhead reverse mapping scheme.
   * Please try to keep this thing as modular as possible.
   *
   * Provides methods for unmapping each kind of mapped page:
   * the anon methods track anonymous pages, and
   * the file methods track pages belonging to an inode.
   *
   * Original design by Rik van Riel <riel@conectiva.com.br> 2001
   * File methods by Dave McCracken <dmccr@us.ibm.com> 2003, 2004
   * Anonymous methods by Andrea Arcangeli <andrea@suse.de> 2004
98f32602d   Hugh Dickins   hugh: update emai...
17
   * Contributions by Hugh Dickins 2003, 2004
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
18
19
20
21
22
   */
  
  /*
   * Lock ordering in mm:
   *
1b1dcc1b5   Jes Sorensen   [PATCH] mutex sub...
23
   * inode->i_mutex	(while writing or truncating, not reading or faulting)
82591e6ea   Nick Piggin   [PATCH] mm: more ...
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
   *   inode->i_alloc_sem (vmtruncate_range)
   *   mm->mmap_sem
   *     page->flags PG_locked (lock_page)
   *       mapping->i_mmap_lock
   *         anon_vma->lock
   *           mm->page_table_lock or pte_lock
   *             zone->lru_lock (in mark_page_accessed, isolate_lru_page)
   *             swap_lock (in swap_duplicate, swap_info_get)
   *               mmlist_lock (in mmput, drain_mmlist and others)
   *               mapping->private_lock (in __set_page_dirty_buffers)
   *               inode_lock (in set_page_dirty's __mark_inode_dirty)
   *                 sb_lock (within inode_lock in fs/fs-writeback.c)
   *                 mapping->tree_lock (widely used, in set_page_dirty,
   *                           in arch-dependent flush_dcache_mmap_lock,
   *                           within inode_lock in __sync_single_inode)
6a46079cf   Andi Kleen   HWPOISON: The hig...
39
40
41
42
43
   *
   * (code doesn't rely on that order so it could be switched around)
   * ->tasklist_lock
   *   anon_vma->lock      (memory_failure, collect_procs_anon)
   *     pte map lock
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
44
45
46
47
48
49
50
51
   */
  
  #include <linux/mm.h>
  #include <linux/pagemap.h>
  #include <linux/swap.h>
  #include <linux/swapops.h>
  #include <linux/slab.h>
  #include <linux/init.h>
5ad646880   Hugh Dickins   ksm: let shared p...
52
  #include <linux/ksm.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
53
54
  #include <linux/rmap.h>
  #include <linux/rcupdate.h>
a48d07afd   Christoph Lameter   [PATCH] Direct Mi...
55
  #include <linux/module.h>
8a9f3ccd2   Balbir Singh   Memory controller...
56
  #include <linux/memcontrol.h>
cddb8a5c1   Andrea Arcangeli   mmu-notifiers: core
57
  #include <linux/mmu_notifier.h>
64cdd548f   KOSAKI Motohiro   mm: cleanup: remo...
58
  #include <linux/migrate.h>
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
59
  #include <linux/hugetlb.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
60
61
  
  #include <asm/tlbflush.h>
b291f0003   Nick Piggin   mlock: mlocked pa...
62
  #include "internal.h"
fdd2e5f88   Adrian Bunk   make mm/rmap.c:an...
63
  static struct kmem_cache *anon_vma_cachep;
5beb49305   Rik van Riel   mm: change anon_v...
64
  static struct kmem_cache *anon_vma_chain_cachep;
fdd2e5f88   Adrian Bunk   make mm/rmap.c:an...
65
66
67
68
69
  
  static inline struct anon_vma *anon_vma_alloc(void)
  {
  	return kmem_cache_alloc(anon_vma_cachep, GFP_KERNEL);
  }
db114b83a   Hugh Dickins   ksm: hold anon_vm...
70
  void anon_vma_free(struct anon_vma *anon_vma)
fdd2e5f88   Adrian Bunk   make mm/rmap.c:an...
71
72
73
  {
  	kmem_cache_free(anon_vma_cachep, anon_vma);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
74

5beb49305   Rik van Riel   mm: change anon_v...
75
76
77
78
  static inline struct anon_vma_chain *anon_vma_chain_alloc(void)
  {
  	return kmem_cache_alloc(anon_vma_chain_cachep, GFP_KERNEL);
  }
e574b5fd2   Namhyung Kim   rmap: make anon_v...
79
  static void anon_vma_chain_free(struct anon_vma_chain *anon_vma_chain)
5beb49305   Rik van Riel   mm: change anon_v...
80
81
82
  {
  	kmem_cache_free(anon_vma_chain_cachep, anon_vma_chain);
  }
d9d332e08   Linus Torvalds   anon_vma_prepare:...
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
  /**
   * anon_vma_prepare - attach an anon_vma to a memory region
   * @vma: the memory region in question
   *
   * This makes sure the memory mapping described by 'vma' has
   * an 'anon_vma' attached to it, so that we can associate the
   * anonymous pages mapped into it with that anon_vma.
   *
   * The common case will be that we already have one, but if
   * if not we either need to find an adjacent mapping that we
   * can re-use the anon_vma from (very common when the only
   * reason for splitting a vma has been mprotect()), or we
   * allocate a new one.
   *
   * Anon-vma allocations are very subtle, because we may have
   * optimistically looked up an anon_vma in page_lock_anon_vma()
   * and that may actually touch the spinlock even in the newly
   * allocated vma (it depends on RCU to make sure that the
   * anon_vma isn't actually destroyed).
   *
   * As a result, we need to do proper anon_vma locking even
   * for the new allocation. At the same time, we do not want
   * to do any locking for the common case of already having
   * an anon_vma.
   *
   * This must be called with the mmap_sem held for reading.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
110
111
112
  int anon_vma_prepare(struct vm_area_struct *vma)
  {
  	struct anon_vma *anon_vma = vma->anon_vma;
5beb49305   Rik van Riel   mm: change anon_v...
113
  	struct anon_vma_chain *avc;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
114
115
116
117
  
  	might_sleep();
  	if (unlikely(!anon_vma)) {
  		struct mm_struct *mm = vma->vm_mm;
d9d332e08   Linus Torvalds   anon_vma_prepare:...
118
  		struct anon_vma *allocated;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
119

5beb49305   Rik van Riel   mm: change anon_v...
120
121
122
  		avc = anon_vma_chain_alloc();
  		if (!avc)
  			goto out_enomem;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
123
  		anon_vma = find_mergeable_anon_vma(vma);
d9d332e08   Linus Torvalds   anon_vma_prepare:...
124
125
  		allocated = NULL;
  		if (!anon_vma) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
126
127
  			anon_vma = anon_vma_alloc();
  			if (unlikely(!anon_vma))
5beb49305   Rik van Riel   mm: change anon_v...
128
  				goto out_enomem_free_avc;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
129
  			allocated = anon_vma;
5c341ee1d   Rik van Riel   mm: track the roo...
130
131
132
133
134
  			/*
  			 * This VMA had no anon_vma yet.  This anon_vma is
  			 * the root of any anon_vma tree that might form.
  			 */
  			anon_vma->root = anon_vma;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
135
  		}
cba48b98f   Rik van Riel   mm: change direct...
136
  		anon_vma_lock(anon_vma);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
137
138
139
140
  		/* page_table_lock to protect against threads */
  		spin_lock(&mm->page_table_lock);
  		if (likely(!vma->anon_vma)) {
  			vma->anon_vma = anon_vma;
5beb49305   Rik van Riel   mm: change anon_v...
141
142
143
  			avc->anon_vma = anon_vma;
  			avc->vma = vma;
  			list_add(&avc->same_vma, &vma->anon_vma_chain);
26ba0cb63   Andrea Arcangeli   rmap: always add ...
144
  			list_add_tail(&avc->same_anon_vma, &anon_vma->head);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
145
  			allocated = NULL;
31f2b0ebc   Oleg Nesterov   rmap: anon_vma_pr...
146
  			avc = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
147
148
  		}
  		spin_unlock(&mm->page_table_lock);
cba48b98f   Rik van Riel   mm: change direct...
149
  		anon_vma_unlock(anon_vma);
31f2b0ebc   Oleg Nesterov   rmap: anon_vma_pr...
150
151
  
  		if (unlikely(allocated))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
152
  			anon_vma_free(allocated);
31f2b0ebc   Oleg Nesterov   rmap: anon_vma_pr...
153
  		if (unlikely(avc))
5beb49305   Rik van Riel   mm: change anon_v...
154
  			anon_vma_chain_free(avc);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
155
156
  	}
  	return 0;
5beb49305   Rik van Riel   mm: change anon_v...
157
158
159
160
161
  
   out_enomem_free_avc:
  	anon_vma_chain_free(avc);
   out_enomem:
  	return -ENOMEM;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
162
  }
5beb49305   Rik van Riel   mm: change anon_v...
163
164
165
  static void anon_vma_chain_link(struct vm_area_struct *vma,
  				struct anon_vma_chain *avc,
  				struct anon_vma *anon_vma)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
166
  {
5beb49305   Rik van Riel   mm: change anon_v...
167
168
169
  	avc->vma = vma;
  	avc->anon_vma = anon_vma;
  	list_add(&avc->same_vma, &vma->anon_vma_chain);
cba48b98f   Rik van Riel   mm: change direct...
170
  	anon_vma_lock(anon_vma);
5beb49305   Rik van Riel   mm: change anon_v...
171
  	list_add_tail(&avc->same_anon_vma, &anon_vma->head);
cba48b98f   Rik van Riel   mm: change direct...
172
  	anon_vma_unlock(anon_vma);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
173
  }
5beb49305   Rik van Riel   mm: change anon_v...
174
175
176
177
178
  /*
   * Attach the anon_vmas from src to dst.
   * Returns 0 on success, -ENOMEM on failure.
   */
  int anon_vma_clone(struct vm_area_struct *dst, struct vm_area_struct *src)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
179
  {
5beb49305   Rik van Riel   mm: change anon_v...
180
  	struct anon_vma_chain *avc, *pavc;
646d87b48   Linus Torvalds   anon_vma: clone t...
181
  	list_for_each_entry_reverse(pavc, &src->anon_vma_chain, same_vma) {
5beb49305   Rik van Riel   mm: change anon_v...
182
183
184
185
186
187
  		avc = anon_vma_chain_alloc();
  		if (!avc)
  			goto enomem_failure;
  		anon_vma_chain_link(dst, avc, pavc->anon_vma);
  	}
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
188

5beb49305   Rik van Riel   mm: change anon_v...
189
190
191
   enomem_failure:
  	unlink_anon_vmas(dst);
  	return -ENOMEM;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
192
  }
5beb49305   Rik van Riel   mm: change anon_v...
193
194
195
196
197
198
  /*
   * Attach vma to its own anon_vma, as well as to the anon_vmas that
   * the corresponding VMA in the parent process is attached to.
   * Returns 0 on success, non-zero on failure.
   */
  int anon_vma_fork(struct vm_area_struct *vma, struct vm_area_struct *pvma)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
199
  {
5beb49305   Rik van Riel   mm: change anon_v...
200
201
  	struct anon_vma_chain *avc;
  	struct anon_vma *anon_vma;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
202

5beb49305   Rik van Riel   mm: change anon_v...
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
  	/* Don't bother if the parent process has no anon_vma here. */
  	if (!pvma->anon_vma)
  		return 0;
  
  	/*
  	 * First, attach the new VMA to the parent VMA's anon_vmas,
  	 * so rmap can find non-COWed pages in child processes.
  	 */
  	if (anon_vma_clone(vma, pvma))
  		return -ENOMEM;
  
  	/* Then add our own anon_vma. */
  	anon_vma = anon_vma_alloc();
  	if (!anon_vma)
  		goto out_error;
  	avc = anon_vma_chain_alloc();
  	if (!avc)
  		goto out_error_free_anon_vma;
5c341ee1d   Rik van Riel   mm: track the roo...
221
222
223
224
225
226
  
  	/*
  	 * The root anon_vma's spinlock is the lock actually used when we
  	 * lock any of the anon_vmas in this anon_vma tree.
  	 */
  	anon_vma->root = pvma->anon_vma->root;
76545066c   Rik van Riel   mm: extend KSM re...
227
228
229
230
231
232
  	/*
  	 * With KSM refcounts, an anon_vma can stay around longer than the
  	 * process it belongs to.  The root anon_vma needs to be pinned
  	 * until this anon_vma is freed, because the lock lives in the root.
  	 */
  	get_anon_vma(anon_vma->root);
5beb49305   Rik van Riel   mm: change anon_v...
233
234
  	/* Mark this anon_vma as the one where our new (COWed) pages go. */
  	vma->anon_vma = anon_vma;
5c341ee1d   Rik van Riel   mm: track the roo...
235
  	anon_vma_chain_link(vma, avc, anon_vma);
5beb49305   Rik van Riel   mm: change anon_v...
236
237
238
239
240
241
  
  	return 0;
  
   out_error_free_anon_vma:
  	anon_vma_free(anon_vma);
   out_error:
4946d54cb   Rik van Riel   rmap: fix anon_vm...
242
  	unlink_anon_vmas(vma);
5beb49305   Rik van Riel   mm: change anon_v...
243
  	return -ENOMEM;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
244
  }
5beb49305   Rik van Riel   mm: change anon_v...
245
  static void anon_vma_unlink(struct anon_vma_chain *anon_vma_chain)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
246
  {
5beb49305   Rik van Riel   mm: change anon_v...
247
  	struct anon_vma *anon_vma = anon_vma_chain->anon_vma;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
248
  	int empty;
5beb49305   Rik van Riel   mm: change anon_v...
249
  	/* If anon_vma_fork fails, we can get an empty anon_vma_chain. */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
250
251
  	if (!anon_vma)
  		return;
cba48b98f   Rik van Riel   mm: change direct...
252
  	anon_vma_lock(anon_vma);
5beb49305   Rik van Riel   mm: change anon_v...
253
  	list_del(&anon_vma_chain->same_anon_vma);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
254
255
  
  	/* We must garbage collect the anon_vma if it's empty */
7f60c214f   Mel Gorman   mm: migration: sh...
256
  	empty = list_empty(&anon_vma->head) && !anonvma_external_refcount(anon_vma);
cba48b98f   Rik van Riel   mm: change direct...
257
  	anon_vma_unlock(anon_vma);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
258

76545066c   Rik van Riel   mm: extend KSM re...
259
260
261
262
  	if (empty) {
  		/* We no longer need the root anon_vma */
  		if (anon_vma->root != anon_vma)
  			drop_anon_vma(anon_vma->root);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
263
  		anon_vma_free(anon_vma);
76545066c   Rik van Riel   mm: extend KSM re...
264
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
265
  }
5beb49305   Rik van Riel   mm: change anon_v...
266
267
268
  void unlink_anon_vmas(struct vm_area_struct *vma)
  {
  	struct anon_vma_chain *avc, *next;
5c341ee1d   Rik van Riel   mm: track the roo...
269
270
271
272
  	/*
  	 * Unlink each anon_vma chained to the VMA.  This list is ordered
  	 * from newest to oldest, ensuring the root anon_vma gets freed last.
  	 */
5beb49305   Rik van Riel   mm: change anon_v...
273
274
275
276
277
278
  	list_for_each_entry_safe(avc, next, &vma->anon_vma_chain, same_vma) {
  		anon_vma_unlink(avc);
  		list_del(&avc->same_vma);
  		anon_vma_chain_free(avc);
  	}
  }
51cc50685   Alexey Dobriyan   SL*B: drop kmem c...
279
  static void anon_vma_ctor(void *data)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
280
  {
a35afb830   Christoph Lameter   Remove SLAB_CTOR_...
281
  	struct anon_vma *anon_vma = data;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
282

a35afb830   Christoph Lameter   Remove SLAB_CTOR_...
283
  	spin_lock_init(&anon_vma->lock);
7f60c214f   Mel Gorman   mm: migration: sh...
284
  	anonvma_external_refcount_init(anon_vma);
a35afb830   Christoph Lameter   Remove SLAB_CTOR_...
285
  	INIT_LIST_HEAD(&anon_vma->head);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
286
287
288
289
290
  }
  
  void __init anon_vma_init(void)
  {
  	anon_vma_cachep = kmem_cache_create("anon_vma", sizeof(struct anon_vma),
20c2df83d   Paul Mundt   mm: Remove slab d...
291
  			0, SLAB_DESTROY_BY_RCU|SLAB_PANIC, anon_vma_ctor);
5beb49305   Rik van Riel   mm: change anon_v...
292
  	anon_vma_chain_cachep = KMEM_CACHE(anon_vma_chain, SLAB_PANIC);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
293
294
295
296
297
298
  }
  
  /*
   * Getting a lock on a stable anon_vma from a page off the LRU is
   * tricky: page_lock_anon_vma rely on RCU to guard against the races.
   */
ea4525b60   Namhyung Kim   rmap: annotate lo...
299
  struct anon_vma *__page_lock_anon_vma(struct page *page)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
300
  {
f18194275   Hugh Dickins   mm: fix hang on a...
301
  	struct anon_vma *anon_vma, *root_anon_vma;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
302
303
304
  	unsigned long anon_mapping;
  
  	rcu_read_lock();
80e148226   Hugh Dickins   ksm: share anon p...
305
  	anon_mapping = (unsigned long) ACCESS_ONCE(page->mapping);
3ca7b3c5b   Hugh Dickins   mm: define PAGE_M...
306
  	if ((anon_mapping & PAGE_MAPPING_FLAGS) != PAGE_MAPPING_ANON)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
307
308
309
310
311
  		goto out;
  	if (!page_mapped(page))
  		goto out;
  
  	anon_vma = (struct anon_vma *) (anon_mapping - PAGE_MAPPING_ANON);
f18194275   Hugh Dickins   mm: fix hang on a...
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
  	root_anon_vma = ACCESS_ONCE(anon_vma->root);
  	spin_lock(&root_anon_vma->lock);
  
  	/*
  	 * If this page is still mapped, then its anon_vma cannot have been
  	 * freed.  But if it has been unmapped, we have no security against
  	 * the anon_vma structure being freed and reused (for another anon_vma:
  	 * SLAB_DESTROY_BY_RCU guarantees that - so the spin_lock above cannot
  	 * corrupt): with anon_vma_prepare() or anon_vma_fork() redirecting
  	 * anon_vma->root before page_unlock_anon_vma() is called to unlock.
  	 */
  	if (page_mapped(page))
  		return anon_vma;
  
  	spin_unlock(&root_anon_vma->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
327
328
  out:
  	rcu_read_unlock();
34bbd7040   Oleg Nesterov   [PATCH] adapt pag...
329
330
  	return NULL;
  }
10be22dfe   Andi Kleen   HWPOISON: Export ...
331
  void page_unlock_anon_vma(struct anon_vma *anon_vma)
ea4525b60   Namhyung Kim   rmap: annotate lo...
332
333
  	__releases(&anon_vma->root->lock)
  	__releases(RCU)
34bbd7040   Oleg Nesterov   [PATCH] adapt pag...
334
  {
cba48b98f   Rik van Riel   mm: change direct...
335
  	anon_vma_unlock(anon_vma);
34bbd7040   Oleg Nesterov   [PATCH] adapt pag...
336
  	rcu_read_unlock();
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
337
338
339
  }
  
  /*
3ad33b243   Lee Schermerhorn   Migration: find c...
340
341
342
   * At what user virtual address is page expected in @vma?
   * Returns virtual address or -EFAULT if page's index/offset is not
   * within the range mapped the @vma.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
343
344
345
346
347
348
   */
  static inline unsigned long
  vma_address(struct page *page, struct vm_area_struct *vma)
  {
  	pgoff_t pgoff = page->index << (PAGE_CACHE_SHIFT - PAGE_SHIFT);
  	unsigned long address;
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
349
350
  	if (unlikely(is_vm_hugetlb_page(vma)))
  		pgoff = page->index << huge_page_order(page_hstate(page));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
351
352
  	address = vma->vm_start + ((pgoff - vma->vm_pgoff) << PAGE_SHIFT);
  	if (unlikely(address < vma->vm_start || address >= vma->vm_end)) {
3ad33b243   Lee Schermerhorn   Migration: find c...
353
  		/* page should be within @vma mapping range */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
354
355
356
357
358
359
  		return -EFAULT;
  	}
  	return address;
  }
  
  /*
bf89c8c86   Huang Shijie   mm/rmap.c: fix co...
360
   * At what user virtual address is page expected in vma?
ab941e0ff   Naoya Horiguchi   rmap: remove anon...
361
   * Caller should check the page is actually part of the vma.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
362
363
364
   */
  unsigned long page_address_in_vma(struct page *page, struct vm_area_struct *vma)
  {
21d0d443c   Andrea Arcangeli   rmap: resurrect p...
365
  	if (PageAnon(page)) {
4829b906c   Hugh Dickins   ksm: fix page_add...
366
367
368
369
370
371
372
  		struct anon_vma *page__anon_vma = page_anon_vma(page);
  		/*
  		 * Note: swapoff's unuse_vma() is more efficient with this
  		 * check, and needs it to match anon_vma when KSM is active.
  		 */
  		if (!vma->anon_vma || !page__anon_vma ||
  		    vma->anon_vma->root != page__anon_vma->root)
21d0d443c   Andrea Arcangeli   rmap: resurrect p...
373
374
  			return -EFAULT;
  	} else if (page->mapping && !(vma->vm_flags & VM_NONLINEAR)) {
ee498ed73   Hugh Dickins   [PATCH] unpaged: ...
375
376
  		if (!vma->vm_file ||
  		    vma->vm_file->f_mapping != page->mapping)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
377
378
379
380
381
382
383
  			return -EFAULT;
  	} else
  		return -EFAULT;
  	return vma_address(page, vma);
  }
  
  /*
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
384
385
   * Check that @page is mapped at @address into @mm.
   *
479db0bf4   Nick Piggin   mm: dirty page tr...
386
387
388
389
   * If @sync is false, page_check_address may perform a racy check to avoid
   * the page table lock when the pte is not present (helpful when reclaiming
   * highly shared pages).
   *
b8072f099   Hugh Dickins   [PATCH] mm: updat...
390
   * On success returns with pte mapped and locked.
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
391
   */
e9a81a821   Namhyung Kim   rmap: wrap page_c...
392
  pte_t *__page_check_address(struct page *page, struct mm_struct *mm,
479db0bf4   Nick Piggin   mm: dirty page tr...
393
  			  unsigned long address, spinlock_t **ptlp, int sync)
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
394
395
396
397
398
  {
  	pgd_t *pgd;
  	pud_t *pud;
  	pmd_t *pmd;
  	pte_t *pte;
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
399
  	spinlock_t *ptl;
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
400

0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
401
402
403
404
405
  	if (unlikely(PageHuge(page))) {
  		pte = huge_pte_offset(mm, address);
  		ptl = &mm->page_table_lock;
  		goto check;
  	}
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
406
  	pgd = pgd_offset(mm, address);
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
407
408
409
410
411
412
413
414
415
416
417
418
419
  	if (!pgd_present(*pgd))
  		return NULL;
  
  	pud = pud_offset(pgd, address);
  	if (!pud_present(*pud))
  		return NULL;
  
  	pmd = pmd_offset(pud, address);
  	if (!pmd_present(*pmd))
  		return NULL;
  
  	pte = pte_offset_map(pmd, address);
  	/* Make a quick check before getting the lock */
479db0bf4   Nick Piggin   mm: dirty page tr...
420
  	if (!sync && !pte_present(*pte)) {
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
421
422
423
  		pte_unmap(pte);
  		return NULL;
  	}
4c21e2f24   Hugh Dickins   [PATCH] mm: split...
424
  	ptl = pte_lockptr(mm, pmd);
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
425
  check:
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
426
427
428
429
  	spin_lock(ptl);
  	if (pte_present(*pte) && page_to_pfn(page) == pte_pfn(*pte)) {
  		*ptlp = ptl;
  		return pte;
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
430
  	}
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
431
432
  	pte_unmap_unlock(pte, ptl);
  	return NULL;
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
433
  }
b291f0003   Nick Piggin   mlock: mlocked pa...
434
435
436
437
438
439
440
441
442
  /**
   * page_mapped_in_vma - check whether a page is really mapped in a VMA
   * @page: the page to test
   * @vma: the VMA to test
   *
   * Returns 1 if the page is mapped into the page tables of the VMA, 0
   * if the page is not mapped into the page tables of this VMA.  Only
   * valid for normal file or anonymous VMAs.
   */
6a46079cf   Andi Kleen   HWPOISON: The hig...
443
  int page_mapped_in_vma(struct page *page, struct vm_area_struct *vma)
b291f0003   Nick Piggin   mlock: mlocked pa...
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
  {
  	unsigned long address;
  	pte_t *pte;
  	spinlock_t *ptl;
  
  	address = vma_address(page, vma);
  	if (address == -EFAULT)		/* out of vma range */
  		return 0;
  	pte = page_check_address(page, vma->vm_mm, address, &ptl, 1);
  	if (!pte)			/* the page is not in this mm */
  		return 0;
  	pte_unmap_unlock(pte, ptl);
  
  	return 1;
  }
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
459
  /*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
460
461
462
   * Subfunctions of page_referenced: page_referenced_one called
   * repeatedly from either page_referenced_anon or page_referenced_file.
   */
5ad646880   Hugh Dickins   ksm: let shared p...
463
464
465
  int page_referenced_one(struct page *page, struct vm_area_struct *vma,
  			unsigned long address, unsigned int *mapcount,
  			unsigned long *vm_flags)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
466
467
  {
  	struct mm_struct *mm = vma->vm_mm;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
468
  	pte_t *pte;
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
469
  	spinlock_t *ptl;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
470
  	int referenced = 0;
479db0bf4   Nick Piggin   mm: dirty page tr...
471
  	pte = page_check_address(page, mm, address, &ptl, 0);
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
472
473
  	if (!pte)
  		goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
474

b291f0003   Nick Piggin   mlock: mlocked pa...
475
476
477
478
479
  	/*
  	 * Don't want to elevate referenced for mlocked page that gets this far,
  	 * in order that it progresses to try_to_unmap and is moved to the
  	 * unevictable list.
  	 */
5a9bbdcd2   Hugh Dickins   mm: don't waste s...
480
  	if (vma->vm_flags & VM_LOCKED) {
5a9bbdcd2   Hugh Dickins   mm: don't waste s...
481
  		*mapcount = 1;	/* break early from loop */
03ef83af5   Minchan Kim   mm: fix for infin...
482
  		*vm_flags |= VM_LOCKED;
b291f0003   Nick Piggin   mlock: mlocked pa...
483
484
  		goto out_unmap;
  	}
4917e5d04   Johannes Weiner   mm: more likely r...
485
486
487
488
489
490
491
492
493
494
495
  	if (ptep_clear_flush_young_notify(vma, address, pte)) {
  		/*
  		 * Don't treat a reference through a sequentially read
  		 * mapping as such.  If the page has been used in
  		 * another mapping, we will catch it; if this other
  		 * mapping is already gone, the unmap path will have
  		 * set PG_referenced or activated the page.
  		 */
  		if (likely(!VM_SequentialReadHint(vma)))
  			referenced++;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
496

c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
497
498
  	/* Pretend the page is referenced if the task has the
  	   swap token and is in the middle of a page fault. */
f7b7fd8f3   Rik van Riel   [PATCH] temporari...
499
  	if (mm != current->mm && has_swap_token(mm) &&
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
500
501
  			rwsem_is_locked(&mm->mmap_sem))
  		referenced++;
b291f0003   Nick Piggin   mlock: mlocked pa...
502
  out_unmap:
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
503
504
  	(*mapcount)--;
  	pte_unmap_unlock(pte, ptl);
273f047e3   Huang Shijie   rmap: move label ...
505

6fe6b7e35   Wu Fengguang   vmscan: report vm...
506
507
  	if (referenced)
  		*vm_flags |= vma->vm_flags;
273f047e3   Huang Shijie   rmap: move label ...
508
  out:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
509
510
  	return referenced;
  }
bed7161a5   Balbir Singh   Memory controller...
511
  static int page_referenced_anon(struct page *page,
6fe6b7e35   Wu Fengguang   vmscan: report vm...
512
513
  				struct mem_cgroup *mem_cont,
  				unsigned long *vm_flags)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
514
515
516
  {
  	unsigned int mapcount;
  	struct anon_vma *anon_vma;
5beb49305   Rik van Riel   mm: change anon_v...
517
  	struct anon_vma_chain *avc;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
518
519
520
521
522
523
524
  	int referenced = 0;
  
  	anon_vma = page_lock_anon_vma(page);
  	if (!anon_vma)
  		return referenced;
  
  	mapcount = page_mapcount(page);
5beb49305   Rik van Riel   mm: change anon_v...
525
526
  	list_for_each_entry(avc, &anon_vma->head, same_anon_vma) {
  		struct vm_area_struct *vma = avc->vma;
1cb1729b1   Hugh Dickins   mm: pass address ...
527
528
529
  		unsigned long address = vma_address(page, vma);
  		if (address == -EFAULT)
  			continue;
bed7161a5   Balbir Singh   Memory controller...
530
531
532
533
534
  		/*
  		 * If we are reclaiming on behalf of a cgroup, skip
  		 * counting on behalf of references from different
  		 * cgroups
  		 */
bd845e38c   Hugh Dickins   memcg: mm_match_c...
535
  		if (mem_cont && !mm_match_cgroup(vma->vm_mm, mem_cont))
bed7161a5   Balbir Singh   Memory controller...
536
  			continue;
1cb1729b1   Hugh Dickins   mm: pass address ...
537
  		referenced += page_referenced_one(page, vma, address,
6fe6b7e35   Wu Fengguang   vmscan: report vm...
538
  						  &mapcount, vm_flags);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
539
540
541
  		if (!mapcount)
  			break;
  	}
34bbd7040   Oleg Nesterov   [PATCH] adapt pag...
542
543
  
  	page_unlock_anon_vma(anon_vma);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
544
545
546
547
548
549
  	return referenced;
  }
  
  /**
   * page_referenced_file - referenced check for object-based rmap
   * @page: the page we're checking references on.
43d8eac44   Randy Dunlap   mm: rmap kernel-d...
550
   * @mem_cont: target memory controller
6fe6b7e35   Wu Fengguang   vmscan: report vm...
551
   * @vm_flags: collect encountered vma->vm_flags who actually referenced the page
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
552
553
554
555
556
557
558
559
   *
   * For an object-based mapped page, find all the places it is mapped and
   * check/clear the referenced flag.  This is done by following the page->mapping
   * pointer, then walking the chain of vmas it holds.  It returns the number
   * of references it found.
   *
   * This function is only called from page_referenced for object-based pages.
   */
bed7161a5   Balbir Singh   Memory controller...
560
  static int page_referenced_file(struct page *page,
6fe6b7e35   Wu Fengguang   vmscan: report vm...
561
562
  				struct mem_cgroup *mem_cont,
  				unsigned long *vm_flags)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
  {
  	unsigned int mapcount;
  	struct address_space *mapping = page->mapping;
  	pgoff_t pgoff = page->index << (PAGE_CACHE_SHIFT - PAGE_SHIFT);
  	struct vm_area_struct *vma;
  	struct prio_tree_iter iter;
  	int referenced = 0;
  
  	/*
  	 * The caller's checks on page->mapping and !PageAnon have made
  	 * sure that this is a file page: the check for page->mapping
  	 * excludes the case just before it gets set on an anon page.
  	 */
  	BUG_ON(PageAnon(page));
  
  	/*
  	 * The page lock not only makes sure that page->mapping cannot
  	 * suddenly be NULLified by truncation, it makes sure that the
  	 * structure at mapping cannot be freed and reused yet,
  	 * so we can safely take mapping->i_mmap_lock.
  	 */
  	BUG_ON(!PageLocked(page));
  
  	spin_lock(&mapping->i_mmap_lock);
  
  	/*
  	 * i_mmap_lock does not stabilize mapcount at all, but mapcount
  	 * is more likely to be accurate if we note it after spinning.
  	 */
  	mapcount = page_mapcount(page);
  
  	vma_prio_tree_foreach(vma, &iter, &mapping->i_mmap, pgoff, pgoff) {
1cb1729b1   Hugh Dickins   mm: pass address ...
595
596
597
  		unsigned long address = vma_address(page, vma);
  		if (address == -EFAULT)
  			continue;
bed7161a5   Balbir Singh   Memory controller...
598
599
600
601
602
  		/*
  		 * If we are reclaiming on behalf of a cgroup, skip
  		 * counting on behalf of references from different
  		 * cgroups
  		 */
bd845e38c   Hugh Dickins   memcg: mm_match_c...
603
  		if (mem_cont && !mm_match_cgroup(vma->vm_mm, mem_cont))
bed7161a5   Balbir Singh   Memory controller...
604
  			continue;
1cb1729b1   Hugh Dickins   mm: pass address ...
605
  		referenced += page_referenced_one(page, vma, address,
6fe6b7e35   Wu Fengguang   vmscan: report vm...
606
  						  &mapcount, vm_flags);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
607
608
609
610
611
612
613
614
615
616
617
618
  		if (!mapcount)
  			break;
  	}
  
  	spin_unlock(&mapping->i_mmap_lock);
  	return referenced;
  }
  
  /**
   * page_referenced - test if the page was referenced
   * @page: the page to test
   * @is_locked: caller holds lock on the page
43d8eac44   Randy Dunlap   mm: rmap kernel-d...
619
   * @mem_cont: target memory controller
6fe6b7e35   Wu Fengguang   vmscan: report vm...
620
   * @vm_flags: collect encountered vma->vm_flags who actually referenced the page
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
621
622
623
624
   *
   * Quick test_and_clear_referenced for all mappings to a page,
   * returns the number of ptes which referenced the page.
   */
6fe6b7e35   Wu Fengguang   vmscan: report vm...
625
626
627
628
  int page_referenced(struct page *page,
  		    int is_locked,
  		    struct mem_cgroup *mem_cont,
  		    unsigned long *vm_flags)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
629
630
  {
  	int referenced = 0;
5ad646880   Hugh Dickins   ksm: let shared p...
631
  	int we_locked = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
632

6fe6b7e35   Wu Fengguang   vmscan: report vm...
633
  	*vm_flags = 0;
3ca7b3c5b   Hugh Dickins   mm: define PAGE_M...
634
  	if (page_mapped(page) && page_rmapping(page)) {
5ad646880   Hugh Dickins   ksm: let shared p...
635
636
637
638
639
640
641
642
643
644
645
  		if (!is_locked && (!PageAnon(page) || PageKsm(page))) {
  			we_locked = trylock_page(page);
  			if (!we_locked) {
  				referenced++;
  				goto out;
  			}
  		}
  		if (unlikely(PageKsm(page)))
  			referenced += page_referenced_ksm(page, mem_cont,
  								vm_flags);
  		else if (PageAnon(page))
6fe6b7e35   Wu Fengguang   vmscan: report vm...
646
647
  			referenced += page_referenced_anon(page, mem_cont,
  								vm_flags);
5ad646880   Hugh Dickins   ksm: let shared p...
648
  		else if (page->mapping)
6fe6b7e35   Wu Fengguang   vmscan: report vm...
649
650
  			referenced += page_referenced_file(page, mem_cont,
  								vm_flags);
5ad646880   Hugh Dickins   ksm: let shared p...
651
  		if (we_locked)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
652
  			unlock_page(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
653
  	}
5ad646880   Hugh Dickins   ksm: let shared p...
654
  out:
5b7baf057   Christian Borntraeger   s390: KVM prepara...
655
656
  	if (page_test_and_clear_young(page))
  		referenced++;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
657
658
  	return referenced;
  }
1cb1729b1   Hugh Dickins   mm: pass address ...
659
660
  static int page_mkclean_one(struct page *page, struct vm_area_struct *vma,
  			    unsigned long address)
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
661
662
  {
  	struct mm_struct *mm = vma->vm_mm;
c2fda5fed   Peter Zijlstra   [PATCH] Fix up pa...
663
  	pte_t *pte;
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
664
665
  	spinlock_t *ptl;
  	int ret = 0;
479db0bf4   Nick Piggin   mm: dirty page tr...
666
  	pte = page_check_address(page, mm, address, &ptl, 1);
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
667
668
  	if (!pte)
  		goto out;
c2fda5fed   Peter Zijlstra   [PATCH] Fix up pa...
669
670
  	if (pte_dirty(*pte) || pte_write(*pte)) {
  		pte_t entry;
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
671

c2fda5fed   Peter Zijlstra   [PATCH] Fix up pa...
672
  		flush_cache_page(vma, address, pte_pfn(*pte));
cddb8a5c1   Andrea Arcangeli   mmu-notifiers: core
673
  		entry = ptep_clear_flush_notify(vma, address, pte);
c2fda5fed   Peter Zijlstra   [PATCH] Fix up pa...
674
675
  		entry = pte_wrprotect(entry);
  		entry = pte_mkclean(entry);
d6e88e671   Al Viro   [PATCH] page_mkcl...
676
  		set_pte_at(mm, address, pte, entry);
c2fda5fed   Peter Zijlstra   [PATCH] Fix up pa...
677
678
  		ret = 1;
  	}
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
679

d08b3851d   Peter Zijlstra   [PATCH] mm: track...
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
  	pte_unmap_unlock(pte, ptl);
  out:
  	return ret;
  }
  
  static int page_mkclean_file(struct address_space *mapping, struct page *page)
  {
  	pgoff_t pgoff = page->index << (PAGE_CACHE_SHIFT - PAGE_SHIFT);
  	struct vm_area_struct *vma;
  	struct prio_tree_iter iter;
  	int ret = 0;
  
  	BUG_ON(PageAnon(page));
  
  	spin_lock(&mapping->i_mmap_lock);
  	vma_prio_tree_foreach(vma, &iter, &mapping->i_mmap, pgoff, pgoff) {
1cb1729b1   Hugh Dickins   mm: pass address ...
696
697
698
699
700
701
  		if (vma->vm_flags & VM_SHARED) {
  			unsigned long address = vma_address(page, vma);
  			if (address == -EFAULT)
  				continue;
  			ret += page_mkclean_one(page, vma, address);
  		}
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
702
703
704
705
706
707
708
709
710
711
712
713
714
  	}
  	spin_unlock(&mapping->i_mmap_lock);
  	return ret;
  }
  
  int page_mkclean(struct page *page)
  {
  	int ret = 0;
  
  	BUG_ON(!PageLocked(page));
  
  	if (page_mapped(page)) {
  		struct address_space *mapping = page_mapping(page);
ce7e9fae8   Christian Borntraeger   [S390] Optimize s...
715
  		if (mapping) {
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
716
  			ret = page_mkclean_file(mapping, page);
ce7e9fae8   Christian Borntraeger   [S390] Optimize s...
717
  			if (page_test_dirty(page)) {
e2b8d7af0   Martin Schwidefsky   [S390] add suppor...
718
  				page_clear_dirty(page, 1);
ce7e9fae8   Christian Borntraeger   [S390] Optimize s...
719
720
  				ret = 1;
  			}
6c210482a   Martin Schwidefsky   [S390] split page...
721
  		}
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
722
723
724
725
  	}
  
  	return ret;
  }
60b59beaf   Jaya Kumar   fbdev: mm: Deferr...
726
  EXPORT_SYMBOL_GPL(page_mkclean);
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
727

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
728
  /**
c44b67432   Rik van Riel   rmap: move exclus...
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
   * page_move_anon_rmap - move a page to our anon_vma
   * @page:	the page to move to our anon_vma
   * @vma:	the vma the page belongs to
   * @address:	the user virtual address mapped
   *
   * When a page belongs exclusively to one process after a COW event,
   * that page can be moved into the anon_vma that belongs to just that
   * process, so the rmap code will not search the parent or sibling
   * processes.
   */
  void page_move_anon_rmap(struct page *page,
  	struct vm_area_struct *vma, unsigned long address)
  {
  	struct anon_vma *anon_vma = vma->anon_vma;
  
  	VM_BUG_ON(!PageLocked(page));
  	VM_BUG_ON(!anon_vma);
  	VM_BUG_ON(page->index != linear_page_index(vma, address));
  
  	anon_vma = (void *) anon_vma + PAGE_MAPPING_ANON;
  	page->mapping = (struct address_space *) anon_vma;
  }
  
  /**
4e1c19750   Andi Kleen   Clean up __page_s...
753
754
755
756
   * __page_set_anon_rmap - set up new anonymous rmap
   * @page:	Page to add to rmap	
   * @vma:	VM area to add page to.
   * @address:	User virtual address of the mapping	
e8a03feb5   Rik van Riel   rmap: add exclusi...
757
   * @exclusive:	the page is exclusively owned by the current process
9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
758
759
   */
  static void __page_set_anon_rmap(struct page *page,
e8a03feb5   Rik van Riel   rmap: add exclusi...
760
  	struct vm_area_struct *vma, unsigned long address, int exclusive)
9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
761
  {
e8a03feb5   Rik van Riel   rmap: add exclusi...
762
  	struct anon_vma *anon_vma = vma->anon_vma;
ea90002b0   Linus Torvalds   anonvma: when set...
763

e8a03feb5   Rik van Riel   rmap: add exclusi...
764
  	BUG_ON(!anon_vma);
ea90002b0   Linus Torvalds   anonvma: when set...
765

4e1c19750   Andi Kleen   Clean up __page_s...
766
767
  	if (PageAnon(page))
  		return;
ea90002b0   Linus Torvalds   anonvma: when set...
768
  	/*
e8a03feb5   Rik van Riel   rmap: add exclusi...
769
770
771
  	 * If the page isn't exclusively mapped into this vma,
  	 * we must use the _oldest_ possible anon_vma for the
  	 * page mapping!
ea90002b0   Linus Torvalds   anonvma: when set...
772
  	 */
4e1c19750   Andi Kleen   Clean up __page_s...
773
  	if (!exclusive)
288468c33   Andrea Arcangeli   rmap: always use ...
774
  		anon_vma = anon_vma->root;
9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
775

9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
776
777
  	anon_vma = (void *) anon_vma + PAGE_MAPPING_ANON;
  	page->mapping = (struct address_space *) anon_vma;
9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
778
  	page->index = linear_page_index(vma, address);
9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
779
780
781
  }
  
  /**
43d8eac44   Randy Dunlap   mm: rmap kernel-d...
782
   * __page_check_anon_rmap - sanity check anonymous rmap addition
c97a9e10e   Nick Piggin   mm: more rmap che...
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
   * @page:	the page to add the mapping to
   * @vma:	the vm area in which the mapping is added
   * @address:	the user virtual address mapped
   */
  static void __page_check_anon_rmap(struct page *page,
  	struct vm_area_struct *vma, unsigned long address)
  {
  #ifdef CONFIG_DEBUG_VM
  	/*
  	 * The page's anon-rmap details (mapping and index) are guaranteed to
  	 * be set up correctly at this point.
  	 *
  	 * We have exclusion against page_add_anon_rmap because the caller
  	 * always holds the page locked, except if called from page_dup_rmap,
  	 * in which case the page is already known to be setup.
  	 *
  	 * We have exclusion against page_add_new_anon_rmap because those pages
  	 * are initially only visible via the pagetables, and the pte is locked
  	 * over the call to page_add_new_anon_rmap.
  	 */
44ab57a06   Andrea Arcangeli   rmap: add anon_vm...
803
  	BUG_ON(page_anon_vma(page)->root != vma->anon_vma->root);
c97a9e10e   Nick Piggin   mm: more rmap che...
804
805
806
807
808
  	BUG_ON(page->index != linear_page_index(vma, address));
  #endif
  }
  
  /**
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
809
810
811
812
813
   * page_add_anon_rmap - add pte mapping to an anonymous page
   * @page:	the page to add the mapping to
   * @vma:	the vm area in which the mapping is added
   * @address:	the user virtual address mapped
   *
5ad646880   Hugh Dickins   ksm: let shared p...
814
   * The caller needs to hold the pte lock, and the page must be locked in
80e148226   Hugh Dickins   ksm: share anon p...
815
816
817
   * the anon_vma case: to serialize mapping,index checking after setting,
   * and to ensure that PageAnon is not being upgraded racily to PageKsm
   * (but PageKsm is never downgraded to PageAnon).
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
818
819
820
821
   */
  void page_add_anon_rmap(struct page *page,
  	struct vm_area_struct *vma, unsigned long address)
  {
ad8c2ee80   Rik van Riel   rmap: add exclusi...
822
823
824
825
826
827
828
829
830
831
832
  	do_page_add_anon_rmap(page, vma, address, 0);
  }
  
  /*
   * Special version of the above for do_swap_page, which often runs
   * into pages that are exclusively owned by the current process.
   * Everybody else should continue to use page_add_anon_rmap above.
   */
  void do_page_add_anon_rmap(struct page *page,
  	struct vm_area_struct *vma, unsigned long address, int exclusive)
  {
5ad646880   Hugh Dickins   ksm: let shared p...
833
834
835
836
837
  	int first = atomic_inc_and_test(&page->_mapcount);
  	if (first)
  		__inc_zone_page_state(page, NR_ANON_PAGES);
  	if (unlikely(PageKsm(page)))
  		return;
c97a9e10e   Nick Piggin   mm: more rmap che...
838
839
  	VM_BUG_ON(!PageLocked(page));
  	VM_BUG_ON(address < vma->vm_start || address >= vma->vm_end);
5ad646880   Hugh Dickins   ksm: let shared p...
840
  	if (first)
ad8c2ee80   Rik van Riel   rmap: add exclusi...
841
  		__page_set_anon_rmap(page, vma, address, exclusive);
69029cd55   KAMEZAWA Hiroyuki   memcg: remove ref...
842
  	else
c97a9e10e   Nick Piggin   mm: more rmap che...
843
  		__page_check_anon_rmap(page, vma, address);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
844
  }
43d8eac44   Randy Dunlap   mm: rmap kernel-d...
845
  /**
9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
846
847
848
849
850
851
852
   * page_add_new_anon_rmap - add pte mapping to a new anonymous page
   * @page:	the page to add the mapping to
   * @vma:	the vm area in which the mapping is added
   * @address:	the user virtual address mapped
   *
   * Same as page_add_anon_rmap but must only be called on *new* pages.
   * This means the inc-and-test can be bypassed.
c97a9e10e   Nick Piggin   mm: more rmap che...
853
   * Page does not have to be locked.
9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
854
855
856
857
   */
  void page_add_new_anon_rmap(struct page *page,
  	struct vm_area_struct *vma, unsigned long address)
  {
b5934c531   Hugh Dickins   mm: add_active_or...
858
  	VM_BUG_ON(address < vma->vm_start || address >= vma->vm_end);
cbf84b7ad   Hugh Dickins   mm: further clean...
859
860
  	SetPageSwapBacked(page);
  	atomic_set(&page->_mapcount, 0); /* increment count (starts at -1) */
5ad646880   Hugh Dickins   ksm: let shared p...
861
  	__inc_zone_page_state(page, NR_ANON_PAGES);
e8a03feb5   Rik van Riel   rmap: add exclusi...
862
  	__page_set_anon_rmap(page, vma, address, 1);
b5934c531   Hugh Dickins   mm: add_active_or...
863
  	if (page_evictable(page, vma))
cbf84b7ad   Hugh Dickins   mm: further clean...
864
  		lru_cache_add_lru(page, LRU_ACTIVE_ANON);
b5934c531   Hugh Dickins   mm: add_active_or...
865
866
  	else
  		add_page_to_unevictable_list(page);
9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
867
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
868
869
870
871
  /**
   * page_add_file_rmap - add pte mapping to a file page
   * @page: the page to add the mapping to
   *
b8072f099   Hugh Dickins   [PATCH] mm: updat...
872
   * The caller needs to hold the pte lock.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
873
874
875
   */
  void page_add_file_rmap(struct page *page)
  {
d69b042f3   Balbir Singh   memcg: add file-b...
876
  	if (atomic_inc_and_test(&page->_mapcount)) {
65ba55f50   Christoph Lameter   [PATCH] zoned vm ...
877
  		__inc_zone_page_state(page, NR_FILE_MAPPED);
d8046582d   KAMEZAWA Hiroyuki   memcg: make memcg...
878
  		mem_cgroup_update_file_mapped(page, 1);
d69b042f3   Balbir Singh   memcg: add file-b...
879
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
880
881
882
883
884
885
  }
  
  /**
   * page_remove_rmap - take down pte mapping from a page
   * @page: page to remove mapping from
   *
b8072f099   Hugh Dickins   [PATCH] mm: updat...
886
   * The caller needs to hold the pte lock.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
887
   */
edc315fd2   Hugh Dickins   badpage: remove v...
888
  void page_remove_rmap(struct page *page)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
889
  {
b904dcfed   KOSAKI Motohiro   mm: clean up page...
890
891
892
893
894
895
896
897
898
899
900
901
  	/* page still mapped by someone else? */
  	if (!atomic_add_negative(-1, &page->_mapcount))
  		return;
  
  	/*
  	 * Now that the last pte has gone, s390 must transfer dirty
  	 * flag from storage key to struct page.  We can usually skip
  	 * this if the page is anon, so about to be freed; but perhaps
  	 * not if it's in swapcache - there might be another pte slot
  	 * containing the swap entry, but page not yet written to swap.
  	 */
  	if ((!PageAnon(page) || PageSwapCache(page)) && page_test_dirty(page)) {
e2b8d7af0   Martin Schwidefsky   [S390] add suppor...
902
  		page_clear_dirty(page, 1);
b904dcfed   KOSAKI Motohiro   mm: clean up page...
903
  		set_page_dirty(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
904
  	}
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
905
906
907
908
909
910
  	/*
  	 * Hugepages are not counted in NR_ANON_PAGES nor NR_FILE_MAPPED
  	 * and not charged by memcg for now.
  	 */
  	if (unlikely(PageHuge(page)))
  		return;
b904dcfed   KOSAKI Motohiro   mm: clean up page...
911
912
913
914
915
  	if (PageAnon(page)) {
  		mem_cgroup_uncharge_page(page);
  		__dec_zone_page_state(page, NR_ANON_PAGES);
  	} else {
  		__dec_zone_page_state(page, NR_FILE_MAPPED);
d8046582d   KAMEZAWA Hiroyuki   memcg: make memcg...
916
  		mem_cgroup_update_file_mapped(page, -1);
b904dcfed   KOSAKI Motohiro   mm: clean up page...
917
  	}
b904dcfed   KOSAKI Motohiro   mm: clean up page...
918
919
920
921
922
923
924
925
926
  	/*
  	 * It would be tidy to reset the PageAnon mapping here,
  	 * but that might overwrite a racing page_add_anon_rmap
  	 * which increments mapcount after us but sets mapping
  	 * before us: so leave the reset to free_hot_cold_page,
  	 * and remember that it's only reliable while mapped.
  	 * Leaving it set also helps swapoff to reinstate ptes
  	 * faster for those pages still in swapcache.
  	 */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
927
928
929
930
931
932
  }
  
  /*
   * Subfunctions of try_to_unmap: try_to_unmap_one called
   * repeatedly from either try_to_unmap_anon or try_to_unmap_file.
   */
5ad646880   Hugh Dickins   ksm: let shared p...
933
934
  int try_to_unmap_one(struct page *page, struct vm_area_struct *vma,
  		     unsigned long address, enum ttu_flags flags)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
935
936
  {
  	struct mm_struct *mm = vma->vm_mm;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
937
938
  	pte_t *pte;
  	pte_t pteval;
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
939
  	spinlock_t *ptl;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
940
  	int ret = SWAP_AGAIN;
479db0bf4   Nick Piggin   mm: dirty page tr...
941
  	pte = page_check_address(page, mm, address, &ptl, 0);
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
942
  	if (!pte)
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
943
  		goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
944
945
946
947
948
949
  
  	/*
  	 * If the page is mlock()d, we cannot swap it out.
  	 * If it's recently referenced (perhaps page_referenced
  	 * skipped over this mm) then we should reactivate it.
  	 */
14fa31b89   Andi Kleen   HWPOISON: Use bit...
950
  	if (!(flags & TTU_IGNORE_MLOCK)) {
caed0f486   KOSAKI Motohiro   mm: simplify try_...
951
952
  		if (vma->vm_flags & VM_LOCKED)
  			goto out_mlock;
af8e3354b   Hugh Dickins   mm: CONFIG_MMU fo...
953
  		if (TTU_ACTION(flags) == TTU_MUNLOCK)
53f79acb6   Hugh Dickins   mm: mlocking in t...
954
  			goto out_unmap;
14fa31b89   Andi Kleen   HWPOISON: Use bit...
955
956
  	}
  	if (!(flags & TTU_IGNORE_ACCESS)) {
b291f0003   Nick Piggin   mlock: mlocked pa...
957
958
959
960
961
  		if (ptep_clear_flush_young_notify(vma, address, pte)) {
  			ret = SWAP_FAIL;
  			goto out_unmap;
  		}
    	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
962

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
963
964
  	/* Nuke the page table entry. */
  	flush_cache_page(vma, address, page_to_pfn(page));
cddb8a5c1   Andrea Arcangeli   mmu-notifiers: core
965
  	pteval = ptep_clear_flush_notify(vma, address, pte);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
966
967
968
969
  
  	/* Move the dirty bit to the physical page now the pte is gone. */
  	if (pte_dirty(pteval))
  		set_page_dirty(page);
365e9c87a   Hugh Dickins   [PATCH] mm: updat...
970
971
  	/* Update high watermark before we lower rss */
  	update_hiwater_rss(mm);
888b9f7c5   Andi Kleen   HWPOISON: Handle ...
972
973
  	if (PageHWPoison(page) && !(flags & TTU_IGNORE_HWPOISON)) {
  		if (PageAnon(page))
d559db086   KAMEZAWA Hiroyuki   mm: clean up mm_c...
974
  			dec_mm_counter(mm, MM_ANONPAGES);
888b9f7c5   Andi Kleen   HWPOISON: Handle ...
975
  		else
d559db086   KAMEZAWA Hiroyuki   mm: clean up mm_c...
976
  			dec_mm_counter(mm, MM_FILEPAGES);
888b9f7c5   Andi Kleen   HWPOISON: Handle ...
977
978
979
  		set_pte_at(mm, address, pte,
  				swp_entry_to_pte(make_hwpoison_entry(page)));
  	} else if (PageAnon(page)) {
4c21e2f24   Hugh Dickins   [PATCH] mm: split...
980
  		swp_entry_t entry = { .val = page_private(page) };
0697212a4   Christoph Lameter   [PATCH] Swapless ...
981
982
983
984
985
986
  
  		if (PageSwapCache(page)) {
  			/*
  			 * Store the swap location in the pte.
  			 * See handle_pte_fault() ...
  			 */
570a335b8   Hugh Dickins   swap_info: swap c...
987
988
989
990
991
  			if (swap_duplicate(entry) < 0) {
  				set_pte_at(mm, address, pte, pteval);
  				ret = SWAP_FAIL;
  				goto out_unmap;
  			}
0697212a4   Christoph Lameter   [PATCH] Swapless ...
992
993
994
995
996
997
  			if (list_empty(&mm->mmlist)) {
  				spin_lock(&mmlist_lock);
  				if (list_empty(&mm->mmlist))
  					list_add(&mm->mmlist, &init_mm.mmlist);
  				spin_unlock(&mmlist_lock);
  			}
d559db086   KAMEZAWA Hiroyuki   mm: clean up mm_c...
998
  			dec_mm_counter(mm, MM_ANONPAGES);
b084d4353   KAMEZAWA Hiroyuki   mm: count swap usage
999
  			inc_mm_counter(mm, MM_SWAPENTS);
64cdd548f   KOSAKI Motohiro   mm: cleanup: remo...
1000
  		} else if (PAGE_MIGRATION) {
0697212a4   Christoph Lameter   [PATCH] Swapless ...
1001
1002
1003
1004
1005
  			/*
  			 * Store the pfn of the page in a special migration
  			 * pte. do_swap_page() will wait until the migration
  			 * pte is removed and then restart fault handling.
  			 */
14fa31b89   Andi Kleen   HWPOISON: Use bit...
1006
  			BUG_ON(TTU_ACTION(flags) != TTU_MIGRATION);
0697212a4   Christoph Lameter   [PATCH] Swapless ...
1007
  			entry = make_migration_entry(page, pte_write(pteval));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1008
1009
1010
  		}
  		set_pte_at(mm, address, pte, swp_entry_to_pte(entry));
  		BUG_ON(pte_file(*pte));
14fa31b89   Andi Kleen   HWPOISON: Use bit...
1011
  	} else if (PAGE_MIGRATION && (TTU_ACTION(flags) == TTU_MIGRATION)) {
04e62a29b   Christoph Lameter   [PATCH] More page...
1012
1013
1014
1015
1016
  		/* Establish migration entry for a file page */
  		swp_entry_t entry;
  		entry = make_migration_entry(page, pte_write(pteval));
  		set_pte_at(mm, address, pte, swp_entry_to_pte(entry));
  	} else
d559db086   KAMEZAWA Hiroyuki   mm: clean up mm_c...
1017
  		dec_mm_counter(mm, MM_FILEPAGES);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1018

edc315fd2   Hugh Dickins   badpage: remove v...
1019
  	page_remove_rmap(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1020
1021
1022
  	page_cache_release(page);
  
  out_unmap:
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
1023
  	pte_unmap_unlock(pte, ptl);
caed0f486   KOSAKI Motohiro   mm: simplify try_...
1024
1025
  out:
  	return ret;
53f79acb6   Hugh Dickins   mm: mlocking in t...
1026

caed0f486   KOSAKI Motohiro   mm: simplify try_...
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
  out_mlock:
  	pte_unmap_unlock(pte, ptl);
  
  
  	/*
  	 * We need mmap_sem locking, Otherwise VM_LOCKED check makes
  	 * unstable result and race. Plus, We can't wait here because
  	 * we now hold anon_vma->lock or mapping->i_mmap_lock.
  	 * if trylock failed, the page remain in evictable lru and later
  	 * vmscan could retry to move the page to unevictable lru if the
  	 * page is actually mlocked.
  	 */
  	if (down_read_trylock(&vma->vm_mm->mmap_sem)) {
  		if (vma->vm_flags & VM_LOCKED) {
  			mlock_vma_page(page);
  			ret = SWAP_MLOCK;
53f79acb6   Hugh Dickins   mm: mlocking in t...
1043
  		}
caed0f486   KOSAKI Motohiro   mm: simplify try_...
1044
  		up_read(&vma->vm_mm->mmap_sem);
53f79acb6   Hugh Dickins   mm: mlocking in t...
1045
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
1060
1061
1062
1063
1064
1065
1066
  	return ret;
  }
  
  /*
   * objrmap doesn't work for nonlinear VMAs because the assumption that
   * offset-into-file correlates with offset-into-virtual-addresses does not hold.
   * Consequently, given a particular page and its ->index, we cannot locate the
   * ptes which are mapping that page without an exhaustive linear search.
   *
   * So what this code does is a mini "virtual scan" of each nonlinear VMA which
   * maps the file to which the target page belongs.  The ->vm_private_data field
   * holds the current cursor into that scan.  Successive searches will circulate
   * around the vma's virtual address space.
   *
   * So as more replacement pressure is applied to the pages in a nonlinear VMA,
   * more scanning pressure is placed against them as well.   Eventually pages
   * will become fully unmapped and are eligible for eviction.
   *
   * For very sparsely populated VMAs this is a little inefficient - chances are
   * there there won't be many ptes located within the scan cluster.  In this case
   * maybe we could scan further - to the end of the pte page, perhaps.
b291f0003   Nick Piggin   mlock: mlocked pa...
1067
1068
1069
1070
1071
   *
   * Mlocked pages:  check VM_LOCKED under mmap_sem held for read, if we can
   * acquire it without blocking.  If vma locked, mlock the pages in the cluster,
   * rather than unmapping them.  If we encounter the "check_page" that vmscan is
   * trying to unmap, return SWAP_MLOCK, else default SWAP_AGAIN.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1072
1073
1074
   */
  #define CLUSTER_SIZE	min(32*PAGE_SIZE, PMD_SIZE)
  #define CLUSTER_MASK	(~(CLUSTER_SIZE - 1))
b291f0003   Nick Piggin   mlock: mlocked pa...
1075
1076
  static int try_to_unmap_cluster(unsigned long cursor, unsigned int *mapcount,
  		struct vm_area_struct *vma, struct page *check_page)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1077
1078
1079
1080
1081
  {
  	struct mm_struct *mm = vma->vm_mm;
  	pgd_t *pgd;
  	pud_t *pud;
  	pmd_t *pmd;
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
1082
  	pte_t *pte;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1083
  	pte_t pteval;
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
1084
  	spinlock_t *ptl;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1085
1086
1087
  	struct page *page;
  	unsigned long address;
  	unsigned long end;
b291f0003   Nick Piggin   mlock: mlocked pa...
1088
1089
  	int ret = SWAP_AGAIN;
  	int locked_vma = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1090

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1091
1092
1093
1094
1095
1096
1097
1098
1099
  	address = (vma->vm_start + cursor) & CLUSTER_MASK;
  	end = address + CLUSTER_SIZE;
  	if (address < vma->vm_start)
  		address = vma->vm_start;
  	if (end > vma->vm_end)
  		end = vma->vm_end;
  
  	pgd = pgd_offset(mm, address);
  	if (!pgd_present(*pgd))
b291f0003   Nick Piggin   mlock: mlocked pa...
1100
  		return ret;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1101
1102
1103
  
  	pud = pud_offset(pgd, address);
  	if (!pud_present(*pud))
b291f0003   Nick Piggin   mlock: mlocked pa...
1104
  		return ret;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1105
1106
1107
  
  	pmd = pmd_offset(pud, address);
  	if (!pmd_present(*pmd))
b291f0003   Nick Piggin   mlock: mlocked pa...
1108
1109
1110
  		return ret;
  
  	/*
af8e3354b   Hugh Dickins   mm: CONFIG_MMU fo...
1111
  	 * If we can acquire the mmap_sem for read, and vma is VM_LOCKED,
b291f0003   Nick Piggin   mlock: mlocked pa...
1112
1113
  	 * keep the sem while scanning the cluster for mlocking pages.
  	 */
af8e3354b   Hugh Dickins   mm: CONFIG_MMU fo...
1114
  	if (down_read_trylock(&vma->vm_mm->mmap_sem)) {
b291f0003   Nick Piggin   mlock: mlocked pa...
1115
1116
1117
1118
  		locked_vma = (vma->vm_flags & VM_LOCKED);
  		if (!locked_vma)
  			up_read(&vma->vm_mm->mmap_sem); /* don't need it */
  	}
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
1119
1120
  
  	pte = pte_offset_map_lock(mm, pmd, address, &ptl);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1121

365e9c87a   Hugh Dickins   [PATCH] mm: updat...
1122
1123
  	/* Update high watermark before we lower rss */
  	update_hiwater_rss(mm);
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
1124
  	for (; address < end; pte++, address += PAGE_SIZE) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1125
1126
  		if (!pte_present(*pte))
  			continue;
6aab341e0   Linus Torvalds   mm: re-architect ...
1127
1128
  		page = vm_normal_page(vma, address, *pte);
  		BUG_ON(!page || PageAnon(page));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1129

b291f0003   Nick Piggin   mlock: mlocked pa...
1130
1131
1132
1133
1134
1135
  		if (locked_vma) {
  			mlock_vma_page(page);   /* no-op if already mlocked */
  			if (page == check_page)
  				ret = SWAP_MLOCK;
  			continue;	/* don't unmap */
  		}
cddb8a5c1   Andrea Arcangeli   mmu-notifiers: core
1136
  		if (ptep_clear_flush_young_notify(vma, address, pte))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1137
1138
1139
  			continue;
  
  		/* Nuke the page table entry. */
eca351336   Ben Collins   [PATCH] Fix missi...
1140
  		flush_cache_page(vma, address, pte_pfn(*pte));
cddb8a5c1   Andrea Arcangeli   mmu-notifiers: core
1141
  		pteval = ptep_clear_flush_notify(vma, address, pte);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1142
1143
1144
1145
1146
1147
1148
1149
  
  		/* If nonlinear, store the file page offset in the pte. */
  		if (page->index != linear_page_index(vma, address))
  			set_pte_at(mm, address, pte, pgoff_to_pte(page->index));
  
  		/* Move the dirty bit to the physical page now the pte is gone. */
  		if (pte_dirty(pteval))
  			set_page_dirty(page);
edc315fd2   Hugh Dickins   badpage: remove v...
1150
  		page_remove_rmap(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1151
  		page_cache_release(page);
d559db086   KAMEZAWA Hiroyuki   mm: clean up mm_c...
1152
  		dec_mm_counter(mm, MM_FILEPAGES);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1153
1154
  		(*mapcount)--;
  	}
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
1155
  	pte_unmap_unlock(pte - 1, ptl);
b291f0003   Nick Piggin   mlock: mlocked pa...
1156
1157
1158
  	if (locked_vma)
  		up_read(&vma->vm_mm->mmap_sem);
  	return ret;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1159
  }
a8bef8ff6   Mel Gorman   mm: migration: av...
1160
1161
1162
1163
1164
1165
1166
1167
1168
1169
1170
1171
1172
  static bool is_vma_temporary_stack(struct vm_area_struct *vma)
  {
  	int maybe_stack = vma->vm_flags & (VM_GROWSDOWN | VM_GROWSUP);
  
  	if (!maybe_stack)
  		return false;
  
  	if ((vma->vm_flags & VM_STACK_INCOMPLETE_SETUP) ==
  						VM_STACK_INCOMPLETE_SETUP)
  		return true;
  
  	return false;
  }
b291f0003   Nick Piggin   mlock: mlocked pa...
1173
1174
1175
1176
  /**
   * try_to_unmap_anon - unmap or unlock anonymous page using the object-based
   * rmap method
   * @page: the page to unmap/unlock
8051be5e6   Huang Shijie   rmap: fix the com...
1177
   * @flags: action and flags
b291f0003   Nick Piggin   mlock: mlocked pa...
1178
1179
1180
1181
1182
1183
1184
1185
1186
1187
1188
   *
   * Find all the mappings of a page using the mapping pointer and the vma chains
   * contained in the anon_vma struct it points to.
   *
   * This function is only called from try_to_unmap/try_to_munlock for
   * anonymous pages.
   * When called from try_to_munlock(), the mmap_sem of the mm containing the vma
   * where the page was found will be held for write.  So, we won't recheck
   * vm_flags for that VMA.  That should be OK, because that vma shouldn't be
   * 'LOCKED.
   */
14fa31b89   Andi Kleen   HWPOISON: Use bit...
1189
  static int try_to_unmap_anon(struct page *page, enum ttu_flags flags)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1190
1191
  {
  	struct anon_vma *anon_vma;
5beb49305   Rik van Riel   mm: change anon_v...
1192
  	struct anon_vma_chain *avc;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1193
  	int ret = SWAP_AGAIN;
b291f0003   Nick Piggin   mlock: mlocked pa...
1194

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1195
1196
1197
  	anon_vma = page_lock_anon_vma(page);
  	if (!anon_vma)
  		return ret;
5beb49305   Rik van Riel   mm: change anon_v...
1198
1199
  	list_for_each_entry(avc, &anon_vma->head, same_anon_vma) {
  		struct vm_area_struct *vma = avc->vma;
a8bef8ff6   Mel Gorman   mm: migration: av...
1200
1201
1202
1203
1204
1205
1206
1207
1208
1209
1210
1211
1212
1213
1214
  		unsigned long address;
  
  		/*
  		 * During exec, a temporary VMA is setup and later moved.
  		 * The VMA is moved under the anon_vma lock but not the
  		 * page tables leading to a race where migration cannot
  		 * find the migration ptes. Rather than increasing the
  		 * locking requirements of exec(), migration skips
  		 * temporary VMAs until after exec() completes.
  		 */
  		if (PAGE_MIGRATION && (flags & TTU_MIGRATION) &&
  				is_vma_temporary_stack(vma))
  			continue;
  
  		address = vma_address(page, vma);
1cb1729b1   Hugh Dickins   mm: pass address ...
1215
1216
1217
  		if (address == -EFAULT)
  			continue;
  		ret = try_to_unmap_one(page, vma, address, flags);
53f79acb6   Hugh Dickins   mm: mlocking in t...
1218
1219
  		if (ret != SWAP_AGAIN || !page_mapped(page))
  			break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1220
  	}
34bbd7040   Oleg Nesterov   [PATCH] adapt pag...
1221
1222
  
  	page_unlock_anon_vma(anon_vma);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1223
1224
1225
1226
  	return ret;
  }
  
  /**
b291f0003   Nick Piggin   mlock: mlocked pa...
1227
1228
   * try_to_unmap_file - unmap/unlock file page using the object-based rmap method
   * @page: the page to unmap/unlock
14fa31b89   Andi Kleen   HWPOISON: Use bit...
1229
   * @flags: action and flags
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1230
1231
1232
1233
   *
   * Find all the mappings of a page using the mapping pointer and the vma chains
   * contained in the address_space struct it points to.
   *
b291f0003   Nick Piggin   mlock: mlocked pa...
1234
1235
1236
1237
1238
1239
   * This function is only called from try_to_unmap/try_to_munlock for
   * object-based pages.
   * When called from try_to_munlock(), the mmap_sem of the mm containing the vma
   * where the page was found will be held for write.  So, we won't recheck
   * vm_flags for that VMA.  That should be OK, because that vma shouldn't be
   * 'LOCKED.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1240
   */
14fa31b89   Andi Kleen   HWPOISON: Use bit...
1241
  static int try_to_unmap_file(struct page *page, enum ttu_flags flags)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1242
1243
1244
1245
1246
1247
1248
1249
1250
1251
1252
1253
1254
  {
  	struct address_space *mapping = page->mapping;
  	pgoff_t pgoff = page->index << (PAGE_CACHE_SHIFT - PAGE_SHIFT);
  	struct vm_area_struct *vma;
  	struct prio_tree_iter iter;
  	int ret = SWAP_AGAIN;
  	unsigned long cursor;
  	unsigned long max_nl_cursor = 0;
  	unsigned long max_nl_size = 0;
  	unsigned int mapcount;
  
  	spin_lock(&mapping->i_mmap_lock);
  	vma_prio_tree_foreach(vma, &iter, &mapping->i_mmap, pgoff, pgoff) {
1cb1729b1   Hugh Dickins   mm: pass address ...
1255
1256
1257
1258
  		unsigned long address = vma_address(page, vma);
  		if (address == -EFAULT)
  			continue;
  		ret = try_to_unmap_one(page, vma, address, flags);
53f79acb6   Hugh Dickins   mm: mlocking in t...
1259
1260
  		if (ret != SWAP_AGAIN || !page_mapped(page))
  			goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1261
1262
1263
1264
  	}
  
  	if (list_empty(&mapping->i_mmap_nonlinear))
  		goto out;
53f79acb6   Hugh Dickins   mm: mlocking in t...
1265
1266
1267
1268
1269
1270
1271
  	/*
  	 * We don't bother to try to find the munlocked page in nonlinears.
  	 * It's costly. Instead, later, page reclaim logic may call
  	 * try_to_unmap(TTU_MUNLOCK) and recover PG_mlocked lazily.
  	 */
  	if (TTU_ACTION(flags) == TTU_MUNLOCK)
  		goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1272
1273
  	list_for_each_entry(vma, &mapping->i_mmap_nonlinear,
  						shared.vm_set.list) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1274
1275
1276
1277
1278
1279
1280
  		cursor = (unsigned long) vma->vm_private_data;
  		if (cursor > max_nl_cursor)
  			max_nl_cursor = cursor;
  		cursor = vma->vm_end - vma->vm_start;
  		if (cursor > max_nl_size)
  			max_nl_size = cursor;
  	}
b291f0003   Nick Piggin   mlock: mlocked pa...
1281
  	if (max_nl_size == 0) {	/* all nonlinears locked or reserved ? */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1282
1283
1284
1285
1286
1287
1288
1289
1290
1291
1292
1293
1294
1295
1296
1297
1298
1299
1300
1301
1302
1303
1304
  		ret = SWAP_FAIL;
  		goto out;
  	}
  
  	/*
  	 * We don't try to search for this page in the nonlinear vmas,
  	 * and page_referenced wouldn't have found it anyway.  Instead
  	 * just walk the nonlinear vmas trying to age and unmap some.
  	 * The mapcount of the page we came in with is irrelevant,
  	 * but even so use it as a guide to how hard we should try?
  	 */
  	mapcount = page_mapcount(page);
  	if (!mapcount)
  		goto out;
  	cond_resched_lock(&mapping->i_mmap_lock);
  
  	max_nl_size = (max_nl_size + CLUSTER_SIZE - 1) & CLUSTER_MASK;
  	if (max_nl_cursor == 0)
  		max_nl_cursor = CLUSTER_SIZE;
  
  	do {
  		list_for_each_entry(vma, &mapping->i_mmap_nonlinear,
  						shared.vm_set.list) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1305
  			cursor = (unsigned long) vma->vm_private_data;
839b9685e   Hugh Dickins   [PATCH] rmap: don...
1306
  			while ( cursor < max_nl_cursor &&
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1307
  				cursor < vma->vm_end - vma->vm_start) {
53f79acb6   Hugh Dickins   mm: mlocking in t...
1308
1309
1310
  				if (try_to_unmap_cluster(cursor, &mapcount,
  						vma, page) == SWAP_MLOCK)
  					ret = SWAP_MLOCK;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1311
1312
1313
1314
1315
1316
1317
1318
1319
1320
1321
1322
1323
1324
1325
1326
  				cursor += CLUSTER_SIZE;
  				vma->vm_private_data = (void *) cursor;
  				if ((int)mapcount <= 0)
  					goto out;
  			}
  			vma->vm_private_data = (void *) max_nl_cursor;
  		}
  		cond_resched_lock(&mapping->i_mmap_lock);
  		max_nl_cursor += CLUSTER_SIZE;
  	} while (max_nl_cursor <= max_nl_size);
  
  	/*
  	 * Don't loop forever (perhaps all the remaining pages are
  	 * in locked vmas).  Reset cursor on all unreserved nonlinear
  	 * vmas, now forgetting on which ones it had fallen behind.
  	 */
101d2be76   Hugh Dickins   [PATCH] unpaged: ...
1327
1328
  	list_for_each_entry(vma, &mapping->i_mmap_nonlinear, shared.vm_set.list)
  		vma->vm_private_data = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1329
1330
1331
1332
1333
1334
1335
1336
  out:
  	spin_unlock(&mapping->i_mmap_lock);
  	return ret;
  }
  
  /**
   * try_to_unmap - try to remove all page table mappings to a page
   * @page: the page to get unmapped
14fa31b89   Andi Kleen   HWPOISON: Use bit...
1337
   * @flags: action and flags
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1338
1339
1340
1341
1342
1343
1344
1345
   *
   * Tries to remove all the page table entries which are mapping this
   * page, used in the pageout path.  Caller must hold the page lock.
   * Return values are:
   *
   * SWAP_SUCCESS	- we succeeded in removing all mappings
   * SWAP_AGAIN	- we missed a mapping, try again later
   * SWAP_FAIL	- the page is unswappable
b291f0003   Nick Piggin   mlock: mlocked pa...
1346
   * SWAP_MLOCK	- page is mlocked.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1347
   */
14fa31b89   Andi Kleen   HWPOISON: Use bit...
1348
  int try_to_unmap(struct page *page, enum ttu_flags flags)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1349
1350
  {
  	int ret;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1351
  	BUG_ON(!PageLocked(page));
5ad646880   Hugh Dickins   ksm: let shared p...
1352
1353
1354
  	if (unlikely(PageKsm(page)))
  		ret = try_to_unmap_ksm(page, flags);
  	else if (PageAnon(page))
14fa31b89   Andi Kleen   HWPOISON: Use bit...
1355
  		ret = try_to_unmap_anon(page, flags);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1356
  	else
14fa31b89   Andi Kleen   HWPOISON: Use bit...
1357
  		ret = try_to_unmap_file(page, flags);
b291f0003   Nick Piggin   mlock: mlocked pa...
1358
  	if (ret != SWAP_MLOCK && !page_mapped(page))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1359
1360
1361
  		ret = SWAP_SUCCESS;
  	return ret;
  }
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
1362

b291f0003   Nick Piggin   mlock: mlocked pa...
1363
1364
1365
1366
1367
1368
1369
1370
1371
1372
  /**
   * try_to_munlock - try to munlock a page
   * @page: the page to be munlocked
   *
   * Called from munlock code.  Checks all of the VMAs mapping the page
   * to make sure nobody else has this page mlocked. The page will be
   * returned with PG_mlocked cleared if no other vmas have it mlocked.
   *
   * Return values are:
   *
53f79acb6   Hugh Dickins   mm: mlocking in t...
1373
   * SWAP_AGAIN	- no vma is holding page mlocked, or,
b291f0003   Nick Piggin   mlock: mlocked pa...
1374
   * SWAP_AGAIN	- page mapped in mlocked vma -- couldn't acquire mmap sem
5ad646880   Hugh Dickins   ksm: let shared p...
1375
   * SWAP_FAIL	- page cannot be located at present
b291f0003   Nick Piggin   mlock: mlocked pa...
1376
1377
1378
1379
1380
   * SWAP_MLOCK	- page is now mlocked.
   */
  int try_to_munlock(struct page *page)
  {
  	VM_BUG_ON(!PageLocked(page) || PageLRU(page));
5ad646880   Hugh Dickins   ksm: let shared p...
1381
1382
1383
  	if (unlikely(PageKsm(page)))
  		return try_to_unmap_ksm(page, TTU_MUNLOCK);
  	else if (PageAnon(page))
14fa31b89   Andi Kleen   HWPOISON: Use bit...
1384
  		return try_to_unmap_anon(page, TTU_MUNLOCK);
b291f0003   Nick Piggin   mlock: mlocked pa...
1385
  	else
14fa31b89   Andi Kleen   HWPOISON: Use bit...
1386
  		return try_to_unmap_file(page, TTU_MUNLOCK);
b291f0003   Nick Piggin   mlock: mlocked pa...
1387
  }
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1388

76545066c   Rik van Riel   mm: extend KSM re...
1389
1390
1391
1392
1393
1394
1395
1396
1397
  #if defined(CONFIG_KSM) || defined(CONFIG_MIGRATION)
  /*
   * Drop an anon_vma refcount, freeing the anon_vma and anon_vma->root
   * if necessary.  Be careful to do all the tests under the lock.  Once
   * we know we are the last user, nobody else can get a reference and we
   * can do the freeing without the lock.
   */
  void drop_anon_vma(struct anon_vma *anon_vma)
  {
44ab57a06   Andrea Arcangeli   rmap: add anon_vm...
1398
  	BUG_ON(atomic_read(&anon_vma->external_refcount) <= 0);
76545066c   Rik van Riel   mm: extend KSM re...
1399
1400
1401
1402
1403
1404
1405
1406
1407
1408
1409
  	if (atomic_dec_and_lock(&anon_vma->external_refcount, &anon_vma->root->lock)) {
  		struct anon_vma *root = anon_vma->root;
  		int empty = list_empty(&anon_vma->head);
  		int last_root_user = 0;
  		int root_empty = 0;
  
  		/*
  		 * The refcount on a non-root anon_vma got dropped.  Drop
  		 * the refcount on the root and check if we need to free it.
  		 */
  		if (empty && anon_vma != root) {
44ab57a06   Andrea Arcangeli   rmap: add anon_vm...
1410
  			BUG_ON(atomic_read(&root->external_refcount) <= 0);
76545066c   Rik van Riel   mm: extend KSM re...
1411
1412
1413
1414
1415
1416
1417
1418
1419
1420
1421
1422
1423
  			last_root_user = atomic_dec_and_test(&root->external_refcount);
  			root_empty = list_empty(&root->head);
  		}
  		anon_vma_unlock(anon_vma);
  
  		if (empty) {
  			anon_vma_free(anon_vma);
  			if (root_empty && last_root_user)
  				anon_vma_free(root);
  		}
  	}
  }
  #endif
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1424
1425
1426
1427
1428
1429
1430
1431
1432
  #ifdef CONFIG_MIGRATION
  /*
   * rmap_walk() and its helpers rmap_walk_anon() and rmap_walk_file():
   * Called by migrate.c to remove migration ptes, but might be used more later.
   */
  static int rmap_walk_anon(struct page *page, int (*rmap_one)(struct page *,
  		struct vm_area_struct *, unsigned long, void *), void *arg)
  {
  	struct anon_vma *anon_vma;
5beb49305   Rik van Riel   mm: change anon_v...
1433
  	struct anon_vma_chain *avc;
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1434
1435
1436
1437
1438
  	int ret = SWAP_AGAIN;
  
  	/*
  	 * Note: remove_migration_ptes() cannot use page_lock_anon_vma()
  	 * because that depends on page_mapped(); but not all its usages
3f6c82728   Mel Gorman   mm: migration: ta...
1439
1440
  	 * are holding mmap_sem. Users without mmap_sem are required to
  	 * take a reference count to prevent the anon_vma disappearing
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1441
1442
1443
1444
  	 */
  	anon_vma = page_anon_vma(page);
  	if (!anon_vma)
  		return ret;
cba48b98f   Rik van Riel   mm: change direct...
1445
  	anon_vma_lock(anon_vma);
5beb49305   Rik van Riel   mm: change anon_v...
1446
1447
  	list_for_each_entry(avc, &anon_vma->head, same_anon_vma) {
  		struct vm_area_struct *vma = avc->vma;
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1448
1449
1450
1451
1452
1453
1454
  		unsigned long address = vma_address(page, vma);
  		if (address == -EFAULT)
  			continue;
  		ret = rmap_one(page, vma, address, arg);
  		if (ret != SWAP_AGAIN)
  			break;
  	}
cba48b98f   Rik van Riel   mm: change direct...
1455
  	anon_vma_unlock(anon_vma);
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1456
1457
1458
1459
1460
1461
1462
1463
1464
1465
1466
1467
1468
1469
1470
1471
1472
1473
1474
1475
1476
1477
1478
1479
1480
1481
1482
1483
1484
1485
1486
1487
1488
1489
1490
1491
1492
1493
1494
1495
1496
1497
1498
1499
1500
  	return ret;
  }
  
  static int rmap_walk_file(struct page *page, int (*rmap_one)(struct page *,
  		struct vm_area_struct *, unsigned long, void *), void *arg)
  {
  	struct address_space *mapping = page->mapping;
  	pgoff_t pgoff = page->index << (PAGE_CACHE_SHIFT - PAGE_SHIFT);
  	struct vm_area_struct *vma;
  	struct prio_tree_iter iter;
  	int ret = SWAP_AGAIN;
  
  	if (!mapping)
  		return ret;
  	spin_lock(&mapping->i_mmap_lock);
  	vma_prio_tree_foreach(vma, &iter, &mapping->i_mmap, pgoff, pgoff) {
  		unsigned long address = vma_address(page, vma);
  		if (address == -EFAULT)
  			continue;
  		ret = rmap_one(page, vma, address, arg);
  		if (ret != SWAP_AGAIN)
  			break;
  	}
  	/*
  	 * No nonlinear handling: being always shared, nonlinear vmas
  	 * never contain migration ptes.  Decide what to do about this
  	 * limitation to linear when we need rmap_walk() on nonlinear.
  	 */
  	spin_unlock(&mapping->i_mmap_lock);
  	return ret;
  }
  
  int rmap_walk(struct page *page, int (*rmap_one)(struct page *,
  		struct vm_area_struct *, unsigned long, void *), void *arg)
  {
  	VM_BUG_ON(!PageLocked(page));
  
  	if (unlikely(PageKsm(page)))
  		return rmap_walk_ksm(page, rmap_one, arg);
  	else if (PageAnon(page))
  		return rmap_walk_anon(page, rmap_one, arg);
  	else
  		return rmap_walk_file(page, rmap_one, arg);
  }
  #endif /* CONFIG_MIGRATION */
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
1501

e3390f67a   Naoya Horiguchi   hwpoison: rename ...
1502
  #ifdef CONFIG_HUGETLB_PAGE
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
1503
1504
1505
1506
1507
1508
1509
1510
1511
  /*
   * The following three functions are for anonymous (private mapped) hugepages.
   * Unlike common anonymous pages, anonymous hugepages have no accounting code
   * and no lru code, because we handle hugepages differently from common pages.
   */
  static void __hugepage_set_anon_rmap(struct page *page,
  	struct vm_area_struct *vma, unsigned long address, int exclusive)
  {
  	struct anon_vma *anon_vma = vma->anon_vma;
433abed6c   Naoya Horiguchi   hugetlb, rmap: al...
1512

0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
1513
  	BUG_ON(!anon_vma);
433abed6c   Naoya Horiguchi   hugetlb, rmap: al...
1514
1515
1516
1517
1518
  
  	if (PageAnon(page))
  		return;
  	if (!exclusive)
  		anon_vma = anon_vma->root;
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
1519
1520
1521
1522
1523
1524
1525
1526
1527
1528
  	anon_vma = (void *) anon_vma + PAGE_MAPPING_ANON;
  	page->mapping = (struct address_space *) anon_vma;
  	page->index = linear_page_index(vma, address);
  }
  
  void hugepage_add_anon_rmap(struct page *page,
  			    struct vm_area_struct *vma, unsigned long address)
  {
  	struct anon_vma *anon_vma = vma->anon_vma;
  	int first;
a850ea303   Naoya Horiguchi   hugetlb, rmap: ad...
1529
1530
  
  	BUG_ON(!PageLocked(page));
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
1531
1532
1533
1534
1535
1536
1537
1538
1539
1540
1541
1542
1543
1544
  	BUG_ON(!anon_vma);
  	BUG_ON(address < vma->vm_start || address >= vma->vm_end);
  	first = atomic_inc_and_test(&page->_mapcount);
  	if (first)
  		__hugepage_set_anon_rmap(page, vma, address, 0);
  }
  
  void hugepage_add_new_anon_rmap(struct page *page,
  			struct vm_area_struct *vma, unsigned long address)
  {
  	BUG_ON(address < vma->vm_start || address >= vma->vm_end);
  	atomic_set(&page->_mapcount, 0);
  	__hugepage_set_anon_rmap(page, vma, address, 1);
  }
e3390f67a   Naoya Horiguchi   hwpoison: rename ...
1545
  #endif /* CONFIG_HUGETLB_PAGE */