mirror of
https://gitlab.com/freepascal.org/lazarus/lazarus.git
synced 2025-04-06 17:57:54 +02:00
FpDebug: improve TX86AsmDecoder.UnwindFrame
This commit is contained in:
parent
8e2532bdfc
commit
ac1a3e6fd2
@ -5536,51 +5536,174 @@ var
|
||||
const
|
||||
MAX_SEARCH_ADDR = 8000;
|
||||
MAX_SEARCH_CNT = 400;
|
||||
MAX_ADDR_DONE_BLOCKS = 10;
|
||||
MAX_FORWARD_ADDR = 5;
|
||||
var
|
||||
MaxAddr, MaxAddr2, StartAddr, StartStack, Tmp: TDBGPtr;
|
||||
ConditionalForwardAddr, BackwardJumpAddress: TDBGPtr;
|
||||
AddressDoneBlocks: array [0..MAX_ADDR_DONE_BLOCKS] of record
|
||||
First, Last: TDBGPtr;
|
||||
end;
|
||||
CurAddressDoneBlock: integer;
|
||||
MaxAddrCurrentBlock: TDBGPtr;
|
||||
ConditionalForwardAddr: array [0..MAX_FORWARD_ADDR+1] of TDBGPtr;
|
||||
CurConditionalForwardAddr: integer;
|
||||
|
||||
function AddrWasDone(AnAddr: TDBGPtr): boolean; inline;
|
||||
var
|
||||
i: Integer;
|
||||
begin
|
||||
i := CurAddressDoneBlock;
|
||||
Result := (AnAddr >= AddressDoneBlocks[i].First) and (AnAddr < NewAddr);
|
||||
while (not Result) and (i > 0) do begin
|
||||
dec(i);
|
||||
Result := (AnAddr >= AddressDoneBlocks[i].First) and (AnAddr < AddressDoneBlocks[i].Last);
|
||||
end;
|
||||
end;
|
||||
|
||||
function NextDoneAddrFor(AnAddr: TDBGPtr): TDBGPtr; inline;
|
||||
var
|
||||
i: Integer;
|
||||
begin
|
||||
Result := high(TDBGPtr);
|
||||
i := CurAddressDoneBlock - 1;
|
||||
while (i >= 0) do begin
|
||||
if (AnAddr < AddressDoneBlocks[i].First) and
|
||||
(Result > AddressDoneBlocks[i].First)
|
||||
then
|
||||
Result := AddressDoneBlocks[i].First;
|
||||
dec(i);
|
||||
end;
|
||||
end;
|
||||
|
||||
procedure FinishCurAddrBlock; inline;
|
||||
begin
|
||||
if CurAddressDoneBlock < MAX_ADDR_DONE_BLOCKS then begin
|
||||
AddressDoneBlocks[CurAddressDoneBlock].Last := NewAddr;
|
||||
inc(CurAddressDoneBlock);
|
||||
AddressDoneBlocks[CurAddressDoneBlock].First := high(TDBGPtr);
|
||||
end
|
||||
else
|
||||
if NewAddr > AddressDoneBlocks[CurAddressDoneBlock].Last then begin
|
||||
AddressDoneBlocks[CurAddressDoneBlock].Last := NewAddr;
|
||||
end;
|
||||
end;
|
||||
procedure StartNextAddrBlock; inline;
|
||||
begin
|
||||
if CurAddressDoneBlock < MAX_ADDR_DONE_BLOCKS then
|
||||
AddressDoneBlocks[CurAddressDoneBlock].First := NewAddr
|
||||
else
|
||||
if NewAddr < AddressDoneBlocks[CurAddressDoneBlock].First then
|
||||
AddressDoneBlocks[CurAddressDoneBlock].First := NewAddr;
|
||||
|
||||
MaxAddrCurrentBlock := NextDoneAddrFor(NewAddr);
|
||||
end;
|
||||
|
||||
procedure CheckConditionalForwAddr; inline;
|
||||
var
|
||||
j, i: Integer;
|
||||
begin
|
||||
j := 0;
|
||||
for i := 0 to CurConditionalForwardAddr do
|
||||
if (ConditionalForwardAddr[i] >= NewAddr) or
|
||||
(ConditionalForwardAddr[i] < AddressDoneBlocks[CurAddressDoneBlock].First)
|
||||
then begin
|
||||
ConditionalForwardAddr[j] := ConditionalForwardAddr[i];
|
||||
inc(j);
|
||||
end;
|
||||
CurConditionalForwardAddr := j - 1;
|
||||
end;
|
||||
|
||||
procedure AddConditionalForwAddr(AnAddr: TDBGPtr); inline;
|
||||
var
|
||||
i, j: Integer;
|
||||
begin
|
||||
i := CurConditionalForwardAddr;
|
||||
while (i >= 0) and (AnAddr <> ConditionalForwardAddr[i]) do
|
||||
dec(i);
|
||||
if i >= 0 then
|
||||
exit;
|
||||
|
||||
if CurConditionalForwardAddr < MAX_FORWARD_ADDR then begin
|
||||
inc(CurConditionalForwardAddr);
|
||||
ConditionalForwardAddr[CurConditionalForwardAddr] := AnAddr;
|
||||
end
|
||||
else begin
|
||||
j := 0;
|
||||
for i := 0 to CurConditionalForwardAddr do
|
||||
if (ConditionalForwardAddr[i] >= NewAddr) or
|
||||
(ConditionalForwardAddr[i] < AddressDoneBlocks[CurAddressDoneBlock].First)
|
||||
then begin
|
||||
ConditionalForwardAddr[j] := ConditionalForwardAddr[i];
|
||||
inc(j);
|
||||
end;
|
||||
CurConditionalForwardAddr := j - 1;
|
||||
if CurConditionalForwardAddr < MAX_FORWARD_ADDR then
|
||||
inc(CurConditionalForwardAddr);
|
||||
ConditionalForwardAddr[CurConditionalForwardAddr] := AnAddr;
|
||||
end;
|
||||
end;
|
||||
|
||||
var
|
||||
MaxAddr, StartStack, Tmp: TDBGPtr;
|
||||
BackwardJumpAddress: TDBGPtr;
|
||||
Cnt: Integer;
|
||||
instr: TX86AsmInstruction;
|
||||
RSize: Cardinal;
|
||||
Val: Int64;
|
||||
ClearRecValList: Boolean;
|
||||
ClearRecValList, ForceDifferentBranch: Boolean;
|
||||
FullName: String;
|
||||
begin
|
||||
Result := False;
|
||||
NewAddr := AnAddress;
|
||||
NewStack := AStackPtr;
|
||||
NewFrame := AFramePtr;
|
||||
StartAddr := AnAddress;
|
||||
StartStack := AStackPtr;
|
||||
ConditionalForwardAddr := 0;
|
||||
CurConditionalForwardAddr := -1;
|
||||
BackwardJumpAddress := 0;
|
||||
|
||||
{$PUSH}{$R-}{$Q-}
|
||||
MaxAddr := AnAddress + MAX_SEARCH_ADDR;
|
||||
{$POP}
|
||||
MaxAddr2 := MaxAddr;
|
||||
Cnt := MAX_SEARCH_CNT;
|
||||
if AQuick then Cnt := 10;
|
||||
|
||||
CurAddressDoneBlock := 0;
|
||||
AddressDoneBlocks[CurAddressDoneBlock].First := AnAddress;
|
||||
MaxAddrCurrentBlock := MaxAddr;
|
||||
|
||||
ClearRecValList := False;
|
||||
while (NewAddr < MaxAddr) and (Cnt > 0) do begin
|
||||
ForceDifferentBranch := False;
|
||||
while (Cnt > 0) do begin
|
||||
if ClearRecValList then ARegisterValueList.Clear;
|
||||
|
||||
if NewAddr > MaxAddr2 then begin
|
||||
if (ConditionalForwardAddr > 0) and (BackwardJumpAddress > 0) and
|
||||
(ConditionalForwardAddr >= BackwardJumpAddress)
|
||||
then begin
|
||||
NewAddr := ConditionalForwardAddr;
|
||||
MaxAddr2 := MaxAddr;
|
||||
if ForceDifferentBranch or (NewAddr >= MaxAddr) or( NewAddr > MaxAddrCurrentBlock) then begin
|
||||
CheckConditionalForwAddr;
|
||||
FinishCurAddrBlock;
|
||||
while (CurConditionalForwardAddr >= 0) and
|
||||
AddrWasDone(ConditionalForwardAddr[CurConditionalForwardAddr])
|
||||
do
|
||||
dec(CurConditionalForwardAddr);
|
||||
if (CurConditionalForwardAddr >= 0) then begin
|
||||
NewAddr := ConditionalForwardAddr[CurConditionalForwardAddr];
|
||||
dec(CurConditionalForwardAddr);
|
||||
end
|
||||
else
|
||||
if (BackwardJumpAddress > 0) and (not AddrWasDone(BackwardJumpAddress)) then begin
|
||||
NewAddr := BackwardJumpAddress;
|
||||
BackwardJumpAddress := 0;
|
||||
end
|
||||
else
|
||||
exit;
|
||||
StartNextAddrBlock;
|
||||
ARegisterValueList.Clear;
|
||||
end;
|
||||
ForceDifferentBranch := False;
|
||||
|
||||
dec(Cnt);
|
||||
instr := TX86AsmInstruction(GetInstructionInfo(NewAddr));
|
||||
if instr.InstructionLength <= 0 then
|
||||
exit;
|
||||
if instr.InstructionLength <= 0 then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
{$PUSH}{$R-}{$Q-}
|
||||
NewAddr := NewAddr + instr.InstructionLength;
|
||||
{$POP}
|
||||
@ -5624,7 +5747,8 @@ begin
|
||||
IsRegister(instr.X86Instruction.Operand[1].Value, 'bp') or
|
||||
IsRegister(instr.X86Instruction.Operand[1].Value, 'sp')
|
||||
then begin
|
||||
exit; // false
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
{$PUSH}{$R-}{$Q-}
|
||||
NewStack := NewStack - RegisterSize(instr.X86Instruction.Operand[1].Value);
|
||||
@ -5633,8 +5757,10 @@ begin
|
||||
OPpusha:
|
||||
begin
|
||||
ClearRecValList := False;
|
||||
if (FProcess.Mode <> dm32) or (instr.X86Instruction.OpCode.Suffix <> OPSx_d) then
|
||||
exit;
|
||||
if (FProcess.Mode <> dm32) or (instr.X86Instruction.OpCode.Suffix <> OPSx_d) then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
// push 8 registers
|
||||
{$PUSH}{$R-}{$Q-}
|
||||
NewStack := NewStack - (8*4);
|
||||
@ -5643,19 +5769,28 @@ begin
|
||||
OPpushf:
|
||||
begin
|
||||
ClearRecValList := False;
|
||||
exit; // false
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
OPpopa, OPpopad:
|
||||
exit; // false
|
||||
begin
|
||||
ClearRecValList := False;
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
OPpop:
|
||||
begin
|
||||
if instr.X86Instruction.OperCnt <> 1 then
|
||||
exit;
|
||||
if instr.X86Instruction.OperCnt <> 1 then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
ClearRecValList := False;
|
||||
ClearRegister(instr.X86Instruction.Operand[1].Value);
|
||||
if not(instr.X86Instruction.Operand[1].Value[1] in ['e', 'r'])
|
||||
then
|
||||
exit; // false
|
||||
then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
if IsRegister(instr.X86Instruction.Operand[1].Value, 'bp')
|
||||
then begin
|
||||
if NewStack < StartStack then
|
||||
@ -5696,15 +5831,19 @@ begin
|
||||
end;
|
||||
OPmov:
|
||||
begin
|
||||
if instr.X86Instruction.OperCnt <> 2 then
|
||||
exit;
|
||||
if instr.X86Instruction.OperCnt <> 2 then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
ClearRecValList := False;
|
||||
|
||||
if IsRegister(instr.X86Instruction.Operand[1].Value, 'sp') and
|
||||
not(ofMemory in Instr.X86Instruction.Operand[1].Flags)
|
||||
then begin
|
||||
if not ValueFromOperand(instr.X86Instruction.Operand[2], Tmp) then
|
||||
exit;
|
||||
if not ValueFromOperand(instr.X86Instruction.Operand[2], Tmp) then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
NewStack := Tmp;
|
||||
end
|
||||
else
|
||||
@ -5728,16 +5867,20 @@ begin
|
||||
end;
|
||||
OPlea:
|
||||
begin
|
||||
if instr.X86Instruction.OperCnt <> 2 then
|
||||
exit;
|
||||
if instr.X86Instruction.OperCnt <> 2 then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
ClearRecValList := False;
|
||||
if IsRegister(instr.X86Instruction.Operand[1].Value, 'bp')
|
||||
then begin
|
||||
if (Instr.X86Instruction.Operand[2].ByteCount = 0) or
|
||||
(Instr.X86Instruction.Operand[2].ByteCount2 <> 0) or
|
||||
not(ofMemory in Instr.X86Instruction.Operand[2].Flags)
|
||||
then
|
||||
exit;
|
||||
then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
|
||||
Val := ValueFromMem(CurAddr[Instr.X86Instruction.Operand[2].CodeIndex], Instr.X86Instruction.Operand[2].ByteCount, Instr.X86Instruction.Operand[2].FormatFlags);
|
||||
if (IsRegister(instr.X86Instruction.Operand[2].Value, 'sp%s')) then begin
|
||||
@ -5762,8 +5905,10 @@ begin
|
||||
if (Instr.X86Instruction.Operand[2].ByteCount = 0) or
|
||||
(Instr.X86Instruction.Operand[2].ByteCount2 <> 0) or
|
||||
not(ofMemory in Instr.X86Instruction.Operand[2].Flags)
|
||||
then
|
||||
exit;
|
||||
then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
|
||||
Val := ValueFromMem(CurAddr[Instr.X86Instruction.Operand[2].CodeIndex], Instr.X86Instruction.Operand[2].ByteCount, Instr.X86Instruction.Operand[2].FormatFlags);
|
||||
if (IsRegister(instr.X86Instruction.Operand[2].Value, 'sp%s')) then begin
|
||||
@ -5781,8 +5926,10 @@ begin
|
||||
if ValueFromOperand(instr.X86Instruction.Operand[2], Tmp, True) then begin
|
||||
NewStack := Tmp;
|
||||
end
|
||||
else
|
||||
exit;
|
||||
else begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
end
|
||||
else begin
|
||||
FullName := LowerCase(FullRegisterName(instr.X86Instruction.Operand[1].Value));
|
||||
@ -5803,16 +5950,20 @@ begin
|
||||
end;
|
||||
OPadd:
|
||||
begin
|
||||
if instr.X86Instruction.OperCnt <> 2 then
|
||||
exit;
|
||||
if instr.X86Instruction.OperCnt <> 2 then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
ClearRecValList := False;
|
||||
ClearRegister(instr.X86Instruction.Operand[1].Value);
|
||||
|
||||
if IsRegister(instr.X86Instruction.Operand[1].Value, 'sp') and
|
||||
not(ofMemory in Instr.X86Instruction.Operand[1].Flags)
|
||||
then begin
|
||||
if not ValueFromOperand(instr.X86Instruction.Operand[2], Tmp) then
|
||||
exit;
|
||||
if not ValueFromOperand(instr.X86Instruction.Operand[2], Tmp) then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
{$PUSH}{$R-}{$Q-}
|
||||
NewStack := NewStack + int64(Tmp);
|
||||
{$POP}
|
||||
@ -5820,8 +5971,10 @@ begin
|
||||
if IsRegister(instr.X86Instruction.Operand[1].Value, 'bp') and
|
||||
not(ofMemory in Instr.X86Instruction.Operand[1].Flags)
|
||||
then begin
|
||||
if not ValueFromOperand(instr.X86Instruction.Operand[2], Tmp) then
|
||||
exit;
|
||||
if not ValueFromOperand(instr.X86Instruction.Operand[2], Tmp) then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
{$PUSH}{$R-}{$Q-}
|
||||
NewFrame := NewFrame + int64(Tmp);
|
||||
{$POP}
|
||||
@ -5829,16 +5982,20 @@ begin
|
||||
end;
|
||||
OPsub:
|
||||
begin
|
||||
if instr.X86Instruction.OperCnt <> 2 then
|
||||
exit;
|
||||
if instr.X86Instruction.OperCnt <> 2 then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
ClearRecValList := False;
|
||||
ClearRegister(instr.X86Instruction.Operand[1].Value);
|
||||
|
||||
if IsRegister(instr.X86Instruction.Operand[1].Value, 'sp') and
|
||||
not(ofMemory in Instr.X86Instruction.Operand[1].Flags)
|
||||
then begin
|
||||
if not ValueFromOperand(instr.X86Instruction.Operand[2], Tmp) then
|
||||
exit;
|
||||
if not ValueFromOperand(instr.X86Instruction.Operand[2], Tmp) then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
{$PUSH}{$R-}{$Q-}
|
||||
NewStack := NewStack - int64(Tmp);
|
||||
{$POP}
|
||||
@ -5846,8 +6003,10 @@ begin
|
||||
if IsRegister(instr.X86Instruction.Operand[1].Value, 'bp') and
|
||||
not(ofMemory in Instr.X86Instruction.Operand[1].Flags)
|
||||
then begin
|
||||
if not ValueFromOperand(instr.X86Instruction.Operand[2], Tmp) then
|
||||
exit;
|
||||
if not ValueFromOperand(instr.X86Instruction.Operand[2], Tmp) then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
{$PUSH}{$R-}{$Q-}
|
||||
NewFrame := NewFrame - int64(Tmp);
|
||||
{$POP}
|
||||
@ -5862,47 +6021,74 @@ begin
|
||||
then
|
||||
begin
|
||||
Val := ValueFromMem(CurAddr[Instr.X86Instruction.Operand[1].CodeIndex], Instr.X86Instruction.Operand[1].ByteCount, Instr.X86Instruction.Operand[1].FormatFlags);
|
||||
{$PUSH}{$R-}{$Q-}
|
||||
Tmp := NewAddr + Val;
|
||||
{$POP}
|
||||
if Val > 0 then begin
|
||||
{$PUSH}{$R-}{$Q-}
|
||||
Tmp := NewAddr + Val;
|
||||
{$POP}
|
||||
if (Tmp > ConditionalForwardAddr) and (Tmp < MaxAddr) then
|
||||
ConditionalForwardAddr := Tmp;
|
||||
if (Tmp < MaxAddr) and (not AddrWasDone(Tmp)) then
|
||||
AddConditionalForwAddr(Tmp);
|
||||
end
|
||||
else begin
|
||||
if ((BackwardJumpAddress = 0) or (Tmp < BackwardJumpAddress)) and
|
||||
(not AddrWasDone(Tmp))
|
||||
then
|
||||
BackwardJumpAddress := Tmp;
|
||||
end;
|
||||
end;
|
||||
OPjmp:
|
||||
begin
|
||||
if AQuick then
|
||||
exit;
|
||||
if instr.X86Instruction.OperCnt <> 1 then
|
||||
exit;
|
||||
if (instr.X86Instruction.Operand[1].Value <> '%s') then
|
||||
exit; // false
|
||||
if instr.X86Instruction.OperCnt <> 1 then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
if (instr.X86Instruction.Operand[1].Value <> '%s') then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
if (Instr.X86Instruction.Operand[1].ByteCount = 0) or
|
||||
(Instr.X86Instruction.Operand[1].ByteCount2 <> 0)
|
||||
then
|
||||
exit;
|
||||
then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
|
||||
Val := ValueFromMem(CurAddr[Instr.X86Instruction.Operand[1].CodeIndex], Instr.X86Instruction.Operand[1].ByteCount, Instr.X86Instruction.Operand[1].FormatFlags);
|
||||
{$PUSH}{$R-}{$Q-}
|
||||
Tmp := NewAddr + Val;
|
||||
{$POP}
|
||||
if (Val < 0) then begin
|
||||
if ConditionalForwardAddr >= NewAddr then begin
|
||||
NewAddr := ConditionalForwardAddr;
|
||||
CheckConditionalForwAddr;
|
||||
if (CurConditionalForwardAddr >= 0) then begin
|
||||
FinishCurAddrBlock;
|
||||
NewAddr := ConditionalForwardAddr[CurConditionalForwardAddr];
|
||||
dec(CurConditionalForwardAddr);
|
||||
StartNextAddrBlock;
|
||||
ClearRecValList := True;
|
||||
|
||||
if ((BackwardJumpAddress = 0) or (Tmp < BackwardJumpAddress)) and
|
||||
(not AddrWasDone(Tmp))
|
||||
then
|
||||
BackwardJumpAddress := Tmp;
|
||||
continue;
|
||||
end;
|
||||
|
||||
if (Tmp > StartAddr) or (Tmp < StartAddr - MAX_SEARCH_ADDR) then
|
||||
exit;
|
||||
MaxAddr2 := StartAddr;
|
||||
BackwardJumpAddress := NewAddr;
|
||||
end;
|
||||
|
||||
if AddrWasDone(Tmp) then begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
|
||||
FinishCurAddrBlock;
|
||||
NewAddr := Tmp;
|
||||
StartNextAddrBlock;
|
||||
end;
|
||||
OPjmpe, OPint, OPint1, OPint3: begin
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
OPjmpe, OPint, OPint1, OPint3:
|
||||
exit; // false
|
||||
OPtest, OPtpause, OPnop, OPcmp, OPcmps:
|
||||
ClearRecValList := False;
|
||||
else
|
||||
@ -5913,7 +6099,8 @@ begin
|
||||
IsRegister(instr.X86Instruction.Operand[1].Value, 'sp')
|
||||
)
|
||||
then begin
|
||||
exit; // false
|
||||
ForceDifferentBranch := True;
|
||||
continue;
|
||||
end;
|
||||
end;
|
||||
end;
|
||||
|
Loading…
Reference in New Issue
Block a user