Blame view

mm/rmap.c 49 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
   *   mm->mmap_sem
   *     page->flags PG_locked (lock_page)
3d48ae45e   Peter Zijlstra   mm: Convert i_mma...
26
   *       mapping->i_mmap_mutex
5a505085f   Ingo Molnar   mm/rmap: Convert ...
27
   *         anon_vma->rwsem
82591e6ea   Nick Piggin   [PATCH] mm: more ...
28
29
30
31
32
   *           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)
250df6ed2   Dave Chinner   fs: protect inode...
33
   *               inode->i_lock (in set_page_dirty's __mark_inode_dirty)
f758eeabe   Christoph Hellwig   writeback: split ...
34
   *               bdi.wb->list_lock (in set_page_dirty's __mark_inode_dirty)
82591e6ea   Nick Piggin   [PATCH] mm: more ...
35
36
37
   *                 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,
f758eeabe   Christoph Hellwig   writeback: split ...
38
   *                           within bdi.wb->list_lock in __sync_single_inode)
6a46079cf   Andi Kleen   HWPOISON: The hig...
39
   *
5a505085f   Ingo Molnar   mm/rmap: Convert ...
40
   * anon_vma->rwsem,mapping->i_mutex      (memory_failure, collect_procs_anon)
9b679320a   Peter Zijlstra   mm/memory-failure...
41
   *   ->tasklist_lock
6a46079cf   Andi Kleen   HWPOISON: The hig...
42
   *     pte map lock
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
43
44
45
46
47
48
49
50
   */
  
  #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...
51
  #include <linux/ksm.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
52
53
  #include <linux/rmap.h>
  #include <linux/rcupdate.h>
b95f1b31b   Paul Gortmaker   mm: Map most file...
54
  #include <linux/export.h>
8a9f3ccd2   Balbir Singh   Memory controller...
55
  #include <linux/memcontrol.h>
cddb8a5c1   Andrea Arcangeli   mmu-notifiers: core
56
  #include <linux/mmu_notifier.h>
64cdd548f   KOSAKI Motohiro   mm: cleanup: remo...
57
  #include <linux/migrate.h>
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
58
  #include <linux/hugetlb.h>
ef5d437f7   Jan Kara   mm: fix XFS oops ...
59
  #include <linux/backing-dev.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
  
  static inline struct anon_vma *anon_vma_alloc(void)
  {
01d8b20de   Peter Zijlstra   mm: simplify anon...
68
69
70
71
72
73
74
75
76
77
78
79
80
  	struct anon_vma *anon_vma;
  
  	anon_vma = kmem_cache_alloc(anon_vma_cachep, GFP_KERNEL);
  	if (anon_vma) {
  		atomic_set(&anon_vma->refcount, 1);
  		/*
  		 * Initialise the anon_vma root to point to itself. If called
  		 * from fork, the root will be reset to the parents anon_vma.
  		 */
  		anon_vma->root = anon_vma;
  	}
  
  	return anon_vma;
fdd2e5f88   Adrian Bunk   make mm/rmap.c:an...
81
  }
01d8b20de   Peter Zijlstra   mm: simplify anon...
82
  static inline void anon_vma_free(struct anon_vma *anon_vma)
fdd2e5f88   Adrian Bunk   make mm/rmap.c:an...
83
  {
01d8b20de   Peter Zijlstra   mm: simplify anon...
84
  	VM_BUG_ON(atomic_read(&anon_vma->refcount));
88c22088b   Peter Zijlstra   mm: optimize page...
85
86
  
  	/*
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
87
  	 * Synchronize against page_lock_anon_vma_read() such that
88c22088b   Peter Zijlstra   mm: optimize page...
88
89
90
91
92
  	 * we can safely hold the lock without the anon_vma getting
  	 * freed.
  	 *
  	 * Relies on the full mb implied by the atomic_dec_and_test() from
  	 * put_anon_vma() against the acquire barrier implied by
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
93
  	 * down_read_trylock() from page_lock_anon_vma_read(). This orders:
88c22088b   Peter Zijlstra   mm: optimize page...
94
  	 *
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
95
96
  	 * page_lock_anon_vma_read()	VS	put_anon_vma()
  	 *   down_read_trylock()		  atomic_dec_and_test()
88c22088b   Peter Zijlstra   mm: optimize page...
97
  	 *   LOCK				  MB
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
98
  	 *   atomic_read()			  rwsem_is_locked()
88c22088b   Peter Zijlstra   mm: optimize page...
99
100
101
102
  	 *
  	 * LOCK should suffice since the actual taking of the lock must
  	 * happen _before_ what follows.
  	 */
7f39dda9d   Hugh Dickins   mm: fix sleeping ...
103
  	might_sleep();
5a505085f   Ingo Molnar   mm/rmap: Convert ...
104
  	if (rwsem_is_locked(&anon_vma->root->rwsem)) {
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
105
  		anon_vma_lock_write(anon_vma);
08b52706d   Konstantin Khlebnikov   mm/rmap: rename a...
106
  		anon_vma_unlock_write(anon_vma);
88c22088b   Peter Zijlstra   mm: optimize page...
107
  	}
fdd2e5f88   Adrian Bunk   make mm/rmap.c:an...
108
109
  	kmem_cache_free(anon_vma_cachep, anon_vma);
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
110

dd34739c0   Linus Torvalds   mm: avoid anon_vm...
111
  static inline struct anon_vma_chain *anon_vma_chain_alloc(gfp_t gfp)
5beb49305   Rik van Riel   mm: change anon_v...
112
  {
dd34739c0   Linus Torvalds   mm: avoid anon_vm...
113
  	return kmem_cache_alloc(anon_vma_chain_cachep, gfp);
5beb49305   Rik van Riel   mm: change anon_v...
114
  }
e574b5fd2   Namhyung Kim   rmap: make anon_v...
115
  static void anon_vma_chain_free(struct anon_vma_chain *anon_vma_chain)
5beb49305   Rik van Riel   mm: change anon_v...
116
117
118
  {
  	kmem_cache_free(anon_vma_chain_cachep, anon_vma_chain);
  }
6583a8430   Kautuk Consul   rmap: anon_vma_pr...
119
120
121
122
123
124
125
  static void anon_vma_chain_link(struct vm_area_struct *vma,
  				struct anon_vma_chain *avc,
  				struct anon_vma *anon_vma)
  {
  	avc->vma = vma;
  	avc->anon_vma = anon_vma;
  	list_add(&avc->same_vma, &vma->anon_vma_chain);
bf181b9f9   Michel Lespinasse   mm anon rmap: rep...
126
  	anon_vma_interval_tree_insert(avc, &anon_vma->rb_root);
6583a8430   Kautuk Consul   rmap: anon_vma_pr...
127
  }
d9d332e08   Linus Torvalds   anon_vma_prepare:...
128
129
130
131
132
133
134
135
136
  /**
   * 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
23a0790af   Figo.zhang   mm/rmap.c: fix co...
137
   * not we either need to find an adjacent mapping that we
d9d332e08   Linus Torvalds   anon_vma_prepare:...
138
139
140
141
142
   * 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
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
143
   * optimistically looked up an anon_vma in page_lock_anon_vma_read()
d9d332e08   Linus Torvalds   anon_vma_prepare:...
144
145
146
147
148
149
150
151
152
153
154
   * 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
155
156
157
  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...
158
  	struct anon_vma_chain *avc;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
159
160
161
162
  
  	might_sleep();
  	if (unlikely(!anon_vma)) {
  		struct mm_struct *mm = vma->vm_mm;
d9d332e08   Linus Torvalds   anon_vma_prepare:...
163
  		struct anon_vma *allocated;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
164

dd34739c0   Linus Torvalds   mm: avoid anon_vm...
165
  		avc = anon_vma_chain_alloc(GFP_KERNEL);
5beb49305   Rik van Riel   mm: change anon_v...
166
167
  		if (!avc)
  			goto out_enomem;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
168
  		anon_vma = find_mergeable_anon_vma(vma);
d9d332e08   Linus Torvalds   anon_vma_prepare:...
169
170
  		allocated = NULL;
  		if (!anon_vma) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
171
172
  			anon_vma = anon_vma_alloc();
  			if (unlikely(!anon_vma))
5beb49305   Rik van Riel   mm: change anon_v...
173
  				goto out_enomem_free_avc;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
174
  			allocated = anon_vma;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
175
  		}
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
176
  		anon_vma_lock_write(anon_vma);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
177
178
179
180
  		/* page_table_lock to protect against threads */
  		spin_lock(&mm->page_table_lock);
  		if (likely(!vma->anon_vma)) {
  			vma->anon_vma = anon_vma;
6583a8430   Kautuk Consul   rmap: anon_vma_pr...
181
  			anon_vma_chain_link(vma, avc, anon_vma);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
182
  			allocated = NULL;
31f2b0ebc   Oleg Nesterov   rmap: anon_vma_pr...
183
  			avc = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
184
185
  		}
  		spin_unlock(&mm->page_table_lock);
08b52706d   Konstantin Khlebnikov   mm/rmap: rename a...
186
  		anon_vma_unlock_write(anon_vma);
31f2b0ebc   Oleg Nesterov   rmap: anon_vma_pr...
187
188
  
  		if (unlikely(allocated))
01d8b20de   Peter Zijlstra   mm: simplify anon...
189
  			put_anon_vma(allocated);
31f2b0ebc   Oleg Nesterov   rmap: anon_vma_pr...
190
  		if (unlikely(avc))
5beb49305   Rik van Riel   mm: change anon_v...
191
  			anon_vma_chain_free(avc);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
192
193
  	}
  	return 0;
5beb49305   Rik van Riel   mm: change anon_v...
194
195
196
197
198
  
   out_enomem_free_avc:
  	anon_vma_chain_free(avc);
   out_enomem:
  	return -ENOMEM;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
199
  }
bb4aa3967   Linus Torvalds   mm: avoid repeate...
200
201
202
203
204
205
206
207
208
209
210
211
212
  /*
   * This is a useful helper function for locking the anon_vma root as
   * we traverse the vma->anon_vma_chain, looping over anon_vma's that
   * have the same vma.
   *
   * Such anon_vma's should have the same root, so you'd expect to see
   * just a single mutex_lock for the whole traversal.
   */
  static inline struct anon_vma *lock_anon_vma_root(struct anon_vma *root, struct anon_vma *anon_vma)
  {
  	struct anon_vma *new_root = anon_vma->root;
  	if (new_root != root) {
  		if (WARN_ON_ONCE(root))
5a505085f   Ingo Molnar   mm/rmap: Convert ...
213
  			up_write(&root->rwsem);
bb4aa3967   Linus Torvalds   mm: avoid repeate...
214
  		root = new_root;
5a505085f   Ingo Molnar   mm/rmap: Convert ...
215
  		down_write(&root->rwsem);
bb4aa3967   Linus Torvalds   mm: avoid repeate...
216
217
218
219
220
221
222
  	}
  	return root;
  }
  
  static inline void unlock_anon_vma_root(struct anon_vma *root)
  {
  	if (root)
5a505085f   Ingo Molnar   mm/rmap: Convert ...
223
  		up_write(&root->rwsem);
bb4aa3967   Linus Torvalds   mm: avoid repeate...
224
  }
