2008-03-03 17:17:00 +00:00
|
|
|
/*-
|
2009-05-21 11:43:37 +00:00
|
|
|
* Copyright (C) 2007-2009 Semihalf, Rafal Jaworowski <raj@semihalf.com>
|
2008-03-03 17:17:00 +00:00
|
|
|
* Copyright (C) 2006 Semihalf, Marian Balakowicz <m8@semihalf.com>
|
|
|
|
* 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 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.
|
|
|
|
*
|
|
|
|
* Some hw specific parts of this pmap were derived or influenced
|
|
|
|
* by NetBSD's ibm4xx pmap module. More generic code is shared with
|
|
|
|
* a few other pmap modules from the FreeBSD tree.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* VM layout notes:
|
|
|
|
*
|
|
|
|
* Kernel and user threads run within one common virtual address space
|
|
|
|
* defined by AS=0.
|
|
|
|
*
|
|
|
|
* Virtual address space layout:
|
|
|
|
* -----------------------------
|
2009-01-13 16:15:49 +00:00
|
|
|
* 0x0000_0000 - 0xafff_ffff : user process
|
|
|
|
* 0xb000_0000 - 0xbfff_ffff : pmap_mapdev()-ed area (PCI/PCIE etc.)
|
|
|
|
* 0xc000_0000 - 0xc0ff_ffff : kernel reserved
|
2009-04-04 22:01:43 +00:00
|
|
|
* 0xc000_0000 - data_end : kernel code+data, env, metadata etc.
|
2009-01-13 16:15:49 +00:00
|
|
|
* 0xc100_0000 - 0xfeef_ffff : KVA
|
|
|
|
* 0xc100_0000 - 0xc100_3fff : reserved for page zero/copy
|
|
|
|
* 0xc100_4000 - 0xc200_3fff : reserved for ptbl bufs
|
|
|
|
* 0xc200_4000 - 0xc200_8fff : guard page + kstack0
|
|
|
|
* 0xc200_9000 - 0xfeef_ffff : actual free KVA space
|
|
|
|
* 0xfef0_0000 - 0xffff_ffff : I/O devices region
|
2008-03-03 17:17:00 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sys/cdefs.h>
|
|
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
|
2015-07-16 10:46:52 +00:00
|
|
|
#include "opt_kstack_pages.h"
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
#include <sys/param.h>
|
2015-01-07 01:01:39 +00:00
|
|
|
#include <sys/conf.h>
|
2008-03-03 17:17:00 +00:00
|
|
|
#include <sys/malloc.h>
|
2009-01-13 15:41:58 +00:00
|
|
|
#include <sys/ktr.h>
|
2008-03-03 17:17:00 +00:00
|
|
|
#include <sys/proc.h>
|
|
|
|
#include <sys/user.h>
|
|
|
|
#include <sys/queue.h>
|
|
|
|
#include <sys/systm.h>
|
|
|
|
#include <sys/kernel.h>
|
2015-01-07 01:01:39 +00:00
|
|
|
#include <sys/kerneldump.h>
|
2011-08-02 15:35:43 +00:00
|
|
|
#include <sys/linker.h>
|
2008-03-03 17:17:00 +00:00
|
|
|
#include <sys/msgbuf.h>
|
|
|
|
#include <sys/lock.h>
|
|
|
|
#include <sys/mutex.h>
|
2012-11-03 23:22:49 +00:00
|
|
|
#include <sys/rwlock.h>
|
2011-05-17 22:03:01 +00:00
|
|
|
#include <sys/sched.h>
|
2009-05-21 11:43:37 +00:00
|
|
|
#include <sys/smp.h>
|
2008-03-03 17:17:00 +00:00
|
|
|
#include <sys/vmmeter.h>
|
|
|
|
|
|
|
|
#include <vm/vm.h>
|
|
|
|
#include <vm/vm_page.h>
|
|
|
|
#include <vm/vm_kern.h>
|
|
|
|
#include <vm/vm_pageout.h>
|
|
|
|
#include <vm/vm_extern.h>
|
|
|
|
#include <vm/vm_object.h>
|
|
|
|
#include <vm/vm_param.h>
|
|
|
|
#include <vm/vm_map.h>
|
|
|
|
#include <vm/vm_pager.h>
|
|
|
|
#include <vm/uma.h>
|
|
|
|
|
|
|
|
#include <machine/cpu.h>
|
|
|
|
#include <machine/pcb.h>
|
2009-05-14 00:34:26 +00:00
|
|
|
#include <machine/platform.h>
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
#include <machine/tlb.h>
|
|
|
|
#include <machine/spr.h>
|
|
|
|
#include <machine/md_var.h>
|
|
|
|
#include <machine/mmuvar.h>
|
|
|
|
#include <machine/pmap.h>
|
|
|
|
#include <machine/pte.h>
|
|
|
|
|
|
|
|
#include "mmu_if.h"
|
|
|
|
|
2016-04-10 15:48:09 +00:00
|
|
|
#define SPARSE_MAPDEV
|
2008-03-03 17:17:00 +00:00
|
|
|
#ifdef DEBUG
|
|
|
|
#define debugf(fmt, args...) printf(fmt, ##args)
|
|
|
|
#else
|
|
|
|
#define debugf(fmt, args...)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define TODO panic("%s: not implemented", __func__);
|
|
|
|
|
2009-04-04 22:01:43 +00:00
|
|
|
extern unsigned char _etext[];
|
|
|
|
extern unsigned char _end[];
|
|
|
|
|
2011-08-02 15:35:43 +00:00
|
|
|
extern uint32_t *bootinfo;
|
|
|
|
|
|
|
|
vm_paddr_t kernload;
|
2009-04-04 22:01:43 +00:00
|
|
|
vm_offset_t kernstart;
|
|
|
|
vm_size_t kernsize;
|
|
|
|
|
|
|
|
/* Message buffer and tables. */
|
|
|
|
static vm_offset_t data_start;
|
|
|
|
static vm_size_t data_end;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-05-14 00:34:26 +00:00
|
|
|
/* Phys/avail memory regions. */
|
|
|
|
static struct mem_region *availmem_regions;
|
|
|
|
static int availmem_regions_sz;
|
|
|
|
static struct mem_region *physmem_regions;
|
|
|
|
static int physmem_regions_sz;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Reserved KVA space and mutex for mmu_booke_zero_page. */
|
|
|
|
static vm_offset_t zero_page_va;
|
|
|
|
static struct mtx zero_page_mutex;
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
static struct mtx tlbivax_mutex;
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/*
|
|
|
|
* Reserved KVA space for mmu_booke_zero_page_idle. This is used
|
|
|
|
* by idle thred only, no lock required.
|
|
|
|
*/
|
|
|
|
static vm_offset_t zero_page_idle_va;
|
|
|
|
|
|
|
|
/* Reserved KVA space and mutex for mmu_booke_copy_page. */
|
|
|
|
static vm_offset_t copy_page_src_va;
|
|
|
|
static vm_offset_t copy_page_dst_va;
|
|
|
|
static struct mtx copy_page_mutex;
|
|
|
|
|
|
|
|
/**************************************************************************/
|
|
|
|
/* PMAP */
|
|
|
|
/**************************************************************************/
|
|
|
|
|
2014-08-08 17:12:03 +00:00
|
|
|
static int mmu_booke_enter_locked(mmu_t, pmap_t, vm_offset_t, vm_page_t,
|
|
|
|
vm_prot_t, u_int flags, int8_t psind);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
unsigned int kptbl_min; /* Index of the first kernel ptbl. */
|
|
|
|
unsigned int kernel_ptbls; /* Number of KVA ptbls. */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If user pmap is processed with mmu_booke_remove and the resident count
|
|
|
|
* drops to 0, there are no more pages to remove, so we need not continue.
|
|
|
|
*/
|
|
|
|
#define PMAP_REMOVE_DONE(pmap) \
|
|
|
|
((pmap) != kernel_pmap && (pmap)->pm_stats.resident_count == 0)
|
|
|
|
|
2015-03-01 21:47:38 +00:00
|
|
|
extern int elf32_nxstack;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/**************************************************************************/
|
|
|
|
/* TLB and TID handling */
|
|
|
|
/**************************************************************************/
|
|
|
|
|
|
|
|
/* Translation ID busy table */
|
2009-01-13 15:41:58 +00:00
|
|
|
static volatile pmap_t tidbusy[MAXCPU][TID_MAX + 1];
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/*
|
2009-01-13 15:41:58 +00:00
|
|
|
* TLB0 capabilities (entry, way numbers etc.). These can vary between e500
|
|
|
|
* core revisions and should be read from h/w registers during early config.
|
2008-03-03 17:17:00 +00:00
|
|
|
*/
|
2009-01-13 15:41:58 +00:00
|
|
|
uint32_t tlb0_entries;
|
|
|
|
uint32_t tlb0_ways;
|
|
|
|
uint32_t tlb0_entries_per_way;
|
2015-08-28 03:03:09 +00:00
|
|
|
uint32_t tlb1_entries;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
#define TLB0_ENTRIES (tlb0_entries)
|
|
|
|
#define TLB0_WAYS (tlb0_ways)
|
|
|
|
#define TLB0_ENTRIES_PER_WAY (tlb0_entries_per_way)
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2015-08-28 03:03:09 +00:00
|
|
|
#define TLB1_ENTRIES (tlb1_entries)
|
|
|
|
#define TLB1_MAXENTRIES 64
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2016-04-10 15:50:45 +00:00
|
|
|
static vm_offset_t tlb1_map_base = VM_MAXUSER_ADDRESS + PAGE_SIZE;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
static tlbtid_t tid_alloc(struct pmap *);
|
2015-12-30 02:23:14 +00:00
|
|
|
static void tid_flush(tlbtid_t tid);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
static void tlb_print_entry(int, uint32_t, uint32_t, uint32_t, uint32_t);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2016-04-19 01:48:18 +00:00
|
|
|
static void tlb1_read_entry(tlb_entry_t *, unsigned int);
|
|
|
|
static void tlb1_write_entry(tlb_entry_t *, unsigned int);
|
2008-03-03 17:17:00 +00:00
|
|
|
static int tlb1_iomapped(int, vm_paddr_t, vm_size_t, vm_offset_t *);
|
2011-08-02 15:35:43 +00:00
|
|
|
static vm_size_t tlb1_mapin_region(vm_offset_t, vm_paddr_t, vm_size_t);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
static vm_size_t tsize2size(unsigned int);
|
|
|
|
static unsigned int size2tsize(vm_size_t);
|
|
|
|
static unsigned int ilog2(unsigned int);
|
|
|
|
|
|
|
|
static void set_mas4_defaults(void);
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
static inline void tlb0_flush_entry(vm_offset_t);
|
2008-03-03 17:17:00 +00:00
|
|
|
static inline unsigned int tlb0_tableidx(vm_offset_t, unsigned int);
|
|
|
|
|
|
|
|
/**************************************************************************/
|
|
|
|
/* Page table management */
|
|
|
|
/**************************************************************************/
|
|
|
|
|
2012-11-03 23:22:49 +00:00
|
|
|
static struct rwlock_padalign pvh_global_lock;
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/* Data for the pv entry allocation mechanism */
|
|
|
|
static uma_zone_t pvzone;
|
|
|
|
static int pv_entry_count = 0, pv_entry_max = 0, pv_entry_high_water = 0;
|
|
|
|
|
|
|
|
#define PV_ENTRY_ZONE_MIN 2048 /* min pv entries in uma zone */
|
|
|
|
|
|
|
|
#ifndef PMAP_SHPGPERPROC
|
|
|
|
#define PMAP_SHPGPERPROC 200
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static void ptbl_init(void);
|
|
|
|
static struct ptbl_buf *ptbl_buf_alloc(void);
|
|
|
|
static void ptbl_buf_free(struct ptbl_buf *);
|
|
|
|
static void ptbl_free_pmap_ptbl(pmap_t, pte_t *);
|
|
|
|
|
2014-08-08 17:12:03 +00:00
|
|
|
static pte_t *ptbl_alloc(mmu_t, pmap_t, unsigned int, boolean_t);
|
2008-03-03 17:17:00 +00:00
|
|
|
static void ptbl_free(mmu_t, pmap_t, unsigned int);
|
|
|
|
static void ptbl_hold(mmu_t, pmap_t, unsigned int);
|
|
|
|
static int ptbl_unhold(mmu_t, pmap_t, unsigned int);
|
|
|
|
|
|
|
|
static vm_paddr_t pte_vatopa(mmu_t, pmap_t, vm_offset_t);
|
|
|
|
static pte_t *pte_find(mmu_t, pmap_t, vm_offset_t);
|
2014-08-08 17:12:03 +00:00
|
|
|
static int pte_enter(mmu_t, pmap_t, vm_page_t, vm_offset_t, uint32_t, boolean_t);
|
2009-01-13 15:41:58 +00:00
|
|
|
static int pte_remove(mmu_t, pmap_t, vm_offset_t, uint8_t);
|
2016-01-19 03:07:25 +00:00
|
|
|
static void kernel_pte_alloc(vm_offset_t data_end, vm_offset_t addr,
|
|
|
|
vm_offset_t pdir);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
static pv_entry_t pv_alloc(void);
|
2008-03-03 17:17:00 +00:00
|
|
|
static void pv_free(pv_entry_t);
|
|
|
|
static void pv_insert(pmap_t, vm_offset_t, vm_page_t);
|
|
|
|
static void pv_remove(pmap_t, vm_offset_t, vm_page_t);
|
|
|
|
|
2015-08-04 19:46:13 +00:00
|
|
|
static void booke_pmap_init_qpages(void);
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/* Number of kva ptbl buffers, each covering one ptbl (PTBL_PAGES). */
|
|
|
|
#define PTBL_BUFS (128 * 16)
|
|
|
|
|
|
|
|
struct ptbl_buf {
|
|
|
|
TAILQ_ENTRY(ptbl_buf) link; /* list link */
|
|
|
|
vm_offset_t kva; /* va of mapping */
|
|
|
|
};
|
|
|
|
|
|
|
|
/* ptbl free list and a lock used for access synchronization. */
|
|
|
|
static TAILQ_HEAD(, ptbl_buf) ptbl_buf_freelist;
|
|
|
|
static struct mtx ptbl_buf_freelist_lock;
|
|
|
|
|
|
|
|
/* Base address of kva space allocated fot ptbl bufs. */
|
|
|
|
static vm_offset_t ptbl_buf_pool_vabase;
|
|
|
|
|
|
|
|
/* Pointer to ptbl_buf structures. */
|
|
|
|
static struct ptbl_buf *ptbl_bufs;
|
|
|
|
|
2015-03-28 21:39:42 +00:00
|
|
|
#ifdef SMP
|
2016-04-19 01:48:18 +00:00
|
|
|
extern tlb_entry_t __boot_tlb1[];
|
2009-05-21 11:43:37 +00:00
|
|
|
void pmap_bootstrap_ap(volatile uint32_t *);
|
2015-03-28 21:39:42 +00:00
|
|
|
#endif
|
2009-05-21 11:43:37 +00:00
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/*
|
|
|
|
* Kernel MMU interface
|
|
|
|
*/
|
|
|
|
static void mmu_booke_clear_modify(mmu_t, vm_page_t);
|
2009-06-13 08:57:04 +00:00
|
|
|
static void mmu_booke_copy(mmu_t, pmap_t, pmap_t, vm_offset_t,
|
|
|
|
vm_size_t, vm_offset_t);
|
2008-03-03 17:17:00 +00:00
|
|
|
static void mmu_booke_copy_page(mmu_t, vm_page_t, vm_page_t);
|
Add pmap function pmap_copy_pages(), which copies the content of the
pages around, taking array of vm_page_t both for source and
destination. Starting offsets and total transfer size are specified.
The function implements optimal algorithm for copying using the
platform-specific optimizations. For instance, on the architectures
were the direct map is available, no transient mappings are created,
for i386 the per-cpu ephemeral page frame is used. The code was
typically borrowed from the pmap_copy_page() for the same
architecture.
Only i386/amd64, powerpc aim and arm/arm-v6 implementations were
tested at the time of commit. High-level code, not committed yet to
the tree, ensures that the use of the function is only allowed after
explicit enablement.
For sparc64, the existing code has known issues and a stab is added
instead, to allow the kernel linking.
Sponsored by: The FreeBSD Foundation
Tested by: pho (i386, amd64), scottl (amd64), ian (arm and arm-v6)
MFC after: 2 weeks
2013-03-14 20:18:12 +00:00
|
|
|
static void mmu_booke_copy_pages(mmu_t, vm_page_t *,
|
|
|
|
vm_offset_t, vm_page_t *, vm_offset_t, int);
|
2014-08-08 17:12:03 +00:00
|
|
|
static int mmu_booke_enter(mmu_t, pmap_t, vm_offset_t, vm_page_t,
|
|
|
|
vm_prot_t, u_int flags, int8_t psind);
|
2008-03-03 17:17:00 +00:00
|
|
|
static void mmu_booke_enter_object(mmu_t, pmap_t, vm_offset_t, vm_offset_t,
|
|
|
|
vm_page_t, vm_prot_t);
|
|
|
|
static void mmu_booke_enter_quick(mmu_t, pmap_t, vm_offset_t, vm_page_t,
|
|
|
|
vm_prot_t);
|
|
|
|
static vm_paddr_t mmu_booke_extract(mmu_t, pmap_t, vm_offset_t);
|
|
|
|
static vm_page_t mmu_booke_extract_and_hold(mmu_t, pmap_t, vm_offset_t,
|
|
|
|
vm_prot_t);
|
|
|
|
static void mmu_booke_init(mmu_t);
|
|
|
|
static boolean_t mmu_booke_is_modified(mmu_t, vm_page_t);
|
|
|
|
static boolean_t mmu_booke_is_prefaultable(mmu_t, pmap_t, vm_offset_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
|
|
|
static boolean_t mmu_booke_is_referenced(mmu_t, vm_page_t);
|
2012-07-10 22:10:21 +00:00
|
|
|
static int mmu_booke_ts_referenced(mmu_t, vm_page_t);
|
2012-05-24 21:13:24 +00:00
|
|
|
static vm_offset_t mmu_booke_map(mmu_t, vm_offset_t *, vm_paddr_t, vm_paddr_t,
|
2008-03-03 17:17:00 +00:00
|
|
|
int);
|
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
|
|
|
static int mmu_booke_mincore(mmu_t, pmap_t, vm_offset_t,
|
|
|
|
vm_paddr_t *);
|
2008-03-03 17:17:00 +00:00
|
|
|
static void mmu_booke_object_init_pt(mmu_t, pmap_t, vm_offset_t,
|
|
|
|
vm_object_t, vm_pindex_t, vm_size_t);
|
|
|
|
static boolean_t mmu_booke_page_exists_quick(mmu_t, pmap_t, vm_page_t);
|
|
|
|
static void mmu_booke_page_init(mmu_t, vm_page_t);
|
|
|
|
static int mmu_booke_page_wired_mappings(mmu_t, vm_page_t);
|
|
|
|
static void mmu_booke_pinit(mmu_t, pmap_t);
|
|
|
|
static void mmu_booke_pinit0(mmu_t, pmap_t);
|
|
|
|
static void mmu_booke_protect(mmu_t, pmap_t, vm_offset_t, vm_offset_t,
|
|
|
|
vm_prot_t);
|
|
|
|
static void mmu_booke_qenter(mmu_t, vm_offset_t, vm_page_t *, int);
|
|
|
|
static void mmu_booke_qremove(mmu_t, vm_offset_t, int);
|
|
|
|
static void mmu_booke_release(mmu_t, pmap_t);
|
|
|
|
static void mmu_booke_remove(mmu_t, pmap_t, vm_offset_t, vm_offset_t);
|
|
|
|
static void mmu_booke_remove_all(mmu_t, vm_page_t);
|
|
|
|
static void mmu_booke_remove_write(mmu_t, vm_page_t);
|
2014-07-13 16:27:57 +00:00
|
|
|
static void mmu_booke_unwire(mmu_t, pmap_t, vm_offset_t, vm_offset_t);
|
2008-03-03 17:17:00 +00:00
|
|
|
static void mmu_booke_zero_page(mmu_t, vm_page_t);
|
|
|
|
static void mmu_booke_zero_page_area(mmu_t, vm_page_t, int, int);
|
|
|
|
static void mmu_booke_zero_page_idle(mmu_t, vm_page_t);
|
|
|
|
static void mmu_booke_activate(mmu_t, struct thread *);
|
|
|
|
static void mmu_booke_deactivate(mmu_t, struct thread *);
|
|
|
|
static void mmu_booke_bootstrap(mmu_t, vm_offset_t, vm_offset_t);
|
2012-05-24 21:13:24 +00:00
|
|
|
static void *mmu_booke_mapdev(mmu_t, vm_paddr_t, vm_size_t);
|
2013-10-26 14:52:55 +00:00
|
|
|
static void *mmu_booke_mapdev_attr(mmu_t, vm_paddr_t, vm_size_t, vm_memattr_t);
|
2008-03-03 17:17:00 +00:00
|
|
|
static void mmu_booke_unmapdev(mmu_t, vm_offset_t, vm_size_t);
|
2012-05-24 21:13:24 +00:00
|
|
|
static vm_paddr_t mmu_booke_kextract(mmu_t, vm_offset_t);
|
|
|
|
static void mmu_booke_kenter(mmu_t, vm_offset_t, vm_paddr_t);
|
2013-10-26 14:52:55 +00:00
|
|
|
static void mmu_booke_kenter_attr(mmu_t, vm_offset_t, vm_paddr_t, vm_memattr_t);
|
2008-03-03 17:17:00 +00:00
|
|
|
static void mmu_booke_kremove(mmu_t, vm_offset_t);
|
2012-05-24 21:13:24 +00:00
|
|
|
static boolean_t mmu_booke_dev_direct_mapped(mmu_t, vm_paddr_t, vm_size_t);
|
2009-10-21 18:38:02 +00:00
|
|
|
static void mmu_booke_sync_icache(mmu_t, pmap_t, vm_offset_t,
|
|
|
|
vm_size_t);
|
2015-01-07 01:01:39 +00:00
|
|
|
static void mmu_booke_dumpsys_map(mmu_t, vm_paddr_t pa, size_t,
|
|
|
|
void **);
|
|
|
|
static void mmu_booke_dumpsys_unmap(mmu_t, vm_paddr_t pa, size_t,
|
|
|
|
void *);
|
|
|
|
static void mmu_booke_scan_init(mmu_t);
|
2015-08-04 19:46:13 +00:00
|
|
|
static vm_offset_t mmu_booke_quick_enter_page(mmu_t mmu, vm_page_t m);
|
|
|
|
static void mmu_booke_quick_remove_page(mmu_t mmu, vm_offset_t addr);
|
2016-02-27 20:39:36 +00:00
|
|
|
static int mmu_booke_change_attr(mmu_t mmu, vm_offset_t addr,
|
|
|
|
vm_size_t sz, vm_memattr_t mode);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
static mmu_method_t mmu_booke_methods[] = {
|
|
|
|
/* pmap dispatcher interface */
|
|
|
|
MMUMETHOD(mmu_clear_modify, mmu_booke_clear_modify),
|
|
|
|
MMUMETHOD(mmu_copy, mmu_booke_copy),
|
|
|
|
MMUMETHOD(mmu_copy_page, mmu_booke_copy_page),
|
Add pmap function pmap_copy_pages(), which copies the content of the
pages around, taking array of vm_page_t both for source and
destination. Starting offsets and total transfer size are specified.
The function implements optimal algorithm for copying using the
platform-specific optimizations. For instance, on the architectures
were the direct map is available, no transient mappings are created,
for i386 the per-cpu ephemeral page frame is used. The code was
typically borrowed from the pmap_copy_page() for the same
architecture.
Only i386/amd64, powerpc aim and arm/arm-v6 implementations were
tested at the time of commit. High-level code, not committed yet to
the tree, ensures that the use of the function is only allowed after
explicit enablement.
For sparc64, the existing code has known issues and a stab is added
instead, to allow the kernel linking.
Sponsored by: The FreeBSD Foundation
Tested by: pho (i386, amd64), scottl (amd64), ian (arm and arm-v6)
MFC after: 2 weeks
2013-03-14 20:18:12 +00:00
|
|
|
MMUMETHOD(mmu_copy_pages, mmu_booke_copy_pages),
|
2008-03-03 17:17:00 +00:00
|
|
|
MMUMETHOD(mmu_enter, mmu_booke_enter),
|
|
|
|
MMUMETHOD(mmu_enter_object, mmu_booke_enter_object),
|
|
|
|
MMUMETHOD(mmu_enter_quick, mmu_booke_enter_quick),
|
|
|
|
MMUMETHOD(mmu_extract, mmu_booke_extract),
|
|
|
|
MMUMETHOD(mmu_extract_and_hold, mmu_booke_extract_and_hold),
|
|
|
|
MMUMETHOD(mmu_init, mmu_booke_init),
|
|
|
|
MMUMETHOD(mmu_is_modified, mmu_booke_is_modified),
|
|
|
|
MMUMETHOD(mmu_is_prefaultable, mmu_booke_is_prefaultable),
|
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, mmu_booke_is_referenced),
|
2008-03-03 17:17:00 +00:00
|
|
|
MMUMETHOD(mmu_ts_referenced, mmu_booke_ts_referenced),
|
|
|
|
MMUMETHOD(mmu_map, mmu_booke_map),
|
|
|
|
MMUMETHOD(mmu_mincore, mmu_booke_mincore),
|
|
|
|
MMUMETHOD(mmu_object_init_pt, mmu_booke_object_init_pt),
|
|
|
|
MMUMETHOD(mmu_page_exists_quick,mmu_booke_page_exists_quick),
|
|
|
|
MMUMETHOD(mmu_page_init, mmu_booke_page_init),
|
|
|
|
MMUMETHOD(mmu_page_wired_mappings, mmu_booke_page_wired_mappings),
|
|
|
|
MMUMETHOD(mmu_pinit, mmu_booke_pinit),
|
|
|
|
MMUMETHOD(mmu_pinit0, mmu_booke_pinit0),
|
|
|
|
MMUMETHOD(mmu_protect, mmu_booke_protect),
|
|
|
|
MMUMETHOD(mmu_qenter, mmu_booke_qenter),
|
|
|
|
MMUMETHOD(mmu_qremove, mmu_booke_qremove),
|
|
|
|
MMUMETHOD(mmu_release, mmu_booke_release),
|
|
|
|
MMUMETHOD(mmu_remove, mmu_booke_remove),
|
|
|
|
MMUMETHOD(mmu_remove_all, mmu_booke_remove_all),
|
|
|
|
MMUMETHOD(mmu_remove_write, mmu_booke_remove_write),
|
2009-10-21 18:38:02 +00:00
|
|
|
MMUMETHOD(mmu_sync_icache, mmu_booke_sync_icache),
|
2014-07-13 16:27:57 +00:00
|
|
|
MMUMETHOD(mmu_unwire, mmu_booke_unwire),
|
2008-03-03 17:17:00 +00:00
|
|
|
MMUMETHOD(mmu_zero_page, mmu_booke_zero_page),
|
|
|
|
MMUMETHOD(mmu_zero_page_area, mmu_booke_zero_page_area),
|
|
|
|
MMUMETHOD(mmu_zero_page_idle, mmu_booke_zero_page_idle),
|
|
|
|
MMUMETHOD(mmu_activate, mmu_booke_activate),
|
|
|
|
MMUMETHOD(mmu_deactivate, mmu_booke_deactivate),
|
2015-08-04 19:46:13 +00:00
|
|
|
MMUMETHOD(mmu_quick_enter_page, mmu_booke_quick_enter_page),
|
|
|
|
MMUMETHOD(mmu_quick_remove_page, mmu_booke_quick_remove_page),
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Internal interfaces */
|
|
|
|
MMUMETHOD(mmu_bootstrap, mmu_booke_bootstrap),
|
|
|
|
MMUMETHOD(mmu_dev_direct_mapped,mmu_booke_dev_direct_mapped),
|
|
|
|
MMUMETHOD(mmu_mapdev, mmu_booke_mapdev),
|
2013-10-26 14:52:55 +00:00
|
|
|
MMUMETHOD(mmu_mapdev_attr, mmu_booke_mapdev_attr),
|
2008-03-03 17:17:00 +00:00
|
|
|
MMUMETHOD(mmu_kenter, mmu_booke_kenter),
|
2013-10-26 14:52:55 +00:00
|
|
|
MMUMETHOD(mmu_kenter_attr, mmu_booke_kenter_attr),
|
2008-03-03 17:17:00 +00:00
|
|
|
MMUMETHOD(mmu_kextract, mmu_booke_kextract),
|
|
|
|
/* MMUMETHOD(mmu_kremove, mmu_booke_kremove), */
|
|
|
|
MMUMETHOD(mmu_unmapdev, mmu_booke_unmapdev),
|
2016-02-27 20:39:36 +00:00
|
|
|
MMUMETHOD(mmu_change_attr, mmu_booke_change_attr),
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-04-04 22:01:43 +00:00
|
|
|
/* dumpsys() support */
|
|
|
|
MMUMETHOD(mmu_dumpsys_map, mmu_booke_dumpsys_map),
|
|
|
|
MMUMETHOD(mmu_dumpsys_unmap, mmu_booke_dumpsys_unmap),
|
2015-01-07 01:01:39 +00:00
|
|
|
MMUMETHOD(mmu_scan_init, mmu_booke_scan_init),
|
2009-04-04 22:01:43 +00:00
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
{ 0, 0 }
|
|
|
|
};
|
|
|
|
|
2010-09-15 00:17:52 +00:00
|
|
|
MMU_DEF(booke_mmu, MMU_TYPE_BOOKE, mmu_booke_methods, 0);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2013-10-26 14:52:55 +00:00
|
|
|
static __inline uint32_t
|
2015-07-04 19:00:38 +00:00
|
|
|
tlb_calc_wimg(vm_paddr_t pa, vm_memattr_t ma)
|
2013-10-26 14:52:55 +00:00
|
|
|
{
|
|
|
|
uint32_t attrib;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (ma != VM_MEMATTR_DEFAULT) {
|
|
|
|
switch (ma) {
|
|
|
|
case VM_MEMATTR_UNCACHEABLE:
|
2016-02-11 13:15:37 +00:00
|
|
|
return (MAS2_I | MAS2_G);
|
2013-10-26 14:52:55 +00:00
|
|
|
case VM_MEMATTR_WRITE_COMBINING:
|
|
|
|
case VM_MEMATTR_WRITE_BACK:
|
|
|
|
case VM_MEMATTR_PREFETCHABLE:
|
2016-02-11 13:15:37 +00:00
|
|
|
return (MAS2_I);
|
2013-10-26 14:52:55 +00:00
|
|
|
case VM_MEMATTR_WRITE_THROUGH:
|
2016-02-11 13:15:37 +00:00
|
|
|
return (MAS2_W | MAS2_M);
|
2016-02-27 20:39:36 +00:00
|
|
|
case VM_MEMATTR_CACHEABLE:
|
|
|
|
return (MAS2_M);
|
2013-10-26 14:52:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Assume the page is cache inhibited and access is guarded unless
|
|
|
|
* it's in our available memory array.
|
|
|
|
*/
|
|
|
|
attrib = _TLB_ENTRY_IO;
|
|
|
|
for (i = 0; i < physmem_regions_sz; i++) {
|
|
|
|
if ((pa >= physmem_regions[i].mr_start) &&
|
|
|
|
(pa < (physmem_regions[i].mr_start +
|
|
|
|
physmem_regions[i].mr_size))) {
|
|
|
|
attrib = _TLB_ENTRY_MEM;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return (attrib);
|
|
|
|
}
|
|
|
|
|
2009-05-21 11:43:37 +00:00
|
|
|
static inline void
|
|
|
|
tlb_miss_lock(void)
|
|
|
|
{
|
|
|
|
#ifdef SMP
|
|
|
|
struct pcpu *pc;
|
|
|
|
|
|
|
|
if (!smp_started)
|
|
|
|
return;
|
|
|
|
|
2011-05-31 21:22:44 +00:00
|
|
|
STAILQ_FOREACH(pc, &cpuhead, pc_allcpu) {
|
2009-05-21 11:43:37 +00:00
|
|
|
if (pc != pcpup) {
|
|
|
|
|
|
|
|
CTR3(KTR_PMAP, "%s: tlb miss LOCK of CPU=%d, "
|
|
|
|
"tlb_lock=%p", __func__, pc->pc_cpuid, pc->pc_booke_tlb_lock);
|
|
|
|
|
|
|
|
KASSERT((pc->pc_cpuid != PCPU_GET(cpuid)),
|
|
|
|
("tlb_miss_lock: tried to lock self"));
|
|
|
|
|
|
|
|
tlb_lock(pc->pc_booke_tlb_lock);
|
|
|
|
|
|
|
|
CTR1(KTR_PMAP, "%s: locked", __func__);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
tlb_miss_unlock(void)
|
|
|
|
{
|
|
|
|
#ifdef SMP
|
|
|
|
struct pcpu *pc;
|
|
|
|
|
|
|
|
if (!smp_started)
|
|
|
|
return;
|
|
|
|
|
2011-05-31 21:22:44 +00:00
|
|
|
STAILQ_FOREACH(pc, &cpuhead, pc_allcpu) {
|
2009-05-21 11:43:37 +00:00
|
|
|
if (pc != pcpup) {
|
|
|
|
CTR2(KTR_PMAP, "%s: tlb miss UNLOCK of CPU=%d",
|
|
|
|
__func__, pc->pc_cpuid);
|
|
|
|
|
|
|
|
tlb_unlock(pc->pc_booke_tlb_lock);
|
|
|
|
|
|
|
|
CTR1(KTR_PMAP, "%s: unlocked", __func__);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/* Return number of entries in TLB0. */
|
|
|
|
static __inline void
|
|
|
|
tlb0_get_tlbconf(void)
|
|
|
|
{
|
|
|
|
uint32_t tlb0_cfg;
|
|
|
|
|
|
|
|
tlb0_cfg = mfspr(SPR_TLB0CFG);
|
2009-01-13 15:41:58 +00:00
|
|
|
tlb0_entries = tlb0_cfg & TLBCFG_NENTRY_MASK;
|
|
|
|
tlb0_ways = (tlb0_cfg & TLBCFG_ASSOC_MASK) >> TLBCFG_ASSOC_SHIFT;
|
|
|
|
tlb0_entries_per_way = tlb0_entries / tlb0_ways;
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
2015-08-28 03:03:09 +00:00
|
|
|
/* Return number of entries in TLB1. */
|
|
|
|
static __inline void
|
|
|
|
tlb1_get_tlbconf(void)
|
|
|
|
{
|
|
|
|
uint32_t tlb1_cfg;
|
|
|
|
|
|
|
|
tlb1_cfg = mfspr(SPR_TLB1CFG);
|
|
|
|
tlb1_entries = tlb1_cfg & TLBCFG_NENTRY_MASK;
|
|
|
|
}
|
|
|
|
|
2016-01-19 03:07:25 +00:00
|
|
|
/**************************************************************************/
|
|
|
|
/* Page table related */
|
|
|
|
/**************************************************************************/
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/* Initialize pool of kva ptbl buffers. */
|
|
|
|
static void
|
|
|
|
ptbl_init(void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
2009-01-13 16:15:49 +00:00
|
|
|
CTR3(KTR_PMAP, "%s: s (ptbl_bufs = 0x%08x size 0x%08x)", __func__,
|
|
|
|
(uint32_t)ptbl_bufs, sizeof(struct ptbl_buf) * PTBL_BUFS);
|
|
|
|
CTR3(KTR_PMAP, "%s: s (ptbl_buf_pool_vabase = 0x%08x size = 0x%08x)",
|
|
|
|
__func__, ptbl_buf_pool_vabase, PTBL_BUFS * PTBL_PAGES * PAGE_SIZE);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
mtx_init(&ptbl_buf_freelist_lock, "ptbl bufs lock", NULL, MTX_DEF);
|
|
|
|
TAILQ_INIT(&ptbl_buf_freelist);
|
|
|
|
|
|
|
|
for (i = 0; i < PTBL_BUFS; i++) {
|
|
|
|
ptbl_bufs[i].kva = ptbl_buf_pool_vabase + i * PTBL_PAGES * PAGE_SIZE;
|
|
|
|
TAILQ_INSERT_TAIL(&ptbl_buf_freelist, &ptbl_bufs[i], link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-08-28 07:38:08 +00:00
|
|
|
/* Get a ptbl_buf from the freelist. */
|
2008-03-03 17:17:00 +00:00
|
|
|
static struct ptbl_buf *
|
|
|
|
ptbl_buf_alloc(void)
|
|
|
|
{
|
|
|
|
struct ptbl_buf *buf;
|
|
|
|
|
|
|
|
mtx_lock(&ptbl_buf_freelist_lock);
|
|
|
|
buf = TAILQ_FIRST(&ptbl_buf_freelist);
|
|
|
|
if (buf != NULL)
|
|
|
|
TAILQ_REMOVE(&ptbl_buf_freelist, buf, link);
|
|
|
|
mtx_unlock(&ptbl_buf_freelist_lock);
|
|
|
|
|
2009-01-13 16:15:49 +00:00
|
|
|
CTR2(KTR_PMAP, "%s: buf = %p", __func__, buf);
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
return (buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Return ptbl buff to free pool. */
|
|
|
|
static void
|
|
|
|
ptbl_buf_free(struct ptbl_buf *buf)
|
|
|
|
{
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
CTR2(KTR_PMAP, "%s: buf = %p", __func__, buf);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
mtx_lock(&ptbl_buf_freelist_lock);
|
|
|
|
TAILQ_INSERT_TAIL(&ptbl_buf_freelist, buf, link);
|
|
|
|
mtx_unlock(&ptbl_buf_freelist_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2009-01-13 15:41:58 +00:00
|
|
|
* Search the list of allocated ptbl bufs and find on list of allocated ptbls
|
2008-03-03 17:17:00 +00:00
|
|
|
*/
|
|
|
|
static void
|
|
|
|
ptbl_free_pmap_ptbl(pmap_t pmap, pte_t *ptbl)
|
|
|
|
{
|
|
|
|
struct ptbl_buf *pbuf;
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
CTR2(KTR_PMAP, "%s: ptbl = %p", __func__, ptbl);
|
|
|
|
|
|
|
|
PMAP_LOCK_ASSERT(pmap, MA_OWNED);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
TAILQ_FOREACH(pbuf, &pmap->pm_ptbl_list, link)
|
2008-03-03 17:17:00 +00:00
|
|
|
if (pbuf->kva == (vm_offset_t)ptbl) {
|
|
|
|
/* Remove from pmap ptbl buf list. */
|
2009-01-13 15:41:58 +00:00
|
|
|
TAILQ_REMOVE(&pmap->pm_ptbl_list, pbuf, link);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
/* Free corresponding ptbl buf. */
|
2008-03-03 17:17:00 +00:00
|
|
|
ptbl_buf_free(pbuf);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate page table. */
|
2009-01-13 15:41:58 +00:00
|
|
|
static pte_t *
|
2014-08-08 17:12:03 +00:00
|
|
|
ptbl_alloc(mmu_t mmu, pmap_t pmap, unsigned int pdir_idx, boolean_t nosleep)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
|
|
|
vm_page_t mtbl[PTBL_PAGES];
|
|
|
|
vm_page_t m;
|
|
|
|
struct ptbl_buf *pbuf;
|
|
|
|
unsigned int pidx;
|
2009-01-13 15:41:58 +00:00
|
|
|
pte_t *ptbl;
|
2014-08-08 17:12:03 +00:00
|
|
|
int i, j;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
CTR4(KTR_PMAP, "%s: pmap = %p su = %d pdir_idx = %d", __func__, pmap,
|
|
|
|
(pmap == kernel_pmap), pdir_idx);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
KASSERT((pdir_idx <= (VM_MAXUSER_ADDRESS / PDIR_SIZE)),
|
|
|
|
("ptbl_alloc: invalid pdir_idx"));
|
|
|
|
KASSERT((pmap->pm_pdir[pdir_idx] == NULL),
|
|
|
|
("pte_alloc: valid ptbl entry exists!"));
|
|
|
|
|
|
|
|
pbuf = ptbl_buf_alloc();
|
|
|
|
if (pbuf == NULL)
|
|
|
|
panic("pte_alloc: couldn't alloc kernel virtual memory");
|
2009-01-13 15:41:58 +00:00
|
|
|
|
|
|
|
ptbl = (pte_t *)pbuf->kva;
|
|
|
|
|
|
|
|
CTR2(KTR_PMAP, "%s: ptbl kva = %p", __func__, ptbl);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Allocate ptbl pages, this will sleep! */
|
|
|
|
for (i = 0; i < PTBL_PAGES; i++) {
|
|
|
|
pidx = (PTBL_PAGES * pdir_idx) + i;
|
2009-01-13 15:41:58 +00:00
|
|
|
while ((m = vm_page_alloc(NULL, pidx,
|
|
|
|
VM_ALLOC_NOOBJ | VM_ALLOC_WIRED)) == NULL) {
|
2008-03-03 17:17:00 +00:00
|
|
|
PMAP_UNLOCK(pmap);
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wunlock(&pvh_global_lock);
|
2014-08-08 17:12:03 +00:00
|
|
|
if (nosleep) {
|
|
|
|
ptbl_free_pmap_ptbl(pmap, ptbl);
|
|
|
|
for (j = 0; j < i; j++)
|
|
|
|
vm_page_free(mtbl[j]);
|
|
|
|
atomic_subtract_int(&vm_cnt.v_wire_count, i);
|
|
|
|
return (NULL);
|
|
|
|
}
|
2008-03-03 17:17:00 +00:00
|
|
|
VM_WAIT;
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
PMAP_LOCK(pmap);
|
|
|
|
}
|
|
|
|
mtbl[i] = m;
|
|
|
|
}
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
/* Map allocated pages into kernel_pmap. */
|
|
|
|
mmu_booke_qenter(mmu, (vm_offset_t)ptbl, mtbl, PTBL_PAGES);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Zero whole ptbl. */
|
2009-01-13 15:41:58 +00:00
|
|
|
bzero((caddr_t)ptbl, PTBL_PAGES * PAGE_SIZE);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Add pbuf to the pmap ptbl bufs list. */
|
2009-01-13 15:41:58 +00:00
|
|
|
TAILQ_INSERT_TAIL(&pmap->pm_ptbl_list, pbuf, link);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
return (ptbl);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Free ptbl pages and invalidate pdir entry. */
|
|
|
|
static void
|
|
|
|
ptbl_free(mmu_t mmu, pmap_t pmap, unsigned int pdir_idx)
|
|
|
|
{
|
|
|
|
pte_t *ptbl;
|
|
|
|
vm_paddr_t pa;
|
|
|
|
vm_offset_t va;
|
|
|
|
vm_page_t m;
|
|
|
|
int i;
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
CTR4(KTR_PMAP, "%s: pmap = %p su = %d pdir_idx = %d", __func__, pmap,
|
|
|
|
(pmap == kernel_pmap), pdir_idx);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
KASSERT((pdir_idx <= (VM_MAXUSER_ADDRESS / PDIR_SIZE)),
|
|
|
|
("ptbl_free: invalid pdir_idx"));
|
|
|
|
|
|
|
|
ptbl = pmap->pm_pdir[pdir_idx];
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
CTR2(KTR_PMAP, "%s: ptbl = %p", __func__, ptbl);
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
KASSERT((ptbl != NULL), ("ptbl_free: null ptbl"));
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
/*
|
|
|
|
* Invalidate the pdir entry as soon as possible, so that other CPUs
|
|
|
|
* don't attempt to look up the page tables we are releasing.
|
|
|
|
*/
|
|
|
|
mtx_lock_spin(&tlbivax_mutex);
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_lock();
|
2009-01-13 15:41:58 +00:00
|
|
|
|
|
|
|
pmap->pm_pdir[pdir_idx] = NULL;
|
|
|
|
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_unlock();
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_unlock_spin(&tlbivax_mutex);
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
for (i = 0; i < PTBL_PAGES; i++) {
|
|
|
|
va = ((vm_offset_t)ptbl + (i * PAGE_SIZE));
|
|
|
|
pa = pte_vatopa(mmu, kernel_pmap, va);
|
|
|
|
m = PHYS_TO_VM_PAGE(pa);
|
|
|
|
vm_page_free_zero(m);
|
2014-03-22 10:26:09 +00:00
|
|
|
atomic_subtract_int(&vm_cnt.v_wire_count, 1);
|
2008-03-03 17:17:00 +00:00
|
|
|
mmu_booke_kremove(mmu, va);
|
|
|
|
}
|
|
|
|
|
|
|
|
ptbl_free_pmap_ptbl(pmap, ptbl);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Decrement ptbl pages hold count and attempt to free ptbl pages.
|
|
|
|
* Called when removing pte entry from ptbl.
|
|
|
|
*
|
|
|
|
* Return 1 if ptbl pages were freed.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
ptbl_unhold(mmu_t mmu, pmap_t pmap, unsigned int pdir_idx)
|
|
|
|
{
|
|
|
|
pte_t *ptbl;
|
|
|
|
vm_paddr_t pa;
|
|
|
|
vm_page_t m;
|
|
|
|
int i;
|
|
|
|
|
2009-01-13 16:15:49 +00:00
|
|
|
CTR4(KTR_PMAP, "%s: pmap = %p su = %d pdir_idx = %d", __func__, pmap,
|
|
|
|
(pmap == kernel_pmap), pdir_idx);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
KASSERT((pdir_idx <= (VM_MAXUSER_ADDRESS / PDIR_SIZE)),
|
|
|
|
("ptbl_unhold: invalid pdir_idx"));
|
|
|
|
KASSERT((pmap != kernel_pmap),
|
|
|
|
("ptbl_unhold: unholding kernel ptbl!"));
|
|
|
|
|
|
|
|
ptbl = pmap->pm_pdir[pdir_idx];
|
|
|
|
|
|
|
|
//debugf("ptbl_unhold: ptbl = 0x%08x\n", (u_int32_t)ptbl);
|
|
|
|
KASSERT(((vm_offset_t)ptbl >= VM_MIN_KERNEL_ADDRESS),
|
|
|
|
("ptbl_unhold: non kva ptbl"));
|
|
|
|
|
|
|
|
/* decrement hold count */
|
|
|
|
for (i = 0; i < PTBL_PAGES; i++) {
|
2009-01-13 16:15:49 +00:00
|
|
|
pa = pte_vatopa(mmu, kernel_pmap,
|
|
|
|
(vm_offset_t)ptbl + (i * PAGE_SIZE));
|
2008-03-03 17:17:00 +00:00
|
|
|
m = PHYS_TO_VM_PAGE(pa);
|
|
|
|
m->wire_count--;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Free ptbl pages if there are no pte etries in this ptbl.
|
2009-01-13 16:15:49 +00:00
|
|
|
* wire_count has the same value for all ptbl pages, so check the last
|
|
|
|
* page.
|
2008-03-03 17:17:00 +00:00
|
|
|
*/
|
|
|
|
if (m->wire_count == 0) {
|
|
|
|
ptbl_free(mmu, pmap, pdir_idx);
|
|
|
|
|
|
|
|
//debugf("ptbl_unhold: e (freed ptbl)\n");
|
|
|
|
return (1);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2009-01-13 16:15:49 +00:00
|
|
|
* Increment hold count for ptbl pages. This routine is used when a new pte
|
|
|
|
* entry is being inserted into the ptbl.
|
2008-03-03 17:17:00 +00:00
|
|
|
*/
|
|
|
|
static void
|
|
|
|
ptbl_hold(mmu_t mmu, pmap_t pmap, unsigned int pdir_idx)
|
|
|
|
{
|
|
|
|
vm_paddr_t pa;
|
|
|
|
pte_t *ptbl;
|
|
|
|
vm_page_t m;
|
|
|
|
int i;
|
|
|
|
|
2009-01-13 16:15:49 +00:00
|
|
|
CTR3(KTR_PMAP, "%s: pmap = %p pdir_idx = %d", __func__, pmap,
|
|
|
|
pdir_idx);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
KASSERT((pdir_idx <= (VM_MAXUSER_ADDRESS / PDIR_SIZE)),
|
|
|
|
("ptbl_hold: invalid pdir_idx"));
|
|
|
|
KASSERT((pmap != kernel_pmap),
|
|
|
|
("ptbl_hold: holding kernel ptbl!"));
|
|
|
|
|
|
|
|
ptbl = pmap->pm_pdir[pdir_idx];
|
|
|
|
|
|
|
|
KASSERT((ptbl != NULL), ("ptbl_hold: null ptbl"));
|
|
|
|
|
|
|
|
for (i = 0; i < PTBL_PAGES; i++) {
|
2009-01-13 16:15:49 +00:00
|
|
|
pa = pte_vatopa(mmu, kernel_pmap,
|
|
|
|
(vm_offset_t)ptbl + (i * PAGE_SIZE));
|
2008-03-03 17:17:00 +00:00
|
|
|
m = PHYS_TO_VM_PAGE(pa);
|
|
|
|
m->wire_count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate pv_entry structure. */
|
|
|
|
pv_entry_t
|
|
|
|
pv_alloc(void)
|
|
|
|
{
|
|
|
|
pv_entry_t pv;
|
|
|
|
|
|
|
|
pv_entry_count++;
|
2009-06-13 18:35:29 +00:00
|
|
|
if (pv_entry_count > pv_entry_high_water)
|
|
|
|
pagedaemon_wakeup();
|
2008-03-03 17:17:00 +00:00
|
|
|
pv = uma_zalloc(pvzone, M_NOWAIT);
|
|
|
|
|
|
|
|
return (pv);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Free pv_entry structure. */
|
|
|
|
static __inline void
|
|
|
|
pv_free(pv_entry_t pve)
|
|
|
|
{
|
|
|
|
|
|
|
|
pv_entry_count--;
|
|
|
|
uma_zfree(pvzone, pve);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Allocate and initialize pv_entry structure. */
|
|
|
|
static void
|
|
|
|
pv_insert(pmap_t pmap, vm_offset_t va, vm_page_t m)
|
|
|
|
{
|
|
|
|
pv_entry_t pve;
|
|
|
|
|
|
|
|
//int su = (pmap == kernel_pmap);
|
|
|
|
//debugf("pv_insert: s (su = %d pmap = 0x%08x va = 0x%08x m = 0x%08x)\n", su,
|
|
|
|
// (u_int32_t)pmap, va, (u_int32_t)m);
|
|
|
|
|
|
|
|
pve = pv_alloc();
|
|
|
|
if (pve == NULL)
|
|
|
|
panic("pv_insert: no pv entries!");
|
|
|
|
|
|
|
|
pve->pv_pmap = pmap;
|
|
|
|
pve->pv_va = va;
|
|
|
|
|
|
|
|
/* add to pv_list */
|
|
|
|
PMAP_LOCK_ASSERT(pmap, MA_OWNED);
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_assert(&pvh_global_lock, RA_WLOCKED);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
TAILQ_INSERT_TAIL(&m->md.pv_list, pve, pv_link);
|
|
|
|
|
|
|
|
//debugf("pv_insert: e\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Destroy pv entry. */
|
|
|
|
static void
|
|
|
|
pv_remove(pmap_t pmap, vm_offset_t va, vm_page_t m)
|
|
|
|
{
|
|
|
|
pv_entry_t pve;
|
|
|
|
|
|
|
|
//int su = (pmap == kernel_pmap);
|
|
|
|
//debugf("pv_remove: s (su = %d pmap = 0x%08x va = 0x%08x)\n", su, (u_int32_t)pmap, va);
|
|
|
|
|
|
|
|
PMAP_LOCK_ASSERT(pmap, MA_OWNED);
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_assert(&pvh_global_lock, RA_WLOCKED);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* find pv entry */
|
|
|
|
TAILQ_FOREACH(pve, &m->md.pv_list, pv_link) {
|
|
|
|
if ((pmap == pve->pv_pmap) && (va == pve->pv_va)) {
|
|
|
|
/* remove from pv_list */
|
|
|
|
TAILQ_REMOVE(&m->md.pv_list, pve, pv_link);
|
|
|
|
if (TAILQ_EMPTY(&m->md.pv_list))
|
2011-09-06 10:30:11 +00:00
|
|
|
vm_page_aflag_clear(m, PGA_WRITEABLE);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* free pv entry struct */
|
|
|
|
pv_free(pve);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//debugf("pv_remove: e\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Clean pte entry, try to free page table page if requested.
|
|
|
|
*
|
|
|
|
* Return 1 if ptbl pages were freed, otherwise return 0.
|
|
|
|
*/
|
|
|
|
static int
|
2009-01-13 16:15:49 +00:00
|
|
|
pte_remove(mmu_t mmu, pmap_t pmap, vm_offset_t va, uint8_t flags)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
|
|
|
unsigned int pdir_idx = PDIR_IDX(va);
|
|
|
|
unsigned int ptbl_idx = PTBL_IDX(va);
|
|
|
|
vm_page_t m;
|
|
|
|
pte_t *ptbl;
|
|
|
|
pte_t *pte;
|
|
|
|
|
|
|
|
//int su = (pmap == kernel_pmap);
|
|
|
|
//debugf("pte_remove: s (su = %d pmap = 0x%08x va = 0x%08x flags = %d)\n",
|
|
|
|
// su, (u_int32_t)pmap, va, flags);
|
|
|
|
|
|
|
|
ptbl = pmap->pm_pdir[pdir_idx];
|
|
|
|
KASSERT(ptbl, ("pte_remove: null ptbl"));
|
|
|
|
|
|
|
|
pte = &ptbl[ptbl_idx];
|
|
|
|
|
|
|
|
if (pte == NULL || !PTE_ISVALID(pte))
|
|
|
|
return (0);
|
|
|
|
|
|
|
|
if (PTE_ISWIRED(pte))
|
|
|
|
pmap->pm_stats.wired_count--;
|
|
|
|
|
2009-04-24 02:53:38 +00:00
|
|
|
/* Handle managed entry. */
|
|
|
|
if (PTE_ISMANAGED(pte)) {
|
|
|
|
/* Get vm_page_t for mapped pte. */
|
|
|
|
m = PHYS_TO_VM_PAGE(PTE_PA(pte));
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-04-24 02:53:38 +00:00
|
|
|
if (PTE_ISMODIFIED(pte))
|
|
|
|
vm_page_dirty(m);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-04-24 02:53:38 +00:00
|
|
|
if (PTE_ISREFERENCED(pte))
|
2011-09-06 10:30:11 +00:00
|
|
|
vm_page_aflag_set(m, PGA_REFERENCED);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-04-24 02:53:38 +00:00
|
|
|
pv_remove(pmap, va, m);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_lock_spin(&tlbivax_mutex);
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_lock();
|
2009-01-13 15:41:58 +00:00
|
|
|
|
|
|
|
tlb0_flush_entry(va);
|
2016-02-11 13:15:37 +00:00
|
|
|
*pte = 0;
|
2009-01-13 15:41:58 +00:00
|
|
|
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_unlock();
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_unlock_spin(&tlbivax_mutex);
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
pmap->pm_stats.resident_count--;
|
|
|
|
|
|
|
|
if (flags & PTBL_UNHOLD) {
|
|
|
|
//debugf("pte_remove: e (unhold)\n");
|
|
|
|
return (ptbl_unhold(mmu, pmap, pdir_idx));
|
|
|
|
}
|
|
|
|
|
|
|
|
//debugf("pte_remove: e\n");
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Insert PTE for a given page and virtual address.
|
|
|
|
*/
|
2014-08-08 17:12:03 +00:00
|
|
|
static int
|
|
|
|
pte_enter(mmu_t mmu, pmap_t pmap, vm_page_t m, vm_offset_t va, uint32_t flags,
|
|
|
|
boolean_t nosleep)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
|
|
|
unsigned int pdir_idx = PDIR_IDX(va);
|
|
|
|
unsigned int ptbl_idx = PTBL_IDX(va);
|
2009-01-13 15:41:58 +00:00
|
|
|
pte_t *ptbl, *pte;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
CTR4(KTR_PMAP, "%s: su = %d pmap = %p va = %p", __func__,
|
|
|
|
pmap == kernel_pmap, pmap, va);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Get the page table pointer. */
|
|
|
|
ptbl = pmap->pm_pdir[pdir_idx];
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
if (ptbl == NULL) {
|
|
|
|
/* Allocate page table pages. */
|
2014-08-08 17:12:03 +00:00
|
|
|
ptbl = ptbl_alloc(mmu, pmap, pdir_idx, nosleep);
|
|
|
|
if (ptbl == NULL) {
|
|
|
|
KASSERT(nosleep, ("nosleep and NULL ptbl"));
|
|
|
|
return (ENOMEM);
|
|
|
|
}
|
2009-01-13 15:41:58 +00:00
|
|
|
} else {
|
2008-03-03 17:17:00 +00:00
|
|
|
/*
|
|
|
|
* Check if there is valid mapping for requested
|
|
|
|
* va, if there is, remove it.
|
|
|
|
*/
|
|
|
|
pte = &pmap->pm_pdir[pdir_idx][ptbl_idx];
|
|
|
|
if (PTE_ISVALID(pte)) {
|
|
|
|
pte_remove(mmu, pmap, va, PTBL_HOLD);
|
|
|
|
} else {
|
|
|
|
/*
|
|
|
|
* pte is not used, increment hold count
|
|
|
|
* for ptbl pages.
|
|
|
|
*/
|
|
|
|
if (pmap != kernel_pmap)
|
|
|
|
ptbl_hold(mmu, pmap, pdir_idx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2009-01-13 15:41:58 +00:00
|
|
|
* Insert pv_entry into pv_list for mapped page if part of managed
|
|
|
|
* memory.
|
2008-03-03 17:17:00 +00:00
|
|
|
*/
|
2011-08-09 21:01:36 +00:00
|
|
|
if ((m->oflags & VPO_UNMANAGED) == 0) {
|
|
|
|
flags |= PTE_MANAGED;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2011-08-09 21:01:36 +00:00
|
|
|
/* Create and insert pv entry. */
|
|
|
|
pv_insert(pmap, va, m);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pmap->pm_stats.resident_count++;
|
2009-01-13 15:41:58 +00:00
|
|
|
|
|
|
|
mtx_lock_spin(&tlbivax_mutex);
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_lock();
|
2009-01-13 15:41:58 +00:00
|
|
|
|
|
|
|
tlb0_flush_entry(va);
|
|
|
|
if (pmap->pm_pdir[pdir_idx] == NULL) {
|
|
|
|
/*
|
|
|
|
* If we just allocated a new page table, hook it in
|
|
|
|
* the pdir.
|
|
|
|
*/
|
|
|
|
pmap->pm_pdir[pdir_idx] = ptbl;
|
|
|
|
}
|
|
|
|
pte = &(pmap->pm_pdir[pdir_idx][ptbl_idx]);
|
2016-02-11 13:15:37 +00:00
|
|
|
*pte = PTE_RPN_FROM_PA(VM_PAGE_TO_PHYS(m));
|
|
|
|
*pte |= (PTE_VALID | flags | PTE_PS_4KB); /* 4KB pages only */
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_unlock();
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_unlock_spin(&tlbivax_mutex);
|
2014-08-08 17:12:03 +00:00
|
|
|
return (0);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Return the pa for the given pmap/va. */
|
|
|
|
static vm_paddr_t
|
|
|
|
pte_vatopa(mmu_t mmu, pmap_t pmap, vm_offset_t va)
|
|
|
|
{
|
|
|
|
vm_paddr_t pa = 0;
|
|
|
|
pte_t *pte;
|
|
|
|
|
|
|
|
pte = pte_find(mmu, pmap, va);
|
|
|
|
if ((pte != NULL) && PTE_ISVALID(pte))
|
|
|
|
pa = (PTE_PA(pte) | (va & PTE_PA_MASK));
|
|
|
|
return (pa);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get a pointer to a PTE in a page table. */
|
|
|
|
static pte_t *
|
|
|
|
pte_find(mmu_t mmu, pmap_t pmap, vm_offset_t va)
|
|
|
|
{
|
|
|
|
unsigned int pdir_idx = PDIR_IDX(va);
|
|
|
|
unsigned int ptbl_idx = PTBL_IDX(va);
|
|
|
|
|
|
|
|
KASSERT((pmap != NULL), ("pte_find: invalid pmap"));
|
|
|
|
|
|
|
|
if (pmap->pm_pdir[pdir_idx])
|
|
|
|
return (&(pmap->pm_pdir[pdir_idx][ptbl_idx]));
|
|
|
|
|
|
|
|
return (NULL);
|
|
|
|
}
|
|
|
|
|
2016-01-19 03:07:25 +00:00
|
|
|
/* Set up kernel page tables. */
|
|
|
|
static void
|
|
|
|
kernel_pte_alloc(vm_offset_t data_end, vm_offset_t addr, vm_offset_t pdir)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
vm_offset_t va;
|
|
|
|
pte_t *pte;
|
|
|
|
|
|
|
|
/* Initialize kernel pdir */
|
|
|
|
for (i = 0; i < kernel_ptbls; i++)
|
|
|
|
kernel_pmap->pm_pdir[kptbl_min + i] =
|
|
|
|
(pte_t *)(pdir + (i * PAGE_SIZE * PTBL_PAGES));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fill in PTEs covering kernel code and data. They are not required
|
|
|
|
* for address translation, as this area is covered by static TLB1
|
|
|
|
* entries, but for pte_vatopa() to work correctly with kernel area
|
|
|
|
* addresses.
|
|
|
|
*/
|
|
|
|
for (va = addr; va < data_end; va += PAGE_SIZE) {
|
|
|
|
pte = &(kernel_pmap->pm_pdir[PDIR_IDX(va)][PTBL_IDX(va)]);
|
2016-02-11 13:15:37 +00:00
|
|
|
*pte = PTE_RPN_FROM_PA(kernload + (va - kernstart));
|
|
|
|
*pte |= PTE_M | PTE_SR | PTE_SW | PTE_SX | PTE_WIRED |
|
|
|
|
PTE_VALID | PTE_PS_4KB;
|
2016-01-19 03:07:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/**************************************************************************/
|
|
|
|
/* PMAP related */
|
|
|
|
/**************************************************************************/
|
|
|
|
|
|
|
|
/*
|
2011-05-28 04:10:44 +00:00
|
|
|
* This is called during booke_init, before the system is really initialized.
|
2008-03-03 17:17:00 +00:00
|
|
|
*/
|
|
|
|
static void
|
2009-04-04 22:01:43 +00:00
|
|
|
mmu_booke_bootstrap(mmu_t mmu, vm_offset_t start, vm_offset_t kernelend)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
2015-12-24 04:30:15 +00:00
|
|
|
vm_paddr_t phys_kernelend;
|
2008-03-03 17:17:00 +00:00
|
|
|
struct mem_region *mp, *mp1;
|
|
|
|
int cnt, i, j;
|
2015-12-24 04:30:15 +00:00
|
|
|
vm_paddr_t s, e, sz;
|
|
|
|
vm_paddr_t physsz, hwphyssz;
|
2008-03-03 17:17:00 +00:00
|
|
|
u_int phys_avail_count;
|
2015-12-24 04:30:15 +00:00
|
|
|
vm_size_t kstack0_sz;
|
2016-01-19 03:07:25 +00:00
|
|
|
vm_offset_t kernel_pdir, kstack0;
|
2008-08-26 17:07:37 +00:00
|
|
|
vm_paddr_t kstack0_phys;
|
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;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
debugf("mmu_booke_bootstrap: entered\n");
|
|
|
|
|
2015-03-01 21:23:23 +00:00
|
|
|
/* Set interesting system properties */
|
|
|
|
hw_direct_map = 0;
|
|
|
|
elf32_nxstack = 1;
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
/* Initialize invalidation mutex */
|
|
|
|
mtx_init(&tlbivax_mutex, "tlbivax", NULL, MTX_SPIN);
|
|
|
|
|
|
|
|
/* Read TLB0 size and associativity. */
|
|
|
|
tlb0_get_tlbconf();
|
|
|
|
|
2011-08-02 15:35:43 +00:00
|
|
|
/*
|
|
|
|
* Align kernel start and end address (kernel image).
|
|
|
|
* Note that kernel end does not necessarily relate to kernsize.
|
|
|
|
* kernsize is the size of the kernel that is actually mapped.
|
|
|
|
*/
|
2015-04-30 01:24:25 +00:00
|
|
|
kernstart = trunc_page(start);
|
2009-04-04 22:01:43 +00:00
|
|
|
data_start = round_page(kernelend);
|
|
|
|
data_end = data_start;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2011-08-02 15:35:43 +00:00
|
|
|
/*
|
|
|
|
* Addresses of preloaded modules (like file systems) use
|
|
|
|
* physical addresses. Make sure we relocate those into
|
|
|
|
* virtual addresses.
|
|
|
|
*/
|
|
|
|
preload_addr_relocate = kernstart - kernload;
|
|
|
|
|
|
|
|
/* Allocate the dynamic per-cpu area. */
|
|
|
|
dpcpu = (void *)data_end;
|
|
|
|
data_end += DPCPU_SIZE;
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/* Allocate space for the message buffer. */
|
2009-04-04 22:01:43 +00:00
|
|
|
msgbufp = (struct msgbuf *)data_end;
|
2011-01-21 10:26:26 +00:00
|
|
|
data_end += msgbufsize;
|
2009-01-13 15:41:58 +00:00
|
|
|
debugf(" msgbufp at 0x%08x end = 0x%08x\n", (uint32_t)msgbufp,
|
2009-04-04 22:01:43 +00:00
|
|
|
data_end);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-04-04 22:01:43 +00:00
|
|
|
data_end = round_page(data_end);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Allocate space for ptbl_bufs. */
|
2009-04-04 22:01:43 +00:00
|
|
|
ptbl_bufs = (struct ptbl_buf *)data_end;
|
|
|
|
data_end += sizeof(struct ptbl_buf) * PTBL_BUFS;
|
2009-01-13 15:41:58 +00:00
|
|
|
debugf(" ptbl_bufs at 0x%08x end = 0x%08x\n", (uint32_t)ptbl_bufs,
|
2009-04-04 22:01:43 +00:00
|
|
|
data_end);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-04-04 22:01:43 +00:00
|
|
|
data_end = round_page(data_end);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Allocate PTE tables for kernel KVA. */
|
2009-04-04 22:01:43 +00:00
|
|
|
kernel_pdir = data_end;
|
2016-04-26 14:44:49 +00:00
|
|
|
kernel_ptbls = howmany(VM_MAX_KERNEL_ADDRESS - VM_MIN_KERNEL_ADDRESS,
|
|
|
|
PDIR_SIZE);
|
2009-04-04 22:01:43 +00:00
|
|
|
data_end += kernel_ptbls * PTBL_PAGES * PAGE_SIZE;
|
2008-03-03 17:17:00 +00:00
|
|
|
debugf(" kernel ptbls: %d\n", kernel_ptbls);
|
2009-04-04 22:01:43 +00:00
|
|
|
debugf(" kernel pdir at 0x%08x end = 0x%08x\n", kernel_pdir, data_end);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-04-04 22:01:43 +00:00
|
|
|
debugf(" data_end: 0x%08x\n", data_end);
|
2011-08-02 15:35:43 +00:00
|
|
|
if (data_end - kernstart > kernsize) {
|
|
|
|
kernsize += tlb1_mapin_region(kernstart + kernsize,
|
|
|
|
kernload + kernsize, (data_end - kernstart) - kernsize);
|
|
|
|
}
|
|
|
|
data_end = kernstart + kernsize;
|
2009-04-04 22:01:43 +00:00
|
|
|
debugf(" updated data_end: 0x%08x\n", data_end);
|
|
|
|
|
2008-08-28 07:38:08 +00:00
|
|
|
/*
|
|
|
|
* Clear the structures - note we can only do it safely after the
|
2009-01-13 15:41:58 +00:00
|
|
|
* possible additional TLB1 translations are in place (above) so that
|
2009-04-04 22:01:43 +00:00
|
|
|
* all range up to the currently calculated 'data_end' is covered.
|
2008-08-28 07:38:08 +00:00
|
|
|
*/
|
2011-08-02 15:35:43 +00:00
|
|
|
dpcpu_init(dpcpu, 0);
|
2008-08-28 07:38:08 +00:00
|
|
|
memset((void *)ptbl_bufs, 0, sizeof(struct ptbl_buf) * PTBL_SIZE);
|
|
|
|
memset((void *)kernel_pdir, 0, kernel_ptbls * PTBL_PAGES * PAGE_SIZE);
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/*******************************************************/
|
|
|
|
/* Set the start and end of kva. */
|
|
|
|
/*******************************************************/
|
2009-04-04 22:01:43 +00:00
|
|
|
virtual_avail = round_page(data_end);
|
2008-03-03 17:17:00 +00:00
|
|
|
virtual_end = VM_MAX_KERNEL_ADDRESS;
|
|
|
|
|
|
|
|
/* Allocate KVA space for page zero/copy operations. */
|
|
|
|
zero_page_va = virtual_avail;
|
|
|
|
virtual_avail += PAGE_SIZE;
|
|
|
|
zero_page_idle_va = virtual_avail;
|
|
|
|
virtual_avail += PAGE_SIZE;
|
|
|
|
copy_page_src_va = virtual_avail;
|
|
|
|
virtual_avail += PAGE_SIZE;
|
|
|
|
copy_page_dst_va = virtual_avail;
|
|
|
|
virtual_avail += PAGE_SIZE;
|
2009-01-13 15:41:58 +00:00
|
|
|
debugf("zero_page_va = 0x%08x\n", zero_page_va);
|
|
|
|
debugf("zero_page_idle_va = 0x%08x\n", zero_page_idle_va);
|
|
|
|
debugf("copy_page_src_va = 0x%08x\n", copy_page_src_va);
|
|
|
|
debugf("copy_page_dst_va = 0x%08x\n", copy_page_dst_va);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Initialize page zero/copy mutexes. */
|
|
|
|
mtx_init(&zero_page_mutex, "mmu_booke_zero_page", NULL, MTX_DEF);
|
|
|
|
mtx_init(©_page_mutex, "mmu_booke_copy_page", NULL, MTX_DEF);
|
|
|
|
|
|
|
|
/* Allocate KVA space for ptbl bufs. */
|
|
|
|
ptbl_buf_pool_vabase = virtual_avail;
|
|
|
|
virtual_avail += PTBL_BUFS * PTBL_PAGES * PAGE_SIZE;
|
2009-01-13 15:41:58 +00:00
|
|
|
debugf("ptbl_buf_pool_vabase = 0x%08x end = 0x%08x\n",
|
|
|
|
ptbl_buf_pool_vabase, virtual_avail);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Calculate corresponding physical addresses for the kernel region. */
|
2009-04-04 22:01:43 +00:00
|
|
|
phys_kernelend = kernload + kernsize;
|
2008-03-03 17:17:00 +00:00
|
|
|
debugf("kernel image and allocated data:\n");
|
2015-08-22 07:20:03 +00:00
|
|
|
debugf(" kernload = 0x%09llx\n", (uint64_t)kernload);
|
2009-04-04 22:01:43 +00:00
|
|
|
debugf(" kernstart = 0x%08x\n", kernstart);
|
|
|
|
debugf(" kernsize = 0x%08x\n", kernsize);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
if (sizeof(phys_avail) / sizeof(phys_avail[0]) < availmem_regions_sz)
|
|
|
|
panic("mmu_booke_bootstrap: phys_avail too small");
|
|
|
|
|
|
|
|
/*
|
2009-01-13 16:15:49 +00:00
|
|
|
* Remove kernel physical address range from avail regions list. Page
|
|
|
|
* align all regions. Non-page aligned memory isn't very interesting
|
|
|
|
* to us. Also, sort the entries for ascending addresses.
|
2008-03-03 17:17:00 +00:00
|
|
|
*/
|
2009-05-14 00:34:26 +00:00
|
|
|
|
|
|
|
/* Retrieve phys/avail mem regions */
|
|
|
|
mem_regions(&physmem_regions, &physmem_regions_sz,
|
|
|
|
&availmem_regions, &availmem_regions_sz);
|
2008-03-03 17:17:00 +00:00
|
|
|
sz = 0;
|
|
|
|
cnt = availmem_regions_sz;
|
|
|
|
debugf("processing avail regions:\n");
|
|
|
|
for (mp = availmem_regions; mp->mr_size; mp++) {
|
|
|
|
s = mp->mr_start;
|
|
|
|
e = mp->mr_start + mp->mr_size;
|
2015-12-24 04:30:15 +00:00
|
|
|
debugf(" %09jx-%09jx -> ", (uintmax_t)s, (uintmax_t)e);
|
2008-03-03 17:17:00 +00:00
|
|
|
/* Check whether this region holds all of the kernel. */
|
|
|
|
if (s < kernload && e > phys_kernelend) {
|
|
|
|
availmem_regions[cnt].mr_start = phys_kernelend;
|
|
|
|
availmem_regions[cnt++].mr_size = e - phys_kernelend;
|
|
|
|
e = kernload;
|
|
|
|
}
|
|
|
|
/* Look whether this regions starts within the kernel. */
|
|
|
|
if (s >= kernload && s < phys_kernelend) {
|
|
|
|
if (e <= phys_kernelend)
|
|
|
|
goto empty;
|
|
|
|
s = phys_kernelend;
|
|
|
|
}
|
|
|
|
/* Now look whether this region ends within the kernel. */
|
|
|
|
if (e > kernload && e <= phys_kernelend) {
|
|
|
|
if (s >= kernload)
|
|
|
|
goto empty;
|
|
|
|
e = kernload;
|
|
|
|
}
|
|
|
|
/* Now page align the start and size of the region. */
|
|
|
|
s = round_page(s);
|
|
|
|
e = trunc_page(e);
|
|
|
|
if (e < s)
|
|
|
|
e = s;
|
|
|
|
sz = e - s;
|
2015-12-24 04:30:15 +00:00
|
|
|
debugf("%09jx-%09jx = %jx\n",
|
|
|
|
(uintmax_t)s, (uintmax_t)e, (uintmax_t)sz);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Check whether some memory is left here. */
|
|
|
|
if (sz == 0) {
|
|
|
|
empty:
|
|
|
|
memmove(mp, mp + 1,
|
|
|
|
(cnt - (mp - availmem_regions)) * sizeof(*mp));
|
|
|
|
cnt--;
|
|
|
|
mp--;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Do an insertion sort. */
|
|
|
|
for (mp1 = availmem_regions; mp1 < mp; mp1++)
|
|
|
|
if (s < mp1->mr_start)
|
|
|
|
break;
|
|
|
|
if (mp1 < mp) {
|
|
|
|
memmove(mp1 + 1, mp1, (char *)mp - (char *)mp1);
|
|
|
|
mp1->mr_start = s;
|
|
|
|
mp1->mr_size = sz;
|
|
|
|
} else {
|
|
|
|
mp->mr_start = s;
|
|
|
|
mp->mr_size = sz;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
availmem_regions_sz = cnt;
|
|
|
|
|
2008-08-26 17:07:37 +00:00
|
|
|
/*******************************************************/
|
|
|
|
/* Steal physical memory for kernel stack from the end */
|
|
|
|
/* of the first avail region */
|
|
|
|
/*******************************************************/
|
2015-08-10 17:18:21 +00:00
|
|
|
kstack0_sz = kstack_pages * PAGE_SIZE;
|
2008-08-26 17:07:37 +00:00
|
|
|
kstack0_phys = availmem_regions[0].mr_start +
|
|
|
|
availmem_regions[0].mr_size;
|
|
|
|
kstack0_phys -= kstack0_sz;
|
|
|
|
availmem_regions[0].mr_size -= kstack0_sz;
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/*******************************************************/
|
|
|
|
/* Fill in phys_avail table, based on availmem_regions */
|
|
|
|
/*******************************************************/
|
|
|
|
phys_avail_count = 0;
|
|
|
|
physsz = 0;
|
|
|
|
hwphyssz = 0;
|
|
|
|
TUNABLE_ULONG_FETCH("hw.physmem", (u_long *) &hwphyssz);
|
|
|
|
|
|
|
|
debugf("fill in phys_avail:\n");
|
|
|
|
for (i = 0, j = 0; i < availmem_regions_sz; i++, j += 2) {
|
|
|
|
|
2015-08-28 03:03:09 +00:00
|
|
|
debugf(" region: 0x%jx - 0x%jx (0x%jx)\n",
|
2015-12-24 04:30:15 +00:00
|
|
|
(uintmax_t)availmem_regions[i].mr_start,
|
|
|
|
(uintmax_t)availmem_regions[i].mr_start +
|
2009-01-13 16:15:49 +00:00
|
|
|
availmem_regions[i].mr_size,
|
2015-12-24 04:30:15 +00:00
|
|
|
(uintmax_t)availmem_regions[i].mr_size);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2008-08-28 07:38:08 +00:00
|
|
|
if (hwphyssz != 0 &&
|
|
|
|
(physsz + availmem_regions[i].mr_size) >= hwphyssz) {
|
2008-03-03 17:17:00 +00:00
|
|
|
debugf(" hw.physmem adjust\n");
|
|
|
|
if (physsz < hwphyssz) {
|
|
|
|
phys_avail[j] = availmem_regions[i].mr_start;
|
2008-08-28 07:38:08 +00:00
|
|
|
phys_avail[j + 1] =
|
|
|
|
availmem_regions[i].mr_start +
|
2008-03-03 17:17:00 +00:00
|
|
|
hwphyssz - physsz;
|
|
|
|
physsz = hwphyssz;
|
|
|
|
phys_avail_count++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
phys_avail[j] = availmem_regions[i].mr_start;
|
|
|
|
phys_avail[j + 1] = availmem_regions[i].mr_start +
|
|
|
|
availmem_regions[i].mr_size;
|
|
|
|
phys_avail_count++;
|
|
|
|
physsz += availmem_regions[i].mr_size;
|
|
|
|
}
|
|
|
|
physmem = btoc(physsz);
|
|
|
|
|
|
|
|
/* Calculate the last available physical address. */
|
|
|
|
for (i = 0; phys_avail[i + 2] != 0; i += 2)
|
|
|
|
;
|
|
|
|
Maxmem = powerpc_btop(phys_avail[i + 1]);
|
|
|
|
|
|
|
|
debugf("Maxmem = 0x%08lx\n", Maxmem);
|
|
|
|
debugf("phys_avail_count = %d\n", phys_avail_count);
|
2016-01-26 03:52:14 +00:00
|
|
|
debugf("physsz = 0x%09jx physmem = %jd (0x%09jx)\n",
|
|
|
|
(uintmax_t)physsz, (uintmax_t)physmem, (uintmax_t)physmem);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/*******************************************************/
|
|
|
|
/* Initialize (statically allocated) kernel pmap. */
|
|
|
|
/*******************************************************/
|
|
|
|
PMAP_LOCK_INIT(kernel_pmap);
|
|
|
|
kptbl_min = VM_MIN_KERNEL_ADDRESS / PDIR_SIZE;
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
debugf("kernel_pmap = 0x%08x\n", (uint32_t)kernel_pmap);
|
|
|
|
debugf("kptbl_min = %d, kernel_ptbls = %d\n", kptbl_min, kernel_ptbls);
|
2008-03-03 17:17:00 +00:00
|
|
|
debugf("kernel pdir range: 0x%08x - 0x%08x\n",
|
|
|
|
kptbl_min * PDIR_SIZE, (kptbl_min + kernel_ptbls) * PDIR_SIZE - 1);
|
|
|
|
|
2016-01-19 03:07:25 +00:00
|
|
|
kernel_pte_alloc(data_end, kernstart, kernel_pdir);
|
2009-01-13 15:41:58 +00:00
|
|
|
for (i = 0; i < MAXCPU; i++) {
|
|
|
|
kernel_pmap->pm_tid[i] = TID_KERNEL;
|
|
|
|
|
|
|
|
/* Initialize each CPU's tidbusy entry 0 with kernel_pmap */
|
2015-08-22 07:20:03 +00:00
|
|
|
tidbusy[i][TID_KERNEL] = kernel_pmap;
|
2009-01-13 15:41:58 +00:00
|
|
|
}
|
2009-06-05 09:09:46 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
/* Mark kernel_pmap active on all CPUs */
|
2011-05-09 16:16:15 +00:00
|
|
|
CPU_FILL(&kernel_pmap->pm_active);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2012-11-03 23:22:49 +00:00
|
|
|
/*
|
|
|
|
* Initialize the global pv list lock.
|
|
|
|
*/
|
|
|
|
rw_init(&pvh_global_lock, "pmap pv global");
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/*******************************************************/
|
|
|
|
/* Final setup */
|
|
|
|
/*******************************************************/
|
2009-01-13 15:41:58 +00:00
|
|
|
|
2008-08-26 17:07:37 +00:00
|
|
|
/* Enter kstack0 into kernel map, provide guard page */
|
|
|
|
kstack0 = virtual_avail + KSTACK_GUARD_PAGES * PAGE_SIZE;
|
|
|
|
thread0.td_kstack = kstack0;
|
2015-08-10 17:18:21 +00:00
|
|
|
thread0.td_kstack_pages = kstack_pages;
|
2008-08-26 17:07:37 +00:00
|
|
|
|
|
|
|
debugf("kstack_sz = 0x%08x\n", kstack0_sz);
|
2015-08-22 07:20:03 +00:00
|
|
|
debugf("kstack0_phys at 0x%09llx - 0x%09llx\n",
|
2008-08-26 17:07:37 +00:00
|
|
|
kstack0_phys, kstack0_phys + kstack0_sz);
|
|
|
|
debugf("kstack0 at 0x%08x - 0x%08x\n", kstack0, kstack0 + kstack0_sz);
|
|
|
|
|
|
|
|
virtual_avail += KSTACK_GUARD_PAGES * PAGE_SIZE + kstack0_sz;
|
2015-08-10 17:18:21 +00:00
|
|
|
for (i = 0; i < kstack_pages; i++) {
|
2008-08-26 17:07:37 +00:00
|
|
|
mmu_booke_kenter(mmu, kstack0, kstack0_phys);
|
|
|
|
kstack0 += PAGE_SIZE;
|
|
|
|
kstack0_phys += PAGE_SIZE;
|
|
|
|
}
|
2015-03-01 21:23:23 +00:00
|
|
|
|
|
|
|
pmap_bootstrapped = 1;
|
2009-01-13 15:41:58 +00:00
|
|
|
|
|
|
|
debugf("virtual_avail = %08x\n", virtual_avail);
|
|
|
|
debugf("virtual_end = %08x\n", virtual_end);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
debugf("mmu_booke_bootstrap: exit\n");
|
|
|
|
}
|
|
|
|
|
2015-03-28 21:39:42 +00:00
|
|
|
#ifdef SMP
|
2016-04-19 01:48:18 +00:00
|
|
|
void
|
|
|
|
tlb1_ap_prep(void)
|
|
|
|
{
|
|
|
|
tlb_entry_t *e, tmp;
|
|
|
|
unsigned int i;
|
|
|
|
|
|
|
|
/* Prepare TLB1 image for AP processors */
|
|
|
|
e = __boot_tlb1;
|
|
|
|
for (i = 0; i < TLB1_ENTRIES; i++) {
|
|
|
|
tlb1_read_entry(&tmp, i);
|
|
|
|
|
|
|
|
if ((tmp.mas1 & MAS1_VALID) && (tmp.mas2 & _TLB_ENTRY_SHARED))
|
|
|
|
memcpy(e++, &tmp, sizeof(tmp));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-21 11:43:37 +00:00
|
|
|
void
|
|
|
|
pmap_bootstrap_ap(volatile uint32_t *trcp __unused)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Finish TLB1 configuration: the BSP already set up its TLB1 and we
|
2016-04-19 01:48:18 +00:00
|
|
|
* have the snapshot of its contents in the s/w __boot_tlb1[] table
|
|
|
|
* created by tlb1_ap_prep(), so use these values directly to
|
|
|
|
* (re)program AP's TLB1 hardware.
|
|
|
|
*
|
|
|
|
* Start at index 1 because index 0 has the kernel map.
|
2009-05-21 11:43:37 +00:00
|
|
|
*/
|
2016-04-19 01:48:18 +00:00
|
|
|
for (i = 1; i < TLB1_ENTRIES; i++) {
|
|
|
|
if (__boot_tlb1[i].mas1 & MAS1_VALID)
|
|
|
|
tlb1_write_entry(&__boot_tlb1[i], i);
|
2009-05-21 11:43:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
set_mas4_defaults();
|
|
|
|
}
|
2015-03-28 21:39:42 +00:00
|
|
|
#endif
|
2009-05-21 11:43:37 +00:00
|
|
|
|
2015-08-04 19:46:13 +00:00
|
|
|
static void
|
|
|
|
booke_pmap_init_qpages(void)
|
|
|
|
{
|
|
|
|
struct pcpu *pc;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
CPU_FOREACH(i) {
|
|
|
|
pc = pcpu_find(i);
|
|
|
|
pc->pc_qmap_addr = kva_alloc(PAGE_SIZE);
|
|
|
|
if (pc->pc_qmap_addr == 0)
|
|
|
|
panic("pmap_init_qpages: unable to allocate KVA");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SYSINIT(qpages_init, SI_SUB_CPU, SI_ORDER_ANY, booke_pmap_init_qpages, NULL);
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/*
|
|
|
|
* Get the physical page address for the given pmap/virtual address.
|
|
|
|
*/
|
|
|
|
static vm_paddr_t
|
|
|
|
mmu_booke_extract(mmu_t mmu, pmap_t pmap, vm_offset_t va)
|
|
|
|
{
|
|
|
|
vm_paddr_t pa;
|
|
|
|
|
|
|
|
PMAP_LOCK(pmap);
|
|
|
|
pa = pte_vatopa(mmu, pmap, va);
|
|
|
|
PMAP_UNLOCK(pmap);
|
|
|
|
|
|
|
|
return (pa);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Extract the physical page address associated with the given
|
|
|
|
* kernel virtual address.
|
|
|
|
*/
|
|
|
|
static vm_paddr_t
|
|
|
|
mmu_booke_kextract(mmu_t mmu, vm_offset_t va)
|
|
|
|
{
|
2016-04-19 01:48:18 +00:00
|
|
|
tlb_entry_t e;
|
2013-10-26 16:49:41 +00:00
|
|
|
int i;
|
|
|
|
|
|
|
|
/* Check TLB1 mappings */
|
2016-04-19 01:48:18 +00:00
|
|
|
for (i = 0; i < TLB1_ENTRIES; i++) {
|
|
|
|
tlb1_read_entry(&e, i);
|
|
|
|
if (!(e.mas1 & MAS1_VALID))
|
2013-10-26 16:49:41 +00:00
|
|
|
continue;
|
2016-04-19 01:48:18 +00:00
|
|
|
if (va >= e.virt && va < e.virt + e.size)
|
|
|
|
return (e.phys + (va - e.virt));
|
2013-10-26 16:49:41 +00:00
|
|
|
}
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
return (pte_vatopa(mmu, kernel_pmap, va));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize the pmap module.
|
|
|
|
* Called by vm_init, to initialize any structures that the pmap
|
|
|
|
* system needs to map virtual memory.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_init(mmu_t mmu)
|
|
|
|
{
|
|
|
|
int shpgperproc = PMAP_SHPGPERPROC;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize the address space (zone) for the pv entries. Set a
|
|
|
|
* high water mark so that the system can recover from excessive
|
|
|
|
* numbers of pv entries.
|
|
|
|
*/
|
|
|
|
pvzone = uma_zcreate("PV ENTRY", sizeof(struct pv_entry), NULL, NULL,
|
|
|
|
NULL, NULL, UMA_ALIGN_PTR, UMA_ZONE_VM | UMA_ZONE_NOFREE);
|
|
|
|
|
|
|
|
TUNABLE_INT_FETCH("vm.pmap.shpgperproc", &shpgperproc);
|
2014-03-22 10:26:09 +00:00
|
|
|
pv_entry_max = shpgperproc * maxproc + vm_cnt.v_page_count;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
TUNABLE_INT_FETCH("vm.pmap.pv_entries", &pv_entry_max);
|
|
|
|
pv_entry_high_water = 9 * (pv_entry_max / 10);
|
|
|
|
|
2013-02-26 23:52:23 +00:00
|
|
|
uma_zone_reserve_kva(pvzone, pv_entry_max);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Pre-fill pvzone with initial number of pv entries. */
|
|
|
|
uma_prealloc(pvzone, PV_ENTRY_ZONE_MIN);
|
|
|
|
|
|
|
|
/* Initialize ptbl allocation. */
|
|
|
|
ptbl_init();
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_qenter(mmu_t mmu, vm_offset_t sva, vm_page_t *m, int count)
|
|
|
|
{
|
|
|
|
vm_offset_t va;
|
|
|
|
|
|
|
|
va = sva;
|
|
|
|
while (count-- > 0) {
|
|
|
|
mmu_booke_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 mmu_booke_qenter.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_qremove(mmu_t mmu, vm_offset_t sva, int count)
|
|
|
|
{
|
|
|
|
vm_offset_t va;
|
|
|
|
|
|
|
|
va = sva;
|
|
|
|
while (count-- > 0) {
|
|
|
|
mmu_booke_kremove(mmu, va);
|
|
|
|
va += PAGE_SIZE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Map a wired page into kernel virtual address space.
|
|
|
|
*/
|
|
|
|
static void
|
2012-05-24 21:13:24 +00:00
|
|
|
mmu_booke_kenter(mmu_t mmu, vm_offset_t va, vm_paddr_t pa)
|
2013-10-26 14:52:55 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
mmu_booke_kenter_attr(mmu, va, pa, VM_MEMATTR_DEFAULT);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
mmu_booke_kenter_attr(mmu_t mmu, vm_offset_t va, vm_paddr_t pa, vm_memattr_t ma)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
2009-01-13 16:15:49 +00:00
|
|
|
uint32_t flags;
|
2008-03-03 17:17:00 +00:00
|
|
|
pte_t *pte;
|
|
|
|
|
2009-01-13 16:15:49 +00:00
|
|
|
KASSERT(((va >= VM_MIN_KERNEL_ADDRESS) &&
|
|
|
|
(va <= VM_MAX_KERNEL_ADDRESS)), ("mmu_booke_kenter: invalid va"));
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2013-10-26 14:52:55 +00:00
|
|
|
flags = PTE_SR | PTE_SW | PTE_SX | PTE_WIRED | PTE_VALID;
|
2016-02-11 13:15:37 +00:00
|
|
|
flags |= tlb_calc_wimg(pa, ma) << PTE_MAS2_SHIFT;
|
|
|
|
flags |= PTE_PS_4KB;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2016-01-19 03:07:25 +00:00
|
|
|
pte = pte_find(mmu, kernel_pmap, va);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_lock_spin(&tlbivax_mutex);
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_lock();
|
2009-01-13 15:41:58 +00:00
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
if (PTE_ISVALID(pte)) {
|
2009-01-13 15:41:58 +00:00
|
|
|
|
|
|
|
CTR1(KTR_PMAP, "%s: replacing entry!", __func__);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Flush entry from TLB0 */
|
2009-01-13 15:41:58 +00:00
|
|
|
tlb0_flush_entry(va);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
2016-02-11 13:15:37 +00:00
|
|
|
*pte = PTE_RPN_FROM_PA(pa) | flags;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
//debugf("mmu_booke_kenter: pdir_idx = %d ptbl_idx = %d va=0x%08x "
|
|
|
|
// "pa=0x%08x rpn=0x%08x flags=0x%08x\n",
|
|
|
|
// pdir_idx, ptbl_idx, va, pa, pte->rpn, pte->flags);
|
|
|
|
|
|
|
|
/* Flush the real memory from the instruction cache. */
|
2016-02-11 13:15:37 +00:00
|
|
|
if ((flags & (PTE_I | PTE_G)) == 0)
|
2008-03-03 17:17:00 +00:00
|
|
|
__syncicache((void *)va, PAGE_SIZE);
|
|
|
|
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_unlock();
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_unlock_spin(&tlbivax_mutex);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Remove a page from kernel page table.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_kremove(mmu_t mmu, vm_offset_t va)
|
|
|
|
{
|
|
|
|
pte_t *pte;
|
|
|
|
|
2016-01-19 03:07:25 +00:00
|
|
|
CTR2(KTR_PMAP,"%s: s (va = 0x%08x)\n", __func__, va);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
KASSERT(((va >= VM_MIN_KERNEL_ADDRESS) &&
|
|
|
|
(va <= VM_MAX_KERNEL_ADDRESS)),
|
2008-03-03 17:17:00 +00:00
|
|
|
("mmu_booke_kremove: invalid va"));
|
|
|
|
|
2016-01-19 03:07:25 +00:00
|
|
|
pte = pte_find(mmu, kernel_pmap, va);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
if (!PTE_ISVALID(pte)) {
|
2009-01-13 15:41:58 +00:00
|
|
|
|
|
|
|
CTR1(KTR_PMAP, "%s: invalid pte", __func__);
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_lock_spin(&tlbivax_mutex);
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_lock();
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
/* Invalidate entry in TLB0, update PTE. */
|
|
|
|
tlb0_flush_entry(va);
|
2016-02-11 13:15:37 +00:00
|
|
|
*pte = 0;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_unlock();
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_unlock_spin(&tlbivax_mutex);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize pmap associated with process 0.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_pinit0(mmu_t mmu, pmap_t pmap)
|
|
|
|
{
|
2009-01-13 16:15:49 +00:00
|
|
|
|
2013-08-22 18:12:24 +00:00
|
|
|
PMAP_LOCK_INIT(pmap);
|
2008-03-03 17:17:00 +00:00
|
|
|
mmu_booke_pinit(mmu, pmap);
|
|
|
|
PCPU_SET(curpmap, pmap);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize a preallocated and zeroed pmap structure,
|
|
|
|
* such as one in a vmspace structure.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_pinit(mmu_t mmu, pmap_t pmap)
|
|
|
|
{
|
2009-01-13 15:41:58 +00:00
|
|
|
int i;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
CTR4(KTR_PMAP, "%s: pmap = %p, proc %d '%s'", __func__, pmap,
|
|
|
|
curthread->td_proc->p_pid, curthread->td_proc->p_comm);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
KASSERT((pmap != kernel_pmap), ("pmap_pinit: initializing kernel_pmap"));
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
for (i = 0; i < MAXCPU; i++)
|
|
|
|
pmap->pm_tid[i] = TID_NONE;
|
2011-05-09 16:16:15 +00:00
|
|
|
CPU_ZERO(&kernel_pmap->pm_active);
|
2008-03-03 17:17:00 +00:00
|
|
|
bzero(&pmap->pm_stats, sizeof(pmap->pm_stats));
|
|
|
|
bzero(&pmap->pm_pdir, sizeof(pte_t *) * PDIR_NENTRIES);
|
2009-01-13 15:41:58 +00:00
|
|
|
TAILQ_INIT(&pmap->pm_ptbl_list);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Release any resources held by the given physical map.
|
|
|
|
* Called when a pmap initialized by mmu_booke_pinit is being released.
|
|
|
|
* Should only be called if the map contains no valid mappings.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_release(mmu_t mmu, pmap_t pmap)
|
|
|
|
{
|
|
|
|
|
2009-01-13 16:15:49 +00:00
|
|
|
KASSERT(pmap->pm_stats.resident_count == 0,
|
|
|
|
("pmap_release: pmap resident count %ld != 0",
|
|
|
|
pmap->pm_stats.resident_count));
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Insert the given physical page at the specified virtual address in the
|
|
|
|
* target physical map with the protection requested. If specified the page
|
|
|
|
* will be wired down.
|
|
|
|
*/
|
2014-08-08 17:12:03 +00:00
|
|
|
static int
|
2008-03-03 17:17:00 +00:00
|
|
|
mmu_booke_enter(mmu_t mmu, pmap_t pmap, vm_offset_t va, vm_page_t m,
|
2014-08-08 17:12:03 +00:00
|
|
|
vm_prot_t prot, u_int flags, int8_t psind)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
2014-08-08 17:12:03 +00:00
|
|
|
int error;
|
2009-01-13 16:15:49 +00:00
|
|
|
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
PMAP_LOCK(pmap);
|
2014-08-08 17:12:03 +00:00
|
|
|
error = mmu_booke_enter_locked(mmu, pmap, va, m, prot, flags, psind);
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wunlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
PMAP_UNLOCK(pmap);
|
2014-08-08 17:12:03 +00:00
|
|
|
return (error);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
2014-08-08 17:12:03 +00:00
|
|
|
static int
|
2008-03-03 17:17:00 +00:00
|
|
|
mmu_booke_enter_locked(mmu_t mmu, pmap_t pmap, vm_offset_t va, vm_page_t m,
|
2014-08-08 17:12:03 +00:00
|
|
|
vm_prot_t prot, u_int pmap_flags, int8_t psind __unused)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
|
|
|
pte_t *pte;
|
|
|
|
vm_paddr_t pa;
|
2009-01-13 16:15:49 +00:00
|
|
|
uint32_t flags;
|
2014-08-08 17:12:03 +00:00
|
|
|
int error, su, sync;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
pa = VM_PAGE_TO_PHYS(m);
|
|
|
|
su = (pmap == kernel_pmap);
|
|
|
|
sync = 0;
|
|
|
|
|
|
|
|
//debugf("mmu_booke_enter_locked: s (pmap=0x%08x su=%d tid=%d m=0x%08x va=0x%08x "
|
2014-08-08 17:12:03 +00:00
|
|
|
// "pa=0x%08x prot=0x%08x flags=%#x)\n",
|
2008-03-03 17:17:00 +00:00
|
|
|
// (u_int32_t)pmap, su, pmap->pm_tid,
|
2014-08-08 17:12:03 +00:00
|
|
|
// (u_int32_t)m, va, pa, prot, flags);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
if (su) {
|
2009-01-13 16:15:49 +00:00
|
|
|
KASSERT(((va >= virtual_avail) &&
|
|
|
|
(va <= VM_MAX_KERNEL_ADDRESS)),
|
|
|
|
("mmu_booke_enter_locked: kernel pmap, non kernel va"));
|
2008-03-03 17:17:00 +00:00
|
|
|
} else {
|
|
|
|
KASSERT((va <= VM_MAXUSER_ADDRESS),
|
2009-01-13 16:15:49 +00:00
|
|
|
("mmu_booke_enter_locked: user pmap, non user va"));
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
2013-08-09 11:11:11 +00:00
|
|
|
if ((m->oflags & VPO_UNMANAGED) == 0 && !vm_page_xbusied(m))
|
2013-05-17 18:59:00 +00:00
|
|
|
VM_OBJECT_ASSERT_LOCKED(m->object);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
PMAP_LOCK_ASSERT(pmap, MA_OWNED);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If there is an existing mapping, and the physical address has not
|
|
|
|
* changed, must be protection or wiring change.
|
|
|
|
*/
|
|
|
|
if (((pte = pte_find(mmu, pmap, va)) != NULL) &&
|
|
|
|
(PTE_ISVALID(pte)) && (PTE_PA(pte) == pa)) {
|
2009-01-13 15:41:58 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Before actually updating pte->flags we calculate and
|
|
|
|
* prepare its new value in a helper var.
|
|
|
|
*/
|
2016-02-11 13:15:37 +00:00
|
|
|
flags = *pte;
|
2009-01-13 15:41:58 +00:00
|
|
|
flags &= ~(PTE_UW | PTE_UX | PTE_SW | PTE_SX | PTE_MODIFIED);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Wiring change, just update stats. */
|
2014-08-08 17:12:03 +00:00
|
|
|
if ((pmap_flags & PMAP_ENTER_WIRED) != 0) {
|
2008-03-03 17:17:00 +00:00
|
|
|
if (!PTE_ISWIRED(pte)) {
|
2009-01-13 15:41:58 +00:00
|
|
|
flags |= PTE_WIRED;
|
2008-03-03 17:17:00 +00:00
|
|
|
pmap->pm_stats.wired_count++;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (PTE_ISWIRED(pte)) {
|
2009-01-13 15:41:58 +00:00
|
|
|
flags &= ~PTE_WIRED;
|
2008-03-03 17:17:00 +00:00
|
|
|
pmap->pm_stats.wired_count--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (prot & VM_PROT_WRITE) {
|
|
|
|
/* Add write permissions. */
|
2009-01-13 15:41:58 +00:00
|
|
|
flags |= PTE_SW;
|
2008-03-03 17:17:00 +00:00
|
|
|
if (!su)
|
2009-01-13 15:41:58 +00:00
|
|
|
flags |= PTE_UW;
|
2009-05-26 06:24:50 +00:00
|
|
|
|
2010-06-05 18:20:09 +00:00
|
|
|
if ((flags & PTE_MANAGED) != 0)
|
2011-09-06 10:30:11 +00:00
|
|
|
vm_page_aflag_set(m, PGA_WRITEABLE);
|
2008-03-03 17:17:00 +00:00
|
|
|
} else {
|
|
|
|
/* Handle modified pages, sense modify status. */
|
2009-01-13 15:41:58 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The PTE_MODIFIED flag could be set by underlying
|
|
|
|
* TLB misses since we last read it (above), possibly
|
|
|
|
* other CPUs could update it so we check in the PTE
|
|
|
|
* directly rather than rely on that saved local flags
|
|
|
|
* copy.
|
|
|
|
*/
|
2008-04-27 21:04:54 +00:00
|
|
|
if (PTE_ISMODIFIED(pte))
|
|
|
|
vm_page_dirty(m);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (prot & VM_PROT_EXECUTE) {
|
2009-01-13 15:41:58 +00:00
|
|
|
flags |= PTE_SX;
|
2008-03-03 17:17:00 +00:00
|
|
|
if (!su)
|
2009-01-13 15:41:58 +00:00
|
|
|
flags |= PTE_UX;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
/*
|
|
|
|
* Check existing flags for execute permissions: if we
|
|
|
|
* are turning execute permissions on, icache should
|
|
|
|
* be flushed.
|
|
|
|
*/
|
2016-02-11 13:15:37 +00:00
|
|
|
if ((*pte & (PTE_UX | PTE_SX)) == 0)
|
2008-03-03 17:17:00 +00:00
|
|
|
sync++;
|
|
|
|
}
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
flags &= ~PTE_REFERENCED;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The new flags value is all calculated -- only now actually
|
|
|
|
* update the PTE.
|
|
|
|
*/
|
|
|
|
mtx_lock_spin(&tlbivax_mutex);
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_lock();
|
2009-01-13 15:41:58 +00:00
|
|
|
|
|
|
|
tlb0_flush_entry(va);
|
2016-02-16 02:13:55 +00:00
|
|
|
*pte &= ~PTE_FLAGS_MASK;
|
|
|
|
*pte |= flags;
|
2009-01-13 15:41:58 +00:00
|
|
|
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_unlock();
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_unlock_spin(&tlbivax_mutex);
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
} else {
|
|
|
|
/*
|
2009-01-13 15:41:58 +00:00
|
|
|
* If there is an existing mapping, but it's for a different
|
2008-03-03 17:17:00 +00:00
|
|
|
* physical address, pte_enter() will delete the old mapping.
|
|
|
|
*/
|
|
|
|
//if ((pte != NULL) && PTE_ISVALID(pte))
|
|
|
|
// debugf("mmu_booke_enter_locked: replace\n");
|
|
|
|
//else
|
|
|
|
// debugf("mmu_booke_enter_locked: new\n");
|
|
|
|
|
|
|
|
/* Now set up the flags and install the new mapping. */
|
|
|
|
flags = (PTE_SR | PTE_VALID);
|
2009-01-13 15:41:58 +00:00
|
|
|
flags |= PTE_M;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
if (!su)
|
|
|
|
flags |= PTE_UR;
|
|
|
|
|
|
|
|
if (prot & VM_PROT_WRITE) {
|
|
|
|
flags |= PTE_SW;
|
|
|
|
if (!su)
|
|
|
|
flags |= PTE_UW;
|
2009-05-26 06:24:50 +00:00
|
|
|
|
2011-08-09 21:01:36 +00:00
|
|
|
if ((m->oflags & VPO_UNMANAGED) == 0)
|
2011-09-06 10:30:11 +00:00
|
|
|
vm_page_aflag_set(m, PGA_WRITEABLE);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (prot & VM_PROT_EXECUTE) {
|
|
|
|
flags |= PTE_SX;
|
|
|
|
if (!su)
|
|
|
|
flags |= PTE_UX;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If its wired update stats. */
|
2014-08-08 17:12:03 +00:00
|
|
|
if ((pmap_flags & PMAP_ENTER_WIRED) != 0)
|
2008-03-03 17:17:00 +00:00
|
|
|
flags |= PTE_WIRED;
|
|
|
|
|
2014-08-08 17:12:03 +00:00
|
|
|
error = pte_enter(mmu, pmap, m, va, flags,
|
|
|
|
(pmap_flags & PMAP_ENTER_NOSLEEP) != 0);
|
|
|
|
if (error != 0)
|
|
|
|
return (KERN_RESOURCE_SHORTAGE);
|
|
|
|
|
|
|
|
if ((flags & PMAP_ENTER_WIRED) != 0)
|
|
|
|
pmap->pm_stats.wired_count++;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Flush the real memory from the instruction cache. */
|
|
|
|
if (prot & VM_PROT_EXECUTE)
|
|
|
|
sync++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sync && (su || pmap == PCPU_GET(curpmap))) {
|
|
|
|
__syncicache((void *)va, PAGE_SIZE);
|
|
|
|
sync = 0;
|
|
|
|
}
|
2014-08-08 17:12:03 +00:00
|
|
|
|
|
|
|
return (KERN_SUCCESS);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_enter_object(mmu_t mmu, pmap_t pmap, 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;
|
|
|
|
|
2013-05-21 20:38:19 +00:00
|
|
|
VM_OBJECT_ASSERT_LOCKED(m_start->object);
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
psize = atop(end - start);
|
|
|
|
m = m_start;
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
PMAP_LOCK(pmap);
|
|
|
|
while (m != NULL && (diff = m->pindex - m_start->pindex) < psize) {
|
2009-01-13 16:15:49 +00:00
|
|
|
mmu_booke_enter_locked(mmu, pmap, start + ptoa(diff), m,
|
2014-08-08 17:12:03 +00:00
|
|
|
prot & (VM_PROT_READ | VM_PROT_EXECUTE),
|
|
|
|
PMAP_ENTER_NOSLEEP, 0);
|
2008-03-03 17:17:00 +00:00
|
|
|
m = TAILQ_NEXT(m, listq);
|
|
|
|
}
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wunlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
PMAP_UNLOCK(pmap);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
mmu_booke_enter_quick(mmu_t mmu, pmap_t pmap, vm_offset_t va, vm_page_t m,
|
|
|
|
vm_prot_t prot)
|
|
|
|
{
|
|
|
|
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
PMAP_LOCK(pmap);
|
|
|
|
mmu_booke_enter_locked(mmu, pmap, va, m,
|
2014-08-08 17:12:03 +00:00
|
|
|
prot & (VM_PROT_READ | VM_PROT_EXECUTE), PMAP_ENTER_NOSLEEP,
|
|
|
|
0);
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wunlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
PMAP_UNLOCK(pmap);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Remove the given range of addresses from the specified map.
|
|
|
|
*
|
|
|
|
* It is assumed that the start and end are properly rounded to the page size.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_remove(mmu_t mmu, pmap_t pmap, vm_offset_t va, vm_offset_t endva)
|
|
|
|
{
|
|
|
|
pte_t *pte;
|
2009-01-13 16:15:49 +00:00
|
|
|
uint8_t hold_flag;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
int su = (pmap == kernel_pmap);
|
|
|
|
|
|
|
|
//debugf("mmu_booke_remove: s (su = %d pmap=0x%08x tid=%d va=0x%08x endva=0x%08x)\n",
|
|
|
|
// su, (u_int32_t)pmap, pmap->pm_tid, va, endva);
|
|
|
|
|
|
|
|
if (su) {
|
2009-01-13 16:15:49 +00:00
|
|
|
KASSERT(((va >= virtual_avail) &&
|
|
|
|
(va <= VM_MAX_KERNEL_ADDRESS)),
|
|
|
|
("mmu_booke_remove: kernel pmap, non kernel va"));
|
2008-03-03 17:17:00 +00:00
|
|
|
} else {
|
|
|
|
KASSERT((va <= VM_MAXUSER_ADDRESS),
|
2009-01-13 16:15:49 +00:00
|
|
|
("mmu_booke_remove: user pmap, non user va"));
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (PMAP_REMOVE_DONE(pmap)) {
|
|
|
|
//debugf("mmu_booke_remove: e (empty)\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
hold_flag = PTBL_HOLD_FLAG(pmap);
|
|
|
|
//debugf("mmu_booke_remove: hold_flag = %d\n", hold_flag);
|
|
|
|
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
PMAP_LOCK(pmap);
|
|
|
|
for (; va < endva; va += PAGE_SIZE) {
|
|
|
|
pte = pte_find(mmu, pmap, va);
|
2009-01-13 15:41:58 +00:00
|
|
|
if ((pte != NULL) && PTE_ISVALID(pte))
|
2008-03-03 17:17:00 +00:00
|
|
|
pte_remove(mmu, pmap, va, hold_flag);
|
|
|
|
}
|
|
|
|
PMAP_UNLOCK(pmap);
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wunlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
//debugf("mmu_booke_remove: e\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Remove physical page from all pmaps in which it resides.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_remove_all(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
pv_entry_t pv, pvn;
|
2009-01-13 16:15:49 +00:00
|
|
|
uint8_t hold_flag;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
for (pv = TAILQ_FIRST(&m->md.pv_list); pv != NULL; pv = pvn) {
|
|
|
|
pvn = TAILQ_NEXT(pv, pv_link);
|
|
|
|
|
|
|
|
PMAP_LOCK(pv->pv_pmap);
|
|
|
|
hold_flag = PTBL_HOLD_FLAG(pv->pv_pmap);
|
|
|
|
pte_remove(mmu, pv->pv_pmap, pv->pv_va, hold_flag);
|
|
|
|
PMAP_UNLOCK(pv->pv_pmap);
|
|
|
|
}
|
2011-09-06 10:30:11 +00:00
|
|
|
vm_page_aflag_clear(m, PGA_WRITEABLE);
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wunlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Map a range of physical addresses into kernel virtual address space.
|
|
|
|
*/
|
|
|
|
static vm_offset_t
|
2012-05-24 21:13:24 +00:00
|
|
|
mmu_booke_map(mmu_t mmu, vm_offset_t *virt, vm_paddr_t pa_start,
|
|
|
|
vm_paddr_t pa_end, int prot)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
|
|
|
vm_offset_t sva = *virt;
|
|
|
|
vm_offset_t va = sva;
|
|
|
|
|
|
|
|
//debugf("mmu_booke_map: s (sva = 0x%08x pa_start = 0x%08x pa_end = 0x%08x)\n",
|
|
|
|
// sva, pa_start, pa_end);
|
|
|
|
|
|
|
|
while (pa_start < pa_end) {
|
|
|
|
mmu_booke_kenter(mmu, va, pa_start);
|
|
|
|
va += PAGE_SIZE;
|
|
|
|
pa_start += PAGE_SIZE;
|
|
|
|
}
|
|
|
|
*virt = va;
|
|
|
|
|
|
|
|
//debugf("mmu_booke_map: e (va = 0x%08x)\n", va);
|
|
|
|
return (sva);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The pmap must be activated before it's address space can be accessed in any
|
|
|
|
* way.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_activate(mmu_t mmu, struct thread *td)
|
|
|
|
{
|
|
|
|
pmap_t pmap;
|
2011-07-04 12:04:52 +00:00
|
|
|
u_int cpuid;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
pmap = &td->td_proc->p_vmspace->vm_pmap;
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
CTR5(KTR_PMAP, "%s: s (td = %p, proc = '%s', id = %d, pmap = 0x%08x)",
|
|
|
|
__func__, td, td->td_proc->p_comm, td->td_proc->p_pid, pmap);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
KASSERT((pmap != kernel_pmap), ("mmu_booke_activate: kernel_pmap!"));
|
|
|
|
|
2013-11-11 17:37:50 +00:00
|
|
|
sched_pin();
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2011-07-04 12:04:52 +00:00
|
|
|
cpuid = PCPU_GET(cpuid);
|
|
|
|
CPU_SET_ATOMIC(cpuid, &pmap->pm_active);
|
2008-03-03 17:17:00 +00:00
|
|
|
PCPU_SET(curpmap, pmap);
|
2009-01-13 15:41:58 +00:00
|
|
|
|
2011-07-04 12:04:52 +00:00
|
|
|
if (pmap->pm_tid[cpuid] == TID_NONE)
|
2008-03-03 17:17:00 +00:00
|
|
|
tid_alloc(pmap);
|
|
|
|
|
|
|
|
/* Load PID0 register with pmap tid value. */
|
2011-07-04 12:04:52 +00:00
|
|
|
mtspr(SPR_PID0, pmap->pm_tid[cpuid]);
|
2009-01-13 15:41:58 +00:00
|
|
|
__asm __volatile("isync");
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2015-03-04 16:45:31 +00:00
|
|
|
mtspr(SPR_DBCR0, td->td_pcb->pcb_cpu.booke.dbcr0);
|
|
|
|
|
2013-11-11 17:37:50 +00:00
|
|
|
sched_unpin();
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
CTR3(KTR_PMAP, "%s: e (tid = %d for '%s')", __func__,
|
|
|
|
pmap->pm_tid[PCPU_GET(cpuid)], td->td_proc->p_comm);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Deactivate the specified process's address space.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_deactivate(mmu_t mmu, struct thread *td)
|
|
|
|
{
|
|
|
|
pmap_t pmap;
|
|
|
|
|
|
|
|
pmap = &td->td_proc->p_vmspace->vm_pmap;
|
2009-01-13 15:41:58 +00:00
|
|
|
|
|
|
|
CTR5(KTR_PMAP, "%s: td=%p, proc = '%s', id = %d, pmap = 0x%08x",
|
|
|
|
__func__, td, td->td_proc->p_comm, td->td_proc->p_pid, pmap);
|
|
|
|
|
2015-03-04 16:45:31 +00:00
|
|
|
td->td_pcb->pcb_cpu.booke.dbcr0 = mfspr(SPR_DBCR0);
|
|
|
|
|
2011-07-04 12:04:52 +00:00
|
|
|
CPU_CLR_ATOMIC(PCPU_GET(cpuid), &pmap->pm_active);
|
2008-03-03 17:17:00 +00:00
|
|
|
PCPU_SET(curpmap, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copy the range specified by src_addr/len
|
|
|
|
* from the source map to the range dst_addr/len
|
|
|
|
* in the destination map.
|
|
|
|
*
|
|
|
|
* This routine is only advisory and need not do anything.
|
|
|
|
*/
|
|
|
|
static void
|
2009-06-13 08:57:04 +00:00
|
|
|
mmu_booke_copy(mmu_t mmu, pmap_t dst_pmap, pmap_t src_pmap,
|
|
|
|
vm_offset_t dst_addr, vm_size_t len, vm_offset_t src_addr)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the physical protection on the specified range of this map as requested.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_protect(mmu_t mmu, pmap_t pmap, vm_offset_t sva, vm_offset_t eva,
|
|
|
|
vm_prot_t prot)
|
|
|
|
{
|
|
|
|
vm_offset_t va;
|
|
|
|
vm_page_t m;
|
|
|
|
pte_t *pte;
|
|
|
|
|
|
|
|
if ((prot & VM_PROT_READ) == VM_PROT_NONE) {
|
|
|
|
mmu_booke_remove(mmu, pmap, sva, eva);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (prot & VM_PROT_WRITE)
|
|
|
|
return;
|
|
|
|
|
|
|
|
PMAP_LOCK(pmap);
|
|
|
|
for (va = sva; va < eva; va += PAGE_SIZE) {
|
|
|
|
if ((pte = pte_find(mmu, pmap, va)) != NULL) {
|
|
|
|
if (PTE_ISVALID(pte)) {
|
|
|
|
m = PHYS_TO_VM_PAGE(PTE_PA(pte));
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_lock_spin(&tlbivax_mutex);
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_lock();
|
2009-01-13 15:41:58 +00:00
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/* Handle modified pages. */
|
2010-04-30 15:22:52 +00:00
|
|
|
if (PTE_ISMODIFIED(pte) && PTE_ISMANAGED(pte))
|
2008-04-27 21:04:54 +00:00
|
|
|
vm_page_dirty(m);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
tlb0_flush_entry(va);
|
2016-02-11 13:15:37 +00:00
|
|
|
*pte &= ~(PTE_UW | PTE_SW | PTE_MODIFIED);
|
2009-01-13 15:41:58 +00:00
|
|
|
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_unlock();
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_unlock_spin(&tlbivax_mutex);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
PMAP_UNLOCK(pmap);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Clear the write and modified bits in each of the given page's mappings.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_remove_write(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
pv_entry_t pv;
|
|
|
|
pte_t *pte;
|
|
|
|
|
2011-08-09 21:01:36 +00:00
|
|
|
KASSERT((m->oflags & VPO_UNMANAGED) == 0,
|
2010-05-16 23:45:10 +00:00
|
|
|
("mmu_booke_remove_write: page %p is not managed", m));
|
|
|
|
|
|
|
|
/*
|
2013-08-09 11:11:11 +00:00
|
|
|
* If the page is not exclusive busied, then PGA_WRITEABLE cannot be
|
|
|
|
* set by another thread while the object is locked. Thus,
|
|
|
|
* if PGA_WRITEABLE is clear, no page table entries need updating.
|
2010-05-16 23:45:10 +00:00
|
|
|
*/
|
2013-02-21 21:54:53 +00:00
|
|
|
VM_OBJECT_ASSERT_WLOCKED(m->object);
|
2013-08-09 11:11:11 +00:00
|
|
|
if (!vm_page_xbusied(m) && (m->aflags & PGA_WRITEABLE) == 0)
|
2008-03-03 17:17:00 +00:00
|
|
|
return;
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
TAILQ_FOREACH(pv, &m->md.pv_list, pv_link) {
|
|
|
|
PMAP_LOCK(pv->pv_pmap);
|
|
|
|
if ((pte = pte_find(mmu, pv->pv_pmap, pv->pv_va)) != NULL) {
|
|
|
|
if (PTE_ISVALID(pte)) {
|
|
|
|
m = PHYS_TO_VM_PAGE(PTE_PA(pte));
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_lock_spin(&tlbivax_mutex);
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_lock();
|
2009-01-13 15:41:58 +00:00
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/* Handle modified pages. */
|
2008-04-27 21:04:54 +00:00
|
|
|
if (PTE_ISMODIFIED(pte))
|
|
|
|
vm_page_dirty(m);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Flush mapping from TLB0. */
|
2016-02-11 13:15:37 +00:00
|
|
|
*pte &= ~(PTE_UW | PTE_SW | PTE_MODIFIED);
|
2009-01-13 15:41:58 +00:00
|
|
|
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_unlock();
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_unlock_spin(&tlbivax_mutex);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
PMAP_UNLOCK(pv->pv_pmap);
|
|
|
|
}
|
2011-09-06 10:30:11 +00:00
|
|
|
vm_page_aflag_clear(m, PGA_WRITEABLE);
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wunlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
2009-10-21 18:38:02 +00:00
|
|
|
static void
|
|
|
|
mmu_booke_sync_icache(mmu_t mmu, pmap_t pm, vm_offset_t va, vm_size_t sz)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
|
|
|
pte_t *pte;
|
2009-10-21 18:38:02 +00:00
|
|
|
pmap_t pmap;
|
|
|
|
vm_page_t m;
|
|
|
|
vm_offset_t addr;
|
2014-02-01 20:06:52 +00:00
|
|
|
vm_paddr_t pa = 0;
|
2009-10-21 18:38:02 +00:00
|
|
|
int active, valid;
|
|
|
|
|
|
|
|
va = trunc_page(va);
|
|
|
|
sz = round_page(sz);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wlock(&pvh_global_lock);
|
2009-10-21 18:38:02 +00:00
|
|
|
pmap = PCPU_GET(curpmap);
|
|
|
|
active = (pm == kernel_pmap || pm == pmap) ? 1 : 0;
|
|
|
|
while (sz > 0) {
|
|
|
|
PMAP_LOCK(pm);
|
|
|
|
pte = pte_find(mmu, pm, va);
|
|
|
|
valid = (pte != NULL && PTE_ISVALID(pte)) ? 1 : 0;
|
|
|
|
if (valid)
|
|
|
|
pa = PTE_PA(pte);
|
|
|
|
PMAP_UNLOCK(pm);
|
|
|
|
if (valid) {
|
|
|
|
if (!active) {
|
|
|
|
/* Create a mapping in the active pmap. */
|
|
|
|
addr = 0;
|
|
|
|
m = PHYS_TO_VM_PAGE(pa);
|
|
|
|
PMAP_LOCK(pmap);
|
|
|
|
pte_enter(mmu, pmap, m, addr,
|
2014-08-08 17:12:03 +00:00
|
|
|
PTE_SR | PTE_VALID | PTE_UR, FALSE);
|
2009-10-21 18:38:02 +00:00
|
|
|
__syncicache((void *)addr, PAGE_SIZE);
|
|
|
|
pte_remove(mmu, pmap, addr, PTBL_UNHOLD);
|
|
|
|
PMAP_UNLOCK(pmap);
|
|
|
|
} else
|
|
|
|
__syncicache((void *)va, PAGE_SIZE);
|
|
|
|
}
|
|
|
|
va += PAGE_SIZE;
|
|
|
|
sz -= PAGE_SIZE;
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wunlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Atomically extract and hold the physical page with the given
|
|
|
|
* pmap and virtual address pair if that mapping permits the given
|
|
|
|
* protection.
|
|
|
|
*/
|
|
|
|
static vm_page_t
|
|
|
|
mmu_booke_extract_and_hold(mmu_t mmu, pmap_t pmap, vm_offset_t va,
|
|
|
|
vm_prot_t prot)
|
|
|
|
{
|
|
|
|
pte_t *pte;
|
|
|
|
vm_page_t m;
|
2009-01-13 16:15:49 +00:00
|
|
|
uint32_t pte_wbit;
|
2010-04-30 00:46:43 +00:00
|
|
|
vm_paddr_t pa;
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
m = NULL;
|
2010-04-30 00:46:43 +00:00
|
|
|
pa = 0;
|
2008-03-03 17:17:00 +00:00
|
|
|
PMAP_LOCK(pmap);
|
2010-04-30 00:46:43 +00:00
|
|
|
retry:
|
2009-01-13 16:15:49 +00:00
|
|
|
pte = pte_find(mmu, pmap, va);
|
2008-03-03 17:17:00 +00:00
|
|
|
if ((pte != NULL) && PTE_ISVALID(pte)) {
|
|
|
|
if (pmap == kernel_pmap)
|
|
|
|
pte_wbit = PTE_SW;
|
|
|
|
else
|
|
|
|
pte_wbit = PTE_UW;
|
|
|
|
|
2016-02-11 13:15:37 +00:00
|
|
|
if ((*pte & pte_wbit) || ((prot & VM_PROT_WRITE) == 0)) {
|
2010-04-30 00:46:43 +00:00
|
|
|
if (vm_page_pa_tryrelock(pmap, PTE_PA(pte), &pa))
|
|
|
|
goto retry;
|
2008-03-03 17:17:00 +00:00
|
|
|
m = PHYS_TO_VM_PAGE(PTE_PA(pte));
|
|
|
|
vm_page_hold(m);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-04-30 00:46:43 +00:00
|
|
|
PA_UNLOCK_COND(pa);
|
2008-03-03 17:17:00 +00:00
|
|
|
PMAP_UNLOCK(pmap);
|
|
|
|
return (m);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize a vm_page's machine-dependent fields.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_page_init(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
|
|
|
|
TAILQ_INIT(&m->md.pv_list);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* mmu_booke_zero_page_area zeros the specified hardware page by
|
|
|
|
* mapping it into virtual memory and using bzero to clear
|
|
|
|
* its contents.
|
|
|
|
*
|
|
|
|
* off and size must reside within a single page.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_zero_page_area(mmu_t mmu, vm_page_t m, int off, int size)
|
|
|
|
{
|
|
|
|
vm_offset_t va;
|
|
|
|
|
2009-01-13 16:15:49 +00:00
|
|
|
/* XXX KASSERT off and size are within a single page? */
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
mtx_lock(&zero_page_mutex);
|
|
|
|
va = zero_page_va;
|
|
|
|
|
|
|
|
mmu_booke_kenter(mmu, va, VM_PAGE_TO_PHYS(m));
|
|
|
|
bzero((caddr_t)va + off, size);
|
|
|
|
mmu_booke_kremove(mmu, va);
|
|
|
|
|
|
|
|
mtx_unlock(&zero_page_mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* mmu_booke_zero_page zeros the specified hardware page.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_zero_page(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
2015-12-30 02:26:04 +00:00
|
|
|
vm_offset_t off, va;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2015-12-30 02:26:04 +00:00
|
|
|
mtx_lock(&zero_page_mutex);
|
|
|
|
va = zero_page_va;
|
|
|
|
|
|
|
|
mmu_booke_kenter(mmu, va, VM_PAGE_TO_PHYS(m));
|
|
|
|
for (off = 0; off < PAGE_SIZE; off += cacheline_size)
|
2016-01-26 04:41:18 +00:00
|
|
|
__asm __volatile("dcbz 0,%0" :: "r"(va + off));
|
2015-12-30 02:26:04 +00:00
|
|
|
mmu_booke_kremove(mmu, va);
|
|
|
|
|
|
|
|
mtx_unlock(&zero_page_mutex);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* mmu_booke_copy_page copies the specified (machine independent) page by
|
|
|
|
* mapping the page into virtual memory and using memcopy to copy the page,
|
|
|
|
* one machine dependent page at a time.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_copy_page(mmu_t mmu, vm_page_t sm, vm_page_t dm)
|
|
|
|
{
|
|
|
|
vm_offset_t sva, dva;
|
|
|
|
|
|
|
|
sva = copy_page_src_va;
|
|
|
|
dva = copy_page_dst_va;
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_lock(©_page_mutex);
|
2008-03-03 17:17:00 +00:00
|
|
|
mmu_booke_kenter(mmu, sva, VM_PAGE_TO_PHYS(sm));
|
|
|
|
mmu_booke_kenter(mmu, dva, VM_PAGE_TO_PHYS(dm));
|
|
|
|
memcpy((caddr_t)dva, (caddr_t)sva, PAGE_SIZE);
|
|
|
|
mmu_booke_kremove(mmu, dva);
|
|
|
|
mmu_booke_kremove(mmu, sva);
|
|
|
|
mtx_unlock(©_page_mutex);
|
|
|
|
}
|
|
|
|
|
Add pmap function pmap_copy_pages(), which copies the content of the
pages around, taking array of vm_page_t both for source and
destination. Starting offsets and total transfer size are specified.
The function implements optimal algorithm for copying using the
platform-specific optimizations. For instance, on the architectures
were the direct map is available, no transient mappings are created,
for i386 the per-cpu ephemeral page frame is used. The code was
typically borrowed from the pmap_copy_page() for the same
architecture.
Only i386/amd64, powerpc aim and arm/arm-v6 implementations were
tested at the time of commit. High-level code, not committed yet to
the tree, ensures that the use of the function is only allowed after
explicit enablement.
For sparc64, the existing code has known issues and a stab is added
instead, to allow the kernel linking.
Sponsored by: The FreeBSD Foundation
Tested by: pho (i386, amd64), scottl (amd64), ian (arm and arm-v6)
MFC after: 2 weeks
2013-03-14 20:18:12 +00:00
|
|
|
static inline void
|
|
|
|
mmu_booke_copy_pages(mmu_t mmu, vm_page_t *ma, vm_offset_t a_offset,
|
|
|
|
vm_page_t *mb, vm_offset_t b_offset, int xfersize)
|
|
|
|
{
|
|
|
|
void *a_cp, *b_cp;
|
|
|
|
vm_offset_t a_pg_offset, b_pg_offset;
|
|
|
|
int cnt;
|
|
|
|
|
|
|
|
mtx_lock(©_page_mutex);
|
|
|
|
while (xfersize > 0) {
|
|
|
|
a_pg_offset = a_offset & PAGE_MASK;
|
|
|
|
cnt = min(xfersize, PAGE_SIZE - a_pg_offset);
|
|
|
|
mmu_booke_kenter(mmu, copy_page_src_va,
|
|
|
|
VM_PAGE_TO_PHYS(ma[a_offset >> PAGE_SHIFT]));
|
|
|
|
a_cp = (char *)copy_page_src_va + a_pg_offset;
|
|
|
|
b_pg_offset = b_offset & PAGE_MASK;
|
|
|
|
cnt = min(cnt, PAGE_SIZE - b_pg_offset);
|
|
|
|
mmu_booke_kenter(mmu, copy_page_dst_va,
|
|
|
|
VM_PAGE_TO_PHYS(mb[b_offset >> PAGE_SHIFT]));
|
|
|
|
b_cp = (char *)copy_page_dst_va + b_pg_offset;
|
|
|
|
bcopy(a_cp, b_cp, cnt);
|
|
|
|
mmu_booke_kremove(mmu, copy_page_dst_va);
|
|
|
|
mmu_booke_kremove(mmu, copy_page_src_va);
|
|
|
|
a_offset += cnt;
|
|
|
|
b_offset += cnt;
|
|
|
|
xfersize -= cnt;
|
|
|
|
}
|
|
|
|
mtx_unlock(©_page_mutex);
|
|
|
|
}
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/*
|
|
|
|
* mmu_booke_zero_page_idle zeros the specified hardware page by mapping it
|
|
|
|
* into virtual memory and using bzero to clear its contents. This is intended
|
|
|
|
* to be called from the vm_pagezero process only and outside of Giant. No
|
|
|
|
* lock is required.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_zero_page_idle(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
vm_offset_t va;
|
|
|
|
|
|
|
|
va = zero_page_idle_va;
|
|
|
|
mmu_booke_kenter(mmu, va, VM_PAGE_TO_PHYS(m));
|
|
|
|
bzero((caddr_t)va, PAGE_SIZE);
|
|
|
|
mmu_booke_kremove(mmu, va);
|
|
|
|
}
|
|
|
|
|
2015-08-04 19:46:13 +00:00
|
|
|
static vm_offset_t
|
|
|
|
mmu_booke_quick_enter_page(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
vm_paddr_t paddr;
|
|
|
|
vm_offset_t qaddr;
|
|
|
|
uint32_t flags;
|
|
|
|
pte_t *pte;
|
|
|
|
|
|
|
|
paddr = VM_PAGE_TO_PHYS(m);
|
|
|
|
|
|
|
|
flags = PTE_SR | PTE_SW | PTE_SX | PTE_WIRED | PTE_VALID;
|
2016-02-11 13:15:37 +00:00
|
|
|
flags |= tlb_calc_wimg(paddr, pmap_page_get_memattr(m)) << PTE_MAS2_SHIFT;
|
|
|
|
flags |= PTE_PS_4KB;
|
2015-08-04 19:46:13 +00:00
|
|
|
|
|
|
|
critical_enter();
|
|
|
|
qaddr = PCPU_GET(qmap_addr);
|
|
|
|
|
2016-01-19 03:07:25 +00:00
|
|
|
pte = pte_find(mmu, kernel_pmap, qaddr);
|
2015-08-04 19:46:13 +00:00
|
|
|
|
2016-02-11 13:15:37 +00:00
|
|
|
KASSERT(*pte == 0, ("mmu_booke_quick_enter_page: PTE busy"));
|
2015-08-04 19:46:13 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* XXX: tlbivax is broadcast to other cores, but qaddr should
|
|
|
|
* not be present in other TLBs. Is there a better instruction
|
|
|
|
* sequence to use? Or just forget it & use mmu_booke_kenter()...
|
|
|
|
*/
|
|
|
|
__asm __volatile("tlbivax 0, %0" :: "r"(qaddr & MAS2_EPN_MASK));
|
|
|
|
__asm __volatile("isync; msync");
|
|
|
|
|
2016-02-11 13:15:37 +00:00
|
|
|
*pte = PTE_RPN_FROM_PA(paddr) | flags;
|
2015-08-04 19:46:13 +00:00
|
|
|
|
|
|
|
/* Flush the real memory from the instruction cache. */
|
|
|
|
if ((flags & (PTE_I | PTE_G)) == 0)
|
|
|
|
__syncicache((void *)qaddr, PAGE_SIZE);
|
|
|
|
|
|
|
|
return (qaddr);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
mmu_booke_quick_remove_page(mmu_t mmu, vm_offset_t addr)
|
|
|
|
{
|
|
|
|
pte_t *pte;
|
|
|
|
|
2016-01-19 03:07:25 +00:00
|
|
|
pte = pte_find(mmu, kernel_pmap, addr);
|
2015-08-04 19:46:13 +00:00
|
|
|
|
|
|
|
KASSERT(PCPU_GET(qmap_addr) == addr,
|
|
|
|
("mmu_booke_quick_remove_page: invalid address"));
|
2016-02-11 13:15:37 +00:00
|
|
|
KASSERT(*pte != 0,
|
2015-08-04 19:46:13 +00:00
|
|
|
("mmu_booke_quick_remove_page: PTE not in use"));
|
|
|
|
|
2016-02-11 13:15:37 +00:00
|
|
|
*pte = 0;
|
2015-08-04 19:46:13 +00:00
|
|
|
critical_exit();
|
|
|
|
}
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/*
|
|
|
|
* Return whether or not the specified physical page was modified
|
|
|
|
* in any of physical maps.
|
|
|
|
*/
|
|
|
|
static boolean_t
|
|
|
|
mmu_booke_is_modified(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
pte_t *pte;
|
|
|
|
pv_entry_t pv;
|
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
|
|
|
boolean_t rv;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2011-08-09 21:01:36 +00:00
|
|
|
KASSERT((m->oflags & VPO_UNMANAGED) == 0,
|
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
|
|
|
("mmu_booke_is_modified: page %p is not managed", m));
|
|
|
|
rv = FALSE;
|
2008-03-03 17:17:00 +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
|
|
|
/*
|
2013-08-09 11:11:11 +00:00
|
|
|
* If the page is not exclusive busied, then PGA_WRITEABLE cannot be
|
2011-09-06 10:30:11 +00:00
|
|
|
* concurrently set while the object is locked. Thus, if PGA_WRITEABLE
|
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
|
|
|
* is clear, no PTEs can be modified.
|
|
|
|
*/
|
2013-02-21 21:54:53 +00:00
|
|
|
VM_OBJECT_ASSERT_WLOCKED(m->object);
|
2013-08-09 11:11:11 +00:00
|
|
|
if (!vm_page_xbusied(m) && (m->aflags & PGA_WRITEABLE) == 0)
|
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
|
|
|
return (rv);
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
TAILQ_FOREACH(pv, &m->md.pv_list, pv_link) {
|
|
|
|
PMAP_LOCK(pv->pv_pmap);
|
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 ((pte = pte_find(mmu, pv->pv_pmap, pv->pv_va)) != NULL &&
|
|
|
|
PTE_ISVALID(pte)) {
|
|
|
|
if (PTE_ISMODIFIED(pte))
|
|
|
|
rv = TRUE;
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
PMAP_UNLOCK(pv->pv_pmap);
|
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 (rv)
|
|
|
|
break;
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wunlock(&pvh_global_lock);
|
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
|
|
|
return (rv);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2009-01-13 16:15:49 +00:00
|
|
|
* Return whether or not the specified virtual address is eligible
|
2008-03-03 17:17:00 +00:00
|
|
|
* for prefault.
|
|
|
|
*/
|
|
|
|
static boolean_t
|
|
|
|
mmu_booke_is_prefaultable(mmu_t mmu, pmap_t pmap, vm_offset_t addr)
|
|
|
|
{
|
|
|
|
|
|
|
|
return (FALSE);
|
|
|
|
}
|
|
|
|
|
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 whether or not the specified physical page was referenced
|
|
|
|
* in any physical maps.
|
|
|
|
*/
|
|
|
|
static boolean_t
|
|
|
|
mmu_booke_is_referenced(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
pte_t *pte;
|
|
|
|
pv_entry_t pv;
|
|
|
|
boolean_t rv;
|
|
|
|
|
2011-08-09 21:01:36 +00:00
|
|
|
KASSERT((m->oflags & VPO_UNMANAGED) == 0,
|
2010-05-26 18:00:44 +00:00
|
|
|
("mmu_booke_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
|
|
|
rv = FALSE;
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wlock(&pvh_global_lock);
|
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
|
|
|
TAILQ_FOREACH(pv, &m->md.pv_list, pv_link) {
|
|
|
|
PMAP_LOCK(pv->pv_pmap);
|
|
|
|
if ((pte = pte_find(mmu, pv->pv_pmap, pv->pv_va)) != NULL &&
|
2010-05-26 18:00:44 +00:00
|
|
|
PTE_ISVALID(pte)) {
|
|
|
|
if (PTE_ISREFERENCED(pte))
|
|
|
|
rv = 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
|
|
|
PMAP_UNLOCK(pv->pv_pmap);
|
|
|
|
if (rv)
|
|
|
|
break;
|
|
|
|
}
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wunlock(&pvh_global_lock);
|
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 (rv);
|
|
|
|
}
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/*
|
|
|
|
* Clear the modify bits on the specified physical page.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_clear_modify(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
pte_t *pte;
|
|
|
|
pv_entry_t pv;
|
|
|
|
|
2011-08-09 21:01:36 +00:00
|
|
|
KASSERT((m->oflags & VPO_UNMANAGED) == 0,
|
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
|
|
|
("mmu_booke_clear_modify: page %p is not managed", m));
|
2013-02-21 21:54:53 +00:00
|
|
|
VM_OBJECT_ASSERT_WLOCKED(m->object);
|
2013-08-09 11:11:11 +00:00
|
|
|
KASSERT(!vm_page_xbusied(m),
|
|
|
|
("mmu_booke_clear_modify: page %p is exclusive busied", m));
|
2008-03-03 17:17:00 +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
|
|
|
/*
|
2011-09-06 10:30:11 +00:00
|
|
|
* If the page is not PG_AWRITEABLE, then no PTEs can be modified.
|
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 object containing the page is locked and the page is not
|
2013-08-09 11:11:11 +00:00
|
|
|
* exclusive busied, then PG_AWRITEABLE cannot be concurrently set.
|
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
|
|
|
*/
|
2011-09-06 10:30:11 +00:00
|
|
|
if ((m->aflags & PGA_WRITEABLE) == 0)
|
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
|
|
|
return;
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
TAILQ_FOREACH(pv, &m->md.pv_list, pv_link) {
|
|
|
|
PMAP_LOCK(pv->pv_pmap);
|
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 ((pte = pte_find(mmu, pv->pv_pmap, pv->pv_va)) != NULL &&
|
|
|
|
PTE_ISVALID(pte)) {
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_lock_spin(&tlbivax_mutex);
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_lock();
|
2009-01-13 15:41:58 +00:00
|
|
|
|
2016-02-11 13:15:37 +00:00
|
|
|
if (*pte & (PTE_SW | PTE_UW | PTE_MODIFIED)) {
|
2009-01-13 15:41:58 +00:00
|
|
|
tlb0_flush_entry(pv->pv_va);
|
2016-02-11 13:15:37 +00:00
|
|
|
*pte &= ~(PTE_SW | PTE_UW | PTE_MODIFIED |
|
2008-03-03 17:17:00 +00:00
|
|
|
PTE_REFERENCED);
|
|
|
|
}
|
2009-01-13 15:41:58 +00:00
|
|
|
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_unlock();
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_unlock_spin(&tlbivax_mutex);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
PMAP_UNLOCK(pv->pv_pmap);
|
|
|
|
}
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wunlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
mmu_booke_ts_referenced(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
pte_t *pte;
|
|
|
|
pv_entry_t pv;
|
|
|
|
int count;
|
|
|
|
|
2011-08-09 21:01:36 +00:00
|
|
|
KASSERT((m->oflags & VPO_UNMANAGED) == 0,
|
2010-06-10 16:56:35 +00:00
|
|
|
("mmu_booke_ts_referenced: page %p is not managed", m));
|
2008-03-03 17:17:00 +00:00
|
|
|
count = 0;
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
TAILQ_FOREACH(pv, &m->md.pv_list, pv_link) {
|
|
|
|
PMAP_LOCK(pv->pv_pmap);
|
2010-06-10 16:56:35 +00:00
|
|
|
if ((pte = pte_find(mmu, pv->pv_pmap, pv->pv_va)) != NULL &&
|
|
|
|
PTE_ISVALID(pte)) {
|
2008-03-03 17:17:00 +00:00
|
|
|
if (PTE_ISREFERENCED(pte)) {
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_lock_spin(&tlbivax_mutex);
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_lock();
|
2009-01-13 15:41:58 +00:00
|
|
|
|
|
|
|
tlb0_flush_entry(pv->pv_va);
|
2016-02-11 13:15:37 +00:00
|
|
|
*pte &= ~PTE_REFERENCED;
|
2009-01-13 15:41:58 +00:00
|
|
|
|
2009-05-21 11:43:37 +00:00
|
|
|
tlb_miss_unlock();
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_unlock_spin(&tlbivax_mutex);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
if (++count > 4) {
|
|
|
|
PMAP_UNLOCK(pv->pv_pmap);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
PMAP_UNLOCK(pv->pv_pmap);
|
|
|
|
}
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wunlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
return (count);
|
|
|
|
}
|
|
|
|
|
2014-07-13 16:27:57 +00:00
|
|
|
/*
|
|
|
|
* Clear the wired attribute from the mappings for the specified range of
|
|
|
|
* addresses in the given pmap. Every valid mapping within that range must
|
|
|
|
* have the wired attribute set. In contrast, invalid mappings cannot have
|
|
|
|
* the wired attribute set, so they are ignored.
|
|
|
|
*
|
|
|
|
* The wired attribute of the page table entry is not a hardware feature, so
|
|
|
|
* there is no need to invalidate any TLB entries.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_unwire(mmu_t mmu, pmap_t pmap, vm_offset_t sva, vm_offset_t eva)
|
|
|
|
{
|
|
|
|
vm_offset_t va;
|
|
|
|
pte_t *pte;
|
|
|
|
|
|
|
|
PMAP_LOCK(pmap);
|
|
|
|
for (va = sva; va < eva; va += PAGE_SIZE) {
|
|
|
|
if ((pte = pte_find(mmu, pmap, va)) != NULL &&
|
|
|
|
PTE_ISVALID(pte)) {
|
|
|
|
if (!PTE_ISWIRED(pte))
|
|
|
|
panic("mmu_booke_unwire: pte %p isn't wired",
|
|
|
|
pte);
|
2016-02-11 13:15:37 +00:00
|
|
|
*pte &= ~PTE_WIRED;
|
2014-07-13 16:27:57 +00:00
|
|
|
pmap->pm_stats.wired_count--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
PMAP_UNLOCK(pmap);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/*
|
|
|
|
* Return 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.
|
|
|
|
*/
|
|
|
|
static boolean_t
|
|
|
|
mmu_booke_page_exists_quick(mmu_t mmu, pmap_t pmap, vm_page_t m)
|
|
|
|
{
|
|
|
|
pv_entry_t pv;
|
|
|
|
int loops;
|
2010-06-10 16:56:35 +00:00
|
|
|
boolean_t rv;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2011-08-09 21:01:36 +00:00
|
|
|
KASSERT((m->oflags & VPO_UNMANAGED) == 0,
|
2010-06-10 16:56:35 +00:00
|
|
|
("mmu_booke_page_exists_quick: page %p is not managed", m));
|
2008-03-03 17:17:00 +00:00
|
|
|
loops = 0;
|
2010-06-10 16:56:35 +00:00
|
|
|
rv = FALSE;
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
TAILQ_FOREACH(pv, &m->md.pv_list, pv_link) {
|
2010-06-10 16:56:35 +00:00
|
|
|
if (pv->pv_pmap == pmap) {
|
|
|
|
rv = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
2008-03-03 17:17:00 +00:00
|
|
|
if (++loops >= 16)
|
|
|
|
break;
|
|
|
|
}
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wunlock(&pvh_global_lock);
|
2010-06-10 16:56:35 +00:00
|
|
|
return (rv);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return the number of managed mappings to the given physical page that are
|
|
|
|
* wired.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
mmu_booke_page_wired_mappings(mmu_t mmu, vm_page_t m)
|
|
|
|
{
|
|
|
|
pv_entry_t pv;
|
|
|
|
pte_t *pte;
|
|
|
|
int count = 0;
|
|
|
|
|
2011-08-09 21:01:36 +00:00
|
|
|
if ((m->oflags & VPO_UNMANAGED) != 0)
|
2008-03-03 17:17:00 +00:00
|
|
|
return (count);
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
TAILQ_FOREACH(pv, &m->md.pv_list, pv_link) {
|
|
|
|
PMAP_LOCK(pv->pv_pmap);
|
|
|
|
if ((pte = pte_find(mmu, pv->pv_pmap, pv->pv_va)) != NULL)
|
|
|
|
if (PTE_ISVALID(pte) && PTE_ISWIRED(pte))
|
|
|
|
count++;
|
|
|
|
PMAP_UNLOCK(pv->pv_pmap);
|
|
|
|
}
|
2012-11-03 23:22:49 +00:00
|
|
|
rw_wunlock(&pvh_global_lock);
|
2008-03-03 17:17:00 +00:00
|
|
|
return (count);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2012-05-24 21:13:24 +00:00
|
|
|
mmu_booke_dev_direct_mapped(mmu_t mmu, vm_paddr_t pa, vm_size_t size)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
vm_offset_t va;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This currently does not work for entries that
|
|
|
|
* overlap TLB1 entries.
|
|
|
|
*/
|
2016-04-19 01:48:18 +00:00
|
|
|
for (i = 0; i < TLB1_ENTRIES; i ++) {
|
2008-03-03 17:17:00 +00:00
|
|
|
if (tlb1_iomapped(i, pa, size, &va) == 0)
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
return (EFAULT);
|
|
|
|
}
|
|
|
|
|
2015-01-07 01:01:39 +00:00
|
|
|
void
|
|
|
|
mmu_booke_dumpsys_map(mmu_t mmu, vm_paddr_t pa, size_t sz, void **va)
|
2009-04-04 22:01:43 +00:00
|
|
|
{
|
2015-01-07 01:01:39 +00:00
|
|
|
vm_paddr_t ppa;
|
|
|
|
vm_offset_t ofs;
|
2009-04-04 22:01:43 +00:00
|
|
|
vm_size_t gran;
|
|
|
|
|
|
|
|
/* Minidumps are based on virtual memory addresses. */
|
2015-01-07 01:01:39 +00:00
|
|
|
if (do_minidump) {
|
2015-08-22 07:20:03 +00:00
|
|
|
*va = (void *)(vm_offset_t)pa;
|
2015-01-07 01:01:39 +00:00
|
|
|
return;
|
2009-04-04 22:01:43 +00:00
|
|
|
}
|
2015-01-07 01:01:39 +00:00
|
|
|
|
|
|
|
/* Raw physical memory dumps don't have a virtual address. */
|
|
|
|
/* We always map a 256MB page at 256M. */
|
|
|
|
gran = 256 * 1024 * 1024;
|
2016-04-21 19:57:40 +00:00
|
|
|
ppa = rounddown2(pa, gran);
|
2015-01-07 01:01:39 +00:00
|
|
|
ofs = pa - ppa;
|
|
|
|
*va = (void *)gran;
|
|
|
|
tlb1_set_entry((vm_offset_t)va, ppa, gran, _TLB_ENTRY_IO);
|
|
|
|
|
|
|
|
if (sz > (gran - ofs))
|
|
|
|
tlb1_set_entry((vm_offset_t)(va + gran), ppa + gran, gran,
|
|
|
|
_TLB_ENTRY_IO);
|
2009-04-04 22:01:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2015-01-07 01:01:39 +00:00
|
|
|
mmu_booke_dumpsys_unmap(mmu_t mmu, vm_paddr_t pa, size_t sz, void *va)
|
2009-04-04 22:01:43 +00:00
|
|
|
{
|
2015-01-07 01:01:39 +00:00
|
|
|
vm_paddr_t ppa;
|
|
|
|
vm_offset_t ofs;
|
|
|
|
vm_size_t gran;
|
2016-04-19 01:48:18 +00:00
|
|
|
tlb_entry_t e;
|
|
|
|
int i;
|
2015-01-07 01:01:39 +00:00
|
|
|
|
|
|
|
/* Minidumps are based on virtual memory addresses. */
|
|
|
|
/* Nothing to do... */
|
|
|
|
if (do_minidump)
|
|
|
|
return;
|
2009-04-04 22:01:43 +00:00
|
|
|
|
2016-04-19 01:48:18 +00:00
|
|
|
for (i = 0; i < TLB1_ENTRIES; i++) {
|
|
|
|
tlb1_read_entry(&e, i);
|
|
|
|
if (!(e.mas1 & MAS1_VALID))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2009-04-04 22:01:43 +00:00
|
|
|
/* Raw physical memory dumps don't have a virtual address. */
|
2016-04-19 01:48:18 +00:00
|
|
|
i--;
|
|
|
|
e.mas1 = 0;
|
|
|
|
e.mas2 = 0;
|
|
|
|
e.mas3 = 0;
|
|
|
|
tlb1_write_entry(&e, i);
|
2015-01-07 01:01:39 +00:00
|
|
|
|
|
|
|
gran = 256 * 1024 * 1024;
|
2016-04-21 19:57:40 +00:00
|
|
|
ppa = rounddown2(pa, gran);
|
2015-01-07 01:01:39 +00:00
|
|
|
ofs = pa - ppa;
|
|
|
|
if (sz > (gran - ofs)) {
|
2016-04-19 01:48:18 +00:00
|
|
|
i--;
|
|
|
|
e.mas1 = 0;
|
|
|
|
e.mas2 = 0;
|
|
|
|
e.mas3 = 0;
|
|
|
|
tlb1_write_entry(&e, i);
|
2009-04-04 22:01:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-07 01:01:39 +00:00
|
|
|
extern struct dump_pa dump_map[PHYS_AVAIL_SZ + 1];
|
|
|
|
|
|
|
|
void
|
|
|
|
mmu_booke_scan_init(mmu_t mmu)
|
2009-04-04 22:01:43 +00:00
|
|
|
{
|
|
|
|
vm_offset_t va;
|
2015-01-07 01:01:39 +00:00
|
|
|
pte_t *pte;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!do_minidump) {
|
|
|
|
/* Initialize phys. segments for dumpsys(). */
|
|
|
|
memset(&dump_map, 0, sizeof(dump_map));
|
|
|
|
mem_regions(&physmem_regions, &physmem_regions_sz, &availmem_regions,
|
|
|
|
&availmem_regions_sz);
|
|
|
|
for (i = 0; i < physmem_regions_sz; i++) {
|
|
|
|
dump_map[i].pa_start = physmem_regions[i].mr_start;
|
|
|
|
dump_map[i].pa_size = physmem_regions[i].mr_size;
|
2009-04-04 22:01:43 +00:00
|
|
|
}
|
2015-01-07 01:01:39 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Virtual segments for minidumps: */
|
|
|
|
memset(&dump_map, 0, sizeof(dump_map));
|
|
|
|
|
|
|
|
/* 1st: kernel .data and .bss. */
|
|
|
|
dump_map[0].pa_start = trunc_page((uintptr_t)_etext);
|
|
|
|
dump_map[0].pa_size =
|
|
|
|
round_page((uintptr_t)_end) - dump_map[0].pa_start;
|
|
|
|
|
|
|
|
/* 2nd: msgbuf and tables (see pmap_bootstrap()). */
|
|
|
|
dump_map[1].pa_start = data_start;
|
|
|
|
dump_map[1].pa_size = data_end - data_start;
|
|
|
|
|
|
|
|
/* 3rd: kernel VM. */
|
|
|
|
va = dump_map[1].pa_start + dump_map[1].pa_size;
|
|
|
|
/* Find start of next chunk (from va). */
|
|
|
|
while (va < virtual_end) {
|
|
|
|
/* Don't dump the buffer cache. */
|
|
|
|
if (va >= kmi.buffer_sva && va < kmi.buffer_eva) {
|
|
|
|
va = kmi.buffer_eva;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
pte = pte_find(mmu, kernel_pmap, va);
|
|
|
|
if (pte != NULL && PTE_ISVALID(pte))
|
2009-04-04 22:01:43 +00:00
|
|
|
break;
|
2015-01-07 01:01:39 +00:00
|
|
|
va += PAGE_SIZE;
|
|
|
|
}
|
|
|
|
if (va < virtual_end) {
|
|
|
|
dump_map[2].pa_start = va;
|
|
|
|
va += PAGE_SIZE;
|
|
|
|
/* Find last page in chunk. */
|
|
|
|
while (va < virtual_end) {
|
|
|
|
/* Don't run into the buffer cache. */
|
|
|
|
if (va == kmi.buffer_sva)
|
2009-04-04 22:01:43 +00:00
|
|
|
break;
|
2015-01-07 01:01:39 +00:00
|
|
|
pte = pte_find(mmu, kernel_pmap, va);
|
|
|
|
if (pte == NULL || !PTE_ISVALID(pte))
|
|
|
|
break;
|
|
|
|
va += PAGE_SIZE;
|
2009-04-04 22:01:43 +00:00
|
|
|
}
|
2015-01-07 01:01:39 +00:00
|
|
|
dump_map[2].pa_size = va - dump_map[2].pa_start;
|
2009-04-04 22:01:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-03-03 17:17:00 +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.
|
|
|
|
*/
|
|
|
|
static void *
|
2012-05-24 21:13:24 +00:00
|
|
|
mmu_booke_mapdev(mmu_t mmu, vm_paddr_t pa, vm_size_t size)
|
2013-10-26 14:52:55 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
return (mmu_booke_mapdev_attr(mmu, pa, size, VM_MEMATTR_DEFAULT));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *
|
|
|
|
mmu_booke_mapdev_attr(mmu_t mmu, vm_paddr_t pa, vm_size_t size, vm_memattr_t ma)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
2016-04-19 01:48:18 +00:00
|
|
|
tlb_entry_t e;
|
2008-10-25 03:36:21 +00:00
|
|
|
void *res;
|
2016-04-10 15:48:09 +00:00
|
|
|
uintptr_t va, tmpva;
|
2008-10-25 03:36:21 +00:00
|
|
|
vm_size_t sz;
|
2013-10-26 18:18:14 +00:00
|
|
|
int i;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2012-11-03 22:02:12 +00:00
|
|
|
/*
|
2013-10-26 18:18:14 +00:00
|
|
|
* Check if this is premapped in TLB1. Note: this should probably also
|
|
|
|
* check whether a sequence of TLB1 entries exist that match the
|
|
|
|
* requirement, but now only checks the easy case.
|
2012-11-03 22:02:12 +00:00
|
|
|
*/
|
2013-10-26 18:18:14 +00:00
|
|
|
if (ma == VM_MEMATTR_DEFAULT) {
|
2016-04-19 01:48:18 +00:00
|
|
|
for (i = 0; i < TLB1_ENTRIES; i++) {
|
|
|
|
tlb1_read_entry(&e, i);
|
|
|
|
if (!(e.mas1 & MAS1_VALID))
|
2013-10-26 18:18:14 +00:00
|
|
|
continue;
|
2016-04-19 01:48:18 +00:00
|
|
|
if (pa >= e.phys &&
|
|
|
|
(pa + size) <= (e.phys + e.size))
|
|
|
|
return (void *)(e.virt +
|
|
|
|
(vm_offset_t)(pa - e.phys));
|
2013-10-26 18:18:14 +00:00
|
|
|
}
|
2012-11-03 22:02:12 +00:00
|
|
|
}
|
|
|
|
|
2013-10-26 18:18:14 +00:00
|
|
|
size = roundup(size, PAGE_SIZE);
|
2008-10-25 03:36:21 +00:00
|
|
|
|
2013-10-26 20:57:26 +00:00
|
|
|
/*
|
2016-04-10 15:48:09 +00:00
|
|
|
* The device mapping area is between VM_MAXUSER_ADDRESS and
|
|
|
|
* VM_MIN_KERNEL_ADDRESS. This gives 1GB of device addressing.
|
2013-10-26 20:57:26 +00:00
|
|
|
*/
|
2016-04-10 15:48:09 +00:00
|
|
|
#ifdef SPARSE_MAPDEV
|
|
|
|
/*
|
|
|
|
* With a sparse mapdev, align to the largest starting region. This
|
|
|
|
* could feasibly be optimized for a 'best-fit' alignment, but that
|
|
|
|
* calculation could be very costly.
|
|
|
|
*/
|
|
|
|
do {
|
|
|
|
tmpva = tlb1_map_base;
|
|
|
|
va = roundup(tlb1_map_base, 1 << flsl(size));
|
|
|
|
} while (!atomic_cmpset_int(&tlb1_map_base, tmpva, va + size));
|
|
|
|
#else
|
|
|
|
va = atomic_fetchadd_int(&tlb1_map_base, size);
|
|
|
|
#endif
|
2013-10-26 14:52:55 +00:00
|
|
|
res = (void *)va;
|
|
|
|
|
2008-10-25 03:36:21 +00:00
|
|
|
do {
|
|
|
|
sz = 1 << (ilog2(size) & ~1);
|
2015-08-28 03:03:09 +00:00
|
|
|
if (va % sz != 0) {
|
|
|
|
do {
|
2015-08-29 06:52:14 +00:00
|
|
|
sz >>= 2;
|
2015-08-28 03:03:09 +00:00
|
|
|
} while (va % sz != 0);
|
|
|
|
}
|
2008-10-25 03:36:21 +00:00
|
|
|
if (bootverbose)
|
2016-04-19 01:48:18 +00:00
|
|
|
printf("Wiring VA=%x to PA=%jx (size=%x)\n",
|
|
|
|
va, (uintmax_t)pa, sz);
|
|
|
|
tlb1_set_entry(va, pa, sz,
|
|
|
|
_TLB_ENTRY_SHARED | tlb_calc_wimg(pa, ma));
|
2008-10-25 03:36:21 +00:00
|
|
|
size -= sz;
|
|
|
|
pa += sz;
|
|
|
|
va += sz;
|
|
|
|
} while (size > 0);
|
|
|
|
|
|
|
|
return (res);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 'Unmap' a range mapped by mmu_booke_mapdev().
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_unmapdev(mmu_t mmu, vm_offset_t va, vm_size_t size)
|
|
|
|
{
|
2013-10-26 16:49:41 +00:00
|
|
|
#ifdef SUPPORTS_SHRINKING_TLB1
|
2008-03-03 17:17:00 +00:00
|
|
|
vm_offset_t base, offset;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Unmap only if this is inside kernel virtual space.
|
|
|
|
*/
|
|
|
|
if ((va >= VM_MIN_KERNEL_ADDRESS) && (va <= VM_MAX_KERNEL_ADDRESS)) {
|
|
|
|
base = trunc_page(va);
|
|
|
|
offset = va & PAGE_MASK;
|
|
|
|
size = roundup(offset + size, PAGE_SIZE);
|
2013-08-07 06:21:20 +00:00
|
|
|
kva_free(base, size);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
2013-10-26 16:49:41 +00:00
|
|
|
#endif
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2009-01-13 16:15:49 +00:00
|
|
|
* mmu_booke_object_init_pt preloads the ptes for a given object into the
|
|
|
|
* specified pmap. This eliminates the blast of soft faults on process startup
|
|
|
|
* and immediately after an mmap.
|
2008-03-03 17:17:00 +00:00
|
|
|
*/
|
|
|
|
static void
|
|
|
|
mmu_booke_object_init_pt(mmu_t mmu, pmap_t pmap, vm_offset_t addr,
|
|
|
|
vm_object_t object, vm_pindex_t pindex, vm_size_t size)
|
|
|
|
{
|
2009-01-13 16:15:49 +00:00
|
|
|
|
2013-02-21 21:54:53 +00:00
|
|
|
VM_OBJECT_ASSERT_WLOCKED(object);
|
2009-07-24 13:50:29 +00:00
|
|
|
KASSERT(object->type == OBJT_DEVICE || object->type == OBJT_SG,
|
2008-03-03 17:17:00 +00:00
|
|
|
("mmu_booke_object_init_pt: non-device object"));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Perform the pmap work for mincore.
|
|
|
|
*/
|
|
|
|
static int
|
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
|
|
|
mmu_booke_mincore(mmu_t mmu, pmap_t pmap, vm_offset_t addr,
|
|
|
|
vm_paddr_t *locked_pa)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
|
|
|
|
2013-11-06 14:36:38 +00:00
|
|
|
/* XXX: this should be implemented at some point */
|
2008-03-03 17:17:00 +00:00
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
2016-02-27 20:39:36 +00:00
|
|
|
static int
|
|
|
|
mmu_booke_change_attr(mmu_t mmu, vm_offset_t addr, vm_size_t sz,
|
|
|
|
vm_memattr_t mode)
|
|
|
|
{
|
|
|
|
vm_offset_t va;
|
|
|
|
pte_t *pte;
|
|
|
|
int i, j;
|
2016-04-19 01:48:18 +00:00
|
|
|
tlb_entry_t e;
|
2016-02-27 20:39:36 +00:00
|
|
|
|
|
|
|
/* Check TLB1 mappings */
|
2016-04-19 01:48:18 +00:00
|
|
|
for (i = 0; i < TLB1_ENTRIES; i++) {
|
|
|
|
tlb1_read_entry(&e, i);
|
|
|
|
if (!(e.mas1 & MAS1_VALID))
|
2016-02-27 20:39:36 +00:00
|
|
|
continue;
|
2016-04-19 01:48:18 +00:00
|
|
|
if (addr >= e.virt && addr < e.virt + e.size)
|
2016-02-27 20:39:36 +00:00
|
|
|
break;
|
|
|
|
}
|
2016-04-19 01:48:18 +00:00
|
|
|
if (i < TLB1_ENTRIES) {
|
2016-02-27 20:39:36 +00:00
|
|
|
/* Only allow full mappings to be modified for now. */
|
|
|
|
/* Validate the range. */
|
2016-04-19 01:48:18 +00:00
|
|
|
for (j = i, va = addr; va < addr + sz; va += e.size, j++) {
|
|
|
|
tlb1_read_entry(&e, j);
|
|
|
|
if (va != e.virt || (sz - (va - addr) < e.size))
|
2016-02-27 20:39:36 +00:00
|
|
|
return (EINVAL);
|
|
|
|
}
|
2016-04-19 01:48:18 +00:00
|
|
|
for (va = addr; va < addr + sz; va += e.size, i++) {
|
|
|
|
tlb1_read_entry(&e, i);
|
|
|
|
e.mas2 &= ~MAS2_WIMGE_MASK;
|
|
|
|
e.mas2 |= tlb_calc_wimg(e.phys, mode);
|
2016-02-27 20:39:36 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Write it out to the TLB. Should really re-sync with other
|
|
|
|
* cores.
|
|
|
|
*/
|
2016-04-19 01:48:18 +00:00
|
|
|
tlb1_write_entry(&e, i);
|
2016-02-27 20:39:36 +00:00
|
|
|
}
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Not in TLB1, try through pmap */
|
|
|
|
/* First validate the range. */
|
|
|
|
for (va = addr; va < addr + sz; va += PAGE_SIZE) {
|
|
|
|
pte = pte_find(mmu, kernel_pmap, va);
|
|
|
|
if (pte == NULL || !PTE_ISVALID(pte))
|
|
|
|
return (EINVAL);
|
|
|
|
}
|
|
|
|
|
|
|
|
mtx_lock_spin(&tlbivax_mutex);
|
|
|
|
tlb_miss_lock();
|
|
|
|
for (va = addr; va < addr + sz; va += PAGE_SIZE) {
|
|
|
|
pte = pte_find(mmu, kernel_pmap, va);
|
|
|
|
*pte &= ~(PTE_MAS2_MASK << PTE_MAS2_SHIFT);
|
|
|
|
*pte |= tlb_calc_wimg(PTE_PA(pte), mode << PTE_MAS2_SHIFT);
|
|
|
|
tlb0_flush_entry(va);
|
|
|
|
}
|
|
|
|
tlb_miss_unlock();
|
|
|
|
mtx_unlock_spin(&tlbivax_mutex);
|
|
|
|
|
|
|
|
return (pte_vatopa(mmu, kernel_pmap, va));
|
|
|
|
}
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/**************************************************************************/
|
|
|
|
/* TID handling */
|
|
|
|
/**************************************************************************/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Allocate a TID. If necessary, steal one from someone else.
|
|
|
|
* The new TID is flushed from the TLB before returning.
|
|
|
|
*/
|
|
|
|
static tlbtid_t
|
|
|
|
tid_alloc(pmap_t pmap)
|
|
|
|
{
|
|
|
|
tlbtid_t tid;
|
2009-01-13 15:41:58 +00:00
|
|
|
int thiscpu;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
KASSERT((pmap != kernel_pmap), ("tid_alloc: kernel pmap"));
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
CTR2(KTR_PMAP, "%s: s (pmap = %p)", __func__, pmap);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
thiscpu = PCPU_GET(cpuid);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
tid = PCPU_GET(tid_next);
|
|
|
|
if (tid > TID_MAX)
|
|
|
|
tid = TID_MIN;
|
|
|
|
PCPU_SET(tid_next, tid + 1);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
/* If we are stealing TID then clear the relevant pmap's field */
|
|
|
|
if (tidbusy[thiscpu][tid] != NULL) {
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
CTR2(KTR_PMAP, "%s: warning: stealing tid %d", __func__, tid);
|
|
|
|
|
|
|
|
tidbusy[thiscpu][tid]->pm_tid[thiscpu] = TID_NONE;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
/* Flush all entries from TLB0 matching this TID. */
|
2015-12-30 02:23:14 +00:00
|
|
|
tid_flush(tid);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
tidbusy[thiscpu][tid] = pmap;
|
|
|
|
pmap->pm_tid[thiscpu] = tid;
|
|
|
|
__asm __volatile("msync; isync");
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
CTR3(KTR_PMAP, "%s: e (%02d next = %02d)", __func__, tid,
|
|
|
|
PCPU_GET(tid_next));
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
return (tid);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************/
|
|
|
|
/* TLB0 handling */
|
|
|
|
/**************************************************************************/
|
|
|
|
|
|
|
|
static void
|
2009-01-13 15:41:58 +00:00
|
|
|
tlb_print_entry(int i, uint32_t mas1, uint32_t mas2, uint32_t mas3,
|
|
|
|
uint32_t mas7)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
|
|
|
int as;
|
|
|
|
char desc[3];
|
|
|
|
tlbtid_t tid;
|
|
|
|
vm_size_t size;
|
|
|
|
unsigned int tsize;
|
|
|
|
|
|
|
|
desc[2] = '\0';
|
|
|
|
if (mas1 & MAS1_VALID)
|
|
|
|
desc[0] = 'V';
|
|
|
|
else
|
|
|
|
desc[0] = ' ';
|
|
|
|
|
|
|
|
if (mas1 & MAS1_IPROT)
|
|
|
|
desc[1] = 'P';
|
|
|
|
else
|
|
|
|
desc[1] = ' ';
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
as = (mas1 & MAS1_TS_MASK) ? 1 : 0;
|
2008-03-03 17:17:00 +00:00
|
|
|
tid = MAS1_GETTID(mas1);
|
|
|
|
|
|
|
|
tsize = (mas1 & MAS1_TSIZE_MASK) >> MAS1_TSIZE_SHIFT;
|
|
|
|
size = 0;
|
|
|
|
if (tsize)
|
|
|
|
size = tsize2size(tsize);
|
|
|
|
|
|
|
|
debugf("%3d: (%s) [AS=%d] "
|
|
|
|
"sz = 0x%08x tsz = %d tid = %d mas1 = 0x%08x "
|
|
|
|
"mas2(va) = 0x%08x mas3(pa) = 0x%08x mas7 = 0x%08x\n",
|
|
|
|
i, desc, as, size, tsize, tid, mas1, mas2, mas3, mas7);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Convert TLB0 va and way number to tlb0[] table index. */
|
|
|
|
static inline unsigned int
|
|
|
|
tlb0_tableidx(vm_offset_t va, unsigned int way)
|
|
|
|
{
|
|
|
|
unsigned int idx;
|
|
|
|
|
|
|
|
idx = (way * TLB0_ENTRIES_PER_WAY);
|
|
|
|
idx += (va & MAS2_TLB0_ENTRY_IDX_MASK) >> MAS2_TLB0_ENTRY_IDX_SHIFT;
|
|
|
|
return (idx);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2009-01-13 15:41:58 +00:00
|
|
|
* Invalidate TLB0 entry.
|
2008-03-03 17:17:00 +00:00
|
|
|
*/
|
2009-01-13 15:41:58 +00:00
|
|
|
static inline void
|
|
|
|
tlb0_flush_entry(vm_offset_t va)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
CTR2(KTR_PMAP, "%s: s va=0x%08x", __func__, va);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
mtx_assert(&tlbivax_mutex, MA_OWNED);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
__asm __volatile("tlbivax 0, %0" :: "r"(va & MAS2_EPN_MASK));
|
|
|
|
__asm __volatile("isync; msync");
|
|
|
|
__asm __volatile("tlbsync; msync");
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
CTR1(KTR_PMAP, "%s: e", __func__);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Print out contents of the MAS registers for each TLB0 entry */
|
2009-01-13 15:41:58 +00:00
|
|
|
void
|
2008-03-03 17:17:00 +00:00
|
|
|
tlb0_print_tlbentries(void)
|
|
|
|
{
|
2009-01-13 15:41:58 +00:00
|
|
|
uint32_t mas0, mas1, mas2, mas3, mas7;
|
2008-03-03 17:17:00 +00:00
|
|
|
int entryidx, way, idx;
|
|
|
|
|
|
|
|
debugf("TLB0 entries:\n");
|
2009-01-13 15:41:58 +00:00
|
|
|
for (way = 0; way < TLB0_WAYS; way ++)
|
2008-03-03 17:17:00 +00:00
|
|
|
for (entryidx = 0; entryidx < TLB0_ENTRIES_PER_WAY; entryidx++) {
|
|
|
|
|
|
|
|
mas0 = MAS0_TLBSEL(0) | MAS0_ESEL(way);
|
|
|
|
mtspr(SPR_MAS0, mas0);
|
2009-01-13 15:41:58 +00:00
|
|
|
__asm __volatile("isync");
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
mas2 = entryidx << MAS2_TLB0_ENTRY_IDX_SHIFT;
|
|
|
|
mtspr(SPR_MAS2, mas2);
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
__asm __volatile("isync; tlbre");
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
mas1 = mfspr(SPR_MAS1);
|
|
|
|
mas2 = mfspr(SPR_MAS2);
|
|
|
|
mas3 = mfspr(SPR_MAS3);
|
|
|
|
mas7 = mfspr(SPR_MAS7);
|
|
|
|
|
|
|
|
idx = tlb0_tableidx(mas2, way);
|
|
|
|
tlb_print_entry(idx, mas1, mas2, mas3, mas7);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**************************************************************************/
|
|
|
|
/* TLB1 handling */
|
|
|
|
/**************************************************************************/
|
2009-01-13 15:41:58 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* TLB1 mapping notes:
|
|
|
|
*
|
2013-10-26 18:18:14 +00:00
|
|
|
* TLB1[0] Kernel text and data.
|
|
|
|
* TLB1[1-15] Additional kernel text and data mappings (if required), PCI
|
2009-01-13 15:41:58 +00:00
|
|
|
* windows, other devices mappings.
|
|
|
|
*/
|
|
|
|
|
2016-04-19 01:48:18 +00:00
|
|
|
/*
|
|
|
|
* Read an entry from given TLB1 slot.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
tlb1_read_entry(tlb_entry_t *entry, unsigned int slot)
|
|
|
|
{
|
|
|
|
uint32_t mas0;
|
|
|
|
|
|
|
|
KASSERT((entry != NULL), ("%s(): Entry is NULL!", __func__));
|
|
|
|
|
|
|
|
mas0 = MAS0_TLBSEL(1) | MAS0_ESEL(slot);
|
|
|
|
mtspr(SPR_MAS0, mas0);
|
|
|
|
__asm __volatile("isync; tlbre");
|
|
|
|
|
|
|
|
entry->mas1 = mfspr(SPR_MAS1);
|
|
|
|
entry->mas2 = mfspr(SPR_MAS2);
|
|
|
|
entry->mas3 = mfspr(SPR_MAS3);
|
|
|
|
|
|
|
|
switch ((mfpvr() >> 16) & 0xFFFF) {
|
|
|
|
case FSL_E500v2:
|
|
|
|
case FSL_E500mc:
|
|
|
|
case FSL_E5500:
|
|
|
|
entry->mas7 = mfspr(SPR_MAS7);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
entry->mas7 = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
entry->virt = entry->mas2 & MAS2_EPN_MASK;
|
|
|
|
entry->phys = ((vm_paddr_t)(entry->mas7 & MAS7_RPN) << 32) |
|
|
|
|
(entry->mas3 & MAS3_RPN);
|
|
|
|
entry->size =
|
|
|
|
tsize2size((entry->mas1 & MAS1_TSIZE_MASK) >> MAS1_TSIZE_SHIFT);
|
|
|
|
}
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/*
|
|
|
|
* Write given entry to TLB1 hardware.
|
|
|
|
* Use 32 bit pa, clear 4 high-order bits of RPN (mas7).
|
|
|
|
*/
|
|
|
|
static void
|
2016-04-19 01:48:18 +00:00
|
|
|
tlb1_write_entry(tlb_entry_t *e, unsigned int idx)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
2015-08-22 07:20:03 +00:00
|
|
|
uint32_t mas0;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
//debugf("tlb1_write_entry: s\n");
|
|
|
|
|
|
|
|
/* Select entry */
|
|
|
|
mas0 = MAS0_TLBSEL(1) | MAS0_ESEL(idx);
|
|
|
|
//debugf("tlb1_write_entry: mas0 = 0x%08x\n", mas0);
|
|
|
|
|
|
|
|
mtspr(SPR_MAS0, mas0);
|
2009-01-13 16:15:49 +00:00
|
|
|
__asm __volatile("isync");
|
2016-04-19 01:48:18 +00:00
|
|
|
mtspr(SPR_MAS1, e->mas1);
|
2009-01-13 16:15:49 +00:00
|
|
|
__asm __volatile("isync");
|
2016-04-19 01:48:18 +00:00
|
|
|
mtspr(SPR_MAS2, e->mas2);
|
2009-01-13 16:15:49 +00:00
|
|
|
__asm __volatile("isync");
|
2016-04-19 01:48:18 +00:00
|
|
|
mtspr(SPR_MAS3, e->mas3);
|
2009-01-13 16:15:49 +00:00
|
|
|
__asm __volatile("isync");
|
2015-08-28 03:03:09 +00:00
|
|
|
switch ((mfpvr() >> 16) & 0xFFFF) {
|
|
|
|
case FSL_E500mc:
|
|
|
|
case FSL_E5500:
|
|
|
|
mtspr(SPR_MAS8, 0);
|
|
|
|
__asm __volatile("isync");
|
|
|
|
/* FALLTHROUGH */
|
|
|
|
case FSL_E500v2:
|
2016-04-19 01:48:18 +00:00
|
|
|
mtspr(SPR_MAS7, e->mas7);
|
2015-08-28 03:03:09 +00:00
|
|
|
__asm __volatile("isync");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
__asm __volatile("tlbwe; isync; msync");
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2010-01-07 21:01:37 +00:00
|
|
|
//debugf("tlb1_write_entry: e\n");
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return the largest uint value log such that 2^log <= num.
|
|
|
|
*/
|
|
|
|
static unsigned int
|
|
|
|
ilog2(unsigned int num)
|
|
|
|
{
|
|
|
|
int lz;
|
|
|
|
|
|
|
|
__asm ("cntlzw %0, %1" : "=r" (lz) : "r" (num));
|
|
|
|
return (31 - lz);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Convert TLB TSIZE value to mapped region size.
|
|
|
|
*/
|
|
|
|
static vm_size_t
|
|
|
|
tsize2size(unsigned int tsize)
|
|
|
|
{
|
|
|
|
|
|
|
|
/*
|
|
|
|
* size = 4^tsize KB
|
|
|
|
* size = 4^tsize * 2^10 = 2^(2 * tsize - 10)
|
|
|
|
*/
|
|
|
|
|
|
|
|
return ((1 << (2 * tsize)) * 1024);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Convert region size (must be power of 4) to TLB TSIZE value.
|
|
|
|
*/
|
|
|
|
static unsigned int
|
|
|
|
size2tsize(vm_size_t size)
|
|
|
|
{
|
|
|
|
|
|
|
|
return (ilog2(size) / 2 - 5);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2009-01-13 15:41:58 +00:00
|
|
|
* Register permanent kernel mapping in TLB1.
|
2008-03-03 17:17:00 +00:00
|
|
|
*
|
2009-01-13 15:41:58 +00:00
|
|
|
* Entries are created starting from index 0 (current free entry is
|
|
|
|
* kept in tlb1_idx) and are not supposed to be invalidated.
|
2008-03-03 17:17:00 +00:00
|
|
|
*/
|
Add support for the Freescale dTSEC DPAA-based ethernet controller.
Freescale's QorIQ line includes a new ethernet controller, based on their
Datapath Acceleration Architecture (DPAA). This uses a combination of a Frame
manager, Buffer manager, and Queue manager to improve performance across all
interfaces by being able to pass data directly between hardware acceleration
interfaces.
As part of this import, Freescale's Netcomm Software (ncsw) driver is imported.
This was an attempt by Freescale to create an OS-agnostic sub-driver for
managing the hardware, using shims to interface to the OS-specific APIs. This
work was abandoned, and Freescale's primary work is in the Linux driver (dual
BSD/GPL license). Hence, this was imported directly to sys/contrib, rather than
going through the vendor area. Going forward, FreeBSD-specific changes may be
made to the ncsw code, diverging from the upstream in potentially incompatible
ways. An alternative could be to import the Linux driver itself, using the
linuxKPI layer, as that would maintain parity with the vendor-maintained driver.
However, the Linux driver has not been evaluated for reliability yet, and may
have issues with the import, whereas the ncsw-based driver in this commit was
completed by Semihalf 4 years ago, and is very stable.
Other SoC modules based on DPAA, which could be added in the future:
* Security and Encryption engine (SEC4.x, SEC5.x)
* RAID engine
Additional work to be done:
* Implement polling mode
* Test vlan support
* Add support for the Pattern Matching Engine, which can do regular expression
matching on packets.
This driver has been tested on the P5020 QorIQ SoC. Others listed in the
dtsec(4) manual page are expected to work as the same DPAA engine is included in
all.
Obtained from: Semihalf
Relnotes: Yes
Sponsored by: Alex Perez/Inertial Computing
2016-02-29 03:38:00 +00:00
|
|
|
int
|
2015-07-04 19:00:38 +00:00
|
|
|
tlb1_set_entry(vm_offset_t va, vm_paddr_t pa, vm_size_t size,
|
2009-01-13 15:41:58 +00:00
|
|
|
uint32_t flags)
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
2016-04-19 01:48:18 +00:00
|
|
|
tlb_entry_t e;
|
2009-01-13 15:41:58 +00:00
|
|
|
uint32_t ts, tid;
|
2013-10-26 16:49:41 +00:00
|
|
|
int tsize, index;
|
|
|
|
|
2016-04-19 01:48:18 +00:00
|
|
|
for (index = 0; index < TLB1_ENTRIES; index++) {
|
|
|
|
tlb1_read_entry(&e, index);
|
|
|
|
if ((e.mas1 & MAS1_VALID) == 0)
|
|
|
|
break;
|
|
|
|
/* Check if we're just updating the flags, and update them. */
|
|
|
|
if (e.phys == pa && e.virt == va && e.size == size) {
|
|
|
|
e.mas2 = (va & MAS2_EPN_MASK) | flags;
|
|
|
|
tlb1_write_entry(&e, index);
|
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
}
|
2013-10-26 16:49:41 +00:00
|
|
|
if (index >= TLB1_ENTRIES) {
|
2009-01-13 15:41:58 +00:00
|
|
|
printf("tlb1_set_entry: TLB1 full!\n");
|
|
|
|
return (-1);
|
|
|
|
}
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Convert size to TSIZE */
|
|
|
|
tsize = size2tsize(size);
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
tid = (TID_KERNEL << MAS1_TID_SHIFT) & MAS1_TID_MASK;
|
|
|
|
/* XXX TS is hard coded to 0 for now as we only use single address space */
|
|
|
|
ts = (0 << MAS1_TS_SHIFT) & MAS1_TS_MASK;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2016-04-19 01:48:18 +00:00
|
|
|
e.phys = pa;
|
|
|
|
e.virt = va;
|
|
|
|
e.size = size;
|
|
|
|
e.mas1 = MAS1_VALID | MAS1_IPROT | ts | tid;
|
|
|
|
e.mas1 |= ((tsize << MAS1_TSIZE_SHIFT) & MAS1_TSIZE_MASK);
|
|
|
|
e.mas2 = (va & MAS2_EPN_MASK) | flags;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
/* Set supervisor RWX permission bits */
|
2016-04-19 01:48:18 +00:00
|
|
|
e.mas3 = (pa & MAS3_RPN) | MAS3_SR | MAS3_SW | MAS3_SX;
|
|
|
|
e.mas7 = (pa >> 32) & MAS7_RPN;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2016-04-19 01:48:18 +00:00
|
|
|
tlb1_write_entry(&e, index);
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
/*
|
|
|
|
* XXX in general TLB1 updates should be propagated between CPUs,
|
|
|
|
* since current design assumes to have the same TLB1 set-up on all
|
|
|
|
* cores.
|
|
|
|
*/
|
2008-03-03 17:17:00 +00:00
|
|
|
return (0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2009-01-13 16:15:49 +00:00
|
|
|
* Map in contiguous RAM region into the TLB1 using maximum of
|
2008-03-03 17:17:00 +00:00
|
|
|
* KERNEL_REGION_MAX_TLB_ENTRIES entries.
|
|
|
|
*
|
2009-01-13 16:15:49 +00:00
|
|
|
* If necessary round up last entry size and return total size
|
2008-03-03 17:17:00 +00:00
|
|
|
* used by all allocated entries.
|
|
|
|
*/
|
|
|
|
vm_size_t
|
2011-08-02 15:35:43 +00:00
|
|
|
tlb1_mapin_region(vm_offset_t va, vm_paddr_t pa, vm_size_t size)
|
|
|
|
{
|
|
|
|
vm_size_t pgs[KERNEL_REGION_MAX_TLB_ENTRIES];
|
|
|
|
vm_size_t mapped, pgsz, base, mask;
|
|
|
|
int idx, nents;
|
|
|
|
|
|
|
|
/* Round up to the next 1M */
|
2016-04-21 19:57:40 +00:00
|
|
|
size = roundup2(size, 1 << 20);
|
2011-08-02 15:35:43 +00:00
|
|
|
|
|
|
|
mapped = 0;
|
|
|
|
idx = 0;
|
|
|
|
base = va;
|
|
|
|
pgsz = 64*1024*1024;
|
|
|
|
while (mapped < size) {
|
|
|
|
while (mapped < size && idx < KERNEL_REGION_MAX_TLB_ENTRIES) {
|
|
|
|
while (pgsz > (size - mapped))
|
|
|
|
pgsz >>= 2;
|
|
|
|
pgs[idx++] = pgsz;
|
|
|
|
mapped += pgsz;
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
2011-08-02 15:35:43 +00:00
|
|
|
/* We under-map. Correct for this. */
|
|
|
|
if (mapped < size) {
|
|
|
|
while (pgs[idx - 1] == pgsz) {
|
|
|
|
idx--;
|
|
|
|
mapped -= pgsz;
|
|
|
|
}
|
|
|
|
/* XXX We may increase beyond out starting point. */
|
|
|
|
pgsz <<= 2;
|
|
|
|
pgs[idx++] = pgsz;
|
|
|
|
mapped += pgsz;
|
|
|
|
}
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
2011-08-02 15:35:43 +00:00
|
|
|
nents = idx;
|
|
|
|
mask = pgs[0] - 1;
|
|
|
|
/* Align address to the boundary */
|
|
|
|
if (va & mask) {
|
|
|
|
va = (va + mask) & ~mask;
|
|
|
|
pa = (pa + mask) & ~mask;
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
2011-08-02 15:35:43 +00:00
|
|
|
for (idx = 0; idx < nents; idx++) {
|
|
|
|
pgsz = pgs[idx];
|
2015-08-22 07:20:03 +00:00
|
|
|
debugf("%u: %llx -> %x, size=%x\n", idx, pa, va, pgsz);
|
2016-04-19 01:48:18 +00:00
|
|
|
tlb1_set_entry(va, pa, pgsz,
|
|
|
|
_TLB_ENTRY_SHARED | _TLB_ENTRY_MEM);
|
2011-08-02 15:35:43 +00:00
|
|
|
pa += pgsz;
|
|
|
|
va += pgsz;
|
|
|
|
}
|
2008-03-03 17:17:00 +00:00
|
|
|
|
2011-08-02 15:35:43 +00:00
|
|
|
mapped = (va - base);
|
2016-01-19 03:07:25 +00:00
|
|
|
#ifdef __powerpc64__
|
|
|
|
printf("mapped size 0x%016lx (wasted space 0x%16lx)\n",
|
|
|
|
#else
|
2013-10-26 20:57:26 +00:00
|
|
|
printf("mapped size 0x%08x (wasted space 0x%08x)\n",
|
2016-01-19 03:07:25 +00:00
|
|
|
#endif
|
2011-08-02 15:35:43 +00:00
|
|
|
mapped, mapped - size);
|
|
|
|
return (mapped);
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* TLB1 initialization routine, to be called after the very first
|
|
|
|
* assembler level setup done in locore.S.
|
|
|
|
*/
|
|
|
|
void
|
2013-10-26 18:18:14 +00:00
|
|
|
tlb1_init()
|
2008-03-03 17:17:00 +00:00
|
|
|
{
|
2015-08-22 07:20:03 +00:00
|
|
|
uint32_t mas0, mas1, mas2, mas3, mas7;
|
2011-08-02 15:35:43 +00:00
|
|
|
uint32_t tsz;
|
|
|
|
|
2015-08-28 03:03:09 +00:00
|
|
|
tlb1_get_tlbconf();
|
2011-08-02 15:35:43 +00:00
|
|
|
|
2015-08-28 03:03:09 +00:00
|
|
|
mas0 = MAS0_TLBSEL(1) | MAS0_ESEL(0);
|
|
|
|
mtspr(SPR_MAS0, mas0);
|
|
|
|
__asm __volatile("isync; tlbre");
|
|
|
|
|
|
|
|
mas1 = mfspr(SPR_MAS1);
|
|
|
|
mas2 = mfspr(SPR_MAS2);
|
|
|
|
mas3 = mfspr(SPR_MAS3);
|
|
|
|
mas7 = mfspr(SPR_MAS7);
|
|
|
|
|
2016-04-19 01:48:18 +00:00
|
|
|
kernload = ((vm_paddr_t)(mas7 & MAS7_RPN) << 32) |
|
2015-08-28 03:03:09 +00:00
|
|
|
(mas3 & MAS3_RPN);
|
|
|
|
|
|
|
|
tsz = (mas1 & MAS1_TSIZE_MASK) >> MAS1_TSIZE_SHIFT;
|
2016-04-19 01:48:18 +00:00
|
|
|
kernsize += (tsz > 0) ? tsize2size(tsz) : 0;
|
2012-07-02 21:15:56 +00:00
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/* Setup TLB miss defaults */
|
|
|
|
set_mas4_defaults();
|
|
|
|
}
|
|
|
|
|
2013-10-26 18:18:14 +00:00
|
|
|
vm_offset_t
|
|
|
|
pmap_early_io_map(vm_paddr_t pa, vm_size_t size)
|
|
|
|
{
|
|
|
|
vm_paddr_t pa_base;
|
|
|
|
vm_offset_t va, sz;
|
|
|
|
int i;
|
2016-04-19 01:48:18 +00:00
|
|
|
tlb_entry_t e;
|
2013-10-26 18:18:14 +00:00
|
|
|
|
|
|
|
KASSERT(!pmap_bootstrapped, ("Do not use after PMAP is up!"));
|
|
|
|
|
2016-04-19 01:48:18 +00:00
|
|
|
for (i = 0; i < TLB1_ENTRIES; i++) {
|
|
|
|
tlb1_read_entry(&e, i);
|
|
|
|
if (!(e.mas1 & MAS1_VALID))
|
2013-10-26 18:18:14 +00:00
|
|
|
continue;
|
2016-04-19 01:48:18 +00:00
|
|
|
if (pa >= e.phys && (pa + size) <=
|
|
|
|
(e.phys + e.size))
|
|
|
|
return (e.virt + (pa - e.phys));
|
2013-10-26 18:18:14 +00:00
|
|
|
}
|
|
|
|
|
2015-11-21 06:03:46 +00:00
|
|
|
pa_base = rounddown(pa, PAGE_SIZE);
|
2013-10-26 18:18:14 +00:00
|
|
|
size = roundup(size + (pa - pa_base), PAGE_SIZE);
|
2013-11-17 01:59:42 +00:00
|
|
|
tlb1_map_base = roundup2(tlb1_map_base, 1 << (ilog2(size) & ~1));
|
2013-10-26 20:57:26 +00:00
|
|
|
va = tlb1_map_base + (pa - pa_base);
|
2013-10-26 18:18:14 +00:00
|
|
|
|
|
|
|
do {
|
|
|
|
sz = 1 << (ilog2(size) & ~1);
|
2016-04-19 01:48:18 +00:00
|
|
|
tlb1_set_entry(tlb1_map_base, pa_base, sz,
|
|
|
|
_TLB_ENTRY_SHARED | _TLB_ENTRY_IO);
|
2013-10-26 18:18:14 +00:00
|
|
|
size -= sz;
|
|
|
|
pa_base += sz;
|
2013-10-26 20:57:26 +00:00
|
|
|
tlb1_map_base += sz;
|
2013-10-26 18:18:14 +00:00
|
|
|
} while (size > 0);
|
|
|
|
|
|
|
|
return (va);
|
|
|
|
}
|
|
|
|
|
2008-03-03 17:17:00 +00:00
|
|
|
/*
|
|
|
|
* Setup MAS4 defaults.
|
|
|
|
* These values are loaded to MAS0-2 on a TLB miss.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
set_mas4_defaults(void)
|
|
|
|
{
|
2009-01-13 16:15:49 +00:00
|
|
|
uint32_t mas4;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
/* Defaults: TLB0, PID0, TSIZED=4K */
|
|
|
|
mas4 = MAS4_TLBSELD0;
|
|
|
|
mas4 |= (TLB_SIZE_4K << MAS4_TSIZED_SHIFT) & MAS4_TSIZED_MASK;
|
2009-05-21 11:43:37 +00:00
|
|
|
#ifdef SMP
|
|
|
|
mas4 |= MAS4_MD;
|
|
|
|
#endif
|
2008-03-03 17:17:00 +00:00
|
|
|
mtspr(SPR_MAS4, mas4);
|
2009-01-13 16:15:49 +00:00
|
|
|
__asm __volatile("isync");
|
2008-03-03 17:17:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Print out contents of the MAS registers for each TLB1 entry
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
tlb1_print_tlbentries(void)
|
|
|
|
{
|
2009-01-13 15:41:58 +00:00
|
|
|
uint32_t mas0, mas1, mas2, mas3, mas7;
|
2008-03-03 17:17:00 +00:00
|
|
|
int i;
|
|
|
|
|
|
|
|
debugf("TLB1 entries:\n");
|
2009-01-13 15:41:58 +00:00
|
|
|
for (i = 0; i < TLB1_ENTRIES; i++) {
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
mas0 = MAS0_TLBSEL(1) | MAS0_ESEL(i);
|
|
|
|
mtspr(SPR_MAS0, mas0);
|
|
|
|
|
2009-01-13 15:41:58 +00:00
|
|
|
__asm __volatile("isync; tlbre");
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
mas1 = mfspr(SPR_MAS1);
|
|
|
|
mas2 = mfspr(SPR_MAS2);
|
|
|
|
mas3 = mfspr(SPR_MAS3);
|
|
|
|
mas7 = mfspr(SPR_MAS7);
|
|
|
|
|
|
|
|
tlb_print_entry(i, mas1, mas2, mas3, mas7);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return 0 if the physical IO range is encompassed by one of the
|
|
|
|
* the TLB1 entries, otherwise return related error code.
|
|
|
|
*/
|
|
|
|
static int
|
|
|
|
tlb1_iomapped(int i, vm_paddr_t pa, vm_size_t size, vm_offset_t *va)
|
|
|
|
{
|
2009-01-13 16:15:49 +00:00
|
|
|
uint32_t prot;
|
2008-03-03 17:17:00 +00:00
|
|
|
vm_paddr_t pa_start;
|
|
|
|
vm_paddr_t pa_end;
|
|
|
|
unsigned int entry_tsize;
|
|
|
|
vm_size_t entry_size;
|
2016-04-19 01:48:18 +00:00
|
|
|
tlb_entry_t e;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
*va = (vm_offset_t)NULL;
|
|
|
|
|
2016-04-19 01:48:18 +00:00
|
|
|
tlb1_read_entry(&e, i);
|
2008-03-03 17:17:00 +00:00
|
|
|
/* Skip invalid entries */
|
2016-04-19 01:48:18 +00:00
|
|
|
if (!(e.mas1 & MAS1_VALID))
|
2008-03-03 17:17:00 +00:00
|
|
|
return (EINVAL);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The entry must be cache-inhibited, guarded, and r/w
|
|
|
|
* so it can function as an i/o page
|
|
|
|
*/
|
2016-04-19 01:48:18 +00:00
|
|
|
prot = e.mas2 & (MAS2_I | MAS2_G);
|
2008-03-03 17:17:00 +00:00
|
|
|
if (prot != (MAS2_I | MAS2_G))
|
|
|
|
return (EPERM);
|
|
|
|
|
2016-04-19 01:48:18 +00:00
|
|
|
prot = e.mas3 & (MAS3_SR | MAS3_SW);
|
2008-03-03 17:17:00 +00:00
|
|
|
if (prot != (MAS3_SR | MAS3_SW))
|
|
|
|
return (EPERM);
|
|
|
|
|
|
|
|
/* The address should be within the entry range. */
|
2016-04-19 01:48:18 +00:00
|
|
|
entry_tsize = (e.mas1 & MAS1_TSIZE_MASK) >> MAS1_TSIZE_SHIFT;
|
2008-03-03 17:17:00 +00:00
|
|
|
KASSERT((entry_tsize), ("tlb1_iomapped: invalid entry tsize"));
|
|
|
|
|
|
|
|
entry_size = tsize2size(entry_tsize);
|
2016-04-19 01:48:18 +00:00
|
|
|
pa_start = (((vm_paddr_t)e.mas7 & MAS7_RPN) << 32) |
|
|
|
|
(e.mas3 & MAS3_RPN);
|
2015-08-22 07:20:03 +00:00
|
|
|
pa_end = pa_start + entry_size;
|
2008-03-03 17:17:00 +00:00
|
|
|
|
|
|
|
if ((pa < pa_start) || ((pa + size) > pa_end))
|
|
|
|
return (ERANGE);
|
|
|
|
|
|
|
|
/* Return virtual address of this mapping. */
|
2016-04-19 01:48:18 +00:00
|
|
|
*va = (e.mas2 & MAS2_EPN_MASK) + (pa - pa_start);
|
2008-03-03 17:17:00 +00:00
|
|
|
return (0);
|
|
|
|
}
|
2015-12-30 02:23:14 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Invalidate all TLB0 entries which match the given TID. Note this is
|
|
|
|
* dedicated for cases when invalidations should NOT be propagated to other
|
|
|
|
* CPUs.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
tid_flush(tlbtid_t tid)
|
|
|
|
{
|
|
|
|
register_t msr;
|
|
|
|
uint32_t mas0, mas1, mas2;
|
|
|
|
int entry, way;
|
|
|
|
|
|
|
|
|
|
|
|
/* Don't evict kernel translations */
|
|
|
|
if (tid == TID_KERNEL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
msr = mfmsr();
|
|
|
|
__asm __volatile("wrteei 0");
|
|
|
|
|
|
|
|
for (way = 0; way < TLB0_WAYS; way++)
|
|
|
|
for (entry = 0; entry < TLB0_ENTRIES_PER_WAY; entry++) {
|
|
|
|
|
|
|
|
mas0 = MAS0_TLBSEL(0) | MAS0_ESEL(way);
|
|
|
|
mtspr(SPR_MAS0, mas0);
|
|
|
|
__asm __volatile("isync");
|
|
|
|
|
|
|
|
mas2 = entry << MAS2_TLB0_ENTRY_IDX_SHIFT;
|
|
|
|
mtspr(SPR_MAS2, mas2);
|
|
|
|
|
|
|
|
__asm __volatile("isync; tlbre");
|
|
|
|
|
|
|
|
mas1 = mfspr(SPR_MAS1);
|
|
|
|
|
|
|
|
if (!(mas1 & MAS1_VALID))
|
|
|
|
continue;
|
|
|
|
if (((mas1 & MAS1_TID_MASK) >> MAS1_TID_SHIFT) != tid)
|
|
|
|
continue;
|
|
|
|
mas1 &= ~MAS1_VALID;
|
|
|
|
mtspr(SPR_MAS1, mas1);
|
|
|
|
__asm __volatile("isync; tlbwe; isync; msync");
|
|
|
|
}
|
|
|
|
mtmsr(msr);
|
|
|
|
}
|