0
0
mirror of https://github.com/vim/vim.git synced 2025-07-25 10:54:51 -04:00

1521 lines
44 KiB
VimL
Raw Normal View History

" Debugger plugin using gdb.
2017-08-27 16:52:01 +02:00
"
" Author: Bram Moolenaar
" Copyright: Vim license applies, see ":help license"
2022-06-25 18:01:32 +01:00
" Last Change: 2022 Jun 24
"
2022-03-19 15:18:53 +00:00
" WORK IN PROGRESS - The basics works stable, more to come
" Note: In general you need at least GDB 7.12 because this provides the
" frame= response in MI thread-selected events we need to sync stack to file.
" The one included with "old" MingW is too old (7.6.1), you may upgrade it or
" use a newer version from http://www.equation.com/servlet/equation.cmd?fa=gdb
"
" There are two ways to run gdb:
" - In a terminal window; used if possible, does not work on MS-Windows
" Not used when g:termdebug_use_prompt is set to 1.
" - Using a "prompt" buffer; may use a terminal window for the program
2017-08-27 16:52:01 +02:00
"
" For both the current window is used to view source code and shows the
" current statement from gdb.
"
" USING A TERMINAL WINDOW
"
" Opens two visible terminal windows:
" 1. runs a pty for the debugged program, as with ":term NONE"
" 2. runs gdb, passing the pty of the debugged program
" A third terminal window is hidden, it is used for communication with gdb.
"
" USING A PROMPT BUFFER
"
" Opens a window with a prompt buffer to communicate with gdb.
" Gdb is run as a job with callbacks for I/O.
" On Unix another terminal window is opened to run the debugged program
" On MS-Windows a separate console is opened to run the debugged program
"
" The communication with gdb uses GDB/MI. See:
" https://sourceware.org/gdb/current/onlinedocs/gdb/GDB_002fMI.html
2017-08-27 16:52:01 +02:00
" In case this gets sourced twice.
2017-09-19 22:06:03 +02:00
if exists(':Termdebug')
finish
endif
" Need either the +terminal feature or +channel and the prompt buffer.
" The terminal feature does not work with gdb on win32.
if has('terminal') && !has('win32')
let s:way = 'terminal'
elseif has('channel') && exists('*prompt_setprompt')
let s:way = 'prompt'
else
if has('terminal')
let s:err = 'Cannot debug, missing prompt buffer support'
else
let s:err = 'Cannot debug, +channel feature is not supported'
endif
command -nargs=* -complete=file -bang Termdebug echoerr s:err
command -nargs=+ -complete=file -bang TermdebugCommand echoerr s:err
finish
endif
let s:keepcpo = &cpo
set cpo&vim
" The command that starts debugging, e.g. ":Termdebug vim".
" To end type "quit" in the gdb window.
command -nargs=* -complete=file -bang Termdebug call s:StartDebug(<bang>0, <f-args>)
command -nargs=+ -complete=file -bang TermdebugCommand call s:StartDebugCommand(<bang>0, <f-args>)
2017-08-27 16:52:01 +02:00
let s:pc_id = 12
2021-01-11 19:40:15 +01:00
let s:asm_id = 13
let s:break_id = 14 " breakpoint number is added to this
let s:stopped = 1
2021-01-11 19:40:15 +01:00
let s:parsing_disasm_msg = 0
let s:asm_lines = []
let s:asm_addr = ''
" Take a breakpoint number as used by GDB and turn it into an integer.
" The breakpoint may contain a dot: 123.4 -> 123004
" The main breakpoint has a zero subid.
func s:Breakpoint2SignNumber(id, subid)
return s:break_id + a:id * 1000 + a:subid
endfunction
func s:Highlight(init, old, new)
let default = a:init ? 'default ' : ''
if a:new ==# 'light' && a:old !=# 'light'
exe "hi " . default . "debugPC term=reverse ctermbg=lightblue guibg=lightblue"
elseif a:new ==# 'dark' && a:old !=# 'dark'
exe "hi " . default . "debugPC term=reverse ctermbg=darkblue guibg=darkblue"
endif
endfunc
call s:Highlight(1, '', &background)
hi default debugBreakpoint term=reverse ctermbg=red guibg=red
2021-11-27 10:57:26 +00:00
hi default debugBreakpointDisabled term=reverse ctermbg=gray guibg=gray
" Get the command to execute the debugger as a list, defaults to ["gdb"].
2021-12-24 13:18:38 +00:00
func s:GetCommand()
if exists('g:termdebug_config')
let cmd = get(g:termdebug_config, 'command', 'gdb')
elseif exists('g:termdebugger')
let cmd = g:termdebugger
else
let cmd = 'gdb'
endif
return type(cmd) == v:t_list ? copy(cmd) : [cmd]
2021-12-24 13:18:38 +00:00
endfunc
func s:StartDebug(bang, ...)
" First argument is the command to debug, second core file or process ID.
call s:StartDebug_internal({'gdb_args': a:000, 'bang': a:bang})
endfunc
func s:StartDebugCommand(bang, ...)
" First argument is the command to debug, rest are run arguments.
call s:StartDebug_internal({'gdb_args': [a:1], 'proc_args': a:000[1:], 'bang': a:bang})
endfunc
func s:StartDebug_internal(dict)
if exists('s:gdbwin')
echoerr 'Terminal debugger already running, cannot run two'
return
endif
2021-12-24 13:18:38 +00:00
let gdbcmd = s:GetCommand()
if !executable(gdbcmd[0])
echoerr 'Cannot execute debugger program "' .. gdbcmd[0] .. '"'
return
endif
let s:ptywin = 0
let s:pid = 0
2021-01-11 19:40:15 +01:00
let s:asmwin = 0
2021-08-14 21:25:52 +02:00
if exists('#User#TermdebugStartPre')
doauto <nomodeline> User TermdebugStartPre
endif
" Uncomment this line to write logging in "debuglog".
" call ch_logfile('debuglog', 'w')
let s:sourcewin = win_getid(winnr())
2020-10-26 21:12:46 +01:00
" Remember the old value of 'signcolumn' for each buffer that it's set in, so
" that we can restore the value for all buffers.
let b:save_signcolumn = &signcolumn
let s:signcolumn_buflist = [bufnr()]
2017-09-27 22:23:55 +02:00
let s:save_columns = 0
2019-05-26 21:33:31 +02:00
let s:allleft = 0
let wide = 0
if exists('g:termdebug_config')
let wide = get(g:termdebug_config, 'wide', 0)
elseif exists('g:termdebug_wide')
let wide = g:termdebug_wide
endif
if wide > 0
if &columns < wide
2017-09-27 22:23:55 +02:00
let s:save_columns = &columns
let &columns = wide
2022-01-08 21:51:59 +00:00
" If we make the Vim window wider, use the whole left half for the debug
2019-05-26 21:33:31 +02:00
" windows.
let s:allleft = 1
2017-09-27 22:23:55 +02:00
endif
let s:vertical = 1
else
let s:vertical = 0
endif
" Override using a terminal window by setting g:termdebug_use_prompt to 1.
let use_prompt = 0
if exists('g:termdebug_config')
let use_prompt = get(g:termdebug_config, 'use_prompt', 0)
elseif exists('g:termdebug_use_prompt')
let use_prompt = g:termdebug_use_prompt
endif
if has('terminal') && !has('win32') && !use_prompt
let s:way = 'terminal'
else
let s:way = 'prompt'
endif
if s:way == 'prompt'
call s:StartDebug_prompt(a:dict)
else
call s:StartDebug_term(a:dict)
endif
2021-01-11 19:40:15 +01:00
if s:GetDisasmWindow()
let curwinid = win_getid(winnr())
call s:GotoAsmwinOrCreateIt()
call win_gotoid(curwinid)
2021-01-11 19:40:15 +01:00
endif
2021-08-14 21:25:52 +02:00
if exists('#User#TermdebugStartPost')
doauto <nomodeline> User TermdebugStartPost
endif
endfunc
" Use when debugger didn't start or ended.
func s:CloseBuffers()
exe 'bwipe! ' . s:ptybuf
exe 'bwipe! ' . s:commbuf
unlet! s:gdbwin
endfunc
2021-05-30 20:54:13 +02:00
func s:CheckGdbRunning()
let gdbproc = term_getjob(s:gdbbuf)
if gdbproc == v:null || job_status(gdbproc) !=# 'run'
2021-12-24 13:18:38 +00:00
echoerr string(s:GetCommand()[0]) . ' exited unexpectedly'
2021-05-30 20:54:13 +02:00
call s:CloseBuffers()
return ''
endif
return 'ok'
endfunc
2022-05-09 19:50:35 +01:00
" Open a terminal window without a job, to run the debugged program in.
func s:StartDebug_term(dict)
let s:ptybuf = term_start('NONE', {
2019-09-20 14:38:13 +02:00
\ 'term_name': 'debugged program',
\ 'vertical': s:vertical,
\ })
if s:ptybuf == 0
echoerr 'Failed to open the program terminal window'
return
endif
let pty = job_info(term_getjob(s:ptybuf))['tty_out']
let s:ptywin = win_getid(winnr())
if s:vertical
" Assuming the source code window will get a signcolumn, use two more
" columns for that, thus one less for the terminal window.
exe (&columns / 2 - 1) . "wincmd |"
2019-05-26 21:33:31 +02:00
if s:allleft
" use the whole left column
wincmd H
endif
endif
" Create a hidden terminal window to communicate with gdb
let s:commbuf = term_start('NONE', {
2019-09-20 14:38:13 +02:00
\ '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-08-27 16:52:01 +02:00
let gdb_args = get(a:dict, 'gdb_args', [])
let proc_args = get(a:dict, 'proc_args', [])
2021-12-24 13:18:38 +00:00
let gdb_cmd = s:GetCommand()
if exists('g:termdebug_config') && has_key(g:termdebug_config, 'command_add_args')
let gdb_cmd = g:termdebug_config.command_add_args(gdb_cmd, pty)
else
" Add -quiet to avoid the intro message causing a hit-enter prompt.
let gdb_cmd += ['-quiet']
" Disable pagination, it causes everything to stop at the gdb
let gdb_cmd += ['-iex', 'set pagination off']
" Interpret commands while the target is running. This should usually only
" be exec-interrupt, since many commands don't work properly while the
" target is running (so execute during startup).
let gdb_cmd += ['-iex', 'set mi-async on']
" Open a terminal window to run the debugger.
let gdb_cmd += ['-tty', pty]
" Command executed _after_ startup is done, provides us with the necessary
" feedback
let gdb_cmd += ['-ex', 'echo startupdone\n']
endif
if exists('g:termdebug_config') && has_key(g:termdebug_config, 'command_filter')
let gdb_cmd = g:termdebug_config.command_filter(gdb_cmd)
endif
2021-11-27 10:57:26 +00:00
" Adding arguments requested by the user
let gdb_cmd += gdb_args
call ch_log('executing "' . join(gdb_cmd) . '"')
let s:gdbbuf = term_start(gdb_cmd, {
2019-09-20 14:38:13 +02:00
\ 'term_finish': 'close',
\ })
if s:gdbbuf == 0
echoerr 'Failed to open the gdb terminal window'
call s:CloseBuffers()
return
endif
let s:gdbwin = win_getid(winnr())
2021-05-30 20:54:13 +02:00
" Wait for the "startupdone" message before sending any commands.
let try_count = 0
while 1
if s:CheckGdbRunning() != 'ok'
return
endif
for lnum in range(1, 200)
if term_getline(s:gdbbuf, lnum) =~ 'startupdone'
2021-11-27 10:57:26 +00:00
let try_count = 9999
break
2021-05-30 20:54:13 +02:00
endif
endfor
let try_count += 1
if try_count > 300
" done or give up after five seconds
break
endif
sleep 10m
endwhile
" Set arguments to be run.
if len(proc_args)
2021-11-27 10:57:26 +00:00
call term_sendkeys(s:gdbbuf, 'server set args ' . join(proc_args) . "\r")
endif
2021-11-21 21:13:36 +00:00
" Connect gdb to the communication pty, using the GDB/MI interface.
" Prefix "server" to avoid adding this to the history.
call term_sendkeys(s:gdbbuf, 'server new-ui mi ' . commpty . "\r")
" Wait for the response to show up, users may not notice the error and wonder
" why the debugger doesn't work.
let try_count = 0
while 1
2021-05-30 20:54:13 +02:00
if s:CheckGdbRunning() != 'ok'
return
endif
let response = ''
for lnum in range(1, 200)
let line1 = term_getline(s:gdbbuf, lnum)
let line2 = term_getline(s:gdbbuf, lnum + 1)
if line1 =~ 'new-ui mi '
2021-11-27 10:57:26 +00:00
" response can be in the same line or the next line
let response = line1 . line2
if response =~ 'Undefined command'
echoerr 'Sorry, your gdb is too old, gdb 7.12 is required'
2022-03-19 15:18:53 +00:00
" CHECKME: possibly send a "server show version" here
2021-11-27 10:57:26 +00:00
call s:CloseBuffers()
return
endif
if response =~ 'New UI allocated'
" Success!
break
endif
elseif line1 =~ 'Reading symbols from' && line2 !~ 'new-ui mi '
2021-11-27 10:57:26 +00:00
" Reading symbols might take a while, try more times
let try_count -= 1
endif
endfor
if response =~ 'New UI allocated'
break
endif
let try_count += 1
if try_count > 100
echoerr 'Cannot check if your gdb works, continuing anyway'
break
endif
sleep 10m
endwhile
2021-05-30 20:54:13 +02:00
call job_setoptions(term_getjob(s:gdbbuf), {'exit_cb': function('s:EndTermDebug')})
2021-08-29 21:55:35 +02:00
" Set the filetype, this can be used to add mappings.
set filetype=termdebug
call s:StartDebugCommon(a:dict)
endfunc
2022-05-09 19:50:35 +01:00
" Open a window with a prompt buffer to run gdb in.
func s:StartDebug_prompt(dict)
if s:vertical
vertical new
else
new
endif
let s:gdbwin = win_getid(winnr())
let s:promptbuf = bufnr('')
call prompt_setprompt(s:promptbuf, 'gdb> ')
set buftype=prompt
file gdb
call prompt_setcallback(s:promptbuf, function('s:PromptCallback'))
call prompt_setinterrupt(s:promptbuf, function('s:PromptInterrupt'))
if s:vertical
" Assuming the source code window will get a signcolumn, use two more
" columns for that, thus one less for the terminal window.
exe (&columns / 2 - 1) . "wincmd |"
endif
let gdb_args = get(a:dict, 'gdb_args', [])
let proc_args = get(a:dict, 'proc_args', [])
2021-12-24 13:18:38 +00:00
let gdb_cmd = s:GetCommand()
2021-11-27 10:57:26 +00:00
" Add -quiet to avoid the intro message causing a hit-enter prompt.
let gdb_cmd += ['-quiet']
" Disable pagination, it causes everything to stop at the gdb, needs to be run early
let gdb_cmd += ['-iex', 'set pagination off']
" Interpret commands while the target is running. This should usually only
" be exec-interrupt, since many commands don't work properly while the
" target is running (so execute during startup).
let gdb_cmd += ['-iex', 'set mi-async on']
" directly communicate via mi2
let gdb_cmd += ['--interpreter=mi2']
" Adding arguments requested by the user
let gdb_cmd += gdb_args
2021-11-27 10:57:26 +00:00
call ch_log('executing "' . join(gdb_cmd) . '"')
let s:gdbjob = job_start(gdb_cmd, {
2019-09-20 14:38:13 +02:00
\ 'exit_cb': function('s:EndPromptDebug'),
\ 'out_cb': function('s:GdbOutCallback'),
\ })
if job_status(s:gdbjob) != "run"
echoerr 'Failed to start gdb'
exe 'bwipe! ' . s:promptbuf
return
endif
" Mark the buffer modified so that it's not easy to close.
set modified
2022-03-19 15:18:53 +00:00
let s:gdb_channel = job_getchannel(s:gdbjob)
let s:ptybuf = 0
if has('win32')
" MS-Windows: run in a new console window for maximum compatibility
call s:SendCommand('set new-console on')
elseif has('terminal')
" Unix: Run the debugged program in a terminal window. Open it below the
" gdb window.
belowright let s:ptybuf = term_start('NONE', {
2019-09-20 14:38:13 +02:00
\ 'term_name': 'debugged program',
\ })
if s:ptybuf == 0
echoerr 'Failed to open the program terminal window'
call job_stop(s:gdbjob)
return
endif
let s:ptywin = win_getid(winnr())
let pty = job_info(term_getjob(s:ptybuf))['tty_out']
call s:SendCommand('tty ' . pty)
" Since GDB runs in a prompt window, the environment has not been set to
" match a terminal window, need to do that now.
call s:SendCommand('set env TERM = xterm-color')
call s:SendCommand('set env ROWS = ' . winheight(s:ptywin))
call s:SendCommand('set env LINES = ' . winheight(s:ptywin))
call s:SendCommand('set env COLUMNS = ' . winwidth(s:ptywin))
call s:SendCommand('set env COLORS = ' . &t_Co)
call s:SendCommand('set env VIM_TERMINAL = ' . v:version)
else
2022-01-08 21:51:59 +00:00
" TODO: open a new terminal, get the tty name, pass on to gdb
call s:SendCommand('show inferior-tty')
endif
call s:SendCommand('set print pretty on')
call s:SendCommand('set breakpoint pending on')
" Set arguments to be run
if len(proc_args)
call s:SendCommand('set args ' . join(proc_args))
endif
call s:StartDebugCommon(a:dict)
startinsert
endfunc
func s:StartDebugCommon(dict)
" Sign used to highlight the line where the program has stopped.
" There can be only one.
2022-04-18 15:36:40 +01:00
call sign_define('debugPC', #{linehl: 'debugPC'})
" Install debugger commands in the text window.
call win_gotoid(s:sourcewin)
call s:InstallCommands()
call win_gotoid(s:gdbwin)
" Enable showing a balloon with eval info
if has("balloon_eval") || has("balloon_eval_term")
set balloonexpr=TermDebugBalloonExpr()
if has("balloon_eval")
set ballooneval
endif
if has("balloon_eval_term")
set balloonevalterm
endif
endif
" Contains breakpoints that have been placed, key is a string with the GDB
" breakpoint number.
" Each entry is a dict, containing the sub-breakpoints. Key is the subid.
" For a breakpoint that is just a number the subid is zero.
" For a breakpoint "123.4" the id is "123" and subid is "4".
" Example, when breakpoint "44", "123", "123.1" and "123.2" exist:
" {'44': {'0': entry}, '123': {'0': entry, '1': entry, '2': entry}}
let s:breakpoints = {}
" Contains breakpoints by file/lnum. The key is "fname:lnum".
" Each entry is a list of breakpoint IDs at that position.
let s:breakpoint_locations = {}
augroup TermDebug
au BufRead * call s:BufRead()
au BufUnload * call s:BufUnloaded()
au OptionSet background call s:Highlight(0, v:option_old, v:option_new)
augroup END
" Run the command if the bang attribute was given and got to the debug
" window.
if get(a:dict, 'bang', 0)
2021-12-05 21:54:04 +00:00
call s:SendResumingCommand('-exec-run')
call win_gotoid(s:ptywin)
endif
endfunc
" Send a command to gdb. "cmd" is the string without line terminator.
func s:SendCommand(cmd)
call ch_log('sending to gdb: ' . a:cmd)
if s:way == 'prompt'
call ch_sendraw(s:gdb_channel, a:cmd . "\n")
else
call term_sendkeys(s:commbuf, a:cmd . "\r")
endif
2017-08-27 16:52:01 +02:00
endfunc
" This is global so that a user can create their mappings with this.
func TermDebugSendCommand(cmd)
if s:way == 'prompt'
call ch_sendraw(s:gdb_channel, a:cmd . "\n")
else
let do_continue = 0
if !s:stopped
let do_continue = 1
2021-11-21 21:13:36 +00:00
Stop
sleep 10m
endif
2022-06-25 18:01:32 +01:00
" TODO: should we prepend CTRL-U to clear the command?
call term_sendkeys(s:gdbbuf, a:cmd . "\r")
if do_continue
Continue
endif
endif
endfunc
2021-12-05 21:54:04 +00:00
" Send a command that resumes the program. If the program isn't stopped the
" command is not sent (to avoid a repeated command to cause trouble).
" If the command is sent then reset s:stopped.
func s:SendResumingCommand(cmd)
2021-11-27 10:57:26 +00:00
if s:stopped
2021-12-05 21:54:04 +00:00
" reset s:stopped here, it may take a bit of time before we get a response
let s:stopped = 0
call ch_log('assume that program is running after this command')
2021-11-27 10:57:26 +00:00
call s:SendCommand(a:cmd)
else
call ch_log('dropping command, program is running: ' . a:cmd)
endif
endfunc
" Function called when entering a line in the prompt buffer.
func s:PromptCallback(text)
call s:SendCommand(a:text)
endfunc
" Function called when pressing CTRL-C in the prompt buffer and when placing a
" breakpoint.
func s:PromptInterrupt()
call ch_log('Interrupting gdb')
if has('win32')
" Using job_stop() does not work on MS-Windows, need to send SIGTRAP to
" the debugger program so that gdb responds again.
if s:pid == 0
echoerr 'Cannot interrupt gdb, did not find a process ID'
else
call debugbreak(s:pid)
endif
else
call job_stop(s:gdbjob, 'int')
endif
endfunc
" Function called when gdb outputs text.
func s:GdbOutCallback(channel, text)
call ch_log('received from gdb: ' . a:text)
" Drop the gdb prompt, we have our own.
" Drop status and echo'd commands.
if a:text == '(gdb) ' || a:text == '^done' || a:text[0] == '&'
return
endif
2021-11-07 20:27:04 +00:00
if a:text =~ '^\^error,msg='
let text = s:DecodeMessage(a:text[11:])
if exists('s:evalexpr') && text =~ 'A syntax error in expression, near\|No symbol .* in current context'
" Silently drop evaluation errors.
unlet s:evalexpr
return
endif
elseif a:text[0] == '~'
let text = s:DecodeMessage(a:text[1:])
else
call s:CommOutput(a:channel, a:text)
return
endif
let curwinid = win_getid(winnr())
call win_gotoid(s:gdbwin)
" Add the output above the current prompt.
call append(line('$') - 1, text)
set modified
call win_gotoid(curwinid)
endfunc
" Decode a message from gdb. quotedText starts with a ", return the text up
2021-11-27 10:57:26 +00:00
" to the next ", unescaping characters:
" - remove line breaks
" - change \\t to \t
2021-12-16 14:41:10 +00:00
" - change \0xhh to \xhh (disabled for now)
2021-11-27 10:57:26 +00:00
" - change \ooo to octal
" - change \\ to \
func s:DecodeMessage(quotedText)
if a:quotedText[0] != '"'
echoerr 'DecodeMessage(): missing quote in ' . a:quotedText
return
endif
2021-11-27 10:57:26 +00:00
return a:quotedText
2021-12-16 14:41:10 +00:00
\ ->substitute('^"\|".*\|\\n', '', 'g')
\ ->substitute('\\t', "\t", 'g')
" multi-byte characters arrive in octal form
" NULL-values must be kept encoded as those break the string otherwise
\ ->substitute('\\000', s:NullRepl, 'g')
\ ->substitute('\\\o\o\o', {-> eval('"' .. submatch(0) .. '"')}, 'g')
" Note: GDB docs also mention hex encodings - the translations below work
" but we keep them out for performance-reasons until we actually see
" those in mi-returns
" \ ->substitute('\\0x\(\x\x\)', {-> eval('"\x' .. submatch(1) .. '"')}, 'g')
" \ ->substitute('\\0x00', s:NullRepl, 'g')
\ ->substitute('\\\\', '\', 'g')
\ ->substitute(s:NullRepl, '\\000', 'g')
endfunc
2021-12-16 14:41:10 +00:00
const s:NullRepl = 'XXXNULLXXX'
" Extract the "name" value from a gdb message with fullname="name".
func s:GetFullname(msg)
if a:msg !~ 'fullname'
return ''
endif
let name = s:DecodeMessage(substitute(a:msg, '.*fullname=', '', ''))
if has('win32') && name =~ ':\\\\'
" sometimes the name arrives double-escaped
let name = substitute(name, '\\\\', '\\', 'g')
endif
return name
endfunc
2021-01-11 19:40:15 +01:00
" Extract the "addr" value from a gdb message with addr="0x0001234".
func s:GetAsmAddr(msg)
if a:msg !~ 'addr='
return ''
endif
let addr = s:DecodeMessage(substitute(a:msg, '.*addr=', '', ''))
return addr
endfunc
2021-08-14 21:25:52 +02:00
func s:EndTermDebug(job, status)
2021-08-14 21:25:52 +02:00
if exists('#User#TermdebugStopPre')
doauto <nomodeline> User TermdebugStopPre
endif
exe 'bwipe! ' . s:commbuf
unlet s:gdbwin
call s:EndDebugCommon()
endfunc
func s:EndDebugCommon()
let curwinid = win_getid(winnr())
if exists('s:ptybuf') && s:ptybuf
exe 'bwipe! ' . s:ptybuf
endif
2020-10-26 21:12:46 +01:00
" Restore 'signcolumn' in all buffers for which it was set.
call win_gotoid(s:sourcewin)
2020-10-26 21:12:46 +01:00
let was_buf = bufnr()
for bufnr in s:signcolumn_buflist
if bufexists(bufnr)
exe bufnr .. "buf"
if exists('b:save_signcolumn')
2022-03-19 15:18:53 +00:00
let &signcolumn = b:save_signcolumn
unlet b:save_signcolumn
2020-10-26 21:12:46 +01:00
endif
endif
endfor
2022-04-18 15:36:40 +01:00
if bufexists(was_buf)
exe was_buf .. "buf"
endif
2020-10-26 21:12:46 +01:00
call s:DeleteCommands()
call win_gotoid(curwinid)
if s:save_columns > 0
let &columns = s:save_columns
endif
if has("balloon_eval") || has("balloon_eval_term")
set balloonexpr=
if has("balloon_eval")
set noballooneval
endif
if has("balloon_eval_term")
set noballoonevalterm
endif
endif
2021-08-14 21:25:52 +02:00
if exists('#User#TermdebugStopPost')
doauto <nomodeline> User TermdebugStopPost
endif
au! TermDebug
endfunc
func s:EndPromptDebug(job, status)
2021-08-14 21:25:52 +02:00
if exists('#User#TermdebugStopPre')
doauto <nomodeline> User TermdebugStopPre
endif
let curwinid = win_getid(winnr())
call win_gotoid(s:gdbwin)
set nomodified
close
if curwinid != s:gdbwin
call win_gotoid(curwinid)
endif
call s:EndDebugCommon()
unlet s:gdbwin
call ch_log("Returning from EndPromptDebug()")
endfunc
2021-01-11 19:40:15 +01:00
" Disassembly window - added by Michael Sartain
"
" - CommOutput: disassemble $pc
" - CommOutput: &"disassemble $pc\n"
" - CommOutput: ~"Dump of assembler code for function main(int, char**):\n"
" - CommOutput: ~" 0x0000555556466f69 <+0>:\tpush rbp\n"
" ...
" - CommOutput: ~" 0x0000555556467cd0:\tpop rbp\n"
" - CommOutput: ~" 0x0000555556467cd1:\tret \n"
" - CommOutput: ~"End of assembler dump.\n"
" - CommOutput: ^done
" - CommOutput: disassemble $pc
" - CommOutput: &"disassemble $pc\n"
" - CommOutput: &"No function contains specified address.\n"
" - CommOutput: ^error,msg="No function contains specified address."
func s:HandleDisasmMsg(msg)
if a:msg =~ '^\^done'
let curwinid = win_getid(winnr())
if win_gotoid(s:asmwin)
silent normal! gg0"_dG
call setline(1, s:asm_lines)
set nomodified
set filetype=asm
let lnum = search('^' . s:asm_addr)
if lnum != 0
2022-04-18 15:36:40 +01:00
call sign_unplace('TermDebug', #{id: s:asm_id})
call sign_place(s:asm_id, 'TermDebug', 'debugPC', '%', #{lnum: lnum})
2021-01-11 19:40:15 +01:00
endif
call win_gotoid(curwinid)
endif
let s:parsing_disasm_msg = 0
let s:asm_lines = []
elseif a:msg =~ '^\^error,msg='
if s:parsing_disasm_msg == 1
" Disassemble call ran into an error. This can happen when gdb can't
" find the function frame address, so let's try to disassemble starting
" at current PC
call s:SendCommand('disassemble $pc,+100')
endif
let s:parsing_disasm_msg = 0
elseif a:msg =~ '\&\"disassemble \$pc'
if a:msg =~ '+100'
" This is our second disasm attempt
let s:parsing_disasm_msg = 2
endif
else
let value = substitute(a:msg, '^\~\"[ ]*', '', '')
let value = substitute(value, '^=>[ ]*', '', '')
2021-11-07 20:27:04 +00:00
let value = substitute(value, '\\n\"\r$', '', '')
2021-01-11 19:40:15 +01:00
let value = substitute(value, '\\n\"$', '', '')
2021-11-07 20:27:04 +00:00
let value = substitute(value, '\r', '', '')
2021-01-11 19:40:15 +01:00
let value = substitute(value, '\\t', ' ', 'g')
if value != '' || !empty(s:asm_lines)
call add(s:asm_lines, value)
endif
endif
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
2021-01-11 19:40:15 +01:00
if s:parsing_disasm_msg
call s:HandleDisasmMsg(msg)
elseif msg != ''
if msg =~ '^\(\*stopped\|\*running\|=thread-selected\)'
2022-03-19 15:18:53 +00:00
call s:HandleCursor(msg)
2021-11-27 10:57:26 +00:00
elseif msg =~ '^\^done,bkpt=' || msg =~ '^=breakpoint-created,'
call s:HandleNewBreakpoint(msg, 0)
elseif msg =~ '^=breakpoint-modified,'
call s:HandleNewBreakpoint(msg, 1)
elseif msg =~ '^=breakpoint-deleted,'
2022-03-19 15:18:53 +00:00
call s:HandleBreakpointDelete(msg)
elseif msg =~ '^=thread-group-started'
2022-03-19 15:18:53 +00:00
call s:HandleProgramRun(msg)
elseif msg =~ '^\^done,value='
2022-03-19 15:18:53 +00:00
call s:HandleEvaluate(msg)
elseif msg =~ '^\^error,msg='
2022-03-19 15:18:53 +00:00
call s:HandleError(msg)
2021-01-11 19:40:15 +01:00
elseif msg =~ '^disassemble'
2022-03-19 15:18:53 +00:00
let s:parsing_disasm_msg = 1
let s:asm_lines = []
endif
endif
endfor
endfunc
2019-09-20 14:38:13 +02:00
func s:GotoProgram()
if has('win32')
if executable('powershell')
call system(printf('powershell -Command "add-type -AssemblyName microsoft.VisualBasic;[Microsoft.VisualBasic.Interaction]::AppActivate(%d);"', s:pid))
endif
else
2019-12-11 23:05:48 +01:00
call win_gotoid(s:ptywin)
2019-09-20 14:38:13 +02:00
endif
endfunc
" Install commands in the current window to control the debugger.
func s:InstallCommands()
let save_cpo = &cpo
set cpo&vim
2019-09-20 14:38:13 +02:00
command -nargs=? Break call s:SetBreakpoint(<q-args>)
command Clear call s:ClearBreakpoint()
2021-12-05 21:54:04 +00:00
command Step call s:SendResumingCommand('-exec-step')
command Over call s:SendResumingCommand('-exec-next')
2022-04-18 15:36:40 +01:00
command -nargs=? Until call s:Until(<q-args>)
2021-12-05 21:54:04 +00:00
command Finish call s:SendResumingCommand('-exec-finish')
command -nargs=* Run call s:Run(<q-args>)
2021-12-05 21:54:04 +00:00
command -nargs=* Arguments call s:SendResumingCommand('-exec-arguments ' . <q-args>)
if s:way == 'prompt'
2021-11-21 21:13:36 +00:00
command Stop call s:PromptInterrupt()
command Continue call s:SendCommand('continue')
else
2021-11-21 21:13:36 +00:00
command Stop call s:SendCommand('-exec-interrupt')
" using -exec-continue results in CTRL-C in the gdb window not working,
" communicating via commbuf (= use of SendCommand) has the same result
"command Continue call s:SendCommand('-exec-continue')
command Continue call term_sendkeys(s:gdbbuf, "continue\r")
endif
command -range -nargs=* Evaluate call s:Evaluate(<range>, <q-args>)
command Gdb call win_gotoid(s:gdbwin)
2019-09-20 14:38:13 +02:00
command Program call s:GotoProgram()
command Source call s:GotoSourcewinOrCreateIt()
2021-01-11 19:40:15 +01:00
command Asm call s:GotoAsmwinOrCreateIt()
command Winbar call s:InstallWinbar()
let map = 1
if exists('g:termdebug_config')
let map = get(g:termdebug_config, 'map_K', 1)
elseif exists('g:termdebug_map_K')
let map = g:termdebug_map_K
endif
if map
2020-05-26 21:20:45 +02:00
let s:k_map_saved = maparg('K', 'n', 0, 1)
nnoremap K :Evaluate<CR>
endif
2017-12-17 17:17:07 +01:00
if has('menu') && &mouse != ''
call s:InstallWinbar()
let popup = 1
if exists('g:termdebug_config')
let popup = get(g:termdebug_config, 'popup', 1)
elseif exists('g:termdebug_popup')
let popup = g:termdebug_popup
endif
if popup
let s:saved_mousemodel = &mousemodel
let &mousemodel = 'popup_setpos'
an 1.200 PopUp.-SEP3- <Nop>
an 1.210 PopUp.Set\ breakpoint :Break<CR>
an 1.220 PopUp.Clear\ breakpoint :Clear<CR>
2022-04-18 15:36:40 +01:00
an 1.230 PopUp.Run\ until :Until<CR>
an 1.240 PopUp.Evaluate :Evaluate<CR>
endif
endif
let &cpo = save_cpo
endfunc
let s:winbar_winids = []
" Install the window toolbar in the current window.
func s:InstallWinbar()
if has('menu') && &mouse != ''
nnoremenu WinBar.Step :Step<CR>
nnoremenu WinBar.Next :Over<CR>
nnoremenu WinBar.Finish :Finish<CR>
nnoremenu WinBar.Cont :Continue<CR>
nnoremenu WinBar.Stop :Stop<CR>
nnoremenu WinBar.Eval :Evaluate<CR>
call add(s:winbar_winids, win_getid(winnr()))
endif
endfunc
" Delete installed debugger commands in the current window.
func s:DeleteCommands()
delcommand Break
delcommand Clear
delcommand Step
delcommand Over
2022-04-18 15:36:40 +01:00
delcommand Until
delcommand Finish
delcommand Run
delcommand Arguments
delcommand Stop
delcommand Continue
delcommand Evaluate
delcommand Gdb
delcommand Program
delcommand Source
2021-01-11 19:40:15 +01:00
delcommand Asm
delcommand Winbar
2020-12-10 21:11:27 +01:00
if exists('s:k_map_saved')
if empty(s:k_map_saved)
nunmap K
else
2022-05-07 21:54:03 +01:00
call mapset(s:k_map_saved)
2020-12-10 21:11:27 +01:00
endif
2020-05-26 21:20:45 +02:00
unlet s:k_map_saved
endif
if has('menu')
" Remove the WinBar entries from all windows where it was added.
let curwinid = win_getid(winnr())
for winid in s:winbar_winids
if win_gotoid(winid)
2019-09-20 14:38:13 +02:00
aunmenu WinBar.Step
aunmenu WinBar.Next
aunmenu WinBar.Finish
aunmenu WinBar.Cont
aunmenu WinBar.Stop
aunmenu WinBar.Eval
endif
endfor
call win_gotoid(curwinid)
let s:winbar_winids = []
if exists('s:saved_mousemodel')
let &mousemodel = s:saved_mousemodel
unlet s:saved_mousemodel
aunmenu PopUp.-SEP3-
aunmenu PopUp.Set\ breakpoint
aunmenu PopUp.Clear\ breakpoint
2022-04-18 15:36:40 +01:00
aunmenu PopUp.Run\ until
aunmenu PopUp.Evaluate
endif
endif
2022-04-18 15:36:40 +01:00
call sign_unplace('TermDebug')
unlet s:breakpoints
unlet s:breakpoint_locations
2022-04-18 15:36:40 +01:00
call sign_undefine('debugPC')
call sign_undefine(s:BreakpointSigns->map("'debugBreakpoint' .. v:val"))
let s:BreakpointSigns = []
endfunc
2022-04-18 15:36:40 +01:00
" :Until - Execute until past a specified position or current line
func s:Until(at)
if s:stopped
" reset s:stopped here, it may take a bit of time before we get a response
let s:stopped = 0
call ch_log('assume that program is running after this command')
" Use the fname:lnum format
let at = empty(a:at) ?
\ fnameescape(expand('%:p')) . ':' . line('.') : a:at
call s:SendCommand('-exec-until ' . at)
else
call ch_log('dropping command, program is running: exec-until')
endif
endfunc
" :Break - Set a breakpoint at the cursor position.
2019-09-20 14:38:13 +02:00
func s:SetBreakpoint(at)
" Setting a breakpoint may not work while the program is running.
" Interrupt to make it work.
let do_continue = 0
if !s:stopped
let do_continue = 1
2021-11-21 21:13:36 +00:00
Stop
sleep 10m
endif
2019-09-20 14:38:13 +02:00
" Use the fname:lnum format, older gdb can't handle --source.
2019-09-20 14:38:13 +02:00
let at = empty(a:at) ?
2021-11-21 21:13:36 +00:00
\ fnameescape(expand('%:p')) . ':' . line('.') : a:at
2019-09-20 14:38:13 +02:00
call s:SendCommand('-break-insert ' . at)
if do_continue
2021-11-21 21:13:36 +00:00
Continue
endif
endfunc
" :Clear - Delete a breakpoint at the cursor position.
func s:ClearBreakpoint()
let fname = fnameescape(expand('%:p'))
let lnum = line('.')
let bploc = printf('%s:%d', fname, lnum)
if has_key(s:breakpoint_locations, bploc)
let idx = 0
2021-11-21 21:13:36 +00:00
let nr = 0
for id in s:breakpoint_locations[bploc]
if has_key(s:breakpoints, id)
2022-03-19 15:18:53 +00:00
" Assume this always works, the reply is simply "^done".
call s:SendCommand('-break-delete ' . id)
for subid in keys(s:breakpoints[id])
2022-04-18 15:36:40 +01:00
call sign_unplace('TermDebug',
\ #{id: s:Breakpoint2SignNumber(id, subid)})
2022-03-19 15:18:53 +00:00
endfor
unlet s:breakpoints[id]
unlet s:breakpoint_locations[bploc][idx]
let nr = id
break
else
2022-03-19 15:18:53 +00:00
let idx += 1
endif
endfor
2021-11-21 21:13:36 +00:00
if nr != 0
if empty(s:breakpoint_locations[bploc])
2022-03-19 15:18:53 +00:00
unlet s:breakpoint_locations[bploc]
2021-11-21 21:13:36 +00:00
endif
echomsg 'Breakpoint ' . id . ' cleared from line ' . lnum . '.'
else
echoerr 'Internal error trying to remove breakpoint at line ' . lnum . '!'
endif
2021-11-21 21:13:36 +00:00
else
echomsg 'No breakpoint to remove at line ' . lnum . '.'
endif
endfunc
func s:Run(args)
if a:args != ''
2021-12-05 21:54:04 +00:00
call s:SendResumingCommand('-exec-arguments ' . a:args)
endif
2021-12-05 21:54:04 +00:00
call s:SendResumingCommand('-exec-run')
endfunc
func s:SendEval(expr)
2021-11-21 21:13:36 +00:00
" check for "likely" boolean expressions, in which case we take it as lhs
if a:expr =~ "[=!<>]="
let exprLHS = a:expr
2021-11-07 20:27:04 +00:00
else
2021-11-21 21:13:36 +00:00
" remove text that is likely an assignment
let exprLHS = substitute(a:expr, ' *=.*', '', '')
2021-11-07 20:27:04 +00:00
endif
2022-03-19 15:18:53 +00:00
2021-11-21 21:13:36 +00:00
" encoding expression to prevent bad errors
let expr = a:expr
let expr = substitute(expr, '\\', '\\\\', 'g')
let expr = substitute(expr, '"', '\\"', 'g')
2021-11-07 20:27:04 +00:00
call s:SendCommand('-data-evaluate-expression "' . expr . '"')
2021-11-21 21:13:36 +00:00
let s:evalexpr = exprLHS
endfunc
2022-03-19 15:18:53 +00:00
" :Evaluate - evaluate what is specified / under the cursor
func s:Evaluate(range, arg)
2021-11-21 21:13:36 +00:00
let expr = s:GetEvaluationExpression(a:range, a:arg)
let s:ignoreEvalError = 0
call s:SendEval(expr)
endfunc
2022-03-19 15:18:53 +00:00
" get what is specified / under the cursor
2021-11-21 21:13:36 +00:00
func s:GetEvaluationExpression(range, arg)
if a:arg != ''
2021-11-21 21:13:36 +00:00
" user supplied evaluation
let expr = s:CleanupExpr(a:arg)
" DSW: replace "likely copy + paste" assignment
let expr = substitute(expr, '"\([^"]*\)": *', '\1=', 'g')
elseif a:range == 2
2021-11-21 21:13:36 +00:00
" no evaluation but provided but range set
let pos = getcurpos()
let reg = getreg('v', 1, 1)
let regt = getregtype('v')
normal! gv"vy
2021-11-21 21:13:36 +00:00
let expr = s:CleanupExpr(@v)
call setpos('.', pos)
call setreg('v', reg, regt)
else
2021-11-21 21:13:36 +00:00
" no evaluation provided: get from C-expression under cursor
" TODO: allow filetype specific lookup #9057
let expr = expand('<cexpr>')
endif
2021-11-21 21:13:36 +00:00
return expr
endfunc
2022-01-08 21:51:59 +00:00
" clean up expression that may get in because of range
2021-11-21 21:13:36 +00:00
" (newlines and surrounding whitespace)
2021-11-27 10:57:26 +00:00
" As it can also be specified via ex-command for assignments this function
2022-01-08 21:51:59 +00:00
" may not change the "content" parts (like replacing contained spaces)
2021-11-21 21:13:36 +00:00
func s:CleanupExpr(expr)
" replace all embedded newlines/tabs/...
2021-11-27 10:57:26 +00:00
let expr = substitute(a:expr, '\_s', ' ', 'g')
2021-11-21 21:13:36 +00:00
if &filetype ==# 'cobol'
2021-11-27 10:57:26 +00:00
" extra cleanup for COBOL:
" - a semicolon nmay be used instead of a space
" - a trailing comma or period is ignored as it commonly separates/ends
" multiple expr
2021-11-21 21:13:36 +00:00
let expr = substitute(expr, ';', ' ', 'g')
2021-11-27 10:57:26 +00:00
let expr = substitute(expr, '[,.]\+ *$', '', '')
2021-11-21 21:13:36 +00:00
endif
2021-11-27 10:57:26 +00:00
" get rid of leading and trailing spaces
let expr = substitute(expr, '^ *', '', '')
let expr = substitute(expr, ' *$', '', '')
2021-11-21 21:13:36 +00:00
return expr
endfunc
let s:ignoreEvalError = 0
let s:evalFromBalloonExpr = 0
" Handle the result of data-evaluate-expression
func s:HandleEvaluate(msg)
2021-12-16 14:41:10 +00:00
let value = a:msg
\ ->substitute('.*value="\(.*\)"', '\1', '')
\ ->substitute('\\"', '"', 'g')
\ ->substitute('\\\\', '\\', 'g')
2022-01-08 21:51:59 +00:00
"\ multi-byte characters arrive in octal form, replace everything but NULL values
2021-12-16 14:41:10 +00:00
\ ->substitute('\\000', s:NullRepl, 'g')
\ ->substitute('\\\o\o\o', {-> eval('"' .. submatch(0) .. '"')}, 'g')
"\ Note: GDB docs also mention hex encodings - the translations below work
"\ but we keep them out for performance-reasons until we actually see
"\ those in mi-returns
"\ ->substitute('\\0x00', s:NullRep, 'g')
"\ ->substitute('\\0x\(\x\x\)', {-> eval('"\x' .. submatch(1) .. '"')}, 'g')
\ ->substitute(s:NullRepl, '\\000', 'g')
if s:evalFromBalloonExpr
if s:evalFromBalloonExprResult == ''
let s:evalFromBalloonExprResult = s:evalexpr . ': ' . value
else
let s:evalFromBalloonExprResult .= ' = ' . value
endif
call balloon_show(s:evalFromBalloonExprResult)
else
echomsg '"' . s:evalexpr . '": ' . value
endif
2017-11-11 20:58:53 +01:00
if s:evalexpr[0] != '*' && value =~ '^0x' && value != '0x0' && value !~ '"$'
" Looks like a pointer, also display what it points to.
let s:ignoreEvalError = 1
call s:SendEval('*' . s:evalexpr)
else
let s:evalFromBalloonExpr = 0
endif
endfunc
" Show a balloon with information of the variable under the mouse pointer,
" if there is any.
func TermDebugBalloonExpr()
if v:beval_winid != s:sourcewin
2019-07-13 23:04:31 +02:00
return ''
endif
if !s:stopped
" Only evaluate when stopped, otherwise setting a breakpoint using the
" mouse triggers a balloon.
2019-07-13 23:04:31 +02:00
return ''
endif
let s:evalFromBalloonExpr = 1
let s:evalFromBalloonExprResult = ''
let s:ignoreEvalError = 1
2021-11-21 21:13:36 +00:00
let expr = s:CleanupExpr(v:beval_text)
call s:SendEval(expr)
return ''
endfunc
" Handle an error.
func s:HandleError(msg)
if s:ignoreEvalError
" Result of s:SendEval() failed, ignore.
let s:ignoreEvalError = 0
let s:evalFromBalloonExpr = 0
return
endif
2021-11-07 20:27:04 +00:00
let msgVal = substitute(a:msg, '.*msg="\(.*\)"', '\1', '')
echoerr substitute(msgVal, '\\"', '"', 'g')
endfunc
func s:GotoSourcewinOrCreateIt()
if !win_gotoid(s:sourcewin)
new
let s:sourcewin = win_getid(winnr())
call s:InstallWinbar()
endif
endfunc
func s:GetDisasmWindow()
if exists('g:termdebug_config')
return get(g:termdebug_config, 'disasm_window', 0)
endif
if exists('g:termdebug_disasm_window')
return g:termdebug_disasm_window
endif
return 0
endfunc
func s:GetDisasmWindowHeight()
if exists('g:termdebug_config')
return get(g:termdebug_config, 'disasm_window_height', 0)
endif
if exists('g:termdebug_disasm_window') && g:termdebug_disasm_window > 1
return g:termdebug_disasm_window
endif
return 0
endfunc
2021-01-11 19:40:15 +01:00
func s:GotoAsmwinOrCreateIt()
if !win_gotoid(s:asmwin)
if win_gotoid(s:sourcewin)
exe 'rightbelow new'
else
exe 'new'
endif
let s:asmwin = win_getid(winnr())
setlocal nowrap
setlocal number
setlocal noswapfile
setlocal buftype=nofile
2021-11-16 19:18:26 +00:00
setlocal modifiable
2021-01-11 19:40:15 +01:00
let asmbuf = bufnr('Termdebug-asm-listing')
if asmbuf > 0
exe 'buffer' . asmbuf
else
exe 'file Termdebug-asm-listing'
endif
if s:GetDisasmWindowHeight() > 0
exe 'resize ' .. s:GetDisasmWindowHeight()
2021-01-11 19:40:15 +01:00
endif
endif
if s:asm_addr != ''
let lnum = search('^' . s:asm_addr)
if lnum == 0
if s:stopped
2022-03-19 15:18:53 +00:00
call s:SendCommand('disassemble $pc')
2021-01-11 19:40:15 +01:00
endif
else
2022-04-18 15:36:40 +01:00
call sign_unplace('TermDebug', #{id: s:asm_id})
call sign_place(s:asm_id, 'TermDebug', 'debugPC', '%', #{lnum: lnum})
2021-01-11 19:40:15 +01:00
endif
endif
endfunc
" 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 a:msg =~ '^\*stopped'
call ch_log('program stopped')
let s:stopped = 1
elseif a:msg =~ '^\*running'
call ch_log('program running')
let s:stopped = 0
endif
if a:msg =~ 'fullname='
let fname = s:GetFullname(a:msg)
else
let fname = ''
endif
2021-01-11 19:40:15 +01:00
if a:msg =~ 'addr='
let asm_addr = s:GetAsmAddr(a:msg)
if asm_addr != ''
let s:asm_addr = asm_addr
let curwinid = win_getid(winnr())
if win_gotoid(s:asmwin)
2022-03-19 15:18:53 +00:00
let lnum = search('^' . s:asm_addr)
if lnum == 0
call s:SendCommand('disassemble $pc')
else
2022-04-18 15:36:40 +01:00
call sign_unplace('TermDebug', #{id: s:asm_id})
call sign_place(s:asm_id, 'TermDebug', 'debugPC', '%', #{lnum: lnum})
2022-03-19 15:18:53 +00:00
endif
call win_gotoid(curwinid)
2021-01-11 19:40:15 +01:00
endif
endif
endif
if a:msg =~ '^\(\*stopped\|=thread-selected\)' && filereadable(fname)
let lnum = substitute(a:msg, '.*line="\([^"]*\)".*', '\1', '')
if lnum =~ '^[0-9]*$'
2022-01-16 14:46:06 +00:00
call s:GotoSourcewinOrCreateIt()
if expand('%:p') != fnamemodify(fname, ':p')
2022-01-23 12:07:04 +00:00
echomsg 'different fname: "' .. expand('%:p') .. '" vs "' .. fnamemodify(fname, ':p') .. '"'
2022-01-16 14:46:06 +00:00
augroup Termdebug
" Always open a file read-only instead of showing the ATTENTION
2022-06-23 13:04:20 +01:00
" prompt, since it is unlikely we want to edit the file.
2022-01-16 14:46:06 +00:00
" The file may be changed but not saved, warn for that.
au SwapExists * echohl WarningMsg
\ | echo 'Warning: file is being edited elsewhere'
\ | echohl None
2022-06-23 13:04:20 +01:00
\ | let v:swapchoice = 'o'
2022-03-19 15:18:53 +00:00
augroup END
if &modified
" TODO: find existing window
exe 'split ' . fnameescape(fname)
let s:sourcewin = win_getid(winnr())
call s:InstallWinbar()
else
exe 'edit ' . fnameescape(fname)
endif
augroup Termdebug
au! SwapExists
augroup END
endif
exe lnum
2021-11-16 19:18:26 +00:00
normal! zv
2022-04-18 15:36:40 +01:00
call sign_unplace('TermDebug', #{id: s:pc_id})
call sign_place(s:pc_id, 'TermDebug', 'debugPC', fname,
\ #{lnum: lnum, priority: 110})
2020-10-26 21:12:46 +01:00
if !exists('b:save_signcolumn')
2022-03-19 15:18:53 +00:00
let b:save_signcolumn = &signcolumn
call add(s:signcolumn_buflist, bufnr())
2020-10-26 21:12:46 +01:00
endif
setlocal signcolumn=yes
endif
elseif !s:stopped || fname != ''
2022-04-18 15:36:40 +01:00
call sign_unplace('TermDebug', #{id: s:pc_id})
endif
call win_gotoid(wid)
endfunc
let s:BreakpointSigns = []
2021-11-27 10:57:26 +00:00
func s:CreateBreakpoint(id, subid, enabled)
let nr = printf('%d.%d', a:id, a:subid)
if index(s:BreakpointSigns, nr) == -1
call add(s:BreakpointSigns, nr)
2021-11-27 10:57:26 +00:00
if a:enabled == "n"
let hiName = "debugBreakpointDisabled"
else
let hiName = "debugBreakpoint"
endif
2022-04-18 15:36:40 +01:00
call sign_define('debugBreakpoint' .. nr,
\ #{text: substitute(nr, '\..*', '', ''),
\ texthl: hiName})
endif
endfunc
func! s:SplitMsg(s)
return split(a:s, '{.\{-}}\zs')
endfunction
" Handle setting a breakpoint
" Will update the sign that shows the breakpoint
2021-11-27 10:57:26 +00:00
func s:HandleNewBreakpoint(msg, modifiedFlag)
if a:msg !~ 'fullname='
2021-11-21 21:13:36 +00:00
" a watch or a pending breakpoint does not have a file name
if a:msg =~ 'pending='
let nr = substitute(a:msg, '.*number=\"\([0-9.]*\)\".*', '\1', '')
let target = substitute(a:msg, '.*pending=\"\([^"]*\)\".*', '\1', '')
echomsg 'Breakpoint ' . nr . ' (' . target . ') pending.'
endif
return
endif
for msg in s:SplitMsg(a:msg)
let fname = s:GetFullname(msg)
if empty(fname)
continue
endif
let nr = substitute(msg, '.*number="\([0-9.]*\)\".*', '\1', '')
if empty(nr)
return
endif
" If "nr" is 123 it becomes "123.0" and subid is "0".
" If "nr" is 123.4 it becomes "123.4.0" and subid is "4"; "0" is discarded.
let [id, subid; _] = map(split(nr . '.0', '\.'), 'v:val + 0')
2021-11-27 10:57:26 +00:00
let enabled = substitute(msg, '.*enabled="\([yn]\)".*', '\1', '')
call s:CreateBreakpoint(id, subid, enabled)
if has_key(s:breakpoints, id)
let entries = s:breakpoints[id]
else
let entries = {}
let s:breakpoints[id] = entries
endif
if has_key(entries, subid)
let entry = entries[subid]
else
let entry = {}
let entries[subid] = entry
endif
let lnum = substitute(msg, '.*line="\([^"]*\)".*', '\1', '')
let entry['fname'] = fname
let entry['lnum'] = lnum
let bploc = printf('%s:%d', fname, lnum)
if !has_key(s:breakpoint_locations, bploc)
let s:breakpoint_locations[bploc] = []
endif
let s:breakpoint_locations[bploc] += [id]
if bufloaded(fname)
call s:PlaceSign(id, subid, entry)
2021-11-27 10:57:26 +00:00
let posMsg = ' at line ' . lnum . '.'
else
let posMsg = ' in ' . fname . ' at line ' . lnum . '.'
endif
if !a:modifiedFlag
let actionTaken = 'created'
elseif enabled == 'n'
let actionTaken = 'disabled'
else
let actionTaken = 'enabled'
endif
2021-11-27 10:57:26 +00:00
echomsg 'Breakpoint ' . nr . ' ' . actionTaken . posMsg
endfor
endfunc
func s:PlaceSign(id, subid, entry)
let nr = printf('%d.%d', a:id, a:subid)
2022-04-18 15:36:40 +01:00
call sign_place(s:Breakpoint2SignNumber(a:id, a:subid), 'TermDebug',
\ 'debugBreakpoint' .. nr, a:entry['fname'],
\ #{lnum: a:entry['lnum'], priority: 110})
let a:entry['placed'] = 1
endfunc
" Handle deleting a breakpoint
" Will remove the sign that shows the breakpoint
func s:HandleBreakpointDelete(msg)
let id = substitute(a:msg, '.*id="\([0-9]*\)\".*', '\1', '') + 0
if empty(id)
return
endif
if has_key(s:breakpoints, id)
for [subid, entry] in items(s:breakpoints[id])
if has_key(entry, 'placed')
2022-04-18 15:36:40 +01:00
call sign_unplace('TermDebug',
\ #{id: s:Breakpoint2SignNumber(id, subid)})
2022-03-19 15:18:53 +00:00
unlet entry['placed']
endif
endfor
unlet s:breakpoints[id]
2021-11-21 21:13:36 +00:00
echomsg 'Breakpoint ' . id . ' cleared.'
endif
2017-08-27 16:52:01 +02:00
endfunc
" Handle the debugged program starting to run.
" Will store the process ID in s:pid
func s:HandleProgramRun(msg)
let nr = substitute(a:msg, '.*pid="\([0-9]*\)\".*', '\1', '') + 0
if nr == 0
return
endif
let s:pid = nr
call ch_log('Detected process ID: ' . s:pid)
endfunc
" Handle a BufRead autocommand event: place any signs.
func s:BufRead()
let fname = expand('<afile>:p')
for [id, entries] in items(s:breakpoints)
for [subid, entry] in items(entries)
if entry['fname'] == fname
2022-03-19 15:18:53 +00:00
call s:PlaceSign(id, subid, entry)
endif
endfor
endfor
endfunc
" Handle a BufUnloaded autocommand event: unplace any signs.
func s:BufUnloaded()
let fname = expand('<afile>:p')
for [id, entries] in items(s:breakpoints)
for [subid, entry] in items(entries)
if entry['fname'] == fname
2022-03-19 15:18:53 +00:00
let entry['placed'] = 0
endif
endfor
endfor
endfunc
let &cpo = s:keepcpo
unlet s:keepcpo