5beb49305   Rik van Riel   mm: change anon_v...
225
226
227
228
229
  /*
   * 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
230
  {
5beb49305   Rik van Riel   mm: change anon_v...
231
  	struct anon_vma_chain *avc, *pavc;
bb4aa3967   Linus Torvalds   mm: avoid repeate...
232
  	struct anon_vma *root = NULL;
5beb49305   Rik van Riel   mm: change anon_v...
233

646d87b48   Linus Torvalds   anon_vma: clone t...
234
  	list_for_each_entry_reverse(pavc, &src->anon_vma_chain, same_vma) {
bb4aa3967   Linus Torvalds   mm: avoid repeate...
235
  		struct anon_vma *anon_vma;
dd34739c0   Linus Torvalds   mm: avoid anon_vm...
236
237
238
239
240
241
242
243
  		avc = anon_vma_chain_alloc(GFP_NOWAIT | __GFP_NOWARN);
  		if (unlikely(!avc)) {
  			unlock_anon_vma_root(root);
  			root = NULL;
  			avc = anon_vma_chain_alloc(GFP_KERNEL);
  			if (!avc)
  				goto enomem_failure;
  		}
bb4aa3967   Linus Torvalds   mm: avoid repeate...
244
245
246
  		anon_vma = pavc->anon_vma;
  		root = lock_anon_vma_root(root, anon_vma);
  		anon_vma_chain_link(dst, avc, anon_vma);
5beb49305   Rik van Riel   mm: change anon_v...
247
  	}
bb4aa3967   Linus Torvalds   mm: avoid repeate...
248
  	unlock_anon_vma_root(root);
5beb49305   Rik van Riel   mm: change anon_v...
249
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
250

5beb49305   Rik van Riel   mm: change anon_v...
251
252
253
   enomem_failure:
  	unlink_anon_vmas(dst);
  	return -ENOMEM;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
254
  }
5beb49305   Rik van Riel   mm: change anon_v...
255
256
257
258
259
260
  /*
   * 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
261
  {
5beb49305   Rik van Riel   mm: change anon_v...
262
263
  	struct anon_vma_chain *avc;
  	struct anon_vma *anon_vma;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
264

5beb49305   Rik van Riel   mm: change anon_v...
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
  	/* 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;
dd34739c0   Linus Torvalds   mm: avoid anon_vm...
280
  	avc = anon_vma_chain_alloc(GFP_KERNEL);
5beb49305   Rik van Riel   mm: change anon_v...
281
282
  	if (!avc)
  		goto out_error_free_anon_vma;
5c341ee1d   Rik van Riel   mm: track the roo...
283
284
285
286
287
288
  
  	/*
  	 * 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...
289
  	/*
01d8b20de   Peter Zijlstra   mm: simplify anon...
290
291
292
  	 * With 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.
76545066c   Rik van Riel   mm: extend KSM re...
293
294
  	 */
  	get_anon_vma(anon_vma->root);
5beb49305   Rik van Riel   mm: change anon_v...
295
296
  	/* Mark this anon_vma as the one where our new (COWed) pages go. */
  	vma->anon_vma = anon_vma;
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
297
  	anon_vma_lock_write(anon_vma);
5c341ee1d   Rik van Riel   mm: track the roo...
298
  	anon_vma_chain_link(vma, avc, anon_vma);
08b52706d   Konstantin Khlebnikov   mm/rmap: rename a...
299
  	anon_vma_unlock_write(anon_vma);
5beb49305   Rik van Riel   mm: change anon_v...
300
301
302
303
  
  	return 0;
  
   out_error_free_anon_vma:
01d8b20de   Peter Zijlstra   mm: simplify anon...
304
  	put_anon_vma(anon_vma);
5beb49305   Rik van Riel   mm: change anon_v...
305
   out_error:
4946d54cb   Rik van Riel   rmap: fix anon_vm...
306
  	unlink_anon_vmas(vma);
5beb49305   Rik van Riel   mm: change anon_v...
307
  	return -ENOMEM;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
308
  }
5beb49305   Rik van Riel   mm: change anon_v...
309
310
311
  void unlink_anon_vmas(struct vm_area_struct *vma)
  {
  	struct anon_vma_chain *avc, *next;
eee2acbae   Peter Zijlstra   mm: avoid repeate...
312
  	struct anon_vma *root = NULL;
5beb49305   Rik van Riel   mm: change anon_v...
313

5c341ee1d   Rik van Riel   mm: track the roo...
314
315
316
317
  	/*
  	 * 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...
318
  	list_for_each_entry_safe(avc, next, &vma->anon_vma_chain, same_vma) {
eee2acbae   Peter Zijlstra   mm: avoid repeate...
319
320
321
  		struct anon_vma *anon_vma = avc->anon_vma;
  
  		root = lock_anon_vma_root(root, anon_vma);
bf181b9f9   Michel Lespinasse   mm anon rmap: rep...
322
  		anon_vma_interval_tree_remove(avc, &anon_vma->rb_root);
eee2acbae   Peter Zijlstra   mm: avoid repeate...
323
324
325
326
327
  
  		/*
  		 * Leave empty anon_vmas on the list - we'll need
  		 * to free them outside the lock.
  		 */
bf181b9f9   Michel Lespinasse   mm anon rmap: rep...
328
  		if (RB_EMPTY_ROOT(&anon_vma->rb_root))
eee2acbae   Peter Zijlstra   mm: avoid repeate...
329
330
331
332
333
334
335
336
337
338
  			continue;
  
  		list_del(&avc->same_vma);
  		anon_vma_chain_free(avc);
  	}
  	unlock_anon_vma_root(root);
  
  	/*
  	 * Iterate the list once more, it now only contains empty and unlinked
  	 * anon_vmas, destroy them. Could not do before due to __put_anon_vma()
5a505085f   Ingo Molnar   mm/rmap: Convert ...
339
  	 * needing to write-acquire the anon_vma->root->rwsem.
eee2acbae   Peter Zijlstra   mm: avoid repeate...
340
341
342
343
344
  	 */
  	list_for_each_entry_safe(avc, next, &vma->anon_vma_chain, same_vma) {
  		struct anon_vma *anon_vma = avc->anon_vma;
  
  		put_anon_vma(anon_vma);
5beb49305   Rik van Riel   mm: change anon_v...
345
346
347
348
  		list_del(&avc->same_vma);
  		anon_vma_chain_free(avc);
  	}
  }
51cc50685   Alexey Dobriyan   SL*B: drop kmem c...
349
  static void anon_vma_ctor(void *data)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
350
  {
a35afb830   Christoph Lameter   Remove SLAB_CTOR_...
351
  	struct anon_vma *anon_vma = data;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
352

5a505085f   Ingo Molnar   mm/rmap: Convert ...
353
  	init_rwsem(&anon_vma->rwsem);
83813267c   Peter Zijlstra   mm: move anon_vma...
354
  	atomic_set(&anon_vma->refcount, 0);
bf181b9f9   Michel Lespinasse   mm anon rmap: rep...
355
  	anon_vma->rb_root = RB_ROOT;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
356
357
358
359
360
  }
  
  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...
361
  			0, SLAB_DESTROY_BY_RCU|SLAB_PANIC, anon_vma_ctor);
5beb49305   Rik van Riel   mm: change anon_v...
362
  	anon_vma_chain_cachep = KMEM_CACHE(anon_vma_chain, SLAB_PANIC);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
363
364
365
  }
  
  /*
6111e4ca6   Peter Zijlstra   mm: improve page_...
366
367
368
369
370
371
372
373
374
   * Getting a lock on a stable anon_vma from a page off the LRU is tricky!
   *
   * Since there is no serialization what so ever against page_remove_rmap()
   * the best this function can do is return a locked anon_vma that might
   * have been relevant to this page.
   *
   * The page might have been remapped to a different anon_vma or the anon_vma
   * returned may already be freed (and even reused).
   *
bc658c960   Peter Zijlstra   mm, rmap: Add yet...
375
376
377
378
379
   * In case it was remapped to a different anon_vma, the new anon_vma will be a
   * child of the old anon_vma, and the anon_vma lifetime rules will therefore
   * ensure that any anon_vma obtained from the page will still be valid for as
   * long as we observe page_mapped() [ hence all those page_mapped() tests ].
   *
6111e4ca6   Peter Zijlstra   mm: improve page_...
380
381
382
383
384
385
386
   * All users of this function must be very careful when walking the anon_vma
   * chain and verify that the page in question is indeed mapped in it
   * [ something equivalent to page_mapped_in_vma() ].
   *
   * Since anon_vma's slab is DESTROY_BY_RCU and we know from page_remove_rmap()
   * that the anon_vma pointer from page->mapping is valid if there is a
   * mapcount, we can dereference the anon_vma after observing those.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
387
   */
746b18d42   Peter Zijlstra   mm: use refcounts...
388
  struct anon_vma *page_get_anon_vma(struct page *page)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
