2019-02-03 06:48:21 -07:00
|
|
|
" Debugger plugin using gdb.
|
2017-11-06 19:16:52 -07:00
|
|
|
"
|
|
|
|
" WORK IN PROGRESS - much doesn't work yet
|
|
|
|
"
|
2019-02-03 06:48:21 -07:00
|
|
|
" Open two visible terminal windows:
|
2017-11-06 19:16:52 -07:00
|
|
|
" 1. run a pty, as with ":term NONE"
|
|
|
|
" 2. run gdb, passing the pty
|
2019-02-03 06:48:21 -07:00
|
|
|
" The current window is used to view source code and follows gdb.
|
|
|
|
"
|
|
|
|
" A third terminal window is hidden, it is used for communication with gdb.
|
|
|
|
"
|
|
|
|
" The communication with gdb uses GDB/MI. See:
|
|
|
|
" https://sourceware.org/gdb/current/onlinedocs/gdb/GDB_002fMI.html
|
2017-11-06 19:16:52 -07:00
|
|
|
"
|
|
|
|
" Author: Bram Moolenaar
|
2019-02-03 06:48:21 -07:00
|
|
|
" Copyright: Vim license applies, see ":help license"
|
2017-11-06 19:16:52 -07:00
|
|
|
|
2017-11-07 11:34:37 -07:00
|
|
|
" In case this gets loaded twice.
|
|
|
|
if exists(':Termdebug')
|
|
|
|
finish
|
|
|
|
endif
|
|
|
|
|
2019-02-03 06:48:21 -07:00
|
|
|
" The command that starts debugging, e.g. ":Termdebug vim".
|
|
|
|
" To end type "quit" in the gdb window.
|
2017-11-06 19:16:52 -07:00
|
|
|
command -nargs=* -complete=file Termdebug call s:StartDebug(<q-args>)
|
|
|
|
|
2019-02-03 06:48:21 -07:00
|
|
|
" Name of the gdb command, defaults to "gdb".
|
2019-04-15 22:48:52 -07:00
|
|
|
if !exists('termdebugger')
|
|
|
|
let termdebugger = 'gdb'
|
2017-11-06 19:16:52 -07:00
|
|
|
endif
|
|
|
|
|
2019-02-03 06:48:21 -07:00
|
|
|
" Sign used to highlight the line where the program has stopped.
|
2019-04-15 22:48:52 -07:00
|
|
|
" There can be only one.
|
2019-02-03 06:48:21 -07:00
|
|
|
sign define debugPC linehl=debugPC
|
2019-04-15 22:48:52 -07:00
|
|
|
let s:pc_id = 12
|
|
|
|
let s:break_id = 13
|
|
|
|
|
|
|
|
" Sign used to indicate a breakpoint.
|
|
|
|
" Can be used multiple times.
|
|
|
|
sign define debugBreakpoint text=>> texthl=debugBreakpoint
|
|
|
|
|
2019-02-03 06:48:21 -07:00
|
|
|
if &background == 'light'
|
2019-04-15 22:48:52 -07:00
|
|
|
hi default debugPC term=reverse ctermbg=lightblue guibg=lightblue
|
2019-02-03 06:48:21 -07:00
|
|
|
else
|
2019-04-15 22:48:52 -07:00
|
|
|
hi default debugPC term=reverse ctermbg=darkblue guibg=darkblue
|
2019-02-03 06:48:21 -07:00
|
|
|
endif
|
2019-04-15 22:48:52 -07:00
|
|
|
hi default debugBreakpoint term=reverse ctermbg=red guibg=red
|
2019-02-03 06:48:21 -07:00
|
|
|
|
2017-11-06 19:16:52 -07:00
|
|
|
func s:StartDebug(cmd)
|
2019-02-03 06:48:21 -07:00
|
|
|
let s:startwin = win_getid(winnr())
|
|
|
|
let s:startsigncolumn = &signcolumn
|
|
|
|
|
2017-11-06 19:16:52 -07:00
|
|
|
" Open a terminal window without a job, to run the debugged program
|
2019-02-03 06:48:21 -07:00
|
|
|
let s:ptybuf = term_start('NONE', {
|
|
|
|
\ 'term_name': 'gdb program',
|
|
|
|
\ })
|
|
|
|
if s:ptybuf == 0
|
|
|
|
echoerr 'Failed to open the program terminal window'
|
|
|
|
return
|
|
|
|
endif
|
|
|
|
let pty = job_info(term_getjob(s:ptybuf))['tty_out']
|
2019-04-15 22:50:25 -07:00
|
|
|
let s:ptywin = win_getid(winnr())
|
2019-02-03 06:48:21 -07:00
|
|
|
|
|
|
|
" Create a hidden terminal window to communicate with gdb
|
|
|
|
let s:commbuf = term_start('NONE', {
|
|
|
|
\ 'term_name': 'gdb communication',
|
|
|
|
\ 'out_cb': function('s:CommOutput'),
|
|
|
|
\ 'hidden': 1,
|
|
|
|
\ })
|
|
|
|
if s:commbuf == 0
|
|
|
|
echoerr 'Failed to open the communication terminal window'
|
|
|
|
exe 'bwipe! ' . s:ptybuf
|
|
|
|
return
|
|
|
|
endif
|
|
|
|
let commpty = job_info(term_getjob(s:commbuf))['tty_out']
|
2017-11-06 19:16:52 -07:00
|
|
|
|
|
|
|
" Open a terminal window to run the debugger.
|
2019-04-15 22:48:52 -07:00
|
|
|
let cmd = [g:termdebugger, '-tty', pty, a:cmd]
|
2017-11-06 19:16:52 -07:00
|
|
|
echomsg 'executing "' . join(cmd) . '"'
|
|
|
|
let gdbbuf = term_start(cmd, {
|
|
|
|
\ 'exit_cb': function('s:EndDebug'),
|
2019-02-03 06:48:21 -07:00
|
|
|
\ 'term_finish': 'close',
|
2017-11-06 19:16:52 -07:00
|
|
|
\ })
|
2019-02-03 06:48:21 -07:00
|
|
|
if gdbbuf == 0
|
|
|
|
echoerr 'Failed to open the gdb terminal window'
|
|
|
|
exe 'bwipe! ' . s:ptybuf
|
|
|
|
exe 'bwipe! ' . s:commbuf
|
|
|
|
return
|
|
|
|
endif
|
2019-04-15 22:50:25 -07:00
|
|
|
let s:gdbwin = win_getid(winnr())
|
2019-02-03 06:48:21 -07:00
|
|
|
|
|
|
|
" Connect gdb to the communication pty, using the GDB/MI interface
|
|
|
|
call term_sendkeys(gdbbuf, 'new-ui mi ' . commpty . "\r")
|
2019-04-15 22:48:52 -07:00
|
|
|
|
2019-04-15 22:50:25 -07:00
|
|
|
" Install debugger commands in the text window.
|
|
|
|
call win_gotoid(s:startwin)
|
2019-04-15 22:48:52 -07:00
|
|
|
call s:InstallCommands()
|
2019-04-15 22:50:25 -07:00
|
|
|
call win_gotoid(s:gdbwin)
|
2019-04-15 22:48:52 -07:00
|
|
|
|
|
|
|
let s:breakpoints = {}
|
2017-11-06 19:16:52 -07:00
|
|
|
endfunc
|
|
|
|
|
|
|
|
func s:EndDebug(job, status)
|
2019-02-03 06:48:21 -07:00
|
|
|
exe 'bwipe! ' . s:ptybuf
|
|
|
|
exe 'bwipe! ' . s:commbuf
|
2019-04-15 22:48:52 -07:00
|
|
|
|
|
|
|
let curwinid = win_getid(winnr())
|
|
|
|
|
|
|
|
call win_gotoid(s:startwin)
|
|
|
|
let &signcolumn = s:startsigncolumn
|
|
|
|
call s:DeleteCommands()
|
|
|
|
|
|
|
|
call win_gotoid(curwinid)
|
2019-02-03 06:48:21 -07:00
|
|
|
endfunc
|
|
|
|
|
|
|
|
" Handle a message received from gdb on the GDB/MI interface.
|
|
|
|
func s:CommOutput(chan, msg)
|
|
|
|
let msgs = split(a:msg, "\r")
|
|
|
|
|
|
|
|
for msg in msgs
|
|
|
|
" remove prefixed NL
|
|
|
|
if msg[0] == "\n"
|
|
|
|
let msg = msg[1:]
|
|
|
|
endif
|
|
|
|
if msg != ''
|
|
|
|
if msg =~ '^\*\(stopped\|running\)'
|
2019-04-15 22:48:52 -07:00
|
|
|
call s:HandleCursor(msg)
|
2019-04-15 22:50:25 -07:00
|
|
|
elseif msg =~ '^\^done,bkpt=' || msg =~ '^=breakpoint-created,'
|
2019-04-15 22:48:52 -07:00
|
|
|
call s:HandleNewBreakpoint(msg)
|
|
|
|
elseif msg =~ '^=breakpoint-deleted,'
|
|
|
|
call s:HandleBreakpointDelete(msg)
|
2019-04-15 22:50:25 -07:00
|
|
|
elseif msg =~ '^\^done,value='
|
|
|
|
call s:HandleEvaluate(msg)
|
|
|
|
elseif msg =~ '^\^error,msg='
|
|
|
|
call s:HandleError(msg)
|
2019-04-15 22:48:52 -07:00
|
|
|
endif
|
|
|
|
endif
|
|
|
|
endfor
|
|
|
|
endfunc
|
|
|
|
|
|
|
|
" Install commands in the current window to control the debugger.
|
|
|
|
func s:InstallCommands()
|
|
|
|
command Break call s:SetBreakpoint()
|
|
|
|
command Delete call s:DeleteBreakpoint()
|
|
|
|
command Step call s:SendCommand('-exec-step')
|
2019-04-15 22:50:25 -07:00
|
|
|
command Over call s:SendCommand('-exec-next')
|
2019-04-15 22:48:52 -07:00
|
|
|
command Finish call s:SendCommand('-exec-finish')
|
|
|
|
command Continue call s:SendCommand('-exec-continue')
|
2019-04-15 22:50:25 -07:00
|
|
|
command -range -nargs=* Evaluate call s:Evaluate(<range>, <q-args>)
|
|
|
|
command Gdb call win_gotoid(s:gdbwin)
|
|
|
|
command Program call win_gotoid(s:ptywin)
|
|
|
|
|
|
|
|
" TODO: can the K mapping be restored?
|
|
|
|
nnoremap K :Evaluate<CR>
|
2019-04-15 22:48:52 -07:00
|
|
|
endfunc
|
|
|
|
|
|
|
|
" Delete installed debugger commands in the current window.
|
|
|
|
func s:DeleteCommands()
|
|
|
|
delcommand Break
|
|
|
|
delcommand Delete
|
|
|
|
delcommand Step
|
2019-04-15 22:50:25 -07:00
|
|
|
delcommand Over
|
2019-04-15 22:48:52 -07:00
|
|
|
delcommand Finish
|
|
|
|
delcommand Continue
|
2019-04-15 22:50:25 -07:00
|
|
|
delcommand Evaluate
|
|
|
|
delcommand Gdb
|
|
|
|
delcommand Program
|
|
|
|
|
|
|
|
nunmap K
|
|
|
|
sign undefine debugPC
|
|
|
|
sign undefine debugBreakpoint
|
|
|
|
exe 'sign unplace ' . s:pc_id
|
|
|
|
for key in keys(s:breakpoints)
|
|
|
|
exe 'sign unplace ' . (s:break_id + key)
|
|
|
|
endfor
|
|
|
|
unlet s:breakpoints
|
2019-04-15 22:48:52 -07:00
|
|
|
endfunc
|
|
|
|
|
|
|
|
" :Break - Set a breakpoint at the cursor position.
|
|
|
|
func s:SetBreakpoint()
|
|
|
|
call term_sendkeys(s:commbuf, '-break-insert --source '
|
|
|
|
\ . fnameescape(expand('%:p')) . ' --line ' . line('.') . "\r")
|
|
|
|
endfunc
|
|
|
|
|
|
|
|
" :Delete - Delete a breakpoint at the cursor position.
|
|
|
|
func s:DeleteBreakpoint()
|
|
|
|
let fname = fnameescape(expand('%:p'))
|
|
|
|
let lnum = line('.')
|
|
|
|
for [key, val] in items(s:breakpoints)
|
|
|
|
if val['fname'] == fname && val['lnum'] == lnum
|
|
|
|
call term_sendkeys(s:commbuf, '-break-delete ' . key . "\r")
|
|
|
|
" Assume this always wors, the reply is simply "^done".
|
|
|
|
exe 'sign unplace ' . (s:break_id + key)
|
|
|
|
unlet s:breakpoints[key]
|
|
|
|
break
|
|
|
|
endif
|
|
|
|
endfor
|
|
|
|
endfunc
|
|
|
|
|
|
|
|
" :Next, :Continue, etc - send a command to gdb
|
|
|
|
func s:SendCommand(cmd)
|
|
|
|
call term_sendkeys(s:commbuf, a:cmd . "\r")
|
|
|
|
endfunc
|
|
|
|
|
2019-04-15 22:50:25 -07:00
|
|
|
" :Evaluate - evaluate what is under the cursor
|
|
|
|
func s:Evaluate(range, arg)
|
|
|
|
if a:arg != ''
|
|
|
|
let expr = a:arg
|
|
|
|
elseif a:range == 2
|
|
|
|
let pos = getcurpos()
|
|
|
|
let reg = getreg('v', 1, 1)
|
|
|
|
let regt = getregtype('v')
|
|
|
|
normal! gv"vy
|
|
|
|
let expr = @v
|
|
|
|
call setpos('.', pos)
|
|
|
|
call setreg('v', reg, regt)
|
|
|
|
else
|
|
|
|
let expr = expand('<cexpr>')
|
|
|
|
endif
|
|
|
|
call term_sendkeys(s:commbuf, '-data-evaluate-expression "' . expr . "\"\r")
|
|
|
|
let s:evalexpr = expr
|
|
|
|
endfunc
|
|
|
|
|
|
|
|
" Handle the result of data-evaluate-expression
|
|
|
|
func s:HandleEvaluate(msg)
|
|
|
|
echomsg '"' . s:evalexpr . '": ' . substitute(a:msg, '.*value="\(.*\)"', '\1', '')
|
|
|
|
endfunc
|
|
|
|
|
|
|
|
" Handle an error.
|
|
|
|
func s:HandleError(msg)
|
|
|
|
echoerr substitute(a:msg, '.*msg="\(.*\)"', '\1', '')
|
|
|
|
endfunc
|
|
|
|
|
2019-04-15 22:48:52 -07:00
|
|
|
" Handle stopping and running message from gdb.
|
|
|
|
" Will update the sign that shows the current position.
|
|
|
|
func s:HandleCursor(msg)
|
|
|
|
let wid = win_getid(winnr())
|
|
|
|
|
|
|
|
if win_gotoid(s:startwin)
|
|
|
|
if a:msg =~ '^\*stopped'
|
|
|
|
let fname = substitute(a:msg, '.*fullname="\([^"]*\)".*', '\1', '')
|
|
|
|
let lnum = substitute(a:msg, '.*line="\([^"]*\)".*', '\1', '')
|
|
|
|
if lnum =~ '^[0-9]*$'
|
|
|
|
if expand('%:h') != fname
|
|
|
|
if &modified
|
|
|
|
" TODO: find existing window
|
|
|
|
exe 'split ' . fnameescape(fname)
|
|
|
|
let s:startwin = win_getid(winnr())
|
2019-02-03 06:48:21 -07:00
|
|
|
else
|
2019-04-15 22:48:52 -07:00
|
|
|
exe 'edit ' . fnameescape(fname)
|
2019-02-03 06:48:21 -07:00
|
|
|
endif
|
|
|
|
endif
|
2019-04-15 22:48:52 -07:00
|
|
|
exe lnum
|
|
|
|
exe 'sign place ' . s:pc_id . ' line=' . lnum . ' name=debugPC file=' . fnameescape(fname)
|
|
|
|
setlocal signcolumn=yes
|
2019-02-03 06:48:21 -07:00
|
|
|
endif
|
2019-04-15 22:48:52 -07:00
|
|
|
else
|
|
|
|
exe 'sign unplace ' . s:pc_id
|
2019-02-03 06:48:21 -07:00
|
|
|
endif
|
2019-04-15 22:48:52 -07:00
|
|
|
|
|
|
|
call win_gotoid(wid)
|
|
|
|
endif
|
|
|
|
endfunc
|
|
|
|
|
|
|
|
" Handle setting a breakpoint
|
|
|
|
" Will update the sign that shows the breakpoint
|
|
|
|
func s:HandleNewBreakpoint(msg)
|
|
|
|
let nr = substitute(a:msg, '.*number="\([0-9]\)*\".*', '\1', '') + 0
|
|
|
|
if nr == 0
|
|
|
|
return
|
|
|
|
endif
|
|
|
|
|
|
|
|
if has_key(s:breakpoints, nr)
|
|
|
|
let entry = s:breakpoints[nr]
|
|
|
|
else
|
|
|
|
let entry = {}
|
|
|
|
let s:breakpoints[nr] = entry
|
|
|
|
endif
|
|
|
|
|
|
|
|
let fname = substitute(a:msg, '.*fullname="\([^"]*\)".*', '\1', '')
|
|
|
|
let lnum = substitute(a:msg, '.*line="\([^"]*\)".*', '\1', '')
|
|
|
|
|
|
|
|
exe 'sign place ' . (s:break_id + nr) . ' line=' . lnum . ' name=debugBreakpoint file=' . fnameescape(fname)
|
|
|
|
|
|
|
|
let entry['fname'] = fname
|
|
|
|
let entry['lnum'] = lnum
|
|
|
|
endfunc
|
|
|
|
|
|
|
|
" Handle deleting a breakpoint
|
|
|
|
" Will remove the sign that shows the breakpoint
|
|
|
|
func s:HandleBreakpointDelete(msg)
|
|
|
|
let nr = substitute(a:msg, '.*id="\([0-9]*\)\".*', '\1', '') + 0
|
|
|
|
if nr == 0
|
|
|
|
return
|
|
|
|
endif
|
|
|
|
exe 'sign unplace ' . (s:break_id + nr)
|
|
|
|
unlet s:breakpoints[nr]
|
2017-11-06 19:16:52 -07:00
|
|
|
endfunc
|