forked from dolphin-emu/dolphin
		
	* Added 21 Arithmetic instructions to the JIT * Used the DSPAnalyser to identify arithmetic/multiplier instructions that precede a conditional branch. This allows the JIT to skip updating the SR when nothing would read from it. Marked all arithmetic/multiplier instructions in the DSPTable for this purpose. * Converted CheckExternalInterrupt into ASM * Fixed a couple instructions in DSP Load/Store git-svn-id: https://dolphin-emu.googlecode.com/svn/trunk@6633 8ced0084-cf51-0410-be5f-012b33b47a6e
		
			
				
	
	
		
			159 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			159 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
// Copyright (C) 2003 Dolphin Project.
 | 
						|
 | 
						|
// 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, version 2.0.
 | 
						|
 | 
						|
// 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 2.0 for more details.
 | 
						|
 | 
						|
// A copy of the GPL 2.0 should have been included with the program.
 | 
						|
// If not, see http://www.gnu.org/licenses/
 | 
						|
 | 
						|
// Official SVN repository and contact information can be found at
 | 
						|
// http://code.google.com/p/dolphin-emu/
 | 
						|
 | 
						|
// Additional copyrights go to Duddie (c) 2005 (duddie@walla.com)
 | 
						|
 | 
						|
#ifndef _DSPTABLES_H
 | 
						|
#define _DSPTABLES_H
 | 
						|
 | 
						|
//nclude "Common.h"
 | 
						|
#include "DSPEmitter.h"
 | 
						|
#include "DSPCommon.h"
 | 
						|
 | 
						|
// The non-ADDR ones that end with _D are the opposite one - if the bit specify
 | 
						|
// ACC0, then ACC_D will be ACC1.
 | 
						|
 | 
						|
// The values of these are very important.
 | 
						|
// For the reg ones, the value >> 8 is the base register.
 | 
						|
// & 0x80  means it's a "D".
 | 
						|
 | 
						|
enum partype_t
 | 
						|
{
 | 
						|
	P_NONE		= 0x0000,
 | 
						|
	P_VAL       = 0x0001,
 | 
						|
	P_IMM		= 0x0002,
 | 
						|
	P_MEM		= 0x0003,
 | 
						|
	P_STR		= 0x0004,
 | 
						|
	P_ADDR_I	= 0x0005,
 | 
						|
	P_ADDR_D	= 0x0006,
 | 
						|
	P_REG		= 0x8000,
 | 
						|
	P_REG04     = P_REG | 0x0400, // IX
 | 
						|
	P_REG08		= P_REG | 0x0800, 
 | 
						|
	P_REG18		= P_REG | 0x1800,
 | 
						|
	P_REGM18	= P_REG | 0x1810, // used in multiply instructions
 | 
						|
	P_REG19		= P_REG | 0x1900,
 | 
						|
	P_REGM19	= P_REG | 0x1910, // used in multiply instructions
 | 
						|
	P_REG1A		= P_REG | 0x1a80,
 | 
						|
	P_REG1C		= P_REG | 0x1c00,
 | 
						|
//	P_ACC		= P_REG | 0x1c10, // used for global accum (gcdsptool's value)
 | 
						|
	P_ACCL		= P_REG | 0x1c00, // used for low part of accum
 | 
						|
	P_ACCM		= P_REG | 0x1e00, // used for mid part of accum
 | 
						|
	// The following are not in gcdsptool
 | 
						|
	P_ACCM_D	= P_REG | 0x1e80,
 | 
						|
	P_ACC		= P_REG | 0x2000, // used for full accum.
 | 
						|
	P_ACC_D		= P_REG | 0x2080,
 | 
						|
	P_AX		= P_REG | 0x2200,
 | 
						|
	P_REGS_MASK	= 0x03f80, // gcdsptool's value = 0x01f80
 | 
						|
	P_REF       = P_REG | 0x4000,
 | 
						|
	P_PRG       = P_REF | P_REG,
 | 
						|
 | 
						|
	// The following seem like junk:
 | 
						|
	//	P_REG10		= P_REG | 0x1000,
 | 
						|
	//	P_AX_D		= P_REG | 0x2280,
 | 
						|
};
 | 
						|
 | 
						|