389
  {
746b18d42   Peter Zijlstra   mm: use refcounts...
390
  	struct anon_vma *anon_vma = NULL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
391
392
393
  	unsigned long anon_mapping;
  
  	rcu_read_lock();
80e148226   Hugh Dickins   ksm: share anon p...
394
  	anon_mapping = (unsigned long) ACCESS_ONCE(page->mapping);
3ca7b3c5b   Hugh Dickins   mm: define PAGE_M...
395
  	if ((anon_mapping & PAGE_MAPPING_FLAGS) != PAGE_MAPPING_ANON)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
396
397
398
399
400
  		goto out;
  	if (!page_mapped(page))
  		goto out;
  
  	anon_vma = (struct anon_vma *) (anon_mapping - PAGE_MAPPING_ANON);
746b18d42   Peter Zijlstra   mm: use refcounts...
401
402
403
404
  	if (!atomic_inc_not_zero(&anon_vma->refcount)) {
  		anon_vma = NULL;
  		goto out;
  	}
f18194275   Hugh Dickins   mm: fix hang on a...
405
406
407
  
  	/*
  	 * If this page is still mapped, then its anon_vma cannot have been
746b18d42   Peter Zijlstra   mm: use refcounts...
408
409
410
411
  	 * 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 atomic_inc_not_zero()
  	 * above cannot corrupt).
f18194275   Hugh Dickins   mm: fix hang on a...
412
  	 */
746b18d42   Peter Zijlstra   mm: use refcounts...
413
  	if (!page_mapped(page)) {
7f39dda9d   Hugh Dickins   mm: fix sleeping ...
414
  		rcu_read_unlock();
746b18d42   Peter Zijlstra   mm: use refcounts...
415
  		put_anon_vma(anon_vma);
7f39dda9d   Hugh Dickins   mm: fix sleeping ...
416
  		return NULL;
746b18d42   Peter Zijlstra   mm: use refcounts...
417
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
418
419
  out:
  	rcu_read_unlock();
746b18d42   Peter Zijlstra   mm: use refcounts...
420
421
422
  
  	return anon_vma;
  }
88c22088b   Peter Zijlstra   mm: optimize page...
423
424
425
426
427
428
429
  /*
   * Similar to page_get_anon_vma() except it locks the anon_vma.
   *
   * Its a little more complex as it tries to keep the fast path to a single
   * atomic op -- the trylock. If we fail the trylock, we fall back to getting a
   * reference like with page_get_anon_vma() and then block on the mutex.
   */
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
430
  struct anon_vma *page_lock_anon_vma_read(struct page *page)
746b18d42   Peter Zijlstra   mm: use refcounts...
431
  {
88c22088b   Peter Zijlstra   mm: optimize page...
432
  	struct anon_vma *anon_vma = NULL;
eee0f252c   Hugh Dickins   mm: fix page_lock...
433
  	struct anon_vma *root_anon_vma;
88c22088b   Peter Zijlstra   mm: optimize page...
434
  	unsigned long anon_mapping;
746b18d42   Peter Zijlstra   mm: use refcounts...
435

88c22088b   Peter Zijlstra   mm: optimize page...
436
437
438
439
440
441
442
443
  	rcu_read_lock();
  	anon_mapping = (unsigned long) ACCESS_ONCE(page->mapping);
  	if ((anon_mapping & PAGE_MAPPING_FLAGS) != PAGE_MAPPING_ANON)
  		goto out;
  	if (!page_mapped(page))
  		goto out;
  
  	anon_vma = (struct anon_vma *) (anon_mapping - PAGE_MAPPING_ANON);
eee0f252c   Hugh Dickins   mm: fix page_lock...
444
  	root_anon_vma = ACCESS_ONCE(anon_vma->root);
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
445
  	if (down_read_trylock(&root_anon_vma->rwsem)) {
88c22088b   Peter Zijlstra   mm: optimize page...
446
  		/*
eee0f252c   Hugh Dickins   mm: fix page_lock...
447
448
  		 * If the page is still mapped, then this anon_vma is still
  		 * its anon_vma, and holding the mutex ensures that it will
bc658c960   Peter Zijlstra   mm, rmap: Add yet...
449
  		 * not go away, see anon_vma_free().
88c22088b   Peter Zijlstra   mm: optimize page...
450
  		 */
eee0f252c   Hugh Dickins   mm: fix page_lock...
451
  		if (!page_mapped(page)) {
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
452
  			up_read(&root_anon_vma->rwsem);
88c22088b   Peter Zijlstra   mm: optimize page...
453
454
455
456
  			anon_vma = NULL;
  		}
  		goto out;
  	}
746b18d42   Peter Zijlstra   mm: use refcounts...
457

88c22088b   Peter Zijlstra   mm: optimize page...
458
459
460
461
462
463
464
  	/* trylock failed, we got to sleep */
  	if (!atomic_inc_not_zero(&anon_vma->refcount)) {
  		anon_vma = NULL;
  		goto out;
  	}
  
  	if (!page_mapped(page)) {
7f39dda9d   Hugh Dickins   mm: fix sleeping ...
465
  		rcu_read_unlock();
88c22088b   Peter Zijlstra   mm: optimize page...
466
  		put_anon_vma(anon_vma);
7f39dda9d   Hugh Dickins   mm: fix sleeping ...
467
  		return NULL;
88c22088b   Peter Zijlstra   mm: optimize page...
468
469
470
471
  	}
  
  	/* we pinned the anon_vma, its safe to sleep */
  	rcu_read_unlock();
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
472
  	anon_vma_lock_read(anon_vma);
88c22088b   Peter Zijlstra   mm: optimize page...
473
474
475
476
477
  
  	if (atomic_dec_and_test(&anon_vma->refcount)) {
  		/*
  		 * Oops, we held the last refcount, release the lock
  		 * and bail -- can't simply use put_anon_vma() because
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
478
  		 * we'll deadlock on the anon_vma_lock_write() recursion.
88c22088b   Peter Zijlstra   mm: optimize page...
479
  		 */
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
480
  		anon_vma_unlock_read(anon_vma);
88c22088b   Peter Zijlstra   mm: optimize page...
481
482
483
484
485
486
487
488
  		__put_anon_vma(anon_vma);
  		anon_vma = NULL;
  	}
  
  	return anon_vma;
  
  out:
  	rcu_read_unlock();
746b18d42   Peter Zijlstra   mm: use refcounts...
489
  	return anon_vma;
34bbd7040   Oleg Nesterov   [PATCH] adapt pag...
490
  }
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
491
  void page_unlock_anon_vma_read(struct anon_vma *anon_vma)
34bbd7040   Oleg Nesterov   [PATCH] adapt pag...
492
  {
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
493
  	anon_vma_unlock_read(anon_vma);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
494
495
496
  }
  
  /*
3ad33b243   Lee Schermerhorn   Migration: find c...
497
   * At what user virtual address is page expected in @vma?
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
498
   */
86c2ad199   Michel Lespinasse   mm rmap: remove v...
499
500
  static inline unsigned long
  __vma_address(struct page *page, struct vm_area_struct *vma)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
501
502
  {
  	pgoff_t pgoff = page->index << (PAGE_CACHE_SHIFT - PAGE_SHIFT);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
503

0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
504
505
  	if (unlikely(is_vm_hugetlb_page(vma)))
  		pgoff = page->index << huge_page_order(page_hstate(page));
86c2ad199   Michel Lespinasse   mm rmap: remove v...
506
507
508
509
510
511
512
513
514
515
516
  
  	return vma->vm_start + ((pgoff - vma->vm_pgoff) << PAGE_SHIFT);
  }
  
  inline unsigned long
  vma_address(struct page *page, struct vm_area_struct *vma)
  {
  	unsigned long address = __vma_address(page, vma);
  
  	/* page should be within @vma mapping range */
  	VM_BUG_ON(address < vma->vm_start || address >= vma->vm_end);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
517
518
519
520
  	return address;
  }
  
  /*
bf89c8c86   Huang Shijie   mm/rmap.c: fix co...
521
   * At what user virtual address is page expected in vma?
ab941e0ff   Naoya Horiguchi   rmap: remove anon...
522
   * Caller should check the page is actually part of the vma.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
523
524
525
   */
  unsigned long page_address_in_vma(struct page *page, struct vm_area_struct *vma)
  {
86c2ad199   Michel Lespinasse   mm rmap: remove v...
526
  	unsigned long address;
21d0d443c   Andrea Arcangeli   rmap: resurrect p...
527
  	if (PageAnon(page)) {
4829b906c   Hugh Dickins   ksm: fix page_add...
528
529
530
531
532
533
534
  		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...
535
536
  			return -EFAULT;
  	} else if (page->mapping && !(vma->vm_flags & VM_NONLINEAR)) {
ee498ed73   Hugh Dickins   [PATCH] unpaged: ...
537
538
  		if (!vma->vm_file ||
  		    vma->vm_file->f_mapping != page->mapping)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
539
540
541
  			return -EFAULT;
  	} else
  		return -EFAULT;
86c2ad199   Michel Lespinasse   mm rmap: remove v...
542
543
544
545
  	address = __vma_address(page, vma);
  	if (unlikely(address < vma->vm_start || address >= vma->vm_end))
  		return -EFAULT;
  	return address;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
546
  }
6219049ae   Bob Liu   mm: introduce mm_...
547
548
549
550
551
  pmd_t *mm_find_pmd(struct mm_struct *mm, unsigned long address)
  {
  	pgd_t *pgd;
  	pud_t *pud;
  	pmd_t *pmd = NULL;
f72e7dcdd   Hugh Dickins   mm: let mm_find_p...
552
  	pmd_t pmde;
6219049ae   Bob Liu   mm: introduce mm_...
553
554
555
556
557
558
559
560
561
562
  
  	pgd = pgd_offset(mm, address);
  	if (!pgd_present(*pgd))
  		goto out;
  
  	pud = pud_offset(pgd, address);
  	if (!pud_present(*pud))
  		goto out;
  
  	pmd = pmd_offset(pud, address);
f72e7dcdd   Hugh Dickins   mm: let mm_find_p...
563
564
565
566
567
568
569
  	/*
  	 * Some THP functions use the sequence pmdp_clear_flush(), set_pmd_at()
  	 * without holding anon_vma lock for write.  So when looking for a
  	 * genuine pmde (in which to find pte), test present and !THP together.
  	 */
  	pmde = ACCESS_ONCE(*pmd);
  	if (!pmd_present(pmde) || pmd_trans_huge(pmde))
6219049ae   Bob Liu   mm: introduce mm_...
570
571
572
573
  		pmd = NULL;
  out:
  	return pmd;
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
574
  /*
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
575
576
   * Check that @page is mapped at @address into @mm.
   *
479db0bf4   Nick Piggin   mm: dirty page tr...
577
578
579
580
   * 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...
581
   * On success returns with pte mapped and locked.
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
582
   */
e9a81a821   Namhyung Kim   rmap: wrap page_c...
583
  pte_t *__page_check_address(struct page *page, struct mm_struct *mm,
479db0bf4   Nick Piggin   mm: dirty page tr...
584
  			  unsigned long address, spinlock_t **ptlp, int sync)
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
585
  {
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
586
587
  	pmd_t *pmd;
  	pte_t *pte;
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
588
  	spinlock_t *ptl;
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
589

0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
590
  	if (unlikely(PageHuge(page))) {
98398c32f   Jianguo Wu   mm/hugetlb: check...
591
  		/* when pud is not present, pte will be NULL */
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
592
  		pte = huge_pte_offset(mm, address);
98398c32f   Jianguo Wu   mm/hugetlb: check...
593
594
  		if (!pte)
  			return NULL;
cb900f412   Kirill A. Shutemov   mm, hugetlb: conv...
595
  		ptl = huge_pte_lockptr(page_hstate(page), mm, pte);
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
596
597
  		goto check;
  	}
6219049ae   Bob Liu   mm: introduce mm_...
598
599
  	pmd = mm_find_pmd(mm, address);
  	if (!pmd)
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
600
  		return NULL;
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
601
602
  	pte = pte_offset_map(pmd, address);
  	/* Make a quick check before getting the lock */
479db0bf4   Nick Piggin   mm: dirty page tr...
603
  	if (!sync && !pte_present(*pte)) {
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
604
605
606
  		pte_unmap(pte);
  		return NULL;
  	}
4c21e2f24   Hugh Dickins   [PATCH] mm: split...
607
  	ptl = pte_lockptr(mm, pmd);
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
608
  check:
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
609
610
611
612
  	spin_lock(ptl);
  	if (pte_present(*pte) && page_to_pfn(page) == pte_pfn(*pte)) {
  		*ptlp = ptl;
  		return pte;
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
613
  	}
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
614
615
  	pte_unmap_unlock(pte, ptl);
  	return NULL;
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
616
  }
b291f0003   Nick Piggin   mlock: mlocked pa...
617
618
619
620
621
622
623
624
625
  /**
   * 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...
626
  int page_mapped_in_vma(struct page *page, struct vm_area_struct *vma)
b291f0003   Nick Piggin   mlock: mlocked pa...
627
628
629
630
  {
  	unsigned long address;
  	pte_t *pte;
  	spinlock_t *ptl;
86c2ad199   Michel Lespinasse   mm rmap: remove v...
631
632
  	address = __vma_address(page, vma);
  	if (unlikely(address < vma->vm_start || address >= vma->vm_end))
b291f0003   Nick Piggin   mlock: mlocked pa...
633
634
635
636
637
638
639
640
  		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;
  }
9f32624be   Joonsoo Kim   mm/rmap: use rmap...
641
642
643
644
645
646
  struct page_referenced_arg {
  	int mapcount;
  	int referenced;
  	unsigned long vm_flags;
  	struct mem_cgroup *memcg;
  };
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
647
  /*
9f32624be   Joonsoo Kim   mm/rmap: use rmap...
648
   * arg: page_referenced_arg will be passed
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
649
   */
ac7695012   Kirill A. Shutemov   mm/rmap.c: make p...
650
  static int page_referenced_one(struct page *page, struct vm_area_struct *vma,
9f32624be   Joonsoo Kim   mm/rmap: use rmap...
651
  			unsigned long address, void *arg)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
652
653
  {
  	struct mm_struct *mm = vma->vm_mm;
117b0791a   Kirill A. Shutemov   mm, thp: move ptl...
654
  	spinlock_t *ptl;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
655
  	int referenced = 0;
9f32624be   Joonsoo Kim   mm/rmap: use rmap...
656
  	struct page_referenced_arg *pra = arg;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
657

71e3aac07   Andrea Arcangeli   thp: transparent ...
658
659
  	if (unlikely(PageTransHuge(page))) {
  		pmd_t *pmd;
2da28bfd9   Andrea Arcangeli   thp: fix page_ref...
660
661
662
663
  		/*
  		 * rmap might return false positives; we must filter
  		 * these out using page_check_address_pmd().
  		 */
71e3aac07   Andrea Arcangeli   thp: transparent ...
664
  		pmd = page_check_address_pmd(page, mm, address,
117b0791a   Kirill A. Shutemov   mm, thp: move ptl...
665
666
  					     PAGE_CHECK_ADDRESS_PMD_FLAG, &ptl);
  		if (!pmd)
9f32624be   Joonsoo Kim   mm/rmap: use rmap...
667
  			return SWAP_AGAIN;
2da28bfd9   Andrea Arcangeli   thp: fix page_ref...
668
669
  
  		if (vma->vm_flags & VM_LOCKED) {
117b0791a   Kirill A. Shutemov   mm, thp: move ptl...
670
  			spin_unlock(ptl);
9f32624be   Joonsoo Kim   mm/rmap: use rmap...
671
672
  			pra->vm_flags |= VM_LOCKED;
  			return SWAP_FAIL; /* To break the loop */
2da28bfd9   Andrea Arcangeli   thp: fix page_ref...
673
674
675
676
  		}
  
  		/* go ahead even if the pmd is pmd_trans_splitting() */
  		if (pmdp_clear_flush_young_notify(vma, address, pmd))
71e3aac07   Andrea Arcangeli   thp: transparent ...
677
  			referenced++;
117b0791a   Kirill A. Shutemov   mm, thp: move ptl...
678
  		spin_unlock(ptl);
71e3aac07   Andrea Arcangeli   thp: transparent ...
679
680
  	} else {
  		pte_t *pte;
71e3aac07   Andrea Arcangeli   thp: transparent ...
681

2da28bfd9   Andrea Arcangeli   thp: fix page_ref...
682
683
684
685
  		/*
  		 * rmap might return false positives; we must filter
  		 * these out using page_check_address().
  		 */
71e3aac07   Andrea Arcangeli   thp: transparent ...
686
687
  		pte = page_check_address(page, mm, address, &ptl, 0);
  		if (!pte)
9f32624be   Joonsoo Kim   mm/rmap: use rmap...
688
  			return SWAP_AGAIN;
71e3aac07   Andrea Arcangeli   thp: transparent ...
689

2da28bfd9   Andrea Arcangeli   thp: fix page_ref...
690
691
  		if (vma->vm_flags & VM_LOCKED) {
  			pte_unmap_unlock(pte, ptl);
9f32624be   Joonsoo Kim   mm/rmap: use rmap...
692
693
  			pra->vm_flags |= VM_LOCKED;
  			return SWAP_FAIL; /* To break the loop */
2da28bfd9   Andrea Arcangeli   thp: fix page_ref...
694
  		}
71e3aac07   Andrea Arcangeli   thp: transparent ...
695
696
697
698
699
700
701
702
  		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.
  			 */
64363aad5   Joe Perches   mm: remove unused...
703
  			if (likely(!(vma->vm_flags & VM_SEQ_READ)))
71e3aac07   Andrea Arcangeli   thp: transparent ...
704
705
706
707
  				referenced++;
  		}
  		pte_unmap_unlock(pte, ptl);
  	}
9f32624be   Joonsoo Kim   mm/rmap: use rmap...
708
709
710
  	if (referenced) {
  		pra->referenced++;
  		pra->vm_flags |= vma->vm_flags;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
711
  	}
34bbd7040   Oleg Nesterov   [PATCH] adapt pag...
712

9f32624be   Joonsoo Kim   mm/rmap: use rmap...
713
714
715
716
717
  	pra->mapcount--;
  	if (!pra->mapcount)
  		return SWAP_SUCCESS; /* To break the loop */
  
  	return SWAP_AGAIN;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
718
  }
9f32624be   Joonsoo Kim   mm/rmap: use rmap...
719
  static bool invalid_page_referenced_vma(struct vm_area_struct *vma, void *arg)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
720
  {
9f32624be   Joonsoo Kim   mm/rmap: use rmap...
721
722
  	struct page_referenced_arg *pra = arg;
  	struct mem_cgroup *memcg = pra->memcg;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
723

9f32624be   Joonsoo Kim   mm/rmap: use rmap...
724
725
  	if (!mm_match_cgroup(vma->vm_mm, memcg))
  		return true;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
726

9f32624be   Joonsoo Kim   mm/rmap: use rmap...
727
  	return false;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
728
729
730
731
732
733
  }
  
  /**
   * page_referenced - test if the page was referenced
   * @page: the page to test
   * @is_locked: caller holds lock on the page
72835c86c   Johannes Weiner   mm: unify remaini...
734
   * @memcg: target memory cgroup
6fe6b7e35   Wu Fengguang   vmscan: report vm...
735
   * @vm_flags: collect encountered vma->vm_flags who actually referenced the page
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
736
737
738
739
   *
   * 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...
740
741
  int page_referenced(struct page *page,
  		    int is_locked,
72835c86c   Johannes Weiner   mm: unify remaini...
742
  		    struct mem_cgroup *memcg,
6fe6b7e35   Wu Fengguang   vmscan: report vm...
743
  		    unsigned long *vm_flags)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
744
  {
9f32624be   Joonsoo Kim   mm/rmap: use rmap...
745
  	int ret;
5ad646880   Hugh Dickins   ksm: let shared p...
746
  	int we_locked = 0;
9f32624be   Joonsoo Kim   mm/rmap: use rmap...
747
748
749
750
751
752
753
754
755
  	struct page_referenced_arg pra = {
  		.mapcount = page_mapcount(page),
  		.memcg = memcg,
  	};
  	struct rmap_walk_control rwc = {
  		.rmap_one = page_referenced_one,
  		.arg = (void *)&pra,
  		.anon_lock = page_lock_anon_vma_read,
  	};
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
756

6fe6b7e35   Wu Fengguang   vmscan: report vm...
757
  	*vm_flags = 0;
9f32624be   Joonsoo Kim   mm/rmap: use rmap...
758
759
760
761
762
763
764
765
766
767
  	if (!page_mapped(page))
  		return 0;
  
  	if (!page_rmapping(page))
  		return 0;
  
  	if (!is_locked && (!PageAnon(page) || PageKsm(page))) {
  		we_locked = trylock_page(page);
  		if (!we_locked)
  			return 1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
768
  	}
9f32624be   Joonsoo Kim   mm/rmap: use rmap...
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
  
  	/*
  	 * If we are reclaiming on behalf of a cgroup, skip
  	 * counting on behalf of references from different
  	 * cgroups
  	 */
  	if (memcg) {
  		rwc.invalid_vma = invalid_page_referenced_vma;
  	}
  
  	ret = rmap_walk(page, &rwc);
  	*vm_flags = pra.vm_flags;
  
  	if (we_locked)
  		unlock_page(page);
  
  	return pra.referenced;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
786
  }
