2023-08-16 23:29:22 +00:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <Project64-rsp-core/Settings/RspSettings.h>
|
2023-08-10 00:57:11 +00:00
|
|
|
#include <Project64-rsp-core/cpu/RSPOpcode.h>
|
2024-08-22 10:14:07 +00:00
|
|
|
#include <Project64-rsp-core/cpu/RspPipelineStage.h>
|
2023-08-10 00:57:11 +00:00
|
|
|
#include <Project64-rsp-core/cpu/RspTypes.h>
|
2023-08-16 23:29:22 +00:00
|
|
|
#include <Settings/Settings.h>
|
2016-01-27 09:11:59 +00:00
|
|
|
|
2024-08-08 00:09:45 +00:00
|
|
|
class CRSPSystem;
|
2024-08-22 07:00:20 +00:00
|
|
|
class RSPRegisterHandlerPlugin;
|
2024-08-08 00:09:45 +00:00
|
|
|
|
|
|
|
class CRSPRecompiler
|
|
|
|
{
|
|
|
|
friend class CRSPRecompilerOps;
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
uint32_t StartPC, CurrPC; // Block start
|
|
|
|
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
uint32_t TargetPC; // Target for this unknown branch
|
|
|
|
uint32_t * X86JumpLoc; // Our x86 uint32_t to fill
|
|
|
|
} BranchesToResolve[200]; // Branches inside or outside block
|
|
|
|
|
|
|
|
uint32_t ResolveCount; // Branches with NULL jump table
|
|
|
|
} RSP_BLOCK;
|
|
|
|
|
|
|
|
public:
|
|
|
|
CRSPRecompiler(CRSPSystem & System);
|
|
|
|
|
2024-08-22 07:00:20 +00:00
|
|
|
void RunCPU(void);
|
2024-08-08 00:09:45 +00:00
|
|
|
void Branch_AddRef(uint32_t Target, uint32_t * X86Loc);
|
2024-08-22 07:00:20 +00:00
|
|
|
void SetJumpTable(uint32_t End);
|
2024-08-08 00:09:45 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
CRSPRecompiler();
|
|
|
|
CRSPRecompiler(const CRSPRecompiler &);
|
|
|
|
CRSPRecompiler & operator=(const CRSPRecompiler &);
|
|
|
|
|
2024-08-22 10:14:07 +00:00
|
|
|
void CompilerLinkBlocks(void);
|
2024-08-08 00:09:45 +00:00
|
|
|
void CompilerRSPBlock(void);
|
|
|
|
void LinkBranches(RSP_BLOCK * Block);
|
|
|
|
void ReOrderSubBlock(RSP_BLOCK * Block);
|
2024-08-22 07:00:20 +00:00
|
|
|
void ReOrderInstructions(uint32_t StartPC, uint32_t EndPC);
|
|
|
|
void ResetJumpTables(void);
|
2024-08-08 00:09:45 +00:00
|
|
|
|
|
|
|
CRSPSystem & m_System;
|
2024-08-22 07:00:20 +00:00
|
|
|
RSPRegisterHandlerPlugin *& m_RSPRegisterHandler;
|
2024-08-08 03:25:54 +00:00
|
|
|
RSPOpcode & m_OpCode;
|
2024-08-08 00:09:45 +00:00
|
|
|
RSP_BLOCK m_CurrentBlock;
|
2024-08-22 10:14:07 +00:00
|
|
|
RSPPIPELINE_STAGE m_NextInstruction;
|
2024-08-22 07:00:20 +00:00
|
|
|
uint8_t *& m_IMEM;
|
2024-08-08 00:09:45 +00:00
|
|
|
};
|
|
|
|
|
2024-08-22 10:14:07 +00:00
|
|
|
extern uint32_t CompilePC, JumpTableSize;
|
2023-06-29 02:59:07 +00:00
|
|
|
extern bool ChangedPC;
|
2016-01-27 09:11:59 +00:00
|
|
|
|
2023-06-01 11:46:23 +00:00
|
|
|
#define CompilerWarning \
|
2023-08-16 23:29:22 +00:00
|
|
|
if (ShowErrors) g_Notify->DisplayError
|
2016-01-27 09:11:59 +00:00
|
|
|
|
2023-06-01 11:46:23 +00:00
|
|
|
#define High16BitAccum 1
|
|
|
|
#define Middle16BitAccum 2
|
|
|
|
#define Low16BitAccum 4
|
|
|
|
#define EntireAccum (Low16BitAccum | Middle16BitAccum | High16BitAccum)
|
2016-01-27 09:11:59 +00:00
|
|
|
|
2023-06-29 02:59:07 +00:00
|
|
|
bool WriteToAccum(int Location, int PC);
|
2023-08-10 04:46:57 +00:00
|
|
|
bool WriteToVectorDest(uint32_t DestReg, int PC);
|
2023-06-29 02:59:07 +00:00
|
|
|
bool UseRspFlags(int PC);
|
2016-01-27 09:11:59 +00:00
|
|
|
|
2023-08-10 04:46:57 +00:00
|
|
|
bool DelaySlotAffectBranch(uint32_t PC);
|
|
|
|
bool CompareInstructions(uint32_t PC, RSPOpcode * Top, RSPOpcode * Bottom);
|
|
|
|
bool IsOpcodeBranch(uint32_t PC, RSPOpcode RspOp);
|
|
|
|
bool IsOpcodeNop(uint32_t PC);
|
2016-01-27 09:11:59 +00:00
|
|
|
|
2023-08-10 04:46:57 +00:00
|
|
|
bool IsNextInstructionMmx(uint32_t PC);
|
|
|
|
bool IsRegisterConstant(uint32_t Reg, uint32_t * Constant);
|
2016-01-27 09:11:59 +00:00
|
|
|
|
2023-06-01 11:46:23 +00:00
|
|
|
#define MainBuffer 0
|
|
|
|
#define SecondaryBuffer 1
|
2016-01-27 09:11:59 +00:00
|
|
|
|
2023-06-01 11:46:23 +00:00
|
|
|
void BuildRecompilerCPU(void);
|
2016-01-27 09:11:59 +00:00
|
|
|
|
2023-06-01 11:46:23 +00:00
|
|
|
void CompilerToggleBuffer(void);
|
2016-01-27 09:11:59 +00:00
|
|
|
|
2023-06-01 11:46:23 +00:00
|
|
|
typedef struct
|
|
|
|
{
|
2023-08-10 04:46:57 +00:00
|
|
|
bool bIsRegConst[32]; // bool toggle for constant
|
|
|
|
uint32_t MipsRegConst[32]; // Value of register 32-bit
|
|
|
|
uint32_t BranchLabels[250];
|
|
|
|
uint32_t LabelCount;
|
|
|
|
uint32_t BranchLocations[250];
|
|
|
|
uint32_t BranchCount;
|
2016-01-27 09:11:59 +00:00
|
|
|
} RSP_CODE;
|
|
|
|
|
|
|
|
extern RSP_CODE RspCode;
|
|
|
|
|
2023-06-01 11:46:23 +00:00
|
|
|
#define IsRegConst(i) (RspCode.bIsRegConst[i])
|
2016-01-27 09:11:59 +00:00
|
|
|
#define MipsRegConst(i) (RspCode.MipsRegConst[i])
|
|
|
|
|
2023-06-01 11:46:23 +00:00
|
|
|
typedef struct
|
|
|
|
{
|
2023-06-29 02:59:07 +00:00
|
|
|
bool mmx, mmx2, sse; // CPU specs and compiling
|
|
|
|
bool bFlags; // RSP flag analysis
|
|
|
|
bool bReOrdering; // Instruction reordering
|
|
|
|
bool bSections; // Microcode sections
|
|
|
|
bool bDest; // Vector destination toggle
|
|
|
|
bool bAccum; // Accumulator toggle
|
|
|
|
bool bGPRConstants; // Analyze GPR constants
|
|
|
|
bool bAlignVector; // Align known vector loads
|
|
|
|
bool bAudioUcode; // Audio microcode analysis
|
2016-01-27 09:11:59 +00:00
|
|
|
} RSP_COMPILER;
|
|
|
|
|
|
|
|
extern RSP_COMPILER Compiler;
|
|
|
|
|
2023-06-01 11:46:23 +00:00
|
|
|
#define IsMmxEnabled (Compiler.mmx)
|
|
|
|
#define IsMmx2Enabled (Compiler.mmx2)
|
2024-08-02 12:30:01 +00:00
|
|
|
#define IsSseEnabled (Compiler.sse)
|