mirror of https://gitlab.com/basile.b/dexed.git
gdb commander, allow to evaluate the editor expression when mouse motion stops
this does not work very well because of the missing dereference (*) when required so use of temporaries during debug might be advisable.
This commit is contained in:
parent
ac9d54c91a
commit
7a04995921
|
@ -23,9 +23,10 @@ type
|
||||||
procedure addBreakPoint(const fname: string; line: integer; kind: TBreakPointKind = bpkBreak);
|
procedure addBreakPoint(const fname: string; line: integer; kind: TBreakPointKind = bpkBreak);
|
||||||
procedure removeBreakPoint(const fname: string; line: integer; kind: TBreakPointKind = bpkBreak);
|
procedure removeBreakPoint(const fname: string; line: integer; kind: TBreakPointKind = bpkBreak);
|
||||||
procedure removeBreakPoints(const fname: string);
|
procedure removeBreakPoints(const fname: string);
|
||||||
|
function evaluate(const exp: string): string;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
// Enumerates th e reason why debuging breaks.
|
// Enumerates the reason why debuging breaks.
|
||||||
TDebugBreakReason = (
|
TDebugBreakReason = (
|
||||||
dbUnknown, // ?
|
dbUnknown, // ?
|
||||||
dbBreakPoint, // a break point is reached.
|
dbBreakPoint, // a break point is reached.
|
||||||
|
|
|
@ -153,6 +153,12 @@ procedure getImports(list: TLexTokenList; imports: TStrings);
|
||||||
*)
|
*)
|
||||||
function getIndexOfTokenLeftTo(tokens: TLexTokenList; caretPos: TPoint): integer;
|
function getIndexOfTokenLeftTo(tokens: TLexTokenList; caretPos: TPoint): integer;
|
||||||
|
|
||||||
|
(**
|
||||||
|
* Get the expression, as a string, that ends at the caret position.
|
||||||
|
* This helper is mostly used by GDB commander
|
||||||
|
*)
|
||||||
|
function getExpressionAt(tokens: TLexTokenList; caretPos: TPoint): string;
|
||||||
|
|
||||||
implementation
|
implementation
|
||||||
|
|
||||||
{$REGION TReaderHead -----------------------------------------------------------}
|
{$REGION TReaderHead -----------------------------------------------------------}
|
||||||
|
@ -1026,6 +1032,55 @@ begin
|
||||||
end;
|
end;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
function getExpressionAt(tokens: TLexTokenList; caretPos: TPoint): string;
|
||||||
|
var
|
||||||
|
ri: integer;
|
||||||
|
li: integer = -1;
|
||||||
|
p : integer = 1;
|
||||||
|
i : integer;
|
||||||
|
t : PLexToken;
|
||||||
|
begin
|
||||||
|
result := '';
|
||||||
|
ri := getIndexOfTokenLeftTo(tokens, caretPos);
|
||||||
|
if ri <> -1 then
|
||||||
|
for i := ri downto 0 do
|
||||||
|
begin
|
||||||
|
t := tokens[i];
|
||||||
|
// other; a.b.c|.d -> a.b.c
|
||||||
|
if (t^.kind = TLexTokenKind.ltkSymbol) and
|
||||||
|
((t^.Data = ';') or (t^.Data = ',') or (t^.Data = '{') or (t^.Data = ':') or (t^.Data = '?')) then
|
||||||
|
begin
|
||||||
|
li := i+1;
|
||||||
|
break;
|
||||||
|
end
|
||||||
|
// other; a.b c|.d -> c
|
||||||
|
else if (t^.kind = TLexTokenKind.ltkWhite) then
|
||||||
|
begin
|
||||||
|
li := i+1;
|
||||||
|
break;
|
||||||
|
end
|
||||||
|
// other; a + b.c|.d -> b.c
|
||||||
|
else if (t^.kind = TLexTokenKind.ltkOperator) then
|
||||||
|
begin
|
||||||
|
li := i+1;
|
||||||
|
break;
|
||||||
|
end;
|
||||||
|
p += Byte((t^.kind = TLexTokenKind.ltkSymbol) and (t^.Data = ')'));
|
||||||
|
p -= Byte((t^.kind = TLexTokenKind.ltkSymbol) and (t^.Data = '('));
|
||||||
|
// (a.(b).c|.d) -> a.(b).c
|
||||||
|
if p = 0 then
|
||||||
|
begin
|
||||||
|
li := i+1;
|
||||||
|
break;
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
if (li <> -1) and (li <> -1) then
|
||||||
|
begin
|
||||||
|
for i := li to ri do
|
||||||
|
result += tokens[i]^.Data;
|
||||||
|
end;
|
||||||
|
end;
|
||||||
|
|
||||||
function TLexErrorList.getError(index: integer): TLexError;
|
function TLexErrorList.getError(index: integer): TLexError;
|
||||||
begin
|
begin
|
||||||
Result := PLexError(Items[index])^;
|
Result := PLexError(Items[index])^;
|
||||||
|
|
|
@ -36,8 +36,8 @@ inherited GdbWidget: TGdbWidget
|
||||||
Width = 672
|
Width = 672
|
||||||
Align = alClient
|
Align = alClient
|
||||||
Caption = 'CPU'
|
Caption = 'CPU'
|
||||||
ClientHeight = 156
|
ClientHeight = 155
|
||||||
ClientWidth = 670
|
ClientWidth = 668
|
||||||
TabOrder = 0
|
TabOrder = 0
|
||||||
object cpuViewer: TTIPropertyGrid
|
object cpuViewer: TTIPropertyGrid
|
||||||
Left = 0
|
Left = 0
|
||||||
|
@ -77,8 +77,8 @@ inherited GdbWidget: TGdbWidget
|
||||||
OnChange = PageControl2Change
|
OnChange = PageControl2Change
|
||||||
object TabSheet3: TTabSheet
|
object TabSheet3: TTabSheet
|
||||||
Caption = 'Variables'
|
Caption = 'Variables'
|
||||||
ClientHeight = 169
|
ClientHeight = 173
|
||||||
ClientWidth = 662
|
ClientWidth = 670
|
||||||
object lstVariables: TListView
|
object lstVariables: TListView
|
||||||
Left = 2
|
Left = 2
|
||||||
Height = 130
|
Height = 130
|
||||||
|
@ -124,8 +124,8 @@ inherited GdbWidget: TGdbWidget
|
||||||
end
|
end
|
||||||
object TabSheet4: TTabSheet
|
object TabSheet4: TTabSheet
|
||||||
Caption = 'Assembler'
|
Caption = 'Assembler'
|
||||||
ClientHeight = 169
|
ClientHeight = 173
|
||||||
ClientWidth = 662
|
ClientWidth = 670
|
||||||
object lstAsm: TListView
|
object lstAsm: TListView
|
||||||
Left = 2
|
Left = 2
|
||||||
Height = 161
|
Height = 161
|
||||||
|
@ -214,8 +214,8 @@ inherited GdbWidget: TGdbWidget
|
||||||
TabOrder = 3
|
TabOrder = 3
|
||||||
object TabSheet1: TTabSheet
|
object TabSheet1: TTabSheet
|
||||||
Caption = 'Call stack'
|
Caption = 'Call stack'
|
||||||
ClientHeight = 169
|
ClientHeight = 173
|
||||||
ClientWidth = 662
|
ClientWidth = 670
|
||||||
object lstCallStack: TListView
|
object lstCallStack: TListView
|
||||||
Left = 0
|
Left = 0
|
||||||
Height = 169
|
Height = 169
|
||||||
|
@ -251,8 +251,8 @@ inherited GdbWidget: TGdbWidget
|
||||||
end
|
end
|
||||||
object TabSheet2: TTabSheet
|
object TabSheet2: TTabSheet
|
||||||
Caption = 'Thread list'
|
Caption = 'Thread list'
|
||||||
ClientHeight = 169
|
ClientHeight = 173
|
||||||
ClientWidth = 662
|
ClientWidth = 670
|
||||||
object lstThreads: TListView
|
object lstThreads: TListView
|
||||||
Left = 0
|
Left = 0
|
||||||
Height = 169
|
Height = 169
|
||||||
|
@ -307,8 +307,8 @@ inherited GdbWidget: TGdbWidget
|
||||||
end
|
end
|
||||||
object TabSheet5: TTabSheet
|
object TabSheet5: TTabSheet
|
||||||
Caption = 'Debugee options'
|
Caption = 'Debugee options'
|
||||||
ClientHeight = 169
|
ClientHeight = 173
|
||||||
ClientWidth = 662
|
ClientWidth = 670
|
||||||
object dbgeeOptsEd: TTIPropertyGrid
|
object dbgeeOptsEd: TTIPropertyGrid
|
||||||
Left = 2
|
Left = 2
|
||||||
Height = 161
|
Height = 161
|
||||||
|
@ -330,6 +330,7 @@ inherited GdbWidget: TGdbWidget
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
inherited toolbar: TDexedToolBar
|
inherited toolbar: TDexedToolBar
|
||||||
|
Height = 30
|
||||||
Width = 664
|
Width = 664
|
||||||
object btnStack: TDexedToolButton[0]
|
object btnStack: TDexedToolButton[0]
|
||||||
Left = 238
|
Left = 238
|
||||||
|
@ -353,7 +354,7 @@ inherited GdbWidget: TGdbWidget
|
||||||
end
|
end
|
||||||
object button4: TDexedToolButton[2]
|
object button4: TDexedToolButton[2]
|
||||||
Left = 205
|
Left = 205
|
||||||
Height = 28
|
Height = 5
|
||||||
Top = 0
|
Top = 0
|
||||||
AutoSize = True
|
AutoSize = True
|
||||||
Caption = 'button4'
|
Caption = 'button4'
|
||||||
|
|
|
@ -502,6 +502,8 @@ type
|
||||||
fDoc: TDexedMemo;
|
fDoc: TDexedMemo;
|
||||||
fDbgRunnable: boolean;
|
fDbgRunnable: boolean;
|
||||||
fCatchCustomEval: boolean;
|
fCatchCustomEval: boolean;
|
||||||
|
fCatchCustomEvalAsString: boolean;
|
||||||
|
fCaughtCustomEvalAstring: string;
|
||||||
fProj: ICommonProject;
|
fProj: ICommonProject;
|
||||||
fJson: TJsonObject;
|
fJson: TJsonObject;
|
||||||
fLog: TStringList;
|
fLog: TStringList;
|
||||||
|
@ -576,6 +578,7 @@ type
|
||||||
procedure removeBreakPoint(const fname: string; line: integer;
|
procedure removeBreakPoint(const fname: string; line: integer;
|
||||||
kind: TBreakPointKind = bpkBreak);
|
kind: TBreakPointKind = bpkBreak);
|
||||||
procedure removeBreakPoints(const fname: string);
|
procedure removeBreakPoints(const fname: string);
|
||||||
|
function evaluate(const exp: string): string;
|
||||||
procedure executeFromShortcut(sender: TObject);
|
procedure executeFromShortcut(sender: TObject);
|
||||||
public
|
public
|
||||||
constructor create(aOwner: TComponent); override;
|
constructor create(aOwner: TComponent); override;
|
||||||
|
@ -1686,6 +1689,20 @@ begin
|
||||||
updateButtonsState;
|
updateButtonsState;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
function TGdbWidget.evaluate(const exp: string): string;
|
||||||
|
begin
|
||||||
|
result := '';
|
||||||
|
if fGdbState <> gsPaused then
|
||||||
|
exit;
|
||||||
|
fCatchCustomEvalAsString:=true;
|
||||||
|
fCaughtCustomEvalAstring := '';
|
||||||
|
gdbCommand('-data-evaluate-expression "' + exp + '"', @gdboutJsonize);
|
||||||
|
sleep(25);
|
||||||
|
Application.ProcessMessages();
|
||||||
|
sleep(25);
|
||||||
|
result := fCaughtCustomEvalAstring;
|
||||||
|
end;
|
||||||
|
|
||||||
procedure TGdbWidget.updateButtonsState;
|
procedure TGdbWidget.updateButtonsState;
|
||||||
begin
|
begin
|
||||||
case fGdbState of
|
case fGdbState of
|
||||||
|
@ -2256,6 +2273,15 @@ begin
|
||||||
exit;
|
exit;
|
||||||
end;
|
end;
|
||||||
|
|
||||||
|
if fCatchCustomEvalAsString then
|
||||||
|
begin
|
||||||
|
fCatchCustomEvalAsString := false;
|
||||||
|
fCaughtCustomEvalAstring := '';
|
||||||
|
if fJson.findAny('value', val) then
|
||||||
|
fCaughtCustomEvalAstring := val.AsString;
|
||||||
|
exit;
|
||||||
|
end;
|
||||||
|
|
||||||
if fJson.findAny('reason', val) then
|
if fJson.findAny('reason', val) then
|
||||||
begin
|
begin
|
||||||
reason := val.AsString;
|
reason := val.AsString;
|
||||||
|
|
|
@ -2773,12 +2773,28 @@ end;
|
||||||
|
|
||||||
procedure TDexedMemo.showDDocs;
|
procedure TDexedMemo.showDDocs;
|
||||||
var
|
var
|
||||||
|
exp: string;
|
||||||
|
ev1: string;
|
||||||
|
ev2: string;
|
||||||
str: string;
|
str: string;
|
||||||
begin
|
begin
|
||||||
fCanShowHint := false;
|
fCanShowHint := false;
|
||||||
if not fIsDSource and not alwaysAdvancedFeatures then
|
if not fIsDSource and not alwaysAdvancedFeatures then
|
||||||
exit;
|
exit;
|
||||||
DcdWrapper.getDdocFromCursor(str);
|
|
||||||
|
if assigned(fDebugger) and fDebugger.running then
|
||||||
|
begin
|
||||||
|
lexWholeText([TLexOption.lxoNoComments]);
|
||||||
|
exp := getExpressionAt(fLexToks, fMousePos);
|
||||||
|
ev1 := fDebugger.evaluate(exp);
|
||||||
|
if ev1.isEmpty then
|
||||||
|
ev1 := '???';
|
||||||
|
ev2 := fDebugger.evaluate('*' + exp);
|
||||||
|
if ev2.isEmpty then
|
||||||
|
ev2 := '???';
|
||||||
|
str := format('exp: %s'#10'---'#10'%s'#10'---'#10'%s', [exp, ev1, ev2]);
|
||||||
|
end
|
||||||
|
else DcdWrapper.getDdocFromCursor(str);
|
||||||
|
|
||||||
if str.isNotEmpty then
|
if str.isNotEmpty then
|
||||||
begin
|
begin
|
||||||
|
|
Loading…
Reference in New Issue