1cb1729b1   Hugh Dickins   mm: pass address ...
787
  static int page_mkclean_one(struct page *page, struct vm_area_struct *vma,
9853a407b   Joonsoo Kim   mm/rmap: use rmap...
788
  			    unsigned long address, void *arg)
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
789
790
  {
  	struct mm_struct *mm = vma->vm_mm;
c2fda5fed   Peter Zijlstra   [PATCH] Fix up pa...
791
  	pte_t *pte;
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
792
793
  	spinlock_t *ptl;
  	int ret = 0;
9853a407b   Joonsoo Kim   mm/rmap: use rmap...
794
  	int *cleaned = arg;
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
795

479db0bf4   Nick Piggin   mm: dirty page tr...
796
  	pte = page_check_address(page, mm, address, &ptl, 1);
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
797
798
  	if (!pte)
  		goto out;
c2fda5fed   Peter Zijlstra   [PATCH] Fix up pa...
799
800
  	if (pte_dirty(*pte) || pte_write(*pte)) {
  		pte_t entry;
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
801

c2fda5fed   Peter Zijlstra   [PATCH] Fix up pa...
802
  		flush_cache_page(vma, address, pte_pfn(*pte));
2ec74c3ef   Sagi Grimberg   mm: move all mmu ...
803
  		entry = ptep_clear_flush(vma, address, pte);
c2fda5fed   Peter Zijlstra   [PATCH] Fix up pa...
804
805
  		entry = pte_wrprotect(entry);
  		entry = pte_mkclean(entry);
d6e88e671   Al Viro   [PATCH] page_mkcl...
806
  		set_pte_at(mm, address, pte, entry);
c2fda5fed   Peter Zijlstra   [PATCH] Fix up pa...
807
808
  		ret = 1;
  	}
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
809

d08b3851d   Peter Zijlstra   [PATCH] mm: track...
810
  	pte_unmap_unlock(pte, ptl);
2ec74c3ef   Sagi Grimberg   mm: move all mmu ...
811

9853a407b   Joonsoo Kim   mm/rmap: use rmap...
812
  	if (ret) {
2ec74c3ef   Sagi Grimberg   mm: move all mmu ...
813
  		mmu_notifier_invalidate_page(mm, address);
9853a407b   Joonsoo Kim   mm/rmap: use rmap...
814
815
  		(*cleaned)++;
  	}
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
816
  out:
9853a407b   Joonsoo Kim   mm/rmap: use rmap...
817
  	return SWAP_AGAIN;
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
818
  }
