123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346 |
- {
- Copyright (c) 1998 by Peter Vreman
- Lowlevel GDB interface which communicates directly with libgdb
- See the file COPYING.FPC, included in this distribution,
- for details about the copyright.
- This program 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.
- **********************************************************************}
- unit GDBCon;
- interface
- uses
- GDBInt;
- type
- PGDBController=^TGDBController;
- TGDBController=object(TGDBInterface)
- progname,
- progdir,
- progargs : pchar;
- in_command,
- init_count : longint;
- constructor Init;
- destructor Done;
- procedure CommandBegin(const s:string);virtual;
- procedure Command(const s:string);
- procedure CommandEnd(const s:string);virtual;
- procedure Reset;virtual;
- { tracing }
- procedure StartTrace;
- procedure Run;virtual;
- procedure TraceStep;virtual;
- procedure TraceNext;virtual;
- procedure TraceStepI;virtual;
- procedure TraceNextI;virtual;
- procedure Continue;virtual;
- { needed for dos because newlines are only #10 (PM) }
- procedure WriteErrorBuf;
- procedure WriteOutputBuf;
- function GetOutput : Pchar;
- function GetError : Pchar;
- function LoadFile(var fn:string):boolean;
- procedure SetDir(const s : string);
- procedure SetArgs(const s : string);
- procedure ClearSymbols;
- end;
- procedure UnixDir(var s : string);
- implementation
- uses
- {$ifdef win32}
- windows,
- {$endif win32}
- strings;
- {$ifdef win32}
- const
- CygDrivePrefixKey1 = 'Software';
- CygDrivePrefixKey2 = 'Cygnus Solutions';
- CygDrivePrefixKey3 = 'Cygwin';
- CygDrivePrefixKey4 = 'mounts v2';
- CygDrivePrefixKey = 'cygdrive prefix';
- function CygDrivePrefix : string;
- var
- i : longint;
- length : dword;
- Value : pchar;
- _type : dword;
- Key,NKey : HKey;
- begin
- Length:=0;
- Key:=HKEY_CURRENT_USER;
- i := RegOpenKeyEx(Key, CygDrivePrefixKey1, 0, KEY_ENUMERATE_SUB_KEYS, @NKey);
- if i=ERROR_SUCCESS then
- begin
- Key:=NKey;
- i := RegOpenKeyEx(Key, CygDrivePrefixKey2, 0, KEY_ENUMERATE_SUB_KEYS, @NKey);
- end;
- if i=ERROR_SUCCESS then
- begin
- RegCloseKey(Key);
- Key:=NKey;
- i := RegOpenKeyEx(Key, CygDrivePrefixKey3, 0, KEY_ENUMERATE_SUB_KEYS, @NKey);
- end;
- if i=ERROR_SUCCESS then
- begin
- RegCloseKey(Key);
- Key:=NKey;
- i := RegOpenKeyEx(Key, CygDrivePrefixKey4, 0, KEY_ENUMERATE_SUB_KEYS, @NKey);
- end;
- if i=ERROR_SUCCESS then
- begin
- RegCloseKey(Key);
- Key:=NKey;
- i := RegQueryValueEx( Key, CygDrivePrefixKey, nil, @_type, nil, @length);
- end;
- if i<>ERROR_SUCCESS then
- CygDrivePrefix:='/cygdrive'
- else
- Begin
- GetMem(Value,Length);
- i := RegQueryValueEx( Key, CygDrivePrefixKey, nil, @_type, LPByte(Value), @length);
- if i<>ERROR_SUCCESS then
- CygDrivePrefix:='/cygdrive'
- else
- CygDrivePrefix:=StrPas(Value);
- FreeMem(Value,Length);
- End;
- if Key<>HKEY_CURRENT_USER then
- RegCloseKey(Key);
- end;
- {$endif win32}
- procedure UnixDir(var s : string);
- var i : longint;
- begin
- for i:=1 to length(s) do
- if s[i]='\' then
- {$ifdef win32}
- { Don't touch at '\ ' used to escapes spaces in windows file names PM }
- if (i=length(s)) or (s[i+1]<>' ') then
- {$endif win32}
- s[i]:='/';
- {$ifdef win32}
- { for win32 we should convert e:\ into //e/ PM }
- if (length(s)>2) and (s[2]=':') and (s[3]='/') then
- s:=CygDrivePrefix+'/'+s[1]+copy(s,3,length(s));
- {$endif win32}
- end;
- constructor TGDBController.Init;
- begin
- inherited init;
- end;
- destructor TGDBController.Done;
- begin
- if assigned(progname) then
- strdispose(progname);
- if assigned(progdir) then
- strdispose(progdir);
- if assigned(progargs) then
- strdispose(progargs);
- inherited done;
- end;
- procedure TGDBController.Command(const s:string);
- begin
- inc(in_command);
- CommandBegin(s);
- gdboutputbuf.reset;
- gdberrorbuf.reset;
- gdb_command(s);
- {
- What is that for ?? PM
- I had to comment it because
- it resets the debuggere after each command !!
- Maybe it can happen on errors ??
- if in_command<0 then
- begin
- in_command:=0;
- inc(in_command);
- Reset;
- dec(in_command);
- end; }
- CommandEnd(s);
- dec(in_command);
- end;
- procedure TGDBController.CommandBegin(const s:string);
- begin
- end;
- procedure TGDBController.CommandEnd(const s:string);
- begin
- end;
- function TGDBController.LoadFile(var fn:string):boolean;
- var
- cmd : string;
- begin
- getdir(0,cmd);
- UnixDir(cmd);
- cmd:='cd '+cmd;
- Command(cmd);
- GDB__Init;
- UnixDir(fn);
- if assigned(progname) then
- strdispose(progname);
- getmem(progname,length(fn)+1);
- strpcopy(progname,fn);
- if fn<>'' then
- Command('file '+fn);
- LoadFile:=true;
- end;
- procedure TGDBController.SetDir(const s : string);
- var
- hs : string;
- begin
- hs:=s;
- UnixDir(hs);
- if assigned(progdir) then
- strdispose(progdir);
- getmem(progdir,length(hs)+1);
- strpcopy(progdir,hs);
- command('cd '+hs);
- end;
- procedure TGDBController.SetArgs(const s : string);
- begin
- if assigned(progargs) then
- strdispose(progargs);
- getmem(progargs,length(s)+1);
- strpcopy(progargs,s);
- command('set args '+s);
- end;
- procedure TGDBController.Reset;
- begin
- call_reset:=false;
- { DeleteBreakPoints(); }
- if debuggee_started then
- begin
- reset_command:=true;
- BreakSession;
- Command('kill');
- reset_command:=false;
- debuggee_started:=false;
- end;
- end;
- procedure TGDBController.StartTrace;
- begin
- Command('tbreak PASCALMAIN');
- Run;
- end;
- procedure TGDBController.Run;
- begin
- Command('run');
- inc(init_count);
- end;
- procedure TGDBController.TraceStep;
- begin
- Command('step');
- end;
- procedure TGDBController.TraceNext;
- begin
- Command('next');
- end;
- procedure TGDBController.TraceStepI;
- begin
- Command('stepi');
- end;
- procedure TGDBController.TraceNextI;
- begin
- Command('nexti');
- end;
- procedure TGDBController.Continue;
- begin
- Command('continue');
- end;
- procedure TGDBController.ClearSymbols;
- begin
- if debuggee_started then
- Reset;
- if init_count>0 then
- Command('file');
- end;
- procedure BufWrite(Buf : pchar);
- var p,pe : pchar;
- begin
- p:=buf;
- While assigned(p) do
- begin
- pe:=strscan(p,#10);
- if pe<>nil then
- pe^:=#0;
- Writeln(p);
- { restore for dispose }
- if pe<>nil then
- pe^:=#10;
- if pe=nil then
- p:=nil
- else
- begin
- p:=pe;
- inc(p);
- end;
- end;
- end;
- function TGDBController.GetOutput : Pchar;
- begin
- GetOutput:=gdboutputbuf.buf;
- end;
- function TGDBController.GetError : Pchar;
- var p : pchar;
- begin
- p:=gdberrorbuf.buf;
- if (p^=#0) and got_error then
- GetError:=pchar(longint(gdboutputbuf.buf)+gdboutputbuf.idx)
- else
- GetError:=p;
- end;
- procedure TGDBController.WriteErrorBuf;
- begin
- BufWrite(gdberrorbuf.buf);
- end;
- procedure TGDBController.WriteOutputBuf;
- begin
- BufWrite(gdboutputbuf.buf);
- end;
- end.
|