PPCAnalyst: EvaluateBranchTarget improved
This commit is contained in:
parent
ef4a1f0ec7
commit
95d0a48759
|
@ -53,7 +53,15 @@ static u32 EvaluateBranchTarget(UGeckoInstruction instr, u32 pc)
|
|||
{
|
||||
switch (instr.OPCD)
|
||||
{
|
||||
case 18: // branch instruction
|
||||
case 16: // bcx - Branch Conditional instructions
|
||||
{
|
||||
u32 target = SignExt16(instr.BD << 2);
|
||||
if (!instr.AA)
|
||||
target += pc;
|
||||
|
||||
return target;
|
||||
}
|
||||
case 18: // bx - Branch instructions
|
||||
{
|
||||
u32 target = SignExt26(instr.LI << 2);
|
||||
if (!instr.AA)
|
||||
|
@ -83,11 +91,10 @@ bool AnalyzeFunction(u32 startAddr, Symbol& func, int max_size)
|
|||
func.callers.clear();
|
||||
func.size = 0;
|
||||
func.flags = FFLAG_LEAF;
|
||||
u32 addr = startAddr;
|
||||
|
||||
u32 farthestInternalBranchTarget = startAddr;
|
||||
int numInternalBranches = 0;
|
||||
while (true)
|
||||
for (u32 addr = startAddr; true; addr += 4)
|
||||
{
|
||||
func.size += 4;
|
||||
if (func.size >= CODEBUFFER_SIZE * 4 || !PowerPC::HostIsRAMAddress(addr)) // weird
|
||||
|
@ -110,22 +117,19 @@ bool AnalyzeFunction(u32 startAddr, Symbol& func, int max_size)
|
|||
// 4e800021 is blrl, not the end of a function
|
||||
if (instr.hex == 0x4e800020 || instr.hex == 0x4C000064)
|
||||
{
|
||||
// Not this one, continue..
|
||||
if (farthestInternalBranchTarget > addr)
|
||||
{
|
||||
// bah, not this one, continue..
|
||||
}
|
||||
else
|
||||
{
|
||||
// a final blr!
|
||||
// We're done! Looks like we have a neat valid function. Perfect.
|
||||
// Let's calc the checksum and get outta here
|
||||
func.address = startAddr;
|
||||
func.analyzed = true;
|
||||
func.hash = HashSignatureDB::ComputeCodeChecksum(startAddr, addr);
|
||||
if (numInternalBranches == 0)
|
||||
func.flags |= FFLAG_STRAIGHT;
|
||||
return true;
|
||||
}
|
||||
continue;
|
||||
|
||||
// A final blr!
|
||||
// We're done! Looks like we have a neat valid function. Perfect.
|
||||
// Let's calc the checksum and get outta here
|
||||
func.address = startAddr;
|
||||
func.analyzed = true;
|
||||
func.hash = HashSignatureDB::ComputeCodeChecksum(startAddr, addr);
|
||||
if (numInternalBranches == 0)
|
||||
func.flags |= FFLAG_STRAIGHT;
|
||||
return true;
|
||||
}
|
||||
/*
|
||||
else if ((instr.hex & 0xFC000000) == (0x4b000000 & 0xFC000000) && !instr.LK)
|
||||
|
@ -155,36 +159,31 @@ bool AnalyzeFunction(u32 startAddr, Symbol& func, int max_size)
|
|||
}
|
||||
else
|
||||
{
|
||||
if (instr.OPCD == 16)
|
||||
u32 target = EvaluateBranchTarget(instr, addr);
|
||||
if (target == INVALID_BRANCH_TARGET)
|
||||
continue;
|
||||
|
||||
if (instr.LK)
|
||||
{
|
||||
u32 target = SignExt16(instr.BD << 2);
|
||||
|
||||
if (!instr.AA)
|
||||
target += addr;
|
||||
|
||||
if (target > farthestInternalBranchTarget && !instr.LK)
|
||||
// Found a branch-n-link
|
||||
func.calls.emplace_back(target, addr);
|
||||
func.flags &= ~FFLAG_LEAF;
|
||||
}
|
||||
else if (instr.OPCD == 16)
|
||||
{
|
||||
// Found a conditional branch
|
||||
if (target > farthestInternalBranchTarget)
|
||||
{
|
||||
farthestInternalBranchTarget = target;
|
||||
}
|
||||
numInternalBranches++;
|
||||
}
|
||||
else
|
||||
{
|
||||
u32 target = EvaluateBranchTarget(instr, addr);
|
||||
if (target != INVALID_BRANCH_TARGET && instr.LK)
|
||||
{
|
||||
// we found a branch-n-link!
|
||||
func.calls.emplace_back(target, addr);
|
||||
func.flags &= ~FFLAG_LEAF;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
return false;
|
||||
}
|
||||
addr += 4;
|
||||
}
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in New Issue