mirror of
https://gitlab.com/freepascal.org/lazarus/lazarus.git
synced 2025-05-03 07:43:51 +02:00
202 lines
5.1 KiB
ObjectPascal
202 lines
5.1 KiB
ObjectPascal
{ $Id$ }
|
|
{ -------------------------------------------
|
|
dbgutils.pp - Debugger utility routines
|
|
-------------------------------------------
|
|
|
|
@created(Sun Apr 28st WET 2002)
|
|
@lastmod($Date$)
|
|
@author(Marc Weustink <marc@@dommelstein.net>)
|
|
|
|
This unit contains a collection of debugger support routines.
|
|
|
|
***************************************************************************
|
|
* *
|
|
* This source is free software; you can redistribute it and/or modify *
|
|
* it under the terms of the GNU General Public License as published by *
|
|
* the Free Software Foundation; either version 2 of the License, or *
|
|
* (at your option) any later version. *
|
|
* *
|
|
* This code is distributed in the hope that it will be useful, but *
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of *
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU *
|
|
* General Public License for more details. *
|
|
* *
|
|
* A copy of the GNU General Public License is available on the World *
|
|
* Wide Web at <http://www.gnu.org/copyleft/gpl.html>. You can also *
|
|
* obtain it by writing to the Free Software Foundation, *
|
|
* Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *
|
|
* *
|
|
***************************************************************************
|
|
}
|
|
unit DBGUtils;
|
|
|
|
{$mode objfpc}{$H+}
|
|
|
|
interface
|
|
|
|
uses
|
|
Classes, LCLProc;
|
|
|
|
type
|
|
TDelayedUdateItem = class(TCollectionItem)
|
|
private
|
|
FUpdateCount: Integer;
|
|
FDoChanged: Boolean;
|
|
protected
|
|
procedure Changed;
|
|
procedure DoChanged; virtual;
|
|
public
|
|
procedure Assign(ASource: TPersistent); override;
|
|
procedure BeginUpdate;
|
|
constructor Create(ACollection: TCollection); override;
|
|
procedure EndUpdate;
|
|
end;
|
|
|
|
function GetLine(var ABuffer: String): String;
|
|
function ConvertToCString(const AText: String): String;
|
|
function DeleteEscapeChars(const AText: String; const AEscapeChar: Char): String;
|
|
|
|
procedure SmartWriteln(const s: string);
|
|
|
|
implementation
|
|
|
|
uses
|
|
SysUtils;
|
|
|
|
{ SmartWriteln: }
|
|
var
|
|
LastSmartWritelnStr: string;
|
|
LastSmartWritelnCount: integer;
|
|
LastSmartWritelnTime: double;
|
|
|
|
procedure SmartWriteln(const s: string);
|
|
var
|
|
TimeDiff: TTimeStamp;
|
|
i: Integer;
|
|
begin
|
|
if (LastSmartWritelnCount>0) and (s=LastSmartWritelnStr) then begin
|
|
TimeDiff:=DateTimeToTimeStamp(Now-LastSmartWritelnTime);
|
|
if TimeDiff.Time<1000 then begin
|
|
// repeating too fast
|
|
inc(LastSmartWritelnCount);
|
|
// write every 2nd, 4th, 8th, 16th, ... time
|
|
i:=LastSmartWritelnCount;
|
|
while (i>0) and ((i and 1)=0) do begin
|
|
i:=i shr 1;
|
|
if i=1 then begin
|
|
DebugLn('Last message repeated %d times: "%s"',
|
|
[LastSmartWritelnCount, LastSmartWritelnStr]);
|
|
break;
|
|
end;
|
|
end;
|
|
exit;
|
|
end;
|
|
end;
|
|
LastSmartWritelnTime:=Now;
|
|
LastSmartWritelnStr:=s;
|
|
LastSmartWritelnCount:=1;
|
|
DebugLn(LastSmartWritelnStr);
|
|
end;
|
|
|
|
function GetLine(var ABuffer: String): String;
|
|
var
|
|
idx: Integer;
|
|
begin
|
|
idx := Pos(#10, ABuffer);
|
|
if idx = 0
|
|
then Result := ''
|
|
else begin
|
|
Result := Copy(ABuffer, 1, idx);
|
|
Delete(ABuffer, 1, idx);
|
|
end;
|
|
end;
|
|
|
|
function ConvertToCString(const AText: String): String;
|
|
var
|
|
n: Integer;
|
|
begin
|
|
Result := AText;
|
|
n := 1;
|
|
while n <= Length(Result) do
|
|
begin
|
|
case Result[n] of
|
|
'''': begin
|
|
if (n < Length(Result))
|
|
and (Result[n + 1] = '''')
|
|
then Delete(Result, n, 1)
|
|
else Result[n] := '"';
|
|
end;
|
|
'"': begin
|
|
Insert('"', Result, n);
|
|
Inc(n);
|
|
end;
|
|
end;
|
|
Inc(n);
|
|
end;
|
|
end;
|
|
|
|
function DeleteEscapeChars(const AText: String; const AEscapeChar: Char): String;
|
|
var
|
|
i: Integer;
|
|
begin
|
|
Result:=AText;
|
|
i:=1;
|
|
while i<length(Result) do begin
|
|
if Result[i]=AEscapeChar then
|
|
System.Delete(Result,i,1);
|
|
Inc(i);
|
|
end;
|
|
end;
|
|
|
|
{ TDelayedUdateItem }
|
|
|
|
procedure TDelayedUdateItem.Assign(ASource: TPersistent);
|
|
begin
|
|
BeginUpdate;
|
|
try
|
|
inherited Assign(ASource);
|
|
finally
|
|
EndUpdate;
|
|
end;
|
|
end;
|
|
|
|
procedure TDelayedUdateItem.BeginUpdate;
|
|
begin
|
|
Inc(FUpdateCount);
|
|
if FUpdateCount = 1 then FDoChanged := False;
|
|
end;
|
|
|
|
procedure TDelayedUdateItem.Changed;
|
|
begin
|
|
if FUpdateCount > 0
|
|
then FDoChanged := True
|
|
else DoChanged;
|
|
end;
|
|
|
|
constructor TDelayedUdateItem.Create(ACollection: TCollection);
|
|
begin
|
|
inherited Create(ACollection);
|
|
FUpdateCount := 0;
|
|
end;
|
|
|
|
procedure TDelayedUdateItem.DoChanged;
|
|
begin
|
|
inherited Changed(False);
|
|
end;
|
|
|
|
procedure TDelayedUdateItem.EndUpdate;
|
|
begin
|
|
Dec(FUpdateCount);
|
|
if FUpdateCount < 0 then raise EInvalidOperation.Create('TDelayedUdateItem.EndUpdate');
|
|
if (FUpdateCount = 0) and FDoChanged
|
|
then begin
|
|
DoChanged;
|
|
FDoChanged := False;
|
|
end;
|
|
end;
|
|
|
|
initialization
|
|
LastSmartWritelnCount:=0;
|
|
|
|
end.
|