mirror of
https://github.com/RetroDECK/Supermodel.git
synced 2024-11-24 14:45:40 +00:00
392900fee2
On PowerPC round to nearest ties to even, not away from zero Also implement correct behavior for ppc_fresx Fixes "tips to win" sequence in Daytona 2 BOTE
2826 lines
44 KiB
C
2826 lines
44 KiB
C
/**
|
|
** Supermodel
|
|
** A Sega Model 3 Arcade Emulator.
|
|
** Copyright 2011 Bart Trzynadlowski, Nik Henson
|
|
**
|
|
** This file is part of Supermodel.
|
|
**
|
|
** Supermodel 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.
|
|
**
|
|
** Supermodel 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 Supermodel. If not, see <http://www.gnu.org/licenses/>.
|
|
**/
|
|
|
|
/*
|
|
* ppc_ops.c
|
|
*
|
|
* PowerPC common opcodes. Included from ppc.cpp; do not compile compile
|
|
* separately.
|
|
*
|
|
* Changes to opcode handlers since inclusion in new Supermodel:
|
|
* - Feb. 13 2011: Changed stwcx. to always set the EQ flag.
|
|
*/
|
|
|
|
/* PowerPC common opcodes */
|
|
|
|
// it really seems like this should be elsewhere - like maybe the floating point checks can hang out someplace else
|
|
#include <math.h>
|
|
#include <cfenv>
|
|
|
|
static void ppc_unimplemented(UINT32 op)
|
|
{
|
|
ErrorLog("PowerPC hit an unimplemented instruction. Halting emulation until reset.");
|
|
DebugLog("PowerPC encountered an unimplemented opcode %08X at %08X\n", op, ppc.pc);
|
|
ppc.fatalError = true;
|
|
}
|
|
|
|
static void ppc_addx(UINT32 op)
|
|
{
|
|
UINT32 ra = REG(RA);
|
|
UINT32 rb = REG(RB);
|
|
|
|
REG(RT) = ra + rb;
|
|
|
|
if( OEBIT ) {
|
|
SET_ADD_OV(REG(RT), ra, rb);
|
|
}
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RT));
|
|
}
|
|
}
|
|
|
|
static void ppc_addcx(UINT32 op)
|
|
{
|
|
UINT32 ra = REG(RA);
|
|
UINT32 rb = REG(RB);
|
|
|
|
REG(RT) = ra + rb;
|
|
|
|
SET_ADD_CA(REG(RT), ra, rb);
|
|
|
|
if( OEBIT ) {
|
|
SET_ADD_OV(REG(RT), ra, rb);
|
|
}
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RT));
|
|
}
|
|
}
|
|
|
|
static void ppc_addex(UINT32 op)
|
|
{
|
|
UINT32 ra = REG(RA);
|
|
UINT32 rb = REG(RB);
|
|
UINT32 carry = (XER >> 29) & 0x1;
|
|
UINT32 tmp;
|
|
|
|
tmp = rb + carry;
|
|
REG(RT) = ra + tmp;
|
|
|
|
if( ADD_CA(tmp, rb, carry) || ADD_CA(REG(RT), ra, tmp) )
|
|
XER |= XER_CA;
|
|
else
|
|
XER &= ~XER_CA;
|
|
|
|
if( OEBIT ) {
|
|
SET_ADD_OV(REG(RT), ra, rb);
|
|
}
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RT));
|
|
}
|
|
}
|
|
|
|
static void ppc_addi(UINT32 op)
|
|
{
|
|
UINT32 i = SIMM16;
|
|
UINT32 a = RA;
|
|
|
|
if( a )
|
|
i += REG(a);
|
|
|
|
REG(RT) = i;
|
|
}
|
|
|
|
static void ppc_addic(UINT32 op)
|
|
{
|
|
UINT32 i = SIMM16;
|
|
UINT32 ra = REG(RA);
|
|
|
|
REG(RT) = ra + i;
|
|
|
|
if( ADD_CA(REG(RT), ra, i) )
|
|
XER |= XER_CA;
|
|
else
|
|
XER &= ~XER_CA;
|
|
}
|
|
|
|
static void ppc_addic_rc(UINT32 op)
|
|
{
|
|
UINT32 i = SIMM16;
|
|
UINT32 ra = REG(RA);
|
|
|
|
REG(RT) = ra + i;
|
|
|
|
if( ADD_CA(REG(RT), ra, i) )
|
|
XER |= XER_CA;
|
|
else
|
|
XER &= ~XER_CA;
|
|
|
|
SET_CR0(REG(RT));
|
|
}
|
|
|
|
static void ppc_addis(UINT32 op)
|
|
{
|
|
UINT32 i = UIMM16 << 16;
|
|
UINT32 a = RA;
|
|
|
|
if( a )
|
|
i += REG(a);
|
|
|
|
REG(RT) = i;
|
|
}
|
|
|
|
static void ppc_addmex(UINT32 op)
|
|
{
|
|
UINT32 ra = REG(RA);
|
|
UINT32 carry = (XER >> 29) & 0x1;
|
|
UINT32 tmp;
|
|
|
|
tmp = ra + carry;
|
|
REG(RT) = tmp + -1;
|
|
|
|
if( ADD_CA(tmp, ra, carry) || ADD_CA(REG(RT), tmp, -1) )
|
|
XER |= XER_CA;
|
|
else
|
|
XER &= ~XER_CA;
|
|
|
|
if( OEBIT ) {
|
|
SET_ADD_OV(REG(RT), ra, carry - 1);
|
|
}
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RT));
|
|
}
|
|
}
|
|
|
|
static void ppc_addzex(UINT32 op)
|
|
{
|
|
UINT32 ra = REG(RA);
|
|
UINT32 carry = (XER >> 29) & 0x1;
|
|
|
|
REG(RT) = ra + carry;
|
|
|
|
if( ADD_CA(REG(RT), ra, carry) )
|
|
XER |= XER_CA;
|
|
else
|
|
XER &= ~XER_CA;
|
|
|
|
if( OEBIT ) {
|
|
SET_ADD_OV(REG(RT), ra, carry);
|
|
}
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RT));
|
|
}
|
|
}
|
|
|
|
static void ppc_andx(UINT32 op)
|
|
{
|
|
REG(RA) = REG(RS) & REG(RB);
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_andcx(UINT32 op)
|
|
{
|
|
REG(RA) = REG(RS) & ~REG(RB);
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_andi_rc(UINT32 op)
|
|
{
|
|
UINT32 i = UIMM16;
|
|
|
|
REG(RA) = REG(RS) & i;
|
|
|
|
SET_CR0(REG(RA));
|
|
}
|
|
|
|
static void ppc_andis_rc(UINT32 op)
|
|
{
|
|
UINT32 i = UIMM16 << 16;
|
|
|
|
REG(RA) = REG(RS) & i;
|
|
|
|
SET_CR0(REG(RA));
|
|
}
|
|
|
|
static void ppc_bx(UINT32 op)
|
|
{
|
|
INT32 li = op & 0x3fffffc;
|
|
if( li & 0x2000000 )
|
|
li |= 0xfc000000;
|
|
|
|
ppc.npc = li;
|
|
|
|
if( !AABIT ) {
|
|
ppc.npc += ppc.pc;
|
|
}
|
|
|
|
if( LKBIT ) {
|
|
LR = ppc.pc + 4;
|
|
}
|
|
|
|
ppc_change_pc(ppc.npc);
|
|
}
|
|
|
|
static void ppc_bcx(UINT32 op)
|
|
{
|
|
int condition = check_condition_code(BO, BI);
|
|
|
|
if( condition ) {
|
|
ppc.npc = SIMM16 & ~0x3;
|
|
if( !AABIT )
|
|
ppc.npc += ppc.pc;
|
|
|
|
ppc_change_pc(ppc.npc);
|
|
}
|
|
|
|
if( LKBIT ) {
|
|
LR = ppc.pc + 4;
|
|
}
|
|
}
|
|
|
|
static void ppc_bcctrx(UINT32 op)
|
|
{
|
|
int condition = check_condition_code(BO, BI);
|
|
|
|
if( condition ) {
|
|
ppc.npc = CTR & ~0x3;
|
|
ppc_change_pc(ppc.npc);
|
|
}
|
|
|
|
if( LKBIT ) {
|
|
LR = ppc.pc + 4;
|
|
}
|
|
}
|
|
|
|
static void ppc_bclrx(UINT32 op)
|
|
{
|
|
int condition = check_condition_code(BO, BI);
|
|
|
|
if( condition ) {
|
|
ppc.npc = LR & ~0x3;
|
|
ppc_change_pc(ppc.npc);
|
|
}
|
|
|
|
if( LKBIT ) {
|
|
LR = ppc.pc + 4;
|
|
}
|
|
}
|
|
|
|
static void ppc_cmp(UINT32 op)
|
|
{
|
|
INT32 ra = REG(RA);
|
|
INT32 rb = REG(RB);
|
|
int d = CRFD;
|
|
|
|
if( ra < rb )
|
|
CR(d) = 0x8;
|
|
else if( ra > rb )
|
|
CR(d) = 0x4;
|
|
else
|
|
CR(d) = 0x2;
|
|
|
|
if( XER & XER_SO )
|
|
CR(d) |= 0x1;
|
|
}
|
|
|
|
static void ppc_cmpi(UINT32 op)
|
|
{
|
|
INT32 ra = REG(RA);
|
|
INT32 i = SIMM16;
|
|
int d = CRFD;
|
|
|
|
if( ra < i )
|
|
CR(d) = 0x8;
|
|
else if( ra > i )
|
|
CR(d) = 0x4;
|
|
else
|
|
CR(d) = 0x2;
|
|
|
|
if( XER & XER_SO )
|
|
CR(d) |= 0x1;
|
|
}
|
|
|
|
static void ppc_cmpl(UINT32 op)
|
|
{
|
|
UINT32 ra = REG(RA);
|
|
UINT32 rb = REG(RB);
|
|
int d = CRFD;
|
|
|
|
if( ra < rb )
|
|
CR(d) = 0x8;
|
|
else if( ra > rb )
|
|
CR(d) = 0x4;
|
|
else
|
|
CR(d) = 0x2;
|
|
|
|
if( XER & XER_SO )
|
|
CR(d) |= 0x1;
|
|
}
|
|
|
|
static void ppc_cmpli(UINT32 op)
|
|
{
|
|
UINT32 ra = REG(RA);
|
|
UINT32 i = UIMM16;
|
|
int d = CRFD;
|
|
|
|
if( ra < i )
|
|
CR(d) = 0x8;
|
|
else if( ra > i )
|
|
CR(d) = 0x4;
|
|
else
|
|
CR(d) = 0x2;
|
|
|
|
if( XER & XER_SO )
|
|
CR(d) |= 0x1;
|
|
}
|
|
|
|
static void ppc_cntlzw(UINT32 op)
|
|
{
|
|
int n = 0;
|
|
int t = RT;
|
|
UINT32 m = 0x80000000;
|
|
|
|
while(n < 32)
|
|
{
|
|
if( REG(t) & m )
|
|
break;
|
|
m >>= 1;
|
|
n++;
|
|
}
|
|
|
|
REG(RA) = n;
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_crand(UINT32 op)
|
|
{
|
|
int bit = RT;
|
|
int b = CRBIT(RA) & CRBIT(RB);
|
|
if( b )
|
|
CR(bit / 4) |= _BIT(3-(bit % 4));
|
|
else
|
|
CR(bit / 4) &= ~_BIT(3-(bit % 4));
|
|
}
|
|
|
|
static void ppc_crandc(UINT32 op)
|
|
{
|
|
int bit = RT;
|
|
int b = CRBIT(RA) & (CRBIT(RB) ^ 0x1);
|
|
if( b )
|
|
CR(bit / 4) |= _BIT(3-(bit % 4));
|
|
else
|
|
CR(bit / 4) &= ~_BIT(3-(bit % 4));
|
|
}
|
|
|
|
static void ppc_creqv(UINT32 op)
|
|
{
|
|
int bit = RT;
|
|
int b = (CRBIT(RA) ^ CRBIT(RB)) ^ 0x1;
|
|
if( b )
|
|
CR(bit / 4) |= _BIT(3-(bit % 4));
|
|
else
|
|
CR(bit / 4) &= ~_BIT(3-(bit % 4));
|
|
}
|
|
|
|
static void ppc_crnand(UINT32 op)
|
|
{
|
|
int bit = RT;
|
|
int b = (CRBIT(RA) & CRBIT(RB)) ^ 0x1;
|
|
if( b )
|
|
CR(bit / 4) |= _BIT(3-(bit % 4));
|
|
else
|
|
CR(bit / 4) &= ~_BIT(3-(bit % 4));
|
|
}
|
|
|
|
static void ppc_crnor(UINT32 op)
|
|
{
|
|
int bit = RT;
|
|
int b = (CRBIT(RA) | CRBIT(RB)) ^ 0x1;
|
|
if( b )
|
|
CR(bit / 4) |= _BIT(3-(bit % 4));
|
|
else
|
|
CR(bit / 4) &= ~_BIT(3-(bit % 4));
|
|
}
|
|
|
|
static void ppc_cror(UINT32 op)
|
|
{
|
|
int bit = RT;
|
|
int b = CRBIT(RA) | CRBIT(RB);
|
|
if( b )
|
|
CR(bit / 4) |= _BIT(3-(bit % 4));
|
|
else
|
|
CR(bit / 4) &= ~_BIT(3-(bit % 4));
|
|
}
|
|
|
|
static void ppc_crorc(UINT32 op)
|
|
{
|
|
int bit = RT;
|
|
int b = CRBIT(RA) | (CRBIT(RB) ^ 0x1);
|
|
if( b )
|
|
CR(bit / 4) |= _BIT(3-(bit % 4));
|
|
else
|
|
CR(bit / 4) &= ~_BIT(3-(bit % 4));
|
|
}
|
|
|
|
static void ppc_crxor(UINT32 op)
|
|
{
|
|
int bit = RT;
|
|
int b = CRBIT(RA) ^ CRBIT(RB);
|
|
if( b )
|
|
CR(bit / 4) |= _BIT(3-(bit % 4));
|
|
else
|
|
CR(bit / 4) &= ~_BIT(3-(bit % 4));
|
|
}
|
|
|
|
static void ppc_dcbf(UINT32 op)
|
|
{
|
|
|
|
}
|
|
|
|
static void ppc_dcbi(UINT32 op)
|
|
{
|
|
|
|
}
|
|
|
|
static void ppc_dcbst(UINT32 op)
|
|
{
|
|
|
|
}
|
|
|
|
static void ppc_dcbt(UINT32 op)
|
|
{
|
|
|
|
}
|
|
|
|
static void ppc_dcbtst(UINT32 op)
|
|
{
|
|
|
|
}
|
|
|
|
static void ppc_dcbz(UINT32 op)
|
|
{
|
|
|
|
}
|
|
|
|
static void ppc_divwx(UINT32 op)
|
|
{
|
|
if( REG(RB) == 0 && REG(RA) < 0x80000000 )
|
|
{
|
|
REG(RT) = 0;
|
|
if( OEBIT ) {
|
|
XER |= XER_SO | XER_OV;
|
|
}
|
|
}
|
|
else if( REG(RB) == 0 || (REG(RB) == 0xffffffff && REG(RA) == 0x80000000) )
|
|
{
|
|
REG(RT) = 0xffffffff;
|
|
if( OEBIT ) {
|
|
XER |= XER_SO | XER_OV;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
REG(RT) = (INT32)REG(RA) / (INT32)REG(RB);
|
|
if( OEBIT ) {
|
|
XER &= ~XER_OV;
|
|
}
|
|
}
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RT));
|
|
}
|
|
}
|
|
|
|
static void ppc_divwux(UINT32 op)
|
|
{
|
|
if( REG(RB) == 0 )
|
|
{
|
|
REG(RT) = 0;
|
|
if( OEBIT ) {
|
|
XER |= XER_SO | XER_OV;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
REG(RT) = (UINT32)REG(RA) / (UINT32)REG(RB);
|
|
if( OEBIT ) {
|
|
XER &= ~XER_OV;
|
|
}
|
|
}
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RT));
|
|
}
|
|
}
|
|
|
|
static void ppc_eieio(UINT32 op)
|
|
{
|
|
|
|
}
|
|
|
|
static void ppc_eqvx(UINT32 op)
|
|
{
|
|
REG(RA) = ~(REG(RS) ^ REG(RB));
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_extsbx(UINT32 op)
|
|
{
|
|
REG(RA) = (INT32)(INT8)REG(RS);
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_extshx(UINT32 op)
|
|
{
|
|
REG(RA) = (INT32)(INT16)REG(RS);
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_icbi(UINT32 op)
|
|
{
|
|
|
|
}
|
|
|
|
static void ppc_isync(UINT32 op)
|
|
{
|
|
|
|
}
|
|
|
|
static void ppc_lbz(UINT32 op)
|
|
{
|
|
UINT32 ea = SIMM16;
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
REG(RT) = (UINT32)READ8(ea);
|
|
}
|
|
|
|
static void ppc_lbzu(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RA) + SIMM16;
|
|
|
|
REG(RT) = (UINT32)READ8(ea);
|
|
REG(RA) = ea;
|
|
}
|
|
|
|
static void ppc_lbzux(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RA) + REG(RB);
|
|
|
|
REG(RT) = (UINT32)READ8(ea);
|
|
REG(RA) = ea;
|
|
}
|
|
|
|
static void ppc_lbzx(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
REG(RT) = (UINT32)READ8(ea);
|
|
}
|
|
|
|
static void ppc_lha(UINT32 op)
|
|
{
|
|
UINT32 ea = SIMM16;
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
REG(RT) = (INT32)(INT16)READ16(ea);
|
|
}
|
|
|
|
static void ppc_lhau(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RA) + SIMM16;
|
|
|
|
REG(RT) = (INT32)(INT16)READ16(ea);
|
|
REG(RA) = ea;
|
|
}
|
|
|
|
static void ppc_lhaux(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RA) + REG(RB);
|
|
|
|
REG(RT) = (INT32)(INT16)READ16(ea);
|
|
REG(RA) = ea;
|
|
}
|
|
|
|
static void ppc_lhax(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
REG(RT) = (INT32)(INT16)READ16(ea);
|
|
}
|
|
|
|
static void ppc_lhbrx(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
UINT16 w;
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
w = READ16(ea);
|
|
REG(RT) = (UINT32)BYTE_REVERSE16(w);
|
|
}
|
|
|
|
static void ppc_lhz(UINT32 op)
|
|
{
|
|
UINT32 ea = SIMM16;
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
REG(RT) = (UINT32)READ16(ea);
|
|
}
|
|
|
|
static void ppc_lhzu(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RA) + SIMM16;
|
|
|
|
REG(RT) = (UINT32)READ16(ea);
|
|
REG(RA) = ea;
|
|
}
|
|
|
|
static void ppc_lhzux(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RA) + REG(RB);
|
|
|
|
REG(RT) = (UINT32)READ16(ea);
|
|
REG(RA) = ea;
|
|
}
|
|
|
|
static void ppc_lhzx(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
REG(RT) = (UINT32)READ16(ea);
|
|
}
|
|
|
|
static void ppc_lmw(UINT32 op)
|
|
{
|
|
int r = RT;
|
|
UINT32 ea = SIMM16;
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
while( r <= 31 )
|
|
{
|
|
REG(r) = READ32(ea);
|
|
ea += 4;
|
|
r++;
|
|
}
|
|
}
|
|
|
|
static void ppc_lswi(UINT32 op)
|
|
{
|
|
int n, r, i;
|
|
UINT32 ea = (RA != 0) ? REG(RA) : 0;
|
|
|
|
n = (RB == 0) ? 32 : RB;
|
|
|
|
r = RT - 1;
|
|
i = 0;
|
|
|
|
while(n > 0)
|
|
{
|
|
if (i == 0) {
|
|
r = (r + 1) % 32;
|
|
REG(r) = 0;
|
|
}
|
|
REG(r) |= ((READ8(ea) & 0xff) << (24 - i));
|
|
i += 8;
|
|
if (i == 32) {
|
|
i = 0;
|
|
}
|
|
ea++;
|
|
n--;
|
|
}
|
|
}
|
|
|
|
static void ppc_lswx(UINT32 op)
|
|
{
|
|
int n, r, i;
|
|
UINT32 ea = REG(RB);
|
|
if(RA != 0)
|
|
ea += REG(RA);
|
|
|
|
n = ppc.xer & 0x7f;
|
|
|
|
r = RT - 1;
|
|
i = 0;
|
|
|
|
while(n > 0)
|
|
{
|
|
if (i == 0) {
|
|
r = (r + 1) % 32;
|
|
REG(r) = 0;
|
|
}
|
|
REG(r) |= ((READ8(ea) & 0xff) << (24 - i));
|
|
i += 8;
|
|
if (i == 32) {
|
|
i = 0;
|
|
}
|
|
ea++;
|
|
n--;
|
|
}
|
|
}
|
|
|
|
static void ppc_lwarx(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
ppc.reserved_address = ea;
|
|
ppc.reserved = 1;
|
|
|
|
REG(RT) = READ32(ea);
|
|
}
|
|
|
|
static void ppc_lwbrx(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
UINT32 w;
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
w = READ32(ea);
|
|
REG(RT) = BYTE_REVERSE32(w);
|
|
}
|
|
|
|
static void ppc_lwz(UINT32 op)
|
|
{
|
|
UINT32 ea = SIMM16;
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
REG(RT) = READ32(ea);
|
|
}
|
|
|
|
static void ppc_lwzu(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RA) + SIMM16;
|
|
|
|
REG(RT) = READ32(ea);
|
|
REG(RA) = ea;
|
|
}
|
|
|
|
static void ppc_lwzux(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RA) + REG(RB);
|
|
|
|
REG(RT) = READ32(ea);
|
|
REG(RA) = ea;
|
|
}
|
|
|
|
static void ppc_lwzx(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
REG(RT) = READ32(ea);
|
|
}
|
|
|
|
static void ppc_mcrf(UINT32 op)
|
|
{
|
|
CR(RT >> 2) = CR(RA >> 2);
|
|
}
|
|
|
|
static void ppc_mcrxr(UINT32 op)
|
|
{
|
|
CR(RT >> 2) = (XER >> 28) & 0x0F;
|
|
XER &= ~0xf0000000;
|
|
}
|
|
|
|
static void ppc_mfcr(UINT32 op)
|
|
{
|
|
REG(RT) = ppc_get_cr();
|
|
}
|
|
|
|
static void ppc_mfmsr(UINT32 op)
|
|
{
|
|
REG(RT) = ppc_get_msr();
|
|
}
|
|
|
|
static void ppc_mfspr(UINT32 op)
|
|
{
|
|
REG(RT) = ppc_get_spr(SPR);
|
|
}
|
|
|
|
static void ppc_mtcrf(UINT32 op)
|
|
{
|
|
int fxm = FXM;
|
|
int t = RT;
|
|
|
|
if( fxm & 0x80 ) CR(0) = (REG(t) >> 28) & 0xf;
|
|
if( fxm & 0x40 ) CR(1) = (REG(t) >> 24) & 0xf;
|
|
if( fxm & 0x20 ) CR(2) = (REG(t) >> 20) & 0xf;
|
|
if( fxm & 0x10 ) CR(3) = (REG(t) >> 16) & 0xf;
|
|
if( fxm & 0x08 ) CR(4) = (REG(t) >> 12) & 0xf;
|
|
if( fxm & 0x04 ) CR(5) = (REG(t) >> 8) & 0xf;
|
|
if( fxm & 0x02 ) CR(6) = (REG(t) >> 4) & 0xf;
|
|
if( fxm & 0x01 ) CR(7) = (REG(t) >> 0) & 0xf;
|
|
}
|
|
|
|
static void ppc_mtmsr(UINT32 op)
|
|
{
|
|
ppc_set_msr(REG(RS));
|
|
}
|
|
|
|
static void ppc_mtspr(UINT32 op)
|
|
{
|
|
ppc_set_spr(SPR, REG(RS));
|
|
}
|
|
|
|
static void ppc_mulhwx(UINT32 op)
|
|
{
|
|
INT64 ra = (INT64)(INT32)REG(RA);
|
|
INT64 rb = (INT64)(INT32)REG(RB);
|
|
|
|
REG(RT) = (UINT32)((ra * rb) >> 32);
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RT));
|
|
}
|
|
}
|
|
|
|
static void ppc_mulhwux(UINT32 op)
|
|
{
|
|
UINT64 ra = (UINT64)REG(RA);
|
|
UINT64 rb = (UINT64)REG(RB);
|
|
|
|
REG(RT) = (UINT32)((ra * rb) >> 32);
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RT));
|
|
}
|
|
}
|
|
|
|
static void ppc_mulli(UINT32 op)
|
|
{
|
|
INT32 ra = (INT32)REG(RA);
|
|
INT32 i = SIMM16;
|
|
|
|
REG(RT) = ra * i;
|
|
}
|
|
|
|
static void ppc_mullwx(UINT32 op)
|
|
{
|
|
INT64 ra = (INT64)(INT32)REG(RA);
|
|
INT64 rb = (INT64)(INT32)REG(RB);
|
|
INT64 r;
|
|
|
|
r = ra * rb;
|
|
REG(RT) = (UINT32)r;
|
|
|
|
if( OEBIT ) {
|
|
XER &= ~XER_OV;
|
|
|
|
if( r != (INT64)(INT32)r )
|
|
XER |= XER_OV | XER_SO;
|
|
}
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RT));
|
|
}
|
|
}
|
|
|
|
static void ppc_nandx(UINT32 op)
|
|
{
|
|
REG(RA) = ~(REG(RS) & REG(RB));
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_negx(UINT32 op)
|
|
{
|
|
REG(RT) = -(INT32)(REG(RA));
|
|
|
|
if( OEBIT ) {
|
|
if( REG(RT) == 0x80000000 )
|
|
XER |= XER_OV | XER_SO;
|
|
else
|
|
XER &= ~XER_OV;
|
|
}
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RT));
|
|
}
|
|
}
|
|
|
|
static void ppc_norx(UINT32 op)
|
|
{
|
|
REG(RA) = ~(REG(RS) | REG(RB));
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_orx(UINT32 op)
|
|
{
|
|
REG(RA) = REG(RS) | REG(RB);
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_orcx(UINT32 op)
|
|
{
|
|
REG(RA) = REG(RS) | ~REG(RB);
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_ori(UINT32 op)
|
|
{
|
|
REG(RA) = REG(RS) | UIMM16;
|
|
}
|
|
|
|
static void ppc_oris(UINT32 op)
|
|
{
|
|
REG(RA) = REG(RS) | (UIMM16 << 16);
|
|
}
|
|
|
|
static void ppc_rfi(UINT32 op)
|
|
{
|
|
UINT32 msr;
|
|
ppc.npc = ppc_get_spr(SPR_SRR0);
|
|
msr = ppc_get_spr(SPR_SRR1);
|
|
ppc_set_msr( msr );
|
|
|
|
ppc_change_pc(ppc.npc);
|
|
}
|
|
|
|
static void ppc_rlwimix(UINT32 op)
|
|
{
|
|
UINT32 r;
|
|
UINT32 mask = GET_ROTATE_MASK(MB, ME);
|
|
UINT32 rs = REG(RS);
|
|
int sh = SH;
|
|
|
|
r = (rs << sh) | (rs >> (32-sh));
|
|
REG(RA) = (REG(RA) & ~mask) | (r & mask);
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_rlwinmx(UINT32 op)
|
|
{
|
|
UINT32 r;
|
|
UINT32 mask = GET_ROTATE_MASK(MB, ME);
|
|
UINT32 rs = REG(RS);
|
|
int sh = SH;
|
|
|
|
r = (rs << sh) | (rs >> (32-sh));
|
|
REG(RA) = r & mask;
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_rlwnmx(UINT32 op)
|
|
{
|
|
UINT32 r;
|
|
UINT32 mask = GET_ROTATE_MASK(MB, ME);
|
|
UINT32 rs = REG(RS);
|
|
int sh = REG(RB) & 0x1f;
|
|
|
|
r = (rs << sh) | (rs >> (32-sh));
|
|
REG(RA) = r & mask;
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_sc(UINT32 op)
|
|
{
|
|
ppc603_exception(EXCEPTION_SYSTEM_CALL);
|
|
}
|
|
|
|
static void ppc_slwx(UINT32 op)
|
|
{
|
|
int sh = REG(RB) & 0x3f;
|
|
|
|
if( sh > 31 ) {
|
|
REG(RA) = 0;
|
|
}
|
|
else {
|
|
REG(RA) = REG(RS) << sh;
|
|
}
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_srawx(UINT32 op)
|
|
{
|
|
int sh = REG(RB) & 0x3f;
|
|
|
|
XER &= ~XER_CA;
|
|
|
|
if( sh > 31 ) {
|
|
if (REG(RS) & 0x80000000)
|
|
REG(RA) = 0xffffffff;
|
|
else
|
|
REG(RA) = 0;
|
|
if( REG(RA) )
|
|
XER |= XER_CA;
|
|
}
|
|
else {
|
|
REG(RA) = (INT32)(REG(RS)) >> sh;
|
|
if( ((INT32)(REG(RS)) < 0) && (REG(RS) & BITMASK_0(sh)) )
|
|
XER |= XER_CA;
|
|
}
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_srawix(UINT32 op)
|
|
{
|
|
int sh = SH;
|
|
|
|
XER &= ~XER_CA;
|
|
if( ((INT32)(REG(RS)) < 0) && (REG(RS) & BITMASK_0(sh)) )
|
|
XER |= XER_CA;
|
|
|
|
REG(RA) = (INT32)(REG(RS)) >> sh;
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_srwx(UINT32 op)
|
|
{
|
|
int sh = REG(RB) & 0x3f;
|
|
|
|
if( sh > 31 ) {
|
|
REG(RA) = 0;
|
|
}
|
|
else {
|
|
REG(RA) = REG(RS) >> sh;
|
|
}
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_stb(UINT32 op)
|
|
{
|
|
UINT32 ea = SIMM16;
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
WRITE8(ea, (UINT8)REG(RS));
|
|
}
|
|
|
|
static void ppc_stbu(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RA) + SIMM16;
|
|
|
|
WRITE8(ea, (UINT8)REG(RS));
|
|
REG(RA) = ea;
|
|
}
|
|
|
|
static void ppc_stbux(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RA) + REG(RB);
|
|
|
|
WRITE8(ea, (UINT8)REG(RS));
|
|
REG(RA) = ea;
|
|
}
|
|
|
|
static void ppc_stbx(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
WRITE8(ea, (UINT8)REG(RS));
|
|
}
|
|
|
|
static void ppc_sth(UINT32 op)
|
|
{
|
|
UINT32 ea = SIMM16;
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
WRITE16(ea, (UINT16)REG(RS));
|
|
}
|
|
|
|
static void ppc_sthbrx(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
UINT16 w;
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
w = REG(RS);
|
|
WRITE16(ea, (UINT16)BYTE_REVERSE16(w));
|
|
}
|
|
|
|
static void ppc_sthu(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RA) + SIMM16;
|
|
|
|
WRITE16(ea, (UINT16)REG(RS));
|
|
REG(RA) = ea;
|
|
}
|
|
|
|
static void ppc_sthux(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RA) + REG(RB);
|
|
|
|
WRITE16(ea, (UINT16)REG(RS));
|
|
REG(RA) = ea;
|
|
}
|
|
|
|
static void ppc_sthx(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
WRITE16(ea, (UINT16)REG(RS));
|
|
}
|
|
|
|
static void ppc_stmw(UINT32 op)
|
|
{
|
|
UINT32 ea = SIMM16;
|
|
int r = RS;
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
while( r <= 31 )
|
|
{
|
|
WRITE32(ea, REG(r));
|
|
ea += 4;
|
|
r++;
|
|
}
|
|
}
|
|
|
|
static void ppc_stswi(UINT32 op)
|
|
{
|
|
int n, r, i;
|
|
UINT32 ea = (RA != 0) ? REG(RA) : 0;
|
|
|
|
n = (RB == 0) ? 32 : RB;
|
|
|
|
r = RT - 1;
|
|
i = 0;
|
|
|
|
while(n > 0)
|
|
{
|
|
if (i == 0) {
|
|
r = (r + 1) % 32;
|
|
}
|
|
WRITE8(ea, (REG(r) >> (24-i)) & 0xff);
|
|
i += 8;
|
|
if (i == 32) {
|
|
i = 0;
|
|
}
|
|
ea++;
|
|
n--;
|
|
}
|
|
}
|
|
|
|
static void ppc_stswx(UINT32 op)
|
|
{
|
|
int n, r, i;
|
|
UINT32 ea = REG(RB);
|
|
if (RA != 0)
|
|
ea += REG(RA);
|
|
|
|
n = ppc.xer & 0x7f;
|
|
|
|
r = RT - 1;
|
|
i = 0;
|
|
|
|
while(n > 0)
|
|
{
|
|
if (i == 0) {
|
|
r = (r + 1) % 32;
|
|
}
|
|
WRITE8(ea, (REG(r) >> (24-i)) & 0xff);
|
|
i += 8;
|
|
if (i == 32) {
|
|
i = 0;
|
|
}
|
|
ea++;
|
|
n--;
|
|
}
|
|
}
|
|
|
|
static void ppc_stw(UINT32 op)
|
|
{
|
|
UINT32 ea = SIMM16;
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
WRITE32(ea, REG(RS));
|
|
}
|
|
|
|
static void ppc_stwbrx(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
UINT32 w;
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
w = REG(RS);
|
|
WRITE32(ea, BYTE_REVERSE32(w));
|
|
}
|
|
|
|
static void ppc_stwcx_rc(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
if( ppc.reserved ) {
|
|
WRITE32(ea, REG(RS));
|
|
|
|
ppc.reserved = 0;
|
|
ppc.reserved_address = 0;
|
|
|
|
CR(0) = 0x2; // set EQ to indicate success
|
|
} else {
|
|
CR(0) = 0;
|
|
}
|
|
|
|
if (XER & XER_SO)
|
|
CR(0) |= 0x1;
|
|
}
|
|
|
|
static void ppc_stwu(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RA) + SIMM16;
|
|
|
|
WRITE32(ea, REG(RS));
|
|
REG(RA) = ea;
|
|
}
|
|
|
|
static void ppc_stwux(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RA) + REG(RB);
|
|
|
|
WRITE32(ea, REG(RS));
|
|
REG(RA) = ea;
|
|
}
|
|
|
|
static void ppc_stwx(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
|
|
if( RA != 0 )
|
|
ea += REG(RA);
|
|
|
|
WRITE32(ea, REG(RS));
|
|
}
|
|
|
|
static void ppc_subfx(UINT32 op)
|
|
{
|
|
UINT32 ra = REG(RA);
|
|
UINT32 rb = REG(RB);
|
|
REG(RT) = rb - ra;
|
|
|
|
if( OEBIT ) {
|
|
SET_SUB_OV(REG(RT), rb, ra);
|
|
}
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RT));
|
|
}
|
|
}
|
|
|
|
static void ppc_subfcx(UINT32 op)
|
|
{
|
|
UINT32 ra = REG(RA);
|
|
UINT32 rb = REG(RB);
|
|
REG(RT) = rb - ra;
|
|
|
|
SET_SUB_CA(REG(RT), rb, ra);
|
|
|
|
if( OEBIT ) {
|
|
SET_SUB_OV(REG(RT), rb, ra);
|
|
}
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RT));
|
|
}
|
|
}
|
|
|
|
static void ppc_subfex(UINT32 op)
|
|
{
|
|
UINT32 ra = REG(RA);
|
|
UINT32 rb = REG(RB);
|
|
UINT32 carry = (XER >> 29) & 0x1;
|
|
UINT32 r;
|
|
|
|
r = ~ra + carry;
|
|
REG(RT) = rb + r;
|
|
|
|
SET_ADD_CA(r, ~ra, carry); /* step 1 carry */
|
|
if( REG(RT) < r ) /* step 2 carry */
|
|
XER |= XER_CA;
|
|
|
|
if( OEBIT ) {
|
|
SET_SUB_OV(REG(RT), rb, ra);
|
|
}
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RT));
|
|
}
|
|
}
|
|
|
|
static void ppc_subfic(UINT32 op)
|
|
{
|
|
UINT32 i = SIMM16;
|
|
UINT32 ra = REG(RA);
|
|
|
|
REG(RT) = i - ra;
|
|
|
|
SET_SUB_CA(REG(RT), i, ra);
|
|
}
|
|
|
|
static void ppc_subfmex(UINT32 op)
|
|
{
|
|
UINT32 ra = REG(RA);
|
|
UINT32 carry = (XER >> 29) & 0x1;
|
|
UINT32 r;
|
|
|
|
r = ~ra + carry;
|
|
REG(RT) = r - 1;
|
|
|
|
SET_SUB_CA(r, ~ra, carry); /* step 1 carry */
|
|
if( REG(RT) < r )
|
|
XER |= XER_CA; /* step 2 carry */
|
|
|
|
if( OEBIT ) {
|
|
SET_SUB_OV(REG(RT), -1, ra);
|
|
}
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RT));
|
|
}
|
|
}
|
|
|
|
static void ppc_subfzex(UINT32 op)
|
|
{
|
|
UINT32 ra = REG(RA);
|
|
UINT32 carry = (XER >> 29) & 0x1;
|
|
|
|
REG(RT) = ~ra + carry;
|
|
|
|
SET_ADD_CA(REG(RT), ~ra, carry);
|
|
|
|
if( OEBIT ) {
|
|
SET_SUB_OV(REG(RT), 0, REG(RA));
|
|
}
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RT));
|
|
}
|
|
}
|
|
|
|
static void ppc_sync(UINT32 op)
|
|
{
|
|
|
|
}
|
|
|
|
static void ppc_tw(UINT32 op)
|
|
{
|
|
int exception = 0;
|
|
INT32 a = REG(RA);
|
|
INT32 b = REG(RB);
|
|
int to = RT;
|
|
|
|
if(( (a < b) && (to & 0x10) ) ||
|
|
( (a > b) && (to & 0x08) ) ||
|
|
( (a == b) && (to & 0x04) ) ||
|
|
( ((UINT32)a < (UINT32)b) && (to & 0x02) ) ||
|
|
( ((UINT32)a > (UINT32)b) && (to & 0x01) )) {
|
|
exception = 1;
|
|
}
|
|
|
|
if (exception) {
|
|
ppc603_exception(EXCEPTION_TRAP);
|
|
}
|
|
}
|
|
|
|
static void ppc_twi(UINT32 op)
|
|
{
|
|
int exception = 0;
|
|
INT32 a = REG(RA);
|
|
INT32 i = SIMM16;
|
|
int to = RT;
|
|
|
|
if(( (a < i) && (to & 0x10) ) ||
|
|
( (a > i) && (to & 0x08) ) ||
|
|
( (a == i) && (to & 0x04) ) ||
|
|
( ((UINT32)a < (UINT32)i) && (to & 0x02) ) ||
|
|
( ((UINT32)a > (UINT32)i) && (to & 0x01) )) {
|
|
exception = 1;
|
|
}
|
|
|
|
if (exception) {
|
|
ppc603_exception(EXCEPTION_TRAP);
|
|
}
|
|
}
|
|
|
|
static void ppc_xorx(UINT32 op)
|
|
{
|
|
REG(RA) = REG(RS) ^ REG(RB);
|
|
|
|
if( RCBIT ) {
|
|
SET_CR0(REG(RA));
|
|
}
|
|
}
|
|
|
|
static void ppc_xori(UINT32 op)
|
|
{
|
|
REG(RA) = REG(RS) ^ UIMM16;
|
|
}
|
|
|
|
static void ppc_xoris(UINT32 op)
|
|
{
|
|
REG(RA) = REG(RS) ^ (UIMM16 << 16);
|
|
}
|
|
|
|
|
|
|
|
static void ppc_invalid(UINT32 op)
|
|
{
|
|
ErrorLog("PowerPC hit an invalid instruction. Halting emulation until reset.");
|
|
DebugLog("ppc: Invalid opcode %08X PC : %X, %08X\n", op, ppc.pc, ppc.npc);
|
|
ppc.fatalError = true;
|
|
}
|
|
|
|
|
|
|
|
#define DOUBLE_SIGN (0x8000000000000000ULL)
|
|
#define DOUBLE_EXP (0x7ff0000000000000ULL)
|
|
#define DOUBLE_FRAC (0x000fffffffffffffULL)
|
|
#define DOUBLE_ZERO (0ULL)
|
|
|
|
/*
|
|
Floating point operations.
|
|
*/
|
|
|
|
/*************************OLD
|
|
inline int is_nan_double(FPR x)
|
|
{
|
|
return( ((x.id & DOUBLE_EXP) == DOUBLE_EXP) &&
|
|
((x.id & DOUBLE_FRAC) != DOUBLE_ZERO) );
|
|
}
|
|
|
|
inline int is_qnan_double(FPR x)
|
|
{
|
|
return( ((x.id & DOUBLE_EXP) == DOUBLE_EXP) &&
|
|
((x.id & 0x0007fffffffffff) == 0x000000000000000) &&
|
|
((x.id & 0x000800000000000) == 0x000800000000000) );
|
|
}
|
|
|
|
inline int is_snan_double(FPR x)
|
|
{
|
|
return( ((x.id & DOUBLE_EXP) == DOUBLE_EXP) &&
|
|
((x.id & DOUBLE_FRAC) != DOUBLE_ZERO) &&
|
|
((x.id & 0x0008000000000000) == DOUBLE_ZERO) );
|
|
}
|
|
|
|
inline int is_infinity_double(FPR x)
|
|
{
|
|
return( ((x.id & DOUBLE_EXP) == DOUBLE_EXP) &&
|
|
((x.id & DOUBLE_FRAC) == DOUBLE_ZERO) );
|
|
}
|
|
|
|
inline int is_normalized_double(FPR x)
|
|
{
|
|
UINT64 exp;
|
|
|
|
exp = (x.id & DOUBLE_EXP) >> 52;
|
|
|
|
return (exp >= 1) && (exp <= 2046);
|
|
}
|
|
|
|
inline int is_denormalized_double(FPR x)
|
|
{
|
|
return( ((x.id & DOUBLE_EXP) == 0) &&
|
|
((x.id & DOUBLE_FRAC) != DOUBLE_ZERO) );
|
|
}
|
|
|
|
inline int sign_double(FPR x)
|
|
{
|
|
return ((x.id & DOUBLE_SIGN) != 0);
|
|
}
|
|
|
|
inline INT64 round_to_nearest(FPR f)
|
|
{
|
|
//return (INT64)(f.fd + 0.5);
|
|
if (f.fd >= 0)
|
|
{
|
|
return (INT64)(f.fd + 0.5);
|
|
}
|
|
else
|
|
{
|
|
return -(INT64)(-f.fd + 0.5);
|
|
}
|
|
}
|
|
|
|
inline INT64 round_toward_zero(FPR f)
|
|
{
|
|
return (INT64)(f.fd);
|
|
}
|
|
|
|
inline INT64 round_toward_positive_infinity(FPR f)
|
|
{
|
|
double r = ceil(f.fd);
|
|
return (INT64)(r);
|
|
}
|
|
|
|
inline INT64 round_toward_negative_infinity(FPR f)
|
|
{
|
|
double r = floor(f.fd);
|
|
return (INT64)(r);
|
|
}
|
|
*/
|
|
|
|
|
|
// New below, based on changes in MAME
|
|
inline int is_nan_double(FPR x)
|
|
{
|
|
return( ((x.id & DOUBLE_EXP) == DOUBLE_EXP) &&
|
|
((x.id & DOUBLE_FRAC) != DOUBLE_ZERO) );
|
|
}
|
|
|
|
inline int is_qnan_double(FPR x)
|
|
{
|
|
return( ((x.id & DOUBLE_EXP) == DOUBLE_EXP) &&
|
|
((x.id & 0x0007fffffffffffULL) == 0x000000000000000ULL) &&
|
|
((x.id & 0x000800000000000ULL) == 0x000800000000000ULL) );
|
|
}
|
|
|
|
inline int is_snan_double(FPR x)
|
|
{
|
|
return( ((x.id & DOUBLE_EXP) == DOUBLE_EXP) &&
|
|
((x.id & DOUBLE_FRAC) != DOUBLE_ZERO) &&
|
|
((x.id & (0x0008000000000000ULL)) == DOUBLE_ZERO) );
|
|
}
|
|
|
|
inline int is_infinity_double(FPR x)
|
|
{
|
|
return( ((x.id & DOUBLE_EXP) == DOUBLE_EXP) &&
|
|
((x.id & DOUBLE_FRAC) == DOUBLE_ZERO) );
|
|
}
|
|
|
|
inline int is_normalized_double(FPR x)
|
|
{
|
|
UINT64 exp;
|
|
|
|
exp = (x.id & DOUBLE_EXP) >> 52;
|
|
|
|
return (exp >= 1) && (exp <= 2046);
|
|
}
|
|
|
|
inline int is_denormalized_double(FPR x)
|
|
{
|
|
return( ((x.id & DOUBLE_EXP) == 0) &&
|
|
((x.id & DOUBLE_FRAC) != DOUBLE_ZERO) );
|
|
}
|
|
|
|
inline int sign_double(FPR x)
|
|
{
|
|
return ((x.id & DOUBLE_SIGN) != 0);
|
|
}
|
|
|
|
inline void set_rounding_mode(void)
|
|
{
|
|
// may require compiler option to work correctly (-frounding-math for GCC, /fp:strict for Visual Studio)
|
|
// unknown if any games actually change this
|
|
switch (ppc.fpscr & 3)
|
|
{
|
|
case 0: fesetround(FE_TONEAREST); break;
|
|
case 1: fesetround(FE_TOWARDZERO); break;
|
|
case 2: fesetround(FE_UPWARD); break;
|
|
case 3: fesetround(FE_DOWNWARD); break;
|
|
}
|
|
}
|
|
|
|
/*
|
|
inline INT64 smround_to_nearest(FPR f)
|
|
{
|
|
// This method is incorrect; it ties away from zero, while PowerPC ties to even
|
|
if (f.fd >= 0)
|
|
{
|
|
return (INT64)(f.fd + 0.5);
|
|
}
|
|
else
|
|
{
|
|
return -(INT64)(-f.fd + 0.5);
|
|
}
|
|
}
|
|
|
|
inline INT64 smround_toward_zero(FPR f)
|
|
{
|
|
return (INT64)(f.fd);
|
|
}
|
|
|
|
inline INT64 round_toward_positive_infinity(FPR f)
|
|
{
|
|
double r = ceil(f.fd);
|
|
return (INT64)(r);
|
|
}
|
|
|
|
inline INT64 round_toward_negative_infinity(FPR f)
|
|
{
|
|
double r = floor(f.fd);
|
|
return (INT64)(r);
|
|
}
|
|
*/
|
|
|
|
#define SET_VXSNAN(a, b) if (is_snan_double(a) || is_snan_double(b)) ppc.fpscr |= 0x80000000
|
|
#define SET_VXSNAN_1(c) if (is_snan_double(c)) ppc.fpscr |= 0x80000000
|
|
|
|
inline void set_fprf(FPR f)
|
|
{
|
|
UINT32 fprf;
|
|
|
|
// see page 3-30, 3-31
|
|
|
|
if (is_qnan_double(f))
|
|
{
|
|
fprf = 0x11;
|
|
}
|
|
else if (is_infinity_double(f))
|
|
{
|
|
if (sign_double(f)) // -INF
|
|
fprf = 0x09;
|
|
else // +INF
|
|
fprf = 0x05;
|
|
}
|
|
else if (is_normalized_double(f))
|
|
{
|
|
if (sign_double(f)) // -Normalized
|
|
fprf = 0x08;
|
|
else // +Normalized
|
|
fprf = 0x04;
|
|
}
|
|
else if (is_denormalized_double(f))
|
|
{
|
|
if (sign_double(f)) // -Denormalized
|
|
fprf = 0x18;
|
|
else // +Denormalized
|
|
fprf = 0x14;
|
|
}
|
|
else // Zero
|
|
{
|
|
if (sign_double(f)) // -Zero
|
|
fprf = 0x12;
|
|
else // +Zero
|
|
fprf = 0x02;
|
|
}
|
|
|
|
ppc.fpscr &= ~0x0001f000;
|
|
ppc.fpscr |= (fprf << 12);
|
|
}
|
|
|
|
|
|
|
|
|
|
static void ppc_lfs(UINT32 op)
|
|
{
|
|
UINT32 ea = SIMM16;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
FPR32 f;
|
|
|
|
if(a)
|
|
ea += REG(a);
|
|
|
|
f.i = READ32(ea);
|
|
FPR(t).fd = (double)(f.f);
|
|
}
|
|
|
|
static void ppc_lfsu(UINT32 op)
|
|
{
|
|
UINT32 ea = SIMM16;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
FPR32 f;
|
|
|
|
ea += REG(a);
|
|
|
|
f.i = READ32(ea);
|
|
FPR(t).fd = (double)(f.f);
|
|
|
|
REG(a) = ea;
|
|
}
|
|
|
|
static void ppc_lfd(UINT32 op)
|
|
{
|
|
UINT32 ea = SIMM16;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
if(a)
|
|
ea += REG(a);
|
|
|
|
FPR(t).id = READ64(ea);
|
|
}
|
|
|
|
static void ppc_lfdu(UINT32 op)
|
|
{
|
|
UINT32 ea = SIMM16;
|
|
UINT32 a = RA;
|
|
UINT32 d = RD;
|
|
|
|
ea += REG(a);
|
|
|
|
FPR(d).id = READ64(ea);
|
|
|
|
REG(a) = ea;
|
|
}
|
|
|
|
static void ppc_stfs(UINT32 op)
|
|
{
|
|
UINT32 ea = SIMM16;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
FPR32 f;
|
|
|
|
if(a)
|
|
ea += REG(a);
|
|
|
|
f.f = (float)(FPR(t).fd);
|
|
WRITE32(ea, f.i);
|
|
}
|
|
|
|
static void ppc_stfsu(UINT32 op)
|
|
{
|
|
UINT32 ea = SIMM16;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
FPR32 f;
|
|
|
|
ea += REG(a);
|
|
|
|
f.f = (float)(FPR(t).fd);
|
|
WRITE32(ea, f.i);
|
|
|
|
REG(a) = ea;
|
|
}
|
|
|
|
static void ppc_stfd(UINT32 op)
|
|
{
|
|
UINT32 ea = SIMM16;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
if(a)
|
|
ea += REG(a);
|
|
|
|
WRITE64(ea, FPR(t).id);
|
|
}
|
|
|
|
static void ppc_stfdu(UINT32 op)
|
|
{
|
|
UINT32 ea = SIMM16;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
ea += REG(a);
|
|
|
|
WRITE64(ea, FPR(t).id);
|
|
|
|
REG(a) = ea;
|
|
}
|
|
|
|
static void ppc_lfdux(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
UINT32 a = RA;
|
|
UINT32 d = RD;
|
|
|
|
ea += REG(a);
|
|
|
|
FPR(d).id = READ64(ea);
|
|
|
|
REG(a) = ea;
|
|
}
|
|
|
|
static void ppc_lfdx(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
UINT32 a = RA;
|
|
UINT32 d = RD;
|
|
|
|
if(a)
|
|
ea += REG(a);
|
|
|
|
FPR(d).id = READ64(ea);
|
|
}
|
|
|
|
static void ppc_lfsux(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
FPR32 f;
|
|
|
|
ea += REG(a);
|
|
|
|
f.i = READ32(ea);
|
|
FPR(t).fd = (double)(f.f);
|
|
|
|
REG(a) = ea;
|
|
}
|
|
|
|
static void ppc_lfsx(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
FPR32 f;
|
|
|
|
if(a)
|
|
ea += REG(a);
|
|
|
|
f.i = READ32(ea);
|
|
FPR(t).fd = (double)(f.f);
|
|
}
|
|
|
|
static void ppc_mfsr(UINT32 op)
|
|
{
|
|
UINT32 sr = (op >> 16) & 15;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_SUPERVISOR();
|
|
|
|
REG(t) = ppc.sr[sr];
|
|
}
|
|
|
|
static void ppc_mfsrin(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_SUPERVISOR();
|
|
|
|
REG(t) = ppc.sr[REG(b) >> 28];
|
|
}
|
|
|
|
static void ppc_mftb(UINT32 op)
|
|
{
|
|
UINT32 x = SPRF;
|
|
|
|
switch(x)
|
|
{
|
|
case 268: REG(RT) = (UINT32)(ppc_read_timebase()); break;
|
|
case 269: REG(RT) = (UINT32)(ppc_read_timebase() >> 32); break;
|
|
default:
|
|
ErrorLog("PowerPC read from an invalid register. Halting emulation until reset.");
|
|
DebugLog("ppc: Invalid timebase register %d at %08X\n", x, ppc.pc);
|
|
ppc.fatalError = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void ppc_mtsr(UINT32 op)
|
|
{
|
|
UINT32 sr = (op >> 16) & 15;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_SUPERVISOR();
|
|
|
|
ppc.sr[sr] = REG(t);
|
|
}
|
|
|
|
static void ppc_mtsrin(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_SUPERVISOR();
|
|
|
|
ppc.sr[REG(b) >> 28] = REG(t);
|
|
}
|
|
|
|
static void ppc_dcba(UINT32 op)
|
|
{
|
|
/* TODO: Cache not emulated so this opcode doesn't need to be implemented */
|
|
}
|
|
|
|
static void ppc_stfdux(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
ea += REG(a);
|
|
|
|
WRITE64(ea, FPR(t).id);
|
|
|
|
REG(a) = ea;
|
|
}
|
|
|
|
static void ppc_stfdx(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
if(a)
|
|
ea += REG(a);
|
|
|
|
WRITE64(ea, FPR(t).id);
|
|
}
|
|
|
|
static void ppc_stfiwx(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
if(a)
|
|
ea += REG(a);
|
|
|
|
WRITE32(ea, (UINT32)FPR(t).id);
|
|
}
|
|
|
|
static void ppc_stfsux(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
FPR32 f;
|
|
|
|
ea += REG(a);
|
|
|
|
f.f = (float)(FPR(t).fd);
|
|
WRITE32(ea, f.i);
|
|
|
|
REG(a) = ea;
|
|
}
|
|
|
|
static void ppc_stfsx(UINT32 op)
|
|
{
|
|
UINT32 ea = REG(RB);
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
FPR32 f;
|
|
|
|
if(a)
|
|
ea += REG(a);
|
|
|
|
f.f = (float)(FPR(t).fd);
|
|
|
|
WRITE32(ea, f.i);
|
|
}
|
|
|
|
static void ppc_tlbia(UINT32 op)
|
|
{
|
|
/* TODO: TLB not emulated so this opcode doesn't need to implemented */
|
|
}
|
|
|
|
static void ppc_tlbie(UINT32 op)
|
|
{
|
|
/* TODO: TLB not emulated so this opcode doesn't need to implemented */
|
|
}
|
|
|
|
static void ppc_tlbsync(UINT32 op)
|
|
{
|
|
/* TODO: TLB not emulated so this opcode doesn't need to implemented */
|
|
}
|
|
|
|
static void ppc_eciwx(UINT32 op)
|
|
{
|
|
ppc_unimplemented(op);
|
|
}
|
|
|
|
static void ppc_ecowx(UINT32 op)
|
|
{
|
|
ppc_unimplemented(op);
|
|
}
|
|
|
|
static void ppc_fabsx(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
FPR(t).id = FPR(b).id & ~DOUBLE_SIGN;
|
|
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_faddx(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(b));
|
|
|
|
FPR(t).fd = FPR(a).fd + FPR(b).fd;
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fcmpo(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = (RT >> 2);
|
|
UINT32 c;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(b));
|
|
|
|
if(is_nan_double(FPR(a)) || is_nan_double(FPR(b)))
|
|
{
|
|
c = 1; /* OX */
|
|
if(is_snan_double(FPR(a)) || is_snan_double(FPR(b))) {
|
|
ppc.fpscr |= 0x01000000; /* VXSNAN */
|
|
|
|
if(!(ppc.fpscr & 0x40000000) || is_qnan_double(FPR(a)) || is_qnan_double(FPR(b)))
|
|
ppc.fpscr |= 0x00080000; /* VXVC */
|
|
}
|
|
}
|
|
else if(FPR(a).fd < FPR(b).fd){
|
|
c = 8; /* FX */
|
|
}
|
|
else if(FPR(a).fd > FPR(b).fd){
|
|
c = 4; /* FEX */
|
|
}
|
|
else {
|
|
c = 2; /* VX */
|
|
}
|
|
|
|
CR(t) = c;
|
|
|
|
// TODO
|
|
// Enabled by Bart
|
|
ppc.fpscr &= ~0x0001F000;
|
|
ppc.fpscr |= (c << 12);
|
|
}
|
|
|
|
static void ppc_fcmpu(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = (RT >> 2);
|
|
UINT32 c;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(b));
|
|
|
|
if(is_nan_double(FPR(a)) || is_nan_double(FPR(b)))
|
|
{
|
|
c = 1; /* OX */
|
|
if(is_snan_double(FPR(a)) || is_snan_double(FPR(b))) {
|
|
ppc.fpscr |= 0x01000000; /* VXSNAN */
|
|
}
|
|
}
|
|
else if(FPR(a).fd < FPR(b).fd){
|
|
c = 8; /* FX */
|
|
}
|
|
else if(FPR(a).fd > FPR(b).fd){
|
|
c = 4; /* FEX */
|
|
}
|
|
else {
|
|
c = 2; /* VX */
|
|
}
|
|
|
|
CR(t) = c;
|
|
|
|
// TODO
|
|
ppc.fpscr &= ~0x0001F000;
|
|
ppc.fpscr |= (c << 12);
|
|
}
|
|
|
|
static void ppc_fctiwx(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 t = RT;
|
|
INT64 r = 0;
|
|
|
|
// TODO: fix FPSCR flags FX,VXSNAN,VXCVI
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN_1(FPR(b));
|
|
|
|
switch(ppc.fpscr & 3)
|
|
{
|
|
// nearbyint() uses rounding mode set by fesetround()
|
|
// this should be FE_TONEAREST (ties to even) if the case is 0
|
|
case 0: r = (INT64)nearbyint(FPR(b).fd); break;
|
|
case 1: r = (INT64)trunc(FPR(b).fd); break;
|
|
case 2: r = (INT64)ceil(FPR(b).fd); break;
|
|
case 3: r = (INT64)floor(FPR(b).fd); break;
|
|
}
|
|
|
|
if(r > (INT64)((INT32)0x7FFFFFFF))
|
|
{
|
|
FPR(t).id = 0x7FFFFFFF;
|
|
// FPSCR[FR] = 0
|
|
// FPSCR[FI] = 1
|
|
// FPSCR[XX] = 1
|
|
}
|
|
else if(FPR(b).fd < (INT64)((INT32)0x80000000))
|
|
{
|
|
FPR(t).id = 0x80000000;
|
|
// FPSCR[FR] = 1
|
|
// FPSCR[FI] = 1
|
|
// FPSCR[XX] = 1
|
|
}
|
|
else
|
|
{
|
|
FPR(t).id = (UINT32)r;
|
|
// FPSCR[FR] = t.iw > t.fd
|
|
// FPSCR[FI] = t.iw == t.fd
|
|
// FPSCR[XX] = ?
|
|
}
|
|
|
|
// FPSCR[FPRF] = undefined (leave it as is)
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fctiwzx(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 t = RT;
|
|
INT64 r;
|
|
|
|
// TODO: fix FPSCR flags FX,VXSNAN,VXCVI
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN_1(FPR(b));
|
|
r = (INT64)trunc(FPR(b).fd);
|
|
|
|
if(r > (INT64)((INT32)0x7fffffff))
|
|
{
|
|
FPR(t).id = 0x7fffffff;
|
|
// FPSCR[FR] = 0
|
|
// FPSCR[FI] = 1
|
|
// FPSCR[XX] = 1
|
|
|
|
}
|
|
else if(r < (INT64)((INT32)0x80000000))
|
|
{
|
|
FPR(t).id = 0x80000000;
|
|
// FPSCR[FR] = 1
|
|
// FPSCR[FI] = 1
|
|
// FPSCR[XX] = 1
|
|
}
|
|
else
|
|
{
|
|
FPR(t).id = (UINT32)r;
|
|
// FPSCR[FR] = t.iw > t.fd
|
|
// FPSCR[FI] = t.iw == t.fd
|
|
// FPSCR[XX] = ?
|
|
}
|
|
|
|
// FPSCR[FPRF] = undefined (leave it as is)
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fdivx(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(b));
|
|
|
|
FPR(t).fd = FPR(a).fd / FPR(b).fd;
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fmrx(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
FPR(t).fd = FPR(b).fd;
|
|
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fnabsx(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
FPR(t).id = FPR(b).id | DOUBLE_SIGN;
|
|
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fnegx(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
FPR(t).id = FPR(b).id ^ DOUBLE_SIGN;
|
|
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_frspx(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN_1(FPR(b));
|
|
|
|
FPR(t).fd = (float)FPR(b).fd;
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_frsqrtex(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN_1(FPR(b));
|
|
|
|
FPR(t).fd = 1.0 / sqrt(FPR(b).fd); /* verify this */
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fsqrtx(UINT32 op)
|
|
{
|
|
/* NOTE: PPC603e doesn't support this opcode */
|
|
UINT32 b = RB;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN_1(FPR(b));
|
|
|
|
FPR(t).fd = (double)(sqrt(FPR(b).fd));
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fsubx(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(b));
|
|
|
|
FPR(t).fd = FPR(a).fd - FPR(b).fd;
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_mffsx(UINT32 op)
|
|
{
|
|
FPR(RT).id = (UINT32)ppc.fpscr;
|
|
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_mtfsb0x(UINT32 op)
|
|
{
|
|
UINT32 crbD;
|
|
|
|
crbD = (op >> 21) & 0x1F;
|
|
|
|
if (crbD != 1 && crbD != 2) // these bits cannot be explicitly cleared
|
|
ppc.fpscr &= ~(1 << (31 - crbD));
|
|
|
|
set_rounding_mode();
|
|
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_mtfsb1x(UINT32 op)
|
|
{
|
|
UINT32 crbD;
|
|
|
|
crbD = (op >> 21) & 0x1F;
|
|
|
|
if (crbD != 1 && crbD != 2) // these bits cannot be explicitly cleared
|
|
ppc.fpscr |= (1 << (31 - crbD));
|
|
|
|
set_rounding_mode();
|
|
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_mtfsfx(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 f = ppc_field_xlat[FM];
|
|
|
|
ppc.fpscr &= (~f) | ~(FPSCR_FEX | FPSCR_VX);
|
|
ppc.fpscr |= (UINT32)(FPR(b).id) & ~(FPSCR_FEX | FPSCR_VX);
|
|
|
|
set_rounding_mode();
|
|
|
|
// FEX, VX
|
|
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_mtfsfix(UINT32 op)
|
|
{
|
|
UINT32 crfd = CRFD;
|
|
UINT32 imm = (op >> 12) & 0xF;
|
|
|
|
/*
|
|
* According to the manual:
|
|
*
|
|
* If bits 0 and 3 of FPSCR are to be modified, they take the immediate
|
|
* value specified. Bits 1 and 2 (FEX and VX) are set according to the
|
|
* "usual rule" and not from IMM[1-2].
|
|
*
|
|
* The "usual rule" is not emulated, so these bits simply aren't modified
|
|
* at all here.
|
|
*/
|
|
|
|
crfd = (7 - crfd) * 4; // calculate LSB position of field
|
|
|
|
if (crfd == 28) // field containing FEX and VX is special...
|
|
{ // bits 1 and 2 of FPSCR must not be altered
|
|
ppc.fpscr &= 0x9fffffff;
|
|
ppc.fpscr |= (imm & 0x9fffffff);
|
|
}
|
|
|
|
ppc.fpscr &= ~(0xf << crfd); // clear field
|
|
ppc.fpscr |= (imm << crfd); // insert new data
|
|
|
|
set_rounding_mode();
|
|
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_mcrfs(UINT32 op)
|
|
{
|
|
UINT32 crfs, f;
|
|
crfs = CRFA;
|
|
|
|
f = ppc.fpscr >> ((7 - crfs) * 4); // get crfS field from FPSCR
|
|
f &= 0xf;
|
|
|
|
switch(crfs) // determine which exception bits to clear in FPSCR
|
|
{
|
|
case 0: // FX, OX
|
|
ppc.fpscr &= ~0x90000000;
|
|
break;
|
|
case 1: // UX, ZX, XX, VXSNAN
|
|
ppc.fpscr &= ~0x0f000000;
|
|
break;
|
|
case 2: // VXISI, VXIDI, VXZDZ, VXIMZ
|
|
ppc.fpscr &= ~0x00F00000;
|
|
break;
|
|
case 3: // VXVC
|
|
ppc.fpscr &= ~0x00080000;
|
|
break;
|
|
case 5: // VXSOFT, VXSQRT, VXCVI
|
|
ppc.fpscr &= ~0x00000700;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
CR(CRFD) = f;
|
|
}
|
|
|
|
static void ppc_faddsx(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(b));
|
|
|
|
FPR(t).fd = (float)(FPR(a).fd + FPR(b).fd);
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fdivsx(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(b));
|
|
|
|
FPR(t).fd = (float)(FPR(a).fd / FPR(b).fd);
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fresx(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN_1(FPR(b));
|
|
|
|
// On the 603 fres behaves the same as fdivs RT, 1.0, RB
|
|
FPR(t).fd = (float)(1.0 / FPR(b).fd);
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fsqrtsx(UINT32 op)
|
|
{
|
|
/* NOTE: This opcode is not supported in PPC603e */
|
|
UINT32 b = RB;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN_1(FPR(b));
|
|
|
|
FPR(t).fd = (float)(sqrt(FPR(b).fd));
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fsubsx(UINT32 op)
|
|
{
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(b));
|
|
|
|
FPR(t).fd = (float)(FPR(a).fd - FPR(b).fd);
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fmaddx(UINT32 op)
|
|
{
|
|
UINT32 c = RC;
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(b));
|
|
SET_VXSNAN_1(FPR(c));
|
|
|
|
FPR(t).fd = ((FPR(a).fd * FPR(c).fd) + FPR(b).fd);
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fmsubx(UINT32 op)
|
|
{
|
|
UINT32 c = RC;
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(b));
|
|
SET_VXSNAN_1(FPR(c));
|
|
|
|
FPR(t).fd = ((FPR(a).fd * FPR(c).fd) - FPR(b).fd);
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fmulx(UINT32 op)
|
|
{
|
|
UINT32 c = RC;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(c));
|
|
|
|
FPR(t).fd = (FPR(a).fd * FPR(c).fd);
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fnmaddx(UINT32 op)
|
|
{
|
|
UINT32 c = RC;
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(b));
|
|
SET_VXSNAN_1(FPR(c));
|
|
|
|
FPR(t).fd = (-((FPR(a).fd * FPR(c).fd) + FPR(b).fd));
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fnmsubx(UINT32 op)
|
|
{
|
|
UINT32 c = RC;
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(b));
|
|
SET_VXSNAN_1(FPR(c));
|
|
|
|
FPR(t).fd = (-((FPR(a).fd * FPR(c).fd) - FPR(b).fd));
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fselx(UINT32 op)
|
|
{
|
|
UINT32 c = RC;
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
FPR(t).fd = (FPR(a).fd >= 0.0) ? FPR(c).fd : FPR(b).fd;
|
|
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fmaddsx(UINT32 op)
|
|
{
|
|
UINT32 c = RC;
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(b));
|
|
SET_VXSNAN_1(FPR(c));
|
|
|
|
FPR(t).fd = (float)((FPR(a).fd * FPR(c).fd) + FPR(b).fd);
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fmsubsx(UINT32 op)
|
|
{
|
|
UINT32 c = RC;
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(b));
|
|
SET_VXSNAN_1(FPR(c));
|
|
|
|
FPR(t).fd = (float)((FPR(a).fd * FPR(c).fd) - FPR(b).fd);
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fmulsx(UINT32 op)
|
|
{
|
|
UINT32 c = RC;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
SET_VXSNAN(FPR(a), FPR(c));
|
|
|
|
FPR(t).fd = (float)(FPR(a).fd * FPR(c).fd);
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fnmaddsx(UINT32 op)
|
|
{
|
|
UINT32 c = RC;
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(b));
|
|
SET_VXSNAN_1(FPR(c));
|
|
|
|
FPR(t).fd = (float)(-((FPR(a).fd * FPR(c).fd) + FPR(b).fd));
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|
|
|
|
static void ppc_fnmsubsx(UINT32 op)
|
|
{
|
|
UINT32 c = RC;
|
|
UINT32 b = RB;
|
|
UINT32 a = RA;
|
|
UINT32 t = RT;
|
|
|
|
CHECK_FPU_AVAILABLE();
|
|
|
|
SET_VXSNAN(FPR(a), FPR(b));
|
|
SET_VXSNAN_1(FPR(c));
|
|
|
|
FPR(t).fd = (float)(-((FPR(a).fd * FPR(c).fd) - FPR(b).fd));
|
|
|
|
set_fprf(FPR(t));
|
|
if( RCBIT ) {
|
|
SET_CR1();
|
|
}
|
|
}
|