2012-05-30 12:23:30 +08:00
|
|
|
/*
|
|
|
|
* PowerPC MMU, TLB, SLB and BAT emulation helpers for QEMU.
|
|
|
|
*
|
|
|
|
* Copyright (c) 2003-2007 Jocelyn Mayer
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2016-01-27 02:16:58 +08:00
|
|
|
#include "qemu/osdep.h"
|
include/qemu/osdep.h: Don't include qapi/error.h
Commit 57cb38b included qapi/error.h into qemu/osdep.h to get the
Error typedef. Since then, we've moved to include qemu/osdep.h
everywhere. Its file comment explains: "To avoid getting into
possible circular include dependencies, this file should not include
any other QEMU headers, with the exceptions of config-host.h,
compiler.h, os-posix.h and os-win32.h, all of which are doing a
similar job to this file and are under similar constraints."
qapi/error.h doesn't do a similar job, and it doesn't adhere to
similar constraints: it includes qapi-types.h. That's in excess of
100KiB of crap most .c files don't actually need.
Add the typedef to qemu/typedefs.h, and include that instead of
qapi/error.h. Include qapi/error.h in .c files that need it and don't
get it now. Include qapi-types.h in qom/object.h for uint16List.
Update scripts/clean-includes accordingly. Update it further to match
reality: replace config.h by config-target.h, add sysemu/os-posix.h,
sysemu/os-win32.h. Update the list of includes in the qemu/osdep.h
comment quoted above similarly.
This reduces the number of objects depending on qapi/error.h from "all
of them" to less than a third. Unfortunately, the number depending on
qapi-types.h shrinks only a little. More work is needed for that one.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
[Fix compilation without the spice devel packages. - Paolo]
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2016-03-14 16:01:28 +08:00
|
|
|
#include "qapi/error.h"
|
2012-05-30 12:23:30 +08:00
|
|
|
#include "cpu.h"
|
2014-04-08 13:31:41 +08:00
|
|
|
#include "exec/helper-proto.h"
|
2012-12-18 01:20:04 +08:00
|
|
|
#include "sysemu/kvm.h"
|
2012-05-30 12:23:33 +08:00
|
|
|
#include "kvm_ppc.h"
|
2013-03-12 08:31:06 +08:00
|
|
|
#include "mmu-hash64.h"
|
2013-03-12 08:31:07 +08:00
|
|
|
#include "mmu-hash32.h"
|
2016-03-15 20:18:37 +08:00
|
|
|
#include "exec/exec-all.h"
|
2014-03-29 02:42:10 +08:00
|
|
|
#include "exec/cpu_ldst.h"
|
2016-01-07 21:55:28 +08:00
|
|
|
#include "exec/log.h"
|
ppc: Do some batching of TCG tlb flushes
On ppc64 especially, we flush the tlb on any slbie or tlbie instruction.
However, those instructions often come in bursts of 3 or more (context
switch will favor a series of slbie's for example to an slbia if the
SLB has less than a certain number of entries in it, and tlbie's can
happen in a series, with PAPR, H_BULK_REMOVE can remove up to 4 entries
at a time.
Doing a tlb_flush() each time is a waste of time. We end up doing a memset
of the whole TLB, reloading it for the next instruction, memset'ing again,
etc...
Those instructions don't have to take effect immediately. For slbie, they
can wait for the next context synchronizing event. For tlbie, the next
tlbsync.
This implements batching by keeping a flag that indicates that we have a
TLB in need of flushing. We check it on interrupts, rfi's, isync's and
tlbsync and flush the TLB if needed.
This reduces the number of tlb_flush() on a boot to a ubuntu installer
first dialog screen from roughly 360K down to 36K.
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
[clg: added a 'CPUPPCState *' variable in h_remove() and
h_bulk_remove() ]
Signed-off-by: Cédric Le Goater <clg@kaod.org>
[dwg: removed spurious whitespace change, use 0/1 not true/false
consistently, since tlb_need_flush has int type]
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
2016-05-04 00:03:25 +08:00
|
|
|
#include "helper_regs.h"
|
2017-02-24 09:05:12 +08:00
|
|
|
#include "qemu/error-report.h"
|
2017-03-01 14:54:38 +08:00
|
|
|
#include "mmu-book3s-v3.h"
|
2017-07-03 14:19:47 +08:00
|
|
|
#include "mmu-radix64.h"
|
2012-05-30 12:23:30 +08:00
|
|
|
|
2012-05-30 12:23:33 +08:00
|
|
|
//#define DEBUG_MMU
|
|
|
|
//#define DEBUG_BATS
|
2012-05-30 12:23:30 +08:00
|
|
|
//#define DEBUG_SOFTWARE_TLB
|
2012-05-30 12:23:33 +08:00
|
|
|
//#define DUMP_PAGE_TABLES
|
|
|
|
//#define FLUSH_ALL_TLBS
|
|
|
|
|
|
|
|
#ifdef DEBUG_MMU
|
2015-11-13 20:34:23 +08:00
|
|
|
# define LOG_MMU_STATE(cpu) log_cpu_state_mask(CPU_LOG_MMU, (cpu), 0)
|
2012-05-30 12:23:33 +08:00
|
|
|
#else
|
2013-07-03 06:52:23 +08:00
|
|
|
# define LOG_MMU_STATE(cpu) do { } while (0)
|
2012-05-30 12:23:33 +08:00
|
|
|
#endif
|
2012-05-30 12:23:30 +08:00
|
|
|
|
|
|
|
#ifdef DEBUG_SOFTWARE_TLB
|
2015-11-13 20:34:23 +08:00
|
|
|
# define LOG_SWTLB(...) qemu_log_mask(CPU_LOG_MMU, __VA_ARGS__)
|
2012-05-30 12:23:30 +08:00
|
|
|
#else
|
|
|
|
# define LOG_SWTLB(...) do { } while (0)
|
|
|
|
#endif
|
|
|
|
|
2012-05-30 12:23:33 +08:00
|
|
|
#ifdef DEBUG_BATS
|
2015-11-13 20:34:23 +08:00
|
|
|
# define LOG_BATS(...) qemu_log_mask(CPU_LOG_MMU, __VA_ARGS__)
|
2012-05-30 12:23:33 +08:00
|
|
|
#else
|
|
|
|
# define LOG_BATS(...) do { } while (0)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
/* PowerPC MMU emulation */
|
2013-03-12 08:31:17 +08:00
|
|
|
|
|
|
|
/* Context used internally during MMU translations */
|
|
|
|
typedef struct mmu_ctx_t mmu_ctx_t;
|
|
|
|
struct mmu_ctx_t {
|
|
|
|
hwaddr raddr; /* Real address */
|
|
|
|
hwaddr eaddr; /* Effective address */
|
|
|
|
int prot; /* Protection bits */
|
|
|
|
hwaddr hash[2]; /* Pagetable hash values */
|
|
|
|
target_ulong ptem; /* Virtual segment ID | API */
|
|
|
|
int key; /* Access key */
|
|
|
|
int nx; /* Non-execute area */
|
|
|
|
};
|
|
|
|
|
2012-05-30 12:23:33 +08:00
|
|
|
/* Common routines used by software and hardware TLBs emulation */
|
|
|
|
static inline int pte_is_valid(target_ulong pte0)
|
|
|
|
{
|
|
|
|
return pte0 & 0x80000000 ? 1 : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void pte_invalidate(target_ulong *pte0)
|
|
|
|
{
|
|
|
|
*pte0 &= ~0x80000000;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define PTE_PTEM_MASK 0x7FFFFFBF
|
|
|
|
#define PTE_CHECK_MASK (TARGET_PAGE_MASK | 0x7B)
|
|
|
|
|
2013-03-12 08:31:14 +08:00
|
|
|
static int pp_check(int key, int pp, int nx)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
|
|
|
int access;
|
|
|
|
|
|
|
|
/* Compute access rights */
|
|
|
|
access = 0;
|
|
|
|
if (key == 0) {
|
|
|
|
switch (pp) {
|
|
|
|
case 0x0:
|
|
|
|
case 0x1:
|
|
|
|
case 0x2:
|
|
|
|
access |= PAGE_WRITE;
|
|
|
|
/* No break here */
|
|
|
|
case 0x3:
|
|
|
|
access |= PAGE_READ;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
switch (pp) {
|
|
|
|
case 0x0:
|
|
|
|
access = 0;
|
|
|
|
break;
|
|
|
|
case 0x1:
|
|
|
|
case 0x3:
|
|
|
|
access = PAGE_READ;
|
|
|
|
break;
|
|
|
|
case 0x2:
|
|
|
|
access = PAGE_READ | PAGE_WRITE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (nx == 0) {
|
|
|
|
access |= PAGE_EXEC;
|
|
|
|
}
|
|
|
|
|
|
|
|
return access;
|
|
|
|
}
|
|
|
|
|
2013-03-12 08:31:14 +08:00
|
|
|
static int check_prot(int prot, int rw, int access_type)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (access_type == ACCESS_CODE) {
|
|
|
|
if (prot & PAGE_EXEC) {
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
|
|
|
ret = -2;
|
|
|
|
}
|
|
|
|
} else if (rw) {
|
|
|
|
if (prot & PAGE_WRITE) {
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
|
|
|
ret = -2;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (prot & PAGE_READ) {
|
|
|
|
ret = 0;
|
|
|
|
} else {
|
|
|
|
ret = -2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-03-12 08:31:07 +08:00
|
|
|
static inline int ppc6xx_tlb_pte_check(mmu_ctx_t *ctx, target_ulong pte0,
|
|
|
|
target_ulong pte1, int h, int rw, int type)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
|
|
|
target_ulong ptem, mmask;
|
|
|
|
int access, ret, pteh, ptev, pp;
|
|
|
|
|
|
|
|
ret = -1;
|
|
|
|
/* Check validity and table match */
|
2013-03-12 08:31:07 +08:00
|
|
|
ptev = pte_is_valid(pte0);
|
|
|
|
pteh = (pte0 >> 6) & 1;
|
2012-05-30 12:23:33 +08:00
|
|
|
if (ptev && h == pteh) {
|
|
|
|
/* Check vsid & api */
|
2013-03-12 08:31:07 +08:00
|
|
|
ptem = pte0 & PTE_PTEM_MASK;
|
|
|
|
mmask = PTE_CHECK_MASK;
|
|
|
|
pp = pte1 & 0x00000003;
|
2012-05-30 12:23:33 +08:00
|
|
|
if (ptem == ctx->ptem) {
|
2012-10-23 18:30:10 +08:00
|
|
|
if (ctx->raddr != (hwaddr)-1ULL) {
|
2012-05-30 12:23:33 +08:00
|
|
|
/* all matches should have equal RPN, WIMG & PP */
|
|
|
|
if ((ctx->raddr & mmask) != (pte1 & mmask)) {
|
2015-11-13 20:34:23 +08:00
|
|
|
qemu_log_mask(CPU_LOG_MMU, "Bad RPN/WIMG/PP\n");
|
2012-05-30 12:23:33 +08:00
|
|
|
return -3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Compute access rights */
|
|
|
|
access = pp_check(ctx->key, pp, ctx->nx);
|
|
|
|
/* Keep the matching PTE informations */
|
|
|
|
ctx->raddr = pte1;
|
|
|
|
ctx->prot = access;
|
|
|
|
ret = check_prot(ctx->prot, rw, type);
|
|
|
|
if (ret == 0) {
|
|
|
|
/* Access granted */
|
2014-12-14 00:48:18 +08:00
|
|
|
qemu_log_mask(CPU_LOG_MMU, "PTE access granted !\n");
|
2012-05-30 12:23:33 +08:00
|
|
|
} else {
|
|
|
|
/* Access right violation */
|
2014-12-14 00:48:18 +08:00
|
|
|
qemu_log_mask(CPU_LOG_MMU, "PTE access rejected\n");
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-03-12 08:31:14 +08:00
|
|
|
static int pte_update_flags(mmu_ctx_t *ctx, target_ulong *pte1p,
|
|
|
|
int ret, int rw)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
|
|
|
int store = 0;
|
|
|
|
|
|
|
|
/* Update page flags */
|
|
|
|
if (!(*pte1p & 0x00000100)) {
|
|
|
|
/* Update accessed flag */
|
|
|
|
*pte1p |= 0x00000100;
|
|
|
|
store = 1;
|
|
|
|
}
|
|
|
|
if (!(*pte1p & 0x00000080)) {
|
|
|
|
if (rw == 1 && ret == 0) {
|
|
|
|
/* Update changed flag */
|
|
|
|
*pte1p |= 0x00000080;
|
|
|
|
store = 1;
|
|
|
|
} else {
|
|
|
|
/* Force page fault for first write access */
|
|
|
|
ctx->prot &= ~PAGE_WRITE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return store;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Software driven TLB helpers */
|
|
|
|
static inline int ppc6xx_tlb_getnum(CPUPPCState *env, target_ulong eaddr,
|
|
|
|
int way, int is_code)
|
|
|
|
{
|
|
|
|
int nr;
|
|
|
|
|
|
|
|
/* Select TLB num in a way from address */
|
|
|
|
nr = (eaddr >> TARGET_PAGE_BITS) & (env->tlb_per_way - 1);
|
|
|
|
/* Select TLB way */
|
|
|
|
nr += env->tlb_per_way * way;
|
|
|
|
/* 6xx have separate TLBs for instructions and data */
|
|
|
|
if (is_code && env->id_tlbs == 1) {
|
|
|
|
nr += env->nb_tlb;
|
|
|
|
}
|
|
|
|
|
|
|
|
return nr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void ppc6xx_tlb_invalidate_all(CPUPPCState *env)
|
|
|
|
{
|
2013-09-04 08:19:44 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
2012-05-30 12:23:33 +08:00
|
|
|
ppc6xx_tlb_t *tlb;
|
|
|
|
int nr, max;
|
|
|
|
|
|
|
|
/* LOG_SWTLB("Invalidate all TLBs\n"); */
|
|
|
|
/* Invalidate all defined software TLB */
|
|
|
|
max = env->nb_tlb;
|
|
|
|
if (env->id_tlbs == 1) {
|
|
|
|
max *= 2;
|
|
|
|
}
|
|
|
|
for (nr = 0; nr < max; nr++) {
|
|
|
|
tlb = &env->tlb.tlb6[nr];
|
|
|
|
pte_invalidate(&tlb->pte0);
|
|
|
|
}
|
2016-11-14 22:17:28 +08:00
|
|
|
tlb_flush(CPU(cpu));
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void ppc6xx_tlb_invalidate_virt2(CPUPPCState *env,
|
|
|
|
target_ulong eaddr,
|
|
|
|
int is_code, int match_epn)
|
|
|
|
{
|
|
|
|
#if !defined(FLUSH_ALL_TLBS)
|
2013-09-04 07:29:02 +08:00
|
|
|
CPUState *cs = CPU(ppc_env_get_cpu(env));
|
2012-05-30 12:23:33 +08:00
|
|
|
ppc6xx_tlb_t *tlb;
|
|
|
|
int way, nr;
|
|
|
|
|
|
|
|
/* Invalidate ITLB + DTLB, all ways */
|
|
|
|
for (way = 0; way < env->nb_ways; way++) {
|
|
|
|
nr = ppc6xx_tlb_getnum(env, eaddr, way, is_code);
|
|
|
|
tlb = &env->tlb.tlb6[nr];
|
|
|
|
if (pte_is_valid(tlb->pte0) && (match_epn == 0 || eaddr == tlb->EPN)) {
|
|
|
|
LOG_SWTLB("TLB invalidate %d/%d " TARGET_FMT_lx "\n", nr,
|
|
|
|
env->nb_tlb, eaddr);
|
|
|
|
pte_invalidate(&tlb->pte0);
|
2013-09-04 07:29:02 +08:00
|
|
|
tlb_flush_page(cs, tlb->EPN);
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
/* XXX: PowerPC specification say this is valid as well */
|
|
|
|
ppc6xx_tlb_invalidate_all(env);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void ppc6xx_tlb_invalidate_virt(CPUPPCState *env,
|
|
|
|
target_ulong eaddr, int is_code)
|
|
|
|
{
|
|
|
|
ppc6xx_tlb_invalidate_virt2(env, eaddr, is_code, 0);
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:34 +08:00
|
|
|
static void ppc6xx_tlb_store(CPUPPCState *env, target_ulong EPN, int way,
|
|
|
|
int is_code, target_ulong pte0, target_ulong pte1)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
|
|
|
ppc6xx_tlb_t *tlb;
|
|
|
|
int nr;
|
|
|
|
|
|
|
|
nr = ppc6xx_tlb_getnum(env, EPN, way, is_code);
|
|
|
|
tlb = &env->tlb.tlb6[nr];
|
|
|
|
LOG_SWTLB("Set TLB %d/%d EPN " TARGET_FMT_lx " PTE0 " TARGET_FMT_lx
|
|
|
|
" PTE1 " TARGET_FMT_lx "\n", nr, env->nb_tlb, EPN, pte0, pte1);
|
|
|
|
/* Invalidate any pending reference in QEMU for this virtual address */
|
|
|
|
ppc6xx_tlb_invalidate_virt2(env, EPN, is_code, 1);
|
|
|
|
tlb->pte0 = pte0;
|
|
|
|
tlb->pte1 = pte1;
|
|
|
|
tlb->EPN = EPN;
|
|
|
|
/* Store last way for LRU mechanism */
|
|
|
|
env->last_way = way;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int ppc6xx_tlb_check(CPUPPCState *env, mmu_ctx_t *ctx,
|
|
|
|
target_ulong eaddr, int rw, int access_type)
|
|
|
|
{
|
|
|
|
ppc6xx_tlb_t *tlb;
|
|
|
|
int nr, best, way;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
best = -1;
|
|
|
|
ret = -1; /* No TLB found */
|
|
|
|
for (way = 0; way < env->nb_ways; way++) {
|
|
|
|
nr = ppc6xx_tlb_getnum(env, eaddr, way,
|
|
|
|
access_type == ACCESS_CODE ? 1 : 0);
|
|
|
|
tlb = &env->tlb.tlb6[nr];
|
|
|
|
/* This test "emulates" the PTE index match for hardware TLBs */
|
|
|
|
if ((eaddr & TARGET_PAGE_MASK) != tlb->EPN) {
|
|
|
|
LOG_SWTLB("TLB %d/%d %s [" TARGET_FMT_lx " " TARGET_FMT_lx
|
|
|
|
"] <> " TARGET_FMT_lx "\n", nr, env->nb_tlb,
|
|
|
|
pte_is_valid(tlb->pte0) ? "valid" : "inval",
|
|
|
|
tlb->EPN, tlb->EPN + TARGET_PAGE_SIZE, eaddr);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
LOG_SWTLB("TLB %d/%d %s " TARGET_FMT_lx " <> " TARGET_FMT_lx " "
|
|
|
|
TARGET_FMT_lx " %c %c\n", nr, env->nb_tlb,
|
|
|
|
pte_is_valid(tlb->pte0) ? "valid" : "inval",
|
|
|
|
tlb->EPN, eaddr, tlb->pte1,
|
|
|
|
rw ? 'S' : 'L', access_type == ACCESS_CODE ? 'I' : 'D');
|
2013-03-12 08:31:07 +08:00
|
|
|
switch (ppc6xx_tlb_pte_check(ctx, tlb->pte0, tlb->pte1, 0, rw, access_type)) {
|
2012-05-30 12:23:33 +08:00
|
|
|
case -3:
|
|
|
|
/* TLB inconsistency */
|
|
|
|
return -1;
|
|
|
|
case -2:
|
|
|
|
/* Access violation */
|
|
|
|
ret = -2;
|
|
|
|
best = nr;
|
|
|
|
break;
|
|
|
|
case -1:
|
|
|
|
default:
|
|
|
|
/* No match */
|
|
|
|
break;
|
|
|
|
case 0:
|
|
|
|
/* access granted */
|
|
|
|
/* XXX: we should go on looping to check all TLBs consistency
|
|
|
|
* but we can speed-up the whole thing as the
|
|
|
|
* result would be undefined if TLBs are not consistent.
|
|
|
|
*/
|
|
|
|
ret = 0;
|
|
|
|
best = nr;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (best != -1) {
|
|
|
|
done:
|
|
|
|
LOG_SWTLB("found TLB at addr " TARGET_FMT_plx " prot=%01x ret=%d\n",
|
|
|
|
ctx->raddr & TARGET_PAGE_MASK, ctx->prot, ret);
|
|
|
|
/* Update page flags */
|
|
|
|
pte_update_flags(ctx, &env->tlb.tlb6[best].pte1, ret, rw);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Perform BAT hit & translation */
|
|
|
|
static inline void bat_size_prot(CPUPPCState *env, target_ulong *blp,
|
|
|
|
int *validp, int *protp, target_ulong *BATu,
|
|
|
|
target_ulong *BATl)
|
|
|
|
{
|
|
|
|
target_ulong bl;
|
|
|
|
int pp, valid, prot;
|
|
|
|
|
|
|
|
bl = (*BATu & 0x00001FFC) << 15;
|
|
|
|
valid = 0;
|
|
|
|
prot = 0;
|
|
|
|
if (((msr_pr == 0) && (*BATu & 0x00000002)) ||
|
|
|
|
((msr_pr != 0) && (*BATu & 0x00000001))) {
|
|
|
|
valid = 1;
|
|
|
|
pp = *BATl & 0x00000003;
|
|
|
|
if (pp != 0) {
|
|
|
|
prot = PAGE_READ | PAGE_EXEC;
|
|
|
|
if (pp == 0x2) {
|
|
|
|
prot |= PAGE_WRITE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
*blp = bl;
|
|
|
|
*validp = valid;
|
|
|
|
*protp = prot;
|
|
|
|
}
|
|
|
|
|
2013-03-12 08:31:16 +08:00
|
|
|
static int get_bat_6xx_tlb(CPUPPCState *env, mmu_ctx_t *ctx,
|
|
|
|
target_ulong virtual, int rw, int type)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
|
|
|
target_ulong *BATlt, *BATut, *BATu, *BATl;
|
|
|
|
target_ulong BEPIl, BEPIu, bl;
|
|
|
|
int i, valid, prot;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
LOG_BATS("%s: %cBAT v " TARGET_FMT_lx "\n", __func__,
|
|
|
|
type == ACCESS_CODE ? 'I' : 'D', virtual);
|
|
|
|
switch (type) {
|
|
|
|
case ACCESS_CODE:
|
|
|
|
BATlt = env->IBAT[1];
|
|
|
|
BATut = env->IBAT[0];
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
BATlt = env->DBAT[1];
|
|
|
|
BATut = env->DBAT[0];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
for (i = 0; i < env->nb_BATs; i++) {
|
|
|
|
BATu = &BATut[i];
|
|
|
|
BATl = &BATlt[i];
|
|
|
|
BEPIu = *BATu & 0xF0000000;
|
|
|
|
BEPIl = *BATu & 0x0FFE0000;
|
2013-03-12 08:31:16 +08:00
|
|
|
bat_size_prot(env, &bl, &valid, &prot, BATu, BATl);
|
2012-05-30 12:23:33 +08:00
|
|
|
LOG_BATS("%s: %cBAT%d v " TARGET_FMT_lx " BATu " TARGET_FMT_lx
|
|
|
|
" BATl " TARGET_FMT_lx "\n", __func__,
|
|
|
|
type == ACCESS_CODE ? 'I' : 'D', i, virtual, *BATu, *BATl);
|
|
|
|
if ((virtual & 0xF0000000) == BEPIu &&
|
|
|
|
((virtual & 0x0FFE0000) & ~bl) == BEPIl) {
|
|
|
|
/* BAT matches */
|
|
|
|
if (valid != 0) {
|
|
|
|
/* Get physical address */
|
|
|
|
ctx->raddr = (*BATl & 0xF0000000) |
|
|
|
|
((virtual & 0x0FFE0000 & bl) | (*BATl & 0x0FFE0000)) |
|
|
|
|
(virtual & 0x0001F000);
|
|
|
|
/* Compute access rights */
|
|
|
|
ctx->prot = prot;
|
|
|
|
ret = check_prot(ctx->prot, rw, type);
|
|
|
|
if (ret == 0) {
|
|
|
|
LOG_BATS("BAT %d match: r " TARGET_FMT_plx " prot=%c%c\n",
|
|
|
|
i, ctx->raddr, ctx->prot & PAGE_READ ? 'R' : '-',
|
|
|
|
ctx->prot & PAGE_WRITE ? 'W' : '-');
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (ret < 0) {
|
|
|
|
#if defined(DEBUG_BATS)
|
|
|
|
if (qemu_log_enabled()) {
|
|
|
|
LOG_BATS("no BAT match for " TARGET_FMT_lx ":\n", virtual);
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
BATu = &BATut[i];
|
|
|
|
BATl = &BATlt[i];
|
|
|
|
BEPIu = *BATu & 0xF0000000;
|
|
|
|
BEPIl = *BATu & 0x0FFE0000;
|
|
|
|
bl = (*BATu & 0x00001FFC) << 15;
|
|
|
|
LOG_BATS("%s: %cBAT%d v " TARGET_FMT_lx " BATu " TARGET_FMT_lx
|
|
|
|
" BATl " TARGET_FMT_lx "\n\t" TARGET_FMT_lx " "
|
|
|
|
TARGET_FMT_lx " " TARGET_FMT_lx "\n",
|
|
|
|
__func__, type == ACCESS_CODE ? 'I' : 'D', i, virtual,
|
|
|
|
*BATu, *BATl, BEPIu, BEPIl, bl);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
/* No hit */
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Perform segment based translation */
|
2013-03-12 08:31:09 +08:00
|
|
|
static inline int get_segment_6xx_tlb(CPUPPCState *env, mmu_ctx_t *ctx,
|
|
|
|
target_ulong eaddr, int rw, int type)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
target/ppc: Eliminate htab_base and htab_mask variables
CPUPPCState includes fields htab_base and htab_mask which store the base
address (GPA) and size (as a mask) of the guest's hashed page table (HPT).
These are set when the SDR1 register is updated.
Keeping these in sync with the SDR1 is actually a little bit fiddly, and
probably not useful for performance, since keeping them expands the size of
CPUPPCState. It also makes some upcoming changes harder to implement.
This patch removes these fields, in favour of calculating them directly
from the SDR1 contents when necessary.
This does make a change to the behaviour of attempting to write a bad value
(invalid HPT size) to the SDR1 with an mtspr instruction. Previously, the
bad value would be stored in SDR1 and could be retrieved with a later
mfspr, but the HPT size as used by the softmmu would be, clamped to the
allowed values. Now, writing a bad value is treated as a no-op. An error
message is printed in both new and old versions.
I'm not sure which behaviour, if either, matches real hardware. I don't
think it matters that much, since it's pretty clear that if an OS writes
a bad value to SDR1, it's not going to boot.
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
Reviewed-by: Alexey Kardashevskiy <aik@ozlabs.ru>
2017-02-24 13:36:44 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
2012-10-23 18:30:10 +08:00
|
|
|
hwaddr hash;
|
2012-05-30 12:23:33 +08:00
|
|
|
target_ulong vsid;
|
|
|
|
int ds, pr, target_page_bits;
|
2013-03-12 08:31:09 +08:00
|
|
|
int ret;
|
|
|
|
target_ulong sr, pgidx;
|
2012-05-30 12:23:33 +08:00
|
|
|
|
|
|
|
pr = msr_pr;
|
|
|
|
ctx->eaddr = eaddr;
|
|
|
|
|
2013-03-12 08:31:09 +08:00
|
|
|
sr = env->sr[eaddr >> 28];
|
|
|
|
ctx->key = (((sr & 0x20000000) && (pr != 0)) ||
|
|
|
|
((sr & 0x40000000) && (pr == 0))) ? 1 : 0;
|
|
|
|
ds = sr & 0x80000000 ? 1 : 0;
|
|
|
|
ctx->nx = sr & 0x10000000 ? 1 : 0;
|
|
|
|
vsid = sr & 0x00FFFFFF;
|
|
|
|
target_page_bits = TARGET_PAGE_BITS;
|
2014-12-14 00:48:18 +08:00
|
|
|
qemu_log_mask(CPU_LOG_MMU,
|
|
|
|
"Check segment v=" TARGET_FMT_lx " %d " TARGET_FMT_lx
|
|
|
|
" nip=" TARGET_FMT_lx " lr=" TARGET_FMT_lx
|
2013-03-12 08:31:09 +08:00
|
|
|
" ir=%d dr=%d pr=%d %d t=%d\n",
|
|
|
|
eaddr, (int)(eaddr >> 28), sr, env->nip, env->lr, (int)msr_ir,
|
|
|
|
(int)msr_dr, pr != 0 ? 1 : 0, rw, type);
|
|
|
|
pgidx = (eaddr & ~SEGMENT_MASK_256M) >> target_page_bits;
|
|
|
|
hash = vsid ^ pgidx;
|
|
|
|
ctx->ptem = (vsid << 7) | (pgidx >> 10);
|
2012-05-30 12:23:33 +08:00
|
|
|
|
2014-12-14 00:48:18 +08:00
|
|
|
qemu_log_mask(CPU_LOG_MMU,
|
|
|
|
"pte segment: key=%d ds %d nx %d vsid " TARGET_FMT_lx "\n",
|
2012-05-30 12:23:33 +08:00
|
|
|
ctx->key, ds, ctx->nx, vsid);
|
|
|
|
ret = -1;
|
|
|
|
if (!ds) {
|
|
|
|
/* Check if instruction fetch is allowed, if needed */
|
|
|
|
if (type != ACCESS_CODE || ctx->nx == 0) {
|
|
|
|
/* Page address translation */
|
2014-12-14 00:48:18 +08:00
|
|
|
qemu_log_mask(CPU_LOG_MMU, "htab_base " TARGET_FMT_plx
|
|
|
|
" htab_mask " TARGET_FMT_plx
|
2012-05-30 12:23:33 +08:00
|
|
|
" hash " TARGET_FMT_plx "\n",
|
target/ppc: Eliminate htab_base and htab_mask variables
CPUPPCState includes fields htab_base and htab_mask which store the base
address (GPA) and size (as a mask) of the guest's hashed page table (HPT).
These are set when the SDR1 register is updated.
Keeping these in sync with the SDR1 is actually a little bit fiddly, and
probably not useful for performance, since keeping them expands the size of
CPUPPCState. It also makes some upcoming changes harder to implement.
This patch removes these fields, in favour of calculating them directly
from the SDR1 contents when necessary.
This does make a change to the behaviour of attempting to write a bad value
(invalid HPT size) to the SDR1 with an mtspr instruction. Previously, the
bad value would be stored in SDR1 and could be retrieved with a later
mfspr, but the HPT size as used by the softmmu would be, clamped to the
allowed values. Now, writing a bad value is treated as a no-op. An error
message is printed in both new and old versions.
I'm not sure which behaviour, if either, matches real hardware. I don't
think it matters that much, since it's pretty clear that if an OS writes
a bad value to SDR1, it's not going to boot.
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
Reviewed-by: Alexey Kardashevskiy <aik@ozlabs.ru>
2017-02-24 13:36:44 +08:00
|
|
|
ppc_hash32_hpt_base(cpu), ppc_hash32_hpt_mask(cpu), hash);
|
2012-05-30 12:23:33 +08:00
|
|
|
ctx->hash[0] = hash;
|
|
|
|
ctx->hash[1] = ~hash;
|
|
|
|
|
|
|
|
/* Initialize real address with an invalid value */
|
2012-10-23 18:30:10 +08:00
|
|
|
ctx->raddr = (hwaddr)-1ULL;
|
2013-03-12 08:31:09 +08:00
|
|
|
/* Software TLB search */
|
|
|
|
ret = ppc6xx_tlb_check(env, ctx, eaddr, rw, type);
|
2012-05-30 12:23:33 +08:00
|
|
|
#if defined(DUMP_PAGE_TABLES)
|
2016-06-03 21:58:09 +08:00
|
|
|
if (qemu_loglevel_mask(CPU_LOG_MMU)) {
|
|
|
|
CPUState *cs = ENV_GET_CPU(env);
|
2012-10-23 18:30:10 +08:00
|
|
|
hwaddr curaddr;
|
2012-05-30 12:23:33 +08:00
|
|
|
uint32_t a0, a1, a2, a3;
|
|
|
|
|
|
|
|
qemu_log("Page table: " TARGET_FMT_plx " len " TARGET_FMT_plx
|
target/ppc: Eliminate htab_base and htab_mask variables
CPUPPCState includes fields htab_base and htab_mask which store the base
address (GPA) and size (as a mask) of the guest's hashed page table (HPT).
These are set when the SDR1 register is updated.
Keeping these in sync with the SDR1 is actually a little bit fiddly, and
probably not useful for performance, since keeping them expands the size of
CPUPPCState. It also makes some upcoming changes harder to implement.
This patch removes these fields, in favour of calculating them directly
from the SDR1 contents when necessary.
This does make a change to the behaviour of attempting to write a bad value
(invalid HPT size) to the SDR1 with an mtspr instruction. Previously, the
bad value would be stored in SDR1 and could be retrieved with a later
mfspr, but the HPT size as used by the softmmu would be, clamped to the
allowed values. Now, writing a bad value is treated as a no-op. An error
message is printed in both new and old versions.
I'm not sure which behaviour, if either, matches real hardware. I don't
think it matters that much, since it's pretty clear that if an OS writes
a bad value to SDR1, it's not going to boot.
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
Reviewed-by: Alexey Kardashevskiy <aik@ozlabs.ru>
2017-02-24 13:36:44 +08:00
|
|
|
"\n", ppc_hash32_hpt_base(cpu),
|
|
|
|
ppc_hash32_hpt_mask(env) + 0x80);
|
|
|
|
for (curaddr = ppc_hash32_hpt_base(cpu);
|
|
|
|
curaddr < (ppc_hash32_hpt_base(cpu)
|
|
|
|
+ ppc_hash32_hpt_mask(cpu) + 0x80);
|
2012-05-30 12:23:33 +08:00
|
|
|
curaddr += 16) {
|
2016-06-03 21:58:09 +08:00
|
|
|
a0 = ldl_phys(cs->as, curaddr);
|
|
|
|
a1 = ldl_phys(cs->as, curaddr + 4);
|
|
|
|
a2 = ldl_phys(cs->as, curaddr + 8);
|
|
|
|
a3 = ldl_phys(cs->as, curaddr + 12);
|
2012-05-30 12:23:33 +08:00
|
|
|
if (a0 != 0 || a1 != 0 || a2 != 0 || a3 != 0) {
|
|
|
|
qemu_log(TARGET_FMT_plx ": %08x %08x %08x %08x\n",
|
|
|
|
curaddr, a0, a1, a2, a3);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
} else {
|
2014-12-14 00:48:18 +08:00
|
|
|
qemu_log_mask(CPU_LOG_MMU, "No access allowed\n");
|
2012-05-30 12:23:33 +08:00
|
|
|
ret = -3;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
target_ulong sr;
|
|
|
|
|
2014-12-14 00:48:18 +08:00
|
|
|
qemu_log_mask(CPU_LOG_MMU, "direct store...\n");
|
2012-05-30 12:23:33 +08:00
|
|
|
/* Direct-store segment : absolutely *BUGGY* for now */
|
|
|
|
|
|
|
|
/* Direct-store implies a 32-bit MMU.
|
|
|
|
* Check the Segment Register's bus unit ID (BUID).
|
|
|
|
*/
|
|
|
|
sr = env->sr[eaddr >> 28];
|
|
|
|
if ((sr & 0x1FF00000) >> 20 == 0x07f) {
|
|
|
|
/* Memory-forced I/O controller interface access */
|
|
|
|
/* If T=1 and BUID=x'07F', the 601 performs a memory access
|
|
|
|
* to SR[28-31] LA[4-31], bypassing all protection mechanisms.
|
|
|
|
*/
|
|
|
|
ctx->raddr = ((sr & 0xF) << 28) | (eaddr & 0x0FFFFFFF);
|
|
|
|
ctx->prot = PAGE_READ | PAGE_WRITE | PAGE_EXEC;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case ACCESS_INT:
|
|
|
|
/* Integer load/store : only access allowed */
|
|
|
|
break;
|
|
|
|
case ACCESS_CODE:
|
|
|
|
/* No code fetch is allowed in direct-store areas */
|
|
|
|
return -4;
|
|
|
|
case ACCESS_FLOAT:
|
|
|
|
/* Floating point load/store */
|
|
|
|
return -4;
|
|
|
|
case ACCESS_RES:
|
|
|
|
/* lwarx, ldarx or srwcx. */
|
|
|
|
return -4;
|
|
|
|
case ACCESS_CACHE:
|
|
|
|
/* dcba, dcbt, dcbtst, dcbf, dcbi, dcbst, dcbz, or icbi */
|
|
|
|
/* Should make the instruction do no-op.
|
|
|
|
* As it already do no-op, it's quite easy :-)
|
|
|
|
*/
|
|
|
|
ctx->raddr = eaddr;
|
|
|
|
return 0;
|
|
|
|
case ACCESS_EXT:
|
|
|
|
/* eciwx or ecowx */
|
|
|
|
return -4;
|
|
|
|
default:
|
2015-11-13 20:34:23 +08:00
|
|
|
qemu_log_mask(CPU_LOG_MMU, "ERROR: instruction should not need "
|
|
|
|
"address translation\n");
|
2012-05-30 12:23:33 +08:00
|
|
|
return -4;
|
|
|
|
}
|
|
|
|
if ((rw == 1 || ctx->key != 1) && (rw == 0 || ctx->key != 0)) {
|
|
|
|
ctx->raddr = eaddr;
|
|
|
|
ret = 2;
|
|
|
|
} else {
|
|
|
|
ret = -2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Generic TLB check function for embedded PowerPC implementations */
|
2012-05-30 12:23:34 +08:00
|
|
|
static int ppcemb_tlb_check(CPUPPCState *env, ppcemb_tlb_t *tlb,
|
2012-10-23 18:30:10 +08:00
|
|
|
hwaddr *raddrp,
|
2012-05-30 12:23:34 +08:00
|
|
|
target_ulong address, uint32_t pid, int ext,
|
|
|
|
int i)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
|
|
|
target_ulong mask;
|
|
|
|
|
|
|
|
/* Check valid flag */
|
|
|
|
if (!(tlb->prot & PAGE_VALID)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
mask = ~(tlb->size - 1);
|
|
|
|
LOG_SWTLB("%s: TLB %d address " TARGET_FMT_lx " PID %u <=> " TARGET_FMT_lx
|
|
|
|
" " TARGET_FMT_lx " %u %x\n", __func__, i, address, pid, tlb->EPN,
|
|
|
|
mask, (uint32_t)tlb->PID, tlb->prot);
|
|
|
|
/* Check PID */
|
|
|
|
if (tlb->PID != 0 && tlb->PID != pid) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
/* Check effective address */
|
|
|
|
if ((address & mask) != tlb->EPN) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
*raddrp = (tlb->RPN & mask) | (address & ~mask);
|
|
|
|
if (ext) {
|
|
|
|
/* Extend the physical address to 36 bits */
|
2012-10-04 18:36:04 +08:00
|
|
|
*raddrp |= (uint64_t)(tlb->RPN & 0xF) << 32;
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Generic TLB search function for PowerPC embedded implementations */
|
2012-05-30 12:23:34 +08:00
|
|
|
static int ppcemb_tlb_search(CPUPPCState *env, target_ulong address,
|
|
|
|
uint32_t pid)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
|
|
|
ppcemb_tlb_t *tlb;
|
2012-10-23 18:30:10 +08:00
|
|
|
hwaddr raddr;
|
2012-05-30 12:23:33 +08:00
|
|
|
int i, ret;
|
|
|
|
|
|
|
|
/* Default return value is no match */
|
|
|
|
ret = -1;
|
|
|
|
for (i = 0; i < env->nb_tlb; i++) {
|
|
|
|
tlb = &env->tlb.tlbe[i];
|
|
|
|
if (ppcemb_tlb_check(env, tlb, &raddr, address, pid, 0, i) == 0) {
|
|
|
|
ret = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Helpers specific to PowerPC 40x implementations */
|
|
|
|
static inline void ppc4xx_tlb_invalidate_all(CPUPPCState *env)
|
|
|
|
{
|
2013-09-04 08:19:44 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
2012-05-30 12:23:33 +08:00
|
|
|
ppcemb_tlb_t *tlb;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < env->nb_tlb; i++) {
|
|
|
|
tlb = &env->tlb.tlbe[i];
|
|
|
|
tlb->prot &= ~PAGE_VALID;
|
|
|
|
}
|
2016-11-14 22:17:28 +08:00
|
|
|
tlb_flush(CPU(cpu));
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static int mmu40x_get_physical_address(CPUPPCState *env, mmu_ctx_t *ctx,
|
|
|
|
target_ulong address, int rw,
|
|
|
|
int access_type)
|
|
|
|
{
|
|
|
|
ppcemb_tlb_t *tlb;
|
2012-10-23 18:30:10 +08:00
|
|
|
hwaddr raddr;
|
2012-05-30 12:23:33 +08:00
|
|
|
int i, ret, zsel, zpr, pr;
|
|
|
|
|
|
|
|
ret = -1;
|
2012-10-23 18:30:10 +08:00
|
|
|
raddr = (hwaddr)-1ULL;
|
2012-05-30 12:23:33 +08:00
|
|
|
pr = msr_pr;
|
|
|
|
for (i = 0; i < env->nb_tlb; i++) {
|
|
|
|
tlb = &env->tlb.tlbe[i];
|
|
|
|
if (ppcemb_tlb_check(env, tlb, &raddr, address,
|
|
|
|
env->spr[SPR_40x_PID], 0, i) < 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
zsel = (tlb->attr >> 4) & 0xF;
|
|
|
|
zpr = (env->spr[SPR_40x_ZPR] >> (30 - (2 * zsel))) & 0x3;
|
|
|
|
LOG_SWTLB("%s: TLB %d zsel %d zpr %d rw %d attr %08x\n",
|
|
|
|
__func__, i, zsel, zpr, rw, tlb->attr);
|
|
|
|
/* Check execute enable bit */
|
|
|
|
switch (zpr) {
|
|
|
|
case 0x2:
|
|
|
|
if (pr != 0) {
|
|
|
|
goto check_perms;
|
|
|
|
}
|
|
|
|
/* No break here */
|
|
|
|
case 0x3:
|
|
|
|
/* All accesses granted */
|
|
|
|
ctx->prot = PAGE_READ | PAGE_WRITE | PAGE_EXEC;
|
|
|
|
ret = 0;
|
|
|
|
break;
|
|
|
|
case 0x0:
|
|
|
|
if (pr != 0) {
|
|
|
|
/* Raise Zone protection fault. */
|
|
|
|
env->spr[SPR_40x_ESR] = 1 << 22;
|
|
|
|
ctx->prot = 0;
|
|
|
|
ret = -2;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* No break here */
|
|
|
|
case 0x1:
|
|
|
|
check_perms:
|
|
|
|
/* Check from TLB entry */
|
|
|
|
ctx->prot = tlb->prot;
|
|
|
|
ret = check_prot(ctx->prot, rw, access_type);
|
|
|
|
if (ret == -2) {
|
|
|
|
env->spr[SPR_40x_ESR] = 0;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (ret >= 0) {
|
|
|
|
ctx->raddr = raddr;
|
|
|
|
LOG_SWTLB("%s: access granted " TARGET_FMT_lx " => " TARGET_FMT_plx
|
|
|
|
" %d %d\n", __func__, address, ctx->raddr, ctx->prot,
|
|
|
|
ret);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
LOG_SWTLB("%s: access refused " TARGET_FMT_lx " => " TARGET_FMT_plx
|
|
|
|
" %d %d\n", __func__, address, raddr, ctx->prot, ret);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void store_40x_sler(CPUPPCState *env, uint32_t val)
|
|
|
|
{
|
2013-09-03 23:38:47 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
|
|
|
|
2012-05-30 12:23:33 +08:00
|
|
|
/* XXX: TO BE FIXED */
|
|
|
|
if (val != 0x00000000) {
|
2013-09-03 23:38:47 +08:00
|
|
|
cpu_abort(CPU(cpu), "Little-endian regions are not supported by now\n");
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
env->spr[SPR_405_SLER] = val;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int mmubooke_check_tlb(CPUPPCState *env, ppcemb_tlb_t *tlb,
|
2012-10-23 18:30:10 +08:00
|
|
|
hwaddr *raddr, int *prot,
|
2012-05-30 12:23:33 +08:00
|
|
|
target_ulong address, int rw,
|
|
|
|
int access_type, int i)
|
|
|
|
{
|
|
|
|
int ret, prot2;
|
|
|
|
|
|
|
|
if (ppcemb_tlb_check(env, tlb, raddr, address,
|
|
|
|
env->spr[SPR_BOOKE_PID],
|
|
|
|
!env->nb_pids, i) >= 0) {
|
|
|
|
goto found_tlb;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (env->spr[SPR_BOOKE_PID1] &&
|
|
|
|
ppcemb_tlb_check(env, tlb, raddr, address,
|
|
|
|
env->spr[SPR_BOOKE_PID1], 0, i) >= 0) {
|
|
|
|
goto found_tlb;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (env->spr[SPR_BOOKE_PID2] &&
|
|
|
|
ppcemb_tlb_check(env, tlb, raddr, address,
|
|
|
|
env->spr[SPR_BOOKE_PID2], 0, i) >= 0) {
|
|
|
|
goto found_tlb;
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG_SWTLB("%s: TLB entry not found\n", __func__);
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
found_tlb:
|
|
|
|
|
|
|
|
if (msr_pr != 0) {
|
|
|
|
prot2 = tlb->prot & 0xF;
|
|
|
|
} else {
|
|
|
|
prot2 = (tlb->prot >> 4) & 0xF;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check the address space */
|
|
|
|
if (access_type == ACCESS_CODE) {
|
|
|
|
if (msr_ir != (tlb->attr & 1)) {
|
|
|
|
LOG_SWTLB("%s: AS doesn't match\n", __func__);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
*prot = prot2;
|
|
|
|
if (prot2 & PAGE_EXEC) {
|
|
|
|
LOG_SWTLB("%s: good TLB!\n", __func__);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG_SWTLB("%s: no PAGE_EXEC: %x\n", __func__, prot2);
|
|
|
|
ret = -3;
|
|
|
|
} else {
|
|
|
|
if (msr_dr != (tlb->attr & 1)) {
|
|
|
|
LOG_SWTLB("%s: AS doesn't match\n", __func__);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
*prot = prot2;
|
|
|
|
if ((!rw && prot2 & PAGE_READ) || (rw && (prot2 & PAGE_WRITE))) {
|
|
|
|
LOG_SWTLB("%s: found TLB!\n", __func__);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG_SWTLB("%s: PAGE_READ/WRITE doesn't match: %x\n", __func__, prot2);
|
|
|
|
ret = -2;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mmubooke_get_physical_address(CPUPPCState *env, mmu_ctx_t *ctx,
|
|
|
|
target_ulong address, int rw,
|
|
|
|
int access_type)
|
|
|
|
{
|
|
|
|
ppcemb_tlb_t *tlb;
|
2012-10-23 18:30:10 +08:00
|
|
|
hwaddr raddr;
|
2012-05-30 12:23:33 +08:00
|
|
|
int i, ret;
|
|
|
|
|
|
|
|
ret = -1;
|
2012-10-23 18:30:10 +08:00
|
|
|
raddr = (hwaddr)-1ULL;
|
2012-05-30 12:23:33 +08:00
|
|
|
for (i = 0; i < env->nb_tlb; i++) {
|
|
|
|
tlb = &env->tlb.tlbe[i];
|
|
|
|
ret = mmubooke_check_tlb(env, tlb, &raddr, &ctx->prot, address, rw,
|
|
|
|
access_type, i);
|
2017-02-14 19:54:29 +08:00
|
|
|
if (ret != -1) {
|
2012-05-30 12:23:33 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ret >= 0) {
|
|
|
|
ctx->raddr = raddr;
|
|
|
|
LOG_SWTLB("%s: access granted " TARGET_FMT_lx " => " TARGET_FMT_plx
|
|
|
|
" %d %d\n", __func__, address, ctx->raddr, ctx->prot,
|
|
|
|
ret);
|
|
|
|
} else {
|
|
|
|
LOG_SWTLB("%s: access refused " TARGET_FMT_lx " => " TARGET_FMT_plx
|
|
|
|
" %d %d\n", __func__, address, raddr, ctx->prot, ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-10-28 19:04:50 +08:00
|
|
|
static void booke206_flush_tlb(CPUPPCState *env, int flags,
|
|
|
|
const int check_iprot)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
2013-09-04 08:19:44 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
2012-05-30 12:23:33 +08:00
|
|
|
int tlb_size;
|
|
|
|
int i, j;
|
|
|
|
ppcmas_tlb_t *tlb = env->tlb.tlbm;
|
|
|
|
|
|
|
|
for (i = 0; i < BOOKE206_MAX_TLBN; i++) {
|
|
|
|
if (flags & (1 << i)) {
|
|
|
|
tlb_size = booke206_tlb_size(env, i);
|
|
|
|
for (j = 0; j < tlb_size; j++) {
|
|
|
|
if (!check_iprot || !(tlb[j].mas1 & MAS1_IPROT)) {
|
|
|
|
tlb[j].mas1 &= ~MAS1_VALID;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tlb += booke206_tlb_size(env, i);
|
|
|
|
}
|
|
|
|
|
2016-11-14 22:17:28 +08:00
|
|
|
tlb_flush(CPU(cpu));
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
|
2012-10-28 19:04:50 +08:00
|
|
|
static hwaddr booke206_tlb_to_page_size(CPUPPCState *env,
|
|
|
|
ppcmas_tlb_t *tlb)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
|
|
|
int tlbm_size;
|
|
|
|
|
|
|
|
tlbm_size = (tlb->mas1 & MAS1_TSIZE_MASK) >> MAS1_TSIZE_SHIFT;
|
|
|
|
|
|
|
|
return 1024ULL << tlbm_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TLB check function for MAS based SoftTLBs */
|
2013-03-12 08:31:04 +08:00
|
|
|
static int ppcmas_tlb_check(CPUPPCState *env, ppcmas_tlb_t *tlb,
|
2014-07-03 01:09:47 +08:00
|
|
|
hwaddr *raddrp, target_ulong address,
|
|
|
|
uint32_t pid)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
2014-07-03 01:09:47 +08:00
|
|
|
hwaddr mask;
|
2012-05-30 12:23:33 +08:00
|
|
|
uint32_t tlb_pid;
|
|
|
|
|
2014-06-04 06:27:31 +08:00
|
|
|
if (!msr_cm) {
|
|
|
|
/* In 32bit mode we can only address 32bit EAs */
|
|
|
|
address = (uint32_t)address;
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:33 +08:00
|
|
|
/* Check valid flag */
|
|
|
|
if (!(tlb->mas1 & MAS1_VALID)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
mask = ~(booke206_tlb_to_page_size(env, tlb) - 1);
|
|
|
|
LOG_SWTLB("%s: TLB ADDR=0x" TARGET_FMT_lx " PID=0x%x MAS1=0x%x MAS2=0x%"
|
2016-06-03 21:58:09 +08:00
|
|
|
PRIx64 " mask=0x%" HWADDR_PRIx " MAS7_3=0x%" PRIx64 " MAS8=0x%"
|
|
|
|
PRIx32 "\n", __func__, address, pid, tlb->mas1, tlb->mas2, mask,
|
|
|
|
tlb->mas7_3, tlb->mas8);
|
2012-05-30 12:23:33 +08:00
|
|
|
|
|
|
|
/* Check PID */
|
|
|
|
tlb_pid = (tlb->mas1 & MAS1_TID_MASK) >> MAS1_TID_SHIFT;
|
|
|
|
if (tlb_pid != 0 && tlb_pid != pid) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check effective address */
|
|
|
|
if ((address & mask) != (tlb->mas2 & MAS2_EPN_MASK)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (raddrp) {
|
|
|
|
*raddrp = (tlb->mas7_3 & mask) | (address & ~mask);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mmubooke206_check_tlb(CPUPPCState *env, ppcmas_tlb_t *tlb,
|
2012-10-23 18:30:10 +08:00
|
|
|
hwaddr *raddr, int *prot,
|
2012-05-30 12:23:33 +08:00
|
|
|
target_ulong address, int rw,
|
|
|
|
int access_type)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
int prot2 = 0;
|
|
|
|
|
|
|
|
if (ppcmas_tlb_check(env, tlb, raddr, address,
|
|
|
|
env->spr[SPR_BOOKE_PID]) >= 0) {
|
|
|
|
goto found_tlb;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (env->spr[SPR_BOOKE_PID1] &&
|
|
|
|
ppcmas_tlb_check(env, tlb, raddr, address,
|
|
|
|
env->spr[SPR_BOOKE_PID1]) >= 0) {
|
|
|
|
goto found_tlb;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (env->spr[SPR_BOOKE_PID2] &&
|
|
|
|
ppcmas_tlb_check(env, tlb, raddr, address,
|
|
|
|
env->spr[SPR_BOOKE_PID2]) >= 0) {
|
|
|
|
goto found_tlb;
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG_SWTLB("%s: TLB entry not found\n", __func__);
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
found_tlb:
|
|
|
|
|
|
|
|
if (msr_pr != 0) {
|
|
|
|
if (tlb->mas7_3 & MAS3_UR) {
|
|
|
|
prot2 |= PAGE_READ;
|
|
|
|
}
|
|
|
|
if (tlb->mas7_3 & MAS3_UW) {
|
|
|
|
prot2 |= PAGE_WRITE;
|
|
|
|
}
|
|
|
|
if (tlb->mas7_3 & MAS3_UX) {
|
|
|
|
prot2 |= PAGE_EXEC;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (tlb->mas7_3 & MAS3_SR) {
|
|
|
|
prot2 |= PAGE_READ;
|
|
|
|
}
|
|
|
|
if (tlb->mas7_3 & MAS3_SW) {
|
|
|
|
prot2 |= PAGE_WRITE;
|
|
|
|
}
|
|
|
|
if (tlb->mas7_3 & MAS3_SX) {
|
|
|
|
prot2 |= PAGE_EXEC;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check the address space and permissions */
|
|
|
|
if (access_type == ACCESS_CODE) {
|
|
|
|
if (msr_ir != ((tlb->mas1 & MAS1_TS) >> MAS1_TS_SHIFT)) {
|
|
|
|
LOG_SWTLB("%s: AS doesn't match\n", __func__);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
*prot = prot2;
|
|
|
|
if (prot2 & PAGE_EXEC) {
|
|
|
|
LOG_SWTLB("%s: good TLB!\n", __func__);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG_SWTLB("%s: no PAGE_EXEC: %x\n", __func__, prot2);
|
|
|
|
ret = -3;
|
|
|
|
} else {
|
|
|
|
if (msr_dr != ((tlb->mas1 & MAS1_TS) >> MAS1_TS_SHIFT)) {
|
|
|
|
LOG_SWTLB("%s: AS doesn't match\n", __func__);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
*prot = prot2;
|
|
|
|
if ((!rw && prot2 & PAGE_READ) || (rw && (prot2 & PAGE_WRITE))) {
|
|
|
|
LOG_SWTLB("%s: found TLB!\n", __func__);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
LOG_SWTLB("%s: PAGE_READ/WRITE doesn't match: %x\n", __func__, prot2);
|
|
|
|
ret = -2;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int mmubooke206_get_physical_address(CPUPPCState *env, mmu_ctx_t *ctx,
|
|
|
|
target_ulong address, int rw,
|
|
|
|
int access_type)
|
|
|
|
{
|
|
|
|
ppcmas_tlb_t *tlb;
|
2012-10-23 18:30:10 +08:00
|
|
|
hwaddr raddr;
|
2012-05-30 12:23:33 +08:00
|
|
|
int i, j, ret;
|
|
|
|
|
|
|
|
ret = -1;
|
2012-10-23 18:30:10 +08:00
|
|
|
raddr = (hwaddr)-1ULL;
|
2012-05-30 12:23:33 +08:00
|
|
|
|
|
|
|
for (i = 0; i < BOOKE206_MAX_TLBN; i++) {
|
|
|
|
int ways = booke206_tlb_ways(env, i);
|
|
|
|
|
|
|
|
for (j = 0; j < ways; j++) {
|
|
|
|
tlb = booke206_get_tlbm(env, i, address, j);
|
|
|
|
if (!tlb) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
ret = mmubooke206_check_tlb(env, tlb, &raddr, &ctx->prot, address,
|
|
|
|
rw, access_type);
|
|
|
|
if (ret != -1) {
|
|
|
|
goto found_tlb;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
found_tlb:
|
|
|
|
|
|
|
|
if (ret >= 0) {
|
|
|
|
ctx->raddr = raddr;
|
|
|
|
LOG_SWTLB("%s: access granted " TARGET_FMT_lx " => " TARGET_FMT_plx
|
|
|
|
" %d %d\n", __func__, address, ctx->raddr, ctx->prot,
|
|
|
|
ret);
|
|
|
|
} else {
|
|
|
|
LOG_SWTLB("%s: access refused " TARGET_FMT_lx " => " TARGET_FMT_plx
|
|
|
|
" %d %d\n", __func__, address, raddr, ctx->prot, ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char *book3e_tsize_to_str[32] = {
|
|
|
|
"1K", "2K", "4K", "8K", "16K", "32K", "64K", "128K", "256K", "512K",
|
|
|
|
"1M", "2M", "4M", "8M", "16M", "32M", "64M", "128M", "256M", "512M",
|
|
|
|
"1G", "2G", "4G", "8G", "16G", "32G", "64G", "128G", "256G", "512G",
|
|
|
|
"1T", "2T"
|
|
|
|
};
|
|
|
|
|
|
|
|
static void mmubooke_dump_mmu(FILE *f, fprintf_function cpu_fprintf,
|
|
|
|
CPUPPCState *env)
|
|
|
|
{
|
|
|
|
ppcemb_tlb_t *entry;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (kvm_enabled() && !env->kvm_sw_tlb) {
|
|
|
|
cpu_fprintf(f, "Cannot access KVM TLB\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
cpu_fprintf(f, "\nTLB:\n");
|
|
|
|
cpu_fprintf(f, "Effective Physical Size PID Prot "
|
|
|
|
"Attr\n");
|
|
|
|
|
|
|
|
entry = &env->tlb.tlbe[0];
|
|
|
|
for (i = 0; i < env->nb_tlb; i++, entry++) {
|
2012-10-23 18:30:10 +08:00
|
|
|
hwaddr ea, pa;
|
2012-05-30 12:23:33 +08:00
|
|
|
target_ulong mask;
|
|
|
|
uint64_t size = (uint64_t)entry->size;
|
|
|
|
char size_buf[20];
|
|
|
|
|
|
|
|
/* Check valid flag */
|
|
|
|
if (!(entry->prot & PAGE_VALID)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
mask = ~(entry->size - 1);
|
|
|
|
ea = entry->EPN & mask;
|
|
|
|
pa = entry->RPN & mask;
|
|
|
|
/* Extend the physical address to 36 bits */
|
2012-10-23 18:30:10 +08:00
|
|
|
pa |= (hwaddr)(entry->RPN & 0xF) << 32;
|
2012-05-30 12:23:33 +08:00
|
|
|
size /= 1024;
|
|
|
|
if (size >= 1024) {
|
|
|
|
snprintf(size_buf, sizeof(size_buf), "%3" PRId64 "M", size / 1024);
|
|
|
|
} else {
|
|
|
|
snprintf(size_buf, sizeof(size_buf), "%3" PRId64 "k", size);
|
|
|
|
}
|
|
|
|
cpu_fprintf(f, "0x%016" PRIx64 " 0x%016" PRIx64 " %s %-5u %08x %08x\n",
|
|
|
|
(uint64_t)ea, (uint64_t)pa, size_buf, (uint32_t)entry->PID,
|
|
|
|
entry->prot, entry->attr);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static void mmubooke206_dump_one_tlb(FILE *f, fprintf_function cpu_fprintf,
|
|
|
|
CPUPPCState *env, int tlbn, int offset,
|
|
|
|
int tlbsize)
|
|
|
|
{
|
|
|
|
ppcmas_tlb_t *entry;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
cpu_fprintf(f, "\nTLB%d:\n", tlbn);
|
|
|
|
cpu_fprintf(f, "Effective Physical Size TID TS SRWX"
|
|
|
|
" URWX WIMGE U0123\n");
|
|
|
|
|
|
|
|
entry = &env->tlb.tlbm[offset];
|
|
|
|
for (i = 0; i < tlbsize; i++, entry++) {
|
2012-10-23 18:30:10 +08:00
|
|
|
hwaddr ea, pa, size;
|
2012-05-30 12:23:33 +08:00
|
|
|
int tsize;
|
|
|
|
|
|
|
|
if (!(entry->mas1 & MAS1_VALID)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
tsize = (entry->mas1 & MAS1_TSIZE_MASK) >> MAS1_TSIZE_SHIFT;
|
|
|
|
size = 1024ULL << tsize;
|
|
|
|
ea = entry->mas2 & ~(size - 1);
|
|
|
|
pa = entry->mas7_3 & ~(size - 1);
|
|
|
|
|
|
|
|
cpu_fprintf(f, "0x%016" PRIx64 " 0x%016" PRIx64 " %4s %-5u %1u S%c%c%c"
|
|
|
|
"U%c%c%c %c%c%c%c%c U%c%c%c%c\n",
|
|
|
|
(uint64_t)ea, (uint64_t)pa,
|
|
|
|
book3e_tsize_to_str[tsize],
|
|
|
|
(entry->mas1 & MAS1_TID_MASK) >> MAS1_TID_SHIFT,
|
|
|
|
(entry->mas1 & MAS1_TS) >> MAS1_TS_SHIFT,
|
|
|
|
entry->mas7_3 & MAS3_SR ? 'R' : '-',
|
|
|
|
entry->mas7_3 & MAS3_SW ? 'W' : '-',
|
|
|
|
entry->mas7_3 & MAS3_SX ? 'X' : '-',
|
|
|
|
entry->mas7_3 & MAS3_UR ? 'R' : '-',
|
|
|
|
entry->mas7_3 & MAS3_UW ? 'W' : '-',
|
|
|
|
entry->mas7_3 & MAS3_UX ? 'X' : '-',
|
|
|
|
entry->mas2 & MAS2_W ? 'W' : '-',
|
|
|
|
entry->mas2 & MAS2_I ? 'I' : '-',
|
|
|
|
entry->mas2 & MAS2_M ? 'M' : '-',
|
|
|
|
entry->mas2 & MAS2_G ? 'G' : '-',
|
|
|
|
entry->mas2 & MAS2_E ? 'E' : '-',
|
|
|
|
entry->mas7_3 & MAS3_U0 ? '0' : '-',
|
|
|
|
entry->mas7_3 & MAS3_U1 ? '1' : '-',
|
|
|
|
entry->mas7_3 & MAS3_U2 ? '2' : '-',
|
|
|
|
entry->mas7_3 & MAS3_U3 ? '3' : '-');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void mmubooke206_dump_mmu(FILE *f, fprintf_function cpu_fprintf,
|
|
|
|
CPUPPCState *env)
|
|
|
|
{
|
|
|
|
int offset = 0;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (kvm_enabled() && !env->kvm_sw_tlb) {
|
|
|
|
cpu_fprintf(f, "Cannot access KVM TLB\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < BOOKE206_MAX_TLBN; i++) {
|
|
|
|
int size = booke206_tlb_size(env, i);
|
|
|
|
|
|
|
|
if (size == 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
mmubooke206_dump_one_tlb(f, cpu_fprintf, env, i, offset, size);
|
|
|
|
offset += size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-21 21:26:57 +08:00
|
|
|
static void mmu6xx_dump_BATs(FILE *f, fprintf_function cpu_fprintf,
|
|
|
|
CPUPPCState *env, int type)
|
|
|
|
{
|
|
|
|
target_ulong *BATlt, *BATut, *BATu, *BATl;
|
|
|
|
target_ulong BEPIl, BEPIu, bl;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case ACCESS_CODE:
|
|
|
|
BATlt = env->IBAT[1];
|
|
|
|
BATut = env->IBAT[0];
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
BATlt = env->DBAT[1];
|
|
|
|
BATut = env->DBAT[0];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < env->nb_BATs; i++) {
|
|
|
|
BATu = &BATut[i];
|
|
|
|
BATl = &BATlt[i];
|
|
|
|
BEPIu = *BATu & 0xF0000000;
|
|
|
|
BEPIl = *BATu & 0x0FFE0000;
|
|
|
|
bl = (*BATu & 0x00001FFC) << 15;
|
|
|
|
cpu_fprintf(f, "%s BAT%d BATu " TARGET_FMT_lx
|
|
|
|
" BATl " TARGET_FMT_lx "\n\t" TARGET_FMT_lx " "
|
|
|
|
TARGET_FMT_lx " " TARGET_FMT_lx "\n",
|
|
|
|
type == ACCESS_CODE ? "code" : "data", i,
|
|
|
|
*BATu, *BATl, BEPIu, BEPIl, bl);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void mmu6xx_dump_mmu(FILE *f, fprintf_function cpu_fprintf,
|
|
|
|
CPUPPCState *env)
|
|
|
|
{
|
target/ppc: Eliminate htab_base and htab_mask variables
CPUPPCState includes fields htab_base and htab_mask which store the base
address (GPA) and size (as a mask) of the guest's hashed page table (HPT).
These are set when the SDR1 register is updated.
Keeping these in sync with the SDR1 is actually a little bit fiddly, and
probably not useful for performance, since keeping them expands the size of
CPUPPCState. It also makes some upcoming changes harder to implement.
This patch removes these fields, in favour of calculating them directly
from the SDR1 contents when necessary.
This does make a change to the behaviour of attempting to write a bad value
(invalid HPT size) to the SDR1 with an mtspr instruction. Previously, the
bad value would be stored in SDR1 and could be retrieved with a later
mfspr, but the HPT size as used by the softmmu would be, clamped to the
allowed values. Now, writing a bad value is treated as a no-op. An error
message is printed in both new and old versions.
I'm not sure which behaviour, if either, matches real hardware. I don't
think it matters that much, since it's pretty clear that if an OS writes
a bad value to SDR1, it's not going to boot.
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
Reviewed-by: Alexey Kardashevskiy <aik@ozlabs.ru>
2017-02-24 13:36:44 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
2013-06-21 21:26:57 +08:00
|
|
|
ppc6xx_tlb_t *tlb;
|
|
|
|
target_ulong sr;
|
|
|
|
int type, way, entry, i;
|
|
|
|
|
target/ppc: Eliminate htab_base and htab_mask variables
CPUPPCState includes fields htab_base and htab_mask which store the base
address (GPA) and size (as a mask) of the guest's hashed page table (HPT).
These are set when the SDR1 register is updated.
Keeping these in sync with the SDR1 is actually a little bit fiddly, and
probably not useful for performance, since keeping them expands the size of
CPUPPCState. It also makes some upcoming changes harder to implement.
This patch removes these fields, in favour of calculating them directly
from the SDR1 contents when necessary.
This does make a change to the behaviour of attempting to write a bad value
(invalid HPT size) to the SDR1 with an mtspr instruction. Previously, the
bad value would be stored in SDR1 and could be retrieved with a later
mfspr, but the HPT size as used by the softmmu would be, clamped to the
allowed values. Now, writing a bad value is treated as a no-op. An error
message is printed in both new and old versions.
I'm not sure which behaviour, if either, matches real hardware. I don't
think it matters that much, since it's pretty clear that if an OS writes
a bad value to SDR1, it's not going to boot.
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
Reviewed-by: Alexey Kardashevskiy <aik@ozlabs.ru>
2017-02-24 13:36:44 +08:00
|
|
|
cpu_fprintf(f, "HTAB base = 0x%"HWADDR_PRIx"\n", ppc_hash32_hpt_base(cpu));
|
|
|
|
cpu_fprintf(f, "HTAB mask = 0x%"HWADDR_PRIx"\n", ppc_hash32_hpt_mask(cpu));
|
2013-06-21 21:26:57 +08:00
|
|
|
|
|
|
|
cpu_fprintf(f, "\nSegment registers:\n");
|
|
|
|
for (i = 0; i < 32; i++) {
|
|
|
|
sr = env->sr[i];
|
|
|
|
if (sr & 0x80000000) {
|
|
|
|
cpu_fprintf(f, "%02d T=%d Ks=%d Kp=%d BUID=0x%03x "
|
|
|
|
"CNTLR_SPEC=0x%05x\n", i,
|
|
|
|
sr & 0x80000000 ? 1 : 0, sr & 0x40000000 ? 1 : 0,
|
|
|
|
sr & 0x20000000 ? 1 : 0, (uint32_t)((sr >> 20) & 0x1FF),
|
|
|
|
(uint32_t)(sr & 0xFFFFF));
|
|
|
|
} else {
|
|
|
|
cpu_fprintf(f, "%02d T=%d Ks=%d Kp=%d N=%d VSID=0x%06x\n", i,
|
|
|
|
sr & 0x80000000 ? 1 : 0, sr & 0x40000000 ? 1 : 0,
|
|
|
|
sr & 0x20000000 ? 1 : 0, sr & 0x10000000 ? 1 : 0,
|
|
|
|
(uint32_t)(sr & 0x00FFFFFF));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cpu_fprintf(f, "\nBATs:\n");
|
|
|
|
mmu6xx_dump_BATs(f, cpu_fprintf, env, ACCESS_INT);
|
|
|
|
mmu6xx_dump_BATs(f, cpu_fprintf, env, ACCESS_CODE);
|
|
|
|
|
|
|
|
if (env->id_tlbs != 1) {
|
|
|
|
cpu_fprintf(f, "ERROR: 6xx MMU should have separated TLB"
|
|
|
|
" for code and data\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
cpu_fprintf(f, "\nTLBs [EPN EPN + SIZE]\n");
|
|
|
|
|
|
|
|
for (type = 0; type < 2; type++) {
|
|
|
|
for (way = 0; way < env->nb_ways; way++) {
|
|
|
|
for (entry = env->nb_tlb * type + env->tlb_per_way * way;
|
|
|
|
entry < (env->nb_tlb * type + env->tlb_per_way * (way + 1));
|
|
|
|
entry++) {
|
|
|
|
|
|
|
|
tlb = &env->tlb.tlb6[entry];
|
|
|
|
cpu_fprintf(f, "%s TLB %02d/%02d way:%d %s ["
|
|
|
|
TARGET_FMT_lx " " TARGET_FMT_lx "]\n",
|
|
|
|
type ? "code" : "data", entry % env->nb_tlb,
|
|
|
|
env->nb_tlb, way,
|
|
|
|
pte_is_valid(tlb->pte0) ? "valid" : "inval",
|
|
|
|
tlb->EPN, tlb->EPN + TARGET_PAGE_SIZE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:33 +08:00
|
|
|
void dump_mmu(FILE *f, fprintf_function cpu_fprintf, CPUPPCState *env)
|
|
|
|
{
|
2017-03-02 13:38:56 +08:00
|
|
|
switch (POWERPC_MMU_VER(env->mmu_model)) {
|
2012-05-30 12:23:33 +08:00
|
|
|
case POWERPC_MMU_BOOKE:
|
|
|
|
mmubooke_dump_mmu(f, cpu_fprintf, env);
|
|
|
|
break;
|
|
|
|
case POWERPC_MMU_BOOKE206:
|
|
|
|
mmubooke206_dump_mmu(f, cpu_fprintf, env);
|
|
|
|
break;
|
2013-06-21 21:26:57 +08:00
|
|
|
case POWERPC_MMU_SOFT_6xx:
|
|
|
|
case POWERPC_MMU_SOFT_74xx:
|
|
|
|
mmu6xx_dump_mmu(f, cpu_fprintf, env);
|
|
|
|
break;
|
2012-05-30 12:23:33 +08:00
|
|
|
#if defined(TARGET_PPC64)
|
2017-03-02 13:38:56 +08:00
|
|
|
case POWERPC_MMU_VER_64B:
|
|
|
|
case POWERPC_MMU_VER_2_03:
|
|
|
|
case POWERPC_MMU_VER_2_06:
|
|
|
|
case POWERPC_MMU_VER_2_07:
|
2016-01-14 12:33:27 +08:00
|
|
|
dump_slb(f, cpu_fprintf, ppc_env_get_cpu(env));
|
2012-05-30 12:23:33 +08:00
|
|
|
break;
|
2017-03-02 13:38:56 +08:00
|
|
|
case POWERPC_MMU_VER_3_00:
|
2017-03-01 14:54:38 +08:00
|
|
|
if (ppc64_radix_guest(ppc_env_get_cpu(env))) {
|
|
|
|
/* TODO - Unsupported */
|
|
|
|
} else {
|
|
|
|
dump_slb(f, cpu_fprintf, ppc_env_get_cpu(env));
|
|
|
|
break;
|
|
|
|
}
|
2012-05-30 12:23:33 +08:00
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
qemu_log_mask(LOG_UNIMP, "%s: unimplemented\n", __func__);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int check_physical(CPUPPCState *env, mmu_ctx_t *ctx,
|
|
|
|
target_ulong eaddr, int rw)
|
|
|
|
{
|
|
|
|
int in_plb, ret;
|
|
|
|
|
|
|
|
ctx->raddr = eaddr;
|
|
|
|
ctx->prot = PAGE_READ | PAGE_EXEC;
|
|
|
|
ret = 0;
|
|
|
|
switch (env->mmu_model) {
|
|
|
|
case POWERPC_MMU_SOFT_6xx:
|
|
|
|
case POWERPC_MMU_SOFT_74xx:
|
|
|
|
case POWERPC_MMU_SOFT_4xx:
|
|
|
|
case POWERPC_MMU_REAL:
|
|
|
|
case POWERPC_MMU_BOOKE:
|
|
|
|
ctx->prot |= PAGE_WRITE;
|
|
|
|
break;
|
2013-03-12 08:31:11 +08:00
|
|
|
|
2012-05-30 12:23:33 +08:00
|
|
|
case POWERPC_MMU_SOFT_4xx_Z:
|
|
|
|
if (unlikely(msr_pe != 0)) {
|
|
|
|
/* 403 family add some particular protections,
|
|
|
|
* using PBL/PBU registers for accesses with no translation.
|
|
|
|
*/
|
|
|
|
in_plb =
|
|
|
|
/* Check PLB validity */
|
|
|
|
(env->pb[0] < env->pb[1] &&
|
|
|
|
/* and address in plb area */
|
|
|
|
eaddr >= env->pb[0] && eaddr < env->pb[1]) ||
|
|
|
|
(env->pb[2] < env->pb[3] &&
|
|
|
|
eaddr >= env->pb[2] && eaddr < env->pb[3]) ? 1 : 0;
|
|
|
|
if (in_plb ^ msr_px) {
|
|
|
|
/* Access in protected area */
|
|
|
|
if (rw == 1) {
|
|
|
|
/* Access is not allowed */
|
|
|
|
ret = -2;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* Read-write access is allowed */
|
|
|
|
ctx->prot |= PAGE_WRITE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2013-03-12 08:31:11 +08:00
|
|
|
|
2012-05-30 12:23:33 +08:00
|
|
|
default:
|
2013-03-12 08:31:11 +08:00
|
|
|
/* Caller's checks mean we should never get here for other models */
|
|
|
|
abort();
|
2012-05-30 12:23:33 +08:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-10-28 19:04:50 +08:00
|
|
|
static int get_physical_address(CPUPPCState *env, mmu_ctx_t *ctx,
|
|
|
|
target_ulong eaddr, int rw, int access_type)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
2013-09-03 23:38:47 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
2013-03-12 08:31:10 +08:00
|
|
|
int ret = -1;
|
|
|
|
bool real_mode = (access_type == ACCESS_CODE && msr_ir == 0)
|
|
|
|
|| (access_type != ACCESS_CODE && msr_dr == 0);
|
2012-05-30 12:23:33 +08:00
|
|
|
|
|
|
|
#if 0
|
|
|
|
qemu_log("%s\n", __func__);
|
|
|
|
#endif
|
2013-03-12 08:31:10 +08:00
|
|
|
|
|
|
|
switch (env->mmu_model) {
|
|
|
|
case POWERPC_MMU_SOFT_6xx:
|
|
|
|
case POWERPC_MMU_SOFT_74xx:
|
|
|
|
if (real_mode) {
|
|
|
|
ret = check_physical(env, ctx, eaddr, rw);
|
|
|
|
} else {
|
2012-05-30 12:23:33 +08:00
|
|
|
/* Try to find a BAT */
|
|
|
|
if (env->nb_BATs != 0) {
|
2013-03-12 08:31:16 +08:00
|
|
|
ret = get_bat_6xx_tlb(env, ctx, eaddr, rw, access_type);
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
2013-03-12 08:31:09 +08:00
|
|
|
if (ret < 0) {
|
|
|
|
/* We didn't match any BAT entry or don't have BATs */
|
|
|
|
ret = get_segment_6xx_tlb(env, ctx, eaddr, rw, access_type);
|
|
|
|
}
|
2013-03-12 08:31:10 +08:00
|
|
|
}
|
|
|
|
break;
|
2013-03-12 08:31:09 +08:00
|
|
|
|
2013-03-12 08:31:10 +08:00
|
|
|
case POWERPC_MMU_SOFT_4xx:
|
|
|
|
case POWERPC_MMU_SOFT_4xx_Z:
|
|
|
|
if (real_mode) {
|
|
|
|
ret = check_physical(env, ctx, eaddr, rw);
|
|
|
|
} else {
|
2012-05-30 12:23:33 +08:00
|
|
|
ret = mmu40x_get_physical_address(env, ctx, eaddr,
|
|
|
|
rw, access_type);
|
2013-03-12 08:31:10 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case POWERPC_MMU_BOOKE:
|
|
|
|
ret = mmubooke_get_physical_address(env, ctx, eaddr,
|
|
|
|
rw, access_type);
|
|
|
|
break;
|
|
|
|
case POWERPC_MMU_BOOKE206:
|
|
|
|
ret = mmubooke206_get_physical_address(env, ctx, eaddr, rw,
|
2012-05-30 12:23:33 +08:00
|
|
|
access_type);
|
2013-03-12 08:31:10 +08:00
|
|
|
break;
|
|
|
|
case POWERPC_MMU_MPC8xx:
|
|
|
|
/* XXX: TODO */
|
2013-09-03 23:38:47 +08:00
|
|
|
cpu_abort(CPU(cpu), "MPC8xx MMU model is not implemented\n");
|
2013-03-12 08:31:10 +08:00
|
|
|
break;
|
|
|
|
case POWERPC_MMU_REAL:
|
|
|
|
if (real_mode) {
|
|
|
|
ret = check_physical(env, ctx, eaddr, rw);
|
|
|
|
} else {
|
2013-09-03 23:38:47 +08:00
|
|
|
cpu_abort(CPU(cpu), "PowerPC in real mode do not do any translation\n");
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
2013-03-12 08:31:10 +08:00
|
|
|
return -1;
|
|
|
|
default:
|
2013-09-03 23:38:47 +08:00
|
|
|
cpu_abort(CPU(cpu), "Unknown or invalid MMU model\n");
|
2013-03-12 08:31:10 +08:00
|
|
|
return -1;
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
#if 0
|
|
|
|
qemu_log("%s address " TARGET_FMT_lx " => %d " TARGET_FMT_plx "\n",
|
|
|
|
__func__, eaddr, ret, ctx->raddr);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-06-30 00:55:54 +08:00
|
|
|
hwaddr ppc_cpu_get_phys_page_debug(CPUState *cs, vaddr addr)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
2013-06-30 00:55:54 +08:00
|
|
|
PowerPCCPU *cpu = POWERPC_CPU(cs);
|
|
|
|
CPUPPCState *env = &cpu->env;
|
2012-05-30 12:23:33 +08:00
|
|
|
mmu_ctx_t ctx;
|
|
|
|
|
2017-03-02 13:38:56 +08:00
|
|
|
switch (POWERPC_MMU_VER(env->mmu_model)) {
|
2013-03-12 08:31:13 +08:00
|
|
|
#if defined(TARGET_PPC64)
|
2017-03-02 13:38:56 +08:00
|
|
|
case POWERPC_MMU_VER_64B:
|
|
|
|
case POWERPC_MMU_VER_2_03:
|
|
|
|
case POWERPC_MMU_VER_2_06:
|
|
|
|
case POWERPC_MMU_VER_2_07:
|
2016-01-14 12:33:27 +08:00
|
|
|
return ppc_hash64_get_phys_page_debug(cpu, addr);
|
2017-03-02 13:38:56 +08:00
|
|
|
case POWERPC_MMU_VER_3_00:
|
2017-03-01 14:54:38 +08:00
|
|
|
if (ppc64_radix_guest(ppc_env_get_cpu(env))) {
|
2017-07-03 14:19:47 +08:00
|
|
|
return ppc_radix64_get_phys_page_debug(cpu, addr);
|
2017-03-01 14:54:38 +08:00
|
|
|
} else {
|
|
|
|
return ppc_hash64_get_phys_page_debug(cpu, addr);
|
|
|
|
}
|
|
|
|
break;
|
2013-03-12 08:31:13 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
case POWERPC_MMU_32B:
|
|
|
|
case POWERPC_MMU_601:
|
2016-01-14 12:33:27 +08:00
|
|
|
return ppc_hash32_get_phys_page_debug(cpu, addr);
|
2013-03-12 08:31:13 +08:00
|
|
|
|
|
|
|
default:
|
|
|
|
;
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:33 +08:00
|
|
|
if (unlikely(get_physical_address(env, &ctx, addr, 0, ACCESS_INT) != 0)) {
|
2013-06-18 22:53:01 +08:00
|
|
|
|
|
|
|
/* Some MMUs have separate TLBs for code and data. If we only try an
|
|
|
|
* ACCESS_INT, we may not be able to read instructions mapped by code
|
|
|
|
* TLBs, so we also try a ACCESS_CODE.
|
|
|
|
*/
|
|
|
|
if (unlikely(get_physical_address(env, &ctx, addr, 0,
|
|
|
|
ACCESS_CODE) != 0)) {
|
|
|
|
return -1;
|
|
|
|
}
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return ctx.raddr & TARGET_PAGE_MASK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void booke206_update_mas_tlb_miss(CPUPPCState *env, target_ulong address,
|
|
|
|
int rw)
|
|
|
|
{
|
|
|
|
env->spr[SPR_BOOKE_MAS0] = env->spr[SPR_BOOKE_MAS4] & MAS4_TLBSELD_MASK;
|
|
|
|
env->spr[SPR_BOOKE_MAS1] = env->spr[SPR_BOOKE_MAS4] & MAS4_TSIZED_MASK;
|
|
|
|
env->spr[SPR_BOOKE_MAS2] = env->spr[SPR_BOOKE_MAS4] & MAS4_WIMGED_MASK;
|
|
|
|
env->spr[SPR_BOOKE_MAS3] = 0;
|
|
|
|
env->spr[SPR_BOOKE_MAS6] = 0;
|
|
|
|
env->spr[SPR_BOOKE_MAS7] = 0;
|
|
|
|
|
|
|
|
/* AS */
|
|
|
|
if (((rw == 2) && msr_ir) || ((rw != 2) && msr_dr)) {
|
|
|
|
env->spr[SPR_BOOKE_MAS1] |= MAS1_TS;
|
|
|
|
env->spr[SPR_BOOKE_MAS6] |= MAS6_SAS;
|
|
|
|
}
|
|
|
|
|
|
|
|
env->spr[SPR_BOOKE_MAS1] |= MAS1_VALID;
|
|
|
|
env->spr[SPR_BOOKE_MAS2] |= address & MAS2_EPN_MASK;
|
|
|
|
|
|
|
|
switch (env->spr[SPR_BOOKE_MAS4] & MAS4_TIDSELD_PIDZ) {
|
|
|
|
case MAS4_TIDSELD_PID0:
|
|
|
|
env->spr[SPR_BOOKE_MAS1] |= env->spr[SPR_BOOKE_PID] << MAS1_TID_SHIFT;
|
|
|
|
break;
|
|
|
|
case MAS4_TIDSELD_PID1:
|
|
|
|
env->spr[SPR_BOOKE_MAS1] |= env->spr[SPR_BOOKE_PID1] << MAS1_TID_SHIFT;
|
|
|
|
break;
|
|
|
|
case MAS4_TIDSELD_PID2:
|
|
|
|
env->spr[SPR_BOOKE_MAS1] |= env->spr[SPR_BOOKE_PID2] << MAS1_TID_SHIFT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
env->spr[SPR_BOOKE_MAS6] |= env->spr[SPR_BOOKE_PID] << 16;
|
|
|
|
|
|
|
|
/* next victim logic */
|
|
|
|
env->spr[SPR_BOOKE_MAS0] |= env->last_way << MAS0_ESEL_SHIFT;
|
|
|
|
env->last_way++;
|
|
|
|
env->last_way &= booke206_tlb_ways(env, 0) - 1;
|
|
|
|
env->spr[SPR_BOOKE_MAS0] |= env->last_way << MAS0_NV_SHIFT;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Perform address translation */
|
2013-03-12 08:31:49 +08:00
|
|
|
static int cpu_ppc_handle_mmu_fault(CPUPPCState *env, target_ulong address,
|
|
|
|
int rw, int mmu_idx)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
2013-08-26 14:31:06 +08:00
|
|
|
CPUState *cs = CPU(ppc_env_get_cpu(env));
|
2016-01-14 12:33:27 +08:00
|
|
|
PowerPCCPU *cpu = POWERPC_CPU(cs);
|
2012-05-30 12:23:33 +08:00
|
|
|
mmu_ctx_t ctx;
|
|
|
|
int access_type;
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
if (rw == 2) {
|
|
|
|
/* code access */
|
|
|
|
rw = 0;
|
|
|
|
access_type = ACCESS_CODE;
|
|
|
|
} else {
|
|
|
|
/* data access */
|
|
|
|
access_type = env->access_type;
|
|
|
|
}
|
|
|
|
ret = get_physical_address(env, &ctx, address, rw, access_type);
|
|
|
|
if (ret == 0) {
|
2013-09-03 19:59:37 +08:00
|
|
|
tlb_set_page(cs, address & TARGET_PAGE_MASK,
|
2012-05-30 12:23:33 +08:00
|
|
|
ctx.raddr & TARGET_PAGE_MASK, ctx.prot,
|
|
|
|
mmu_idx, TARGET_PAGE_SIZE);
|
|
|
|
ret = 0;
|
|
|
|
} else if (ret < 0) {
|
2013-08-26 14:31:06 +08:00
|
|
|
LOG_MMU_STATE(cs);
|
2012-05-30 12:23:33 +08:00
|
|
|
if (access_type == ACCESS_CODE) {
|
|
|
|
switch (ret) {
|
|
|
|
case -1:
|
|
|
|
/* No matches in page tables or TLB */
|
|
|
|
switch (env->mmu_model) {
|
|
|
|
case POWERPC_MMU_SOFT_6xx:
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_IFTLB;
|
2012-05-30 12:23:33 +08:00
|
|
|
env->error_code = 1 << 18;
|
|
|
|
env->spr[SPR_IMISS] = address;
|
|
|
|
env->spr[SPR_ICMP] = 0x80000000 | ctx.ptem;
|
|
|
|
goto tlb_miss;
|
|
|
|
case POWERPC_MMU_SOFT_74xx:
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_IFTLB;
|
2012-05-30 12:23:33 +08:00
|
|
|
goto tlb_miss_74xx;
|
|
|
|
case POWERPC_MMU_SOFT_4xx:
|
|
|
|
case POWERPC_MMU_SOFT_4xx_Z:
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_ITLB;
|
2012-05-30 12:23:33 +08:00
|
|
|
env->error_code = 0;
|
|
|
|
env->spr[SPR_40x_DEAR] = address;
|
|
|
|
env->spr[SPR_40x_ESR] = 0x00000000;
|
|
|
|
break;
|
|
|
|
case POWERPC_MMU_BOOKE206:
|
2017-08-01 16:44:57 +08:00
|
|
|
booke206_update_mas_tlb_miss(env, address, 2);
|
2012-05-30 12:23:33 +08:00
|
|
|
/* fall through */
|
|
|
|
case POWERPC_MMU_BOOKE:
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_ITLB;
|
2012-05-30 12:23:33 +08:00
|
|
|
env->error_code = 0;
|
|
|
|
env->spr[SPR_BOOKE_DEAR] = address;
|
|
|
|
return -1;
|
|
|
|
case POWERPC_MMU_MPC8xx:
|
|
|
|
/* XXX: TODO */
|
2013-09-03 23:38:47 +08:00
|
|
|
cpu_abort(cs, "MPC8xx MMU model is not implemented\n");
|
2012-05-30 12:23:33 +08:00
|
|
|
break;
|
|
|
|
case POWERPC_MMU_REAL:
|
2013-09-03 23:38:47 +08:00
|
|
|
cpu_abort(cs, "PowerPC in real mode should never raise "
|
2012-05-30 12:23:33 +08:00
|
|
|
"any MMU exceptions\n");
|
|
|
|
return -1;
|
|
|
|
default:
|
2013-09-03 23:38:47 +08:00
|
|
|
cpu_abort(cs, "Unknown or invalid MMU model\n");
|
2012-05-30 12:23:33 +08:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case -2:
|
|
|
|
/* Access rights violation */
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_ISI;
|
2012-05-30 12:23:33 +08:00
|
|
|
env->error_code = 0x08000000;
|
|
|
|
break;
|
|
|
|
case -3:
|
|
|
|
/* No execute protection violation */
|
|
|
|
if ((env->mmu_model == POWERPC_MMU_BOOKE) ||
|
|
|
|
(env->mmu_model == POWERPC_MMU_BOOKE206)) {
|
|
|
|
env->spr[SPR_BOOKE_ESR] = 0x00000000;
|
|
|
|
}
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_ISI;
|
2012-05-30 12:23:33 +08:00
|
|
|
env->error_code = 0x10000000;
|
|
|
|
break;
|
|
|
|
case -4:
|
|
|
|
/* Direct store exception */
|
|
|
|
/* No code fetch is allowed in direct-store areas */
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_ISI;
|
2012-05-30 12:23:33 +08:00
|
|
|
env->error_code = 0x10000000;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
switch (ret) {
|
|
|
|
case -1:
|
|
|
|
/* No matches in page tables or TLB */
|
|
|
|
switch (env->mmu_model) {
|
|
|
|
case POWERPC_MMU_SOFT_6xx:
|
|
|
|
if (rw == 1) {
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_DSTLB;
|
2012-05-30 12:23:33 +08:00
|
|
|
env->error_code = 1 << 16;
|
|
|
|
} else {
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_DLTLB;
|
2012-05-30 12:23:33 +08:00
|
|
|
env->error_code = 0;
|
|
|
|
}
|
|
|
|
env->spr[SPR_DMISS] = address;
|
|
|
|
env->spr[SPR_DCMP] = 0x80000000 | ctx.ptem;
|
|
|
|
tlb_miss:
|
|
|
|
env->error_code |= ctx.key << 19;
|
target/ppc: Eliminate htab_base and htab_mask variables
CPUPPCState includes fields htab_base and htab_mask which store the base
address (GPA) and size (as a mask) of the guest's hashed page table (HPT).
These are set when the SDR1 register is updated.
Keeping these in sync with the SDR1 is actually a little bit fiddly, and
probably not useful for performance, since keeping them expands the size of
CPUPPCState. It also makes some upcoming changes harder to implement.
This patch removes these fields, in favour of calculating them directly
from the SDR1 contents when necessary.
This does make a change to the behaviour of attempting to write a bad value
(invalid HPT size) to the SDR1 with an mtspr instruction. Previously, the
bad value would be stored in SDR1 and could be retrieved with a later
mfspr, but the HPT size as used by the softmmu would be, clamped to the
allowed values. Now, writing a bad value is treated as a no-op. An error
message is printed in both new and old versions.
I'm not sure which behaviour, if either, matches real hardware. I don't
think it matters that much, since it's pretty clear that if an OS writes
a bad value to SDR1, it's not going to boot.
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
Reviewed-by: Alexey Kardashevskiy <aik@ozlabs.ru>
2017-02-24 13:36:44 +08:00
|
|
|
env->spr[SPR_HASH1] = ppc_hash32_hpt_base(cpu) +
|
2016-01-14 12:33:27 +08:00
|
|
|
get_pteg_offset32(cpu, ctx.hash[0]);
|
target/ppc: Eliminate htab_base and htab_mask variables
CPUPPCState includes fields htab_base and htab_mask which store the base
address (GPA) and size (as a mask) of the guest's hashed page table (HPT).
These are set when the SDR1 register is updated.
Keeping these in sync with the SDR1 is actually a little bit fiddly, and
probably not useful for performance, since keeping them expands the size of
CPUPPCState. It also makes some upcoming changes harder to implement.
This patch removes these fields, in favour of calculating them directly
from the SDR1 contents when necessary.
This does make a change to the behaviour of attempting to write a bad value
(invalid HPT size) to the SDR1 with an mtspr instruction. Previously, the
bad value would be stored in SDR1 and could be retrieved with a later
mfspr, but the HPT size as used by the softmmu would be, clamped to the
allowed values. Now, writing a bad value is treated as a no-op. An error
message is printed in both new and old versions.
I'm not sure which behaviour, if either, matches real hardware. I don't
think it matters that much, since it's pretty clear that if an OS writes
a bad value to SDR1, it's not going to boot.
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
Reviewed-by: Alexey Kardashevskiy <aik@ozlabs.ru>
2017-02-24 13:36:44 +08:00
|
|
|
env->spr[SPR_HASH2] = ppc_hash32_hpt_base(cpu) +
|
2016-01-14 12:33:27 +08:00
|
|
|
get_pteg_offset32(cpu, ctx.hash[1]);
|
2012-05-30 12:23:33 +08:00
|
|
|
break;
|
|
|
|
case POWERPC_MMU_SOFT_74xx:
|
|
|
|
if (rw == 1) {
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_DSTLB;
|
2012-05-30 12:23:33 +08:00
|
|
|
} else {
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_DLTLB;
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
tlb_miss_74xx:
|
|
|
|
/* Implement LRU algorithm */
|
|
|
|
env->error_code = ctx.key << 19;
|
|
|
|
env->spr[SPR_TLBMISS] = (address & ~((target_ulong)0x3)) |
|
|
|
|
((env->last_way + 1) & (env->nb_ways - 1));
|
|
|
|
env->spr[SPR_PTEHI] = 0x80000000 | ctx.ptem;
|
|
|
|
break;
|
|
|
|
case POWERPC_MMU_SOFT_4xx:
|
|
|
|
case POWERPC_MMU_SOFT_4xx_Z:
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_DTLB;
|
2012-05-30 12:23:33 +08:00
|
|
|
env->error_code = 0;
|
|
|
|
env->spr[SPR_40x_DEAR] = address;
|
|
|
|
if (rw) {
|
|
|
|
env->spr[SPR_40x_ESR] = 0x00800000;
|
|
|
|
} else {
|
|
|
|
env->spr[SPR_40x_ESR] = 0x00000000;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case POWERPC_MMU_MPC8xx:
|
|
|
|
/* XXX: TODO */
|
2013-09-03 23:38:47 +08:00
|
|
|
cpu_abort(cs, "MPC8xx MMU model is not implemented\n");
|
2012-05-30 12:23:33 +08:00
|
|
|
break;
|
|
|
|
case POWERPC_MMU_BOOKE206:
|
|
|
|
booke206_update_mas_tlb_miss(env, address, rw);
|
|
|
|
/* fall through */
|
|
|
|
case POWERPC_MMU_BOOKE:
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_DTLB;
|
2012-05-30 12:23:33 +08:00
|
|
|
env->error_code = 0;
|
|
|
|
env->spr[SPR_BOOKE_DEAR] = address;
|
|
|
|
env->spr[SPR_BOOKE_ESR] = rw ? ESR_ST : 0;
|
|
|
|
return -1;
|
|
|
|
case POWERPC_MMU_REAL:
|
2013-09-03 23:38:47 +08:00
|
|
|
cpu_abort(cs, "PowerPC in real mode should never raise "
|
2012-05-30 12:23:33 +08:00
|
|
|
"any MMU exceptions\n");
|
|
|
|
return -1;
|
|
|
|
default:
|
2013-09-03 23:38:47 +08:00
|
|
|
cpu_abort(cs, "Unknown or invalid MMU model\n");
|
2012-05-30 12:23:33 +08:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case -2:
|
|
|
|
/* Access rights violation */
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_DSI;
|
2012-05-30 12:23:33 +08:00
|
|
|
env->error_code = 0;
|
|
|
|
if (env->mmu_model == POWERPC_MMU_SOFT_4xx
|
|
|
|
|| env->mmu_model == POWERPC_MMU_SOFT_4xx_Z) {
|
|
|
|
env->spr[SPR_40x_DEAR] = address;
|
|
|
|
if (rw) {
|
|
|
|
env->spr[SPR_40x_ESR] |= 0x00800000;
|
|
|
|
}
|
|
|
|
} else if ((env->mmu_model == POWERPC_MMU_BOOKE) ||
|
|
|
|
(env->mmu_model == POWERPC_MMU_BOOKE206)) {
|
|
|
|
env->spr[SPR_BOOKE_DEAR] = address;
|
|
|
|
env->spr[SPR_BOOKE_ESR] = rw ? ESR_ST : 0;
|
|
|
|
} else {
|
|
|
|
env->spr[SPR_DAR] = address;
|
|
|
|
if (rw == 1) {
|
|
|
|
env->spr[SPR_DSISR] = 0x0A000000;
|
|
|
|
} else {
|
|
|
|
env->spr[SPR_DSISR] = 0x08000000;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case -4:
|
|
|
|
/* Direct store exception */
|
|
|
|
switch (access_type) {
|
|
|
|
case ACCESS_FLOAT:
|
|
|
|
/* Floating point load/store */
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_ALIGN;
|
2012-05-30 12:23:33 +08:00
|
|
|
env->error_code = POWERPC_EXCP_ALIGN_FP;
|
|
|
|
env->spr[SPR_DAR] = address;
|
|
|
|
break;
|
|
|
|
case ACCESS_RES:
|
|
|
|
/* lwarx, ldarx or stwcx. */
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_DSI;
|
2012-05-30 12:23:33 +08:00
|
|
|
env->error_code = 0;
|
|
|
|
env->spr[SPR_DAR] = address;
|
|
|
|
if (rw == 1) {
|
|
|
|
env->spr[SPR_DSISR] = 0x06000000;
|
|
|
|
} else {
|
|
|
|
env->spr[SPR_DSISR] = 0x04000000;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case ACCESS_EXT:
|
|
|
|
/* eciwx or ecowx */
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_DSI;
|
2012-05-30 12:23:33 +08:00
|
|
|
env->error_code = 0;
|
|
|
|
env->spr[SPR_DAR] = address;
|
|
|
|
if (rw == 1) {
|
|
|
|
env->spr[SPR_DSISR] = 0x06100000;
|
|
|
|
} else {
|
|
|
|
env->spr[SPR_DSISR] = 0x04100000;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
printf("DSI: invalid exception (%d)\n", ret);
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception_index = POWERPC_EXCP_PROGRAM;
|
2012-05-30 12:23:33 +08:00
|
|
|
env->error_code =
|
|
|
|
POWERPC_EXCP_INVAL | POWERPC_EXCP_INVAL_INVAL;
|
|
|
|
env->spr[SPR_DAR] = address;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#if 0
|
|
|
|
printf("%s: set exception to %d %02x\n", __func__,
|
2013-08-26 14:31:06 +08:00
|
|
|
cs->exception, env->error_code);
|
2012-05-30 12:23:33 +08:00
|
|
|
#endif
|
|
|
|
ret = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
/* BATs management */
|
|
|
|
#if !defined(FLUSH_ALL_TLBS)
|
|
|
|
static inline void do_invalidate_BAT(CPUPPCState *env, target_ulong BATu,
|
|
|
|
target_ulong mask)
|
|
|
|
{
|
2013-09-04 07:29:02 +08:00
|
|
|
CPUState *cs = CPU(ppc_env_get_cpu(env));
|
2012-05-30 12:23:33 +08:00
|
|
|
target_ulong base, end, page;
|
|
|
|
|
|
|
|
base = BATu & ~0x0001FFFF;
|
|
|
|
end = base + mask + 0x00020000;
|
|
|
|
LOG_BATS("Flush BAT from " TARGET_FMT_lx " to " TARGET_FMT_lx " ("
|
|
|
|
TARGET_FMT_lx ")\n", base, end, mask);
|
|
|
|
for (page = base; page != end; page += TARGET_PAGE_SIZE) {
|
2013-09-04 07:29:02 +08:00
|
|
|
tlb_flush_page(cs, page);
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
LOG_BATS("Flush done\n");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static inline void dump_store_bat(CPUPPCState *env, char ID, int ul, int nr,
|
|
|
|
target_ulong value)
|
|
|
|
{
|
|
|
|
LOG_BATS("Set %cBAT%d%c to " TARGET_FMT_lx " (" TARGET_FMT_lx ")\n", ID,
|
|
|
|
nr, ul == 0 ? 'u' : 'l', value, env->nip);
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:34 +08:00
|
|
|
void helper_store_ibatu(CPUPPCState *env, uint32_t nr, target_ulong value)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
|
|
|
target_ulong mask;
|
2016-06-03 21:58:09 +08:00
|
|
|
#if defined(FLUSH_ALL_TLBS)
|
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
|
|
|
#endif
|
2012-05-30 12:23:33 +08:00
|
|
|
|
|
|
|
dump_store_bat(env, 'I', 0, nr, value);
|
|
|
|
if (env->IBAT[0][nr] != value) {
|
|
|
|
mask = (value << 15) & 0x0FFE0000UL;
|
|
|
|
#if !defined(FLUSH_ALL_TLBS)
|
|
|
|
do_invalidate_BAT(env, env->IBAT[0][nr], mask);
|
|
|
|
#endif
|
|
|
|
/* When storing valid upper BAT, mask BEPI and BRPN
|
|
|
|
* and invalidate all TLBs covered by this BAT
|
|
|
|
*/
|
|
|
|
mask = (value << 15) & 0x0FFE0000UL;
|
|
|
|
env->IBAT[0][nr] = (value & 0x00001FFFUL) |
|
|
|
|
(value & ~0x0001FFFFUL & ~mask);
|
|
|
|
env->IBAT[1][nr] = (env->IBAT[1][nr] & 0x0000007B) |
|
|
|
|
(env->IBAT[1][nr] & ~0x0001FFFF & ~mask);
|
|
|
|
#if !defined(FLUSH_ALL_TLBS)
|
|
|
|
do_invalidate_BAT(env, env->IBAT[0][nr], mask);
|
|
|
|
#else
|
2016-11-14 22:17:28 +08:00
|
|
|
tlb_flush(CPU(cpu));
|
2012-05-30 12:23:33 +08:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:34 +08:00
|
|
|
void helper_store_ibatl(CPUPPCState *env, uint32_t nr, target_ulong value)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
|
|
|
dump_store_bat(env, 'I', 1, nr, value);
|
|
|
|
env->IBAT[1][nr] = value;
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:34 +08:00
|
|
|
void helper_store_dbatu(CPUPPCState *env, uint32_t nr, target_ulong value)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
|
|
|
target_ulong mask;
|
2016-06-03 21:58:09 +08:00
|
|
|
#if defined(FLUSH_ALL_TLBS)
|
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
|
|
|
#endif
|
2012-05-30 12:23:33 +08:00
|
|
|
|
|
|
|
dump_store_bat(env, 'D', 0, nr, value);
|
|
|
|
if (env->DBAT[0][nr] != value) {
|
|
|
|
/* When storing valid upper BAT, mask BEPI and BRPN
|
|
|
|
* and invalidate all TLBs covered by this BAT
|
|
|
|
*/
|
|
|
|
mask = (value << 15) & 0x0FFE0000UL;
|
|
|
|
#if !defined(FLUSH_ALL_TLBS)
|
|
|
|
do_invalidate_BAT(env, env->DBAT[0][nr], mask);
|
|
|
|
#endif
|
|
|
|
mask = (value << 15) & 0x0FFE0000UL;
|
|
|
|
env->DBAT[0][nr] = (value & 0x00001FFFUL) |
|
|
|
|
(value & ~0x0001FFFFUL & ~mask);
|
|
|
|
env->DBAT[1][nr] = (env->DBAT[1][nr] & 0x0000007B) |
|
|
|
|
(env->DBAT[1][nr] & ~0x0001FFFF & ~mask);
|
|
|
|
#if !defined(FLUSH_ALL_TLBS)
|
|
|
|
do_invalidate_BAT(env, env->DBAT[0][nr], mask);
|
|
|
|
#else
|
2016-11-14 22:17:28 +08:00
|
|
|
tlb_flush(CPU(cpu));
|
2012-05-30 12:23:33 +08:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:34 +08:00
|
|
|
void helper_store_dbatl(CPUPPCState *env, uint32_t nr, target_ulong value)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
|
|
|
dump_store_bat(env, 'D', 1, nr, value);
|
|
|
|
env->DBAT[1][nr] = value;
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:34 +08:00
|
|
|
void helper_store_601_batu(CPUPPCState *env, uint32_t nr, target_ulong value)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
|
|
|
target_ulong mask;
|
|
|
|
#if defined(FLUSH_ALL_TLBS)
|
2016-06-03 21:58:09 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
2012-05-30 12:23:33 +08:00
|
|
|
int do_inval;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
dump_store_bat(env, 'I', 0, nr, value);
|
|
|
|
if (env->IBAT[0][nr] != value) {
|
|
|
|
#if defined(FLUSH_ALL_TLBS)
|
|
|
|
do_inval = 0;
|
|
|
|
#endif
|
|
|
|
mask = (env->IBAT[1][nr] << 17) & 0x0FFE0000UL;
|
|
|
|
if (env->IBAT[1][nr] & 0x40) {
|
|
|
|
/* Invalidate BAT only if it is valid */
|
|
|
|
#if !defined(FLUSH_ALL_TLBS)
|
|
|
|
do_invalidate_BAT(env, env->IBAT[0][nr], mask);
|
|
|
|
#else
|
|
|
|
do_inval = 1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
/* When storing valid upper BAT, mask BEPI and BRPN
|
|
|
|
* and invalidate all TLBs covered by this BAT
|
|
|
|
*/
|
|
|
|
env->IBAT[0][nr] = (value & 0x00001FFFUL) |
|
|
|
|
(value & ~0x0001FFFFUL & ~mask);
|
|
|
|
env->DBAT[0][nr] = env->IBAT[0][nr];
|
|
|
|
if (env->IBAT[1][nr] & 0x40) {
|
|
|
|
#if !defined(FLUSH_ALL_TLBS)
|
|
|
|
do_invalidate_BAT(env, env->IBAT[0][nr], mask);
|
|
|
|
#else
|
|
|
|
do_inval = 1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#if defined(FLUSH_ALL_TLBS)
|
|
|
|
if (do_inval) {
|
2016-11-14 22:17:28 +08:00
|
|
|
tlb_flush(CPU(cpu));
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:34 +08:00
|
|
|
void helper_store_601_batl(CPUPPCState *env, uint32_t nr, target_ulong value)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
2013-01-27 11:32:01 +08:00
|
|
|
#if !defined(FLUSH_ALL_TLBS)
|
2012-05-30 12:23:33 +08:00
|
|
|
target_ulong mask;
|
2013-01-27 11:32:01 +08:00
|
|
|
#else
|
2016-06-03 21:58:09 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
2012-05-30 12:23:33 +08:00
|
|
|
int do_inval;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
dump_store_bat(env, 'I', 1, nr, value);
|
|
|
|
if (env->IBAT[1][nr] != value) {
|
|
|
|
#if defined(FLUSH_ALL_TLBS)
|
|
|
|
do_inval = 0;
|
|
|
|
#endif
|
|
|
|
if (env->IBAT[1][nr] & 0x40) {
|
|
|
|
#if !defined(FLUSH_ALL_TLBS)
|
|
|
|
mask = (env->IBAT[1][nr] << 17) & 0x0FFE0000UL;
|
|
|
|
do_invalidate_BAT(env, env->IBAT[0][nr], mask);
|
|
|
|
#else
|
|
|
|
do_inval = 1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
if (value & 0x40) {
|
|
|
|
#if !defined(FLUSH_ALL_TLBS)
|
|
|
|
mask = (value << 17) & 0x0FFE0000UL;
|
|
|
|
do_invalidate_BAT(env, env->IBAT[0][nr], mask);
|
|
|
|
#else
|
|
|
|
do_inval = 1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
env->IBAT[1][nr] = value;
|
|
|
|
env->DBAT[1][nr] = value;
|
|
|
|
#if defined(FLUSH_ALL_TLBS)
|
|
|
|
if (do_inval) {
|
2016-11-14 22:17:28 +08:00
|
|
|
tlb_flush(CPU(cpu));
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
/* TLB management */
|
|
|
|
void ppc_tlb_invalidate_all(CPUPPCState *env)
|
|
|
|
{
|
2013-09-03 23:38:47 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
|
|
|
|
2017-03-02 13:38:56 +08:00
|
|
|
#if defined(TARGET_PPC64)
|
|
|
|
if (env->mmu_model & POWERPC_MMU_64) {
|
|
|
|
env->tlb_need_flush = 0;
|
|
|
|
tlb_flush(CPU(cpu));
|
|
|
|
} else
|
|
|
|
#endif /* defined(TARGET_PPC64) */
|
2012-05-30 12:23:33 +08:00
|
|
|
switch (env->mmu_model) {
|
|
|
|
case POWERPC_MMU_SOFT_6xx:
|
|
|
|
case POWERPC_MMU_SOFT_74xx:
|
|
|
|
ppc6xx_tlb_invalidate_all(env);
|
|
|
|
break;
|
|
|
|
case POWERPC_MMU_SOFT_4xx:
|
|
|
|
case POWERPC_MMU_SOFT_4xx_Z:
|
|
|
|
ppc4xx_tlb_invalidate_all(env);
|
|
|
|
break;
|
|
|
|
case POWERPC_MMU_REAL:
|
2013-09-03 23:38:47 +08:00
|
|
|
cpu_abort(CPU(cpu), "No TLB for PowerPC 4xx in real mode\n");
|
2012-05-30 12:23:33 +08:00
|
|
|
break;
|
|
|
|
case POWERPC_MMU_MPC8xx:
|
|
|
|
/* XXX: TODO */
|
2013-09-03 23:38:47 +08:00
|
|
|
cpu_abort(CPU(cpu), "MPC8xx MMU model is not implemented\n");
|
2012-05-30 12:23:33 +08:00
|
|
|
break;
|
|
|
|
case POWERPC_MMU_BOOKE:
|
2016-11-14 22:17:28 +08:00
|
|
|
tlb_flush(CPU(cpu));
|
2012-05-30 12:23:33 +08:00
|
|
|
break;
|
|
|
|
case POWERPC_MMU_BOOKE206:
|
|
|
|
booke206_flush_tlb(env, -1, 0);
|
|
|
|
break;
|
|
|
|
case POWERPC_MMU_32B:
|
|
|
|
case POWERPC_MMU_601:
|
2016-06-07 10:50:22 +08:00
|
|
|
env->tlb_need_flush = 0;
|
2016-11-14 22:17:28 +08:00
|
|
|
tlb_flush(CPU(cpu));
|
2012-05-30 12:23:33 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
/* XXX: TODO */
|
2017-03-02 13:38:56 +08:00
|
|
|
cpu_abort(CPU(cpu), "Unknown MMU model %x\n", env->mmu_model);
|
2012-05-30 12:23:33 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ppc_tlb_invalidate_one(CPUPPCState *env, target_ulong addr)
|
|
|
|
{
|
|
|
|
#if !defined(FLUSH_ALL_TLBS)
|
|
|
|
addr &= TARGET_PAGE_MASK;
|
2017-03-02 13:38:56 +08:00
|
|
|
#if defined(TARGET_PPC64)
|
|
|
|
if (env->mmu_model & POWERPC_MMU_64) {
|
|
|
|
/* tlbie invalidate TLBs for all segments */
|
|
|
|
/* XXX: given the fact that there are too many segments to invalidate,
|
|
|
|
* and we still don't have a tlb_flush_mask(env, n, mask) in QEMU,
|
|
|
|
* we just invalidate all TLBs
|
|
|
|
*/
|
|
|
|
env->tlb_need_flush |= TLB_NEED_LOCAL_FLUSH;
|
|
|
|
} else
|
|
|
|
#endif /* defined(TARGET_PPC64) */
|
2012-05-30 12:23:33 +08:00
|
|
|
switch (env->mmu_model) {
|
|
|
|
case POWERPC_MMU_SOFT_6xx:
|
|
|
|
case POWERPC_MMU_SOFT_74xx:
|
|
|
|
ppc6xx_tlb_invalidate_virt(env, addr, 0);
|
|
|
|
if (env->id_tlbs == 1) {
|
|
|
|
ppc6xx_tlb_invalidate_virt(env, addr, 1);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case POWERPC_MMU_32B:
|
|
|
|
case POWERPC_MMU_601:
|
2016-06-07 10:50:22 +08:00
|
|
|
/* Actual CPUs invalidate entire congruence classes based on the
|
|
|
|
* geometry of their TLBs and some OSes take that into account,
|
|
|
|
* we just mark the TLB to be flushed later (context synchronizing
|
|
|
|
* event or sync instruction on 32-bit).
|
2016-06-07 10:50:21 +08:00
|
|
|
*/
|
2016-09-21 00:34:59 +08:00
|
|
|
env->tlb_need_flush |= TLB_NEED_LOCAL_FLUSH;
|
2012-05-30 12:23:33 +08:00
|
|
|
break;
|
|
|
|
default:
|
target-ppc: Remove unused mmu models from ppc_tlb_invalidate_one
ppc_tlb_invalidate_one() has a big switch handling many different MMU
types. However, most of those branches can never be reached:
It is called from 3 places: from remove_hpte() and h_protect() in
spapr_hcall.c (which always has a 64-bit hash MMU type), and from
helper_tlbie() in mmu_helper.c.
Calls to helper_tlbie() are generated from gen_tlbiel, gen_tlbiel and
gen_tlbiva. The first two are only used with the PPC_MEM_TLBIE flag,
set only with 32-bit or 64-bit hash MMU models, and gen_tlbiva() is
used only on 440 and 460 models with the BookE mmu model.
These means the exhaustive list of MMU types which may call
ppc_tlb_invalidate_one() is: POWERPC_MMU_SOFT_6xx, POWERPC_MMU_601,
POWERPC_MMU_32B, POWERPC_MMU_SOFT_74xx, POWERPC_MMU_64B, POWERPC_MMU_2_03,
POWERPC_MMU_2_06, POWERPC_MMU_2_07 and POWERPC_MMU_BOOKE.
Clean up by removing logic for all other MMU types from
ppc_tlb_invalidate_one().
This means that ppc4xx_tlb_invalidate_virt() now has no callers, or rather,
makes it obvious that it has no callers. So, we remove that function as
well.
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
2016-01-30 20:49:22 +08:00
|
|
|
/* Should never reach here with other MMU models */
|
|
|
|
assert(0);
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
#else
|
|
|
|
ppc_tlb_invalidate_all(env);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
/* Special registers manipulation */
|
|
|
|
void ppc_store_sdr1(CPUPPCState *env, target_ulong value)
|
|
|
|
{
|
2017-02-23 08:39:18 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
2014-12-14 00:48:18 +08:00
|
|
|
qemu_log_mask(CPU_LOG_MMU, "%s: " TARGET_FMT_lx "\n", __func__, value);
|
2017-02-23 08:39:18 +08:00
|
|
|
assert(!cpu->vhyp);
|
2012-05-30 12:23:33 +08:00
|
|
|
#if defined(TARGET_PPC64)
|
2015-02-10 06:40:47 +08:00
|
|
|
if (env->mmu_model & POWERPC_MMU_64) {
|
2017-02-24 14:35:50 +08:00
|
|
|
target_ulong sdr_mask = SDR_64_HTABORG | SDR_64_HTABSIZE;
|
2017-02-24 09:05:12 +08:00
|
|
|
target_ulong htabsize = value & SDR_64_HTABSIZE;
|
2012-05-30 12:23:33 +08:00
|
|
|
|
2017-02-24 14:35:50 +08:00
|
|
|
if (value & ~sdr_mask) {
|
|
|
|
error_report("Invalid bits 0x"TARGET_FMT_lx" set in SDR1",
|
|
|
|
value & ~sdr_mask);
|
|
|
|
value &= sdr_mask;
|
|
|
|
}
|
2017-02-24 09:05:12 +08:00
|
|
|
if (htabsize > 28) {
|
|
|
|
error_report("Invalid HTABSIZE 0x" TARGET_FMT_lx" stored in SDR1",
|
|
|
|
htabsize);
|
target/ppc: Eliminate htab_base and htab_mask variables
CPUPPCState includes fields htab_base and htab_mask which store the base
address (GPA) and size (as a mask) of the guest's hashed page table (HPT).
These are set when the SDR1 register is updated.
Keeping these in sync with the SDR1 is actually a little bit fiddly, and
probably not useful for performance, since keeping them expands the size of
CPUPPCState. It also makes some upcoming changes harder to implement.
This patch removes these fields, in favour of calculating them directly
from the SDR1 contents when necessary.
This does make a change to the behaviour of attempting to write a bad value
(invalid HPT size) to the SDR1 with an mtspr instruction. Previously, the
bad value would be stored in SDR1 and could be retrieved with a later
mfspr, but the HPT size as used by the softmmu would be, clamped to the
allowed values. Now, writing a bad value is treated as a no-op. An error
message is printed in both new and old versions.
I'm not sure which behaviour, if either, matches real hardware. I don't
think it matters that much, since it's pretty clear that if an OS writes
a bad value to SDR1, it's not going to boot.
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
Reviewed-by: Alexey Kardashevskiy <aik@ozlabs.ru>
2017-02-24 13:36:44 +08:00
|
|
|
return;
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
}
|
target/ppc: Eliminate htab_base and htab_mask variables
CPUPPCState includes fields htab_base and htab_mask which store the base
address (GPA) and size (as a mask) of the guest's hashed page table (HPT).
These are set when the SDR1 register is updated.
Keeping these in sync with the SDR1 is actually a little bit fiddly, and
probably not useful for performance, since keeping them expands the size of
CPUPPCState. It also makes some upcoming changes harder to implement.
This patch removes these fields, in favour of calculating them directly
from the SDR1 contents when necessary.
This does make a change to the behaviour of attempting to write a bad value
(invalid HPT size) to the SDR1 with an mtspr instruction. Previously, the
bad value would be stored in SDR1 and could be retrieved with a later
mfspr, but the HPT size as used by the softmmu would be, clamped to the
allowed values. Now, writing a bad value is treated as a no-op. An error
message is printed in both new and old versions.
I'm not sure which behaviour, if either, matches real hardware. I don't
think it matters that much, since it's pretty clear that if an OS writes
a bad value to SDR1, it's not going to boot.
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
Reviewed-by: Alexey Kardashevskiy <aik@ozlabs.ru>
2017-02-24 13:36:44 +08:00
|
|
|
#endif /* defined(TARGET_PPC64) */
|
|
|
|
/* FIXME: Should check for valid HTABMASK values in 32-bit case */
|
|
|
|
env->spr[SPR_SDR1] = value;
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:34 +08:00
|
|
|
/* Segment registers load and store */
|
|
|
|
target_ulong helper_load_sr(CPUPPCState *env, target_ulong sr_num)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
2012-05-30 12:23:34 +08:00
|
|
|
#if defined(TARGET_PPC64)
|
|
|
|
if (env->mmu_model & POWERPC_MMU_64) {
|
|
|
|
/* XXX */
|
|
|
|
return 0;
|
|
|
|
}
|
2012-05-30 12:23:33 +08:00
|
|
|
#endif
|
2012-05-30 12:23:34 +08:00
|
|
|
return env->sr[sr_num];
|
|
|
|
}
|
2012-05-30 12:23:33 +08:00
|
|
|
|
2012-05-30 12:23:34 +08:00
|
|
|
void helper_store_sr(CPUPPCState *env, target_ulong srnum, target_ulong value)
|
2012-05-30 12:23:33 +08:00
|
|
|
{
|
2014-12-14 00:48:18 +08:00
|
|
|
qemu_log_mask(CPU_LOG_MMU,
|
|
|
|
"%s: reg=%d " TARGET_FMT_lx " " TARGET_FMT_lx "\n", __func__,
|
2012-05-30 12:23:34 +08:00
|
|
|
(int)srnum, value, env->sr[srnum]);
|
2012-05-30 12:23:33 +08:00
|
|
|
#if defined(TARGET_PPC64)
|
|
|
|
if (env->mmu_model & POWERPC_MMU_64) {
|
2016-06-07 10:50:22 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
2016-01-27 08:07:29 +08:00
|
|
|
uint64_t esid, vsid;
|
2012-05-30 12:23:33 +08:00
|
|
|
|
|
|
|
/* ESID = srnum */
|
2016-01-27 08:07:29 +08:00
|
|
|
esid = ((uint64_t)(srnum & 0xf) << 28) | SLB_ESID_V;
|
2012-05-30 12:23:33 +08:00
|
|
|
|
|
|
|
/* VSID = VSID */
|
2016-01-27 08:07:29 +08:00
|
|
|
vsid = (value & 0xfffffff) << 12;
|
2012-05-30 12:23:33 +08:00
|
|
|
/* flags = flags */
|
2016-01-27 08:07:29 +08:00
|
|
|
vsid |= ((value >> 27) & 0xf) << 8;
|
2012-05-30 12:23:33 +08:00
|
|
|
|
2016-01-27 08:07:29 +08:00
|
|
|
ppc_store_slb(cpu, srnum, esid, vsid);
|
2012-05-30 12:23:33 +08:00
|
|
|
} else
|
|
|
|
#endif
|
|
|
|
if (env->sr[srnum] != value) {
|
|
|
|
env->sr[srnum] = value;
|
|
|
|
/* Invalidating 256MB of virtual memory in 4kB pages is way longer than
|
|
|
|
flusing the whole TLB. */
|
|
|
|
#if !defined(FLUSH_ALL_TLBS) && 0
|
|
|
|
{
|
|
|
|
target_ulong page, end;
|
|
|
|
/* Invalidate 256 MB of virtual memory */
|
|
|
|
page = (16 << 20) * srnum;
|
|
|
|
end = page + (16 << 20);
|
|
|
|
for (; page != end; page += TARGET_PAGE_SIZE) {
|
2013-09-04 08:19:44 +08:00
|
|
|
tlb_flush_page(CPU(cpu), page);
|
2012-05-30 12:23:33 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
2016-09-21 00:34:59 +08:00
|
|
|
env->tlb_need_flush |= TLB_NEED_LOCAL_FLUSH;
|
2012-05-30 12:23:33 +08:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:30 +08:00
|
|
|
/* TLB management */
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_tlbia(CPUPPCState *env)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
|
|
|
ppc_tlb_invalidate_all(env);
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_tlbie(CPUPPCState *env, target_ulong addr)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
|
|
|
ppc_tlb_invalidate_one(env, addr);
|
|
|
|
}
|
|
|
|
|
2016-01-28 07:31:04 +08:00
|
|
|
void helper_tlbiva(CPUPPCState *env, target_ulong addr)
|
|
|
|
{
|
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
|
|
|
|
|
|
|
/* tlbiva instruction only exists on BookE */
|
|
|
|
assert(env->mmu_model == POWERPC_MMU_BOOKE);
|
|
|
|
/* XXX: TODO */
|
|
|
|
cpu_abort(CPU(cpu), "BookE MMU model is not implemented\n");
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:30 +08:00
|
|
|
/* Software driven TLBs management */
|
|
|
|
/* PowerPC 602/603 software TLB load instructions helpers */
|
2012-05-30 12:23:31 +08:00
|
|
|
static void do_6xx_tlb(CPUPPCState *env, target_ulong new_EPN, int is_code)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
|
|
|
target_ulong RPN, CMP, EPN;
|
|
|
|
int way;
|
|
|
|
|
|
|
|
RPN = env->spr[SPR_RPA];
|
|
|
|
if (is_code) {
|
|
|
|
CMP = env->spr[SPR_ICMP];
|
|
|
|
EPN = env->spr[SPR_IMISS];
|
|
|
|
} else {
|
|
|
|
CMP = env->spr[SPR_DCMP];
|
|
|
|
EPN = env->spr[SPR_DMISS];
|
|
|
|
}
|
|
|
|
way = (env->spr[SPR_SRR1] >> 17) & 1;
|
|
|
|
(void)EPN; /* avoid a compiler warning */
|
|
|
|
LOG_SWTLB("%s: EPN " TARGET_FMT_lx " " TARGET_FMT_lx " PTE0 " TARGET_FMT_lx
|
|
|
|
" PTE1 " TARGET_FMT_lx " way %d\n", __func__, new_EPN, EPN, CMP,
|
|
|
|
RPN, way);
|
|
|
|
/* Store this TLB */
|
|
|
|
ppc6xx_tlb_store(env, (uint32_t)(new_EPN & TARGET_PAGE_MASK),
|
|
|
|
way, is_code, CMP, RPN);
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_6xx_tlbd(CPUPPCState *env, target_ulong EPN)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
2012-05-30 12:23:31 +08:00
|
|
|
do_6xx_tlb(env, EPN, 0);
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_6xx_tlbi(CPUPPCState *env, target_ulong EPN)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
2012-05-30 12:23:31 +08:00
|
|
|
do_6xx_tlb(env, EPN, 1);
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* PowerPC 74xx software TLB load instructions helpers */
|
2012-05-30 12:23:31 +08:00
|
|
|
static void do_74xx_tlb(CPUPPCState *env, target_ulong new_EPN, int is_code)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
|
|
|
target_ulong RPN, CMP, EPN;
|
|
|
|
int way;
|
|
|
|
|
|
|
|
RPN = env->spr[SPR_PTELO];
|
|
|
|
CMP = env->spr[SPR_PTEHI];
|
|
|
|
EPN = env->spr[SPR_TLBMISS] & ~0x3;
|
|
|
|
way = env->spr[SPR_TLBMISS] & 0x3;
|
|
|
|
(void)EPN; /* avoid a compiler warning */
|
|
|
|
LOG_SWTLB("%s: EPN " TARGET_FMT_lx " " TARGET_FMT_lx " PTE0 " TARGET_FMT_lx
|
|
|
|
" PTE1 " TARGET_FMT_lx " way %d\n", __func__, new_EPN, EPN, CMP,
|
|
|
|
RPN, way);
|
|
|
|
/* Store this TLB */
|
|
|
|
ppc6xx_tlb_store(env, (uint32_t)(new_EPN & TARGET_PAGE_MASK),
|
|
|
|
way, is_code, CMP, RPN);
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_74xx_tlbd(CPUPPCState *env, target_ulong EPN)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
2012-05-30 12:23:31 +08:00
|
|
|
do_74xx_tlb(env, EPN, 0);
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_74xx_tlbi(CPUPPCState *env, target_ulong EPN)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
2012-05-30 12:23:31 +08:00
|
|
|
do_74xx_tlb(env, EPN, 1);
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
/* PowerPC 601 specific instructions (POWER bridge) */
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
target_ulong helper_rac(CPUPPCState *env, target_ulong addr)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
|
|
|
mmu_ctx_t ctx;
|
|
|
|
int nb_BATs;
|
|
|
|
target_ulong ret = 0;
|
|
|
|
|
|
|
|
/* We don't have to generate many instances of this instruction,
|
|
|
|
* as rac is supervisor only.
|
|
|
|
*/
|
|
|
|
/* XXX: FIX THIS: Pretend we have no BAT */
|
|
|
|
nb_BATs = env->nb_BATs;
|
|
|
|
env->nb_BATs = 0;
|
|
|
|
if (get_physical_address(env, &ctx, addr, 0, ACCESS_INT) == 0) {
|
|
|
|
ret = ctx.raddr;
|
|
|
|
}
|
|
|
|
env->nb_BATs = nb_BATs;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline target_ulong booke_tlb_to_page_size(int size)
|
|
|
|
{
|
|
|
|
return 1024 << (2 * size);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int booke_page_size_to_tlb(target_ulong page_size)
|
|
|
|
{
|
|
|
|
int size;
|
|
|
|
|
|
|
|
switch (page_size) {
|
|
|
|
case 0x00000400UL:
|
|
|
|
size = 0x0;
|
|
|
|
break;
|
|
|
|
case 0x00001000UL:
|
|
|
|
size = 0x1;
|
|
|
|
break;
|
|
|
|
case 0x00004000UL:
|
|
|
|
size = 0x2;
|
|
|
|
break;
|
|
|
|
case 0x00010000UL:
|
|
|
|
size = 0x3;
|
|
|
|
break;
|
|
|
|
case 0x00040000UL:
|
|
|
|
size = 0x4;
|
|
|
|
break;
|
|
|
|
case 0x00100000UL:
|
|
|
|
size = 0x5;
|
|
|
|
break;
|
|
|
|
case 0x00400000UL:
|
|
|
|
size = 0x6;
|
|
|
|
break;
|
|
|
|
case 0x01000000UL:
|
|
|
|
size = 0x7;
|
|
|
|
break;
|
|
|
|
case 0x04000000UL:
|
|
|
|
size = 0x8;
|
|
|
|
break;
|
|
|
|
case 0x10000000UL:
|
|
|
|
size = 0x9;
|
|
|
|
break;
|
|
|
|
case 0x40000000UL:
|
|
|
|
size = 0xA;
|
|
|
|
break;
|
|
|
|
#if defined(TARGET_PPC64)
|
|
|
|
case 0x000100000000ULL:
|
|
|
|
size = 0xB;
|
|
|
|
break;
|
|
|
|
case 0x000400000000ULL:
|
|
|
|
size = 0xC;
|
|
|
|
break;
|
|
|
|
case 0x001000000000ULL:
|
|
|
|
size = 0xD;
|
|
|
|
break;
|
|
|
|
case 0x004000000000ULL:
|
|
|
|
size = 0xE;
|
|
|
|
break;
|
|
|
|
case 0x010000000000ULL:
|
|
|
|
size = 0xF;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
size = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Helpers for 4xx TLB management */
|
|
|
|
#define PPC4XX_TLB_ENTRY_MASK 0x0000003f /* Mask for 64 TLB entries */
|
|
|
|
|
|
|
|
#define PPC4XX_TLBHI_V 0x00000040
|
|
|
|
#define PPC4XX_TLBHI_E 0x00000020
|
|
|
|
#define PPC4XX_TLBHI_SIZE_MIN 0
|
|
|
|
#define PPC4XX_TLBHI_SIZE_MAX 7
|
|
|
|
#define PPC4XX_TLBHI_SIZE_DEFAULT 1
|
|
|
|
#define PPC4XX_TLBHI_SIZE_SHIFT 7
|
|
|
|
#define PPC4XX_TLBHI_SIZE_MASK 0x00000007
|
|
|
|
|
|
|
|
#define PPC4XX_TLBLO_EX 0x00000200
|
|
|
|
#define PPC4XX_TLBLO_WR 0x00000100
|
|
|
|
#define PPC4XX_TLBLO_ATTR_MASK 0x000000FF
|
|
|
|
#define PPC4XX_TLBLO_RPN_MASK 0xFFFFFC00
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
target_ulong helper_4xx_tlbre_hi(CPUPPCState *env, target_ulong entry)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
|
|
|
ppcemb_tlb_t *tlb;
|
|
|
|
target_ulong ret;
|
|
|
|
int size;
|
|
|
|
|
|
|
|
entry &= PPC4XX_TLB_ENTRY_MASK;
|
|
|
|
tlb = &env->tlb.tlbe[entry];
|
|
|
|
ret = tlb->EPN;
|
|
|
|
if (tlb->prot & PAGE_VALID) {
|
|
|
|
ret |= PPC4XX_TLBHI_V;
|
|
|
|
}
|
|
|
|
size = booke_page_size_to_tlb(tlb->size);
|
|
|
|
if (size < PPC4XX_TLBHI_SIZE_MIN || size > PPC4XX_TLBHI_SIZE_MAX) {
|
|
|
|
size = PPC4XX_TLBHI_SIZE_DEFAULT;
|
|
|
|
}
|
|
|
|
ret |= size << PPC4XX_TLBHI_SIZE_SHIFT;
|
|
|
|
env->spr[SPR_40x_PID] = tlb->PID;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
target_ulong helper_4xx_tlbre_lo(CPUPPCState *env, target_ulong entry)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
|
|
|
ppcemb_tlb_t *tlb;
|
|
|
|
target_ulong ret;
|
|
|
|
|
|
|
|
entry &= PPC4XX_TLB_ENTRY_MASK;
|
|
|
|
tlb = &env->tlb.tlbe[entry];
|
|
|
|
ret = tlb->RPN;
|
|
|
|
if (tlb->prot & PAGE_EXEC) {
|
|
|
|
ret |= PPC4XX_TLBLO_EX;
|
|
|
|
}
|
|
|
|
if (tlb->prot & PAGE_WRITE) {
|
|
|
|
ret |= PPC4XX_TLBLO_WR;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_4xx_tlbwe_hi(CPUPPCState *env, target_ulong entry,
|
|
|
|
target_ulong val)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
2013-09-03 23:38:47 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
2013-09-04 07:29:02 +08:00
|
|
|
CPUState *cs = CPU(cpu);
|
2012-05-30 12:23:30 +08:00
|
|
|
ppcemb_tlb_t *tlb;
|
|
|
|
target_ulong page, end;
|
|
|
|
|
|
|
|
LOG_SWTLB("%s entry %d val " TARGET_FMT_lx "\n", __func__, (int)entry,
|
|
|
|
val);
|
|
|
|
entry &= PPC4XX_TLB_ENTRY_MASK;
|
|
|
|
tlb = &env->tlb.tlbe[entry];
|
|
|
|
/* Invalidate previous TLB (if it's valid) */
|
|
|
|
if (tlb->prot & PAGE_VALID) {
|
|
|
|
end = tlb->EPN + tlb->size;
|
|
|
|
LOG_SWTLB("%s: invalidate old TLB %d start " TARGET_FMT_lx " end "
|
|
|
|
TARGET_FMT_lx "\n", __func__, (int)entry, tlb->EPN, end);
|
|
|
|
for (page = tlb->EPN; page < end; page += TARGET_PAGE_SIZE) {
|
2013-09-04 07:29:02 +08:00
|
|
|
tlb_flush_page(cs, page);
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
tlb->size = booke_tlb_to_page_size((val >> PPC4XX_TLBHI_SIZE_SHIFT)
|
|
|
|
& PPC4XX_TLBHI_SIZE_MASK);
|
|
|
|
/* We cannot handle TLB size < TARGET_PAGE_SIZE.
|
|
|
|
* If this ever occurs, one should use the ppcemb target instead
|
|
|
|
* of the ppc or ppc64 one
|
|
|
|
*/
|
|
|
|
if ((val & PPC4XX_TLBHI_V) && tlb->size < TARGET_PAGE_SIZE) {
|
2013-09-04 07:29:02 +08:00
|
|
|
cpu_abort(cs, "TLB size " TARGET_FMT_lu " < %u "
|
2012-05-30 12:23:30 +08:00
|
|
|
"are not supported (%d)\n",
|
|
|
|
tlb->size, TARGET_PAGE_SIZE, (int)((val >> 7) & 0x7));
|
|
|
|
}
|
|
|
|
tlb->EPN = val & ~(tlb->size - 1);
|
|
|
|
if (val & PPC4XX_TLBHI_V) {
|
|
|
|
tlb->prot |= PAGE_VALID;
|
|
|
|
if (val & PPC4XX_TLBHI_E) {
|
|
|
|
/* XXX: TO BE FIXED */
|
2013-09-04 07:29:02 +08:00
|
|
|
cpu_abort(cs,
|
2012-05-30 12:23:30 +08:00
|
|
|
"Little-endian TLB entries are not supported by now\n");
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
tlb->prot &= ~PAGE_VALID;
|
|
|
|
}
|
|
|
|
tlb->PID = env->spr[SPR_40x_PID]; /* PID */
|
|
|
|
LOG_SWTLB("%s: set up TLB %d RPN " TARGET_FMT_plx " EPN " TARGET_FMT_lx
|
|
|
|
" size " TARGET_FMT_lx " prot %c%c%c%c PID %d\n", __func__,
|
|
|
|
(int)entry, tlb->RPN, tlb->EPN, tlb->size,
|
|
|
|
tlb->prot & PAGE_READ ? 'r' : '-',
|
|
|
|
tlb->prot & PAGE_WRITE ? 'w' : '-',
|
|
|
|
tlb->prot & PAGE_EXEC ? 'x' : '-',
|
|
|
|
tlb->prot & PAGE_VALID ? 'v' : '-', (int)tlb->PID);
|
|
|
|
/* Invalidate new TLB (if valid) */
|
|
|
|
if (tlb->prot & PAGE_VALID) {
|
|
|
|
end = tlb->EPN + tlb->size;
|
|
|
|
LOG_SWTLB("%s: invalidate TLB %d start " TARGET_FMT_lx " end "
|
|
|
|
TARGET_FMT_lx "\n", __func__, (int)entry, tlb->EPN, end);
|
|
|
|
for (page = tlb->EPN; page < end; page += TARGET_PAGE_SIZE) {
|
2013-09-04 07:29:02 +08:00
|
|
|
tlb_flush_page(cs, page);
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_4xx_tlbwe_lo(CPUPPCState *env, target_ulong entry,
|
|
|
|
target_ulong val)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
|
|
|
ppcemb_tlb_t *tlb;
|
|
|
|
|
|
|
|
LOG_SWTLB("%s entry %i val " TARGET_FMT_lx "\n", __func__, (int)entry,
|
|
|
|
val);
|
|
|
|
entry &= PPC4XX_TLB_ENTRY_MASK;
|
|
|
|
tlb = &env->tlb.tlbe[entry];
|
|
|
|
tlb->attr = val & PPC4XX_TLBLO_ATTR_MASK;
|
|
|
|
tlb->RPN = val & PPC4XX_TLBLO_RPN_MASK;
|
|
|
|
tlb->prot = PAGE_READ;
|
|
|
|
if (val & PPC4XX_TLBLO_EX) {
|
|
|
|
tlb->prot |= PAGE_EXEC;
|
|
|
|
}
|
|
|
|
if (val & PPC4XX_TLBLO_WR) {
|
|
|
|
tlb->prot |= PAGE_WRITE;
|
|
|
|
}
|
|
|
|
LOG_SWTLB("%s: set up TLB %d RPN " TARGET_FMT_plx " EPN " TARGET_FMT_lx
|
|
|
|
" size " TARGET_FMT_lx " prot %c%c%c%c PID %d\n", __func__,
|
|
|
|
(int)entry, tlb->RPN, tlb->EPN, tlb->size,
|
|
|
|
tlb->prot & PAGE_READ ? 'r' : '-',
|
|
|
|
tlb->prot & PAGE_WRITE ? 'w' : '-',
|
|
|
|
tlb->prot & PAGE_EXEC ? 'x' : '-',
|
|
|
|
tlb->prot & PAGE_VALID ? 'v' : '-', (int)tlb->PID);
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
target_ulong helper_4xx_tlbsx(CPUPPCState *env, target_ulong address)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
|
|
|
return ppcemb_tlb_search(env, address, env->spr[SPR_40x_PID]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* PowerPC 440 TLB management */
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_440_tlbwe(CPUPPCState *env, uint32_t word, target_ulong entry,
|
|
|
|
target_ulong value)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
2013-09-04 08:19:44 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
2012-05-30 12:23:30 +08:00
|
|
|
ppcemb_tlb_t *tlb;
|
|
|
|
target_ulong EPN, RPN, size;
|
|
|
|
int do_flush_tlbs;
|
|
|
|
|
|
|
|
LOG_SWTLB("%s word %d entry %d value " TARGET_FMT_lx "\n",
|
|
|
|
__func__, word, (int)entry, value);
|
|
|
|
do_flush_tlbs = 0;
|
|
|
|
entry &= 0x3F;
|
|
|
|
tlb = &env->tlb.tlbe[entry];
|
|
|
|
switch (word) {
|
|
|
|
default:
|
|
|
|
/* Just here to please gcc */
|
|
|
|
case 0:
|
|
|
|
EPN = value & 0xFFFFFC00;
|
|
|
|
if ((tlb->prot & PAGE_VALID) && EPN != tlb->EPN) {
|
|
|
|
do_flush_tlbs = 1;
|
|
|
|
}
|
|
|
|
tlb->EPN = EPN;
|
|
|
|
size = booke_tlb_to_page_size((value >> 4) & 0xF);
|
|
|
|
if ((tlb->prot & PAGE_VALID) && tlb->size < size) {
|
|
|
|
do_flush_tlbs = 1;
|
|
|
|
}
|
|
|
|
tlb->size = size;
|
|
|
|
tlb->attr &= ~0x1;
|
|
|
|
tlb->attr |= (value >> 8) & 1;
|
|
|
|
if (value & 0x200) {
|
|
|
|
tlb->prot |= PAGE_VALID;
|
|
|
|
} else {
|
|
|
|
if (tlb->prot & PAGE_VALID) {
|
|
|
|
tlb->prot &= ~PAGE_VALID;
|
|
|
|
do_flush_tlbs = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tlb->PID = env->spr[SPR_440_MMUCR] & 0x000000FF;
|
|
|
|
if (do_flush_tlbs) {
|
2016-11-14 22:17:28 +08:00
|
|
|
tlb_flush(CPU(cpu));
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
RPN = value & 0xFFFFFC0F;
|
|
|
|
if ((tlb->prot & PAGE_VALID) && tlb->RPN != RPN) {
|
2016-11-14 22:17:28 +08:00
|
|
|
tlb_flush(CPU(cpu));
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
tlb->RPN = RPN;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
tlb->attr = (tlb->attr & 0x1) | (value & 0x0000FF00);
|
|
|
|
tlb->prot = tlb->prot & PAGE_VALID;
|
|
|
|
if (value & 0x1) {
|
|
|
|
tlb->prot |= PAGE_READ << 4;
|
|
|
|
}
|
|
|
|
if (value & 0x2) {
|
|
|
|
tlb->prot |= PAGE_WRITE << 4;
|
|
|
|
}
|
|
|
|
if (value & 0x4) {
|
|
|
|
tlb->prot |= PAGE_EXEC << 4;
|
|
|
|
}
|
|
|
|
if (value & 0x8) {
|
|
|
|
tlb->prot |= PAGE_READ;
|
|
|
|
}
|
|
|
|
if (value & 0x10) {
|
|
|
|
tlb->prot |= PAGE_WRITE;
|
|
|
|
}
|
|
|
|
if (value & 0x20) {
|
|
|
|
tlb->prot |= PAGE_EXEC;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
target_ulong helper_440_tlbre(CPUPPCState *env, uint32_t word,
|
|
|
|
target_ulong entry)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
|
|
|
ppcemb_tlb_t *tlb;
|
|
|
|
target_ulong ret;
|
|
|
|
int size;
|
|
|
|
|
|
|
|
entry &= 0x3F;
|
|
|
|
tlb = &env->tlb.tlbe[entry];
|
|
|
|
switch (word) {
|
|
|
|
default:
|
|
|
|
/* Just here to please gcc */
|
|
|
|
case 0:
|
|
|
|
ret = tlb->EPN;
|
|
|
|
size = booke_page_size_to_tlb(tlb->size);
|
|
|
|
if (size < 0 || size > 0xF) {
|
|
|
|
size = 1;
|
|
|
|
}
|
|
|
|
ret |= size << 4;
|
|
|
|
if (tlb->attr & 0x1) {
|
|
|
|
ret |= 0x100;
|
|
|
|
}
|
|
|
|
if (tlb->prot & PAGE_VALID) {
|
|
|
|
ret |= 0x200;
|
|
|
|
}
|
|
|
|
env->spr[SPR_440_MMUCR] &= ~0x000000FF;
|
|
|
|
env->spr[SPR_440_MMUCR] |= tlb->PID;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
ret = tlb->RPN;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
ret = tlb->attr & ~0x1;
|
|
|
|
if (tlb->prot & (PAGE_READ << 4)) {
|
|
|
|
ret |= 0x1;
|
|
|
|
}
|
|
|
|
if (tlb->prot & (PAGE_WRITE << 4)) {
|
|
|
|
ret |= 0x2;
|
|
|
|
}
|
|
|
|
if (tlb->prot & (PAGE_EXEC << 4)) {
|
|
|
|
ret |= 0x4;
|
|
|
|
}
|
|
|
|
if (tlb->prot & PAGE_READ) {
|
|
|
|
ret |= 0x8;
|
|
|
|
}
|
|
|
|
if (tlb->prot & PAGE_WRITE) {
|
|
|
|
ret |= 0x10;
|
|
|
|
}
|
|
|
|
if (tlb->prot & PAGE_EXEC) {
|
|
|
|
ret |= 0x20;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
target_ulong helper_440_tlbsx(CPUPPCState *env, target_ulong address)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
|
|
|
return ppcemb_tlb_search(env, address, env->spr[SPR_440_MMUCR] & 0xFF);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* PowerPC BookE 2.06 TLB management */
|
|
|
|
|
|
|
|
static ppcmas_tlb_t *booke206_cur_tlb(CPUPPCState *env)
|
|
|
|
{
|
2013-09-03 23:38:47 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
2012-05-30 12:23:30 +08:00
|
|
|
uint32_t tlbncfg = 0;
|
|
|
|
int esel = (env->spr[SPR_BOOKE_MAS0] & MAS0_ESEL_MASK) >> MAS0_ESEL_SHIFT;
|
|
|
|
int ea = (env->spr[SPR_BOOKE_MAS2] & MAS2_EPN_MASK);
|
|
|
|
int tlb;
|
|
|
|
|
|
|
|
tlb = (env->spr[SPR_BOOKE_MAS0] & MAS0_TLBSEL_MASK) >> MAS0_TLBSEL_SHIFT;
|
|
|
|
tlbncfg = env->spr[SPR_BOOKE_TLB0CFG + tlb];
|
|
|
|
|
|
|
|
if ((tlbncfg & TLBnCFG_HES) && (env->spr[SPR_BOOKE_MAS0] & MAS0_HES)) {
|
2013-09-03 23:38:47 +08:00
|
|
|
cpu_abort(CPU(cpu), "we don't support HES yet\n");
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return booke206_get_tlbm(env, tlb, ea, esel);
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_booke_setpid(CPUPPCState *env, uint32_t pidn, target_ulong pid)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
2013-09-04 08:19:44 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
|
|
|
|
2012-05-30 12:23:30 +08:00
|
|
|
env->spr[pidn] = pid;
|
|
|
|
/* changing PIDs mean we're in a different address space now */
|
2016-11-14 22:17:28 +08:00
|
|
|
tlb_flush(CPU(cpu));
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_booke206_tlbwe(CPUPPCState *env)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
2013-09-03 23:38:47 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
2012-05-30 12:23:30 +08:00
|
|
|
uint32_t tlbncfg, tlbn;
|
|
|
|
ppcmas_tlb_t *tlb;
|
|
|
|
uint32_t size_tlb, size_ps;
|
2012-05-21 14:11:06 +08:00
|
|
|
target_ulong mask;
|
|
|
|
|
2012-05-30 12:23:30 +08:00
|
|
|
|
|
|
|
switch (env->spr[SPR_BOOKE_MAS0] & MAS0_WQ_MASK) {
|
|
|
|
case MAS0_WQ_ALWAYS:
|
|
|
|
/* good to go, write that entry */
|
|
|
|
break;
|
|
|
|
case MAS0_WQ_COND:
|
|
|
|
/* XXX check if reserved */
|
|
|
|
if (0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case MAS0_WQ_CLR_RSRV:
|
|
|
|
/* XXX clear entry */
|
|
|
|
return;
|
|
|
|
default:
|
|
|
|
/* no idea what to do */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (((env->spr[SPR_BOOKE_MAS0] & MAS0_ATSEL) == MAS0_ATSEL_LRAT) &&
|
|
|
|
!msr_gs) {
|
|
|
|
/* XXX we don't support direct LRAT setting yet */
|
|
|
|
fprintf(stderr, "cpu: don't support LRAT setting yet\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
tlbn = (env->spr[SPR_BOOKE_MAS0] & MAS0_TLBSEL_MASK) >> MAS0_TLBSEL_SHIFT;
|
|
|
|
tlbncfg = env->spr[SPR_BOOKE_TLB0CFG + tlbn];
|
|
|
|
|
|
|
|
tlb = booke206_cur_tlb(env);
|
|
|
|
|
|
|
|
if (!tlb) {
|
2016-07-27 14:56:37 +08:00
|
|
|
raise_exception_err_ra(env, POWERPC_EXCP_PROGRAM,
|
|
|
|
POWERPC_EXCP_INVAL |
|
|
|
|
POWERPC_EXCP_INVAL_INVAL, GETPC());
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* check that we support the targeted size */
|
|
|
|
size_tlb = (env->spr[SPR_BOOKE_MAS1] & MAS1_TSIZE_MASK) >> MAS1_TSIZE_SHIFT;
|
|
|
|
size_ps = booke206_tlbnps(env, tlbn);
|
|
|
|
if ((env->spr[SPR_BOOKE_MAS1] & MAS1_VALID) && (tlbncfg & TLBnCFG_AVAIL) &&
|
|
|
|
!(size_ps & (1 << size_tlb))) {
|
2016-07-27 14:56:37 +08:00
|
|
|
raise_exception_err_ra(env, POWERPC_EXCP_PROGRAM,
|
|
|
|
POWERPC_EXCP_INVAL |
|
|
|
|
POWERPC_EXCP_INVAL_INVAL, GETPC());
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (msr_gs) {
|
2013-09-03 23:38:47 +08:00
|
|
|
cpu_abort(CPU(cpu), "missing HV implementation\n");
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
tlb->mas7_3 = ((uint64_t)env->spr[SPR_BOOKE_MAS7] << 32) |
|
|
|
|
env->spr[SPR_BOOKE_MAS3];
|
|
|
|
tlb->mas1 = env->spr[SPR_BOOKE_MAS1];
|
|
|
|
|
|
|
|
/* MAV 1.0 only */
|
|
|
|
if (!(tlbncfg & TLBnCFG_AVAIL)) {
|
|
|
|
/* force !AVAIL TLB entries to correct page size */
|
|
|
|
tlb->mas1 &= ~MAS1_TSIZE_MASK;
|
|
|
|
/* XXX can be configured in MMUCSR0 */
|
|
|
|
tlb->mas1 |= (tlbncfg & TLBnCFG_MINSIZE) >> 12;
|
|
|
|
}
|
|
|
|
|
2012-05-21 14:11:06 +08:00
|
|
|
/* Make a mask from TLB size to discard invalid bits in EPN field */
|
|
|
|
mask = ~(booke206_tlb_to_page_size(env, tlb) - 1);
|
|
|
|
/* Add a mask for page attributes */
|
|
|
|
mask |= MAS2_ACM | MAS2_VLE | MAS2_W | MAS2_I | MAS2_M | MAS2_G | MAS2_E;
|
|
|
|
|
|
|
|
if (!msr_cm) {
|
|
|
|
/* Executing a tlbwe instruction in 32-bit mode will set
|
|
|
|
* bits 0:31 of the TLB EPN field to zero.
|
|
|
|
*/
|
|
|
|
mask &= 0xffffffff;
|
|
|
|
}
|
|
|
|
|
|
|
|
tlb->mas2 = env->spr[SPR_BOOKE_MAS2] & mask;
|
2012-05-30 12:23:30 +08:00
|
|
|
|
|
|
|
if (!(tlbncfg & TLBnCFG_IPROT)) {
|
|
|
|
/* no IPROT supported by TLB */
|
|
|
|
tlb->mas1 &= ~MAS1_IPROT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (booke206_tlb_to_page_size(env, tlb) == TARGET_PAGE_SIZE) {
|
2013-09-04 07:29:02 +08:00
|
|
|
tlb_flush_page(CPU(cpu), tlb->mas2 & MAS2_EPN_MASK);
|
2012-05-30 12:23:30 +08:00
|
|
|
} else {
|
2016-11-14 22:17:28 +08:00
|
|
|
tlb_flush(CPU(cpu));
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void booke206_tlb_to_mas(CPUPPCState *env, ppcmas_tlb_t *tlb)
|
|
|
|
{
|
|
|
|
int tlbn = booke206_tlbm_to_tlbn(env, tlb);
|
|
|
|
int way = booke206_tlbm_to_way(env, tlb);
|
|
|
|
|
|
|
|
env->spr[SPR_BOOKE_MAS0] = tlbn << MAS0_TLBSEL_SHIFT;
|
|
|
|
env->spr[SPR_BOOKE_MAS0] |= way << MAS0_ESEL_SHIFT;
|
|
|
|
env->spr[SPR_BOOKE_MAS0] |= env->last_way << MAS0_NV_SHIFT;
|
|
|
|
|
|
|
|
env->spr[SPR_BOOKE_MAS1] = tlb->mas1;
|
|
|
|
env->spr[SPR_BOOKE_MAS2] = tlb->mas2;
|
|
|
|
env->spr[SPR_BOOKE_MAS3] = tlb->mas7_3;
|
|
|
|
env->spr[SPR_BOOKE_MAS7] = tlb->mas7_3 >> 32;
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_booke206_tlbre(CPUPPCState *env)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
|
|
|
ppcmas_tlb_t *tlb = NULL;
|
|
|
|
|
|
|
|
tlb = booke206_cur_tlb(env);
|
|
|
|
if (!tlb) {
|
|
|
|
env->spr[SPR_BOOKE_MAS1] = 0;
|
|
|
|
} else {
|
|
|
|
booke206_tlb_to_mas(env, tlb);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_booke206_tlbsx(CPUPPCState *env, target_ulong address)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
|
|
|
ppcmas_tlb_t *tlb = NULL;
|
|
|
|
int i, j;
|
2012-10-23 18:30:10 +08:00
|
|
|
hwaddr raddr;
|
2012-05-30 12:23:30 +08:00
|
|
|
uint32_t spid, sas;
|
|
|
|
|
|
|
|
spid = (env->spr[SPR_BOOKE_MAS6] & MAS6_SPID_MASK) >> MAS6_SPID_SHIFT;
|
|
|
|
sas = env->spr[SPR_BOOKE_MAS6] & MAS6_SAS;
|
|
|
|
|
|
|
|
for (i = 0; i < BOOKE206_MAX_TLBN; i++) {
|
|
|
|
int ways = booke206_tlb_ways(env, i);
|
|
|
|
|
|
|
|
for (j = 0; j < ways; j++) {
|
|
|
|
tlb = booke206_get_tlbm(env, i, address, j);
|
|
|
|
|
|
|
|
if (!tlb) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ppcmas_tlb_check(env, tlb, &raddr, address, spid)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sas != ((tlb->mas1 & MAS1_TS) >> MAS1_TS_SHIFT)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
booke206_tlb_to_mas(env, tlb);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* no entry found, fill with defaults */
|
|
|
|
env->spr[SPR_BOOKE_MAS0] = env->spr[SPR_BOOKE_MAS4] & MAS4_TLBSELD_MASK;
|
|
|
|
env->spr[SPR_BOOKE_MAS1] = env->spr[SPR_BOOKE_MAS4] & MAS4_TSIZED_MASK;
|
|
|
|
env->spr[SPR_BOOKE_MAS2] = env->spr[SPR_BOOKE_MAS4] & MAS4_WIMGED_MASK;
|
|
|
|
env->spr[SPR_BOOKE_MAS3] = 0;
|
|
|
|
env->spr[SPR_BOOKE_MAS7] = 0;
|
|
|
|
|
|
|
|
if (env->spr[SPR_BOOKE_MAS6] & MAS6_SAS) {
|
|
|
|
env->spr[SPR_BOOKE_MAS1] |= MAS1_TS;
|
|
|
|
}
|
|
|
|
|
|
|
|
env->spr[SPR_BOOKE_MAS1] |= (env->spr[SPR_BOOKE_MAS6] >> 16)
|
|
|
|
<< MAS1_TID_SHIFT;
|
|
|
|
|
|
|
|
/* next victim logic */
|
|
|
|
env->spr[SPR_BOOKE_MAS0] |= env->last_way << MAS0_ESEL_SHIFT;
|
|
|
|
env->last_way++;
|
|
|
|
env->last_way &= booke206_tlb_ways(env, 0) - 1;
|
|
|
|
env->spr[SPR_BOOKE_MAS0] |= env->last_way << MAS0_NV_SHIFT;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void booke206_invalidate_ea_tlb(CPUPPCState *env, int tlbn,
|
|
|
|
uint32_t ea)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
int ways = booke206_tlb_ways(env, tlbn);
|
|
|
|
target_ulong mask;
|
|
|
|
|
|
|
|
for (i = 0; i < ways; i++) {
|
|
|
|
ppcmas_tlb_t *tlb = booke206_get_tlbm(env, tlbn, ea, i);
|
|
|
|
if (!tlb) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
mask = ~(booke206_tlb_to_page_size(env, tlb) - 1);
|
|
|
|
if (((tlb->mas2 & MAS2_EPN_MASK) == (ea & mask)) &&
|
|
|
|
!(tlb->mas1 & MAS1_IPROT)) {
|
|
|
|
tlb->mas1 &= ~MAS1_VALID;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_booke206_tlbivax(CPUPPCState *env, target_ulong address)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
2016-09-21 00:35:01 +08:00
|
|
|
CPUState *cs;
|
2013-09-04 07:29:02 +08:00
|
|
|
|
2012-05-30 12:23:30 +08:00
|
|
|
if (address & 0x4) {
|
|
|
|
/* flush all entries */
|
|
|
|
if (address & 0x8) {
|
|
|
|
/* flush all of TLB1 */
|
|
|
|
booke206_flush_tlb(env, BOOKE206_FLUSH_TLB1, 1);
|
|
|
|
} else {
|
|
|
|
/* flush all of TLB0 */
|
|
|
|
booke206_flush_tlb(env, BOOKE206_FLUSH_TLB0, 0);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (address & 0x8) {
|
|
|
|
/* flush TLB1 entries */
|
|
|
|
booke206_invalidate_ea_tlb(env, 1, address);
|
2016-09-21 00:35:01 +08:00
|
|
|
CPU_FOREACH(cs) {
|
2016-11-14 22:17:28 +08:00
|
|
|
tlb_flush(cs);
|
2016-09-21 00:35:01 +08:00
|
|
|
}
|
2012-05-30 12:23:30 +08:00
|
|
|
} else {
|
|
|
|
/* flush TLB0 entries */
|
|
|
|
booke206_invalidate_ea_tlb(env, 0, address);
|
2016-09-21 00:35:01 +08:00
|
|
|
CPU_FOREACH(cs) {
|
|
|
|
tlb_flush_page(cs, address & MAS2_EPN_MASK);
|
|
|
|
}
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_booke206_tlbilx0(CPUPPCState *env, target_ulong address)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
|
|
|
/* XXX missing LPID handling */
|
|
|
|
booke206_flush_tlb(env, -1, 1);
|
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_booke206_tlbilx1(CPUPPCState *env, target_ulong address)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
2013-09-04 08:19:44 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
2012-05-30 12:23:30 +08:00
|
|
|
int i, j;
|
|
|
|
int tid = (env->spr[SPR_BOOKE_MAS6] & MAS6_SPID);
|
|
|
|
ppcmas_tlb_t *tlb = env->tlb.tlbm;
|
|
|
|
int tlb_size;
|
|
|
|
|
|
|
|
/* XXX missing LPID handling */
|
|
|
|
for (i = 0; i < BOOKE206_MAX_TLBN; i++) {
|
|
|
|
tlb_size = booke206_tlb_size(env, i);
|
|
|
|
for (j = 0; j < tlb_size; j++) {
|
|
|
|
if (!(tlb[j].mas1 & MAS1_IPROT) &&
|
|
|
|
((tlb[j].mas1 & MAS1_TID_MASK) == tid)) {
|
|
|
|
tlb[j].mas1 &= ~MAS1_VALID;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tlb += booke206_tlb_size(env, i);
|
|
|
|
}
|
2016-11-14 22:17:28 +08:00
|
|
|
tlb_flush(CPU(cpu));
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
|
2012-05-30 12:23:31 +08:00
|
|
|
void helper_booke206_tlbilx3(CPUPPCState *env, target_ulong address)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
2013-09-04 08:19:44 +08:00
|
|
|
PowerPCCPU *cpu = ppc_env_get_cpu(env);
|
2012-05-30 12:23:30 +08:00
|
|
|
int i, j;
|
|
|
|
ppcmas_tlb_t *tlb;
|
|
|
|
int tid = (env->spr[SPR_BOOKE_MAS6] & MAS6_SPID);
|
|
|
|
int pid = tid >> MAS6_SPID_SHIFT;
|
|
|
|
int sgs = env->spr[SPR_BOOKE_MAS5] & MAS5_SGS;
|
|
|
|
int ind = (env->spr[SPR_BOOKE_MAS6] & MAS6_SIND) ? MAS1_IND : 0;
|
|
|
|
/* XXX check for unsupported isize and raise an invalid opcode then */
|
|
|
|
int size = env->spr[SPR_BOOKE_MAS6] & MAS6_ISIZE_MASK;
|
|
|
|
/* XXX implement MAV2 handling */
|
|
|
|
bool mav2 = false;
|
|
|
|
|
|
|
|
/* XXX missing LPID handling */
|
|
|
|
/* flush by pid and ea */
|
|
|
|
for (i = 0; i < BOOKE206_MAX_TLBN; i++) {
|
|
|
|
int ways = booke206_tlb_ways(env, i);
|
|
|
|
|
|
|
|
for (j = 0; j < ways; j++) {
|
|
|
|
tlb = booke206_get_tlbm(env, i, address, j);
|
|
|
|
if (!tlb) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if ((ppcmas_tlb_check(env, tlb, NULL, address, pid) != 0) ||
|
|
|
|
(tlb->mas1 & MAS1_IPROT) ||
|
|
|
|
((tlb->mas1 & MAS1_IND) != ind) ||
|
|
|
|
((tlb->mas8 & MAS8_TGS) != sgs)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (mav2 && ((tlb->mas1 & MAS1_TSIZE_MASK) != size)) {
|
|
|
|
/* XXX only check when MMUCFG[TWC] || TLBnCFG[HES] */
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
/* XXX e500mc doesn't match SAS, but other cores might */
|
|
|
|
tlb->mas1 &= ~MAS1_VALID;
|
|
|
|
}
|
|
|
|
}
|
2016-11-14 22:17:28 +08:00
|
|
|
tlb_flush(CPU(cpu));
|
2012-05-30 12:23:30 +08:00
|
|
|
}
|
|
|
|
|
2014-05-29 01:25:36 +08:00
|
|
|
void helper_booke206_tlbflush(CPUPPCState *env, target_ulong type)
|
2012-05-30 12:23:30 +08:00
|
|
|
{
|
|
|
|
int flags = 0;
|
|
|
|
|
|
|
|
if (type & 2) {
|
|
|
|
flags |= BOOKE206_FLUSH_TLB1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (type & 4) {
|
|
|
|
flags |= BOOKE206_FLUSH_TLB0;
|
|
|
|
}
|
|
|
|
|
|
|
|
booke206_flush_tlb(env, flags, 1);
|
|
|
|
}
|
2013-03-12 08:31:49 +08:00
|
|
|
|
|
|
|
|
2016-09-21 00:35:00 +08:00
|
|
|
void helper_check_tlb_flush_local(CPUPPCState *env)
|
ppc: Do some batching of TCG tlb flushes
On ppc64 especially, we flush the tlb on any slbie or tlbie instruction.
However, those instructions often come in bursts of 3 or more (context
switch will favor a series of slbie's for example to an slbia if the
SLB has less than a certain number of entries in it, and tlbie's can
happen in a series, with PAPR, H_BULK_REMOVE can remove up to 4 entries
at a time.
Doing a tlb_flush() each time is a waste of time. We end up doing a memset
of the whole TLB, reloading it for the next instruction, memset'ing again,
etc...
Those instructions don't have to take effect immediately. For slbie, they
can wait for the next context synchronizing event. For tlbie, the next
tlbsync.
This implements batching by keeping a flag that indicates that we have a
TLB in need of flushing. We check it on interrupts, rfi's, isync's and
tlbsync and flush the TLB if needed.
This reduces the number of tlb_flush() on a boot to a ubuntu installer
first dialog screen from roughly 360K down to 36K.
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
[clg: added a 'CPUPPCState *' variable in h_remove() and
h_bulk_remove() ]
Signed-off-by: Cédric Le Goater <clg@kaod.org>
[dwg: removed spurious whitespace change, use 0/1 not true/false
consistently, since tlb_need_flush has int type]
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
2016-05-04 00:03:25 +08:00
|
|
|
{
|
2016-09-21 00:35:00 +08:00
|
|
|
check_tlb_flush(env, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
void helper_check_tlb_flush_global(CPUPPCState *env)
|
|
|
|
{
|
|
|
|
check_tlb_flush(env, true);
|
ppc: Do some batching of TCG tlb flushes
On ppc64 especially, we flush the tlb on any slbie or tlbie instruction.
However, those instructions often come in bursts of 3 or more (context
switch will favor a series of slbie's for example to an slbia if the
SLB has less than a certain number of entries in it, and tlbie's can
happen in a series, with PAPR, H_BULK_REMOVE can remove up to 4 entries
at a time.
Doing a tlb_flush() each time is a waste of time. We end up doing a memset
of the whole TLB, reloading it for the next instruction, memset'ing again,
etc...
Those instructions don't have to take effect immediately. For slbie, they
can wait for the next context synchronizing event. For tlbie, the next
tlbsync.
This implements batching by keeping a flag that indicates that we have a
TLB in need of flushing. We check it on interrupts, rfi's, isync's and
tlbsync and flush the TLB if needed.
This reduces the number of tlb_flush() on a boot to a ubuntu installer
first dialog screen from roughly 360K down to 36K.
Signed-off-by: Benjamin Herrenschmidt <benh@kernel.crashing.org>
[clg: added a 'CPUPPCState *' variable in h_remove() and
h_bulk_remove() ]
Signed-off-by: Cédric Le Goater <clg@kaod.org>
[dwg: removed spurious whitespace change, use 0/1 not true/false
consistently, since tlb_need_flush has int type]
Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
2016-05-04 00:03:25 +08:00
|
|
|
}
|
|
|
|
|
2013-03-12 08:31:49 +08:00
|
|
|
/*****************************************************************************/
|
|
|
|
|
|
|
|
/* try to fill the TLB and return an exception if error. If retaddr is
|
|
|
|
NULL, it means that the function was called in C code (i.e. not
|
|
|
|
from generated code or from helper.c) */
|
|
|
|
/* XXX: fix it to restore all registers */
|
2016-06-14 20:26:17 +08:00
|
|
|
void tlb_fill(CPUState *cs, target_ulong addr, MMUAccessType access_type,
|
|
|
|
int mmu_idx, uintptr_t retaddr)
|
2013-03-12 08:31:49 +08:00
|
|
|
{
|
2013-08-27 06:28:06 +08:00
|
|
|
PowerPCCPU *cpu = POWERPC_CPU(cs);
|
|
|
|
PowerPCCPUClass *pcc = POWERPC_CPU_GET_CLASS(cs);
|
|
|
|
CPUPPCState *env = &cpu->env;
|
2013-03-12 08:31:49 +08:00
|
|
|
int ret;
|
|
|
|
|
2013-03-13 08:40:33 +08:00
|
|
|
if (pcc->handle_mmu_fault) {
|
2016-06-14 20:26:17 +08:00
|
|
|
ret = pcc->handle_mmu_fault(cpu, addr, access_type, mmu_idx);
|
2013-03-13 08:40:33 +08:00
|
|
|
} else {
|
2016-06-14 20:26:17 +08:00
|
|
|
ret = cpu_ppc_handle_mmu_fault(env, addr, access_type, mmu_idx);
|
2013-03-13 08:40:33 +08:00
|
|
|
}
|
2013-03-12 08:31:49 +08:00
|
|
|
if (unlikely(ret != 0)) {
|
2016-07-27 14:56:32 +08:00
|
|
|
raise_exception_err_ra(env, cs->exception_index, env->error_code,
|
|
|
|
retaddr);
|
2013-03-12 08:31:49 +08:00
|
|
|
}
|
|
|
|
}
|