15118ebe8SLucas Mateus Castro (alqotel) /* 25118ebe8SLucas Mateus Castro (alqotel) * PowerPC MMU, TLB, SLB and BAT emulation helpers for QEMU. 35118ebe8SLucas Mateus Castro (alqotel) * 45118ebe8SLucas Mateus Castro (alqotel) * Copyright (c) 2003-2007 Jocelyn Mayer 55118ebe8SLucas Mateus Castro (alqotel) * 65118ebe8SLucas Mateus Castro (alqotel) * This library is free software; you can redistribute it and/or 75118ebe8SLucas Mateus Castro (alqotel) * modify it under the terms of the GNU Lesser General Public 85118ebe8SLucas Mateus Castro (alqotel) * License as published by the Free Software Foundation; either 95118ebe8SLucas Mateus Castro (alqotel) * version 2.1 of the License, or (at your option) any later version. 105118ebe8SLucas Mateus Castro (alqotel) * 115118ebe8SLucas Mateus Castro (alqotel) * This library is distributed in the hope that it will be useful, 125118ebe8SLucas Mateus Castro (alqotel) * but WITHOUT ANY WARRANTY; without even the implied warranty of 135118ebe8SLucas Mateus Castro (alqotel) * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 145118ebe8SLucas Mateus Castro (alqotel) * Lesser General Public License for more details. 155118ebe8SLucas Mateus Castro (alqotel) * 165118ebe8SLucas Mateus Castro (alqotel) * You should have received a copy of the GNU Lesser General Public 175118ebe8SLucas Mateus Castro (alqotel) * License along with this library; if not, see <http://www.gnu.org/licenses/>. 185118ebe8SLucas Mateus Castro (alqotel) */ 195118ebe8SLucas Mateus Castro (alqotel) 205118ebe8SLucas Mateus Castro (alqotel) #include "qemu/osdep.h" 215118ebe8SLucas Mateus Castro (alqotel) #include "qemu/units.h" 225118ebe8SLucas Mateus Castro (alqotel) #include "cpu.h" 235118ebe8SLucas Mateus Castro (alqotel) #include "sysemu/kvm.h" 245118ebe8SLucas Mateus Castro (alqotel) #include "kvm_ppc.h" 255118ebe8SLucas Mateus Castro (alqotel) #include "mmu-hash64.h" 265118ebe8SLucas Mateus Castro (alqotel) #include "mmu-hash32.h" 275118ebe8SLucas Mateus Castro (alqotel) #include "exec/exec-all.h" 285118ebe8SLucas Mateus Castro (alqotel) #include "exec/log.h" 295118ebe8SLucas Mateus Castro (alqotel) #include "helper_regs.h" 305118ebe8SLucas Mateus Castro (alqotel) #include "qemu/error-report.h" 315118ebe8SLucas Mateus Castro (alqotel) #include "qemu/main-loop.h" 325118ebe8SLucas Mateus Castro (alqotel) #include "qemu/qemu-print.h" 335118ebe8SLucas Mateus Castro (alqotel) #include "internal.h" 345118ebe8SLucas Mateus Castro (alqotel) #include "mmu-book3s-v3.h" 355118ebe8SLucas Mateus Castro (alqotel) #include "mmu-radix64.h" 365118ebe8SLucas Mateus Castro (alqotel) 375118ebe8SLucas Mateus Castro (alqotel) /* #define DEBUG_MMU */ 385118ebe8SLucas Mateus Castro (alqotel) /* #define DEBUG_BATS */ 395118ebe8SLucas Mateus Castro (alqotel) /* #define DEBUG_SOFTWARE_TLB */ 405118ebe8SLucas Mateus Castro (alqotel) /* #define DUMP_PAGE_TABLES */ 415118ebe8SLucas Mateus Castro (alqotel) /* #define FLUSH_ALL_TLBS */ 425118ebe8SLucas Mateus Castro (alqotel) 435118ebe8SLucas Mateus Castro (alqotel) #ifdef DEBUG_MMU 445118ebe8SLucas Mateus Castro (alqotel) # define LOG_MMU_STATE(cpu) log_cpu_state_mask(CPU_LOG_MMU, (cpu), 0) 455118ebe8SLucas Mateus Castro (alqotel) #else 465118ebe8SLucas Mateus Castro (alqotel) # define LOG_MMU_STATE(cpu) do { } while (0) 475118ebe8SLucas Mateus Castro (alqotel) #endif 485118ebe8SLucas Mateus Castro (alqotel) 495118ebe8SLucas Mateus Castro (alqotel) #ifdef DEBUG_SOFTWARE_TLB 505118ebe8SLucas Mateus Castro (alqotel) # define LOG_SWTLB(...) qemu_log_mask(CPU_LOG_MMU, __VA_ARGS__) 515118ebe8SLucas Mateus Castro (alqotel) #else 525118ebe8SLucas Mateus Castro (alqotel) # define LOG_SWTLB(...) do { } while (0) 535118ebe8SLucas Mateus Castro (alqotel) #endif 545118ebe8SLucas Mateus Castro (alqotel) 555118ebe8SLucas Mateus Castro (alqotel) #ifdef DEBUG_BATS 565118ebe8SLucas Mateus Castro (alqotel) # define LOG_BATS(...) qemu_log_mask(CPU_LOG_MMU, __VA_ARGS__) 575118ebe8SLucas Mateus Castro (alqotel) #else 585118ebe8SLucas Mateus Castro (alqotel) # define LOG_BATS(...) do { } while (0) 595118ebe8SLucas Mateus Castro (alqotel) #endif 605118ebe8SLucas Mateus Castro (alqotel) 61d6ae8ec6SLucas Mateus Castro (alqotel) void ppc_store_sdr1(CPUPPCState *env, target_ulong value) 62d6ae8ec6SLucas Mateus Castro (alqotel) { 63d6ae8ec6SLucas Mateus Castro (alqotel) PowerPCCPU *cpu = env_archcpu(env); 64d6ae8ec6SLucas Mateus Castro (alqotel) qemu_log_mask(CPU_LOG_MMU, "%s: " TARGET_FMT_lx "\n", __func__, value); 65d6ae8ec6SLucas Mateus Castro (alqotel) assert(!cpu->env.has_hv_mode || !cpu->vhyp); 66d6ae8ec6SLucas Mateus Castro (alqotel) #if defined(TARGET_PPC64) 67d6ae8ec6SLucas Mateus Castro (alqotel) if (mmu_is_64bit(env->mmu_model)) { 68d6ae8ec6SLucas Mateus Castro (alqotel) target_ulong sdr_mask = SDR_64_HTABORG | SDR_64_HTABSIZE; 69d6ae8ec6SLucas Mateus Castro (alqotel) target_ulong htabsize = value & SDR_64_HTABSIZE; 70d6ae8ec6SLucas Mateus Castro (alqotel) 71d6ae8ec6SLucas Mateus Castro (alqotel) if (value & ~sdr_mask) { 72d6ae8ec6SLucas Mateus Castro (alqotel) qemu_log_mask(LOG_GUEST_ERROR, "Invalid bits 0x"TARGET_FMT_lx 73d6ae8ec6SLucas Mateus Castro (alqotel) " set in SDR1", value & ~sdr_mask); 74d6ae8ec6SLucas Mateus Castro (alqotel) value &= sdr_mask; 75d6ae8ec6SLucas Mateus Castro (alqotel) } 76d6ae8ec6SLucas Mateus Castro (alqotel) if (htabsize > 28) { 77d6ae8ec6SLucas Mateus Castro (alqotel) qemu_log_mask(LOG_GUEST_ERROR, "Invalid HTABSIZE 0x" TARGET_FMT_lx 78d6ae8ec6SLucas Mateus Castro (alqotel) " stored in SDR1", htabsize); 79d6ae8ec6SLucas Mateus Castro (alqotel) return; 80d6ae8ec6SLucas Mateus Castro (alqotel) } 81d6ae8ec6SLucas Mateus Castro (alqotel) } 82d6ae8ec6SLucas Mateus Castro (alqotel) #endif /* defined(TARGET_PPC64) */ 83d6ae8ec6SLucas Mateus Castro (alqotel) /* FIXME: Should check for valid HTABMASK values in 32-bit case */ 84d6ae8ec6SLucas Mateus Castro (alqotel) env->spr[SPR_SDR1] = value; 85d6ae8ec6SLucas Mateus Castro (alqotel) } 86d6ae8ec6SLucas Mateus Castro (alqotel) 875118ebe8SLucas Mateus Castro (alqotel) /*****************************************************************************/ 885118ebe8SLucas Mateus Castro (alqotel) /* PowerPC MMU emulation */ 895118ebe8SLucas Mateus Castro (alqotel) 905118ebe8SLucas Mateus Castro (alqotel) static int pp_check(int key, int pp, int nx) 915118ebe8SLucas Mateus Castro (alqotel) { 925118ebe8SLucas Mateus Castro (alqotel) int access; 935118ebe8SLucas Mateus Castro (alqotel) 945118ebe8SLucas Mateus Castro (alqotel) /* Compute access rights */ 955118ebe8SLucas Mateus Castro (alqotel) access = 0; 965118ebe8SLucas Mateus Castro (alqotel) if (key == 0) { 975118ebe8SLucas Mateus Castro (alqotel) switch (pp) { 985118ebe8SLucas Mateus Castro (alqotel) case 0x0: 995118ebe8SLucas Mateus Castro (alqotel) case 0x1: 1005118ebe8SLucas Mateus Castro (alqotel) case 0x2: 1015118ebe8SLucas Mateus Castro (alqotel) access |= PAGE_WRITE; 1025118ebe8SLucas Mateus Castro (alqotel) /* fall through */ 1035118ebe8SLucas Mateus Castro (alqotel) case 0x3: 1045118ebe8SLucas Mateus Castro (alqotel) access |= PAGE_READ; 1055118ebe8SLucas Mateus Castro (alqotel) break; 1065118ebe8SLucas Mateus Castro (alqotel) } 1075118ebe8SLucas Mateus Castro (alqotel) } else { 1085118ebe8SLucas Mateus Castro (alqotel) switch (pp) { 1095118ebe8SLucas Mateus Castro (alqotel) case 0x0: 1105118ebe8SLucas Mateus Castro (alqotel) access = 0; 1115118ebe8SLucas Mateus Castro (alqotel) break; 1125118ebe8SLucas Mateus Castro (alqotel) case 0x1: 1135118ebe8SLucas Mateus Castro (alqotel) case 0x3: 1145118ebe8SLucas Mateus Castro (alqotel) access = PAGE_READ; 1155118ebe8SLucas Mateus Castro (alqotel) break; 1165118ebe8SLucas Mateus Castro (alqotel) case 0x2: 1175118ebe8SLucas Mateus Castro (alqotel) access = PAGE_READ | PAGE_WRITE; 1185118ebe8SLucas Mateus Castro (alqotel) break; 1195118ebe8SLucas Mateus Castro (alqotel) } 1205118ebe8SLucas Mateus Castro (alqotel) } 1215118ebe8SLucas Mateus Castro (alqotel) if (nx == 0) { 1225118ebe8SLucas Mateus Castro (alqotel) access |= PAGE_EXEC; 1235118ebe8SLucas Mateus Castro (alqotel) } 1245118ebe8SLucas Mateus Castro (alqotel) 1255118ebe8SLucas Mateus Castro (alqotel) return access; 1265118ebe8SLucas Mateus Castro (alqotel) } 1275118ebe8SLucas Mateus Castro (alqotel) 1285118ebe8SLucas Mateus Castro (alqotel) static int check_prot(int prot, MMUAccessType access_type) 1295118ebe8SLucas Mateus Castro (alqotel) { 1305118ebe8SLucas Mateus Castro (alqotel) return prot & prot_for_access_type(access_type) ? 0 : -2; 1315118ebe8SLucas Mateus Castro (alqotel) } 1325118ebe8SLucas Mateus Castro (alqotel) 1335118ebe8SLucas Mateus Castro (alqotel) int ppc6xx_tlb_getnum(CPUPPCState *env, target_ulong eaddr, 1345118ebe8SLucas Mateus Castro (alqotel) int way, int is_code) 1355118ebe8SLucas Mateus Castro (alqotel) { 1365118ebe8SLucas Mateus Castro (alqotel) int nr; 1375118ebe8SLucas Mateus Castro (alqotel) 1385118ebe8SLucas Mateus Castro (alqotel) /* Select TLB num in a way from address */ 1395118ebe8SLucas Mateus Castro (alqotel) nr = (eaddr >> TARGET_PAGE_BITS) & (env->tlb_per_way - 1); 1405118ebe8SLucas Mateus Castro (alqotel) /* Select TLB way */ 1415118ebe8SLucas Mateus Castro (alqotel) nr += env->tlb_per_way * way; 1425118ebe8SLucas Mateus Castro (alqotel) /* 6xx have separate TLBs for instructions and data */ 1435118ebe8SLucas Mateus Castro (alqotel) if (is_code && env->id_tlbs == 1) { 1445118ebe8SLucas Mateus Castro (alqotel) nr += env->nb_tlb; 1455118ebe8SLucas Mateus Castro (alqotel) } 1465118ebe8SLucas Mateus Castro (alqotel) 1475118ebe8SLucas Mateus Castro (alqotel) return nr; 1485118ebe8SLucas Mateus Castro (alqotel) } 1495118ebe8SLucas Mateus Castro (alqotel) 1505118ebe8SLucas Mateus Castro (alqotel) static int ppc6xx_tlb_pte_check(mmu_ctx_t *ctx, target_ulong pte0, 1515118ebe8SLucas Mateus Castro (alqotel) target_ulong pte1, int h, 1525118ebe8SLucas Mateus Castro (alqotel) MMUAccessType access_type) 1535118ebe8SLucas Mateus Castro (alqotel) { 1545118ebe8SLucas Mateus Castro (alqotel) target_ulong ptem, mmask; 1555118ebe8SLucas Mateus Castro (alqotel) int access, ret, pteh, ptev, pp; 1565118ebe8SLucas Mateus Castro (alqotel) 1575118ebe8SLucas Mateus Castro (alqotel) ret = -1; 1585118ebe8SLucas Mateus Castro (alqotel) /* Check validity and table match */ 1595118ebe8SLucas Mateus Castro (alqotel) ptev = pte_is_valid(pte0); 1605118ebe8SLucas Mateus Castro (alqotel) pteh = (pte0 >> 6) & 1; 1615118ebe8SLucas Mateus Castro (alqotel) if (ptev && h == pteh) { 1625118ebe8SLucas Mateus Castro (alqotel) /* Check vsid & api */ 1635118ebe8SLucas Mateus Castro (alqotel) ptem = pte0 & PTE_PTEM_MASK; 1645118ebe8SLucas Mateus Castro (alqotel) mmask = PTE_CHECK_MASK; 1655118ebe8SLucas Mateus Castro (alqotel) pp = pte1 & 0x00000003; 1665118ebe8SLucas Mateus Castro (alqotel) if (ptem == ctx->ptem) { 1675118ebe8SLucas Mateus Castro (alqotel) if (ctx->raddr != (hwaddr)-1ULL) { 1685118ebe8SLucas Mateus Castro (alqotel) /* all matches should have equal RPN, WIMG & PP */ 1695118ebe8SLucas Mateus Castro (alqotel) if ((ctx->raddr & mmask) != (pte1 & mmask)) { 1705118ebe8SLucas Mateus Castro (alqotel) qemu_log_mask(CPU_LOG_MMU, "Bad RPN/WIMG/PP\n"); 1715118ebe8SLucas Mateus Castro (alqotel) return -3; 1725118ebe8SLucas Mateus Castro (alqotel) } 1735118ebe8SLucas Mateus Castro (alqotel) } 1745118ebe8SLucas Mateus Castro (alqotel) /* Compute access rights */ 1755118ebe8SLucas Mateus Castro (alqotel) access = pp_check(ctx->key, pp, ctx->nx); 1765118ebe8SLucas Mateus Castro (alqotel) /* Keep the matching PTE information */ 1775118ebe8SLucas Mateus Castro (alqotel) ctx->raddr = pte1; 1785118ebe8SLucas Mateus Castro (alqotel) ctx->prot = access; 1795118ebe8SLucas Mateus Castro (alqotel) ret = check_prot(ctx->prot, access_type); 1805118ebe8SLucas Mateus Castro (alqotel) if (ret == 0) { 1815118ebe8SLucas Mateus Castro (alqotel) /* Access granted */ 1825118ebe8SLucas Mateus Castro (alqotel) qemu_log_mask(CPU_LOG_MMU, "PTE access granted !\n"); 1835118ebe8SLucas Mateus Castro (alqotel) } else { 1845118ebe8SLucas Mateus Castro (alqotel) /* Access right violation */ 1855118ebe8SLucas Mateus Castro (alqotel) qemu_log_mask(CPU_LOG_MMU, "PTE access rejected\n"); 1865118ebe8SLucas Mateus Castro (alqotel) } 1875118ebe8SLucas Mateus Castro (alqotel) } 1885118ebe8SLucas Mateus Castro (alqotel) } 1895118ebe8SLucas Mateus Castro (alqotel) 1905118ebe8SLucas Mateus Castro (alqotel) return ret; 1915118ebe8SLucas Mateus Castro (alqotel) } 1925118ebe8SLucas Mateus Castro (alqotel) 1935118ebe8SLucas Mateus Castro (alqotel) static int pte_update_flags(mmu_ctx_t *ctx, target_ulong *pte1p, 1945118ebe8SLucas Mateus Castro (alqotel) int ret, MMUAccessType access_type) 1955118ebe8SLucas Mateus Castro (alqotel) { 1965118ebe8SLucas Mateus Castro (alqotel) int store = 0; 1975118ebe8SLucas Mateus Castro (alqotel) 1985118ebe8SLucas Mateus Castro (alqotel) /* Update page flags */ 1995118ebe8SLucas Mateus Castro (alqotel) if (!(*pte1p & 0x00000100)) { 2005118ebe8SLucas Mateus Castro (alqotel) /* Update accessed flag */ 2015118ebe8SLucas Mateus Castro (alqotel) *pte1p |= 0x00000100; 2025118ebe8SLucas Mateus Castro (alqotel) store = 1; 2035118ebe8SLucas Mateus Castro (alqotel) } 2045118ebe8SLucas Mateus Castro (alqotel) if (!(*pte1p & 0x00000080)) { 2055118ebe8SLucas Mateus Castro (alqotel) if (access_type == MMU_DATA_STORE && ret == 0) { 2065118ebe8SLucas Mateus Castro (alqotel) /* Update changed flag */ 2075118ebe8SLucas Mateus Castro (alqotel) *pte1p |= 0x00000080; 2085118ebe8SLucas Mateus Castro (alqotel) store = 1; 2095118ebe8SLucas Mateus Castro (alqotel) } else { 2105118ebe8SLucas Mateus Castro (alqotel) /* Force page fault for first write access */ 2115118ebe8SLucas Mateus Castro (alqotel) ctx->prot &= ~PAGE_WRITE; 2125118ebe8SLucas Mateus Castro (alqotel) } 2135118ebe8SLucas Mateus Castro (alqotel) } 2145118ebe8SLucas Mateus Castro (alqotel) 2155118ebe8SLucas Mateus Castro (alqotel) return store; 2165118ebe8SLucas Mateus Castro (alqotel) } 2175118ebe8SLucas Mateus Castro (alqotel) 2185118ebe8SLucas Mateus Castro (alqotel) /* Software driven TLB helpers */ 2195118ebe8SLucas Mateus Castro (alqotel) 2205118ebe8SLucas Mateus Castro (alqotel) static int ppc6xx_tlb_check(CPUPPCState *env, mmu_ctx_t *ctx, 2215118ebe8SLucas Mateus Castro (alqotel) target_ulong eaddr, MMUAccessType access_type) 2225118ebe8SLucas Mateus Castro (alqotel) { 2235118ebe8SLucas Mateus Castro (alqotel) ppc6xx_tlb_t *tlb; 2245118ebe8SLucas Mateus Castro (alqotel) int nr, best, way; 2255118ebe8SLucas Mateus Castro (alqotel) int ret; 2265118ebe8SLucas Mateus Castro (alqotel) 2275118ebe8SLucas Mateus Castro (alqotel) best = -1; 2285118ebe8SLucas Mateus Castro (alqotel) ret = -1; /* No TLB found */ 2295118ebe8SLucas Mateus Castro (alqotel) for (way = 0; way < env->nb_ways; way++) { 2305118ebe8SLucas Mateus Castro (alqotel) nr = ppc6xx_tlb_getnum(env, eaddr, way, access_type == MMU_INST_FETCH); 2315118ebe8SLucas Mateus Castro (alqotel) tlb = &env->tlb.tlb6[nr]; 2325118ebe8SLucas Mateus Castro (alqotel) /* This test "emulates" the PTE index match for hardware TLBs */ 2335118ebe8SLucas Mateus Castro (alqotel) if ((eaddr & TARGET_PAGE_MASK) != tlb->EPN) { 2345118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("TLB %d/%d %s [" TARGET_FMT_lx " " TARGET_FMT_lx 2355118ebe8SLucas Mateus Castro (alqotel) "] <> " TARGET_FMT_lx "\n", nr, env->nb_tlb, 2365118ebe8SLucas Mateus Castro (alqotel) pte_is_valid(tlb->pte0) ? "valid" : "inval", 2375118ebe8SLucas Mateus Castro (alqotel) tlb->EPN, tlb->EPN + TARGET_PAGE_SIZE, eaddr); 2385118ebe8SLucas Mateus Castro (alqotel) continue; 2395118ebe8SLucas Mateus Castro (alqotel) } 2405118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("TLB %d/%d %s " TARGET_FMT_lx " <> " TARGET_FMT_lx " " 2415118ebe8SLucas Mateus Castro (alqotel) TARGET_FMT_lx " %c %c\n", nr, env->nb_tlb, 2425118ebe8SLucas Mateus Castro (alqotel) pte_is_valid(tlb->pte0) ? "valid" : "inval", 2435118ebe8SLucas Mateus Castro (alqotel) tlb->EPN, eaddr, tlb->pte1, 2445118ebe8SLucas Mateus Castro (alqotel) access_type == MMU_DATA_STORE ? 'S' : 'L', 2455118ebe8SLucas Mateus Castro (alqotel) access_type == MMU_INST_FETCH ? 'I' : 'D'); 2465118ebe8SLucas Mateus Castro (alqotel) switch (ppc6xx_tlb_pte_check(ctx, tlb->pte0, tlb->pte1, 2475118ebe8SLucas Mateus Castro (alqotel) 0, access_type)) { 2485118ebe8SLucas Mateus Castro (alqotel) case -3: 2495118ebe8SLucas Mateus Castro (alqotel) /* TLB inconsistency */ 2505118ebe8SLucas Mateus Castro (alqotel) return -1; 2515118ebe8SLucas Mateus Castro (alqotel) case -2: 2525118ebe8SLucas Mateus Castro (alqotel) /* Access violation */ 2535118ebe8SLucas Mateus Castro (alqotel) ret = -2; 2545118ebe8SLucas Mateus Castro (alqotel) best = nr; 2555118ebe8SLucas Mateus Castro (alqotel) break; 2565118ebe8SLucas Mateus Castro (alqotel) case -1: 2575118ebe8SLucas Mateus Castro (alqotel) default: 2585118ebe8SLucas Mateus Castro (alqotel) /* No match */ 2595118ebe8SLucas Mateus Castro (alqotel) break; 2605118ebe8SLucas Mateus Castro (alqotel) case 0: 2615118ebe8SLucas Mateus Castro (alqotel) /* access granted */ 2625118ebe8SLucas Mateus Castro (alqotel) /* 2635118ebe8SLucas Mateus Castro (alqotel) * XXX: we should go on looping to check all TLBs 2645118ebe8SLucas Mateus Castro (alqotel) * consistency but we can speed-up the whole thing as 2655118ebe8SLucas Mateus Castro (alqotel) * the result would be undefined if TLBs are not 2665118ebe8SLucas Mateus Castro (alqotel) * consistent. 2675118ebe8SLucas Mateus Castro (alqotel) */ 2685118ebe8SLucas Mateus Castro (alqotel) ret = 0; 2695118ebe8SLucas Mateus Castro (alqotel) best = nr; 2705118ebe8SLucas Mateus Castro (alqotel) goto done; 2715118ebe8SLucas Mateus Castro (alqotel) } 2725118ebe8SLucas Mateus Castro (alqotel) } 2735118ebe8SLucas Mateus Castro (alqotel) if (best != -1) { 2745118ebe8SLucas Mateus Castro (alqotel) done: 2755118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("found TLB at addr " TARGET_FMT_plx " prot=%01x ret=%d\n", 2765118ebe8SLucas Mateus Castro (alqotel) ctx->raddr & TARGET_PAGE_MASK, ctx->prot, ret); 2775118ebe8SLucas Mateus Castro (alqotel) /* Update page flags */ 2785118ebe8SLucas Mateus Castro (alqotel) pte_update_flags(ctx, &env->tlb.tlb6[best].pte1, ret, access_type); 2795118ebe8SLucas Mateus Castro (alqotel) } 2805118ebe8SLucas Mateus Castro (alqotel) 2815118ebe8SLucas Mateus Castro (alqotel) return ret; 2825118ebe8SLucas Mateus Castro (alqotel) } 2835118ebe8SLucas Mateus Castro (alqotel) 2845118ebe8SLucas Mateus Castro (alqotel) /* Perform BAT hit & translation */ 2855118ebe8SLucas Mateus Castro (alqotel) static inline void bat_size_prot(CPUPPCState *env, target_ulong *blp, 2865118ebe8SLucas Mateus Castro (alqotel) int *validp, int *protp, target_ulong *BATu, 2875118ebe8SLucas Mateus Castro (alqotel) target_ulong *BATl) 2885118ebe8SLucas Mateus Castro (alqotel) { 2895118ebe8SLucas Mateus Castro (alqotel) target_ulong bl; 2905118ebe8SLucas Mateus Castro (alqotel) int pp, valid, prot; 2915118ebe8SLucas Mateus Castro (alqotel) 2925118ebe8SLucas Mateus Castro (alqotel) bl = (*BATu & 0x00001FFC) << 15; 2935118ebe8SLucas Mateus Castro (alqotel) valid = 0; 2945118ebe8SLucas Mateus Castro (alqotel) prot = 0; 2955118ebe8SLucas Mateus Castro (alqotel) if (((msr_pr == 0) && (*BATu & 0x00000002)) || 2965118ebe8SLucas Mateus Castro (alqotel) ((msr_pr != 0) && (*BATu & 0x00000001))) { 2975118ebe8SLucas Mateus Castro (alqotel) valid = 1; 2985118ebe8SLucas Mateus Castro (alqotel) pp = *BATl & 0x00000003; 2995118ebe8SLucas Mateus Castro (alqotel) if (pp != 0) { 3005118ebe8SLucas Mateus Castro (alqotel) prot = PAGE_READ | PAGE_EXEC; 3015118ebe8SLucas Mateus Castro (alqotel) if (pp == 0x2) { 3025118ebe8SLucas Mateus Castro (alqotel) prot |= PAGE_WRITE; 3035118ebe8SLucas Mateus Castro (alqotel) } 3045118ebe8SLucas Mateus Castro (alqotel) } 3055118ebe8SLucas Mateus Castro (alqotel) } 3065118ebe8SLucas Mateus Castro (alqotel) *blp = bl; 3075118ebe8SLucas Mateus Castro (alqotel) *validp = valid; 3085118ebe8SLucas Mateus Castro (alqotel) *protp = prot; 3095118ebe8SLucas Mateus Castro (alqotel) } 3105118ebe8SLucas Mateus Castro (alqotel) 3115118ebe8SLucas Mateus Castro (alqotel) static int get_bat_6xx_tlb(CPUPPCState *env, mmu_ctx_t *ctx, 3125118ebe8SLucas Mateus Castro (alqotel) target_ulong virtual, MMUAccessType access_type) 3135118ebe8SLucas Mateus Castro (alqotel) { 3145118ebe8SLucas Mateus Castro (alqotel) target_ulong *BATlt, *BATut, *BATu, *BATl; 3155118ebe8SLucas Mateus Castro (alqotel) target_ulong BEPIl, BEPIu, bl; 3165118ebe8SLucas Mateus Castro (alqotel) int i, valid, prot; 3175118ebe8SLucas Mateus Castro (alqotel) int ret = -1; 3185118ebe8SLucas Mateus Castro (alqotel) bool ifetch = access_type == MMU_INST_FETCH; 3195118ebe8SLucas Mateus Castro (alqotel) 3205118ebe8SLucas Mateus Castro (alqotel) LOG_BATS("%s: %cBAT v " TARGET_FMT_lx "\n", __func__, 3215118ebe8SLucas Mateus Castro (alqotel) ifetch ? 'I' : 'D', virtual); 3225118ebe8SLucas Mateus Castro (alqotel) if (ifetch) { 3235118ebe8SLucas Mateus Castro (alqotel) BATlt = env->IBAT[1]; 3245118ebe8SLucas Mateus Castro (alqotel) BATut = env->IBAT[0]; 3255118ebe8SLucas Mateus Castro (alqotel) } else { 3265118ebe8SLucas Mateus Castro (alqotel) BATlt = env->DBAT[1]; 3275118ebe8SLucas Mateus Castro (alqotel) BATut = env->DBAT[0]; 3285118ebe8SLucas Mateus Castro (alqotel) } 3295118ebe8SLucas Mateus Castro (alqotel) for (i = 0; i < env->nb_BATs; i++) { 3305118ebe8SLucas Mateus Castro (alqotel) BATu = &BATut[i]; 3315118ebe8SLucas Mateus Castro (alqotel) BATl = &BATlt[i]; 3325118ebe8SLucas Mateus Castro (alqotel) BEPIu = *BATu & 0xF0000000; 3335118ebe8SLucas Mateus Castro (alqotel) BEPIl = *BATu & 0x0FFE0000; 3345118ebe8SLucas Mateus Castro (alqotel) bat_size_prot(env, &bl, &valid, &prot, BATu, BATl); 3355118ebe8SLucas Mateus Castro (alqotel) LOG_BATS("%s: %cBAT%d v " TARGET_FMT_lx " BATu " TARGET_FMT_lx 3365118ebe8SLucas Mateus Castro (alqotel) " BATl " TARGET_FMT_lx "\n", __func__, 3375118ebe8SLucas Mateus Castro (alqotel) ifetch ? 'I' : 'D', i, virtual, *BATu, *BATl); 3385118ebe8SLucas Mateus Castro (alqotel) if ((virtual & 0xF0000000) == BEPIu && 3395118ebe8SLucas Mateus Castro (alqotel) ((virtual & 0x0FFE0000) & ~bl) == BEPIl) { 3405118ebe8SLucas Mateus Castro (alqotel) /* BAT matches */ 3415118ebe8SLucas Mateus Castro (alqotel) if (valid != 0) { 3425118ebe8SLucas Mateus Castro (alqotel) /* Get physical address */ 3435118ebe8SLucas Mateus Castro (alqotel) ctx->raddr = (*BATl & 0xF0000000) | 3445118ebe8SLucas Mateus Castro (alqotel) ((virtual & 0x0FFE0000 & bl) | (*BATl & 0x0FFE0000)) | 3455118ebe8SLucas Mateus Castro (alqotel) (virtual & 0x0001F000); 3465118ebe8SLucas Mateus Castro (alqotel) /* Compute access rights */ 3475118ebe8SLucas Mateus Castro (alqotel) ctx->prot = prot; 3485118ebe8SLucas Mateus Castro (alqotel) ret = check_prot(ctx->prot, access_type); 3495118ebe8SLucas Mateus Castro (alqotel) if (ret == 0) { 3505118ebe8SLucas Mateus Castro (alqotel) LOG_BATS("BAT %d match: r " TARGET_FMT_plx " prot=%c%c\n", 3515118ebe8SLucas Mateus Castro (alqotel) i, ctx->raddr, ctx->prot & PAGE_READ ? 'R' : '-', 3525118ebe8SLucas Mateus Castro (alqotel) ctx->prot & PAGE_WRITE ? 'W' : '-'); 3535118ebe8SLucas Mateus Castro (alqotel) } 3545118ebe8SLucas Mateus Castro (alqotel) break; 3555118ebe8SLucas Mateus Castro (alqotel) } 3565118ebe8SLucas Mateus Castro (alqotel) } 3575118ebe8SLucas Mateus Castro (alqotel) } 3585118ebe8SLucas Mateus Castro (alqotel) if (ret < 0) { 3595118ebe8SLucas Mateus Castro (alqotel) #if defined(DEBUG_BATS) 3605118ebe8SLucas Mateus Castro (alqotel) if (qemu_log_enabled()) { 3615118ebe8SLucas Mateus Castro (alqotel) LOG_BATS("no BAT match for " TARGET_FMT_lx ":\n", virtual); 3625118ebe8SLucas Mateus Castro (alqotel) for (i = 0; i < 4; i++) { 3635118ebe8SLucas Mateus Castro (alqotel) BATu = &BATut[i]; 3645118ebe8SLucas Mateus Castro (alqotel) BATl = &BATlt[i]; 3655118ebe8SLucas Mateus Castro (alqotel) BEPIu = *BATu & 0xF0000000; 3665118ebe8SLucas Mateus Castro (alqotel) BEPIl = *BATu & 0x0FFE0000; 3675118ebe8SLucas Mateus Castro (alqotel) bl = (*BATu & 0x00001FFC) << 15; 3685118ebe8SLucas Mateus Castro (alqotel) LOG_BATS("%s: %cBAT%d v " TARGET_FMT_lx " BATu " TARGET_FMT_lx 3695118ebe8SLucas Mateus Castro (alqotel) " BATl " TARGET_FMT_lx "\n\t" TARGET_FMT_lx " " 3705118ebe8SLucas Mateus Castro (alqotel) TARGET_FMT_lx " " TARGET_FMT_lx "\n", 3715118ebe8SLucas Mateus Castro (alqotel) __func__, ifetch ? 'I' : 'D', i, virtual, 3725118ebe8SLucas Mateus Castro (alqotel) *BATu, *BATl, BEPIu, BEPIl, bl); 3735118ebe8SLucas Mateus Castro (alqotel) } 3745118ebe8SLucas Mateus Castro (alqotel) } 3755118ebe8SLucas Mateus Castro (alqotel) #endif 3765118ebe8SLucas Mateus Castro (alqotel) } 3775118ebe8SLucas Mateus Castro (alqotel) /* No hit */ 3785118ebe8SLucas Mateus Castro (alqotel) return ret; 3795118ebe8SLucas Mateus Castro (alqotel) } 3805118ebe8SLucas Mateus Castro (alqotel) 3815118ebe8SLucas Mateus Castro (alqotel) /* Perform segment based translation */ 3825118ebe8SLucas Mateus Castro (alqotel) static int get_segment_6xx_tlb(CPUPPCState *env, mmu_ctx_t *ctx, 3835118ebe8SLucas Mateus Castro (alqotel) target_ulong eaddr, MMUAccessType access_type, 3845118ebe8SLucas Mateus Castro (alqotel) int type) 3855118ebe8SLucas Mateus Castro (alqotel) { 3865118ebe8SLucas Mateus Castro (alqotel) PowerPCCPU *cpu = env_archcpu(env); 3875118ebe8SLucas Mateus Castro (alqotel) hwaddr hash; 3885118ebe8SLucas Mateus Castro (alqotel) target_ulong vsid; 3895118ebe8SLucas Mateus Castro (alqotel) int ds, pr, target_page_bits; 3905118ebe8SLucas Mateus Castro (alqotel) int ret; 3915118ebe8SLucas Mateus Castro (alqotel) target_ulong sr, pgidx; 3925118ebe8SLucas Mateus Castro (alqotel) 3935118ebe8SLucas Mateus Castro (alqotel) pr = msr_pr; 3945118ebe8SLucas Mateus Castro (alqotel) ctx->eaddr = eaddr; 3955118ebe8SLucas Mateus Castro (alqotel) 3965118ebe8SLucas Mateus Castro (alqotel) sr = env->sr[eaddr >> 28]; 3975118ebe8SLucas Mateus Castro (alqotel) ctx->key = (((sr & 0x20000000) && (pr != 0)) || 3985118ebe8SLucas Mateus Castro (alqotel) ((sr & 0x40000000) && (pr == 0))) ? 1 : 0; 3995118ebe8SLucas Mateus Castro (alqotel) ds = sr & 0x80000000 ? 1 : 0; 4005118ebe8SLucas Mateus Castro (alqotel) ctx->nx = sr & 0x10000000 ? 1 : 0; 4015118ebe8SLucas Mateus Castro (alqotel) vsid = sr & 0x00FFFFFF; 4025118ebe8SLucas Mateus Castro (alqotel) target_page_bits = TARGET_PAGE_BITS; 4035118ebe8SLucas Mateus Castro (alqotel) qemu_log_mask(CPU_LOG_MMU, 4045118ebe8SLucas Mateus Castro (alqotel) "Check segment v=" TARGET_FMT_lx " %d " TARGET_FMT_lx 4055118ebe8SLucas Mateus Castro (alqotel) " nip=" TARGET_FMT_lx " lr=" TARGET_FMT_lx 4065118ebe8SLucas Mateus Castro (alqotel) " ir=%d dr=%d pr=%d %d t=%d\n", 4075118ebe8SLucas Mateus Castro (alqotel) eaddr, (int)(eaddr >> 28), sr, env->nip, env->lr, (int)msr_ir, 4085118ebe8SLucas Mateus Castro (alqotel) (int)msr_dr, pr != 0 ? 1 : 0, access_type == MMU_DATA_STORE, type); 4095118ebe8SLucas Mateus Castro (alqotel) pgidx = (eaddr & ~SEGMENT_MASK_256M) >> target_page_bits; 4105118ebe8SLucas Mateus Castro (alqotel) hash = vsid ^ pgidx; 4115118ebe8SLucas Mateus Castro (alqotel) ctx->ptem = (vsid << 7) | (pgidx >> 10); 4125118ebe8SLucas Mateus Castro (alqotel) 4135118ebe8SLucas Mateus Castro (alqotel) qemu_log_mask(CPU_LOG_MMU, 4145118ebe8SLucas Mateus Castro (alqotel) "pte segment: key=%d ds %d nx %d vsid " TARGET_FMT_lx "\n", 4155118ebe8SLucas Mateus Castro (alqotel) ctx->key, ds, ctx->nx, vsid); 4165118ebe8SLucas Mateus Castro (alqotel) ret = -1; 4175118ebe8SLucas Mateus Castro (alqotel) if (!ds) { 4185118ebe8SLucas Mateus Castro (alqotel) /* Check if instruction fetch is allowed, if needed */ 4195118ebe8SLucas Mateus Castro (alqotel) if (type != ACCESS_CODE || ctx->nx == 0) { 4205118ebe8SLucas Mateus Castro (alqotel) /* Page address translation */ 4215118ebe8SLucas Mateus Castro (alqotel) qemu_log_mask(CPU_LOG_MMU, "htab_base " TARGET_FMT_plx 4225118ebe8SLucas Mateus Castro (alqotel) " htab_mask " TARGET_FMT_plx 4235118ebe8SLucas Mateus Castro (alqotel) " hash " TARGET_FMT_plx "\n", 4245118ebe8SLucas Mateus Castro (alqotel) ppc_hash32_hpt_base(cpu), ppc_hash32_hpt_mask(cpu), hash); 4255118ebe8SLucas Mateus Castro (alqotel) ctx->hash[0] = hash; 4265118ebe8SLucas Mateus Castro (alqotel) ctx->hash[1] = ~hash; 4275118ebe8SLucas Mateus Castro (alqotel) 4285118ebe8SLucas Mateus Castro (alqotel) /* Initialize real address with an invalid value */ 4295118ebe8SLucas Mateus Castro (alqotel) ctx->raddr = (hwaddr)-1ULL; 4305118ebe8SLucas Mateus Castro (alqotel) /* Software TLB search */ 4315118ebe8SLucas Mateus Castro (alqotel) ret = ppc6xx_tlb_check(env, ctx, eaddr, access_type); 4325118ebe8SLucas Mateus Castro (alqotel) #if defined(DUMP_PAGE_TABLES) 4335118ebe8SLucas Mateus Castro (alqotel) if (qemu_loglevel_mask(CPU_LOG_MMU)) { 4345118ebe8SLucas Mateus Castro (alqotel) CPUState *cs = env_cpu(env); 4355118ebe8SLucas Mateus Castro (alqotel) hwaddr curaddr; 4365118ebe8SLucas Mateus Castro (alqotel) uint32_t a0, a1, a2, a3; 4375118ebe8SLucas Mateus Castro (alqotel) 4385118ebe8SLucas Mateus Castro (alqotel) qemu_log("Page table: " TARGET_FMT_plx " len " TARGET_FMT_plx 4395118ebe8SLucas Mateus Castro (alqotel) "\n", ppc_hash32_hpt_base(cpu), 4405118ebe8SLucas Mateus Castro (alqotel) ppc_hash32_hpt_mask(cpu) + 0x80); 4415118ebe8SLucas Mateus Castro (alqotel) for (curaddr = ppc_hash32_hpt_base(cpu); 4425118ebe8SLucas Mateus Castro (alqotel) curaddr < (ppc_hash32_hpt_base(cpu) 4435118ebe8SLucas Mateus Castro (alqotel) + ppc_hash32_hpt_mask(cpu) + 0x80); 4445118ebe8SLucas Mateus Castro (alqotel) curaddr += 16) { 4455118ebe8SLucas Mateus Castro (alqotel) a0 = ldl_phys(cs->as, curaddr); 4465118ebe8SLucas Mateus Castro (alqotel) a1 = ldl_phys(cs->as, curaddr + 4); 4475118ebe8SLucas Mateus Castro (alqotel) a2 = ldl_phys(cs->as, curaddr + 8); 4485118ebe8SLucas Mateus Castro (alqotel) a3 = ldl_phys(cs->as, curaddr + 12); 4495118ebe8SLucas Mateus Castro (alqotel) if (a0 != 0 || a1 != 0 || a2 != 0 || a3 != 0) { 4505118ebe8SLucas Mateus Castro (alqotel) qemu_log(TARGET_FMT_plx ": %08x %08x %08x %08x\n", 4515118ebe8SLucas Mateus Castro (alqotel) curaddr, a0, a1, a2, a3); 4525118ebe8SLucas Mateus Castro (alqotel) } 4535118ebe8SLucas Mateus Castro (alqotel) } 4545118ebe8SLucas Mateus Castro (alqotel) } 4555118ebe8SLucas Mateus Castro (alqotel) #endif 4565118ebe8SLucas Mateus Castro (alqotel) } else { 4575118ebe8SLucas Mateus Castro (alqotel) qemu_log_mask(CPU_LOG_MMU, "No access allowed\n"); 4585118ebe8SLucas Mateus Castro (alqotel) ret = -3; 4595118ebe8SLucas Mateus Castro (alqotel) } 4605118ebe8SLucas Mateus Castro (alqotel) } else { 4615118ebe8SLucas Mateus Castro (alqotel) target_ulong sr; 4625118ebe8SLucas Mateus Castro (alqotel) 4635118ebe8SLucas Mateus Castro (alqotel) qemu_log_mask(CPU_LOG_MMU, "direct store...\n"); 4645118ebe8SLucas Mateus Castro (alqotel) /* Direct-store segment : absolutely *BUGGY* for now */ 4655118ebe8SLucas Mateus Castro (alqotel) 4665118ebe8SLucas Mateus Castro (alqotel) /* 4675118ebe8SLucas Mateus Castro (alqotel) * Direct-store implies a 32-bit MMU. 4685118ebe8SLucas Mateus Castro (alqotel) * Check the Segment Register's bus unit ID (BUID). 4695118ebe8SLucas Mateus Castro (alqotel) */ 4705118ebe8SLucas Mateus Castro (alqotel) sr = env->sr[eaddr >> 28]; 4715118ebe8SLucas Mateus Castro (alqotel) if ((sr & 0x1FF00000) >> 20 == 0x07f) { 4725118ebe8SLucas Mateus Castro (alqotel) /* 4735118ebe8SLucas Mateus Castro (alqotel) * Memory-forced I/O controller interface access 4745118ebe8SLucas Mateus Castro (alqotel) * 4755118ebe8SLucas Mateus Castro (alqotel) * If T=1 and BUID=x'07F', the 601 performs a memory 4765118ebe8SLucas Mateus Castro (alqotel) * access to SR[28-31] LA[4-31], bypassing all protection 4775118ebe8SLucas Mateus Castro (alqotel) * mechanisms. 4785118ebe8SLucas Mateus Castro (alqotel) */ 4795118ebe8SLucas Mateus Castro (alqotel) ctx->raddr = ((sr & 0xF) << 28) | (eaddr & 0x0FFFFFFF); 4805118ebe8SLucas Mateus Castro (alqotel) ctx->prot = PAGE_READ | PAGE_WRITE | PAGE_EXEC; 4815118ebe8SLucas Mateus Castro (alqotel) return 0; 4825118ebe8SLucas Mateus Castro (alqotel) } 4835118ebe8SLucas Mateus Castro (alqotel) 4845118ebe8SLucas Mateus Castro (alqotel) switch (type) { 4855118ebe8SLucas Mateus Castro (alqotel) case ACCESS_INT: 4865118ebe8SLucas Mateus Castro (alqotel) /* Integer load/store : only access allowed */ 4875118ebe8SLucas Mateus Castro (alqotel) break; 4885118ebe8SLucas Mateus Castro (alqotel) case ACCESS_CODE: 4895118ebe8SLucas Mateus Castro (alqotel) /* No code fetch is allowed in direct-store areas */ 4905118ebe8SLucas Mateus Castro (alqotel) return -4; 4915118ebe8SLucas Mateus Castro (alqotel) case ACCESS_FLOAT: 4925118ebe8SLucas Mateus Castro (alqotel) /* Floating point load/store */ 4935118ebe8SLucas Mateus Castro (alqotel) return -4; 4945118ebe8SLucas Mateus Castro (alqotel) case ACCESS_RES: 4955118ebe8SLucas Mateus Castro (alqotel) /* lwarx, ldarx or srwcx. */ 4965118ebe8SLucas Mateus Castro (alqotel) return -4; 4975118ebe8SLucas Mateus Castro (alqotel) case ACCESS_CACHE: 4985118ebe8SLucas Mateus Castro (alqotel) /* 4995118ebe8SLucas Mateus Castro (alqotel) * dcba, dcbt, dcbtst, dcbf, dcbi, dcbst, dcbz, or icbi 5005118ebe8SLucas Mateus Castro (alqotel) * 5015118ebe8SLucas Mateus Castro (alqotel) * Should make the instruction do no-op. As it already do 5025118ebe8SLucas Mateus Castro (alqotel) * no-op, it's quite easy :-) 5035118ebe8SLucas Mateus Castro (alqotel) */ 5045118ebe8SLucas Mateus Castro (alqotel) ctx->raddr = eaddr; 5055118ebe8SLucas Mateus Castro (alqotel) return 0; 5065118ebe8SLucas Mateus Castro (alqotel) case ACCESS_EXT: 5075118ebe8SLucas Mateus Castro (alqotel) /* eciwx or ecowx */ 5085118ebe8SLucas Mateus Castro (alqotel) return -4; 5095118ebe8SLucas Mateus Castro (alqotel) default: 5105118ebe8SLucas Mateus Castro (alqotel) qemu_log_mask(CPU_LOG_MMU, "ERROR: instruction should not need " 5115118ebe8SLucas Mateus Castro (alqotel) "address translation\n"); 5125118ebe8SLucas Mateus Castro (alqotel) return -4; 5135118ebe8SLucas Mateus Castro (alqotel) } 5145118ebe8SLucas Mateus Castro (alqotel) if ((access_type == MMU_DATA_STORE || ctx->key != 1) && 5155118ebe8SLucas Mateus Castro (alqotel) (access_type == MMU_DATA_LOAD || ctx->key != 0)) { 5165118ebe8SLucas Mateus Castro (alqotel) ctx->raddr = eaddr; 5175118ebe8SLucas Mateus Castro (alqotel) ret = 2; 5185118ebe8SLucas Mateus Castro (alqotel) } else { 5195118ebe8SLucas Mateus Castro (alqotel) ret = -2; 5205118ebe8SLucas Mateus Castro (alqotel) } 5215118ebe8SLucas Mateus Castro (alqotel) } 5225118ebe8SLucas Mateus Castro (alqotel) 5235118ebe8SLucas Mateus Castro (alqotel) return ret; 5245118ebe8SLucas Mateus Castro (alqotel) } 5255118ebe8SLucas Mateus Castro (alqotel) 5265118ebe8SLucas Mateus Castro (alqotel) /* Generic TLB check function for embedded PowerPC implementations */ 5275118ebe8SLucas Mateus Castro (alqotel) int ppcemb_tlb_check(CPUPPCState *env, ppcemb_tlb_t *tlb, 5285118ebe8SLucas Mateus Castro (alqotel) hwaddr *raddrp, 5295118ebe8SLucas Mateus Castro (alqotel) target_ulong address, uint32_t pid, int ext, 5305118ebe8SLucas Mateus Castro (alqotel) int i) 5315118ebe8SLucas Mateus Castro (alqotel) { 5325118ebe8SLucas Mateus Castro (alqotel) target_ulong mask; 5335118ebe8SLucas Mateus Castro (alqotel) 5345118ebe8SLucas Mateus Castro (alqotel) /* Check valid flag */ 5355118ebe8SLucas Mateus Castro (alqotel) if (!(tlb->prot & PAGE_VALID)) { 5365118ebe8SLucas Mateus Castro (alqotel) return -1; 5375118ebe8SLucas Mateus Castro (alqotel) } 5385118ebe8SLucas Mateus Castro (alqotel) mask = ~(tlb->size - 1); 5395118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: TLB %d address " TARGET_FMT_lx " PID %u <=> " TARGET_FMT_lx 5405118ebe8SLucas Mateus Castro (alqotel) " " TARGET_FMT_lx " %u %x\n", __func__, i, address, pid, tlb->EPN, 5415118ebe8SLucas Mateus Castro (alqotel) mask, (uint32_t)tlb->PID, tlb->prot); 5425118ebe8SLucas Mateus Castro (alqotel) /* Check PID */ 5435118ebe8SLucas Mateus Castro (alqotel) if (tlb->PID != 0 && tlb->PID != pid) { 5445118ebe8SLucas Mateus Castro (alqotel) return -1; 5455118ebe8SLucas Mateus Castro (alqotel) } 5465118ebe8SLucas Mateus Castro (alqotel) /* Check effective address */ 5475118ebe8SLucas Mateus Castro (alqotel) if ((address & mask) != tlb->EPN) { 5485118ebe8SLucas Mateus Castro (alqotel) return -1; 5495118ebe8SLucas Mateus Castro (alqotel) } 5505118ebe8SLucas Mateus Castro (alqotel) *raddrp = (tlb->RPN & mask) | (address & ~mask); 5515118ebe8SLucas Mateus Castro (alqotel) if (ext) { 5525118ebe8SLucas Mateus Castro (alqotel) /* Extend the physical address to 36 bits */ 5535118ebe8SLucas Mateus Castro (alqotel) *raddrp |= (uint64_t)(tlb->RPN & 0xF) << 32; 5545118ebe8SLucas Mateus Castro (alqotel) } 5555118ebe8SLucas Mateus Castro (alqotel) 5565118ebe8SLucas Mateus Castro (alqotel) return 0; 5575118ebe8SLucas Mateus Castro (alqotel) } 5585118ebe8SLucas Mateus Castro (alqotel) 5595118ebe8SLucas Mateus Castro (alqotel) static int mmu40x_get_physical_address(CPUPPCState *env, mmu_ctx_t *ctx, 5605118ebe8SLucas Mateus Castro (alqotel) target_ulong address, 5615118ebe8SLucas Mateus Castro (alqotel) MMUAccessType access_type) 5625118ebe8SLucas Mateus Castro (alqotel) { 5635118ebe8SLucas Mateus Castro (alqotel) ppcemb_tlb_t *tlb; 5645118ebe8SLucas Mateus Castro (alqotel) hwaddr raddr; 5655118ebe8SLucas Mateus Castro (alqotel) int i, ret, zsel, zpr, pr; 5665118ebe8SLucas Mateus Castro (alqotel) 5675118ebe8SLucas Mateus Castro (alqotel) ret = -1; 5685118ebe8SLucas Mateus Castro (alqotel) raddr = (hwaddr)-1ULL; 5695118ebe8SLucas Mateus Castro (alqotel) pr = msr_pr; 5705118ebe8SLucas Mateus Castro (alqotel) for (i = 0; i < env->nb_tlb; i++) { 5715118ebe8SLucas Mateus Castro (alqotel) tlb = &env->tlb.tlbe[i]; 5725118ebe8SLucas Mateus Castro (alqotel) if (ppcemb_tlb_check(env, tlb, &raddr, address, 5735118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_40x_PID], 0, i) < 0) { 5745118ebe8SLucas Mateus Castro (alqotel) continue; 5755118ebe8SLucas Mateus Castro (alqotel) } 5765118ebe8SLucas Mateus Castro (alqotel) zsel = (tlb->attr >> 4) & 0xF; 5775118ebe8SLucas Mateus Castro (alqotel) zpr = (env->spr[SPR_40x_ZPR] >> (30 - (2 * zsel))) & 0x3; 5785118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: TLB %d zsel %d zpr %d ty %d attr %08x\n", 5795118ebe8SLucas Mateus Castro (alqotel) __func__, i, zsel, zpr, access_type, tlb->attr); 5805118ebe8SLucas Mateus Castro (alqotel) /* Check execute enable bit */ 5815118ebe8SLucas Mateus Castro (alqotel) switch (zpr) { 5825118ebe8SLucas Mateus Castro (alqotel) case 0x2: 5835118ebe8SLucas Mateus Castro (alqotel) if (pr != 0) { 5845118ebe8SLucas Mateus Castro (alqotel) goto check_perms; 5855118ebe8SLucas Mateus Castro (alqotel) } 5865118ebe8SLucas Mateus Castro (alqotel) /* fall through */ 5875118ebe8SLucas Mateus Castro (alqotel) case 0x3: 5885118ebe8SLucas Mateus Castro (alqotel) /* All accesses granted */ 5895118ebe8SLucas Mateus Castro (alqotel) ctx->prot = PAGE_READ | PAGE_WRITE | PAGE_EXEC; 5905118ebe8SLucas Mateus Castro (alqotel) ret = 0; 5915118ebe8SLucas Mateus Castro (alqotel) break; 5925118ebe8SLucas Mateus Castro (alqotel) case 0x0: 5935118ebe8SLucas Mateus Castro (alqotel) if (pr != 0) { 5945118ebe8SLucas Mateus Castro (alqotel) /* Raise Zone protection fault. */ 5955118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_40x_ESR] = 1 << 22; 5965118ebe8SLucas Mateus Castro (alqotel) ctx->prot = 0; 5975118ebe8SLucas Mateus Castro (alqotel) ret = -2; 5985118ebe8SLucas Mateus Castro (alqotel) break; 5995118ebe8SLucas Mateus Castro (alqotel) } 6005118ebe8SLucas Mateus Castro (alqotel) /* fall through */ 6015118ebe8SLucas Mateus Castro (alqotel) case 0x1: 6025118ebe8SLucas Mateus Castro (alqotel) check_perms: 6035118ebe8SLucas Mateus Castro (alqotel) /* Check from TLB entry */ 6045118ebe8SLucas Mateus Castro (alqotel) ctx->prot = tlb->prot; 6055118ebe8SLucas Mateus Castro (alqotel) ret = check_prot(ctx->prot, access_type); 6065118ebe8SLucas Mateus Castro (alqotel) if (ret == -2) { 6075118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_40x_ESR] = 0; 6085118ebe8SLucas Mateus Castro (alqotel) } 6095118ebe8SLucas Mateus Castro (alqotel) break; 6105118ebe8SLucas Mateus Castro (alqotel) } 6115118ebe8SLucas Mateus Castro (alqotel) if (ret >= 0) { 6125118ebe8SLucas Mateus Castro (alqotel) ctx->raddr = raddr; 6135118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: access granted " TARGET_FMT_lx " => " TARGET_FMT_plx 6145118ebe8SLucas Mateus Castro (alqotel) " %d %d\n", __func__, address, ctx->raddr, ctx->prot, 6155118ebe8SLucas Mateus Castro (alqotel) ret); 6165118ebe8SLucas Mateus Castro (alqotel) return 0; 6175118ebe8SLucas Mateus Castro (alqotel) } 6185118ebe8SLucas Mateus Castro (alqotel) } 6195118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: access refused " TARGET_FMT_lx " => " TARGET_FMT_plx 6205118ebe8SLucas Mateus Castro (alqotel) " %d %d\n", __func__, address, raddr, ctx->prot, ret); 6215118ebe8SLucas Mateus Castro (alqotel) 6225118ebe8SLucas Mateus Castro (alqotel) return ret; 6235118ebe8SLucas Mateus Castro (alqotel) } 6245118ebe8SLucas Mateus Castro (alqotel) 6255118ebe8SLucas Mateus Castro (alqotel) static int mmubooke_check_tlb(CPUPPCState *env, ppcemb_tlb_t *tlb, 6265118ebe8SLucas Mateus Castro (alqotel) hwaddr *raddr, int *prot, target_ulong address, 6275118ebe8SLucas Mateus Castro (alqotel) MMUAccessType access_type, int i) 6285118ebe8SLucas Mateus Castro (alqotel) { 6295118ebe8SLucas Mateus Castro (alqotel) int prot2; 6305118ebe8SLucas Mateus Castro (alqotel) 6315118ebe8SLucas Mateus Castro (alqotel) if (ppcemb_tlb_check(env, tlb, raddr, address, 6325118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_PID], 6335118ebe8SLucas Mateus Castro (alqotel) !env->nb_pids, i) >= 0) { 6345118ebe8SLucas Mateus Castro (alqotel) goto found_tlb; 6355118ebe8SLucas Mateus Castro (alqotel) } 6365118ebe8SLucas Mateus Castro (alqotel) 6375118ebe8SLucas Mateus Castro (alqotel) if (env->spr[SPR_BOOKE_PID1] && 6385118ebe8SLucas Mateus Castro (alqotel) ppcemb_tlb_check(env, tlb, raddr, address, 6395118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_PID1], 0, i) >= 0) { 6405118ebe8SLucas Mateus Castro (alqotel) goto found_tlb; 6415118ebe8SLucas Mateus Castro (alqotel) } 6425118ebe8SLucas Mateus Castro (alqotel) 6435118ebe8SLucas Mateus Castro (alqotel) if (env->spr[SPR_BOOKE_PID2] && 6445118ebe8SLucas Mateus Castro (alqotel) ppcemb_tlb_check(env, tlb, raddr, address, 6455118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_PID2], 0, i) >= 0) { 6465118ebe8SLucas Mateus Castro (alqotel) goto found_tlb; 6475118ebe8SLucas Mateus Castro (alqotel) } 6485118ebe8SLucas Mateus Castro (alqotel) 6495118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: TLB entry not found\n", __func__); 6505118ebe8SLucas Mateus Castro (alqotel) return -1; 6515118ebe8SLucas Mateus Castro (alqotel) 6525118ebe8SLucas Mateus Castro (alqotel) found_tlb: 6535118ebe8SLucas Mateus Castro (alqotel) 6545118ebe8SLucas Mateus Castro (alqotel) if (msr_pr != 0) { 6555118ebe8SLucas Mateus Castro (alqotel) prot2 = tlb->prot & 0xF; 6565118ebe8SLucas Mateus Castro (alqotel) } else { 6575118ebe8SLucas Mateus Castro (alqotel) prot2 = (tlb->prot >> 4) & 0xF; 6585118ebe8SLucas Mateus Castro (alqotel) } 6595118ebe8SLucas Mateus Castro (alqotel) 6605118ebe8SLucas Mateus Castro (alqotel) /* Check the address space */ 6615118ebe8SLucas Mateus Castro (alqotel) if ((access_type == MMU_INST_FETCH ? msr_ir : msr_dr) != (tlb->attr & 1)) { 6625118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: AS doesn't match\n", __func__); 6635118ebe8SLucas Mateus Castro (alqotel) return -1; 6645118ebe8SLucas Mateus Castro (alqotel) } 6655118ebe8SLucas Mateus Castro (alqotel) 6665118ebe8SLucas Mateus Castro (alqotel) *prot = prot2; 6675118ebe8SLucas Mateus Castro (alqotel) if (prot2 & prot_for_access_type(access_type)) { 6685118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: good TLB!\n", __func__); 6695118ebe8SLucas Mateus Castro (alqotel) return 0; 6705118ebe8SLucas Mateus Castro (alqotel) } 6715118ebe8SLucas Mateus Castro (alqotel) 6725118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: no prot match: %x\n", __func__, prot2); 6735118ebe8SLucas Mateus Castro (alqotel) return access_type == MMU_INST_FETCH ? -3 : -2; 6745118ebe8SLucas Mateus Castro (alqotel) } 6755118ebe8SLucas Mateus Castro (alqotel) 6765118ebe8SLucas Mateus Castro (alqotel) static int mmubooke_get_physical_address(CPUPPCState *env, mmu_ctx_t *ctx, 6775118ebe8SLucas Mateus Castro (alqotel) target_ulong address, 6785118ebe8SLucas Mateus Castro (alqotel) MMUAccessType access_type) 6795118ebe8SLucas Mateus Castro (alqotel) { 6805118ebe8SLucas Mateus Castro (alqotel) ppcemb_tlb_t *tlb; 6815118ebe8SLucas Mateus Castro (alqotel) hwaddr raddr; 6825118ebe8SLucas Mateus Castro (alqotel) int i, ret; 6835118ebe8SLucas Mateus Castro (alqotel) 6845118ebe8SLucas Mateus Castro (alqotel) ret = -1; 6855118ebe8SLucas Mateus Castro (alqotel) raddr = (hwaddr)-1ULL; 6865118ebe8SLucas Mateus Castro (alqotel) for (i = 0; i < env->nb_tlb; i++) { 6875118ebe8SLucas Mateus Castro (alqotel) tlb = &env->tlb.tlbe[i]; 6885118ebe8SLucas Mateus Castro (alqotel) ret = mmubooke_check_tlb(env, tlb, &raddr, &ctx->prot, address, 6895118ebe8SLucas Mateus Castro (alqotel) access_type, i); 6905118ebe8SLucas Mateus Castro (alqotel) if (ret != -1) { 6915118ebe8SLucas Mateus Castro (alqotel) break; 6925118ebe8SLucas Mateus Castro (alqotel) } 6935118ebe8SLucas Mateus Castro (alqotel) } 6945118ebe8SLucas Mateus Castro (alqotel) 6955118ebe8SLucas Mateus Castro (alqotel) if (ret >= 0) { 6965118ebe8SLucas Mateus Castro (alqotel) ctx->raddr = raddr; 6975118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: access granted " TARGET_FMT_lx " => " TARGET_FMT_plx 6985118ebe8SLucas Mateus Castro (alqotel) " %d %d\n", __func__, address, ctx->raddr, ctx->prot, 6995118ebe8SLucas Mateus Castro (alqotel) ret); 7005118ebe8SLucas Mateus Castro (alqotel) } else { 7015118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: access refused " TARGET_FMT_lx " => " TARGET_FMT_plx 7025118ebe8SLucas Mateus Castro (alqotel) " %d %d\n", __func__, address, raddr, ctx->prot, ret); 7035118ebe8SLucas Mateus Castro (alqotel) } 7045118ebe8SLucas Mateus Castro (alqotel) 7055118ebe8SLucas Mateus Castro (alqotel) return ret; 7065118ebe8SLucas Mateus Castro (alqotel) } 7075118ebe8SLucas Mateus Castro (alqotel) 7085118ebe8SLucas Mateus Castro (alqotel) hwaddr booke206_tlb_to_page_size(CPUPPCState *env, 7095118ebe8SLucas Mateus Castro (alqotel) ppcmas_tlb_t *tlb) 7105118ebe8SLucas Mateus Castro (alqotel) { 7115118ebe8SLucas Mateus Castro (alqotel) int tlbm_size; 7125118ebe8SLucas Mateus Castro (alqotel) 7135118ebe8SLucas Mateus Castro (alqotel) tlbm_size = (tlb->mas1 & MAS1_TSIZE_MASK) >> MAS1_TSIZE_SHIFT; 7145118ebe8SLucas Mateus Castro (alqotel) 7155118ebe8SLucas Mateus Castro (alqotel) return 1024ULL << tlbm_size; 7165118ebe8SLucas Mateus Castro (alqotel) } 7175118ebe8SLucas Mateus Castro (alqotel) 7185118ebe8SLucas Mateus Castro (alqotel) /* TLB check function for MAS based SoftTLBs */ 7195118ebe8SLucas Mateus Castro (alqotel) int ppcmas_tlb_check(CPUPPCState *env, ppcmas_tlb_t *tlb, 7205118ebe8SLucas Mateus Castro (alqotel) hwaddr *raddrp, target_ulong address, 7215118ebe8SLucas Mateus Castro (alqotel) uint32_t pid) 7225118ebe8SLucas Mateus Castro (alqotel) { 7235118ebe8SLucas Mateus Castro (alqotel) hwaddr mask; 7245118ebe8SLucas Mateus Castro (alqotel) uint32_t tlb_pid; 7255118ebe8SLucas Mateus Castro (alqotel) 7265118ebe8SLucas Mateus Castro (alqotel) if (!msr_cm) { 7275118ebe8SLucas Mateus Castro (alqotel) /* In 32bit mode we can only address 32bit EAs */ 7285118ebe8SLucas Mateus Castro (alqotel) address = (uint32_t)address; 7295118ebe8SLucas Mateus Castro (alqotel) } 7305118ebe8SLucas Mateus Castro (alqotel) 7315118ebe8SLucas Mateus Castro (alqotel) /* Check valid flag */ 7325118ebe8SLucas Mateus Castro (alqotel) if (!(tlb->mas1 & MAS1_VALID)) { 7335118ebe8SLucas Mateus Castro (alqotel) return -1; 7345118ebe8SLucas Mateus Castro (alqotel) } 7355118ebe8SLucas Mateus Castro (alqotel) 7365118ebe8SLucas Mateus Castro (alqotel) mask = ~(booke206_tlb_to_page_size(env, tlb) - 1); 7375118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: TLB ADDR=0x" TARGET_FMT_lx " PID=0x%x MAS1=0x%x MAS2=0x%" 7385118ebe8SLucas Mateus Castro (alqotel) PRIx64 " mask=0x%" HWADDR_PRIx " MAS7_3=0x%" PRIx64 " MAS8=0x%" 7395118ebe8SLucas Mateus Castro (alqotel) PRIx32 "\n", __func__, address, pid, tlb->mas1, tlb->mas2, mask, 7405118ebe8SLucas Mateus Castro (alqotel) tlb->mas7_3, tlb->mas8); 7415118ebe8SLucas Mateus Castro (alqotel) 7425118ebe8SLucas Mateus Castro (alqotel) /* Check PID */ 7435118ebe8SLucas Mateus Castro (alqotel) tlb_pid = (tlb->mas1 & MAS1_TID_MASK) >> MAS1_TID_SHIFT; 7445118ebe8SLucas Mateus Castro (alqotel) if (tlb_pid != 0 && tlb_pid != pid) { 7455118ebe8SLucas Mateus Castro (alqotel) return -1; 7465118ebe8SLucas Mateus Castro (alqotel) } 7475118ebe8SLucas Mateus Castro (alqotel) 7485118ebe8SLucas Mateus Castro (alqotel) /* Check effective address */ 7495118ebe8SLucas Mateus Castro (alqotel) if ((address & mask) != (tlb->mas2 & MAS2_EPN_MASK)) { 7505118ebe8SLucas Mateus Castro (alqotel) return -1; 7515118ebe8SLucas Mateus Castro (alqotel) } 7525118ebe8SLucas Mateus Castro (alqotel) 7535118ebe8SLucas Mateus Castro (alqotel) if (raddrp) { 7545118ebe8SLucas Mateus Castro (alqotel) *raddrp = (tlb->mas7_3 & mask) | (address & ~mask); 7555118ebe8SLucas Mateus Castro (alqotel) } 7565118ebe8SLucas Mateus Castro (alqotel) 7575118ebe8SLucas Mateus Castro (alqotel) return 0; 7585118ebe8SLucas Mateus Castro (alqotel) } 7595118ebe8SLucas Mateus Castro (alqotel) 7605118ebe8SLucas Mateus Castro (alqotel) static bool is_epid_mmu(int mmu_idx) 7615118ebe8SLucas Mateus Castro (alqotel) { 7625118ebe8SLucas Mateus Castro (alqotel) return mmu_idx == PPC_TLB_EPID_STORE || mmu_idx == PPC_TLB_EPID_LOAD; 7635118ebe8SLucas Mateus Castro (alqotel) } 7645118ebe8SLucas Mateus Castro (alqotel) 7655118ebe8SLucas Mateus Castro (alqotel) static uint32_t mmubooke206_esr(int mmu_idx, MMUAccessType access_type) 7665118ebe8SLucas Mateus Castro (alqotel) { 7675118ebe8SLucas Mateus Castro (alqotel) uint32_t esr = 0; 7685118ebe8SLucas Mateus Castro (alqotel) if (access_type == MMU_DATA_STORE) { 7695118ebe8SLucas Mateus Castro (alqotel) esr |= ESR_ST; 7705118ebe8SLucas Mateus Castro (alqotel) } 7715118ebe8SLucas Mateus Castro (alqotel) if (is_epid_mmu(mmu_idx)) { 7725118ebe8SLucas Mateus Castro (alqotel) esr |= ESR_EPID; 7735118ebe8SLucas Mateus Castro (alqotel) } 7745118ebe8SLucas Mateus Castro (alqotel) return esr; 7755118ebe8SLucas Mateus Castro (alqotel) } 7765118ebe8SLucas Mateus Castro (alqotel) 7775118ebe8SLucas Mateus Castro (alqotel) /* 7785118ebe8SLucas Mateus Castro (alqotel) * Get EPID register given the mmu_idx. If this is regular load, 7795118ebe8SLucas Mateus Castro (alqotel) * construct the EPID access bits from current processor state 7805118ebe8SLucas Mateus Castro (alqotel) * 7815118ebe8SLucas Mateus Castro (alqotel) * Get the effective AS and PR bits and the PID. The PID is returned 7825118ebe8SLucas Mateus Castro (alqotel) * only if EPID load is requested, otherwise the caller must detect 7835118ebe8SLucas Mateus Castro (alqotel) * the correct EPID. Return true if valid EPID is returned. 7845118ebe8SLucas Mateus Castro (alqotel) */ 7855118ebe8SLucas Mateus Castro (alqotel) static bool mmubooke206_get_as(CPUPPCState *env, 7865118ebe8SLucas Mateus Castro (alqotel) int mmu_idx, uint32_t *epid_out, 7875118ebe8SLucas Mateus Castro (alqotel) bool *as_out, bool *pr_out) 7885118ebe8SLucas Mateus Castro (alqotel) { 7895118ebe8SLucas Mateus Castro (alqotel) if (is_epid_mmu(mmu_idx)) { 7905118ebe8SLucas Mateus Castro (alqotel) uint32_t epidr; 7915118ebe8SLucas Mateus Castro (alqotel) if (mmu_idx == PPC_TLB_EPID_STORE) { 7925118ebe8SLucas Mateus Castro (alqotel) epidr = env->spr[SPR_BOOKE_EPSC]; 7935118ebe8SLucas Mateus Castro (alqotel) } else { 7945118ebe8SLucas Mateus Castro (alqotel) epidr = env->spr[SPR_BOOKE_EPLC]; 7955118ebe8SLucas Mateus Castro (alqotel) } 7965118ebe8SLucas Mateus Castro (alqotel) *epid_out = (epidr & EPID_EPID) >> EPID_EPID_SHIFT; 7975118ebe8SLucas Mateus Castro (alqotel) *as_out = !!(epidr & EPID_EAS); 7985118ebe8SLucas Mateus Castro (alqotel) *pr_out = !!(epidr & EPID_EPR); 7995118ebe8SLucas Mateus Castro (alqotel) return true; 8005118ebe8SLucas Mateus Castro (alqotel) } else { 8015118ebe8SLucas Mateus Castro (alqotel) *as_out = msr_ds; 8025118ebe8SLucas Mateus Castro (alqotel) *pr_out = msr_pr; 8035118ebe8SLucas Mateus Castro (alqotel) return false; 8045118ebe8SLucas Mateus Castro (alqotel) } 8055118ebe8SLucas Mateus Castro (alqotel) } 8065118ebe8SLucas Mateus Castro (alqotel) 8075118ebe8SLucas Mateus Castro (alqotel) /* Check if the tlb found by hashing really matches */ 8085118ebe8SLucas Mateus Castro (alqotel) static int mmubooke206_check_tlb(CPUPPCState *env, ppcmas_tlb_t *tlb, 8095118ebe8SLucas Mateus Castro (alqotel) hwaddr *raddr, int *prot, 8105118ebe8SLucas Mateus Castro (alqotel) target_ulong address, 8115118ebe8SLucas Mateus Castro (alqotel) MMUAccessType access_type, int mmu_idx) 8125118ebe8SLucas Mateus Castro (alqotel) { 8135118ebe8SLucas Mateus Castro (alqotel) int prot2 = 0; 8145118ebe8SLucas Mateus Castro (alqotel) uint32_t epid; 8155118ebe8SLucas Mateus Castro (alqotel) bool as, pr; 8165118ebe8SLucas Mateus Castro (alqotel) bool use_epid = mmubooke206_get_as(env, mmu_idx, &epid, &as, &pr); 8175118ebe8SLucas Mateus Castro (alqotel) 8185118ebe8SLucas Mateus Castro (alqotel) if (!use_epid) { 8195118ebe8SLucas Mateus Castro (alqotel) if (ppcmas_tlb_check(env, tlb, raddr, address, 8205118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_PID]) >= 0) { 8215118ebe8SLucas Mateus Castro (alqotel) goto found_tlb; 8225118ebe8SLucas Mateus Castro (alqotel) } 8235118ebe8SLucas Mateus Castro (alqotel) 8245118ebe8SLucas Mateus Castro (alqotel) if (env->spr[SPR_BOOKE_PID1] && 8255118ebe8SLucas Mateus Castro (alqotel) ppcmas_tlb_check(env, tlb, raddr, address, 8265118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_PID1]) >= 0) { 8275118ebe8SLucas Mateus Castro (alqotel) goto found_tlb; 8285118ebe8SLucas Mateus Castro (alqotel) } 8295118ebe8SLucas Mateus Castro (alqotel) 8305118ebe8SLucas Mateus Castro (alqotel) if (env->spr[SPR_BOOKE_PID2] && 8315118ebe8SLucas Mateus Castro (alqotel) ppcmas_tlb_check(env, tlb, raddr, address, 8325118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_PID2]) >= 0) { 8335118ebe8SLucas Mateus Castro (alqotel) goto found_tlb; 8345118ebe8SLucas Mateus Castro (alqotel) } 8355118ebe8SLucas Mateus Castro (alqotel) } else { 8365118ebe8SLucas Mateus Castro (alqotel) if (ppcmas_tlb_check(env, tlb, raddr, address, epid) >= 0) { 8375118ebe8SLucas Mateus Castro (alqotel) goto found_tlb; 8385118ebe8SLucas Mateus Castro (alqotel) } 8395118ebe8SLucas Mateus Castro (alqotel) } 8405118ebe8SLucas Mateus Castro (alqotel) 8415118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: TLB entry not found\n", __func__); 8425118ebe8SLucas Mateus Castro (alqotel) return -1; 8435118ebe8SLucas Mateus Castro (alqotel) 8445118ebe8SLucas Mateus Castro (alqotel) found_tlb: 8455118ebe8SLucas Mateus Castro (alqotel) 8465118ebe8SLucas Mateus Castro (alqotel) if (pr) { 8475118ebe8SLucas Mateus Castro (alqotel) if (tlb->mas7_3 & MAS3_UR) { 8485118ebe8SLucas Mateus Castro (alqotel) prot2 |= PAGE_READ; 8495118ebe8SLucas Mateus Castro (alqotel) } 8505118ebe8SLucas Mateus Castro (alqotel) if (tlb->mas7_3 & MAS3_UW) { 8515118ebe8SLucas Mateus Castro (alqotel) prot2 |= PAGE_WRITE; 8525118ebe8SLucas Mateus Castro (alqotel) } 8535118ebe8SLucas Mateus Castro (alqotel) if (tlb->mas7_3 & MAS3_UX) { 8545118ebe8SLucas Mateus Castro (alqotel) prot2 |= PAGE_EXEC; 8555118ebe8SLucas Mateus Castro (alqotel) } 8565118ebe8SLucas Mateus Castro (alqotel) } else { 8575118ebe8SLucas Mateus Castro (alqotel) if (tlb->mas7_3 & MAS3_SR) { 8585118ebe8SLucas Mateus Castro (alqotel) prot2 |= PAGE_READ; 8595118ebe8SLucas Mateus Castro (alqotel) } 8605118ebe8SLucas Mateus Castro (alqotel) if (tlb->mas7_3 & MAS3_SW) { 8615118ebe8SLucas Mateus Castro (alqotel) prot2 |= PAGE_WRITE; 8625118ebe8SLucas Mateus Castro (alqotel) } 8635118ebe8SLucas Mateus Castro (alqotel) if (tlb->mas7_3 & MAS3_SX) { 8645118ebe8SLucas Mateus Castro (alqotel) prot2 |= PAGE_EXEC; 8655118ebe8SLucas Mateus Castro (alqotel) } 8665118ebe8SLucas Mateus Castro (alqotel) } 8675118ebe8SLucas Mateus Castro (alqotel) 8685118ebe8SLucas Mateus Castro (alqotel) /* Check the address space and permissions */ 8695118ebe8SLucas Mateus Castro (alqotel) if (access_type == MMU_INST_FETCH) { 8705118ebe8SLucas Mateus Castro (alqotel) /* There is no way to fetch code using epid load */ 8715118ebe8SLucas Mateus Castro (alqotel) assert(!use_epid); 8725118ebe8SLucas Mateus Castro (alqotel) as = msr_ir; 8735118ebe8SLucas Mateus Castro (alqotel) } 8745118ebe8SLucas Mateus Castro (alqotel) 8755118ebe8SLucas Mateus Castro (alqotel) if (as != ((tlb->mas1 & MAS1_TS) >> MAS1_TS_SHIFT)) { 8765118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: AS doesn't match\n", __func__); 8775118ebe8SLucas Mateus Castro (alqotel) return -1; 8785118ebe8SLucas Mateus Castro (alqotel) } 8795118ebe8SLucas Mateus Castro (alqotel) 8805118ebe8SLucas Mateus Castro (alqotel) *prot = prot2; 8815118ebe8SLucas Mateus Castro (alqotel) if (prot2 & prot_for_access_type(access_type)) { 8825118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: good TLB!\n", __func__); 8835118ebe8SLucas Mateus Castro (alqotel) return 0; 8845118ebe8SLucas Mateus Castro (alqotel) } 8855118ebe8SLucas Mateus Castro (alqotel) 8865118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: no prot match: %x\n", __func__, prot2); 8875118ebe8SLucas Mateus Castro (alqotel) return access_type == MMU_INST_FETCH ? -3 : -2; 8885118ebe8SLucas Mateus Castro (alqotel) } 8895118ebe8SLucas Mateus Castro (alqotel) 8905118ebe8SLucas Mateus Castro (alqotel) static int mmubooke206_get_physical_address(CPUPPCState *env, mmu_ctx_t *ctx, 8915118ebe8SLucas Mateus Castro (alqotel) target_ulong address, 8925118ebe8SLucas Mateus Castro (alqotel) MMUAccessType access_type, 8935118ebe8SLucas Mateus Castro (alqotel) int mmu_idx) 8945118ebe8SLucas Mateus Castro (alqotel) { 8955118ebe8SLucas Mateus Castro (alqotel) ppcmas_tlb_t *tlb; 8965118ebe8SLucas Mateus Castro (alqotel) hwaddr raddr; 8975118ebe8SLucas Mateus Castro (alqotel) int i, j, ret; 8985118ebe8SLucas Mateus Castro (alqotel) 8995118ebe8SLucas Mateus Castro (alqotel) ret = -1; 9005118ebe8SLucas Mateus Castro (alqotel) raddr = (hwaddr)-1ULL; 9015118ebe8SLucas Mateus Castro (alqotel) 9025118ebe8SLucas Mateus Castro (alqotel) for (i = 0; i < BOOKE206_MAX_TLBN; i++) { 9035118ebe8SLucas Mateus Castro (alqotel) int ways = booke206_tlb_ways(env, i); 9045118ebe8SLucas Mateus Castro (alqotel) 9055118ebe8SLucas Mateus Castro (alqotel) for (j = 0; j < ways; j++) { 9065118ebe8SLucas Mateus Castro (alqotel) tlb = booke206_get_tlbm(env, i, address, j); 9075118ebe8SLucas Mateus Castro (alqotel) if (!tlb) { 9085118ebe8SLucas Mateus Castro (alqotel) continue; 9095118ebe8SLucas Mateus Castro (alqotel) } 9105118ebe8SLucas Mateus Castro (alqotel) ret = mmubooke206_check_tlb(env, tlb, &raddr, &ctx->prot, address, 9115118ebe8SLucas Mateus Castro (alqotel) access_type, mmu_idx); 9125118ebe8SLucas Mateus Castro (alqotel) if (ret != -1) { 9135118ebe8SLucas Mateus Castro (alqotel) goto found_tlb; 9145118ebe8SLucas Mateus Castro (alqotel) } 9155118ebe8SLucas Mateus Castro (alqotel) } 9165118ebe8SLucas Mateus Castro (alqotel) } 9175118ebe8SLucas Mateus Castro (alqotel) 9185118ebe8SLucas Mateus Castro (alqotel) found_tlb: 9195118ebe8SLucas Mateus Castro (alqotel) 9205118ebe8SLucas Mateus Castro (alqotel) if (ret >= 0) { 9215118ebe8SLucas Mateus Castro (alqotel) ctx->raddr = raddr; 9225118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: access granted " TARGET_FMT_lx " => " TARGET_FMT_plx 9235118ebe8SLucas Mateus Castro (alqotel) " %d %d\n", __func__, address, ctx->raddr, ctx->prot, 9245118ebe8SLucas Mateus Castro (alqotel) ret); 9255118ebe8SLucas Mateus Castro (alqotel) } else { 9265118ebe8SLucas Mateus Castro (alqotel) LOG_SWTLB("%s: access refused " TARGET_FMT_lx " => " TARGET_FMT_plx 9275118ebe8SLucas Mateus Castro (alqotel) " %d %d\n", __func__, address, raddr, ctx->prot, ret); 9285118ebe8SLucas Mateus Castro (alqotel) } 9295118ebe8SLucas Mateus Castro (alqotel) 9305118ebe8SLucas Mateus Castro (alqotel) return ret; 9315118ebe8SLucas Mateus Castro (alqotel) } 9325118ebe8SLucas Mateus Castro (alqotel) 9335118ebe8SLucas Mateus Castro (alqotel) static const char *book3e_tsize_to_str[32] = { 9345118ebe8SLucas Mateus Castro (alqotel) "1K", "2K", "4K", "8K", "16K", "32K", "64K", "128K", "256K", "512K", 9355118ebe8SLucas Mateus Castro (alqotel) "1M", "2M", "4M", "8M", "16M", "32M", "64M", "128M", "256M", "512M", 9365118ebe8SLucas Mateus Castro (alqotel) "1G", "2G", "4G", "8G", "16G", "32G", "64G", "128G", "256G", "512G", 9375118ebe8SLucas Mateus Castro (alqotel) "1T", "2T" 9385118ebe8SLucas Mateus Castro (alqotel) }; 9395118ebe8SLucas Mateus Castro (alqotel) 9405118ebe8SLucas Mateus Castro (alqotel) static void mmubooke_dump_mmu(CPUPPCState *env) 9415118ebe8SLucas Mateus Castro (alqotel) { 9425118ebe8SLucas Mateus Castro (alqotel) ppcemb_tlb_t *entry; 9435118ebe8SLucas Mateus Castro (alqotel) int i; 9445118ebe8SLucas Mateus Castro (alqotel) 9455118ebe8SLucas Mateus Castro (alqotel) if (kvm_enabled() && !env->kvm_sw_tlb) { 9465118ebe8SLucas Mateus Castro (alqotel) qemu_printf("Cannot access KVM TLB\n"); 9475118ebe8SLucas Mateus Castro (alqotel) return; 9485118ebe8SLucas Mateus Castro (alqotel) } 9495118ebe8SLucas Mateus Castro (alqotel) 9505118ebe8SLucas Mateus Castro (alqotel) qemu_printf("\nTLB:\n"); 9515118ebe8SLucas Mateus Castro (alqotel) qemu_printf("Effective Physical Size PID Prot " 9525118ebe8SLucas Mateus Castro (alqotel) "Attr\n"); 9535118ebe8SLucas Mateus Castro (alqotel) 9545118ebe8SLucas Mateus Castro (alqotel) entry = &env->tlb.tlbe[0]; 9555118ebe8SLucas Mateus Castro (alqotel) for (i = 0; i < env->nb_tlb; i++, entry++) { 9565118ebe8SLucas Mateus Castro (alqotel) hwaddr ea, pa; 9575118ebe8SLucas Mateus Castro (alqotel) target_ulong mask; 9585118ebe8SLucas Mateus Castro (alqotel) uint64_t size = (uint64_t)entry->size; 9595118ebe8SLucas Mateus Castro (alqotel) char size_buf[20]; 9605118ebe8SLucas Mateus Castro (alqotel) 9615118ebe8SLucas Mateus Castro (alqotel) /* Check valid flag */ 9625118ebe8SLucas Mateus Castro (alqotel) if (!(entry->prot & PAGE_VALID)) { 9635118ebe8SLucas Mateus Castro (alqotel) continue; 9645118ebe8SLucas Mateus Castro (alqotel) } 9655118ebe8SLucas Mateus Castro (alqotel) 9665118ebe8SLucas Mateus Castro (alqotel) mask = ~(entry->size - 1); 9675118ebe8SLucas Mateus Castro (alqotel) ea = entry->EPN & mask; 9685118ebe8SLucas Mateus Castro (alqotel) pa = entry->RPN & mask; 9695118ebe8SLucas Mateus Castro (alqotel) /* Extend the physical address to 36 bits */ 9705118ebe8SLucas Mateus Castro (alqotel) pa |= (hwaddr)(entry->RPN & 0xF) << 32; 9715118ebe8SLucas Mateus Castro (alqotel) if (size >= 1 * MiB) { 9725118ebe8SLucas Mateus Castro (alqotel) snprintf(size_buf, sizeof(size_buf), "%3" PRId64 "M", size / MiB); 9735118ebe8SLucas Mateus Castro (alqotel) } else { 9745118ebe8SLucas Mateus Castro (alqotel) snprintf(size_buf, sizeof(size_buf), "%3" PRId64 "k", size / KiB); 9755118ebe8SLucas Mateus Castro (alqotel) } 9765118ebe8SLucas Mateus Castro (alqotel) qemu_printf("0x%016" PRIx64 " 0x%016" PRIx64 " %s %-5u %08x %08x\n", 9775118ebe8SLucas Mateus Castro (alqotel) (uint64_t)ea, (uint64_t)pa, size_buf, (uint32_t)entry->PID, 9785118ebe8SLucas Mateus Castro (alqotel) entry->prot, entry->attr); 9795118ebe8SLucas Mateus Castro (alqotel) } 9805118ebe8SLucas Mateus Castro (alqotel) 9815118ebe8SLucas Mateus Castro (alqotel) } 9825118ebe8SLucas Mateus Castro (alqotel) 9835118ebe8SLucas Mateus Castro (alqotel) static void mmubooke206_dump_one_tlb(CPUPPCState *env, int tlbn, int offset, 9845118ebe8SLucas Mateus Castro (alqotel) int tlbsize) 9855118ebe8SLucas Mateus Castro (alqotel) { 9865118ebe8SLucas Mateus Castro (alqotel) ppcmas_tlb_t *entry; 9875118ebe8SLucas Mateus Castro (alqotel) int i; 9885118ebe8SLucas Mateus Castro (alqotel) 9895118ebe8SLucas Mateus Castro (alqotel) qemu_printf("\nTLB%d:\n", tlbn); 9905118ebe8SLucas Mateus Castro (alqotel) qemu_printf("Effective Physical Size TID TS SRWX" 9915118ebe8SLucas Mateus Castro (alqotel) " URWX WIMGE U0123\n"); 9925118ebe8SLucas Mateus Castro (alqotel) 9935118ebe8SLucas Mateus Castro (alqotel) entry = &env->tlb.tlbm[offset]; 9945118ebe8SLucas Mateus Castro (alqotel) for (i = 0; i < tlbsize; i++, entry++) { 9955118ebe8SLucas Mateus Castro (alqotel) hwaddr ea, pa, size; 9965118ebe8SLucas Mateus Castro (alqotel) int tsize; 9975118ebe8SLucas Mateus Castro (alqotel) 9985118ebe8SLucas Mateus Castro (alqotel) if (!(entry->mas1 & MAS1_VALID)) { 9995118ebe8SLucas Mateus Castro (alqotel) continue; 10005118ebe8SLucas Mateus Castro (alqotel) } 10015118ebe8SLucas Mateus Castro (alqotel) 10025118ebe8SLucas Mateus Castro (alqotel) tsize = (entry->mas1 & MAS1_TSIZE_MASK) >> MAS1_TSIZE_SHIFT; 10035118ebe8SLucas Mateus Castro (alqotel) size = 1024ULL << tsize; 10045118ebe8SLucas Mateus Castro (alqotel) ea = entry->mas2 & ~(size - 1); 10055118ebe8SLucas Mateus Castro (alqotel) pa = entry->mas7_3 & ~(size - 1); 10065118ebe8SLucas Mateus Castro (alqotel) 10075118ebe8SLucas Mateus Castro (alqotel) qemu_printf("0x%016" PRIx64 " 0x%016" PRIx64 " %4s %-5u %1u S%c%c%c" 10085118ebe8SLucas Mateus Castro (alqotel) "U%c%c%c %c%c%c%c%c U%c%c%c%c\n", 10095118ebe8SLucas Mateus Castro (alqotel) (uint64_t)ea, (uint64_t)pa, 10105118ebe8SLucas Mateus Castro (alqotel) book3e_tsize_to_str[tsize], 10115118ebe8SLucas Mateus Castro (alqotel) (entry->mas1 & MAS1_TID_MASK) >> MAS1_TID_SHIFT, 10125118ebe8SLucas Mateus Castro (alqotel) (entry->mas1 & MAS1_TS) >> MAS1_TS_SHIFT, 10135118ebe8SLucas Mateus Castro (alqotel) entry->mas7_3 & MAS3_SR ? 'R' : '-', 10145118ebe8SLucas Mateus Castro (alqotel) entry->mas7_3 & MAS3_SW ? 'W' : '-', 10155118ebe8SLucas Mateus Castro (alqotel) entry->mas7_3 & MAS3_SX ? 'X' : '-', 10165118ebe8SLucas Mateus Castro (alqotel) entry->mas7_3 & MAS3_UR ? 'R' : '-', 10175118ebe8SLucas Mateus Castro (alqotel) entry->mas7_3 & MAS3_UW ? 'W' : '-', 10185118ebe8SLucas Mateus Castro (alqotel) entry->mas7_3 & MAS3_UX ? 'X' : '-', 10195118ebe8SLucas Mateus Castro (alqotel) entry->mas2 & MAS2_W ? 'W' : '-', 10205118ebe8SLucas Mateus Castro (alqotel) entry->mas2 & MAS2_I ? 'I' : '-', 10215118ebe8SLucas Mateus Castro (alqotel) entry->mas2 & MAS2_M ? 'M' : '-', 10225118ebe8SLucas Mateus Castro (alqotel) entry->mas2 & MAS2_G ? 'G' : '-', 10235118ebe8SLucas Mateus Castro (alqotel) entry->mas2 & MAS2_E ? 'E' : '-', 10245118ebe8SLucas Mateus Castro (alqotel) entry->mas7_3 & MAS3_U0 ? '0' : '-', 10255118ebe8SLucas Mateus Castro (alqotel) entry->mas7_3 & MAS3_U1 ? '1' : '-', 10265118ebe8SLucas Mateus Castro (alqotel) entry->mas7_3 & MAS3_U2 ? '2' : '-', 10275118ebe8SLucas Mateus Castro (alqotel) entry->mas7_3 & MAS3_U3 ? '3' : '-'); 10285118ebe8SLucas Mateus Castro (alqotel) } 10295118ebe8SLucas Mateus Castro (alqotel) } 10305118ebe8SLucas Mateus Castro (alqotel) 10315118ebe8SLucas Mateus Castro (alqotel) static void mmubooke206_dump_mmu(CPUPPCState *env) 10325118ebe8SLucas Mateus Castro (alqotel) { 10335118ebe8SLucas Mateus Castro (alqotel) int offset = 0; 10345118ebe8SLucas Mateus Castro (alqotel) int i; 10355118ebe8SLucas Mateus Castro (alqotel) 10365118ebe8SLucas Mateus Castro (alqotel) if (kvm_enabled() && !env->kvm_sw_tlb) { 10375118ebe8SLucas Mateus Castro (alqotel) qemu_printf("Cannot access KVM TLB\n"); 10385118ebe8SLucas Mateus Castro (alqotel) return; 10395118ebe8SLucas Mateus Castro (alqotel) } 10405118ebe8SLucas Mateus Castro (alqotel) 10415118ebe8SLucas Mateus Castro (alqotel) for (i = 0; i < BOOKE206_MAX_TLBN; i++) { 10425118ebe8SLucas Mateus Castro (alqotel) int size = booke206_tlb_size(env, i); 10435118ebe8SLucas Mateus Castro (alqotel) 10445118ebe8SLucas Mateus Castro (alqotel) if (size == 0) { 10455118ebe8SLucas Mateus Castro (alqotel) continue; 10465118ebe8SLucas Mateus Castro (alqotel) } 10475118ebe8SLucas Mateus Castro (alqotel) 10485118ebe8SLucas Mateus Castro (alqotel) mmubooke206_dump_one_tlb(env, i, offset, size); 10495118ebe8SLucas Mateus Castro (alqotel) offset += size; 10505118ebe8SLucas Mateus Castro (alqotel) } 10515118ebe8SLucas Mateus Castro (alqotel) } 10525118ebe8SLucas Mateus Castro (alqotel) 10535118ebe8SLucas Mateus Castro (alqotel) static void mmu6xx_dump_BATs(CPUPPCState *env, int type) 10545118ebe8SLucas Mateus Castro (alqotel) { 10555118ebe8SLucas Mateus Castro (alqotel) target_ulong *BATlt, *BATut, *BATu, *BATl; 10565118ebe8SLucas Mateus Castro (alqotel) target_ulong BEPIl, BEPIu, bl; 10575118ebe8SLucas Mateus Castro (alqotel) int i; 10585118ebe8SLucas Mateus Castro (alqotel) 10595118ebe8SLucas Mateus Castro (alqotel) switch (type) { 10605118ebe8SLucas Mateus Castro (alqotel) case ACCESS_CODE: 10615118ebe8SLucas Mateus Castro (alqotel) BATlt = env->IBAT[1]; 10625118ebe8SLucas Mateus Castro (alqotel) BATut = env->IBAT[0]; 10635118ebe8SLucas Mateus Castro (alqotel) break; 10645118ebe8SLucas Mateus Castro (alqotel) default: 10655118ebe8SLucas Mateus Castro (alqotel) BATlt = env->DBAT[1]; 10665118ebe8SLucas Mateus Castro (alqotel) BATut = env->DBAT[0]; 10675118ebe8SLucas Mateus Castro (alqotel) break; 10685118ebe8SLucas Mateus Castro (alqotel) } 10695118ebe8SLucas Mateus Castro (alqotel) 10705118ebe8SLucas Mateus Castro (alqotel) for (i = 0; i < env->nb_BATs; i++) { 10715118ebe8SLucas Mateus Castro (alqotel) BATu = &BATut[i]; 10725118ebe8SLucas Mateus Castro (alqotel) BATl = &BATlt[i]; 10735118ebe8SLucas Mateus Castro (alqotel) BEPIu = *BATu & 0xF0000000; 10745118ebe8SLucas Mateus Castro (alqotel) BEPIl = *BATu & 0x0FFE0000; 10755118ebe8SLucas Mateus Castro (alqotel) bl = (*BATu & 0x00001FFC) << 15; 10765118ebe8SLucas Mateus Castro (alqotel) qemu_printf("%s BAT%d BATu " TARGET_FMT_lx 10775118ebe8SLucas Mateus Castro (alqotel) " BATl " TARGET_FMT_lx "\n\t" TARGET_FMT_lx " " 10785118ebe8SLucas Mateus Castro (alqotel) TARGET_FMT_lx " " TARGET_FMT_lx "\n", 10795118ebe8SLucas Mateus Castro (alqotel) type == ACCESS_CODE ? "code" : "data", i, 10805118ebe8SLucas Mateus Castro (alqotel) *BATu, *BATl, BEPIu, BEPIl, bl); 10815118ebe8SLucas Mateus Castro (alqotel) } 10825118ebe8SLucas Mateus Castro (alqotel) } 10835118ebe8SLucas Mateus Castro (alqotel) 10845118ebe8SLucas Mateus Castro (alqotel) static void mmu6xx_dump_mmu(CPUPPCState *env) 10855118ebe8SLucas Mateus Castro (alqotel) { 10865118ebe8SLucas Mateus Castro (alqotel) PowerPCCPU *cpu = env_archcpu(env); 10875118ebe8SLucas Mateus Castro (alqotel) ppc6xx_tlb_t *tlb; 10885118ebe8SLucas Mateus Castro (alqotel) target_ulong sr; 10895118ebe8SLucas Mateus Castro (alqotel) int type, way, entry, i; 10905118ebe8SLucas Mateus Castro (alqotel) 10915118ebe8SLucas Mateus Castro (alqotel) qemu_printf("HTAB base = 0x%"HWADDR_PRIx"\n", ppc_hash32_hpt_base(cpu)); 10925118ebe8SLucas Mateus Castro (alqotel) qemu_printf("HTAB mask = 0x%"HWADDR_PRIx"\n", ppc_hash32_hpt_mask(cpu)); 10935118ebe8SLucas Mateus Castro (alqotel) 10945118ebe8SLucas Mateus Castro (alqotel) qemu_printf("\nSegment registers:\n"); 10955118ebe8SLucas Mateus Castro (alqotel) for (i = 0; i < 32; i++) { 10965118ebe8SLucas Mateus Castro (alqotel) sr = env->sr[i]; 10975118ebe8SLucas Mateus Castro (alqotel) if (sr & 0x80000000) { 10985118ebe8SLucas Mateus Castro (alqotel) qemu_printf("%02d T=%d Ks=%d Kp=%d BUID=0x%03x " 10995118ebe8SLucas Mateus Castro (alqotel) "CNTLR_SPEC=0x%05x\n", i, 11005118ebe8SLucas Mateus Castro (alqotel) sr & 0x80000000 ? 1 : 0, sr & 0x40000000 ? 1 : 0, 11015118ebe8SLucas Mateus Castro (alqotel) sr & 0x20000000 ? 1 : 0, (uint32_t)((sr >> 20) & 0x1FF), 11025118ebe8SLucas Mateus Castro (alqotel) (uint32_t)(sr & 0xFFFFF)); 11035118ebe8SLucas Mateus Castro (alqotel) } else { 11045118ebe8SLucas Mateus Castro (alqotel) qemu_printf("%02d T=%d Ks=%d Kp=%d N=%d VSID=0x%06x\n", i, 11055118ebe8SLucas Mateus Castro (alqotel) sr & 0x80000000 ? 1 : 0, sr & 0x40000000 ? 1 : 0, 11065118ebe8SLucas Mateus Castro (alqotel) sr & 0x20000000 ? 1 : 0, sr & 0x10000000 ? 1 : 0, 11075118ebe8SLucas Mateus Castro (alqotel) (uint32_t)(sr & 0x00FFFFFF)); 11085118ebe8SLucas Mateus Castro (alqotel) } 11095118ebe8SLucas Mateus Castro (alqotel) } 11105118ebe8SLucas Mateus Castro (alqotel) 11115118ebe8SLucas Mateus Castro (alqotel) qemu_printf("\nBATs:\n"); 11125118ebe8SLucas Mateus Castro (alqotel) mmu6xx_dump_BATs(env, ACCESS_INT); 11135118ebe8SLucas Mateus Castro (alqotel) mmu6xx_dump_BATs(env, ACCESS_CODE); 11145118ebe8SLucas Mateus Castro (alqotel) 11155118ebe8SLucas Mateus Castro (alqotel) if (env->id_tlbs != 1) { 11165118ebe8SLucas Mateus Castro (alqotel) qemu_printf("ERROR: 6xx MMU should have separated TLB" 11175118ebe8SLucas Mateus Castro (alqotel) " for code and data\n"); 11185118ebe8SLucas Mateus Castro (alqotel) } 11195118ebe8SLucas Mateus Castro (alqotel) 11205118ebe8SLucas Mateus Castro (alqotel) qemu_printf("\nTLBs [EPN EPN + SIZE]\n"); 11215118ebe8SLucas Mateus Castro (alqotel) 11225118ebe8SLucas Mateus Castro (alqotel) for (type = 0; type < 2; type++) { 11235118ebe8SLucas Mateus Castro (alqotel) for (way = 0; way < env->nb_ways; way++) { 11245118ebe8SLucas Mateus Castro (alqotel) for (entry = env->nb_tlb * type + env->tlb_per_way * way; 11255118ebe8SLucas Mateus Castro (alqotel) entry < (env->nb_tlb * type + env->tlb_per_way * (way + 1)); 11265118ebe8SLucas Mateus Castro (alqotel) entry++) { 11275118ebe8SLucas Mateus Castro (alqotel) 11285118ebe8SLucas Mateus Castro (alqotel) tlb = &env->tlb.tlb6[entry]; 11295118ebe8SLucas Mateus Castro (alqotel) qemu_printf("%s TLB %02d/%02d way:%d %s [" 11305118ebe8SLucas Mateus Castro (alqotel) TARGET_FMT_lx " " TARGET_FMT_lx "]\n", 11315118ebe8SLucas Mateus Castro (alqotel) type ? "code" : "data", entry % env->nb_tlb, 11325118ebe8SLucas Mateus Castro (alqotel) env->nb_tlb, way, 11335118ebe8SLucas Mateus Castro (alqotel) pte_is_valid(tlb->pte0) ? "valid" : "inval", 11345118ebe8SLucas Mateus Castro (alqotel) tlb->EPN, tlb->EPN + TARGET_PAGE_SIZE); 11355118ebe8SLucas Mateus Castro (alqotel) } 11365118ebe8SLucas Mateus Castro (alqotel) } 11375118ebe8SLucas Mateus Castro (alqotel) } 11385118ebe8SLucas Mateus Castro (alqotel) } 11395118ebe8SLucas Mateus Castro (alqotel) 11405118ebe8SLucas Mateus Castro (alqotel) void dump_mmu(CPUPPCState *env) 11415118ebe8SLucas Mateus Castro (alqotel) { 11425118ebe8SLucas Mateus Castro (alqotel) switch (env->mmu_model) { 11435118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_BOOKE: 11445118ebe8SLucas Mateus Castro (alqotel) mmubooke_dump_mmu(env); 11455118ebe8SLucas Mateus Castro (alqotel) break; 11465118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_BOOKE206: 11475118ebe8SLucas Mateus Castro (alqotel) mmubooke206_dump_mmu(env); 11485118ebe8SLucas Mateus Castro (alqotel) break; 11495118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_SOFT_6xx: 11505118ebe8SLucas Mateus Castro (alqotel) mmu6xx_dump_mmu(env); 11515118ebe8SLucas Mateus Castro (alqotel) break; 11525118ebe8SLucas Mateus Castro (alqotel) #if defined(TARGET_PPC64) 11535118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_64B: 11545118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_2_03: 11555118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_2_06: 11565118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_2_07: 11575118ebe8SLucas Mateus Castro (alqotel) dump_slb(env_archcpu(env)); 11585118ebe8SLucas Mateus Castro (alqotel) break; 11595118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_3_00: 11605118ebe8SLucas Mateus Castro (alqotel) if (ppc64_v3_radix(env_archcpu(env))) { 11615118ebe8SLucas Mateus Castro (alqotel) qemu_log_mask(LOG_UNIMP, "%s: the PPC64 MMU is unsupported\n", 11625118ebe8SLucas Mateus Castro (alqotel) __func__); 11635118ebe8SLucas Mateus Castro (alqotel) } else { 11645118ebe8SLucas Mateus Castro (alqotel) dump_slb(env_archcpu(env)); 11655118ebe8SLucas Mateus Castro (alqotel) } 11665118ebe8SLucas Mateus Castro (alqotel) break; 11675118ebe8SLucas Mateus Castro (alqotel) #endif 11685118ebe8SLucas Mateus Castro (alqotel) default: 11695118ebe8SLucas Mateus Castro (alqotel) qemu_log_mask(LOG_UNIMP, "%s: unimplemented\n", __func__); 11705118ebe8SLucas Mateus Castro (alqotel) } 11715118ebe8SLucas Mateus Castro (alqotel) } 11725118ebe8SLucas Mateus Castro (alqotel) 11735118ebe8SLucas Mateus Castro (alqotel) static int check_physical(CPUPPCState *env, mmu_ctx_t *ctx, target_ulong eaddr, 11745118ebe8SLucas Mateus Castro (alqotel) MMUAccessType access_type) 11755118ebe8SLucas Mateus Castro (alqotel) { 11765118ebe8SLucas Mateus Castro (alqotel) ctx->raddr = eaddr; 11775118ebe8SLucas Mateus Castro (alqotel) ctx->prot = PAGE_READ | PAGE_EXEC; 1178*c8f49e6bSCédric Le Goater 11795118ebe8SLucas Mateus Castro (alqotel) switch (env->mmu_model) { 11805118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_SOFT_6xx: 11815118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_SOFT_4xx: 11825118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_REAL: 11835118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_BOOKE: 11845118ebe8SLucas Mateus Castro (alqotel) ctx->prot |= PAGE_WRITE; 11855118ebe8SLucas Mateus Castro (alqotel) break; 11865118ebe8SLucas Mateus Castro (alqotel) 11875118ebe8SLucas Mateus Castro (alqotel) default: 11885118ebe8SLucas Mateus Castro (alqotel) /* Caller's checks mean we should never get here for other models */ 1189*c8f49e6bSCédric Le Goater g_assert_not_reached(); 11905118ebe8SLucas Mateus Castro (alqotel) } 11915118ebe8SLucas Mateus Castro (alqotel) 1192*c8f49e6bSCédric Le Goater return 0; 11935118ebe8SLucas Mateus Castro (alqotel) } 11945118ebe8SLucas Mateus Castro (alqotel) 11955118ebe8SLucas Mateus Castro (alqotel) int get_physical_address_wtlb(CPUPPCState *env, mmu_ctx_t *ctx, 11965118ebe8SLucas Mateus Castro (alqotel) target_ulong eaddr, 11975118ebe8SLucas Mateus Castro (alqotel) MMUAccessType access_type, int type, 11985118ebe8SLucas Mateus Castro (alqotel) int mmu_idx) 11995118ebe8SLucas Mateus Castro (alqotel) { 12005118ebe8SLucas Mateus Castro (alqotel) int ret = -1; 12015118ebe8SLucas Mateus Castro (alqotel) bool real_mode = (type == ACCESS_CODE && msr_ir == 0) 12025118ebe8SLucas Mateus Castro (alqotel) || (type != ACCESS_CODE && msr_dr == 0); 12035118ebe8SLucas Mateus Castro (alqotel) 12045118ebe8SLucas Mateus Castro (alqotel) switch (env->mmu_model) { 12055118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_SOFT_6xx: 12065118ebe8SLucas Mateus Castro (alqotel) if (real_mode) { 12075118ebe8SLucas Mateus Castro (alqotel) ret = check_physical(env, ctx, eaddr, access_type); 12085118ebe8SLucas Mateus Castro (alqotel) } else { 12095118ebe8SLucas Mateus Castro (alqotel) /* Try to find a BAT */ 12105118ebe8SLucas Mateus Castro (alqotel) if (env->nb_BATs != 0) { 12115118ebe8SLucas Mateus Castro (alqotel) ret = get_bat_6xx_tlb(env, ctx, eaddr, access_type); 12125118ebe8SLucas Mateus Castro (alqotel) } 12135118ebe8SLucas Mateus Castro (alqotel) if (ret < 0) { 12145118ebe8SLucas Mateus Castro (alqotel) /* We didn't match any BAT entry or don't have BATs */ 12155118ebe8SLucas Mateus Castro (alqotel) ret = get_segment_6xx_tlb(env, ctx, eaddr, access_type, type); 12165118ebe8SLucas Mateus Castro (alqotel) } 12175118ebe8SLucas Mateus Castro (alqotel) } 12185118ebe8SLucas Mateus Castro (alqotel) break; 12195118ebe8SLucas Mateus Castro (alqotel) 12205118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_SOFT_4xx: 12215118ebe8SLucas Mateus Castro (alqotel) if (real_mode) { 12225118ebe8SLucas Mateus Castro (alqotel) ret = check_physical(env, ctx, eaddr, access_type); 12235118ebe8SLucas Mateus Castro (alqotel) } else { 12245118ebe8SLucas Mateus Castro (alqotel) ret = mmu40x_get_physical_address(env, ctx, eaddr, access_type); 12255118ebe8SLucas Mateus Castro (alqotel) } 12265118ebe8SLucas Mateus Castro (alqotel) break; 12275118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_BOOKE: 12285118ebe8SLucas Mateus Castro (alqotel) ret = mmubooke_get_physical_address(env, ctx, eaddr, access_type); 12295118ebe8SLucas Mateus Castro (alqotel) break; 12305118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_BOOKE206: 12315118ebe8SLucas Mateus Castro (alqotel) ret = mmubooke206_get_physical_address(env, ctx, eaddr, access_type, 12325118ebe8SLucas Mateus Castro (alqotel) mmu_idx); 12335118ebe8SLucas Mateus Castro (alqotel) break; 12345118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_MPC8xx: 12355118ebe8SLucas Mateus Castro (alqotel) /* XXX: TODO */ 12365118ebe8SLucas Mateus Castro (alqotel) cpu_abort(env_cpu(env), "MPC8xx MMU model is not implemented\n"); 12375118ebe8SLucas Mateus Castro (alqotel) break; 12385118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_REAL: 12395118ebe8SLucas Mateus Castro (alqotel) if (real_mode) { 12405118ebe8SLucas Mateus Castro (alqotel) ret = check_physical(env, ctx, eaddr, access_type); 12415118ebe8SLucas Mateus Castro (alqotel) } else { 12425118ebe8SLucas Mateus Castro (alqotel) cpu_abort(env_cpu(env), 12435118ebe8SLucas Mateus Castro (alqotel) "PowerPC in real mode do not do any translation\n"); 12445118ebe8SLucas Mateus Castro (alqotel) } 12455118ebe8SLucas Mateus Castro (alqotel) return -1; 12465118ebe8SLucas Mateus Castro (alqotel) default: 12475118ebe8SLucas Mateus Castro (alqotel) cpu_abort(env_cpu(env), "Unknown or invalid MMU model\n"); 12485118ebe8SLucas Mateus Castro (alqotel) return -1; 12495118ebe8SLucas Mateus Castro (alqotel) } 12505118ebe8SLucas Mateus Castro (alqotel) 12515118ebe8SLucas Mateus Castro (alqotel) return ret; 12525118ebe8SLucas Mateus Castro (alqotel) } 12535118ebe8SLucas Mateus Castro (alqotel) 12545118ebe8SLucas Mateus Castro (alqotel) static void booke206_update_mas_tlb_miss(CPUPPCState *env, target_ulong address, 12555118ebe8SLucas Mateus Castro (alqotel) MMUAccessType access_type, int mmu_idx) 12565118ebe8SLucas Mateus Castro (alqotel) { 12575118ebe8SLucas Mateus Castro (alqotel) uint32_t epid; 12585118ebe8SLucas Mateus Castro (alqotel) bool as, pr; 12595118ebe8SLucas Mateus Castro (alqotel) uint32_t missed_tid = 0; 12605118ebe8SLucas Mateus Castro (alqotel) bool use_epid = mmubooke206_get_as(env, mmu_idx, &epid, &as, &pr); 12615118ebe8SLucas Mateus Castro (alqotel) 12625118ebe8SLucas Mateus Castro (alqotel) if (access_type == MMU_INST_FETCH) { 12635118ebe8SLucas Mateus Castro (alqotel) as = msr_ir; 12645118ebe8SLucas Mateus Castro (alqotel) } 12655118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_MAS0] = env->spr[SPR_BOOKE_MAS4] & MAS4_TLBSELD_MASK; 12665118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_MAS1] = env->spr[SPR_BOOKE_MAS4] & MAS4_TSIZED_MASK; 12675118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_MAS2] = env->spr[SPR_BOOKE_MAS4] & MAS4_WIMGED_MASK; 12685118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_MAS3] = 0; 12695118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_MAS6] = 0; 12705118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_MAS7] = 0; 12715118ebe8SLucas Mateus Castro (alqotel) 12725118ebe8SLucas Mateus Castro (alqotel) /* AS */ 12735118ebe8SLucas Mateus Castro (alqotel) if (as) { 12745118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_MAS1] |= MAS1_TS; 12755118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_MAS6] |= MAS6_SAS; 12765118ebe8SLucas Mateus Castro (alqotel) } 12775118ebe8SLucas Mateus Castro (alqotel) 12785118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_MAS1] |= MAS1_VALID; 12795118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_MAS2] |= address & MAS2_EPN_MASK; 12805118ebe8SLucas Mateus Castro (alqotel) 12815118ebe8SLucas Mateus Castro (alqotel) if (!use_epid) { 12825118ebe8SLucas Mateus Castro (alqotel) switch (env->spr[SPR_BOOKE_MAS4] & MAS4_TIDSELD_PIDZ) { 12835118ebe8SLucas Mateus Castro (alqotel) case MAS4_TIDSELD_PID0: 12845118ebe8SLucas Mateus Castro (alqotel) missed_tid = env->spr[SPR_BOOKE_PID]; 12855118ebe8SLucas Mateus Castro (alqotel) break; 12865118ebe8SLucas Mateus Castro (alqotel) case MAS4_TIDSELD_PID1: 12875118ebe8SLucas Mateus Castro (alqotel) missed_tid = env->spr[SPR_BOOKE_PID1]; 12885118ebe8SLucas Mateus Castro (alqotel) break; 12895118ebe8SLucas Mateus Castro (alqotel) case MAS4_TIDSELD_PID2: 12905118ebe8SLucas Mateus Castro (alqotel) missed_tid = env->spr[SPR_BOOKE_PID2]; 12915118ebe8SLucas Mateus Castro (alqotel) break; 12925118ebe8SLucas Mateus Castro (alqotel) } 12935118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_MAS6] |= env->spr[SPR_BOOKE_PID] << 16; 12945118ebe8SLucas Mateus Castro (alqotel) } else { 12955118ebe8SLucas Mateus Castro (alqotel) missed_tid = epid; 12965118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_MAS6] |= missed_tid << 16; 12975118ebe8SLucas Mateus Castro (alqotel) } 12985118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_MAS1] |= (missed_tid << MAS1_TID_SHIFT); 12995118ebe8SLucas Mateus Castro (alqotel) 13005118ebe8SLucas Mateus Castro (alqotel) 13015118ebe8SLucas Mateus Castro (alqotel) /* next victim logic */ 13025118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_MAS0] |= env->last_way << MAS0_ESEL_SHIFT; 13035118ebe8SLucas Mateus Castro (alqotel) env->last_way++; 13045118ebe8SLucas Mateus Castro (alqotel) env->last_way &= booke206_tlb_ways(env, 0) - 1; 13055118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_MAS0] |= env->last_way << MAS0_NV_SHIFT; 13065118ebe8SLucas Mateus Castro (alqotel) } 13075118ebe8SLucas Mateus Castro (alqotel) 13085118ebe8SLucas Mateus Castro (alqotel) /* Perform address translation */ 13095118ebe8SLucas Mateus Castro (alqotel) /* TODO: Split this by mmu_model. */ 13105118ebe8SLucas Mateus Castro (alqotel) static bool ppc_jumbo_xlate(PowerPCCPU *cpu, vaddr eaddr, 13115118ebe8SLucas Mateus Castro (alqotel) MMUAccessType access_type, 13125118ebe8SLucas Mateus Castro (alqotel) hwaddr *raddrp, int *psizep, int *protp, 13135118ebe8SLucas Mateus Castro (alqotel) int mmu_idx, bool guest_visible) 13145118ebe8SLucas Mateus Castro (alqotel) { 13155118ebe8SLucas Mateus Castro (alqotel) CPUState *cs = CPU(cpu); 13165118ebe8SLucas Mateus Castro (alqotel) CPUPPCState *env = &cpu->env; 13175118ebe8SLucas Mateus Castro (alqotel) mmu_ctx_t ctx; 13185118ebe8SLucas Mateus Castro (alqotel) int type; 13195118ebe8SLucas Mateus Castro (alqotel) int ret; 13205118ebe8SLucas Mateus Castro (alqotel) 13215118ebe8SLucas Mateus Castro (alqotel) if (access_type == MMU_INST_FETCH) { 13225118ebe8SLucas Mateus Castro (alqotel) /* code access */ 13235118ebe8SLucas Mateus Castro (alqotel) type = ACCESS_CODE; 13245118ebe8SLucas Mateus Castro (alqotel) } else if (guest_visible) { 13255118ebe8SLucas Mateus Castro (alqotel) /* data access */ 13265118ebe8SLucas Mateus Castro (alqotel) type = env->access_type; 13275118ebe8SLucas Mateus Castro (alqotel) } else { 13285118ebe8SLucas Mateus Castro (alqotel) type = ACCESS_INT; 13295118ebe8SLucas Mateus Castro (alqotel) } 13305118ebe8SLucas Mateus Castro (alqotel) 13315118ebe8SLucas Mateus Castro (alqotel) ret = get_physical_address_wtlb(env, &ctx, eaddr, access_type, 13325118ebe8SLucas Mateus Castro (alqotel) type, mmu_idx); 13335118ebe8SLucas Mateus Castro (alqotel) if (ret == 0) { 13345118ebe8SLucas Mateus Castro (alqotel) *raddrp = ctx.raddr; 13355118ebe8SLucas Mateus Castro (alqotel) *protp = ctx.prot; 13365118ebe8SLucas Mateus Castro (alqotel) *psizep = TARGET_PAGE_BITS; 13375118ebe8SLucas Mateus Castro (alqotel) return true; 13385118ebe8SLucas Mateus Castro (alqotel) } 13395118ebe8SLucas Mateus Castro (alqotel) 13405118ebe8SLucas Mateus Castro (alqotel) if (guest_visible) { 13415118ebe8SLucas Mateus Castro (alqotel) LOG_MMU_STATE(cs); 13425118ebe8SLucas Mateus Castro (alqotel) if (type == ACCESS_CODE) { 13435118ebe8SLucas Mateus Castro (alqotel) switch (ret) { 13445118ebe8SLucas Mateus Castro (alqotel) case -1: 13455118ebe8SLucas Mateus Castro (alqotel) /* No matches in page tables or TLB */ 13465118ebe8SLucas Mateus Castro (alqotel) switch (env->mmu_model) { 13475118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_SOFT_6xx: 13485118ebe8SLucas Mateus Castro (alqotel) cs->exception_index = POWERPC_EXCP_IFTLB; 13495118ebe8SLucas Mateus Castro (alqotel) env->error_code = 1 << 18; 13505118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_IMISS] = eaddr; 13515118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_ICMP] = 0x80000000 | ctx.ptem; 13525118ebe8SLucas Mateus Castro (alqotel) goto tlb_miss; 13535118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_SOFT_4xx: 13545118ebe8SLucas Mateus Castro (alqotel) cs->exception_index = POWERPC_EXCP_ITLB; 13555118ebe8SLucas Mateus Castro (alqotel) env->error_code = 0; 13565118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_40x_DEAR] = eaddr; 13575118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_40x_ESR] = 0x00000000; 13585118ebe8SLucas Mateus Castro (alqotel) break; 13595118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_BOOKE206: 13605118ebe8SLucas Mateus Castro (alqotel) booke206_update_mas_tlb_miss(env, eaddr, 2, mmu_idx); 13615118ebe8SLucas Mateus Castro (alqotel) /* fall through */ 13625118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_BOOKE: 13635118ebe8SLucas Mateus Castro (alqotel) cs->exception_index = POWERPC_EXCP_ITLB; 13645118ebe8SLucas Mateus Castro (alqotel) env->error_code = 0; 13655118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_DEAR] = eaddr; 13665118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_ESR] = mmubooke206_esr(mmu_idx, MMU_DATA_LOAD); 13675118ebe8SLucas Mateus Castro (alqotel) break; 13685118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_MPC8xx: 13695118ebe8SLucas Mateus Castro (alqotel) cpu_abort(cs, "MPC8xx MMU model is not implemented\n"); 13705118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_REAL: 13715118ebe8SLucas Mateus Castro (alqotel) cpu_abort(cs, "PowerPC in real mode should never raise " 13725118ebe8SLucas Mateus Castro (alqotel) "any MMU exceptions\n"); 13735118ebe8SLucas Mateus Castro (alqotel) default: 13745118ebe8SLucas Mateus Castro (alqotel) cpu_abort(cs, "Unknown or invalid MMU model\n"); 13755118ebe8SLucas Mateus Castro (alqotel) } 13765118ebe8SLucas Mateus Castro (alqotel) break; 13775118ebe8SLucas Mateus Castro (alqotel) case -2: 13785118ebe8SLucas Mateus Castro (alqotel) /* Access rights violation */ 13795118ebe8SLucas Mateus Castro (alqotel) cs->exception_index = POWERPC_EXCP_ISI; 13805118ebe8SLucas Mateus Castro (alqotel) env->error_code = 0x08000000; 13815118ebe8SLucas Mateus Castro (alqotel) break; 13825118ebe8SLucas Mateus Castro (alqotel) case -3: 13835118ebe8SLucas Mateus Castro (alqotel) /* No execute protection violation */ 13845118ebe8SLucas Mateus Castro (alqotel) if ((env->mmu_model == POWERPC_MMU_BOOKE) || 13855118ebe8SLucas Mateus Castro (alqotel) (env->mmu_model == POWERPC_MMU_BOOKE206)) { 13865118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_ESR] = 0x00000000; 13875118ebe8SLucas Mateus Castro (alqotel) } 13885118ebe8SLucas Mateus Castro (alqotel) cs->exception_index = POWERPC_EXCP_ISI; 13895118ebe8SLucas Mateus Castro (alqotel) env->error_code = 0x10000000; 13905118ebe8SLucas Mateus Castro (alqotel) break; 13915118ebe8SLucas Mateus Castro (alqotel) case -4: 13925118ebe8SLucas Mateus Castro (alqotel) /* Direct store exception */ 13935118ebe8SLucas Mateus Castro (alqotel) /* No code fetch is allowed in direct-store areas */ 13945118ebe8SLucas Mateus Castro (alqotel) cs->exception_index = POWERPC_EXCP_ISI; 13955118ebe8SLucas Mateus Castro (alqotel) env->error_code = 0x10000000; 13965118ebe8SLucas Mateus Castro (alqotel) break; 13975118ebe8SLucas Mateus Castro (alqotel) } 13985118ebe8SLucas Mateus Castro (alqotel) } else { 13995118ebe8SLucas Mateus Castro (alqotel) switch (ret) { 14005118ebe8SLucas Mateus Castro (alqotel) case -1: 14015118ebe8SLucas Mateus Castro (alqotel) /* No matches in page tables or TLB */ 14025118ebe8SLucas Mateus Castro (alqotel) switch (env->mmu_model) { 14035118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_SOFT_6xx: 14045118ebe8SLucas Mateus Castro (alqotel) if (access_type == MMU_DATA_STORE) { 14055118ebe8SLucas Mateus Castro (alqotel) cs->exception_index = POWERPC_EXCP_DSTLB; 14065118ebe8SLucas Mateus Castro (alqotel) env->error_code = 1 << 16; 14075118ebe8SLucas Mateus Castro (alqotel) } else { 14085118ebe8SLucas Mateus Castro (alqotel) cs->exception_index = POWERPC_EXCP_DLTLB; 14095118ebe8SLucas Mateus Castro (alqotel) env->error_code = 0; 14105118ebe8SLucas Mateus Castro (alqotel) } 14115118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_DMISS] = eaddr; 14125118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_DCMP] = 0x80000000 | ctx.ptem; 14135118ebe8SLucas Mateus Castro (alqotel) tlb_miss: 14145118ebe8SLucas Mateus Castro (alqotel) env->error_code |= ctx.key << 19; 14155118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_HASH1] = ppc_hash32_hpt_base(cpu) + 14165118ebe8SLucas Mateus Castro (alqotel) get_pteg_offset32(cpu, ctx.hash[0]); 14175118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_HASH2] = ppc_hash32_hpt_base(cpu) + 14185118ebe8SLucas Mateus Castro (alqotel) get_pteg_offset32(cpu, ctx.hash[1]); 14195118ebe8SLucas Mateus Castro (alqotel) break; 14205118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_SOFT_4xx: 14215118ebe8SLucas Mateus Castro (alqotel) cs->exception_index = POWERPC_EXCP_DTLB; 14225118ebe8SLucas Mateus Castro (alqotel) env->error_code = 0; 14235118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_40x_DEAR] = eaddr; 14245118ebe8SLucas Mateus Castro (alqotel) if (access_type == MMU_DATA_STORE) { 14255118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_40x_ESR] = 0x00800000; 14265118ebe8SLucas Mateus Castro (alqotel) } else { 14275118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_40x_ESR] = 0x00000000; 14285118ebe8SLucas Mateus Castro (alqotel) } 14295118ebe8SLucas Mateus Castro (alqotel) break; 14305118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_MPC8xx: 14315118ebe8SLucas Mateus Castro (alqotel) /* XXX: TODO */ 14325118ebe8SLucas Mateus Castro (alqotel) cpu_abort(cs, "MPC8xx MMU model is not implemented\n"); 14335118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_BOOKE206: 14345118ebe8SLucas Mateus Castro (alqotel) booke206_update_mas_tlb_miss(env, eaddr, access_type, mmu_idx); 14355118ebe8SLucas Mateus Castro (alqotel) /* fall through */ 14365118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_BOOKE: 14375118ebe8SLucas Mateus Castro (alqotel) cs->exception_index = POWERPC_EXCP_DTLB; 14385118ebe8SLucas Mateus Castro (alqotel) env->error_code = 0; 14395118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_DEAR] = eaddr; 14405118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_ESR] = mmubooke206_esr(mmu_idx, access_type); 14415118ebe8SLucas Mateus Castro (alqotel) break; 14425118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_REAL: 14435118ebe8SLucas Mateus Castro (alqotel) cpu_abort(cs, "PowerPC in real mode should never raise " 14445118ebe8SLucas Mateus Castro (alqotel) "any MMU exceptions\n"); 14455118ebe8SLucas Mateus Castro (alqotel) default: 14465118ebe8SLucas Mateus Castro (alqotel) cpu_abort(cs, "Unknown or invalid MMU model\n"); 14475118ebe8SLucas Mateus Castro (alqotel) } 14485118ebe8SLucas Mateus Castro (alqotel) break; 14495118ebe8SLucas Mateus Castro (alqotel) case -2: 14505118ebe8SLucas Mateus Castro (alqotel) /* Access rights violation */ 14515118ebe8SLucas Mateus Castro (alqotel) cs->exception_index = POWERPC_EXCP_DSI; 14525118ebe8SLucas Mateus Castro (alqotel) env->error_code = 0; 1453*c8f49e6bSCédric Le Goater if (env->mmu_model == POWERPC_MMU_SOFT_4xx) { 14545118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_40x_DEAR] = eaddr; 14555118ebe8SLucas Mateus Castro (alqotel) if (access_type == MMU_DATA_STORE) { 14565118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_40x_ESR] |= 0x00800000; 14575118ebe8SLucas Mateus Castro (alqotel) } 14585118ebe8SLucas Mateus Castro (alqotel) } else if ((env->mmu_model == POWERPC_MMU_BOOKE) || 14595118ebe8SLucas Mateus Castro (alqotel) (env->mmu_model == POWERPC_MMU_BOOKE206)) { 14605118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_DEAR] = eaddr; 14615118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_BOOKE_ESR] = mmubooke206_esr(mmu_idx, access_type); 14625118ebe8SLucas Mateus Castro (alqotel) } else { 14635118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_DAR] = eaddr; 14645118ebe8SLucas Mateus Castro (alqotel) if (access_type == MMU_DATA_STORE) { 14655118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_DSISR] = 0x0A000000; 14665118ebe8SLucas Mateus Castro (alqotel) } else { 14675118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_DSISR] = 0x08000000; 14685118ebe8SLucas Mateus Castro (alqotel) } 14695118ebe8SLucas Mateus Castro (alqotel) } 14705118ebe8SLucas Mateus Castro (alqotel) break; 14715118ebe8SLucas Mateus Castro (alqotel) case -4: 14725118ebe8SLucas Mateus Castro (alqotel) /* Direct store exception */ 14735118ebe8SLucas Mateus Castro (alqotel) switch (type) { 14745118ebe8SLucas Mateus Castro (alqotel) case ACCESS_FLOAT: 14755118ebe8SLucas Mateus Castro (alqotel) /* Floating point load/store */ 14765118ebe8SLucas Mateus Castro (alqotel) cs->exception_index = POWERPC_EXCP_ALIGN; 14775118ebe8SLucas Mateus Castro (alqotel) env->error_code = POWERPC_EXCP_ALIGN_FP; 14785118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_DAR] = eaddr; 14795118ebe8SLucas Mateus Castro (alqotel) break; 14805118ebe8SLucas Mateus Castro (alqotel) case ACCESS_RES: 14815118ebe8SLucas Mateus Castro (alqotel) /* lwarx, ldarx or stwcx. */ 14825118ebe8SLucas Mateus Castro (alqotel) cs->exception_index = POWERPC_EXCP_DSI; 14835118ebe8SLucas Mateus Castro (alqotel) env->error_code = 0; 14845118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_DAR] = eaddr; 14855118ebe8SLucas Mateus Castro (alqotel) if (access_type == MMU_DATA_STORE) { 14865118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_DSISR] = 0x06000000; 14875118ebe8SLucas Mateus Castro (alqotel) } else { 14885118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_DSISR] = 0x04000000; 14895118ebe8SLucas Mateus Castro (alqotel) } 14905118ebe8SLucas Mateus Castro (alqotel) break; 14915118ebe8SLucas Mateus Castro (alqotel) case ACCESS_EXT: 14925118ebe8SLucas Mateus Castro (alqotel) /* eciwx or ecowx */ 14935118ebe8SLucas Mateus Castro (alqotel) cs->exception_index = POWERPC_EXCP_DSI; 14945118ebe8SLucas Mateus Castro (alqotel) env->error_code = 0; 14955118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_DAR] = eaddr; 14965118ebe8SLucas Mateus Castro (alqotel) if (access_type == MMU_DATA_STORE) { 14975118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_DSISR] = 0x06100000; 14985118ebe8SLucas Mateus Castro (alqotel) } else { 14995118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_DSISR] = 0x04100000; 15005118ebe8SLucas Mateus Castro (alqotel) } 15015118ebe8SLucas Mateus Castro (alqotel) break; 15025118ebe8SLucas Mateus Castro (alqotel) default: 15035118ebe8SLucas Mateus Castro (alqotel) printf("DSI: invalid exception (%d)\n", ret); 15045118ebe8SLucas Mateus Castro (alqotel) cs->exception_index = POWERPC_EXCP_PROGRAM; 15055118ebe8SLucas Mateus Castro (alqotel) env->error_code = 15065118ebe8SLucas Mateus Castro (alqotel) POWERPC_EXCP_INVAL | POWERPC_EXCP_INVAL_INVAL; 15075118ebe8SLucas Mateus Castro (alqotel) env->spr[SPR_DAR] = eaddr; 15085118ebe8SLucas Mateus Castro (alqotel) break; 15095118ebe8SLucas Mateus Castro (alqotel) } 15105118ebe8SLucas Mateus Castro (alqotel) break; 15115118ebe8SLucas Mateus Castro (alqotel) } 15125118ebe8SLucas Mateus Castro (alqotel) } 15135118ebe8SLucas Mateus Castro (alqotel) } 15145118ebe8SLucas Mateus Castro (alqotel) return false; 15155118ebe8SLucas Mateus Castro (alqotel) } 15165118ebe8SLucas Mateus Castro (alqotel) 15175118ebe8SLucas Mateus Castro (alqotel) /*****************************************************************************/ 15185118ebe8SLucas Mateus Castro (alqotel) 15195118ebe8SLucas Mateus Castro (alqotel) bool ppc_xlate(PowerPCCPU *cpu, vaddr eaddr, MMUAccessType access_type, 15205118ebe8SLucas Mateus Castro (alqotel) hwaddr *raddrp, int *psizep, int *protp, 15215118ebe8SLucas Mateus Castro (alqotel) int mmu_idx, bool guest_visible) 15225118ebe8SLucas Mateus Castro (alqotel) { 15235118ebe8SLucas Mateus Castro (alqotel) switch (cpu->env.mmu_model) { 15245118ebe8SLucas Mateus Castro (alqotel) #if defined(TARGET_PPC64) 15255118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_3_00: 15265118ebe8SLucas Mateus Castro (alqotel) if (ppc64_v3_radix(cpu)) { 15275118ebe8SLucas Mateus Castro (alqotel) return ppc_radix64_xlate(cpu, eaddr, access_type, raddrp, 15285118ebe8SLucas Mateus Castro (alqotel) psizep, protp, mmu_idx, guest_visible); 15295118ebe8SLucas Mateus Castro (alqotel) } 15305118ebe8SLucas Mateus Castro (alqotel) /* fall through */ 15315118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_64B: 15325118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_2_03: 15335118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_2_06: 15345118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_2_07: 15355118ebe8SLucas Mateus Castro (alqotel) return ppc_hash64_xlate(cpu, eaddr, access_type, 15365118ebe8SLucas Mateus Castro (alqotel) raddrp, psizep, protp, mmu_idx, guest_visible); 15375118ebe8SLucas Mateus Castro (alqotel) #endif 15385118ebe8SLucas Mateus Castro (alqotel) 15395118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_32B: 15405118ebe8SLucas Mateus Castro (alqotel) case POWERPC_MMU_601: 15415118ebe8SLucas Mateus Castro (alqotel) return ppc_hash32_xlate(cpu, eaddr, access_type, raddrp, 15425118ebe8SLucas Mateus Castro (alqotel) psizep, protp, mmu_idx, guest_visible); 15435118ebe8SLucas Mateus Castro (alqotel) 15445118ebe8SLucas Mateus Castro (alqotel) default: 15455118ebe8SLucas Mateus Castro (alqotel) return ppc_jumbo_xlate(cpu, eaddr, access_type, raddrp, 15465118ebe8SLucas Mateus Castro (alqotel) psizep, protp, mmu_idx, guest_visible); 15475118ebe8SLucas Mateus Castro (alqotel) } 15485118ebe8SLucas Mateus Castro (alqotel) } 15495118ebe8SLucas Mateus Castro (alqotel) 15505118ebe8SLucas Mateus Castro (alqotel) hwaddr ppc_cpu_get_phys_page_debug(CPUState *cs, vaddr addr) 15515118ebe8SLucas Mateus Castro (alqotel) { 15525118ebe8SLucas Mateus Castro (alqotel) PowerPCCPU *cpu = POWERPC_CPU(cs); 15535118ebe8SLucas Mateus Castro (alqotel) hwaddr raddr; 15545118ebe8SLucas Mateus Castro (alqotel) int s, p; 15555118ebe8SLucas Mateus Castro (alqotel) 15565118ebe8SLucas Mateus Castro (alqotel) /* 15575118ebe8SLucas Mateus Castro (alqotel) * Some MMUs have separate TLBs for code and data. If we only 15585118ebe8SLucas Mateus Castro (alqotel) * try an MMU_DATA_LOAD, we may not be able to read instructions 15595118ebe8SLucas Mateus Castro (alqotel) * mapped by code TLBs, so we also try a MMU_INST_FETCH. 15605118ebe8SLucas Mateus Castro (alqotel) */ 15615118ebe8SLucas Mateus Castro (alqotel) if (ppc_xlate(cpu, addr, MMU_DATA_LOAD, &raddr, &s, &p, 15625118ebe8SLucas Mateus Castro (alqotel) cpu_mmu_index(&cpu->env, false), false) || 15635118ebe8SLucas Mateus Castro (alqotel) ppc_xlate(cpu, addr, MMU_INST_FETCH, &raddr, &s, &p, 15645118ebe8SLucas Mateus Castro (alqotel) cpu_mmu_index(&cpu->env, true), false)) { 15655118ebe8SLucas Mateus Castro (alqotel) return raddr & TARGET_PAGE_MASK; 15665118ebe8SLucas Mateus Castro (alqotel) } 15675118ebe8SLucas Mateus Castro (alqotel) return -1; 15685118ebe8SLucas Mateus Castro (alqotel) } 1569