⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 op_helper.c

📁 qemu虚拟机代码
💻 C
📖 第 1 页 / 共 2 页
字号:
        val = T0;        old = env->CP0_Compare;        cpu_mips_store_compare(env, val);        rn = "Compare";        break;    case 12:        val = T0 & 0xFA78FF01;        if (T0 & (1 << CP0St_UM))            env->hflags |= MIPS_HFLAG_UM;        else            env->hflags &= ~MIPS_HFLAG_UM;        if (T0 & (1 << CP0St_ERL))            env->hflags |= MIPS_HFLAG_ERL;        else            env->hflags &= ~MIPS_HFLAG_ERL;        if (T0 & (1 << CP0St_EXL))            env->hflags |= MIPS_HFLAG_EXL;        else            env->hflags &= ~MIPS_HFLAG_EXL;        old = env->CP0_Status;        env->CP0_Status = val;        /* If we unmasked an asserted IRQ, raise it */        mask = 0x0000FF00;        if (loglevel & CPU_LOG_TB_IN_ASM) {            fprintf(logfile, "Status %08x => %08x Cause %08x (%08x %08x %08x)\n",                    old, val, env->CP0_Cause, old & mask, val & mask,                    env->CP0_Cause & mask);        }#if 1        if ((val & (1 << CP0St_IE)) && !(old & (1 << CP0St_IE)) &&            !(env->hflags & MIPS_HFLAG_EXL) &&            !(env->hflags & MIPS_HFLAG_ERL) &&            !(env->hflags & MIPS_HFLAG_DM) &&             (env->CP0_Status & env->CP0_Cause & mask)) {            if (logfile)                fprintf(logfile, "Raise pending IRQs\n");            env->interrupt_request |= CPU_INTERRUPT_HARD;            do_raise_exception(EXCP_EXT_INTERRUPT);        } else if (!(val & 0x00000001) && (old & 0x00000001)) {            env->interrupt_request &= ~CPU_INTERRUPT_HARD;        }#endif        rn = "Status";        break;    case 13:        val = (env->CP0_Cause & 0xB000F87C) | (T0 & 0x000C00300);        old = env->CP0_Cause;        env->CP0_Cause = val;#if 0        {            int i;            /* Check if we ever asserted a software IRQ */            for (i = 0; i < 2; i++) {                mask = 0x100 << i;                if ((val & mask) & !(old & mask))                    mips_set_irq(i);            }        }#endif        rn = "Cause";        break;    case 14:        val = T0;        old = env->CP0_EPC;        env->CP0_EPC = val;        rn = "EPC";        break;    case 16:        switch (sel) {        case 0:#if defined(MIPS_USES_R4K_TLB)            val = (env->CP0_Config0 & 0x8017FF80) | (T0 & 0x7E000001);#else            val = (env->CP0_Config0 & 0xFE17FF80) | (T0 & 0x00000001);#endif            old = env->CP0_Config0;            env->CP0_Config0 = val;            rn = "Config0";            break;        default:            val = -1;            old = -1;            rn = "bad config selector";            break;        }        break;    case 18:        val = T0;        old = env->CP0_WatchLo;        env->CP0_WatchLo = val;        rn = "WatchLo";        break;    case 19:        val = T0 & 0x40FF0FF8;        old = env->CP0_WatchHi;        env->CP0_WatchHi = val;        rn = "WatchHi";        break;    case 23:        val = (env->CP0_Debug & 0x8C03FC1F) | (T0 & 0x13300120);        if (T0 & (1 << CP0DB_DM))            env->hflags |= MIPS_HFLAG_DM;        else            env->hflags &= ~MIPS_HFLAG_DM;        old = env->CP0_Debug;        env->CP0_Debug = val;        rn = "Debug";        break;    case 24:        val = T0;        old = env->CP0_DEPC;        env->CP0_DEPC = val;        rn = "DEPC";        break;    case 28:        switch (sel) {        case 0:            val = T0 & 0xFFFFFCF6;            old = env->CP0_TagLo;            env->CP0_TagLo = val;            rn = "TagLo";            break;        default:            val = -1;            old = -1;            rn = "invalid sel";            break;        }        break;    case 30:        val = T0;        old = env->CP0_ErrorEPC;        env->CP0_ErrorEPC = val;        rn = "EPC";        break;    case 31:        val = T0;        old = env->CP0_DESAVE;        env->CP0_DESAVE = val;        rn = "DESAVE";        break;    default:        val = -1;        old = -1;        rn = "unknown";        break;    } print:#if defined MIPS_DEBUG_DISAS    if (loglevel & CPU_LOG_TB_IN_ASM) {        fprintf(logfile, "%08x mtc0 %s %08x => %08x (%d %d %08x)\n",                env->PC, rn, T0, val, reg, sel, old);    }#endif    return;}/* TLB management */#if defined(MIPS_USES_R4K_TLB)static void invalidate_tlb (int idx){    tlb_t *tlb;    target_ulong addr;    tlb = &env->tlb[idx];    if (tlb->V0) {        tb_invalidate_page_range(tlb->PFN[0], tlb->end - tlb->VPN);        addr = tlb->VPN;        while (addr < tlb->end) {            tlb_flush_page (env, addr);            addr += TARGET_PAGE_SIZE;        }    }    if (tlb->V1) {        tb_invalidate_page_range(tlb->PFN[1], tlb->end2 - tlb->end);        addr = tlb->end;        while (addr < tlb->end2) {            tlb_flush_page (env, addr);            addr += TARGET_PAGE_SIZE;        }    }}static void fill_tlb (int idx){    tlb_t *tlb;    int size;    /* XXX: detect conflicting TLBs and raise a MCHECK exception when needed */    tlb = &env->tlb[idx];    tlb->VPN = env->CP0_EntryHi & 0xFFFFE000;    tlb->ASID = env->CP0_EntryHi & 0xFF;    size = env->CP0_PageMask >> 13;    size = 4 * (size + 1);    tlb->end = tlb->VPN + (1 << (8 + size));    tlb->end2 = tlb->end + (1 << (8 + size));    tlb->G = env->CP0_EntryLo0 & env->CP0_EntryLo1 & 1;    tlb->V0 = (env->CP0_EntryLo0 & 2) != 0;    tlb->D0 = (env->CP0_EntryLo0 & 4) != 0;    tlb->C0 = (env->CP0_EntryLo0 >> 3) & 0x7;    tlb->PFN[0] = (env->CP0_EntryLo0 >> 6) << 12;    tlb->V1 = (env->CP0_EntryLo1 & 2) != 0;    tlb->D1 = (env->CP0_EntryLo1 & 4) != 0;    tlb->C1 = (env->CP0_EntryLo1 >> 3) & 0x7;    tlb->PFN[1] = (env->CP0_EntryLo1 >> 6) << 12;}void do_tlbwi (void){    /* Wildly undefined effects for CP0_index containing a too high value and       MIPS_TLB_NB not being a power of two.  But so does real silicon.  */    invalidate_tlb(env->CP0_index & (MIPS_TLB_NB - 1));    fill_tlb(env->CP0_index & (MIPS_TLB_NB - 1));}void do_tlbwr (void){    int r = cpu_mips_get_random(env);    invalidate_tlb(r);    fill_tlb(r);}void do_tlbp (void){    tlb_t *tlb;    target_ulong tag;    uint8_t ASID;    int i;    tag = (env->CP0_EntryHi & 0xFFFFE000);    ASID = env->CP0_EntryHi & 0x000000FF;        for (i = 0; i < MIPS_TLB_NB; i++) {        tlb = &env->tlb[i];        /* Check ASID, virtual page number & size */        if ((tlb->G == 1 || tlb->ASID == ASID) && tlb->VPN == tag) {            /* TLB match */            env->CP0_index = i;            break;        }    }    if (i == MIPS_TLB_NB) {        env->CP0_index |= 0x80000000;    }}void do_tlbr (void){    tlb_t *tlb;    uint8_t ASID;    int size;    ASID = env->CP0_EntryHi & 0xFF;    tlb = &env->tlb[env->CP0_index & (MIPS_TLB_NB - 1)];    /* If this will change the current ASID, flush qemu's TLB.  */    if (ASID != tlb->ASID && tlb->G != 1)      tlb_flush (env, 1);    env->CP0_EntryHi = tlb->VPN | tlb->ASID;    size = (tlb->end - tlb->VPN) >> 12;    env->CP0_PageMask = (size - 1) << 13;    env->CP0_EntryLo0 = tlb->G | (tlb->V0 << 1) | (tlb->D0 << 2)		| (tlb->C0 << 3) | (tlb->PFN[0] >> 6);    env->CP0_EntryLo1 = tlb->G | (tlb->V1 << 1) | (tlb->D1 << 2)		| (tlb->C1 << 3) | (tlb->PFN[1] >> 6);}#endif#endif /* !CONFIG_USER_ONLY */void op_dump_ldst (const unsigned char *func){    if (loglevel)        fprintf(logfile, "%s => %08x %08x\n", __func__, T0, T1);}void dump_sc (void){    if (loglevel) {        fprintf(logfile, "%s %08x at %08x (%08x)\n", __func__,                T1, T0, env->CP0_LLAddr);    }}void debug_eret (void){    if (loglevel) {        fprintf(logfile, "ERET: pc %08x EPC %08x ErrorEPC %08x (%d)\n",                env->PC, env->CP0_EPC, env->CP0_ErrorEPC,                env->hflags & MIPS_HFLAG_ERL ? 1 : 0);    }}void do_pmon (int function){    function /= 2;    switch (function) {    case 2: /* TODO: char inbyte(int waitflag); */        if (env->gpr[4] == 0)            env->gpr[2] = -1;        /* Fall through */    case 11: /* TODO: char inbyte (void); */        env->gpr[2] = -1;        break;    case 3:    case 12:        printf("%c", env->gpr[4] & 0xFF);        break;    case 17:        break;    case 158:        {            unsigned char *fmt = (void *)env->gpr[4];            printf("%s", fmt);        }        break;    }}#if !defined(CONFIG_USER_ONLY) static void do_unaligned_access (target_ulong addr, int is_write, int is_user, void *retaddr);#define MMUSUFFIX _mmu#define ALIGNED_ONLY#define SHIFT 0#include "softmmu_template.h"#define SHIFT 1#include "softmmu_template.h"#define SHIFT 2#include "softmmu_template.h"#define SHIFT 3#include "softmmu_template.h"static void do_unaligned_access (target_ulong addr, int is_write, int is_user, void *retaddr){    env->CP0_BadVAddr = addr;    do_restore_state (retaddr);    do_raise_exception ((is_write == 1) ? EXCP_AdES : EXCP_AdEL);}void tlb_fill (target_ulong addr, int is_write, int is_user, void *retaddr){    TranslationBlock *tb;    CPUState *saved_env;    unsigned long pc;    int ret;    /* XXX: hack to restore env in all cases, even if not called from       generated code */    saved_env = env;    env = cpu_single_env;    ret = cpu_mips_handle_mmu_fault(env, addr, is_write, is_user, 1);    if (ret) {        if (retaddr) {            /* now we have a real cpu fault */            pc = (unsigned long)retaddr;            tb = tb_find_pc(pc);            if (tb) {                /* the PC is inside the translated code. It means that we have                   a virtual CPU fault */                cpu_restore_state(tb, env, pc, NULL);            }        }        do_raise_exception_err(env->exception_index, env->error_code);    }    env = saved_env;}#endif

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -