2020-01-05 17:38:32 +00:00
|
|
|
//DingusPPC
|
2019-10-16 04:15:12 +00:00
|
|
|
//Written by divingkatae and maximum
|
|
|
|
//(c)2018-20 (theweirdo) spatium
|
2019-07-02 02:15:33 +00:00
|
|
|
//Please ask for permission
|
|
|
|
//if you want to distribute this.
|
2020-01-05 17:38:32 +00:00
|
|
|
//(divingkatae#1017 or powermax#2286 on Discord)
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-03 19:36:00 +00:00
|
|
|
/** @file PowerPC Memory management unit emulation. */
|
|
|
|
|
|
|
|
/* TODO:
|
|
|
|
- implement TLB
|
|
|
|
- implement 601-style BATs
|
|
|
|
- add proper error and exception handling
|
|
|
|
- clarify what to do in the case of unaligned memory accesses
|
|
|
|
*/
|
2019-07-02 02:15:33 +00:00
|
|
|
|
|
|
|
#include <iostream>
|
|
|
|
#include <cstdint>
|
|
|
|
#include <cinttypes>
|
|
|
|
#include <string>
|
|
|
|
#include <array>
|
2020-01-03 15:08:00 +00:00
|
|
|
#include "memreadwrite.h"
|
2019-12-27 19:10:36 +00:00
|
|
|
#include "ppcemu.h"
|
2019-12-27 19:00:53 +00:00
|
|
|
#include "ppcmmu.h"
|
2019-08-21 06:33:01 +00:00
|
|
|
#include "devices/memctrlbase.h"
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-02-04 13:20:10 +00:00
|
|
|
/** PowerPC-style MMU BAT arrays (NULL initialization isn't prescribed). */
|
2020-01-26 02:30:55 +00:00
|
|
|
PPC_BAT_entry ibat_array[4] = { {0} };
|
|
|
|
PPC_BAT_entry dbat_array[4] = { {0} };
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-26 22:45:29 +00:00
|
|
|
/** remember recently used physical memory regions for quicker translation. */
|
2020-02-04 13:20:10 +00:00
|
|
|
AddressMapEntry last_read_area = { 0 };
|
2020-01-26 22:45:29 +00:00
|
|
|
AddressMapEntry last_write_area = { 0 };
|
2020-02-04 13:20:10 +00:00
|
|
|
AddressMapEntry last_exec_area = { 0 };
|
|
|
|
AddressMapEntry last_ptab_area = { 0 };
|
2020-01-26 22:45:29 +00:00
|
|
|
|
|
|
|
|
|
|
|
/* macro for generating code reading from physical memory */
|
|
|
|
#define READ_PHYS_MEM(ENTRY, ADDR, OP, SIZE, UNVAL) \
|
|
|
|
{ \
|
|
|
|
if ((ADDR) >= (ENTRY).start && (ADDR) <= (ENTRY).end) { \
|
|
|
|
ret = OP((ENTRY).mem_ptr + ((ADDR) - (ENTRY).start)); \
|
|
|
|
} else { \
|
|
|
|
AddressMapEntry* entry = mem_ctrl_instance->find_range((ADDR)); \
|
|
|
|
if (entry) { \
|
|
|
|
if (entry->type & (RT_ROM | RT_RAM)) { \
|
|
|
|
(ENTRY).start = entry->start; \
|
|
|
|
(ENTRY).end = entry->end; \
|
|
|
|
(ENTRY).mem_ptr = entry->mem_ptr; \
|
|
|
|
ret = OP((ENTRY).mem_ptr + ((ADDR) - (ENTRY).start)); \
|
|
|
|
} \
|
|
|
|
else if (entry->type & RT_MMIO) { \
|
|
|
|
ret = entry->devobj->read((ADDR) - entry->start, (SIZE)); \
|
|
|
|
} \
|
|
|
|
else { \
|
|
|
|
printf("Please check your address map!\n"); \
|
|
|
|
ret = (UNVAL); \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
else { \
|
|
|
|
printf("WARNING: read from unmapped memory at 0x%08X!\n", (ADDR)); \
|
|
|
|
ret = (UNVAL); \
|
|
|
|
} \
|
|
|
|
} \
|
2020-01-26 02:30:55 +00:00
|
|
|
}
|
|
|
|
|
2020-01-26 22:45:29 +00:00
|
|
|
/* macro for generating code writing to physical memory */
|
|
|
|
#define WRITE_PHYS_MEM(ENTRY, ADDR, OP, VAL, SIZE) \
|
|
|
|
{ \
|
|
|
|
if ((ADDR) >= (ENTRY).start && (ADDR) <= (ENTRY).end) { \
|
|
|
|
OP((ENTRY).mem_ptr + ((ADDR) - (ENTRY).start), (VAL)); \
|
|
|
|
} else { \
|
|
|
|
AddressMapEntry* entry = mem_ctrl_instance->find_range((ADDR)); \
|
|
|
|
if (entry) { \
|
|
|
|
if (entry->type & RT_RAM) { \
|
|
|
|
(ENTRY).start = entry->start; \
|
|
|
|
(ENTRY).end = entry->end; \
|
|
|
|
(ENTRY).mem_ptr = entry->mem_ptr; \
|
|
|
|
OP((ENTRY).mem_ptr + ((ADDR) - (ENTRY).start), (VAL)); \
|
|
|
|
} \
|
|
|
|
else if (entry->type & RT_MMIO) { \
|
|
|
|
entry->devobj->write((ADDR) - entry->start, (VAL), (SIZE)); \
|
|
|
|
} \
|
|
|
|
else { \
|
|
|
|
printf("Please check your address map!\n"); \
|
|
|
|
} \
|
|
|
|
} \
|
|
|
|
else { \
|
|
|
|
printf("WARNING: write to unmapped memory at 0x%08X!\n", (ADDR)); \
|
|
|
|
} \
|
|
|
|
} \
|
2020-01-26 02:30:55 +00:00
|
|
|
}
|
2019-07-19 06:31:16 +00:00
|
|
|
|
2020-01-26 22:45:29 +00:00
|
|
|
void ppc_set_cur_instruction(const uint8_t* ptr)
|
|
|
|
{
|
|
|
|
ppc_cur_instruction = READ_DWORD_BE_A(ptr);
|
2020-01-26 02:30:55 +00:00
|
|
|
}
|
2020-01-06 02:46:23 +00:00
|
|
|
|
2019-08-02 23:05:48 +00:00
|
|
|
void ibat_update(uint32_t bat_reg)
|
|
|
|
{
|
|
|
|
int upper_reg_num;
|
|
|
|
uint32_t bl, lo_mask;
|
2020-01-26 02:30:55 +00:00
|
|
|
PPC_BAT_entry* bat_entry;
|
2019-08-02 23:05:48 +00:00
|
|
|
|
|
|
|
upper_reg_num = bat_reg & 0xFFFFFFFE;
|
|
|
|
|
|
|
|
if (ppc_state.ppc_spr[upper_reg_num] & 3) { // is that BAT pair valid?
|
|
|
|
bat_entry = &ibat_array[(bat_reg - 528) >> 1];
|
|
|
|
bl = (ppc_state.ppc_spr[upper_reg_num] >> 2) & 0x7FF;
|
|
|
|
lo_mask = (bl << 17) | 0x1FFFF;
|
|
|
|
|
2020-01-26 02:30:55 +00:00
|
|
|
bat_entry->access = ppc_state.ppc_spr[upper_reg_num] & 3;
|
|
|
|
bat_entry->prot = ppc_state.ppc_spr[upper_reg_num + 1] & 3;
|
2019-08-02 23:05:48 +00:00
|
|
|
bat_entry->lo_mask = lo_mask;
|
|
|
|
bat_entry->phys_hi = ppc_state.ppc_spr[upper_reg_num + 1] & ~lo_mask;
|
2020-01-26 02:30:55 +00:00
|
|
|
bat_entry->bepi = ppc_state.ppc_spr[upper_reg_num] & ~lo_mask;
|
2019-07-02 02:15:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-02 00:02:01 +00:00
|
|
|
void dbat_update(uint32_t bat_reg)
|
|
|
|
{
|
|
|
|
int upper_reg_num;
|
|
|
|
uint32_t bl, lo_mask;
|
2020-01-26 02:30:55 +00:00
|
|
|
PPC_BAT_entry* bat_entry;
|
2019-08-02 00:02:01 +00:00
|
|
|
|
|
|
|
upper_reg_num = bat_reg & 0xFFFFFFFE;
|
|
|
|
|
|
|
|
if (ppc_state.ppc_spr[upper_reg_num] & 3) { // is that BAT pair valid?
|
|
|
|
bat_entry = &dbat_array[(bat_reg - 536) >> 1];
|
|
|
|
bl = (ppc_state.ppc_spr[upper_reg_num] >> 2) & 0x7FF;
|
|
|
|
lo_mask = (bl << 17) | 0x1FFFF;
|
|
|
|
|
2020-01-26 02:30:55 +00:00
|
|
|
bat_entry->access = ppc_state.ppc_spr[upper_reg_num] & 3;
|
|
|
|
bat_entry->prot = ppc_state.ppc_spr[upper_reg_num + 1] & 3;
|
2019-08-02 00:02:01 +00:00
|
|
|
bat_entry->lo_mask = lo_mask;
|
|
|
|
bat_entry->phys_hi = ppc_state.ppc_spr[upper_reg_num + 1] & ~lo_mask;
|
2020-01-26 02:30:55 +00:00
|
|
|
bat_entry->bepi = ppc_state.ppc_spr[upper_reg_num] & ~lo_mask;
|
2019-07-02 02:15:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-26 02:30:55 +00:00
|
|
|
static inline uint8_t* calc_pteg_addr(uint32_t hash)
|
2020-01-03 15:17:45 +00:00
|
|
|
{
|
|
|
|
uint32_t sdr1_val, pteg_addr;
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-28 03:39:23 +00:00
|
|
|
sdr1_val = ppc_state.ppc_spr[SPR::SDR1];
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-03 15:17:45 +00:00
|
|
|
pteg_addr = sdr1_val & 0xFE000000;
|
|
|
|
pteg_addr |= (sdr1_val & 0x01FF0000) |
|
2020-01-26 02:30:55 +00:00
|
|
|
(((sdr1_val & 0x1FF) << 16) & ((hash & 0x7FC00) << 6));
|
2020-01-03 15:17:45 +00:00
|
|
|
pteg_addr |= (hash & 0x3FF) << 6;
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-26 22:45:29 +00:00
|
|
|
if (pteg_addr >= last_ptab_area.start && pteg_addr <= last_ptab_area.end) {
|
|
|
|
return last_ptab_area.mem_ptr + (pteg_addr - last_ptab_area.start);
|
2020-01-26 02:30:55 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
AddressMapEntry* entry = mem_ctrl_instance->find_range(pteg_addr);
|
2020-01-03 15:17:45 +00:00
|
|
|
if (entry && entry->type & (RT_ROM | RT_RAM)) {
|
2020-02-04 13:20:10 +00:00
|
|
|
last_ptab_area.start = entry->start;
|
|
|
|
last_ptab_area.end = entry->end;
|
2020-01-26 22:45:29 +00:00
|
|
|
last_ptab_area.mem_ptr = entry->mem_ptr;
|
|
|
|
return last_ptab_area.mem_ptr + (pteg_addr - last_ptab_area.start);
|
2020-01-26 02:30:55 +00:00
|
|
|
}
|
|
|
|
else {
|
2020-01-03 15:17:45 +00:00
|
|
|
printf("SOS: no page table region was found at %08X!\n", pteg_addr);
|
|
|
|
exit(-1); // FIXME: ugly error handling, must be the proper exception!
|
2019-07-02 02:15:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-26 02:30:55 +00:00
|
|
|
static bool search_pteg(uint8_t* pteg_addr, uint8_t** ret_pte_addr,
|
|
|
|
uint32_t vsid, uint16_t page_index, uint8_t pteg_num)
|
2020-01-03 15:17:45 +00:00
|
|
|
{
|
|
|
|
/* construct PTE matching word */
|
|
|
|
uint32_t pte_check = 0x80000000 | (vsid << 7) | (pteg_num << 6) |
|
2020-01-26 02:30:55 +00:00
|
|
|
(page_index >> 10);
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-03 19:28:37 +00:00
|
|
|
#ifdef MMU_INTEGRITY_CHECKS
|
|
|
|
/* PTEG integrity check that ensures that all matching PTEs have
|
|
|
|
identical RPN, WIMG and PP bits (PPC PEM 32-bit 7.6.2, rule 5). */
|
|
|
|
uint32_t pte_word2_check;
|
|
|
|
bool match_found = false;
|
|
|
|
|
|
|
|
for (int i = 0; i < 8; i++, pteg_addr += 8) {
|
2020-01-13 02:04:06 +00:00
|
|
|
if (pte_check == READ_DWORD_BE_A(pteg_addr)) {
|
2020-01-03 19:28:37 +00:00
|
|
|
if (match_found) {
|
2020-01-13 02:04:06 +00:00
|
|
|
if ((READ_DWORD_BE_A(pteg_addr) & 0xFFFFF07B) != pte_word2_check) {
|
2020-01-03 19:28:37 +00:00
|
|
|
printf("Multiple PTEs with different RPN/WIMG/PP found!\n");
|
|
|
|
exit(-1);
|
|
|
|
}
|
2020-01-26 02:30:55 +00:00
|
|
|
}
|
|
|
|
else {
|
2020-01-03 19:28:37 +00:00
|
|
|
/* isolate RPN, WIMG and PP fields */
|
2020-01-13 02:04:06 +00:00
|
|
|
pte_word2_check = READ_DWORD_BE_A(pteg_addr) & 0xFFFFF07B;
|
2020-01-03 19:28:37 +00:00
|
|
|
*ret_pte_addr = pteg_addr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
2020-01-03 15:17:45 +00:00
|
|
|
for (int i = 0; i < 8; i++, pteg_addr += 8) {
|
2020-01-13 02:04:06 +00:00
|
|
|
if (pte_check == READ_DWORD_BE_A(pteg_addr)) {
|
2020-01-03 15:17:45 +00:00
|
|
|
*ret_pte_addr = pteg_addr;
|
|
|
|
return true;
|
2019-07-02 02:15:33 +00:00
|
|
|
}
|
|
|
|
}
|
2020-01-03 19:28:37 +00:00
|
|
|
#endif
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-03 15:17:45 +00:00
|
|
|
return false;
|
2019-07-02 02:15:33 +00:00
|
|
|
}
|
|
|
|
|
2020-01-03 15:17:45 +00:00
|
|
|
static uint32_t page_address_translate(uint32_t la, bool is_instr_fetch,
|
2020-01-26 02:30:55 +00:00
|
|
|
unsigned msr_pr, int is_write)
|
2020-01-03 15:17:45 +00:00
|
|
|
{
|
2020-01-03 19:09:07 +00:00
|
|
|
uint32_t sr_val, page_index, pteg_hash1, vsid, pte_word2;
|
2020-01-03 15:17:45 +00:00
|
|
|
unsigned key, pp;
|
2020-01-26 02:30:55 +00:00
|
|
|
uint8_t* pte_addr;
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-03 15:17:45 +00:00
|
|
|
sr_val = ppc_state.ppc_sr[(la >> 28) & 0x0F];
|
|
|
|
if (sr_val & 0x80000000) {
|
|
|
|
printf("Direct-store segments not supported, LA=%0xX\n", la);
|
|
|
|
exit(-1); // FIXME: ugly error handling, must be the proper exception!
|
|
|
|
}
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-03 15:17:45 +00:00
|
|
|
/* instruction fetch from a no-execute segment will cause ISI exception */
|
|
|
|
if ((sr_val & 0x10000000) && is_instr_fetch) {
|
2020-01-11 20:48:56 +00:00
|
|
|
ppc_exception_handler(Except_Type::EXC_ISI, 0x10000000);
|
2020-01-03 15:17:45 +00:00
|
|
|
}
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-03 15:17:45 +00:00
|
|
|
page_index = (la >> 12) & 0xFFFF;
|
|
|
|
pteg_hash1 = (sr_val & 0x7FFFF) ^ page_index;
|
|
|
|
vsid = sr_val & 0x0FFFFFF;
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-03 15:17:45 +00:00
|
|
|
if (!search_pteg(calc_pteg_addr(pteg_hash1), &pte_addr, vsid, page_index, 0)) {
|
|
|
|
if (!search_pteg(calc_pteg_addr(~pteg_hash1), &pte_addr, vsid, page_index, 1)) {
|
|
|
|
if (is_instr_fetch) {
|
2020-01-11 20:48:56 +00:00
|
|
|
ppc_exception_handler(Except_Type::EXC_ISI, 0x40000000);
|
2020-01-26 02:30:55 +00:00
|
|
|
}
|
|
|
|
else {
|
2020-01-28 03:39:23 +00:00
|
|
|
ppc_state.ppc_spr[SPR::DSISR] = 0x40000000 | (is_write << 25);
|
|
|
|
ppc_state.ppc_spr[SPR::DAR] = la;
|
2020-01-11 20:48:56 +00:00
|
|
|
ppc_exception_handler(Except_Type::EXC_DSI, 0);
|
2019-07-02 02:15:33 +00:00
|
|
|
}
|
|
|
|
}
|
2020-01-03 15:17:45 +00:00
|
|
|
}
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-13 02:04:06 +00:00
|
|
|
pte_word2 = READ_DWORD_BE_A(pte_addr + 4);
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-26 02:30:55 +00:00
|
|
|
key = (((sr_val >> 29) & 1)& msr_pr) | (((sr_val >> 30) & 1)& (msr_pr ^ 1));
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-03 15:17:45 +00:00
|
|
|
/* check page access */
|
|
|
|
pp = pte_word2 & 3;
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-03 15:17:45 +00:00
|
|
|
// the following scenarios cause DSI/ISI exception:
|
|
|
|
// any access with key = 1 and PP = %00
|
|
|
|
// write access with key = 1 and PP = %01
|
|
|
|
// write access with PP = %11
|
|
|
|
if ((key && (!pp || (pp == 1 && is_write))) || (pp == 3 && is_write)) {
|
|
|
|
if (is_instr_fetch) {
|
2020-01-11 20:48:56 +00:00
|
|
|
ppc_exception_handler(Except_Type::EXC_ISI, 0x08000000);
|
2020-01-26 02:30:55 +00:00
|
|
|
}
|
|
|
|
else {
|
2020-01-28 03:39:23 +00:00
|
|
|
ppc_state.ppc_spr[SPR::DSISR] = 0x08000000 | (is_write << 25);
|
|
|
|
ppc_state.ppc_spr[SPR::DAR] = la;
|
2020-01-11 20:48:56 +00:00
|
|
|
ppc_exception_handler(Except_Type::EXC_DSI, 0);
|
2019-07-02 02:15:33 +00:00
|
|
|
}
|
2020-01-03 15:17:45 +00:00
|
|
|
}
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-03 15:17:45 +00:00
|
|
|
/* update R and C bits */
|
|
|
|
/* For simplicity, R is set on each access, C is set only for writes */
|
|
|
|
pte_addr[6] |= 0x01;
|
|
|
|
if (is_write) {
|
|
|
|
pte_addr[7] |= 0x80;
|
|
|
|
}
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-03 15:17:45 +00:00
|
|
|
/* return physical address */
|
|
|
|
return ((pte_word2 & 0xFFFFF000) | (la & 0x00000FFF));
|
2019-07-02 02:15:33 +00:00
|
|
|
}
|
|
|
|
|
2019-08-02 23:05:48 +00:00
|
|
|
/** PowerPC-style MMU instruction address translation. */
|
2020-01-11 20:48:56 +00:00
|
|
|
static uint32_t ppc_mmu_instr_translate(uint32_t la)
|
2019-08-02 00:02:01 +00:00
|
|
|
{
|
2019-08-02 23:05:48 +00:00
|
|
|
uint32_t pa; /* translated physical address */
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2019-08-02 23:05:48 +00:00
|
|
|
bool bat_hit = false;
|
|
|
|
unsigned msr_pr = !!(ppc_state.ppc_msr & 0x4000);
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2019-08-02 23:05:48 +00:00
|
|
|
// Format: %XY
|
|
|
|
// X - supervisor access bit, Y - problem/user access bit
|
|
|
|
// Those bits are mutually exclusive
|
|
|
|
unsigned access_bits = (~msr_pr << 1) | msr_pr;
|
|
|
|
|
2020-01-26 02:30:55 +00:00
|
|
|
for (int bat_index = 0; bat_index < 4; bat_index++) {
|
|
|
|
PPC_BAT_entry* bat_entry = &ibat_array[bat_index];
|
2019-08-02 23:05:48 +00:00
|
|
|
|
|
|
|
if ((bat_entry->access & access_bits) &&
|
|
|
|
((la & ~bat_entry->lo_mask) == bat_entry->bepi)) {
|
|
|
|
bat_hit = true;
|
2020-01-29 23:57:45 +00:00
|
|
|
|
|
|
|
if (!bat_entry->prot) {
|
|
|
|
ppc_exception_handler(Except_Type::EXC_ISI, 0x08000000);
|
|
|
|
}
|
2019-08-02 23:05:48 +00:00
|
|
|
|
|
|
|
// logical to physical translation
|
|
|
|
pa = bat_entry->phys_hi | (la & bat_entry->lo_mask);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2019-08-02 00:02:01 +00:00
|
|
|
|
2020-01-03 15:17:45 +00:00
|
|
|
/* page address translation */
|
|
|
|
if (!bat_hit) {
|
2020-01-11 20:48:56 +00:00
|
|
|
pa = page_address_translate(la, true, msr_pr, 0);
|
2019-08-02 23:05:48 +00:00
|
|
|
}
|
2019-08-02 00:02:01 +00:00
|
|
|
|
2019-08-02 23:05:48 +00:00
|
|
|
return pa;
|
|
|
|
}
|
2019-08-02 00:02:01 +00:00
|
|
|
|
2019-08-02 23:05:48 +00:00
|
|
|
/** PowerPC-style MMU data address translation. */
|
2020-01-11 20:48:56 +00:00
|
|
|
static uint32_t ppc_mmu_addr_translate(uint32_t la, int is_write)
|
2019-08-02 23:05:48 +00:00
|
|
|
{
|
2020-01-28 01:20:18 +00:00
|
|
|
#ifdef PROFILER
|
2020-01-15 03:50:01 +00:00
|
|
|
mmu_translations_num++;
|
|
|
|
#endif
|
|
|
|
|
2019-08-02 23:05:48 +00:00
|
|
|
uint32_t pa; /* translated physical address */
|
2019-08-02 00:02:01 +00:00
|
|
|
|
2019-08-02 23:05:48 +00:00
|
|
|
bool bat_hit = false;
|
|
|
|
unsigned msr_pr = !!(ppc_state.ppc_msr & 0x4000);
|
2019-08-02 00:02:01 +00:00
|
|
|
|
2019-08-02 23:05:48 +00:00
|
|
|
// Format: %XY
|
|
|
|
// X - supervisor access bit, Y - problem/user access bit
|
|
|
|
// Those bits are mutually exclusive
|
|
|
|
unsigned access_bits = (~msr_pr << 1) | msr_pr;
|
|
|
|
|
2020-01-26 02:30:55 +00:00
|
|
|
for (int bat_index = 0; bat_index < 4; bat_index++) {
|
|
|
|
PPC_BAT_entry* bat_entry = &dbat_array[bat_index];
|
2019-08-02 23:05:48 +00:00
|
|
|
|
|
|
|
if ((bat_entry->access & access_bits) &&
|
|
|
|
((la & ~bat_entry->lo_mask) == bat_entry->bepi)) {
|
|
|
|
bat_hit = true;
|
2020-01-29 23:57:45 +00:00
|
|
|
|
|
|
|
if (!bat_entry->prot || ((bat_entry->prot & 1) && is_write)) {
|
|
|
|
ppc_state.ppc_spr[SPR::DSISR] = 0x08000000 | (is_write << 25);
|
|
|
|
ppc_state.ppc_spr[SPR::DAR] = la;
|
|
|
|
ppc_exception_handler(Except_Type::EXC_DSI, 0);
|
|
|
|
}
|
2019-08-02 23:05:48 +00:00
|
|
|
|
|
|
|
// logical to physical translation
|
|
|
|
pa = bat_entry->phys_hi | (la & bat_entry->lo_mask);
|
|
|
|
break;
|
2019-07-02 02:15:33 +00:00
|
|
|
}
|
2019-08-02 23:05:48 +00:00
|
|
|
}
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-03 15:17:45 +00:00
|
|
|
/* page address translation */
|
|
|
|
if (!bat_hit) {
|
|
|
|
pa = page_address_translate(la, false, msr_pr, is_write);
|
2019-07-02 02:15:33 +00:00
|
|
|
}
|
|
|
|
|
2019-08-02 23:05:48 +00:00
|
|
|
return pa;
|
|
|
|
}
|
|
|
|
|
2020-01-29 23:45:39 +00:00
|
|
|
static void mem_write_unaligned(uint32_t addr, uint32_t value, uint32_t size)
|
|
|
|
{
|
|
|
|
printf("WARNING! Attempt to write unaligned %d bytes to 0x%08X\n", size, addr);
|
|
|
|
|
|
|
|
if (((addr & 0xFFF) + size) > 0x1000) {
|
|
|
|
printf("SOS! Cross-page unaligned write, addr=%08X, size=%d\n", addr, size);
|
|
|
|
exit(-1); //FIXME!
|
2020-02-04 13:20:10 +00:00
|
|
|
} else {
|
2020-01-29 23:45:39 +00:00
|
|
|
/* data address translation if enabled */
|
|
|
|
if (ppc_state.ppc_msr & 0x10) {
|
|
|
|
addr = ppc_mmu_addr_translate(addr, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (size == 2) {
|
|
|
|
WRITE_PHYS_MEM(last_write_area, addr, WRITE_WORD_BE_U, value, 2);
|
2020-02-04 13:20:10 +00:00
|
|
|
} else {
|
2020-01-29 23:45:39 +00:00
|
|
|
WRITE_PHYS_MEM(last_write_area, addr, WRITE_DWORD_BE_U, value, 4);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-27 00:36:22 +00:00
|
|
|
void mem_write_byte(uint32_t addr, uint8_t value)
|
2019-08-21 06:33:01 +00:00
|
|
|
{
|
|
|
|
/* data address translation if enabled */
|
2019-08-02 23:05:48 +00:00
|
|
|
if (ppc_state.ppc_msr & 0x10) {
|
2020-01-11 20:48:56 +00:00
|
|
|
addr = ppc_mmu_addr_translate(addr, 1);
|
2019-08-21 06:33:01 +00:00
|
|
|
}
|
|
|
|
|
2020-02-04 13:20:10 +00:00
|
|
|
#define WRITE_BYTE(addr, val) (*(addr) = val)
|
2020-01-26 22:45:29 +00:00
|
|
|
|
|
|
|
WRITE_PHYS_MEM(last_write_area, addr, WRITE_BYTE, value, 1);
|
2019-08-21 06:33:01 +00:00
|
|
|
}
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-27 00:36:22 +00:00
|
|
|
void mem_write_word(uint32_t addr, uint16_t value)
|
2020-01-26 02:30:55 +00:00
|
|
|
{
|
2020-01-29 23:45:39 +00:00
|
|
|
if (addr & 1) {
|
|
|
|
mem_write_unaligned(addr, value, 2);
|
|
|
|
}
|
|
|
|
|
2020-01-26 02:30:55 +00:00
|
|
|
/* data address translation if enabled */
|
|
|
|
if (ppc_state.ppc_msr & 0x10) {
|
|
|
|
addr = ppc_mmu_addr_translate(addr, 1);
|
|
|
|
}
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-26 22:45:29 +00:00
|
|
|
WRITE_PHYS_MEM(last_write_area, addr, WRITE_WORD_BE_A, value, 2);
|
2020-01-26 02:30:55 +00:00
|
|
|
}
|
|
|
|
|
2020-01-27 00:36:22 +00:00
|
|
|
void mem_write_dword(uint32_t addr, uint32_t value)
|
2020-01-26 02:30:55 +00:00
|
|
|
{
|
2020-01-29 23:45:39 +00:00
|
|
|
if (addr & 3) {
|
|
|
|
mem_write_unaligned(addr, value, 4);
|
|
|
|
}
|
|
|
|
|
2020-01-26 02:30:55 +00:00
|
|
|
/* data address translation if enabled */
|
|
|
|
if (ppc_state.ppc_msr & 0x10) {
|
|
|
|
addr = ppc_mmu_addr_translate(addr, 1);
|
|
|
|
}
|
|
|
|
|
2020-01-26 22:45:29 +00:00
|
|
|
WRITE_PHYS_MEM(last_write_area, addr, WRITE_DWORD_BE_A, value, 4);
|
2020-01-26 02:30:55 +00:00
|
|
|
}
|
|
|
|
|
2020-01-27 00:36:22 +00:00
|
|
|
void mem_write_qword(uint32_t addr, uint64_t value)
|
2020-01-26 02:30:55 +00:00
|
|
|
{
|
2020-01-29 23:45:39 +00:00
|
|
|
if (addr & 7) {
|
|
|
|
printf("SOS! Attempt to write unaligned QWORD to 0x%08X\n", addr);
|
|
|
|
exit(-1); //FIXME!
|
|
|
|
}
|
|
|
|
|
2020-01-26 02:30:55 +00:00
|
|
|
/* data address translation if enabled */
|
|
|
|
if (ppc_state.ppc_msr & 0x10) {
|
|
|
|
addr = ppc_mmu_addr_translate(addr, 1);
|
|
|
|
}
|
|
|
|
|
2020-01-26 22:45:29 +00:00
|
|
|
WRITE_PHYS_MEM(last_write_area, addr, WRITE_QWORD_BE_A, value, 8);
|
2020-01-26 02:30:55 +00:00
|
|
|
}
|
2019-07-02 02:15:33 +00:00
|
|
|
|
2020-01-29 23:45:39 +00:00
|
|
|
static uint32_t mem_grab_unaligned(uint32_t addr, uint32_t size)
|
|
|
|
{
|
|
|
|
uint32_t ret = 0;
|
|
|
|
|
|
|
|
printf("WARNING! Attempt to read unaligned %d bytes from 0x%08X\n", size, addr);
|
|
|
|
|
|
|
|
if (((addr & 0xFFF) + size) > 0x1000) {
|
|
|
|
printf("SOS! Cross-page unaligned read, addr=%08X, size=%d\n", addr, size);
|
|
|
|
exit(-1); //FIXME!
|
2020-02-04 13:20:10 +00:00
|
|
|
} else {
|
2020-01-29 23:45:39 +00:00
|
|
|
/* data address translation if enabled */
|
|
|
|
if (ppc_state.ppc_msr & 0x10) {
|
|
|
|
addr = ppc_mmu_addr_translate(addr, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (size == 2) {
|
|
|
|
READ_PHYS_MEM(last_read_area, addr, READ_WORD_BE_U, 2, 0xFFFFU);
|
2020-02-04 13:20:10 +00:00
|
|
|
} else {
|
2020-01-29 23:45:39 +00:00
|
|
|
READ_PHYS_MEM(last_read_area, addr, READ_DWORD_BE_U, 4, 0xFFFFFFFFUL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2019-08-02 23:05:48 +00:00
|
|
|
/** Grab a value from memory into a register */
|
2020-01-27 00:36:22 +00:00
|
|
|
uint8_t mem_grab_byte(uint32_t addr)
|
2020-01-26 02:30:55 +00:00
|
|
|
{
|
2020-01-26 22:45:29 +00:00
|
|
|
uint8_t ret;
|
|
|
|
|
2020-01-26 02:30:55 +00:00
|
|
|
/* data address translation if enabled */
|
|
|
|
if (ppc_state.ppc_msr & 0x10) {
|
|
|
|
addr = ppc_mmu_addr_translate(addr, 0);
|
|
|
|
}
|
|
|
|
|
2020-01-26 22:45:29 +00:00
|
|
|
READ_PHYS_MEM(last_read_area, addr, *, 1, 0xFFU);
|
2020-01-27 00:36:22 +00:00
|
|
|
return ret;
|
2020-01-26 02:30:55 +00:00
|
|
|
}
|
|
|
|
|
2020-01-27 00:36:22 +00:00
|
|
|
uint16_t mem_grab_word(uint32_t addr)
|
2020-01-26 02:30:55 +00:00
|
|
|
{
|
2020-01-26 22:45:29 +00:00
|
|
|
uint16_t ret;
|
|
|
|
|
2020-01-29 23:45:39 +00:00
|
|
|
if (addr & 1) {
|
|
|
|
return mem_grab_unaligned(addr, 2);
|
|
|
|
}
|
|
|
|
|
2020-01-26 02:30:55 +00:00
|
|
|
/* data address translation if enabled */
|
|
|
|
if (ppc_state.ppc_msr & 0x10) {
|
|
|
|
addr = ppc_mmu_addr_translate(addr, 0);
|
|
|
|
}
|
|
|
|
|
2020-01-26 22:45:29 +00:00
|
|
|
READ_PHYS_MEM(last_read_area, addr, READ_WORD_BE_A, 2, 0xFFFFU);
|
2020-01-27 00:36:22 +00:00
|
|
|
return ret;
|
2020-01-26 02:30:55 +00:00
|
|
|
}
|
|
|
|
|
2020-01-27 00:36:22 +00:00
|
|
|
uint32_t mem_grab_dword(uint32_t addr)
|
2019-08-02 23:05:48 +00:00
|
|
|
{
|
2020-01-26 22:45:29 +00:00
|
|
|
uint32_t ret;
|
|
|
|
|
2020-01-29 23:45:39 +00:00
|
|
|
if (addr & 3) {
|
|
|
|
return mem_grab_unaligned(addr, 4);
|
|
|
|
}
|
|
|
|
|
2019-08-02 23:05:48 +00:00
|
|
|
/* data address translation if enabled */
|
|
|
|
if (ppc_state.ppc_msr & 0x10) {
|
2020-01-11 20:48:56 +00:00
|
|
|
addr = ppc_mmu_addr_translate(addr, 0);
|
2019-08-21 06:33:01 +00:00
|
|
|
}
|
|
|
|
|
2020-01-26 22:45:29 +00:00
|
|
|
READ_PHYS_MEM(last_read_area, addr, READ_DWORD_BE_A, 4, 0xFFFFFFFFUL);
|
2020-01-27 00:36:22 +00:00
|
|
|
return ret;
|
2020-01-26 02:30:55 +00:00
|
|
|
}
|
|
|
|
|
2020-01-27 00:36:22 +00:00
|
|
|
uint64_t mem_grab_qword(uint32_t addr)
|
2020-01-26 02:30:55 +00:00
|
|
|
{
|
2020-01-26 22:45:29 +00:00
|
|
|
uint64_t ret;
|
|
|
|
|
2020-01-29 23:45:39 +00:00
|
|
|
if (addr & 7) {
|
|
|
|
printf("SOS! Attempt to read unaligned QWORD at 0x%08X\n", addr);
|
|
|
|
exit(-1); //FIXME!
|
|
|
|
}
|
|
|
|
|
2020-01-26 02:30:55 +00:00
|
|
|
/* data address translation if enabled */
|
|
|
|
if (ppc_state.ppc_msr & 0x10) {
|
|
|
|
addr = ppc_mmu_addr_translate(addr, 0);
|
|
|
|
}
|
|
|
|
|
2020-01-26 22:45:29 +00:00
|
|
|
READ_PHYS_MEM(last_read_area, addr, READ_QWORD_BE_A, 8, 0xFFFFFFFFFFFFFFFFULL);
|
2020-01-27 00:36:22 +00:00
|
|
|
return ret;
|
2019-07-02 02:15:33 +00:00
|
|
|
}
|
|
|
|
|
2020-01-26 02:30:55 +00:00
|
|
|
uint8_t* quickinstruction_translate(uint32_t addr)
|
2019-08-21 06:33:01 +00:00
|
|
|
{
|
2020-01-26 02:30:55 +00:00
|
|
|
uint8_t* real_addr;
|
2020-01-03 15:17:45 +00:00
|
|
|
|
|
|
|
/* perform instruction address translation if enabled */
|
2019-08-02 23:05:48 +00:00
|
|
|
if (ppc_state.ppc_msr & 0x20) {
|
2019-08-21 06:33:01 +00:00
|
|
|
addr = ppc_mmu_instr_translate(addr);
|
2019-07-02 02:15:33 +00:00
|
|
|
}
|
|
|
|
|
2020-01-26 22:45:29 +00:00
|
|
|
if (addr >= last_exec_area.start && addr <= last_exec_area.end) {
|
|
|
|
real_addr = last_exec_area.mem_ptr + (addr - last_exec_area.start);
|
2020-01-03 15:17:45 +00:00
|
|
|
ppc_set_cur_instruction(real_addr);
|
2020-01-26 02:30:55 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
AddressMapEntry* entry = mem_ctrl_instance->find_range(addr);
|
2019-08-21 06:33:01 +00:00
|
|
|
if (entry && entry->type & (RT_ROM | RT_RAM)) {
|
2020-01-26 22:45:29 +00:00
|
|
|
last_exec_area.start = entry->start;
|
|
|
|
last_exec_area.end = entry->end;
|
|
|
|
last_exec_area.mem_ptr = entry->mem_ptr;
|
|
|
|
real_addr = last_exec_area.mem_ptr + (addr - last_exec_area.start);
|
2020-01-03 15:17:45 +00:00
|
|
|
ppc_set_cur_instruction(real_addr);
|
2020-01-26 02:30:55 +00:00
|
|
|
}
|
|
|
|
else {
|
2019-08-21 06:33:01 +00:00
|
|
|
printf("WARNING: attempt to execute code at %08X!\n", addr);
|
2020-01-03 15:17:45 +00:00
|
|
|
exit(-1); // FIXME: ugly error handling, must be the proper exception!
|
2019-07-02 02:15:33 +00:00
|
|
|
}
|
|
|
|
}
|
2020-01-03 15:17:45 +00:00
|
|
|
|
|
|
|
return real_addr;
|
2020-02-04 13:20:10 +00:00
|
|
|
}
|