9853a407b   Joonsoo Kim   mm/rmap: use rmap...
819
  static bool invalid_mkclean_vma(struct vm_area_struct *vma, void *arg)
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
820
  {
9853a407b   Joonsoo Kim   mm/rmap: use rmap...
821
  	if (vma->vm_flags & VM_SHARED)
871beb8c3   Fengguang Wu   mm/rmap: fix cocc...
822
  		return false;
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
823

871beb8c3   Fengguang Wu   mm/rmap: fix cocc...
824
  	return true;
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
825
826
827
828
  }
  
  int page_mkclean(struct page *page)
  {
9853a407b   Joonsoo Kim   mm/rmap: use rmap...
829
830
831
832
833
834
835
  	int cleaned = 0;
  	struct address_space *mapping;
  	struct rmap_walk_control rwc = {
  		.arg = (void *)&cleaned,
  		.rmap_one = page_mkclean_one,
  		.invalid_vma = invalid_mkclean_vma,
  	};
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
836
837
  
  	BUG_ON(!PageLocked(page));
9853a407b   Joonsoo Kim   mm/rmap: use rmap...
838
839
840
841
842
843
844
845
  	if (!page_mapped(page))
  		return 0;
  
  	mapping = page_mapping(page);
  	if (!mapping)
  		return 0;
  
  	rmap_walk(page, &rwc);
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
846

9853a407b   Joonsoo Kim   mm/rmap: use rmap...
847
  	return cleaned;
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
848
  }
60b59beaf   Jaya Kumar   fbdev: mm: Deferr...
849
  EXPORT_SYMBOL_GPL(page_mkclean);
d08b3851d   Peter Zijlstra   [PATCH] mm: track...
850

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
851
  /**
c44b67432   Rik van Riel   rmap: move exclus...
852
853
854
855
856
857
858
859
860
861
862
863
864
865
   * 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;
309381fea   Sasha Levin   mm: dump page whe...
866
  	VM_BUG_ON_PAGE(!PageLocked(page), page);
c44b67432   Rik van Riel   rmap: move exclus...
867
  	VM_BUG_ON(!anon_vma);
309381fea   Sasha Levin   mm: dump page whe...
868
  	VM_BUG_ON_PAGE(page->index != linear_page_index(vma, address), page);
c44b67432   Rik van Riel   rmap: move exclus...
869
870
871
872
873
874
  
  	anon_vma = (void *) anon_vma + PAGE_MAPPING_ANON;
  	page->mapping = (struct address_space *) anon_vma;
  }
  
  /**
4e1c19750   Andi Kleen   Clean up __page_s...
875
876
877
878
   * __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...
879
   * @exclusive:	the page is exclusively owned by the current process
9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
880
881
   */
  static void __page_set_anon_rmap(struct page *page,
e8a03feb5   Rik van Riel   rmap: add exclusi...
882
  	struct vm_area_struct *vma, unsigned long address, int exclusive)
9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
883
  {
e8a03feb5   Rik van Riel   rmap: add exclusi...
884
  	struct anon_vma *anon_vma = vma->anon_vma;
ea90002b0   Linus Torvalds   anonvma: when set...
885

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

4e1c19750   Andi Kleen   Clean up __page_s...
888
889
  	if (PageAnon(page))
  		return;
ea90002b0   Linus Torvalds   anonvma: when set...
890
  	/*
e8a03feb5   Rik van Riel   rmap: add exclusi...
891
892
893
  	 * 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...
894
  	 */
4e1c19750   Andi Kleen   Clean up __page_s...
895
  	if (!exclusive)
288468c33   Andrea Arcangeli   rmap: always use ...
896
  		anon_vma = anon_vma->root;
9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
897

9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
898
899
  	anon_vma = (void *) anon_vma + PAGE_MAPPING_ANON;
  	page->mapping = (struct address_space *) anon_vma;
9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
900
  	page->index = linear_page_index(vma, address);
9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
901
902
903
  }
  
  /**
43d8eac44   Randy Dunlap   mm: rmap kernel-d...
904
   * __page_check_anon_rmap - sanity check anonymous rmap addition
c97a9e10e   Nick Piggin   mm: more rmap che...
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
   * @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...
925
  	BUG_ON(page_anon_vma(page)->root != vma->anon_vma->root);
c97a9e10e   Nick Piggin   mm: more rmap che...
926
927
928
929
930
  	BUG_ON(page->index != linear_page_index(vma, address));
  #endif
  }
  
  /**
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
931
932
933
934
935
   * 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...
936
   * The caller needs to hold the pte lock, and the page must be locked in
80e148226   Hugh Dickins   ksm: share anon p...
937
938
939
   * 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
940
941
942
943
   */
  void page_add_anon_rmap(struct page *page,
  	struct vm_area_struct *vma, unsigned long address)
  {
ad8c2ee80   Rik van Riel   rmap: add exclusi...
944
945
946
947
948
949
950
951
952
953
954
  	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...
955
  	int first = atomic_inc_and_test(&page->_mapcount);
79134171d   Andrea Arcangeli   thp: transparent ...
956
  	if (first) {
bea04b073   Jianyu Zhan   mm: use the light...
957
958
959
960
961
962
  		/*
  		 * We use the irq-unsafe __{inc|mod}_zone_page_stat because
  		 * these counters are not modified in interrupt context, and
  		 * pte lock(a spinlock) is held, which implies preemption
  		 * disabled.
  		 */
3cd14fcd3   Kirill A. Shutemov   thp: account anon...
963
  		if (PageTransHuge(page))
79134171d   Andrea Arcangeli   thp: transparent ...
964
965
  			__inc_zone_page_state(page,
  					      NR_ANON_TRANSPARENT_HUGEPAGES);
3cd14fcd3   Kirill A. Shutemov   thp: account anon...
966
967
  		__mod_zone_page_state(page_zone(page), NR_ANON_PAGES,
  				hpage_nr_pages(page));
79134171d   Andrea Arcangeli   thp: transparent ...
968
  	}
5ad646880   Hugh Dickins   ksm: let shared p...
969
970
  	if (unlikely(PageKsm(page)))
  		return;
309381fea   Sasha Levin   mm: dump page whe...
971
  	VM_BUG_ON_PAGE(!PageLocked(page), page);
5dbe0af47   Hugh Dickins   mm: fix kernel BU...
972
  	/* address might be in next vma when migration races vma_adjust */
5ad646880   Hugh Dickins   ksm: let shared p...
973
  	if (first)
ad8c2ee80   Rik van Riel   rmap: add exclusi...
974
  		__page_set_anon_rmap(page, vma, address, exclusive);
69029cd55   KAMEZAWA Hiroyuki   memcg: remove ref...
975
  	else
c97a9e10e   Nick Piggin   mm: more rmap che...
976
  		__page_check_anon_rmap(page, vma, address);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
977
  }
