{ $Id$ } { ---------------------------------------- Debugger.pp - Debugger base classes ---------------------------------------- @created(Wed Feb 25st WET 2001) @author(Marc Weustink ) This unit contains the base class definitions of the debugger. These classes are only definitions. Implemented debuggers should be derived from these. *************************************************************************** * * * 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 . You can also * * obtain it by writing to the Free Software Foundation, * * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * * * *************************************************************************** } unit Debugger; {$mode objfpc}{$H+} {$IFDEF linux} {$DEFINE DBG_ENABLE_TERMINAL} {$ENDIF} interface uses TypInfo, Classes, SysUtils, Laz2_XMLCfg, math, FileUtil, LazLoggerBase, LCLProc, LazConfigStorage, LazClasses, maps, DbgIntfBaseTypes, DbgIntfMiscClasses, DbgIntfDebuggerBase; const XMLBreakPointsNode = 'BreakPoints'; XMLBreakPointGroupsNode = 'BreakPointGroups'; XMLWatchesNode = 'Watches'; XMLExceptionsNode = 'Exceptions'; type { TDebuggerConfigStore } (* TODO: maybe revert relations. Create this in Debugger, and call environmentoptions for the configstore only? *) { TDebuggerConfigStoreBase } TDebuggerConfigStoreBase = class(TPersistent) private FConfigStore: TConfigStorage; public property ConfigStore: TConfigStorage read FConfigStore write FConfigStore; procedure Init; virtual; procedure Load; virtual; procedure Save; virtual; end; { TDebuggerWatchesDlgConfig } TDebuggerWatchesDlgConfig = class(TDebuggerConfigStoreBase) private FColumnNameWidth: Integer; FColumnValueWidth: Integer; public constructor Create; procedure Init; override; published property ColumnNameWidth: Integer read FColumnNameWidth write FColumnNameWidth; property ColumnValueWidth: Integer read FColumnValueWidth write FColumnValueWidth; end; TDebuggerConfigStore = class(TDebuggerConfigStoreBase) private FDebuggerClass: String; FTDebuggerWatchesDlgConfig: TDebuggerWatchesDlgConfig; public procedure Load; override; procedure Save; override; public constructor Create; destructor Destroy; override; property DebuggerClass: String read FDebuggerClass write FDebuggerClass; property DlgWatchesConfig: TDebuggerWatchesDlgConfig read FTDebuggerWatchesDlgConfig; published end; TDebuggerLocationType = (dltUnknown, // not jet looked up dltUnresolvable, // lookup failed dltProject, dltPackage ); TDebuggerLocationFlag = (dlfLoadError, // resolved but failed to load dlfSearchByFunctionName ); TDebuggerLocationFlags = set of TDebuggerLocationFlag; { TDebuggerUnitInfo } TDebuggerUnitInfo = class(TRefCountedObject) private FFunctionArgs: String; FSrcClassName: String; FFileName, FDbgFullName: String; FFlags: TDebuggerLocationFlags; FFunctionName: String; FLocationName, FLocationOwnerName, FLocationFullFile: String; FLocationType: TDebuggerLocationType; FSrcLine: Integer; FUnitName: String; function GetFileName: String; function GetDbgFullName: String; function GetLocationFullFile: String; function GetLocationName: String; function GetLocationOwnerName: String; function GetLocationType: TDebuggerLocationType; procedure SetLocationFullFile(AValue: String); procedure SetLocationType(AValue: TDebuggerLocationType); public constructor Create(const AFileName: String; const AFullFileName: String); constructor Create(const AUnitName, AClassName, AFunctionName, AFunctionArgs: String); function DebugText: String; function IsEqual(const AFileName: String; const AFullFileName: String): boolean; function IsEqual(const AUnitName, AClassName, AFunctionName, AFunctionArgs: String): boolean; function IsEqual(AnOther: TDebuggerUnitInfo): boolean; procedure LoadDataFromXMLConfig(const AConfig: TXMLConfig; const APath: string); virtual; procedure SaveDataToXMLConfig(const AConfig: TXMLConfig; const APath: string); virtual; property FileName: String read GetFileName; property SrcLine: Integer read FSrcLine write FSrcLine; property DbgFullName: String read GetDbgFullName; property LocationType: TDebuggerLocationType read GetLocationType write SetLocationType; property LocationOwnerName: String read GetLocationOwnerName; property LocationName: String read GetLocationName; property LocationFullFile: String read GetLocationFullFile write SetLocationFullFile; property Flags: TDebuggerLocationFlags read FFlags write FFlags; property UnitName: String read FUnitName; property SrcClassName: String read FSrcClassName; property FunctionName: String read FFunctionName; property FunctionArgs: String read FFunctionArgs; // comma separated list of types. e.g. "integer, boolean" // functions have result type at end, after ",," end; { TDebuggerUnitInfoList } TDebuggerUnitInfoList = class(TRefCntObjList) private function GetInfo(Index: Integer): TDebuggerUnitInfo; procedure PutInfo(Index: Integer; AValue: TDebuggerUnitInfo); public property Items[Index: Integer]: TDebuggerUnitInfo read GetInfo write PutInfo; default; end; { TDebuggerUnitInfoProvider } TDebuggerUnitInfoProvider = class private FList: TDebuggerUnitInfoList; FLoader: TDebuggerUnitInfo; function GetInfo(Index: Integer): TDebuggerUnitInfo; public constructor Create; destructor Destroy; override; procedure Clear; function GetUnitInfoFor(const AFileName: String; const AFullFileName: String): TDebuggerUnitInfo; function GetUnitInfoByFunction(const AUnitName, AClassName, AFunctionName, AFunctionArgs: String): TDebuggerUnitInfo; function IndexOf(AnInfo: TDebuggerUnitInfo; AddIfNotExists: Boolean = False): Integer; function Count: integer; property Items[Index: Integer]: TDebuggerUnitInfo read GetInfo; default; public // Load/Save all entries with ID procedure LoadDataFromXMLConfig(const AConfig: TXMLConfig; const APath: string); procedure SaveDataToXMLConfig(const AConfig: TXMLConfig; const APath: string); end; { ---------------------------------------------------------
TDebuggerNotification is a reference counted baseclass for handling notifications for locals, watches, breakpoints etc.
---------------------------------------------------------} TDebuggerNotification = class(TRefCountedObject) end; TDebuggerChangeNotification = class(TDebuggerNotification) private FOnChange: TNotifyEvent; FOnCurrent: TNotifyEvent; protected property OnCurrent: TNotifyEvent read FOnCurrent write FOnCurrent; property OnChange: TNotifyEvent read FOnChange write FOnChange; end; { TDebuggerNotificationList } TDebuggerNotificationList = class(TObject) private FList: TList; function GetItem(AIndex: Integer): TDebuggerNotification; protected function NextDownIndex(var Index: integer): boolean; public constructor Create; destructor Destroy; override; procedure Add(const ANotification: TDebuggerNotification); procedure Remove(const ANotification: TDebuggerNotification); function Count: Integer; procedure Clear; property Items[AIndex: Integer]: TDebuggerNotification read GetItem; default; end; { TDebuggerChangeNotificationList } TDebuggerChangeNotificationList = class(TDebuggerNotificationList) private function GetItem(AIndex: Integer): TDebuggerChangeNotification; reintroduce; public procedure NotifyChange(Sender: TObject); procedure NotifyCurrent(Sender: TObject); property Items[AIndex: Integer]: TDebuggerChangeNotification read GetItem; default; end; TIDEBreakPoints = class; TIDEBreakPointGroup = class; TIDEBreakPointGroups = class; TIdeWatch = class; TIdeWatches = class; TCurrentWatch = class; TCurrentWatches = class; TIdeWatchesMonitor = class; TIdeLocalsMonitor = class; TCurrentLocals = class; TIDELineInfo = class; TIdeCallStack = class; TIdeCallStackMonitor = class; TIdeThreadsMonitor = class; TSnapshotManager = class; TDebugger = class; TOnSaveFilenameToConfig = procedure(var Filename: string) of object; TOnLoadFilenameFromConfig = procedure(var Filename: string) of object; TOnGetGroupByName = function(const GroupName: string): TIDEBreakPointGroup of object; TNullableBool = (nbUnknown, nbTrue, nbFalse); { TDebuggerDataSnapShot } TDebuggerDataSnapShot = class public destructor Destroy; override; public DataObject: TObject; SnapShotId: Pointer; end; { TDebuggerDataSnapShotList } TDebuggerDataSnapShotList = class private FList: TList; function GetSnapShot(AnID: Pointer): TObject; public constructor Create; destructor Destroy; override; procedure Clear; procedure AddSnapShot(AnID: Pointer; AnObject: TObject); procedure RemoveSnapShot(AnID: Pointer); property SnapShot[AnID: Pointer]: TObject read GetSnapShot; end; {$region Breakpoints **********************************************************} (******************************************************************************) (** **) (** B R E A K P O I N T S **) (** **) (******************************************************************************) (******************************************************************************) { TIDEBreakPoint } // The TBaseBreakpoint family is the common ancestor for the "public" available // TIDEBreakPoint through the DebugBoss as well as the "private" TDBGBreakPoint // used by the debugboss itself. // The BreakPointGroups are no longer part of the debugger, but they are now // managed by the debugboss. TIDEBreakPointAction = ( bpaStop, bpaEnableGroup, bpaDisableGroup, bpaLogMessage, bpaEValExpression, bpaLogCallStack, bpaTakeSnapshot ); TIDEBreakPointActions = set of TIDEBreakPointAction; TIDEBreakPoint = class; { TIDEBreakPointGroupList } TIDEBreakPointGroupList = class private FList: TFPList; FOwner: TIDEBreakPoint; function GetItem(AIndex: Integer): TIDEBreakPointGroup; public constructor Create(AOwner: TIDEBreakPoint); destructor Destroy; override; procedure Assign(ASrc: TIDEBreakPointGroupList); procedure Clear; function Add(const AGroup: TIDEBreakPointGroup): Integer; procedure Remove(const AGroup: TIDEBreakPointGroup); function IndexOf(const AGroup: TIDEBreakPointGroup): Integer; function Count: Integer; property Items[AIndex: Integer]: TIDEBreakPointGroup read GetItem; default; end; TIDEBreakPoint = class(TBaseBreakPoint) private FLogEvalExpression: String; FMaster: TDBGBreakPoint; FAutoContinueTime: Cardinal; FActions: TIDEBreakPointActions; FDisableGroupList: TIDEBreakPointGroupList; FEnableGroupList: TIDEBreakPointGroupList; FGroup: TIDEBreakPointGroup; FLoading: Boolean; FLogMessage: String; FLogCallStackLimit: Integer; FUserModified: Boolean; protected procedure AssignLocationTo(Dest: TPersistent); override; procedure AssignTo(Dest: TPersistent); override; procedure DoChanged; override; procedure DoUserChanged; // User changed settings function GetHitCount: Integer; override; function GetValid: TValidState; override; procedure SetBreakHitCount(const AValue: Integer); override; procedure SetEnabled(const AValue: Boolean); override; procedure SetInitialEnabled(const AValue: Boolean); override; procedure SetExpression(const AValue: String); override; function DebugExeLine: Integer; virtual; // Same as line, but in Subclass: the line in the compiled exe procedure DisableGroups; procedure DoActionChange; virtual; procedure DoHit(const ACount: Integer; var AContinue: Boolean); override; procedure EnableGroups; procedure ClearAllGroupLists; {$IFDEF DBG_BREAKPOINT} function DebugText: string; {$ENDIF} protected // virtual properties function GetActions: TIDEBreakPointActions; virtual; function GetGroup: TIDEBreakPointGroup; virtual; function GetAutoContinueTime: Cardinal; virtual; function GetLogMessage: String; virtual; function GetLogCallStackLimit: Integer; procedure SetActions(const AValue: TIDEBreakPointActions); virtual; procedure SetGroup(const AValue: TIDEBreakPointGroup); virtual; procedure SetAutoContinueTime(const AValue: Cardinal); virtual; procedure SetLogEvalExpression(AValue: String); procedure SetLogMessage(const AValue: String); virtual; procedure SetLogCallStackLimit(const AValue: Integer); public constructor Create(ACollection: TCollection); override; destructor Destroy; override; procedure LoadFromXMLConfig(XMLConfig: TXMLConfig; const Path: string; const OnLoadFilename: TOnLoadFilenameFromConfig; const OnGetGroup: TOnGetGroupByName); virtual; procedure SaveToXMLConfig(const AConfig: TXMLConfig; const APath: string; const OnSaveFilename: TOnSaveFilenameToConfig); virtual; procedure SetAddress(const AValue: TDBGPtr); override; procedure SetLocation(const ASource: String; const ALine: Integer); override; procedure SetWatch(const AData: String; const AScope: TDBGWatchPointScope; const AKind: TDBGWatchPointKind); override; procedure ResetMaster; property UserModified: Boolean read FUserModified write FUserModified; // Indicator for DoChanged public property Actions: TIDEBreakPointActions read GetActions write SetActions; property AutoContinueTime: Cardinal read GetAutoContinueTime write SetAutoContinueTime; property Group: TIDEBreakPointGroup read GetGroup write SetGroup; property DisableGroupList: TIDEBreakPointGroupList read FDisableGroupList; property EnableGroupList: TIDEBreakPointGroupList read FEnableGroupList; property LogEvalExpression: String read FLogEvalExpression write SetLogEvalExpression; property Loading: Boolean read FLoading; property LogMessage: String read GetLogMessage write SetLogMessage; property LogCallStackLimit: Integer read GetLogCallStackLimit write SetLogCallStackLimit; end; TIDEBreakPointClass = class of TIDEBreakPoint; { TIDEBreakPoints } TIDEBreakPointsEvent = procedure(const ASender: TIDEBreakPoints; const ABreakpoint: TIDEBreakPoint) of object; TIDEBreakPointsNotification = class(TDebuggerNotification) private FOnAdd: TIDEBreakPointsEvent; FOnUpdate: TIDEBreakPointsEvent;//Item will be nil in case all items need to be updated FOnRemove: TIDEBreakPointsEvent; public property OnAdd: TIDEBreakPointsEvent read FOnAdd write FOnAdd; property OnUpdate: TIDEBreakPointsEvent read FOnUpdate write FOnUpdate; property OnRemove: TIDEBreakPointsEvent read FOnRemove write FonRemove; end; TIDEBreakPoints = class(TBaseBreakPoints) private FNotificationList: TList; FMaster: TDBGBreakPoints; procedure SetMaster(const AValue: TDBGBreakPoints); function GetItem(const AnIndex: Integer): TIDEBreakPoint; procedure SetItem(const AnIndex: Integer; const AValue: TIDEBreakPoint); protected procedure NotifyAdd(const ABreakPoint: TIDEBreakPoint); virtual; // called when a breakpoint is added procedure NotifyRemove(const ABreakpoint: TIDEBreakPoint); virtual; // called by breakpoint when destructed procedure Update(Item: TCollectionItem); override; public constructor Create(const ABreakPointClass: TIDEBreakPointClass); destructor Destroy; override; function Add(const ASource: String; const ALine: Integer): TIDEBreakPoint; overload; function Add(const AAddress: TDBGPtr): TIDEBreakPoint; overload; function Add(const AData: String; const AScope: TDBGWatchPointScope; const AKind: TDBGWatchPointKind): TIDEBreakPoint; overload; function Find(const ASource: String; const ALine: Integer): TIDEBreakPoint; overload; function Find(const ASource: String; const ALine: Integer; const AIgnore: TIDEBreakPoint): TIDEBreakPoint; overload; function Find(const AAddress: TDBGPtr): TIDEBreakPoint; overload; function Find(const AAddress: TDBGPtr; const AIgnore: TIDEBreakPoint): TIDEBreakPoint; overload; function Find(const AData: String; const AScope: TDBGWatchPointScope; const AKind: TDBGWatchPointKind): TIDEBreakPoint; overload; function Find(const AData: String; const AScope: TDBGWatchPointScope; const AKind: TDBGWatchPointKind; const AIgnore: TIDEBreakPoint): TIDEBreakPoint; overload; procedure AddNotification(const ANotification: TIDEBreakPointsNotification); procedure RemoveNotification(const ANotification: TIDEBreakPointsNotification); procedure LoadFromXMLConfig(XMLConfig: TXMLConfig; const Path: string; const OnLoadFilename: TOnLoadFilenameFromConfig; const OnGetGroup: TOnGetGroupByName); virtual; procedure SaveToXMLConfig(XMLConfig: TXMLConfig; const Path: string; const OnSaveFilename: TOnSaveFilenameToConfig); virtual; property Master: TDBGBreakPoints read FMaster write SetMaster; public property Items[const AnIndex: Integer]: TIDEBreakPoint read GetItem write SetItem; default; end; { TIDEBreakPointGroup } TIDEBreakPointGroup = class(TCollectionItem) private FEnabled: Boolean; FInitialEnabled: Boolean; FName: String; FBreakpoints: TList;// A list of breakpoints that member FReferences: TList; // A list of breakpoints that refer to us through En/disable group function GetBreakpoint(const AIndex: Integer): TIDEBreakPoint; procedure SetEnabled(const AValue: Boolean); procedure SetInitialEnabled(const AValue: Boolean); procedure SetName(const AValue: String); protected procedure AssignTo(Dest: TPersistent); override; procedure AddReference(const ABreakPointList: TIDEBreakPointGroupList); procedure RemoveReference(const ABreakPointList: TIDEBreakPointGroupList); public function Add(const ABreakPoint: TIDEBreakPoint): Integer; function Count: Integer; constructor Create(ACollection: TCollection); override; procedure Delete(const AIndex: Integer); destructor Destroy; override; function Remove(const ABreakPoint: TIDEBreakPoint): Integer; procedure LoadFromXMLConfig(XMLConfig: TXMLConfig; const Path: string); virtual; procedure SaveToXMLConfig(XMLConfig: TXMLConfig; const Path: string); virtual; class function CheckName(const AName: String): Boolean; public property Breakpoints[const AIndex: Integer]: TIDEBreakPoint read GetBreakpoint; property Enabled: Boolean read FEnabled write SetEnabled; //property InitialEnabled: Boolean read FInitialEnabled write SetInitialEnabled; property Name: String read FName write SetName; end; { TIDEBreakPointGroups } TIDEBreakPointGroups = class(TCollection) private function GetItem(const AnIndex: Integer): TIDEBreakPointGroup; procedure SetItem(const AnIndex: Integer; const AValue: TIDEBreakPointGroup); protected public constructor Create; procedure LoadFromXMLConfig(XMLConfig: TXMLConfig; const Path: string); virtual; procedure SaveToXMLConfig(XMLConfig: TXMLConfig; const Path: string); virtual; function GetGroupByName(const GroupName: string): TIDEBreakPointGroup; function FindGroupByName(const GroupName: string; Ignore: TIDEBreakPointGroup): TIDEBreakPointGroup; function IndexOfGroupWithName(const GroupName: string; Ignore : TIDEBreakPointGroup): integer; procedure InitTargetStart; virtual; // procedure Regroup(SrcGroups: TIDEBreakPointGroups; // SrcBreakPoints, DestBreakPoints: TIDEBreakPoints); public property Items[const AnIndex: Integer]: TIDEBreakPointGroup read GetItem write SetItem; default; end; {%endregion ^^^^^ Breakpoints ^^^^^ } {%region Watches ************************************************************** ****************************************************************************** ** ** ** W A T C H E S ** ** ** ****************************************************************************** ******************************************************************************} const TWatchDisplayFormatNames: array [TWatchDisplayFormat] of string = ('wdfDefault', 'wdfStructure', 'wdfChar', 'wdfString', 'wdfDecimal', 'wdfUnsigned', 'wdfFloat', 'wdfHex', 'wdfPointer', 'wdfMemDump' ); type TWatchesEvent = procedure(const ASender: TIdeWatches; const AWatch: TIdeWatch) of object; TWatchesNotification = class(TDebuggerNotification) private FOnAdd: TWatchesEvent; FOnUpdate: TWatchesEvent;//Item will be nil in case all items need to be updated FOnRemove: TWatchesEvent; public property OnAdd: TWatchesEvent read FOnAdd write FOnAdd; property OnUpdate: TWatchesEvent read FOnUpdate write FOnUpdate; property OnRemove: TWatchesEvent read FOnRemove write FonRemove; end; { TWatchesNotificationList } TWatchesNotificationList = class(TDebuggerNotificationList) private function GetItem(AIndex: Integer): TWatchesNotification; public procedure NotifyAdd(const ASender: TCurrentWatches; const AWatch: TCurrentWatch); procedure NotifyUpdate(const ASender: TCurrentWatches; const AWatch: TCurrentWatch); procedure NotifyRemove(const ASender: TCurrentWatches; const AWatch: TCurrentWatch); property Items[AIndex: Integer]: TWatchesNotification read GetItem; default; end; { TWatchValue } { TIdeWatchValue } TIdeWatchValue = class(TWatchValue) private function GetWatch: TIdeWatch; protected function GetTypeInfo: TDBGType; override; function GetValue: String; override; procedure RequestData; virtual; procedure LoadDataFromXMLConfig(const AConfig: TXMLConfig; const APath: string); procedure SaveDataToXMLConfig(const AConfig: TXMLConfig; const APath: string); public constructor Create(AOwnerWatch: TIdeWatch); constructor Create(AOwnerWatch: TIdeWatch; const AThreadId: Integer; const AStackFrame: Integer ); procedure Assign(AnOther: TWatchValue); override; property Watch: TIdeWatch read GetWatch; end; { TIdeWatchValueList } TIdeWatchValueList = class(TWatchValueList) private function GetEntry(const AThreadId: Integer; const AStackFrame: Integer): TIdeWatchValue; function GetEntryByIdx(AnIndex: integer): TIdeWatchValue; function GetWatch: TIdeWatch; protected function CopyEntry(AnEntry: TWatchValue): TWatchValue; override; procedure LoadDataFromXMLConfig(const AConfig: TXMLConfig; APath: string); procedure SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string); public constructor Create(AOwnerWatch: TIdeWatch); property EntriesByIdx[AnIndex: integer]: TIdeWatchValue read GetEntryByIdx; property Entries[const AThreadId: Integer; const AStackFrame: Integer]: TIdeWatchValue read GetEntry; default; property Watch: TIdeWatch read GetWatch; end; { TIdeWatch } TIdeWatch = class(TWatch) private function GetValue(const AThreadId: Integer; const AStackFrame: Integer): TIdeWatchValue; protected function CreateValueList: TWatchValueList; override; procedure DoEnableChange; override; procedure DoExpressionChange; override; procedure DoDisplayFormatChanged; override; protected procedure LoadDataFromXMLConfig(const AConfig: TXMLConfig; const APath: string); procedure SaveDataToXMLConfig(const AConfig: TXMLConfig; const APath: string); public constructor Create(ACollection: TCollection); override; procedure ClearValues; override; public property Values[const AThreadId: Integer; const AStackFrame: Integer]: TIdeWatchValue read GetValue; end; { TIdeWatches } TIdeWatches = class(TWatches) private function GetItem(const AnIndex: Integer): TIdeWatch; procedure SetItem(const AnIndex: Integer; const AValue: TIdeWatch); protected function WatchClass: TWatchClass; override; procedure LoadDataFromXMLConfig(const AConfig: TXMLConfig; APath: string); procedure SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string); public function Add(const AExpression: String): TIdeWatch; function Find(const AExpression: String): TIdeWatch; reintroduce; property Items[const AnIndex: Integer]: TIdeWatch read GetItem write SetItem; default; end; { TCurrentWatchValue } TCurrentWatchValue = class(TIdeWatchValue) private FSnapShot: TIdeWatchValue; procedure SetSnapShot(const AValue: TIdeWatchValue); protected procedure RequestData; override; procedure DoDataValidityChanged(AnOldValidity: TDebuggerDataState); override; public property SnapShot: TIdeWatchValue read FSnapShot write SetSnapShot; end; { TCurrentWatchValueList } TCurrentWatchValueList = class(TIdeWatchValueList) private FSnapShot: TIdeWatchValueList; procedure SetSnapShot(const AValue: TIdeWatchValueList); protected function CreateEntry(const AThreadId: Integer; const AStackFrame: Integer): TIdeWatchValue; override; property SnapShot: TIdeWatchValueList read FSnapShot write SetSnapShot; end; { TCurrentWatch } TCurrentWatch = class(TIdeWatch) private FSnapShot: TIdeWatch; procedure SetSnapShot(const AValue: TIdeWatch); protected function CreateValueList: TWatchValueList; override; procedure DoChanged; override; procedure DoModified; override; procedure RequestData(AWatchValue: TCurrentWatchValue); property SnapShot: TIdeWatch read FSnapShot write SetSnapShot; public destructor Destroy; override; procedure LoadFromXMLConfig(const AConfig: TXMLConfig; const APath: string); virtual; procedure SaveToXMLConfig(const AConfig: TXMLConfig; const APath: string); virtual; end; TIDEWatchClass = class of TCurrentWatch; { TCurrentWatches } TCurrentWatches = class(TIdeWatches) private FMonitor: TIdeWatchesMonitor; FSnapShot: TIdeWatches; FDestroying: Boolean; procedure SetSnapShot(const AValue: TIdeWatches); procedure WatchesChanged(Sender: TObject); protected function GetItem(const AnIndex: Integer): TCurrentWatch; procedure SetItem(const AnIndex: Integer; const AValue: TCurrentWatch); protected function WatchClass: TWatchClass; override; procedure NotifyAdd(const AWatch: TCurrentWatch); virtual; // called when a watch is added procedure NotifyRemove(const AWatch: TCurrentWatch); virtual; // called by watch when destructed procedure DoModified; procedure Update(Item: TCollectionItem); override; procedure RequestData(AWatchValue: TCurrentWatchValue); property SnapShot: TIdeWatches read FSnapShot write SetSnapShot; public constructor Create(AMonitor: TIdeWatchesMonitor); destructor Destroy; override; // Watch function Add(const AExpression: String): TCurrentWatch; function Find(const AExpression: String): TCurrentWatch; reintroduce; // IDE procedure LoadFromXMLConfig(const AConfig: TXMLConfig; const APath: string); procedure SaveToXMLConfig(const AConfig: TXMLConfig; const APath: string); public property Items[const AnIndex: Integer]: TCurrentWatch read GetItem write SetItem; default; end; { TIdeWatchesMonitor } TIdeWatchesMonitor = class(TWatchesMonitor) private FSnapshots: TDebuggerDataSnapShotList; FOnModified: TNotifyEvent; FIgnoreModified: Integer; FNotificationList: TWatchesNotificationList; function GetCurrentWatches: TCurrentWatches; function GetSnapshot(AnID: Pointer): TIdeWatches; protected procedure DoStateEnterPause; override; procedure DoStateLeavePause; override; procedure DoStateLeavePauseClean; override; procedure DoModified; override; //procedure NotifyChange procedure NotifyAdd(const AWatches: TCurrentWatches; const AWatch: TCurrentWatch); procedure NotifyRemove(const AWatches: TCurrentWatches; const AWatch: TCurrentWatch); procedure NotifyUpdate(const AWatches: TCurrentWatches; const AWatch: TCurrentWatch); procedure RequestData(AWatchValue: TCurrentWatchValue); function CreateWatches: TWatches; override; function CreateSnapshot(CreateEmpty: Boolean = False): TObject; public constructor Create; destructor Destroy; override; procedure AddNotification(const ANotification: TWatchesNotification); procedure RemoveNotification(const ANotification: TWatchesNotification); procedure NewSnapshot(AnID: Pointer; CreateEmpty: Boolean = False); procedure RemoveSnapshot(AnID: Pointer); property CurrentWatches: TCurrentWatches read GetCurrentWatches;// FCurrentWatches; property Snapshots[AnID: Pointer]: TIdeWatches read GetSnapshot; public procedure Clear; procedure LoadFromXMLConfig(const AConfig: TXMLConfig; const APath: string); procedure SaveToXMLConfig(const AConfig: TXMLConfig; const APath: string); procedure BeginIgnoreModified; procedure EndIgnoreModified; property OnModified: TNotifyEvent read FOnModified write FOnModified; // user-modified / xml-storable data modified end; {%endregion ^^^^^ Watches ^^^^^ } {%region Locals *************************************************************** ****************************************************************************** ** ** ** L O C A L S ** ** ** ****************************************************************************** ******************************************************************************} TLocalsNotification = class(TDebuggerChangeNotification) public property OnChange; end; { TIDELocals } TIDELocals = class(TLocals) protected procedure LoadDataFromXMLConfig(const AConfig: TXMLConfig; APath: string); procedure SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string); public constructor CreateFromXMLConfig(const AConfig: TXMLConfig; APath: string); procedure SetDataValidity(AValidity: TDebuggerDataState); override; end; { TCurrentLocals } TCurrentLocals = class(TIDELocals) private FMonitor: TIdeLocalsMonitor; FSnapShot: TIDELocals; FDataValidity: TDebuggerDataState; procedure SetSnapShot(const AValue: TIDELocals); protected property SnapShot: TIDELocals read FSnapShot write SetSnapShot; public constructor Create(AMonitor: TIdeLocalsMonitor; AThreadId, AStackFrame: Integer); function Count: Integer; override; procedure SetDataValidity(AValidity: TDebuggerDataState); override; end; { TLocalsList } { TIDELocalsList } TIDELocalsList = class(TLocalsList) private function GetEntry(const AThreadId: Integer; const AStackFrame: Integer): TIDELocals; function GetEntryByIdx(const AnIndex: Integer): TIDELocals; protected function CreateEntry(AThreadId, AStackFrame: Integer): TDbgEntityValuesList; override; procedure DoAssign(AnOther: TDbgEntitiesThreadStackList); override; procedure DoAdded(AnEntry: TDbgEntityValuesList); override; procedure LoadDataFromXMLConfig(const AConfig: TXMLConfig; APath: string); procedure SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string); public property EntriesByIdx[const AnIndex: Integer]: TIDELocals read GetEntryByIdx; property Entries[const AThreadId: Integer; const AStackFrame: Integer]: TIDELocals read GetEntry; default; end; { TCurrentLocalsList } TCurrentLocalsList = class(TIDELocalsList) private FMonitor: TIdeLocalsMonitor; FSnapShot: TIDELocalsList; procedure SetSnapShot(const AValue: TIDELocalsList); protected procedure DoCleared; override; procedure DoAdded(AnEntry: TDbgEntityValuesList); override; function CreateEntry(AThreadId, AStackFrame: Integer): TIDELocals; override; property SnapShot: TIDELocalsList read FSnapShot write SetSnapShot; public constructor Create(AMonitor: TIdeLocalsMonitor); end; { TIdeLocalsMonitor } TIdeLocalsMonitor = class(TLocalsMonitor) private FSnapshots: TDebuggerDataSnapShotList; FNotificationList: TDebuggerChangeNotificationList; function GetCurrentLocalsList: TCurrentLocalsList; function GetSnapshot(AnID: Pointer): TIDELocalsList; protected procedure DoStateEnterPause; override; procedure DoStateLeavePause; override; procedure DoStateLeavePauseClean; override; procedure NotifyChange(ALocals: TCurrentLocals); procedure DoNewSupplier; override; procedure RequestData(ALocals: TCurrentLocals); function CreateSnapshot(CreateEmpty: Boolean = False): TObject; function CreateLocalsList: TLocalsList; override; public constructor Create; destructor Destroy; override; procedure Clear; procedure AddNotification(const ANotification: TLocalsNotification); procedure RemoveNotification(const ANotification: TLocalsNotification); procedure NewSnapshot(AnID: Pointer; CreateEmpty: Boolean = False); procedure RemoveSnapshot(AnID: Pointer); property CurrentLocalsList: TCurrentLocalsList read GetCurrentLocalsList; property Snapshots[AnID: Pointer]: TIDELocalsList read GetSnapshot; end; {%endregion ^^^^^ Locals ^^^^^ } {%region Line Info ************************************************************ ****************************************************************************** ** ** ** L I N E I N F O ** ** ** ****************************************************************************** ******************************************************************************} { TIDELineInfo } TIDELineInfoNotification = class(TDebuggerNotification) private FOnChange: TIDELineInfoEvent; public property OnChange: TIDELineInfoEvent read FOnChange write FOnChange; end; TIDELineInfo = class(TBaseLineInfo) private FNotificationList: TList; FMaster: TDBGLineInfo; procedure LineInfoChanged(const {%H-}ASender: TObject; const ASource: String); procedure SetMaster(const AMaster: TDBGLineInfo); protected function GetSource(const AIndex: Integer): String; override; protected procedure NotifyChange(ASource: String); public constructor Create; destructor Destroy; override; procedure AddNotification(const ANotification: TIDELineInfoNotification); procedure RemoveNotification(const ANotification: TIDELineInfoNotification); function Count: Integer; override; function GetAddress(const AIndex: Integer; const ALine: Integer): TDbgPtr; override; function GetInfo(AAdress: TDbgPtr; out ASource, ALine, AOffset: Integer): Boolean; override; function IndexOf(const ASource: String): integer; override; procedure Request(const ASource: String); override; procedure Cancel(const ASource: String); override; property Master: TDBGLineInfo read FMaster write SetMaster; end; {%endregion ^^^^^ Line Info ^^^^^ } {%region Register ************************************************************* ****************************************************************************** ** ** ** R E G I S T E R S ** ** ** ****************************************************************************** ******************************************************************************} TIdeRegistersMonitor = class; TRegistersNotification = class(TDebuggerChangeNotification) public property OnChange; end; { TIDERegisterValue } TIDERegisterValue = class(TRegisterValue) protected procedure DoDataValidityChanged(AnOldValidity: TDebuggerDataState); override; procedure DoDisplayFormatChanged(AnOldFormat: TRegisterDisplayFormat); override; end; { TIDERegisters } TIDERegisters = class(TRegisters) protected function CreateEntry: TDbgEntityValue; override; end; { TCurrentIDERegisters } TCurrentIDERegisters = class(TIDERegisters) private FMonitor: TIdeRegistersMonitor; protected procedure DoDataValidityChanged(AnOldValidity: TDebuggerDataState); override; public constructor Create(AMonitor: TIdeRegistersMonitor; AThreadId, AStackFrame: Integer); function Count: Integer; override; end; TIDERegistersList = class(TRegistersList) private //function GetEntry(const AThreadId: Integer; const AStackFrame: Integer): TIDERegisters; //function GetEntryByIdx(const AnIndex: Integer): TIDERegisters; protected //function CreateEntry(AThreadId, AStackFrame: Integer): TDbgEntityValuesList; override; // TIDERegisters //procedure DoAssign(AnOther: TDbgEntitiesThreadStackList); override; // Immutable // XML public //property EntriesByIdx[const AnIndex: Integer]: TIDERegisters read GetEntryByIdx; //property Entries[const AThreadId: Integer; const AStackFrame: Integer]: TIDERegisters // read GetEntry; default; end; { TCurrentIDERegistersList } TCurrentIDERegistersList = class(TIDERegistersList) private FMonitor: TIdeRegistersMonitor; protected procedure DoCleared; override; function CreateEntry(AThreadId, AStackFrame: Integer): TRegisters; override; // TIDERegisters public constructor Create(AMonitor: TIdeRegistersMonitor); end; { TIdeRegistersMonitor } TIdeRegistersMonitor = class(TRegistersMonitor) private FNotificationList: TDebuggerChangeNotificationList; FFlags: set of (rmNeedNotifyChange); function GetCurrentRegistersList: TCurrentIDERegistersList; protected procedure DoStateEnterPause; override; //procedure DoStateLeavePause; override; procedure DoStateLeavePauseClean; override; procedure DoEndUpdate; override; procedure NotifyChange(ARegisters: TCurrentIDERegisters); procedure DoNewSupplier; override; procedure RequestData(ARegisters: TCurrentIDERegisters); //function CreateSnapshot(CreateEmpty: Boolean = False): TObject; override; function CreateRegistersList: TRegistersList; override; public constructor Create; destructor Destroy; override; procedure Clear; procedure AddNotification(const ANotification: TRegistersNotification); procedure RemoveNotification(const ANotification: TRegistersNotification); property CurrentRegistersList: TCurrentIDERegistersList read GetCurrentRegistersList; //property Snapshots[AnID: Pointer]: TIDERegistersList read GetSnapshot; end; {%endregion ^^^^^ Register ^^^^^ } {%region Callstack ************************************************************ ****************************************************************************** ** ** ** C A L L S T A C K ** ** ** ****************************************************************************** ****************************************************************************** * The entries for the callstack are created on demand. This way when the * * first entry is needed, it isn't required to create the whole stack * * * * TCallStackEntry needs to stay a readonly object so its data can be shared * ******************************************************************************} { TCallStackNotification } TCallStackNotification = class(TDebuggerChangeNotification) public property OnChange; property OnCurrent; end; { TCallStackEntry } { TIdeCallStackEntry } TIdeCallStackEntry = class(TCallStackEntry) private FOwner: TIdeCallStack; FUnitInfo: TDebuggerUnitInfo; procedure SetUnitInfo(AUnitInfo: TDebuggerUnitInfo); protected function GetUnitInfoProvider: TDebuggerUnitInfoProvider; virtual; protected function GetFunctionName: String; override; function GetSource: String; override; procedure LoadDataFromXMLConfig(const AConfig: TXMLConfig; const APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil ); procedure SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil ); public constructor Create(const AIndex:Integer; const AnAdress: TDbgPtr; const AnArguments: TStrings; const AFunctionName: String; const AUnitInfo: TDebuggerUnitInfo; const ALine: Integer; AState: TDebuggerDataState = ddsValid); overload; function CreateCopy: TCallStackEntry; override; procedure Assign(AnOther: TCallStackEntry); override; destructor Destroy; override; procedure Init(const AnAdress: TDbgPtr; const AnArguments: TStrings; const AFunctionName: String; const AUnitName, AClassName, AProcName, AFunctionArgs: String; const ALine: Integer; AState: TDebuggerDataState = ddsValid); override; procedure Init(const AnAdress: TDbgPtr; const AnArguments: TStrings; const AFunctionName: String; const FileName, FullName: String; const ALine: Integer; AState: TDebuggerDataState = ddsValid); override; procedure ClearLocation; override; // TODO need a way to call Changed on TCallStack or TThreads // corrently done in SetThreadState function IsCurrent: Boolean; procedure MakeCurrent; property UnitInfo: TDebuggerUnitInfo read FUnitInfo; end; { TIdeCallStack } TIdeCallStack = class(TCallStackBase) private FList: TList; protected function IndexError(AIndex: Integer): TIdeCallStackEntry; function GetEntryBase(AIndex: Integer): TCallStackEntry; override; function GetRawEntries: TMap; override; function GetNewCurrentIndex: Integer; override; procedure Clear; virtual; function GetCount: Integer; override; procedure SetCount({%H-}ACount: Integer); override; function GetEntry(AIndex: Integer): TIdeCallStackEntry; virtual; procedure AddEntry(AnEntry: TIdeCallStackEntry); virtual; // must be added in correct order procedure AssignEntriesTo(AnOther: TIdeCallStack); virtual; procedure LoadDataFromXMLConfig(const AConfig: TXMLConfig; APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil ); procedure SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil ); public procedure DoEntriesCreated; override; procedure DoEntriesUpdated; override; procedure SetCountValidity(AValidity: TDebuggerDataState); override; procedure SetHasAtLeastCountInfo(AValidity: TDebuggerDataState; AMinCount: Integer = - 1); override; procedure SetCurrentValidity(AValidity: TDebuggerDataState); override; public constructor Create; function CreateCopy: TCallStackBase; override; destructor Destroy; override; procedure Assign(AnOther: TCallStackBase); override; procedure PrepareRange({%H-}AIndex, {%H-}ACount: Integer); override; procedure ChangeCurrentIndex(ANewIndex: Integer); virtual; function HasAtLeastCount(ARequiredMinCount: Integer): TNullableBool; virtual; // Can be faster than getting the full count function CountLimited(ALimit: Integer): Integer; override; property Entries[AIndex: Integer]: TIdeCallStackEntry read GetEntry; end; { TCallStackList } TIdeCallStackList = class(TCallStackList) private function GetEntry(const AIndex: Integer): TIdeCallStack; function GetEntryForThread(const AThreadId: Integer): TIdeCallStack; protected procedure LoadDataFromXMLConfig(const AConfig: TXMLConfig; APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil ); procedure SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil ); public property Entries[const AIndex: Integer]: TIdeCallStack read GetEntry; default; property EntriesForThreads[const AThreadId: Integer]: TIdeCallStack read GetEntryForThread; end; { TCurrentCallStack } TCurrentCallStack = class(TIdeCallStack) private FMonitor: TIdeCallStackMonitor; FCountValidity, FAtLeastCountValidity: TDebuggerDataState; FCurrentValidity: TDebuggerDataState; FNewCurrentIndex: Integer; FPreparing: Boolean; FSnapShot: TIdeCallStack; FEntries: TMap; // list of created entries FCount, FAtLeastCount, FAtLeastCountOld: Integer; FLowestUnknown, FHighestUnknown: Integer; procedure SetSnapShot(const AValue: TIdeCallStack); protected function GetCurrent: Integer; override; procedure SetCurrent(AValue: Integer); override; procedure Clear; override; function GetCount: Integer; override; procedure SetCount(ACount: Integer); override; function GetEntry(AIndex: Integer): TIdeCallStackEntry; override; procedure AddEntry(AnEntry: TIdeCallStackEntry); override; procedure AssignEntriesTo(AnOther: TIdeCallStack); override; function GetRawEntries: TMap; override; function GetLowestUnknown: Integer; override; function GetHighestUnknown: Integer; override; function GetNewCurrentIndex: Integer; override; public constructor Create(AMonitor: TIdeCallStackMonitor); destructor Destroy; override; procedure Assign(AnOther: TCallStackBase); override; procedure PrepareRange(AIndex, ACount: Integer); override; procedure ChangeCurrentIndex(ANewIndex: Integer); override; procedure DoEntriesCreated; override; procedure DoEntriesUpdated; override; function HasAtLeastCount(ARequiredMinCount: Integer): TNullableBool; override; property NewCurrentIndex: Integer read FNewCurrentIndex; property SnapShot: TIdeCallStack read FSnapShot write SetSnapShot; public procedure SetCountValidity(AValidity: TDebuggerDataState); override; procedure SetHasAtLeastCountInfo(AValidity: TDebuggerDataState; AMinCount: Integer = -1); override; procedure SetCurrentValidity(AValidity: TDebuggerDataState); override; end; { TCurrentCallStackList } TCurrentCallStackList = class(TIdeCallStackList) private FMonitor: TIdeCallStackMonitor; FSnapShot: TIdeCallStackList; procedure SetSnapShot(const AValue: TIdeCallStackList); protected function NewEntryForThread(const AThreadId: Integer): TCallStackBase; override; property SnapShot: TIdeCallStackList read FSnapShot write SetSnapShot; public constructor Create(AMonitor: TIdeCallStackMonitor); end; { TIdeCallStackMonitor } TIdeCallStackMonitor = class(TCallStackMonitor) private FSnapshots: TDebuggerDataSnapShotList; FNotificationList: TDebuggerChangeNotificationList; FUnitInfoProvider: TDebuggerUnitInfoProvider; procedure CallStackClear(Sender: TObject); function GetCurrentCallStackList: TCurrentCallStackList; function GetSnapshot(AnID: Pointer): TIdeCallStackList; protected procedure DoStateEnterPause; override; procedure DoStateLeavePause; override; procedure DoStateLeavePauseClean; override; procedure DoModified; override; procedure RequestCount(ACallstack: TIdeCallStack); procedure RequestAtLeastCount(ACallstack: TIdeCallStack; ARequiredMinCount: Integer); procedure RequestCurrent(ACallstack: TIdeCallStack); procedure RequestEntries(ACallstack: TIdeCallStack); procedure UpdateCurrentIndex; procedure DoNewSupplier; override; function CreateSnapshot(CreateEmpty: Boolean = False): TObject; function CreateCallStackList: TCallStackList; override; public constructor Create; destructor Destroy; override; procedure AddNotification(const ANotification: TCallStackNotification); procedure RemoveNotification(const ANotification: TCallStackNotification); procedure NewSnapshot(AnID: Pointer; CreateEmpty: Boolean = False); procedure RemoveSnapshot(AnID: Pointer); procedure NotifyChange; // (sender) procedure NotifyCurrent; property CurrentCallStackList: TCurrentCallStackList read GetCurrentCallStackList; property Snapshots[AnID: Pointer]: TIdeCallStackList read GetSnapshot; property UnitInfoProvider: TDebuggerUnitInfoProvider // Provided by DebugBoss, to map files to packages or project read FUnitInfoProvider write FUnitInfoProvider; end; {%endregion ^^^^^ Callstack ^^^^^ } {%region ***** Disassembler ***** } (******************************************************************************) (******************************************************************************) (** **) (** D I S A S S E M B L E R **) (** **) (******************************************************************************) (******************************************************************************) { TIDEDisassemblerNotification } TIDEDisassemblerNotification = class(TDebuggerNotification) private FOnChange: TNotifyEvent; public property OnChange: TNotifyEvent read FOnChange write FOnChange; end; TIDEDisassembler = class(TBaseDisassembler) private FNotificationList: TList; FMaster: TDBGDisassembler; procedure DisassemblerChanged(Sender: TObject); procedure SetMaster(AMaster: TDBGDisassembler); protected procedure DoChanged; override; function InternalGetEntry(AIndex: Integer): TDisassemblerEntry; override; function InternalGetEntryPtr(AIndex: Integer): PDisassemblerEntry; override; public constructor Create; destructor Destroy; override; procedure AddNotification(const ANotification: TIDEDisassemblerNotification); procedure RemoveNotification(const ANotification: TIDEDisassemblerNotification); procedure Clear; override; function PrepareRange(AnAddr: TDbgPtr; ALinesBefore, ALinesAfter: Integer): Boolean; override; property Master: TDBGDisassembler read FMaster write SetMaster; end; {%endregion ^^^^^ Disassembler ^^^^^ } {%region Threads ************************************************************** ****************************************************************************** ** ** ** T H R E A D S ** ** ** ****************************************************************************** ******************************************************************************} { TThreadsNotification } TThreadsNotification = class(TDebuggerChangeNotification) public property OnChange; // fires for all changes (incl OnCurrent) property OnCurrent; end; TIdeThreadEntry = class; TIdeThreads = class; { TIdeThreadFrameEntry } TIdeThreadFrameEntry = class(TIdeCallStackEntry) private FThread: TIdeThreadEntry; protected function GetUnitInfoProvider: TDebuggerUnitInfoProvider; override; end; { TThreadEntry } { TIdeThreadEntry } TIdeThreadEntry = class(TThreadEntry) private FThreadOwner: TIdeThreads; function GetTopFrame: TIdeThreadFrameEntry; protected function CreateStackEntry: TCallStackEntry; override; function GetUnitInfoProvider: TDebuggerUnitInfoProvider; procedure SetThreadState(AValue: String); override; procedure LoadDataFromXMLConfig(const AConfig: TXMLConfig; const APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil ); reintroduce; procedure SaveDataToXMLConfig(const AConfig: TXMLConfig; const APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil ); reintroduce; public function CreateCopy: TThreadEntry; override; property TopFrame: TIdeThreadFrameEntry read GetTopFrame; end; { TIdeThreads } TIdeThreads = class(TThreads) private function GetEntry(const AnIndex: Integer): TIdeThreadEntry; function GetEntryById(const AnID: Integer): TIdeThreadEntry; protected procedure LoadDataFromXMLConfig(const AConfig: TXMLConfig; APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil ); procedure SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil ); public function CreateEntry(const AnAdress: TDbgPtr; const AnArguments: TStrings; const AFunctionName: String; const FileName, FullName: String; const ALine: Integer; const AThreadId: Integer; const AThreadName: String; const AThreadState: String; AState: TDebuggerDataState = ddsValid): TThreadEntry; override; procedure SetValidity(AValidity: TDebuggerDataState); override; property Entries[const AnIndex: Integer]: TIdeThreadEntry read GetEntry; default; property EntryById[const AnID: Integer]: TIdeThreadEntry read GetEntryById; end; { TCurrentThreads } TCurrentThreads = class(TIdeThreads) private FMonitor: TIdeThreadsMonitor; FDataValidity: TDebuggerDataState; FSnapShot: TIdeThreads; procedure SetSnapShot(const AValue: TIdeThreads); protected Paused: Boolean; // Todo: introduce Supplie.ReadyForRequest procedure SetCurrentThreadId(AValue: Integer); override; property SnapShot: TIdeThreads read FSnapShot write SetSnapShot; public constructor Create(AMonitor: TIdeThreadsMonitor); function Count: Integer; override; procedure Clear; override; function CreateEntry(const AnAdress: TDbgPtr; const AnArguments: TStrings; const AFunctionName: String; const FileName, FullName: String; const ALine: Integer; const AThreadId: Integer; const AThreadName: String; const AThreadState: String; AState: TDebuggerDataState = ddsValid): TThreadEntry; override; procedure SetValidity(AValidity: TDebuggerDataState); override; end; { TIdeThreadsMonitor } TIdeThreadsMonitor = class(TThreadsMonitor) private FSnapshots: TDebuggerDataSnapShotList; FUnitInfoProvider: TDebuggerUnitInfoProvider; FNotificationList: TDebuggerChangeNotificationList; function GetCurrentThreads: TCurrentThreads; function GetSnapshot(AnID: Pointer): TIdeThreads; protected procedure DoModified; override; procedure DoStateEnterPause; override; procedure DoStateLeavePause; override; procedure DoStateLeavePauseClean; override; procedure DoNewSupplier; override; procedure Changed; procedure RequestData; function CreateSnapshot(CreateEmpty: Boolean = False): TObject; function CreateThreads: TThreads; override; public constructor Create; destructor Destroy; override; procedure Clear; procedure AddNotification(const ANotification: TThreadsNotification); procedure RemoveNotification(const ANotification: TThreadsNotification); procedure NewSnapshot(AnID: Pointer; CreateEmpty: Boolean = False); procedure RemoveSnapshot(AnID: Pointer); procedure ChangeCurrentThread(ANewId: Integer); procedure CurrentChanged; property CurrentThreads: TCurrentThreads read GetCurrentThreads; property Snapshots[AnID: Pointer]: TIdeThreads read GetSnapshot; property UnitInfoProvider: TDebuggerUnitInfoProvider // Provided by DebugBoss, to map files to packages or project read FUnitInfoProvider write FUnitInfoProvider; end; {%endregion ^^^^^ Threads ^^^^^ } {%region ***** Snapshots ***** } TSnapshotNotification = class(TDebuggerChangeNotification) public property OnChange; // fires for all changes (incl OnCurrent) property OnCurrent; end; { TSnapshot } TSnapshot = class(TRefCountedObject) private FLocation: TDBGLocationRec; FTimeStamp: TDateTime; FSnapMgr: TSnapshotManager; function GetLocationAsText: String; protected procedure LoadDataFromXMLConfig(const AConfig: TXMLConfig; APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil ); procedure SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil ); public constructor Create(ASnapMgr: TSnapshotManager); destructor Destroy; override; property TimeStamp: TDateTime read FTimeStamp; property Location: TDBGLocationRec read FLocation write FLocation; property LocationAsText: String read GetLocationAsText; public procedure AddToSnapshots; procedure AddToHistory; procedure RemoveFromSnapshots; procedure RemoveFromHistory; function IsCurrent: Boolean; function IsHistory: Boolean; function IsSnapshot: Boolean; end; { TSnapshotList } TSnapshotList = class(TRefCntObjList) private function Get(Index: Integer): TSnapshot; procedure Put(Index: Integer; const AValue: TSnapshot); public property Items[Index: Integer]: TSnapshot read Get write Put; default; end; { TSnapshotManager } TSnapshotManagerRequestedFlags = set of (smrThreads, smrCallStackCnt, smrCallStack, smrLocals, smrWatches); TSnapshotManager = class private FDebugger: TDebuggerIntf; FNotificationList: TDebuggerChangeNotificationList; FLocals: TIdeLocalsMonitor; FWatches: TIdeWatchesMonitor; FCallStack: TIdeCallStackMonitor; FCallStackNotification: TCallStackNotification; FThreads: TIdeThreadsMonitor; procedure SetCallStack(AValue: TIdeCallStackMonitor); procedure DoCallStackChanged(Sender: TObject); private FActive: Boolean; FForcedIdle: Boolean; FUnitInfoProvider: TDebuggerUnitInfoProvider; FUpdateLock: Integer; FUpdateFlags: set of (ufSnapChanged, ufSnapCurrent, ufInDebuggerIdle); FCurrentState: TDBGState; FRequestsDone: TSnapshotManagerRequestedFlags; FCurrentSnapshot: TSnapshot; // snapshot for current pause. Not yet in list procedure SetActive(const AValue: Boolean); procedure SetDebugger(AValue: TDebuggerIntf); protected FHistoryCapacity: Integer; FHistoryIndex: Integer; FHistoryList: TSnapshotList; FHistorySelected: Boolean; function GetHistoryEntry(AIndex: Integer): TSnapshot; procedure SetHistoryIndex(const AValue: Integer); procedure SetHistorySelected(AValue: Boolean); procedure CreateHistoryEntry; procedure RemoveHistoryEntry(AIndex: Integer); procedure RemoveHistoryEntry(ASnapShot: TSnapshot); procedure RemoveHistoryEntryFromMonitors(AnEntry: TSnapshot); protected FSnapshotIndex: Integer; FSnapshotList: TSnapshotList; FSnapshotSelected: Boolean; function GetSnapshotEntry(AIndex: Integer): TSnapshot; procedure SetSnapshotIndex(const AValue: Integer); procedure SetSnapshotSelected(AValue: Boolean); procedure AddSnapshotEntry(ASnapShot: TSnapshot); procedure RemoveSnapshotEntry(ASnapShot: TSnapshot); procedure AddHistoryEntry(ASnapShot: TSnapshot); protected procedure DoSnapShotDestroy(ASnapShot: TSnapshot); procedure BeginUpdate; procedure EndUpdate; procedure DoChanged; procedure DoCurrent; protected procedure LoadDataFromXMLConfig(const AConfig: TXMLConfig; APath: string); procedure SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string); public constructor Create; destructor Destroy; override; procedure AddNotification(const ANotification: TSnapshotNotification); procedure RemoveNotification(const ANotification: TSnapshotNotification); procedure DoStateChange(const AOldState: TDBGState); procedure DoDebuggerIdle(AForce: Boolean = False); property Active: Boolean read FActive write SetActive; public function SelectedId: Pointer; function SelectedEntry: TSnapshot; procedure Clear; procedure ClearHistory; procedure ClearSnapshots; function GetAsXML: String; procedure SetFromXML(aXML: String); property Current: TSnapshot read FCurrentSnapshot; public property HistoryIndex: Integer read FHistoryIndex write SetHistoryIndex; property HistoryCapacity: Integer read FHistoryCapacity write FHistoryCapacity; property HistorySelected: Boolean read FHistorySelected write SetHistorySelected; property History: TSnapshotList read FHistoryList; public property SnapshotIndex: Integer read FSnapshotIndex write SetSnapshotIndex; property SnapshotSelected: Boolean read FSnapshotSelected write SetSnapshotSelected; property Snapshots: TSnapshotList read FSnapshotList; public property Locals: TIdeLocalsMonitor read FLocals write FLocals; property Watches: TIdeWatchesMonitor read FWatches write FWatches; property CallStack: TIdeCallStackMonitor read FCallStack write SetCallStack; property Threads: TIdeThreadsMonitor read FThreads write FThreads; property Debugger: TDebuggerIntf read FDebugger write SetDebugger; property UnitInfoProvider: TDebuggerUnitInfoProvider read FUnitInfoProvider write FUnitInfoProvider; end; {%endregion ^^^^^ Snapshots ^^^^^ } {%region Signals / Exceptions *************************************************} (******************************************************************************) (** **) (** S I G N A L S and E X C E P T I O N S **) (** **) (******************************************************************************) (******************************************************************************) { TIDESignal } TIDESignal = class(TBaseSignal) private FMaster: TDBGSignal; protected procedure AssignTo(Dest: TPersistent); override; public procedure LoadFromXMLConfig(const {%H-}AXMLConfig: TXMLConfig; const {%H-}APath: string); procedure SaveToXMLConfig(const {%H-}AXMLConfig: TXMLConfig; const {%H-}APath: string); procedure ResetMaster; end; { TIDESignals } TIDESignals = class(TBaseSignals) private FMaster: TDBGSignals; procedure SetMaster(const AValue: TDBGSignals); function GetItem(const AIndex: Integer): TIDESignal; procedure SetItem(const AIndex: Integer; const AValue: TIDESignal); protected procedure AddDefault; public constructor Create; procedure Reset; override; function Add(const AName: String; AID: Integer): TIDESignal; function Find(const AName: String): TIDESignal; property Master: TDBGSignals read FMaster write SetMaster; public procedure LoadFromXMLConfig(const {%H-}AXMLConfig: TXMLConfig; const {%H-}APath: string); procedure SaveToXMLConfig(const {%H-}AXMLConfig: TXMLConfig; const {%H-}APath: string); property Items[const AIndex: Integer]: TIDESignal read GetItem write SetItem; default; end; { TIDEException } TIDEException = class(TBaseException) private FMaster: TDBGException; public constructor Create(ACollection: TCollection); override; procedure LoadFromXMLConfig(const AXMLConfig: TXMLConfig; const APath: string); procedure SaveToXMLConfig(const AXMLConfig: TXMLConfig; const APath: string); procedure ResetMaster; end; { TIDEExceptions } TIDEExceptions = class(TBaseExceptions) private function GetItem(const AIndex: Integer): TIDEException; procedure SetItem(const AIndex: Integer; const AValue: TIDEException); protected procedure AddDefault; public function Add(const AName: String): TIDEException; function Find(const AName: String): TIDEException; public constructor Create; procedure LoadFromXMLConfig(const AXMLConfig: TXMLConfig; const APath: string); procedure SaveToXMLConfig(const AXMLConfig: TXMLConfig; const APath: string); procedure AddIfNeeded(AName: string); procedure Reset; override; property Items[const AIndex: Integer]: TIDEException read GetItem write SetItem; default; end; {%endregion ^^^^^ Signals / Exceptions ^^^^^ } (******************************************************************************) (******************************************************************************) (** **) (** D E B U G G E R **) (** **) (******************************************************************************) (******************************************************************************) TDBGEventRec = packed record case Boolean of False: ( Category: Word; EventType: Word); True: (Ptr: Pointer); end; { TDebugger } TDebugger = class(TDebuggerIntf) end; const DBGCommandNames: array[TDBGCommand] of string = ( 'Run', 'Pause', 'Stop', 'StepOver', 'StepInto', 'StepOut', 'RunTo', 'Jumpto', 'Attach', 'Detach', 'Break', 'Watch', 'Local', 'Evaluate', 'Modify', 'Environment', 'SetStackFrame', 'Disassemble', 'StepOverInstr', 'StepIntoInstr', 'SendConsoleInput' ); DBGStateNames: array[TDBGState] of string = ( 'None', 'Idle', 'Stop', 'Pause', 'InternalPause', 'Init', 'Run', 'Error', 'Destroying' ); DBGBreakPointActionNames: array[TIDEBreakPointAction] of string = ( 'Stop', 'EnableGroup', 'DisableGroup', 'LogMessage', 'EvalExpression', 'LogCallStack', 'TakeSnapshot' ); function DBGCommandNameToCommand(const s: string): TDBGCommand; function DBGStateNameToState(const s: string): TDBGState; deprecated; function DBGBreakPointActionNameToAction(const s: string): TIDEBreakPointAction; function dbgs(AFlag: TDebuggerLocationFlag): String; overload; function dbgs(AFlags: TDebuggerLocationFlags): String; overload; function HasConsoleSupport: Boolean; (******************************************************************************) (******************************************************************************) (******************************************************************************) (******************************************************************************) implementation var DBG_DATA_MONITORS, DBG_LOCATION_INFO: PLazLoggerLogGroup; function dbgs(AFlag: TDebuggerLocationFlag): String; begin writestr(Result{%H-}, AFlag); end; function dbgs(AFlags: TDebuggerLocationFlags): String; var i: TDebuggerLocationFlag; begin Result:=''; for i := low(TDebuggerLocationFlags) to high(TDebuggerLocationFlags) do if i in AFlags then begin if Result <> '' then Result := Result + ', '; Result := Result + dbgs(i); end; if Result <> '' then Result := '[' + Result + ']'; end; function HasConsoleSupport: Boolean; begin {$IFDEF DBG_ENABLE_TERMINAL} Result := True; {$ELSE} Result := False; {$ENDIF} end; function DBGCommandNameToCommand(const s: string): TDBGCommand; begin for Result:=Low(TDBGCommand) to High(TDBGCommand) do if AnsiCompareText(s,DBGCommandNames[Result])=0 then exit; Result:=dcStop; end; function DBGStateNameToState(const s: string): TDBGState; begin for Result:=Low(TDBGState) to High(TDBGState) do if AnsiCompareText(s,DBGStateNames[Result])=0 then exit; Result:=dsNone; end; function DBGBreakPointActionNameToAction(const s: string): TIDEBreakPointAction; begin for Result:=Low(TIDEBreakPointAction) to High(TIDEBreakPointAction) do if AnsiCompareText(s,DBGBreakPointActionNames[Result])=0 then exit; Result:=bpaStop; end; { TIdeThreadFrameEntry } function TIdeThreadFrameEntry.GetUnitInfoProvider: TDebuggerUnitInfoProvider; begin Result := FThread.GetUnitInfoProvider; end; { TIDEBreakPointGroupList } function TIDEBreakPointGroupList.GetItem(AIndex: Integer): TIDEBreakPointGroup; begin Result := TIDEBreakPointGroup(FList[AIndex]); end; constructor TIDEBreakPointGroupList.Create(AOwner: TIDEBreakPoint); begin FList := TFPList.Create; FOwner := AOwner; end; destructor TIDEBreakPointGroupList.Destroy; begin inherited Destroy; FList.Free; end; procedure TIDEBreakPointGroupList.Assign(ASrc: TIDEBreakPointGroupList); var i: Integer; begin Clear; for i := 0 to ASrc.Count - 1 do Add(ASrc[i]); end; procedure TIDEBreakPointGroupList.Clear; var i: Integer; begin for i:=0 to Count-1 do Items[i].RemoveReference(Self); FList.Clear; end; function TIDEBreakPointGroupList.Add(const AGroup: TIDEBreakPointGroup): Integer; begin if (AGroup = nil) or (IndexOf(AGroup) >= 0) then exit(-1); Result := FList.Add(AGroup); AGroup.AddReference(Self); FOwner.DoChanged; end; procedure TIDEBreakPointGroupList.Remove(const AGroup: TIDEBreakPointGroup); begin if (AGroup = nil) then exit; AGroup.RemoveReference(Self); if (IndexOf(AGroup) < 0) then exit; FList.Remove(AGroup); FOwner.DoChanged; end; function TIDEBreakPointGroupList.IndexOf(const AGroup: TIDEBreakPointGroup): Integer; begin Result := FList.IndexOf(AGroup); end; function TIDEBreakPointGroupList.Count: Integer; begin Result := FList.Count; end; { TDebuggerWatchesDlgConfig } constructor TDebuggerWatchesDlgConfig.Create; begin Init; end; procedure TDebuggerWatchesDlgConfig.Init; begin FColumnNameWidth := -1; FColumnValueWidth := -1; end; { TDebuggerConfigStoreBase } procedure TDebuggerConfigStoreBase.Init; begin // end; procedure TDebuggerConfigStoreBase.Load; begin Init; ConfigStore.ReadObject('', self); end; procedure TDebuggerConfigStoreBase.Save; begin ConfigStore.WriteObject('', self); end; { TDebuggerConfigStore } procedure TDebuggerConfigStore.Load; const OLD_GDB_DBG_NAME = 'GNU debugger (gdb)'; OLD_SSH_DBG_NAME = 'GNU debugger through SSH (gdb)'; var s: String; begin inherited; FDebuggerClass := ConfigStore.GetValue('Class', ''); if FDebuggerClass='' then begin // try old format s := ConfigStore.GetValue('Type', ''); if s = OLD_GDB_DBG_NAME then FDebuggerClass:='TGDBMIDEBUGGER'; if s = OLD_SSH_DBG_NAME then FDebuggerClass:='TSSHGDBMIDEBUGGER'; end; ConfigStore.AppendBasePath('WatchesDlg/'); try FTDebuggerWatchesDlgConfig.ConfigStore := ConfigStore; FTDebuggerWatchesDlgConfig.Load; finally ConfigStore.UndoAppendBasePath; end; end; procedure TDebuggerConfigStore.Save; begin inherited; ConfigStore.SetDeleteValue('Class', FDebuggerClass, ''); ConfigStore.DeletePath('Type'); ConfigStore.AppendBasePath('WatchesDlg/'); try FTDebuggerWatchesDlgConfig.ConfigStore := ConfigStore; FTDebuggerWatchesDlgConfig.Save; finally ConfigStore.UndoAppendBasePath; end; end; constructor TDebuggerConfigStore.Create; begin FTDebuggerWatchesDlgConfig := TDebuggerWatchesDlgConfig.Create; end; destructor TDebuggerConfigStore.Destroy; begin inherited Destroy; FreeAndNil(FTDebuggerWatchesDlgConfig); end; { TDebuggerUnitInfoProvider } function TDebuggerUnitInfoProvider.GetInfo(Index: Integer): TDebuggerUnitInfo; begin Result := FList.Items[Index]; end; constructor TDebuggerUnitInfoProvider.Create; begin FList := TDebuggerUnitInfoList.Create; FLoader := TDebuggerUnitInfo.Create('', ''); end; destructor TDebuggerUnitInfoProvider.Destroy; begin FList.Clear; inherited Destroy; FreeAndNil(FLoader); FreeAndNil(FList); end; procedure TDebuggerUnitInfoProvider.Clear; begin FList.Clear; end; function TDebuggerUnitInfoProvider.GetUnitInfoFor(const AFileName: String; const AFullFileName: String): TDebuggerUnitInfo; var i: Integer; begin i := FList.Count - 1; while i >= 0 do begin if (not(dlfSearchByFunctionName in FList[i].Flags)) and FList[i].IsEqual(AFileName, AFullFileName) then begin debugln(DBG_LOCATION_INFO, ['TDebuggerLocationProvider.GetLocationInfoFor Found entry for: ', AFileName, ' / ', AFullFileName]); exit(FList[i]); end; dec(i); end; Result := TDebuggerUnitInfo.Create(AFileName, AFullFileName); FList.Add(Result); debugln(DBG_LOCATION_INFO, ['TDebuggerLocationProvider.GetLocationInfoFor Created new entry (Cnt=',FList.Count,') for: ', AFileName, ' / ', AFullFileName]); end; function TDebuggerUnitInfoProvider.GetUnitInfoByFunction(const AUnitName, AClassName, AFunctionName, AFunctionArgs: String): TDebuggerUnitInfo; var i: Integer; begin i := FList.Count - 1; while i >= 0 do begin if (dlfSearchByFunctionName in FList[i].Flags) and FList[i].IsEqual(AUnitName, AClassName, AFunctionName, AFunctionArgs) then begin debugln(DBG_LOCATION_INFO, ['TDebuggerLocationProvider.GetLocationInfoFor Found entry for: ', AUnitName, ' / ', AClassName, ' / ', AFunctionName]); exit(FList[i]); end; dec(i); end; Result := TDebuggerUnitInfo.Create(AUnitName, AClassName, AFunctionName, AFunctionArgs); FList.Add(Result); debugln(DBG_LOCATION_INFO, ['TDebuggerLocationProvider.GetLocationInfoFor Created new entry (Cnt=',FList.Count,') for: ', AUnitName, ' / ', AClassName, ' / ', AFunctionName]); end; function TDebuggerUnitInfoProvider.IndexOf(AnInfo: TDebuggerUnitInfo; AddIfNotExists: Boolean): Integer; begin Result := FList.Count - 1; while Result >= 0 do begin if FList[Result].IsEqual(AnInfo) then begin exit; end; dec(Result); end; if AddIfNotExists then Result := FList.Add(AnInfo); end; function TDebuggerUnitInfoProvider.Count: integer; begin Result := FList.Count; end; procedure TDebuggerUnitInfoProvider.LoadDataFromXMLConfig(const AConfig: TXMLConfig; const APath: string); var i, c: Integer; Item: TDebuggerUnitInfo; begin c := AConfig.GetValue(APath + 'UnitInfoCount', 0); for i := 0 to c - 1 do begin Item := TDebuggerUnitInfo.Create('', ''); Item.LoadDataFromXMLConfig(AConfig, APath + 'UnitInfo_' + IntToStr(i) + '/'); FList.Add(Item); end; end; procedure TDebuggerUnitInfoProvider.SaveDataToXMLConfig(const AConfig: TXMLConfig; const APath: string); var i: Integer; begin AConfig.SetValue(APath + 'UnitInfoCount', FList.Count); for i := 0 to FList.Count - 1 do FList[i].SaveDataToXMLConfig(AConfig, APath + 'UnitInfo_' + IntToStr(i) + '/'); end; { TDebuggerUnitInfoList } function TDebuggerUnitInfoList.GetInfo(Index: Integer): TDebuggerUnitInfo; begin Result := TDebuggerUnitInfo(inherited Items[Index]); end; procedure TDebuggerUnitInfoList.PutInfo(Index: Integer; AValue: TDebuggerUnitInfo); begin inherited Items[Index] := AValue; end; { TDebuggerUnitInfo } function TDebuggerUnitInfo.GetFileName: String; begin Result := FFileName; end; function TDebuggerUnitInfo.GetDbgFullName: String; begin Result := FDbgFullName; end; function TDebuggerUnitInfo.GetLocationFullFile: String; begin Result := FLocationFullFile;; end; function TDebuggerUnitInfo.GetLocationName: String; begin Result := FLocationName; end; function TDebuggerUnitInfo.GetLocationOwnerName: String; begin Result := FLocationOwnerName; end; function TDebuggerUnitInfo.GetLocationType: TDebuggerLocationType; begin Result := FLocationType; end; procedure TDebuggerUnitInfo.SetLocationFullFile(AValue: String); begin FLocationFullFile := AValue; end; procedure TDebuggerUnitInfo.SetLocationType(AValue: TDebuggerLocationType); begin FLocationType := AValue; end; constructor TDebuggerUnitInfo.Create(const AFileName: String; const AFullFileName: String); begin FFileName := AFileName; FDbgFullName := TrimFilename(AFullFileName); FLocationType := dltUnknown; end; constructor TDebuggerUnitInfo.Create(const AUnitName, AClassName, AFunctionName, AFunctionArgs: String); begin include(FFlags, dlfSearchByFunctionName); FUnitName := AUnitName; FSrcClassName := AClassName; FFunctionName := AFunctionName; FFunctionArgs := AFunctionArgs; FLocationType := dltUnknown; end; function TDebuggerUnitInfo.DebugText: String; var s: String; begin writestr(s{%H-}, FLocationType); Result := ' FileName="'+FFileName+'" ' + 'DbgFullName="' + FDbgFullName+'" ' + 'UnitName="' + FUnitName+'" ' + 'ClassName="' + FSrcClassName+'" ' + 'FunctionName="' + FFunctionName+'" ' + 'Flags="' + dbgs(FFlags)+'" ' + 'LocationName="' + FLocationName+'" ' + 'LocationOwnerName="' + FLocationOwnerName+'" ' + 'LocationFullFile="' + FLocationFullFile+'" ' + 'LocationType="' + s+'"' + 'FunctionArgs"' + FFunctionArgs +'" '; end; function TDebuggerUnitInfo.IsEqual(const AFileName: String; const AFullFileName: String): boolean; begin Result := (FFileName = AFileName) and (FDbgFullName = AFullFileName); end; function TDebuggerUnitInfo.IsEqual(const AUnitName, AClassName, AFunctionName, AFunctionArgs: String): boolean; begin Result := (FUnitName = AUnitName) and (FSrcClassName = AClassName) and (FFunctionName = AFunctionName) and (FFunctionArgs = AFunctionArgs); end; function TDebuggerUnitInfo.IsEqual(AnOther: TDebuggerUnitInfo): boolean; begin Result := (FFileName = AnOther.FFileName); if not Result then exit; case LocationType of dltUnknown, dltUnresolvable: Result := Result and (FDbgFullName = AnOther.FDbgFullName); dltProject, dltPackage: Result := Result and (FLocationType = AnOther.FLocationType) and (FLocationOwnerName = AnOther.FLocationOwnerName) and (FLocationName = AnOther.FLocationName) and (FSrcLine = AnOther.FSrcLine); end; end; procedure TDebuggerUnitInfo.LoadDataFromXMLConfig(const AConfig: TXMLConfig; const APath: string); begin try ReadStr(AConfig.GetValue(APath + 'Type', 'dltUnknown'), FLocationType); if LocationType = dltUnresolvable then LocationType := dltUnknown; except FLocationType := dltUnknown; end; if AConfig.GetValue(APath + 'ByFunction', False) then include(FFlags, dlfSearchByFunctionName) else exclude(FFlags, dlfSearchByFunctionName); FFileName := AConfig.GetValue(APath + 'File', ''); FSrcLine := AConfig.GetValue(APath + 'SrcLine', 0); FLocationOwnerName := AConfig.GetValue(APath + 'UnitOwner', ''); FLocationName := AConfig.GetValue(APath + 'UnitFile', ''); FDbgFullName := AConfig.GetValue(APath + 'DbgFile', ''); FLocationFullFile := ''; FUnitName := AConfig.GetValue(APath + 'UnitName', ''); FSrcClassName := AConfig.GetValue(APath + 'SrcClassName', ''); FFunctionName := AConfig.GetValue(APath + 'FunctionName', ''); FFunctionArgs := AConfig.GetValue(APath + 'FunctionArgs', ''); end; procedure TDebuggerUnitInfo.SaveDataToXMLConfig(const AConfig: TXMLConfig; const APath: string); var s: String; begin WriteStr(s{%H-}, LocationType); AConfig.SetValue(APath + 'Type', s); AConfig.SetValue(APath + 'File', FileName); AConfig.SetValue(APath + 'SrcLine', FSrcLine); AConfig.SetDeleteValue(APath + 'ByFunction', dlfSearchByFunctionName in FFlags, False); AConfig.SetValue(APath + 'UnitOwner', LocationOwnerName); AConfig.SetValue(APath + 'UnitFile', LocationName); AConfig.SetValue(APath + 'DbgFile', FDbgFullName); AConfig.SetDeleteValue(APath + 'UnitName', FUnitName, ''); AConfig.SetDeleteValue(APath + 'SrcClassName', FSrcClassName, ''); AConfig.SetDeleteValue(APath + 'FunctionName', FFunctionName, ''); AConfig.SetDeleteValue(APath + 'FunctionArgs', FFunctionArgs, ''); end; { TSnapshotList } function TSnapshotList.Get(Index: Integer): TSnapshot; begin Result := TSnapshot(inherited Items[Index]) end; procedure TSnapshotList.Put(Index: Integer; const AValue: TSnapshot); begin inherited Items[Index] := AValue; end; { TDebuggerDataSnapShot } destructor TDebuggerDataSnapShot.Destroy; begin inherited Destroy; DataObject.Free; end; function TSnapshot.GetLocationAsText: String; begin if FLocation.SrcFile <> '' then Result := FLocation.SrcFile + ' ' + IntToStr(FLocation.SrcLine) else Result := ':' + IntToHex(FLocation.Address, 8); if FLocation.FuncName <> '' then Result := FLocation.FuncName + ' (' + Result + ')'; end; constructor TSnapshot.Create(ASnapMgr: TSnapshotManager); begin FTimeStamp := Now; FSnapMgr := ASnapMgr; AddReference; end; destructor TSnapshot.Destroy; begin FSnapMgr.DoSnapShotDestroy(Self); inherited Destroy; end; procedure TSnapshot.LoadDataFromXMLConfig(const AConfig: TXMLConfig; APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil); begin FLocation.Address := StrToQWordDef(AConfig.GetValue(APath + 'LocationAddress', '0'), 0); FLocation.FuncName := AConfig.GetValue(APath + 'LocationFuncName', ''); FLocation.SrcFile := AConfig.GetValue(APath + 'LocationSrcFile', ''); FLocation.SrcFullName := AConfig.GetValue(APath + 'LocationSrcFullName', ''); FLocation.SrcLine := AConfig.GetValue(APath + 'LocationSrcLine', -1); try FTimeStamp := StrToDouble(AConfig.GetValue(APath + 'TimeStamp', '0')); except FTimeStamp := 0; end; if FSnapMgr.Threads.Snapshots[Pointer(Self)] <> nil then FSnapMgr.Threads.Snapshots[Pointer(Self)].LoadDataFromXMLConfig(AConfig, APath + 'SnapThreads/', AUnitInvoPrv); if FSnapMgr.CallStack.Snapshots[Pointer(Self)] <> nil then FSnapMgr.CallStack.Snapshots[Pointer(Self)].LoadDataFromXMLConfig(AConfig, APath + 'SnapCallstack/', AUnitInvoPrv); if FSnapMgr.Locals.Snapshots[Pointer(Self)] <> nil then FSnapMgr.Locals.Snapshots[Pointer(Self)].LoadDataFromXMLConfig(AConfig, APath + 'SnapLocals/'); if FSnapMgr.Watches.Snapshots[Pointer(Self)] <> nil then FSnapMgr.Watches.Snapshots[Pointer(Self)].LoadDataFromXMLConfig(AConfig, APath + 'SnapWatches/'); if AConfig.GetValue(APath + 'IsSnapshot', False) then AddToSnapshots; if AConfig.GetValue(APath + 'IsHistory', True) then AddToHistory; end; procedure TSnapshot.SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil); begin AConfig.SetValue(APath + 'LocationAddress', IntToStr(FLocation.Address)); AConfig.SetValue(APath + 'LocationFuncName', FLocation.FuncName); AConfig.SetValue(APath + 'LocationSrcFile', FLocation.SrcFile); AConfig.SetValue(APath + 'LocationSrcFullName', FLocation.SrcFullName); AConfig.SetValue(APath + 'LocationSrcLine', FLocation.SrcLine); AConfig.SetValue(APath + 'TimeStamp', FloatToStr(FTimeStamp)); AConfig.SetValue(APath + 'IsHistory', IsHistory); AConfig.SetValue(APath + 'IsSnapshot', IsSnapshot); if FSnapMgr.Threads.Snapshots[Pointer(Self)] <> nil then FSnapMgr.Threads.Snapshots[Pointer(Self)].SaveDataToXMLConfig(AConfig, APath + 'SnapThreads/', AUnitInvoPrv); if FSnapMgr.CallStack.Snapshots[Pointer(Self)] <> nil then FSnapMgr.CallStack.Snapshots[Pointer(Self)].SaveDataToXMLConfig(AConfig, APath + 'SnapCallstack/', AUnitInvoPrv); if FSnapMgr.Locals.Snapshots[Pointer(Self)] <> nil then FSnapMgr.Locals.Snapshots[Pointer(Self)].SaveDataToXMLConfig(AConfig, APath + 'SnapLocals/'); if FSnapMgr.Watches.Snapshots[Pointer(Self)] <> nil then FSnapMgr.Watches.Snapshots[Pointer(Self)].SaveDataToXMLConfig(AConfig, APath + 'SnapWatches/'); end; procedure TSnapshot.AddToSnapshots; begin FSnapMgr.AddSnapshotEntry(Self); end; procedure TSnapshot.AddToHistory; begin FSnapMgr.AddHistoryEntry(Self); end; procedure TSnapshot.RemoveFromSnapshots; begin FSnapMgr.RemoveSnapshotEntry(Self); end; procedure TSnapshot.RemoveFromHistory; begin FSnapMgr.RemoveHistoryEntry(Self); end; function TSnapshot.IsCurrent: Boolean; begin Result := Self = FSnapMgr.Current; end; function TSnapshot.IsHistory: Boolean; begin Result := FSnapMgr.FHistoryList.IndexOf(Self) >= 0; end; function TSnapshot.IsSnapshot: Boolean; begin Result := FSnapMgr.FSnapshotList.IndexOf(Self) >= 0; end; { TSnapshotManager } function TSnapshotManager.GetHistoryEntry(AIndex: Integer): TSnapshot; begin Result := FHistoryList[AIndex]; end; procedure TSnapshotManager.SetActive(const AValue: Boolean); begin if FActive = AValue then exit; FActive := AValue; if Active and (FCurrentState = dsPause) then DoDebuggerIdle; end; procedure TSnapshotManager.SetDebugger(AValue: TDebuggerIntf); begin if FDebugger = AValue then Exit; FDebugger := AValue; FCurrentState := dsNone; end; procedure TSnapshotManager.DoCallStackChanged(Sender: TObject); begin if FForcedIdle then DoDebuggerIdle(True); end; procedure TSnapshotManager.SetCallStack(AValue: TIdeCallStackMonitor); begin if FCallStack = AValue then Exit; if (FCallStackNotification <> nil) and (FCallStack <> nil) then begin FCallStack.RemoveNotification(FCallStackNotification); end; FCallStack := AValue; if (FCallStack <> nil) then begin if FCallStackNotification = nil then begin FCallStackNotification := TCallStackNotification.Create; FCallStackNotification.AddReference; FCallStackNotification.OnChange := @DoCallStackChanged; end; FCallStack.AddNotification(FCallStackNotification); end; end; procedure TSnapshotManager.SetHistoryIndex(const AValue: Integer); begin if FHistoryindex = AValue then exit; FHistoryindex := AValue; if FHistorySelected then DoCurrent; end; procedure TSnapshotManager.SetHistorySelected(AValue: Boolean); begin if FHistoryList.Count = 0 then AValue := False; if FHistorySelected = AValue then exit; FHistorySelected := AValue; if AValue then SnapshotSelected := False; DoCurrent; end; function TSnapshotManager.GetSnapshotEntry(AIndex: Integer): TSnapshot; begin Result := FSnapshotList[AIndex]; end; procedure TSnapshotManager.SetSnapshotIndex(const AValue: Integer); begin if FSnapshotIndex = AValue then exit; FSnapshotIndex := AValue; if FSnapshotSelected then DoCurrent; end; procedure TSnapshotManager.SetSnapshotSelected(AValue: Boolean); begin if FSnapshotList.Count = 0 then AValue := False; if FSnapshotSelected = AValue then exit; FSnapshotSelected := AValue; if AValue then HistorySelected := False; DoCurrent; end; procedure TSnapshotManager.DoSnapShotDestroy(ASnapShot: TSnapshot); begin FHistoryList.Remove(ASnapShot); RemoveHistoryEntryFromMonitors(ASnapShot); if FHistoryList.Count = 0 then HistorySelected := False; if FSnapshotList.Count = 0 then SnapshotSelected := False; end; procedure TSnapshotManager.BeginUpdate; begin inc(FUpdateLock); end; procedure TSnapshotManager.EndUpdate; begin Assert(FUpdateLock > 0, 'TSnapshotManager.EndUpdate no locked'); if FUpdateLock > 0 then dec(FUpdateLock); if FUpdateLock = 0 then begin if ufSnapChanged in FUpdateFlags then DoChanged; if ufSnapCurrent in FUpdateFlags then DoCurrent; end; end; procedure TSnapshotManager.DoChanged; begin if FUpdateLock > 0 then begin Include(FUpdateFlags, ufSnapChanged); exit; end; Exclude(FUpdateFlags, ufSnapChanged); FNotificationList.NotifyChange(Self); end; procedure TSnapshotManager.DoCurrent; begin if FUpdateLock > 0 then begin Include(FUpdateFlags, ufSnapCurrent); exit; end; Exclude(FUpdateFlags, ufSnapCurrent); FNotificationList.NotifyCurrent(Self); end; procedure TSnapshotManager.LoadDataFromXMLConfig(const AConfig: TXMLConfig; APath: string); var c, i: Integer; NewSnap: TSnapshot; UIProv: TDebuggerUnitInfoProvider; begin Clear; UIProv := TDebuggerUnitInfoProvider.Create; UIProv.LoadDataFromXMLConfig(AConfig, APath + 'UnitInfos/'); c := AConfig.GetValue(APath + 'SnapCount', 0); for i := 0 to c - 1 do begin NewSnap := TSnapshot.Create(Self); FThreads.NewSnapshot(NewSnap, True); FCallStack.NewSnapshot(NewSnap, True); FLocals.NewSnapshot(NewSnap, True); FWatches.NewSnapshot(NewSnap, True); NewSnap.LoadDataFromXMLConfig(AConfig, APath + 'SnapEntry' + IntToStr(i) + '/', UIProv); if not(NewSnap.IsHistory or NewSnap.IsSnapshot) then begin RemoveHistoryEntryFromMonitors(NewSnap); // TODO: add user feedback / warning debugln(['************** Snapshot loaded, but not kept']); end; NewSnap.ReleaseReference; end; c := AConfig.GetValue(APath + 'HistCount', 0); for i := 0 to c - 1 do begin NewSnap := TSnapshot.Create(Self); FThreads.NewSnapshot(NewSnap, True); FCallStack.NewSnapshot(NewSnap, True); FLocals.NewSnapshot(NewSnap, True); FWatches.NewSnapshot(NewSnap, True); NewSnap.LoadDataFromXMLConfig(AConfig, APath + 'HistEntry' + IntToStr(i) + '/', UIProv); if not(NewSnap.IsHistory or NewSnap.IsSnapshot) then begin RemoveHistoryEntryFromMonitors(NewSnap); // TODO: add user feedback / warning debugln(['************** Snapshot loaded, but not kept']); end; NewSnap.ReleaseReference; end; UIProv.Free; //FThreads.CurrentThreads.SnapShot := nil; //FCallStack.CurrentCallStackList.SnapShot := nil; //FLocals.CurrentLocalsList.SnapShot := nil; //FWatches.CurrentWatches.SnapShot := nil; DoChanged; DoCurrent; end; procedure TSnapshotManager.SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string); var c, i: Integer; UIProv: TDebuggerUnitInfoProvider; begin UIProv := TDebuggerUnitInfoProvider.Create; c := 0; for i := 0 to FSnapshotList.Count - 1 do begin if FSnapshotList[i].IsHistory then continue; FSnapshotList[i].SaveDataToXMLConfig(AConfig, APath + 'SnapEntry' + IntToStr(i) + '/', UIProv); inc(c); end; AConfig.SetValue(APath + 'SnapCount', c); c := 0; for i := 0 to FHistoryList.Count - 1 do begin FHistoryList[i].SaveDataToXMLConfig(AConfig, APath + 'HistEntry' + IntToStr(i) + '/', UIProv); inc(c); end; AConfig.SetValue(APath + 'HistCount', c); UIProv.SaveDataToXMLConfig(AConfig, APath + 'UnitInfos/'); UIProv.Free; end; procedure TSnapshotManager.ClearHistory; begin FHistoryList.Clear; HistorySelected := False; end; procedure TSnapshotManager.ClearSnapshots; begin FSnapshotList.Clear; SnapshotSelected := False; end; function TSnapshotManager.GetAsXML: String; var XmlConf: TXMLConfig; s: TStringStream; begin XmlConf := TXMLConfig.CreateClean(''); XmlConf.Clear; SaveDataToXMLConfig(XmlConf, 'History/'); s := TStringStream.Create(''); XmlConf.WriteToStream(s); Result := s.DataString; s.WriteAnsiString(Result); XmlConf.Free; s.Free; end; procedure TSnapshotManager.SetFromXML(aXML: String); var XmlConf: TXMLConfig; s: TStringStream; begin XmlConf := TXMLConfig.CreateClean(''); XmlConf.Clear; s := TStringStream.Create(aXML); XmlConf.ReadFromStream(s); LoadDataFromXMLConfig(XmlConf, 'History/'); XmlConf.Free; s.Free; end; procedure TSnapshotManager.CreateHistoryEntry; var t: LongInt; begin ReleaseRefAndNil(FCurrentSnapshot); // should be nil already FCurrentSnapshot := TSnapshot.Create(Self); FCurrentSnapshot.Location := Debugger.GetLocation; FThreads.NewSnapshot(FCurrentSnapshot); FCallStack.NewSnapshot(FCurrentSnapshot); FLocals.NewSnapshot(FCurrentSnapshot); FWatches.NewSnapshot(FCurrentSnapshot); // acces them , so they will be present t := FThreads.CurrentThreads.CurrentThreadId; FCallStack.CurrentCallStackList.EntriesForThreads[t]; DoDebuggerIdle; DoChanged; end; procedure TSnapshotManager.RemoveHistoryEntry(AIndex: Integer); begin BeginUpdate; try FHistoryList.Delete(AIndex); if FHistoryList.Count = 0 then HistorySelected := False; DoChanged; finally EndUpdate; end; end; procedure TSnapshotManager.RemoveHistoryEntry(ASnapShot: TSnapshot); begin BeginUpdate; try FHistoryList.Remove(ASnapShot); if FHistoryList.Count = 0 then HistorySelected := False; DoChanged; finally EndUpdate; end; end; procedure TSnapshotManager.RemoveHistoryEntryFromMonitors(AnEntry: TSnapshot); begin if FThreads <> nil then FThreads.RemoveSnapshot(AnEntry); if FCallStack <> nil then FCallStack.RemoveSnapshot(AnEntry); if FLocals <> nil then FLocals.RemoveSnapshot(AnEntry); if FWatches <> nil then FWatches.RemoveSnapshot(AnEntry); end; procedure TSnapshotManager.AddSnapshotEntry(ASnapShot: TSnapshot); begin FSnapshotList.Add(ASnapShot); DoChanged; end; procedure TSnapshotManager.RemoveSnapshotEntry(ASnapShot: TSnapshot); begin BeginUpdate; try FSnapshotList.Remove(ASnapShot); if FSnapshotList.Count = 0 then SnapshotSelected := False; DoChanged; finally EndUpdate; end; end; procedure TSnapshotManager.AddHistoryEntry(ASnapShot: TSnapshot); begin FHistoryList.Add(ASnapShot); DoChanged; end; constructor TSnapshotManager.Create; begin FNotificationList := TDebuggerChangeNotificationList.Create; FActive := True; FHistorySelected := False; FHistoryList := TSnapshotList.Create; FHistoryCapacity := 25; FSnapshotList := TSnapshotList.Create; inherited Create; end; destructor TSnapshotManager.Destroy; begin FCallStackNotification.OnChange := nil; FNotificationList.Clear; ReleaseRefAndNil(FCurrentSnapshot); Clear; CallStack := nil; ReleaseRefAndNil(FCallStackNotification); inherited Destroy; FreeAndNil(FHistoryList); FreeAndNil(FSnapshotList); FreeAndNil(FNotificationList); end; procedure TSnapshotManager.AddNotification(const ANotification: TSnapshotNotification); begin FNotificationList.Add(ANotification); end; procedure TSnapshotManager.RemoveNotification(const ANotification: TSnapshotNotification); begin FNotificationList.Remove(ANotification); end; procedure TSnapshotManager.DoStateChange(const AOldState: TDBGState); begin if FDebugger = nil then exit; FCurrentState := Debugger.State; FForcedIdle := False; DebugLnEnter(DBG_DATA_MONITORS, ['DebugDataMonitor: >>ENTER: TSnapshotManager.DoStateChange New-State=', DBGStateNames[FCurrentState]]); BeginUpdate; try if FDebugger.State in [dsPause, dsInternalPause] then begin Exclude(FUpdateFlags, ufInDebuggerIdle); FRequestsDone := []; CreateHistoryEntry; HistorySelected := False; SnapshotSelected := False; end else begin if (FCurrentSnapshot <> nil) and (FActive or (AOldState = dsInternalPause)) then begin HistoryIndex := FHistoryList.Add(FCurrentSnapshot); ReleaseRefAndNil(FCurrentSnapshot); while FHistoryList.Count > HistoryCapacity do RemoveHistoryEntry(0); DoChanged; end; end; if (FDebugger.State = dsInit) then begin Clear; end; finally EndUpdate; end; DebugLnExit(DBG_DATA_MONITORS, ['DebugDataMonitor: <>ENTER: TSnapshotManager.DoDebuggerIdle New-State=', DBGStateNames[FCurrentState]]); try if not(smrThreads in FRequestsDone) then begin include(FRequestsDone, smrThreads); FThreads.CurrentThreads.Count; if (not(FCurrentState in [dsPause, dsInternalPause])) or (Debugger = nil) or ( (not Debugger.IsIdle) and (not AForce) ) then exit; if CurSnap <> FCurrentSnapshot then exit; // Debugger did "run" in between end; if not(smrCallStack in FRequestsDone) then begin i := FThreads.CurrentThreads.CurrentThreadId; k := FCallStack.CurrentCallStackList.EntriesForThreads[i].CountLimited(5); if CurSnap <> FCurrentSnapshot then exit; // Debugger did "run" in between if (k > 0) or (smrCallStackCnt in FRequestsDone) then begin // Since DoDebuggerIdle was re-entered // and smrCallStackCnt is set, the count should be valid include(FRequestsDone, smrCallStack); if k > 0 then FCallStack.CurrentCallStackList.EntriesForThreads[i].PrepareRange(0, Min(5, k)); if (not(FCurrentState in [dsPause, dsInternalPause])) or (Debugger = nil) or ( (not Debugger.IsIdle) and (not AForce) ) then exit; if CurSnap <> FCurrentSnapshot then exit; // Debugger did "run" in between end else if AForce then // request re-entry, even if not idle FForcedIdle := True; end; if not(smrCallStackCnt in FRequestsDone) then begin include(FRequestsDone, smrCallStackCnt); i := FThreads.CurrentThreads.CurrentThreadId; FCallStack.CurrentCallStackList.EntriesForThreads[i].CountLimited(5); if (not(FCurrentState in [dsPause, dsInternalPause])) or (Debugger = nil) or ( (not Debugger.IsIdle) and (not AForce) ) then exit; if CurSnap <> FCurrentSnapshot then exit; // Debugger did "run" in between end; if not(smrLocals in FRequestsDone) then begin include(FRequestsDone, smrLocals); i := FThreads.CurrentThreads.CurrentThreadId; j := FCallStack.CurrentCallStackList.EntriesForThreads[i].CurrentIndex; FLocals.CurrentLocalsList.Entries[i, j].Count; if (not(FCurrentState in [dsPause, dsInternalPause])) or (Debugger = nil) or ( (not Debugger.IsIdle) and (not AForce) ) then exit; if CurSnap <> FCurrentSnapshot then exit; // Debugger did "run" in between end; if not(smrWatches in FRequestsDone) then begin include(FRequestsDone, smrWatches); i := FThreads.CurrentThreads.CurrentThreadId; j := FCallStack.CurrentCallStackList.EntriesForThreads[i].CurrentIndex; w := FWatches.CurrentWatches; k := 0; while k < w.Count do begin if CurSnap <> FCurrentSnapshot then exit; // Debugger did "run" in between w[k].Values[i, j].Value; inc(k); end; if (not(FCurrentState in [dsPause, dsInternalPause])) or (Debugger = nil) or ( (not Debugger.IsIdle) and (not AForce) ) then exit; if CurSnap <> FCurrentSnapshot then exit; // Debugger did "run" in between end; finally Exclude(FUpdateFlags, ufInDebuggerIdle); DebugLnExit(DBG_DATA_MONITORS, ['DebugDataMonitor: <= 0) and (HistoryIndex < FHistoryList.Count) and (FHistorySelected) then Result := FHistoryList[HistoryIndex]; if (SnapshotIndex >= 0) and (SnapshotIndex < FSnapshotList.Count) and (FSnapshotSelected) then Result := FSnapshotList[HistoryIndex]; end; function TSnapshotManager.SelectedEntry: TSnapshot; begin Result := nil; if (HistoryIndex >= 0) and (HistoryIndex < FHistoryList.Count) and (FHistorySelected) then Result := FHistoryList[HistoryIndex]; if (SnapshotIndex >= 0) and (SnapshotIndex < FSnapshotList.Count) and (FSnapshotSelected) then Result := FSnapshotList[SnapshotIndex]; end; procedure TSnapshotManager.Clear; begin BeginUpdate; try ClearHistory; ClearSnapshots; DoChanged; DoCurrent; finally EndUpdate; end; end; { TDebuggerDataSnapShotList } function TDebuggerDataSnapShotList.GetSnapShot(AnID: Pointer): TObject; var i: Integer; begin i := FList.Count - 1; while i >= 0 do begin Result := TObject(FList[i]); if TDebuggerDataSnapShot(Result).SnapShotId = AnID then exit(TDebuggerDataSnapShot(Result).DataObject); dec(i); end; Result := nil; end; constructor TDebuggerDataSnapShotList.Create; begin inherited Create; FList := TList.Create; end; destructor TDebuggerDataSnapShotList.Destroy; begin Clear; inherited Destroy; FreeAndNil(FList); end; procedure TDebuggerDataSnapShotList.Clear; begin while FList.Count > 0 do begin TDebuggerDataSnapShot(FList[0]).Free; FList.Delete(0); end; end; procedure TDebuggerDataSnapShotList.AddSnapShot(AnID: Pointer; AnObject: TObject); var NewSn: TDebuggerDataSnapShot; begin NewSn := TDebuggerDataSnapShot.Create; NewSn.SnapShotId := AnID; NewSn.DataObject := AnObject; FList.Add(NewSn); end; procedure TDebuggerDataSnapShotList.RemoveSnapShot(AnID: Pointer); var R: TDebuggerDataSnapShot; i: Integer; begin i := FList.Count - 1; while i >= 0 do begin R := TDebuggerDataSnapShot(FList[i]); if TDebuggerDataSnapShot(R).SnapShotId = AnID then break; dec(i); end; if i >= 0 then begin FList.Delete(i); R.Free; end; end; { TCurrentLocalsList } procedure TCurrentLocalsList.SetSnapShot(const AValue: TIDELocalsList); var i: Integer; E, R: TIDELocals; begin assert((FSnapShot=nil) or (AValue=nil), 'TCurrentLocalsList already have snapshot'); if FSnapShot = AValue then exit; if FSnapShot <> nil then FSnapShot.Immutable := True; FSnapShot := AValue; if FSnapShot = nil then begin for i := 0 to Count-1 do TCurrentLocals(EntriesByIdx[i]).SnapShot := nil; end else begin //FSnapShot.Assign(Self); FSnapShot.Clear; for i := 0 to Count-1 do begin E := EntriesByIdx[i]; R := TIDELocals.Create(e.ThreadId, e.StackFrame); FSnapShot.Add(R); TCurrentLocals(E).SnapShot := R; end; end; end; procedure TCurrentLocalsList.DoCleared; begin FMonitor.NotifyChange(nil); end; procedure TCurrentLocalsList.DoAdded(AnEntry: TDbgEntityValuesList); var R: TIDELocals; begin Assert(AnEntry is TCurrentLocals, 'TCurrentLocalsList.DoAdded'); inherited DoAdded(AnEntry); if FSnapShot <> nil then begin R := TIDELocals.Create(AnEntry.ThreadId, AnEntry.StackFrame); FSnapShot.Add(R); TCurrentLocals(AnEntry).SnapShot := R; end; end; function TCurrentLocalsList.CreateEntry(AThreadId, AStackFrame: Integer): TIDELocals; begin Result := TCurrentLocals.Create(FMonitor, AThreadId, AStackFrame); end; constructor TCurrentLocalsList.Create(AMonitor: TIdeLocalsMonitor); begin FMonitor := AMonitor; inherited Create; end; { TLocalsList } function TIDELocalsList.GetEntry(const AThreadId: Integer; const AStackFrame: Integer): TIDELocals; begin Result := TIDELocals(inherited Entries[AThreadId, AStackFrame]); end; function TIDELocalsList.GetEntryByIdx(const AnIndex: Integer): TIDELocals; begin Result := TIDELocals(inherited EntriesByIdx[AnIndex]); end; function TIDELocalsList.CreateEntry(AThreadId, AStackFrame: Integer): TDbgEntityValuesList; begin Result := TIDELocals.Create(AThreadId, AStackFrame); end; procedure TIDELocalsList.DoAssign(AnOther: TDbgEntitiesThreadStackList); begin inherited DoAssign(AnOther); Immutable := not(Self is TCurrentLocalsList); end; procedure TIDELocalsList.DoAdded(AnEntry: TDbgEntityValuesList); begin inherited DoAdded(AnEntry); //AnEntry.Immutable := not(Self is TCurrentLocalsList); end; procedure TIDELocalsList.LoadDataFromXMLConfig(const AConfig: TXMLConfig; APath: string); var e: TIDELocals; c, i: Integer; begin Clear; c := AConfig.GetValue(APath + 'Count', 0); APath := APath + 'LocalsEntry'; for i := 0 to c - 1 do begin e := TIDELocals.CreateFromXMLConfig(AConfig, APath + IntToStr(i) + '/'); Add(e); end; end; procedure TIDELocalsList.SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string); var i: Integer; begin AConfig.SetDeleteValue(APath + 'Count', Count, 0); APath := APath + 'LocalsEntry'; for i := 0 to Count - 1 do EntriesByIdx[i].SaveDataToXMLConfig(AConfig, APath + IntToStr(i) + '/'); end; { TIdeLocalsMonitor } function TIdeLocalsMonitor.GetSnapshot(AnID: Pointer): TIDELocalsList; begin Result := TIDELocalsList(FSnapshots.SnapShot[AnID]); end; function TIdeLocalsMonitor.GetCurrentLocalsList: TCurrentLocalsList; begin Result := TCurrentLocalsList(LocalsList);; end; procedure TIdeLocalsMonitor.DoStateEnterPause; begin inherited DoStateEnterPause; if (CurrentLocalsList = nil) then Exit; Clear; end; procedure TIdeLocalsMonitor.DoStateLeavePause; begin inherited DoStateLeavePause; if (CurrentLocalsList = nil) then Exit; CurrentLocalsList.SnapShot := nil; end; procedure TIdeLocalsMonitor.DoStateLeavePauseClean; begin inherited DoStateLeavePauseClean; if (CurrentLocalsList = nil) then Exit; CurrentLocalsList.SnapShot := nil; Clear; end; procedure TIdeLocalsMonitor.NotifyChange(ALocals: TCurrentLocals); begin FNotificationList.NotifyChange(ALocals); end; procedure TIdeLocalsMonitor.DoNewSupplier; begin inherited DoNewSupplier; NotifyChange(nil); end; procedure TIdeLocalsMonitor.RequestData(ALocals: TCurrentLocals); begin if Supplier <> nil then Supplier.RequestData(ALocals) else ALocals.SetDataValidity(ddsInvalid); end; function TIdeLocalsMonitor.CreateSnapshot(CreateEmpty: Boolean = False): TObject; begin Result := TIDELocalsList.Create; if not CreateEmpty then CurrentLocalsList.SnapShot := TIDELocalsList(Result); end; function TIdeLocalsMonitor.CreateLocalsList: TLocalsList; begin Result := TCurrentLocalsList.Create(Self); end; constructor TIdeLocalsMonitor.Create; begin FSnapshots := TDebuggerDataSnapShotList.Create; inherited; FNotificationList := TDebuggerChangeNotificationList.Create; end; destructor TIdeLocalsMonitor.Destroy; begin FSnapshots.Clear; FNotificationList.Clear; inherited Destroy; FreeAndNil(FNotificationList); FreeAndNil(FSnapshots); end; procedure TIdeLocalsMonitor.Clear; begin CurrentLocalsList.Clear; end; procedure TIdeLocalsMonitor.AddNotification(const ANotification: TLocalsNotification); begin FNotificationList.Add(ANotification); end; procedure TIdeLocalsMonitor.RemoveNotification(const ANotification: TLocalsNotification); begin FNotificationList.Remove(ANotification); end; procedure TIdeLocalsMonitor.NewSnapshot(AnID: Pointer; CreateEmpty: Boolean); var S: TObject; begin S := CreateSnapshot(CreateEmpty); FSnapshots.AddSnapShot(AnID, S); end; procedure TIdeLocalsMonitor.RemoveSnapshot(AnID: Pointer); begin FSnapshots.RemoveSnapShot(AnID); end; { TCurrentWatchValue } procedure TCurrentWatchValue.SetSnapShot(const AValue: TIdeWatchValue); begin assert((FSnapShot=nil) or (AValue=nil), 'TCurrentWatchValue already have snapshot'); if FSnapShot = AValue then exit; FSnapShot := AValue; if FSnapShot <> nil then FSnapShot.Assign(self); end; procedure TCurrentWatchValue.RequestData; begin TCurrentWatch(Watch).RequestData(self); end; procedure TCurrentWatchValue.DoDataValidityChanged(AnOldValidity: TDebuggerDataState); begin if Validity = ddsRequested then exit; TCurrentWatches(TCurrentWatch(Watch).Collection).Update(Watch); if FSnapShot <> nil then FSnapShot.Assign(self); end; { TCurrentWatchValueList } procedure TCurrentWatchValueList.SetSnapShot(const AValue: TIdeWatchValueList); var R: TIdeWatchValue; i: Integer; begin assert((FSnapShot=nil) or (AValue=nil), 'TCurrentWatchValueList already have snapshot'); if FSnapShot = AValue then exit; FSnapShot := AValue; if FSnapShot = nil then begin for i := 0 to Count - 1 do TCurrentWatchValue(EntriesByIdx[i]).SnapShot := nil; end else begin // Assign FSnapShot.Clear; for i := 0 to Count - 1 do begin R := TIdeWatchValue.Create(FSnapShot.Watch); R.Assign(EntriesByIdx[i]); FSnapShot.Add(R); TCurrentWatchValue(EntriesByIdx[i]).SnapShot := R; end; end; end; function TCurrentWatchValueList.CreateEntry(const AThreadId: Integer; const AStackFrame: Integer): TIdeWatchValue; var R: TIdeWatchValue; begin try DebugLnEnter(DBG_DATA_MONITORS, ['DebugDataMonitor: >>ENTER: TCurrentWatchValueList.CreateEntry AThreadId=', AThreadId, ' AStackFrame=',AStackFrame, ' Expr=', Watch.Expression]); Result := TCurrentWatchValue.Create(Watch, AThreadId, AStackFrame); Add(Result); if FSnapShot <> nil then begin R := TIdeWatchValue.Create(FSnapShot.Watch); FSnapShot.Add(R); TCurrentWatchValue(Result).SnapShot := R; end; finally DebugLnExit(DBG_DATA_MONITORS, ['DebugDataMonitor: <'); i := DbgStateChangeCounter; // workaround for state changes during TWatchValue.GetValue if Validity = ddsUnknown then begin Result := ''; Validity := ddsRequested; RequestData; if i <> DbgStateChangeCounter then exit; // in case the debugger did run. // TODO: The watch can also be deleted by the user end; case Validity of ddsRequested, ddsEvaluating: Result := ''; ddsValid: Result := inherited GetValue; ddsInvalid: Result := ''; ddsError: Result := ''; end; end; function TIdeWatchValue.GetWatch: TIdeWatch; begin Result := TIdeWatch(inherited Watch); end; function TIdeWatchValue.GetTypeInfo: TDBGType; var i: Integer; begin Result := nil; if not Watch.Enabled then exit; i := DbgStateChangeCounter; // workaround for state changes during TWatchValue.GetValue if Validity = ddsUnknown then begin Validity := ddsRequested; RequestData; if i <> DbgStateChangeCounter then exit; end; case Validity of ddsRequested, ddsEvaluating: Result := nil; ddsValid: Result := inherited GetTypeInfo; ddsInvalid, ddsError: Result := nil; end; end; procedure TIdeWatchValue.RequestData; begin Validity := ddsInvalid; end; procedure TIdeWatchValue.LoadDataFromXMLConfig(const AConfig: TXMLConfig; const APath: string); var NewValidity: TDebuggerDataState; begin FThreadId := AConfig.GetValue(APath + 'ThreadId', -1); FStackFrame := AConfig.GetValue(APath + 'StackFrame', -1); Value := AConfig.GetValue(APath + 'Value', ''); if AConfig.GetValue(APath + 'ClassAutoCast', False) then Include(FEvaluateFlags, defClassAutoCast) else Exclude(FEvaluateFlags, defClassAutoCast); FRepeatCount := AConfig.GetValue(APath + 'RepeatCount', 0); try ReadStr(AConfig.GetValue(APath + 'DisplayFormat', 'wdfDefault'), FDisplayFormat); except FDisplayFormat := wdfDefault; end; try ReadStr(AConfig.GetValue(APath + 'Validity', 'ddsValid'), NewValidity); Validity := NewValidity; except Validity := ddsUnknown; end; end; procedure TIdeWatchValue.SaveDataToXMLConfig(const AConfig: TXMLConfig; const APath: string); var s: String; begin AConfig.SetValue(APath + 'ThreadId', ThreadId); AConfig.SetValue(APath + 'StackFrame', StackFrame); AConfig.SetValue(APath + 'Value', Value); WriteStr(s{%H-}, DisplayFormat); AConfig.SetDeleteValue(APath + 'DisplayFormat', s, 'wdfDefault'); WriteStr(s, Validity); AConfig.SetDeleteValue(APath + 'Validity', s, 'ddsValid'); AConfig.SetDeleteValue(APath + 'ClassAutoCast', defClassAutoCast in EvaluateFlags, False); AConfig.SetDeleteValue(APath + 'RepeatCount', RepeatCount, 0); end; constructor TIdeWatchValue.Create(AOwnerWatch: TIdeWatch); begin inherited Create(AOwnerWatch); Validity := ddsUnknown; FDisplayFormat := Watch.DisplayFormat; FEvaluateFlags := Watch.EvaluateFlags; FRepeatCount := Watch.RepeatCount; end; constructor TIdeWatchValue.Create(AOwnerWatch: TIdeWatch; const AThreadId: Integer; const AStackFrame: Integer); begin Create(AOwnerWatch); FThreadId := AThreadId; FStackFrame := AStackFrame; end; procedure TIdeWatchValue.Assign(AnOther: TWatchValue); begin inherited Assign(AnOther); FThreadId := TIdeWatchValue(AnOther).FThreadId; FStackFrame := TIdeWatchValue(AnOther).FStackFrame; FDisplayFormat := TIdeWatchValue(AnOther).FDisplayFormat; end; { TIdeWatchesMonitor } function TIdeWatchesMonitor.GetSnapshot(AnID: Pointer): TIdeWatches; begin Result := TIdeWatches(FSnapshots.SnapShot[AnID]); end; function TIdeWatchesMonitor.GetCurrentWatches: TCurrentWatches; begin Result := TCurrentWatches(Watches); end; procedure TIdeWatchesMonitor.DoStateEnterPause; begin inherited DoStateEnterPause; if (CurrentWatches = nil) then Exit; CurrentWatches.ClearValues; NotifyUpdate(CurrentWatches, nil); end; procedure TIdeWatchesMonitor.DoStateLeavePause; begin inherited DoStateLeavePause; if (CurrentWatches = nil) then Exit; CurrentWatches.SnapShot := nil; end; procedure TIdeWatchesMonitor.DoStateLeavePauseClean; begin inherited DoStateLeavePauseClean; if (CurrentWatches = nil) then Exit; CurrentWatches.SnapShot := nil; CurrentWatches.ClearValues; // TODO: block the update calls, update will be done for all on next line NotifyUpdate(CurrentWatches, nil); end; procedure TIdeWatchesMonitor.DoModified; begin if (FIgnoreModified = 0) and Assigned(FOnModified) then FOnModified(Self); end; procedure TIdeWatchesMonitor.NotifyAdd(const AWatches: TCurrentWatches; const AWatch: TCurrentWatch); begin FNotificationList.NotifyAdd(AWatches, AWatch); end; procedure TIdeWatchesMonitor.NotifyRemove(const AWatches: TCurrentWatches; const AWatch: TCurrentWatch); begin FNotificationList.NotifyRemove(AWatches, AWatch); end; procedure TIdeWatchesMonitor.NotifyUpdate(const AWatches: TCurrentWatches; const AWatch: TCurrentWatch); begin FNotificationList.NotifyUpdate(AWatches, AWatch); end; procedure TIdeWatchesMonitor.RequestData(AWatchValue: TCurrentWatchValue); begin if Supplier <> nil then Supplier.RequestData(AWatchValue) else AWatchValue.Validity := ddsInvalid; end; function TIdeWatchesMonitor.CreateWatches: TWatches; begin Result := TCurrentWatches.Create(Self); end; function TIdeWatchesMonitor.CreateSnapshot(CreateEmpty: Boolean = False): TObject; begin Result := TIdeWatches.Create; if not CreateEmpty then CurrentWatches.SnapShot := TIdeWatches(Result); end; constructor TIdeWatchesMonitor.Create; begin FSnapshots := TDebuggerDataSnapShotList.Create; FIgnoreModified := 0; FNotificationList := TWatchesNotificationList.Create; inherited; end; destructor TIdeWatchesMonitor.Destroy; begin FSnapshots.Clear; FNotificationList.Clear; inherited Destroy; FreeAndNil(FNotificationList); FreeAndNil(FSnapshots); end; procedure TIdeWatchesMonitor.AddNotification(const ANotification: TWatchesNotification); begin FNotificationList.Add(ANotification); end; procedure TIdeWatchesMonitor.RemoveNotification(const ANotification: TWatchesNotification); begin FNotificationList.Remove(ANotification); end; procedure TIdeWatchesMonitor.NewSnapshot(AnID: Pointer; CreateEmpty: Boolean); var S: TObject; begin S := CreateSnapshot(CreateEmpty); FSnapshots.AddSnapShot(AnID, S); end; procedure TIdeWatchesMonitor.RemoveSnapshot(AnID: Pointer); begin FSnapshots.RemoveSnapShot(AnID); end; procedure TIdeWatchesMonitor.Clear; begin CurrentWatches.Clear; end; procedure TIdeWatchesMonitor.LoadFromXMLConfig(const AConfig: TXMLConfig; const APath: string); begin CurrentWatches.LoadFromXMLConfig(AConfig, APath); end; procedure TIdeWatchesMonitor.SaveToXMLConfig(const AConfig: TXMLConfig; const APath: string); begin CurrentWatches.SaveToXMLConfig(AConfig, APath); end; procedure TIdeWatchesMonitor.BeginIgnoreModified; begin inc(FIgnoreModified); end; procedure TIdeWatchesMonitor.EndIgnoreModified; begin dec(FIgnoreModified); end; { TWatchesNotificationList } function TWatchesNotificationList.GetItem(AIndex: Integer): TWatchesNotification; begin Result := TWatchesNotification(FList[AIndex]); end; procedure TWatchesNotificationList.NotifyAdd(const ASender: TCurrentWatches; const AWatch: TCurrentWatch); var i: LongInt; begin i := Count; while NextDownIndex(i) do if Assigned(Items[i].OnAdd) then Items[i].OnAdd(ASender, AWatch); end; procedure TWatchesNotificationList.NotifyUpdate(const ASender: TCurrentWatches; const AWatch: TCurrentWatch); var i: LongInt; begin i := Count; while NextDownIndex(i) do if Assigned(Items[i].OnUpdate) then Items[i].OnUpdate(ASender, AWatch); end; procedure TWatchesNotificationList.NotifyRemove(const ASender: TCurrentWatches; const AWatch: TCurrentWatch); var i: LongInt; begin i := Count; while NextDownIndex(i) do if Assigned(Items[i].OnRemove) then Items[i].OnRemove(ASender, AWatch); end; procedure TCurrentCallStack.SetCurrent(AValue: Integer); begin inherited SetCurrent(AValue); FMonitor.NotifyCurrent; end; function TCurrentCallStack.GetCurrent: Integer; begin case FCurrentValidity of ddsUnknown: begin Result := 0; FCurrentValidity := ddsRequested; FMonitor.RequestCurrent(self); if FCurrentValidity = ddsValid then Result := inherited GetCurrent(); end; ddsRequested, ddsEvaluating: Result := 0; ddsValid: Result := inherited GetCurrent; ddsInvalid, ddsError: Result := 0; end; end; procedure TCurrentCallStack.Clear; var Iterator: TMapIterator; begin Iterator:= TMapIterator.Create(FEntries); while not Iterator.EOM do begin TObject(Iterator.DataPtr^).Free; Iterator.Next; end; Iterator.Free; FEntries.Clear; FCount := -1; FAtLeastCount := -1; FAtLeastCountOld := -1; end; constructor TCurrentCallStack.Create(AMonitor: TIdeCallStackMonitor); begin FCount := 0; FAtLeastCount := 0; FAtLeastCountOld := -1; FEntries:= TMap.Create(its4, SizeOf(TIdeCallStackEntry)); FMonitor := AMonitor; FPreparing := False; FCountValidity := ddsUnknown; FAtLeastCountValidity := ddsUnknown; FCurrentValidity := ddsUnknown; FLowestUnknown := -1; FHighestUnknown := -1; inherited Create; end; destructor TCurrentCallStack.Destroy; begin Clear; inherited Destroy; FreeAndNil(FEntries); end; procedure TCurrentCallStack.Assign(AnOther: TCallStackBase); begin inherited Assign(AnOther); if AnOther is TCurrentCallStack then begin FCount := TCurrentCallStack(AnOther).FCount; FCountValidity := TCurrentCallStack(AnOther).FCountValidity; FAtLeastCount := TCurrentCallStack(AnOther).FAtLeastCount; FAtLeastCountOld := TCurrentCallStack(AnOther).FAtLeastCountOld; end else begin FCount := AnOther.Count; FAtLeastCount := -1; FAtLeastCountOld := -1; end; end; procedure TCurrentCallStack.SetSnapShot(const AValue: TIdeCallStack); begin assert((FSnapShot=nil) or (AValue=nil), 'TCurrentCallStack already have snapshot'); if FSnapShot = AValue then exit; if (FSnapShot <> nil) and (AValue = nil) then FSnapShot.Assign(Self); FSnapShot := AValue; end; function TCurrentCallStack.GetCount: Integer; begin case FCountValidity of ddsUnknown: begin Result := 0; FCountValidity := ddsRequested; FMonitor.RequestCount(self); if FCountValidity = ddsValid then Result := FCount; end; ddsRequested, ddsEvaluating: Result := 0; ddsValid: Result := FCount; ddsInvalid, ddsError: Result := 0; end; end; procedure TCurrentCallStack.SetCount(ACount: Integer); begin if FCount = ACount then exit; FCount := ACount; FAtLeastCount := ACount; if FCountValidity = ddsValid then FMonitor.NotifyChange; end; function TCurrentCallStack.GetEntry(AIndex: Integer): TIdeCallStackEntry; begin if (AIndex < 0) or (AIndex >= CountLimited(AIndex+1)) then IndexError(Aindex); Result := nil; if FEntries.GetData(AIndex, Result) then Exit; Result := TIdeCallStackEntry.Create(AIndex, 0, nil, '', nil, 0, ddsRequested); if Result = nil then Exit; FEntries.Add(AIndex, Result); Result.FOwner := Self; if (FLowestUnknown < 0) or (FLowestUnknown > AIndex) then FLowestUnknown := AIndex; if (FHighestUnknown < AIndex) then FHighestUnknown := AIndex; DoEntriesCreated; end; procedure TCurrentCallStack.AddEntry(AnEntry: TIdeCallStackEntry); begin FEntries.Add(AnEntry.Index, AnEntry); AnEntry.FOwner := Self; end; procedure TCurrentCallStack.AssignEntriesTo(AnOther: TIdeCallStack); var It: TMapIterator; begin It := TMapIterator.Create(FEntries); It.First; while (not IT.EOM) do begin AnOther.AddEntry(TIdeCallStackEntry(It.DataPtr^).CreateCopy as TIdeCallStackEntry); It.Next; end; It.Free; end; function TCurrentCallStack.GetRawEntries: TMap; begin Result := FEntries; end; function TCurrentCallStack.GetLowestUnknown: Integer; begin Result := FLowestUnknown; end; function TCurrentCallStack.GetHighestUnknown: Integer; begin Result := FHighestUnknown; end; function TCurrentCallStack.GetNewCurrentIndex: Integer; begin Result := FNewCurrentIndex; end; procedure TCurrentCallStack.PrepareRange(AIndex, ACount: Integer); var It: TMapIterator; EndIndex: Integer; begin It := TMapIterator.Create(FEntries); if It.Locate(AIndex) then repeat // start searching for the first unavailable Inc(AIndex); Dec(ACount); It.Next; until It.EOM or (ACount <= 0) or (TIdeCallStackEntry(It.DataPtr^).Index <> AIndex); if ACount > 1 then begin EndIndex := AIndex + ACount - 1; if It.Locate(EndIndex) then repeat // start searching for the last unavailable Dec(EndIndex); Dec(ACount); It.Previous; until It.BOM or (ACount <= 0) or (TIdeCallStackEntry(It.DataPtr^).Index <> EndIndex); end; It.Free; if ACount <= 0 then Exit; FPreparing := True; while ACount > 0 do begin Entries[AIndex]; // Request unknown entries: will set LowesUnknown / HighesUnknown inc(AIndex); dec(ACount); end; FPreparing := False; DoEntriesCreated; end; procedure TCurrentCallStack.ChangeCurrentIndex(ANewIndex: Integer); begin FNewCurrentIndex := ANewIndex; FMonitor.UpdateCurrentIndex; end; procedure TCurrentCallStack.DoEntriesCreated; begin if not FPreparing then FMonitor.RequestEntries(Self); end; procedure TCurrentCallStack.DoEntriesUpdated; begin FLowestUnknown := -1; FHighestUnknown := -1; FMonitor.NotifyChange; end; function TCurrentCallStack.HasAtLeastCount(ARequiredMinCount: Integer): TNullableBool; begin if FCountValidity = ddsValid then begin Result := inherited HasAtLeastCount(ARequiredMinCount); exit; end; if FAtLeastCountOld >= ARequiredMinCount then begin Result := nbTrue; exit; end; if (FAtLeastCountValidity = ddsValid) and (FAtLeastCount < ARequiredMinCount) then begin FAtLeastCountOld := FAtLeastCount; FAtLeastCountValidity := ddsUnknown; end; case FAtLeastCountValidity of ddsUnknown: begin Result := nbUnknown; if FCountValidity in [ddsRequested, ddsEvaluating] then exit; FAtLeastCountValidity := ddsRequested; FMonitor.RequestAtLeastCount(self, ARequiredMinCount); if FCountValidity = ddsValid then Result := inherited HasAtLeastCount(ARequiredMinCount) else if FAtLeastCountValidity = ddsValid then begin if ARequiredMinCount <= FAtLeastCount then Result := nbTrue else Result := nbFalse; end; end; ddsRequested, ddsEvaluating: Result := nbUnknown; ddsValid: begin if ARequiredMinCount <= FAtLeastCount then Result := nbTrue else Result := nbFalse; end; ddsInvalid, ddsError: Result := nbFalse; end; end; procedure TCurrentCallStack.SetCountValidity(AValidity: TDebuggerDataState); begin if FCountValidity = AValidity then exit; DebugLn(DBG_DATA_MONITORS, ['DebugDataMonitor: TCurrentCallStack.SetCountValidity: FThreadId=', FThreadId, ' AValidity=',dbgs(AValidity)]); FCountValidity := AValidity; FMonitor.NotifyChange; end; procedure TCurrentCallStack.SetHasAtLeastCountInfo(AValidity: TDebuggerDataState; AMinCount: Integer); begin if (FAtLeastCountValidity = AValidity) then exit; DebugLn(DBG_DATA_MONITORS, ['DebugDataMonitor: TCurrentCallStack.SetCountMinValidity: FThreadId=', FThreadId, ' AValidity=',dbgs(AValidity)]); FAtLeastCountOld := -1; FAtLeastCountValidity := AValidity; FAtLeastCount := AMinCount; FMonitor.NotifyChange; end; procedure TCurrentCallStack.SetCurrentValidity(AValidity: TDebuggerDataState); begin if FCurrentValidity = AValidity then exit; DebugLn(DBG_DATA_MONITORS, ['DebugDataMonitor: TCurrentCallStack.SetCurrentValidity: FThreadId=', FThreadId, ' AValidity=',dbgs(AValidity)]); FCurrentValidity := AValidity; if FCurrentValidity = ddsValid then FMonitor.NotifyChange; FMonitor.NotifyCurrent; end; { TCurrentCallStackList } constructor TCurrentCallStackList.Create(AMonitor: TIdeCallStackMonitor); begin FMonitor := AMonitor; inherited Create; end; procedure TCurrentCallStackList.SetSnapShot(const AValue: TIdeCallStackList); var R: TIdeCallStack; i: Integer; begin assert((FSnapShot=nil) or (AValue=nil), 'Callstack already have snapshot'); if FSnapShot = AValue then exit; FSnapShot := AValue; if FSnapShot = nil then begin for i := 0 to Count - 1 do TCurrentCallStack(Entries[i]).SnapShot := nil; end else begin // Assign FSnapShot.Clear; for i := 0 to Count - 1 do begin R := TIdeCallStack.Create; R.ThreadId := Entries[i].ThreadId; FSnapShot.Add(R); TCurrentCallStack(Entries[i]).SnapShot := R; end; end; end; function TCurrentCallStackList.NewEntryForThread(const AThreadId: Integer): TCallStackBase; var R: TIdeCallStack; begin try DebugLnEnter(DBG_DATA_MONITORS, ['DebugDataMonitor: >>ENTER: TCurrentCallStackList.GetEntryForThread: ThreadId=', AThreadId]); Result := TCurrentCallStack.Create(FMonitor); Result.ThreadId := AThreadId; Add(Result); if FSnapShot <> nil then begin R := TIdeCallStack.Create; R.ThreadId := AThreadId; FSnapShot.Add(R); TCurrentCallStack(Result).SnapShot := R; end; finally DebugLnExit(DBG_DATA_MONITORS, ['DebugDataMonitor: < nil) then FSnapShot.Assign(self); FDataValidity := AValidity; if FDataValidity = ddsUnknown then Clear; FMonitor.Changed; end; procedure TCurrentThreads.SetCurrentThreadId(AValue: Integer); begin if CurrentThreadId = AValue then exit; DebugLn(DBG_DATA_MONITORS, ['DebugDataMonitor: TCurrentThreads.SetCurrentThreadId ', AValue]); inherited SetCurrentThreadId(AValue); FMonitor.CurrentChanged; // TODO ChangedSelection end; procedure TCurrentThreads.SetSnapShot(const AValue: TIdeThreads); begin assert((FSnapShot=nil) or (AValue=nil), 'Threads already have snapshot'); if FSnapShot = AValue then exit; FSnapShot := AValue; if FSnapShot <> nil then FSnapShot.Assign(self); end; constructor TCurrentThreads.Create(AMonitor: TIdeThreadsMonitor); begin FMonitor := AMonitor; FDataValidity := ddsUnknown; inherited Create; end; function TCurrentThreads.Count: Integer; begin if (FDataValidity = ddsUnknown) and Paused then begin FDataValidity := ddsRequested; Paused := False; FMonitor.RequestData; end; Result := inherited Count; //case FDataValidity of // ddsUnknown: begin // Result := 0; // FDataValidity := ddsRequested; // FMonitor.RequestData; // if FDataValidity = ddsValid then Result := inherited Count(); // end; // ddsRequested, ddsEvaluating: Result := 0; // ddsValid: Result := inherited Count; // ddsInvalid, ddsError: Result := 0; //end; end; procedure TCurrentThreads.Clear; begin FDataValidity := ddsUnknown; inherited Clear; end; function TCurrentThreads.CreateEntry(const AnAdress: TDbgPtr; const AnArguments: TStrings; const AFunctionName: String; const FileName, FullName: String; const ALine: Integer; const AThreadId: Integer; const AThreadName: String; const AThreadState: String; AState: TDebuggerDataState): TThreadEntry; begin Result := inherited CreateEntry(AnAdress, AnArguments, AFunctionName, FileName, FullName, ALine, AThreadId, AThreadName, AThreadState, AState); TIdeThreadEntry(Result).FThreadOwner := self; end; { TIdeThreadsMonitor } function TIdeThreadsMonitor.GetSnapshot(AnID: Pointer): TIdeThreads; begin Result := TIdeThreads(FSnapshots.SnapShot[AnID]); end; function TIdeThreadsMonitor.GetCurrentThreads: TCurrentThreads; begin Result :=TCurrentThreads(Threads); end; procedure TIdeThreadsMonitor.DoModified; begin Changed; end; procedure TIdeThreadsMonitor.DoStateEnterPause; begin inherited DoStateEnterPause; if (CurrentThreads = nil) then Exit; CurrentThreads.SetValidity(ddsUnknown); CurrentThreads.Paused := True; end; procedure TIdeThreadsMonitor.DoStateLeavePause; begin inherited DoStateLeavePause; if (CurrentThreads = nil) then Exit; CurrentThreads.SnapShot := nil; end; procedure TIdeThreadsMonitor.DoStateLeavePauseClean; begin inherited DoStateLeavePauseClean; if (CurrentThreads = nil) then Exit; CurrentThreads.SnapShot := nil; end; procedure TIdeThreadsMonitor.DoNewSupplier; begin inherited DoNewSupplier; if CurrentThreads <> nil then CurrentThreads.SetValidity(ddsUnknown); end; procedure TIdeThreadsMonitor.RequestData; begin if Supplier <> nil then Supplier.RequestMasterData; end; function TIdeThreadsMonitor.CreateSnapshot(CreateEmpty: Boolean = False): TObject; begin Result := TIdeThreads.Create; if not CreateEmpty then CurrentThreads.SnapShot := TIdeThreads(Result); end; function TIdeThreadsMonitor.CreateThreads: TThreads; begin Result := TCurrentThreads.Create(self); end; procedure TIdeThreadsMonitor.Changed; begin FNotificationList.NotifyChange(Self); end; procedure TIdeThreadsMonitor.CurrentChanged; begin FNotificationList.NotifyChange(Self); // TODO: is this required?? It should not FNotificationList.NotifyCurrent(Self); end; constructor TIdeThreadsMonitor.Create; begin FSnapshots := TDebuggerDataSnapShotList.Create; inherited; FNotificationList := TDebuggerChangeNotificationList.Create; end; destructor TIdeThreadsMonitor.Destroy; begin FSnapshots.Clear; FNotificationList.Clear; inherited Destroy; FreeAndNil(FNotificationList); FreeAndNil(FSnapshots); end; procedure TIdeThreadsMonitor.Clear; begin DebugLn(DBG_DATA_MONITORS, ['DebugDataMonitor: TIdeThreadsMonitor.Clear']); CurrentThreads.Clear; Changed; end; procedure TIdeThreadsMonitor.AddNotification(const ANotification: TThreadsNotification); begin FNotificationList.Add(ANotification); end; procedure TIdeThreadsMonitor.RemoveNotification(const ANotification: TThreadsNotification); begin FNotificationList.Remove(ANotification); end; procedure TIdeThreadsMonitor.NewSnapshot(AnID: Pointer; CreateEmpty: Boolean); var S: TObject; begin S := CreateSnapshot(CreateEmpty); FSnapshots.AddSnapShot(AnID, S); end; procedure TIdeThreadsMonitor.RemoveSnapshot(AnID: Pointer); begin FSnapshots.RemoveSnapShot(AnID); end; procedure TIdeThreadsMonitor.ChangeCurrentThread(ANewId: Integer); begin if Supplier <> nil then Supplier.ChangeCurrentThread(ANewId); end; { TDebuggerChangeNotificationList } function TDebuggerChangeNotificationList.GetItem(AIndex: Integer): TDebuggerChangeNotification; begin Result := TDebuggerChangeNotification(FList[AIndex]); end; procedure TDebuggerChangeNotificationList.NotifyChange(Sender: TObject); var i: LongInt; begin i := Count; while NextDownIndex(i) do if Assigned(Items[i].OnChange) then Items[i].OnChange(Sender); end; procedure TDebuggerChangeNotificationList.NotifyCurrent(Sender: TObject); var i: LongInt; begin i := Count; while NextDownIndex(i) do if Assigned(Items[i].OnCurrent) then Items[i].OnCurrent(Sender); end; { TDebuggerNotificationList } function TDebuggerNotificationList.GetItem(AIndex: Integer): TDebuggerNotification; begin Result := TDebuggerNotification(FList[AIndex]); end; function TDebuggerNotificationList.NextDownIndex(var Index: integer): boolean; begin dec(Index); if (Index >= FList.Count) then Index := FList.Count-1; Result := Index >= 0; end; function TDebuggerNotificationList.Count: Integer; begin Result := FList.Count; end; procedure TDebuggerNotificationList.Clear; begin while Count > 0 do Remove(Items[0]); end; constructor TDebuggerNotificationList.Create; begin FList := TList.Create; end; destructor TDebuggerNotificationList.Destroy; begin inherited Destroy; Clear; FreeAndNil(FList); end; procedure TDebuggerNotificationList.Add(const ANotification: TDebuggerNotification); begin FList.Add(ANotification); ANotification.AddReference; end; procedure TDebuggerNotificationList.Remove(const ANotification: TDebuggerNotification); begin ANotification.ReleaseReference; FList.Remove(ANotification); end; { TThreadEntry } function TIdeThreadEntry.GetTopFrame: TIdeThreadFrameEntry; begin Result := TIdeThreadFrameEntry(inherited TopFrame); end; function TIdeThreadEntry.CreateStackEntry: TCallStackEntry; begin Result := TIdeThreadFrameEntry.Create; TIdeThreadFrameEntry(Result).FThread := Self; end; function TIdeThreadEntry.GetUnitInfoProvider: TDebuggerUnitInfoProvider; begin if FThreadOwner = nil then Result := nil else Result := (FThreadOwner as TCurrentThreads).FMonitor.UnitInfoProvider; end; procedure TIdeThreadEntry.SetThreadState(AValue: String); begin if ThreadState = AValue then Exit; inherited SetThreadState(AValue); TopFrame.ClearLocation; end; procedure TIdeThreadEntry.LoadDataFromXMLConfig(const AConfig: TXMLConfig; const APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil); begin TIdeCallStackEntry(TopFrame).LoadDataFromXMLConfig(AConfig, APath, AUnitInvoPrv); FThreadId := AConfig.GetValue(APath + 'ThreadId', -1); FThreadName := AConfig.GetValue(APath + 'ThreadName', ''); FThreadState := AConfig.GetValue(APath + 'ThreadState', ''); end; procedure TIdeThreadEntry.SaveDataToXMLConfig(const AConfig: TXMLConfig; const APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil); begin TIdeCallStackEntry(TopFrame).SaveDataToXMLConfig(AConfig, APath, AUnitInvoPrv); AConfig.SetValue(APath + 'ThreadId', ThreadId); AConfig.SetValue(APath + 'ThreadName', ThreadName); AConfig.SetValue(APath + 'ThreadState', ThreadState); end; function TIdeThreadEntry.CreateCopy: TThreadEntry; begin Result := TIdeThreadEntry.Create; Result.Assign(Self); end; { TIdeThreads } function TIdeThreads.GetEntry(const AnIndex: Integer): TIdeThreadEntry; begin Result := TIdeThreadEntry(inherited Entries[AnIndex]); end; function TIdeThreads.GetEntryById(const AnID: Integer): TIdeThreadEntry; begin Result := TIdeThreadEntry(inherited EntryById[AnID]); end; procedure TIdeThreads.LoadDataFromXMLConfig(const AConfig: TXMLConfig; APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil); var c, i: Integer; e: TIdeThreadEntry; NewCurrentThreadId: Integer; begin Clear; NewCurrentThreadId := AConfig.GetValue(APath + 'CurrentThreadId', -1); inherited SetCurrentThreadId(NewCurrentThreadId); c := AConfig.GetValue(APath + 'Count', 0); APath := APath + 'Entry'; for i := 0 to c - 1 do begin e := TIdeThreadEntry.Create; e.LoadDataFromXMLConfig(AConfig, APath + IntToStr(i) + '/', AUnitInvoPrv); List.Add(e); end; end; procedure TIdeThreads.SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil); var i: Integer; begin AConfig.SetValue(APath + 'CurrentThreadId', CurrentThreadId); AConfig.SetDeleteValue(APath + 'Count', Count, 0); APath := APath + 'Entry'; for i := 0 to Count - 1 do Entries[i].SaveDataToXMLConfig(AConfig, APath + IntToStr(i) + '/', AUnitInvoPrv); end; function TIdeThreads.CreateEntry(const AnAdress: TDbgPtr; const AnArguments: TStrings; const AFunctionName: String; const FileName, FullName: String; const ALine: Integer; const AThreadId: Integer; const AThreadName: String; const AThreadState: String; AState: TDebuggerDataState): TThreadEntry; begin Result := TIdeThreadEntry.Create(AnAdress, AnArguments, AFunctionName, FileName, FullName, ALine, AThreadId, AThreadName, AThreadState, AState); TIdeThreadEntry(Result).FThreadOwner := self; end; procedure TIdeThreads.SetValidity(AValidity: TDebuggerDataState); begin assert(false, 'TIdeThreads.SetValidity'); end; (******************************************************************************) (******************************************************************************) (** **) (** B R E A K P O I N T S **) (** **) (******************************************************************************) (******************************************************************************) { =========================================================================== } { TIDEBreakPoint } { =========================================================================== } function TIDEBreakPoint.GetAutoContinueTime: Cardinal; begin Result := FAutoContinueTime; end; procedure TIDEBreakPoint.SetAutoContinueTime(const AValue: Cardinal); begin if FAutoContinueTime = AValue then Exit; FAutoContinueTime := AValue; //Changed; DoUserChanged; end; procedure TIDEBreakPoint.SetLogEvalExpression(AValue: String); begin if FLogEvalExpression <> AValue then begin FLogEvalExpression := AValue; //Changed; DoUserChanged; end; end; procedure TIDEBreakPoint.SetLogMessage(const AValue: String); begin if FLogMessage <> AValue then begin FLogMessage := AValue; //Changed; DoUserChanged; end; end; function TIDEBreakPoint.GetLogMessage: String; begin Result := FLogMessage; end; function TIDEBreakPoint.GetLogCallStackLimit: Integer; begin Result := FLogCallStackLimit; end; procedure TIDEBreakPoint.SetLogCallStackLimit(const AValue: Integer); begin if FLogCallStackLimit <> AValue then begin FLogCallStackLimit := AValue; //Changed; DoUserChanged; end; end; procedure TIDEBreakPoint.AssignLocationTo(Dest: TPersistent); var DestBreakPoint: TBaseBreakPoint absolute Dest; begin if DestBreakPoint is TDBGBreakPoint then DestBreakPoint.SetLocation(Source, DebugExeLine) else inherited; end; procedure TIDEBreakPoint.AssignTo(Dest: TPersistent); begin inherited; if Dest is TIDEBreakPoint then begin TIDEBreakPoint(Dest).Actions := FActions; TIDEBreakPoint(Dest).AutoContinueTime := FAutoContinueTime; TIDEBreakPoint(Dest).Group := FGroup; TIDEBreakPoint(Dest).LogEvalExpression := FLogEvalExpression; TIDEBreakPoint(Dest).LogMessage := FLogMessage; TIDEBreakPoint(Dest).LogCallStackLimit := FLogCallStackLimit; TIDEBreakPoint(Dest).EnableGroupList.Assign(FEnableGroupList); TIDEBreakPoint(Dest).DisableGroupList.Assign(FDisableGroupList); end; if (Collection <> nil) and (TIDEBreakPoints(Collection).FMaster <> nil) and (Dest is TDBGBreakPoint) then begin Assert(FMaster=nil, 'TManagedBreakPoint.AssignTO already has Master'); if FMaster <> nil then FMaster.Slave := nil; FMaster := TDBGBreakPoint(Dest); FMaster.Slave := Self; end; end; procedure TIDEBreakPoint.DoChanged; begin if (FMaster <> nil) and (FMaster.Slave = nil) then FMaster := nil; inherited DoChanged; end; procedure TIDEBreakPoint.DoUserChanged; begin FUserModified := True; DoChanged; end; function TIDEBreakPoint.GetHitCount: Integer; begin if FMaster = nil then Result := 0 else Result := FMaster.HitCount; end; function TIDEBreakPoint.GetValid: TValidState; begin if FMaster = nil then Result := vsUnknown else Result := FMaster.Valid; end; procedure TIDEBreakPoint.SetBreakHitCount(const AValue: Integer); begin if BreakHitCount = AValue then exit; inherited SetBreakHitCount(AValue); DoUserChanged; if FMaster <> nil then FMaster.BreakHitCount := AValue; end; procedure TIDEBreakPoint.SetEnabled(const AValue: Boolean); begin if Enabled = AValue then exit; inherited SetEnabled(AValue); InitialEnabled:=Enabled; if FMaster <> nil then FMaster.Enabled := AValue; end; procedure TIDEBreakPoint.SetInitialEnabled(const AValue: Boolean); begin if InitialEnabled = AValue then exit; inherited SetInitialEnabled(AValue); DoUserChanged; if FMaster <> nil then FMaster.InitialEnabled := AValue; end; procedure TIDEBreakPoint.SetExpression(const AValue: String); begin if AValue=Expression then exit; inherited SetExpression(AValue); DoUserChanged; if FMaster <> nil then FMaster.Expression := AValue; end; function TIDEBreakPoint.DebugExeLine: Integer; begin Result := Line; end; procedure TIDEBreakPoint.ClearAllGroupLists; begin FDisableGroupList.Clear; FEnableGroupList.Clear; end; {$IFDEF DBG_BREAKPOINT} function TIDEBreakPoint.DebugText: string; var s: String; begin WriteStr(s, FKind); Result := dbgs(self) + ' ' + s + ' at ' + Source +':' + IntToStr(Line); end; {$ENDIF} constructor TIDEBreakPoint.Create(ACollection: TCollection); begin inherited Create(ACollection); FGroup := nil; FActions := [bpaStop]; FDisableGroupList := TIDEBreakPointGroupList.Create(Self); FEnableGroupList := TIDEBreakPointGroupList.Create(Self); end; destructor TIDEBreakPoint.Destroy; var Grp: TIDEBreakPointGroup; begin if FMaster <> nil then begin FMaster.Slave := nil; ReleaseRefAndNil(FMaster); end; if (TIDEBreakPoints(Collection) <> nil) then TIDEBreakPoints(Collection).NotifyRemove(Self); Grp := FGroup; FGroup := nil; if Grp <> nil then Grp.Remove(Self); ClearAllGroupLists; inherited; FreeAndNil(FDisableGroupList); FreeAndNil(FEnableGroupList); end; procedure TIDEBreakPoint.DisableGroups; var n: Integer; begin {$IFDEF DBG_BREAKPOINT} DebugLn(['DisableGroups: ', DebugText, ' Cnt=', FDisableGroupList.Count]); {$ENDIF} for n := 0 to FDisableGroupList.Count - 1 do FDisableGroupList[n].Enabled := False; end; procedure TIDEBreakPoint.DoActionChange; begin //Changed; DoUserChanged; end; procedure TIDEBreakPoint.DoHit(const ACount: Integer; var AContinue: Boolean); begin inherited DoHit(ACount, AContinue); AContinue := AContinue or not (bpaStop in Actions); if bpaLogMessage in Actions then FMaster.DoLogMessage(FLogMessage); if (bpaEValExpression in Actions) and (Trim(FLogEvalExpression) <> '') then FMaster.DoLogExpression(Trim(FLogEvalExpression)); if bpaLogCallStack in Actions then FMaster.DoLogCallStack(FLogCallStackLimit); // SnapShot is taken in TDebugManager.DebuggerChangeState if bpaEnableGroup in Actions then EnableGroups; if bpaDisableGroup in Actions then DisableGroups; end; procedure TIDEBreakPoint.EnableGroups; var n: Integer; begin {$IFDEF DBG_BREAKPOINT} DebugLn(['EnableGroups: ', DebugText, ' Cnt=', FEnableGroupList.Count]); {$ENDIF} for n := 0 to FEnableGroupList.Count - 1 do FEnableGroupList[n].Enabled := True; end; function TIDEBreakPoint.GetActions: TIDEBreakPointActions; begin Result := FActions; end; function TIDEBreakPoint.GetGroup: TIDEBreakPointGroup; begin Result := FGroup; end; procedure TIDEBreakPoint.LoadFromXMLConfig(XMLConfig: TXMLConfig; const Path: string; const OnLoadFilename: TOnLoadFilenameFromConfig; const OnGetGroup: TOnGetGroupByName); procedure LoadGroupList(GroupList: TIDEBreakPointGroupList; const ListPath: string); var i: Integer; CurGroup: TIDEBreakPointGroup; NewCount: Integer; GroupName: String; begin GroupList.Clear; NewCount:=XMLConfig.GetValue(ListPath+'Count',0); for i:=0 to NewCount-1 do begin GroupName:=XMLConfig.GetValue(ListPath+'Group'+IntToStr(i+1)+'/Name',''); if GroupName='' then continue; CurGroup:=OnGetGroup(GroupName); if CurGroup=nil then continue; GroupList.Add(CurGroup); end; end; var Filename: String; GroupName: String; NewActions: TIDEBreakPointActions; CurAction: TIDEBreakPointAction; begin FLoading:=true; try Kind:=TDBGBreakPointKind(GetEnumValueDef(TypeInfo(TDBGBreakPointKind),XMLConfig.GetValue(Path+'Kind/Value',''),0)); GroupName:=XMLConfig.GetValue(Path+'Group/Name',''); Group:=OnGetGroup(GroupName); Expression:=XMLConfig.GetValue(Path+'Expression/Value',''); AutoContinueTime:=XMLConfig.GetValue(Path+'AutoContinueTime/Value',0); BreakHitCount := XMLConfig.GetValue(Path+'BreakHitCount/Value',0); Address:=XMLConfig.GetValue(Path+'Address/Value',0); FWatchData := XMLConfig.GetValue(Path+'WatchData/Value', ''); try ReadStr(XMLConfig.GetValue(Path+'WatchScope/Value', 'wpsGlobal'), FWatchScope); except FWatchScope := wpsGlobal; end; try ReadStr(XMLConfig.GetValue(Path+'WatchKind/Value', 'wpkWrite'), FWatchKind); except FWatchKind:= wpkWrite; end; Filename:=XMLConfig.GetValue(Path+'Source/Value',''); if Assigned(OnLoadFilename) then OnLoadFilename(Filename); FSource:=Filename; InitialEnabled:=XMLConfig.GetValue(Path+'InitialEnabled/Value',true); Enabled:=FInitialEnabled; FLine:=XMLConfig.GetValue(Path+'Line/Value',-1); FLogEvalExpression := XMLConfig.GetValue(Path+'LogEvalExpression/Value', ''); FLogMessage:=XMLConfig.GetValue(Path+'LogMessage/Value',''); FLogCallStackLimit:=XMLConfig.GetValue(Path+'LogCallStackLimit/Value',0); NewActions:=[]; for CurAction:=Low(TIDEBreakPointAction) to High(TIDEBreakPointAction) do if XMLConfig.GetValue( Path+'Actions/'+DBGBreakPointActionNames[CurAction], CurAction in [bpaStop]) then Include(NewActions,CurAction); Actions:=NewActions; LoadGroupList(FDisableGroupList,Path+'DisableGroups/'); LoadGroupList(FEnableGroupList,Path+'EnableGroups/'); finally FLoading:=false; end; end; procedure TIDEBreakPoint.SaveToXMLConfig(const AConfig: TXMLConfig; const APath: string; const OnSaveFilename: TOnSaveFilenameToConfig); procedure SaveGroupList(const AList: TIDEBreakPointGroupList; const AListPath: string); var i: Integer; CurGroup: TIDEBreakPointGroup; begin AConfig.SetDeleteValue(AListPath + 'Count', AList.Count,0); for i := 0 to AList.Count - 1 do begin CurGroup := AList[i]; AConfig.SetDeleteValue(AListPath+'Group'+IntToStr(i+1)+'/Name', CurGroup.Name, ''); end; end; var s, Filename: String; CurAction: TIDEBreakPointAction; begin AConfig.SetDeleteValue(APath+'Kind/Value',GetEnumName(TypeInfo(TDBGBreakPointKind), Ord(Kind)), ''); AConfig.SetDeleteValue(APath+'Address/Value',Address,0); AConfig.SetDeleteValue(APath+'WatchData/Value', FWatchData, ''); WriteStr(s{%H-}, FWatchScope); AConfig.SetDeleteValue(APath+'WatchScope/Value', s, ''); WriteStr(s, FWatchKind); AConfig.SetDeleteValue(APath+'WatchKind/Value', s, ''); if Group <> nil then AConfig.SetDeleteValue(APath+'Group/Name',Group.Name,''); AConfig.SetDeleteValue(APath+'Expression/Value',Expression,''); AConfig.SetDeleteValue(APath+'AutoContinueTime/Value',AutoContinueTime,0); AConfig.SetDeleteValue(APath+'BreakHitCount/Value',BreakHitCount,0); Filename := Source; if Assigned(OnSaveFilename) then OnSaveFilename(Filename); AConfig.SetDeleteValue(APath+'Source/Value',Filename,''); AConfig.SetDeleteValue(APath+'InitialEnabled/Value',InitialEnabled,true); AConfig.SetDeleteValue(APath+'Line/Value',Line,-1); AConfig.SetDeleteValue(APath+'LogEvalExpression/Value', FLogEvalExpression,''); AConfig.SetDeleteValue(APath+'LogMessage/Value',LogMessage,''); AConfig.SetDeleteValue(APath+'LogCallStackLimit/Value',LogCallStackLimit,0); for CurAction := Low(TIDEBreakPointAction) to High(TIDEBreakPointAction) do begin AConfig.SetDeleteValue( APath+'Actions/'+DBGBreakPointActionNames[CurAction], CurAction in Actions, CurAction in [bpaStop]); end; SaveGroupList(FDisableGroupList, APath + 'DisableGroups/'); SaveGroupList(FEnableGroupList, APath + 'EnableGroups/'); end; procedure TIDEBreakPoint.SetAddress(const AValue: TDBGPtr); begin inherited SetAddress(AValue); if FMaster<>nil then FMaster.Address := Address; end; procedure TIDEBreakPoint.SetLocation(const ASource: String; const ALine: Integer); begin inherited SetLocation(ASource, ALine); if FMaster<>nil then FMaster.SetLocation(ASource, DebugExeLine); end; procedure TIDEBreakPoint.SetWatch(const AData: String; const AScope: TDBGWatchPointScope; const AKind: TDBGWatchPointKind); begin inherited SetWatch(AData, AScope, AKind); if FMaster<>nil then FMaster.SetWatch(AData, AScope, AKind); end; procedure TIDEBreakPoint.ResetMaster; begin if FMaster <> nil then FMaster.Slave := nil; FMaster := nil; Changed; end; procedure TIDEBreakPoint.SetActions(const AValue: TIDEBreakPointActions); begin if FActions <> AValue then begin FActions := AValue; DoActionChange; end; end; procedure TIDEBreakPoint.SetGroup(const AValue: TIDEBreakPointGroup); var Grp: TIDEBreakPointGroup; begin if FGroup <> AValue then begin if FGroup <> nil then begin Grp := FGroup; FGroup := nil; // avoid second entrance Grp.Remove(Self); end; FGroup := AValue; if FGroup <> nil then begin FGroup.Add(Self); end; //Changed; DoUserChanged; end; end; (* procedure TIDEBreakPoint.CopyGroupList(SrcGroupList, DestGroupList: TIDEBreakPointGroupList; DestGroups: TIDEBreakPointGroups); var i: Integer; CurGroup: TIDEBreakPointGroup; NewGroup: TIDEBreakPointGroup; begin DestGroupList.clear; for i:=0 to SrcGroupList.Count-1 do begin CurGroup:=TIDEBreakPointGroup(SrcGroupList[i]); NewGroup:=DestGroups.GetGroupByName(CurGroup.Name); DestGroupList.Add(NewGroup); end; end; procedure TIDEBreakPoint.CopyAllGroupLists(SrcBreakPoint: TIDEBreakPoint; DestGroups: TIDEBreakPointGroups); begin CopyGroupList(SrcBreakPoint.FEnableGroupList,FEnableGroupList,DestGroups); CopyGroupList(SrcBreakPoint.FDisableGroupList,FDisableGroupList,DestGroups); end; *) { =========================================================================== } { TIDEBreakPoints } { =========================================================================== } function TIDEBreakPoints.Add(const ASource: String; const ALine: Integer): TIDEBreakPoint; begin Result := TIDEBreakPoint(inherited Add(ASource, ALine)); NotifyAdd(Result); end; function TIDEBreakPoints.Add(const AAddress: TDBGPtr): TIDEBreakPoint; begin Result := TIDEBreakPoint(inherited Add(AAddress)); NotifyAdd(Result); end; function TIDEBreakPoints.Add(const AData: String; const AScope: TDBGWatchPointScope; const AKind: TDBGWatchPointKind): TIDEBreakPoint; begin Result := TIDEBreakPoint(inherited Add(AData, AScope, AKind)); NotifyAdd(Result); end; procedure TIDEBreakPoints.AddNotification( const ANotification: TIDEBreakPointsNotification); begin FNotificationList.Add(ANotification); ANotification.AddReference; end; constructor TIDEBreakPoints.Create(const ABreakPointClass: TIDEBreakPointClass); begin FMaster := nil; FNotificationList := TList.Create; inherited Create(ABreakPointClass); end; destructor TIDEBreakPoints.Destroy; var n: Integer; begin for n := FNotificationList.Count - 1 downto 0 do TDebuggerNotification(FNotificationList[n]).ReleaseReference; inherited; FreeAndNil(FNotificationList); end; function TIDEBreakPoints.Find(const ASource: String; const ALine: Integer): TIDEBreakPoint; begin Result := TIDEBreakPoint(inherited Find(ASource, ALine, nil)); end; function TIDEBreakPoints.Find(const ASource: String; const ALine: Integer; const AIgnore: TIDEBreakPoint): TIDEBreakPoint; begin Result := TIDEBreakPoint(inherited Find(ASource, ALine, AIgnore)); end; function TIDEBreakPoints.Find(const AAddress: TDBGPtr): TIDEBreakPoint; begin Result := TIDEBreakPoint(inherited Find(AAddress)); end; function TIDEBreakPoints.Find(const AAddress: TDBGPtr; const AIgnore: TIDEBreakPoint): TIDEBreakPoint; begin Result := TIDEBreakPoint(inherited Find(AAddress, AIgnore)); end; function TIDEBreakPoints.Find(const AData: String; const AScope: TDBGWatchPointScope; const AKind: TDBGWatchPointKind): TIDEBreakPoint; begin Result := TIDEBreakPoint(inherited Find(AData, AScope, AKind)); end; function TIDEBreakPoints.Find(const AData: String; const AScope: TDBGWatchPointScope; const AKind: TDBGWatchPointKind; const AIgnore: TIDEBreakPoint): TIDEBreakPoint; begin Result := TIDEBreakPoint(inherited Find(AData, AScope, AKind, AIgnore)); end; procedure TIDEBreakPoints.SetMaster(const AValue: TDBGBreakPoints); var n: Integer; begin if FMaster = AValue then Exit; FMaster := AValue; if FMaster = nil then begin for n := 0 to Count - 1 do Items[n].ResetMaster; end else begin FMaster.Assign(Self); end; end; function TIDEBreakPoints.GetItem(const AnIndex: Integer): TIDEBreakPoint; begin Result := TIDEBreakPoint(inherited GetItem(AnIndex)); end; procedure TIDEBreakPoints.NotifyAdd(const ABreakPoint: TIDEBreakPoint); var n: Integer; Notification: TIDEBreakPointsNotification; BP: TBaseBreakPoint; begin ABreakpoint.InitialEnabled := True; ABreakpoint.Enabled := True; for n := 0 to FNotificationList.Count - 1 do begin Notification := TIDEBreakPointsNotification(FNotificationList[n]); if Assigned(Notification.FOnAdd) then Notification.FOnAdd(Self, ABreakPoint); end; if FMaster <> nil then begin // create without source. it will be set in assign (but during Begin/EndUpdate) BP := FMaster.Add('', 0); BP.Assign(ABreakPoint); // will set ABreakPoint.FMaster := BP; end; end; procedure TIDEBreakPoints.NotifyRemove(const ABreakpoint: TIDEBreakPoint); var n: Integer; Notification: TIDEBreakPointsNotification; begin for n := 0 to FNotificationList.Count - 1 do begin Notification := TIDEBreakPointsNotification(FNotificationList[n]); if Assigned(Notification.FOnRemove) then Notification.FOnRemove(Self, ABreakpoint); end; end; procedure TIDEBreakPoints.RemoveNotification( const ANotification: TIDEBreakPointsNotification); begin FNotificationList.Remove(ANotification); ANotification.ReleaseReference; end; procedure TIDEBreakPoints.LoadFromXMLConfig(XMLConfig: TXMLConfig; const Path: string; const OnLoadFilename: TOnLoadFilenameFromConfig; const OnGetGroup: TOnGetGroupByName); var NewCount: Integer; i: Integer; LoadBreakPoint: TIDEBreakPoint; BreakPoint: TIDEBreakPoint; begin Clear; NewCount:=XMLConfig.GetValue(Path+'Count',0); for i:=0 to NewCount-1 do begin LoadBreakPoint := TIDEBreakPoint.Create(nil); LoadBreakPoint.LoadFromXMLConfig(XMLConfig, Path+'Item'+IntToStr(i+1)+'/',OnLoadFilename,OnGetGroup); case LoadBreakPoint.Kind of bpkSource: begin BreakPoint := Find(LoadBreakPoint.Source, LoadBreakPoint.Line, LoadBreakPoint); if BreakPoint = nil then BreakPoint := Add(LoadBreakPoint.Source, LoadBreakPoint.Line); end; bpkAddress: begin BreakPoint := Find(LoadBreakPoint.Address, LoadBreakPoint); if BreakPoint = nil then BreakPoint := Add(LoadBreakPoint.Address); end; bpkData: begin BreakPoint := Find(LoadBreakPoint.WatchData, LoadBreakPoint.WatchScope, LoadBreakPoint.WatchKind, LoadBreakPoint); if BreakPoint = nil then BreakPoint := Add(LoadBreakPoint.WatchData, LoadBreakPoint.WatchScope, LoadBreakPoint.WatchKind); end; end; BreakPoint.Assign(LoadBreakPoint); ReleaseRefAndNil(LoadBreakPoint) end; end; procedure TIDEBreakPoints.SaveToXMLConfig(XMLConfig: TXMLConfig; const Path: string; const OnSaveFilename: TOnSaveFilenameToConfig); var Cnt: Integer; i: Integer; CurBreakPoint: TIDEBreakPoint; begin Cnt:=Count; XMLConfig.SetDeleteValue(Path+'Count',Cnt,0); for i:=0 to Cnt-1 do begin CurBreakPoint:=Items[i]; CurBreakPoint.SaveToXMLConfig(XMLConfig, Path+'Item'+IntToStr(i+1)+'/',OnSaveFilename); end; end; procedure TIDEBreakPoints.SetItem(const AnIndex: Integer; const AValue: TIDEBreakPoint); begin inherited SetItem(AnIndex, AValue); end; procedure TIDEBreakPoints.Update(Item: TCollectionItem); var n: Integer; Notification: TIDEBreakPointsNotification; begin // Note: Item will be nil in case all items need to be updated for n := 0 to FNotificationList.Count - 1 do begin Notification := TIDEBreakPointsNotification(FNotificationList[n]); if Assigned(Notification.FOnUpdate) then Notification.FOnUpdate(Self, TIDEBreakPoint(Item)); end; end; { =========================================================================== } { TIDEBreakPointGroup } { =========================================================================== } function TIDEBreakPointGroup.Add(const ABreakPoint: TIDEBreakPoint): Integer; begin Result := FBreakpoints.IndexOf(ABreakPoint); //avoid dups if Result = -1 then begin Result := FBreakpoints.Add(ABreakPoint); ABreakpoint.Group := Self; end; end; procedure TIDEBreakPointGroup.AddReference(const ABreakPointList: TIDEBreakPointGroupList); begin FReferences.Add(ABreakPointList); end; function TIDEBreakPointGroup.Count: Integer; begin Result := FBreakpoints.Count; end; constructor TIDEBreakPointGroup.Create(ACollection: TCollection); begin inherited Create(ACollection); FBreakpoints := TList.Create; FReferences := TList.Create; FEnabled := True; end; procedure TIDEBreakPointGroup.Delete(const AIndex: Integer); begin Remove(TIDEBreakPoint(FBreakPoints[AIndex])); end; destructor TIDEBreakPointGroup.Destroy; var n: Integer; begin for n := FBreakpoints.Count - 1 downto 0 do TIDEBreakPoint(FBreakpoints[n]).Group := nil; for n := FReferences.Count - 1 downto 0 do TIDEBreakPointGroupList(FReferences[n]).Remove(Self); inherited Destroy; FreeAndNil(FBreakpoints); FreeAndNil(FReferences); end; function TIDEBreakPointGroup.GetBreakpoint(const AIndex: Integer): TIDEBreakPoint; begin Result := TIDEBreakPoint(FBreakPoints[AIndex]); end; function TIDEBreakPointGroup.Remove(const ABreakPoint: TIDEBreakPoint): Integer; begin Result := FBreakpoints.Remove(ABreakPoint); if ABreakpoint.Group = Self then ABreakpoint.Group := nil; end; procedure TIDEBreakPointGroup.LoadFromXMLConfig(XMLConfig: TXMLConfig; const Path: string); begin Name:=XMLConfig.GetValue(Path+'Name/Value',''); // the breakpoints of this group are not loaded here. // They are loaded by the TIDEBreakPoints object. FInitialEnabled:=XMLConfig.GetValue(Path+'InitialEnabled/Value',true); FEnabled:=FInitialEnabled; end; procedure TIDEBreakPointGroup.SaveToXMLConfig(XMLConfig: TXMLConfig; const Path: string); begin XMLConfig.SetDeleteValue(Path+'Name/Value',Name,''); // the breakpoints of this group are not saved here. // They are saved by the TIDEBreakPoints object. XMLConfig.SetDeleteValue(Path+'InitialEnabled/Value',FInitialEnabled,true); end; class function TIDEBreakPointGroup.CheckName(const AName: String): Boolean; var i: Integer; begin for i := 1 to Length(AName) do if not (AName[i] in ['A'..'Z', 'a'..'z', '0'..'9', '_']) then Exit(False); Result := True; end; procedure TIDEBreakPointGroup.RemoveReference(const ABreakPointList: TIDEBreakPointGroupList); begin FReferences.Remove(ABreakPointList); end; procedure TIDEBreakPointGroup.SetEnabled(const AValue: Boolean); var n: Integer; begin for n := 0 to FBreakPoints.Count - 1 do TIDEBreakPoint(FBreakPoints[n]).Enabled := AValue; end; procedure TIDEBreakPointGroup.SetInitialEnabled(const AValue: Boolean); begin if FInitialEnabled=AValue then exit; FInitialEnabled:=AValue; end; procedure TIDEBreakPointGroup.SetName(const AValue: String); begin if FName = AValue then Exit; FName := AValue; Changed(False); end; procedure TIDEBreakPointGroup.AssignTo(Dest: TPersistent); var DestGroup: TIDEBreakPointGroup; begin if Dest is TIDEBreakPointGroup then begin DestGroup:=TIDEBreakPointGroup(Dest); DestGroup.Name:=Name; //DestGroup.InitialEnabled:=InitialEnabled; DestGroup.Enabled:=Enabled; end else inherited AssignTo(Dest); end; { =========================================================================== } { TIDEBreakPointGroups } { =========================================================================== } constructor TIDEBreakPointGroups.Create; begin inherited Create(TIDEBreakPointGroup); end; procedure TIDEBreakPointGroups.LoadFromXMLConfig(XMLConfig: TXMLConfig; const Path: string); var NewCount: integer; NewGroup: TIDEBreakPointGroup; i: Integer; OldGroup: TIDEBreakPointGroup; begin Clear; NewCount := XMLConfig.GetValue(Path+'Count', 0); for i := 0 to NewCount - 1 do begin NewGroup := TIDEBreakPointGroup(inherited Add); NewGroup.LoadFromXMLConfig(XMLConfig, Path+'Item'+IntToStr(i+1)+'/'); OldGroup := FindGroupByName(NewGroup.Name, NewGroup); if OldGroup <> nil then NewGroup.Free; end; end; procedure TIDEBreakPointGroups.SaveToXMLConfig(XMLConfig: TXMLConfig; const Path: string); var Cnt: Integer; CurGroup: TIDEBreakPointGroup; i: Integer; begin Cnt:=Count; XMLConfig.SetDeleteValue(Path+'Count',Cnt,0); for i := 0 to Cnt - 1 do begin CurGroup := Items[i]; CurGroup.SaveToXMLConfig(XMLConfig, Path+'Item'+IntToStr(i+1)+'/'); end; end; function TIDEBreakPointGroups.GetGroupByName(const GroupName: string): TIDEBreakPointGroup; begin Result := FindGroupByName(GroupName, nil); end; function TIDEBreakPointGroups.FindGroupByName(const GroupName: string; Ignore: TIDEBreakPointGroup): TIDEBreakPointGroup; var i: Integer; begin i := Count - 1; while i >= 0 do begin Result := Items[i]; if (AnsiCompareText(Result.Name, GroupName) = 0) and (Ignore <> Result) then Exit; Dec(i); end; Result := nil; end; function TIDEBreakPointGroups.IndexOfGroupWithName(const GroupName: string; Ignore : TIDEBreakPointGroup): integer; begin Result:=Count-1; while (Result>=0) and ((AnsiCompareText(Items[Result].Name,GroupName)<>0) or (Items[Result]=Ignore)) do dec(Result); end; procedure TIDEBreakPointGroups.InitTargetStart; var i: Integer; begin for i:=0 to Count-1 do Items[i].Enabled:=Items[i].fInitialEnabled; end; function TIDEBreakPointGroups.GetItem(const AnIndex: Integer ): TIDEBreakPointGroup; begin Result := TIDEBreakPointGroup(inherited GetItem(AnIndex)); end; procedure TIDEBreakPointGroups.SetItem(const AnIndex: Integer; const AValue: TIDEBreakPointGroup); begin inherited SetItem(AnIndex, AValue); end; (******************************************************************************) (******************************************************************************) (** **) (** W A T C H E S **) (** **) (******************************************************************************) (******************************************************************************) { =========================================================================== } { TIdeWatch } { =========================================================================== } function TIdeWatch.CreateValueList: TWatchValueList; begin Result := TIdeWatchValueList.Create(Self); end; constructor TIdeWatch.Create(ACollection: TCollection); begin assert(((Self is TCurrentWatch) and (ACollection is TCurrentWatches)) or ((not(Self is TCurrentWatch)) and not(ACollection is TCurrentWatches)), 'TIdewatch.Create: Watch and collection differ (current and none current)'); inherited Create(ACollection); end; procedure TIdeWatch.ClearValues; begin inherited ClearValues; TCurrentWatches(Collection).Update(Self); end; procedure TIdeWatch.DoEnableChange; begin Changed; DoModified; end; procedure TIdeWatch.DoExpressionChange; begin Changed; DoModified; end; procedure TIdeWatch.DoDisplayFormatChanged; begin Changed; DoModified; end; function TIdeWatch.GetValue(const AThreadId: Integer; const AStackFrame: Integer): TIdeWatchValue; begin Result := TIdeWatchValue(inherited Values[AThreadId, AStackFrame]); end; procedure TIdeWatch.LoadDataFromXMLConfig(const AConfig: TXMLConfig; const APath: string); begin FEnabled := AConfig.GetValue(APath + 'Enabled', True); FExpression := AConfig.GetValue(APath + 'Expression', ''); if AConfig.GetValue(APath + 'ClassAutoCast', False) then Include(FEvaluateFlags, defClassAutoCast) else Exclude(FEvaluateFlags, defClassAutoCast); try ReadStr(AConfig.GetValue(APath + 'DisplayFormat', 'wdfDefault'), FDisplayFormat); except FDisplayFormat := wdfDefault; end; FRepeatCount := AConfig.GetValue(APath + 'RepeatCount', 0); TIdeWatchValueList(FValueList).LoadDataFromXMLConfig(AConfig, APath + 'ValueList/'); end; procedure TIdeWatch.SaveDataToXMLConfig(const AConfig: TXMLConfig; const APath: string); var s: String; begin AConfig.SetDeleteValue(APath + 'Enabled', FEnabled, True); AConfig.SetDeleteValue(APath + 'Expression', FExpression, ''); WriteStr(s{%H-}, FDisplayFormat); AConfig.SetDeleteValue(APath + 'DisplayFormat', s, 'wdfDefault'); AConfig.SetDeleteValue(APath + 'ClassAutoCast', defClassAutoCast in FEvaluateFlags, False); AConfig.SetDeleteValue(APath + 'RepeatCount', FRepeatCount, 0); TIdeWatchValueList(FValueList).SaveDataToXMLConfig(AConfig, APath + 'ValueList/'); end; { =========================================================================== } { TCurrentWatch } { =========================================================================== } procedure TCurrentWatch.SetSnapShot(const AValue: TIdeWatch); begin assert((FSnapShot=nil) or (AValue=nil), 'TCurrentWatch already have snapshot'); if FSnapShot = AValue then exit; FSnapShot := AValue; if FSnapShot = nil then begin TCurrentWatchValueList(FValueList).SnapShot := nil; end else begin // TODO: FValueList is copied twice ? FSnapShot.Assign(self); FSnapShot.Enabled := True; // Snapshots are always enabled TCurrentWatchValueList(FValueList).SnapShot := TIdeWatchValueList(FSnapShot.FValueList); end; end; function TCurrentWatch.CreateValueList: TWatchValueList; begin Result := TCurrentWatchValueList.Create(Self); end; procedure TCurrentWatch.DoChanged; begin inherited DoChanged; if Collection <> nil then TCurrentWatches(Collection).Update(Self); end; procedure TCurrentWatch.DoModified; begin inherited DoModified; TCurrentWatches(Collection).DoModified; end; procedure TCurrentWatch.RequestData(AWatchValue: TCurrentWatchValue); begin if Collection <> nil then TCurrentWatches(Collection).RequestData(AWatchValue) else AWatchValue.Validity := ddsInvalid; end; destructor TCurrentWatch.Destroy; begin if (TCurrentWatches(Collection) <> nil) then begin TCurrentWatches(Collection).NotifyRemove(Self); TCurrentWatches(Collection).DoModified; end; inherited Destroy; end; procedure TCurrentWatch.LoadFromXMLConfig(const AConfig: TXMLConfig; const APath: string); var i: Integer; begin Expression := AConfig.GetValue(APath + 'Expression/Value', ''); Enabled := AConfig.GetValue(APath + 'Enabled/Value', true); if AConfig.GetValue(APath + 'ClassAutoCast', False) then Include(FEvaluateFlags, defClassAutoCast) else Exclude(FEvaluateFlags, defClassAutoCast); i := StringCase (AConfig.GetValue(APath + 'DisplayStyle/Value', TWatchDisplayFormatNames[wdfDefault]), TWatchDisplayFormatNames); if i >= 0 then DisplayFormat := TWatchDisplayFormat(i) else DisplayFormat := wdfDefault; FRepeatCount := AConfig.GetValue(APath + 'RepeatCount', 0); end; procedure TCurrentWatch.SaveToXMLConfig(const AConfig: TXMLConfig; const APath: string); begin AConfig.SetDeleteValue(APath + 'Expression/Value', Expression, ''); AConfig.SetDeleteValue(APath + 'Enabled/Value', Enabled, true); AConfig.SetDeleteValue(APath + 'DisplayStyle/Value', TWatchDisplayFormatNames[DisplayFormat], TWatchDisplayFormatNames[wdfDefault]); AConfig.SetDeleteValue(APath + 'ClassAutoCast', defClassAutoCast in FEvaluateFlags, False); AConfig.SetDeleteValue(APath + 'RepeatCount', FRepeatCount, 0); end; { =========================================================================== } { TIdeWatches } { =========================================================================== } function TIdeWatches.Add(const AExpression: String): TIdeWatch; begin BeginUpdate; Result := TIdeWatch(inherited Add); Result.Expression := AExpression; EndUpdate; end; function TIdeWatches.GetItem(const AnIndex: Integer): TIdeWatch; begin Result := TIdeWatch(inherited Items[AnIndex]); end; procedure TIdeWatches.SetItem(const AnIndex: Integer; const AValue: TIdeWatch); begin inherited Items[AnIndex] := AValue; end; function TIdeWatches.WatchClass: TWatchClass; begin Result := TIdeWatch; end; procedure TIdeWatches.LoadDataFromXMLConfig(const AConfig: TXMLConfig; APath: string); var c, i: Integer; begin Clear; c := AConfig.GetValue(APath + 'Count', 0); APath := APath + 'Entry'; for i := 0 to c - 1 do Add('').LoadDataFromXMLConfig(AConfig, APath + IntToStr(i) + '/'); end; procedure TIdeWatches.SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string); var i: Integer; begin AConfig.SetDeleteValue(APath + 'Count', Count, 0); APath := APath + 'Entry'; for i := 0 to Count - 1 do Items[i].SaveDataToXMLConfig(AConfig, APath + IntToStr(i) + '/'); end; function TIdeWatches.Find(const AExpression: String): TIdeWatch; begin Result := TIdeWatch(inherited Find(AExpression)); end; { =========================================================================== } { TCurrentWatches } { =========================================================================== } function TCurrentWatches.Add(const AExpression: String): TCurrentWatch; var R: TIdeWatch; begin // if this is modified, then also update LoadFromXMLConfig Result := TCurrentWatch(inherited Add(AExpression)); if FSnapShot <> nil then begin R := FSnapShot.Add(AExpression); Result.SnapShot := R; end; NotifyAdd(Result); DoModified; end; constructor TCurrentWatches.Create(AMonitor: TIdeWatchesMonitor); begin FDestroying := False; FMonitor := AMonitor; inherited Create; end; destructor TCurrentWatches.Destroy; begin FDestroying := True; inherited Destroy; end; function TCurrentWatches.Find(const AExpression: String): TCurrentWatch; begin Result := TCurrentWatch(inherited Find(AExpression)); end; procedure TCurrentWatches.WatchesChanged(Sender: TObject); begin Changed; end; procedure TCurrentWatches.SetSnapShot(const AValue: TIdeWatches); var R: TIdeWatch; i: Integer; begin assert((FSnapShot=nil) or (AValue=nil), 'TCurrentWatches already have snapshot'); if FSnapShot = AValue then exit; FSnapShot := AValue; if FSnapShot = nil then begin for i := 0 to Count - 1 do Items[i].SnapShot := nil; end else begin // FSnapShot.Assign(Self); FSnapShot.Clear; for i := 0 to Count - 1 do begin R := FSnapShot.Add(''); R.Assign(Items[i]); Items[i].SnapShot := R; end; end; end; function TCurrentWatches.GetItem(const AnIndex: Integer): TCurrentWatch; begin Result := TCurrentWatch(inherited GetItem(AnIndex)); end; procedure TCurrentWatches.LoadFromXMLConfig(const AConfig: TXMLConfig; const APath: string); var NewCount: Integer; i: Integer; Watch: TCurrentWatch; begin if FMonitor <> nil then FMonitor.BeginIgnoreModified; try Clear; NewCount := AConfig.GetValue(APath + 'Count', 0); for i := 0 to NewCount-1 do begin // Call inherited Add, so NotifyAdd can be send, after the Watch was loaded Watch := TCurrentWatch(inherited Add('')); Watch.LoadFromXMLConfig(AConfig, Format('%sItem%d/', [APath, i + 1])); NotifyAdd(Watch); end; finally if FMonitor <> nil then FMonitor.EndIgnoreModified; end; end; procedure TCurrentWatches.NotifyAdd(const AWatch: TCurrentWatch); begin FMonitor.NotifyAdd(Self, AWatch); end; procedure TCurrentWatches.NotifyRemove(const AWatch: TCurrentWatch); begin FMonitor.NotifyRemove(Self, AWatch); end; procedure TCurrentWatches.DoModified; begin if (FMonitor <> nil) and (not FDestroying) then FMonitor.DoModified; end; procedure TCurrentWatches.SaveToXMLConfig(const AConfig: TXMLConfig; const APath: string); var Cnt: Integer; i: Integer; Watch: TCurrentWatch; begin Cnt := Count; AConfig.SetDeleteValue(APath + 'Count', Cnt, 0); for i := 0 to Cnt - 1 do begin Watch := Items[i]; Watch.SaveToXMLConfig(AConfig, Format('%sItem%d/', [APath, i + 1])); end; end; procedure TCurrentWatches.SetItem(const AnIndex: Integer; const AValue: TCurrentWatch); begin inherited SetItem(AnIndex, AValue); end; function TCurrentWatches.WatchClass: TWatchClass; begin Result := TCurrentWatch; end; procedure TCurrentWatches.Update(Item: TCollectionItem); var m, c: Integer; begin if Item <> nil then begin FMonitor.NotifyUpdate(Self, TCurrentWatch(Item)); end else begin m := 0; c := Count; while m < c do begin FMonitor.NotifyUpdate(Self, Items[m]); if c <> Count then begin m := Max(0, m - Max(0, Count - c)); c := Count; end; inc(m); end; end; end; procedure TCurrentWatches.RequestData(AWatchValue: TCurrentWatchValue); begin FMonitor.RequestData(AWatchValue); end; (******************************************************************************) (******************************************************************************) (** **) (** L O C A L S **) (** **) (******************************************************************************) (******************************************************************************) { =========================================================================== } { TLocals } { =========================================================================== } procedure TIDELocals.SetDataValidity(AValidity: TDebuggerDataState); begin assert(Self is TCurrentLocals, 'TLocals.SetDataValidity'); end; procedure TIDELocals.LoadDataFromXMLConfig(const AConfig: TXMLConfig; APath: string); var c, i: Integer; begin c := AConfig.GetValue(APath + 'Count', 0); APath := APath + 'Entry'; for i := 0 to c - 1 do begin Add(AConfig.GetValue(APath + IntToStr(i) + '/Expression', ''), AConfig.GetValue(APath + IntToStr(i) + '/Value', '')); end; end; procedure TIDELocals.SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string); var i: Integer; begin AConfig.SetValue(APath + 'ThreadId', ThreadId); AConfig.SetValue(APath + 'StackFrame', StackFrame); AConfig.SetDeleteValue(APath + 'Count', Count, 0); APath := APath + 'Entry'; for i := 0 to Count - 1 do begin AConfig.SetValue(APath + IntToStr(i) + '/Expression', Names[i]); AConfig.SetValue(APath + IntToStr(i) + '/Value', Values[i]); end; end; constructor TIDELocals.CreateFromXMLConfig(const AConfig: TXMLConfig; APath: string); var LoadThreadId, LoadStackFrame: Integer; begin LoadThreadId := AConfig.GetValue(APath + 'ThreadId', -1); LoadStackFrame := AConfig.GetValue(APath + 'StackFrame', -1); Create(LoadThreadId, LoadStackFrame); LoadDataFromXMLConfig(AConfig, APath); end; { =========================================================================== } { TCurrentLocals } { =========================================================================== } procedure TCurrentLocals.SetSnapShot(const AValue: TIDELocals); begin assert((FSnapShot=nil) or (AValue=nil), 'TCurrentLocals already have snapshot'); if FSnapShot = AValue then exit; FSnapShot := AValue; if FSnapShot <> nil then FSnapShot.Assign(Self); end; constructor TCurrentLocals.Create(AMonitor: TIdeLocalsMonitor; AThreadId, AStackFrame: Integer); begin FMonitor := AMonitor; FDataValidity := ddsUnknown; inherited Create(AThreadId, AStackFrame); end; function TCurrentLocals.Count: Integer; begin case FDataValidity of ddsUnknown: begin AddReference; try Result := 0; FDataValidity := ddsRequested; FMonitor.RequestData(Self); // Locals can be cleared, if debugger is "run" again if FDataValidity = ddsValid then Result := inherited Count(); finally ReleaseReference; end; end; ddsRequested, ddsEvaluating: Result := 0; ddsValid: Result := inherited Count; ddsInvalid, ddsError: Result := 0; end; end; procedure TCurrentLocals.SetDataValidity(AValidity: TDebuggerDataState); begin if FDataValidity = AValidity then exit; if (FDataValidity in [ddsUnknown, ddsEvaluating, ddsRequested]) and (FSnapShot <> nil) then FSnapShot.Assign(Self); FDataValidity := AValidity; FMonitor.NotifyChange(Self); end; (******************************************************************************) (******************************************************************************) (** **) (** R E G I S T E R S **) (** **) (******************************************************************************) (******************************************************************************) { TIDERegisterValue } procedure TIDERegisterValue.DoDataValidityChanged(AnOldValidity: TDebuggerDataState); begin if (Owner <> nil) and (Owner is TCurrentIDERegisters) then TCurrentIDERegisters(Owner).DoDataValidityChanged(AnOldValidity); end; procedure TIDERegisterValue.DoDisplayFormatChanged(AnOldFormat: TRegisterDisplayFormat); begin if not HasValueFormat[DisplayFormat] then begin DataValidity := ddsRequested; if (Owner <> nil) and (Owner is TCurrentIDERegisters) then TCurrentIDERegisters(Owner).FMonitor.RequestData(TCurrentIDERegisters(Owner)); end else if (Owner <> nil) and (Owner is TCurrentIDERegisters) then TCurrentIDERegisters(Owner).FMonitor.NotifyChange(TCurrentIDERegisters(Owner)); end; { TIDERegisters } function TIDERegisters.CreateEntry: TDbgEntityValue; begin Result := TIDERegisterValue.Create; end; { TCurrentIDERegisters } procedure TCurrentIDERegisters.DoDataValidityChanged(AnOldValidity: TDebuggerDataState); begin inherited DoDataValidityChanged(AnOldValidity); if not( (DataValidity in [ddsRequested, ddsEvaluating]) and (AnOldValidity in [ddsUnknown, ddsRequested, ddsEvaluating]) ) then FMonitor.NotifyChange(Self); end; constructor TCurrentIDERegisters.Create(AMonitor: TIdeRegistersMonitor; AThreadId, AStackFrame: Integer); begin FMonitor := AMonitor; inherited Create(AThreadId, AStackFrame); end; function TCurrentIDERegisters.Count: Integer; begin case DataValidity of ddsUnknown: begin AddReference; try Result := 0; DataValidity := ddsRequested; FMonitor.RequestData(Self); // Locals can be cleared, if debugger is "run" again if DataValidity = ddsValid then Result := inherited Count(); finally ReleaseReference; end; end; ddsRequested, ddsEvaluating: Result := 0; ddsValid: Result := inherited Count; ddsInvalid, ddsError: Result := 0; end; end; { TCurrentIDERegistersList } procedure TCurrentIDERegistersList.DoCleared; begin inherited DoCleared; FMonitor.NotifyChange(nil); end; function TCurrentIDERegistersList.CreateEntry(AThreadId, AStackFrame: Integer): TRegisters; begin Result := TCurrentIDERegisters.Create(FMonitor, AThreadId, AStackFrame); end; constructor TCurrentIDERegistersList.Create(AMonitor: TIdeRegistersMonitor); begin FMonitor := AMonitor; inherited Create; end; { TIdeRegistersMonitor } function TIdeRegistersMonitor.GetCurrentRegistersList: TCurrentIDERegistersList; begin Result := TCurrentIDERegistersList(RegistersList); end; procedure TIdeRegistersMonitor.DoStateEnterPause; begin inherited DoStateEnterPause; if CurrentRegistersList = nil then exit; Clear; end; procedure TIdeRegistersMonitor.DoStateLeavePauseClean; begin inherited DoStateLeavePauseClean; if CurrentRegistersList = nil then exit; Clear; end; procedure TIdeRegistersMonitor.DoEndUpdate; begin inherited DoEndUpdate; if rmNeedNotifyChange in FFlags then NotifyChange(nil); end; procedure TIdeRegistersMonitor.NotifyChange(ARegisters: TCurrentIDERegisters); begin if IsUpdating then begin Include(FFlags, rmNeedNotifyChange); exit; end; Exclude(FFlags, rmNeedNotifyChange); FNotificationList.NotifyChange(ARegisters); end; procedure TIdeRegistersMonitor.DoNewSupplier; begin inherited DoNewSupplier; NotifyChange(nil); end; procedure TIdeRegistersMonitor.RequestData(ARegisters: TCurrentIDERegisters); begin if Supplier <> nil then Supplier.RequestData(ARegisters) else ARegisters.DataValidity := ddsInvalid; end; function TIdeRegistersMonitor.CreateRegistersList: TRegistersList; begin Result := TCurrentIDERegistersList.Create(Self); end; constructor TIdeRegistersMonitor.Create; begin inherited Create; FNotificationList := TDebuggerChangeNotificationList.Create; end; destructor TIdeRegistersMonitor.Destroy; begin FNotificationList.Clear; inherited Destroy; FreeAndNil(FNotificationList); end; procedure TIdeRegistersMonitor.Clear; begin CurrentRegistersList.Clear; end; procedure TIdeRegistersMonitor.AddNotification(const ANotification: TRegistersNotification); begin FNotificationList.Add(ANotification); end; procedure TIdeRegistersMonitor.RemoveNotification(const ANotification: TRegistersNotification); begin FNotificationList.Remove(ANotification); end; (******************************************************************************) (******************************************************************************) (** **) (** C A L L S T A C K **) (** **) (******************************************************************************) (******************************************************************************) { =========================================================================== } { TDBGCallStackEntry } { =========================================================================== } constructor TIdeCallStackEntry.Create(const AIndex: Integer; const AnAdress: TDbgPtr; const AnArguments: TStrings; const AFunctionName: String; const AUnitInfo: TDebuggerUnitInfo; const ALine: Integer; AState: TDebuggerDataState = ddsValid); begin inherited Create; SetUnitInfo(AUnitInfo); InitFields(AIndex, AnAdress, AnArguments, AFunctionName, ALine, AState); end; function TIdeCallStackEntry.CreateCopy: TCallStackEntry; begin Result := TIdeCallStackEntry.Create; Result.Assign(Self); end; procedure TIdeCallStackEntry.Assign(AnOther: TCallStackEntry); begin FUnitInfo.ReleaseReference; inherited Assign(AnOther); if AnOther is TIdeCallStackEntry then begin FUnitInfo := TIdeCallStackEntry(AnOther).FUnitInfo; if FUnitInfo <> nil then FUnitInfo.AddReference; end; end; destructor TIdeCallStackEntry.Destroy; begin inherited; if FUnitInfo <> nil then FUnitInfo.ReleaseReference; end; procedure TIdeCallStackEntry.Init(const AnAdress: TDbgPtr; const AnArguments: TStrings; const AFunctionName: String; const AUnitName, AClassName, AProcName, AFunctionArgs: String; const ALine: Integer; AState: TDebuggerDataState); var loc: TDebuggerUnitInfo; begin assert((FOwner = nil) or (FOwner is TCurrentCallStack), 'FOwner is TCurrentCallStack'); inherited Init(AnAdress, AnArguments, AFunctionName, AUnitName, AClassName, AProcName, AFunctionArgs, ALine, AState); if GetUnitInfoProvider = nil then loc := nil else loc := GetUnitInfoProvider.GetUnitInfoByFunction(AUnitName, AClassName, AProcName, AFunctionArgs); SetUnitInfo(loc); end; procedure TIdeCallStackEntry.Init(const AnAdress: TDbgPtr; const AnArguments: TStrings; const AFunctionName: String; const FileName, FullName: String; const ALine: Integer; AState: TDebuggerDataState); var loc: TDebuggerUnitInfo; begin assert((FOwner = nil) or (FOwner is TCurrentCallStack), 'FOwner is TCurrentCallStack'); inherited Init(AnAdress, AnArguments, AFunctionName, FileName, FullName, ALine, AState); if GetUnitInfoProvider = nil then loc := nil else loc := GetUnitInfoProvider.GetUnitInfoFor(FileName, FullName); SetUnitInfo(loc); end; function TIdeCallStackEntry.IsCurrent: Boolean; begin Result := (FOwner <> nil) and (FOwner.CurrentIndex = Self.Index); //TODO: check current thread end; procedure TIdeCallStackEntry.MakeCurrent; begin if FOwner = nil then Exit; if IsCurrent then exit; FOwner.ChangeCurrentIndex(Self.Index); end; function TIdeCallStackEntry.GetFunctionName: String; begin case Validity of ddsValid: Result := inherited GetFunctionName; ddsError: Result := ''; ddsInvalid: Result := ''; ddsRequested, ddsEvaluating: Result := ''; ddsUnknown: Result := ''; end; end; function TIdeCallStackEntry.GetSource: String; begin if (Validity = ddsValid) and (FUnitInfo <> nil) then Result := FUnitInfo.FileName else Result := ''; end; procedure TIdeCallStackEntry.SetUnitInfo(AUnitInfo: TDebuggerUnitInfo); begin if FUnitInfo <> nil then FUnitInfo.ReleaseReference; FUnitInfo := AUnitInfo; if FUnitInfo <> nil then FUnitInfo.AddReference; end; function TIdeCallStackEntry.GetUnitInfoProvider: TDebuggerUnitInfoProvider; begin Result := (FOwner as TCurrentCallStack).FMonitor.UnitInfoProvider; end; procedure TIdeCallStackEntry.LoadDataFromXMLConfig(const AConfig: TXMLConfig; const APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil); var UInfo: TDebuggerUnitInfo; i: Integer; AState: TDebuggerDataState; NewIndex, NewLine: Integer; NewAddress: QWord; NewFunctionName: String; begin NewIndex := AConfig.GetValue(APath + 'Index', 0); NewAddress := StrToQWordDef(AConfig.GetValue(APath + 'Address', '0'), 0); NewFunctionName := AConfig.GetValue(APath + 'FunctionName', ''); NewLine := AConfig.GetValue(APath + 'Line', 0); InitFields(NewIndex, NewAddress, nil, NewFunctionName, NewLine, ddsUnknown); Arguments.Text := AConfig.GetValue(APath + 'Arguments', ''); i := AConfig.GetValue(APath + 'UnitInfoRef', -1); UInfo := nil; if (i >= 0) and (AUnitInvoPrv <> nil) then begin if i < AUnitInvoPrv.Count then UInfo := AUnitInvoPrv[i]; end else begin UInfo := TDebuggerUnitInfo.Create('',''); UInfo.LoadDataFromXMLConfig(AConfig, APath + 'UnitInfo/'); end; SetUnitInfo(UInfo); try ReadStr(AConfig.GetValue(APath + 'State', 'ddsUnknown'), AState); Validity := AState; except Validity := ddsUnknown; end; end; procedure TIdeCallStackEntry.SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil); var s: string; i: Integer; begin AConfig.SetValue(APath + 'Index', Index); AConfig.SetValue(APath + 'Address', IntToStr(Address)); AConfig.SetValue(APath + 'FunctionName', FunctionName); AConfig.SetValue(APath + 'Line', Line); AConfig.SetValue(APath + 'Arguments', Arguments.Text); if FUnitInfo <> nil then begin if AUnitInvoPrv <> nil then begin i := AUnitInvoPrv.IndexOf(FUnitInfo, True); AConfig.SetValue(APath + 'UnitInfoRef', i); end else FUnitInfo.SaveDataToXMLConfig(AConfig, APath + 'UnitInfo/'); end; WriteStr(s{%H-}, Validity); AConfig.SetValue(APath + 'State', s); end; procedure TIdeCallStackEntry.ClearLocation; begin inherited ClearLocation; SetUnitInfo(TDebuggerUnitInfo.Create('','')); end; { =========================================================================== } { TCallStack } { =========================================================================== } procedure TIdeCallStack.Clear; var i: Integer; begin for i := 0 to FList.Count - 1 do TObject(FList[i]).Free; FList.Clear; end; function TIdeCallStack.GetCount: Integer; begin Result := FList.Count; end; destructor TIdeCallStack.Destroy; begin Clear; inherited Destroy; FreeAndNil(FList); end; function TIdeCallStack.GetEntry(AIndex: Integer): TIdeCallStackEntry; begin if (AIndex < 0) or (AIndex >= CountLimited(AIndex+1)) then IndexError(Aindex); Result := TIdeCallStackEntry(FList[AIndex]); end; procedure TIdeCallStack.AddEntry(AnEntry: TIdeCallStackEntry); begin // must be added in correct order Flist.Add(AnEntry); AnEntry.FOwner := Self; end; procedure TIdeCallStack.AssignEntriesTo(AnOther: TIdeCallStack); var i: Integer; begin for i := 0 to FList.Count-1 do begin AnOther.AddEntry(TIdeCallStackEntry(FList[i]).CreateCopy as TIdeCallStackEntry); end; end; procedure TIdeCallStack.LoadDataFromXMLConfig(const AConfig: TXMLConfig; APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil); var c, i: Integer; e: TIdeCallStackEntry; begin Clear; FThreadId := AConfig.GetValue(APath + 'ThreadId', -1); FCurrent := AConfig.GetValue(APath + 'Current', -1); c := AConfig.GetValue(APath + 'Count', 0); APath := APath + 'Entry'; for i := 0 to c - 1 do begin e := TIdeCallStackEntry.Create(); e.FOwner := self; e.LoadDataFromXMLConfig(AConfig, APath + IntToStr(i) + '/', AUnitInvoPrv); FList.Add(e); end; end; procedure TIdeCallStack.SaveDataToXMLConfig(const AConfig: TXMLConfig; APath: string; AUnitInvoPrv: TDebuggerUnitInfoProvider = nil); var i: Integer; begin AConfig.SetValue(APath + 'ThreadId', FThreadId); AConfig.SetValue(APath + 'Current', FCurrent); AConfig.SetDeleteValue(APath + 'Count', FList.Count, 0); APath := APath + 'Entry'; for i := 0 to FList.Count - 1 do TIdeCallStackEntry(FList[i]).SaveDataToXMLConfig(AConfig, APath + IntToStr(i) + '/', AUnitInvoPrv); end; procedure TIdeCallStack.DoEntriesCreated; begin assert(False, 'TCallStack.DoEntriesCreated'); end; procedure TIdeCallStack.DoEntriesUpdated; begin assert(False, 'TCallStack.DoEntriesUpdated'); end; procedure TIdeCallStack.SetCountValidity(AValidity: TDebuggerDataState); begin assert(False, 'TCallStack.SetCountValidity'); end; procedure TIdeCallStack.SetHasAtLeastCountInfo(AValidity: TDebuggerDataState; AMinCount: Integer); begin assert(False, 'TCallStack.SetHasAtLeastCountInfo'); end; procedure TIdeCallStack.SetCurrentValidity(AValidity: TDebuggerDataState); begin assert(False, 'TCallStack.SetCurrentValidity'); end; function TIdeCallStack.IndexError(AIndex: Integer): TIdeCallStackEntry; begin Result:=nil; raise EInvalidOperation.CreateFmt('Index out of range (%d)', [AIndex]); end; function TIdeCallStack.GetEntryBase(AIndex: Integer): TCallStackEntry; begin Result := TCallStackEntry(GetEntry(AIndex)); end; procedure TIdeCallStack.PrepareRange(AIndex, ACount: Integer); begin end; procedure TIdeCallStack.ChangeCurrentIndex(ANewIndex: Integer); begin CurrentIndex := ANewIndex; end; function TIdeCallStack.HasAtLeastCount(ARequiredMinCount: Integer): TNullableBool; begin if ARequiredMinCount <= Count then Result := nbTrue else Result := nbFalse; end; function TIdeCallStack.CountLimited(ALimit: Integer): Integer; begin case HasAtLeastCount(ALimit) of nbUnknown: Result := 0; nbTrue: Result := ALimit; nbFalse: Result := Count; end; end; procedure TIdeCallStack.SetCount(ACount: Integer); begin // can not set count assert(False, 'TCallStack.SetCount should not be called') end; procedure TIdeCallStack.Assign(AnOther: TCallStackBase); begin Clear; inherited Assign(AnOther); TIdeCallStack(AnOther).AssignEntriesTo(Self); end; constructor TIdeCallStack.Create; begin FList := TList.Create; inherited; end; function TIdeCallStack.CreateCopy: TCallStackBase; begin Result := TIdeCallStack.Create; Result.Assign(Self); end; function TIdeCallStack.GetRawEntries: TMap; begin assert(False, 'TCallStack.GetRawEntries'); Result := nil; end; function TIdeCallStack.GetNewCurrentIndex: Integer; begin assert(False, 'TCallStack.GetNewCurrentIndex'); Result := inherited GetNewCurrentIndex; end; { =========================================================================== } { TIdeCallStackMonitor } { =========================================================================== } procedure TIdeCallStackMonitor.AddNotification(const ANotification: TCallStackNotification); begin FNotificationList.Add(ANotification); end; constructor TIdeCallStackMonitor.Create; begin FSnapshots := TDebuggerDataSnapShotList.Create; FNotificationList := TDebuggerChangeNotificationList.Create; inherited Create; end; destructor TIdeCallStackMonitor.Destroy; begin FSnapshots.Clear; FNotificationList.Clear; inherited; FreeAndNil(FNotificationList); FreeAndNil(FSnapshots); end; procedure TIdeCallStackMonitor.DoStateEnterPause; begin inherited DoStateEnterPause; if (CurrentCallStackList = nil) then Exit; CurrentCallStackList.Clear; DoModified; end; procedure TIdeCallStackMonitor.DoStateLeavePause; begin inherited DoStateLeavePause; if (CurrentCallStackList = nil) then Exit; CurrentCallStackList.SnapShot := nil; end; procedure TIdeCallStackMonitor.DoStateLeavePauseClean; begin inherited DoStateLeavePauseClean; if (CurrentCallStackList = nil) then Exit; CurrentCallStackList.SnapShot := nil; CurrentCallStackList.Clear; CallStackClear(Self); end; procedure TIdeCallStackMonitor.DoModified; begin NotifyChange; end; procedure TIdeCallStackMonitor.RequestCount(ACallstack: TIdeCallStack); begin if (Supplier <> nil) and (ACallstack is TCurrentCallStack) then Supplier.RequestCount(TCurrentCallStack(ACallstack)); end; procedure TIdeCallStackMonitor.RequestAtLeastCount(ACallstack: TIdeCallStack; ARequiredMinCount: Integer); begin if (Supplier <> nil) and (ACallstack is TCurrentCallStack) then Supplier.RequestAtLeastCount(TCurrentCallStack(ACallstack), ARequiredMinCount); end; procedure TIdeCallStackMonitor.RequestCurrent(ACallstack: TIdeCallStack); begin if (Supplier <> nil) and (ACallstack is TCurrentCallStack) then Supplier.RequestCurrent(TCurrentCallStack(ACallstack)); end; procedure TIdeCallStackMonitor.RequestEntries(ACallstack: TIdeCallStack); begin if (Supplier <> nil) and (ACallstack is TCurrentCallStack) then Supplier.RequestEntries(TCurrentCallStack(ACallstack)); end; procedure TIdeCallStackMonitor.UpdateCurrentIndex; begin DebugLn(DBG_DATA_MONITORS, ['DebugDataMonitor: TIdeCallStackMonitor.UpdateCurrentIndex']); if Supplier <> nil then Supplier.UpdateCurrentIndex; NotifyCurrent; end; procedure TIdeCallStackMonitor.DoNewSupplier; begin inherited DoNewSupplier; NotifyChange; end; procedure TIdeCallStackMonitor.CallStackClear(Sender: TObject); begin DebugLn(DBG_DATA_MONITORS, ['DebugDataMonitor: TIdeCallStackMonitor.CallStackClear']); // Don't clear, set it to 0 so there are no entries shown //SetCount(0); NotifyChange; end; function TIdeCallStackMonitor.GetCurrentCallStackList: TCurrentCallStackList; begin Result := TCurrentCallStackList(CallStackList); end; function TIdeCallStackMonitor.GetSnapshot(AnID: Pointer): TIdeCallStackList; begin Result := TIdeCallStackList(FSnapshots.SnapShot[AnID]); end; procedure TIdeCallStackMonitor.NotifyChange; begin FNotificationList.NotifyChange(Self); end; procedure TIdeCallStackMonitor.NotifyCurrent; begin FNotificationList.NotifyCurrent(Self); end; function TIdeCallStackMonitor.CreateSnapshot(CreateEmpty: Boolean = False): TObject; begin Result := TIdeCallStackList.Create; if not CreateEmpty then CurrentCallStackList.SnapShot := TIdeCallStackList(Result); end; function TIdeCallStackMonitor.CreateCallStackList: TCallStackList; begin Result := TCurrentCallStackList.Create(Self); end; procedure TIdeCallStackMonitor.RemoveNotification(const ANotification: TCallStackNotification); begin FNotificationList.Remove(ANotification); end; procedure TIdeCallStackMonitor.NewSnapshot(AnID: Pointer; CreateEmpty: Boolean); var S: TObject; begin S := CreateSnapshot(CreateEmpty); FSnapshots.AddSnapShot(AnID, S); end; procedure TIdeCallStackMonitor.RemoveSnapshot(AnID: Pointer); begin FSnapshots.RemoveSnapShot(AnID); end; (******************************************************************************) (******************************************************************************) (** **) (** S I G N A L S and E X C E P T I O N S **) (** **) (******************************************************************************) (******************************************************************************) { =========================================================================== } { TIDESignal } { =========================================================================== } procedure TIDESignal.AssignTo(Dest: TPersistent); begin inherited AssignTo(Dest); if (TIDESignals(Collection).FMaster <> nil) and (Dest is TDBGSignal) then begin FMaster := TDBGSignal(Dest); end; end; procedure TIDESignal.LoadFromXMLConfig (const AXMLConfig: TXMLConfig; const APath: string ); begin // TODO end; procedure TIDESignal.SaveToXMLConfig (const AXMLConfig: TXMLConfig; const APath: string ); begin // TODO end; procedure TIDESignal.ResetMaster; begin FMaster := nil; end; { =========================================================================== } { TIDESignals } { =========================================================================== } function TIDESignals.Add(const AName: String; AID: Integer): TIDESignal; begin Result := TIDESignal(inherited Add(AName, AID)); end; function TIDESignals.Find(const AName: String): TIDESignal; begin Result := TIDESignal(inherited Find(AName)); end; procedure TIDESignals.SetMaster(const AValue: TDBGSignals); var n: Integer; begin if FMaster = AValue then Exit; FMaster := AValue; if FMaster = nil then begin for n := 0 to Count - 1 do Items[n].ResetMaster; end else begin FMaster.Assign(Self); end; end; function TIDESignals.GetItem(const AIndex: Integer): TIDESignal; begin Result := TIDESignal(inherited GetItem(AIndex)); end; procedure TIDESignals.LoadFromXMLConfig(const AXMLConfig: TXMLConfig; const APath: string); begin // TODO end; procedure TIDESignals.SaveToXMLConfig(const AXMLConfig: TXMLConfig; const APath: string); begin // TODO end; procedure TIDESignals.SetItem(const AIndex: Integer; const AValue: TIDESignal); begin inherited SetItem(AIndex, AValue); end; procedure TIDESignals.AddDefault; begin // todo: add default signals end; constructor TIDESignals.Create; begin FMaster := nil; inherited Create(TIDESignal); AddDefault; end; procedure TIDESignals.Reset; begin inherited Reset; AddDefault; end; { =========================================================================== } { TIDEException } { =========================================================================== } constructor TIDEException.Create (ACollection: TCollection ); begin FEnabled := True; inherited Create(ACollection); end; procedure TIDEException.LoadFromXMLConfig(const AXMLConfig: TXMLConfig; const APath: string); begin FName:=AXMLConfig.GetValue(APath+'Name/Value',''); FEnabled:=AXMLConfig.GetValue(APath+'Enabled/Value',true); end; procedure TIDEException.SaveToXMLConfig(const AXMLConfig: TXMLConfig; const APath: string); begin AXMLConfig.SetDeleteValue(APath+'Name/Value',FName,''); AXMLConfig.SetDeleteValue(APath+'Enabled/Value',FEnabled,true); end; procedure TIDEException.ResetMaster; begin FMaster := nil; end; { =========================================================================== } { TIDEExceptions } { =========================================================================== } function TIDEExceptions.Add(const AName: String): TIDEException; begin Result := TIDEException(inherited Add(AName)); end; function TIDEExceptions.Find(const AName: String): TIDEException; begin Result := TIDEException(inherited Find(AName)); end; constructor TIDEExceptions.Create; begin inherited Create(TIDEException); AddDefault; end; function TIDEExceptions.GetItem(const AIndex: Integer): TIDEException; begin Result := TIDEException(inherited GetItem(AIndex)); end; procedure TIDEExceptions.LoadFromXMLConfig (const AXMLConfig: TXMLConfig; const APath: string); var NewCount: Integer; i: Integer; IDEException: TIDEException; begin Clear; NewCount := AXMLConfig.GetValue(APath + 'Count', 0); FIgnoreAll := AXMLConfig.GetValue(APath + 'IgnoreAll', False); for i := 0 to NewCount-1 do begin IDEException := TIDEException(inherited Add('')); IDEException.LoadFromXMLConfig(AXMLConfig, Format('%sItem%d/', [APath, i + 1])); end; end; procedure TIDEExceptions.SaveToXMLConfig (const AXMLConfig: TXMLConfig; const APath: string); var Cnt: Integer; i: Integer; IDEException: TIDEException; begin Cnt := Count; AXMLConfig.SetDeleteValue(APath + 'Count', Cnt, 0); AXMLConfig.SetDeleteValue(APath + 'IgnoreAll', IgnoreAll, False); for i := 0 to Cnt - 1 do begin IDEException := Items[i]; IDEException.SaveToXMLConfig(AXMLConfig, Format('%sItem%d/', [APath, i + 1])); end; end; procedure TIDEExceptions.AddIfNeeded(AName: string); begin if Find(AName) = nil then Add(AName); end; procedure TIDEExceptions.Reset; begin inherited Reset; AddDefault; end; procedure TIDEExceptions.SetItem(const AIndex: Integer; const AValue: TIDEException); begin inherited SetItem(Aindex, AValue); end; procedure TIDEExceptions.AddDefault; begin AddIfNeeded('EAbort'); AddIfNeeded('ECodetoolError'); AddIfNeeded('EFOpenError'); end; { TIDELineInfo } procedure TIDELineInfo.LineInfoChanged(const ASender: TObject; const ASource: String); begin NotifyChange(ASource); end; procedure TIDELineInfo.SetMaster(const AMaster: TDBGLineInfo); begin if FMaster = AMaster then Exit; if FMaster <> nil then begin FMaster.OnChange := nil; end; FMaster := AMaster; if FMaster <> nil then begin FMaster.OnChange := @LineInfoChanged; end; end; function TIDELineInfo.GetSource(const AIndex: Integer): String; begin if Master = nil then Result := inherited GetSource(AIndex) else Result := Master.Sources[AIndex]; end; procedure TIDELineInfo.NotifyChange(ASource: String); var n: Integer; Notification: TIDELineInfoNotification; begin for n := 0 to FNotificationList.Count - 1 do begin Notification := TIDELineInfoNotification(FNotificationList[n]); if Assigned(Notification.FOnChange) then Notification.FOnChange(Self, ASource); end; end; constructor TIDELineInfo.Create; begin FNotificationList := TList.Create; inherited Create; end; destructor TIDELineInfo.Destroy; var n: Integer; begin for n := FNotificationList.Count - 1 downto 0 do TDebuggerNotification(FNotificationList[n]).ReleaseReference; inherited; FreeAndNil(FNotificationList); end; procedure TIDELineInfo.AddNotification(const ANotification: TIDELineInfoNotification); begin FNotificationList.Add(ANotification); ANotification.AddReference; end; procedure TIDELineInfo.RemoveNotification(const ANotification: TIDELineInfoNotification); begin if FNotificationList.IndexOf(ANotification) >= 0 then begin FNotificationList.Remove(ANotification); ANotification.ReleaseReference; end; end; function TIDELineInfo.Count: Integer; begin if Master = nil then Result := inherited Count else Result := Master.Count; end; function TIDELineInfo.GetAddress(const AIndex: Integer; const ALine: Integer): TDbgPtr; begin if Master = nil then Result := inherited GetAddress(AIndex, ALine) else Result := Master.GetAddress(AIndex, ALine); end; function TIDELineInfo.GetInfo(AAdress: TDbgPtr; out ASource, ALine, AOffset: Integer): Boolean; begin if Master = nil then Result := inherited GetInfo(AAdress, ASource, ALine, AOffset) else Result := Master.GetInfo(AAdress, ASource, ALine, AOffset); end; function TIDELineInfo.IndexOf(const ASource: String): integer; begin if Master = nil then Result := inherited IndexOf(ASource) else Result := Master.IndexOf(ASource); end; procedure TIDELineInfo.Request(const ASource: String); begin if Master = nil then inherited Request(ASource) else Master.Request(ASource); end; procedure TIDELineInfo.Cancel(const ASource: String); begin if Master = nil then inherited Cancel(ASource) else Master.Cancel(ASource); end; { TIDEDisassembler } procedure TIDEDisassembler.DisassemblerChanged(Sender: TObject); begin Changed; end; procedure TIDEDisassembler.SetMaster(AMaster: TDBGDisassembler); begin if FMaster = AMaster then Exit; if FMaster <> nil then FMaster.OnChange := nil; FMaster := AMaster; if FMaster <> nil then FMaster.OnChange := @DisassemblerChanged; Changed; end; procedure TIDEDisassembler.DoChanged; var n: Integer; Notification: TIDEDisassemblerNotification; begin if FMaster <> nil then begin SetCountBefore(FMaster.CountBefore); SetCountAfter(FMaster.CountAfter); SetBaseAddr(FMaster.BaseAddr); end else Clear; for n := 0 to FNotificationList.Count - 1 do begin Notification := TIDEDisassemblerNotification(FNotificationList[n]); if Assigned(Notification.FOnChange) then Notification.FOnChange(Self); end; end; function TIDEDisassembler.InternalGetEntry(AIndex: Integer): TDisassemblerEntry; begin if FMaster <> nil then Result := FMaster.Entries[AIndex] else Result := inherited InternalGetEntry(AIndex); end; function TIDEDisassembler.InternalGetEntryPtr(AIndex: Integer): PDisassemblerEntry; begin if FMaster <> nil then Result := FMaster.EntriesPtr[AIndex] else Result := inherited InternalGetEntryPtr(AIndex); end; constructor TIDEDisassembler.Create; begin FNotificationList := TList.Create; inherited Create; end; destructor TIDEDisassembler.Destroy; var n: Integer; begin if FMaster <> nil then FMaster.OnChange := nil; FMaster := nil; for n := FNotificationList.Count - 1 downto 0 do TDebuggerNotification(FNotificationList[n]).ReleaseReference; inherited; FreeAndNil(FNotificationList); end; procedure TIDEDisassembler.AddNotification(const ANotification: TIDEDisassemblerNotification); begin FNotificationList.Add(ANotification); ANotification.AddReference; end; procedure TIDEDisassembler.RemoveNotification(const ANotification: TIDEDisassemblerNotification); begin FNotificationList.Remove(ANotification); ANotification.ReleaseReference; end; procedure TIDEDisassembler.Clear; begin if FMaster <> nil then FMaster.Clear else inherited Clear; end; function TIDEDisassembler.PrepareRange(AnAddr: TDbgPtr; ALinesBefore, ALinesAfter: Integer): Boolean; begin if (AnAddr = BaseAddr) and (ALinesBefore < CountBefore) and (ALinesAfter < CountAfter) then exit(True); if FMaster <> nil then Result := FMaster.PrepareRange(AnAddr, ALinesBefore, ALinesAfter) else Result := inherited PrepareRange(AnAddr, ALinesBefore, ALinesAfter); end; initialization DBG_DATA_MONITORS := DebugLogger.FindOrRegisterLogGroup('DBG_DATA_MONITORS' {$IFDEF DBG_DATA_MONITORS} , True {$ENDIF} ); DBG_LOCATION_INFO := DebugLogger.FindOrRegisterLogGroup('DBG_LOCATION_INFO' {$IFDEF DBG_LOCATION_INFO} , True {$ENDIF} ); end.