2020-02-28 09:04:28 -07:00
|
|
|
/*
|
|
|
|
DingusPPC - The Experimental PowerPC Macintosh emulator
|
2023-12-24 02:35:38 +01: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/>.
|
|
|
|
*/
|
2019-07-01 19:15:33 -07:00
|
|
|
|
|
|
|
// General opcodes for the processor - ppcopcodes.cpp
|
|
|
|
|
2022-01-10 16:36:14 +01:00
|
|
|
#include <core/timermanager.h>
|
2022-09-15 21:22:37 -07:00
|
|
|
#include <core/mathutils.h>
|
2020-05-12 23:55:45 +05:00
|
|
|
#include "ppcemu.h"
|
|
|
|
#include "ppcmmu.h"
|
|
|
|
#include <cinttypes>
|
2021-08-03 16:01:32 +02:00
|
|
|
#include <vector>
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
uint32_t reg_s;
|
|
|
|
uint32_t reg_d;
|
|
|
|
uint32_t reg_a;
|
|
|
|
uint32_t reg_b;
|
2020-05-12 23:55:45 +05:00
|
|
|
uint32_t reg_c; // used only for floating point multiplication operations
|
2020-01-25 19:30:55 -07:00
|
|
|
uint32_t uimm;
|
|
|
|
int32_t simm;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
// Used for GP calcs
|
2020-01-25 19:30:55 -07:00
|
|
|
uint32_t ppc_result_a = 0;
|
|
|
|
uint32_t ppc_result_b = 0;
|
|
|
|
uint32_t ppc_result_d = 0;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
Extract the registers desired and the values of the registers
|
|
|
|
This also takes the MSR into account, mainly to determine
|
|
|
|
what endian the numbers are to be stored in.
|
|
|
|
**/
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
// Storage and register retrieval functions for the integer functions.
|
2021-01-24 22:50:07 +01:00
|
|
|
void ppc_store_result_regd() {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.gpr[reg_d] = ppc_result_d;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2021-01-24 22:50:07 +01:00
|
|
|
void ppc_store_result_rega() {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.gpr[reg_a] = ppc_result_a;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2021-10-05 00:40:12 +02:00
|
|
|
void ppc_grab_regsdasimm() {
|
2020-05-12 23:55:45 +05:00
|
|
|
reg_d = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
reg_a = (ppc_cur_instruction >> 16) & 31;
|
2024-01-31 08:06:33 -07:00
|
|
|
simm = int32_t(int16_t(ppc_cur_instruction));
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_result_a = ppc_state.gpr[reg_a];
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2021-01-24 11:59:16 -07:00
|
|
|
inline void ppc_grab_regsdauimm() {
|
2020-05-12 23:55:45 +05:00
|
|
|
reg_d = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
reg_a = (ppc_cur_instruction >> 16) & 31;
|
2024-01-07 17:04:51 -07:00
|
|
|
uimm = ppc_cur_instruction & 0xFFFFUL;
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_result_a = ppc_state.gpr[reg_a];
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2021-01-24 11:59:16 -07:00
|
|
|
inline void ppc_grab_regsasimm() {
|
2020-05-12 23:55:45 +05:00
|
|
|
reg_a = (ppc_cur_instruction >> 16) & 31;
|
2024-01-31 08:06:33 -07:00
|
|
|
simm = int32_t(int16_t(ppc_cur_instruction));
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_result_a = ppc_state.gpr[reg_a];
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2021-01-24 11:59:16 -07:00
|
|
|
inline void ppc_grab_regssauimm() {
|
2020-05-12 23:55:45 +05:00
|
|
|
reg_s = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
reg_a = (ppc_cur_instruction >> 16) & 31;
|
2024-01-07 17:04:51 -07:00
|
|
|
uimm = ppc_cur_instruction & 0xFFFFUL;
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_result_d = ppc_state.gpr[reg_s];
|
|
|
|
ppc_result_a = ppc_state.gpr[reg_a];
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2023-12-17 05:31:16 -08:00
|
|
|
inline void ppc_grab_dab() {
|
|
|
|
reg_d = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
reg_a = (ppc_cur_instruction >> 16) & 31;
|
|
|
|
reg_b = (ppc_cur_instruction >> 11) & 31;
|
|
|
|
}
|
|
|
|
|
2021-01-24 22:50:07 +01:00
|
|
|
void ppc_grab_regsdab() {
|
2020-05-12 23:55:45 +05:00
|
|
|
reg_d = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
reg_a = (ppc_cur_instruction >> 16) & 31;
|
|
|
|
reg_b = (ppc_cur_instruction >> 11) & 31;
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_result_a = ppc_state.gpr[reg_a];
|
|
|
|
ppc_result_b = ppc_state.gpr[reg_b];
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2021-01-24 22:50:07 +01:00
|
|
|
void ppc_grab_regssab() {
|
2020-05-12 23:55:45 +05:00
|
|
|
reg_s = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
reg_a = (ppc_cur_instruction >> 16) & 31;
|
|
|
|
reg_b = (ppc_cur_instruction >> 11) & 31;
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_result_d = ppc_state.gpr[reg_s];
|
|
|
|
ppc_result_a = ppc_state.gpr[reg_a];
|
|
|
|
ppc_result_b = ppc_state.gpr[reg_b];
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2021-01-24 22:50:07 +01:00
|
|
|
void ppc_grab_regssa() {
|
2020-05-12 23:55:45 +05:00
|
|
|
reg_s = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
reg_a = (ppc_cur_instruction >> 16) & 31;
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_result_d = ppc_state.gpr[reg_s];
|
|
|
|
ppc_result_a = ppc_state.gpr[reg_a];
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2021-01-24 11:59:16 -07:00
|
|
|
inline void ppc_grab_regssb() {
|
2020-05-12 23:55:45 +05:00
|
|
|
reg_s = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
reg_b = (ppc_cur_instruction >> 11) & 31;
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_result_d = ppc_state.gpr[reg_s];
|
|
|
|
ppc_result_b = ppc_state.gpr[reg_b];
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2021-01-24 22:50:07 +01:00
|
|
|
void ppc_grab_regsda() {
|
2020-05-12 23:55:45 +05:00
|
|
|
reg_d = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
reg_a = (ppc_cur_instruction >> 16) & 31;
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_result_a = ppc_state.gpr[reg_a];
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2021-01-24 11:59:16 -07:00
|
|
|
inline void ppc_grab_regsdb() {
|
2020-05-12 23:55:45 +05:00
|
|
|
reg_d = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
reg_b = (ppc_cur_instruction >> 11) & 31;
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_result_b = ppc_state.gpr[reg_b];
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
// Affects CR Field 0 - For integer operations
|
2020-01-25 19:30:55 -07:00
|
|
|
void ppc_changecrf0(uint32_t set_result) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr &= 0x0FFFFFFFUL;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if (set_result == 0) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr |= 0x20000000UL;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-02-10 23:07:39 +01:00
|
|
|
if (set_result & 0x80000000) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr |= 0x80000000UL;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr |= 0x40000000UL;
|
2020-02-10 23:07:39 +01:00
|
|
|
}
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-02-10 23:07:39 +01:00
|
|
|
/* copy XER[SO] into CR0[SO]. */
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr |= (ppc_state.spr[SPR::XER] >> 3) & 0x10000000UL;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
// Affects the XER register's Carry Bit
|
2020-02-09 08:01:26 +01:00
|
|
|
inline void ppc_carry(uint32_t a, uint32_t b) {
|
|
|
|
if (b < a) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.spr[SPR::XER] |= 0x20000000UL;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.spr[SPR::XER] &= 0xDFFFFFFFUL;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-09 07:41:03 +01:00
|
|
|
inline void ppc_carry_sub(uint32_t a, uint32_t b) {
|
2020-02-09 08:01:26 +01:00
|
|
|
if (b >= a) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.spr[SPR::XER] |= 0x20000000UL;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.spr[SPR::XER] &= 0xDFFFFFFFUL;
|
2020-02-09 07:41:03 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
// Affects the XER register's SO and OV Bits
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-02-03 19:42:33 +01:00
|
|
|
inline void ppc_setsoov(uint32_t a, uint32_t b, uint32_t d) {
|
|
|
|
if ((a ^ b) & (a ^ d) & 0x80000000UL) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.spr[SPR::XER] |= 0xC0000000UL;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.spr[SPR::XER] &= 0xBFFFFFFFUL;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-03 16:01:32 +02:00
|
|
|
typedef std::function<void()> CtxSyncCallback;
|
|
|
|
std::vector<CtxSyncCallback> gCtxSyncCallbacks;
|
|
|
|
|
|
|
|
// perform context synchronization by executing registered actions if any
|
|
|
|
void do_ctx_sync() {
|
|
|
|
while (!gCtxSyncCallbacks.empty()) {
|
|
|
|
gCtxSyncCallbacks.back()();
|
|
|
|
gCtxSyncCallbacks.pop_back();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void add_ctx_sync_action(const CtxSyncCallback &cb) {
|
|
|
|
gCtxSyncCallbacks.push_back(cb);
|
|
|
|
}
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
/**
|
|
|
|
The core functionality of this PPC emulation is within all of these void functions.
|
2020-05-12 23:55:45 +05:00
|
|
|
This is where the opcode tables in the ppcemumain.h come into play - reducing the number of
|
|
|
|
comparisons needed. This means loads of functions, but less CPU cycles needed to determine the
|
|
|
|
function (theoretically).
|
2019-07-01 19:15:33 -07:00
|
|
|
**/
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_addi() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdasimm();
|
2020-01-25 19:30:55 -07:00
|
|
|
ppc_result_d = (reg_a == 0) ? simm : (ppc_result_a + simm);
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_addic() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdasimm();
|
|
|
|
ppc_result_d = (ppc_result_a + simm);
|
|
|
|
ppc_carry(ppc_result_a, ppc_result_d);
|
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_addicdot() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdasimm();
|
|
|
|
ppc_result_d = (ppc_result_a + simm);
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
|
|
|
ppc_carry(ppc_result_a, ppc_result_d);
|
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_addis() {
|
2020-01-25 19:30:55 -07:00
|
|
|
ppc_grab_regsdasimm();
|
|
|
|
ppc_result_d = (reg_a == 0) ? (simm << 16) : (ppc_result_a + (simm << 16));
|
|
|
|
ppc_store_result_regd();
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_add() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
|
|
|
ppc_result_d = ppc_result_a + ppc_result_b;
|
2020-10-17 14:30:37 -07:00
|
|
|
if (oe_flag)
|
|
|
|
ppc_setsoov(ppc_result_a, ~ppc_result_b, ppc_result_d);
|
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_addc() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
|
|
|
ppc_result_d = ppc_result_a + ppc_result_b;
|
|
|
|
ppc_carry(ppc_result_a, ppc_result_d);
|
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (oe_flag)
|
|
|
|
ppc_setsoov(ppc_result_a, ~ppc_result_b, ppc_result_d);
|
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
|
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_adde() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2020-03-04 21:29:04 -07:00
|
|
|
uint32_t xer_ca = !!(ppc_state.spr[SPR::XER] & 0x20000000);
|
2020-05-12 23:55:45 +05:00
|
|
|
ppc_result_d = ppc_result_a + ppc_result_b + xer_ca;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if ((ppc_result_d < ppc_result_a) || (xer_ca && (ppc_result_d == ppc_result_a))) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.spr[SPR::XER] |= 0x20000000UL;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.spr[SPR::XER] &= 0xDFFFFFFFUL;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (oe_flag)
|
|
|
|
ppc_setsoov(ppc_result_a, ~ppc_result_b, ppc_result_d);
|
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
|
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_addme() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsda();
|
2020-03-04 21:29:04 -07:00
|
|
|
uint32_t xer_ca = !!(ppc_state.spr[SPR::XER] & 0x20000000);
|
2020-05-12 23:55:45 +05:00
|
|
|
ppc_result_d = ppc_result_a + xer_ca - 1;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2021-01-23 15:10:08 -07:00
|
|
|
if (((xer_ca - 1) < 0xFFFFFFFFUL) || (ppc_result_d < ppc_result_a)) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.spr[SPR::XER] |= 0x20000000UL;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.spr[SPR::XER] &= 0xDFFFFFFFUL;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (oe_flag)
|
|
|
|
ppc_setsoov(ppc_result_a, 0, ppc_result_d);
|
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
|
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_addze() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsda();
|
2020-03-04 21:29:04 -07:00
|
|
|
uint32_t grab_xer = !!(ppc_state.spr[SPR::XER] & 0x20000000);
|
2020-05-12 23:55:45 +05:00
|
|
|
ppc_result_d = ppc_result_a + grab_xer;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if (ppc_result_d < ppc_result_a) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.spr[SPR::XER] |= 0x20000000UL;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.spr[SPR::XER] &= 0xDFFFFFFFUL;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2020-11-29 23:53:03 +01:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (oe_flag)
|
|
|
|
ppc_setsoov(ppc_result_a, 0xFFFFFFFFUL, ppc_result_d);
|
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
|
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_subf() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2020-02-09 08:04:38 +01:00
|
|
|
ppc_result_d = ppc_result_b - ppc_result_a;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (oe_flag)
|
|
|
|
ppc_setsoov(ppc_result_b, ppc_result_a, ppc_result_d);
|
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
|
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_subfc() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2020-02-03 22:32:56 -07:00
|
|
|
ppc_result_d = ppc_result_b - ppc_result_a;
|
2020-02-09 07:41:03 +01:00
|
|
|
ppc_carry_sub(ppc_result_a, ppc_result_b);
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (oe_flag)
|
|
|
|
ppc_setsoov(ppc_result_b, ppc_result_a, ppc_result_d);
|
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
|
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_subfic() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdasimm();
|
2020-02-03 22:32:56 -07:00
|
|
|
ppc_result_d = simm - ppc_result_a;
|
2024-01-10 05:38:13 -08:00
|
|
|
if (simm == -1)
|
|
|
|
ppc_state.spr[SPR::XER] |= XER::CA;
|
|
|
|
else
|
|
|
|
ppc_carry(~ppc_result_a, ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_subfe() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2023-12-17 16:59:07 -08:00
|
|
|
uint32_t grab_ca = !!(ppc_state.spr[SPR::XER] & XER::CA);
|
|
|
|
ppc_result_d = ~ppc_result_a + ppc_result_b + grab_ca;
|
|
|
|
if (grab_ca && ppc_result_b == 0xFFFFFFFFUL)
|
|
|
|
ppc_state.spr[SPR::XER] |= XER::CA;
|
|
|
|
else
|
|
|
|
ppc_carry(~ppc_result_a, ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (oe_flag)
|
|
|
|
ppc_setsoov(ppc_result_b, ppc_result_a, ppc_result_d);
|
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
2020-02-04 00:58:04 +01:00
|
|
|
|
2020-02-03 22:32:56 -07:00
|
|
|
ppc_store_result_regd();
|
2020-02-04 00:58:04 +01:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_subfme() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsda();
|
2023-12-24 02:35:38 +01:00
|
|
|
uint32_t grab_ca = !!(ppc_state.spr[SPR::XER] & XER::CA);
|
|
|
|
ppc_result_d = ~ppc_result_a + grab_ca - 1;
|
2020-11-29 23:53:03 +01:00
|
|
|
|
2023-12-24 02:35:38 +01:00
|
|
|
if (ppc_result_a == 0xFFFFFFFFUL && !grab_ca)
|
|
|
|
ppc_state.spr[SPR::XER] &= ~XER::CA;
|
|
|
|
else
|
|
|
|
ppc_state.spr[SPR::XER] |= XER::CA;
|
|
|
|
|
|
|
|
if (oe_flag) {
|
|
|
|
if (ppc_result_d == ppc_result_a && (int32_t)ppc_result_d > 0)
|
|
|
|
ppc_state.spr[SPR::XER] |= XER::SO | XER::OV;
|
|
|
|
else
|
|
|
|
ppc_state.spr[SPR::XER] &= ~XER::OV;
|
2023-12-16 19:09:09 -08:00
|
|
|
}
|
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-02-03 22:32:56 -07:00
|
|
|
ppc_store_result_regd();
|
2020-02-04 00:58:04 +01:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_subfze() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsda();
|
2023-12-30 15:58:21 +01:00
|
|
|
uint32_t grab_ca = !!(ppc_state.spr[SPR::XER] & XER::CA);
|
|
|
|
ppc_result_d = ~ppc_result_a + grab_ca;
|
|
|
|
|
|
|
|
if (!ppc_result_d && grab_ca) // special case: ppc_result_d = 0 and CA=1
|
|
|
|
ppc_state.spr[SPR::XER] |= XER::CA;
|
|
|
|
else
|
|
|
|
ppc_state.spr[SPR::XER] &= ~XER::CA;
|
|
|
|
|
|
|
|
if (oe_flag) {
|
|
|
|
if (ppc_result_d && ppc_result_d == ppc_result_a)
|
|
|
|
ppc_state.spr[SPR::XER] |= XER::SO | XER::OV;
|
|
|
|
else
|
|
|
|
ppc_state.spr[SPR::XER] &= ~XER::OV;
|
2023-12-16 23:31:54 -08:00
|
|
|
}
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
2020-02-04 00:58:04 +01:00
|
|
|
|
2020-02-03 22:32:56 -07:00
|
|
|
ppc_store_result_regd();
|
2020-02-04 00:58:04 +01:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_and() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
|
|
|
ppc_result_a = ppc_result_d & ppc_result_b;
|
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_andc() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
|
|
|
ppc_result_a = ppc_result_d & ~(ppc_result_b);
|
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_andidot() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssauimm();
|
|
|
|
ppc_result_a = ppc_result_d & uimm;
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_andisdot() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssauimm();
|
|
|
|
ppc_result_a = ppc_result_d & (uimm << 16);
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_nand() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
|
|
|
ppc_result_a = ~(ppc_result_d & ppc_result_b);
|
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_or() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
|
|
|
ppc_result_a = ppc_result_d | ppc_result_b;
|
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_orc() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
|
|
|
ppc_result_a = ppc_result_d | ~(ppc_result_b);
|
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_ori() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssauimm();
|
|
|
|
ppc_result_a = ppc_result_d | uimm;
|
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_oris() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssauimm();
|
2020-01-31 16:54:37 -07:00
|
|
|
ppc_result_a = (uimm << 16) | ppc_result_d;
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_eqv() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
|
|
|
ppc_result_a = ~(ppc_result_d ^ ppc_result_b);
|
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_nor() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
|
|
|
ppc_result_a = ~(ppc_result_d | ppc_result_b);
|
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_xor() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
|
|
|
ppc_result_a = ppc_result_d ^ ppc_result_b;
|
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_xori() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssauimm();
|
|
|
|
ppc_result_a = ppc_result_d ^ uimm;
|
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_xoris() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssauimm();
|
|
|
|
ppc_result_a = ppc_result_d ^ (uimm << 16);
|
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_neg() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsda();
|
2020-05-12 23:55:45 +05:00
|
|
|
ppc_result_d = ~(ppc_result_a) + 1;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (oe_flag) {
|
|
|
|
if (ppc_result_a == 0x80000000)
|
|
|
|
ppc_state.spr[SPR::XER] |= 0xC0000000;
|
|
|
|
else
|
|
|
|
ppc_state.spr[SPR::XER] &= 0xBFFFFFFF;
|
|
|
|
}
|
2020-11-29 23:53:03 +01:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
|
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_cntlzw() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssa();
|
2019-10-08 18:39:39 -07:00
|
|
|
|
|
|
|
uint32_t bit_check = ppc_result_d;
|
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
#ifdef USE_GCC_BUILTINS
|
2020-12-18 18:31:05 +01:00
|
|
|
uint32_t lead = !bit_check ? 32 : __builtin_clz(bit_check);
|
2020-01-31 18:03:27 +01:00
|
|
|
#elif defined USE_VS_BUILTINS
|
2020-12-18 18:31:05 +01:00
|
|
|
uint32_t lead = __lzcnt(bit_check);
|
2020-01-25 19:30:55 -07:00
|
|
|
#else
|
2020-12-18 18:31:05 +01:00
|
|
|
uint32_t lead, mask;
|
|
|
|
|
|
|
|
for (mask = 0x80000000UL, lead = 0; mask != 0; lead++, mask >>= 1) {
|
2020-01-31 18:03:27 +01:00
|
|
|
if (bit_check & mask)
|
|
|
|
break;
|
|
|
|
}
|
2020-01-25 19:30:55 -07:00
|
|
|
#endif
|
2019-10-08 18:39:39 -07:00
|
|
|
ppc_result_a = lead;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag) {
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
2020-01-31 18:03:27 +01:00
|
|
|
}
|
2020-10-17 14:30:37 -07:00
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_mulhwu() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2020-02-10 18:36:20 +01:00
|
|
|
uint64_t product = (uint64_t)ppc_result_a * (uint64_t)ppc_result_b;
|
2020-05-12 23:55:45 +05:00
|
|
|
ppc_result_d = (uint32_t)(product >> 32);
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_mulhw() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2020-02-10 18:36:20 +01:00
|
|
|
int64_t product = (int64_t)(int32_t)ppc_result_a * (int64_t)(int32_t)ppc_result_b;
|
2020-05-12 23:55:45 +05:00
|
|
|
ppc_result_d = product >> 32;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_mullw() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2020-02-10 18:36:20 +01:00
|
|
|
int64_t product = (int64_t)(int32_t)ppc_result_a * (int64_t)(int32_t)ppc_result_b;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (oe_flag) {
|
|
|
|
if (product != (int64_t)(int32_t)product) {
|
|
|
|
ppc_state.spr[SPR::XER] |= 0xC0000000;
|
|
|
|
} else {
|
|
|
|
ppc_state.spr[SPR::XER] &= 0xBFFFFFFFUL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-08 14:34:01 -07:00
|
|
|
ppc_result_d = (uint32_t)product;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
|
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_mulli() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdasimm();
|
2020-02-10 18:36:20 +01:00
|
|
|
int64_t product = (int64_t)(int32_t)ppc_result_a * (int64_t)(int32_t)simm;
|
2020-05-12 23:55:45 +05:00
|
|
|
ppc_result_d = (uint32_t)product;
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_divw() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2019-07-11 22:27:14 -07:00
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
if (!ppc_result_b) { /* handle the "anything / 0" case */
|
2023-12-17 05:32:54 -08:00
|
|
|
ppc_result_d = 0; // tested on G4 in Mac OS X 10.4 and Open Firmware.
|
|
|
|
// ppc_result_d = (ppc_result_a & 0x80000000) ? -1 : 0; /* UNDOCUMENTED! */
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (oe_flag)
|
|
|
|
ppc_state.spr[SPR::XER] |= 0xC0000000;
|
2019-07-11 22:27:14 -07:00
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
} else if (ppc_result_a == 0x80000000UL && ppc_result_b == 0xFFFFFFFFUL) {
|
2020-02-10 16:06:56 +01:00
|
|
|
ppc_result_d = 0xFFFFFFFF;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (oe_flag)
|
|
|
|
ppc_state.spr[SPR::XER] |= 0xC0000000;
|
2019-10-08 18:39:39 -07:00
|
|
|
|
2021-01-08 14:34:01 -07:00
|
|
|
} else { /* normal signed devision */
|
2020-02-10 16:06:56 +01:00
|
|
|
ppc_result_d = (int32_t)ppc_result_a / (int32_t)ppc_result_b;
|
2019-07-11 22:27:14 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (oe_flag)
|
|
|
|
ppc_state.spr[SPR::XER] &= 0xBFFFFFFFUL;
|
2019-07-11 22:27:14 -07:00
|
|
|
}
|
2020-02-10 16:06:56 +01:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
|
|
|
|
2020-02-10 16:06:56 +01:00
|
|
|
ppc_store_result_regd();
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_divwu() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2019-07-11 22:27:14 -07:00
|
|
|
|
2020-02-10 17:16:18 +01:00
|
|
|
if (!ppc_result_b) { /* division by zero */
|
2019-07-11 22:27:14 -07:00
|
|
|
ppc_result_d = 0;
|
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (oe_flag)
|
|
|
|
ppc_state.spr[SPR::XER] |= 0xC0000000;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_state.cr |= 0x20000000;
|
2019-07-11 22:27:14 -07:00
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-02-10 17:16:18 +01:00
|
|
|
ppc_result_d = ppc_result_a / ppc_result_b;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (oe_flag)
|
|
|
|
ppc_state.spr[SPR::XER] &= 0xBFFFFFFFUL;
|
2019-07-11 22:27:14 -07:00
|
|
|
}
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_d);
|
|
|
|
|
2020-02-10 17:16:18 +01:00
|
|
|
ppc_store_result_regd();
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
// Value shifting
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_slw() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
2020-02-03 20:36:22 +01:00
|
|
|
if (ppc_result_b & 0x20) {
|
|
|
|
ppc_result_a = 0;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-02-03 20:36:22 +01:00
|
|
|
ppc_result_a = ppc_result_d << (ppc_result_b & 0x1F);
|
|
|
|
}
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_srw() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
2020-02-03 20:36:22 +01:00
|
|
|
if (ppc_result_b & 0x20) {
|
|
|
|
ppc_result_a = 0;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-02-03 20:36:22 +01:00
|
|
|
ppc_result_a = ppc_result_d >> (ppc_result_b & 0x1F);
|
|
|
|
}
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_sraw() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
2023-12-16 22:01:02 -08:00
|
|
|
|
|
|
|
// clear XER[CA] by default
|
|
|
|
ppc_state.spr[SPR::XER] &= ~XER::CA;
|
|
|
|
|
2020-02-03 20:36:22 +01:00
|
|
|
if (ppc_result_b & 0x20) {
|
2023-12-16 22:01:02 -08:00
|
|
|
// fill rA with the sign bit of rS
|
2020-02-03 20:36:22 +01:00
|
|
|
ppc_result_a = (int32_t)ppc_result_d >> 31;
|
2023-12-16 22:01:02 -08:00
|
|
|
if (ppc_result_a) // if rA is negative
|
|
|
|
ppc_state.spr[SPR::XER] |= XER::CA;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-02-03 20:36:22 +01:00
|
|
|
uint32_t shift = ppc_result_b & 0x1F;
|
2023-12-16 22:01:02 -08:00
|
|
|
ppc_result_a = (int32_t)ppc_result_d >> shift;
|
|
|
|
if ((ppc_result_d & 0x80000000UL) && (ppc_result_d & ((1U << shift) - 1)))
|
|
|
|
ppc_state.spr[SPR::XER] |= XER::CA;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_srawi() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssa();
|
2020-02-03 20:36:22 +01:00
|
|
|
unsigned shift = (ppc_cur_instruction >> 11) & 0x1F;
|
2023-12-07 23:59:49 -08:00
|
|
|
uint32_t mask = (1U << shift) - 1;
|
2020-05-12 23:55:45 +05:00
|
|
|
ppc_result_a = (int32_t)ppc_result_d >> shift;
|
2020-02-03 20:36:22 +01:00
|
|
|
if ((ppc_result_d & 0x80000000UL) && (ppc_result_d & mask)) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.spr[SPR::XER] |= 0x20000000UL;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.spr[SPR::XER] &= 0xDFFFFFFFUL;
|
2020-02-03 20:36:22 +01:00
|
|
|
}
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2019-08-05 18:42:15 +02:00
|
|
|
/** mask generator for rotate and shift instructions (§ 4.2.1.4 PowerpC PEM) */
|
2020-05-12 23:55:45 +05:00
|
|
|
static inline uint32_t rot_mask(unsigned rot_mb, unsigned rot_me) {
|
2019-08-05 18:42:15 +02:00
|
|
|
uint32_t m1 = 0xFFFFFFFFUL >> rot_mb;
|
2022-12-21 03:20:39 -08:00
|
|
|
uint32_t m2 = (uint32_t)(0xFFFFFFFFUL << (31 - rot_me));
|
2019-08-05 18:42:15 +02:00
|
|
|
return ((rot_mb <= rot_me) ? m2 & m1 : m1 | m2);
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_rlwimi() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssa();
|
2019-08-05 18:42:15 +02:00
|
|
|
unsigned rot_sh = (ppc_cur_instruction >> 11) & 31;
|
|
|
|
unsigned rot_mb = (ppc_cur_instruction >> 6) & 31;
|
|
|
|
unsigned rot_me = (ppc_cur_instruction >> 1) & 31;
|
2020-05-12 23:55:45 +05:00
|
|
|
uint32_t mask = rot_mask(rot_mb, rot_me);
|
2023-12-07 23:59:49 -08:00
|
|
|
uint32_t r = rot_sh ? ((ppc_result_d << rot_sh) | (ppc_result_d >> (32 - rot_sh))) : ppc_result_d;
|
2020-05-12 23:55:45 +05:00
|
|
|
ppc_result_a = (ppc_result_a & ~mask) | (r & mask);
|
2020-01-25 19:30:55 -07:00
|
|
|
if ((ppc_cur_instruction & 0x01) == 1) {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
}
|
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_rlwinm() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssa();
|
2019-08-05 18:42:15 +02:00
|
|
|
unsigned rot_sh = (ppc_cur_instruction >> 11) & 31;
|
|
|
|
unsigned rot_mb = (ppc_cur_instruction >> 6) & 31;
|
|
|
|
unsigned rot_me = (ppc_cur_instruction >> 1) & 31;
|
2020-05-12 23:55:45 +05:00
|
|
|
uint32_t mask = rot_mask(rot_mb, rot_me);
|
2023-12-07 23:59:49 -08:00
|
|
|
uint32_t r = rot_sh ? ((ppc_result_d << rot_sh) | (ppc_result_d >> (32 - rot_sh))) : ppc_result_d;
|
2020-05-12 23:55:45 +05:00
|
|
|
ppc_result_a = r & mask;
|
2020-01-25 19:30:55 -07:00
|
|
|
if ((ppc_cur_instruction & 0x01) == 1) {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
}
|
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_rlwnm() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
2023-10-30 17:10:51 -07:00
|
|
|
ppc_result_b &= 31;
|
2019-08-05 18:42:15 +02:00
|
|
|
unsigned rot_mb = (ppc_cur_instruction >> 6) & 31;
|
|
|
|
unsigned rot_me = (ppc_cur_instruction >> 1) & 31;
|
2020-05-12 23:55:45 +05:00
|
|
|
uint32_t mask = rot_mask(rot_mb, rot_me);
|
2023-12-07 23:59:49 -08:00
|
|
|
uint32_t rot = ppc_result_b & 0x1F;
|
|
|
|
uint32_t r = rot ? ((ppc_result_d << rot) | (ppc_result_d >> (32 - rot))) : ppc_result_d;
|
2020-05-12 23:55:45 +05:00
|
|
|
ppc_result_a = r & mask;
|
2020-01-25 19:30:55 -07:00
|
|
|
if ((ppc_cur_instruction & 0x01) == 1) {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
}
|
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_mfcr() {
|
2020-05-12 23:55:45 +05:00
|
|
|
reg_d = (ppc_cur_instruction >> 21) & 31;
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.gpr[reg_d] = ppc_state.cr;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_mtsr() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_supervisor_instrs++;
|
2020-01-14 20:50:01 -07:00
|
|
|
#endif
|
2023-11-21 08:06:50 -07:00
|
|
|
if (ppc_state.msr & MSR::PR) {
|
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::NOT_ALLOWED);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2023-11-21 08:06:50 -07:00
|
|
|
reg_s = (ppc_cur_instruction >> 21) & 31;
|
2024-01-05 15:11:37 -07:00
|
|
|
uint32_t grab_sr = (ppc_cur_instruction >> 16) & 15;
|
2023-11-21 08:06:50 -07:00
|
|
|
ppc_state.sr[grab_sr] = ppc_state.gpr[reg_s];
|
|
|
|
mmu_pat_ctx_changed();
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_mtsrin() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_supervisor_instrs++;
|
2020-01-14 20:50:01 -07:00
|
|
|
#endif
|
2023-11-21 08:06:50 -07:00
|
|
|
if (ppc_state.msr & MSR::PR) {
|
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::NOT_ALLOWED);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2023-11-21 08:06:50 -07:00
|
|
|
ppc_grab_regssb();
|
2024-01-05 15:11:37 -07:00
|
|
|
uint32_t grab_sr = ppc_result_b >> 28;
|
2023-11-21 08:06:50 -07:00
|
|
|
ppc_state.sr[grab_sr] = ppc_result_d;
|
|
|
|
mmu_pat_ctx_changed();
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_mfsr() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_supervisor_instrs++;
|
2020-01-14 20:50:01 -07:00
|
|
|
#endif
|
2023-11-21 08:06:50 -07:00
|
|
|
if (ppc_state.msr & MSR::PR) {
|
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::NOT_ALLOWED);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2023-11-21 08:06:50 -07:00
|
|
|
reg_d = (ppc_cur_instruction >> 21) & 31;
|
2024-01-05 15:11:37 -07:00
|
|
|
uint32_t grab_sr = (ppc_cur_instruction >> 16) & 15;
|
2023-11-21 08:06:50 -07:00
|
|
|
ppc_state.gpr[reg_d] = ppc_state.sr[grab_sr];
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_mfsrin() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_supervisor_instrs++;
|
2020-01-14 20:50:01 -07:00
|
|
|
#endif
|
2023-11-21 08:06:50 -07:00
|
|
|
if (ppc_state.msr & MSR::PR) {
|
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::NOT_ALLOWED);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2023-11-21 08:06:50 -07:00
|
|
|
ppc_grab_regsdb();
|
2024-01-05 15:11:37 -07:00
|
|
|
uint32_t grab_sr = ppc_result_b >> 28;
|
2023-11-21 08:06:50 -07:00
|
|
|
ppc_state.gpr[reg_d] = ppc_state.sr[grab_sr];
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_mfmsr() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_supervisor_instrs++;
|
2020-01-14 20:50:01 -07:00
|
|
|
#endif
|
2023-11-21 08:06:50 -07:00
|
|
|
if (ppc_state.msr & MSR::PR) {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::NOT_ALLOWED);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2020-05-12 23:55:45 +05:00
|
|
|
reg_d = (ppc_cur_instruction >> 21) & 31;
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.gpr[reg_d] = ppc_state.msr;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_mtmsr() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_supervisor_instrs++;
|
2020-01-14 20:50:01 -07:00
|
|
|
#endif
|
2023-11-21 08:06:50 -07:00
|
|
|
if (ppc_state.msr & MSR::PR) {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::NOT_ALLOWED);
|
2020-01-11 21:48:56 +01:00
|
|
|
}
|
2020-05-12 23:55:45 +05:00
|
|
|
reg_s = (ppc_cur_instruction >> 21) & 31;
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.msr = ppc_state.gpr[reg_s];
|
2022-08-24 12:51:43 +02:00
|
|
|
|
|
|
|
// generate External Interrupt Exception
|
|
|
|
// if CPU interrupt line is asserted
|
2023-11-21 08:06:50 -07:00
|
|
|
if (ppc_state.msr & MSR::EE && int_pin) {
|
2023-09-25 03:25:50 +02:00
|
|
|
//LOG_F(WARNING, "MTMSR: CPU INT pending, generate CPU exception");
|
2022-08-24 12:51:43 +02:00
|
|
|
ppc_exception_handler(Except_Type::EXC_EXT_INT, 0);
|
2023-11-21 08:06:50 -07:00
|
|
|
} else if ((ppc_state.msr & MSR::EE) && dec_exception_pending) {
|
2023-08-07 11:11:02 -07:00
|
|
|
dec_exception_pending = false;
|
|
|
|
//LOG_F(WARNING, "MTMSR: decrementer exception triggered");
|
|
|
|
ppc_exception_handler(Except_Type::EXC_DECR, 0);
|
2022-08-24 12:51:43 +02:00
|
|
|
} else {
|
|
|
|
mmu_change_mode();
|
|
|
|
}
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2022-09-15 20:54:22 -07:00
|
|
|
static inline void calc_rtcl_value()
|
2022-03-22 12:23:54 +01:00
|
|
|
{
|
2022-09-15 20:54:22 -07:00
|
|
|
uint64_t new_ts = get_virt_time_ns();
|
2022-09-15 21:05:08 -07:00
|
|
|
uint64_t rtc_l = new_ts - rtc_timestamp + rtc_lo;
|
2022-03-22 12:23:54 +01:00
|
|
|
if (rtc_l >= ONE_BILLION_NS) { // check RTCL overflow
|
2022-12-21 03:20:39 -08:00
|
|
|
rtc_hi += (uint32_t)(rtc_l / ONE_BILLION_NS);
|
2022-03-22 12:23:54 +01:00
|
|
|
rtc_lo = rtc_l % ONE_BILLION_NS;
|
|
|
|
}
|
2022-09-15 20:54:22 -07:00
|
|
|
else {
|
2022-12-21 03:20:39 -08:00
|
|
|
rtc_lo = (uint32_t)rtc_l;
|
2022-09-15 20:54:22 -07:00
|
|
|
}
|
|
|
|
rtc_timestamp = new_ts;
|
2022-03-22 12:23:54 +01:00
|
|
|
}
|
|
|
|
|
2023-02-09 01:38:55 +01:00
|
|
|
static inline uint64_t calc_tbr_value()
|
|
|
|
{
|
|
|
|
uint64_t tbr_inc;
|
|
|
|
uint32_t tbr_inc_lo;
|
|
|
|
uint64_t diff = get_virt_time_ns() - tbr_wr_timestamp;
|
|
|
|
_u32xu64(tbr_freq_ghz, diff, tbr_inc, tbr_inc_lo);
|
|
|
|
return (tbr_wr_value + tbr_inc);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline uint32_t calc_dec_value() {
|
|
|
|
uint64_t dec_adj;
|
|
|
|
uint32_t dec_adj_lo;
|
|
|
|
uint64_t diff = get_virt_time_ns() - dec_wr_timestamp;
|
|
|
|
_u32xu64(tbr_freq_ghz, diff, dec_adj, dec_adj_lo);
|
|
|
|
return (dec_wr_value - dec_adj);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void update_timebase(uint64_t mask, uint64_t new_val)
|
|
|
|
{
|
|
|
|
uint64_t tbr_value = calc_tbr_value();
|
|
|
|
tbr_wr_value = (tbr_value & mask) | new_val;
|
|
|
|
tbr_wr_timestamp = get_virt_time_ns();
|
|
|
|
}
|
|
|
|
|
2023-08-07 11:11:02 -07:00
|
|
|
|
|
|
|
static uint32_t decrementer_timer_id = 0;
|
|
|
|
|
|
|
|
static void trigger_decrementer_exception() {
|
|
|
|
decrementer_timer_id = 0;
|
|
|
|
dec_wr_value = -1;
|
|
|
|
dec_wr_timestamp = get_virt_time_ns();
|
2023-11-21 08:06:50 -07:00
|
|
|
if (ppc_state.msr & MSR::EE) {
|
2023-08-07 11:11:02 -07:00
|
|
|
dec_exception_pending = false;
|
|
|
|
//LOG_F(WARNING, "decrementer exception triggered");
|
|
|
|
ppc_exception_handler(Except_Type::EXC_DECR, 0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
//LOG_F(WARNING, "decrementer exception pending");
|
|
|
|
dec_exception_pending = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-09 01:38:55 +01:00
|
|
|
static void update_decrementer(uint32_t val) {
|
|
|
|
dec_wr_value = val;
|
|
|
|
dec_wr_timestamp = get_virt_time_ns();
|
2023-08-07 11:11:02 -07:00
|
|
|
|
|
|
|
dec_exception_pending = false;
|
|
|
|
|
2023-09-18 21:20:02 +02:00
|
|
|
if (is_601)
|
|
|
|
return;
|
|
|
|
|
2023-08-07 11:11:02 -07:00
|
|
|
if (decrementer_timer_id) {
|
|
|
|
//LOG_F(WARNING, "decrementer cancel timer");
|
|
|
|
TimerManager::get_instance()->cancel_timer(decrementer_timer_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t time_out;
|
|
|
|
uint32_t time_out_lo;
|
|
|
|
_u32xu64(val, tbr_period_ns, time_out, time_out_lo);
|
|
|
|
//LOG_F(WARNING, "decrementer:0x%08X ns:%llu", val, time_out);
|
|
|
|
decrementer_timer_id = TimerManager::get_instance()->add_oneshot_timer(
|
|
|
|
time_out,
|
|
|
|
trigger_decrementer_exception
|
|
|
|
);
|
2023-02-09 01:38:55 +01:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_mfspr() {
|
2019-07-01 19:15:33 -07:00
|
|
|
uint32_t ref_spr = (((ppc_cur_instruction >> 11) & 31) << 5) | ((ppc_cur_instruction >> 16) & 31);
|
2020-01-14 20:50:01 -07:00
|
|
|
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
2020-01-14 20:50:01 -07:00
|
|
|
if (ref_spr > 31) {
|
2021-04-29 02:26:17 +02:00
|
|
|
num_supervisor_instrs++;
|
2020-01-14 20:50:01 -07:00
|
|
|
}
|
|
|
|
#endif
|
2022-03-22 12:23:54 +01:00
|
|
|
|
|
|
|
switch (ref_spr) {
|
|
|
|
case SPR::RTCL_U:
|
2022-09-15 20:54:22 -07:00
|
|
|
calc_rtcl_value();
|
|
|
|
ppc_state.spr[SPR::RTCL_U] = rtc_lo & 0x3FFFFF80UL;
|
2022-03-22 12:23:54 +01:00
|
|
|
break;
|
|
|
|
case SPR::RTCU_U:
|
2022-09-15 20:54:22 -07:00
|
|
|
calc_rtcl_value();
|
2022-09-14 00:33:44 -07:00
|
|
|
ppc_state.spr[SPR::RTCU_U] = rtc_hi;
|
2022-03-22 12:23:54 +01:00
|
|
|
break;
|
2023-02-09 01:38:55 +01:00
|
|
|
case SPR::DEC:
|
|
|
|
ppc_state.spr[SPR::DEC] = calc_dec_value();
|
|
|
|
break;
|
2022-03-22 12:23:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ppc_state.gpr[(ppc_cur_instruction >> 21) & 31] = ppc_state.spr[ref_spr];
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_mtspr() {
|
2019-07-01 19:15:33 -07:00
|
|
|
uint32_t ref_spr = (((ppc_cur_instruction >> 11) & 31) << 5) | ((ppc_cur_instruction >> 16) & 31);
|
2020-05-12 23:55:45 +05:00
|
|
|
reg_s = (ppc_cur_instruction >> 21) & 31;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
2020-01-14 20:50:01 -07:00
|
|
|
if (ref_spr > 31) {
|
2021-04-29 02:26:17 +02:00
|
|
|
num_supervisor_instrs++;
|
2020-01-14 20:50:01 -07:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2022-03-22 12:23:54 +01:00
|
|
|
uint32_t val = ppc_state.gpr[reg_s];
|
|
|
|
|
|
|
|
if (ref_spr != SPR::PVR) { // prevent writes to the read-only PVR
|
|
|
|
ppc_state.spr[ref_spr] = val;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2022-03-22 12:23:54 +01:00
|
|
|
if (ref_spr == SPR::SDR1) { // adapt to SDR1 changes
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_pat_ctx_changed();
|
|
|
|
}
|
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
switch (ref_spr) {
|
2022-03-22 12:23:54 +01:00
|
|
|
case SPR::RTCL_S:
|
2022-09-15 20:59:56 -07:00
|
|
|
calc_rtcl_value();
|
2022-03-22 12:23:54 +01:00
|
|
|
rtc_lo = val & 0x3FFFFF80UL;
|
|
|
|
break;
|
|
|
|
case SPR::RTCU_S:
|
2022-09-15 20:59:56 -07:00
|
|
|
calc_rtcl_value();
|
2022-03-22 12:23:54 +01:00
|
|
|
rtc_hi = val;
|
|
|
|
break;
|
2023-02-09 01:38:55 +01:00
|
|
|
case SPR::DEC:
|
|
|
|
update_decrementer(val);
|
|
|
|
break;
|
2022-03-22 12:23:54 +01:00
|
|
|
case SPR::TBL_S:
|
|
|
|
update_timebase(0xFFFFFFFF00000000ULL, val);
|
2020-01-25 19:30:55 -07:00
|
|
|
break;
|
2022-03-22 12:23:54 +01:00
|
|
|
case SPR::TBU_S:
|
|
|
|
update_timebase(0x00000000FFFFFFFFULL, (uint64_t)(val) << 32);
|
2020-01-25 19:30:55 -07:00
|
|
|
break;
|
|
|
|
case 528:
|
|
|
|
case 529:
|
|
|
|
case 530:
|
|
|
|
case 531:
|
|
|
|
case 532:
|
|
|
|
case 533:
|
|
|
|
case 534:
|
|
|
|
case 535:
|
|
|
|
ibat_update(ref_spr);
|
|
|
|
break;
|
|
|
|
case 536:
|
|
|
|
case 537:
|
|
|
|
case 538:
|
|
|
|
case 539:
|
|
|
|
case 540:
|
|
|
|
case 541:
|
|
|
|
case 542:
|
|
|
|
case 543:
|
|
|
|
dbat_update(ref_spr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_mftb() {
|
2019-07-01 19:15:33 -07:00
|
|
|
uint32_t ref_spr = (((ppc_cur_instruction >> 11) & 31) << 5) | ((ppc_cur_instruction >> 16) & 31);
|
2021-10-05 17:40:09 -07:00
|
|
|
reg_d = (ppc_cur_instruction >> 21) & 31;
|
2021-12-20 00:12:44 +01:00
|
|
|
|
|
|
|
uint64_t tbr_value = calc_tbr_value();
|
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
switch (ref_spr) {
|
2022-03-22 12:23:54 +01:00
|
|
|
case SPR::TBL_U:
|
2021-12-20 00:12:44 +01:00
|
|
|
ppc_state.gpr[reg_d] = tbr_value & 0xFFFFFFFFUL;
|
|
|
|
break;
|
2022-03-22 12:23:54 +01:00
|
|
|
case SPR::TBU_U:
|
2021-12-20 00:12:44 +01:00
|
|
|
ppc_state.gpr[reg_d] = (tbr_value >> 32) & 0xFFFFFFFFUL;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::ILLEGAL_OP);
|
2021-10-05 17:40:09 -07:00
|
|
|
}
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_mtcrf() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssa();
|
2024-01-02 17:43:17 +01:00
|
|
|
uint8_t crm = (ppc_cur_instruction >> 12) & 0xFFU;
|
|
|
|
|
|
|
|
uint32_t cr_mask = 0;
|
|
|
|
|
|
|
|
if (crm == 0xFFU) // the fast case
|
|
|
|
cr_mask = 0xFFFFFFFFUL;
|
|
|
|
else { // the slow case
|
|
|
|
if (crm & 0x80) cr_mask |= 0xF0000000UL;
|
|
|
|
if (crm & 0x40) cr_mask |= 0x0F000000UL;
|
|
|
|
if (crm & 0x20) cr_mask |= 0x00F00000UL;
|
|
|
|
if (crm & 0x10) cr_mask |= 0x000F0000UL;
|
|
|
|
if (crm & 0x08) cr_mask |= 0x0000F000UL;
|
|
|
|
if (crm & 0x04) cr_mask |= 0x00000F00UL;
|
|
|
|
if (crm & 0x02) cr_mask |= 0x000000F0UL;
|
|
|
|
if (crm & 0x01) cr_mask |= 0x0000000FUL;
|
|
|
|
}
|
|
|
|
ppc_state.cr = (ppc_state.cr & ~cr_mask) | (ppc_result_d & cr_mask);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_mcrxr() {
|
2021-01-08 14:34:01 -07:00
|
|
|
int crf_d = (ppc_cur_instruction >> 21) & 0x1C;
|
2020-05-12 23:55:45 +05:00
|
|
|
ppc_state.cr = (ppc_state.cr & ~(0xF0000000UL >> crf_d)) |
|
|
|
|
((ppc_state.spr[SPR::XER] & 0xF0000000UL) >> crf_d);
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.spr[SPR::XER] &= 0x0FFFFFFF;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_extsb() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssa();
|
2020-01-31 16:54:37 -07:00
|
|
|
ppc_result_d = ppc_result_d & 0xFF;
|
2024-01-07 17:04:51 -07:00
|
|
|
ppc_result_a = (int32_t(int8_t(ppc_result_d)));
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_extsh() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssa();
|
2020-01-31 16:54:37 -07:00
|
|
|
ppc_result_d = ppc_result_d & 0xFFFF;
|
2024-01-07 17:04:51 -07:00
|
|
|
ppc_result_a = (int32_t(int16_t(ppc_result_d)));
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag)
|
|
|
|
ppc_changecrf0(ppc_result_a);
|
2019-07-01 19:15:33 -07:00
|
|
|
|
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
// Branching Instructions
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
// The last two bytes of the instruction are used for determining how the branch happens.
|
|
|
|
// The middle 24 bytes are the 24-bit address to use for branching to.
|
2019-07-01 19:15:33 -07:00
|
|
|
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_b() {
|
2024-01-07 17:04:51 -07:00
|
|
|
uint32_t quick_test = (ppc_cur_instruction & 0x03FFFFFCUL);
|
|
|
|
int32_t adr_li = (quick_test < 0x2000000UL) ? quick_test : (0xFC000000UL + quick_test);
|
2020-03-05 07:48:10 -07:00
|
|
|
ppc_next_instruction_address = (uint32_t)(ppc_state.pc + adr_li);
|
2022-03-02 16:55:20 +01:00
|
|
|
exec_flags = EXEF_BRANCH;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_bl() {
|
2024-01-07 17:04:51 -07:00
|
|
|
uint32_t quick_test = (ppc_cur_instruction & 0x03FFFFFCUL);
|
|
|
|
int32_t adr_li = (quick_test < 0x2000000UL) ? quick_test : (0xFC000000UL + quick_test);
|
2020-03-05 07:48:10 -07:00
|
|
|
ppc_next_instruction_address = (uint32_t)(ppc_state.pc + adr_li);
|
2021-01-08 14:34:01 -07:00
|
|
|
ppc_state.spr[SPR::LR] = (uint32_t)(ppc_state.pc + 4);
|
2022-03-02 16:55:20 +01:00
|
|
|
exec_flags = EXEF_BRANCH;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_ba() {
|
2024-01-07 17:04:51 -07:00
|
|
|
uint32_t quick_test = (ppc_cur_instruction & 0x03FFFFFCUL);
|
|
|
|
int32_t adr_li = (quick_test < 0x2000000UL) ? quick_test : (0xFC000000UL + quick_test);
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_next_instruction_address = adr_li;
|
2022-03-02 16:55:20 +01:00
|
|
|
exec_flags = EXEF_BRANCH;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_bla() {
|
2024-01-07 17:04:51 -07:00
|
|
|
uint32_t quick_test = (ppc_cur_instruction & 0x03FFFFFCUL);
|
|
|
|
int32_t adr_li = (quick_test < 0x2000000UL) ? quick_test : (0xFC000000UL + quick_test);
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_next_instruction_address = adr_li;
|
2021-01-08 14:34:01 -07:00
|
|
|
ppc_state.spr[SPR::LR] = ppc_state.pc + 4;
|
2022-03-02 16:55:20 +01:00
|
|
|
exec_flags = EXEF_BRANCH;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_bc() {
|
2019-08-03 07:00:04 +02:00
|
|
|
uint32_t ctr_ok;
|
|
|
|
uint32_t cnd_ok;
|
|
|
|
uint32_t br_bo = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
uint32_t br_bi = (ppc_cur_instruction >> 16) & 31;
|
2020-05-12 23:55:45 +05:00
|
|
|
int32_t br_bd = (int32_t)((int16_t)(ppc_cur_instruction & 0xFFFC));
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if (!(br_bo & 0x04)) {
|
2020-03-04 21:29:04 -07:00
|
|
|
(ppc_state.spr[SPR::CTR])--; /* decrement CTR */
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2024-01-05 15:11:37 -07:00
|
|
|
ctr_ok = (br_bo & 0x04) | ((ppc_state.spr[SPR::CTR] != 0) == !(br_bo & 0x02));
|
|
|
|
cnd_ok = (br_bo & 0x10) | (!(ppc_state.cr & (0x80000000UL >> br_bi)) == !(br_bo & 0x08));
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if (ctr_ok && cnd_ok) {
|
2020-03-05 07:48:10 -07:00
|
|
|
ppc_next_instruction_address = (ppc_state.pc + br_bd);
|
2022-03-02 16:55:20 +01:00
|
|
|
exec_flags = EXEF_BRANCH;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_bca() {
|
2019-08-03 07:00:04 +02:00
|
|
|
uint32_t ctr_ok;
|
|
|
|
uint32_t cnd_ok;
|
|
|
|
uint32_t br_bo = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
uint32_t br_bi = (ppc_cur_instruction >> 16) & 31;
|
2020-05-12 23:55:45 +05:00
|
|
|
int32_t br_bd = (int32_t)((int16_t)(ppc_cur_instruction & 0xFFFC));
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if (!(br_bo & 0x04)) {
|
2020-03-04 21:29:04 -07:00
|
|
|
(ppc_state.spr[SPR::CTR])--; /* decrement CTR */
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2024-01-05 15:11:37 -07:00
|
|
|
ctr_ok = (br_bo & 0x04) | ((ppc_state.spr[SPR::CTR] != 0) == !(br_bo & 0x02));
|
|
|
|
cnd_ok = (br_bo & 0x10) | (!(ppc_state.cr & (0x80000000UL >> br_bi)) == !(br_bo & 0x08));
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if (ctr_ok && cnd_ok) {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_next_instruction_address = br_bd;
|
2022-03-02 16:55:20 +01:00
|
|
|
exec_flags = EXEF_BRANCH;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_bcl() {
|
2019-08-03 07:00:04 +02:00
|
|
|
uint32_t ctr_ok;
|
|
|
|
uint32_t cnd_ok;
|
|
|
|
uint32_t br_bo = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
uint32_t br_bi = (ppc_cur_instruction >> 16) & 31;
|
2020-05-12 23:55:45 +05:00
|
|
|
int32_t br_bd = (int32_t)((int16_t)(ppc_cur_instruction & 0xFFFC));
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if (!(br_bo & 0x04)) {
|
2020-03-04 21:29:04 -07:00
|
|
|
(ppc_state.spr[SPR::CTR])--; /* decrement CTR */
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2024-01-05 15:11:37 -07:00
|
|
|
ctr_ok = (br_bo & 0x04) | ((ppc_state.spr[SPR::CTR] != 0) == !(br_bo & 0x02));
|
|
|
|
cnd_ok = (br_bo & 0x10) | (!(ppc_state.cr & (0x80000000UL >> br_bi)) == !(br_bo & 0x08));
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if (ctr_ok && cnd_ok) {
|
2020-03-05 07:48:10 -07:00
|
|
|
ppc_next_instruction_address = (ppc_state.pc + br_bd);
|
2022-03-02 16:55:20 +01:00
|
|
|
exec_flags = EXEF_BRANCH;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2020-03-05 07:48:10 -07:00
|
|
|
ppc_state.spr[SPR::LR] = ppc_state.pc + 4;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_bcla() {
|
2019-08-03 07:00:04 +02:00
|
|
|
uint32_t ctr_ok;
|
|
|
|
uint32_t cnd_ok;
|
|
|
|
uint32_t br_bo = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
uint32_t br_bi = (ppc_cur_instruction >> 16) & 31;
|
2020-05-12 23:55:45 +05:00
|
|
|
int32_t br_bd = (int32_t)((int16_t)(ppc_cur_instruction & 0xFFFC));
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if (!(br_bo & 0x04)) {
|
2020-03-04 21:29:04 -07:00
|
|
|
(ppc_state.spr[SPR::CTR])--; /* decrement CTR */
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2024-01-05 15:11:37 -07:00
|
|
|
ctr_ok = (br_bo & 0x04) | ((ppc_state.spr[SPR::CTR] != 0) == !(br_bo & 0x02));
|
|
|
|
cnd_ok = (br_bo & 0x10) | (!(ppc_state.cr & (0x80000000UL >> br_bi)) == !(br_bo & 0x08));
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if (ctr_ok && cnd_ok) {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_next_instruction_address = br_bd;
|
2022-03-02 16:55:20 +01:00
|
|
|
exec_flags = EXEF_BRANCH;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2020-03-05 07:48:10 -07:00
|
|
|
ppc_state.spr[SPR::LR] = ppc_state.pc + 4;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_bcctr() {
|
2019-08-03 07:00:04 +02:00
|
|
|
uint32_t br_bo = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
uint32_t br_bi = (ppc_cur_instruction >> 16) & 31;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2024-01-05 15:11:37 -07:00
|
|
|
uint32_t cnd_ok = (br_bo & 0x10) |
|
2020-05-12 23:55:45 +05:00
|
|
|
(!(ppc_state.cr & (0x80000000UL >> br_bi)) == !(br_bo & 0x08));
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if (cnd_ok) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_next_instruction_address = (ppc_state.spr[SPR::CTR] & 0xFFFFFFFCUL);
|
2022-03-02 16:55:20 +01:00
|
|
|
exec_flags = EXEF_BRANCH;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_bcctrl() {
|
2019-08-03 07:00:04 +02:00
|
|
|
uint32_t br_bo = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
uint32_t br_bi = (ppc_cur_instruction >> 16) & 31;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2024-01-05 15:11:37 -07:00
|
|
|
uint32_t cnd_ok = (br_bo & 0x10) |
|
2020-05-12 23:55:45 +05:00
|
|
|
(!(ppc_state.cr & (0x80000000UL >> br_bi)) == !(br_bo & 0x08));
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if (cnd_ok) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_next_instruction_address = (ppc_state.spr[SPR::CTR] & 0xFFFFFFFCUL);
|
2022-03-02 16:55:20 +01:00
|
|
|
exec_flags = EXEF_BRANCH;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2020-03-05 07:48:10 -07:00
|
|
|
ppc_state.spr[SPR::LR] = ppc_state.pc + 4;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_bclr() {
|
2019-08-03 07:00:04 +02:00
|
|
|
uint32_t br_bo = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
uint32_t br_bi = (ppc_cur_instruction >> 16) & 31;
|
|
|
|
uint32_t ctr_ok;
|
|
|
|
uint32_t cnd_ok;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if (!(br_bo & 0x04)) {
|
2020-03-04 21:29:04 -07:00
|
|
|
(ppc_state.spr[SPR::CTR])--; /* decrement CTR */
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2024-01-05 15:11:37 -07:00
|
|
|
ctr_ok = (br_bo & 0x04) | ((ppc_state.spr[SPR::CTR] != 0) == !(br_bo & 0x02));
|
|
|
|
cnd_ok = (br_bo & 0x10) | (!(ppc_state.cr & (0x80000000UL >> br_bi)) == !(br_bo & 0x08));
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if (ctr_ok && cnd_ok) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_next_instruction_address = (ppc_state.spr[SPR::LR] & 0xFFFFFFFCUL);
|
2022-03-02 16:55:20 +01:00
|
|
|
exec_flags = EXEF_BRANCH;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_bclrl() {
|
2019-08-03 07:00:04 +02:00
|
|
|
uint32_t br_bo = (ppc_cur_instruction >> 21) & 31;
|
|
|
|
uint32_t br_bi = (ppc_cur_instruction >> 16) & 31;
|
|
|
|
uint32_t ctr_ok;
|
|
|
|
uint32_t cnd_ok;
|
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if (!(br_bo & 0x04)) {
|
2020-03-04 21:29:04 -07:00
|
|
|
(ppc_state.spr[SPR::CTR])--; /* decrement CTR */
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2024-01-05 15:11:37 -07:00
|
|
|
ctr_ok = (br_bo & 0x04) | ((ppc_state.spr[SPR::CTR] != 0) == !(br_bo & 0x02));
|
|
|
|
cnd_ok = (br_bo & 0x10) | (!(ppc_state.cr & (0x80000000UL >> br_bi)) == !(br_bo & 0x08));
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-01-25 19:30:55 -07:00
|
|
|
if (ctr_ok && cnd_ok) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_next_instruction_address = (ppc_state.spr[SPR::LR] & 0xFFFFFFFCUL);
|
2022-03-02 16:55:20 +01:00
|
|
|
exec_flags = EXEF_BRANCH;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2020-03-05 07:48:10 -07:00
|
|
|
ppc_state.spr[SPR::LR] = ppc_state.pc + 4;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2020-05-12 23:55:45 +05:00
|
|
|
// Compare Instructions
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_cmp() {
|
2020-02-24 19:50:52 -07:00
|
|
|
#ifdef CHECK_INVALID
|
|
|
|
if (ppc_cur_instruction & 0x200000) {
|
2022-08-14 05:26:56 -07:00
|
|
|
LOG_F(WARNING, "Invalid CMP instruction form (L=1)!");
|
2020-02-24 19:50:52 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2021-01-08 14:34:01 -07:00
|
|
|
int crf_d = (ppc_cur_instruction >> 21) & 0x1C;
|
2020-01-25 19:30:55 -07:00
|
|
|
ppc_grab_regssab();
|
2024-01-05 15:11:37 -07:00
|
|
|
uint32_t xercon = (ppc_state.spr[SPR::XER] & 0x80000000UL) >> 3;
|
2023-12-16 05:30:02 -08:00
|
|
|
uint32_t cmp_c = (((int32_t)ppc_result_a) == ((int32_t)ppc_result_b))
|
2020-05-12 23:55:45 +05:00
|
|
|
? 0x20000000UL
|
|
|
|
: (((int32_t)ppc_result_a) > ((int32_t)ppc_result_b)) ? 0x40000000UL : 0x80000000UL;
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr = ((ppc_state.cr & ~(0xf0000000UL >> crf_d)) | ((cmp_c + xercon) >> crf_d));
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_cmpi() {
|
2020-01-25 19:30:55 -07:00
|
|
|
#ifdef CHECK_INVALID
|
2019-08-03 06:38:23 +02:00
|
|
|
if (ppc_cur_instruction & 0x200000) {
|
2022-08-14 05:26:56 -07:00
|
|
|
LOG_F(WARNING, "Invalid CMPI instruction form (L=1)!");
|
2019-08-03 06:38:23 +02:00
|
|
|
return;
|
|
|
|
}
|
2020-01-25 19:30:55 -07:00
|
|
|
#endif
|
2019-08-03 06:38:23 +02:00
|
|
|
|
2021-01-08 14:34:01 -07:00
|
|
|
int crf_d = (ppc_cur_instruction >> 21) & 0x1C;
|
2019-08-03 06:38:23 +02:00
|
|
|
ppc_grab_regsasimm();
|
2024-01-05 15:11:37 -07:00
|
|
|
uint32_t xercon = (ppc_state.spr[SPR::XER] & 0x80000000UL) >> 3;
|
2023-12-16 05:30:02 -08:00
|
|
|
uint32_t cmp_c = (((int32_t)ppc_result_a) == simm)
|
2020-05-12 23:55:45 +05:00
|
|
|
? 0x20000000UL
|
|
|
|
: (((int32_t)ppc_result_a) > simm) ? 0x40000000UL : 0x80000000UL;
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr = ((ppc_state.cr & ~(0xf0000000UL >> crf_d)) | ((cmp_c + xercon) >> crf_d));
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_cmpl() {
|
2020-01-25 19:30:55 -07:00
|
|
|
#ifdef CHECK_INVALID
|
2019-08-03 06:38:23 +02:00
|
|
|
if (ppc_cur_instruction & 0x200000) {
|
2022-08-14 05:26:56 -07:00
|
|
|
LOG_F(WARNING, "Invalid CMPL instruction form (L=1)!");
|
2019-08-03 06:38:23 +02:00
|
|
|
return;
|
|
|
|
}
|
2020-01-25 19:30:55 -07:00
|
|
|
#endif
|
2019-08-03 06:38:23 +02:00
|
|
|
|
2021-01-08 14:34:01 -07:00
|
|
|
int crf_d = (ppc_cur_instruction >> 21) & 0x1C;
|
2019-08-03 06:38:23 +02:00
|
|
|
ppc_grab_regssab();
|
2024-01-05 15:11:37 -07:00
|
|
|
uint32_t xercon = (ppc_state.spr[SPR::XER] & 0x80000000UL) >> 3;
|
2023-12-16 05:30:02 -08:00
|
|
|
uint32_t cmp_c = (ppc_result_a == ppc_result_b)
|
2020-05-12 23:55:45 +05:00
|
|
|
? 0x20000000UL
|
|
|
|
: (ppc_result_a > ppc_result_b) ? 0x40000000UL : 0x80000000UL;
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr = ((ppc_state.cr & ~(0xf0000000UL >> crf_d)) | ((cmp_c + xercon) >> crf_d));
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_cmpli() {
|
2020-01-25 19:30:55 -07:00
|
|
|
#ifdef CHECK_INVALID
|
2019-08-03 06:38:23 +02:00
|
|
|
if (ppc_cur_instruction & 0x200000) {
|
2022-08-14 05:26:56 -07:00
|
|
|
LOG_F(WARNING, "Invalid CMPLI instruction form (L=1)!");
|
2019-08-03 06:38:23 +02:00
|
|
|
return;
|
|
|
|
}
|
2020-01-25 19:30:55 -07:00
|
|
|
#endif
|
2019-08-03 06:38:23 +02:00
|
|
|
|
2021-01-08 14:34:01 -07:00
|
|
|
int crf_d = (ppc_cur_instruction >> 21) & 0x1C;
|
2019-08-03 06:38:23 +02:00
|
|
|
ppc_grab_regssauimm();
|
2024-01-05 15:11:37 -07:00
|
|
|
uint32_t xercon = (ppc_state.spr[SPR::XER] & 0x80000000UL) >> 3;
|
2023-12-16 05:30:02 -08:00
|
|
|
uint32_t cmp_c = (ppc_result_a == uimm) ? 0x20000000UL
|
2020-05-12 23:55:45 +05:00
|
|
|
: (ppc_result_a > uimm) ? 0x40000000UL : 0x80000000UL;
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr = ((ppc_state.cr & ~(0xf0000000UL >> crf_d)) | ((cmp_c + xercon) >> crf_d));
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
// Condition Register Changes
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2021-01-07 20:38:32 -07:00
|
|
|
void dppc_interpreter::ppc_mcrf() {
|
2021-01-08 14:34:01 -07:00
|
|
|
int crf_d = (ppc_cur_instruction >> 21) & 0x1C;
|
|
|
|
int crf_s = (ppc_cur_instruction >> 16) & 0x1C;
|
2021-01-08 21:58:57 +01:00
|
|
|
|
2022-05-17 08:44:50 +02:00
|
|
|
// extract and right justify source flags field
|
|
|
|
uint32_t grab_s = (ppc_state.cr >> (28 - crf_s)) & 0xF;
|
|
|
|
|
|
|
|
ppc_state.cr = (ppc_state.cr & ~(0xf0000000UL >> crf_d)) | (grab_s << (28 - crf_d));
|
2021-01-07 20:38:32 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_crand() {
|
2023-12-17 05:31:16 -08:00
|
|
|
ppc_grab_dab();
|
2022-01-21 14:56:10 +01:00
|
|
|
uint8_t ir = (ppc_state.cr >> (31 - reg_a)) & (ppc_state.cr >> (31 - reg_b));
|
|
|
|
if (ir & 1) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr |= (0x80000000UL >> reg_d);
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr &= ~(0x80000000UL >> reg_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
2021-01-07 20:38:32 -07:00
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_crandc() {
|
2023-12-17 05:31:16 -08:00
|
|
|
ppc_grab_dab();
|
2022-01-10 18:10:39 +01:00
|
|
|
if ((ppc_state.cr & (0x80000000UL >> reg_a)) && !(ppc_state.cr & (0x80000000UL >> reg_b))) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr |= (0x80000000UL >> reg_d);
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr &= ~(0x80000000UL >> reg_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_creqv() {
|
2023-12-17 05:31:16 -08:00
|
|
|
ppc_grab_dab();
|
2022-01-21 16:32:07 +01:00
|
|
|
uint8_t ir = (ppc_state.cr >> (31 - reg_a)) ^ (ppc_state.cr >> (31 - reg_b));
|
|
|
|
if (ir & 1) { // compliment is implemented by swapping the following if/else bodies
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr &= ~(0x80000000UL >> reg_d);
|
2022-01-21 16:32:07 +01:00
|
|
|
} else {
|
|
|
|
ppc_state.cr |= (0x80000000UL >> reg_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_crnand() {
|
2023-12-17 05:31:16 -08:00
|
|
|
ppc_grab_dab();
|
2022-01-22 22:33:13 -07:00
|
|
|
uint8_t ir = (ppc_state.cr >> (31 - reg_a)) & (ppc_state.cr >> (31 - reg_b));
|
|
|
|
if (ir & 1) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr &= ~(0x80000000UL >> reg_d);
|
2022-01-22 22:33:13 -07:00
|
|
|
} else {
|
|
|
|
ppc_state.cr |= (0x80000000UL >> reg_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
2022-01-22 22:33:13 -07:00
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_crnor() {
|
2023-12-17 05:31:16 -08:00
|
|
|
ppc_grab_dab();
|
2022-01-22 22:33:13 -07:00
|
|
|
uint8_t ir = (ppc_state.cr >> (31 - reg_a)) | (ppc_state.cr >> (31 - reg_b));
|
|
|
|
if (ir & 1) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr &= ~(0x80000000UL >> reg_d);
|
2022-01-22 22:33:13 -07:00
|
|
|
} else {
|
|
|
|
ppc_state.cr |= (0x80000000UL >> reg_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_cror() {
|
2023-12-17 05:31:16 -08:00
|
|
|
ppc_grab_dab();
|
2022-01-21 14:37:51 +01:00
|
|
|
uint8_t ir = (ppc_state.cr >> (31 - reg_a)) | (ppc_state.cr >> (31 - reg_b));
|
|
|
|
if (ir & 1) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr |= (0x80000000UL >> reg_d);
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr &= ~(0x80000000UL >> reg_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
2022-01-21 14:37:51 +01:00
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_crorc() {
|
2023-12-17 05:31:16 -08:00
|
|
|
ppc_grab_dab();
|
2022-01-10 18:10:39 +01:00
|
|
|
if ((ppc_state.cr & (0x80000000UL >> reg_a)) || !(ppc_state.cr & (0x80000000UL >> reg_b))) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr |= (0x80000000UL >> reg_d);
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr &= ~(0x80000000UL >> reg_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_crxor() {
|
2023-12-17 05:31:16 -08:00
|
|
|
ppc_grab_dab();
|
2022-01-08 04:21:09 +01:00
|
|
|
uint8_t ir = (ppc_state.cr >> (31 - reg_a)) ^ (ppc_state.cr >> (31 - reg_b));
|
|
|
|
if (ir & 1) {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr |= (0x80000000UL >> reg_d);
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.cr &= ~(0x80000000UL >> reg_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
// Processor MGMT Fns.
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_rfi() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_supervisor_instrs++;
|
2020-01-14 20:50:01 -07:00
|
|
|
#endif
|
2020-05-12 23:55:45 +05:00
|
|
|
uint32_t new_srr1_val = (ppc_state.spr[SPR::SRR1] & 0x87C0FF73UL);
|
|
|
|
uint32_t new_msr_val = (ppc_state.msr & ~(0x87C0FF73UL));
|
|
|
|
ppc_state.msr = (new_msr_val | new_srr1_val) & 0xFFFBFFFFUL;
|
2022-08-24 12:51:43 +02:00
|
|
|
|
|
|
|
// generate External Interrupt Exception
|
|
|
|
// if CPU interrupt line is still asserted
|
2023-11-21 08:06:50 -07:00
|
|
|
if (ppc_state.msr & MSR::EE && int_pin) {
|
2022-08-24 12:51:43 +02:00
|
|
|
uint32_t save_srr0 = ppc_state.spr[SPR::SRR0] & 0xFFFFFFFCUL;
|
|
|
|
ppc_exception_handler(Except_Type::EXC_EXT_INT, 0);
|
|
|
|
ppc_state.spr[SPR::SRR0] = save_srr0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-11-21 08:06:50 -07:00
|
|
|
if ((ppc_state.msr & MSR::EE) && dec_exception_pending) {
|
2023-08-07 11:11:02 -07:00
|
|
|
dec_exception_pending = false;
|
|
|
|
//LOG_F(WARNING, "decrementer exception from rfi msr:0x%X", ppc_state.msr);
|
|
|
|
uint32_t save_srr0 = ppc_state.spr[SPR::SRR0] & 0xFFFFFFFCUL;
|
|
|
|
ppc_exception_handler(Except_Type::EXC_DECR, 0);
|
|
|
|
ppc_state.spr[SPR::SRR0] = save_srr0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_next_instruction_address = ppc_state.spr[SPR::SRR0] & 0xFFFFFFFCUL;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2021-10-13 20:58:09 +02:00
|
|
|
do_ctx_sync(); // RFI is context synchronizing
|
|
|
|
|
2021-05-16 00:53:15 +02:00
|
|
|
mmu_change_mode();
|
|
|
|
|
2019-07-01 19:15:33 -07:00
|
|
|
grab_return = true;
|
2022-03-02 16:55:20 +01:00
|
|
|
exec_flags = EXEF_RFI;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_sc() {
|
2021-10-13 20:58:09 +02:00
|
|
|
do_ctx_sync(); // SC is context synchronizing!
|
2020-01-11 21:48:56 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_SYSCALL, 0x20000);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_tw() {
|
2020-05-12 23:55:45 +05:00
|
|
|
reg_a = (ppc_cur_instruction >> 11) & 31;
|
|
|
|
reg_b = (ppc_cur_instruction >> 16) & 31;
|
2024-01-07 17:04:51 -07:00
|
|
|
uint32_t ppc_to = (ppc_cur_instruction >> 21) & 31;
|
2021-01-07 20:38:32 -07:00
|
|
|
if ((((int32_t)ppc_state.gpr[reg_a] < (int32_t)ppc_state.gpr[reg_b]) && (ppc_to & 0x10)) ||
|
|
|
|
(((int32_t)ppc_state.gpr[reg_a] > (int32_t)ppc_state.gpr[reg_b]) && (ppc_to & 0x08)) ||
|
|
|
|
(((int32_t)ppc_state.gpr[reg_a] == (int32_t)ppc_state.gpr[reg_b]) && (ppc_to & 0x04)) ||
|
|
|
|
((ppc_state.gpr[reg_a] < ppc_state.gpr[reg_b]) && (ppc_to & 0x02)) ||
|
|
|
|
((ppc_state.gpr[reg_a] > ppc_state.gpr[reg_b]) && (ppc_to & 0x01))) {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::TRAP);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_twi() {
|
2020-05-12 23:55:45 +05:00
|
|
|
simm = (int32_t)((int16_t)((ppc_cur_instruction)&0xFFFF));
|
2020-04-21 04:47:08 +02:00
|
|
|
reg_a = (ppc_cur_instruction >> 16) & 0x1F;
|
2024-01-07 17:04:51 -07:00
|
|
|
uint32_t ppc_to = (ppc_cur_instruction >> 21) & 0x1F;
|
2020-05-12 23:55:45 +05:00
|
|
|
if ((((int32_t)ppc_state.gpr[reg_a] < simm) && (ppc_to & 0x10)) ||
|
|
|
|
(((int32_t)ppc_state.gpr[reg_a] > simm) && (ppc_to & 0x08)) ||
|
2020-04-21 04:47:08 +02:00
|
|
|
(((int32_t)ppc_state.gpr[reg_a] == simm) && (ppc_to & 0x04)) ||
|
|
|
|
((ppc_state.gpr[reg_a] < (uint32_t)simm) && (ppc_to & 0x02)) ||
|
|
|
|
((ppc_state.gpr[reg_a] > (uint32_t)simm) && (ppc_to & 0x01))) {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::TRAP);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_eieio() {
|
2020-02-27 16:31:22 +01:00
|
|
|
/* placeholder */
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_isync() {
|
2021-08-03 16:01:32 +02:00
|
|
|
do_ctx_sync();
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_sync() {
|
2020-02-27 16:31:22 +01:00
|
|
|
/* placeholder */
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_icbi() {
|
2020-02-27 16:31:22 +01:00
|
|
|
/* placeholder */
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_dcbf() {
|
2020-02-27 16:31:22 +01:00
|
|
|
/* placeholder */
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_dcbi() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_supervisor_instrs++;
|
2020-01-25 19:30:55 -07:00
|
|
|
#endif
|
2020-02-27 16:31:22 +01:00
|
|
|
/* placeholder */
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_dcbst() {
|
2023-09-18 20:45:35 +02:00
|
|
|
/* placeholder */
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_dcbt() {
|
2020-05-12 23:55:45 +05:00
|
|
|
// Not needed, the HDI reg is touched to no-op this instruction.
|
2019-07-01 19:15:33 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_dcbtst() {
|
2020-05-12 23:55:45 +05:00
|
|
|
// Not needed, the HDI reg is touched to no-op this instruction.
|
2019-07-01 19:15:33 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_dcbz() {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2020-01-25 19:30:55 -07:00
|
|
|
ppc_effective_address = (reg_a == 0) ? ppc_result_b : (ppc_result_a + ppc_result_b);
|
2021-01-07 04:25:27 +01:00
|
|
|
|
2023-06-17 20:57:48 +02:00
|
|
|
ppc_effective_address &= 0xFFFFFFE0UL; // align EA on a 32-byte boundary
|
|
|
|
|
|
|
|
// the following is not especially efficient but necessary
|
|
|
|
// to make BlockZero under Mac OS 8.x and later to work
|
|
|
|
mmu_write_vmem<uint64_t>(ppc_effective_address + 0, 0);
|
|
|
|
mmu_write_vmem<uint64_t>(ppc_effective_address + 8, 0);
|
|
|
|
mmu_write_vmem<uint64_t>(ppc_effective_address + 16, 0);
|
|
|
|
mmu_write_vmem<uint64_t>(ppc_effective_address + 24, 0);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-01-21 19:25:50 -07:00
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
// Integer Load and Store Functions
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_stb() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssa();
|
2024-01-31 08:06:33 -07:00
|
|
|
ppc_effective_address = int32_t(int16_t(ppc_cur_instruction));
|
2021-01-25 21:03:17 -07:00
|
|
|
ppc_effective_address += reg_a ? ppc_result_a : 0;
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_write_vmem<uint8_t>(ppc_effective_address, ppc_result_d);
|
|
|
|
//mem_write_byte(ppc_effective_address, ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_stbx() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
2021-01-25 21:03:17 -07:00
|
|
|
ppc_effective_address = reg_a ? (ppc_result_a + ppc_result_b) : ppc_result_b;
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_write_vmem<uint8_t>(ppc_effective_address, ppc_result_d);
|
|
|
|
//mem_write_byte(ppc_effective_address, ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_stbu() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssa();
|
2020-01-21 19:25:50 -07:00
|
|
|
if (reg_a != 0) {
|
2024-01-31 08:06:33 -07:00
|
|
|
ppc_effective_address = int32_t(int16_t(ppc_cur_instruction));
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_effective_address += ppc_result_a;
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_write_vmem<uint8_t>(ppc_effective_address, ppc_result_d);
|
|
|
|
//mem_write_byte(ppc_effective_address, ppc_result_d);
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.gpr[reg_a] = ppc_effective_address;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::ILLEGAL_OP);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_stbux() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
2020-01-21 19:25:50 -07:00
|
|
|
if (reg_a != 0) {
|
2020-01-17 21:26:09 -07:00
|
|
|
ppc_effective_address = ppc_result_a + ppc_result_b;
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_write_vmem<uint8_t>(ppc_effective_address, ppc_result_d);
|
|
|
|
//mem_write_byte(ppc_effective_address, ppc_result_d);
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.gpr[reg_a] = ppc_effective_address;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::ILLEGAL_OP);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_sth() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssa();
|
2024-01-31 08:06:33 -07:00
|
|
|
ppc_effective_address = int32_t(int16_t(ppc_cur_instruction));
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_effective_address += (reg_a > 0) ? ppc_result_a : 0;
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_write_vmem<uint16_t>(ppc_effective_address, ppc_result_d);
|
|
|
|
//mem_write_word(ppc_effective_address, ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_sthu() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssa();
|
2020-01-17 21:06:26 -07:00
|
|
|
if (reg_a != 0) {
|
2024-01-31 08:06:33 -07:00
|
|
|
ppc_effective_address = int32_t(int16_t(ppc_cur_instruction));
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_effective_address += ppc_result_a;
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_write_vmem<uint16_t>(ppc_effective_address, ppc_result_d);
|
|
|
|
//mem_write_word(ppc_effective_address, ppc_result_d);
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.gpr[reg_a] = ppc_effective_address;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::ILLEGAL_OP);
|
2020-01-17 21:06:26 -07:00
|
|
|
}
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_sthux() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
2020-01-17 21:06:26 -07:00
|
|
|
if (reg_a != 0) {
|
2020-01-17 21:26:09 -07:00
|
|
|
ppc_effective_address = ppc_result_a + ppc_result_b;
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_write_vmem<uint16_t>(ppc_effective_address, ppc_result_d);
|
|
|
|
//mem_write_word(ppc_effective_address, ppc_result_d);
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.gpr[reg_a] = ppc_effective_address;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::ILLEGAL_OP);
|
2020-01-17 21:06:26 -07:00
|
|
|
}
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_sthx() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
2021-01-25 21:03:17 -07:00
|
|
|
ppc_effective_address = reg_a ? (ppc_result_a + ppc_result_b) : ppc_result_b;
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_write_vmem<uint16_t>(ppc_effective_address, ppc_result_d);
|
|
|
|
//mem_write_word(ppc_effective_address, ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_sthbrx() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_effective_address = (reg_a == 0) ? ppc_result_b : (ppc_result_a + ppc_result_b);
|
2020-05-12 23:55:45 +05:00
|
|
|
ppc_result_d = (uint32_t)(BYTESWAP_16((uint16_t)ppc_result_d));
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_write_vmem<uint16_t>(ppc_effective_address, ppc_result_d);
|
|
|
|
//mem_write_word(ppc_effective_address, ppc_result_d);
|
2020-01-21 19:25:50 -07:00
|
|
|
}
|
2021-08-03 16:01:32 +02:00
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_stw() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssa();
|
2024-01-31 08:06:33 -07:00
|
|
|
ppc_effective_address = int32_t(int16_t(ppc_cur_instruction));
|
2021-01-25 21:03:17 -07:00
|
|
|
ppc_effective_address += reg_a ? ppc_result_a : 0;
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_write_vmem<uint32_t>(ppc_effective_address, ppc_result_d);
|
|
|
|
//mem_write_dword(ppc_effective_address, ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_stwx() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
2021-01-25 21:03:17 -07:00
|
|
|
ppc_effective_address = reg_a ? (ppc_result_a + ppc_result_b) : ppc_result_b;
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_write_vmem<uint32_t>(ppc_effective_address, ppc_result_d);
|
|
|
|
//mem_write_dword(ppc_effective_address, ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_stwcx() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2020-10-17 14:30:37 -07:00
|
|
|
if (rc_flag == 0) {
|
2020-11-29 23:53:03 +01:00
|
|
|
ppc_illegalop();
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-10-17 14:30:37 -07:00
|
|
|
ppc_grab_regssab();
|
|
|
|
ppc_effective_address = (reg_a == 0) ? ppc_result_b : (ppc_result_a + ppc_result_b);
|
2023-08-21 04:50:02 +02:00
|
|
|
ppc_state.cr &= 0x0FFFFFFFUL; // clear CR0
|
|
|
|
ppc_state.cr |= (ppc_state.spr[SPR::XER] & 0x80000000UL) >> 3; // copy XER[SO] to CR0[SO]
|
2020-10-17 14:30:37 -07:00
|
|
|
if (ppc_state.reserve) {
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_write_vmem<uint32_t>(ppc_effective_address, ppc_result_d);
|
2020-10-17 14:30:37 -07:00
|
|
|
ppc_state.reserve = false;
|
2023-08-21 04:50:02 +02:00
|
|
|
ppc_state.cr |= 0x20000000UL; // set CR0[EQ]
|
2020-10-17 14:30:37 -07:00
|
|
|
}
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_stwu() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssa();
|
2020-01-17 21:06:26 -07:00
|
|
|
if (reg_a != 0) {
|
2024-01-31 08:06:33 -07:00
|
|
|
ppc_effective_address = int32_t(int16_t(ppc_cur_instruction));
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_effective_address += ppc_result_a;
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_write_vmem<uint32_t>(ppc_effective_address, ppc_result_d);
|
|
|
|
//mem_write_dword(ppc_effective_address, ppc_result_d);
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.gpr[reg_a] = ppc_effective_address;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::ILLEGAL_OP);
|
2020-01-17 21:06:26 -07:00
|
|
|
}
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_stwux() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
2020-01-17 21:06:26 -07:00
|
|
|
if (reg_a != 0) {
|
2020-01-17 21:26:09 -07:00
|
|
|
ppc_effective_address = ppc_result_a + ppc_result_b;
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_write_vmem<uint32_t>(ppc_effective_address, ppc_result_d);
|
|
|
|
//mem_write_dword(ppc_effective_address, ppc_result_d);
|
2020-03-04 21:29:04 -07:00
|
|
|
ppc_state.gpr[reg_a] = ppc_effective_address;
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::ILLEGAL_OP);
|
2020-01-17 21:06:26 -07:00
|
|
|
}
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_stwbrx() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
2021-01-25 21:03:17 -07:00
|
|
|
ppc_effective_address = reg_a ? (ppc_result_a + ppc_result_b) : ppc_result_b;
|
2020-05-12 23:55:45 +05:00
|
|
|
ppc_result_d = BYTESWAP_32(ppc_result_d);
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_write_vmem<uint32_t>(ppc_effective_address, ppc_result_d);
|
|
|
|
//mem_write_dword(ppc_effective_address, ppc_result_d);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_stmw() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssa();
|
2024-01-31 08:06:33 -07:00
|
|
|
ppc_effective_address = int32_t(int16_t(ppc_cur_instruction));
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_effective_address += (reg_a > 0) ? ppc_result_a : 0;
|
2020-04-21 04:47:51 +02:00
|
|
|
|
|
|
|
/* what should we do if EA is unaligned? */
|
|
|
|
if (ppc_effective_address & 3) {
|
|
|
|
ppc_exception_handler(Except_Type::EXC_ALIGNMENT, 0x00000);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (; reg_s <= 31; reg_s++) {
|
2021-08-03 16:01:32 +02:00
|
|
|
mmu_write_vmem<uint32_t>(ppc_effective_address, ppc_state.gpr[reg_s]);
|
|
|
|
//mem_write_dword(ppc_effective_address, ppc_state.gpr[reg_s]);
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_effective_address += 4;
|
2020-04-21 04:47:51 +02:00
|
|
|
}
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lbz() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsda();
|
2024-01-31 08:06:33 -07:00
|
|
|
ppc_effective_address = int32_t(int16_t(ppc_cur_instruction));
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_effective_address += (reg_a > 0) ? ppc_result_a : 0;
|
2021-06-20 22:33:03 +02:00
|
|
|
//ppc_result_d = mem_grab_byte(ppc_effective_address);
|
|
|
|
ppc_result_d = mmu_read_vmem<uint8_t>(ppc_effective_address);
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lbzu() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsda();
|
2024-01-31 08:06:33 -07:00
|
|
|
ppc_effective_address = int32_t(int16_t(ppc_cur_instruction));
|
2020-01-21 19:25:50 -07:00
|
|
|
if ((reg_a != reg_d) || reg_a != 0) {
|
|
|
|
ppc_effective_address += ppc_result_a;
|
2021-06-20 22:33:03 +02:00
|
|
|
//ppc_result_d = mem_grab_byte(ppc_effective_address);
|
|
|
|
ppc_result_d = mmu_read_vmem<uint8_t>(ppc_effective_address);
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_result_a = ppc_effective_address;
|
|
|
|
ppc_store_result_regd();
|
|
|
|
ppc_store_result_rega();
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::ILLEGAL_OP);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lbzx() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2021-01-25 21:03:17 -07:00
|
|
|
ppc_effective_address = reg_a ? (ppc_result_a + ppc_result_b) : ppc_result_b;
|
2021-06-20 22:33:03 +02:00
|
|
|
//ppc_result_d = mem_grab_byte(ppc_effective_address);
|
|
|
|
ppc_result_d = mmu_read_vmem<uint8_t>(ppc_effective_address);
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lbzux() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2020-01-21 19:25:50 -07:00
|
|
|
if ((reg_a != reg_d) || reg_a != 0) {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_effective_address = ppc_result_a + ppc_result_b;
|
2021-06-20 22:33:03 +02:00
|
|
|
//ppc_result_d = mem_grab_byte(ppc_effective_address);
|
|
|
|
ppc_result_d = mmu_read_vmem<uint8_t>(ppc_effective_address);
|
2020-05-12 23:55:45 +05:00
|
|
|
ppc_result_a = ppc_effective_address;
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
ppc_store_result_rega();
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::ILLEGAL_OP);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lhz() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsda();
|
2024-01-31 08:06:33 -07:00
|
|
|
ppc_effective_address = int32_t(int16_t(ppc_cur_instruction));
|
2021-01-25 21:03:17 -07:00
|
|
|
ppc_effective_address += reg_a ? ppc_result_a : 0;
|
2021-06-20 22:33:03 +02:00
|
|
|
//ppc_result_d = mem_grab_word(ppc_effective_address);
|
|
|
|
ppc_result_d = mmu_read_vmem<uint16_t>(ppc_effective_address);
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lhzu() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsda();
|
2020-01-21 19:25:50 -07:00
|
|
|
if ((reg_a != reg_d) || reg_a != 0) {
|
2024-01-31 08:06:33 -07:00
|
|
|
ppc_effective_address = int32_t(int16_t(ppc_cur_instruction));
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_effective_address += ppc_result_a;
|
2021-06-20 22:33:03 +02:00
|
|
|
//ppc_result_d = mem_grab_word(ppc_effective_address);
|
|
|
|
ppc_result_d = mmu_read_vmem<uint16_t>(ppc_effective_address);
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_result_a = ppc_effective_address;
|
|
|
|
ppc_store_result_regd();
|
|
|
|
ppc_store_result_rega();
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::ILLEGAL_OP);
|
2020-01-21 19:25:50 -07:00
|
|
|
}
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lhzx() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2021-01-25 21:03:17 -07:00
|
|
|
ppc_effective_address = reg_a ? (ppc_result_a + ppc_result_b) : ppc_result_b;
|
2021-06-20 22:33:03 +02:00
|
|
|
//ppc_result_d = mem_grab_word(ppc_effective_address);
|
|
|
|
ppc_result_d = mmu_read_vmem<uint16_t>(ppc_effective_address);
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lhzux() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2020-01-21 19:25:50 -07:00
|
|
|
if ((reg_a != reg_d) || reg_a != 0) {
|
|
|
|
ppc_effective_address = ppc_result_a + ppc_result_b;
|
2021-06-20 22:33:03 +02:00
|
|
|
//ppc_result_d = mem_grab_word(ppc_effective_address);
|
|
|
|
ppc_result_d = mmu_read_vmem<uint16_t>(ppc_effective_address);
|
|
|
|
ppc_result_a = ppc_effective_address;
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
ppc_store_result_rega();
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::ILLEGAL_OP);
|
2020-01-21 19:25:50 -07:00
|
|
|
}
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lha() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsda();
|
2024-01-31 08:06:33 -07:00
|
|
|
ppc_effective_address = int32_t(int16_t(ppc_cur_instruction));
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_effective_address += (reg_a > 0) ? ppc_result_a : 0;
|
2021-06-20 22:33:03 +02:00
|
|
|
//uint16_t val = mem_grab_word(ppc_effective_address);
|
|
|
|
uint16_t val = mmu_read_vmem<uint16_t>(ppc_effective_address);
|
2024-01-05 19:10:05 -07:00
|
|
|
ppc_result_d = int32_t(int16_t(val));
|
2020-01-27 01:36:22 +01:00
|
|
|
ppc_store_result_regd();
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lhau() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsda();
|
2020-01-21 19:25:50 -07:00
|
|
|
if ((reg_a != reg_d) || reg_a != 0) {
|
2024-01-31 08:06:33 -07:00
|
|
|
ppc_effective_address = int32_t(int16_t(ppc_cur_instruction));
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_effective_address += ppc_result_a;
|
2021-06-20 22:33:03 +02:00
|
|
|
//uint16_t val = mem_grab_word(ppc_effective_address);
|
|
|
|
uint16_t val = mmu_read_vmem<uint16_t>(ppc_effective_address);
|
2024-01-05 19:10:05 -07:00
|
|
|
ppc_result_d = int32_t(int16_t(val));
|
2020-01-27 01:36:22 +01:00
|
|
|
ppc_store_result_regd();
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_result_a = ppc_effective_address;
|
|
|
|
ppc_store_result_rega();
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::ILLEGAL_OP);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lhaux() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2024-01-05 19:10:05 -07:00
|
|
|
if ((reg_a != reg_d) || reg_a != 0) {
|
|
|
|
ppc_effective_address = ppc_result_a + ppc_result_b;
|
|
|
|
// uint16_t val = mem_grab_word(ppc_effective_address);
|
|
|
|
uint16_t val = mmu_read_vmem<uint16_t>(ppc_effective_address);
|
|
|
|
ppc_result_d = int32_t(int16_t(val));
|
|
|
|
ppc_store_result_regd();
|
|
|
|
ppc_result_a = ppc_effective_address;
|
|
|
|
ppc_store_result_rega();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::ILLEGAL_OP);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lhax() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2021-01-25 21:03:17 -07:00
|
|
|
ppc_effective_address = reg_a ? (ppc_result_a + ppc_result_b) : ppc_result_b;
|
2021-06-20 22:33:03 +02:00
|
|
|
//uint16_t val = mem_grab_word(ppc_effective_address);
|
|
|
|
uint16_t val = mmu_read_vmem<uint16_t>(ppc_effective_address);
|
2024-01-05 19:10:05 -07:00
|
|
|
ppc_result_d = int32_t(int16_t(val));
|
2020-01-27 01:36:22 +01:00
|
|
|
ppc_store_result_regd();
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lhbrx() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2021-01-25 21:03:17 -07:00
|
|
|
ppc_effective_address = reg_a ? (ppc_result_a + ppc_result_b) : ppc_result_b;
|
2021-06-20 22:33:03 +02:00
|
|
|
//ppc_result_d = (uint32_t)(BYTESWAP_16(mem_grab_word(ppc_effective_address)));
|
|
|
|
ppc_result_d = (uint32_t)(BYTESWAP_16(mmu_read_vmem<uint16_t>(ppc_effective_address)));
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lwz() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsda();
|
2024-01-31 08:06:33 -07:00
|
|
|
ppc_effective_address = int32_t(int16_t(ppc_cur_instruction));
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_effective_address += (reg_a > 0) ? ppc_result_a : 0;
|
2021-06-20 22:33:03 +02:00
|
|
|
//ppc_result_d = mem_grab_dword(ppc_effective_address);
|
|
|
|
ppc_result_d = mmu_read_vmem<uint32_t>(ppc_effective_address);
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lwbrx() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2021-01-25 21:03:17 -07:00
|
|
|
ppc_effective_address = reg_a ? (ppc_result_a + ppc_result_b) : ppc_result_b;
|
2021-06-20 22:33:03 +02:00
|
|
|
//ppc_result_d = BYTESWAP_32(mem_grab_dword(ppc_effective_address));
|
|
|
|
ppc_result_d = BYTESWAP_32(mmu_read_vmem<uint32_t>(ppc_effective_address));
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lwzu() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsda();
|
2024-01-31 08:06:33 -07:00
|
|
|
ppc_effective_address = int32_t(int16_t(ppc_cur_instruction));
|
2020-01-21 19:25:50 -07:00
|
|
|
if ((reg_a != reg_d) || reg_a != 0) {
|
|
|
|
ppc_effective_address += ppc_result_a;
|
2021-06-20 22:33:03 +02:00
|
|
|
//ppc_result_d = mem_grab_dword(ppc_effective_address);
|
|
|
|
ppc_result_d = mmu_read_vmem<uint32_t>(ppc_effective_address);
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
ppc_result_a = ppc_effective_address;
|
|
|
|
ppc_store_result_rega();
|
2020-05-12 23:55:45 +05:00
|
|
|
} else {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::ILLEGAL_OP);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lwzx() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2021-01-25 21:03:17 -07:00
|
|
|
ppc_effective_address = reg_a ? (ppc_result_a + ppc_result_b) : ppc_result_b;
|
2021-06-20 22:33:03 +02:00
|
|
|
//ppc_result_d = mem_grab_dword(ppc_effective_address);
|
|
|
|
ppc_result_d = mmu_read_vmem<uint32_t>(ppc_effective_address);
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lwzux() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2020-01-21 19:25:50 -07:00
|
|
|
if ((reg_a != reg_d) || reg_a != 0) {
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_effective_address = ppc_result_a + ppc_result_b;
|
2024-01-07 17:04:51 -07:00
|
|
|
// ppc_result_d = mem_grab_dword(ppc_effective_address);
|
|
|
|
ppc_result_d = mmu_read_vmem<uint32_t>(ppc_effective_address);
|
|
|
|
ppc_result_a = ppc_effective_address;
|
|
|
|
ppc_store_result_regd();
|
|
|
|
ppc_store_result_rega();
|
2024-01-07 17:45:05 -07:00
|
|
|
}
|
|
|
|
else {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::ILLEGAL_OP);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lwarx() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2020-05-12 23:55:45 +05:00
|
|
|
// Placeholder - Get the reservation of memory implemented!
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2020-01-21 19:25:50 -07:00
|
|
|
ppc_effective_address = (reg_a == 0) ? ppc_result_b : (ppc_result_a + ppc_result_b);
|
2020-05-12 23:55:45 +05:00
|
|
|
ppc_state.reserve = true;
|
2021-06-20 22:33:03 +02:00
|
|
|
//ppc_result_d = mem_grab_dword(ppc_effective_address);
|
|
|
|
ppc_result_d = mmu_read_vmem<uint32_t>(ppc_effective_address);
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lmw() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsda();
|
2024-01-07 17:45:05 -07:00
|
|
|
ppc_effective_address = int32_t(int16_t(ppc_cur_instruction & 0xFFFF));
|
|
|
|
ppc_effective_address += (reg_a > 0) ? ppc_result_a : 0;
|
|
|
|
// How many words to load in memory - using a do-while for this
|
|
|
|
do {
|
|
|
|
//ppc_state.gpr[reg_d] = mem_grab_dword(ppc_effective_address);
|
|
|
|
ppc_state.gpr[reg_d] = mmu_read_vmem<uint32_t>(ppc_effective_address);
|
|
|
|
ppc_effective_address += 4;
|
|
|
|
reg_d++;
|
|
|
|
} while (reg_d < 32);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lswi() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsda();
|
2022-02-19 23:21:18 +01:00
|
|
|
ppc_effective_address = reg_a ? ppc_result_a : 0;
|
2024-01-05 15:11:37 -07:00
|
|
|
uint32_t grab_inb = (ppc_cur_instruction >> 11) & 0x1F;
|
2022-02-19 23:21:18 +01:00
|
|
|
grab_inb = grab_inb ? grab_inb : 32;
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2023-10-29 17:23:31 -07:00
|
|
|
while (grab_inb >= 4) {
|
|
|
|
ppc_state.gpr[reg_d] = mmu_read_vmem<uint32_t>(ppc_effective_address);
|
|
|
|
reg_d++;
|
|
|
|
if (reg_d >= 32) { // wrap around through GPR0
|
|
|
|
reg_d = 0;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2023-10-29 17:23:31 -07:00
|
|
|
ppc_effective_address += 4;
|
|
|
|
grab_inb -= 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
// handle remaining bytes
|
|
|
|
switch (grab_inb) {
|
|
|
|
case 1:
|
|
|
|
ppc_state.gpr[reg_d] = mmu_read_vmem<uint8_t>(ppc_effective_address) << 24;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
ppc_state.gpr[reg_d] = mmu_read_vmem<uint16_t>(ppc_effective_address) << 16;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
ppc_state.gpr[reg_d] = mmu_read_vmem<uint16_t>(ppc_effective_address) << 16;
|
|
|
|
ppc_state.gpr[reg_d] += mmu_read_vmem<uint8_t>(ppc_effective_address + 2) << 8;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_lswx() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_loads++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regsdab();
|
2021-01-25 21:03:17 -07:00
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
// Invalid instruction forms
|
2024-01-07 17:21:11 -07:00
|
|
|
if ((reg_d == 0 && reg_a == 0) || (reg_d == reg_a) || (reg_d == reg_b)) {
|
2020-11-30 20:59:36 +01:00
|
|
|
ppc_exception_handler(Except_Type::EXC_PROGRAM, Exc_Cause::ILLEGAL_OP);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2021-01-25 21:03:17 -07:00
|
|
|
|
|
|
|
ppc_effective_address = reg_a ? (ppc_result_a + ppc_result_b) : ppc_result_b;
|
2024-01-05 15:11:37 -07:00
|
|
|
uint32_t grab_inb = ppc_state.spr[SPR::XER] & 0x7F;
|
2021-01-25 21:03:17 -07:00
|
|
|
|
2023-10-29 17:23:31 -07:00
|
|
|
while (grab_inb >= 4) {
|
|
|
|
ppc_state.gpr[reg_d] = mmu_read_vmem<uint32_t>(ppc_effective_address);
|
|
|
|
reg_d++;
|
|
|
|
if (reg_d >= 32) { // wrap around through GPR0
|
|
|
|
reg_d = 0;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2023-10-29 17:23:31 -07:00
|
|
|
ppc_effective_address += 4;
|
|
|
|
grab_inb -= 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
// handle remaining bytes
|
|
|
|
switch (grab_inb) {
|
|
|
|
case 1:
|
|
|
|
ppc_state.gpr[reg_d] = mmu_read_vmem<uint8_t>(ppc_effective_address) << 24;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
ppc_state.gpr[reg_d] = mmu_read_vmem<uint16_t>(ppc_effective_address) << 16;
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
ppc_state.gpr[reg_d] = mmu_read_vmem<uint16_t>(ppc_effective_address) << 16;
|
|
|
|
ppc_state.gpr[reg_d] += mmu_read_vmem<uint8_t>(ppc_effective_address + 2) << 8;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_stswi() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssa();
|
2022-02-19 23:21:18 +01:00
|
|
|
ppc_effective_address = reg_a ? ppc_result_a : 0;
|
2024-01-05 15:11:37 -07:00
|
|
|
uint32_t grab_inb = (ppc_cur_instruction >> 11) & 0x1F;
|
2022-02-19 23:21:18 +01:00
|
|
|
grab_inb = grab_inb ? grab_inb : 32;
|
2021-01-25 21:03:17 -07:00
|
|
|
|
2023-10-29 17:23:31 -07:00
|
|
|
while (grab_inb >= 4) {
|
|
|
|
mmu_write_vmem<uint32_t>(ppc_effective_address, ppc_state.gpr[reg_s]);
|
|
|
|
reg_s++;
|
|
|
|
if (reg_s >= 32) { // wrap around through GPR0
|
|
|
|
reg_s = 0;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2023-10-29 17:23:31 -07:00
|
|
|
ppc_effective_address += 4;
|
|
|
|
grab_inb -= 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
// handle remaining bytes
|
|
|
|
switch (grab_inb) {
|
|
|
|
case 1:
|
|
|
|
mmu_write_vmem<uint8_t>(ppc_effective_address, ppc_state.gpr[reg_s] >> 24);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
mmu_write_vmem<uint16_t>(ppc_effective_address, ppc_state.gpr[reg_s] >> 16);
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
mmu_write_vmem<uint16_t>(ppc_effective_address, ppc_state.gpr[reg_s] >> 16);
|
|
|
|
mmu_write_vmem<uint8_t>(ppc_effective_address + 2, (ppc_state.gpr[reg_s] >> 8) & 0xFF);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_stswx() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_int_stores++;
|
|
|
|
#endif
|
2019-07-01 19:15:33 -07:00
|
|
|
ppc_grab_regssab();
|
2021-01-25 21:03:17 -07:00
|
|
|
ppc_effective_address = reg_a ? (ppc_result_a + ppc_result_b) : ppc_result_b;
|
2024-01-05 15:11:37 -07:00
|
|
|
uint32_t grab_inb = ppc_state.spr[SPR::XER] & 127;
|
2021-01-25 21:03:17 -07:00
|
|
|
|
2023-10-29 17:23:31 -07:00
|
|
|
while (grab_inb >= 4) {
|
|
|
|
mmu_write_vmem<uint32_t>(ppc_effective_address, ppc_state.gpr[reg_s]);
|
|
|
|
reg_s++;
|
|
|
|
if (reg_s >= 32) { // wrap around through GPR0
|
|
|
|
reg_s = 0;
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2023-10-29 17:23:31 -07:00
|
|
|
ppc_effective_address += 4;
|
|
|
|
grab_inb -= 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
// handle remaining bytes
|
|
|
|
switch (grab_inb) {
|
|
|
|
case 1:
|
|
|
|
mmu_write_vmem<uint8_t>(ppc_effective_address, ppc_state.gpr[reg_s] >> 24);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
mmu_write_vmem<uint16_t>(ppc_effective_address, ppc_state.gpr[reg_s] >> 16);
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
mmu_write_vmem<uint16_t>(ppc_effective_address, ppc_state.gpr[reg_s] >> 16);
|
|
|
|
mmu_write_vmem<uint8_t>(ppc_effective_address + 2, (ppc_state.gpr[reg_s] >> 8) & 0xFF);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2020-01-25 19:30:55 -07:00
|
|
|
}
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2021-10-05 17:40:09 -07:00
|
|
|
void dppc_interpreter::ppc_eciwx() {
|
|
|
|
uint32_t ear_enable = 0x80000000;
|
|
|
|
|
|
|
|
// error if EAR[E] != 1
|
|
|
|
if (!(ppc_state.spr[282] && ear_enable)) {
|
|
|
|
ppc_exception_handler(Except_Type::EXC_DSI, 0x0);
|
|
|
|
}
|
|
|
|
|
|
|
|
ppc_grab_regsdab();
|
|
|
|
ppc_effective_address = (reg_a == 0) ? ppc_result_b : (ppc_result_a + ppc_result_b);
|
|
|
|
|
|
|
|
if (ppc_effective_address & 0x3) {
|
|
|
|
ppc_exception_handler(Except_Type::EXC_ALIGNMENT, 0x0);
|
|
|
|
}
|
|
|
|
|
|
|
|
ppc_result_d = mmu_read_vmem<uint32_t>(ppc_effective_address);
|
|
|
|
|
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
|
|
|
void dppc_interpreter::ppc_ecowx() {
|
|
|
|
uint32_t ear_enable = 0x80000000;
|
|
|
|
|
|
|
|
// error if EAR[E] != 1
|
|
|
|
if (!(ppc_state.spr[282] && ear_enable)) {
|
|
|
|
ppc_exception_handler(Except_Type::EXC_DSI, 0x0);
|
|
|
|
}
|
|
|
|
|
|
|
|
ppc_grab_regssab();
|
|
|
|
ppc_effective_address = (reg_a == 0) ? ppc_result_b : (ppc_result_a + ppc_result_b);
|
|
|
|
|
|
|
|
if (ppc_effective_address & 0x3) {
|
|
|
|
ppc_exception_handler(Except_Type::EXC_ALIGNMENT, 0x0);
|
|
|
|
}
|
|
|
|
|
|
|
|
mmu_write_vmem<uint32_t>(ppc_effective_address, ppc_result_d);
|
|
|
|
|
|
|
|
ppc_store_result_regd();
|
|
|
|
}
|
|
|
|
|
2020-05-12 23:55:45 +05:00
|
|
|
// TLB Instructions
|
2019-07-01 19:15:33 -07:00
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_tlbie() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_supervisor_instrs++;
|
2020-01-14 20:50:01 -07:00
|
|
|
#endif
|
2021-05-16 13:32:21 +02:00
|
|
|
|
2021-08-03 16:01:32 +02:00
|
|
|
tlb_flush_entry(ppc_state.gpr[(ppc_cur_instruction >> 11) & 31]);
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_tlbia() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_supervisor_instrs++;
|
2020-01-14 20:50:01 -07:00
|
|
|
#endif
|
2020-02-27 16:31:22 +01:00
|
|
|
/* placeholder */
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_tlbld() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_supervisor_instrs++;
|
2020-01-14 20:50:01 -07:00
|
|
|
#endif
|
2020-02-27 16:31:22 +01:00
|
|
|
/* placeholder */
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_tlbli() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_supervisor_instrs++;
|
2020-01-14 20:50:01 -07:00
|
|
|
#endif
|
2020-02-27 16:31:22 +01:00
|
|
|
/* placeholder */
|
2019-07-01 19:15:33 -07:00
|
|
|
}
|
2020-01-14 20:50:01 -07:00
|
|
|
|
2020-10-17 21:46:38 -07:00
|
|
|
void dppc_interpreter::ppc_tlbsync() {
|
2021-04-29 02:26:17 +02:00
|
|
|
#ifdef CPU_PROFILING
|
|
|
|
num_supervisor_instrs++;
|
2020-01-14 20:50:01 -07:00
|
|
|
#endif
|
2020-02-27 16:31:22 +01:00
|
|
|
/* placeholder */
|
|
|
|
}
|