Blame view

mm/fremap.c 5.96 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
17
18
19
20
21
  /*
   *   linux/mm/fremap.c
   * 
   * Explicit pagetable population and nonlinear (random) mappings support.
   *
   * started by Ingo Molnar, Copyright (C) 2002, 2003
   */
  
  #include <linux/mm.h>
  #include <linux/swap.h>
  #include <linux/file.h>
  #include <linux/mman.h>
  #include <linux/pagemap.h>
  #include <linux/swapops.h>
  #include <linux/rmap.h>
  #include <linux/module.h>
  #include <linux/syscalls.h>
  
  #include <asm/mmu_context.h>
  #include <asm/cacheflush.h>
  #include <asm/tlbflush.h>
d0217ac04   Nick Piggin   mm: fault feedbac...
22
  static void zap_pte(struct mm_struct *mm, struct vm_area_struct *vma,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
23
24
25
  			unsigned long addr, pte_t *ptep)
  {
  	pte_t pte = *ptep;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
26
  	if (pte_present(pte)) {
d0217ac04   Nick Piggin   mm: fault feedbac...
27
  		struct page *page;
6aab341e0   Linus Torvalds   mm: re-architect ...
28
  		flush_cache_page(vma, addr, pte_pfn(pte));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
29
  		pte = ptep_clear_flush(vma, addr, ptep);
6aab341e0   Linus Torvalds   mm: re-architect ...
30
31
32
33
  		page = vm_normal_page(vma, addr, pte);
  		if (page) {
  			if (pte_dirty(pte))
  				set_page_dirty(page);
7de6b8057   Nick Piggin   [PATCH] mm: more ...
34
  			page_remove_rmap(page, vma);
6aab341e0   Linus Torvalds   mm: re-architect ...
35
  			page_cache_release(page);
d0217ac04   Nick Piggin   mm: fault feedbac...
36
37
  			update_hiwater_rss(mm);
  			dec_mm_counter(mm, file_rss);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
38
39
40
41
  		}
  	} else {
  		if (!pte_file(pte))
  			free_swap_and_cache(pte_to_swp_entry(pte));
9888a1cae   Zachary Amsden   [PATCH] paravirt:...
42
  		pte_clear_not_present_full(mm, addr, ptep, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
43
44
45
46
  	}
  }
  
  /*
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
47
48
49
   * Install a file pte to a given virtual memory address, release any
   * previously existing mapping.
   */
d0217ac04   Nick Piggin   mm: fault feedbac...
50
  static int install_file_pte(struct mm_struct *mm, struct vm_area_struct *vma,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
51
52
53
54
  		unsigned long addr, unsigned long pgoff, pgprot_t prot)
  {
  	int err = -ENOMEM;
  	pte_t *pte;
c74df32c7   Hugh Dickins   [PATCH] mm: ptd_a...
55
  	spinlock_t *ptl;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
56

c9cfcddfd   Linus Torvalds   VM: add common he...
57
  	pte = get_locked_pte(mm, addr, &ptl);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
58
  	if (!pte)
c74df32c7   Hugh Dickins   [PATCH] mm: ptd_a...
59
  		goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
60

d0217ac04   Nick Piggin   mm: fault feedbac...
61
62
  	if (!pte_none(*pte))
  		zap_pte(mm, vma, addr, pte);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
63
64
  
  	set_pte_at(mm, addr, pte, pgoff_to_pte(pgoff));
668e0d8f1   Hugh Dickins   [PATCH] fix updat...
65
66
67
68
69
70
71
  	/*
  	 * We don't need to run update_mmu_cache() here because the "file pte"
  	 * being installed by install_file_pte() is not a real pte - it's a
  	 * non-present entry (like a swap entry), noting what file offset should
  	 * be mapped there when there's a fault (in a non-linear vma where
  	 * that's not obvious).
  	 */
c74df32c7   Hugh Dickins   [PATCH] mm: ptd_a...
72
73
74
  	pte_unmap_unlock(pte, ptl);
  	err = 0;
  out:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
75
76
  	return err;
  }
54cb8821d   Nick Piggin   mm: merge populat...
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
  static int populate_range(struct mm_struct *mm, struct vm_area_struct *vma,
  			unsigned long addr, unsigned long size, pgoff_t pgoff)
  {
  	int err;
  
  	do {
  		err = install_file_pte(mm, vma, addr, pgoff, vma->vm_page_prot);
  		if (err)
  			return err;
  
  		size -= PAGE_SIZE;
  		addr += PAGE_SIZE;
  		pgoff++;
  	} while (size);
  
          return 0;
  
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
  /***
   * sys_remap_file_pages - remap arbitrary pages of a shared backing store
   *                        file within an existing vma.
   * @start: start of the remapped virtual memory range
   * @size: size of the remapped virtual memory range
   * @prot: new protection bits of the range
   * @pgoff: to be mapped page of the backing store file
   * @flags: 0 or MAP_NONBLOCKED - the later will cause no IO.
   *
   * this syscall works purely via pagetables, so it's the most efficient
   * way to map the same (large) file into a given virtual window. Unlike
   * mmap()/mremap() it does not create any new vmas. The new mappings are
   * also safe across swapout.
   *
   * NOTE: the 'prot' parameter right now is ignored, and the vma's default
   * protection is used. Arbitrary protections might be implemented in the
   * future.
   */
  asmlinkage long sys_remap_file_pages(unsigned long start, unsigned long size,
  	unsigned long __prot, unsigned long pgoff, unsigned long flags)
  {
  	struct mm_struct *mm = current->mm;
  	struct address_space *mapping;
  	unsigned long end = start + size;
  	struct vm_area_struct *vma;
  	int err = -EINVAL;
  	int has_write_lock = 0;
  
  	if (__prot)
  		return err;
  	/*
  	 * Sanitize the syscall parameters:
  	 */
  	start = start & PAGE_MASK;
  	size = size & PAGE_MASK;
  
  	/* Does the address range wrap, or is the span zero-sized? */
  	if (start + size <= start)
  		return err;
  
  	/* Can we represent this offset inside this architecture's pte's? */
  #if PTE_FILE_MAX_BITS < BITS_PER_LONG
  	if (pgoff + (size >> PAGE_SHIFT) >= (1UL << PTE_FILE_MAX_BITS))
  		return err;
  #endif
  
  	/* We need down_write() to change vma->vm_flags. */
  	down_read(&mm->mmap_sem);
   retry:
  	vma = find_vma(mm, start);
  
  	/*
  	 * Make sure the vma is shared, that it supports prefaulting,
  	 * and that the remapped range is valid and fully within
  	 * the single existing vma.  vm_private_data is used as a
101d2be76   Hugh Dickins   [PATCH] unpaged: ...
150
  	 * swapout cursor in a VM_NONLINEAR vma.
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
151
  	 */
54cb8821d   Nick Piggin   mm: merge populat...
152
153
154
155
156
  	if (!vma || !(vma->vm_flags & VM_SHARED))
  		goto out;
  
  	if (vma->vm_private_data && !(vma->vm_flags & VM_NONLINEAR))
  		goto out;
d0217ac04   Nick Piggin   mm: fault feedbac...
157
  	if (!vma->vm_flags & VM_CAN_NONLINEAR)
54cb8821d   Nick Piggin   mm: merge populat...
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
  		goto out;
  
  	if (end <= start || start < vma->vm_start || end > vma->vm_end)
  		goto out;
  
  	/* Must set VM_NONLINEAR before any pages are populated. */
  	if (!(vma->vm_flags & VM_NONLINEAR)) {
  		/* Don't need a nonlinear mapping, exit success */
  		if (pgoff == linear_page_index(vma, start)) {
  			err = 0;
  			goto out;
  		}
  
  		if (!has_write_lock) {
  			up_read(&mm->mmap_sem);
  			down_write(&mm->mmap_sem);
  			has_write_lock = 1;
  			goto retry;
  		}
  		mapping = vma->vm_file->f_mapping;
3ee6dafc6   Miklos Szeredi   only allow nonlin...
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
  		/*
  		 * page_mkclean doesn't work on nonlinear vmas, so if
  		 * dirty pages need to be accounted, emulate with linear
  		 * vmas.
  		 */
  		if (mapping_cap_account_dirty(mapping)) {
  			unsigned long addr;
  
  			flags &= MAP_NONBLOCK;
  			addr = mmap_region(vma->vm_file, start, size,
  					flags, vma->vm_flags, pgoff, 1);
  			if (IS_ERR_VALUE(addr)) {
  				err = addr;
  			} else {
  				BUG_ON(addr != start);
  				err = 0;
  			}
  			goto out;
  		}
54cb8821d   Nick Piggin   mm: merge populat...
197
198
199
200
201
202
203
204
  		spin_lock(&mapping->i_mmap_lock);
  		flush_dcache_mmap_lock(mapping);
  		vma->vm_flags |= VM_NONLINEAR;
  		vma_prio_tree_remove(vma, &mapping->i_mmap);
  		vma_nonlinear_insert(vma, &mapping->i_mmap_nonlinear);
  		flush_dcache_mmap_unlock(mapping);
  		spin_unlock(&mapping->i_mmap_lock);
  	}
d0217ac04   Nick Piggin   mm: fault feedbac...
205
206
207
208
209
  	err = populate_range(mm, vma, start, size, pgoff);
  	if (!err && !(flags & MAP_NONBLOCK)) {
  		if (unlikely(has_write_lock)) {
  			downgrade_write(&mm->mmap_sem);
  			has_write_lock = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
210
  		}
d0217ac04   Nick Piggin   mm: fault feedbac...
211
212
  		make_pages_present(start, start+size);
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
213

54cb8821d   Nick Piggin   mm: merge populat...
214
215
216
217
218
  	/*
  	 * We can't clear VM_NONLINEAR because we'd have to do
  	 * it after ->populate completes, and that would prevent
  	 * downgrading the lock.  (Locks can't be upgraded).
  	 */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
219

54cb8821d   Nick Piggin   mm: merge populat...
220
  out:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
221
222
223
224
225
226
227
  	if (likely(!has_write_lock))
  		up_read(&mm->mmap_sem);
  	else
  		up_write(&mm->mmap_sem);
  
  	return err;
  }