1 /* $NetBSD: uvm_emap.c,v 1.5 2009/08/29 00:06:43 rmind Exp $ */
4 * Copyright (c) 2009 The NetBSD Foundation, Inc.
7 * This code is derived from software contributed to The NetBSD Foundation
8 * by Mindaugas Rasiukevicius and Andrew Doran.
10 * Redistribution and use in source and binary forms, with or without
11 * modification, are permitted provided that the following conditions
13 * 1. Redistributions of source code must retain the above copyright
14 * notice, this list of conditions and the following disclaimer.
15 * 2. Redistributions in binary form must reproduce the above copyright
16 * notice, this list of conditions and the following disclaimer in the
17 * documentation and/or other materials provided with the distribution.
19 * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29 * POSSIBILITY OF SUCH DAMAGE.
33 * UVM ephemeral mapping interface.
35 * Generic (more expensive) stubs are implemented for architectures which
36 * do not support pmap.
38 * Note that uvm_emap_update() is called from lower pmap(9) layer, while
39 * other functions call to pmap(9). Typical pattern of update in pmap:
41 * u_int gen = uvm_emap_gen_return();
45 * It is also used from IPI context, therefore functions must safe.
48 #include <sys/cdefs.h>
49 __KERNEL_RCSID(0, "$NetBSD: uvm_emap.c,v 1.5 2009/08/29 00:06:43 rmind Exp $");
51 #include <sys/param.h>
52 #include <sys/kernel.h>
54 #include <sys/atomic.h>
57 #include <sys/types.h>
60 #include <uvm/uvm_extern.h>
64 #define UVM_EMAP_SIZE (128 * 1024 * 1024) /* 128 MB */
66 #define UVM_EMAP_SIZE (32 * 1024 * 1024) /* 32 MB */
69 static u_int _uvm_emap_gen
[COHERENCY_UNIT
- sizeof(u_int
)]
70 __aligned(COHERENCY_UNIT
);
72 #define uvm_emap_gen (_uvm_emap_gen[0])
74 u_int uvm_emap_size
= UVM_EMAP_SIZE
;
75 static vaddr_t uvm_emap_va
;
76 static vmem_t
* uvm_emap_vmem
;
79 * uvm_emap_init: initialize subsystem.
82 uvm_emap_sysinit(void)
88 uvm_emap_size
= roundup(uvm_emap_size
, PAGE_SIZE
);
89 qmax
= 16 * PAGE_SIZE
;
91 uvm_emap_va
= uvm_km_alloc(kernel_map
, uvm_emap_size
, 0,
92 UVM_KMF_VAONLY
| UVM_KMF_WAITVA
);
93 if (uvm_emap_va
== 0) {
94 panic("uvm_emap_init: KVA allocation failed");
97 uvm_emap_vmem
= vmem_create("emap", uvm_emap_va
, uvm_emap_size
,
98 PAGE_SIZE
, NULL
, NULL
, NULL
, qmax
, VM_SLEEP
, IPL_NONE
);
99 if (uvm_emap_vmem
== NULL
) {
100 panic("uvm_emap_init: vmem creation failed");
104 uvm_emap_vmem
= NULL
;
106 /* Initial generation value is 1. */
108 for (i
= 0; i
< MAXCPUS
; i
++) {
115 * uvm_emap_alloc: allocate a window.
118 uvm_emap_alloc(vsize_t size
, bool waitok
)
122 KASSERT(round_page(size
) == size
);
124 return vmem_alloc(uvm_emap_vmem
, size
,
125 VM_INSTANTFIT
| (waitok
? VM_SLEEP
: VM_NOSLEEP
));
129 * uvm_emap_free: free a window.
132 uvm_emap_free(vaddr_t va
, size_t size
)
135 KASSERT(va
>= uvm_emap_va
);
136 KASSERT(size
<= uvm_emap_size
);
137 KASSERT(va
+ size
<= uvm_emap_va
+ uvm_emap_size
);
139 vmem_free(uvm_emap_vmem
, va
, size
);
142 #ifdef __HAVE_PMAP_EMAP
145 * uvm_emap_enter: enter a new mapping, without TLB flush.
148 uvm_emap_enter(vaddr_t va
, struct vm_page
**pgs
, u_int npages
)
153 for (n
= 0; n
< npages
; n
++, va
+= PAGE_SIZE
) {
154 pa
= VM_PAGE_TO_PHYS(pgs
[n
]);
155 pmap_emap_enter(va
, pa
, VM_PROT_READ
);
160 * uvm_emap_remove: remove a mapping.
163 uvm_emap_remove(vaddr_t sva
, vsize_t len
)
166 pmap_emap_remove(sva
, len
);
170 * uvm_emap_gen_return: get the global generation number.
172 * => can be called from IPI handler, therefore function must be safe.
175 uvm_emap_gen_return(void)
180 if (__predict_false(gen
== UVM_EMAP_INACTIVE
)) {
182 * Instead of looping, just increase in our side.
183 * Other thread could race and increase it again,
184 * but without any negative effect.
186 gen
= atomic_inc_uint_nv(&uvm_emap_gen
);
188 KASSERT(gen
!= UVM_EMAP_INACTIVE
);
193 * uvm_emap_switch: if the CPU is 'behind' the LWP in emap visibility,
194 * perform TLB flush and thus update the local view. Main purpose is
195 * to handle kernel preemption, while emap is in use.
197 * => called from mi_switch(), when LWP returns after block or preempt.
200 uvm_emap_switch(lwp_t
*l
)
202 struct uvm_cpu
*ucpu
;
205 KASSERT(kpreempt_disabled());
207 /* If LWP did not use emap, then nothing to do. */
208 if (__predict_true(l
->l_emap_gen
== UVM_EMAP_INACTIVE
)) {
213 * No need to synchronise if generation number of current CPU is
214 * newer than the number of this LWP.
216 * This test assumes two's complement arithmetic and allows
217 * ~2B missed updates before it will produce bad results.
219 ucpu
= curcpu()->ci_data
.cpu_uvm
;
220 curgen
= ucpu
->emap_gen
;
222 if (__predict_true((signed int)(curgen
- gen
) >= 0)) {
227 * See comments in uvm_emap_consume() about memory
228 * barriers and race conditions.
230 curgen
= uvm_emap_gen_return();
231 pmap_emap_sync(false);
232 ucpu
->emap_gen
= curgen
;
236 * uvm_emap_consume: update the current CPU and LWP to the given generation
237 * of the emap. In a case of LWP migration to a different CPU after block
238 * or preempt, uvm_emap_switch() will synchronise.
240 * => may be called from both interrupt and thread context.
243 uvm_emap_consume(u_int gen
)
246 struct uvm_cpu
*ucpu
;
250 if (gen
== UVM_EMAP_INACTIVE
) {
255 * No need to synchronise if generation number of current CPU is
256 * newer than the number of this LWP.
258 * This test assumes two's complement arithmetic and allows
259 * ~2B missed updates before it will produce bad results.
263 ucpu
= ci
->ci_data
.cpu_uvm
;
264 if (__predict_true((signed int)(ucpu
->emap_gen
- gen
) >= 0)) {
265 l
->l_emap_gen
= ucpu
->emap_gen
;
271 * Record the current generation _before_ issuing the TLB flush.
272 * No need for a memory barrier before, as reading a stale value
273 * for uvm_emap_gen is not a problem.
275 * pmap_emap_sync() must implicitly perform a full memory barrier,
276 * which prevents us from fetching a value from after the TLB flush
277 * has occurred (which would be bad).
279 * We can race with an interrupt on the current CPU updating the
280 * counter to a newer value. This could cause us to set a stale
281 * value into ucpu->emap_gen, overwriting a newer update from the
282 * interrupt. However, it does not matter since:
283 * (1) Interrupts always run to completion or block.
284 * (2) Interrupts will only ever install a newer value and,
285 * (3) We will roll the value forward later.
287 curgen
= uvm_emap_gen_return();
288 pmap_emap_sync(true);
289 ucpu
->emap_gen
= curgen
;
290 l
->l_emap_gen
= curgen
;
291 KASSERT((signed int)(curgen
- gen
) >= 0);
296 * uvm_emap_produce: increment emap generation counter.
298 * => pmap updates must be globally visible.
299 * => caller must have already entered mappings.
300 * => may be called from both interrupt and thread context.
303 uvm_emap_produce(void)
307 gen
= atomic_inc_uint_nv(&uvm_emap_gen
);
308 if (__predict_false(gen
== UVM_EMAP_INACTIVE
)) {
315 * uvm_emap_update: update global emap generation number for current CPU.
317 * Function is called by MD code (eg. pmap) to take advantage of TLB flushes
318 * initiated for other reasons, that sync the emap as a side effect. Note
319 * update should be performed before the actual TLB flush, to avoid race
320 * with newly generated number.
322 * => can be called from IPI handler, therefore function must be safe.
323 * => should be called _after_ TLB flush.
324 * => emap generation number should be taken _before_ TLB flush.
325 * => must be called with preemption disabled.
328 uvm_emap_update(u_int gen
)
330 struct uvm_cpu
*ucpu
;
333 * See comments in uvm_emap_consume() about memory barriers and
334 * race conditions. Store is atomic if emap_gen size is word.
336 CTASSERT(sizeof(ucpu
->emap_gen
) == sizeof(int));
337 /* XXX: KASSERT(kpreempt_disabled()); */
339 ucpu
= curcpu()->ci_data
.cpu_uvm
;
340 ucpu
->emap_gen
= gen
;
346 * Stubs for architectures which do not support emap.
350 uvm_emap_enter(vaddr_t va
, struct vm_page
**pgs
, u_int npages
)
355 for (n
= 0; n
< npages
; n
++, va
+= PAGE_SIZE
) {
356 pa
= VM_PAGE_TO_PHYS(pgs
[n
]);
357 pmap_kenter_pa(va
, pa
, VM_PROT_READ
, 0);
359 pmap_update(pmap_kernel());
363 uvm_emap_remove(vaddr_t sva
, vsize_t len
)
366 pmap_kremove(sva
, len
);
367 pmap_update(pmap_kernel());