/* DingusPPC - The Experimental PowerPC Macintosh emulator Copyright (C) 2018-21 divingkatae and maximum (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 . */ /** @file Handling of low-level PPC exceptions. */ #include "ppcemu.h" #include "ppcmmu.h" #include #include #include jmp_buf exc_env; /* Global exception environment. */ void ppc_exception_handler(Except_Type exception_type, uint32_t srr1_bits) { #ifdef CPU_PROFILING exceptions_processed++; #endif 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: if (!(ppc_state.msr & 0x1000)) { /* 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: ppc_state.spr[SPR::SRR0] = ppc_next_instruction_address; ppc_next_instruction_address = 0x0400; break; case Except_Type::EXC_EXT_INT: if (bb_kind != BB_end_kind::BB_NONE) { ppc_state.spr[SPR::SRR0] = ppc_next_instruction_address; } else { ppc_state.spr[SPR::SRR0] = (ppc_state.pc & 0xFFFFFFFCUL) + 4; } 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: ppc_state.spr[SPR::SRR0] = (ppc_state.pc & 0xFFFFFFFC) + 4; 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: // printf("Unknown exception occured: %X\n", exception_type); // exit(-1); break; } ppc_state.spr[SPR::SRR1] = (ppc_state.msr & 0x0000FF73) | srr1_bits; ppc_state.msr &= 0xFFFB1041; /* copy MSR[ILE] to MSR[LE] */ ppc_state.msr = (ppc_state.msr & 0xFFFFFFFE) | ((ppc_state.msr >> 16) & 1); if (ppc_state.msr & 0x40) { ppc_next_instruction_address |= 0xFFF00000; } bb_kind = BB_end_kind::BB_EXCEPTION; // perform context synchronization for recoverable exceptions if (exception_type != Except_Type::EXC_MACHINE_CHECK && exception_type != Except_Type::EXC_SYSTEM_RESET) { do_ctx_sync(); } mmu_change_mode(); if (exception_type != Except_Type::EXC_EXT_INT) { longjmp(exc_env, 2); /* return to the main execution loop. */ } } [[noreturn]] void dbg_exception_handler(Except_Type exception_type, uint32_t srr1_bits) { std::string exc_descriptor; switch (exception_type) { case Except_Type::EXC_SYSTEM_RESET: exc_descriptor = "System reset exception occured"; break; case Except_Type::EXC_MACHINE_CHECK: exc_descriptor = "Machine check exception occured"; 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: exc_descriptor = "External interrupt exception occured"; break; case Except_Type::EXC_ALIGNMENT: exc_descriptor = "Alignment exception occured"; break; case Except_Type::EXC_PROGRAM: exc_descriptor = "Program exception occured"; break; case Except_Type::EXC_NO_FPU: exc_descriptor = "Floating-Point unavailable exception occured"; break; case Except_Type::EXC_DECR: exc_descriptor = "Decrementer exception occured"; break; case Except_Type::EXC_SYSCALL: exc_descriptor = "Syscall exception occured"; break; case Except_Type::EXC_TRACE: exc_descriptor = "Trace exception occured"; break; } throw std::invalid_argument(exc_descriptor); }