43d8eac44   Randy Dunlap   mm: rmap kernel-d...
978
  /**
9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
979
980
981
982
983
984
985
   * 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...
986
   * Page does not have to be locked.
9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
987
988
989
990
   */
  void page_add_new_anon_rmap(struct page *page,
  	struct vm_area_struct *vma, unsigned long address)
  {
b5934c531   Hugh Dickins   mm: add_active_or...
991
  	VM_BUG_ON(address < vma->vm_start || address >= vma->vm_end);
cbf84b7ad   Hugh Dickins   mm: further clean...
992
993
  	SetPageSwapBacked(page);
  	atomic_set(&page->_mapcount, 0); /* increment count (starts at -1) */
3cd14fcd3   Kirill A. Shutemov   thp: account anon...
994
  	if (PageTransHuge(page))
79134171d   Andrea Arcangeli   thp: transparent ...
995
  		__inc_zone_page_state(page, NR_ANON_TRANSPARENT_HUGEPAGES);
3cd14fcd3   Kirill A. Shutemov   thp: account anon...
996
997
  	__mod_zone_page_state(page_zone(page), NR_ANON_PAGES,
  			hpage_nr_pages(page));
e8a03feb5   Rik van Riel   rmap: add exclusi...
998
  	__page_set_anon_rmap(page, vma, address, 1);
7ee07a44e   Jianyu Zhan   mm: fold mlocked_...
999
1000
1001
  
  	VM_BUG_ON_PAGE(PageLRU(page), page);
  	if (likely((vma->vm_flags & (VM_LOCKED | VM_SPECIAL)) != VM_LOCKED)) {
c53954a09   Mel Gorman   mm: remove lru pa...
1002
1003
  		SetPageActive(page);
  		lru_cache_add(page);
7ee07a44e   Jianyu Zhan   mm: fold mlocked_...
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
  		return;
  	}
  
  	if (!TestSetPageMlocked(page)) {
  		/*
  		 * We use the irq-unsafe __mod_zone_page_stat because this
  		 * counter is not modified from interrupt context, and the pte
  		 * lock is held(spinlock), which implies preemption disabled.
  		 */
  		__mod_zone_page_state(page_zone(page), NR_MLOCK,
  				    hpage_nr_pages(page));
  		count_vm_event(UNEVICTABLE_PGMLOCKED);
  	}
  	add_page_to_unevictable_list(page);
9617d95e6   Nick Piggin   [PATCH] mm: rmap ...
1018
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1019
1020
1021
1022
  /**
   * 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...
1023
   * The caller needs to hold the pte lock.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1024
1025
1026
   */
  void page_add_file_rmap(struct page *page)
  {
89c06bd52   KAMEZAWA Hiroyuki   memcg: use new lo...
1027
1028
1029
1030
  	bool locked;
  	unsigned long flags;
  
  	mem_cgroup_begin_update_page_stat(page, &locked, &flags);
d69b042f3   Balbir Singh   memcg: add file-b...
1031
  	if (atomic_inc_and_test(&page->_mapcount)) {
65ba55f50   Christoph Lameter   [PATCH] zoned vm ...
1032
  		__inc_zone_page_state(page, NR_FILE_MAPPED);
68b4876d9   Sha Zhengju   memcg: remove MEM...
1033
  		mem_cgroup_inc_page_stat(page, MEM_CGROUP_STAT_FILE_MAPPED);
d69b042f3   Balbir Singh   memcg: add file-b...
1034
  	}
89c06bd52   KAMEZAWA Hiroyuki   memcg: use new lo...
1035
  	mem_cgroup_end_update_page_stat(page, &locked, &flags);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1036
1037
1038
1039
1040
1041
  }
  
  /**
   * page_remove_rmap - take down pte mapping from a page
   * @page: page to remove mapping from
   *
b8072f099   Hugh Dickins   [PATCH] mm: updat...
1042
   * The caller needs to hold the pte lock.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1043
   */
edc315fd2   Hugh Dickins   badpage: remove v...
1044
  void page_remove_rmap(struct page *page)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1045
  {
89c06bd52   KAMEZAWA Hiroyuki   memcg: use new lo...
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
  	bool anon = PageAnon(page);
  	bool locked;
  	unsigned long flags;
  
  	/*
  	 * The anon case has no mem_cgroup page_stat to update; but may
  	 * uncharge_page() below, where the lock ordering can deadlock if
  	 * we hold the lock against page_stat move: so avoid it on anon.
  	 */
  	if (!anon)
  		mem_cgroup_begin_update_page_stat(page, &locked, &flags);
b904dcfed   KOSAKI Motohiro   mm: clean up page...
1057
1058
  	/* page still mapped by someone else? */
  	if (!atomic_add_negative(-1, &page->_mapcount))
89c06bd52   KAMEZAWA Hiroyuki   memcg: use new lo...
1059
  		goto out;
b904dcfed   KOSAKI Motohiro   mm: clean up page...
1060
1061
  
  	/*
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
1062
1063
  	 * Hugepages are not counted in NR_ANON_PAGES nor NR_FILE_MAPPED
  	 * and not charged by memcg for now.
bea04b073   Jianyu Zhan   mm: use the light...
1064
1065
1066
1067
1068
  	 *
  	 * We use the irq-unsafe __{inc|mod}_zone_page_stat because
  	 * these counters are not modified in interrupt context, and
  	 * these counters are not modified in interrupt context, and
  	 * pte lock(a spinlock) is held, which implies preemption disabled.
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
1069
1070
  	 */
  	if (unlikely(PageHuge(page)))
89c06bd52   KAMEZAWA Hiroyuki   memcg: use new lo...
1071
1072
  		goto out;
  	if (anon) {
b904dcfed   KOSAKI Motohiro   mm: clean up page...
1073
  		mem_cgroup_uncharge_page(page);
3cd14fcd3   Kirill A. Shutemov   thp: account anon...
1074
  		if (PageTransHuge(page))
79134171d   Andrea Arcangeli   thp: transparent ...
1075
1076
  			__dec_zone_page_state(page,
  					      NR_ANON_TRANSPARENT_HUGEPAGES);
3cd14fcd3   Kirill A. Shutemov   thp: account anon...
1077
1078
  		__mod_zone_page_state(page_zone(page), NR_ANON_PAGES,
  				-hpage_nr_pages(page));
b904dcfed   KOSAKI Motohiro   mm: clean up page...
1079
1080
  	} else {
  		__dec_zone_page_state(page, NR_FILE_MAPPED);
68b4876d9   Sha Zhengju   memcg: remove MEM...
1081
  		mem_cgroup_dec_page_stat(page, MEM_CGROUP_STAT_FILE_MAPPED);
e6c509f85   Hugh Dickins   mm: use clear_pag...
1082
  		mem_cgroup_end_update_page_stat(page, &locked, &flags);
b904dcfed   KOSAKI Motohiro   mm: clean up page...
1083
  	}
e6c509f85   Hugh Dickins   mm: use clear_pag...
1084
1085
  	if (unlikely(PageMlocked(page)))
  		clear_page_mlock(page);
b904dcfed   KOSAKI Motohiro   mm: clean up page...
1086
1087
1088
1089
1090
1091
1092
1093
1094
  	/*
  	 * 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.
  	 */
e6c509f85   Hugh Dickins   mm: use clear_pag...
1095
  	return;
89c06bd52   KAMEZAWA Hiroyuki   memcg: use new lo...
1096
1097
1098
  out:
  	if (!anon)
  		mem_cgroup_end_update_page_stat(page, &locked, &flags);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1099
1100
1101
  }
  
  /*
526295064   Joonsoo Kim   mm/rmap: use rmap...
1102
   * @arg: enum ttu_flags will be passed to this argument
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1103
   */
ac7695012   Kirill A. Shutemov   mm/rmap.c: make p...
1104
  static int try_to_unmap_one(struct page *page, struct vm_area_struct *vma,
526295064   Joonsoo Kim   mm/rmap: use rmap...
1105
  		     unsigned long address, void *arg)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1106
1107
  {
  	struct mm_struct *mm = vma->vm_mm;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1108
1109
  	pte_t *pte;
  	pte_t pteval;
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
1110
  	spinlock_t *ptl;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1111
  	int ret = SWAP_AGAIN;
526295064   Joonsoo Kim   mm/rmap: use rmap...
1112
  	enum ttu_flags flags = (enum ttu_flags)arg;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1113

479db0bf4   Nick Piggin   mm: dirty page tr...
1114
  	pte = page_check_address(page, mm, address, &ptl, 0);
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
1115
  	if (!pte)
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
1116
  		goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1117
1118
1119
1120
1121
1122
  
  	/*
  	 * 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...
1123
  	if (!(flags & TTU_IGNORE_MLOCK)) {
caed0f486   KOSAKI Motohiro   mm: simplify try_...
1124
1125
  		if (vma->vm_flags & VM_LOCKED)
  			goto out_mlock;
daa5ba768   Konstantin Khlebnikov   mm/rmap.c: cleanu...
1126
  		if (flags & TTU_MUNLOCK)
53f79acb6   Hugh Dickins   mm: mlocking in t...
1127
  			goto out_unmap;
14fa31b89   Andi Kleen   HWPOISON: Use bit...
1128
1129
  	}
  	if (!(flags & TTU_IGNORE_ACCESS)) {
b291f0003   Nick Piggin   mlock: mlocked pa...
1130
1131
1132
1133
1134
  		if (ptep_clear_flush_young_notify(vma, address, pte)) {
  			ret = SWAP_FAIL;
  			goto out_unmap;
  		}
    	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1135

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1136
1137
  	/* Nuke the page table entry. */
  	flush_cache_page(vma, address, page_to_pfn(page));
2ec74c3ef   Sagi Grimberg   mm: move all mmu ...
1138
  	pteval = ptep_clear_flush(vma, address, pte);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1139
1140
1141
1142
  
  	/* 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...
1143
1144
  	/* Update high watermark before we lower rss */
  	update_hiwater_rss(mm);
888b9f7c5   Andi Kleen   HWPOISON: Handle ...
1145
  	if (PageHWPoison(page) && !(flags & TTU_IGNORE_HWPOISON)) {
5f24ae585   Naoya Horiguchi   hwpoison, hugetlb...
1146
1147
1148
1149
1150
1151
  		if (!PageHuge(page)) {
  			if (PageAnon(page))
  				dec_mm_counter(mm, MM_ANONPAGES);
  			else
  				dec_mm_counter(mm, MM_FILEPAGES);
  		}
888b9f7c5   Andi Kleen   HWPOISON: Handle ...
1152
  		set_pte_at(mm, address, pte,
5f24ae585   Naoya Horiguchi   hwpoison, hugetlb...
1153
  			   swp_entry_to_pte(make_hwpoison_entry(page)));
45961722f   Konstantin Weitz   mm: add support f...
1154
1155
1156
1157
1158
1159
1160
1161
1162
1163
  	} else if (pte_unused(pteval)) {
  		/*
  		 * The guest indicated that the page content is of no
  		 * interest anymore. Simply discard the pte, vmscan
  		 * will take care of the rest.
  		 */
  		if (PageAnon(page))
  			dec_mm_counter(mm, MM_ANONPAGES);
  		else
  			dec_mm_counter(mm, MM_FILEPAGES);
888b9f7c5   Andi Kleen   HWPOISON: Handle ...
1164
  	} else if (PageAnon(page)) {
4c21e2f24   Hugh Dickins   [PATCH] mm: split...
1165
  		swp_entry_t entry = { .val = page_private(page) };
179ef71cb   Cyrill Gorcunov   mm: save soft-dir...
1166
  		pte_t swp_pte;
0697212a4   Christoph Lameter   [PATCH] Swapless ...
1167
1168
1169
1170
1171
1172
  
  		if (PageSwapCache(page)) {
  			/*
  			 * Store the swap location in the pte.
  			 * See handle_pte_fault() ...
  			 */
570a335b8   Hugh Dickins   swap_info: swap c...
1173
1174
1175
1176
1177
  			if (swap_duplicate(entry) < 0) {
  				set_pte_at(mm, address, pte, pteval);
  				ret = SWAP_FAIL;
  				goto out_unmap;
  			}
0697212a4   Christoph Lameter   [PATCH] Swapless ...
1178
1179
1180
1181
1182
1183
  			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...
1184
  			dec_mm_counter(mm, MM_ANONPAGES);
b084d4353   KAMEZAWA Hiroyuki   mm: count swap usage
1185
  			inc_mm_counter(mm, MM_SWAPENTS);
ce1744f4e   Konstantin Khlebnikov   mm: replace PAGE_...
1186
  		} else if (IS_ENABLED(CONFIG_MIGRATION)) {
0697212a4   Christoph Lameter   [PATCH] Swapless ...
1187
1188
1189
1190
1191
  			/*
  			 * 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.
  			 */
daa5ba768   Konstantin Khlebnikov   mm/rmap.c: cleanu...
1192
  			BUG_ON(!(flags & TTU_MIGRATION));
0697212a4   Christoph Lameter   [PATCH] Swapless ...
1193
  			entry = make_migration_entry(page, pte_write(pteval));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1194
  		}
179ef71cb   Cyrill Gorcunov   mm: save soft-dir...
1195
1196
1197
1198
  		swp_pte = swp_entry_to_pte(entry);
  		if (pte_soft_dirty(pteval))
  			swp_pte = pte_swp_mksoft_dirty(swp_pte);
  		set_pte_at(mm, address, pte, swp_pte);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1199
  		BUG_ON(pte_file(*pte));
ce1744f4e   Konstantin Khlebnikov   mm: replace PAGE_...
1200
  	} else if (IS_ENABLED(CONFIG_MIGRATION) &&
daa5ba768   Konstantin Khlebnikov   mm/rmap.c: cleanu...
1201
  		   (flags & TTU_MIGRATION)) {
04e62a29b   Christoph Lameter   [PATCH] More page...
1202
1203
1204
1205
1206
  		/* 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...
1207
  		dec_mm_counter(mm, MM_FILEPAGES);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1208

edc315fd2   Hugh Dickins   badpage: remove v...
1209
  	page_remove_rmap(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1210
1211
1212
  	page_cache_release(page);
  
  out_unmap:
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
1213
  	pte_unmap_unlock(pte, ptl);
daa5ba768   Konstantin Khlebnikov   mm/rmap.c: cleanu...
1214
  	if (ret != SWAP_FAIL && !(flags & TTU_MUNLOCK))
2ec74c3ef   Sagi Grimberg   mm: move all mmu ...
1215
  		mmu_notifier_invalidate_page(mm, address);
caed0f486   KOSAKI Motohiro   mm: simplify try_...
1216
1217
  out:
  	return ret;
53f79acb6   Hugh Dickins   mm: mlocking in t...
1218

caed0f486   KOSAKI Motohiro   mm: simplify try_...
1219
1220
1221
1222
1223
1224
1225
  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
5a505085f   Ingo Molnar   mm/rmap: Convert ...
1226
  	 * we now hold anon_vma->rwsem or mapping->i_mmap_mutex.
caed0f486   KOSAKI Motohiro   mm: simplify try_...
1227
1228
1229
1230
1231
1232
1233
1234
  	 * 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...
1235
  		}
caed0f486   KOSAKI Motohiro   mm: simplify try_...
1236
  		up_read(&vma->vm_mm->mmap_sem);
53f79acb6   Hugh Dickins   mm: mlocking in t...
1237
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1238
1239
1240
1241
1242
1243
1244
1245
1246
1247
1248
1249
1250
1251
1252
1253
1254
1255
1256
1257
1258
  	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...
1259
1260
1261
1262
1263
   *
   * 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
1264
1265
1266
   */
  #define CLUSTER_SIZE	min(32*PAGE_SIZE, PMD_SIZE)
  #define CLUSTER_MASK	(~(CLUSTER_SIZE - 1))
b291f0003   Nick Piggin   mlock: mlocked pa...
1267
1268
  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
1269
1270
  {
  	struct mm_struct *mm = vma->vm_mm;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1271
  	pmd_t *pmd;
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
1272
  	pte_t *pte;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1273
  	pte_t pteval;
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
1274
  	spinlock_t *ptl;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1275
1276
  	struct page *page;
  	unsigned long address;
2ec74c3ef   Sagi Grimberg   mm: move all mmu ...
1277
1278
  	unsigned long mmun_start;	/* For mmu_notifiers */
  	unsigned long mmun_end;		/* For mmu_notifiers */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1279
  	unsigned long end;
b291f0003   Nick Piggin   mlock: mlocked pa...
1280
1281
  	int ret = SWAP_AGAIN;
  	int locked_vma = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1282

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1283
1284
1285
1286
1287
1288
  	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;
6219049ae   Bob Liu   mm: introduce mm_...
1289
1290
  	pmd = mm_find_pmd(mm, address);
  	if (!pmd)
b291f0003   Nick Piggin   mlock: mlocked pa...
1291
  		return ret;
2ec74c3ef   Sagi Grimberg   mm: move all mmu ...
1292
1293
1294
  	mmun_start = address;
  	mmun_end   = end;
  	mmu_notifier_invalidate_range_start(mm, mmun_start, mmun_end);
b291f0003   Nick Piggin   mlock: mlocked pa...
1295
  	/*
af8e3354b   Hugh Dickins   mm: CONFIG_MMU fo...
1296
  	 * If we can acquire the mmap_sem for read, and vma is VM_LOCKED,
b291f0003   Nick Piggin   mlock: mlocked pa...
1297
1298
  	 * keep the sem while scanning the cluster for mlocking pages.
  	 */
af8e3354b   Hugh Dickins   mm: CONFIG_MMU fo...
1299
  	if (down_read_trylock(&vma->vm_mm->mmap_sem)) {
b291f0003   Nick Piggin   mlock: mlocked pa...
1300
1301
1302
1303
  		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 ...
1304
1305
  
  	pte = pte_offset_map_lock(mm, pmd, address, &ptl);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1306

365e9c87a   Hugh Dickins   [PATCH] mm: updat...
1307
1308
  	/* Update high watermark before we lower rss */
  	update_hiwater_rss(mm);
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
1309
  	for (; address < end; pte++, address += PAGE_SIZE) {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1310
1311
  		if (!pte_present(*pte))
  			continue;
6aab341e0   Linus Torvalds   mm: re-architect ...
1312
1313
  		page = vm_normal_page(vma, address, *pte);
  		BUG_ON(!page || PageAnon(page));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1314

b291f0003   Nick Piggin   mlock: mlocked pa...
1315
  		if (locked_vma) {
57e68e9cd   Vlastimil Babka   mm: try_to_unmap_...
1316
1317
1318
  			if (page == check_page) {
  				/* we know we have check_page locked */
  				mlock_vma_page(page);
b291f0003   Nick Piggin   mlock: mlocked pa...
1319
  				ret = SWAP_MLOCK;
57e68e9cd   Vlastimil Babka   mm: try_to_unmap_...
1320
1321
1322
1323
1324
1325
1326
1327
1328
  			} else if (trylock_page(page)) {
  				/*
  				 * If we can lock the page, perform mlock.
  				 * Otherwise leave the page alone, it will be
  				 * eventually encountered again later.
  				 */
  				mlock_vma_page(page);
  				unlock_page(page);
  			}
b291f0003   Nick Piggin   mlock: mlocked pa...
1329
1330
  			continue;	/* don't unmap */
  		}
cddb8a5c1   Andrea Arcangeli   mmu-notifiers: core
1331
  		if (ptep_clear_flush_young_notify(vma, address, pte))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1332
1333
1334
  			continue;
  
  		/* Nuke the page table entry. */
eca351336   Ben Collins   [PATCH] Fix missi...
1335
  		flush_cache_page(vma, address, pte_pfn(*pte));
2ec74c3ef   Sagi Grimberg   mm: move all mmu ...
1336
  		pteval = ptep_clear_flush(vma, address, pte);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1337
1338
  
  		/* If nonlinear, store the file page offset in the pte. */
41bb3476b   Cyrill Gorcunov   mm: save soft-dir...
1339
1340
1341
  		if (page->index != linear_page_index(vma, address)) {
  			pte_t ptfile = pgoff_to_pte(page->index);
  			if (pte_soft_dirty(pteval))
b43790eed   Cyrill Gorcunov   mm: softdirty: do...
1342
  				ptfile = pte_file_mksoft_dirty(ptfile);
41bb3476b   Cyrill Gorcunov   mm: save soft-dir...
1343
1344
  			set_pte_at(mm, address, pte, ptfile);
  		}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1345
1346
1347
1348
  
  		/* 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...
1349
  		page_remove_rmap(page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1350
  		page_cache_release(page);
d559db086   KAMEZAWA Hiroyuki   mm: clean up mm_c...
1351
  		dec_mm_counter(mm, MM_FILEPAGES);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1352
1353
  		(*mapcount)--;
  	}
c0718806c   Hugh Dickins   [PATCH] mm: rmap ...
1354
  	pte_unmap_unlock(pte - 1, ptl);
2ec74c3ef   Sagi Grimberg   mm: move all mmu ...
1355
  	mmu_notifier_invalidate_range_end(mm, mmun_start, mmun_end);
b291f0003   Nick Piggin   mlock: mlocked pa...
1356
1357
1358
  	if (locked_vma)
  		up_read(&vma->vm_mm->mmap_sem);
  	return ret;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1359
  }
0f843c6ac   Joonsoo Kim   mm/rmap: factor n...
1360
  static int try_to_unmap_nonlinear(struct page *page,
7e09e738a   Hugh Dickins   mm: fix swapops.h...
1361
  		struct address_space *mapping, void *arg)
0f843c6ac   Joonsoo Kim   mm/rmap: factor n...
1362
  {
7e09e738a   Hugh Dickins   mm: fix swapops.h...
1363
  	struct vm_area_struct *vma;
0f843c6ac   Joonsoo Kim   mm/rmap: factor n...
1364
1365
1366
1367
1368
1369
1370
1371
1372
1373
1374
1375
1376
1377
1378
1379
1380
1381
1382
1383
1384
1385
1386
1387
1388
1389
1390
1391
1392
1393
1394
1395
1396
1397
1398
1399
1400
1401
1402
1403
1404
1405
1406
1407
1408
1409
1410
1411
1412
1413
1414
1415
1416
1417
1418
1419
1420
1421
1422
1423
1424
1425
1426
1427
1428
1429
1430
1431
1432
  	int ret = SWAP_AGAIN;
  	unsigned long cursor;
  	unsigned long max_nl_cursor = 0;
  	unsigned long max_nl_size = 0;
  	unsigned int mapcount;
  
  	list_for_each_entry(vma,
  		&mapping->i_mmap_nonlinear, shared.nonlinear) {
  
  		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;
  	}
  
  	if (max_nl_size == 0) {	/* all nonlinears locked or reserved ? */
  		return SWAP_FAIL;
  	}
  
  	/*
  	 * 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)
  		return ret;
  
  	cond_resched();
  
  	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.nonlinear) {
  
  			cursor = (unsigned long) vma->vm_private_data;
  			while (cursor < max_nl_cursor &&
  				cursor < vma->vm_end - vma->vm_start) {
  				if (try_to_unmap_cluster(cursor, &mapcount,
  						vma, page) == SWAP_MLOCK)
  					ret = SWAP_MLOCK;
  				cursor += CLUSTER_SIZE;
  				vma->vm_private_data = (void *) cursor;
  				if ((int)mapcount <= 0)
  					return ret;
  			}
  			vma->vm_private_data = (void *) max_nl_cursor;
  		}
  		cond_resched();
  		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.
  	 */
  	list_for_each_entry(vma, &mapping->i_mmap_nonlinear, shared.nonlinear)
  		vma->vm_private_data = NULL;
  
  	return ret;
  }
71e3aac07   Andrea Arcangeli   thp: transparent ...
1433
  bool is_vma_temporary_stack(struct vm_area_struct *vma)
a8bef8ff6   Mel Gorman   mm: migration: av...
1434
1435
1436
1437
1438
1439
1440
1441
1442
1443
1444
1445
  {
  	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;
  }
526295064   Joonsoo Kim   mm/rmap: use rmap...
1446
1447
1448
1449
  static bool invalid_migration_vma(struct vm_area_struct *vma, void *arg)
  {
  	return is_vma_temporary_stack(vma);
  }
526295064   Joonsoo Kim   mm/rmap: use rmap...
1450
1451
1452
1453
  static int page_not_mapped(struct page *page)
  {
  	return !page_mapped(page);
  };
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1454
1455
1456
  /**
   * 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...
1457
   * @flags: action and flags
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1458
1459
1460
1461
1462
1463
1464
1465
   *
   * 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...
1466
   * SWAP_MLOCK	- page is mlocked.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1467
   */
14fa31b89   Andi Kleen   HWPOISON: Use bit...
1468
  int try_to_unmap(struct page *page, enum ttu_flags flags)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1469
1470
  {
  	int ret;
526295064   Joonsoo Kim   mm/rmap: use rmap...
1471
1472
1473
1474
1475
1476
1477
  	struct rmap_walk_control rwc = {
  		.rmap_one = try_to_unmap_one,
  		.arg = (void *)flags,
  		.done = page_not_mapped,
  		.file_nonlinear = try_to_unmap_nonlinear,
  		.anon_lock = page_lock_anon_vma_read,
  	};
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1478

309381fea   Sasha Levin   mm: dump page whe...
1479
  	VM_BUG_ON_PAGE(!PageHuge(page) && PageTransHuge(page), page);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1480

526295064   Joonsoo Kim   mm/rmap: use rmap...
1481
1482
1483
1484
1485
1486
1487
1488
  	/*
  	 * 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.
  	 */
daa5ba768   Konstantin Khlebnikov   mm/rmap.c: cleanu...
1489
  	if ((flags & TTU_MIGRATION) && !PageKsm(page) && PageAnon(page))
526295064   Joonsoo Kim   mm/rmap: use rmap...
1490
1491
1492
  		rwc.invalid_vma = invalid_migration_vma;
  
  	ret = rmap_walk(page, &rwc);
b291f0003   Nick Piggin   mlock: mlocked pa...
1493
  	if (ret != SWAP_MLOCK && !page_mapped(page))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1494
1495
1496
  		ret = SWAP_SUCCESS;
  	return ret;
  }
81b4082dc   Nikita Danilov   [PATCH] mm: rmap....
1497

b291f0003   Nick Piggin   mlock: mlocked pa...
1498
1499
1500
1501
1502
1503
1504
1505
1506
1507
  /**
   * 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...
1508
   * SWAP_AGAIN	- no vma is holding page mlocked, or,
b291f0003   Nick Piggin   mlock: mlocked pa...
1509
   * SWAP_AGAIN	- page mapped in mlocked vma -- couldn't acquire mmap sem
5ad646880   Hugh Dickins   ksm: let shared p...
1510
   * SWAP_FAIL	- page cannot be located at present
b291f0003   Nick Piggin   mlock: mlocked pa...
1511
1512
1513
1514
   * SWAP_MLOCK	- page is now mlocked.
   */
  int try_to_munlock(struct page *page)
  {
e8351ac9b   Joonsoo Kim   mm/rmap: use rmap...
1515
1516
1517
1518
1519
1520
1521
1522
1523
1524
1525
1526
1527
1528
  	int ret;
  	struct rmap_walk_control rwc = {
  		.rmap_one = try_to_unmap_one,
  		.arg = (void *)TTU_MUNLOCK,
  		.done = page_not_mapped,
  		/*
  		 * 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() and recover PG_mlocked lazily.
  		 */
  		.file_nonlinear = NULL,
  		.anon_lock = page_lock_anon_vma_read,
  
  	};
309381fea   Sasha Levin   mm: dump page whe...
1529
  	VM_BUG_ON_PAGE(!PageLocked(page) || PageLRU(page), page);
b291f0003   Nick Piggin   mlock: mlocked pa...
1530

e8351ac9b   Joonsoo Kim   mm/rmap: use rmap...
1531
1532
  	ret = rmap_walk(page, &rwc);
  	return ret;
b291f0003   Nick Piggin   mlock: mlocked pa...
1533
  }
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1534

01d8b20de   Peter Zijlstra   mm: simplify anon...
1535
  void __put_anon_vma(struct anon_vma *anon_vma)
76545066c   Rik van Riel   mm: extend KSM re...
1536
  {
01d8b20de   Peter Zijlstra   mm: simplify anon...
1537
  	struct anon_vma *root = anon_vma->root;
76545066c   Rik van Riel   mm: extend KSM re...
1538

624483f3e   Andrey Ryabinin   mm: rmap: fix use...
1539
  	anon_vma_free(anon_vma);
01d8b20de   Peter Zijlstra   mm: simplify anon...
1540
1541
  	if (root != anon_vma && atomic_dec_and_test(&root->refcount))
  		anon_vma_free(root);
76545066c   Rik van Riel   mm: extend KSM re...
1542
  }
76545066c   Rik van Riel   mm: extend KSM re...
1543

0dd1c7bbc   Joonsoo Kim   mm/rmap: extend r...
1544
1545
  static struct anon_vma *rmap_walk_anon_lock(struct page *page,
  					struct rmap_walk_control *rwc)
faecd8dd8   Joonsoo Kim   mm/rmap: factor l...
1546
1547
  {
  	struct anon_vma *anon_vma;
0dd1c7bbc   Joonsoo Kim   mm/rmap: extend r...
1548
1549
  	if (rwc->anon_lock)
  		return rwc->anon_lock(page);
faecd8dd8   Joonsoo Kim   mm/rmap: factor l...
1550
1551
1552
1553
1554
1555
1556
1557
1558
1559
1560
1561
1562
  	/*
  	 * Note: remove_migration_ptes() cannot use page_lock_anon_vma_read()
  	 * because that depends on page_mapped(); but not all its usages
  	 * are holding mmap_sem. Users without mmap_sem are required to
  	 * take a reference count to prevent the anon_vma disappearing
  	 */
  	anon_vma = page_anon_vma(page);
  	if (!anon_vma)
  		return NULL;
  
  	anon_vma_lock_read(anon_vma);
  	return anon_vma;
  }
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1563
  /*
e8351ac9b   Joonsoo Kim   mm/rmap: use rmap...
1564
1565
1566
1567
1568
1569
1570
1571
1572
1573
1574
1575
   * rmap_walk_anon - do something to anonymous page using the object-based
   * rmap method
   * @page: the page to be handled
   * @rwc: control variable according to each walk type
   *
   * Find all the mappings of a page using the mapping pointer and the vma chains
   * contained in the anon_vma struct it points to.
   *
   * 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.
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1576
   */
051ac83ad   Joonsoo Kim   mm/rmap: make rma...
1577
  static int rmap_walk_anon(struct page *page, struct rmap_walk_control *rwc)
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1578
1579
  {
  	struct anon_vma *anon_vma;
bf181b9f9   Michel Lespinasse   mm anon rmap: rep...
1580
  	pgoff_t pgoff = page->index << (PAGE_CACHE_SHIFT - PAGE_SHIFT);
5beb49305   Rik van Riel   mm: change anon_v...
1581
  	struct anon_vma_chain *avc;
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1582
  	int ret = SWAP_AGAIN;
0dd1c7bbc   Joonsoo Kim   mm/rmap: extend r...
1583
  	anon_vma = rmap_walk_anon_lock(page, rwc);
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1584
1585
  	if (!anon_vma)
  		return ret;
faecd8dd8   Joonsoo Kim   mm/rmap: factor l...
1586

bf181b9f9   Michel Lespinasse   mm anon rmap: rep...
1587
  	anon_vma_interval_tree_foreach(avc, &anon_vma->rb_root, pgoff, pgoff) {
5beb49305   Rik van Riel   mm: change anon_v...
1588
  		struct vm_area_struct *vma = avc->vma;
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1589
  		unsigned long address = vma_address(page, vma);
0dd1c7bbc   Joonsoo Kim   mm/rmap: extend r...
1590
1591
1592
  
  		if (rwc->invalid_vma && rwc->invalid_vma(vma, rwc->arg))
  			continue;
051ac83ad   Joonsoo Kim   mm/rmap: make rma...
1593
  		ret = rwc->rmap_one(page, vma, address, rwc->arg);
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1594
1595
  		if (ret != SWAP_AGAIN)
  			break;
0dd1c7bbc   Joonsoo Kim   mm/rmap: extend r...
1596
1597
  		if (rwc->done && rwc->done(page))
  			break;
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1598
  	}
4fc3f1d66   Ingo Molnar   mm/rmap, migratio...
1599
  	anon_vma_unlock_read(anon_vma);
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1600
1601
  	return ret;
  }
e8351ac9b   Joonsoo Kim   mm/rmap: use rmap...
1602
1603
1604
1605
1606
1607
1608
1609
1610
1611
1612
1613
1614
  /*
   * rmap_walk_file - do something to file page using the object-based rmap method
   * @page: the page to be handled
   * @rwc: control variable according to each walk type
   *
   * Find all the mappings of a page using the mapping pointer and the vma chains
   * contained in the address_space struct it points to.
   *
   * 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.
   */
051ac83ad   Joonsoo Kim   mm/rmap: make rma...
1615
  static int rmap_walk_file(struct page *page, struct rmap_walk_control *rwc)
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1616
1617
  {
  	struct address_space *mapping = page->mapping;
b854f711f   Joonsoo Kim   mm/rmap: recomput...
1618
  	pgoff_t pgoff = page->index << compound_order(page);
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1619
  	struct vm_area_struct *vma;
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1620
  	int ret = SWAP_AGAIN;
9f32624be   Joonsoo Kim   mm/rmap: use rmap...
1621
1622
1623
1624
1625
1626
1627
  	/*
  	 * 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_mutex.
  	 */
  	VM_BUG_ON(!PageLocked(page));
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1628
1629
  	if (!mapping)
  		return ret;
3d48ae45e   Peter Zijlstra   mm: Convert i_mma...
1630
  	mutex_lock(&mapping->i_mmap_mutex);
6b2dbba8b   Michel Lespinasse   mm: replace vma p...
1631
  	vma_interval_tree_foreach(vma, &mapping->i_mmap, pgoff, pgoff) {
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1632
  		unsigned long address = vma_address(page, vma);
0dd1c7bbc   Joonsoo Kim   mm/rmap: extend r...
1633
1634
1635
  
  		if (rwc->invalid_vma && rwc->invalid_vma(vma, rwc->arg))
  			continue;
051ac83ad   Joonsoo Kim   mm/rmap: make rma...
1636
  		ret = rwc->rmap_one(page, vma, address, rwc->arg);
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1637
  		if (ret != SWAP_AGAIN)
0dd1c7bbc   Joonsoo Kim   mm/rmap: extend r...
1638
1639
1640
  			goto done;
  		if (rwc->done && rwc->done(page))
  			goto done;
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1641
  	}
0dd1c7bbc   Joonsoo Kim   mm/rmap: extend r...
1642
1643
1644
1645
1646
1647
  
  	if (!rwc->file_nonlinear)
  		goto done;
  
  	if (list_empty(&mapping->i_mmap_nonlinear))
  		goto done;
7e09e738a   Hugh Dickins   mm: fix swapops.h...
1648
  	ret = rwc->file_nonlinear(page, mapping, rwc->arg);
0dd1c7bbc   Joonsoo Kim   mm/rmap: extend r...
1649
1650
  
  done:
3d48ae45e   Peter Zijlstra   mm: Convert i_mma...
1651
  	mutex_unlock(&mapping->i_mmap_mutex);
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1652
1653
  	return ret;
  }
051ac83ad   Joonsoo Kim   mm/rmap: make rma...
1654
  int rmap_walk(struct page *page, struct rmap_walk_control *rwc)
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1655
  {
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1656
  	if (unlikely(PageKsm(page)))
051ac83ad   Joonsoo Kim   mm/rmap: make rma...
1657
  		return rmap_walk_ksm(page, rwc);
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1658
  	else if (PageAnon(page))
051ac83ad   Joonsoo Kim   mm/rmap: make rma...
1659
  		return rmap_walk_anon(page, rwc);
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1660
  	else
051ac83ad   Joonsoo Kim   mm/rmap: make rma...
1661
  		return rmap_walk_file(page, rwc);
e9995ef97   Hugh Dickins   ksm: rmap_walk to...
1662
  }
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
1663

e3390f67a   Naoya Horiguchi   hwpoison: rename ...
1664
  #ifdef CONFIG_HUGETLB_PAGE
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
1665
1666
1667
1668
1669
1670
1671
1672
1673
  /*
   * 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...
1674

0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
1675
  	BUG_ON(!anon_vma);
433abed6c   Naoya Horiguchi   hugetlb, rmap: al...
1676
1677
1678
1679
1680
  
  	if (PageAnon(page))
  		return;
  	if (!exclusive)
  		anon_vma = anon_vma->root;
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
1681
1682
1683
1684
1685
1686
1687
1688
1689
1690
  	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...
1691
1692
  
  	BUG_ON(!PageLocked(page));
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
1693
  	BUG_ON(!anon_vma);
5dbe0af47   Hugh Dickins   mm: fix kernel BU...
1694
  	/* address might be in next vma when migration races vma_adjust */
0fe6e20b9   Naoya Horiguchi   hugetlb, rmap: ad...
1695
1696
1697
1698
1699
1700
1701
1702
1703
1704
1705
1706
  	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 ...
1707
  #endif /* CONFIG_HUGETLB_PAGE */