summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--sys/arch/hppa/hppa/pmap.c1364
1 files changed, 1364 insertions, 0 deletions
diff --git a/sys/arch/hppa/hppa/pmap.c b/sys/arch/hppa/hppa/pmap.c
new file mode 100644
index 00000000000..926a508e3ec
--- /dev/null
+++ b/sys/arch/hppa/hppa/pmap.c
@@ -0,0 +1,1364 @@
+/* $OpenBSD: pmap.c,v 1.1 1998/08/20 15:46:49 mickey Exp $ */
+
+/*
+ * Copyright (c) 1998 Michael Shalayeff
+ * 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 Michael Shalayeff.
+ * 4. The name of the author may not be used to endorse or promote products
+ * derived from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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 1996 1995 by Open Software Foundation, Inc.
+ * All Rights Reserved
+ *
+ * Permission to use, copy, modify, and distribute this software and
+ * its documentation for any purpose and without fee is hereby granted,
+ * provided that the above copyright notice appears in all copies and
+ * that both the copyright notice and this permission notice appear in
+ * supporting documentation.
+ *
+ * OSF DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE
+ * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
+ * FOR A PARTICULAR PURPOSE.
+ *
+ * IN NO EVENT SHALL OSF BE LIABLE FOR ANY SPECIAL, INDIRECT, OR
+ * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
+ * LOSS OF USE, DATA OR PROFITS, WHETHER IN ACTION OF CONTRACT,
+ * NEGLIGENCE, OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION
+ * WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+ */
+/*
+ * Mach Operating System
+ * Copyright (c) 1990,1991,1992,1993,1994 The University of Utah and
+ * the Computer Systems Laboratory (CSL).
+ * Copyright (c) 1991,1987 Carnegie Mellon University.
+ * All rights reserved.
+ *
+ * Permission to use, copy, modify and distribute this software and its
+ * documentation is hereby granted, provided that both the copyright
+ * notice and this permission notice appear in all copies of the
+ * software, derivative works or modified versions, and any portions
+ * thereof, and that both notices appear in supporting documentation,
+ * and that all advertising materials mentioning features or use of
+ * this software display the following acknowledgement: ``This product
+ * includes software developed by the Computer Systems Laboratory at
+ * the University of Utah.''
+ *
+ * CARNEGIE MELLON, THE UNIVERSITY OF UTAH AND CSL ALLOW FREE USE OF
+ * THIS SOFTWARE IN ITS "AS IS" CONDITION, AND DISCLAIM ANY LIABILITY
+ * OF ANY KIND FOR ANY DAMAGES WHATSOEVER RESULTING FROM THE USE OF
+ * THIS SOFTWARE.
+ *
+ * CSL requests users of this software to return to csl-dist@cs.utah.edu any
+ * improvements that they make and grant CSL redistribution rights.
+ *
+ * Carnegie Mellon requests users of this software to return to
+ * Software Distribution Coordinator or Software.Distribution@CS.CMU.EDU
+ * School of Computer Science
+ * Carnegie Mellon University
+ * Pittsburgh PA 15213-3890
+ * any improvements or extensions that they make and grant Carnegie Mellon
+ * the rights to redistribute these changes.
+ *
+ * Utah $Hdr: pmap.c 1.49 94/12/15$
+ * Author: Mike Hibler, Bob Wheeler, University of Utah CSL, 10/90
+ */
+/*
+ * Manages physical address maps for hppa.
+ *
+ * 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 virtual-to-physical 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 or reduced protection
+ * operations until such time as they are actually
+ * necessary. This module is given full information to
+ * when physical maps must be made correct.
+ *
+ */
+/*
+ * CAVAETS:
+ *
+ * PAGE_SIZE must equal NBPG
+ * Needs more work for MP support
+ */
+
+#include <sys/param.h>
+#include <sys/systm.h>
+#include <sys/lock.h>
+#include <sys/malloc.h>
+#include <sys/user.h>
+#include <sys/proc.h>
+
+#include <vm/vm.h>
+#include <vm/vm_kern.h>
+#include <vm/vm_page.h>
+#include <vm/pmap.h>
+
+#include <machine/reg.h>
+#include <machine/psl.h>
+#include <machine/cpu.h>
+#include <machine/pmap.h>
+#include <machine/pte.h>
+#include <machine/cpufunc.h>
+
+#ifdef DEBUG
+struct {
+ int kernel; /* entering kernel mapping */
+ int user; /* entering user mapping */
+ int ptpneeded; /* needed to allocate a PT page */
+ int pwchange; /* no mapping change, just wiring or protection */
+ int wchange; /* no mapping change, just wiring */
+ int mchange; /* was mapped but mapping to different page */
+ int managed; /* a managed page */
+ int firstpv; /* first mapping for this PA */
+ int secondpv; /* second mapping for this PA */
+ int ci; /* cache inhibited */
+ int unmanaged; /* not a managed page */
+ int flushes; /* cache flushes */
+} enter_stats;
+struct {
+ int calls;
+ int removes;
+ int pvfirst;
+ int pvsearch;
+ int ptinvalid;
+ int uflushes;
+ int sflushes;
+} remove_stats;
+
+int pmapdebug = 0 /* 0xffff */;
+#define PDB_FOLLOW 0x0001
+#define PDB_INIT 0x0002
+#define PDB_ENTER 0x0004
+#define PDB_REMOVE 0x0008
+#define PDB_CREATE 0x0010
+#define PDB_PTPAGE 0x0020
+#define PDB_CACHE 0x0040
+#define PDB_BITS 0x0080
+#define PDB_COLLECT 0x0100
+#define PDB_PROTECT 0x0200
+#define PDB_PDRTAB 0x0400
+#define PDB_PARANOIA 0x2000
+#define PDB_WIRING 0x4000
+#define PDB_PVDUMP 0x8000
+#endif
+
+vm_offset_t virtual_avail;
+vm_offset_t virtual_end;
+
+long equiv_end = 0;
+
+struct pmap kernel_pmap_store;
+pmap_t kernel_pmap;
+boolean_t pmap_initialized = FALSE;
+
+#ifdef USEALIGNMENT
+/*
+ * Mask to determine if two VAs naturally align in the data cache:
+ * 834/835: 0 (cannot do)
+ * 720/730/750: 256k
+ * 705/710: 64k
+ * 715: 64k
+ * Note that this is a mask only for the data cache since we are
+ * only attempting to prevent excess cache flushing of RW data.
+ */
+vm_offset_t pmap_alignmask;
+
+#define pmap_aligned(va1, va2) \
+ (((va1) & pmap_alignmask) == ((va2) & pmap_alignmask))
+#endif
+
+/*
+ * Hashed (Hardware) Page Table, for use as a software cache, or as a
+ * hardware accessible cache on machines with hardware TLB walkers.
+ */
+struct hpt_entry *hpt_table;
+u_int hpt_hashsize;
+
+TAILQ_HEAD(, pmap) pmap_freelist; /* list of free pmaps */
+u_int pmap_nfree;
+struct simplelock pmap_freelock; /* and lock */
+
+struct simplelock pmap_lock; /* XXX this is all broken */
+struct simplelock sid_pid_lock; /* pids */
+
+u_int pages_per_vm_page;
+u_int pid_counter;
+static const u_int kern_prot[8] = {
+ TLB_AR_NA, /* VM_PROT_NONE | VM_PROT_NONE | VM_PROT_NONE */
+ TLB_AR_KR, /* VM_PROT_READ | VM_PROT_NONE | VM_PROT_NONE */
+ TLB_AR_KRW, /* VM_PROT_NONE | VM_PROT_WRITE | VM_PROT_NONE */
+ TLB_AR_KRW, /* VM_PROT_READ | VM_PROT_WRITE | VM_PROT_NONE */
+ TLB_AR_KRX, /* VM_PROT_NONE | VM_PROT_NONE | VM_PROT_EXECUTE */
+ TLB_AR_KRX, /* VM_PROT_READ | VM_PROT_NONE | VM_PROT_EXECUTE */
+ TLB_AR_KRWX, /* VM_PROT_NONE | VM_PROT_WRITE | VM_PROT_EXECUTE */
+ TLB_AR_KRWX /* VM_PROT_READ | VM_PROT_WRITE | VM_PROT_EXECUTE */
+};
+static const u_int user_prot[8] = {
+ TLB_AR_NA, /* VM_PROT_NONE | VM_PROT_NONE | VM_PROT_NONE */
+ TLB_AR_UR, /* VM_PROT_READ | VM_PROT_NONE | VM_PROT_NONE */
+ TLB_AR_URW, /* VM_PROT_NONE | VM_PROT_WRITE | VM_PROT_NONE */
+ TLB_AR_URW, /* VM_PROT_READ | VM_PROT_WRITE | VM_PROT_NONE */
+ TLB_AR_URX, /* VM_PROT_NONE | VM_PROT_NONE | VM_PROT_EXECUTE */
+ TLB_AR_URX, /* VM_PROT_READ | VM_PROT_NONE | VM_PROT_EXECUTE */
+ TLB_AR_URWX, /* VM_PROT_NONE | VM_PROT_WRITE | VM_PROT_EXECUTE */
+ TLB_AR_URWX /* VM_PROT_READ | VM_PROT_WRITE | VM_PROT_EXECUTE */
+};
+
+TAILQ_HEAD(, pv_page) pv_page_freelist;
+u_int pv_nfree;
+
+/* pv_entry operation */
+static __inline struct pv_entry *pmap_alloc_pv __P((void));
+static __inline void pmap_free_pv __P((struct pv_entry *));
+static __inline struct pv_entry *pmap_find_pv __P((vm_offset_t));
+static __inline void pmap_enter_pv __P((pmap_t, vm_offset_t, u_int, u_int,
+ u_int, struct pv_entry *));
+static __inline void pmap_remove_pv __P((pmap_t, vm_offset_t,
+ struct pv_entry *));
+static __inline void pmap_clear_pv __P((vm_offset_t, struct pv_entry *));
+
+/* hpt_table operation */
+static __inline void pmap_enter_va __P((pa_space_t, vm_offset_t,
+ struct pv_entry *));
+static __inline struct pv_entry *pmap_find_va __P((pa_space_t, vm_offset_t));
+static __inline void pmap_remove_va __P((struct pv_entry *));
+static __inline void pmap_clear_va __P((pa_space_t, vm_offset_t));
+
+#define pmap_sid(pmap, va) \
+ (((va & 0xc0000000) != 0xc0000000)? pmap->pmap_space : HPPA_SID_KERNEL)
+
+#define pmap_prot(pmap, prot) \
+ (((pmap) == kernel_pmap ? kern_prot : user_prot)[prot])
+
+/*
+ * Clear the HPT table entry for the corresponding space/offset to reflect
+ * the fact that we have possibly changed the mapping, and need to pick
+ * up new values from the mapping structure on the next access.
+ */
+static __inline void
+pmap_clear_va(space, va)
+ pa_space_t space;
+ vm_offset_t va;
+{
+ register int hash = pmap_hash(space, va);
+
+ hpt_table[hash].hpt_valid = 0;
+ hpt_table[hash].hpt_space = -1;
+}
+
+static __inline void
+pmap_enter_va(space, va, pv)
+ pa_space_t space;
+ vm_offset_t va;
+ struct pv_entry *pv;
+{
+ register struct hpt_entry *hpt = &hpt_table[pmap_hash(space, va)];
+#ifdef DIAGNOSTIC
+ register struct pv_entry *pvp = hpt->hpt_entry;
+ va = btop(va);
+ while(pvp && pvp->pv_va != va && pvp->pv_space != space)
+ pvp = pvp->pv_hash;
+ if (pvp)
+ panic("pmap_enter_va: pv_entry is already in hpt_table");
+#endif
+ /* we assume that normally there are no duplicate entries
+ would be inserted (use DIAGNOSTIC should you want a proof) */
+ pv->pv_hash = hpt->hpt_entry;
+ hpt->hpt_entry = pv;
+}
+
+static __inline struct pv_entry *
+pmap_find_va(space, va)
+ pa_space_t space;
+ vm_offset_t va;
+{
+ register struct pv_entry *pvp =
+ hpt_table[pmap_hash(space, va)].hpt_entry;
+
+ va = btop(va);
+ while(pvp && pvp->pv_va != va && pvp->pv_space != space)
+ pvp = pvp->pv_hash;
+
+ return pvp;
+}
+
+static __inline void
+pmap_remove_va(pv)
+ struct pv_entry *pv;
+{
+ register struct hpt_entry *hpt =
+ &hpt_table[pmap_hash(pv->pv_space, ptob(pv->pv_va))];
+ register struct pv_entry **pvp = hpt->hpt_entry;
+
+ while(*pvp && *pvp != pv)
+ pvp = &(*pvp)->pv_hash;
+ if (*pvp) {
+ *pvp = (*pvp)->pv_hash;
+ pv->pv_hash = NULL;
+ if (pv->pv_va == hpt->hpt_vpn &&
+ pv->pv_space == hpt->hpt_space) {
+ hpt->hpt_space = -1;
+ hpt->hpt_valid = 0;
+ }
+ } else {
+#ifdef DIAGNOSTIC
+ printf("pmap_remove_va: entry not found\n");
+#endif
+ }
+}
+
+static __inline struct pv_entry *
+pmap_alloc_pv()
+{
+ struct pv_page *pvp;
+ struct pv_entry *pv;
+ int i;
+
+ if (pv_nfree == 0) {
+ MALLOC(pvp, struct pv_page *, NBPG, M_VMPVENT, M_WAITOK);
+
+ if (pvp == 0)
+ panic("pmap_alloc_pv: alloc failed");
+ pvp->pvp_freelist = pv = &pvp->pvp_pv[1];
+ for (i = NPVPPG - 2; i; i--, pv++)
+ pv->pv_next = pv + 1;
+ pv->pv_next = NULL;
+ pv_nfree += pvp->pvp_nfree = NPVPPG - 1;
+ TAILQ_INSERT_HEAD(&pv_page_freelist, pvp, pvp_list);
+ pv = &pvp->pvp_pv[0];
+ } else {
+ --pv_nfree;
+ pvp = pv_page_freelist.tqh_first;
+ if (--pvp->pvp_nfree == 0) {
+ TAILQ_REMOVE(&pv_page_freelist, pvp, pvp_list);
+ }
+ pv = pvp->pvp_freelist;
+#ifdef DIAGNOSTIC
+ if (pv == 0)
+ panic("pmap_alloc_pv: pgi_nfree inconsistent");
+#endif
+ pvp->pvp_freelist = pv->pv_next;
+ }
+ return pv;
+}
+
+static __inline void
+pmap_free_pv(pv)
+ struct pv_entry *pv;
+{
+ register struct pv_page *pvp;
+
+ pvp = (struct pv_page *) trunc_page(pv);
+ switch (++pvp->pvp_nfree) {
+ case 1:
+ TAILQ_INSERT_TAIL(&pv_page_freelist, pvp, pvp_list);
+ default:
+ pv->pv_next = pvp->pvp_freelist;
+ pvp->pvp_freelist = pv;
+ ++pv_nfree;
+ break;
+ case NPVPPG:
+ pv_nfree -= NPVPPG - 1;
+ TAILQ_REMOVE(&pv_page_freelist, pvp, pvp_list);
+ FREE((vm_offset_t) pvp, M_VMPVENT);
+ break;
+ }
+}
+
+#if 0
+void
+pmap_collect_pv()
+{
+ struct pv_page_list pv_page_collectlist;
+ struct pv_page *pvp, *npvp;
+ struct pv_entry *ph, *ppv, *pv, *npv;
+ int s;
+
+ TAILQ_INIT(&pv_page_collectlist);
+
+ for (pvp = pv_page_freelist.tqh_first; pvp; pvp = npvp) {
+ if (pv_nfree < NPVPPG)
+ break;
+ npvp = pvp->pvp_pgi.pgi_list.tqe_next;
+ if (pvp->pvp_pgi.pgi_nfree > NPVPPG / 3) {
+ TAILQ_REMOVE(&pv_page_freelist, pvp, pvp_pgi.pgi_list);
+ TAILQ_INSERT_TAIL(&pv_page_collectlist, pvp, pvp_pgi.pgi_list);
+ pv_nfree -= pvp->pvp_pgi.pgi_nfree;
+ pvp->pvp_pgi.pgi_nfree = -1;
+ }
+ }
+
+ if (pv_page_collectlist.tqh_first == 0)
+ return;
+
+ for (ph = &pv_table[npages - 1]; ph >= &pv_table[0]; ph--) {
+ if (ph->pv_pmap == 0)
+ continue;
+ s = splimp();
+ for (ppv = ph; (pv = ppv->pv_next) != 0; ) {
+ pvp = (struct pv_page *) trunc_page(pv);
+ if (pvp->pvp_pgi.pgi_nfree == -1) {
+ pvp = pv_page_freelist.tqh_first;
+ if (--pvp->pvp_pgi.pgi_nfree == 0) {
+ TAILQ_REMOVE(&pv_page_freelist, pvp, pvp_pgi.pgi_list);
+ }
+ npv = pvp->pvp_pgi.pgi_freelist;
+#ifdef DIAGNOSTIC
+ if (npv == 0)
+ panic("pmap_collect_pv: pgi_nfree inconsistent");
+#endif
+ pvp->pvp_pgi.pgi_freelist = npv->pv_next;
+ *npv = *pv;
+ ppv->pv_next = npv;
+ ppv = npv;
+ } else
+ ppv = pv;
+ }
+ splx(s);
+ }
+
+ for (pvp = pv_page_collectlist.tqh_first; pvp; pvp = npvp) {
+ npvp = pvp->pvp_pgi.pgi_list.tqe_next;
+ FREE((vm_offset_t) pvp, M_VMPVENT);
+ }
+}
+#endif
+
+static __inline void
+pmap_enter_pv(pmap, va, tlbprot, tlbpage, tlbsw, pv)
+ register pmap_t pmap;
+ vm_offset_t va;
+ u_int tlbprot, tlbpage, tlbsw;
+ register struct pv_entry *pv;
+{
+ register struct pv_entry *npv, *hpv;
+ int s;
+
+ if (!pmap_initialized)
+ return;
+
+ s = splimp();
+
+#ifdef DEBUG
+ if (pmapdebug & PDB_ENTER)
+ printf("pmap_enter_pv: pv %p: %lx/%p/%p\n",
+ pv, pv->pv_va, pv->pv_pmap, pv->pv_next);
+#endif
+
+ if (pv->pv_pmap == NULL) {
+ /*
+ * No entries yet, use header as the first entry
+ */
+#ifdef DEBUG
+ enter_stats.firstpv++;
+#endif
+ hpv = npv = NULL;
+ } else {
+ /*
+ * There is at least one other VA mapping this page.
+ * Place this entry after the header.
+ */
+#ifdef DEBUG
+ for (npv = pv; npv; npv = npv->pv_next)
+ if (pmap == npv->pv_pmap && va == npv->pv_va)
+ panic("pmap_enter_pv: already in pv_tab");
+#endif
+ hpv = pv;
+ npv = pv->pv_next;
+ pv = pmap_alloc_pv();
+#ifdef DEBUG
+ if (!npv->pv_next)
+ enter_stats.secondpv++;
+#endif
+ }
+ pv->pv_va = va;
+ pv->pv_pmap = pmap;
+ pv->pv_tlbprot = tlbprot;
+ pv->pv_tlbpage = tlbpage;
+ pv->pv_tlbsw = tlbsw;
+ pv->pv_next = npv;
+ if (hpv)
+ hpv->pv_next = pv;
+ pmap_enter_va(pmap->pmap_space, va, pv);
+
+ splx(s);
+}
+
+static __inline void
+pmap_remove_pv(pmap, va, pv)
+ register pmap_t pmap;
+ vm_offset_t va;
+ struct pv_entry *pv;
+{
+ register struct pv_entry *npv;
+ int s;
+
+ /*
+ * Remove from the PV table (raise IPL since we
+ * may be called at interrupt time).
+ */
+ s = splimp();
+
+ /*
+ * If it is the first entry on the list, it is actually
+ * in the header and we must copy the following entry up
+ * to the header. Otherwise we must search the list for
+ * the entry. In either case we free the now unused entry.
+ */
+ if (pmap == pv->pv_pmap && va == pv->pv_va) {
+ npv = pv->pv_next;
+ pmap_remove_va(pv);
+ if (npv) {
+ *pv = *npv;
+ pmap_free_pv(npv);
+ } else
+ pv->pv_pmap = NULL;
+ } else {
+ for (npv = pv->pv_next; npv; pv = npv, npv = npv->pv_next) {
+ if (pmap == npv->pv_pmap && va == npv->pv_va)
+ break;
+ }
+ if (npv) {
+ pv->pv_next = npv->pv_next;
+ pmap_remove_va(pv);
+ pmap_free_pv(npv);
+ }
+ }
+ splx(s);
+}
+
+/*
+ * Flush caches and TLB entries refering to physical page pa. If cmp is
+ * non-zero, we do not affect the cache or TLB entires for that mapping.
+ */
+static __inline void
+pmap_clear_pv(pa, cpv)
+ vm_offset_t pa;
+ struct pv_entry *cpv;
+{
+ register struct pv_entry *pv;
+ int s;
+
+ if (!(pv = pmap_find_pv(pa)))
+ return;
+
+ s = splimp();
+ for (; pv; pv = pv->pv_next) {
+ if (pv == cpv)
+ continue;
+ /*
+ * have to clear the icache first since fic uses the dtlb.
+ */
+ if (pv->pv_tlbsw & TLB_ICACHE)
+ ficache(pv->pv_space, pv->pv_va, NBPG);
+ pitlb(pv->pv_space, pv->pv_va);
+ if (pv->pv_tlbsw & TLB_DCACHE)
+ fdcache(pv->pv_space, pv->pv_va, NBPG);
+ pdtlb(pv->pv_space, pv->pv_va);
+
+ pv->pv_tlbsw &= ~(TLB_ICACHE|TLB_DCACHE);
+#ifdef USEALIGNMENT
+ pv->pv_tlbprot &= ~TLB_DIRTY;
+#endif
+ pmap_clear_va(pv->pv_space, pv->pv_va);
+ }
+ splx(s);
+}
+
+static __inline struct pv_entry *
+pmap_find_pv(pa)
+ vm_offset_t pa;
+{
+ register int bank;
+ int off;
+ if ((bank = vm_physseg_find(atop(pa), &off)) != -1) {
+ return &vm_physmem[bank].pmseg.pvent[off];
+ } else
+ return NULL;
+}
+
+/*
+ * Bootstrap the system enough to run with virtual memory.
+ * Map the kernel's code and data, and allocate the system page table.
+ * Called with mapping OFF.
+ *
+ * Parameters:
+ * vstart PA of first available physical page
+ * vend PA of last available physical page
+ */
+void
+pmap_bootstrap(vstart, vend)
+ vm_offset_t *vstart;
+ vm_offset_t *vend;
+{
+ vm_offset_t addr;
+ vm_size_t size;
+ int i;
+#ifdef PMAPDEBUG
+ vm_offset_t saddr;
+#endif
+#ifdef USEALIGNMENT
+ extern int dcache_size;
+#endif
+
+ vm_set_page_size();
+
+ pages_per_vm_page = PAGE_SIZE / NBPG;
+ /* XXX for now */
+ if (pages_per_vm_page != 1)
+ panic("HPPA page != MACH page");
+
+ /*
+ * Initialize kernel pmap
+ */
+ kernel_pmap = &kernel_pmap_store;
+#if NCPUS > 1
+ lock_init(&pmap_lock, FALSE, ETAP_VM_PMAP_SYS, ETAP_VM_PMAP_SYS_I);
+#endif /* NCPUS > 1 */
+ simple_lock_init(&kernel_pmap->pmap_lock);
+ simple_lock_init(&pmap_freelock);
+ simple_lock_init(&sid_pid_lock);
+
+ kernel_pmap->pmap_refcnt = 1;
+ kernel_pmap->pmap_space = HPPA_SID_KERNEL;
+ kernel_pmap->pmap_pid = HPPA_PID_KERNEL;
+
+#ifdef USEALIGNMENT
+ /* If we can take advantage of natural alignments in the cache
+ set that up now. */
+ pmap_alignmask = dcache_size - 1;
+#endif
+ /*
+ * Allocate various tables and structures.
+ */
+ addr = *vstart;
+
+ /* here will be a hole due to the HPT alignment XXX */
+
+ /* Allocate the HPT */
+ size = sizeof(struct hpt_entry) * hpt_hashsize;
+#ifdef PMAPDEBUG
+ saddr =
+#endif
+ addr = ((u_int)addr + size-1) & ~(size-1);
+ hpt_table = (struct hpt_entry *) addr;
+ for (i = 0; i < hpt_hashsize; i++) {
+ hpt_table[i].hpt_valid = 0;
+ hpt_table[i].hpt_vpn = 0;
+ hpt_table[i].hpt_space = -1;
+ hpt_table[i].hpt_tlbpage = 0;
+ hpt_table[i].hpt_tlbprot = 0;
+ hpt_table[i].hpt_entry = NULL;
+ }
+ addr = round_page((vm_offset_t)&hpt_table[hpt_hashsize]);
+#ifdef PMAPDEBUG
+ printf("hpt_table: 0x%x @ 0x%x\n", addr - saddr, saddr);
+#endif
+
+ /* Allocate the physical to virtual table. */
+#ifdef PMAPDEBUG
+ saddr =
+#endif
+ addr = cache_align(addr);
+ size = sizeof(struct pv_entry *) * atop(*vend - *vstart + 1);
+ addr = *vstart = cache_align(addr + size);
+ vm_page_physload(atop(*vstart), atop(*vend),
+ atop(*vstart), atop(*vend));
+ /* we have only one initial phys memory segment */
+ vm_physmem[0].pmseg.pvent = (struct pv_entry *) addr;
+
+#ifdef PMAPDEBUG
+ printf("pv_array: 0x%x @ 0x%x\n", addr - saddr, saddr);
+#endif
+ virtual_avail = *vstart;
+ virtual_end = *vend;
+
+ /* load cr25 with the address of the HPT table
+ NB: It sez CR_VTOP, but we (and the TLB handlers) know better ... */
+ mtctl(hpt_table, CR_VTOP);
+
+#ifdef BTLB
+ cpu_disable_sid_hashing();
+#endif
+
+ /* Load the kernel's PID */
+ mtctl(HPPA_PID_KERNEL, CR_PIDR1);
+}
+
+void
+pmap_virtual_space(startp, endp)
+ vm_offset_t *startp;
+ vm_offset_t *endp;
+{
+ *startp = virtual_avail;
+ *endp = virtual_end;
+}
+
+/*
+ * Finishes the initialization of the pmap module.
+ * This procedure is called from vm_mem_init() in vm/vm_init.c
+ * to initialize any remaining data structures that the pmap module
+ * needs to map virtual memory (VM is already ON).
+ */
+void
+pmap_init(void)
+{
+ TAILQ_INIT(&pmap_freelist);
+ pid_counter = HPPA_PID_KERNEL + 2;
+
+ pmap_initialized = TRUE;
+}
+
+/*
+ * Initialize a preallocated and zeroed pmap structure,
+ * such as one in a vmspace structure.
+ */
+void
+pmap_pinit(pmap)
+ pmap_t pmap;
+{
+ u_int pid;
+ if (!pmap->pmap_pid) {
+
+ /*
+ * Allocate space and protection IDs for the pmap.
+ * If all are allocated, there is nothing we can do.
+ */
+ simple_lock(&sid_pid_lock);
+ if (pid_counter <= MAX_PID) {
+ pid = pid_counter;
+ pid_counter += 2;
+ } else
+ pid = 0;
+ simple_unlock(&sid_pid_lock);
+
+ if (pid == 0)
+ printf("Warning no more pmap id\n");
+
+ /*
+ * Initialize the sids and pid. This is a user pmap, so
+ * sr4, sr5, and sr6 are identical. sr7 is always KERNEL_SID.
+ */
+ pmap->pmap_pid = pid;
+ simple_lock_init(&pmap->pmap_lock);
+ }
+
+ simple_lock(&pmap->pmap_lock);
+ pmap->pmap_space = pmap->pmap_pid;
+ pmap->pmap_refcnt = 1;
+ pmap->pmap_stats.resident_count = 0;
+ pmap->pmap_stats.wired_count = 0;
+ simple_unlock(&pmap->pmap_lock);
+}
+
+/*
+ * pmap_create(size)
+ *
+ * Create and return a physical map.
+ *
+ * If the size specified for the map is zero, the map is an actual physical
+ * map, and may be referenced by the hardware.
+ *
+ * If the size specified is non-zero, the map will be used in software
+ * only, and is bounded by that size.
+ */
+pmap_t
+pmap_create(size)
+ vm_size_t size;
+{
+ register pmap_t pmap;
+
+ /*
+ * A software use-only map doesn't even need a pmap structure.
+ */
+ if (size)
+ return(NULL);
+
+ /*
+ * If there is a pmap in the pmap free list, reuse it.
+ */
+ simple_lock(&pmap_freelock);
+ if (pmap_nfree) {
+ pmap = pmap_freelist.tqh_first;
+ TAILQ_REMOVE(&pmap_freelist, pmap, pmap_list);
+ pmap_nfree--;
+ simple_unlock(&pmap_freelock);
+ } else {
+ MALLOC(pmap, struct pmap *, sizeof(*pmap), M_VMMAP, M_NOWAIT);
+ if (pmap == NULL)
+ return NULL;
+ bzero(pmap, sizeof(*pmap));
+ }
+
+ pmap_pinit(pmap);
+
+ return(pmap);
+}
+
+/*
+ * pmap_destroy(pmap)
+ * Gives up a reference to the specified pmap. When the reference count
+ * reaches zero the pmap structure is added to the pmap free list.
+ * Should only be called if the map contains no valid mappings.
+ */
+void
+pmap_destroy(pmap)
+ pmap_t pmap;
+{
+ int ref_count;
+
+ if (pmap == NULL)
+ return;
+
+ simple_lock(&pmap->pmap_lock);
+
+ ref_count = --pmap->pmap_refcnt;
+
+ if (ref_count < 0)
+ panic("pmap_destroy(): ref_count < 0");
+ if (!ref_count) {
+ assert(pmap->pmap_stats.resident_count == 0);
+
+ simple_unlock(&pmap->pmap_lock);
+
+ /*
+ * Add the pmap to the pmap free list
+ * We cannot free() disposed pmaps because of
+ * PID shortage of 2^16
+ */
+ simple_lock(&pmap_freelock);
+ TAILQ_INSERT_HEAD(&pmap_freelist, pmap, pmap_list);
+ pmap_nfree++;
+ simple_unlock(&pmap_freelock);
+ }
+}
+
+/*
+ * pmap_enter(pmap, va, pa, prot, wired)
+ * Create a translation for the virtual address (va) to the physical
+ * address (pa) in the pmap with the protection requested. If the
+ * translation is wired then we can not allow a page fault to occur
+ * for this mapping.
+ */
+void
+pmap_enter(pmap, va, pa, prot, wired)
+ pmap_t pmap;
+ vm_offset_t va;
+ vm_offset_t pa;
+ vm_prot_t prot;
+ boolean_t wired;
+{
+ register struct pv_entry *pv, *ppv;
+ u_int tlbpage = btop(pa), tlbprot;
+ pa_space_t space;
+ boolean_t waswired;
+
+ if (pmap == NULL)
+ return;
+
+ simple_lock(&pmap->pmap_lock);
+
+ space = pmap_sid(pmap, va);
+ pv = pmap_find_pv(pa);
+
+ tlbprot = pmap_prot(pmap, prot) | pmap->pmap_pid;
+ if (!(ppv = pmap_find_va(space, va))) {
+ /*
+ * Mapping for this virtual address doesn't exist.
+ * Enter a new mapping.
+ */
+ pmap_enter_pv(pmap, va, tlbprot, tlbpage, 0, pv);
+ pmap->pmap_stats.resident_count++;
+ }
+ else {
+ /*
+ * We are just changing the protection.
+ * Flush the current TLB entry to force a fault and reload.
+ */
+ pv = ppv;
+ pdtlb(space, va);
+ pitlb(space, va);
+ }
+
+ /*
+ * Determine the protection information for this mapping.
+ */
+ tlbprot |= (pv->pv_tlbprot & ~(TLB_AR_MASK|TLB_PID_MASK));
+
+ /*
+ * Add in software bits and adjust statistics
+ */
+ waswired = tlbprot & TLB_WIRED;
+ if (wired && !waswired) {
+ tlbprot |= TLB_WIRED;
+ pmap->pmap_stats.wired_count++;
+ } else if (!wired && waswired) {
+ tlbprot &= ~TLB_WIRED;
+ pmap->pmap_stats.wired_count--;
+ }
+ pv->pv_tlbprot = tlbprot;
+
+ pmap_clear_va(space, pv->pv_va);
+ simple_unlock(&pmap->pmap_lock);
+}
+
+/*
+ * pmap_remove(pmap, sva, eva)
+ * unmaps all virtual addresses v in the virtual address
+ * range determined by [sva, eva) and pmap.
+ * sva and eva must be on machine independent page boundaries and
+ * sva must be less than or equal to eva.
+ */
+void
+pmap_remove(pmap, sva, eva)
+ register pmap_t pmap;
+ register vm_offset_t sva;
+ register vm_offset_t eva;
+{
+ register struct pv_entry *pv;
+ register pa_space_t space;
+
+ if(pmap == NULL)
+ return;
+
+ simple_lock(&pmap->pmap_lock);
+
+ space = pmap_sid(pmap, sva);
+
+ while (pmap->pmap_stats.resident_count && ((sva < eva))) {
+ if ((pv = pmap_find_va(space, sva))) {
+ pmap_remove_pv(pmap, sva, pv);
+ pmap->pmap_stats.resident_count--;
+ }
+ sva += PAGE_SIZE;
+ }
+
+ simple_unlock(&pmap->pmap_lock);
+}
+
+/*
+ * Used to map a range of physical addresses into kernel
+ * virtual address space.
+ *
+ * For now, VM is already on, we only need to map the
+ * specified memory.
+ */
+vm_offset_t
+pmap_map(va, spa, epa, prot)
+ vm_offset_t va;
+ vm_offset_t spa;
+ vm_offset_t epa;
+ vm_prot_t prot;
+{
+
+#ifdef DEBUG
+ if (pmapdebug & PDB_FOLLOW)
+ printf("pmap_map(%x, %x, %x, %x)\n", va, spa, epa, prot);
+#endif
+
+ while (spa < epa) {
+ pmap_enter(pmap_kernel(), va, spa, prot, FALSE);
+ va += NBPG;
+ spa += NBPG;
+ }
+ return va;
+}
+
+#ifdef BTLB
+/*
+ * pmap_block_map(pa, size, prot, entry, dtlb)
+ * Block map a physical region. Size must be a power of 2. Address must
+ * must be aligned to the size. Entry is the block TLB entry to use.
+ *
+ * S-CHIP: Entries 0,2 and 1,3 must be the same size for each TLB type.
+ */
+void
+pmap_block_map(pa, size, prot, entry, tlbflag)
+ vm_offset_t pa;
+ vm_size_t size;
+ vm_prot_t prot;
+ int entry, tlbflag;
+{
+ u_int tlbprot;
+
+ tlbprot = pmap_prot(kernel_pmap, prot) | kernel_pmap->pmap_pid;
+
+ switch (tlbflag) {
+ case BLK_ICACHE:
+ insert_block_itlb(entry, pa, size, tlbprot);
+ break;
+ case BLK_DCACHE:
+ insert_block_dtlb(entry, pa, size, tlbprot);
+ break;
+ case BLK_COMBINED:
+ insert_block_ctlb(entry, pa, size, tlbprot);
+ break;
+ case BLK_LCOMBINED:
+ insert_L_block_ctlb(entry, pa, size, tlbprot);
+ break;
+ default:
+ printf("pmap_block_map routine: unknown flag %d\n",tlbflag);
+ }
+}
+#endif
+
+/*
+ * pmap_page_protect(pa, prot)
+ *
+ * Lower the permission for all mappings to a given page.
+ */
+void
+pmap_page_protect(pa, prot)
+ vm_offset_t pa;
+ vm_prot_t prot;
+{
+ register struct pv_entry *pv;
+ register pmap_t pmap;
+ u_int tlbprot;
+ int s;
+
+ switch (prot) {
+ case VM_PROT_ALL:
+ return;
+ case VM_PROT_READ:
+ case VM_PROT_READ|VM_PROT_EXECUTE:
+ s = splimp();
+ for (pv = pmap_find_pv(pa); pv; pv = pv->pv_next) {
+ /*
+ * Compare new protection with old to see if
+ * anything needs to be changed.
+ */
+ tlbprot = pmap_prot(pv->pv_pmap, prot);
+
+ if ((pv->pv_tlbprot & TLB_AR_MASK) != tlbprot) {
+ pv->pv_tlbprot &= ~TLB_AR_MASK;
+ pv->pv_tlbprot |= tlbprot;
+
+ pmap_clear_va(pv->pv_space, pv->pv_va);
+
+ /*
+ * Purge the current TLB entry (if any)
+ * to force a fault and reload with the
+ * new protection.
+ */
+ pdtlb(pv->pv_space, pv->pv_va);
+ pitlb(pv->pv_space, pv->pv_va);
+ }
+ }
+ splx(s);
+ break;
+ default:
+ s = splimp();
+ while ((pv = pmap_find_pv(pa)) && pv->pv_pmap) {
+ pmap = pv->pv_pmap;
+ simple_lock(&pmap->pmap_lock);
+ pmap_remove_pv(pmap, pv->pv_va, pv);
+ pmap->pmap_stats.resident_count--;
+ simple_unlock(&pmap->pmap_lock);
+ }
+ splx(s);
+ break;
+ }
+}
+
+/*
+ * pmap_protect(pmap, s, e, prot)
+ * changes the protection on all virtual addresses v in the
+ * virtual address range determined by [s, e) and pmap to prot.
+ * s and e must be on machine independent page boundaries and
+ * s must be less than or equal to e.
+ */
+void
+pmap_protect(pmap, sva, eva, prot)
+ pmap_t pmap;
+ vm_offset_t sva;
+ vm_offset_t eva;
+ vm_prot_t prot;
+{
+ register struct pv_entry *pv;
+ u_int tlbprot;
+ pa_space_t space;
+
+ if (pmap == NULL)
+ return;
+
+ if (prot == VM_PROT_NONE) {
+ pmap_remove(pmap, sva, eva);
+ return;
+ }
+ if (prot & VM_PROT_WRITE)
+ return;
+
+ simple_lock(&pmap->pmap_lock);
+
+ space = pmap_sid(pmap, sva);
+
+ for(; sva < eva; sva += PAGE_SIZE) {
+ if((pv = pmap_find_va(space, sva))) {
+ /*
+ * Determine if mapping is changing.
+ * If not, nothing to do.
+ */
+ tlbprot = pmap_prot(pmap, prot);
+ if ((pv->pv_tlbprot & TLB_AR_MASK) == tlbprot)
+ continue;
+
+ pv->pv_tlbprot &= ~TLB_AR_MASK;
+ pv->pv_tlbprot |= tlbprot;
+
+ pmap_clear_va(space, pv->pv_va);
+
+ /*
+ * Purge the current TLB entry (if any) to force
+ * a fault and reload with the new protection.
+ */
+ pdtlb(space, pv->pv_va);
+ pitlb(space, pv->pv_va);
+ }
+ }
+ simple_unlock(&pmap->pmap_lock);
+}
+
+/*
+ * Routine: pmap_change_wiring
+ * Function: Change the wiring attribute for a map/virtual-address
+ * pair.
+ * In/out conditions:
+ * The mapping must already exist in the pmap.
+ *
+ * Change the wiring for a given virtual page. This routine currently is
+ * only used to unwire pages and hence the mapping entry will exist.
+ */
+void
+pmap_change_wiring(pmap, va, wired)
+ register pmap_t pmap;
+ vm_offset_t va;
+ boolean_t wired;
+{
+ register struct pv_entry *pv;
+ boolean_t waswired;
+
+ if (pmap == NULL)
+ return;
+
+ simple_lock(&pmap->pmap_lock);
+
+ if ((pv = pmap_find_va(pmap_sid(pmap, va), va)) == NULL)
+ panic("pmap_change_wiring: can't find mapping entry");
+
+ waswired = pv->pv_tlbprot & TLB_WIRED;
+ if (wired && !waswired) {
+ pv->pv_tlbprot |= TLB_WIRED;
+ pmap->pmap_stats.wired_count++;
+ } else if (!wired && waswired) {
+ pv->pv_tlbprot &= ~TLB_WIRED;
+ pmap->pmap_stats.wired_count--;
+ }
+ simple_unlock(&pmap->pmap_lock);
+}
+
+/*
+ * pmap_extract(pmap, va)
+ * returns the physical address corrsponding to the
+ * virtual address specified by pmap and va if the
+ * virtual address is mapped and 0 if it is not.
+ */
+vm_offset_t
+pmap_extract(pmap, va)
+ pmap_t pmap;
+ vm_offset_t va;
+{
+ register struct pv_entry *pv;
+
+ if (!(pv = pmap_find_va(pmap_sid(pmap, va), va)))
+ return(0);
+ else
+ return ptob(pv->pv_tlbpage) + (va & PGOFSET);
+}
+
+/*
+ * pmap_zero_page(pa)
+ *
+ * Zeros the specified page.
+ */
+void
+pmap_zero_page(pa)
+ register vm_offset_t pa;
+{
+ extern int dcache_line_mask;
+ register int psw;
+ register vm_offset_t pe = pa + PAGE_SIZE;
+
+ pmap_clear_pv(pa, NULL);
+
+ rsm(PSW_I,psw);
+ while (pa < pe) {
+ __asm volatile("stwas,ma %%r0,4(%0)\n\t"
+ : "=r" (pa):: "memory");
+
+ if (!(pa & dcache_line_mask))
+ __asm volatile("rsm %1, %%r0\n\t"
+ "fdc %2(%0)\n\t"
+ "ssm %1, %%r0"
+ : "=r" (pa): "i" (PSW_D), "r" (-4)
+ : "memory");
+ }
+
+ sync_caches();
+ mtsm(psw);
+}
+
+/*
+ * pmap_copy_page(src, dst)
+ *
+ * pmap_copy_page copies the src page to the destination page. If a mapping
+ * can be found for the source, we use that virtual address. Otherwise, a
+ * slower physical page copy must be done. The destination is always a
+ * physical address sivnce there is usually no mapping for it.
+ */
+void
+pmap_copy_page(spa, dpa)
+ vm_offset_t spa;
+ vm_offset_t dpa;
+{
+ extern int dcache_line_mask;
+ register int psw;
+ register vm_offset_t spe = spa + PAGE_SIZE;
+
+ pmap_clear_pv(spa, NULL);
+ pmap_clear_pv(dpa, NULL);
+
+ rsm(PSW_I,psw);
+
+ while (spa < spe) {
+ __asm volatile("ldwas,ma 4(%0),%%r23\n\t"
+ "stwas,ma %%r23,4(%1)\n\t"
+ : "=r" (spa), "=r" (dpa):: "r23", "memory");
+
+ if (!(spa & dcache_line_mask))
+ __asm volatile("rsm %2, %%r0\n\t"
+ "pdc %3(%0)\n\t"
+ "fdc %3(%1)\n\t"
+ "ssm %2, %%r0"
+ : "=r" (spa), "=r" (dpa)
+ : "i" (PSW_D), "r" (-4)
+ : "memory");
+ }
+
+ sync_caches();
+ mtsm(psw);
+}
+
+/*
+ * pmap_clear_modify(pa)
+ * clears the hardware modified ("dirty") bit for one
+ * machine independant page starting at the given
+ * physical address. phys must be aligned on a machine
+ * independant page boundary.
+ */
+void
+pmap_clear_modify(pa)
+ vm_offset_t pa;
+{
+ register struct pv_entry *pv;
+
+ if ((pv = pmap_find_pv(pa)))
+ pv->pv_tlbprot &= ~TLB_DIRTY;
+}
+
+/*
+ * pmap_is_modified(pa)
+ * returns TRUE iff the given physical page has been modified
+ * since the last call to pmap_clear_modify().
+ */
+boolean_t
+pmap_is_modified(pa)
+ vm_offset_t pa;
+{
+ register struct pv_entry *pv = pmap_find_pv(pa);
+ return pv != NULL && (pv->pv_tlbprot & TLB_DIRTY);
+}
+
+/*
+ * pmap_clear_reference(pa)
+ * clears the hardware referenced bit in the given machine
+ * independant physical page.
+ *
+ * Currently, we treat a TLB miss as a reference; i.e. to clear
+ * the reference bit we flush all mappings for pa from the TLBs.
+ */
+void
+pmap_clear_reference(pa)
+ vm_offset_t pa;
+{
+ register struct pv_entry *pv;
+ int s;
+
+ s = splimp();
+ for (pv = pmap_find_pv(pa); pv; pv = pv->pv_next) {
+ pitlb(pv->pv_space, pv->pv_va);
+ pdtlb(pv->pv_space, pv->pv_va);
+ pv->pv_tlbprot &= ~(TLB_REF);
+
+ pmap_clear_va(pv->pv_space, pv->pv_va);
+ }
+ pv->pv_tlbprot &= ~TLB_REF;
+ splx(s);
+}
+
+/*
+ * pmap_is_referenced(pa)
+ * returns TRUE if the given physical page has been referenced
+ * since the last call to pmap_clear_reference().
+ */
+boolean_t
+pmap_is_referenced(pa)
+ vm_offset_t pa;
+{
+ register struct pv_entry *pv;
+ int s;
+
+ s = splvm();
+ for (pv = pmap_find_pv(pa); pv && !(pv->pv_tlbprot & TLB_REF);)
+ pv = pv->pv_next;
+ splx(s);
+
+ return pv != NULL;
+}
+