#define OPTABLE_SIZE 0xffff + 1
 | 
						|
#define EXT_OPTABLE_SIZE 0xff + 1
 | 
						|
 | 
						|
typedef void (*dspIntFunc)(const UDSPInstruction);
 | 
						|
typedef void (DSPEmitter::*dspJitFunc)(const UDSPInstruction);
 | 
						|
 | 
						|
struct param2_t
 | 
						|
{
 | 
						|
	partype_t type;
 | 
						|
	u8 size;
 | 
						|
	u8 loc;
 | 
						|
	s8 lshift;
 | 
						|
	u16 mask;
 | 
						|
};
 | 
						|
 | 
						|
typedef struct
 | 
						|
{
 | 
						|
	const char *name;
 | 
						|
	u16 opcode;
 | 
						|
	u16 opcode_mask;
 | 
						|
 | 
						|
	dspIntFunc intFunc;
 | 
						|
	dspJitFunc jitFunc;
 | 
						|
 | 
						|
	u8 size;
 | 
						|
	u8 param_count;
 | 
						|
	param2_t params[8];
 | 
						|
	bool extended;
 | 
						|
	bool branch;
 | 
						|
	bool uncond_branch;
 | 
						|
	bool reads_pc;
 | 
						|
	bool updates_sr;
 | 
						|
} DSPOPCTemplate;
 | 
						|
 | 
						|
typedef DSPOPCTemplate opc_t;
 | 
						|
 | 
						|
// Opcodes
 | 
						|
extern const DSPOPCTemplate opcodes[];
 | 
						|
extern const int opcodes_size;
 | 
						|
extern const DSPOPCTemplate opcodes_ext[];
 | 
						|
extern const int opcodes_ext_size;
 | 
						|
extern const DSPOPCTemplate cw;
 | 
						|
 | 
						|
#define WRITEBACKLOGSIZE 5
 | 
						|
 | 
						|
extern const DSPOPCTemplate *opTable[OPTABLE_SIZE];
 | 
						|
extern const DSPOPCTemplate *extOpTable[EXT_OPTABLE_SIZE];
 | 
						|
extern u16 writeBackLog[WRITEBACKLOGSIZE];
 | 
						|
extern int writeBackLogIdx[WRITEBACKLOGSIZE];
 | 
						|
 | 
						|
// Predefined labels
 | 
						|
struct pdlabel_t
 | 
						|
{
 | 
						|
	u16 addr;
 | 
						|
	const char* name;
 | 
						|
	const char* description;
 | 
						|
};
 | 
						|
 | 
						|
extern const pdlabel_t regnames[];
 | 
						|
extern const pdlabel_t pdlabels[];
 | 
						|
extern const u32 pdlabels_size;
 | 
						|
 | 
						|
const char *pdname(u16 val);
 | 
						|
const char *pdregname(int val);
 | 
						|
const char *pdregnamelong(int val);
 | 
						|
 | 
						|
void InitInstructionTable();
 | 
						|
void applyWriteBackLog();
 | 
						|
void zeroWriteBackLog();
 | 
						|
void zeroWriteBackLogPreserveAcc(u8 acc);
 | 
						|
 | 
						|
const DSPOPCTemplate *GetOpTemplate(const UDSPInstruction &inst);
 | 
						|
 | 
						|
inline void ExecuteInstruction(const UDSPInstruction inst)
 | 
						|
{
 | 
						|
	const DSPOPCTemplate *tinst = GetOpTemplate(inst);
 | 
						|
 | 
						|
	if (tinst->extended) {
 | 
						|
		if ((inst >> 12) == 0x3)
 | 
						|
			extOpTable[inst & 0x7F]->intFunc(inst);
 | 
						|
		else
 | 
						|
			extOpTable[inst & 0xFF]->intFunc(inst);
 | 
						|
	}
 | 
						|
	tinst->intFunc(inst);
 | 
						|
	if (tinst->extended) {
 | 
						|
		applyWriteBackLog();
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
#endif // _DSPTABLES_H
 |