sparc64: handle asi referencing nucleus and secondary MMU contexts

- increase max supported MMU modes to 6
- handle nucleus context asi
- handle secondary context asi
- handle non-faulting loads from secondary context

Signed-off-by: Igor V. Kovalenko <igor.v.kovalenko@gmail.com>
Signed-off-by: Blue Swirl <blauwirbel@gmail.com>
This commit is contained in:
Igor V. Kovalenko 2010-05-03 11:29:44 +04:00 committed by Blue Swirl
parent 299b520cd4
commit 2065061ede
5 changed files with 257 additions and 94 deletions

View File

@ -100,9 +100,28 @@
#undef MEMSUFFIX #undef MEMSUFFIX
#endif /* (NB_MMU_MODES >= 5) */ #endif /* (NB_MMU_MODES >= 5) */
#if (NB_MMU_MODES > 5) #if (NB_MMU_MODES >= 6)
#error "NB_MMU_MODES > 5 is not supported for now"
#endif /* (NB_MMU_MODES > 5) */ #define ACCESS_TYPE 5
#define MEMSUFFIX MMU_MODE5_SUFFIX
#define DATA_SIZE 1
#include "softmmu_header.h"
#define DATA_SIZE 2
#include "softmmu_header.h"
#define DATA_SIZE 4
#include "softmmu_header.h"
#define DATA_SIZE 8
#include "softmmu_header.h"
#undef ACCESS_TYPE
#undef MEMSUFFIX
#endif /* (NB_MMU_MODES >= 6) */
#if (NB_MMU_MODES > 6)
#error "NB_MMU_MODES > 6 is not supported for now"
#endif /* (NB_MMU_MODES > 6) */
/* these access are slower, they must be as rare as possible */ /* these access are slower, they must be as rare as possible */
#define ACCESS_TYPE (NB_MMU_MODES) #define ACCESS_TYPE (NB_MMU_MODES)

View File

