2020-02-28 09:04:28 -07:00
|
|
|
|
/*
|
|
|
|
|
DingusPPC - The Experimental PowerPC Macintosh emulator
|
2023-07-10 14:06:20 +02:00
|
|
|
|
Copyright (C) 2018-23 divingkatae and maximum
|
2020-02-28 09:04:28 -07:00
|
|
|
|
(theweirdo) spatium
|
|
|
|
|
|
|
|
|
|
(Contact divingkatae#1017 or powermax#2286 on Discord for more info)
|
|
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
|
*/
|
|
|
|
|
|
2020-01-11 21:48:56 +01:00
|
|
|
|
/** @file Handling of low-level PPC exceptions. */
|
|
|
|
|
|
2022-03-02 16:55:20 +01:00
|
|
|
|
#include <loguru.hpp>
|
2020-05-12 23:55:45 +05:00
|
|
|
|
#include "ppcemu.h"
|
2021-05-16 00:53:15 +02:00
|
|
|
|
#include "ppcmmu.h"
|
2022-03-02 16:55:20 +01:00
|
|
|
|
|
2020-01-11 21:48:56 +01:00
|
|
|
|
#include <setjmp.h>
|
2020-02-23 16:41:44 +01:00
|
|
|
|
#include <stdexcept>
|
2020-05-12 23:55:45 +05:00
|
|
|
|
#include <string>
|
2020-01-11 21:48:56 +01:00
|
|
|
|
|
|
|
|
|
jmp_buf exc_env; /* Global exception environment. */
|
|
|
|
|
|
2022-01-10 17:27:27 +01:00
|
|
|
|
void ppc_exception_handler(Except_Type exception_type, uint32_t srr1_bits) {
|
2021-04-29 02:26:17 +02:00
|
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
|
exceptions_processed++;
|
2020-05-12 23:55:45 +05:00
|
|
|
|
#endif
|
2020-01-11 21:48:56 +01:00
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
|
switch (exception_type) {
|
|
|
|
|
case Except_Type::EXC_SYSTEM_RESET:
|
|
|
|
|
ppc_state.spr[SPR::SRR0] = ppc_state.pc & 0xFFFFFFFC;
|
|
|
|
|
ppc_next_instruction_address = 0x0100;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_MACHINE_CHECK:
|
2023-11-23 16:56:58 -07:00
|
|
|
|
if (!(ppc_state.msr & MSR::ME)) {
|
2020-05-12 23:55:45 +05:00
|
|
|
|
/* TODO: handle internal checkstop */
|
|
|
|
|
}
|
|
|
|
|
ppc_state.spr[SPR::SRR0] = ppc_state.pc & 0xFFFFFFFC;
|
|
|
|
|
ppc_next_instruction_address = 0x0200;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_DSI:
|
|
|
|
|
ppc_state.spr[SPR::SRR0] = ppc_state.pc & 0xFFFFFFFC;
|
|
|
|
|
ppc_next_instruction_address = 0x0300;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_ISI:
|
2024-02-21 23:10:14 -08:00
|
|
|
|
if (exec_flags) {
|
2023-07-10 14:06:20 +02:00
|
|
|
|
ppc_state.spr[SPR::SRR0] = ppc_next_instruction_address;
|
|
|
|
|
} else {
|
|
|
|
|
ppc_state.spr[SPR::SRR0] = ppc_state.pc & 0xFFFFFFFCUL;
|
|
|
|
|
}
|
2020-05-12 23:55:45 +05:00
|
|
|
|
ppc_next_instruction_address = 0x0400;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_EXT_INT:
|
2024-02-21 23:10:14 -08:00
|
|
|
|
if (exec_flags) {
|
2022-01-20 01:32:45 +01:00
|
|
|
|
ppc_state.spr[SPR::SRR0] = ppc_next_instruction_address;
|
|
|
|
|
} else {
|
|
|
|
|
ppc_state.spr[SPR::SRR0] = (ppc_state.pc & 0xFFFFFFFCUL) + 4;
|
|
|
|
|
}
|
2020-05-12 23:55:45 +05:00
|
|
|
|
ppc_next_instruction_address = 0x0500;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_ALIGNMENT:
|
|
|
|
|
ppc_state.spr[SPR::SRR0] = ppc_state.pc & 0xFFFFFFFC;
|
|
|
|
|
ppc_next_instruction_address = 0x0600;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_PROGRAM:
|
|
|
|
|
ppc_state.spr[SPR::SRR0] = ppc_state.pc & 0xFFFFFFFC;
|
|
|
|
|
ppc_next_instruction_address = 0x0700;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_NO_FPU:
|
|
|
|
|
ppc_state.spr[SPR::SRR0] = ppc_state.pc & 0xFFFFFFFC;
|
|
|
|
|
ppc_next_instruction_address = 0x0800;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_DECR:
|
2024-02-21 23:10:14 -08:00
|
|
|
|
if (exec_flags) {
|
2023-08-09 12:53:48 +02:00
|
|
|
|
ppc_state.spr[SPR::SRR0] = ppc_next_instruction_address;
|
|
|
|
|
} else {
|
|
|
|
|
ppc_state.spr[SPR::SRR0] = (ppc_state.pc & 0xFFFFFFFCUL) + 4;
|
|
|
|
|
}
|
2020-05-12 23:55:45 +05:00
|
|
|
|
ppc_next_instruction_address = 0x0900;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_SYSCALL:
|
|
|
|
|
ppc_state.spr[SPR::SRR0] = (ppc_state.pc & 0xFFFFFFFC) + 4;
|
|
|
|
|
ppc_next_instruction_address = 0x0C00;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_TRACE:
|
|
|
|
|
ppc_state.spr[SPR::SRR0] = (ppc_state.pc & 0xFFFFFFFC) + 4;
|
|
|
|
|
ppc_next_instruction_address = 0x0D00;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default:
|
2023-08-07 22:47:26 -07:00
|
|
|
|
ABORT_F("Unknown exception occurred: %X\n", (unsigned)exception_type);
|
2020-05-12 23:55:45 +05:00
|
|
|
|
break;
|
2020-01-11 21:48:56 +01:00
|
|
|
|
}
|
|
|
|
|
|
2020-03-04 21:29:04 -07:00
|
|
|
|
ppc_state.spr[SPR::SRR1] = (ppc_state.msr & 0x0000FF73) | srr1_bits;
|
|
|
|
|
ppc_state.msr &= 0xFFFB1041;
|
2020-01-11 21:48:56 +01:00
|
|
|
|
/* copy MSR[ILE] to MSR[LE] */
|
2024-03-20 01:29:44 -07:00
|
|
|
|
ppc_state.msr = (ppc_state.msr & ~MSR::LE) | !!(ppc_state.msr & MSR::ILE);
|
2020-01-11 21:48:56 +01:00
|
|
|
|
|
2023-11-23 16:56:58 -07:00
|
|
|
|
if (ppc_state.msr & MSR::IP) {
|
2020-01-11 21:48:56 +01:00
|
|
|
|
ppc_next_instruction_address |= 0xFFF00000;
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-02 16:55:20 +01:00
|
|
|
|
exec_flags = EXEF_EXCEPTION;
|
2022-01-20 01:32:45 +01:00
|
|
|
|
|
2021-10-13 20:58:09 +02:00
|
|
|
|
// perform context synchronization for recoverable exceptions
|
|
|
|
|
if (exception_type != Except_Type::EXC_MACHINE_CHECK &&
|
|
|
|
|
exception_type != Except_Type::EXC_SYSTEM_RESET) {
|
|
|
|
|
do_ctx_sync();
|
|
|
|
|
}
|
|
|
|
|
|
2021-05-16 00:53:15 +02:00
|
|
|
|
mmu_change_mode();
|
|
|
|
|
|
2023-08-07 11:11:02 -07:00
|
|
|
|
if (exception_type != Except_Type::EXC_EXT_INT && exception_type != Except_Type::EXC_DECR) {
|
2022-01-10 17:27:27 +01:00
|
|
|
|
longjmp(exc_env, 2); /* return to the main execution loop. */
|
|
|
|
|
}
|
2020-02-04 14:20:10 +01:00
|
|
|
|
}
|
2020-02-23 16:41:44 +01:00
|
|
|
|
|
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
|
[[noreturn]] void dbg_exception_handler(Except_Type exception_type, uint32_t srr1_bits) {
|
2020-02-23 16:41:44 +01:00
|
|
|
|
std::string exc_descriptor;
|
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
|
switch (exception_type) {
|
|
|
|
|
case Except_Type::EXC_SYSTEM_RESET:
|
2023-08-07 22:47:26 -07:00
|
|
|
|
exc_descriptor = "System reset exception occurred";
|
2020-05-12 23:55:45 +05:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_MACHINE_CHECK:
|
2023-08-07 22:47:26 -07:00
|
|
|
|
exc_descriptor = "Machine check exception occurred";
|
2020-05-12 23:55:45 +05:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_DSI:
|
|
|
|
|
case Except_Type::EXC_ISI:
|
|
|
|
|
if (ppc_state.spr[SPR::DSISR] & 0x40000000)
|
|
|
|
|
exc_descriptor = "DSI/ISI exception: unmapped memory access";
|
|
|
|
|
else if (ppc_state.spr[SPR::DSISR] & 0x08000000)
|
|
|
|
|
exc_descriptor = "DSI/ISI exception: access protection violation";
|
|
|
|
|
else {
|
|
|
|
|
if (exception_type == Except_Type::EXC_DSI)
|
|
|
|
|
exc_descriptor = "DSI exception";
|
|
|
|
|
else
|
|
|
|
|
exc_descriptor = "ISI exception";
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_EXT_INT:
|
2023-08-07 22:47:26 -07:00
|
|
|
|
exc_descriptor = "External interrupt exception occurred";
|
2020-05-12 23:55:45 +05:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_ALIGNMENT:
|
2023-08-07 22:47:26 -07:00
|
|
|
|
exc_descriptor = "Alignment exception occurred";
|
2020-05-12 23:55:45 +05:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_PROGRAM:
|
2023-08-07 22:47:26 -07:00
|
|
|
|
exc_descriptor = "Program exception occurred";
|
2020-05-12 23:55:45 +05:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_NO_FPU:
|
2023-08-07 22:47:26 -07:00
|
|
|
|
exc_descriptor = "Floating-Point unavailable exception occurred";
|
2020-05-12 23:55:45 +05:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_DECR:
|
2023-08-07 22:47:26 -07:00
|
|
|
|
exc_descriptor = "Decrementer exception occurred";
|
2020-05-12 23:55:45 +05:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_SYSCALL:
|
2023-08-07 22:47:26 -07:00
|
|
|
|
exc_descriptor = "Syscall exception occurred";
|
2020-05-12 23:55:45 +05:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case Except_Type::EXC_TRACE:
|
2023-08-07 22:47:26 -07:00
|
|
|
|
exc_descriptor = "Trace exception occurred";
|
2020-05-12 23:55:45 +05:00
|
|
|
|
break;
|
2020-02-23 16:41:44 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
throw std::invalid_argument(exc_descriptor);
|
|
|
|
|
}
|
2023-12-17 05:06:38 -08:00
|
|
|
|
|
|
|
|
|
void ppc_floating_point_exception() {
|
|
|
|
|
LOG_F(ERROR, "Floating point exception at 0x%08x for instruction 0x%08x",
|
|
|
|
|
ppc_state.pc, ppc_cur_instruction);
|
|
|
|
|
// mmu_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::FPU_EXCEPTION);
|
|
|
|
|
}
|
2023-12-16 05:34:43 -08:00
|
|
|
|
|
|
|
|
|
void ppc_alignment_exception(uint32_t ea)
|
|
|
|
|
{
|
|
|
|
|
uint32_t dsisr;
|
|
|
|
|
|
|
|
|
|
switch (ppc_cur_instruction & 0xfc000000) {
|
|
|
|
|
case 0x80000000: // lwz
|
|
|
|
|
case 0x90000000: // stw
|
|
|
|
|
case 0xa0000000: // lhz
|
|
|
|
|
case 0xa8000000: // lha
|
|
|
|
|
case 0xb0000000: // sth
|
|
|
|
|
case 0xb8000000: // lmw
|
|
|
|
|
case 0xc0000000: // lfs
|
|
|
|
|
case 0xc8000000: // lfd
|
|
|
|
|
case 0xd0000000: // stfs
|
|
|
|
|
case 0xd8000000: // stfd
|
|
|
|
|
case 0x84000000: // lwzu
|
|
|
|
|
case 0x94000000: // stwu
|
|
|
|
|
case 0xa4000000: // lhzu
|
|
|
|
|
case 0xac000000: // lhau
|
|
|
|
|
case 0xb4000000: // sthu
|
|
|
|
|
case 0xbc000000: // stmw
|
|
|
|
|
case 0xc4000000: // lfsu
|
|
|
|
|
case 0xcc000000: // lfdu
|
|
|
|
|
case 0xd4000000: // stfsu
|
|
|
|
|
case 0xdc000000: // stfdu
|
2024-08-27 04:11:15 -07:00
|
|
|
|
// indirect with immediate index
|
2023-12-16 05:34:43 -08:00
|
|
|
|
dsisr = ((ppc_cur_instruction >> 12) & 0x00004000) // bit 17 — Set to bit 5 of the instruction.
|
|
|
|
|
| ((ppc_cur_instruction >> 17) & 0x00003c00); // bits 18–21 - set to bits 1–4 of the instruction.
|
|
|
|
|
break;
|
|
|
|
|
case 0x7c000000:
|
|
|
|
|
switch (ppc_cur_instruction & 0xfc0007ff) {
|
|
|
|
|
case 0x7c000028: // lwarx (invalid form - bits 15-21 of DSISR are identical to those of lwz)
|
|
|
|
|
case 0x7c0002aa: // lwax (64-bit only)
|
|
|
|
|
case 0x7c00042a: // lswx
|
|
|
|
|
case 0x7c0004aa: // lswi
|
|
|
|
|
case 0x7c00052a: // stswx
|
|
|
|
|
case 0x7c0005aa: // stswi
|
|
|
|
|
case 0x7c0002ea: // lwaux (64 bit only)
|
|
|
|
|
case 0x7c00012c: // stwcx
|
|
|
|
|
case 0x7c00042c: // lwbrx
|
|
|
|
|
case 0x7c00052c: // stwbrx
|
|
|
|
|
case 0x7c00062c: // lhbrx
|
|
|
|
|
case 0x7c00072c: // sthbrx
|
|
|
|
|
case 0x7c00026c: // eciwx // MPC7451
|
|
|
|
|
case 0x7c00036c: // ecowx // MPC7451
|
|
|
|
|
case 0x7c00002e: // lwzx
|
|
|
|
|
case 0x7c00012e: // stwx
|
|
|
|
|
case 0x7c00022e: // lhzx
|
|
|
|
|
case 0x7c0002ae: // lhax
|
|
|
|
|
case 0x7c00032e: // sthx
|
|
|
|
|
case 0x7c00042e: // lfsx
|
|
|
|
|
case 0x7c0004ae: // lfdx
|
|
|
|
|
case 0x7c00052e: // stfsx
|
|
|
|
|
case 0x7c0005ae: // stfdx
|
|
|
|
|
case 0x7c00006e: // lwzux
|
|
|
|
|
case 0x7c00016e: // stwux
|
|
|
|
|
case 0x7c00026e: // lhzux
|
|
|
|
|
case 0x7c0002ee: // lhaux
|
|
|
|
|
case 0x7c00036e: // sthux
|
|
|
|
|
case 0x7c00046e: // lfsux
|
|
|
|
|
case 0x7c0004ee: // lfdux
|
|
|
|
|
case 0x7c00056e: // stfsux
|
|
|
|
|
case 0x7c0005ee: // stfdux
|
|
|
|
|
indirect_with_index:
|
|
|
|
|
dsisr = ((ppc_cur_instruction << 14) & 0x00018000) // bits 15–16 - set to bits 29–30 of the instruction.
|
|
|
|
|
| ((ppc_cur_instruction << 8) & 0x00004000) // bit 17 - set to bit 25 of the instruction.
|
|
|
|
|
| ((ppc_cur_instruction << 3) & 0x00003c00); // bits 18–21 - set to bits 21–24 of the instruction.
|
|
|
|
|
break;
|
|
|
|
|
case 0x7c0007ec:
|
|
|
|
|
if ((ppc_cur_instruction & 0xffe007ff) == 0x7c0007ec) // dcbz
|
|
|
|
|
goto indirect_with_index;
|
|
|
|
|
/* fallthrough */
|
|
|
|
|
default:
|
|
|
|
|
goto unexpected_instruction;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
unexpected_instruction:
|
|
|
|
|
dsisr = 0;
|
|
|
|
|
LOG_F(ERROR, "Alignment exception from unexpected instruction 0x%08x",
|
|
|
|
|
ppc_cur_instruction);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// bits 22–26 - Set to bits 6–10 (source or destination) of the instruction.
|
|
|
|
|
// Undefined for dcbz.
|
|
|
|
|
dsisr |= ((ppc_cur_instruction >> 16) & 0x000003e0);
|
|
|
|
|
|
|
|
|
|
if ((ppc_cur_instruction & 0xfc000000) == 0xb8000000) { // lmw
|
|
|
|
|
LOG_F(ERROR, "Alignment exception from instruction 0x%08x (lmw). "
|
|
|
|
|
"What to set DSISR bits 27-31?", ppc_cur_instruction);
|
|
|
|
|
// dsisr |= ((ppc_cur_instruction >> ?) & 0x0000001f); // bits 27–31
|
|
|
|
|
}
|
|
|
|
|
else if ((ppc_cur_instruction & 0xfc0007ff) == 0x7c0004aa) { // lswi
|
|
|
|
|
LOG_F(ERROR, "Alignment exception from instruction 0x%08x (lswi). "
|
|
|
|
|
"What to set DSISR bits 27-31?", ppc_cur_instruction);
|
|
|
|
|
// dsisr |= ((ppc_cur_instruction >> ?) & 0x0000001f); // bits 27–31
|
|
|
|
|
}
|
|
|
|
|
else if ((ppc_cur_instruction & 0xfc0007ff) == 0x7c00042a) { // lswx
|
|
|
|
|
LOG_F(ERROR, "Alignment exception from instruction 0x%08x (lswx). "
|
|
|
|
|
"What to set DSISR bits 27-31?", ppc_cur_instruction);
|
|
|
|
|
// dsisr |= ((ppc_cur_instruction >> ?) & 0x0000001f); // bits 27–31
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
// bits 27–31 - Set to bits 11–15 of the instruction (rA)
|
|
|
|
|
dsisr |= ((ppc_cur_instruction >> 16) & 0x0000001f);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ppc_state.spr[SPR::DSISR] = dsisr;
|
|
|
|
|
ppc_state.spr[SPR::DAR] = ea;
|
2024-02-20 18:22:55 -07:00
|
|
|
|
ppc_exception_handler(Except_Type::EXC_ALIGNMENT, 0x0);
|
2023-12-16 05:34:43 -08:00
|
|
|
|
}
|