1e8af50a3Sbellard /* 2e8af50a3Sbellard * sparc helpers 3e8af50a3Sbellard * 4e8af50a3Sbellard * Copyright (c) 2003 Fabrice Bellard 5e8af50a3Sbellard * 6e8af50a3Sbellard * This library is free software; you can redistribute it and/or 7e8af50a3Sbellard * modify it under the terms of the GNU Lesser General Public 8e8af50a3Sbellard * License as published by the Free Software Foundation; either 9e8af50a3Sbellard * version 2 of the License, or (at your option) any later version. 10e8af50a3Sbellard * 11e8af50a3Sbellard * This library is distributed in the hope that it will be useful, 12e8af50a3Sbellard * but WITHOUT ANY WARRANTY; without even the implied warranty of 13e8af50a3Sbellard * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 14e8af50a3Sbellard * Lesser General Public License for more details. 15e8af50a3Sbellard * 16e8af50a3Sbellard * You should have received a copy of the GNU Lesser General Public 17e8af50a3Sbellard * License along with this library; if not, write to the Free Software 18e8af50a3Sbellard * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA 19e8af50a3Sbellard */ 20e8af50a3Sbellard #include "exec.h" 21e8af50a3Sbellard 22e80cfcfcSbellard //#define DEBUG_PCALL 23e80cfcfcSbellard //#define DEBUG_MMU 24e8af50a3Sbellard 25e8af50a3Sbellard /* Sparc MMU emulation */ 26e8af50a3Sbellard int cpu_sparc_handle_mmu_fault (CPUState *env, uint32_t address, int rw, 27e8af50a3Sbellard int is_user, int is_softmmu); 28e8af50a3Sbellard 29e8af50a3Sbellard /* thread support */ 30e8af50a3Sbellard 31e8af50a3Sbellard spinlock_t global_cpu_lock = SPIN_LOCK_UNLOCKED; 32e8af50a3Sbellard 33e8af50a3Sbellard void cpu_lock(void) 34e8af50a3Sbellard { 35e8af50a3Sbellard spin_lock(&global_cpu_lock); 36e8af50a3Sbellard } 37e8af50a3Sbellard 38e8af50a3Sbellard void cpu_unlock(void) 39e8af50a3Sbellard { 40e8af50a3Sbellard spin_unlock(&global_cpu_lock); 41e8af50a3Sbellard } 42e8af50a3Sbellard 43e8af50a3Sbellard #if !defined(CONFIG_USER_ONLY) 44e8af50a3Sbellard 45e8af50a3Sbellard #define MMUSUFFIX _mmu 46e8af50a3Sbellard #define GETPC() (__builtin_return_address(0)) 47e8af50a3Sbellard 48e8af50a3Sbellard #define SHIFT 0 49e8af50a3Sbellard #include "softmmu_template.h" 50e8af50a3Sbellard 51e8af50a3Sbellard #define SHIFT 1 52e8af50a3Sbellard #include "softmmu_template.h" 53e8af50a3Sbellard 54e8af50a3Sbellard #define SHIFT 2 55e8af50a3Sbellard #include "softmmu_template.h" 56e8af50a3Sbellard 57e8af50a3Sbellard #define SHIFT 3 58e8af50a3Sbellard #include "softmmu_template.h" 59e8af50a3Sbellard 60e8af50a3Sbellard 61e8af50a3Sbellard /* try to fill the TLB and return an exception if error. If retaddr is 62e8af50a3Sbellard NULL, it means that the function was called in C code (i.e. not 63e8af50a3Sbellard from generated code or from helper.c) */ 64e8af50a3Sbellard /* XXX: fix it to restore all registers */ 65e8af50a3Sbellard void tlb_fill(unsigned long addr, int is_write, int is_user, void *retaddr) 66e8af50a3Sbellard { 67e8af50a3Sbellard TranslationBlock *tb; 68e8af50a3Sbellard int ret; 69e8af50a3Sbellard unsigned long pc; 70e8af50a3Sbellard CPUState *saved_env; 71e8af50a3Sbellard 72e8af50a3Sbellard /* XXX: hack to restore env in all cases, even if not called from 73e8af50a3Sbellard generated code */ 74e8af50a3Sbellard saved_env = env; 75e8af50a3Sbellard env = cpu_single_env; 76e8af50a3Sbellard 77e8af50a3Sbellard ret = cpu_sparc_handle_mmu_fault(env, addr, is_write, is_user, 1); 78e8af50a3Sbellard if (ret) { 79e8af50a3Sbellard if (retaddr) { 80e8af50a3Sbellard /* now we have a real cpu fault */ 81e8af50a3Sbellard pc = (unsigned long)retaddr; 82e8af50a3Sbellard tb = tb_find_pc(pc); 83e8af50a3Sbellard if (tb) { 84e8af50a3Sbellard /* the PC is inside the translated code. It means that we have 85e8af50a3Sbellard a virtual CPU fault */ 86e8af50a3Sbellard cpu_restore_state(tb, env, pc, NULL); 87e8af50a3Sbellard } 88e8af50a3Sbellard } 89e8af50a3Sbellard raise_exception_err(ret, env->error_code); 90e8af50a3Sbellard } 91e8af50a3Sbellard env = saved_env; 92e8af50a3Sbellard } 93e8af50a3Sbellard #endif 94e8af50a3Sbellard 95e8af50a3Sbellard static const int access_table[8][8] = { 96e8af50a3Sbellard { 0, 0, 0, 0, 2, 0, 3, 3 }, 97e8af50a3Sbellard { 0, 0, 0, 0, 2, 0, 0, 0 }, 98e8af50a3Sbellard { 2, 2, 0, 0, 0, 2, 3, 3 }, 99e8af50a3Sbellard { 2, 2, 0, 0, 0, 2, 0, 0 }, 100e8af50a3Sbellard { 2, 0, 2, 0, 2, 2, 3, 3 }, 101e8af50a3Sbellard { 2, 0, 2, 0, 2, 0, 2, 0 }, 102e8af50a3Sbellard { 2, 2, 2, 0, 2, 2, 3, 3 }, 103e8af50a3Sbellard { 2, 2, 2, 0, 2, 2, 2, 0 } 104e8af50a3Sbellard }; 105e8af50a3Sbellard 106e8af50a3Sbellard /* 1 = write OK */ 107e8af50a3Sbellard static const int rw_table[2][8] = { 108e8af50a3Sbellard { 0, 1, 0, 1, 0, 1, 0, 1 }, 109e8af50a3Sbellard { 0, 1, 0, 1, 0, 0, 0, 0 } 110e8af50a3Sbellard }; 111e8af50a3Sbellard 112e80cfcfcSbellard int get_physical_address (CPUState *env, uint32_t *physical, int *prot, 113e80cfcfcSbellard int *access_index, uint32_t address, int rw, 114e80cfcfcSbellard int is_user) 115e8af50a3Sbellard { 116e80cfcfcSbellard int access_perms = 0; 117e80cfcfcSbellard target_phys_addr_t pde_ptr; 118e8af50a3Sbellard uint32_t pde, virt_addr; 119e80cfcfcSbellard int error_code = 0, is_dirty; 120e80cfcfcSbellard unsigned long page_offset; 121e8af50a3Sbellard 122e8af50a3Sbellard virt_addr = address & TARGET_PAGE_MASK; 123e8af50a3Sbellard if ((env->mmuregs[0] & MMU_E) == 0) { /* MMU disabled */ 124e80cfcfcSbellard *physical = address; 125e80cfcfcSbellard *prot = PAGE_READ | PAGE_WRITE; 126e80cfcfcSbellard return 0; 127e8af50a3Sbellard } 128e8af50a3Sbellard 129e8af50a3Sbellard /* SPARC reference MMU table walk: Context table->L1->L2->PTE */ 130e8af50a3Sbellard /* Context base + context number */ 131e80cfcfcSbellard pde_ptr = (env->mmuregs[1] << 4) + (env->mmuregs[2] << 4); 132e80cfcfcSbellard cpu_physical_memory_read(pde_ptr, (uint8_t *)&pde, 4); 133e80cfcfcSbellard bswap32s(&pde); 134e8af50a3Sbellard 135e8af50a3Sbellard /* Ctx pde */ 136e8af50a3Sbellard switch (pde & PTE_ENTRYTYPE_MASK) { 137e80cfcfcSbellard default: 138e8af50a3Sbellard case 0: /* Invalid */ 139e80cfcfcSbellard return 1; 140e80cfcfcSbellard case 2: /* L0 PTE, maybe should not happen? */ 141e8af50a3Sbellard case 3: /* Reserved */ 142e80cfcfcSbellard return 4; 143e80cfcfcSbellard case 1: /* L0 PDE */ 144e80cfcfcSbellard pde_ptr = ((address >> 22) & ~3) + ((pde & ~3) << 4); 145e80cfcfcSbellard cpu_physical_memory_read(pde_ptr, (uint8_t *)&pde, 4); 146e80cfcfcSbellard bswap32s(&pde); 147e80cfcfcSbellard 148e80cfcfcSbellard switch (pde & PTE_ENTRYTYPE_MASK) { 149e80cfcfcSbellard default: 150e80cfcfcSbellard case 0: /* Invalid */ 151e80cfcfcSbellard return 1; 152e80cfcfcSbellard case 3: /* Reserved */ 153e80cfcfcSbellard return 4; 154e8af50a3Sbellard case 1: /* L1 PDE */ 155e80cfcfcSbellard pde_ptr = ((address & 0xfc0000) >> 16) + ((pde & ~3) << 4); 156e80cfcfcSbellard cpu_physical_memory_read(pde_ptr, (uint8_t *)&pde, 4); 157e80cfcfcSbellard bswap32s(&pde); 158e8af50a3Sbellard 159e8af50a3Sbellard switch (pde & PTE_ENTRYTYPE_MASK) { 160e80cfcfcSbellard default: 161e8af50a3Sbellard case 0: /* Invalid */ 162e80cfcfcSbellard return 1; 163e8af50a3Sbellard case 3: /* Reserved */ 164e80cfcfcSbellard return 4; 165e8af50a3Sbellard case 1: /* L2 PDE */ 166e80cfcfcSbellard pde_ptr = ((address & 0x3f000) >> 10) + ((pde & ~3) << 4); 167e80cfcfcSbellard cpu_physical_memory_read(pde_ptr, (uint8_t *)&pde, 4); 168e80cfcfcSbellard bswap32s(&pde); 169e8af50a3Sbellard 170e8af50a3Sbellard switch (pde & PTE_ENTRYTYPE_MASK) { 171e80cfcfcSbellard default: 172e8af50a3Sbellard case 0: /* Invalid */ 173e80cfcfcSbellard return 1; 174e8af50a3Sbellard case 1: /* PDE, should not happen */ 175e8af50a3Sbellard case 3: /* Reserved */ 176e80cfcfcSbellard return 4; 177e8af50a3Sbellard case 2: /* L3 PTE */ 178e8af50a3Sbellard virt_addr = address & TARGET_PAGE_MASK; 179e8af50a3Sbellard page_offset = (address & TARGET_PAGE_MASK) & (TARGET_PAGE_SIZE - 1); 180e8af50a3Sbellard } 181e8af50a3Sbellard break; 182e8af50a3Sbellard case 2: /* L2 PTE */ 183e8af50a3Sbellard virt_addr = address & ~0x3ffff; 184e8af50a3Sbellard page_offset = address & 0x3ffff; 185e8af50a3Sbellard } 186e8af50a3Sbellard break; 187e8af50a3Sbellard case 2: /* L1 PTE */ 188e8af50a3Sbellard virt_addr = address & ~0xffffff; 189e8af50a3Sbellard page_offset = address & 0xffffff; 190e8af50a3Sbellard } 191e8af50a3Sbellard } 192e8af50a3Sbellard 193e8af50a3Sbellard /* update page modified and dirty bits */ 194b769d8feSbellard is_dirty = (rw & 1) && !(pde & PG_MODIFIED_MASK); 195e8af50a3Sbellard if (!(pde & PG_ACCESSED_MASK) || is_dirty) { 196e80cfcfcSbellard uint32_t tmppde; 197e8af50a3Sbellard pde |= PG_ACCESSED_MASK; 198e8af50a3Sbellard if (is_dirty) 199e8af50a3Sbellard pde |= PG_MODIFIED_MASK; 200e80cfcfcSbellard tmppde = bswap32(pde); 201e80cfcfcSbellard cpu_physical_memory_write(pde_ptr, (uint8_t *)&tmppde, 4); 202e8af50a3Sbellard } 203e8af50a3Sbellard /* check access */ 204e80cfcfcSbellard *access_index = ((rw & 1) << 2) | (rw & 2) | (is_user? 0 : 1); 205e8af50a3Sbellard access_perms = (pde & PTE_ACCESS_MASK) >> PTE_ACCESS_SHIFT; 206e80cfcfcSbellard error_code = access_table[*access_index][access_perms]; 207e8af50a3Sbellard if (error_code) 208e80cfcfcSbellard return error_code; 209e8af50a3Sbellard 210e8af50a3Sbellard /* the page can be put in the TLB */ 211e80cfcfcSbellard *prot = PAGE_READ; 212e8af50a3Sbellard if (pde & PG_MODIFIED_MASK) { 213e8af50a3Sbellard /* only set write access if already dirty... otherwise wait 214e8af50a3Sbellard for dirty access */ 215e8af50a3Sbellard if (rw_table[is_user][access_perms]) 216e80cfcfcSbellard *prot |= PAGE_WRITE; 217e8af50a3Sbellard } 218e8af50a3Sbellard 219e8af50a3Sbellard /* Even if large ptes, we map only one 4KB page in the cache to 220e8af50a3Sbellard avoid filling it too fast */ 221e80cfcfcSbellard *physical = ((pde & PTE_ADDR_MASK) << 4) + page_offset; 222e80cfcfcSbellard return 0; 223e80cfcfcSbellard } 224e80cfcfcSbellard 225e80cfcfcSbellard /* Perform address translation */ 226e80cfcfcSbellard int cpu_sparc_handle_mmu_fault (CPUState *env, uint32_t address, int rw, 227e80cfcfcSbellard int is_user, int is_softmmu) 228e80cfcfcSbellard { 229e80cfcfcSbellard int exception = 0; 230e80cfcfcSbellard uint32_t virt_addr, paddr; 231e80cfcfcSbellard unsigned long vaddr; 232e80cfcfcSbellard int error_code = 0, prot, ret = 0, access_index; 233e80cfcfcSbellard 234e80cfcfcSbellard if (env->user_mode_only) { 235e80cfcfcSbellard /* user mode only emulation */ 236e80cfcfcSbellard error_code = -2; 237e80cfcfcSbellard goto do_fault_user; 238e80cfcfcSbellard } 239e80cfcfcSbellard 240e80cfcfcSbellard error_code = get_physical_address(env, &paddr, &prot, &access_index, address, rw, is_user); 241e80cfcfcSbellard if (error_code == 0) { 242e8af50a3Sbellard virt_addr = address & TARGET_PAGE_MASK; 243e8af50a3Sbellard vaddr = virt_addr + ((address & TARGET_PAGE_MASK) & (TARGET_PAGE_SIZE - 1)); 244e8af50a3Sbellard ret = tlb_set_page(env, vaddr, paddr, prot, is_user, is_softmmu); 245e8af50a3Sbellard return ret; 246e80cfcfcSbellard } 247e8af50a3Sbellard 248e8af50a3Sbellard if (env->mmuregs[3]) /* Fault status register */ 249e8af50a3Sbellard env->mmuregs[3] = 1; /* overflow (not read before another fault) */ 250e8af50a3Sbellard env->mmuregs[3] |= (access_index << 5) | (error_code << 2) | 2; 251e8af50a3Sbellard env->mmuregs[4] = address; /* Fault address register */ 252e8af50a3Sbellard 2538d5f07faSbellard if (env->mmuregs[0] & MMU_NF || env->psret == 0) // No fault 254e8af50a3Sbellard return 0; 255e80cfcfcSbellard do_fault_user: 256e8af50a3Sbellard env->exception_index = exception; 257e8af50a3Sbellard env->error_code = error_code; 258e8af50a3Sbellard return error_code; 259e8af50a3Sbellard } 260e8af50a3Sbellard 261e8af50a3Sbellard void memcpy32(uint32_t *dst, const uint32_t *src) 262e8af50a3Sbellard { 263e8af50a3Sbellard dst[0] = src[0]; 264e8af50a3Sbellard dst[1] = src[1]; 265e8af50a3Sbellard dst[2] = src[2]; 266e8af50a3Sbellard dst[3] = src[3]; 267e8af50a3Sbellard dst[4] = src[4]; 268e8af50a3Sbellard dst[5] = src[5]; 269e8af50a3Sbellard dst[6] = src[6]; 270e8af50a3Sbellard dst[7] = src[7]; 271e8af50a3Sbellard } 272e8af50a3Sbellard 273e8af50a3Sbellard void set_cwp(int new_cwp) 274e8af50a3Sbellard { 275e8af50a3Sbellard /* put the modified wrap registers at their proper location */ 276e8af50a3Sbellard if (env->cwp == (NWINDOWS - 1)) 277e8af50a3Sbellard memcpy32(env->regbase, env->regbase + NWINDOWS * 16); 278e8af50a3Sbellard env->cwp = new_cwp; 279e8af50a3Sbellard /* put the wrap registers at their temporary location */ 280e8af50a3Sbellard if (new_cwp == (NWINDOWS - 1)) 281e8af50a3Sbellard memcpy32(env->regbase + NWINDOWS * 16, env->regbase); 282e8af50a3Sbellard env->regwptr = env->regbase + (new_cwp * 16); 283e8af50a3Sbellard } 284e8af50a3Sbellard 285e8af50a3Sbellard /* 286e8af50a3Sbellard * Begin execution of an interruption. is_int is TRUE if coming from 287e8af50a3Sbellard * the int instruction. next_eip is the EIP value AFTER the interrupt 288e8af50a3Sbellard * instruction. It is only relevant if is_int is TRUE. 289e8af50a3Sbellard */ 290e8af50a3Sbellard void do_interrupt(int intno, int is_int, int error_code, 291e8af50a3Sbellard unsigned int next_eip, int is_hw) 292e8af50a3Sbellard { 293e8af50a3Sbellard int cwp; 294e8af50a3Sbellard 295e8af50a3Sbellard #ifdef DEBUG_PCALL 296e8af50a3Sbellard if (loglevel & CPU_LOG_INT) { 297e8af50a3Sbellard static int count; 298e8af50a3Sbellard fprintf(logfile, "%6d: v=%02x e=%04x i=%d pc=%08x npc=%08x SP=%08x\n", 299e8af50a3Sbellard count, intno, error_code, is_int, 300e8af50a3Sbellard env->pc, 3018d5f07faSbellard env->npc, env->regwptr[6]); 302e80cfcfcSbellard #if 1 3037fe48483Sbellard cpu_dump_state(env, logfile, fprintf, 0); 304e8af50a3Sbellard { 305e8af50a3Sbellard int i; 306e8af50a3Sbellard uint8_t *ptr; 307e80cfcfcSbellard 308e8af50a3Sbellard fprintf(logfile, " code="); 309e80cfcfcSbellard ptr = (uint8_t *)env->pc; 310e8af50a3Sbellard for(i = 0; i < 16; i++) { 311e8af50a3Sbellard fprintf(logfile, " %02x", ldub(ptr + i)); 312e8af50a3Sbellard } 313e8af50a3Sbellard fprintf(logfile, "\n"); 314e8af50a3Sbellard } 315e8af50a3Sbellard #endif 316e8af50a3Sbellard count++; 317e8af50a3Sbellard } 318e8af50a3Sbellard #endif 319e80cfcfcSbellard #if !defined(CONFIG_USER_ONLY) 320e80cfcfcSbellard if (env->psret == 0) { 321e80cfcfcSbellard fprintf(logfile, "Trap while interrupts disabled, Error state!\n"); 322e80cfcfcSbellard qemu_system_shutdown_request(); 323e80cfcfcSbellard return; 324e80cfcfcSbellard } 325e80cfcfcSbellard #endif 326e8af50a3Sbellard env->psret = 0; 327e8af50a3Sbellard cwp = (env->cwp - 1) & (NWINDOWS - 1); 328e8af50a3Sbellard set_cwp(cwp); 329e80cfcfcSbellard env->regwptr[9] = env->pc - 4; // XXX? 330e80cfcfcSbellard env->regwptr[10] = env->pc; 331e8af50a3Sbellard env->psrps = env->psrs; 332e8af50a3Sbellard env->psrs = 1; 333e8af50a3Sbellard env->tbr = (env->tbr & TBR_BASE_MASK) | (intno << 4); 334e8af50a3Sbellard env->pc = env->tbr; 335e8af50a3Sbellard env->npc = env->pc + 4; 336e8af50a3Sbellard env->exception_index = 0; 337e8af50a3Sbellard } 338e8af50a3Sbellard 339e8af50a3Sbellard void raise_exception_err(int exception_index, int error_code) 340e8af50a3Sbellard { 341e8af50a3Sbellard raise_exception(exception_index); 342e8af50a3Sbellard } 343e80cfcfcSbellard 344e80cfcfcSbellard uint32_t mmu_probe(uint32_t address, int mmulev) 345e80cfcfcSbellard { 346e80cfcfcSbellard target_phys_addr_t pde_ptr; 347e80cfcfcSbellard uint32_t pde; 348e80cfcfcSbellard 349e80cfcfcSbellard /* Context base + context number */ 350e80cfcfcSbellard pde_ptr = (env->mmuregs[1] << 4) + (env->mmuregs[2] << 4); 351e80cfcfcSbellard cpu_physical_memory_read(pde_ptr, (uint8_t *)&pde, 4); 352e80cfcfcSbellard bswap32s(&pde); 353e80cfcfcSbellard switch (pde & PTE_ENTRYTYPE_MASK) { 354e80cfcfcSbellard default: 355e80cfcfcSbellard case 0: /* Invalid */ 356e80cfcfcSbellard case 2: /* PTE, maybe should not happen? */ 357e80cfcfcSbellard case 3: /* Reserved */ 358e80cfcfcSbellard return 0; 359e80cfcfcSbellard case 1: /* L1 PDE */ 360e80cfcfcSbellard if (mmulev == 3) 361e80cfcfcSbellard return pde; 362e80cfcfcSbellard pde_ptr = ((address >> 22) & ~3) + ((pde & ~3) << 4); 363e80cfcfcSbellard cpu_physical_memory_read(pde_ptr, (uint8_t *)&pde, 4); 364e80cfcfcSbellard bswap32s(&pde); 365e80cfcfcSbellard 366e80cfcfcSbellard switch (pde & PTE_ENTRYTYPE_MASK) { 367e80cfcfcSbellard default: 368e80cfcfcSbellard case 0: /* Invalid */ 369e80cfcfcSbellard case 3: /* Reserved */ 370e80cfcfcSbellard return 0; 371e80cfcfcSbellard case 2: /* L1 PTE */ 372e80cfcfcSbellard return pde; 373e80cfcfcSbellard case 1: /* L2 PDE */ 374e80cfcfcSbellard if (mmulev == 2) 375e80cfcfcSbellard return pde; 376e80cfcfcSbellard pde_ptr = ((address & 0xfc0000) >> 16) + ((pde & ~3) << 4); 377e80cfcfcSbellard cpu_physical_memory_read(pde_ptr, (uint8_t *)&pde, 4); 378e80cfcfcSbellard bswap32s(&pde); 379e80cfcfcSbellard 380e80cfcfcSbellard switch (pde & PTE_ENTRYTYPE_MASK) { 381e80cfcfcSbellard default: 382e80cfcfcSbellard case 0: /* Invalid */ 383e80cfcfcSbellard case 3: /* Reserved */ 384e80cfcfcSbellard return 0; 385e80cfcfcSbellard case 2: /* L2 PTE */ 386e80cfcfcSbellard return pde; 387e80cfcfcSbellard case 1: /* L3 PDE */ 388e80cfcfcSbellard if (mmulev == 1) 389e80cfcfcSbellard return pde; 390e80cfcfcSbellard pde_ptr = ((address & 0x3f000) >> 10) + ((pde & ~3) << 4); 391e80cfcfcSbellard cpu_physical_memory_read(pde_ptr, (uint8_t *)&pde, 4); 392e80cfcfcSbellard bswap32s(&pde); 393e80cfcfcSbellard 394e80cfcfcSbellard switch (pde & PTE_ENTRYTYPE_MASK) { 395e80cfcfcSbellard default: 396e80cfcfcSbellard case 0: /* Invalid */ 397e80cfcfcSbellard case 1: /* PDE, should not happen */ 398e80cfcfcSbellard case 3: /* Reserved */ 399e80cfcfcSbellard return 0; 400e80cfcfcSbellard case 2: /* L3 PTE */ 401e80cfcfcSbellard return pde; 402e80cfcfcSbellard } 403e80cfcfcSbellard } 404e80cfcfcSbellard } 405e80cfcfcSbellard } 406e80cfcfcSbellard return 0; 407e80cfcfcSbellard } 408e80cfcfcSbellard 409e80cfcfcSbellard void dump_mmu(void) 410e80cfcfcSbellard { 411e80cfcfcSbellard #ifdef DEBUG_MMU 412e80cfcfcSbellard uint32_t pa, va, va1, va2; 413e80cfcfcSbellard int n, m, o; 414e80cfcfcSbellard target_phys_addr_t pde_ptr; 415e80cfcfcSbellard uint32_t pde; 416e80cfcfcSbellard 417e80cfcfcSbellard printf("MMU dump:\n"); 418e80cfcfcSbellard pde_ptr = (env->mmuregs[1] << 4) + (env->mmuregs[2] << 4); 419e80cfcfcSbellard cpu_physical_memory_read(pde_ptr, (uint8_t *)&pde, 4); 420e80cfcfcSbellard bswap32s(&pde); 421e80cfcfcSbellard printf("Root ptr: 0x%08x, ctx: %d\n", env->mmuregs[1] << 4, env->mmuregs[2]); 422e80cfcfcSbellard for (n = 0, va = 0; n < 256; n++, va += 16 * 1024 * 1024) { 423e80cfcfcSbellard pde_ptr = mmu_probe(va, 2); 424e80cfcfcSbellard if (pde_ptr) { 425e80cfcfcSbellard pa = cpu_get_phys_page_debug(env, va); 426e80cfcfcSbellard printf("VA: 0x%08x, PA: 0x%08x PDE: 0x%08x\n", va, pa, pde_ptr); 427e80cfcfcSbellard for (m = 0, va1 = va; m < 64; m++, va1 += 256 * 1024) { 428e80cfcfcSbellard pde_ptr = mmu_probe(va1, 1); 429e80cfcfcSbellard if (pde_ptr) { 430e80cfcfcSbellard pa = cpu_get_phys_page_debug(env, va1); 431e80cfcfcSbellard printf(" VA: 0x%08x, PA: 0x%08x PDE: 0x%08x\n", va1, pa, pde_ptr); 432e80cfcfcSbellard for (o = 0, va2 = va1; o < 64; o++, va2 += 4 * 1024) { 433e80cfcfcSbellard pde_ptr = mmu_probe(va2, 0); 434e80cfcfcSbellard if (pde_ptr) { 435e80cfcfcSbellard pa = cpu_get_phys_page_debug(env, va2); 436e80cfcfcSbellard printf(" VA: 0x%08x, PA: 0x%08x PTE: 0x%08x\n", va2, pa, pde_ptr); 437e80cfcfcSbellard } 438e80cfcfcSbellard } 439e80cfcfcSbellard } 440e80cfcfcSbellard } 441e80cfcfcSbellard } 442e80cfcfcSbellard } 443e80cfcfcSbellard printf("MMU dump ends\n"); 444e80cfcfcSbellard #endif 445e80cfcfcSbellard } 446