2009-04-04 00:22:44 +00:00
|
|
|
/*-
|
|
|
|
* Copyright (c) 2001 The NetBSD Foundation, Inc.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* This code is derived from software contributed to The NetBSD Foundation
|
|
|
|
* by Matt Thomas <matt@3am-software.com> of Allegro Networks, Inc.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
|
|
* must display the following acknowledgement:
|
|
|
|
* This product includes software developed by the NetBSD
|
|
|
|
* Foundation, Inc. and its contributors.
|
|
|
|
* 4. Neither the name of The NetBSD Foundation nor the names of its
|
|
|
|
* contributors may be used to endorse or promote products derived
|
|
|
|
* from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
|
|
|
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
|
|
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
|
|
|
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
/*-
|
|
|
|
* Copyright (C) 1995, 1996 Wolfgang Solfrank.
|
|
|
|
* Copyright (C) 1995, 1996 TooLs GmbH.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. All advertising materials mentioning features or use of this software
|
|
|
|
* must display the following acknowledgement:
|
|
|
|
* This product includes software developed by TooLs GmbH.
|
|
|
|
* 4. The name of TooLs GmbH may not be used to endorse or promote products
|
|
|
|
* derived from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY TOOLS GMBH ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
* IN NO EVENT SHALL TOOLS GMBH BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
|
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
|
|
|
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
|
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
|
|
|
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
|
|
|
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
* $NetBSD: pmap.c,v 1.28 2000/03/26 20:42:36 kleink Exp $
|
|
|
|
*/
|
|
|
|
/*-
|
|
|
|
* Copyright (C) 2001 Benno Rice.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY Benno Rice ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
* IN NO EVENT SHALL TOOLS GMBH BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
|
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
|
|
|
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
|
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
|
|
|
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
|
|
|
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/cdefs.h>
|
|
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Manages physical address maps.
|
|
|
|
*
|
|
|
|
* In addition to hardware address maps, this module is called upon to
|
|
|
|
* provide software-use-only maps which may or may not be stored in the
|
|
|
|
* same form as hardware maps. These pseudo-maps are used to store
|
|
|
|
* intermediate results from copy operations to and from address spaces.
|
|
|
|
*
|
|
|
|
* Since the information managed by this module is also stored by the
|
|
|
|
* logical address mapping module, this module may throw away valid virtual
|
|
|
|
* to physical mappings at almost any time. However, invalidations of
|
|
|
|
* mappings must be done as requested.
|
|
|
|
*
|
|
|
|
* In order to cope with hardware architectures which make virtual to
|
|
|
|
* physical map invalidates expensive, this module may delay invalidate
|
|
|
|
* reduced protection operations until such time as they are actually
|
|
|
|
* necessary. This module is given full information as to which processors
|
|
|
|
* are currently using which maps, and to when physical maps must be made
|
|
|
|
* correct.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "opt_kstack_pages.h"
|
|
|
|
|
|
|
|
#include <sys/param.h>
|
|
|
|
#include <sys/kernel.h>
|
|
|
|
#include <sys/ktr.h>
|
|
|
|
#include <sys/lock.h>
|
|
|
|
#include <sys/msgbuf.h>
|
|
|
|
#include <sys/mutex.h>
|
|
|
|
#include <sys/proc.h>
|
|
|
|
#include <sys/sysctl.h>
|
|
|
|
#include <sys/systm.h>
|
|
|
|
#include <sys/vmmeter.h>
|
|
|
|
|
|
|
|
#include <sys/kdb.h>
|
|
|
|
|
|
|
|
#include <dev/ofw/openfirm.h>
|
|
|
|
|
|
|
|
#include <vm/vm.h>
|
|
|
|
#include <vm/vm_param.h>
|
|
|
|
#include <vm/vm_kern.h>
|
|
|
|
#include <vm/vm_page.h>
|
|
|
|
#include <vm/vm_map.h>
|
|
|
|
#include <vm/vm_object.h>
|
|
|
|
#include <vm/vm_extern.h>
|
|
|
|
#include <vm/vm_pageout.h>
|
|
|
|
#include <vm/vm_pager.h>
|
|
|
|
#include <vm/uma.h>
|
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
#include <machine/_inttypes.h>
|
2009-04-04 00:22:44 +00:00
|
|
|
#include <machine/cpu.h>
|
2009-05-14 00:34:26 +00:00
|
|
|
#include <machine/platform.h>
|
2009-04-04 00:22:44 +00:00
|
|
|
#include <machine/frame.h>
|
|
|
|
#include <machine/md_var.h>
|
|
|
|
#include <machine/psl.h>
|
|
|
|
#include <machine/bat.h>
|
2010-07-13 05:32:19 +00:00
|
|
|
#include <machine/hid.h>
|
2009-04-04 00:22:44 +00:00
|
|
|
#include <machine/pte.h>
|
|
|
|
#include <machine/sr.h>
|
|
|
|
#include <machine/trap.h>
|
|
|
|
#include <machine/mmuvar.h>
|
|
|
|
|
|
|
|
#include "mmu_if.h"
|
|
|
|
|
|
|
|
#define MOEA_DEBUG
|
|
|
|
|
|
|
|
#define TODO panic("%s: not implemented", __func__);
|
2010-07-13 05:32:19 +00:00
|
|
|
void moea64_release_vsid(uint64_t vsid);
|
|
|
|
uintptr_t moea64_get_unique_vsid(void);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
static __inline register_t
|
|
|
|
cntlzd(volatile register_t a) {
|
|
|
|
register_t b;
|
|
|
|
__asm ("cntlzd %0, %1" : "=r"(b) : "r"(a));
|
2009-04-04 00:22:44 +00:00
|
|
|
return b;
|
|
|
|
}
|
|
|
|
|
2010-02-18 15:00:43 +00:00
|
|
|
#define PTESYNC() __asm __volatile("ptesync");
|
2009-04-04 00:22:44 +00:00
|
|
|
#define TLBSYNC() __asm __volatile("tlbsync; ptesync");
|
|
|
|
#define SYNC() __asm __volatile("sync");
|
|
|
|
#define EIEIO() __asm __volatile("eieio");
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The tlbie instruction must be executed in 64-bit mode
|
|
|
|
* so we have to twiddle MSR[SF] around every invocation.
|
|
|
|
* Just to add to the fun, exceptions must be off as well
|
|
|
|
* so that we can't trap in 64-bit mode. What a pain.
|
|
|
|
*/
|
2009-10-23 03:17:02 +00:00
|
|
|
struct mtx tlbie_mutex;
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
static __inline void
|
2010-07-13 05:32:19 +00:00
|
|
|
TLBIE(uint64_t vpn) {
|
|
|
|
#ifndef __powerpc64__
|
2009-04-04 00:22:44 +00:00
|
|
|
register_t vpn_hi, vpn_lo;
|
2009-10-23 03:17:02 +00:00
|
|
|
register_t msr;
|
|
|
|
register_t scratch;
|
2010-07-13 05:32:19 +00:00
|
|
|
#endif
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
vpn <<= ADDR_PIDX_SHFT;
|
2010-02-18 15:00:43 +00:00
|
|
|
vpn &= ~(0xffffULL << 48);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
mtx_lock_spin(&tlbie_mutex);
|
|
|
|
#ifdef __powerpc64__
|
|
|
|
__asm __volatile("\
|
|
|
|
ptesync; \
|
|
|
|
tlbie %0; \
|
|
|
|
eieio; \
|
|
|
|
tlbsync; \
|
|
|
|
ptesync;"
|
|
|
|
:: "r"(vpn) : "memory");
|
|
|
|
#else
|
2009-04-04 00:22:44 +00:00
|
|
|
vpn_hi = (uint32_t)(vpn >> 32);
|
|
|
|
vpn_lo = (uint32_t)vpn;
|
|
|
|
|
|
|
|
__asm __volatile("\
|
|
|
|
mfmsr %0; \
|
2010-02-18 15:00:43 +00:00
|
|
|
mr %1, %0; \
|
2009-10-23 03:17:02 +00:00
|
|
|
insrdi %1,%5,1,0; \
|
2009-04-04 00:22:44 +00:00
|
|
|
mtmsrd %1; \
|
|
|
|
ptesync; \
|
|
|
|
\
|
|
|
|
sld %1,%2,%4; \
|
|
|
|
or %1,%1,%3; \
|
|
|
|
tlbie %1; \
|
|
|
|
\
|
|
|
|
mtmsrd %0; \
|
|
|
|
eieio; \
|
|
|
|
tlbsync; \
|
|
|
|
ptesync;"
|
2010-02-18 15:00:43 +00:00
|
|
|
: "=r"(msr), "=r"(scratch) : "r"(vpn_hi), "r"(vpn_lo), "r"(32), "r"(1)
|
|
|
|
: "memory");
|
2010-07-13 05:32:19 +00:00
|
|
|
#endif
|
2009-10-23 03:17:02 +00:00
|
|
|
mtx_unlock_spin(&tlbie_mutex);
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#define DISABLE_TRANS(msr) msr = mfmsr(); mtmsr(msr & ~PSL_DR); isync()
|
|
|
|
#define ENABLE_TRANS(msr) mtmsr(msr); isync()
|
|
|
|
|
|
|
|
#define VSID_MAKE(sr, hash) ((sr) | (((hash) & 0xfffff) << 4))
|
|
|
|
#define VSID_TO_HASH(vsid) (((vsid) >> 4) & 0xfffff)
|
2010-02-24 00:54:37 +00:00
|
|
|
#define VSID_HASH_MASK 0x0000007fffffffffULL
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-02-18 15:00:43 +00:00
|
|
|
#define PVO_PTEGIDX_MASK 0x007UL /* which PTEG slot */
|
|
|
|
#define PVO_PTEGIDX_VALID 0x008UL /* slot is valid */
|
|
|
|
#define PVO_WIRED 0x010UL /* PVO entry is wired */
|
|
|
|
#define PVO_MANAGED 0x020UL /* PVO entry is managed */
|
|
|
|
#define PVO_BOOTSTRAP 0x080UL /* PVO entry allocated during
|
2009-04-04 00:22:44 +00:00
|
|
|
bootstrap */
|
2010-02-18 15:00:43 +00:00
|
|
|
#define PVO_FAKE 0x100UL /* fictitious phys page */
|
2010-07-13 05:32:19 +00:00
|
|
|
#define PVO_LARGE 0x200UL /* large page */
|
2009-04-04 00:22:44 +00:00
|
|
|
#define PVO_VADDR(pvo) ((pvo)->pvo_vaddr & ~ADDR_POFF)
|
|
|
|
#define PVO_ISFAKE(pvo) ((pvo)->pvo_vaddr & PVO_FAKE)
|
|
|
|
#define PVO_PTEGIDX_GET(pvo) ((pvo)->pvo_vaddr & PVO_PTEGIDX_MASK)
|
|
|
|
#define PVO_PTEGIDX_ISSET(pvo) ((pvo)->pvo_vaddr & PVO_PTEGIDX_VALID)
|
|
|
|
#define PVO_PTEGIDX_CLR(pvo) \
|
|
|
|
((void)((pvo)->pvo_vaddr &= ~(PVO_PTEGIDX_VALID|PVO_PTEGIDX_MASK)))
|
|
|
|
#define PVO_PTEGIDX_SET(pvo, i) \
|
|
|
|
((void)((pvo)->pvo_vaddr |= (i)|PVO_PTEGIDX_VALID))
|
2010-07-13 05:32:19 +00:00
|
|
|
#define PVO_VSID(pvo) ((pvo)->pvo_vpn >> 16)
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
#define MOEA_PVO_CHECK(pvo)
|
|
|
|
|
|
|
|
#define LOCK_TABLE() mtx_lock(&moea64_table_mutex)
|
|
|
|
#define UNLOCK_TABLE() mtx_unlock(&moea64_table_mutex);
|
|
|
|
#define ASSERT_TABLE_LOCK() mtx_assert(&moea64_table_mutex, MA_OWNED)
|
|
|
|
|
|
|
|
struct ofw_map {
|
2010-07-13 05:32:19 +00:00
|
|
|
cell_t om_va;
|
|
|
|
cell_t om_len;
|
|
|
|
cell_t om_pa_hi;
|
|
|
|
cell_t om_pa_lo;
|
|
|
|
cell_t om_mode;
|
2009-04-04 00:22:44 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Map of physical memory regions.
|
|
|
|
*/
|
|
|
|
static struct mem_region *regions;
|
|
|
|
static struct mem_region *pregions;
|
2010-07-13 05:32:19 +00:00
|
|
|
static u_int phys_avail_count;
|
|
|
|
static int regions_sz, pregions_sz;
|
2009-04-04 00:22:44 +00:00
|
|
|
extern int ofw_real_mode;
|
|
|
|
|
|
|
|
extern struct pmap ofw_pmap;
|
|
|
|
|
|
|
|
extern void bs_remap_earlyboot(void);
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Lock for the pteg and pvo tables.
|
|
|
|
*/
|
|
|
|
struct mtx moea64_table_mutex;
|
2010-08-29 18:17:38 +00:00
|
|
|
struct mtx moea64_slb_mutex;
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* PTEG data.
|
|
|
|
*/
|
|
|
|
static struct lpteg *moea64_pteg_table;
|
|
|
|
u_int moea64_pteg_count;
|
|
|
|
u_int moea64_pteg_mask;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* PVO data.
|
|
|
|
*/
|
|
|
|
struct pvo_head *moea64_pvo_table; /* pvo entries by pteg index */
|
|
|
|
/* lists of unmanaged pages */
|
|
|
|
struct pvo_head moea64_pvo_kunmanaged =
|
|
|
|
LIST_HEAD_INITIALIZER(moea64_pvo_kunmanaged);
|
|
|
|
struct pvo_head moea64_pvo_unmanaged =
|
|
|
|
LIST_HEAD_INITIALIZER(moea64_pvo_unmanaged);
|
|
|
|
|
|
|
|
uma_zone_t moea64_upvo_zone; /* zone for pvo entries for unmanaged pages */
|
|
|
|
uma_zone_t moea64_mpvo_zone; /* zone for pvo entries for managed pages */
|
|
|
|
|
|
|
|
#define BPVO_POOL_SIZE 327680
|
|
|
|
static struct pvo_entry *moea64_bpvo_pool;
|
|
|
|
static int moea64_bpvo_pool_index = 0;
|
|
|
|
|
|
|
|
#define VSID_NBPW (sizeof(u_int32_t) * 8)
|
2010-07-13 05:32:19 +00:00
|
|
|
#ifdef __powerpc64__
|
|
|
|
#define NVSIDS (NPMAPS * 16)
|
|
|
|
#define VSID_HASHMASK 0xffffffffUL
|
|
|
|
#else
|
|
|
|
#define NVSIDS NPMAPS
|
|
|
|
#define VSID_HASHMASK 0xfffffUL
|
|
|
|
#endif
|
|
|
|
static u_int moea64_vsid_bitmap[NVSIDS / VSID_NBPW];
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
static boolean_t moea64_initialized = FALSE;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Statistics.
|
|
|
|
*/
|
|
|
|
u_int moea64_pte_valid = 0;
|
|
|
|
u_int moea64_pte_overflow = 0;
|
|
|
|
u_int moea64_pvo_entries = 0;
|
|
|
|
u_int moea64_pvo_enter_calls = 0;
|
|
|
|
u_int moea64_pvo_remove_calls = 0;
|
|
|
|
SYSCTL_INT(_machdep, OID_AUTO, moea64_pte_valid, CTLFLAG_RD,
|
|
|
|
&moea64_pte_valid, 0, "");
|
|
|
|
SYSCTL_INT(_machdep, OID_AUTO, moea64_pte_overflow, CTLFLAG_RD,
|
|
|
|
&moea64_pte_overflow, 0, "");
|
|
|
|
SYSCTL_INT(_machdep, OID_AUTO, moea64_pvo_entries, CTLFLAG_RD,
|
|
|
|
&moea64_pvo_entries, 0, "");
|
|
|
|
SYSCTL_INT(_machdep, OID_AUTO, moea64_pvo_enter_calls, CTLFLAG_RD,
|
|
|
|
&moea64_pvo_enter_calls, 0, "");
|
|
|
|
SYSCTL_INT(_machdep, OID_AUTO, moea64_pvo_remove_calls, CTLFLAG_RD,
|
|
|
|
&moea64_pvo_remove_calls, 0, "");
|
|
|
|
|
|
|
|
vm_offset_t moea64_scratchpage_va[2];
|
2010-07-13 05:32:19 +00:00
|
|
|
uint64_t moea64_scratchpage_vpn[2];
|
2009-04-04 00:22:44 +00:00
|
|
|
struct lpte *moea64_scratchpage_pte[2];
|
|
|
|
struct mtx moea64_scratchpage_mtx;
|
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
uint64_t moea64_large_page_mask = 0;
|
|
|
|
int moea64_large_page_size = 0;
|
|
|
|
int moea64_large_page_shift = 0;
|
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
/*
|
|
|
|
* Allocate physical memory for use in moea64_bootstrap.
|
|
|
|
*/
|
|
|
|
static vm_offset_t moea64_bootstrap_alloc(vm_size_t, u_int);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* PTE calls.
|
|
|
|
*/
|
|
|
|
static int moea64_pte_insert(u_int, struct lpte *);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* PVO calls.
|
|
|
|
*/
|
|
|
|
static int moea64_pvo_enter(pmap_t, uma_zone_t, struct pvo_head *,
|
2009-10-23 03:17:02 +00:00
|
|
|
vm_offset_t, vm_offset_t, uint64_t, int);
|
2010-07-13 05:32:19 +00:00
|
|
|
static void moea64_pvo_remove(struct pvo_entry *);
|
|
|
|
static struct pvo_entry *moea64_pvo_find_va(pmap_t, vm_offset_t);
|
|
|
|
static struct lpte *moea64_pvo_to_pte(const struct pvo_entry *);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Utility routines.
|
|
|
|
*/
|
2010-07-13 05:32:19 +00:00
|
|
|
static void moea64_bootstrap(mmu_t mmup,
|
2009-04-04 00:22:44 +00:00
|
|
|
vm_offset_t kernelstart, vm_offset_t kernelend);
|
2010-07-13 05:32:19 +00:00
|
|
|
static void moea64_cpu_bootstrap(mmu_t, int ap);
|
2009-04-04 00:22:44 +00:00
|
|
|
static void moea64_enter_locked(pmap_t, vm_offset_t, vm_page_t,
|
|
|
|
vm_prot_t, boolean_t);
|
|
|
|
static boolean_t moea64_query_bit(vm_page_t, u_int64_t);
|
2010-06-10 16:56:35 +00:00
|
|
|
static u_int moea64_clear_bit(vm_page_t, u_int64_t);
|
2009-04-04 00:22:44 +00:00
|
|
|
static void moea64_kremove(mmu_t, vm_offset_t);
|
|
|
|
static void moea64_syncicache(pmap_t pmap, vm_offset_t va,
|
2009-10-21 18:38:02 +00:00
|
|
|
vm_offset_t pa, vm_size_t sz);
|
2009-04-04 00:22:44 +00:00
|
|
|
static void tlbia(void);
|
2010-07-13 05:32:19 +00:00
|
|
|
#ifdef __powerpc64__
|
|
|
|
static void slbia(void);
|
|
|
|
#endif
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Kernel MMU interface
|
|
|
|
*/
|
|
|
|
void moea64_change_wiring(mmu_t, pmap_t, vm_offset_t, boolean_t);
|
|
|
|
void moea64_clear_modify(mmu_t, vm_page_t);
|
|
|
|
void moea64_clear_reference(mmu_t, vm_page_t);
|
|
|
|
void moea64_copy_page(mmu_t, vm_page_t, vm_page_t);
|
|
|
|
void moea64_enter(mmu_t, pmap_t, vm_offset_t, vm_page_t, vm_prot_t, boolean_t);
|
|
|
|
void moea64_enter_object(mmu_t, pmap_t, vm_offset_t, vm_offset_t, vm_page_t,
|
|
|
|
vm_prot_t);
|
|
|
|
void moea64_enter_quick(mmu_t, pmap_t, vm_offset_t, vm_page_t, vm_prot_t);
|
|
|
|
vm_paddr_t moea64_extract(mmu_t, pmap_t, vm_offset_t);
|
|
|
|
vm_page_t moea64_extract_and_hold(mmu_t, pmap_t, vm_offset_t, vm_prot_t);
|
|
|
|
void moea64_init(mmu_t);
|
|
|
|
boolean_t moea64_is_modified(mmu_t, vm_page_t);
|
Resurrect pmap_is_referenced() and use it in mincore(). Essentially,
pmap_ts_referenced() is not always appropriate for checking whether or
not pages have been referenced because it clears any reference bits
that it encounters. For example, in mincore(), clearing the reference
bits has two negative consequences. First, it throws off the activity
count calculations performed by the page daemon. Specifically, a page
on which mincore() has called pmap_ts_referenced() looks less active
to the page daemon than it should. Consequently, the page could be
deactivated prematurely by the page daemon. Arguably, this problem
could be fixed by having mincore() duplicate the activity count
calculation on the page. However, there is a second problem for which
that is not a solution. In order to clear a reference on a 4KB page,
it may be necessary to demote a 2/4MB page mapping. Thus, a mincore()
by one process can have the side effect of demoting a superpage
mapping within another process!
2010-04-24 17:32:52 +00:00
|
|
|
boolean_t moea64_is_referenced(mmu_t, vm_page_t);
|
2009-04-04 00:22:44 +00:00
|
|
|
boolean_t moea64_ts_referenced(mmu_t, vm_page_t);
|
|
|
|
vm_offset_t moea64_map(mmu_t, vm_offset_t *, vm_offset_t, vm_offset_t, int);
|
|
|
|
boolean_t moea64_page_exists_quick(mmu_t, pmap_t, vm_page_t);
|
|
|
|
int moea64_page_wired_mappings(mmu_t, vm_page_t);
|
|
|
|
void moea64_pinit(mmu_t, pmap_t);
|
|
|
|
void moea64_pinit0(mmu_t, pmap_t);
|
|
|
|
void moea64_protect(mmu_t, pmap_t, vm_offset_t, vm_offset_t, vm_prot_t);
|
|
|
|
void moea64_qenter(mmu_t, vm_offset_t, vm_page_t *, int);
|
|
|
|
void moea64_qremove(mmu_t, vm_offset_t, int);
|
|
|
|
void moea64_release(mmu_t, pmap_t);
|
|
|
|
void moea64_remove(mmu_t, pmap_t, vm_offset_t, vm_offset_t);
|
|
|
|
void moea64_remove_all(mmu_t, vm_page_t);
|
|
|
|
void moea64_remove_write(mmu_t, vm_page_t);
|
|
|
|
void moea64_zero_page(mmu_t, vm_page_t);
|
|
|
|
void moea64_zero_page_area(mmu_t, vm_page_t, int, int);
|
|
|
|
void moea64_zero_page_idle(mmu_t, vm_page_t);
|
|
|
|
void moea64_activate(mmu_t, struct thread *);
|
|
|
|
void moea64_deactivate(mmu_t, struct thread *);
|
|
|
|
void *moea64_mapdev(mmu_t, vm_offset_t, vm_size_t);
|
|
|
|
void moea64_unmapdev(mmu_t, vm_offset_t, vm_size_t);
|
|
|
|
vm_offset_t moea64_kextract(mmu_t, vm_offset_t);
|
|
|
|
void moea64_kenter(mmu_t, vm_offset_t, vm_offset_t);
|
|
|
|
boolean_t moea64_dev_direct_mapped(mmu_t, vm_offset_t, vm_size_t);
|
2009-10-21 18:38:02 +00:00
|
|
|
static void moea64_sync_icache(mmu_t, pmap_t, vm_offset_t, vm_size_t);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
static mmu_method_t moea64_methods[] = {
|
2009-04-04 00:22:44 +00:00
|
|
|
MMUMETHOD(mmu_change_wiring, moea64_change_wiring),
|
|
|
|
MMUMETHOD(mmu_clear_modify, moea64_clear_modify),
|
|
|
|
MMUMETHOD(mmu_clear_reference, moea64_clear_reference),
|
|
|
|
MMUMETHOD(mmu_copy_page, moea64_copy_page),
|
|
|
|
MMUMETHOD(mmu_enter, moea64_enter),
|
|
|
|
MMUMETHOD(mmu_enter_object, moea64_enter_object),
|
|
|
|
MMUMETHOD(mmu_enter_quick, moea64_enter_quick),
|
|
|
|
MMUMETHOD(mmu_extract, moea64_extract),
|
|
|
|
MMUMETHOD(mmu_extract_and_hold, moea64_extract_and_hold),
|
|
|
|
MMUMETHOD(mmu_init, moea64_init),
|
|
|
|
MMUMETHOD(mmu_is_modified, moea64_is_modified),
|
Resurrect pmap_is_referenced() and use it in mincore(). Essentially,
pmap_ts_referenced() is not always appropriate for checking whether or
not pages have been referenced because it clears any reference bits
that it encounters. For example, in mincore(), clearing the reference
bits has two negative consequences. First, it throws off the activity
count calculations performed by the page daemon. Specifically, a page
on which mincore() has called pmap_ts_referenced() looks less active
to the page daemon than it should. Consequently, the page could be
deactivated prematurely by the page daemon. Arguably, this problem
could be fixed by having mincore() duplicate the activity count
calculation on the page. However, there is a second problem for which
that is not a solution. In order to clear a reference on a 4KB page,
it may be necessary to demote a 2/4MB page mapping. Thus, a mincore()
by one process can have the side effect of demoting a superpage
mapping within another process!
2010-04-24 17:32:52 +00:00
|
|
|
MMUMETHOD(mmu_is_referenced, moea64_is_referenced),
|
2009-04-04 00:22:44 +00:00
|
|
|
MMUMETHOD(mmu_ts_referenced, moea64_ts_referenced),
|
|
|
|
MMUMETHOD(mmu_map, moea64_map),
|
|
|
|
MMUMETHOD(mmu_page_exists_quick,moea64_page_exists_quick),
|
|
|
|
MMUMETHOD(mmu_page_wired_mappings,moea64_page_wired_mappings),
|
|
|
|
MMUMETHOD(mmu_pinit, moea64_pinit),
|
|
|
|
MMUMETHOD(mmu_pinit0, moea64_pinit0),
|
|
|
|
MMUMETHOD(mmu_protect, moea64_protect),
|
|
|
|
MMUMETHOD(mmu_qenter, moea64_qenter),
|
|
|
|
MMUMETHOD(mmu_qremove, moea64_qremove),
|
|
|
|
MMUMETHOD(mmu_release, moea64_release),
|
|
|
|
MMUMETHOD(mmu_remove, moea64_remove),
|
|
|
|
MMUMETHOD(mmu_remove_all, moea64_remove_all),
|
|
|
|
MMUMETHOD(mmu_remove_write, moea64_remove_write),
|
2009-10-21 18:38:02 +00:00
|
|
|
MMUMETHOD(mmu_sync_icache, moea64_sync_icache),
|
2009-04-04 00:22:44 +00:00
|
|
|
MMUMETHOD(mmu_zero_page, moea64_zero_page),
|
|
|
|
MMUMETHOD(mmu_zero_page_area, moea64_zero_page_area),
|
|
|
|
MMUMETHOD(mmu_zero_page_idle, moea64_zero_page_idle),
|
|
|
|
MMUMETHOD(mmu_activate, moea64_activate),
|
|
|
|
MMUMETHOD(mmu_deactivate, moea64_deactivate),
|
|
|
|
|
|
|
|
/* Internal interfaces */
|
2010-07-13 05:32:19 +00:00
|
|
|
MMUMETHOD(mmu_bootstrap, moea64_bootstrap),
|
|
|
|
MMUMETHOD(mmu_cpu_bootstrap, moea64_cpu_bootstrap),
|
2009-04-04 00:22:44 +00:00
|
|
|
MMUMETHOD(mmu_mapdev, moea64_mapdev),
|
|
|
|
MMUMETHOD(mmu_unmapdev, moea64_unmapdev),
|
|
|
|
MMUMETHOD(mmu_kextract, moea64_kextract),
|
|
|
|
MMUMETHOD(mmu_kenter, moea64_kenter),
|
|
|
|
MMUMETHOD(mmu_dev_direct_mapped,moea64_dev_direct_mapped),
|
|
|
|
|
|
|
|
{ 0, 0 }
|
|
|
|
};
|
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
static mmu_def_t oea64_mmu = {
|
2009-04-04 00:22:44 +00:00
|
|
|
MMU_TYPE_G5,
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_methods,
|
2009-04-04 00:22:44 +00:00
|
|
|
0
|
|
|
|
};
|
2010-07-13 05:32:19 +00:00
|
|
|
MMU_DEF(oea64_mmu);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
static __inline u_int
|
2010-07-13 05:32:19 +00:00
|
|
|
va_to_pteg(uint64_t vsid, vm_offset_t addr, int large)
|
2009-04-04 00:22:44 +00:00
|
|
|
{
|
2010-02-24 00:54:37 +00:00
|
|
|
uint64_t hash;
|
2010-07-13 05:32:19 +00:00
|
|
|
int shift;
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
shift = large ? moea64_large_page_shift : ADDR_PIDX_SHFT;
|
2010-02-24 00:54:37 +00:00
|
|
|
hash = (vsid & VSID_HASH_MASK) ^ (((uint64_t)addr & ADDR_PIDX) >>
|
2010-07-13 05:32:19 +00:00
|
|
|
shift);
|
2009-04-04 00:22:44 +00:00
|
|
|
return (hash & moea64_pteg_mask);
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline struct pvo_head *
|
|
|
|
pa_to_pvoh(vm_offset_t pa, vm_page_t *pg_p)
|
|
|
|
{
|
|
|
|
struct vm_page *pg;
|
|
|
|
|
|
|
|
pg = PHYS_TO_VM_PAGE(pa);
|
|
|
|
|
|
|
|
if (pg_p != NULL)
|
|
|
|
*pg_p = pg;
|
|
|
|
|
|
|
|
if (pg == NULL)
|
|
|
|
return (&moea64_pvo_unmanaged);
|
|
|
|
|
|
|
|
return (&pg->md.mdpg_pvoh);
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline struct pvo_head *
|
|
|
|
vm_page_to_pvoh(vm_page_t m)
|
|
|
|
{
|
|
|
|
|
|
|
|
return (&m->md.mdpg_pvoh);
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline void
|
|
|
|
moea64_attr_clear(vm_page_t m, u_int64_t ptebit)
|
|
|
|
{
|
|
|
|
|
|
|
|
mtx_assert(&vm_page_queue_mtx, MA_OWNED);
|
|
|
|
m->md.mdpg_attrs &= ~ptebit;
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline u_int64_t
|
|
|
|
moea64_attr_fetch(vm_page_t m)
|
|
|
|
{
|
|
|
|
|
|
|
|
return (m->md.mdpg_attrs);
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline void
|
|
|
|
moea64_attr_save(vm_page_t m, u_int64_t ptebit)
|
|
|
|
{
|
|
|
|
|
|
|
|
mtx_assert(&vm_page_queue_mtx, MA_OWNED);
|
|
|
|
m->md.mdpg_attrs |= ptebit;
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline void
|
|
|
|
moea64_pte_create(struct lpte *pt, uint64_t vsid, vm_offset_t va,
|
2010-07-13 05:32:19 +00:00
|
|
|
uint64_t pte_lo, int flags)
|
2009-04-04 00:22:44 +00:00
|
|
|
{
|
2010-07-13 05:32:19 +00:00
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
ASSERT_TABLE_LOCK();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Construct a PTE. Default to IMB initially. Valid bit only gets
|
|
|
|
* set when the real pte is set in memory.
|
|
|
|
*
|
|
|
|
* Note: Don't set the valid bit for correct operation of tlb update.
|
|
|
|
*/
|
|
|
|
pt->pte_hi = (vsid << LPTE_VSID_SHIFT) |
|
|
|
|
(((uint64_t)(va & ADDR_PIDX) >> ADDR_API_SHFT64) & LPTE_API);
|
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
if (flags & PVO_LARGE)
|
|
|
|
pt->pte_hi |= LPTE_BIG;
|
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
pt->pte_lo = pte_lo;
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline void
|
|
|
|
moea64_pte_synch(struct lpte *pt, struct lpte *pvo_pt)
|
|
|
|
{
|
|
|
|
|
|
|
|
ASSERT_TABLE_LOCK();
|
|
|
|
|
|
|
|
pvo_pt->pte_lo |= pt->pte_lo & (LPTE_REF | LPTE_CHG);
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline void
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_pte_clear(struct lpte *pt, uint64_t vpn, u_int64_t ptebit)
|
2009-04-04 00:22:44 +00:00
|
|
|
{
|
|
|
|
ASSERT_TABLE_LOCK();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* As shown in Section 7.6.3.2.3
|
|
|
|
*/
|
|
|
|
pt->pte_lo &= ~ptebit;
|
2010-07-13 05:32:19 +00:00
|
|
|
TLBIE(vpn);
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static __inline void
|
|
|
|
moea64_pte_set(struct lpte *pt, struct lpte *pvo_pt)
|
|
|
|
{
|
|
|
|
|
|
|
|
ASSERT_TABLE_LOCK();
|
|
|
|
pvo_pt->pte_hi |= LPTE_VALID;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Update the PTE as defined in section 7.6.3.1.
|
|
|
|
* Note that the REF/CHG bits are from pvo_pt and thus should have
|
|
|
|
* been saved so this routine can restore them (if desired).
|
|
|
|
*/
|
|
|
|
pt->pte_lo = pvo_pt->pte_lo;
|
|
|
|
EIEIO();
|
|
|
|
pt->pte_hi = pvo_pt->pte_hi;
|
2010-02-18 15:00:43 +00:00
|
|
|
PTESYNC();
|
2009-04-04 00:22:44 +00:00
|
|
|
moea64_pte_valid++;
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline void
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_pte_unset(struct lpte *pt, struct lpte *pvo_pt, uint64_t vpn)
|
2009-04-04 00:22:44 +00:00
|
|
|
{
|
|
|
|
ASSERT_TABLE_LOCK();
|
|
|
|
pvo_pt->pte_hi &= ~LPTE_VALID;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Force the reg & chg bits back into the PTEs.
|
|
|
|
*/
|
|
|
|
SYNC();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Invalidate the pte.
|
|
|
|
*/
|
|
|
|
pt->pte_hi &= ~LPTE_VALID;
|
2010-07-13 05:32:19 +00:00
|
|
|
TLBIE(vpn);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Save the reg & chg bits.
|
|
|
|
*/
|
|
|
|
moea64_pte_synch(pt, pvo_pt);
|
|
|
|
moea64_pte_valid--;
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline void
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_pte_change(struct lpte *pt, struct lpte *pvo_pt, uint64_t vpn)
|
2009-04-04 00:22:44 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Invalidate the PTE
|
|
|
|
*/
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_pte_unset(pt, pvo_pt, vpn);
|
2009-04-04 00:22:44 +00:00
|
|
|
moea64_pte_set(pt, pvo_pt);
|
|
|
|
}
|
|
|
|
|
|
|
|
static __inline uint64_t
|
|
|
|
moea64_calc_wimg(vm_offset_t pa)
|
|
|
|
{
|
|
|
|
uint64_t pte_lo;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Assume the page is cache inhibited and access is guarded unless
|
|
|
|
* it's in our available memory array.
|
|
|
|
*/
|
|
|
|
pte_lo = LPTE_I | LPTE_G;
|
|
|
|
for (i = 0; i < pregions_sz; i++) {
|
|
|
|
if ((pa >= pregions[i].mr_start) &&
|
|
|
|
(pa < (pregions[i].mr_start + pregions[i].mr_size))) {
|
|
|
|
pte_lo &= ~(LPTE_I | LPTE_G);
|
|
|
|
pte_lo |= LPTE_M;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return pte_lo;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Quick sort callout for comparing memory regions.
|
|
|
|
*/
|
|
|
|
static int mr_cmp(const void *a, const void *b);
|
|
|
|
static int om_cmp(const void *a, const void *b);
|
|
|
|
|
|
|
|
static int
|
|
|
|
mr_cmp(const void *a, const void *b)
|
|
|
|
{
|
|
|
|
const struct mem_region *regiona;
|
|
|
|
const struct mem_region *regionb;
|
|
|
|
|
|
|
|
regiona = a;
|
|
|
|
regionb = b;
|
|
|
|
if (regiona->mr_start < regionb->mr_start)
|
|
|
|
return (-1);
|
|
|
|
else if (regiona->mr_start > regionb->mr_start)
|
|
|
|
return (1);
|
|
|
|
else
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
om_cmp(const void *a, const void *b)
|
|
|
|
{
|
|
|
|
const struct ofw_map *mapa;
|
|
|
|
const struct ofw_map *mapb;
|
|
|
|
|
|
|
|
mapa = a;
|
|
|
|
mapb = b;
|
|
|
|
if (mapa->om_pa_hi < mapb->om_pa_hi)
|
|
|
|
return (-1);
|
|
|
|
else if (mapa->om_pa_hi > mapb->om_pa_hi)
|
|
|
|
return (1);
|
|
|
|
else if (mapa->om_pa_lo < mapb->om_pa_lo)
|
|
|
|
return (-1);
|
|
|
|
else if (mapa->om_pa_lo > mapb->om_pa_lo)
|
|
|
|
return (1);
|
|
|
|
else
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_cpu_bootstrap(mmu_t mmup, int ap)
|
2009-04-04 00:22:44 +00:00
|
|
|
{
|
|
|
|
int i = 0;
|
2010-07-13 05:32:19 +00:00
|
|
|
#ifdef __powerpc64__
|
|
|
|
struct slb *slb = PCPU_GET(slb);
|
|
|
|
#endif
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize segment registers and MMU
|
|
|
|
*/
|
|
|
|
|
|
|
|
mtmsr(mfmsr() & ~PSL_DR & ~PSL_IR); isync();
|
2010-07-13 05:32:19 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Install kernel SLB entries
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef __powerpc64__
|
|
|
|
slbia();
|
|
|
|
|
|
|
|
for (i = 0; i < 64; i++) {
|
|
|
|
if (!(slb[i].slbe & SLBE_VALID))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
__asm __volatile ("slbmte %0, %1" ::
|
|
|
|
"r"(slb[i].slbv), "r"(slb[i].slbe));
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
for (i = 0; i < 16; i++)
|
|
|
|
mtsrin(i << ADDR_SR_SHFT, kernel_pmap->pm_sr[i]);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Install page table
|
|
|
|
*/
|
|
|
|
|
2010-02-18 15:00:43 +00:00
|
|
|
__asm __volatile ("ptesync; mtsdr1 %0; isync"
|
2010-07-13 05:32:19 +00:00
|
|
|
:: "r"((uintptr_t)moea64_pteg_table
|
|
|
|
| (64 - cntlzd(moea64_pteg_mask >> 11))));
|
2009-04-04 00:22:44 +00:00
|
|
|
tlbia();
|
|
|
|
}
|
|
|
|
|
2009-11-12 15:19:09 +00:00
|
|
|
static void
|
|
|
|
moea64_add_ofw_mappings(mmu_t mmup, phandle_t mmu, size_t sz)
|
|
|
|
{
|
|
|
|
struct ofw_map translations[sz/sizeof(struct ofw_map)];
|
|
|
|
register_t msr;
|
|
|
|
vm_offset_t off;
|
2010-02-20 16:23:29 +00:00
|
|
|
vm_paddr_t pa_base;
|
2009-11-12 15:19:09 +00:00
|
|
|
int i, ofw_mappings;
|
|
|
|
|
|
|
|
bzero(translations, sz);
|
|
|
|
if (OF_getprop(mmu, "translations", translations, sz) == -1)
|
|
|
|
panic("moea64_bootstrap: can't get ofw translations");
|
|
|
|
|
|
|
|
CTR0(KTR_PMAP, "moea64_add_ofw_mappings: translations");
|
|
|
|
sz /= sizeof(*translations);
|
|
|
|
qsort(translations, sz, sizeof (*translations), om_cmp);
|
|
|
|
|
|
|
|
for (i = 0, ofw_mappings = 0; i < sz; i++) {
|
|
|
|
CTR3(KTR_PMAP, "translation: pa=%#x va=%#x len=%#x",
|
|
|
|
(uint32_t)(translations[i].om_pa_lo), translations[i].om_va,
|
|
|
|
translations[i].om_len);
|
|
|
|
|
|
|
|
if (translations[i].om_pa_lo % PAGE_SIZE)
|
|
|
|
panic("OFW translation not page-aligned!");
|
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
pa_base = translations[i].om_pa_lo;
|
|
|
|
|
|
|
|
#ifdef __powerpc64__
|
|
|
|
pa_base += (vm_offset_t)translations[i].om_pa_hi << 32;
|
|
|
|
#else
|
2009-11-12 15:19:09 +00:00
|
|
|
if (translations[i].om_pa_hi)
|
|
|
|
panic("OFW translations above 32-bit boundary!");
|
2010-07-13 05:32:19 +00:00
|
|
|
#endif
|
2010-02-20 16:23:29 +00:00
|
|
|
|
2009-11-12 15:19:09 +00:00
|
|
|
/* Now enter the pages for this mapping */
|
|
|
|
|
|
|
|
DISABLE_TRANS(msr);
|
|
|
|
for (off = 0; off < translations[i].om_len; off += PAGE_SIZE) {
|
2010-07-13 05:32:19 +00:00
|
|
|
if (moea64_pvo_find_va(kernel_pmap,
|
|
|
|
translations[i].om_va + off) != NULL)
|
|
|
|
continue;
|
|
|
|
|
2010-02-20 16:23:29 +00:00
|
|
|
moea64_kenter(mmup, translations[i].om_va + off,
|
|
|
|
pa_base + off);
|
2009-11-12 15:19:09 +00:00
|
|
|
|
|
|
|
ofw_mappings++;
|
|
|
|
}
|
|
|
|
ENABLE_TRANS(msr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
#ifdef __powerpc64__
|
|
|
|
static void
|
|
|
|
moea64_probe_large_page(void)
|
|
|
|
{
|
|
|
|
uint16_t pvr = mfpvr() >> 16;
|
|
|
|
|
|
|
|
switch (pvr) {
|
|
|
|
case IBM970:
|
|
|
|
case IBM970FX:
|
|
|
|
case IBM970MP:
|
|
|
|
powerpc_sync(); isync();
|
|
|
|
mtspr(SPR_HID4, mfspr(SPR_HID4) & ~HID4_970_DISABLE_LG_PG);
|
|
|
|
powerpc_sync(); isync();
|
|
|
|
|
|
|
|
/* FALLTHROUGH */
|
|
|
|
case IBMCELLBE:
|
|
|
|
moea64_large_page_size = 0x1000000; /* 16 MB */
|
|
|
|
moea64_large_page_shift = 24;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
moea64_large_page_size = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
moea64_large_page_mask = moea64_large_page_size - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
moea64_bootstrap_slb_prefault(vm_offset_t va, int large)
|
|
|
|
{
|
|
|
|
struct slb *cache;
|
|
|
|
struct slb entry;
|
|
|
|
uint64_t esid, slbe;
|
|
|
|
uint64_t i;
|
|
|
|
|
|
|
|
cache = PCPU_GET(slb);
|
|
|
|
esid = va >> ADDR_SR_SHFT;
|
|
|
|
slbe = (esid << SLBE_ESID_SHIFT) | SLBE_VALID;
|
|
|
|
|
|
|
|
for (i = 0; i < 64; i++) {
|
|
|
|
if (cache[i].slbe == (slbe | i))
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
entry.slbe = slbe;
|
2010-07-31 21:35:15 +00:00
|
|
|
entry.slbv = KERNEL_VSID(esid) << SLBV_VSID_SHIFT;
|
2010-07-13 05:32:19 +00:00
|
|
|
if (large)
|
|
|
|
entry.slbv |= SLBV_L;
|
|
|
|
|
|
|
|
slb_insert(kernel_pmap, cache, &entry);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static void
|
|
|
|
moea64_setup_direct_map(mmu_t mmup, vm_offset_t kernelstart,
|
|
|
|
vm_offset_t kernelend)
|
|
|
|
{
|
|
|
|
register_t msr;
|
|
|
|
vm_paddr_t pa;
|
|
|
|
vm_offset_t size, off;
|
|
|
|
uint64_t pte_lo;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (moea64_large_page_size == 0)
|
|
|
|
hw_direct_map = 0;
|
|
|
|
|
|
|
|
DISABLE_TRANS(msr);
|
|
|
|
if (hw_direct_map) {
|
|
|
|
PMAP_LOCK(kernel_pmap);
|
|
|
|
for (i = 0; i < pregions_sz; i++) {
|
|
|
|
for (pa = pregions[i].mr_start; pa < pregions[i].mr_start +
|
|
|
|
pregions[i].mr_size; pa += moea64_large_page_size) {
|
|
|
|
pte_lo = LPTE_M;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set memory access as guarded if prefetch within
|
|
|
|
* the page could exit the available physmem area.
|
|
|
|
*/
|
|
|
|
if (pa & moea64_large_page_mask) {
|
|
|
|
pa &= moea64_large_page_mask;
|
|
|
|
pte_lo |= LPTE_G;
|
|
|
|
}
|
|
|
|
if (pa + moea64_large_page_size >
|
|
|
|
pregions[i].mr_start + pregions[i].mr_size)
|
|
|
|
pte_lo |= LPTE_G;
|
|
|
|
|
|
|
|
moea64_pvo_enter(kernel_pmap, moea64_upvo_zone,
|
|
|
|
&moea64_pvo_kunmanaged, pa, pa,
|
|
|
|
pte_lo, PVO_WIRED | PVO_LARGE |
|
|
|
|
VM_PROT_EXECUTE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
PMAP_UNLOCK(kernel_pmap);
|
|
|
|
} else {
|
|
|
|
size = moea64_pteg_count * sizeof(struct lpteg);
|
|
|
|
off = (vm_offset_t)(moea64_pteg_table);
|
|
|
|
for (pa = off; pa < off + size; pa += PAGE_SIZE)
|
|
|
|
moea64_kenter(mmup, pa, pa);
|
|
|
|
size = sizeof(struct pvo_head) * moea64_pteg_count;
|
|
|
|
off = (vm_offset_t)(moea64_pvo_table);
|
|
|
|
for (pa = off; pa < off + size; pa += PAGE_SIZE)
|
|
|
|
moea64_kenter(mmup, pa, pa);
|
|
|
|
size = BPVO_POOL_SIZE*sizeof(struct pvo_entry);
|
|
|
|
off = (vm_offset_t)(moea64_bpvo_pool);
|
|
|
|
for (pa = off; pa < off + size; pa += PAGE_SIZE)
|
|
|
|
moea64_kenter(mmup, pa, pa);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Map certain important things, like ourselves.
|
|
|
|
*
|
|
|
|
* NOTE: We do not map the exception vector space. That code is
|
|
|
|
* used only in real mode, and leaving it unmapped allows us to
|
|
|
|
* catch NULL pointer deferences, instead of making NULL a valid
|
|
|
|
* address.
|
|
|
|
*/
|
|
|
|
|
|
|
|
for (pa = kernelstart & ~PAGE_MASK; pa < kernelend;
|
|
|
|
pa += PAGE_SIZE)
|
|
|
|
moea64_kenter(mmup, pa, pa);
|
|
|
|
}
|
|
|
|
ENABLE_TRANS(msr);
|
|
|
|
}
|
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
static void
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_bootstrap(mmu_t mmup, vm_offset_t kernelstart, vm_offset_t kernelend)
|
2009-04-04 00:22:44 +00:00
|
|
|
{
|
|
|
|
ihandle_t mmui;
|
|
|
|
phandle_t chosen;
|
|
|
|
phandle_t mmu;
|
2009-11-12 15:19:09 +00:00
|
|
|
size_t sz;
|
2009-04-04 00:22:44 +00:00
|
|
|
int i, j;
|
|
|
|
vm_size_t size, physsz, hwphyssz;
|
2010-07-13 05:32:19 +00:00
|
|
|
vm_offset_t pa, va;
|
2009-11-12 15:19:09 +00:00
|
|
|
register_t msr;
|
Implement a facility for dynamic per-cpu variables.
- Modules and kernel code alike may use DPCPU_DEFINE(),
DPCPU_GET(), DPCPU_SET(), etc. akin to the statically defined
PCPU_*. Requires only one extra instruction more than PCPU_* and is
virtually the same as __thread for builtin and much faster for shared
objects. DPCPU variables can be initialized when defined.
- Modules are supported by relocating the module's per-cpu linker set
over space reserved in the kernel. Modules may fail to load if there
is insufficient space available.
- Track space available for modules with a one-off extent allocator.
Free may block for memory to allocate space for an extent.
Reviewed by: jhb, rwatson, kan, sam, grehan, marius, marcel, stas
2009-06-23 22:42:39 +00:00
|
|
|
void *dpcpu;
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
#ifndef __powerpc64__
|
2009-04-04 00:22:44 +00:00
|
|
|
/* We don't have a direct map since there is no BAT */
|
|
|
|
hw_direct_map = 0;
|
|
|
|
|
|
|
|
/* Make sure battable is zero, since we have no BAT */
|
|
|
|
for (i = 0; i < 16; i++) {
|
|
|
|
battable[i].batu = 0;
|
|
|
|
battable[i].batl = 0;
|
|
|
|
}
|
2010-07-13 05:32:19 +00:00
|
|
|
#else
|
|
|
|
moea64_probe_large_page();
|
|
|
|
|
|
|
|
/* Use a direct map if we have large page support */
|
|
|
|
if (moea64_large_page_size > 0)
|
|
|
|
hw_direct_map = 1;
|
|
|
|
else
|
|
|
|
hw_direct_map = 0;
|
|
|
|
#endif
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/* Get physical memory regions from firmware */
|
|
|
|
mem_regions(&pregions, &pregions_sz, ®ions, ®ions_sz);
|
|
|
|
CTR0(KTR_PMAP, "moea64_bootstrap: physical memory");
|
|
|
|
|
|
|
|
qsort(pregions, pregions_sz, sizeof(*pregions), mr_cmp);
|
|
|
|
if (sizeof(phys_avail)/sizeof(phys_avail[0]) < regions_sz)
|
|
|
|
panic("moea64_bootstrap: phys_avail too small");
|
|
|
|
qsort(regions, regions_sz, sizeof(*regions), mr_cmp);
|
|
|
|
phys_avail_count = 0;
|
|
|
|
physsz = 0;
|
|
|
|
hwphyssz = 0;
|
|
|
|
TUNABLE_ULONG_FETCH("hw.physmem", (u_long *) &hwphyssz);
|
|
|
|
for (i = 0, j = 0; i < regions_sz; i++, j += 2) {
|
|
|
|
CTR3(KTR_PMAP, "region: %#x - %#x (%#x)", regions[i].mr_start,
|
|
|
|
regions[i].mr_start + regions[i].mr_size,
|
|
|
|
regions[i].mr_size);
|
|
|
|
if (hwphyssz != 0 &&
|
|
|
|
(physsz + regions[i].mr_size) >= hwphyssz) {
|
|
|
|
if (physsz < hwphyssz) {
|
|
|
|
phys_avail[j] = regions[i].mr_start;
|
|
|
|
phys_avail[j + 1] = regions[i].mr_start +
|
|
|
|
hwphyssz - physsz;
|
|
|
|
physsz = hwphyssz;
|
|
|
|
phys_avail_count++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
phys_avail[j] = regions[i].mr_start;
|
|
|
|
phys_avail[j + 1] = regions[i].mr_start + regions[i].mr_size;
|
|
|
|
phys_avail_count++;
|
|
|
|
physsz += regions[i].mr_size;
|
|
|
|
}
|
2010-07-13 05:32:19 +00:00
|
|
|
|
|
|
|
/* Check for overlap with the kernel and exception vectors */
|
|
|
|
for (j = 0; j < 2*phys_avail_count; j+=2) {
|
|
|
|
if (phys_avail[j] < EXC_LAST)
|
|
|
|
phys_avail[j] += EXC_LAST;
|
|
|
|
|
|
|
|
if (kernelstart >= phys_avail[j] &&
|
|
|
|
kernelstart < phys_avail[j+1]) {
|
|
|
|
if (kernelend < phys_avail[j+1]) {
|
|
|
|
phys_avail[2*phys_avail_count] =
|
|
|
|
(kernelend & ~PAGE_MASK) + PAGE_SIZE;
|
|
|
|
phys_avail[2*phys_avail_count + 1] =
|
|
|
|
phys_avail[j+1];
|
|
|
|
phys_avail_count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
phys_avail[j+1] = kernelstart & ~PAGE_MASK;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (kernelend >= phys_avail[j] &&
|
|
|
|
kernelend < phys_avail[j+1]) {
|
|
|
|
if (kernelstart > phys_avail[j]) {
|
|
|
|
phys_avail[2*phys_avail_count] = phys_avail[j];
|
|
|
|
phys_avail[2*phys_avail_count + 1] =
|
|
|
|
kernelstart & ~PAGE_MASK;
|
|
|
|
phys_avail_count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
phys_avail[j] = (kernelend & ~PAGE_MASK) + PAGE_SIZE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
physmem = btoc(physsz);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Allocate PTEG table.
|
|
|
|
*/
|
|
|
|
#ifdef PTEGCOUNT
|
|
|
|
moea64_pteg_count = PTEGCOUNT;
|
|
|
|
#else
|
|
|
|
moea64_pteg_count = 0x1000;
|
|
|
|
|
|
|
|
while (moea64_pteg_count < physmem)
|
|
|
|
moea64_pteg_count <<= 1;
|
2010-07-13 05:32:19 +00:00
|
|
|
|
|
|
|
moea64_pteg_count >>= 1;
|
2009-04-04 00:22:44 +00:00
|
|
|
#endif /* PTEGCOUNT */
|
|
|
|
|
|
|
|
size = moea64_pteg_count * sizeof(struct lpteg);
|
|
|
|
CTR2(KTR_PMAP, "moea64_bootstrap: %d PTEGs, %d bytes",
|
|
|
|
moea64_pteg_count, size);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We now need to allocate memory. This memory, to be allocated,
|
|
|
|
* has to reside in a page table. The page table we are about to
|
|
|
|
* allocate. We don't have BAT. So drop to data real mode for a minute
|
|
|
|
* as a measure of last resort. We do this a couple times.
|
|
|
|
*/
|
|
|
|
|
|
|
|
moea64_pteg_table = (struct lpteg *)moea64_bootstrap_alloc(size, size);
|
|
|
|
DISABLE_TRANS(msr);
|
|
|
|
bzero((void *)moea64_pteg_table, moea64_pteg_count * sizeof(struct lpteg));
|
|
|
|
ENABLE_TRANS(msr);
|
|
|
|
|
|
|
|
moea64_pteg_mask = moea64_pteg_count - 1;
|
|
|
|
|
|
|
|
CTR1(KTR_PMAP, "moea64_bootstrap: PTEG table at %p", moea64_pteg_table);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Allocate pv/overflow lists.
|
|
|
|
*/
|
|
|
|
size = sizeof(struct pvo_head) * moea64_pteg_count;
|
|
|
|
|
|
|
|
moea64_pvo_table = (struct pvo_head *)moea64_bootstrap_alloc(size,
|
|
|
|
PAGE_SIZE);
|
|
|
|
CTR1(KTR_PMAP, "moea64_bootstrap: PVO table at %p", moea64_pvo_table);
|
|
|
|
|
|
|
|
DISABLE_TRANS(msr);
|
|
|
|
for (i = 0; i < moea64_pteg_count; i++)
|
|
|
|
LIST_INIT(&moea64_pvo_table[i]);
|
|
|
|
ENABLE_TRANS(msr);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize the lock that synchronizes access to the pteg and pvo
|
|
|
|
* tables.
|
|
|
|
*/
|
|
|
|
mtx_init(&moea64_table_mutex, "pmap table", NULL, MTX_DEF |
|
|
|
|
MTX_RECURSE);
|
2010-08-29 18:17:38 +00:00
|
|
|
mtx_init(&moea64_slb_mutex, "SLB table", NULL, MTX_DEF);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2009-10-23 03:17:02 +00:00
|
|
|
/*
|
|
|
|
* Initialize the TLBIE lock. TLBIE can only be executed by one CPU.
|
|
|
|
*/
|
|
|
|
mtx_init(&tlbie_mutex, "tlbie mutex", NULL, MTX_SPIN);
|
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
/*
|
|
|
|
* Initialise the unmanaged pvo pool.
|
|
|
|
*/
|
|
|
|
moea64_bpvo_pool = (struct pvo_entry *)moea64_bootstrap_alloc(
|
|
|
|
BPVO_POOL_SIZE*sizeof(struct pvo_entry), 0);
|
|
|
|
moea64_bpvo_pool_index = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Make sure kernel vsid is allocated as well as VSID 0.
|
|
|
|
*/
|
2010-07-13 05:32:19 +00:00
|
|
|
#ifndef __powerpc64__
|
|
|
|
moea64_vsid_bitmap[(KERNEL_VSIDBITS & (NVSIDS - 1)) / VSID_NBPW]
|
2009-04-04 00:22:44 +00:00
|
|
|
|= 1 << (KERNEL_VSIDBITS % VSID_NBPW);
|
|
|
|
moea64_vsid_bitmap[0] |= 1;
|
2010-07-13 05:32:19 +00:00
|
|
|
#endif
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize the kernel pmap (which is statically allocated).
|
|
|
|
*/
|
2010-07-13 05:32:19 +00:00
|
|
|
#ifdef __powerpc64__
|
|
|
|
for (i = 0; i < 64; i++) {
|
|
|
|
pcpup->pc_slb[i].slbv = 0;
|
|
|
|
pcpup->pc_slb[i].slbe = 0;
|
|
|
|
}
|
|
|
|
#else
|
2009-04-04 00:22:44 +00:00
|
|
|
for (i = 0; i < 16; i++)
|
|
|
|
kernel_pmap->pm_sr[i] = EMPTY_SEGMENT + i;
|
2010-07-13 05:32:19 +00:00
|
|
|
#endif
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
kernel_pmap->pmap_phys = kernel_pmap;
|
|
|
|
kernel_pmap->pm_active = ~0;
|
|
|
|
|
|
|
|
PMAP_LOCK_INIT(kernel_pmap);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Now map in all the other buffers we allocated earlier
|
|
|
|
*/
|
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_setup_direct_map(mmup, kernelstart, kernelend);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/*
|
2010-07-13 05:32:19 +00:00
|
|
|
* Set up the Open Firmware pmap and add its mappings if not in real
|
|
|
|
* mode.
|
2009-04-04 00:22:44 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
if (!ofw_real_mode) {
|
2010-07-13 05:32:19 +00:00
|
|
|
#ifndef __powerpc64__
|
2009-04-04 00:22:44 +00:00
|
|
|
moea64_pinit(mmup, &ofw_pmap);
|
2010-07-13 05:32:19 +00:00
|
|
|
|
2010-02-20 16:23:29 +00:00
|
|
|
for (i = 0; i < 16; i++)
|
|
|
|
ofw_pmap.pm_sr[i] = kernel_pmap->pm_sr[i];
|
2010-07-13 05:32:19 +00:00
|
|
|
#endif
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
if ((chosen = OF_finddevice("/chosen")) == -1)
|
|
|
|
panic("moea64_bootstrap: can't find /chosen");
|
|
|
|
OF_getprop(chosen, "mmu", &mmui, 4);
|
2010-07-13 05:32:19 +00:00
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
if ((mmu = OF_instance_to_package(mmui)) == -1)
|
|
|
|
panic("moea64_bootstrap: can't get mmu package");
|
|
|
|
if ((sz = OF_getproplen(mmu, "translations")) == -1)
|
|
|
|
panic("moea64_bootstrap: can't get ofw translation count");
|
2009-11-12 15:19:09 +00:00
|
|
|
if (sz > 6144 /* tmpstksz - 2 KB headroom */)
|
|
|
|
panic("moea64_bootstrap: too many ofw translations");
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2009-11-12 15:19:09 +00:00
|
|
|
moea64_add_ofw_mappings(mmup, mmu, sz);
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SMP
|
|
|
|
TLBSYNC();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Calculate the last available physical address.
|
|
|
|
*/
|
|
|
|
for (i = 0; phys_avail[i + 2] != 0; i += 2)
|
|
|
|
;
|
|
|
|
Maxmem = powerpc_btop(phys_avail[i + 1]);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize MMU and remap early physical mappings
|
|
|
|
*/
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_cpu_bootstrap(mmup,0);
|
2009-04-04 00:22:44 +00:00
|
|
|
mtmsr(mfmsr() | PSL_DR | PSL_IR); isync();
|
|
|
|
pmap_bootstrapped++;
|
|
|
|
bs_remap_earlyboot();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the start and end of kva.
|
|
|
|
*/
|
|
|
|
virtual_avail = VM_MIN_KERNEL_ADDRESS;
|
2010-02-20 16:23:29 +00:00
|
|
|
virtual_end = VM_MAX_SAFE_KERNEL_ADDRESS;
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
/*
|
|
|
|
* Map the entire KVA range into the SLB. We must not fault there.
|
|
|
|
*/
|
|
|
|
#ifdef __powerpc64__
|
|
|
|
for (va = virtual_avail; va < virtual_end; va += SEGMENT_LENGTH)
|
|
|
|
moea64_bootstrap_slb_prefault(va, 0);
|
|
|
|
#endif
|
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
/*
|
2010-02-20 16:23:29 +00:00
|
|
|
* Figure out how far we can extend virtual_end into segment 16
|
|
|
|
* without running into existing mappings. Segment 16 is guaranteed
|
|
|
|
* to contain neither RAM nor devices (at least on Apple hardware),
|
|
|
|
* but will generally contain some OFW mappings we should not
|
|
|
|
* step on.
|
2009-04-04 00:22:44 +00:00
|
|
|
*/
|
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
#ifndef __powerpc64__ /* KVA is in high memory on PPC64 */
|
2010-02-20 16:23:29 +00:00
|
|
|
PMAP_LOCK(kernel_pmap);
|
2010-07-13 05:32:19 +00:00
|
|
|
while (virtual_end < VM_MAX_KERNEL_ADDRESS &&
|
|
|
|
moea64_pvo_find_va(kernel_pmap, virtual_end+1) == NULL)
|
2010-02-20 16:23:29 +00:00
|
|
|
virtual_end += PAGE_SIZE;
|
|
|
|
PMAP_UNLOCK(kernel_pmap);
|
2010-07-13 05:32:19 +00:00
|
|
|
#endif
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/*
|
2010-03-04 06:39:58 +00:00
|
|
|
* Allocate some things for page zeroing. We put this directly
|
|
|
|
* in the page table, marked with LPTE_LOCKED, to avoid any
|
|
|
|
* of the PVO book-keeping or other parts of the VM system
|
|
|
|
* from even knowing that this hack exists.
|
2009-04-04 00:22:44 +00:00
|
|
|
*/
|
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
if (!hw_direct_map) {
|
|
|
|
mtx_init(&moea64_scratchpage_mtx, "pvo zero page", NULL,
|
|
|
|
MTX_DEF);
|
|
|
|
for (i = 0; i < 2; i++) {
|
|
|
|
struct lpte pt;
|
|
|
|
uint64_t vsid;
|
|
|
|
int pteidx, ptegidx;
|
|
|
|
|
|
|
|
moea64_scratchpage_va[i] = (virtual_end+1) - PAGE_SIZE;
|
|
|
|
virtual_end -= PAGE_SIZE;
|
|
|
|
|
|
|
|
LOCK_TABLE();
|
|
|
|
|
|
|
|
vsid = va_to_vsid(kernel_pmap,
|
|
|
|
moea64_scratchpage_va[i]);
|
|
|
|
moea64_pte_create(&pt, vsid, moea64_scratchpage_va[i],
|
|
|
|
LPTE_NOEXEC, 0);
|
|
|
|
pt.pte_hi |= LPTE_LOCKED;
|
|
|
|
|
|
|
|
moea64_scratchpage_vpn[i] = (vsid << 16) |
|
|
|
|
((moea64_scratchpage_va[i] & ADDR_PIDX) >>
|
|
|
|
ADDR_PIDX_SHFT);
|
|
|
|
ptegidx = va_to_pteg(vsid, moea64_scratchpage_va[i], 0);
|
|
|
|
pteidx = moea64_pte_insert(ptegidx, &pt);
|
|
|
|
if (pt.pte_hi & LPTE_HID)
|
|
|
|
ptegidx ^= moea64_pteg_mask;
|
|
|
|
|
|
|
|
moea64_scratchpage_pte[i] =
|
|
|
|
&moea64_pteg_table[ptegidx].pt[pteidx];
|
|
|
|
|
|
|
|
UNLOCK_TABLE();
|
|
|
|
}
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Allocate a kernel stack with a guard page for thread0 and map it
|
|
|
|
* into the kernel page map.
|
|
|
|
*/
|
|
|
|
pa = moea64_bootstrap_alloc(KSTACK_PAGES * PAGE_SIZE, PAGE_SIZE);
|
|
|
|
va = virtual_avail + KSTACK_GUARD_PAGES * PAGE_SIZE;
|
|
|
|
virtual_avail = va + KSTACK_PAGES * PAGE_SIZE;
|
|
|
|
CTR2(KTR_PMAP, "moea_bootstrap: kstack0 at %#x (%#x)", pa, va);
|
|
|
|
thread0.td_kstack = va;
|
|
|
|
thread0.td_kstack_pages = KSTACK_PAGES;
|
|
|
|
for (i = 0; i < KSTACK_PAGES; i++) {
|
2010-01-07 21:01:37 +00:00
|
|
|
moea64_kenter(mmup, va, pa);
|
2009-04-04 00:22:44 +00:00
|
|
|
pa += PAGE_SIZE;
|
|
|
|
va += PAGE_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Allocate virtual address space for the message buffer.
|
|
|
|
*/
|
|
|
|
pa = msgbuf_phys = moea64_bootstrap_alloc(MSGBUF_SIZE, PAGE_SIZE);
|
2010-02-25 03:53:21 +00:00
|
|
|
msgbufp = (struct msgbuf *)virtual_avail;
|
|
|
|
va = virtual_avail;
|
|
|
|
virtual_avail += round_page(MSGBUF_SIZE);
|
|
|
|
while (va < virtual_avail) {
|
|
|
|
moea64_kenter(mmup, va, pa);
|
2009-04-04 00:22:44 +00:00
|
|
|
pa += PAGE_SIZE;
|
2010-02-25 03:53:21 +00:00
|
|
|
va += PAGE_SIZE;
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
Implement a facility for dynamic per-cpu variables.
- Modules and kernel code alike may use DPCPU_DEFINE(),
DPCPU_GET(), DPCPU_SET(), etc. akin to the statically defined
PCPU_*. Requires only one extra instruction more than PCPU_* and is
virtually the same as __thread for builtin and much faster for shared
objects. DPCPU variables can be initialized when defined.
- Modules are supported by relocating the module's per-cpu linker set
over space reserved in the kernel. Modules may fail to load if there
is insufficient space available.
- Track space available for modules with a one-off extent allocator.
Free may block for memory to allocate space for an extent.
Reviewed by: jhb, rwatson, kan, sam, grehan, marius, marcel, stas
2009-06-23 22:42:39 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Allocate virtual address space for the dynamic percpu area.
|
|
|
|
*/
|
|
|
|
pa = moea64_bootstrap_alloc(DPCPU_SIZE, PAGE_SIZE);
|
2010-02-25 03:53:21 +00:00
|
|
|
dpcpu = (void *)virtual_avail;
|
2010-07-13 05:32:19 +00:00
|
|
|
va = virtual_avail;
|
2010-02-25 03:53:21 +00:00
|
|
|
virtual_avail += DPCPU_SIZE;
|
|
|
|
while (va < virtual_avail) {
|
|
|
|
moea64_kenter(mmup, va, pa);
|
Implement a facility for dynamic per-cpu variables.
- Modules and kernel code alike may use DPCPU_DEFINE(),
DPCPU_GET(), DPCPU_SET(), etc. akin to the statically defined
PCPU_*. Requires only one extra instruction more than PCPU_* and is
virtually the same as __thread for builtin and much faster for shared
objects. DPCPU variables can be initialized when defined.
- Modules are supported by relocating the module's per-cpu linker set
over space reserved in the kernel. Modules may fail to load if there
is insufficient space available.
- Track space available for modules with a one-off extent allocator.
Free may block for memory to allocate space for an extent.
Reviewed by: jhb, rwatson, kan, sam, grehan, marius, marcel, stas
2009-06-23 22:42:39 +00:00
|
|
|
pa += PAGE_SIZE;
|
2010-02-25 03:53:21 +00:00
|
|
|
va += PAGE_SIZE;
|
Implement a facility for dynamic per-cpu variables.
- Modules and kernel code alike may use DPCPU_DEFINE(),
DPCPU_GET(), DPCPU_SET(), etc. akin to the statically defined
PCPU_*. Requires only one extra instruction more than PCPU_* and is
virtually the same as __thread for builtin and much faster for shared
objects. DPCPU variables can be initialized when defined.
- Modules are supported by relocating the module's per-cpu linker set
over space reserved in the kernel. Modules may fail to load if there
is insufficient space available.
- Track space available for modules with a one-off extent allocator.
Free may block for memory to allocate space for an extent.
Reviewed by: jhb, rwatson, kan, sam, grehan, marius, marcel, stas
2009-06-23 22:42:39 +00:00
|
|
|
}
|
|
|
|
dpcpu_init(dpcpu, 0);
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2010-07-13 05:32:19 +00:00
|
|
|
* Activate a user pmap. The pmap must be activated before its address
|
2009-04-04 00:22:44 +00:00
|
|
|
* space can be accessed in any way.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
moea64_activate(mmu_t mmu, struct thread *td)
|
|
|
|
{
|
2010-07-13 05:32:19 +00:00
|
|
|
pmap_t pm;
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
pm = &td->td_proc->p_vmspace->vm_pmap;
|
|
|
|
pm->pm_active |= PCPU_GET(cpumask);
|
2010-07-13 05:32:19 +00:00
|
|
|
|
|
|
|
#ifdef __powerpc64__
|
|
|
|
PCPU_SET(userslb, pm->pm_slb);
|
|
|
|
#else
|
|
|
|
PCPU_SET(curpmap, pm->pmap_phys);
|
|
|
|
#endif
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
moea64_deactivate(mmu_t mmu, struct thread *td)
|
|
|
|
{
|
|
|
|
pmap_t pm;
|
|
|
|
|
|
|
|
pm = &td->td_proc->p_vmspace->vm_pmap;
|
|
|
|
pm->pm_active &= ~(PCPU_GET(cpumask));
|
2010-07-13 05:32:19 +00:00
|
|
|
#ifdef __powerpc64__
|
|
|
|
PCPU_SET(userslb, NULL);
|
|
|
|
#else
|
2009-04-04 00:22:44 +00:00
|
|
|
PCPU_SET(curpmap, NULL);
|
2010-07-13 05:32:19 +00:00
|
|
|
#endif
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
moea64_change_wiring(mmu_t mmu, pmap_t pm, vm_offset_t va, boolean_t wired)
|
|
|
|
{
|
|
|
|
struct pvo_entry *pvo;
|
2010-07-13 05:32:19 +00:00
|
|
|
struct lpte *pt;
|
|
|
|
uint64_t vsid;
|
|
|
|
int i, ptegidx;
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
PMAP_LOCK(pm);
|
2010-07-13 05:32:19 +00:00
|
|
|
pvo = moea64_pvo_find_va(pm, va & ~ADDR_POFF);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
if (pvo != NULL) {
|
2010-07-13 05:32:19 +00:00
|
|
|
LOCK_TABLE();
|
|
|
|
pt = moea64_pvo_to_pte(pvo);
|
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
if (wired) {
|
|
|
|
if ((pvo->pvo_vaddr & PVO_WIRED) == 0)
|
|
|
|
pm->pm_stats.wired_count++;
|
|
|
|
pvo->pvo_vaddr |= PVO_WIRED;
|
2010-07-13 05:32:19 +00:00
|
|
|
pvo->pvo_pte.lpte.pte_hi |= LPTE_WIRED;
|
2009-04-04 00:22:44 +00:00
|
|
|
} else {
|
|
|
|
if ((pvo->pvo_vaddr & PVO_WIRED) != 0)
|
|
|
|
pm->pm_stats.wired_count--;
|
|
|
|
pvo->pvo_vaddr &= ~PVO_WIRED;
|
2010-07-13 05:32:19 +00:00
|
|
|
pvo->pvo_pte.lpte.pte_hi &= ~LPTE_WIRED;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pt != NULL) {
|
|
|
|
/* Update wiring flag in page table. */
|
|
|
|
moea64_pte_change(pt, &pvo->pvo_pte.lpte,
|
|
|
|
pvo->pvo_vpn);
|
|
|
|
} else if (wired) {
|
|
|
|
/*
|
|
|
|
* If we are wiring the page, and it wasn't in the
|
|
|
|
* page table before, add it.
|
|
|
|
*/
|
|
|
|
vsid = PVO_VSID(pvo);
|
|
|
|
ptegidx = va_to_pteg(vsid, PVO_VADDR(pvo),
|
|
|
|
pvo->pvo_vaddr & PVO_LARGE);
|
|
|
|
|
|
|
|
i = moea64_pte_insert(ptegidx, &pvo->pvo_pte.lpte);
|
|
|
|
if (i >= 0) {
|
|
|
|
PVO_PTEGIDX_CLR(pvo);
|
|
|
|
PVO_PTEGIDX_SET(pvo, i);
|
|
|
|
}
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
2010-07-13 05:32:19 +00:00
|
|
|
|
|
|
|
UNLOCK_TABLE();
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
PMAP_UNLOCK(pm);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This goes through and sets the physical address of our
|
|
|
|
* special scratch PTE to the PA we want to zero or copy. Because
|
|
|
|
* of locking issues (this can get called in pvo_enter() by
|
|
|
|
* the UMA allocator), we can't use most other utility functions here
|
|
|
|
*/
|
|
|
|
|
|
|
|
static __inline
|
|
|
|
void moea64_set_scratchpage_pa(int which, vm_offset_t pa) {
|
2010-02-24 00:54:37 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
KASSERT(!hw_direct_map, ("Using OEA64 scratchpage with a direct map!"));
|
2010-03-04 06:39:58 +00:00
|
|
|
mtx_assert(&moea64_scratchpage_mtx, MA_OWNED);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
moea64_scratchpage_pte[which]->pte_hi &= ~LPTE_VALID;
|
2010-07-13 05:32:19 +00:00
|
|
|
TLBIE(moea64_scratchpage_vpn[which]);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-03-04 06:39:58 +00:00
|
|
|
moea64_scratchpage_pte[which]->pte_lo &=
|
|
|
|
~(LPTE_WIMG | LPTE_RPGN);
|
|
|
|
moea64_scratchpage_pte[which]->pte_lo |=
|
|
|
|
moea64_calc_wimg(pa) | (uint64_t)pa;
|
2009-04-04 00:22:44 +00:00
|
|
|
EIEIO();
|
|
|
|
|
|
|
|
moea64_scratchpage_pte[which]->pte_hi |= LPTE_VALID;
|
2010-02-18 15:00:43 +00:00
|
|
|
PTESYNC(); isync();
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
moea64_copy_page(mmu_t mmu, vm_page_t msrc, vm_page_t mdst)
|
|
|
|
{
|
|
|
|
vm_offset_t dst;
|
|
|
|
vm_offset_t src;
|
|
|
|
|
|
|
|
dst = VM_PAGE_TO_PHYS(mdst);
|
|
|
|
src = VM_PAGE_TO_PHYS(msrc);
|
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
if (hw_direct_map) {
|
|
|
|
kcopy((void *)src, (void *)dst, PAGE_SIZE);
|
|
|
|
} else {
|
|
|
|
mtx_lock(&moea64_scratchpage_mtx);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_set_scratchpage_pa(0,src);
|
|
|
|
moea64_set_scratchpage_pa(1,dst);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
kcopy((void *)moea64_scratchpage_va[0],
|
|
|
|
(void *)moea64_scratchpage_va[1], PAGE_SIZE);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
mtx_unlock(&moea64_scratchpage_mtx);
|
|
|
|
}
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
moea64_zero_page_area(mmu_t mmu, vm_page_t m, int off, int size)
|
|
|
|
{
|
|
|
|
vm_offset_t pa = VM_PAGE_TO_PHYS(m);
|
|
|
|
|
|
|
|
if (!moea64_initialized)
|
2010-07-13 05:32:19 +00:00
|
|
|
panic("moea64_zero_page: can't zero pa %#" PRIxPTR, pa);
|
2009-04-04 00:22:44 +00:00
|
|
|
if (size + off > PAGE_SIZE)
|
|
|
|
panic("moea64_zero_page: size + off > PAGE_SIZE");
|
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
if (hw_direct_map) {
|
|
|
|
bzero((caddr_t)pa + off, size);
|
|
|
|
} else {
|
|
|
|
mtx_lock(&moea64_scratchpage_mtx);
|
|
|
|
moea64_set_scratchpage_pa(0,pa);
|
|
|
|
bzero((caddr_t)moea64_scratchpage_va[0] + off, size);
|
|
|
|
mtx_unlock(&moea64_scratchpage_mtx);
|
|
|
|
}
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
2010-02-24 00:55:55 +00:00
|
|
|
/*
|
|
|
|
* Zero a page of physical memory by temporarily mapping it
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
moea64_zero_page(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
vm_offset_t pa = VM_PAGE_TO_PHYS(m);
|
2010-07-13 05:32:19 +00:00
|
|
|
vm_offset_t va, off;
|
2010-02-24 00:55:55 +00:00
|
|
|
|
|
|
|
if (!moea64_initialized)
|
2010-07-13 05:32:19 +00:00
|
|
|
panic("moea64_zero_page: can't zero pa %#zx", pa);
|
|
|
|
|
|
|
|
if (!hw_direct_map) {
|
|
|
|
mtx_lock(&moea64_scratchpage_mtx);
|
2010-02-24 00:55:55 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_set_scratchpage_pa(0,pa);
|
|
|
|
va = moea64_scratchpage_va[0];
|
|
|
|
} else {
|
|
|
|
va = pa;
|
|
|
|
}
|
2010-02-24 00:55:55 +00:00
|
|
|
|
|
|
|
for (off = 0; off < PAGE_SIZE; off += cacheline_size)
|
2010-07-13 05:32:19 +00:00
|
|
|
__asm __volatile("dcbz 0,%0" :: "r"(va + off));
|
|
|
|
|
|
|
|
if (!hw_direct_map)
|
|
|
|
mtx_unlock(&moea64_scratchpage_mtx);
|
2010-02-24 00:55:55 +00:00
|
|
|
}
|
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
void
|
|
|
|
moea64_zero_page_idle(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
|
|
|
|
moea64_zero_page(mmu, m);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Map the given physical page at the specified virtual address in the
|
|
|
|
* target pmap with the protection requested. If specified the page
|
|
|
|
* will be wired down.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
moea64_enter(mmu_t mmu, pmap_t pmap, vm_offset_t va, vm_page_t m,
|
|
|
|
vm_prot_t prot, boolean_t wired)
|
|
|
|
{
|
|
|
|
|
|
|
|
vm_page_lock_queues();
|
|
|
|
PMAP_LOCK(pmap);
|
|
|
|
moea64_enter_locked(pmap, va, m, prot, wired);
|
|
|
|
vm_page_unlock_queues();
|
|
|
|
PMAP_UNLOCK(pmap);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Map the given physical page at the specified virtual address in the
|
|
|
|
* target pmap with the protection requested. If specified the page
|
|
|
|
* will be wired down.
|
|
|
|
*
|
|
|
|
* The page queues and pmap must be locked.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void
|
|
|
|
moea64_enter_locked(pmap_t pmap, vm_offset_t va, vm_page_t m, vm_prot_t prot,
|
|
|
|
boolean_t wired)
|
|
|
|
{
|
|
|
|
struct pvo_head *pvo_head;
|
|
|
|
uma_zone_t zone;
|
|
|
|
vm_page_t pg;
|
|
|
|
uint64_t pte_lo;
|
|
|
|
u_int pvo_flags;
|
|
|
|
int error;
|
|
|
|
|
|
|
|
if (!moea64_initialized) {
|
|
|
|
pvo_head = &moea64_pvo_kunmanaged;
|
|
|
|
pg = NULL;
|
|
|
|
zone = moea64_upvo_zone;
|
|
|
|
pvo_flags = 0;
|
|
|
|
} else {
|
|
|
|
pvo_head = vm_page_to_pvoh(m);
|
|
|
|
pg = m;
|
|
|
|
zone = moea64_mpvo_zone;
|
|
|
|
pvo_flags = PVO_MANAGED;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pmap_bootstrapped)
|
|
|
|
mtx_assert(&vm_page_queue_mtx, MA_OWNED);
|
|
|
|
PMAP_LOCK_ASSERT(pmap, MA_OWNED);
|
2010-06-11 15:49:39 +00:00
|
|
|
KASSERT((m->flags & (PG_FICTITIOUS | PG_UNMANAGED)) != 0 ||
|
|
|
|
(m->oflags & VPO_BUSY) != 0 || VM_OBJECT_LOCKED(m->object),
|
2010-05-16 23:45:10 +00:00
|
|
|
("moea64_enter_locked: page %p is not busy", m));
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/* XXX change the pvo head for fake pages */
|
|
|
|
if ((m->flags & PG_FICTITIOUS) == PG_FICTITIOUS) {
|
|
|
|
pvo_flags &= ~PVO_MANAGED;
|
|
|
|
pvo_head = &moea64_pvo_kunmanaged;
|
|
|
|
zone = moea64_upvo_zone;
|
|
|
|
}
|
|
|
|
|
|
|
|
pte_lo = moea64_calc_wimg(VM_PAGE_TO_PHYS(m));
|
|
|
|
|
|
|
|
if (prot & VM_PROT_WRITE) {
|
|
|
|
pte_lo |= LPTE_BW;
|
2010-06-05 06:56:06 +00:00
|
|
|
if (pmap_bootstrapped &&
|
|
|
|
(m->flags & (PG_FICTITIOUS | PG_UNMANAGED)) == 0)
|
2009-04-04 00:22:44 +00:00
|
|
|
vm_page_flag_set(m, PG_WRITEABLE);
|
|
|
|
} else
|
|
|
|
pte_lo |= LPTE_BR;
|
|
|
|
|
|
|
|
if (prot & VM_PROT_EXECUTE)
|
|
|
|
pvo_flags |= VM_PROT_EXECUTE;
|
|
|
|
|
|
|
|
if (wired)
|
|
|
|
pvo_flags |= PVO_WIRED;
|
|
|
|
|
|
|
|
if ((m->flags & PG_FICTITIOUS) != 0)
|
|
|
|
pvo_flags |= PVO_FAKE;
|
|
|
|
|
|
|
|
error = moea64_pvo_enter(pmap, zone, pvo_head, va, VM_PAGE_TO_PHYS(m),
|
2009-10-23 03:17:02 +00:00
|
|
|
pte_lo, pvo_flags);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Flush the page from the instruction cache if this page is
|
|
|
|
* mapped executable and cacheable.
|
|
|
|
*/
|
|
|
|
if ((pte_lo & (LPTE_I | LPTE_G | LPTE_NOEXEC)) == 0) {
|
2009-10-21 18:38:02 +00:00
|
|
|
moea64_syncicache(pmap, va, VM_PAGE_TO_PHYS(m), PAGE_SIZE);
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2009-10-21 18:38:02 +00:00
|
|
|
moea64_syncicache(pmap_t pmap, vm_offset_t va, vm_offset_t pa, vm_size_t sz)
|
2009-04-04 00:22:44 +00:00
|
|
|
{
|
2010-02-18 15:00:43 +00:00
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
/*
|
|
|
|
* This is much trickier than on older systems because
|
|
|
|
* we can't sync the icache on physical addresses directly
|
|
|
|
* without a direct map. Instead we check a couple of cases
|
|
|
|
* where the memory is already mapped in and, failing that,
|
|
|
|
* use the same trick we use for page zeroing to create
|
|
|
|
* a temporary mapping for this physical address.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (!pmap_bootstrapped) {
|
|
|
|
/*
|
|
|
|
* If PMAP is not bootstrapped, we are likely to be
|
|
|
|
* in real mode.
|
|
|
|
*/
|
2009-10-21 18:38:02 +00:00
|
|
|
__syncicache((void *)pa, sz);
|
2009-04-04 00:22:44 +00:00
|
|
|
} else if (pmap == kernel_pmap) {
|
2009-10-21 18:38:02 +00:00
|
|
|
__syncicache((void *)va, sz);
|
2010-07-13 05:32:19 +00:00
|
|
|
} else if (hw_direct_map) {
|
|
|
|
__syncicache((void *)pa, sz);
|
2009-04-04 00:22:44 +00:00
|
|
|
} else {
|
|
|
|
/* Use the scratch page to set up a temp mapping */
|
|
|
|
|
|
|
|
mtx_lock(&moea64_scratchpage_mtx);
|
|
|
|
|
2010-02-18 15:00:43 +00:00
|
|
|
moea64_set_scratchpage_pa(1,pa & ~ADDR_POFF);
|
|
|
|
__syncicache((void *)(moea64_scratchpage_va[1] +
|
|
|
|
(va & ADDR_POFF)), sz);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
mtx_unlock(&moea64_scratchpage_mtx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Maps a sequence of resident pages belonging to the same object.
|
|
|
|
* The sequence begins with the given page m_start. This page is
|
|
|
|
* mapped at the given virtual address start. Each subsequent page is
|
|
|
|
* mapped at a virtual address that is offset from start by the same
|
|
|
|
* amount as the page is offset from m_start within the object. The
|
|
|
|
* last page in the sequence is the page with the largest offset from
|
|
|
|
* m_start that can be mapped at a virtual address less than the given
|
|
|
|
* virtual address end. Not every virtual page between start and end
|
|
|
|
* is mapped; only those for which a resident page exists with the
|
|
|
|
* corresponding offset from m_start are mapped.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
moea64_enter_object(mmu_t mmu, pmap_t pm, vm_offset_t start, vm_offset_t end,
|
|
|
|
vm_page_t m_start, vm_prot_t prot)
|
|
|
|
{
|
|
|
|
vm_page_t m;
|
|
|
|
vm_pindex_t diff, psize;
|
|
|
|
|
|
|
|
psize = atop(end - start);
|
|
|
|
m = m_start;
|
2010-05-26 18:00:44 +00:00
|
|
|
vm_page_lock_queues();
|
2009-04-04 00:22:44 +00:00
|
|
|
PMAP_LOCK(pm);
|
|
|
|
while (m != NULL && (diff = m->pindex - m_start->pindex) < psize) {
|
|
|
|
moea64_enter_locked(pm, start + ptoa(diff), m, prot &
|
|
|
|
(VM_PROT_READ | VM_PROT_EXECUTE), FALSE);
|
|
|
|
m = TAILQ_NEXT(m, listq);
|
|
|
|
}
|
2010-05-26 18:00:44 +00:00
|
|
|
vm_page_unlock_queues();
|
2009-04-04 00:22:44 +00:00
|
|
|
PMAP_UNLOCK(pm);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
moea64_enter_quick(mmu_t mmu, pmap_t pm, vm_offset_t va, vm_page_t m,
|
|
|
|
vm_prot_t prot)
|
|
|
|
{
|
2010-05-08 20:34:01 +00:00
|
|
|
|
|
|
|
vm_page_lock_queues();
|
2009-04-04 00:22:44 +00:00
|
|
|
PMAP_LOCK(pm);
|
|
|
|
moea64_enter_locked(pm, va, m, prot & (VM_PROT_READ | VM_PROT_EXECUTE),
|
|
|
|
FALSE);
|
2010-05-08 20:34:01 +00:00
|
|
|
vm_page_unlock_queues();
|
2009-04-04 00:22:44 +00:00
|
|
|
PMAP_UNLOCK(pm);
|
|
|
|
}
|
|
|
|
|
|
|
|
vm_paddr_t
|
|
|
|
moea64_extract(mmu_t mmu, pmap_t pm, vm_offset_t va)
|
|
|
|
{
|
|
|
|
struct pvo_entry *pvo;
|
|
|
|
vm_paddr_t pa;
|
|
|
|
|
|
|
|
PMAP_LOCK(pm);
|
2010-07-13 05:32:19 +00:00
|
|
|
pvo = moea64_pvo_find_va(pm, va);
|
2009-04-04 00:22:44 +00:00
|
|
|
if (pvo == NULL)
|
|
|
|
pa = 0;
|
|
|
|
else
|
2010-07-13 05:32:19 +00:00
|
|
|
pa = (pvo->pvo_pte.lpte.pte_lo & LPTE_RPGN) |
|
|
|
|
(va - PVO_VADDR(pvo));
|
2009-04-04 00:22:44 +00:00
|
|
|
PMAP_UNLOCK(pm);
|
|
|
|
return (pa);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Atomically extract and hold the physical page with the given
|
|
|
|
* pmap and virtual address pair if that mapping permits the given
|
|
|
|
* protection.
|
|
|
|
*/
|
|
|
|
vm_page_t
|
|
|
|
moea64_extract_and_hold(mmu_t mmu, pmap_t pmap, vm_offset_t va, vm_prot_t prot)
|
|
|
|
{
|
|
|
|
struct pvo_entry *pvo;
|
|
|
|
vm_page_t m;
|
2010-04-30 00:46:43 +00:00
|
|
|
vm_paddr_t pa;
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
m = NULL;
|
2010-04-30 00:46:43 +00:00
|
|
|
pa = 0;
|
2009-04-04 00:22:44 +00:00
|
|
|
PMAP_LOCK(pmap);
|
2010-04-30 00:46:43 +00:00
|
|
|
retry:
|
2010-07-13 05:32:19 +00:00
|
|
|
pvo = moea64_pvo_find_va(pmap, va & ~ADDR_POFF);
|
2009-04-04 00:22:44 +00:00
|
|
|
if (pvo != NULL && (pvo->pvo_pte.lpte.pte_hi & LPTE_VALID) &&
|
|
|
|
((pvo->pvo_pte.lpte.pte_lo & LPTE_PP) == LPTE_RW ||
|
|
|
|
(prot & VM_PROT_WRITE) == 0)) {
|
2010-04-30 00:46:43 +00:00
|
|
|
if (vm_page_pa_tryrelock(pmap,
|
|
|
|
pvo->pvo_pte.lpte.pte_lo & LPTE_RPGN, &pa))
|
|
|
|
goto retry;
|
2009-04-04 00:22:44 +00:00
|
|
|
m = PHYS_TO_VM_PAGE(pvo->pvo_pte.lpte.pte_lo & LPTE_RPGN);
|
|
|
|
vm_page_hold(m);
|
|
|
|
}
|
2010-04-30 00:46:43 +00:00
|
|
|
PA_UNLOCK_COND(pa);
|
2009-04-04 00:22:44 +00:00
|
|
|
PMAP_UNLOCK(pmap);
|
|
|
|
return (m);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *
|
|
|
|
moea64_uma_page_alloc(uma_zone_t zone, int bytes, u_int8_t *flags, int wait)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* This entire routine is a horrible hack to avoid bothering kmem
|
|
|
|
* for new KVA addresses. Because this can get called from inside
|
|
|
|
* kmem allocation routines, calling kmem for a new address here
|
|
|
|
* can lead to multiply locking non-recursive mutexes.
|
|
|
|
*/
|
|
|
|
static vm_pindex_t color;
|
|
|
|
vm_offset_t va;
|
|
|
|
|
|
|
|
vm_page_t m;
|
|
|
|
int pflags, needed_lock;
|
|
|
|
|
|
|
|
*flags = UMA_SLAB_PRIV;
|
|
|
|
needed_lock = !PMAP_LOCKED(kernel_pmap);
|
|
|
|
|
|
|
|
if (needed_lock)
|
|
|
|
PMAP_LOCK(kernel_pmap);
|
|
|
|
|
|
|
|
if ((wait & (M_NOWAIT|M_USE_RESERVE)) == M_NOWAIT)
|
|
|
|
pflags = VM_ALLOC_INTERRUPT | VM_ALLOC_WIRED;
|
|
|
|
else
|
|
|
|
pflags = VM_ALLOC_SYSTEM | VM_ALLOC_WIRED;
|
|
|
|
if (wait & M_ZERO)
|
|
|
|
pflags |= VM_ALLOC_ZERO;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
m = vm_page_alloc(NULL, color++, pflags | VM_ALLOC_NOOBJ);
|
|
|
|
if (m == NULL) {
|
|
|
|
if (wait & M_NOWAIT)
|
|
|
|
return (NULL);
|
|
|
|
VM_WAIT;
|
|
|
|
} else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2010-02-20 16:23:29 +00:00
|
|
|
va = VM_PAGE_TO_PHYS(m);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
moea64_pvo_enter(kernel_pmap, moea64_upvo_zone,
|
2010-02-20 16:23:29 +00:00
|
|
|
&moea64_pvo_kunmanaged, va, VM_PAGE_TO_PHYS(m), LPTE_M,
|
2009-10-23 03:17:02 +00:00
|
|
|
PVO_WIRED | PVO_BOOTSTRAP);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
if (needed_lock)
|
|
|
|
PMAP_UNLOCK(kernel_pmap);
|
2009-10-23 03:17:02 +00:00
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
if ((wait & M_ZERO) && (m->flags & PG_ZERO) == 0)
|
|
|
|
bzero((void *)va, PAGE_SIZE);
|
|
|
|
|
|
|
|
return (void *)va;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
moea64_init(mmu_t mmu)
|
|
|
|
{
|
|
|
|
|
|
|
|
CTR0(KTR_PMAP, "moea64_init");
|
|
|
|
|
|
|
|
moea64_upvo_zone = uma_zcreate("UPVO entry", sizeof (struct pvo_entry),
|
|
|
|
NULL, NULL, NULL, NULL, UMA_ALIGN_PTR,
|
|
|
|
UMA_ZONE_VM | UMA_ZONE_NOFREE);
|
|
|
|
moea64_mpvo_zone = uma_zcreate("MPVO entry", sizeof(struct pvo_entry),
|
|
|
|
NULL, NULL, NULL, NULL, UMA_ALIGN_PTR,
|
|
|
|
UMA_ZONE_VM | UMA_ZONE_NOFREE);
|
|
|
|
|
|
|
|
if (!hw_direct_map) {
|
|
|
|
uma_zone_set_allocf(moea64_upvo_zone,moea64_uma_page_alloc);
|
|
|
|
uma_zone_set_allocf(moea64_mpvo_zone,moea64_uma_page_alloc);
|
|
|
|
}
|
|
|
|
|
|
|
|
moea64_initialized = TRUE;
|
|
|
|
}
|
|
|
|
|
Resurrect pmap_is_referenced() and use it in mincore(). Essentially,
pmap_ts_referenced() is not always appropriate for checking whether or
not pages have been referenced because it clears any reference bits
that it encounters. For example, in mincore(), clearing the reference
bits has two negative consequences. First, it throws off the activity
count calculations performed by the page daemon. Specifically, a page
on which mincore() has called pmap_ts_referenced() looks less active
to the page daemon than it should. Consequently, the page could be
deactivated prematurely by the page daemon. Arguably, this problem
could be fixed by having mincore() duplicate the activity count
calculation on the page. However, there is a second problem for which
that is not a solution. In order to clear a reference on a 4KB page,
it may be necessary to demote a 2/4MB page mapping. Thus, a mincore()
by one process can have the side effect of demoting a superpage
mapping within another process!
2010-04-24 17:32:52 +00:00
|
|
|
boolean_t
|
|
|
|
moea64_is_referenced(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
|
2010-05-26 18:00:44 +00:00
|
|
|
KASSERT((m->flags & (PG_FICTITIOUS | PG_UNMANAGED)) == 0,
|
|
|
|
("moea64_is_referenced: page %p is not managed", m));
|
Resurrect pmap_is_referenced() and use it in mincore(). Essentially,
pmap_ts_referenced() is not always appropriate for checking whether or
not pages have been referenced because it clears any reference bits
that it encounters. For example, in mincore(), clearing the reference
bits has two negative consequences. First, it throws off the activity
count calculations performed by the page daemon. Specifically, a page
on which mincore() has called pmap_ts_referenced() looks less active
to the page daemon than it should. Consequently, the page could be
deactivated prematurely by the page daemon. Arguably, this problem
could be fixed by having mincore() duplicate the activity count
calculation on the page. However, there is a second problem for which
that is not a solution. In order to clear a reference on a 4KB page,
it may be necessary to demote a 2/4MB page mapping. Thus, a mincore()
by one process can have the side effect of demoting a superpage
mapping within another process!
2010-04-24 17:32:52 +00:00
|
|
|
return (moea64_query_bit(m, PTE_REF));
|
|
|
|
}
|
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
boolean_t
|
|
|
|
moea64_is_modified(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
|
Roughly half of a typical pmap_mincore() implementation is machine-
independent code. Move this code into mincore(), and eliminate the
page queues lock from pmap_mincore().
Push down the page queues lock into pmap_clear_modify(),
pmap_clear_reference(), and pmap_is_modified(). Assert that these
functions are never passed an unmanaged page.
Eliminate an inaccurate comment from powerpc/powerpc/mmu_if.m:
Contrary to what the comment says, pmap_mincore() is not simply an
optimization. Without a complete pmap_mincore() implementation,
mincore() cannot return either MINCORE_MODIFIED or MINCORE_REFERENCED
because only the pmap can provide this information.
Eliminate the page queues lock from vfs_setdirty_locked_object(),
vm_pageout_clean(), vm_object_page_collect_flush(), and
vm_object_page_clean(). Generally speaking, these are all accesses
to the page's dirty field, which are synchronized by the containing
vm object's lock.
Reduce the scope of the page queues lock in vm_object_madvise() and
vm_page_dontneed().
Reviewed by: kib (an earlier version)
2010-05-24 14:26:57 +00:00
|
|
|
KASSERT((m->flags & (PG_FICTITIOUS | PG_UNMANAGED)) == 0,
|
|
|
|
("moea64_is_modified: page %p is not managed", m));
|
2009-04-04 00:22:44 +00:00
|
|
|
|
Roughly half of a typical pmap_mincore() implementation is machine-
independent code. Move this code into mincore(), and eliminate the
page queues lock from pmap_mincore().
Push down the page queues lock into pmap_clear_modify(),
pmap_clear_reference(), and pmap_is_modified(). Assert that these
functions are never passed an unmanaged page.
Eliminate an inaccurate comment from powerpc/powerpc/mmu_if.m:
Contrary to what the comment says, pmap_mincore() is not simply an
optimization. Without a complete pmap_mincore() implementation,
mincore() cannot return either MINCORE_MODIFIED or MINCORE_REFERENCED
because only the pmap can provide this information.
Eliminate the page queues lock from vfs_setdirty_locked_object(),
vm_pageout_clean(), vm_object_page_collect_flush(), and
vm_object_page_clean(). Generally speaking, these are all accesses
to the page's dirty field, which are synchronized by the containing
vm object's lock.
Reduce the scope of the page queues lock in vm_object_madvise() and
vm_page_dontneed().
Reviewed by: kib (an earlier version)
2010-05-24 14:26:57 +00:00
|
|
|
/*
|
|
|
|
* If the page is not VPO_BUSY, then PG_WRITEABLE cannot be
|
|
|
|
* concurrently set while the object is locked. Thus, if PG_WRITEABLE
|
|
|
|
* is clear, no PTEs can have LPTE_CHG set.
|
|
|
|
*/
|
|
|
|
VM_OBJECT_LOCK_ASSERT(m->object, MA_OWNED);
|
|
|
|
if ((m->oflags & VPO_BUSY) == 0 &&
|
|
|
|
(m->flags & PG_WRITEABLE) == 0)
|
|
|
|
return (FALSE);
|
2010-05-26 18:00:44 +00:00
|
|
|
return (moea64_query_bit(m, LPTE_CHG));
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
moea64_clear_reference(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
|
Roughly half of a typical pmap_mincore() implementation is machine-
independent code. Move this code into mincore(), and eliminate the
page queues lock from pmap_mincore().
Push down the page queues lock into pmap_clear_modify(),
pmap_clear_reference(), and pmap_is_modified(). Assert that these
functions are never passed an unmanaged page.
Eliminate an inaccurate comment from powerpc/powerpc/mmu_if.m:
Contrary to what the comment says, pmap_mincore() is not simply an
optimization. Without a complete pmap_mincore() implementation,
mincore() cannot return either MINCORE_MODIFIED or MINCORE_REFERENCED
because only the pmap can provide this information.
Eliminate the page queues lock from vfs_setdirty_locked_object(),
vm_pageout_clean(), vm_object_page_collect_flush(), and
vm_object_page_clean(). Generally speaking, these are all accesses
to the page's dirty field, which are synchronized by the containing
vm object's lock.
Reduce the scope of the page queues lock in vm_object_madvise() and
vm_page_dontneed().
Reviewed by: kib (an earlier version)
2010-05-24 14:26:57 +00:00
|
|
|
KASSERT((m->flags & (PG_FICTITIOUS | PG_UNMANAGED)) == 0,
|
|
|
|
("moea64_clear_reference: page %p is not managed", m));
|
2010-06-10 16:56:35 +00:00
|
|
|
moea64_clear_bit(m, LPTE_REF);
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
moea64_clear_modify(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
|
Roughly half of a typical pmap_mincore() implementation is machine-
independent code. Move this code into mincore(), and eliminate the
page queues lock from pmap_mincore().
Push down the page queues lock into pmap_clear_modify(),
pmap_clear_reference(), and pmap_is_modified(). Assert that these
functions are never passed an unmanaged page.
Eliminate an inaccurate comment from powerpc/powerpc/mmu_if.m:
Contrary to what the comment says, pmap_mincore() is not simply an
optimization. Without a complete pmap_mincore() implementation,
mincore() cannot return either MINCORE_MODIFIED or MINCORE_REFERENCED
because only the pmap can provide this information.
Eliminate the page queues lock from vfs_setdirty_locked_object(),
vm_pageout_clean(), vm_object_page_collect_flush(), and
vm_object_page_clean(). Generally speaking, these are all accesses
to the page's dirty field, which are synchronized by the containing
vm object's lock.
Reduce the scope of the page queues lock in vm_object_madvise() and
vm_page_dontneed().
Reviewed by: kib (an earlier version)
2010-05-24 14:26:57 +00:00
|
|
|
KASSERT((m->flags & (PG_FICTITIOUS | PG_UNMANAGED)) == 0,
|
|
|
|
("moea64_clear_modify: page %p is not managed", m));
|
|
|
|
VM_OBJECT_LOCK_ASSERT(m->object, MA_OWNED);
|
|
|
|
KASSERT((m->oflags & VPO_BUSY) == 0,
|
|
|
|
("moea64_clear_modify: page %p is busy", m));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the page is not PG_WRITEABLE, then no PTEs can have LPTE_CHG
|
|
|
|
* set. If the object containing the page is locked and the page is
|
|
|
|
* not VPO_BUSY, then PG_WRITEABLE cannot be concurrently set.
|
|
|
|
*/
|
|
|
|
if ((m->flags & PG_WRITEABLE) == 0)
|
2009-04-04 00:22:44 +00:00
|
|
|
return;
|
2010-06-10 16:56:35 +00:00
|
|
|
moea64_clear_bit(m, LPTE_CHG);
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Clear the write and modified bits in each of the given page's mappings.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
moea64_remove_write(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
struct pvo_entry *pvo;
|
|
|
|
struct lpte *pt;
|
|
|
|
pmap_t pmap;
|
|
|
|
uint64_t lo;
|
|
|
|
|
2010-05-16 23:45:10 +00:00
|
|
|
KASSERT((m->flags & (PG_FICTITIOUS | PG_UNMANAGED)) == 0,
|
|
|
|
("moea64_remove_write: page %p is not managed", m));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the page is not VPO_BUSY, then PG_WRITEABLE cannot be set by
|
|
|
|
* another thread while the object is locked. Thus, if PG_WRITEABLE
|
|
|
|
* is clear, no page table entries need updating.
|
|
|
|
*/
|
|
|
|
VM_OBJECT_LOCK_ASSERT(m->object, MA_OWNED);
|
|
|
|
if ((m->oflags & VPO_BUSY) == 0 &&
|
2009-04-04 00:22:44 +00:00
|
|
|
(m->flags & PG_WRITEABLE) == 0)
|
|
|
|
return;
|
2010-05-08 20:34:01 +00:00
|
|
|
vm_page_lock_queues();
|
2009-04-04 00:22:44 +00:00
|
|
|
lo = moea64_attr_fetch(m);
|
|
|
|
SYNC();
|
|
|
|
LIST_FOREACH(pvo, vm_page_to_pvoh(m), pvo_vlink) {
|
|
|
|
pmap = pvo->pvo_pmap;
|
|
|
|
PMAP_LOCK(pmap);
|
2010-03-20 14:35:24 +00:00
|
|
|
LOCK_TABLE();
|
2009-04-04 00:22:44 +00:00
|
|
|
if ((pvo->pvo_pte.lpte.pte_lo & LPTE_PP) != LPTE_BR) {
|
2010-07-13 05:32:19 +00:00
|
|
|
pt = moea64_pvo_to_pte(pvo);
|
2009-04-04 00:22:44 +00:00
|
|
|
pvo->pvo_pte.lpte.pte_lo &= ~LPTE_PP;
|
|
|
|
pvo->pvo_pte.lpte.pte_lo |= LPTE_BR;
|
|
|
|
if (pt != NULL) {
|
|
|
|
moea64_pte_synch(pt, &pvo->pvo_pte.lpte);
|
|
|
|
lo |= pvo->pvo_pte.lpte.pte_lo;
|
|
|
|
pvo->pvo_pte.lpte.pte_lo &= ~LPTE_CHG;
|
|
|
|
moea64_pte_change(pt, &pvo->pvo_pte.lpte,
|
2010-07-13 05:32:19 +00:00
|
|
|
pvo->pvo_vpn);
|
|
|
|
if (pvo->pvo_pmap == kernel_pmap)
|
|
|
|
isync();
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
}
|
2010-03-20 14:35:24 +00:00
|
|
|
UNLOCK_TABLE();
|
2009-04-04 00:22:44 +00:00
|
|
|
PMAP_UNLOCK(pmap);
|
|
|
|
}
|
|
|
|
if ((lo & LPTE_CHG) != 0) {
|
|
|
|
moea64_attr_clear(m, LPTE_CHG);
|
|
|
|
vm_page_dirty(m);
|
|
|
|
}
|
|
|
|
vm_page_flag_clear(m, PG_WRITEABLE);
|
2010-05-08 20:34:01 +00:00
|
|
|
vm_page_unlock_queues();
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* moea64_ts_referenced:
|
|
|
|
*
|
|
|
|
* Return a count of reference bits for a page, clearing those bits.
|
|
|
|
* It is not necessary for every reference bit to be cleared, but it
|
|
|
|
* is necessary that 0 only be returned when there are truly no
|
|
|
|
* reference bits set.
|
|
|
|
*
|
|
|
|
* XXX: The exact number of bits to check and clear is a matter that
|
|
|
|
* should be tested and standardized at some point in the future for
|
|
|
|
* optimal aging of shared pages.
|
|
|
|
*/
|
|
|
|
boolean_t
|
|
|
|
moea64_ts_referenced(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
|
2010-06-10 16:56:35 +00:00
|
|
|
KASSERT((m->flags & (PG_FICTITIOUS | PG_UNMANAGED)) == 0,
|
|
|
|
("moea64_ts_referenced: page %p is not managed", m));
|
|
|
|
return (moea64_clear_bit(m, LPTE_REF));
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Map a wired page into kernel virtual address space.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
moea64_kenter(mmu_t mmu, vm_offset_t va, vm_offset_t pa)
|
|
|
|
{
|
|
|
|
uint64_t pte_lo;
|
|
|
|
int error;
|
|
|
|
|
2010-02-20 16:23:29 +00:00
|
|
|
#if 0
|
2009-04-04 00:22:44 +00:00
|
|
|
if (!pmap_bootstrapped) {
|
2010-02-20 16:23:29 +00:00
|
|
|
if (va >= VM_MIN_KERNEL_ADDRESS && va < virtual_end)
|
2010-07-13 05:32:19 +00:00
|
|
|
panic("Trying to enter an address in KVA -- %#"
|
|
|
|
PRIxPTR "!\n",pa);
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
2010-02-20 16:23:29 +00:00
|
|
|
#endif
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
pte_lo = moea64_calc_wimg(pa);
|
|
|
|
|
|
|
|
PMAP_LOCK(kernel_pmap);
|
|
|
|
error = moea64_pvo_enter(kernel_pmap, moea64_upvo_zone,
|
|
|
|
&moea64_pvo_kunmanaged, va, pa, pte_lo,
|
2009-10-23 03:17:02 +00:00
|
|
|
PVO_WIRED | VM_PROT_EXECUTE);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
if (error != 0 && error != ENOENT)
|
2010-07-13 05:32:19 +00:00
|
|
|
panic("moea64_kenter: failed to enter va %#zx pa %#zx: %d", va,
|
2009-04-04 00:22:44 +00:00
|
|
|
pa, error);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Flush the memory from the instruction cache.
|
|
|
|
*/
|
|
|
|
if ((pte_lo & (LPTE_I | LPTE_G)) == 0) {
|
|
|
|
__syncicache((void *)va, PAGE_SIZE);
|
|
|
|
}
|
|
|
|
PMAP_UNLOCK(kernel_pmap);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Extract the physical page address associated with the given kernel virtual
|
|
|
|
* address.
|
|
|
|
*/
|
|
|
|
vm_offset_t
|
|
|
|
moea64_kextract(mmu_t mmu, vm_offset_t va)
|
|
|
|
{
|
|
|
|
struct pvo_entry *pvo;
|
|
|
|
vm_paddr_t pa;
|
|
|
|
|
2010-03-20 14:35:24 +00:00
|
|
|
/*
|
|
|
|
* Shortcut the direct-mapped case when applicable. We never put
|
|
|
|
* anything but 1:1 mappings below VM_MIN_KERNEL_ADDRESS.
|
|
|
|
*/
|
|
|
|
if (va < VM_MIN_KERNEL_ADDRESS)
|
|
|
|
return (va);
|
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
PMAP_LOCK(kernel_pmap);
|
2010-07-13 05:32:19 +00:00
|
|
|
pvo = moea64_pvo_find_va(kernel_pmap, va);
|
|
|
|
KASSERT(pvo != NULL, ("moea64_kextract: no addr found for %#" PRIxPTR,
|
|
|
|
va));
|
|
|
|
pa = (pvo->pvo_pte.lpte.pte_lo & LPTE_RPGN) + (va - PVO_VADDR(pvo));
|
2009-04-04 00:22:44 +00:00
|
|
|
PMAP_UNLOCK(kernel_pmap);
|
|
|
|
return (pa);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Remove a wired page from kernel virtual address space.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
moea64_kremove(mmu_t mmu, vm_offset_t va)
|
|
|
|
{
|
|
|
|
moea64_remove(mmu, kernel_pmap, va, va + PAGE_SIZE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Map a range of physical addresses into kernel virtual address space.
|
|
|
|
*
|
|
|
|
* The value passed in *virt is a suggested virtual address for the mapping.
|
|
|
|
* Architectures which can support a direct-mapped physical to virtual region
|
|
|
|
* can return the appropriate address within that region, leaving '*virt'
|
|
|
|
* unchanged. We cannot and therefore do not; *virt is updated with the
|
|
|
|
* first usable address after the mapped region.
|
|
|
|
*/
|
|
|
|
vm_offset_t
|
|
|
|
moea64_map(mmu_t mmu, vm_offset_t *virt, vm_offset_t pa_start,
|
|
|
|
vm_offset_t pa_end, int prot)
|
|
|
|
{
|
|
|
|
vm_offset_t sva, va;
|
|
|
|
|
|
|
|
sva = *virt;
|
|
|
|
va = sva;
|
|
|
|
for (; pa_start < pa_end; pa_start += PAGE_SIZE, va += PAGE_SIZE)
|
|
|
|
moea64_kenter(mmu, va, pa_start);
|
|
|
|
*virt = va;
|
|
|
|
|
|
|
|
return (sva);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Returns true if the pmap's pv is one of the first
|
|
|
|
* 16 pvs linked to from this page. This count may
|
|
|
|
* be changed upwards or downwards in the future; it
|
|
|
|
* is only necessary that true be returned for a small
|
|
|
|
* subset of pmaps for proper page aging.
|
|
|
|
*/
|
|
|
|
boolean_t
|
|
|
|
moea64_page_exists_quick(mmu_t mmu, pmap_t pmap, vm_page_t m)
|
|
|
|
{
|
|
|
|
int loops;
|
|
|
|
struct pvo_entry *pvo;
|
2010-06-10 16:56:35 +00:00
|
|
|
boolean_t rv;
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-06-10 16:56:35 +00:00
|
|
|
KASSERT((m->flags & (PG_FICTITIOUS | PG_UNMANAGED)) == 0,
|
|
|
|
("moea64_page_exists_quick: page %p is not managed", m));
|
2009-04-04 00:22:44 +00:00
|
|
|
loops = 0;
|
2010-06-10 16:56:35 +00:00
|
|
|
rv = FALSE;
|
|
|
|
vm_page_lock_queues();
|
2009-04-04 00:22:44 +00:00
|
|
|
LIST_FOREACH(pvo, vm_page_to_pvoh(m), pvo_vlink) {
|
2010-06-10 16:56:35 +00:00
|
|
|
if (pvo->pvo_pmap == pmap) {
|
|
|
|
rv = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
2009-04-04 00:22:44 +00:00
|
|
|
if (++loops >= 16)
|
|
|
|
break;
|
|
|
|
}
|
2010-06-10 16:56:35 +00:00
|
|
|
vm_page_unlock_queues();
|
|
|
|
return (rv);
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return the number of managed mappings to the given physical page
|
|
|
|
* that are wired.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
moea64_page_wired_mappings(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
struct pvo_entry *pvo;
|
|
|
|
int count;
|
|
|
|
|
|
|
|
count = 0;
|
2010-06-10 16:56:35 +00:00
|
|
|
if ((m->flags & PG_FICTITIOUS) != 0)
|
2009-04-04 00:22:44 +00:00
|
|
|
return (count);
|
2010-05-08 20:34:01 +00:00
|
|
|
vm_page_lock_queues();
|
2009-04-04 00:22:44 +00:00
|
|
|
LIST_FOREACH(pvo, vm_page_to_pvoh(m), pvo_vlink)
|
|
|
|
if ((pvo->pvo_vaddr & PVO_WIRED) != 0)
|
|
|
|
count++;
|
2010-05-08 20:34:01 +00:00
|
|
|
vm_page_unlock_queues();
|
2009-04-04 00:22:44 +00:00
|
|
|
return (count);
|
|
|
|
}
|
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
static uintptr_t moea64_vsidcontext;
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
uintptr_t
|
|
|
|
moea64_get_unique_vsid(void) {
|
|
|
|
u_int entropy;
|
|
|
|
register_t hash;
|
|
|
|
uint32_t mask;
|
|
|
|
int i;
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
entropy = 0;
|
|
|
|
__asm __volatile("mftb %0" : "=r"(entropy));
|
|
|
|
|
2010-08-29 18:17:38 +00:00
|
|
|
mtx_lock(&moea64_slb_mutex);
|
2010-07-13 05:32:19 +00:00
|
|
|
for (i = 0; i < NVSIDS; i += VSID_NBPW) {
|
|
|
|
u_int n;
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Create a new value by mutiplying by a prime and adding in
|
|
|
|
* entropy from the timebase register. This is to make the
|
|
|
|
* VSID more random so that the PT hash function collides
|
|
|
|
* less often. (Note that the prime casues gcc to do shifts
|
|
|
|
* instead of a multiply.)
|
|
|
|
*/
|
|
|
|
moea64_vsidcontext = (moea64_vsidcontext * 0x1105) + entropy;
|
2010-07-13 05:32:19 +00:00
|
|
|
hash = moea64_vsidcontext & (NVSIDS - 1);
|
2009-04-04 00:22:44 +00:00
|
|
|
if (hash == 0) /* 0 is special, avoid it */
|
|
|
|
continue;
|
|
|
|
n = hash >> 5;
|
|
|
|
mask = 1 << (hash & (VSID_NBPW - 1));
|
2010-07-13 05:32:19 +00:00
|
|
|
hash = (moea64_vsidcontext & VSID_HASHMASK);
|
2009-04-04 00:22:44 +00:00
|
|
|
if (moea64_vsid_bitmap[n] & mask) { /* collision? */
|
|
|
|
/* anything free in this bucket? */
|
|
|
|
if (moea64_vsid_bitmap[n] == 0xffffffff) {
|
|
|
|
entropy = (moea64_vsidcontext >> 20);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
i = ffs(~moea64_vsid_bitmap[i]) - 1;
|
|
|
|
mask = 1 << i;
|
2010-07-13 05:32:19 +00:00
|
|
|
hash &= VSID_HASHMASK & ~(VSID_NBPW - 1);
|
2009-04-04 00:22:44 +00:00
|
|
|
hash |= i;
|
|
|
|
}
|
|
|
|
moea64_vsid_bitmap[n] |= mask;
|
2010-08-29 18:17:38 +00:00
|
|
|
mtx_unlock(&moea64_slb_mutex);
|
2010-07-13 05:32:19 +00:00
|
|
|
return (hash);
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
2010-08-29 18:17:38 +00:00
|
|
|
mtx_unlock(&moea64_slb_mutex);
|
2010-07-13 05:32:19 +00:00
|
|
|
panic("%s: out of segments",__func__);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef __powerpc64__
|
|
|
|
void
|
|
|
|
moea64_pinit(mmu_t mmu, pmap_t pmap)
|
|
|
|
{
|
|
|
|
PMAP_LOCK_INIT(pmap);
|
|
|
|
|
|
|
|
SPLAY_INIT(&pmap->pm_slbtree);
|
|
|
|
pmap->pm_slb = slb_alloc_user_cache();
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
void
|
|
|
|
moea64_pinit(mmu_t mmu, pmap_t pmap)
|
|
|
|
{
|
|
|
|
int i;
|
2010-09-07 23:31:48 +00:00
|
|
|
uint32_t hash;
|
2010-07-13 05:32:19 +00:00
|
|
|
|
|
|
|
PMAP_LOCK_INIT(pmap);
|
|
|
|
|
|
|
|
if (pmap_bootstrapped)
|
|
|
|
pmap->pmap_phys = (pmap_t)moea64_kextract(mmu,
|
|
|
|
(vm_offset_t)pmap);
|
|
|
|
else
|
|
|
|
pmap->pmap_phys = pmap;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Allocate some segment registers for this pmap.
|
|
|
|
*/
|
|
|
|
hash = moea64_get_unique_vsid();
|
|
|
|
|
|
|
|
for (i = 0; i < 16; i++)
|
|
|
|
pmap->pm_sr[i] = VSID_MAKE(i, hash);
|
2010-09-07 23:31:48 +00:00
|
|
|
|
|
|
|
KASSERT(pmap->pm_sr[0] != 0, ("moea64_pinit: pm_sr[0] = 0"));
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
2010-07-13 05:32:19 +00:00
|
|
|
#endif
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize the pmap associated with process 0.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
moea64_pinit0(mmu_t mmu, pmap_t pm)
|
|
|
|
{
|
|
|
|
moea64_pinit(mmu, pm);
|
|
|
|
bzero(&pm->pm_stats, sizeof(pm->pm_stats));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the physical protection on the specified range of this map as requested.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
moea64_protect(mmu_t mmu, pmap_t pm, vm_offset_t sva, vm_offset_t eva,
|
|
|
|
vm_prot_t prot)
|
|
|
|
{
|
|
|
|
struct pvo_entry *pvo;
|
|
|
|
struct lpte *pt;
|
|
|
|
|
|
|
|
CTR4(KTR_PMAP, "moea64_protect: pm=%p sva=%#x eva=%#x prot=%#x", pm, sva,
|
|
|
|
eva, prot);
|
|
|
|
|
|
|
|
|
|
|
|
KASSERT(pm == &curproc->p_vmspace->vm_pmap || pm == kernel_pmap,
|
|
|
|
("moea64_protect: non current pmap"));
|
|
|
|
|
|
|
|
if ((prot & VM_PROT_READ) == VM_PROT_NONE) {
|
|
|
|
moea64_remove(mmu, pm, sva, eva);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
vm_page_lock_queues();
|
|
|
|
PMAP_LOCK(pm);
|
|
|
|
for (; sva < eva; sva += PAGE_SIZE) {
|
2010-07-13 05:32:19 +00:00
|
|
|
pvo = moea64_pvo_find_va(pm, sva);
|
2009-04-04 00:22:44 +00:00
|
|
|
if (pvo == NULL)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Grab the PTE pointer before we diddle with the cached PTE
|
|
|
|
* copy.
|
|
|
|
*/
|
|
|
|
LOCK_TABLE();
|
2010-07-13 05:32:19 +00:00
|
|
|
pt = moea64_pvo_to_pte(pvo);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Change the protection of the page.
|
|
|
|
*/
|
|
|
|
pvo->pvo_pte.lpte.pte_lo &= ~LPTE_PP;
|
|
|
|
pvo->pvo_pte.lpte.pte_lo |= LPTE_BR;
|
|
|
|
pvo->pvo_pte.lpte.pte_lo &= ~LPTE_NOEXEC;
|
|
|
|
if ((prot & VM_PROT_EXECUTE) == 0)
|
|
|
|
pvo->pvo_pte.lpte.pte_lo |= LPTE_NOEXEC;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the PVO is in the page table, update that pte as well.
|
|
|
|
*/
|
|
|
|
if (pt != NULL) {
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_pte_change(pt, &pvo->pvo_pte.lpte, pvo->pvo_vpn);
|
2009-04-04 00:22:44 +00:00
|
|
|
if ((pvo->pvo_pte.lpte.pte_lo &
|
|
|
|
(LPTE_I | LPTE_G | LPTE_NOEXEC)) == 0) {
|
2009-10-21 18:38:02 +00:00
|
|
|
moea64_syncicache(pm, sva,
|
|
|
|
pvo->pvo_pte.lpte.pte_lo & LPTE_RPGN,
|
|
|
|
PAGE_SIZE);
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
UNLOCK_TABLE();
|
|
|
|
}
|
|
|
|
vm_page_unlock_queues();
|
|
|
|
PMAP_UNLOCK(pm);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Map a list of wired pages into kernel virtual address space. This is
|
|
|
|
* intended for temporary mappings which do not need page modification or
|
|
|
|
* references recorded. Existing mappings in the region are overwritten.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
moea64_qenter(mmu_t mmu, vm_offset_t va, vm_page_t *m, int count)
|
|
|
|
{
|
|
|
|
while (count-- > 0) {
|
|
|
|
moea64_kenter(mmu, va, VM_PAGE_TO_PHYS(*m));
|
|
|
|
va += PAGE_SIZE;
|
|
|
|
m++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Remove page mappings from kernel virtual address space. Intended for
|
|
|
|
* temporary mappings entered by moea64_qenter.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
moea64_qremove(mmu_t mmu, vm_offset_t va, int count)
|
|
|
|
{
|
|
|
|
while (count-- > 0) {
|
|
|
|
moea64_kremove(mmu, va);
|
|
|
|
va += PAGE_SIZE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_release_vsid(uint64_t vsid)
|
2009-04-04 00:22:44 +00:00
|
|
|
{
|
2010-08-31 02:07:13 +00:00
|
|
|
int idx, mask;
|
2010-07-13 05:32:19 +00:00
|
|
|
|
2010-08-31 02:07:13 +00:00
|
|
|
mtx_lock(&moea64_slb_mutex);
|
|
|
|
idx = vsid & (NVSIDS-1);
|
|
|
|
mask = 1 << (idx % VSID_NBPW);
|
|
|
|
idx /= VSID_NBPW;
|
2010-09-07 23:31:48 +00:00
|
|
|
KASSERT(moea64_vsid_bitmap[idx] & mask,
|
|
|
|
("Freeing unallocated VSID %#jx", vsid));
|
2010-08-31 02:07:13 +00:00
|
|
|
moea64_vsid_bitmap[idx] &= ~mask;
|
|
|
|
mtx_unlock(&moea64_slb_mutex);
|
2010-07-13 05:32:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
moea64_release(mmu_t mmu, pmap_t pmap)
|
|
|
|
{
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/*
|
2010-07-13 05:32:19 +00:00
|
|
|
* Free segment registers' VSIDs
|
2009-04-04 00:22:44 +00:00
|
|
|
*/
|
2010-07-13 05:32:19 +00:00
|
|
|
#ifdef __powerpc64__
|
|
|
|
free_vsids(pmap);
|
|
|
|
slb_free_user_cache(pmap->pm_slb);
|
|
|
|
#else
|
2010-09-07 23:31:48 +00:00
|
|
|
KASSERT(pmap->pm_sr[0] != 0, ("moea64_release: pm_sr[0] = 0"));
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-09-07 23:31:48 +00:00
|
|
|
moea64_release_vsid(VSID_TO_HASH(pmap->pm_sr[0]));
|
2010-07-13 05:32:19 +00:00
|
|
|
#endif
|
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
PMAP_LOCK_DESTROY(pmap);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Remove the given range of addresses from the specified map.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
moea64_remove(mmu_t mmu, pmap_t pm, vm_offset_t sva, vm_offset_t eva)
|
|
|
|
{
|
|
|
|
struct pvo_entry *pvo;
|
|
|
|
|
|
|
|
vm_page_lock_queues();
|
|
|
|
PMAP_LOCK(pm);
|
|
|
|
for (; sva < eva; sva += PAGE_SIZE) {
|
2010-07-13 05:32:19 +00:00
|
|
|
pvo = moea64_pvo_find_va(pm, sva);
|
|
|
|
if (pvo != NULL)
|
|
|
|
moea64_pvo_remove(pvo);
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
vm_page_unlock_queues();
|
|
|
|
PMAP_UNLOCK(pm);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Remove physical page from all pmaps in which it resides. moea64_pvo_remove()
|
|
|
|
* will reflect changes in pte's back to the vm_page.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
moea64_remove_all(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
struct pvo_head *pvo_head;
|
|
|
|
struct pvo_entry *pvo, *next_pvo;
|
|
|
|
pmap_t pmap;
|
|
|
|
|
2010-05-08 20:34:01 +00:00
|
|
|
vm_page_lock_queues();
|
2009-04-04 00:22:44 +00:00
|
|
|
pvo_head = vm_page_to_pvoh(m);
|
|
|
|
for (pvo = LIST_FIRST(pvo_head); pvo != NULL; pvo = next_pvo) {
|
|
|
|
next_pvo = LIST_NEXT(pvo, pvo_vlink);
|
|
|
|
|
|
|
|
MOEA_PVO_CHECK(pvo); /* sanity check */
|
|
|
|
pmap = pvo->pvo_pmap;
|
|
|
|
PMAP_LOCK(pmap);
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_pvo_remove(pvo);
|
2009-04-04 00:22:44 +00:00
|
|
|
PMAP_UNLOCK(pmap);
|
|
|
|
}
|
2010-02-18 15:00:43 +00:00
|
|
|
if ((m->flags & PG_WRITEABLE) && moea64_is_modified(mmu, m)) {
|
|
|
|
moea64_attr_clear(m, LPTE_CHG);
|
|
|
|
vm_page_dirty(m);
|
|
|
|
}
|
2009-04-04 00:22:44 +00:00
|
|
|
vm_page_flag_clear(m, PG_WRITEABLE);
|
2010-05-08 20:34:01 +00:00
|
|
|
vm_page_unlock_queues();
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Allocate a physical page of memory directly from the phys_avail map.
|
|
|
|
* Can only be called from moea64_bootstrap before avail start and end are
|
|
|
|
* calculated.
|
|
|
|
*/
|
|
|
|
static vm_offset_t
|
|
|
|
moea64_bootstrap_alloc(vm_size_t size, u_int align)
|
|
|
|
{
|
|
|
|
vm_offset_t s, e;
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
size = round_page(size);
|
|
|
|
for (i = 0; phys_avail[i + 1] != 0; i += 2) {
|
|
|
|
if (align != 0)
|
|
|
|
s = (phys_avail[i] + align - 1) & ~(align - 1);
|
|
|
|
else
|
|
|
|
s = phys_avail[i];
|
|
|
|
e = s + size;
|
|
|
|
|
|
|
|
if (s < phys_avail[i] || e > phys_avail[i + 1])
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (s == phys_avail[i]) {
|
|
|
|
phys_avail[i] += size;
|
|
|
|
} else if (e == phys_avail[i + 1]) {
|
|
|
|
phys_avail[i + 1] -= size;
|
|
|
|
} else {
|
|
|
|
for (j = phys_avail_count * 2; j > i; j -= 2) {
|
|
|
|
phys_avail[j] = phys_avail[j - 2];
|
|
|
|
phys_avail[j + 1] = phys_avail[j - 1];
|
|
|
|
}
|
|
|
|
|
|
|
|
phys_avail[i + 3] = phys_avail[i + 1];
|
|
|
|
phys_avail[i + 1] = s;
|
|
|
|
phys_avail[i + 2] = e;
|
|
|
|
phys_avail_count++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (s);
|
|
|
|
}
|
|
|
|
panic("moea64_bootstrap_alloc: could not allocate memory");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
tlbia(void)
|
|
|
|
{
|
|
|
|
vm_offset_t i;
|
2010-07-13 05:32:19 +00:00
|
|
|
#ifndef __powerpc64__
|
2009-10-23 03:17:02 +00:00
|
|
|
register_t msr, scratch;
|
2010-07-13 05:32:19 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
TLBSYNC();
|
2009-10-23 03:17:02 +00:00
|
|
|
|
|
|
|
for (i = 0; i < 0xFF000; i += 0x00001000) {
|
2010-07-13 05:32:19 +00:00
|
|
|
#ifdef __powerpc64__
|
|
|
|
__asm __volatile("tlbiel %0" :: "r"(i));
|
|
|
|
#else
|
2009-10-23 03:17:02 +00:00
|
|
|
__asm __volatile("\
|
|
|
|
mfmsr %0; \
|
|
|
|
mr %1, %0; \
|
|
|
|
insrdi %1,%3,1,0; \
|
|
|
|
mtmsrd %1; \
|
2010-07-13 05:32:19 +00:00
|
|
|
isync; \
|
2009-10-23 03:17:02 +00:00
|
|
|
\
|
|
|
|
tlbiel %2; \
|
|
|
|
\
|
|
|
|
mtmsrd %0; \
|
2010-07-13 05:32:19 +00:00
|
|
|
isync;"
|
2009-10-23 03:17:02 +00:00
|
|
|
: "=r"(msr), "=r"(scratch) : "r"(i), "r"(1));
|
2010-07-13 05:32:19 +00:00
|
|
|
#endif
|
2009-10-23 03:17:02 +00:00
|
|
|
}
|
2010-07-13 05:32:19 +00:00
|
|
|
|
|
|
|
EIEIO();
|
|
|
|
TLBSYNC();
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef __powerpc64__
|
|
|
|
static void
|
|
|
|
slbia(void)
|
|
|
|
{
|
|
|
|
register_t seg0;
|
|
|
|
|
|
|
|
__asm __volatile ("slbia");
|
|
|
|
__asm __volatile ("slbmfee %0,%1; slbie %0;" : "=r"(seg0) : "r"(0));
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
2010-07-13 05:32:19 +00:00
|
|
|
#endif
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
static int
|
|
|
|
moea64_pvo_enter(pmap_t pm, uma_zone_t zone, struct pvo_head *pvo_head,
|
2009-10-23 03:17:02 +00:00
|
|
|
vm_offset_t va, vm_offset_t pa, uint64_t pte_lo, int flags)
|
2009-04-04 00:22:44 +00:00
|
|
|
{
|
|
|
|
struct pvo_entry *pvo;
|
|
|
|
uint64_t vsid;
|
|
|
|
int first;
|
|
|
|
u_int ptegidx;
|
|
|
|
int i;
|
|
|
|
int bootstrap;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* One nasty thing that can happen here is that the UMA calls to
|
|
|
|
* allocate new PVOs need to map more memory, which calls pvo_enter(),
|
|
|
|
* which calls UMA...
|
|
|
|
*
|
|
|
|
* We break the loop by detecting recursion and allocating out of
|
|
|
|
* the bootstrap pool.
|
|
|
|
*/
|
|
|
|
|
|
|
|
moea64_pvo_enter_calls++;
|
|
|
|
first = 0;
|
|
|
|
bootstrap = (flags & PVO_BOOTSTRAP);
|
|
|
|
|
|
|
|
if (!moea64_initialized)
|
|
|
|
bootstrap = 1;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Compute the PTE Group index.
|
|
|
|
*/
|
|
|
|
va &= ~ADDR_POFF;
|
|
|
|
vsid = va_to_vsid(pm, va);
|
2010-07-13 05:32:19 +00:00
|
|
|
ptegidx = va_to_pteg(vsid, va, flags & PVO_LARGE);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Remove any existing mapping for this page. Reuse the pvo entry if
|
|
|
|
* there is a mapping.
|
|
|
|
*/
|
2009-10-23 03:17:02 +00:00
|
|
|
LOCK_TABLE();
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
LIST_FOREACH(pvo, &moea64_pvo_table[ptegidx], pvo_olink) {
|
|
|
|
if (pvo->pvo_pmap == pm && PVO_VADDR(pvo) == va) {
|
|
|
|
if ((pvo->pvo_pte.lpte.pte_lo & LPTE_RPGN) == pa &&
|
|
|
|
(pvo->pvo_pte.lpte.pte_lo & LPTE_PP) ==
|
|
|
|
(pte_lo & LPTE_PP)) {
|
2010-07-13 05:32:19 +00:00
|
|
|
if (!(pvo->pvo_pte.lpte.pte_hi & LPTE_VALID)) {
|
|
|
|
/* Re-insert if spilled */
|
|
|
|
i = moea64_pte_insert(ptegidx,
|
|
|
|
&pvo->pvo_pte.lpte);
|
|
|
|
if (i >= 0)
|
|
|
|
PVO_PTEGIDX_SET(pvo, i);
|
|
|
|
moea64_pte_overflow--;
|
|
|
|
}
|
2009-10-23 03:17:02 +00:00
|
|
|
UNLOCK_TABLE();
|
2009-04-04 00:22:44 +00:00
|
|
|
return (0);
|
|
|
|
}
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_pvo_remove(pvo);
|
2009-04-04 00:22:44 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If we aren't overwriting a mapping, try to allocate.
|
|
|
|
*/
|
|
|
|
if (bootstrap) {
|
|
|
|
if (moea64_bpvo_pool_index >= BPVO_POOL_SIZE) {
|
2010-07-13 05:32:19 +00:00
|
|
|
panic("moea64_enter: bpvo pool exhausted, %d, %d, %zd",
|
2009-04-04 00:22:44 +00:00
|
|
|
moea64_bpvo_pool_index, BPVO_POOL_SIZE,
|
|
|
|
BPVO_POOL_SIZE * sizeof(struct pvo_entry));
|
|
|
|
}
|
|
|
|
pvo = &moea64_bpvo_pool[moea64_bpvo_pool_index];
|
|
|
|
moea64_bpvo_pool_index++;
|
|
|
|
bootstrap = 1;
|
|
|
|
} else {
|
2009-10-23 03:17:02 +00:00
|
|
|
/*
|
2010-03-04 17:24:31 +00:00
|
|
|
* Note: drop the table lock around the UMA allocation in
|
2009-10-23 03:17:02 +00:00
|
|
|
* case the UMA allocator needs to manipulate the page
|
|
|
|
* table. The mapping we are working with is already
|
|
|
|
* protected by the PMAP lock.
|
|
|
|
*/
|
|
|
|
UNLOCK_TABLE();
|
2009-04-04 00:22:44 +00:00
|
|
|
pvo = uma_zalloc(zone, M_NOWAIT);
|
2009-10-23 03:17:02 +00:00
|
|
|
LOCK_TABLE();
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (pvo == NULL) {
|
2009-10-23 03:17:02 +00:00
|
|
|
UNLOCK_TABLE();
|
2009-04-04 00:22:44 +00:00
|
|
|
return (ENOMEM);
|
|
|
|
}
|
|
|
|
|
|
|
|
moea64_pvo_entries++;
|
|
|
|
pvo->pvo_vaddr = va;
|
2010-07-13 05:32:19 +00:00
|
|
|
pvo->pvo_vpn = (uint64_t)((va & ADDR_PIDX) >> ADDR_PIDX_SHFT)
|
|
|
|
| (vsid << 16);
|
2009-04-04 00:22:44 +00:00
|
|
|
pvo->pvo_pmap = pm;
|
|
|
|
LIST_INSERT_HEAD(&moea64_pvo_table[ptegidx], pvo, pvo_olink);
|
|
|
|
pvo->pvo_vaddr &= ~ADDR_POFF;
|
|
|
|
|
|
|
|
if (!(flags & VM_PROT_EXECUTE))
|
|
|
|
pte_lo |= LPTE_NOEXEC;
|
|
|
|
if (flags & PVO_WIRED)
|
|
|
|
pvo->pvo_vaddr |= PVO_WIRED;
|
|
|
|
if (pvo_head != &moea64_pvo_kunmanaged)
|
|
|
|
pvo->pvo_vaddr |= PVO_MANAGED;
|
|
|
|
if (bootstrap)
|
|
|
|
pvo->pvo_vaddr |= PVO_BOOTSTRAP;
|
|
|
|
if (flags & PVO_FAKE)
|
|
|
|
pvo->pvo_vaddr |= PVO_FAKE;
|
2010-07-13 05:32:19 +00:00
|
|
|
if (flags & PVO_LARGE)
|
|
|
|
pvo->pvo_vaddr |= PVO_LARGE;
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
moea64_pte_create(&pvo->pvo_pte.lpte, vsid, va,
|
2010-07-13 05:32:19 +00:00
|
|
|
(uint64_t)(pa) | pte_lo, flags);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Remember if the list was empty and therefore will be the first
|
|
|
|
* item.
|
|
|
|
*/
|
|
|
|
if (LIST_FIRST(pvo_head) == NULL)
|
|
|
|
first = 1;
|
|
|
|
LIST_INSERT_HEAD(pvo_head, pvo, pvo_vlink);
|
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
if (pvo->pvo_vaddr & PVO_WIRED) {
|
|
|
|
pvo->pvo_pte.lpte.pte_hi |= LPTE_WIRED;
|
2009-04-04 00:22:44 +00:00
|
|
|
pm->pm_stats.wired_count++;
|
2010-07-13 05:32:19 +00:00
|
|
|
}
|
2009-04-04 00:22:44 +00:00
|
|
|
pm->pm_stats.resident_count++;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We hope this succeeds but it isn't required.
|
|
|
|
*/
|
|
|
|
i = moea64_pte_insert(ptegidx, &pvo->pvo_pte.lpte);
|
|
|
|
if (i >= 0) {
|
|
|
|
PVO_PTEGIDX_SET(pvo, i);
|
|
|
|
} else {
|
|
|
|
panic("moea64_pvo_enter: overflow");
|
|
|
|
moea64_pte_overflow++;
|
|
|
|
}
|
|
|
|
|
2010-02-18 15:00:43 +00:00
|
|
|
if (pm == kernel_pmap)
|
|
|
|
isync();
|
|
|
|
|
2009-10-23 03:17:02 +00:00
|
|
|
UNLOCK_TABLE();
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
#ifdef __powerpc64__
|
|
|
|
/*
|
|
|
|
* Make sure all our bootstrap mappings are in the SLB as soon
|
|
|
|
* as virtual memory is switched on.
|
|
|
|
*/
|
|
|
|
if (!pmap_bootstrapped)
|
|
|
|
moea64_bootstrap_slb_prefault(va, flags & PVO_LARGE);
|
|
|
|
#endif
|
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
return (first ? ENOENT : 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_pvo_remove(struct pvo_entry *pvo)
|
2009-04-04 00:22:44 +00:00
|
|
|
{
|
|
|
|
struct lpte *pt;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If there is an active pte entry, we need to deactivate it (and
|
|
|
|
* save the ref & cfg bits).
|
|
|
|
*/
|
|
|
|
LOCK_TABLE();
|
2010-07-13 05:32:19 +00:00
|
|
|
pt = moea64_pvo_to_pte(pvo);
|
2009-04-04 00:22:44 +00:00
|
|
|
if (pt != NULL) {
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_pte_unset(pt, &pvo->pvo_pte.lpte, pvo->pvo_vpn);
|
2009-04-04 00:22:44 +00:00
|
|
|
PVO_PTEGIDX_CLR(pvo);
|
|
|
|
} else {
|
|
|
|
moea64_pte_overflow--;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Update our statistics.
|
|
|
|
*/
|
|
|
|
pvo->pvo_pmap->pm_stats.resident_count--;
|
2010-02-18 15:00:43 +00:00
|
|
|
if (pvo->pvo_vaddr & PVO_WIRED)
|
2009-04-04 00:22:44 +00:00
|
|
|
pvo->pvo_pmap->pm_stats.wired_count--;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Save the REF/CHG bits into their cache if the page is managed.
|
|
|
|
*/
|
|
|
|
if ((pvo->pvo_vaddr & (PVO_MANAGED|PVO_FAKE)) == PVO_MANAGED) {
|
|
|
|
struct vm_page *pg;
|
|
|
|
|
|
|
|
pg = PHYS_TO_VM_PAGE(pvo->pvo_pte.lpte.pte_lo & LPTE_RPGN);
|
|
|
|
if (pg != NULL) {
|
|
|
|
moea64_attr_save(pg, pvo->pvo_pte.lpte.pte_lo &
|
|
|
|
(LPTE_REF | LPTE_CHG));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Remove this PVO from the PV list.
|
|
|
|
*/
|
|
|
|
LIST_REMOVE(pvo, pvo_vlink);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Remove this from the overflow list and return it to the pool
|
|
|
|
* if we aren't going to reuse it.
|
|
|
|
*/
|
|
|
|
LIST_REMOVE(pvo, pvo_olink);
|
2010-03-04 06:39:58 +00:00
|
|
|
UNLOCK_TABLE();
|
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
if (!(pvo->pvo_vaddr & PVO_BOOTSTRAP))
|
2010-02-18 15:00:43 +00:00
|
|
|
uma_zfree((pvo->pvo_vaddr & PVO_MANAGED) ? moea64_mpvo_zone :
|
2009-04-04 00:22:44 +00:00
|
|
|
moea64_upvo_zone, pvo);
|
2010-03-04 06:39:58 +00:00
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
moea64_pvo_entries--;
|
|
|
|
moea64_pvo_remove_calls++;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct pvo_entry *
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_pvo_find_va(pmap_t pm, vm_offset_t va)
|
2009-04-04 00:22:44 +00:00
|
|
|
{
|
|
|
|
struct pvo_entry *pvo;
|
|
|
|
int ptegidx;
|
|
|
|
uint64_t vsid;
|
2010-07-13 05:32:19 +00:00
|
|
|
#ifdef __powerpc64__
|
|
|
|
struct slb slb;
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
/* The page is not mapped if the segment isn't */
|
|
|
|
if (va_to_slb_entry(pm, va, &slb) != 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
vsid = (slb.slbv & SLBV_VSID_MASK) >> SLBV_VSID_SHIFT;
|
|
|
|
if (slb.slbv & SLBV_L)
|
|
|
|
va &= ~moea64_large_page_mask;
|
|
|
|
else
|
|
|
|
va &= ~ADDR_POFF;
|
|
|
|
ptegidx = va_to_pteg(vsid, va, slb.slbv & SLBV_L);
|
|
|
|
#else
|
2009-04-04 00:22:44 +00:00
|
|
|
va &= ~ADDR_POFF;
|
|
|
|
vsid = va_to_vsid(pm, va);
|
2010-07-13 05:32:19 +00:00
|
|
|
ptegidx = va_to_pteg(vsid, va, 0);
|
|
|
|
#endif
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
LOCK_TABLE();
|
|
|
|
LIST_FOREACH(pvo, &moea64_pvo_table[ptegidx], pvo_olink) {
|
2010-07-13 05:32:19 +00:00
|
|
|
if (pvo->pvo_pmap == pm && PVO_VADDR(pvo) == va)
|
2009-04-04 00:22:44 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
UNLOCK_TABLE();
|
|
|
|
|
|
|
|
return (pvo);
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct lpte *
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_pvo_to_pte(const struct pvo_entry *pvo)
|
2009-04-04 00:22:44 +00:00
|
|
|
{
|
2010-07-13 05:32:19 +00:00
|
|
|
struct lpte *pt;
|
|
|
|
int pteidx, ptegidx;
|
|
|
|
uint64_t vsid;
|
|
|
|
|
|
|
|
ASSERT_TABLE_LOCK();
|
|
|
|
|
|
|
|
/* If the PTEG index is not set, then there is no page table entry */
|
|
|
|
if (!PVO_PTEGIDX_ISSET(pvo))
|
|
|
|
return (NULL);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
/*
|
2010-07-13 05:32:19 +00:00
|
|
|
* Calculate the ptegidx
|
2009-04-04 00:22:44 +00:00
|
|
|
*/
|
2010-07-13 05:32:19 +00:00
|
|
|
vsid = PVO_VSID(pvo);
|
|
|
|
ptegidx = va_to_pteg(vsid, PVO_VADDR(pvo),
|
|
|
|
pvo->pvo_vaddr & PVO_LARGE);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
/*
|
|
|
|
* We can find the actual pte entry without searching by grabbing
|
|
|
|
* the PTEG index from 3 unused bits in pvo_vaddr and by
|
|
|
|
* noticing the HID bit.
|
|
|
|
*/
|
|
|
|
if (pvo->pvo_pte.lpte.pte_hi & LPTE_HID)
|
|
|
|
ptegidx ^= moea64_pteg_mask;
|
2009-04-04 00:22:44 +00:00
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
pteidx = (ptegidx << 3) | PVO_PTEGIDX_GET(pvo);
|
2009-04-04 00:22:44 +00:00
|
|
|
|
|
|
|
if ((pvo->pvo_pte.lpte.pte_hi & LPTE_VALID) &&
|
|
|
|
!PVO_PTEGIDX_ISSET(pvo)) {
|
|
|
|
panic("moea64_pvo_to_pte: pvo %p has valid pte in pvo but no "
|
|
|
|
"valid pte index", pvo);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((pvo->pvo_pte.lpte.pte_hi & LPTE_VALID) == 0 &&
|
|
|
|
PVO_PTEGIDX_ISSET(pvo)) {
|
|
|
|
panic("moea64_pvo_to_pte: pvo %p has valid pte index in pvo "
|
|
|
|
"pvo but no valid pte", pvo);
|
|
|
|
}
|
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
pt = &moea64_pteg_table[pteidx >> 3].pt[pteidx & 7];
|
2009-04-04 00:22:44 +00:00
|
|
|
if ((pt->pte_hi ^ (pvo->pvo_pte.lpte.pte_hi & ~LPTE_VALID)) ==
|
|
|
|
LPTE_VALID) {
|
|
|
|
if ((pvo->pvo_pte.lpte.pte_hi & LPTE_VALID) == 0) {
|
|
|
|
panic("moea64_pvo_to_pte: pvo %p has valid pte in "
|
|
|
|
"moea64_pteg_table %p but invalid in pvo", pvo, pt);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (((pt->pte_lo ^ pvo->pvo_pte.lpte.pte_lo) &
|
2010-03-15 00:27:40 +00:00
|
|
|
~(LPTE_M|LPTE_CHG|LPTE_REF)) != 0) {
|
2009-04-04 00:22:44 +00:00
|
|
|
panic("moea64_pvo_to_pte: pvo %p pte does not match "
|
|
|
|
"pte %p in moea64_pteg_table difference is %#x",
|
|
|
|
pvo, pt,
|
|
|
|
(uint32_t)(pt->pte_lo ^ pvo->pvo_pte.lpte.pte_lo));
|
|
|
|
}
|
|
|
|
|
|
|
|
return (pt);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pvo->pvo_pte.lpte.pte_hi & LPTE_VALID) {
|
|
|
|
panic("moea64_pvo_to_pte: pvo %p has invalid pte %p in "
|
|
|
|
"moea64_pteg_table but valid in pvo", pvo, pt);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (NULL);
|
|
|
|
}
|
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
static __inline int
|
|
|
|
moea64_pte_spillable_ident(u_int ptegidx)
|
|
|
|
{
|
|
|
|
struct lpte *pt;
|
|
|
|
int i, j, k;
|
|
|
|
|
|
|
|
/* Start at a random slot */
|
|
|
|
i = mftb() % 8;
|
|
|
|
k = -1;
|
|
|
|
for (j = 0; j < 8; j++) {
|
|
|
|
pt = &moea64_pteg_table[ptegidx].pt[(i + j) % 8];
|
|
|
|
if (pt->pte_hi & (LPTE_LOCKED | LPTE_WIRED))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
/* This is a candidate, so remember it */
|
|
|
|
k = (i + j) % 8;
|
|
|
|
|
|
|
|
/* Try to get a page that has not been used lately */
|
|
|
|
if (!(pt->pte_lo & LPTE_REF))
|
|
|
|
return (k);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (k);
|
|
|
|
}
|
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
static int
|
|
|
|
moea64_pte_insert(u_int ptegidx, struct lpte *pvo_pt)
|
|
|
|
{
|
|
|
|
struct lpte *pt;
|
2010-07-13 05:32:19 +00:00
|
|
|
struct pvo_entry *pvo;
|
|
|
|
u_int pteg_bktidx;
|
2009-04-04 00:22:44 +00:00
|
|
|
int i;
|
|
|
|
|
|
|
|
ASSERT_TABLE_LOCK();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* First try primary hash.
|
|
|
|
*/
|
2010-07-13 05:32:19 +00:00
|
|
|
pteg_bktidx = ptegidx;
|
|
|
|
for (pt = moea64_pteg_table[pteg_bktidx].pt, i = 0; i < 8; i++, pt++) {
|
|
|
|
if ((pt->pte_hi & (LPTE_VALID | LPTE_LOCKED)) == 0) {
|
2009-04-04 00:22:44 +00:00
|
|
|
pvo_pt->pte_hi &= ~LPTE_HID;
|
|
|
|
moea64_pte_set(pt, pvo_pt);
|
|
|
|
return (i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Now try secondary hash.
|
|
|
|
*/
|
2010-07-13 05:32:19 +00:00
|
|
|
pteg_bktidx ^= moea64_pteg_mask;
|
|
|
|
for (pt = moea64_pteg_table[pteg_bktidx].pt, i = 0; i < 8; i++, pt++) {
|
|
|
|
if ((pt->pte_hi & (LPTE_VALID | LPTE_LOCKED)) == 0) {
|
2009-04-04 00:22:44 +00:00
|
|
|
pvo_pt->pte_hi |= LPTE_HID;
|
|
|
|
moea64_pte_set(pt, pvo_pt);
|
|
|
|
return (i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-13 05:32:19 +00:00
|
|
|
/*
|
|
|
|
* Out of luck. Find a PTE to sacrifice.
|
|
|
|
*/
|
|
|
|
pteg_bktidx = ptegidx;
|
|
|
|
i = moea64_pte_spillable_ident(pteg_bktidx);
|
|
|
|
if (i < 0) {
|
|
|
|
pteg_bktidx ^= moea64_pteg_mask;
|
|
|
|
i = moea64_pte_spillable_ident(pteg_bktidx);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i < 0) {
|
|
|
|
/* No freeable slots in either PTEG? We're hosed. */
|
|
|
|
panic("moea64_pte_insert: overflow");
|
|
|
|
return (-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pteg_bktidx == ptegidx)
|
|
|
|
pvo_pt->pte_hi &= ~LPTE_HID;
|
|
|
|
else
|
|
|
|
pvo_pt->pte_hi |= LPTE_HID;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Synchronize the sacrifice PTE with its PVO, then mark both
|
|
|
|
* invalid. The PVO will be reused when/if the VM system comes
|
|
|
|
* here after a fault.
|
|
|
|
*/
|
|
|
|
pt = &moea64_pteg_table[pteg_bktidx].pt[i];
|
|
|
|
|
|
|
|
if (pt->pte_hi & LPTE_HID)
|
|
|
|
pteg_bktidx ^= moea64_pteg_mask; /* PTEs indexed by primary */
|
|
|
|
|
|
|
|
LIST_FOREACH(pvo, &moea64_pvo_table[pteg_bktidx], pvo_olink) {
|
|
|
|
if (pvo->pvo_pte.lpte.pte_hi == pt->pte_hi) {
|
|
|
|
KASSERT(pvo->pvo_pte.lpte.pte_hi & LPTE_VALID,
|
|
|
|
("Invalid PVO for valid PTE!"));
|
|
|
|
moea64_pte_unset(pt, &pvo->pvo_pte.lpte, pvo->pvo_vpn);
|
|
|
|
PVO_PTEGIDX_CLR(pvo);
|
|
|
|
moea64_pte_overflow++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
KASSERT(pvo->pvo_pte.lpte.pte_hi == pt->pte_hi,
|
|
|
|
("Unable to find PVO for spilled PTE"));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the new PTE.
|
|
|
|
*/
|
|
|
|
moea64_pte_set(pt, pvo_pt);
|
|
|
|
|
|
|
|
return (i);
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static boolean_t
|
|
|
|
moea64_query_bit(vm_page_t m, u_int64_t ptebit)
|
|
|
|
{
|
|
|
|
struct pvo_entry *pvo;
|
|
|
|
struct lpte *pt;
|
|
|
|
|
|
|
|
if (moea64_attr_fetch(m) & ptebit)
|
|
|
|
return (TRUE);
|
|
|
|
|
2010-05-26 18:00:44 +00:00
|
|
|
vm_page_lock_queues();
|
2010-03-20 14:35:24 +00:00
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
LIST_FOREACH(pvo, vm_page_to_pvoh(m), pvo_vlink) {
|
|
|
|
MOEA_PVO_CHECK(pvo); /* sanity check */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* See if we saved the bit off. If so, cache it and return
|
|
|
|
* success.
|
|
|
|
*/
|
|
|
|
if (pvo->pvo_pte.lpte.pte_lo & ptebit) {
|
|
|
|
moea64_attr_save(m, ptebit);
|
|
|
|
MOEA_PVO_CHECK(pvo); /* sanity check */
|
2010-05-26 18:00:44 +00:00
|
|
|
vm_page_unlock_queues();
|
2009-04-04 00:22:44 +00:00
|
|
|
return (TRUE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* No luck, now go through the hard part of looking at the PTEs
|
|
|
|
* themselves. Sync so that any pending REF/CHG bits are flushed to
|
|
|
|
* the PTEs.
|
|
|
|
*/
|
|
|
|
SYNC();
|
|
|
|
LIST_FOREACH(pvo, vm_page_to_pvoh(m), pvo_vlink) {
|
|
|
|
MOEA_PVO_CHECK(pvo); /* sanity check */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* See if this pvo has a valid PTE. if so, fetch the
|
|
|
|
* REF/CHG bits from the valid PTE. If the appropriate
|
|
|
|
* ptebit is set, cache it and return success.
|
|
|
|
*/
|
2010-03-20 14:35:24 +00:00
|
|
|
LOCK_TABLE();
|
2010-07-13 05:32:19 +00:00
|
|
|
pt = moea64_pvo_to_pte(pvo);
|
2009-04-04 00:22:44 +00:00
|
|
|
if (pt != NULL) {
|
|
|
|
moea64_pte_synch(pt, &pvo->pvo_pte.lpte);
|
|
|
|
if (pvo->pvo_pte.lpte.pte_lo & ptebit) {
|
|
|
|
UNLOCK_TABLE();
|
|
|
|
|
|
|
|
moea64_attr_save(m, ptebit);
|
|
|
|
MOEA_PVO_CHECK(pvo); /* sanity check */
|
2010-05-26 18:00:44 +00:00
|
|
|
vm_page_unlock_queues();
|
2009-04-04 00:22:44 +00:00
|
|
|
return (TRUE);
|
|
|
|
}
|
|
|
|
}
|
2010-03-20 14:35:24 +00:00
|
|
|
UNLOCK_TABLE();
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
2010-05-26 18:00:44 +00:00
|
|
|
vm_page_unlock_queues();
|
2009-04-04 00:22:44 +00:00
|
|
|
return (FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
static u_int
|
2010-06-10 16:56:35 +00:00
|
|
|
moea64_clear_bit(vm_page_t m, u_int64_t ptebit)
|
2009-04-04 00:22:44 +00:00
|
|
|
{
|
|
|
|
u_int count;
|
|
|
|
struct pvo_entry *pvo;
|
|
|
|
struct lpte *pt;
|
|
|
|
|
2010-06-10 16:56:35 +00:00
|
|
|
vm_page_lock_queues();
|
2010-03-20 14:35:24 +00:00
|
|
|
|
2009-04-04 00:22:44 +00:00
|
|
|
/*
|
|
|
|
* Clear the cached value.
|
|
|
|
*/
|
|
|
|
moea64_attr_clear(m, ptebit);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Sync so that any pending REF/CHG bits are flushed to the PTEs (so
|
|
|
|
* we can reset the right ones). note that since the pvo entries and
|
|
|
|
* list heads are accessed via BAT0 and are never placed in the page
|
|
|
|
* table, we don't have to worry about further accesses setting the
|
|
|
|
* REF/CHG bits.
|
|
|
|
*/
|
|
|
|
SYNC();
|
|
|
|
|
|
|
|
/*
|
|
|
|
* For each pvo entry, clear the pvo's ptebit. If this pvo has a
|
|
|
|
* valid pte clear the ptebit from the valid pte.
|
|
|
|
*/
|
|
|
|
count = 0;
|
|
|
|
LIST_FOREACH(pvo, vm_page_to_pvoh(m), pvo_vlink) {
|
|
|
|
MOEA_PVO_CHECK(pvo); /* sanity check */
|
|
|
|
|
2010-03-20 14:35:24 +00:00
|
|
|
LOCK_TABLE();
|
2010-07-13 05:32:19 +00:00
|
|
|
pt = moea64_pvo_to_pte(pvo);
|
2009-04-04 00:22:44 +00:00
|
|
|
if (pt != NULL) {
|
|
|
|
moea64_pte_synch(pt, &pvo->pvo_pte.lpte);
|
|
|
|
if (pvo->pvo_pte.lpte.pte_lo & ptebit) {
|
|
|
|
count++;
|
2010-07-13 05:32:19 +00:00
|
|
|
moea64_pte_clear(pt, pvo->pvo_vpn, ptebit);
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
pvo->pvo_pte.lpte.pte_lo &= ~ptebit;
|
|
|
|
MOEA_PVO_CHECK(pvo); /* sanity check */
|
2010-03-20 14:35:24 +00:00
|
|
|
UNLOCK_TABLE();
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
2010-06-10 16:56:35 +00:00
|
|
|
vm_page_unlock_queues();
|
2009-04-04 00:22:44 +00:00
|
|
|
return (count);
|
|
|
|
}
|
|
|
|
|
|
|
|
boolean_t
|
|
|
|
moea64_dev_direct_mapped(mmu_t mmu, vm_offset_t pa, vm_size_t size)
|
|
|
|
{
|
2010-02-25 03:49:17 +00:00
|
|
|
struct pvo_entry *pvo;
|
|
|
|
vm_offset_t ppa;
|
|
|
|
int error = 0;
|
|
|
|
|
|
|
|
PMAP_LOCK(kernel_pmap);
|
|
|
|
for (ppa = pa & ~ADDR_POFF; ppa < pa + size; ppa += PAGE_SIZE) {
|
2010-07-13 05:32:19 +00:00
|
|
|
pvo = moea64_pvo_find_va(kernel_pmap, ppa);
|
2010-02-25 03:49:17 +00:00
|
|
|
if (pvo == NULL ||
|
|
|
|
(pvo->pvo_pte.lpte.pte_lo & LPTE_RPGN) != ppa) {
|
|
|
|
error = EFAULT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
PMAP_UNLOCK(kernel_pmap);
|
|
|
|
|
|
|
|
return (error);
|
2009-04-04 00:22:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Map a set of physical memory pages into the kernel virtual
|
|
|
|
* address space. Return a pointer to where it is mapped. This
|
|
|
|
* routine is intended to be used for mapping device memory,
|
|
|
|
* NOT real memory.
|
|
|
|
*/
|
|
|
|
void *
|
|
|
|
moea64_mapdev(mmu_t mmu, vm_offset_t pa, vm_size_t size)
|
|
|
|
{
|
|
|
|
vm_offset_t va, tmpva, ppa, offset;
|
|
|
|
|
|
|
|
ppa = trunc_page(pa);
|
|
|
|
offset = pa & PAGE_MASK;
|
|
|
|
size = roundup(offset + size, PAGE_SIZE);
|
|
|
|
|
|
|
|
va = kmem_alloc_nofault(kernel_map, size);
|
|
|
|
|
|
|
|
if (!va)
|
|
|
|
panic("moea64_mapdev: Couldn't alloc kernel virtual memory");
|
|
|
|
|
|
|
|
for (tmpva = va; size > 0;) {
|
|
|
|
moea64_kenter(mmu, tmpva, ppa);
|
|
|
|
size -= PAGE_SIZE;
|
|
|
|
tmpva += PAGE_SIZE;
|
|
|
|
ppa += PAGE_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ((void *)(va + offset));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
moea64_unmapdev(mmu_t mmu, vm_offset_t va, vm_size_t size)
|
|
|
|
{
|
|
|
|
vm_offset_t base, offset;
|
|
|
|
|
|
|
|
base = trunc_page(va);
|
|
|
|
offset = va & PAGE_MASK;
|
|
|
|
size = roundup(offset + size, PAGE_SIZE);
|
|
|
|
|
|
|
|
kmem_free(kernel_map, base, size);
|
|
|
|
}
|
|
|
|
|
2009-10-21 18:38:02 +00:00
|
|
|
static void
|
|
|
|
moea64_sync_icache(mmu_t mmu, pmap_t pm, vm_offset_t va, vm_size_t sz)
|
|
|
|
{
|
|
|
|
struct pvo_entry *pvo;
|
|
|
|
vm_offset_t lim;
|
|
|
|
vm_paddr_t pa;
|
|
|
|
vm_size_t len;
|
|
|
|
|
|
|
|
PMAP_LOCK(pm);
|
|
|
|
while (sz > 0) {
|
|
|
|
lim = round_page(va);
|
|
|
|
len = MIN(lim - va, sz);
|
2010-07-13 05:32:19 +00:00
|
|
|
pvo = moea64_pvo_find_va(pm, va & ~ADDR_POFF);
|
2009-10-21 18:38:02 +00:00
|
|
|
if (pvo != NULL) {
|
2010-03-15 00:27:40 +00:00
|
|
|
pa = (pvo->pvo_pte.pte.pte_lo & LPTE_RPGN) |
|
2009-10-21 18:38:02 +00:00
|
|
|
(va & ADDR_POFF);
|
|
|
|
moea64_syncicache(pm, va, pa, len);
|
|
|
|
}
|
|
|
|
va += len;
|
|
|
|
sz -= len;
|
|
|
|
}
|
|
|
|
PMAP_UNLOCK(pm);
|
|
|
|
}
|