/* * Copyright (c) 2011, Max Filippov, Open Source and Linux Lab. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * Neither the name of the Open Source and Linux Lab nor the * names of its contributors may be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #ifndef CPU_XTENSA_H #define CPU_XTENSA_H #define TARGET_LONG_BITS 32 #define ELF_MACHINE EM_XTENSA #define CPUState struct CPUXtensaState #include "config.h" #include "qemu-common.h" #include "cpu-defs.h" #define TARGET_HAS_ICE 1 #define NB_MMU_MODES 4 #define TARGET_PHYS_ADDR_SPACE_BITS 32 #define TARGET_VIRT_ADDR_SPACE_BITS 32 #define TARGET_PAGE_BITS 12 enum { /* Additional instructions */ XTENSA_OPTION_CODE_DENSITY, XTENSA_OPTION_LOOP, XTENSA_OPTION_EXTENDED_L32R, XTENSA_OPTION_16_BIT_IMUL, XTENSA_OPTION_32_BIT_IMUL, XTENSA_OPTION_32_BIT_IMUL_HIGH, XTENSA_OPTION_32_BIT_IDIV, XTENSA_OPTION_MAC16, XTENSA_OPTION_MISC_OP_NSA, XTENSA_OPTION_MISC_OP_MINMAX, XTENSA_OPTION_MISC_OP_SEXT, XTENSA_OPTION_MISC_OP_CLAMPS, XTENSA_OPTION_COPROCESSOR, XTENSA_OPTION_BOOLEAN, XTENSA_OPTION_FP_COPROCESSOR, XTENSA_OPTION_MP_SYNCHRO, XTENSA_OPTION_CONDITIONAL_STORE, /* Interrupts and exceptions */ XTENSA_OPTION_EXCEPTION, XTENSA_OPTION_RELOCATABLE_VECTOR, XTENSA_OPTION_UNALIGNED_EXCEPTION, XTENSA_OPTION_INTERRUPT, XTENSA_OPTION_HIGH_PRIORITY_INTERRUPT, XTENSA_OPTION_TIMER_INTERRUPT, /* Local memory */ XTENSA_OPTION_ICACHE, XTENSA_OPTION_ICACHE_TEST, XTENSA_OPTION_ICACHE_INDEX_LOCK, XTENSA_OPTION_DCACHE, XTENSA_OPTION_DCACHE_TEST, XTENSA_OPTION_DCACHE_INDEX_LOCK, XTENSA_OPTION_IRAM, XTENSA_OPTION_IROM, XTENSA_OPTION_DRAM, XTENSA_OPTION_DROM, XTENSA_OPTION_XLMI, XTENSA_OPTION_HW_ALIGNMENT, XTENSA_OPTION_MEMORY_ECC_PARITY, /* Memory protection and translation */ XTENSA_OPTION_REGION_PROTECTION, XTENSA_OPTION_REGION_TRANSLATION, XTENSA_OPTION_MMU, /* Other */ XTENSA_OPTION_WINDOWED_REGISTER, XTENSA_OPTION_PROCESSOR_INTERFACE, XTENSA_OPTION_MISC_SR, XTENSA_OPTION_THREAD_POINTER, XTENSA_OPTION_PROCESSOR_ID, XTENSA_OPTION_DEBUG, XTENSA_OPTION_TRACE_PORT, }; enum { THREADPTR = 231, FCR = 232, FSR = 233, }; enum { LBEG = 0, LEND = 1, LCOUNT = 2, SAR = 3, BR = 4, LITBASE = 5, SCOMPARE1 = 12, ACCLO = 16, ACCHI = 17, MR = 32, WINDOW_BASE = 72, WINDOW_START = 73, PTEVADDR = 83, RASID = 90, ITLBCFG = 91, DTLBCFG = 92, EPC1 = 177, DEPC = 192, EPS2 = 194, EXCSAVE1 = 209, CPENABLE = 224, INTSET = 226, INTCLEAR = 227, INTENABLE = 228, PS = 230, VECBASE = 231, EXCCAUSE = 232, CCOUNT = 234, PRID = 235, EXCVADDR = 238, CCOMPARE = 240, }; #define PS_INTLEVEL 0xf #define PS_INTLEVEL_SHIFT 0 #define PS_EXCM 0x10 #define PS_UM 0x20 #define PS_RING 0xc0 #define PS_RING_SHIFT 6 #define PS_OWB 0xf00 #define PS_OWB_SHIFT 8 #define PS_CALLINC 0x30000 #define PS_CALLINC_SHIFT 16 #define PS_CALLINC_LEN 2 #define PS_WOE 0x40000 #define MAX_NAREG 64 #define MAX_NINTERRUPT 32 #define MAX_NLEVEL 6 #define MAX_NNMI 1 #define MAX_NCCOMPARE 3 #define MAX_TLB_WAY_SIZE 8 #define REGION_PAGE_MASK 0xe0000000 enum { /* Static vectors */ EXC_RESET, EXC_MEMORY_ERROR, /* Dynamic vectors */ EXC_WINDOW_OVERFLOW4, EXC_WINDOW_UNDERFLOW4, EXC_WINDOW_OVERFLOW8, EXC_WINDOW_UNDERFLOW8, EXC_WINDOW_OVERFLOW12, EXC_WINDOW_UNDERFLOW12, EXC_IRQ, EXC_KERNEL, EXC_USER, EXC_DOUBLE, EXC_MAX }; enum { ILLEGAL_INSTRUCTION_CAUSE = 0, SYSCALL_CAUSE, INSTRUCTION_FETCH_ERROR_CAUSE, LOAD_STORE_ERROR_CAUSE, LEVEL1_INTERRUPT_CAUSE, ALLOCA_CAUSE, INTEGER_DIVIDE_BY_ZERO_CAUSE, PRIVILEGED_CAUSE = 8, LOAD_STORE_ALIGNMENT_CAUSE, INSTR_PIF_DATA_ERROR_CAUSE = 12, LOAD_STORE_PIF_DATA_ERROR_CAUSE, INSTR_PIF_ADDR_ERROR_CAUSE, LOAD_STORE_PIF_ADDR_ERROR_CAUSE, INST_TLB_MISS_CAUSE, INST_TLB_MULTI_HIT_CAUSE, INST_FETCH_PRIVILEGE_CAUSE, INST_FETCH_PROHIBITED_CAUSE = 20, LOAD_STORE_TLB_MISS_CAUSE = 24, LOAD_STORE_TLB_MULTI_HIT_CAUSE, LOAD_STORE_PRIVILEGE_CAUSE, LOAD_PROHIBITED_CAUSE = 28, STORE_PROHIBITED_CAUSE, COPROCESSOR0_DISABLED = 32, }; typedef enum { INTTYPE_LEVEL, INTTYPE_EDGE, INTTYPE_NMI, INTTYPE_SOFTWARE, INTTYPE_TIMER, INTTYPE_DEBUG, INTTYPE_WRITE_ERR, INTTYPE_MAX } interrupt_type; typedef struct xtensa_tlb_entry { uint32_t vaddr; uint32_t paddr; uint8_t asid; uint8_t attr; bool variable; } xtensa_tlb_entry; typedef struct xtensa_tlb { unsigned nways; const unsigned way_size[10]; bool varway56; unsigned nrefillentries; } xtensa_tlb; typedef struct XtensaGdbReg { int targno; int type; int group; } XtensaGdbReg; typedef struct XtensaGdbRegmap { int num_regs; int num_core_regs; /* PC + a + ar + sr + ur */ XtensaGdbReg reg[1 + 16 + 64 + 256 + 256]; } XtensaGdbRegmap; typedef struct XtensaConfig { const char *name; uint64_t options; XtensaGdbRegmap gdb_regmap; unsigned nareg; int excm_level; int ndepc; uint32_t vecbase; uint32_t exception_vector[EXC_MAX]; unsigned ninterrupt; unsigned nlevel; uint32_t interrupt_vector[MAX_NLEVEL + MAX_NNMI + 1]; uint32_t level_mask[MAX_NLEVEL + MAX_NNMI + 1]; uint32_t inttype_mask[INTTYPE_MAX]; struct { uint32_t level; interrupt_type inttype; } interrupt[MAX_NINTERRUPT]; unsigned nccompare; uint32_t timerint[MAX_NCCOMPARE]; unsigned nextint; unsigned extint[MAX_NINTERRUPT]; uint32_t clock_freq_khz; xtensa_tlb itlb; xtensa_tlb dtlb; } XtensaConfig; typedef struct CPUXtensaState { const XtensaConfig *config; uint32_t regs[16]; uint32_t pc; uint32_t sregs[256]; uint32_t uregs[256]; uint32_t phys_regs[MAX_NAREG]; xtensa_tlb_entry itlb[7][MAX_TLB_WAY_SIZE]; xtensa_tlb_entry dtlb[10][MAX_TLB_WAY_SIZE]; unsigned autorefill_idx; int pending_irq_level; /* level of last raised IRQ */ void **irq_inputs; QEMUTimer *ccompare_timer; uint32_t wake_ccount; int64_t halt_clock; int exception_taken; CPU_COMMON } CPUXtensaState; #define cpu_init cpu_xtensa_init #define cpu_exec cpu_xtensa_exec #define cpu_gen_code cpu_xtensa_gen_code #define cpu_signal_handler cpu_xtensa_signal_handler #define cpu_list xtensa_cpu_list CPUXtensaState *cpu_xtensa_init(const char *cpu_model); void xtensa_translate_init(void); int cpu_xtensa_exec(CPUXtensaState *s); void do_interrupt(CPUXtensaState *s); void check_interrupts(CPUXtensaState *s); void xtensa_irq_init(CPUState *env); void *xtensa_get_extint(CPUState *env, unsigned extint); void xtensa_advance_ccount(CPUState *env, uint32_t d); void xtensa_timer_irq(CPUState *env, uint32_t id, uint32_t active); void xtensa_rearm_ccompare_timer(CPUState *env); int cpu_xtensa_signal_handler(int host_signum, void *pinfo, void *puc); void xtensa_cpu_list(FILE *f, fprintf_function cpu_fprintf); void xtensa_sync_window_from_phys(CPUState *env); void xtensa_sync_phys_from_window(CPUState *env); uint32_t xtensa_tlb_get_addr_mask(const CPUState *env, bool dtlb, uint32_t way); void split_tlb_entry_spec_way(const CPUState *env, uint32_t v, bool dtlb, uint32_t *vpn, uint32_t wi, uint32_t *ei); int xtensa_tlb_lookup(const CPUState *env, uint32_t addr, bool dtlb, uint32_t *pwi, uint32_t *pei, uint8_t *pring); void xtensa_tlb_set_entry(CPUState *env, bool dtlb, unsigned wi, unsigned ei, uint32_t vpn, uint32_t pte); int xtensa_get_physical_addr(CPUState *env, uint32_t vaddr, int is_write, int mmu_idx, uint32_t *paddr, uint32_t *page_size, unsigned *access); #define XTENSA_OPTION_BIT(opt) (((uint64_t)1) << (opt)) static inline bool xtensa_option_bits_enabled(const XtensaConfig *config, uint64_t opt) { return (config->options & opt) != 0; } static inline bool xtensa_option_enabled(const XtensaConfig *config, int opt) { return xtensa_option_bits_enabled(config, XTENSA_OPTION_BIT(opt)); } static inline int xtensa_get_cintlevel(const CPUState *env) { int level = (env->sregs[PS] & PS_INTLEVEL) >> PS_INTLEVEL_SHIFT; if ((env->sregs[PS] & PS_EXCM) && env->config->excm_level > level) { level = env->config->excm_level; } return level; } static inline int xtensa_get_ring(const CPUState *env) { if (xtensa_option_enabled(env->config, XTENSA_OPTION_MMU)) { return (env->sregs[PS] & PS_RING) >> PS_RING_SHIFT; } else { return 0; } } static inline int xtensa_get_cring(const CPUState *env) { if (xtensa_option_enabled(env->config, XTENSA_OPTION_MMU) && (env->sregs[PS] & PS_EXCM) == 0) { return (env->sregs[PS] & PS_RING) >> PS_RING_SHIFT; } else { return 0; } } static inline xtensa_tlb_entry *xtensa_tlb_get_entry(CPUState *env, bool dtlb, unsigned wi, unsigned ei) { return dtlb ? env->dtlb[wi] + ei : env->itlb[wi] + ei; } /* MMU modes definitions */ #define MMU_MODE0_SUFFIX _ring0 #define MMU_MODE1_SUFFIX _ring1 #define MMU_MODE2_SUFFIX _ring2 #define MMU_MODE3_SUFFIX _ring3 static inline int cpu_mmu_index(CPUState *env) { return xtensa_get_cring(env); } #define XTENSA_TBFLAG_RING_MASK 0x3 #define XTENSA_TBFLAG_EXCM 0x4 #define XTENSA_TBFLAG_LITBASE 0x8 static inline void cpu_get_tb_cpu_state(CPUState *env, target_ulong *pc, target_ulong *cs_base, int *flags) { *pc = env->pc; *cs_base = 0; *flags = 0; *flags |= xtensa_get_ring(env); if (env->sregs[PS] & PS_EXCM) { *flags |= XTENSA_TBFLAG_EXCM; } if (xtensa_option_enabled(env->config, XTENSA_OPTION_EXTENDED_L32R) && (env->sregs[LITBASE] & 1)) { *flags |= XTENSA_TBFLAG_LITBASE; } } #include "cpu-all.h" #include "exec-all.h" static inline int cpu_has_work(CPUState *env) { return env->pending_irq_level; } static inline void cpu_pc_from_tb(CPUState *env, TranslationBlock *tb) { env->pc = tb->pc; } #endif