@ -224,7 +224,7 @@ enum {
#if !defined(TARGET_SPARC64) #if !defined(TARGET_SPARC64)
#define NB_MMU_MODES 2 #define NB_MMU_MODES 2
#else #else
#define NB_MMU_MODES 3 #define NB_MMU_MODES 6
typedef struct trap_state { typedef struct trap_state {
uint64_t tpc; uint64_t tpc;
uint64_t tnpc; uint64_t tnpc;
@ -571,6 +571,9 @@ static inline void PUT_CWP64(CPUSPARCState *env1, int cwp)
#if !defined(CONFIG_USER_ONLY) #if !defined(CONFIG_USER_ONLY)
void do_unassigned_access(target_phys_addr_t addr, int is_write, int is_exec, void do_unassigned_access(target_phys_addr_t addr, int is_write, int is_exec,
int is_asi, int size); int is_asi, int size);
target_phys_addr_t cpu_get_phys_page_nofault(CPUState *env, target_ulong addr,
int mmu_idx);
#endif #endif
int cpu_sparc_signal_handler(int host_signum, void *pinfo, void *puc); int cpu_sparc_signal_handler(int host_signum, void *pinfo, void *puc);
@ -587,10 +590,18 @@ int cpu_sparc_signal_handler(int host_signum, void *pinfo, void *puc);
#define MMU_MODE1_SUFFIX _kernel #define MMU_MODE1_SUFFIX _kernel
#ifdef TARGET_SPARC64 #ifdef TARGET_SPARC64
#define MMU_MODE2_SUFFIX _hypv #define MMU_MODE2_SUFFIX _hypv
#define MMU_MODE3_SUFFIX _nucleus
#define MMU_MODE4_SUFFIX _user_secondary
#define MMU_MODE5_SUFFIX _kernel_secondary
#endif #endif
#define MMU_USER_IDX 0 #define MMU_USER_IDX 0
#define MMU_KERNEL_IDX 1 #define MMU_KERNEL_IDX 1
#define MMU_HYPV_IDX 2 #define MMU_HYPV_IDX 2
#ifdef TARGET_SPARC64
#define MMU_NUCLEUS_IDX 3
#define MMU_USER_SECONDARY_IDX 4
#define MMU_KERNEL_SECONDARY_IDX 5
#endif
static inline int cpu_mmu_index(CPUState *env1) static inline int cpu_mmu_index(CPUState *env1)
{ {

View File

@ -13,6 +13,10 @@ register struct CPUSPARCState *env asm(AREG0);
#include "cpu.h" #include "cpu.h"
#include "exec-all.h" #include "exec-all.h"
#if !defined(CONFIG_USER_ONLY)
#include "softmmu_exec.h"
#endif /* !defined(CONFIG_USER_ONLY) */
/* op_helper.c */ /* op_helper.c */
void do_interrupt(CPUState *env); void do_interrupt(CPUState *env);

View File

@ -420,21 +420,32 @@ static inline int ultrasparc_tag_match(SparcTLBEntry *tlb,
static int get_physical_address_data(CPUState *env, static int get_physical_address_data(CPUState *env,
target_phys_addr_t *physical, int *prot, target_phys_addr_t *physical, int *prot,
target_ulong address, int rw, int is_user) target_ulong address, int rw, int mmu_idx)
{ {
unsigned int i; unsigned int i;
uint64_t context; uint64_t context;
int is_user = (mmu_idx == MMU_USER_IDX ||
mmu_idx == MMU_USER_SECONDARY_IDX);
if ((env->lsu & DMMU_E) == 0) { /* DMMU disabled */ if ((env->lsu & DMMU_E) == 0) { /* DMMU disabled */
*physical = ultrasparc_truncate_physical(address); *physical = ultrasparc_truncate_physical(address);
*prot = PAGE_READ | PAGE_WRITE; *prot = PAGE_READ | PAGE_WRITE;
return 0; return 0;
} }
if (env->tl == 0) { switch(mmu_idx) {
case MMU_USER_IDX:
case MMU_KERNEL_IDX:
context = env->dmmu.mmu_primary_context & 0x1fff; context = env->dmmu.mmu_primary_context & 0x1fff;
} else { break;
case MMU_USER_SECONDARY_IDX:
case MMU_KERNEL_SECONDARY_IDX:
context = env->dmmu.mmu_secondary_context & 0x1fff;
break;
case MMU_NUCLEUS_IDX:
context = 0; context = 0;
break;
} }
for (i = 0; i < 64; i++) { for (i = 0; i < 64; i++) {
@ -482,11 +493,14 @@ static int get_physical_address_data(CPUState *env,
static int get_physical_address_code(CPUState *env, static int get_physical_address_code(CPUState *env,
target_phys_addr_t *physical, int *prot, target_phys_addr_t *physical, int *prot,
target_ulong address, int is_user) target_ulong address, int mmu_idx)
{ {
unsigned int i; unsigned int i;
uint64_t context; uint64_t context;
int is_user = (mmu_idx == MMU_USER_IDX ||
mmu_idx == MMU_USER_SECONDARY_IDX);
if ((env->lsu & IMMU_E) == 0 || (env->pstate & PS_RED) != 0) { if ((env->lsu & IMMU_E) == 0 || (env->pstate & PS_RED) != 0) {
/* IMMU disabled */ /* IMMU disabled */
*physical = ultrasparc_truncate_physical(address); *physical = ultrasparc_truncate_physical(address);
@ -495,8 +509,10 @@ static int get_physical_address_code(CPUState *env,
} }
if (env->tl == 0) { if (env->tl == 0) {
/* PRIMARY context */
context = env->dmmu.mmu_primary_context & 0x1fff; context = env->dmmu.mmu_primary_context & 0x1fff;
} else { } else {
/* NUCLEUS context */
context = 0; context = 0;
} }
@ -535,17 +551,15 @@ static int get_physical_address(CPUState *env, target_phys_addr_t *physical,
target_ulong address, int rw, int mmu_idx, target_ulong address, int rw, int mmu_idx,
target_ulong *page_size) target_ulong *page_size)
{ {
int is_user = mmu_idx == MMU_USER_IDX;
/* ??? We treat everything as a small page, then explicitly flush /* ??? We treat everything as a small page, then explicitly flush
everything when an entry is evicted. */ everything when an entry is evicted. */
*page_size = TARGET_PAGE_SIZE; *page_size = TARGET_PAGE_SIZE;
if (rw == 2) if (rw == 2)
return get_physical_address_code(env, physical, prot, address, return get_physical_address_code(env, physical, prot, address,
is_user); mmu_idx);
else else
return get_physical_address_data(env, physical, prot, address, rw, return get_physical_address_data(env, physical, prot, address, rw,
is_user); mmu_idx);
} }
/* Perform address translation */ /* Perform address translation */
@ -659,21 +673,27 @@ void dump_mmu(CPUState *env)
#if !defined(CONFIG_USER_ONLY) #if !defined(CONFIG_USER_ONLY)
target_phys_addr_t cpu_get_phys_page_debug(CPUState *env, target_ulong addr) target_phys_addr_t cpu_get_phys_page_nofault(CPUState *env, target_ulong addr,
int mmu_idx)
{ {
target_phys_addr_t phys_addr; target_phys_addr_t phys_addr;
target_ulong page_size; target_ulong page_size;
int prot, access_index; int prot, access_index;
if (get_physical_address(env, &phys_addr, &prot, &access_index, addr, 2, if (get_physical_address(env, &phys_addr, &prot, &access_index, addr, 2,
MMU_KERNEL_IDX, &page_size) != 0) mmu_idx, &page_size) != 0)
if (get_physical_address(env, &phys_addr, &prot, &access_index, addr, if (get_physical_address(env, &phys_addr, &prot, &access_index, addr,
0, MMU_KERNEL_IDX, &page_size) != 0) 0, mmu_idx, &page_size) != 0)
return -1; return -1;
if (cpu_get_physical_page_desc(phys_addr) == IO_MEM_UNASSIGNED) if (cpu_get_physical_page_desc(phys_addr) == IO_MEM_UNASSIGNED)
return -1; return -1;
return phys_addr; return phys_addr;
} }
target_phys_addr_t cpu_get_phys_page_debug(CPUState *env, target_ulong addr)
{
return cpu_get_phys_page_nofault(env, addr, MMU_KERNEL_IDX);
}
#endif #endif
void cpu_reset(CPUSPARCState *env) void cpu_reset(CPUSPARCState *env)

View File

@ -1,9 +1,6 @@
#include "exec.h" #include "exec.h"
#include "host-utils.h" #include "host-utils.h"
#include "helper.h" #include "helper.h"
#if !defined(CONFIG_USER_ONLY)
#include "softmmu_exec.h"
#endif /* !defined(CONFIG_USER_ONLY) */
//#define DEBUG_MMU //#define DEBUG_MMU
//#define DEBUG_MXCC //#define DEBUG_MXCC
@ -2142,17 +2139,29 @@ uint64_t helper_ld_asi(target_ulong addr, int asi, int size, int sign)
switch (asi) { switch (asi) {
case 0x82: // Primary no-fault case 0x82: // Primary no-fault
case 0x8a: // Primary no-fault LE case 0x8a: // Primary no-fault LE
if (cpu_get_phys_page_debug(env, addr) == -1ULL) { case 0x83: // Secondary no-fault
case 0x8b: // Secondary no-fault LE
{
/* secondary space access has lowest asi bit equal to 1 */
int access_mmu_idx = ( asi & 1 ) ? MMU_KERNEL_IDX
: MMU_KERNEL_SECONDARY_IDX;
if (cpu_get_phys_page_nofault(env, addr, access_mmu_idx) == -1ULL) {
#ifdef DEBUG_ASI #ifdef DEBUG_ASI
dump_asi("read ", last_addr, asi, size, ret); dump_asi("read ", last_addr, asi, size, ret);
#endif #endif
return 0; return 0;
}
} }
// Fall through // Fall through
case 0x10: // As if user primary case 0x10: // As if user primary
case 0x11: // As if user secondary
case 0x18: // As if user primary LE case 0x18: // As if user primary LE
case 0x19: // As if user secondary LE
case 0x80: // Primary case 0x80: // Primary
case 0x81: // Secondary
case 0x88: // Primary LE case 0x88: // Primary LE
case 0x89: // Secondary LE
case 0xe2: // UA2007 Primary block init case 0xe2: // UA2007 Primary block init
case 0xe3: // UA2007 Secondary block init case 0xe3: // UA2007 Secondary block init
if ((asi & 0x80) && (env->pstate & PS_PRIV)) { if ((asi & 0x80) && (env->pstate & PS_PRIV)) {
@ -2174,37 +2183,75 @@ uint64_t helper_ld_asi(target_ulong addr, int asi, int size, int sign)
break; break;
} }
} else { } else {
switch(size) { /* secondary space access has lowest asi bit equal to 1 */
case 1: if (asi & 1) {
ret = ldub_kernel(addr); switch(size) {
break; case 1:
case 2: ret = ldub_kernel_secondary(addr);
ret = lduw_kernel(addr); break;
break; case 2:
case 4: ret = lduw_kernel_secondary(addr);
ret = ldl_kernel(addr); break;
break; case 4:
default: ret = ldl_kernel_secondary(addr);
case 8: break;
ret = ldq_kernel(addr); default:
break; case 8:
ret = ldq_kernel_secondary(addr);
break;
}
} else {
switch(size) {
case 1:
ret = ldub_kernel(addr);
break;
case 2:
ret = lduw_kernel(addr);
break;
case 4:
ret = ldl_kernel(addr);
break;
default:
case 8:
ret = ldq_kernel(addr);
break;
}
} }
} }
} else { } else {
switch(size) { /* secondary space access has lowest asi bit equal to 1 */
case 1: if (asi & 1) {
ret = ldub_user(addr); switch(size) {
break; case 1:
case 2: ret = ldub_user_secondary(addr);
ret = lduw_user(addr); break;
break; case 2:
case 4: ret = lduw_user_secondary(addr);
ret = ldl_user(addr); break;
break; case 4:
default: ret = ldl_user_secondary(addr);
case 8: break;
ret = ldq_user(addr); default:
break; case 8:
ret = ldq_user_secondary(addr);
break;
}
} else {
switch(size) {
case 1:
ret = ldub_user(addr);
break;
case 2:
ret = lduw_user(addr);
break;
case 4:
ret = ldl_user(addr);
break;
default:
case 8:
ret = ldq_user(addr);
break;
}
} }
} }
break; break;
@ -2235,22 +2282,27 @@ uint64_t helper_ld_asi(target_ulong addr, int asi, int size, int sign)
// Only ldda allowed // Only ldda allowed
raise_exception(TT_ILL_INSN); raise_exception(TT_ILL_INSN);
return 0; return 0;
case 0x83: // Secondary no-fault
case 0x8b: // Secondary no-fault LE
if (cpu_get_phys_page_debug(env, addr) == -1ULL) {
#ifdef DEBUG_ASI
dump_asi("read ", last_addr, asi, size, ret);
#endif
return 0;
}
// Fall through
case 0x04: // Nucleus case 0x04: // Nucleus
case 0x0c: // Nucleus Little Endian (LE) case 0x0c: // Nucleus Little Endian (LE)
case 0x11: // As if user secondary {
case 0x19: // As if user secondary LE switch(size) {
case 1:
ret = ldub_nucleus(addr);
break;
case 2:
ret = lduw_nucleus(addr);
break;
case 4:
ret = ldl_nucleus(addr);
break;
default:
case 8:
ret = ldq_nucleus(addr);
break;
}
break;
}
case 0x4a: // UPA config case 0x4a: // UPA config
case 0x81: // Secondary
case 0x89: // Secondary LE
// XXX // XXX
break; break;
case 0x45: // LSU case 0x45: // LSU
@ -2464,9 +2516,13 @@ void helper_st_asi(target_ulong addr, target_ulong val, int asi, int size)
switch(asi) { switch(asi) {
case 0x10: // As if user primary case 0x10: // As if user primary
case 0x11: // As if user secondary
case 0x18: // As if user primary LE case 0x18: // As if user primary LE
case 0x19: // As if user secondary LE
case 0x80: // Primary case 0x80: // Primary
case 0x81: // Secondary
case 0x88: // Primary LE case 0x88: // Primary LE
case 0x89: // Secondary LE
case 0xe2: // UA2007 Primary block init case 0xe2: // UA2007 Primary block init
case 0xe3: // UA2007 Secondary block init case 0xe3: // UA2007 Secondary block init
if ((asi & 0x80) && (env->pstate & PS_PRIV)) { if ((asi & 0x80) && (env->pstate & PS_PRIV)) {
@ -2488,37 +2544,75 @@ void helper_st_asi(target_ulong addr, target_ulong val, int asi, int size)
break; break;
} }
} else { } else {
switch(size) { /* secondary space access has lowest asi bit equal to 1 */
case 1: if (asi & 1) {
stb_kernel(addr, val); switch(size) {
break; case 1:
case 2: stb_kernel_secondary(addr, val);
stw_kernel(addr, val); break;
break; case 2:
case 4: stw_kernel_secondary(addr, val);
stl_kernel(addr, val); break;
break; case 4:
case 8: stl_kernel_secondary(addr, val);
default: break;
stq_kernel(addr, val); case 8:
break; default:
stq_kernel_secondary(addr, val);
break;
}
} else {
switch(size) {
case 1:
stb_kernel(addr, val);
break;
case 2:
stw_kernel(addr, val);
break;
case 4:
stl_kernel(addr, val);
break;
case 8:
default:
stq_kernel(addr, val);
break;
}
} }
} }
} else { } else {
switch(size) { /* secondary space access has lowest asi bit equal to 1 */
case 1: if (asi & 1) {
stb_user(addr, val); switch(size) {
break; case 1:
case 2: stb_user_secondary(addr, val);
stw_user(addr, val); break;
break; case 2:
case 4: stw_user_secondary(addr, val);
stl_user(addr, val); break;
break; case 4:
case 8: stl_user_secondary(addr, val);
default: break;
stq_user(addr, val); case 8:
break; default:
stq_user_secondary(addr, val);
break;
}
} else {
switch(size) {
case 1:
stb_user(addr, val);
break;
case 2:
stw_user(addr, val);
break;
case 4:
stl_user(addr, val);
break;
case 8:
default:
stq_user(addr, val);
break;
}
} }
} }
break; break;
@ -2551,11 +2645,26 @@ void helper_st_asi(target_ulong addr, target_ulong val, int asi, int size)
return; return;
case 0x04: // Nucleus case 0x04: // Nucleus
case 0x0c: // Nucleus Little Endian (LE) case 0x0c: // Nucleus Little Endian (LE)
case 0x11: // As if user secondary {
case 0x19: // As if user secondary LE switch(size) {
case 1:
stb_nucleus(addr, val);
break;
case 2:
stw_nucleus(addr, val);
break;
case 4:
stl_nucleus(addr, val);
break;
default:
case 8:
stq_nucleus(addr, val);
break;
}
break;
}
case 0x4a: // UPA config case 0x4a: // UPA config
case 0x81: // Secondary
case 0x89: // Secondary LE
// XXX // XXX
return; return;
case 0x45: // LSU case 0x45: // LSU