mirror of
https://gitlab.com/freepascal.org/lazarus/lazarus.git
synced 2025-04-06 14:17:59 +02:00
885 lines
27 KiB
ObjectPascal
885 lines
27 KiB
ObjectPascal
{ $Id$ }
|
|
{ ----------------------------------------------
|
|
callstackdlg.pp - Overview of the callstack
|
|
----------------------------------------------
|
|
|
|
@created(Sun Apr 28th WET 2002)
|
|
@lastmod($Date$)
|
|
@author(Marc Weustink <marc@@dommelstein.net>)
|
|
|
|
This unit contains the Call Stack debugger dialog.
|
|
|
|
|
|
***************************************************************************
|
|
* *
|
|
* This source 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; either version 2 of the License, or *
|
|
* (at your option) any later version. *
|
|
* *
|
|
* This code 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 for more details. *
|
|
* *
|
|
* A copy of the GNU General Public License is available on the World *
|
|
* Wide Web at <http://www.gnu.org/copyleft/gpl.html>. You can also *
|
|
* obtain it by writing to the Free Software Foundation, *
|
|
* Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1335, USA. *
|
|
* *
|
|
***************************************************************************
|
|
}
|
|
unit CallStackDlg;
|
|
|
|
{$mode objfpc}{$H+}
|
|
|
|
interface
|
|
|
|
uses
|
|
SysUtils, Classes, Controls, Forms, LCLProc, LazLoggerBase,
|
|
IDEWindowIntf, DebuggerStrConst,
|
|
ComCtrls, Debugger, DebuggerDlg, Menus, ClipBrd, ExtCtrls, StdCtrls,
|
|
ActnList, IDEImagesIntf, IDECommands, EnvironmentOpts;
|
|
|
|
type
|
|
|
|
{ TCallStackDlg }
|
|
|
|
TCallStackDlg = class(TDebuggerDlg)
|
|
aclActions: TActionList;
|
|
actCopyAll: TAction;
|
|
actShowDisass: TAction;
|
|
actToggleBreakPoint: TAction;
|
|
actViewBottom: TAction;
|
|
actViewTop: TAction;
|
|
actViewLimit: TAction;
|
|
actViewGoto: TAction;
|
|
actViewMore: TAction;
|
|
actSetCurrent: TAction;
|
|
actShow: TAction;
|
|
popShowDisass: TMenuItem;
|
|
popToggle: TMenuItem;
|
|
ToolButtonPower: TToolButton;
|
|
ToolButton2: TToolButton;
|
|
ToolButtonTop: TToolButton;
|
|
ToolButtonBottom: TToolButton;
|
|
ToolButtonCopyAll: TToolButton;
|
|
ToolButton8: TToolButton;
|
|
ToolButton9: TToolButton;
|
|
txtGoto: TEdit;
|
|
lvCallStack: TListView;
|
|
Panel1: TPanel;
|
|
popLimit50: TMenuItem;
|
|
popLimit25: TMenuItem;
|
|
popLimit10: TMenuItem;
|
|
popCopyAll: TMenuItem;
|
|
N1: TMenuItem;
|
|
popSetAsCurrent: TMenuItem;
|
|
popShow: TMenuItem;
|
|
mnuPopup: TPopupMenu;
|
|
mnuLimit: TPopupMenu;
|
|
ToolBar1: TToolBar;
|
|
ToolButtonShow: TToolButton;
|
|
ToolButtonCurrent: TToolButton;
|
|
ToolButton4: TToolButton;
|
|
ToolButtonMore: TToolButton;
|
|
ToolButtonMax: TToolButton;
|
|
ToolButtonGoto: TToolButton;
|
|
procedure actShowDisassExecute(Sender: TObject);
|
|
procedure actToggleBreakPointExecute(Sender: TObject);
|
|
procedure actViewBottomExecute(Sender: TObject);
|
|
procedure actViewGotoExecute(Sender: TObject);
|
|
procedure actViewMoreExecute(Sender: TObject);
|
|
procedure actViewLimitExecute(Sender: TObject);
|
|
procedure actViewTopExecute(Sender: TObject);
|
|
procedure FormCreate(Sender: TObject);
|
|
procedure lvCallStackClick(Sender: TObject);
|
|
procedure popCountClick(Sender: TObject);
|
|
procedure ToolButtonPowerClick(Sender: TObject);
|
|
procedure txtGotoKeyPress(Sender: TObject; var Key: char);
|
|
procedure lvCallStackDBLCLICK(Sender: TObject);
|
|
procedure actCopyAllClick(Sender: TObject);
|
|
procedure actSetAsCurrentClick(Sender : TObject);
|
|
procedure actShowClick(Sender: TObject);
|
|
private
|
|
FViewCount: Integer;
|
|
FViewLimit: Integer;
|
|
FViewStart: Integer;
|
|
FPowerImgIdx, FPowerImgIdxGrey: Integer;
|
|
FInUpdateView: Boolean;
|
|
FUpdateFlags: set of (ufNeedUpdating);
|
|
function GetImageIndex(Entry: TIdeCallStackEntry): Integer;
|
|
procedure SetViewLimit(const AValue: Integer);
|
|
procedure SetViewStart(AStart: Integer);
|
|
procedure SetViewMax;
|
|
procedure GotoIndex(AIndex: Integer);
|
|
function GetCurrentEntry: TIdeCallStackEntry;
|
|
function GetFunction(const Entry: TIdeCallStackEntry): string;
|
|
procedure UpdateView;
|
|
procedure JumpToSource;
|
|
procedure CopyToClipBoard;
|
|
procedure ToggleBreakpoint(Item: TListItem);
|
|
protected
|
|
procedure DoBeginUpdate; override;
|
|
procedure DoEndUpdate; override;
|
|
procedure DisableAllActions;
|
|
procedure EnableAllActions;
|
|
function GetSelectedSnapshot: TSnapshot;
|
|
function GetSelectedThreads(Snap: TSnapshot): TIdeThreads;
|
|
function GetSelectedCallstack: TIdeCallStack;
|
|
procedure DoBreakPointsChanged; override;
|
|
procedure BreakPointChanged(const ASender: TIDEBreakPoints; const {%H-}ABreakpoint: TIDEBreakPoint);
|
|
procedure CallStackChanged(Sender: TObject);
|
|
procedure CallStackCurrent(Sender: TObject);
|
|
function ColSizeGetter(AColId: Integer; var ASize: Integer): Boolean;
|
|
procedure ColSizeSetter(AColId: Integer; ASize: Integer);
|
|
public
|
|
constructor Create(AOwner: TComponent); override;
|
|
property BreakPoints;
|
|
property CallStackMonitor;
|
|
property ThreadsMonitor;
|
|
property SnapshotManager;
|
|
property ViewLimit: Integer read FViewLimit write SetViewLimit;
|
|
end;
|
|
|
|
|
|
implementation
|
|
|
|
{$R *.lfm}
|
|
|
|
uses
|
|
BaseDebugManager, LazarusIDEStrConsts;
|
|
|
|
var
|
|
DBG_DATA_MONITORS: PLazLoggerLogGroup;
|
|
imgSourceLine: Integer;
|
|
imgNoSourceLine: Integer;
|
|
|
|
CallStackDlgWindowCreator: TIDEWindowCreator;
|
|
|
|
const
|
|
COL_STACK_BRKPOINT = 1;
|
|
COL_STACK_INDEX = 2;
|
|
COL_STACK_SOURCE = 3;
|
|
COL_STACK_LINE = 4;
|
|
COL_STACK_FUNC = 5;
|
|
COL_WIDTHS: Array[0..4] of integer = ( 50, 0, 150, 50, 280);
|
|
|
|
function CallStackDlgColSizeGetter(AForm: TCustomForm; AColId: Integer; var ASize: Integer): Boolean;
|
|
begin
|
|
Result := AForm is TCallStackDlg;
|
|
if Result then
|
|
Result := TCallStackDlg(AForm).ColSizeGetter(AColId, ASize);
|
|
end;
|
|
|
|
procedure CallStackDlgColSizeSetter(AForm: TCustomForm; AColId: Integer; ASize: Integer);
|
|
begin
|
|
if AForm is TCallStackDlg then
|
|
TCallStackDlg(AForm).ColSizeSetter(AColId, ASize);
|
|
end;
|
|
|
|
{ TCallStackDlg }
|
|
|
|
constructor TCallStackDlg.Create(AOwner: TComponent);
|
|
var
|
|
i: Integer;
|
|
begin
|
|
inherited Create(AOwner);
|
|
CallStackNotification.OnChange := @CallStackChanged;
|
|
CallStackNotification.OnCurrent := @CallStackCurrent;
|
|
BreakpointsNotification.OnAdd := @BreakPointChanged;
|
|
BreakpointsNotification.OnUpdate := @BreakPointChanged;
|
|
BreakpointsNotification.OnRemove := @BreakPointChanged;
|
|
ThreadsNotification.OnCurrent := @CallStackChanged;
|
|
SnapshotNotification.OnCurrent := @CallStackChanged;
|
|
|
|
actToggleBreakPoint.ShortCut := IDECommandList.FindIDECommand(ecToggleBreakPoint).AsShortCut;
|
|
|
|
for i := low(COL_WIDTHS) to high(COL_WIDTHS) do
|
|
if COL_WIDTHS[i] > 0 then
|
|
lvCallStack.Column[i].Width := COL_WIDTHS[i]
|
|
else
|
|
lvCallStack.Column[i].AutoSize := True;
|
|
end;
|
|
|
|
procedure TCallStackDlg.CallStackChanged(Sender: TObject);
|
|
begin
|
|
DebugLn(DBG_DATA_MONITORS, ['DebugDataWindow: TCallStackDlg.CallStackChanged from ', DbgSName(Sender), ' Upd:', IsUpdating]);
|
|
if (not ToolButtonPower.Down) or FInUpdateView then exit;
|
|
if FViewStart = 0
|
|
then UpdateView
|
|
else SetViewStart(0);
|
|
SetViewMax;
|
|
end;
|
|
|
|
procedure TCallStackDlg.CallStackCurrent(Sender: TObject);
|
|
begin
|
|
DebugLn(DBG_DATA_MONITORS, ['DebugDataWindow: TCallStackDlg.CallStackCurrent from ', DbgSName(Sender), ' Upd:', IsUpdating]);
|
|
if not ToolButtonPower.Down then exit;
|
|
UpdateView;
|
|
end;
|
|
|
|
function TCallStackDlg.ColSizeGetter(AColId: Integer; var ASize: Integer): Boolean;
|
|
begin
|
|
if (AColId - 1 >= 0) and (AColId - 1 < lvCallStack.ColumnCount) then begin
|
|
ASize := lvCallStack.Column[AColId - 1].Width;
|
|
Result := (ASize <> COL_WIDTHS[AColId - 1]) and (not lvCallStack.Column[AColId - 1].AutoSize);
|
|
end
|
|
else
|
|
Result := False;
|
|
end;
|
|
|
|
procedure TCallStackDlg.ColSizeSetter(AColId: Integer; ASize: Integer);
|
|
begin
|
|
case AColId of
|
|
COL_STACK_BRKPOINT: lvCallStack.Column[0].Width := TWidth(ASize);
|
|
COL_STACK_INDEX: lvCallStack.Column[1].Width := TWidth(ASize);
|
|
COL_STACK_SOURCE: lvCallStack.Column[2].Width := TWidth(ASize);
|
|
COL_STACK_LINE: lvCallStack.Column[3].Width := TWidth(ASize);
|
|
COL_STACK_FUNC: lvCallStack.Column[4].Width := TWidth(ASize);
|
|
end;
|
|
end;
|
|
|
|
function TCallStackDlg.GetImageIndex(Entry: TIdeCallStackEntry): Integer;
|
|
|
|
function GetBreakPoint(Entry: TIdeCallStackEntry): TIDEBreakPoint; inline;
|
|
var
|
|
FileName: String;
|
|
begin
|
|
Result := nil;
|
|
if BreakPoints = nil then Exit;
|
|
if DebugBoss.GetFullFilename(Entry.UnitInfo, FileName, False)
|
|
then Result := BreakPoints.Find(FileName, Entry.Line);
|
|
end;
|
|
|
|
begin
|
|
Result := GetBreakPointImageIndex(GetBreakPoint(Entry), Entry.IsCurrent);
|
|
if Result >= 0
|
|
then exit;
|
|
|
|
if Entry.Source = ''
|
|
then Result := imgNoSourceLine
|
|
else Result := imgSourceLine;
|
|
end;
|
|
|
|
procedure TCallStackDlg.UpdateView;
|
|
function LastDelimPos(const FileName: string): Integer;
|
|
begin
|
|
Result := Length(FileName);
|
|
if FileName[Result] in ['/', '\'] then
|
|
exit(-1);
|
|
while (Result > 0) and not (FileName[Result] in ['/', '\']) do
|
|
Dec(Result);
|
|
end;
|
|
var
|
|
i, n: Integer;
|
|
Item: TListItem;
|
|
Entry: TIdeCallStackEntry;
|
|
First, Count, MaxCnt: Integer;
|
|
Source: String;
|
|
Snap: TSnapshot;
|
|
CStack: TIdeCallStack;
|
|
begin
|
|
if (not ToolButtonPower.Down) or FInUpdateView then exit;
|
|
if IsUpdating then begin
|
|
DebugLn(DBG_DATA_MONITORS, ['DebugDataWindow: TCallStackDlg.UpdateView in IsUpdating']);
|
|
Include(FUpdateFlags, ufNeedUpdating);
|
|
exit;
|
|
end;
|
|
try DebugLnEnter(DBG_DATA_MONITORS, ['DebugDataWindow: >>ENTER: TCallStackDlg.UpdateView']);
|
|
Exclude(FUpdateFlags, ufNeedUpdating);
|
|
|
|
BeginUpdate;
|
|
lvCallStack.BeginUpdate;
|
|
try
|
|
Snap := GetSelectedSnapshot;
|
|
if Snap <> nil
|
|
then Caption:= lisMenuViewCallStack + ' (' + Snap.LocationAsText + ')'
|
|
else Caption:= lisMenuViewCallStack;
|
|
|
|
FInUpdateView := True; // ignore change triggered by count, if there is a change event, then Count will be updated already
|
|
CStack := GetSelectedCallstack;
|
|
MaxCnt := FViewStart + FViewLimit + 1;
|
|
if CStack <> nil then CStack.CountLimited(MaxCnt); // trigger the update-notification, if executed immediately
|
|
FInUpdateView := False;
|
|
// TODO: must make CStack ref-counted
|
|
if CStack <> GetSelectedCallstack then exit; // Something changed, maybe debugger stopped
|
|
|
|
if (CStack = nil) or ((Snap <> nil) and (CStack.CountLimited(MaxCnt) = 0)) then begin
|
|
lvCallStack.Items.Clear;
|
|
Item := lvCallStack.Items.Add;
|
|
Item.SubItems.Add('');
|
|
Item.SubItems.Add(lisCallStackNotEvaluated);
|
|
Item.SubItems.Add('');
|
|
Item.SubItems.Add('');
|
|
exit;
|
|
end;
|
|
|
|
if (CStack.CountLimited(MaxCnt)=0)
|
|
then begin
|
|
txtGoto.Text:= '0';
|
|
lvCallStack.Items.Clear;
|
|
exit;
|
|
end;
|
|
|
|
if Snap <> nil then begin
|
|
First := 0;
|
|
Count := CStack.CountLimited(MaxCnt);
|
|
end else begin
|
|
First := FViewStart;
|
|
if First + FViewLimit <= CStack.CountLimited(MaxCnt)
|
|
then Count := FViewLimit
|
|
else Count := CStack.Count - First;
|
|
end;
|
|
|
|
// Reuse entries, so add and remove only
|
|
// Remove unneded
|
|
for n := lvCallStack.Items.Count - 1 downto Count do
|
|
lvCallStack.Items.Delete(n);
|
|
|
|
// Add needed
|
|
for n := lvCallStack.Items.Count to Count - 1 do
|
|
begin
|
|
Item := lvCallStack.Items.Add;
|
|
Item.SubItems.Add('');
|
|
Item.SubItems.Add('');
|
|
Item.SubItems.Add('');
|
|
Item.SubItems.Add('');
|
|
end;
|
|
|
|
FInUpdateView := True;
|
|
CStack.PrepareRange(First, Count);
|
|
// TODO: must make CStack ref-counted
|
|
FInUpdateView := False;
|
|
if CStack <> GetSelectedCallstack then exit; // Something changed, maybe debugger stopped
|
|
for n := 0 to Count - 1 do
|
|
begin
|
|
Item := lvCallStack.Items[n];
|
|
Entry := CStack.Entries[First + n];
|
|
if Entry = nil
|
|
then begin
|
|
Item.Caption := '';
|
|
Item.ImageIndex := imgNoSourceLine;
|
|
Item.SubItems[0] := '????';
|
|
Item.SubItems[1] := '';
|
|
Item.SubItems[2] := '';
|
|
Item.SubItems[3] := '';
|
|
end
|
|
else begin
|
|
Item.ImageIndex := GetImageIndex(Entry);
|
|
Item.SubItems[0] := IntToStr(Entry.Index);
|
|
Source := Entry.Source;
|
|
if (Source = '') and (Entry.UnitInfo <> nil) and (Entry.UnitInfo.LocationFullFile <> '') then
|
|
Source := Entry.UnitInfo.LocationFullFile;
|
|
if Source = '' then // we do not have a source file => just show an adress
|
|
Source := ':' + IntToHex(Entry.Address, 8)
|
|
else begin
|
|
i := LastDelimPos(Source);
|
|
if i > 1 then
|
|
Source := copy(Source, i+1, length(Source)) + ' (' + copy(Source, 1, i) + ')'
|
|
end;
|
|
Item.SubItems[1] := Source;
|
|
if (Entry.Line = 0) and (Entry.UnitInfo <> nil) and (Entry.UnitInfo.SrcLine > 0) then
|
|
Item.SubItems[2] := '~'+IntToStr(Entry.UnitInfo.SrcLine)
|
|
else
|
|
if Entry.Line > 0 then
|
|
Item.SubItems[2] := IntToStr(Entry.Line) // TODO: if editor is open, map line SrcEdit.DebugToSourceLine
|
|
else
|
|
Item.SubItems[2] := '-';
|
|
Item.SubItems[3] := GetFunction(Entry);
|
|
end;
|
|
end;
|
|
|
|
finally
|
|
FInUpdateView := False;
|
|
lvCallStack.EndUpdate;
|
|
EndUpdate;
|
|
end;
|
|
finally DebugLnExit(DBG_DATA_MONITORS, ['DebugDataWindow: <<EXIT: TCallStackDlg.UpdateView']); end;
|
|
end;
|
|
|
|
procedure TCallStackDlg.DoBeginUpdate;
|
|
begin
|
|
DisableAllActions;
|
|
lvCallStack.BeginUpdate;
|
|
end;
|
|
|
|
procedure TCallStackDlg.DoEndUpdate;
|
|
begin
|
|
if ufNeedUpdating in FUpdateFlags then UpdateView;
|
|
lvCallStack.EndUpdate;
|
|
EnableAllActions;
|
|
end;
|
|
|
|
procedure TCallStackDlg.DisableAllActions;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
for i := 0 to aclActions.ActionCount - 1 do
|
|
(aclActions.Actions[i] as TAction).Enabled := False;
|
|
end;
|
|
|
|
procedure TCallStackDlg.EnableAllActions;
|
|
var
|
|
i: Integer;
|
|
Snap: TSnapshot;
|
|
begin
|
|
for i := 0 to aclActions.ActionCount - 1 do
|
|
(aclActions.Actions[i] as TAction).Enabled := True;
|
|
Snap := GetSelectedSnapshot;
|
|
if snap <> nil then begin
|
|
actViewLimit.Enabled := False;
|
|
actViewMore.Enabled := False;
|
|
end;
|
|
ToolButtonPower.Enabled := Snap = nil;
|
|
end;
|
|
|
|
function TCallStackDlg.GetSelectedSnapshot: TSnapshot;
|
|
begin
|
|
Result := nil;
|
|
if (SnapshotManager <> nil) and (SnapshotManager.SelectedEntry <> nil)
|
|
then Result := SnapshotManager.SelectedEntry;
|
|
end;
|
|
|
|
function TCallStackDlg.GetSelectedThreads(Snap: TSnapshot): TIdeThreads;
|
|
begin
|
|
if ThreadsMonitor = nil then exit(nil);
|
|
if Snap = nil
|
|
then Result := ThreadsMonitor.CurrentThreads
|
|
else Result := ThreadsMonitor.Snapshots[Snap];
|
|
end;
|
|
|
|
function TCallStackDlg.GetSelectedCallstack: TIdeCallStack;
|
|
var
|
|
Snap: TSnapshot;
|
|
Threads: TIdeThreads;
|
|
tid: LongInt;
|
|
begin
|
|
if (CallStackMonitor = nil) or (ThreadsMonitor = nil)
|
|
then begin
|
|
Result := nil;
|
|
exit;
|
|
end;
|
|
|
|
Snap := GetSelectedSnapshot;
|
|
Threads := GetSelectedThreads(Snap);
|
|
// There should always be a thread object
|
|
Assert(Threads<>nil, 'TCallStackDlg.GetSelectedCallstack missing thread object');
|
|
if Threads <> nil
|
|
then tid := Threads.CurrentThreadId
|
|
else tid := 1;
|
|
|
|
if (Snap <> nil)
|
|
then Result := CallStackMonitor.Snapshots[Snap].EntriesForThreads[tid]
|
|
else Result := CallStackMonitor.CurrentCallStackList.EntriesForThreads[tid];
|
|
end;
|
|
|
|
function TCallStackDlg.GetCurrentEntry: TIdeCallStackEntry;
|
|
var
|
|
CurItem: TListItem;
|
|
idx: Integer;
|
|
begin
|
|
Result := nil;
|
|
if GetSelectedCallstack = nil then Exit;
|
|
|
|
CurItem := lvCallStack.Selected;
|
|
if CurItem = nil then Exit;
|
|
|
|
idx := FViewStart + CurItem.Index;
|
|
if idx >= GetSelectedCallstack.CountLimited(idx+1) then Exit;
|
|
|
|
Result := GetSelectedCallstack.Entries[idx];
|
|
end;
|
|
|
|
procedure TCallStackDlg.JumpToSource;
|
|
var
|
|
Entry: TIdeCallStackEntry;
|
|
begin
|
|
Entry := GetCurrentEntry;
|
|
if Entry = nil then Exit;
|
|
|
|
JumpToUnitSource(Entry.UnitInfo, Entry.Line);
|
|
end;
|
|
|
|
procedure TCallStackDlg.CopyToClipBoard;
|
|
var
|
|
n: integer;
|
|
Entry: TIdeCallStackEntry;
|
|
S: String;
|
|
begin
|
|
Clipboard.Clear;
|
|
|
|
if (GetSelectedCallstack=nil) or (GetSelectedCallstack.Count=0) then exit;
|
|
|
|
S := '';
|
|
// GetSelectedCallstack.PrepareRange();
|
|
for n:= 0 to GetSelectedCallstack.Count-1 do
|
|
begin
|
|
Entry:=GetSelectedCallstack.Entries[n];
|
|
if Entry <> nil
|
|
then S := S + format('#%d %s at %s:%d', [n, GetFunction(Entry), Entry.Source, Entry.Line])
|
|
else S := S + format('#%d ????', [n]);
|
|
S := S + LineEnding;
|
|
end;
|
|
ClipBoard.AsText := S;
|
|
end;
|
|
|
|
procedure TCallStackDlg.ToggleBreakpoint(Item: TListItem);
|
|
var
|
|
idx: Integer;
|
|
Entry: TIdeCallStackEntry;
|
|
BreakPoint: TIDEBreakPoint;
|
|
FileName: String;
|
|
Ctrl: Boolean;
|
|
begin
|
|
Ctrl := ssCtrl in GetKeyShiftState;
|
|
|
|
try
|
|
DisableAllActions;
|
|
if (Item <> nil) and (BreakPoints <> nil) then
|
|
begin
|
|
GetSelectedCallstack.CountLimited(FViewStart + FViewLimit + 1); // get max limit
|
|
idx := FViewStart + Item.Index;
|
|
if idx >= GetSelectedCallstack.CountLimited(idx+1) then Exit;
|
|
Entry := GetSelectedCallstack.Entries[idx];
|
|
if Entry.Line <= 0 then exit;
|
|
if not DebugBoss.GetFullFilename(Entry.UnitInfo, FileName, False) then
|
|
Exit;
|
|
BreakPoint := BreakPoints.Find(FileName, Entry.Line);
|
|
if BreakPoint <> nil then begin
|
|
if Ctrl
|
|
then BreakPoint.Enabled := not BreakPoint.Enabled
|
|
else DebugBoss.DoDeleteBreakPoint(BreakPoint.Source, BreakPoint.Line)
|
|
end else begin
|
|
DebugBoss.LockCommandProcessing;
|
|
try
|
|
DebugBoss.DoCreateBreakPoint(FileName, Entry.Line, False, BreakPoint);
|
|
if Ctrl and (BreakPoint <> nil)
|
|
then BreakPoint.Enabled := False;
|
|
finally
|
|
DebugBoss.UnLockCommandProcessing;
|
|
end;
|
|
end;
|
|
end;
|
|
finally
|
|
EnableAllActions;
|
|
end;
|
|
end;
|
|
|
|
procedure TCallStackDlg.DoBreakPointsChanged;
|
|
begin
|
|
UpdateView;
|
|
end;
|
|
|
|
procedure TCallStackDlg.lvCallStackDBLCLICK(Sender: TObject);
|
|
begin
|
|
JumpToSource;
|
|
end;
|
|
|
|
procedure TCallStackDlg.popCountClick(Sender: TObject);
|
|
begin
|
|
if FViewCount = TMenuItem(Sender).Tag then Exit;
|
|
FViewCount := TMenuItem(Sender).Tag;
|
|
ViewLimit := FViewCount;
|
|
EnvironmentOptions.DebuggerConfig.DlgCallStackConfig.ViewCount := FViewCount;
|
|
actViewLimit.Caption := TMenuItem(Sender).Caption;
|
|
end;
|
|
|
|
procedure TCallStackDlg.ToolButtonPowerClick(Sender: TObject);
|
|
begin
|
|
if ToolButtonPower.Down
|
|
then begin
|
|
ToolButtonPower.ImageIndex := FPowerImgIdx;
|
|
UpdateView;
|
|
end
|
|
else ToolButtonPower.ImageIndex := FPowerImgIdxGrey;
|
|
end;
|
|
|
|
procedure TCallStackDlg.txtGotoKeyPress(Sender: TObject; var Key: char);
|
|
begin
|
|
case Key of
|
|
'0'..'9', #8 : ;
|
|
#13 : SetViewStart(StrToIntDef(txtGoto.Text, 0));
|
|
else
|
|
Key := #0;
|
|
end;
|
|
end;
|
|
|
|
procedure TCallStackDlg.actCopyAllClick(Sender: TObject);
|
|
begin
|
|
CopyToClipBoard;
|
|
end;
|
|
|
|
procedure TCallStackDlg.actSetAsCurrentClick(Sender : TObject);
|
|
var
|
|
Entry: TIdeCallStackEntry;
|
|
begin
|
|
try
|
|
DisableAllActions;
|
|
Entry := GetCurrentEntry;
|
|
if Entry = nil then Exit;
|
|
|
|
GetSelectedCallstack.ChangeCurrentIndex(Entry.Index);
|
|
if GetSelectedSnapshot <> nil
|
|
then CallStackMonitor.NotifyCurrent; // TODO: move to snapshot callstack object
|
|
finally
|
|
EnableAllActions;
|
|
end;
|
|
end;
|
|
|
|
procedure TCallStackDlg.actShowClick(Sender: TObject);
|
|
begin
|
|
JumpToSource;
|
|
end;
|
|
|
|
procedure TCallStackDlg.actViewBottomExecute(Sender: TObject);
|
|
begin
|
|
try
|
|
DisableAllActions;
|
|
if GetSelectedCallstack <> nil
|
|
then SetViewStart(GetSelectedCallstack.Count - FViewLimit)
|
|
else SetViewStart(0);
|
|
finally
|
|
EnableAllActions;
|
|
end;
|
|
end;
|
|
|
|
procedure TCallStackDlg.actToggleBreakPointExecute(Sender: TObject);
|
|
begin
|
|
ToggleBreakpoint(lvCallStack.Selected);
|
|
end;
|
|
|
|
procedure TCallStackDlg.actShowDisassExecute(Sender: TObject);
|
|
var
|
|
Entry: TIdeCallStackEntry;
|
|
begin
|
|
Entry := GetCurrentEntry;
|
|
if (Entry = nil) or (Entry.Address = 0) then Exit;
|
|
DebugBoss.ViewDisassembler(Entry.Address);
|
|
end;
|
|
|
|
procedure TCallStackDlg.actViewGotoExecute(Sender: TObject);
|
|
begin
|
|
try
|
|
DisableAllActions;
|
|
SetViewStart(StrToIntDef(txtGoto.Text, 0));
|
|
finally
|
|
EnableAllActions;
|
|
end;
|
|
end;
|
|
|
|
procedure TCallStackDlg.actViewMoreExecute(Sender: TObject);
|
|
begin
|
|
try
|
|
DisableAllActions;
|
|
ToolButtonPower.Down := True;
|
|
ToolButtonPowerClick(nil);
|
|
ViewLimit := ViewLimit + FViewCount;
|
|
finally
|
|
EnableAllActions;
|
|
end;
|
|
end;
|
|
|
|
procedure TCallStackDlg.actViewTopExecute(Sender: TObject);
|
|
begin
|
|
try
|
|
DisableAllActions;
|
|
ToolButtonPower.Down := True;
|
|
ToolButtonPowerClick(nil);
|
|
SetViewStart(0);
|
|
finally
|
|
EnableAllActions;
|
|
end;
|
|
end;
|
|
|
|
procedure TCallStackDlg.BreakPointChanged(const ASender: TIDEBreakPoints;
|
|
const ABreakpoint: TIDEBreakPoint);
|
|
var
|
|
i, idx: Integer;
|
|
Entry: TIdeCallStackEntry;
|
|
Stack: TIdeCallStack;
|
|
begin
|
|
DebugLn(DBG_DATA_MONITORS, ['DebugDataWindow: TCallStackDlg.BreakPointChanged ', DbgSName(ASender), ' Upd:', IsUpdating]);
|
|
Stack := GetSelectedCallstack;
|
|
if (BreakPoints = nil) or (Stack = nil) then
|
|
Exit;
|
|
|
|
Stack.CountLimited(FViewStart + FViewLimit + 1);
|
|
for i := 0 to lvCallStack.Items.Count - 1 do
|
|
begin
|
|
idx := FViewStart + lvCallStack.Items[i].Index;
|
|
if idx >= Stack.CountLimited(idx+1) then
|
|
Continue;
|
|
Entry := Stack.Entries[idx];
|
|
if Entry <> nil then
|
|
lvCallStack.Items[i].ImageIndex := GetImageIndex(Entry)
|
|
else
|
|
lvCallStack.Items[i].ImageIndex := imgNoSourceLine;
|
|
end;
|
|
end;
|
|
|
|
procedure TCallStackDlg.FormCreate(Sender: TObject);
|
|
var
|
|
i: integer;
|
|
curPopLimit: TMenuItem;
|
|
begin
|
|
Caption := lisMenuViewCallStack;
|
|
ToolButtonPower.Caption := lisDbgWinPower;
|
|
ToolButtonPower.Hint := lisDbgWinPowerHint;
|
|
for i:= 0 to mnuLimit.Items.Count-1 do
|
|
mnuLimit.Items[i].Caption:= Format(lisMaxS, [mnuLimit.Items[i].Tag]);
|
|
actViewMore.Caption := lisMore;
|
|
actViewTop.Caption := lisCSTop;
|
|
actViewBottom.Caption := lisCSBottom;
|
|
actViewGoto.Caption := lisGotoSelected;
|
|
actShow.Caption := lisViewSource;
|
|
actShowDisass.Caption := lisViewSourceDisass;
|
|
actToggleBreakPoint.Caption := uemToggleBreakpoint;
|
|
actSetCurrent.Caption := lisCurrent;
|
|
actCopyAll.Caption := lisCopyAll;
|
|
|
|
actViewMore.Hint := lisMore;
|
|
actViewTop.Hint := lisCSTop;
|
|
actViewBottom.Hint := lisCSBottom;
|
|
actViewGoto.Hint := lisGotoSelected;
|
|
actShow.Hint := lisViewSource;
|
|
actShowDisass.Hint := lisViewSourceDisass;
|
|
actToggleBreakPoint.Hint := uemToggleBreakpoint;
|
|
actSetCurrent.Hint := lisCurrent;
|
|
actCopyAll.Hint := lisCopyAll;
|
|
|
|
FViewCount := EnvironmentOptions.DebuggerConfig.DlgCallStackConfig.ViewCount;
|
|
curPopLimit := nil;
|
|
for i := 0 to mnuLimit.Items.Count-1 do
|
|
if mnuLimit.Items[i].Tag = FViewCount then
|
|
begin
|
|
curPopLimit := mnuLimit.Items[i];
|
|
Break;
|
|
end;
|
|
if curPopLimit=nil then
|
|
curPopLimit := popLimit10;
|
|
FViewCount := curPopLimit.Tag;
|
|
FViewLimit := FViewCount;
|
|
FViewStart := 0;
|
|
FInUpdateView := False;
|
|
actViewLimit.Caption := curPopLimit.Caption;
|
|
ToolButtonMax.Caption := actViewLimit.Caption;
|
|
|
|
lvCallStack.Columns[1].Caption:= lisIndex;
|
|
lvCallStack.Columns[2].Caption:= histdlgColumnLoc;
|
|
lvCallStack.Columns[3].Caption:= dlgAddHiAttrGroupLine;
|
|
lvCallStack.Columns[4].Caption:= lisFunction;
|
|
|
|
ToolBar1.Images := IDEImages.Images_16;
|
|
ToolButtonShow.ImageIndex := IDEImages.LoadImage('callstack_show');
|
|
ToolButtonMore.ImageIndex := IDEImages.LoadImage('callstack_more');
|
|
ToolButtonTop.ImageIndex := IDEImages.LoadImage('callstack_top');
|
|
ToolButtonBottom.ImageIndex := IDEImages.LoadImage('callstack_bottom');
|
|
ToolButtonGoto.ImageIndex := IDEImages.LoadImage('callstack_goto');
|
|
ToolButtonCopyAll.ImageIndex := IDEImages.LoadImage('laz_copy');
|
|
ToolButtonCurrent.ImageIndex := IDEImages.LoadImage('debugger_current_line');
|
|
FPowerImgIdx := IDEImages.LoadImage('debugger_power');
|
|
FPowerImgIdxGrey := IDEImages.LoadImage('debugger_power_grey');
|
|
ToolButtonPower.ImageIndex := FPowerImgIdx;
|
|
|
|
lvCallStack.SmallImages := IDEImages.Images_16;
|
|
imgSourceLine := IDEImages.LoadImage('debugger_source_line');
|
|
imgNoSourceLine := IDEImages.LoadImage('debugger_nosource_line');
|
|
|
|
end;
|
|
|
|
procedure TCallStackDlg.lvCallStackClick(Sender: TObject);
|
|
var
|
|
P: TPoint;
|
|
Item: TListItem;
|
|
begin
|
|
// toggle breakpoint
|
|
P := lvCallStack.ScreenToClient(Mouse.CursorPos);
|
|
Item := lvCallStack.GetItemAt(P.X, P.Y);
|
|
// if clicked on the first column of a valid item
|
|
if (Item <> nil) and (P.X <= lvCallStack.Column[0].Width) then
|
|
ToggleBreakPoint(Item);
|
|
end;
|
|
|
|
procedure TCallStackDlg.actViewLimitExecute(Sender: TObject);
|
|
begin
|
|
try
|
|
DisableAllActions;
|
|
ToolButtonPower.Down := True;
|
|
ToolButtonPowerClick(nil);
|
|
ViewLimit := FViewCount;
|
|
finally
|
|
EnableAllActions;
|
|
end;
|
|
end;
|
|
|
|
procedure TCallStackDlg.SetViewStart(AStart: Integer);
|
|
begin
|
|
if GetSelectedCallstack = nil then Exit;
|
|
ToolButtonPower.Down := True;
|
|
ToolButtonPowerClick(nil);
|
|
|
|
if (AStart > GetSelectedCallstack.CountLimited(AStart+FViewLimit+1) - FViewLimit)
|
|
then AStart := GetSelectedCallstack.Count - FViewLimit;
|
|
if AStart < 0 then AStart := 0;
|
|
if FViewStart = AStart then Exit;
|
|
|
|
FViewStart:= AStart;
|
|
txtGoto.Text:= IntToStr(AStart);
|
|
UpdateView;
|
|
end;
|
|
|
|
procedure TCallStackDlg.SetViewMax;
|
|
begin
|
|
// If GetSelectedCallstack = nil
|
|
// then lblViewCnt.Caption:= '0'
|
|
// else lblViewCnt.Caption:= IntToStr(GetSelectedCallstack.Count);
|
|
end;
|
|
|
|
procedure TCallStackDlg.SetViewLimit(const AValue: Integer);
|
|
begin
|
|
ToolButtonPower.Down := True;
|
|
ToolButtonPowerClick(nil);
|
|
if FViewLimit = AValue then Exit;
|
|
if (GetSelectedCallstack <> nil)
|
|
and (FViewStart + FViewLimit >= GetSelectedCallstack.CountLimited(FViewStart + FViewLimit+1))
|
|
and (AValue > FViewLimit)
|
|
then begin
|
|
FViewStart := GetSelectedCallstack.Count - AValue;
|
|
if FViewStart < 0 then FViewStart := 0;
|
|
end;
|
|
FViewLimit := AValue;
|
|
UpdateView;
|
|
end;
|
|
|
|
function TCallStackDlg.GetFunction(const Entry: TIdeCallStackEntry): string;
|
|
begin
|
|
Result := Entry.GetFunctionWithArg;
|
|
end;
|
|
|
|
procedure TCallStackDlg.GotoIndex(AIndex: Integer);
|
|
begin
|
|
if AIndex < 0 then Exit;
|
|
if AIndex >= GetSelectedCallstack.CountLimited(AIndex+1) then Exit;
|
|
|
|
|
|
end;
|
|
|
|
initialization
|
|
|
|
CallStackDlgWindowCreator := IDEWindowCreators.Add(DebugDialogNames[ddtCallStack]);
|
|
CallStackDlgWindowCreator.OnCreateFormProc := @CreateDebugDialog;
|
|
CallStackDlgWindowCreator.OnSetDividerSize := @CallStackDlgColSizeSetter;
|
|
CallStackDlgWindowCreator.OnGetDividerSize := @CallStackDlgColSizeGetter;
|
|
CallStackDlgWindowCreator.DividerTemplate.Add('ColumnCStackBrkPoint', COL_STACK_BRKPOINT, @drsColWidthBrkPointImg);
|
|
CallStackDlgWindowCreator.DividerTemplate.Add('ColumnCStackIndex', COL_STACK_INDEX, @drsColWidthIndex);
|
|
CallStackDlgWindowCreator.DividerTemplate.Add('ColumnCStackSource', COL_STACK_SOURCE, @drsColWidthSource);
|
|
CallStackDlgWindowCreator.DividerTemplate.Add('ColumnCStackLine', COL_STACK_LINE, @drsColWidthLine);
|
|
CallStackDlgWindowCreator.DividerTemplate.Add('ColumnCStackFunc', COL_STACK_FUNC, @drsColWidthFunc);
|
|
CallStackDlgWindowCreator.CreateSimpleLayout;
|
|
|
|
DBG_DATA_MONITORS := DebugLogger.FindOrRegisterLogGroup('DBG_DATA_MONITORS' {$IFDEF DBG_DATA_MONITORS} , True {$ENDIF} );
|
|
|
|
end.
|
|
|