patch 8.1.0575: Termdebug: clearing multi-breakpoint does not work

Problem:    Termdebug: clearing multi-breakpoint does not work.
Solution:   Delete all X.Y breakpoints.  Keep more information about placed
            breakpoints. (Ozaki Kiichi, closes #3641)
This commit is contained in:
Bram Moolenaar
2018-12-09 15:53:01 +01:00
parent 4af7259b2b
commit 37402ed534
2 changed files with 94 additions and 48 deletions

View File

@ -74,10 +74,10 @@ let s:break_id = 13 " breakpoint number is added to this
let s:stopped = 1 let s:stopped = 1
" Take a breakpoint number as used by GDB and turn it into an integer. " Take a breakpoint number as used by GDB and turn it into an integer.
" The breakpoint may contain a dot: 123.4 " The breakpoint may contain a dot: 123.4 -> 123004
func s:Breakpoint2SignNumber(nr) " The main breakpoint has a zero subid.
let t = split(a:nr, '\.') func s:Breakpoint2SignNumber(id, subid)
return t[0] * 1000 + (len(t) == 2 ? t[1] : 0) return s:break_id + a:id * 1000 + a:subid
endfunction endfunction
func s:Highlight(init, old, new) func s:Highlight(init, old, new)
@ -362,8 +362,17 @@ func s:StartDebugCommon(dict)
" Contains breakpoints that have been placed, key is a string with the GDB " Contains breakpoints that have been placed, key is a string with the GDB
" breakpoint number. " 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 = {} 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 augroup TermDebug
au BufRead * call s:BufRead() au BufRead * call s:BufRead()
au BufUnload * call s:BufUnloaded() au BufUnload * call s:BufUnloaded()
@ -683,10 +692,13 @@ func s:DeleteCommands()
endif endif
exe 'sign unplace ' . s:pc_id exe 'sign unplace ' . s:pc_id
for key in keys(s:breakpoints) for [id, entries] in items(s:breakpoints)
exe 'sign unplace ' . (s:break_id + s:Breakpoint2SignNumber(key)) for subid in keys(entries)
exe 'sign unplace ' . s:Breakpoint2SignNumber(id, subid)
endfor
endfor endfor
unlet s:breakpoints unlet s:breakpoints
unlet s:breakpoint_locations
sign undefine debugPC sign undefine debugPC
for val in s:BreakpointSigns for val in s:BreakpointSigns
@ -721,15 +733,27 @@ endfunc
func s:ClearBreakpoint() func s:ClearBreakpoint()
let fname = fnameescape(expand('%:p')) let fname = fnameescape(expand('%:p'))
let lnum = line('.') let lnum = line('.')
for [key, val] in items(s:breakpoints) let bploc = printf('%s:%d', fname, lnum)
if val['fname'] == fname && val['lnum'] == lnum if has_key(s:breakpoint_locations, bploc)
call s:SendCommand('-break-delete ' . key) let idx = 0
" Assume this always wors, the reply is simply "^done". for id in s:breakpoint_locations[bploc]
exe 'sign unplace ' . (s:break_id + s:Breakpoint2SignNumber(key)) if has_key(s:breakpoints, id)
unlet s:breakpoints[key] " Assume this always works, the reply is simply "^done".
break call s:SendCommand('-break-delete ' . id)
for subid in keys(s:breakpoints[id])
exe 'sign unplace ' . s:Breakpoint2SignNumber(id, subid)
endfor
unlet s:breakpoints[id]
unlet s:breakpoint_locations[bploc][idx]
break
else
let idx += 1
endif
endfor
if empty(s:breakpoint_locations[bploc])
unlet s:breakpoint_locations[bploc]
endif endif
endfor endif
endfunc endfunc
func s:Run(args) func s:Run(args)
@ -873,15 +897,16 @@ endfunc
let s:BreakpointSigns = [] let s:BreakpointSigns = []
func s:CreateBreakpoint(nr) func s:CreateBreakpoint(id, subid)
if index(s:BreakpointSigns, a:nr) == -1 let nr = printf('%d.%d', a:id, a:subid)
call add(s:BreakpointSigns, a:nr) if index(s:BreakpointSigns, nr) == -1
exe "sign define debugBreakpoint" . a:nr . " text=" . substitute(a:nr, '\..*', '', '') . " texthl=debugBreakpoint" call add(s:BreakpointSigns, nr)
exe "sign define debugBreakpoint" . nr . " text=" . substitute(nr, '\..*', '', '') . " texthl=debugBreakpoint"
endif endif
endfunc endfunc
func s:SplitMsg(s) func! s:SplitMsg(s)
return split(a:s, '{\%([a-z-]\+=[^,]\+,*\)\+}\zs') return split(a:s, '{.\{-}}\zs')
endfunction endfunction
" Handle setting a breakpoint " Handle setting a breakpoint
@ -900,48 +925,63 @@ func s:HandleNewBreakpoint(msg)
if empty(nr) if empty(nr)
return return
endif endif
call s:CreateBreakpoint(nr)
if has_key(s:breakpoints, nr) " If "nr" is 123 it becomes "123.0" and subid is "0".
let entry = s:breakpoints[nr] " 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')
call s:CreateBreakpoint(id, subid)
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 else
let entry = {} let entry = {}
let s:breakpoints[nr] = entry let entries[subid] = entry
endif endif
let lnum = substitute(msg, '.*line="\([^"]*\)".*', '\1', '') let lnum = substitute(msg, '.*line="\([^"]*\)".*', '\1', '')
let entry['fname'] = fname let entry['fname'] = fname
let entry['lnum'] = lnum 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) if bufloaded(fname)
call s:PlaceSign(nr, entry) call s:PlaceSign(id, subid, entry)
endif endif
endfor endfor
endfunc endfunc
func s:PlaceSign(nr, entry) func s:PlaceSign(id, subid, entry)
exe 'sign place ' . (s:break_id + s:Breakpoint2SignNumber(a:nr)) . ' line=' . a:entry['lnum'] . ' name=debugBreakpoint' . a:nr . ' file=' . a:entry['fname'] let nr = printf('%d.%d', a:id, a:subid)
exe 'sign place ' . s:Breakpoint2SignNumber(a:id, a:subid) . ' line=' . a:entry['lnum'] . ' name=debugBreakpoint' . nr . ' file=' . a:entry['fname']
let a:entry['placed'] = 1 let a:entry['placed'] = 1
endfunc endfunc
" Handle deleting a breakpoint " Handle deleting a breakpoint
" Will remove the sign that shows the breakpoint " Will remove the sign that shows the breakpoint
func s:HandleBreakpointDelete(msg) func s:HandleBreakpointDelete(msg)
let key = substitute(a:msg, '.*id="\([0-9.]*\)\".*', '\1', '') let id = substitute(a:msg, '.*id="\([0-9]*\)\".*', '\1', '') + 0
if empty(key) if empty(id)
return return
endif endif
for [nr, entry] in items(s:breakpoints) if has_key(s:breakpoints, id)
if stridx(nr, key) != 0 for [subid, entry] in items(s:breakpoints[id])
continue if has_key(entry, 'placed')
endif exe 'sign unplace ' . s:Breakpoint2SignNumber(id, subid)
let entry = s:breakpoints[nr] unlet entry['placed']
if has_key(entry, 'placed') endif
exe 'sign unplace ' . (s:break_id + s:Breakpoint2SignNumber(nr)) endfor
unlet entry['placed'] unlet s:breakpoints[id]
endif endif
unlet s:breakpoints[nr]
endfor
endfunc endfunc
" Handle the debugged program starting to run. " Handle the debugged program starting to run.
@ -958,20 +998,24 @@ endfunc
" Handle a BufRead autocommand event: place any signs. " Handle a BufRead autocommand event: place any signs.
func s:BufRead() func s:BufRead()
let fname = expand('<afile>:p') let fname = expand('<afile>:p')
for [nr, entry] in items(s:breakpoints) for [id, entries] in items(s:breakpoints)
if entry['fname'] == fname for [subid, entry] in items(entries)
call s:PlaceSign(nr, entry) if entry['fname'] == fname
endif call s:PlaceSign(id, subid, entry)
endif
endfor
endfor endfor
endfunc endfunc
" Handle a BufUnloaded autocommand event: unplace any signs. " Handle a BufUnloaded autocommand event: unplace any signs.
func s:BufUnloaded() func s:BufUnloaded()
let fname = expand('<afile>:p') let fname = expand('<afile>:p')
for [nr, entry] in items(s:breakpoints) for [id, entries] in items(s:breakpoints)
if entry['fname'] == fname for [subid, entry] in items(entries)
let entry['placed'] = 0 if entry['fname'] == fname
endif let entry['placed'] = 0
endif
endfor
endfor endfor
endfunc endfunc

View File

@ -792,6 +792,8 @@ static char *(features[]) =
static int included_patches[] = static int included_patches[] =
{ /* Add new patch number below this line */ { /* Add new patch number below this line */
/**/
575,
/**/ /**/
574, 574,
/**/ /**/