2016-01-27 09:11:59 +00:00
# include "stdafx.h"
2022-10-10 00:22:17 +00:00
2016-01-27 09:11:59 +00:00
# include "LoopAnalysis.h"
2022-10-10 00:22:17 +00:00
# include <string.h>
2016-01-27 09:11:59 +00:00
2022-10-10 00:22:17 +00:00
# include <Project64-core/N64System/Mips/MemoryVirtualMem.h>
# include <Project64-core/N64System/Mips/R4300iInstruction.h>
2016-01-27 09:11:59 +00:00
# include <Project64-core/N64System/N64Types.h>
# include <Project64-core/N64System/Recompiler/CodeBlock.h>
# include <Project64-core/N64System/SystemGlobals.h>
# ifdef _DEBUG
# define CHECKED_BUILD 1
# endif
2022-08-08 10:52:51 +00:00
LoopAnalysis : : LoopAnalysis ( CCodeBlock & CodeBlock , CCodeSection * Section ) :
2016-07-03 05:22:14 +00:00
m_EnterSection ( Section ) ,
2022-08-08 10:52:51 +00:00
m_CodeBlock ( CodeBlock ) ,
2016-07-03 05:22:14 +00:00
m_PC ( ( uint32_t ) - 1 ) ,
2022-01-18 07:47:21 +00:00
m_PipelineStage ( PIPELINE_STAGE_NORMAL ) ,
2022-08-08 10:52:51 +00:00
m_Test ( CodeBlock . NextTest ( ) ) ,
2022-08-15 03:09:34 +00:00
m_Reg ( CodeBlock , CodeBlock . RecompilerOps ( ) - > Assembler ( ) )
2016-01-27 09:11:59 +00:00
{
memset ( & m_Command , 0 , sizeof ( m_Command ) ) ;
}
LoopAnalysis : : ~ LoopAnalysis ( )
{
for ( RegisterMap : : iterator itr = m_EnterRegisters . begin ( ) ; itr ! = m_EnterRegisters . end ( ) ; itr + + )
{
delete itr - > second ;
}
m_EnterRegisters . clear ( ) ;
for ( RegisterMap : : iterator itr = m_ContinueRegisters . begin ( ) ; itr ! = m_ContinueRegisters . end ( ) ; itr + + )
{
delete itr - > second ;
}
m_ContinueRegisters . clear ( ) ;
for ( RegisterMap : : iterator itr = m_JumpRegisters . begin ( ) ; itr ! = m_JumpRegisters . end ( ) ; itr + + )
{
delete itr - > second ;
}
m_JumpRegisters . clear ( ) ;
}
bool LoopAnalysis : : SetupRegisterForLoop ( )
{
if ( ! m_EnterSection - > m_InLoop )
{
return false ;
}
2022-08-08 10:52:51 +00:00
m_CodeBlock . Log ( " %s: Section ID: %d Test: %X " , __FUNCTION__ , m_EnterSection - > m_SectionID , m_Test ) ;
2016-01-27 09:11:59 +00:00
if ( ! CheckLoopRegisterUsage ( m_EnterSection ) )
{
return false ;
}
RegisterMap : : iterator itr = m_EnterRegisters . find ( m_EnterSection - > m_SectionID ) ;
if ( itr = = m_EnterRegisters . end ( ) )
{
return false ;
}
m_EnterSection - > m_RegEnter = * ( itr - > second ) ;
return true ;
}
bool LoopAnalysis : : SetupEnterSection ( CCodeSection * Section , bool & bChanged , bool & bSkipedSection )
{
bChanged = false ;
bSkipedSection = false ;
2022-10-10 00:22:17 +00:00
if ( Section - > m_ParentSection . empty ( ) )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
return true ;
}
2016-01-27 09:11:59 +00:00
2022-11-23 04:16:55 +00:00
m_CodeBlock . Log ( " %s: Block EnterPC: %X Section ID %d Test: %X Section Test: %X " , __FUNCTION__ , m_CodeBlock . VAddrEnter ( ) , Section - > m_SectionID , m_Test , Section - > m_Test ) ;
2016-01-27 09:11:59 +00:00
bool bFirstParent = true ;
2022-08-15 03:09:34 +00:00
CRegInfo RegEnter ( m_CodeBlock , m_CodeBlock . RecompilerOps ( ) - > Assembler ( ) ) ;
2016-01-27 09:11:59 +00:00
for ( CCodeSection : : SECTION_LIST : : iterator iter = Section - > m_ParentSection . begin ( ) ; iter ! = Section - > m_ParentSection . end ( ) ; iter + + )
{
CCodeSection * Parent = * iter ;
2022-11-23 04:16:55 +00:00
m_CodeBlock . Log ( " %s: Parent Section ID %d Test: %X Section Test: %X " , __FUNCTION__ , Parent - > m_SectionID , m_Test , Parent - > m_Test ) ;
if ( Parent - > m_Test ! = m_Test & & ( m_EnterSection ! = Section | | ! Parent - > m_EnterLabel . isValid ( ) ) & & Parent - > m_InLoop )
2016-01-27 09:11:59 +00:00
{
2022-11-23 04:16:55 +00:00
m_CodeBlock . Log ( " %s: Ignore Parent Section ID %d Test: %X Section Test: %X " , __FUNCTION__ , Parent - > m_SectionID , m_Test , Parent - > m_Test ) ;
2016-01-27 09:11:59 +00:00
bSkipedSection = true ;
continue ;
}
RegisterMap : : iterator Continue_itr = m_ContinueRegisters . find ( Parent - > m_SectionID ) ;
RegisterMap : : iterator Jump_itr = m_JumpRegisters . find ( Parent - > m_SectionID ) ;
2022-10-10 00:22:17 +00:00
CCodeSection * TargetSection [ ] = { Parent - > m_ContinueSection , Parent - > m_JumpSection } ;
2016-01-27 09:11:59 +00:00
CRegInfo * JumpRegInfo [ ] =
2022-10-10 00:22:17 +00:00
{
Continue_itr = = m_ContinueRegisters . end ( ) ? & Parent - > m_Cont . RegSet : Continue_itr - > second ,
Jump_itr = = m_JumpRegisters . end ( ) ? & Parent - > m_Jump . RegSet : Jump_itr - > second } ;
2016-01-27 09:11:59 +00:00
for ( int i = 0 ; i < 2 ; i + + )
{
2022-10-10 00:22:17 +00:00
if ( TargetSection [ i ] ! = Section )
{
continue ;
}
2016-01-27 09:11:59 +00:00
if ( bFirstParent )
{
bFirstParent = false ;
RegEnter = * JumpRegInfo [ i ] ;
}
else
{
if ( * JumpRegInfo [ i ] = = RegEnter )
{
continue ;
}
SyncRegState ( RegEnter , * JumpRegInfo [ i ] ) ;
}
}
}
if ( bFirstParent )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
RegisterMap : : iterator itr = m_EnterRegisters . find ( Section - > m_SectionID ) ;
if ( itr ! = m_EnterRegisters . end ( ) )
{
if ( SyncRegState ( * ( itr - > second ) , RegEnter ) )
{
bChanged = true ;
}
}
else
{
m_EnterRegisters . insert ( RegisterMap : : value_type ( Section - > m_SectionID , new CRegInfo ( RegEnter ) ) ) ;
}
return true ;
}
bool LoopAnalysis : : CheckLoopRegisterUsage ( CCodeSection * Section )
{
2022-10-10 00:22:17 +00:00
if ( Section = = nullptr )
{
return true ;
}
if ( ! Section - > m_InLoop )
{
return true ;
}
2016-01-27 09:11:59 +00:00
2022-08-08 10:52:51 +00:00
m_CodeBlock . Log ( " %s: Section %d Block PC: 0x%X " , __FUNCTION__ , Section - > m_SectionID , m_CodeBlock . VAddrEnter ( ) ) ;
2016-01-27 09:11:59 +00:00
bool bChanged = false , bSkipedSection = false ;
if ( Section = = m_EnterSection & & Section - > m_Test = = m_Test )
{
2022-10-10 00:22:17 +00:00
if ( ! SetupEnterSection ( Section , bChanged , bSkipedSection ) )
{
return false ;
}
2016-01-27 09:11:59 +00:00
return true ;
}
2022-10-10 00:22:17 +00:00
if ( ! SetupEnterSection ( Section , bChanged , bSkipedSection ) )
{
return false ;
}
2016-01-27 09:11:59 +00:00
if ( Section - > m_Test = = m_Test & & ! bChanged )
{
return true ;
}
2022-08-08 10:52:51 +00:00
m_CodeBlock . Log ( " %s: Set Section %d test to %X from %X " , __FUNCTION__ , Section - > m_SectionID , m_Test , Section - > m_Test ) ;
2016-01-27 09:11:59 +00:00
Section - > m_Test = m_Test ;
m_PC = Section - > m_EnterPC ;
RegisterMap : : iterator itr = m_EnterRegisters . find ( Section - > m_SectionID ) ;
m_Reg = itr ! = m_EnterRegisters . end ( ) ? * ( itr - > second ) : Section - > m_RegEnter ;
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_NORMAL ;
2016-01-27 09:11:59 +00:00
uint32_t ContinueSectionPC = Section - > m_ContinueSection ? Section - > m_ContinueSection - > m_EnterPC : ( uint32_t ) - 1 ;
2022-08-08 10:52:51 +00:00
m_CodeBlock . Log ( " ContinueSectionPC = %08X " , ContinueSectionPC ) ;
2016-01-27 09:11:59 +00:00
do
{
2022-07-18 08:31:00 +00:00
if ( ! g_MMU - > MemoryValue32 ( m_PC , m_Command . Value ) )
2016-01-27 09:11:59 +00:00
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
return false ;
}
2024-06-06 04:39:12 +00:00
m_CodeBlock . Log ( " %016llX: %s " , m_PC , R4300iInstruction ( m_PC , m_Command . Value ) . NameAndParam ( ) . c_str ( ) ) ;
2016-01-27 09:11:59 +00:00
switch ( m_Command . op )
{
case R4300i_SPECIAL :
switch ( m_Command . funct )
{
case R4300i_SPECIAL_SLL : SPECIAL_SLL ( ) ; break ;
case R4300i_SPECIAL_SRL : SPECIAL_SRL ( ) ; break ;
case R4300i_SPECIAL_SRA : SPECIAL_SRA ( ) ; break ;
case R4300i_SPECIAL_SLLV : SPECIAL_SLLV ( ) ; break ;
case R4300i_SPECIAL_SRLV : SPECIAL_SRLV ( ) ; break ;
case R4300i_SPECIAL_SRAV : SPECIAL_SRAV ( ) ; break ;
2022-10-10 00:22:17 +00:00
case R4300i_SPECIAL_JR : SPECIAL_JR ( ) ; break ;
2016-01-27 09:11:59 +00:00
case R4300i_SPECIAL_JALR : SPECIAL_JALR ( ) ; break ;
case R4300i_SPECIAL_SYSCALL : SPECIAL_SYSCALL ( Section ) ; break ;
case R4300i_SPECIAL_BREAK : SPECIAL_BREAK ( Section ) ; break ;
case R4300i_SPECIAL_MFHI : SPECIAL_MFHI ( ) ; break ;
case R4300i_SPECIAL_MTHI : SPECIAL_MTHI ( ) ; break ;
case R4300i_SPECIAL_MFLO : SPECIAL_MFLO ( ) ; break ;
case R4300i_SPECIAL_MTLO : SPECIAL_MTLO ( ) ; break ;
case R4300i_SPECIAL_DSLLV : SPECIAL_DSLLV ( ) ; break ;
case R4300i_SPECIAL_DSRLV : SPECIAL_DSRLV ( ) ; break ;
case R4300i_SPECIAL_DSRAV : SPECIAL_DSRAV ( ) ; break ;
case R4300i_SPECIAL_MULT : break ;
case R4300i_SPECIAL_MULTU : break ;
case R4300i_SPECIAL_DIV : break ;
case R4300i_SPECIAL_DIVU : break ;
case R4300i_SPECIAL_DMULT : break ;
case R4300i_SPECIAL_DMULTU : break ;
case R4300i_SPECIAL_DDIV : break ;
case R4300i_SPECIAL_DDIVU : break ;
case R4300i_SPECIAL_ADD : SPECIAL_ADD ( ) ; break ;
case R4300i_SPECIAL_ADDU : SPECIAL_ADDU ( ) ; break ;
case R4300i_SPECIAL_SUB : SPECIAL_SUB ( ) ; break ;
case R4300i_SPECIAL_SUBU : SPECIAL_SUBU ( ) ; break ;
case R4300i_SPECIAL_AND : SPECIAL_AND ( ) ; break ;
case R4300i_SPECIAL_OR : SPECIAL_OR ( ) ; break ;
case R4300i_SPECIAL_XOR : SPECIAL_XOR ( ) ; break ;
case R4300i_SPECIAL_NOR : SPECIAL_NOR ( ) ; break ;
case R4300i_SPECIAL_SLT : SPECIAL_SLT ( ) ; break ;
case R4300i_SPECIAL_SLTU : SPECIAL_SLTU ( ) ; break ;
case R4300i_SPECIAL_DADD : SPECIAL_DADD ( ) ; break ;
case R4300i_SPECIAL_DADDU : SPECIAL_DADDU ( ) ; break ;
case R4300i_SPECIAL_DSUB : SPECIAL_DSUB ( ) ; break ;
case R4300i_SPECIAL_DSUBU : SPECIAL_DSUBU ( ) ; break ;
case R4300i_SPECIAL_DSLL : SPECIAL_DSLL ( ) ; break ;
case R4300i_SPECIAL_DSRL : SPECIAL_DSRL ( ) ; break ;
case R4300i_SPECIAL_DSRA : SPECIAL_DSRA ( ) ; break ;
case R4300i_SPECIAL_DSLL32 : SPECIAL_DSLL32 ( ) ; break ;
case R4300i_SPECIAL_DSRL32 : SPECIAL_DSRL32 ( ) ; break ;
case R4300i_SPECIAL_DSRA32 : SPECIAL_DSRA32 ( ) ; break ;
2022-10-10 00:22:17 +00:00
case R4300i_SPECIAL_TEQ :
case R4300i_SPECIAL_TNE :
case R4300i_SPECIAL_TGE :
case R4300i_SPECIAL_TGEU :
case R4300i_SPECIAL_TLT :
case R4300i_SPECIAL_TLTU :
2019-12-17 15:08:15 +00:00
break ;
2016-01-27 09:11:59 +00:00
default :
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_END_BLOCK ;
2016-01-27 09:11:59 +00:00
m_PC - = 4 ;
}
break ;
case R4300i_REGIMM :
switch ( m_Command . rt )
{
2022-10-10 00:22:17 +00:00
case R4300i_REGIMM_TEQI :
case R4300i_REGIMM_TNEI :
case R4300i_REGIMM_TGEI :
case R4300i_REGIMM_TGEIU :
case R4300i_REGIMM_TLTI :
case R4300i_REGIMM_TLTIU :
2019-12-17 15:08:15 +00:00
break ;
2016-01-27 09:11:59 +00:00
case R4300i_REGIMM_BLTZ :
case R4300i_REGIMM_BGEZ :
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_DELAY_SLOT ;
2016-01-27 09:11:59 +00:00
# ifdef CHECKED_BUILD
if ( Section - > m_Cont . TargetPC ! = m_PC + 8 & &
2021-04-12 11:35:39 +00:00
Section - > m_ContinueSection ! = nullptr & &
2016-01-27 09:11:59 +00:00
Section - > m_Cont . TargetPC ! = ( uint32_t ) - 1 )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
if ( Section - > m_Jump . TargetPC ! = m_PC + ( ( int16_t ) m_Command . offset < < 2 ) + 4 & &
2021-04-12 11:35:39 +00:00
Section - > m_JumpSection ! = nullptr & &
2016-01-27 09:11:59 +00:00
Section - > m_Jump . TargetPC ! = ( uint32_t ) - 1 )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
if ( m_PC = = Section - > m_Jump . TargetPC )
{
2022-07-25 07:05:42 +00:00
R4300iOpcode DelaySlot ;
if ( g_MMU - > MemoryValue32 ( m_PC + 4 , DelaySlot . Value ) & &
! R4300iInstruction ( m_PC , m_Command . Value ) . DelaySlotEffectsCompare ( DelaySlot . Value ) & &
! Section - > m_Jump . PermLoop )
2016-01-27 09:11:59 +00:00
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
}
# endif
break ;
case R4300i_REGIMM_BLTZL :
case R4300i_REGIMM_BGEZL :
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_LIKELY_DELAY_SLOT ;
2016-01-27 09:11:59 +00:00
# ifdef CHECKED_BUILD
if ( Section - > m_Cont . TargetPC ! = m_PC + 8 & &
2021-04-12 11:35:39 +00:00
Section - > m_ContinueSection ! = nullptr & &
2016-01-27 09:11:59 +00:00
Section - > m_Cont . TargetPC ! = ( uint32_t ) - 1 )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
if ( Section - > m_Jump . TargetPC ! = m_PC + 4 & &
2021-04-12 11:35:39 +00:00
Section - > m_JumpSection ! = nullptr & &
2016-01-27 09:11:59 +00:00
Section - > m_Jump . TargetPC ! = ( uint32_t ) - 1 )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
/*if (Section->m_Jump.TargetPC != m_PC + ((int16_t)m_Command.offset << 2) + 4)
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
} */
if ( m_PC = = m_PC + ( ( int16_t ) m_Command . offset < < 2 ) + 4 )
{
2022-07-25 07:05:42 +00:00
R4300iOpcode DelaySlot ;
if ( g_MMU - > MemoryValue32 ( m_PC + 4 , DelaySlot . Value ) & &
! R4300iInstruction ( m_PC , m_Command . Value ) . DelaySlotEffectsCompare ( DelaySlot . Value ) & &
! Section - > m_Jump . PermLoop )
2016-01-27 09:11:59 +00:00
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
}
# endif
break ;
case R4300i_REGIMM_BLTZAL :
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
break ;
case R4300i_REGIMM_BGEZAL :
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
break ;
default :
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
break ;
case R4300i_JAL :
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
break ;
case R4300i_J :
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_DELAY_SLOT ;
2016-01-27 09:11:59 +00:00
# ifdef CHECKED_BUILD
if ( Section - > m_Jump . TargetPC ! = ( m_PC & 0xF0000000 ) + ( m_Command . target < < 2 ) )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
if ( m_PC = = Section - > m_Jump . TargetPC & & ! Section - > m_Jump . PermLoop )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
# endif
break ;
case R4300i_BEQ :
if ( m_PC + ( ( int16_t ) m_Command . offset < < 2 ) + 4 ! = m_PC + 8 )
{
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_DELAY_SLOT ;
2016-01-27 09:11:59 +00:00
# ifdef CHECKED_BUILD
if ( m_Command . rs ! = 0 | | m_Command . rt ! = 0 )
{
if ( Section - > m_Cont . TargetPC ! = m_PC + 8 & &
2021-04-12 11:35:39 +00:00
Section - > m_ContinueSection ! = nullptr & &
2016-01-27 09:11:59 +00:00
Section - > m_Cont . TargetPC ! = ( uint32_t ) - 1 )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
}
else
{
if ( Section - > m_Cont . TargetPC ! = ( uint32_t ) - 1 )
{
//g_Notify->BreakPoint(__FILE__, __LINE__);
}
}
if ( Section - > m_Jump . TargetPC ! = m_PC + ( ( int16_t ) m_Command . offset < < 2 ) + 4 )
{
//g_Notify->BreakPoint(__FILE__, __LINE__);
}
if ( m_PC = = Section - > m_Jump . TargetPC )
{
2022-07-25 07:05:42 +00:00
R4300iOpcode DelaySlot ;
if ( g_MMU - > MemoryValue32 ( m_PC + 4 , DelaySlot . Value ) & &
! R4300iInstruction ( m_PC , m_Command . Value ) . DelaySlotEffectsCompare ( DelaySlot . Value ) & &
! Section - > m_Jump . PermLoop )
2016-01-27 09:11:59 +00:00
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
}
# endif
}
break ;
case R4300i_BNE :
case R4300i_BLEZ :
case R4300i_BGTZ :
if ( m_PC + ( ( int16_t ) m_Command . offset < < 2 ) + 4 ! = m_PC + 8 )
{
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_DELAY_SLOT ;
2016-01-27 09:11:59 +00:00
# ifdef CHECKED_BUILD
if ( Section - > m_Cont . TargetPC ! = m_PC + 8 & &
2021-04-12 11:35:39 +00:00
Section - > m_ContinueSection ! = nullptr & &
2016-01-27 09:11:59 +00:00
Section - > m_Cont . TargetPC ! = ( uint32_t ) - 1 )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
if ( Section - > m_Jump . TargetPC ! = m_PC + ( ( int16_t ) m_Command . offset < < 2 ) + 4 & &
2021-04-12 11:35:39 +00:00
Section - > m_JumpSection ! = nullptr & &
2016-01-27 09:11:59 +00:00
Section - > m_Jump . TargetPC ! = ( uint32_t ) - 1 )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
if ( m_PC = = Section - > m_Jump . TargetPC )
{
2022-07-25 07:05:42 +00:00
R4300iOpcode DelaySlot ;
if ( g_MMU - > MemoryValue32 ( m_PC + 4 , DelaySlot . Value ) & &
! R4300iInstruction ( m_PC , m_Command . Value ) . DelaySlotEffectsCompare ( DelaySlot . Value ) & &
! Section - > m_Jump . PermLoop )
2016-01-27 09:11:59 +00:00
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
}
# endif
}
break ;
case R4300i_ADDI :
case R4300i_ADDIU :
2022-10-10 00:22:17 +00:00
if ( m_Command . rt = = 0 )
{
break ;
}
2016-01-27 09:11:59 +00:00
/*if (m_Command.rs == 0) {
m_Reg . GetMipsRegLo ( m_Command . rt ) = ( int16_t ) m_Command . immediate ;
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_CONST_32_SIGN ) ;
} else { */
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_MODIFIED ) ;
//}
break ;
case R4300i_SLTI :
case R4300i_SLTIU :
2022-10-10 00:22:17 +00:00
if ( m_Command . rt = = 0 )
{
break ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_MODIFIED ) ;
break ;
case R4300i_LUI :
2022-10-10 00:22:17 +00:00
if ( m_Command . rt = = 0 )
{
break ;
}
2016-01-27 09:11:59 +00:00
if ( ! m_Reg . IsModified ( m_Command . rt ) )
{
m_Reg . SetMipsRegLo ( m_Command . rt , ( ( int16_t ) m_Command . offset < < 16 ) ) ;
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_CONST_32_SIGN ) ;
}
break ;
case R4300i_ANDI :
2022-10-10 00:22:17 +00:00
if ( m_Command . rt = = 0 )
{
break ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_MODIFIED ) ;
break ;
case R4300i_ORI :
2022-10-10 00:22:17 +00:00
if ( m_Command . rt = = 0 )
{
break ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rs = = m_Command . rt )
{
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_MODIFIED ) ;
}
2022-10-10 00:22:17 +00:00
if ( m_Reg . IsConst ( m_Command . rs ) )
{
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_CONST_32_SIGN ) ;
m_Reg . SetMipsRegLo ( m_Command . rt , m_Reg . GetMipsRegLo ( m_Command . rs ) | m_Command . immediate ) ;
}
2022-10-10 00:22:17 +00:00
else
{
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_MODIFIED ) ;
}
break ;
case R4300i_XORI :
2022-10-10 00:22:17 +00:00
if ( m_Command . rt = = 0 )
{
break ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rs = = m_Command . rt )
{
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rs ) )
{
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_CONST_32_SIGN ) ;
m_Reg . SetMipsRegLo ( m_Command . rt , m_Reg . GetMipsRegLo ( m_Command . rs ) ^ m_Command . immediate ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_MODIFIED ) ;
}
break ;
case R4300i_CP0 :
switch ( m_Command . rs )
{
case R4300i_COP0_MF :
2022-10-10 00:22:17 +00:00
if ( m_Command . rt = = 0 )
{
break ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_MODIFIED ) ;
break ;
case R4300i_COP0_MT : break ;
default :
if ( ( m_Command . rs & 0x10 ) ! = 0 )
{
switch ( m_Command . funct )
{
case R4300i_COP0_CO_TLBR : break ;
case R4300i_COP0_CO_TLBWI : break ;
case R4300i_COP0_CO_TLBWR : break ;
case R4300i_COP0_CO_TLBP : break ;
2022-01-18 07:47:21 +00:00
case R4300i_COP0_CO_ERET : m_PipelineStage = PIPELINE_STAGE_END_BLOCK ; break ;
2016-01-27 09:11:59 +00:00
default :
2022-07-18 08:31:00 +00:00
g_Notify - > DisplayError ( stdstr_f ( " Unhandled R4300i opcode in FillSectionInfo \n %s " , R4300iInstruction ( m_PC , m_Command . Value ) . NameAndParam ( ) . c_str ( ) ) . c_str ( ) ) ;
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_END_BLOCK ;
2016-01-27 09:11:59 +00:00
m_PC - = 4 ;
}
}
else
{
2022-07-18 08:31:00 +00:00
g_Notify - > DisplayError ( stdstr_f ( " Unhandled R4300i opcode in FillSectionInfo 3 \n %s " , R4300iInstruction ( m_PC , m_Command . Value ) . NameAndParam ( ) . c_str ( ) ) . c_str ( ) ) ;
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_END_BLOCK ;
2016-01-27 09:11:59 +00:00
m_PC - = 4 ;
}
}
break ;
case R4300i_CP1 :
switch ( m_Command . fmt )
{
case R4300i_COP1_CF :
case R4300i_COP1_MF :
case R4300i_COP1_DMF :
2022-10-10 00:22:17 +00:00
if ( m_Command . rt = = 0 )
{
break ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_MODIFIED ) ;
break ;
case R4300i_COP1_BC :
switch ( m_Command . ft )
{
case R4300i_COP1_BC_BCFL :
case R4300i_COP1_BC_BCTL :
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_LIKELY_DELAY_SLOT ;
2016-01-27 09:11:59 +00:00
# ifdef CHECKED_BUILD
if ( Section - > m_Cont . TargetPC ! = m_PC + 8 & &
2021-04-12 11:35:39 +00:00
Section - > m_ContinueSection ! = nullptr & &
2016-01-27 09:11:59 +00:00
Section - > m_Cont . TargetPC ! = ( uint32_t ) - 1 )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
if ( m_PC = = m_PC + ( ( int16_t ) m_Command . offset < < 2 ) + 4 )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
# endif
break ;
case R4300i_COP1_BC_BCF :
case R4300i_COP1_BC_BCT :
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_DELAY_SLOT ;
2016-01-27 09:11:59 +00:00
# ifdef CHECKED_BUILD
if ( Section - > m_Cont . TargetPC ! = m_PC + 8 & &
2021-04-12 11:35:39 +00:00
Section - > m_ContinueSection ! = nullptr & &
2016-01-27 09:11:59 +00:00
Section - > m_Cont . TargetPC ! = ( uint32_t ) - 1 )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
if ( Section - > m_Jump . TargetPC ! = m_PC + ( ( int16_t ) m_Command . offset < < 2 ) + 4 )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
if ( m_PC = = Section - > m_Jump . TargetPC )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
# endif
break ;
}
break ;
case R4300i_COP1_MT : break ;
case R4300i_COP1_DMT : break ;
case R4300i_COP1_CT : break ;
case R4300i_COP1_S : break ;
case R4300i_COP1_D : break ;
case R4300i_COP1_W : break ;
case R4300i_COP1_L : break ;
default :
2022-07-18 08:31:00 +00:00
g_Notify - > DisplayError ( stdstr_f ( " Unhandled R4300i opcode in FillSectionInfo 2 \n %s " , R4300iInstruction ( m_PC , m_Command . Value ) . NameAndParam ( ) . c_str ( ) ) . c_str ( ) ) ;
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_END_BLOCK ;
2016-01-27 09:11:59 +00:00
m_PC - = 4 ;
}
break ;
case R4300i_BEQL :
case R4300i_BNEL :
case R4300i_BLEZL :
case R4300i_BGTZL :
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_LIKELY_DELAY_SLOT ;
2016-01-27 09:11:59 +00:00
# ifdef CHECKED_BUILD
if ( Section - > m_Cont . TargetPC ! = m_PC + 8 & &
2021-04-12 11:35:39 +00:00
Section - > m_ContinueSection ! = nullptr & &
2016-01-27 09:11:59 +00:00
Section - > m_Cont . TargetPC ! = ( uint32_t ) - 1 )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
if ( Section - > m_Jump . TargetPC ! = m_PC + 4 )
{
//g_Notify->BreakPoint(__FILE__, __LINE__);
}
/*if (Section->m_Jump.TargetPC != m_PC + ((int16_t)m_Command.offset << 2) + 4)
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
} */
if ( m_PC = = m_PC + ( ( int16_t ) m_Command . offset < < 2 ) + 4 )
{
2022-07-25 07:05:42 +00:00
R4300iOpcode DelaySlot ;
2022-10-10 00:22:17 +00:00
if ( g_MMU - > MemoryValue32 ( m_PC + 4 , DelaySlot . Value ) & &
2022-07-25 07:05:42 +00:00
! R4300iInstruction ( m_PC , m_Command . Value ) . DelaySlotEffectsCompare ( DelaySlot . Value ) & &
! Section - > m_Jump . PermLoop )
2016-01-27 09:11:59 +00:00
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
}
# endif
break ;
case R4300i_DADDI :
case R4300i_DADDIU :
2022-10-10 00:22:17 +00:00
if ( m_Command . rt = = 0 )
{
break ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rs = = m_Command . rt )
{
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rs ) )
{
if ( m_Reg . Is64Bit ( m_Command . rs ) )
{
int imm32 = ( int16_t ) m_Command . immediate ;
int64_t imm64 = imm32 ;
m_Reg . SetMipsReg_S ( m_Command . rt , m_Reg . GetMipsRegLo_S ( m_Command . rs ) + imm64 ) ;
}
else
{
m_Reg . SetMipsReg_S ( m_Command . rt , m_Reg . GetMipsRegLo_S ( m_Command . rs ) + ( int16_t ) m_Command . immediate ) ;
}
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_CONST_64 ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_MODIFIED ) ;
}
break ;
case R4300i_LDR :
case R4300i_LDL :
case R4300i_LB :
case R4300i_LH :
case R4300i_LWL :
case R4300i_LW :
case R4300i_LWU :
case R4300i_LL :
case R4300i_LBU :
case R4300i_LHU :
case R4300i_LWR :
case R4300i_SC :
2022-10-10 00:22:17 +00:00
if ( m_Command . rt = = 0 )
{
break ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_MODIFIED ) ;
break ;
case R4300i_SB : break ;
case R4300i_SH : break ;
case R4300i_SWL : break ;
case R4300i_SW : break ;
case R4300i_SWR : break ;
case R4300i_SDL : break ;
case R4300i_SDR : break ;
case R4300i_CACHE : break ;
case R4300i_LWC1 : break ;
case R4300i_SWC1 : break ;
case R4300i_LDC1 : break ;
case R4300i_LD :
2022-10-10 00:22:17 +00:00
if ( m_Command . rt = = 0 )
{
break ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rt , CRegInfo : : STATE_MODIFIED ) ;
break ;
case R4300i_SDC1 : break ;
case R4300i_SD : break ;
default :
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_END_BLOCK ;
2016-01-27 09:11:59 +00:00
m_PC - = 4 ;
2022-10-10 00:22:17 +00:00
if ( m_Command . Value = = 0x7C1C97C0 )
{
break ;
}
if ( m_Command . Value = = 0x7FFFFFFF )
{
break ;
}
if ( m_Command . Value = = 0xF1F3F5F7 )
{
break ;
}
if ( m_Command . Value = = 0xC1200000 )
{
break ;
}
if ( m_Command . Value = = 0x4C5A5353 )
{
break ;
}
2022-07-18 08:31:00 +00:00
g_Notify - > DisplayError ( stdstr_f ( " Unhandled R4300i opcode in FillSectionInfo 1 \n %s \n %X " , R4300iInstruction ( m_PC , m_Command . Value ) . NameAndParam ( ) . c_str ( ) , m_Command . Value ) . c_str ( ) ) ;
2016-01-27 09:11:59 +00:00
}
2022-08-08 10:52:51 +00:00
m_CodeBlock . Log ( " %s state: %X value: %X " , CRegName : : GPR [ 5 ] , m_Reg . GetMipsRegState ( 5 ) , m_Reg . GetMipsRegLo ( 5 ) ) ;
2016-01-27 09:11:59 +00:00
if ( Section - > m_DelaySlot )
{
2022-01-18 07:47:21 +00:00
if ( m_PipelineStage ! = PIPELINE_STAGE_NORMAL & & m_PipelineStage ! = PIPELINE_STAGE_END_BLOCK )
2016-01-27 09:11:59 +00:00
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_END_BLOCK ;
2016-01-27 09:11:59 +00:00
SetJumpRegSet ( Section , m_Reg ) ;
}
2022-10-10 00:22:17 +00:00
else
2016-11-27 20:34:02 +00:00
{
2022-01-18 07:47:21 +00:00
switch ( m_PipelineStage )
2016-01-27 09:11:59 +00:00
{
2022-01-18 07:47:21 +00:00
case PIPELINE_STAGE_NORMAL :
2016-01-27 09:11:59 +00:00
m_PC + = 4 ;
break ;
2022-01-18 07:47:21 +00:00
case PIPELINE_STAGE_DELAY_SLOT :
m_PipelineStage = PIPELINE_STAGE_DELAY_SLOT_DONE ;
2016-01-27 09:11:59 +00:00
m_PC + = 4 ;
if ( ( m_PC & 0xFFFFF000 ) ! = ( m_EnterSection - > m_EnterPC & 0xFFFFF000 ) )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
break ;
2022-01-18 07:47:21 +00:00
case PIPELINE_STAGE_LIKELY_DELAY_SLOT :
2016-01-27 09:11:59 +00:00
SetContinueRegSet ( Section , m_Reg ) ;
SetJumpRegSet ( Section , m_Reg ) ;
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_END_BLOCK ;
2016-11-27 20:34:02 +00:00
break ;
2022-01-18 07:47:21 +00:00
case PIPELINE_STAGE_DELAY_SLOT_DONE :
2016-01-27 09:11:59 +00:00
SetContinueRegSet ( Section , m_Reg ) ;
SetJumpRegSet ( Section , m_Reg ) ;
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_END_BLOCK ;
2016-11-27 20:34:02 +00:00
break ;
2022-01-18 07:47:21 +00:00
default :
2016-01-27 09:11:59 +00:00
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_END_BLOCK ;
2016-01-27 09:11:59 +00:00
break ;
}
}
if ( m_PC = = ContinueSectionPC )
{
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_END_BLOCK ;
2016-01-27 09:11:59 +00:00
SetContinueRegSet ( Section , m_Reg ) ;
}
if ( ( m_PC & 0xFFFFF000 ) ! = ( m_EnterSection - > m_EnterPC & 0xFFFFF000 ) )
{
2022-01-18 07:47:21 +00:00
if ( m_PipelineStage ! = PIPELINE_STAGE_END_BLOCK & & m_PipelineStage ! = PIPELINE_STAGE_NORMAL )
2016-01-27 09:11:59 +00:00
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
2022-01-18 07:47:21 +00:00
} while ( m_PipelineStage ! = PIPELINE_STAGE_END_BLOCK ) ;
2016-01-27 09:11:59 +00:00
2022-10-10 00:22:17 +00:00
if ( ! CheckLoopRegisterUsage ( Section - > m_ContinueSection ) )
{
return false ;
}
if ( ! CheckLoopRegisterUsage ( Section - > m_JumpSection ) )
{
return false ;
}
2016-01-27 09:11:59 +00:00
return true ;
}
2022-10-10 00:22:17 +00:00
bool LoopAnalysis : : SyncRegState ( CRegInfo & RegSet , const CRegInfo & SyncReg )
2016-01-27 09:11:59 +00:00
{
bool bChanged = false ;
for ( int x = 0 ; x < 32 ; x + + )
{
if ( RegSet . GetMipsRegState ( x ) ! = SyncReg . GetMipsRegState ( x ) )
{
2022-08-08 10:52:51 +00:00
m_CodeBlock . Log ( " %s: Clear state %s RegEnter State: %X Jump Reg State: %X " , __FUNCTION__ , CRegName : : GPR [ x ] , RegSet . GetMipsRegState ( x ) , SyncReg . GetMipsRegState ( x ) ) ;
2016-01-27 09:11:59 +00:00
RegSet . SetMipsRegState ( x , CRegInfo : : STATE_MODIFIED ) ;
bChanged = true ;
}
else if ( RegSet . IsConst ( x ) & & RegSet . Is32Bit ( x ) & & RegSet . GetMipsRegLo ( x ) ! = SyncReg . GetMipsRegLo ( x ) )
{
2022-08-08 10:52:51 +00:00
m_CodeBlock . Log ( " %s: Clear state %s RegEnter State: %X Jump Reg State: %X " , __FUNCTION__ , CRegName : : GPR [ x ] , RegSet . GetMipsRegState ( x ) , SyncReg . GetMipsRegState ( x ) ) ;
2016-01-27 09:11:59 +00:00
RegSet . SetMipsRegState ( x , CRegInfo : : STATE_MODIFIED ) ;
bChanged = true ;
}
else if ( RegSet . IsConst ( x ) & & RegSet . Is64Bit ( x ) & & RegSet . GetMipsReg_S ( x ) ! = SyncReg . GetMipsReg_S ( x ) )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
}
return bChanged ;
}
2022-10-10 00:22:17 +00:00
void LoopAnalysis : : SetJumpRegSet ( CCodeSection * Section , const CRegInfo & Reg )
2016-01-27 09:11:59 +00:00
{
RegisterMap : : iterator itr = m_JumpRegisters . find ( Section - > m_SectionID ) ;
if ( itr ! = m_JumpRegisters . end ( ) )
{
* ( itr - > second ) = Reg ;
}
else
{
m_JumpRegisters . insert ( RegisterMap : : value_type ( Section - > m_SectionID , new CRegInfo ( Reg ) ) ) ;
}
}
2022-10-10 00:22:17 +00:00
void LoopAnalysis : : SetContinueRegSet ( CCodeSection * Section , const CRegInfo & Reg )
2016-01-27 09:11:59 +00:00
{
RegisterMap : : iterator itr = m_ContinueRegisters . find ( Section - > m_SectionID ) ;
if ( itr ! = m_ContinueRegisters . end ( ) )
{
* ( itr - > second ) = Reg ;
}
else
{
m_ContinueRegisters . insert ( RegisterMap : : value_type ( Section - > m_SectionID , new CRegInfo ( Reg ) ) ) ;
}
}
void LoopAnalysis : : SPECIAL_SLL ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
void LoopAnalysis : : SPECIAL_SRL ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
void LoopAnalysis : : SPECIAL_SRA ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
void LoopAnalysis : : SPECIAL_SLLV ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rt = = m_Command . rd | | m_Command . rs = = m_Command . rd )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rt ) & & m_Reg . IsConst ( m_Command . rs ) )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_CONST_32_SIGN ) ;
m_Reg . SetMipsRegLo ( m_Command . rd , m_Reg . GetMipsRegLo ( m_Command . rt ) < < ( m_Reg . GetMipsRegLo ( m_Command . rs ) & 0x1F ) ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
}
void LoopAnalysis : : SPECIAL_SRLV ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rt = = m_Command . rd | | m_Command . rs = = m_Command . rd )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rt ) & & m_Reg . IsConst ( m_Command . rs ) )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_CONST_32_SIGN ) ;
m_Reg . SetMipsRegLo ( m_Command . rd , m_Reg . GetMipsRegLo ( m_Command . rt ) > > ( m_Reg . GetMipsRegLo ( m_Command . rs ) & 0x1F ) ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
}
void LoopAnalysis : : SPECIAL_SRAV ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rt = = m_Command . rd | | m_Command . rs = = m_Command . rd )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rt ) & & m_Reg . IsConst ( m_Command . rs ) )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_CONST_32_SIGN ) ;
m_Reg . SetMipsRegLo ( m_Command . rd , m_Reg . GetMipsRegLo_S ( m_Command . rt ) > > ( m_Reg . GetMipsRegLo ( m_Command . rs ) & 0x1F ) ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
}
void LoopAnalysis : : SPECIAL_JR ( )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_DELAY_SLOT ;
2016-01-27 09:11:59 +00:00
}
void LoopAnalysis : : SPECIAL_JALR ( )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_DELAY_SLOT ;
2016-01-27 09:11:59 +00:00
}
void LoopAnalysis : : SPECIAL_SYSCALL ( CCodeSection * Section )
{
# ifdef CHECKED_BUILD
2021-04-12 11:35:39 +00:00
if ( Section - > m_ContinueSection ! = nullptr & &
2016-01-27 09:11:59 +00:00
Section - > m_Cont . TargetPC ! = ( uint32_t ) - 1 )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
2021-04-12 11:35:39 +00:00
if ( Section - > m_JumpSection ! = nullptr & &
2016-01-27 09:11:59 +00:00
Section - > m_Jump . TargetPC ! = ( uint32_t ) - 1 )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
# else
Section = Section ;
# endif
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_END_BLOCK ;
2016-01-27 09:11:59 +00:00
m_PC - = 4 ;
}
void LoopAnalysis : : SPECIAL_BREAK ( CCodeSection * Section )
{
# ifdef CHECKED_BUILD
2021-04-12 11:35:39 +00:00
if ( Section - > m_ContinueSection ! = nullptr & &
2016-01-27 09:11:59 +00:00
Section - > m_Cont . TargetPC ! = ( uint32_t ) - 1 )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
2021-04-12 11:35:39 +00:00
if ( Section - > m_JumpSection ! = nullptr & &
2016-01-27 09:11:59 +00:00
Section - > m_Jump . TargetPC ! = ( uint32_t ) - 1 )
{
g_Notify - > BreakPoint ( __FILE__ , __LINE__ ) ;
}
# else
Section = Section ;
# endif
2022-01-18 07:47:21 +00:00
m_PipelineStage = PIPELINE_STAGE_END_BLOCK ;
2016-01-27 09:11:59 +00:00
m_PC - = 4 ;
}
void LoopAnalysis : : SPECIAL_MFHI ( )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
void LoopAnalysis : : SPECIAL_MTHI ( )
{
}
void LoopAnalysis : : SPECIAL_MFLO ( )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
void LoopAnalysis : : SPECIAL_MTLO ( )
{
}
void LoopAnalysis : : SPECIAL_DSLLV ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rt = = m_Command . rd | | m_Command . rs = = m_Command . rd )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rt ) & & m_Reg . IsConst ( m_Command . rs ) )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_CONST_64 ) ;
m_Reg . SetMipsReg ( m_Command . rd , m_Reg . Is64Bit ( m_Command . rt ) ? m_Reg . GetMipsReg ( m_Command . rt ) : ( uint64_t ) m_Reg . GetMipsRegLo_S ( m_Command . rt ) < < ( m_Reg . GetMipsRegLo ( m_Command . rs ) & 0x3F ) ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
}
void LoopAnalysis : : SPECIAL_DSRLV ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rt = = m_Command . rd | | m_Command . rs = = m_Command . rd )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rt ) & & m_Reg . IsConst ( m_Command . rs ) )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_CONST_64 ) ;
m_Reg . SetMipsReg ( m_Command . rd , m_Reg . Is64Bit ( m_Command . rt ) ? m_Reg . GetMipsReg ( m_Command . rt ) : ( uint64_t ) m_Reg . GetMipsRegLo_S ( m_Command . rt ) > > ( m_Reg . GetMipsRegLo ( m_Command . rs ) & 0x3F ) ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
}
void LoopAnalysis : : SPECIAL_DSRAV ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rt = = m_Command . rd | | m_Command . rs = = m_Command . rd )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
2022-10-10 00:22:17 +00:00
if ( m_Reg . IsConst ( m_Command . rt ) & & m_Reg . IsConst ( m_Command . rs ) )
{
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_CONST_64 ) ;
m_Reg . SetMipsReg ( m_Command . rd , m_Reg . Is64Bit ( m_Command . rt ) ? m_Reg . GetMipsReg_S ( m_Command . rt ) : ( int64_t ) m_Reg . GetMipsRegLo_S ( m_Command . rt ) > > ( m_Reg . GetMipsRegLo ( m_Command . rs ) & 0x3F ) ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
}
void LoopAnalysis : : SPECIAL_ADD ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
void LoopAnalysis : : SPECIAL_ADDU ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
void LoopAnalysis : : SPECIAL_SUB ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
void LoopAnalysis : : SPECIAL_SUBU ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
void LoopAnalysis : : SPECIAL_AND ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
void LoopAnalysis : : SPECIAL_OR ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
void LoopAnalysis : : SPECIAL_XOR ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
void LoopAnalysis : : SPECIAL_NOR ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
void LoopAnalysis : : SPECIAL_SLT ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
void LoopAnalysis : : SPECIAL_SLTU ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
void LoopAnalysis : : SPECIAL_DADD ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rt = = m_Command . rd | | m_Command . rs = = m_Command . rd )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rt ) & & m_Reg . IsConst ( m_Command . rs ) )
{
m_Reg . SetMipsReg ( m_Command . rd ,
2022-10-10 00:22:17 +00:00
( m_Reg . Is64Bit ( m_Command . rs ) ? m_Reg . GetMipsReg ( m_Command . rs ) : ( int64_t ) m_Reg . GetMipsRegLo_S ( m_Command . rs ) ) +
( m_Reg . Is64Bit ( m_Command . rt ) ? m_Reg . GetMipsReg ( m_Command . rt ) : ( int64_t ) m_Reg . GetMipsRegLo_S ( m_Command . rt ) ) ) ;
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_CONST_64 ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
}
void LoopAnalysis : : SPECIAL_DADDU ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rt = = m_Command . rd | | m_Command . rs = = m_Command . rd )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rt ) & & m_Reg . IsConst ( m_Command . rs ) )
{
m_Reg . SetMipsReg ( m_Command . rd ,
2022-10-10 00:22:17 +00:00
( m_Reg . Is64Bit ( m_Command . rs ) ? m_Reg . GetMipsReg ( m_Command . rs ) : ( int64_t ) m_Reg . GetMipsRegLo_S ( m_Command . rs ) ) +
( m_Reg . Is64Bit ( m_Command . rt ) ? m_Reg . GetMipsReg ( m_Command . rt ) : ( int64_t ) m_Reg . GetMipsRegLo_S ( m_Command . rt ) ) ) ;
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_CONST_64 ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
}
void LoopAnalysis : : SPECIAL_DSUB ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rt = = m_Command . rd | | m_Command . rs = = m_Command . rd )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rt ) & & m_Reg . IsConst ( m_Command . rs ) )
{
m_Reg . SetMipsReg ( m_Command . rd ,
2022-10-10 00:22:17 +00:00
( m_Reg . Is64Bit ( m_Command . rs ) ? m_Reg . GetMipsReg ( m_Command . rs ) : ( int64_t ) m_Reg . GetMipsRegLo_S ( m_Command . rs ) ) -
( m_Reg . Is64Bit ( m_Command . rt ) ? m_Reg . GetMipsReg ( m_Command . rt ) : ( int64_t ) m_Reg . GetMipsRegLo_S ( m_Command . rt ) ) ) ;
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_CONST_64 ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
}
void LoopAnalysis : : SPECIAL_DSUBU ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rt = = m_Command . rd | | m_Command . rs = = m_Command . rd )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rt ) & & m_Reg . IsConst ( m_Command . rs ) )
{
m_Reg . SetMipsReg ( m_Command . rd ,
2022-10-10 00:22:17 +00:00
( m_Reg . Is64Bit ( m_Command . rs ) ? m_Reg . GetMipsReg ( m_Command . rs ) : ( int64_t ) m_Reg . GetMipsRegLo_S ( m_Command . rs ) ) -
( m_Reg . Is64Bit ( m_Command . rt ) ? m_Reg . GetMipsReg ( m_Command . rt ) : ( int64_t ) m_Reg . GetMipsRegLo_S ( m_Command . rt ) ) ) ;
2016-01-27 09:11:59 +00:00
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_CONST_64 ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
}
void LoopAnalysis : : SPECIAL_DSLL ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rt = = m_Command . rd )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rt ) )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_CONST_64 ) ;
m_Reg . SetMipsReg ( m_Command . rd , m_Reg . Is64Bit ( m_Command . rt ) ? m_Reg . GetMipsReg ( m_Command . rt ) : ( int64_t ) m_Reg . GetMipsRegLo_S ( m_Command . rt ) < < m_Command . sa ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
}
void LoopAnalysis : : SPECIAL_DSRL ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rt = = m_Command . rd )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rt ) )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_CONST_64 ) ;
m_Reg . SetMipsReg ( m_Command . rd , m_Reg . Is64Bit ( m_Command . rt ) ? m_Reg . GetMipsReg ( m_Command . rt ) : ( uint64_t ) m_Reg . GetMipsRegLo_S ( m_Command . rt ) > > m_Command . sa ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
}
void LoopAnalysis : : SPECIAL_DSRA ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rt = = m_Command . rd )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rt ) )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_CONST_64 ) ;
m_Reg . SetMipsReg_S ( m_Command . rd , m_Reg . Is64Bit ( m_Command . rt ) ? m_Reg . GetMipsReg_S ( m_Command . rt ) : ( int64_t ) m_Reg . GetMipsRegLo_S ( m_Command . rt ) > > m_Command . sa ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
}
void LoopAnalysis : : SPECIAL_DSLL32 ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rt = = m_Command . rd )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rt ) )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_CONST_64 ) ;
m_Reg . SetMipsReg ( m_Command . rd , m_Reg . GetMipsRegLo ( m_Command . rt ) < < ( m_Command . sa + 32 ) ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
}
void LoopAnalysis : : SPECIAL_DSRL32 ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rt = = m_Command . rd )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rt ) )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_CONST_32_SIGN ) ;
m_Reg . SetMipsRegLo ( m_Command . rd , ( uint32_t ) ( m_Reg . GetMipsReg ( m_Command . rt ) > > ( m_Command . sa + 32 ) ) ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
}
void LoopAnalysis : : SPECIAL_DSRA32 ( )
{
2022-10-10 00:22:17 +00:00
if ( m_Command . rd = = 0 )
{
return ;
}
2016-01-27 09:11:59 +00:00
if ( m_Command . rt = = m_Command . rd )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
if ( m_Reg . IsConst ( m_Command . rt ) )
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_CONST_32_SIGN ) ;
m_Reg . SetMipsRegLo ( m_Command . rd , ( uint32_t ) ( m_Reg . GetMipsReg_S ( m_Command . rt ) > > ( m_Command . sa + 32 ) ) ) ;
}
else
{
m_Reg . SetMipsRegState ( m_Command . rd , CRegInfo : : STATE_MODIFIED ) ;
}
2015-12-15 06:01:06 +00:00
}