Skip to content

Commit 3dcc5bc

Browse files
committed
proc,service: change FindLocation to work with multiple targets
Changes FindLocation to support multiple targets and adds an AddrPid member to api.Breakpoint so that clients can set breakpoints by address when multiple targets are connected (but at them moment this field is ignored). Updates go-delve#1653 Updates go-delve#2551
1 parent db9770f commit 3dcc5bc

File tree

7 files changed

+114
-62
lines changed

7 files changed

+114
-62
lines changed

pkg/locspec/locations.go

+8-8
Original file line numberDiff line numberDiff line change
@@ -370,7 +370,7 @@ func (ale AmbiguousLocationError) Error() string {
370370
func (loc *NormalLocationSpec) Find(t *proc.Target, processArgs []string, scope *proc.EvalScope, locStr string, includeNonExecutableLines bool, substitutePathRules [][2]string) ([]api.Location, error) {
371371
limit := maxFindLocationCandidates
372372
var candidateFiles []string
373-
for _, sourceFile := range scope.BinInfo.Sources {
373+
for _, sourceFile := range t.BinInfo().Sources {
374374
substFile := sourceFile
375375
if len(substitutePathRules) > 0 {
376376
substFile = SubstitutePath(sourceFile, substitutePathRules)
@@ -387,10 +387,10 @@ func (loc *NormalLocationSpec) Find(t *proc.Target, processArgs []string, scope
387387

388388
var candidateFuncs []string
389389
if loc.FuncBase != nil && limit > 0 {
390-
candidateFuncs = loc.findFuncCandidates(scope, limit)
390+
candidateFuncs = loc.findFuncCandidates(t.BinInfo(), limit)
391391
}
392392

393-
if matching := len(candidateFiles) + len(candidateFuncs); matching == 0 {
393+
if matching := len(candidateFiles) + len(candidateFuncs); matching == 0 && scope != nil {
394394
// if no result was found this locations string could be an
395395
// expression that the user forgot to prefix with '*', try treating it as
396396
// such.
@@ -428,26 +428,26 @@ func (loc *NormalLocationSpec) Find(t *proc.Target, processArgs []string, scope
428428
return []api.Location{addressesToLocation(addrs)}, nil
429429
}
430430

431-
func (loc *NormalLocationSpec) findFuncCandidates(scope *proc.EvalScope, limit int) []string {
431+
func (loc *NormalLocationSpec) findFuncCandidates(bi *proc.BinaryInfo, limit int) []string {
432432
candidateFuncs := map[string]struct{}{}
433433
// See if it matches generic functions first
434-
for fname := range scope.BinInfo.LookupGenericFunc() {
434+
for fname := range bi.LookupGenericFunc() {
435435
if len(candidateFuncs) >= limit {
436436
break
437437
}
438-
if !loc.FuncBase.Match(&proc.Function{Name: fname}, scope.BinInfo.PackageMap) {
438+
if !loc.FuncBase.Match(&proc.Function{Name: fname}, bi.PackageMap) {
439439
continue
440440
}
441441
if loc.Base == fname {
442442
return []string{fname}
443443
}
444444
candidateFuncs[fname] = struct{}{}
445445
}
446-
for _, f := range scope.BinInfo.LookupFunc {
446+
for _, f := range bi.LookupFunc {
447447
if len(candidateFuncs) >= limit {
448448
break
449449
}
450-
if !loc.FuncBase.Match(f, scope.BinInfo.PackageMap) {
450+
if !loc.FuncBase.Match(f, bi.PackageMap) {
451451
continue
452452
}
453453
if loc.Base == f.Name {

pkg/proc/target_group.go

+27-2
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,8 @@ func NewGroup(t *Target) *TargetGroup {
4444
}
4545
}
4646

47-
// Targets returns a slice of targets in the group.
47+
// Targets returns a slice of all targets in the group, including the
48+
// ones that are no longer valid.
4849
func (grp *TargetGroup) Targets() []*Target {
4950
return grp.targets
5051
}
@@ -57,7 +58,9 @@ func (grp *TargetGroup) Valid() (bool, error) {
5758
if ok {
5859
return true, nil
5960
}
60-
err0 = err
61+
if err0 == nil {
62+
err0 = err
63+
}
6164
}
6265
return false, err0
6366
}
@@ -121,3 +124,25 @@ func (grp *TargetGroup) TargetForThread(thread Thread) *Target {
121124
}
122125
return nil
123126
}
127+
128+
// ValidTargets iterates through all valid targets in Group.
129+
type ValidTargets struct {
130+
*Target
131+
Group *TargetGroup
132+
start int
133+
}
134+
135+
// Next moves to the next valid target, returns false if there aren't more
136+
// valid targets in the group.
137+
func (it *ValidTargets) Next() bool {
138+
for i := it.start; i < len(it.Group.targets); i++ {
139+
if ok, _ := it.Group.targets[i].Valid(); ok {
140+
it.Target = it.Group.targets[i]
141+
it.start = i + 1
142+
return true
143+
}
144+
}
145+
it.start = len(it.Group.targets)
146+
it.Target = nil
147+
return false
148+
}

pkg/terminal/command.go

+1
Original file line numberDiff line numberDiff line change
@@ -1749,6 +1749,7 @@ func setBreakpoint(t *Term, ctx callContext, tracepoint bool, argstr string) ([]
17491749
for _, loc := range locs {
17501750
requestedBp.Addr = loc.PC
17511751
requestedBp.Addrs = loc.PCs
1752+
requestedBp.AddrPid = loc.PCPids
17521753
if tracepoint {
17531754
requestedBp.LoadArgs = &ShortLoadConfig
17541755
}

service/api/conversions.go

+3-2
Original file line numberDiff line numberDiff line change
@@ -54,7 +54,7 @@ func ConvertLogicalBreakpoint(lbp *proc.LogicalBreakpoint) *Breakpoint {
5454
}
5555

5656
// ConvertPhysicalBreakpoints adds informations from physical breakpoints to an API breakpoint.
57-
func ConvertPhysicalBreakpoints(b *Breakpoint, bps []*proc.Breakpoint) {
57+
func ConvertPhysicalBreakpoints(b *Breakpoint, pids []int, bps []*proc.Breakpoint) {
5858
if len(bps) == 0 {
5959
return
6060
}
@@ -63,8 +63,9 @@ func ConvertPhysicalBreakpoints(b *Breakpoint, bps []*proc.Breakpoint) {
6363
b.WatchType = WatchType(bps[0].WatchType)
6464

6565
lg := false
66-
for _, bp := range bps {
66+
for i, bp := range bps {
6767
b.Addrs = append(b.Addrs, bp.Addr)
68+
b.AddrPid = append(b.AddrPid, pids[i])
6869
if b.FunctionName != bp.FunctionName && b.FunctionName != "" {
6970
if !lg {
7071
b.FunctionName = removeTypeParams(b.FunctionName)

service/api/types.go

+4
Original file line numberDiff line numberDiff line change
@@ -80,6 +80,9 @@ type Breakpoint struct {
8080
Addr uint64 `json:"addr"`
8181
// Addrs is the list of addresses for this breakpoint.
8282
Addrs []uint64 `json:"addrs"`
83+
// AddrPid[i] is the PID associated with by Addrs[i], when debugging a
84+
// single target process this is optional, otherwise it is mandatory.
85+
AddrPid []int `json:"addrpid"`
8386
// File is the source file for the breakpoint.
8487
File string `json:"file"`
8588
// Line is a line in File for the breakpoint.
@@ -192,6 +195,7 @@ type Location struct {
192195
Line int `json:"line"`
193196
Function *Function `json:"function,omitempty"`
194197
PCs []uint64 `json:"pcs,omitempty"`
198+
PCPids []int `json:"pcpids,omitempty"`
195199
}
196200

197201
// Stackframe describes one frame in a stack trace.

service/dap/server.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -1860,7 +1860,7 @@ func (s *Session) stoppedOnBreakpointGoroutineID(state *api.DebuggerState) (int,
18601860
return goid, nil
18611861
}
18621862
abp := api.ConvertLogicalBreakpoint(bp.Breakpoint.Logical)
1863-
api.ConvertPhysicalBreakpoints(abp, []*proc.Breakpoint{bp.Breakpoint})
1863+
api.ConvertPhysicalBreakpoints(abp, []int{0}, []*proc.Breakpoint{bp.Breakpoint})
18641864
return goid, abp
18651865
}
18661866

service/debugger/debugger.go

+70-49
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,9 @@ var (
4545

4646
// ErrCoreDumpNotSupported is returned when core dumping is not supported
4747
ErrCoreDumpNotSupported = errors.New("core dumping not supported")
48+
49+
// ErrNotImplementedWithMultitarget is returned for operations that are not implemented with multiple targets
50+
ErrNotImplementedWithMultitarget = errors.New("not implemented for multiple targets")
4851
)
4952

5053
// Debugger service.
@@ -386,6 +389,10 @@ func (d *Debugger) FunctionReturnLocations(fnName string) ([]uint64, error) {
386389
d.targetMutex.Lock()
387390
defer d.targetMutex.Unlock()
388391

392+
if len(d.target.Targets()) > 1 {
393+
return nil, ErrNotImplementedWithMultitarget
394+
}
395+
389396
var (
390397
p = d.target.Selected
391398
g = p.SelectedGoroutine()
@@ -604,10 +611,11 @@ func (d *Debugger) state(retLoadCfg *proc.LoadConfig, withBreakpointInfo bool) (
604611
state.When, _ = d.target.When()
605612
}
606613

607-
for _, t := range d.target.Targets() {
614+
t := proc.ValidTargets{Group: d.target}
615+
for t.Next() {
608616
for _, bp := range t.Breakpoints().WatchOutOfScope {
609617
abp := api.ConvertLogicalBreakpoint(bp.Logical)
610-
api.ConvertPhysicalBreakpoints(abp, []*proc.Breakpoint{bp})
618+
api.ConvertPhysicalBreakpoints(abp, []int{t.Pid()}, []*proc.Breakpoint{bp})
611619
state.WatchOutOfScope = append(state.WatchOutOfScope, abp)
612620
}
613621
}
@@ -674,8 +682,9 @@ func (d *Debugger) CreateBreakpoint(requestedBp *api.Breakpoint) (*api.Breakpoin
674682
if runtime.GOOS == "windows" {
675683
// Accept fileName which is case-insensitive and slash-insensitive match
676684
fileNameNormalized := strings.ToLower(filepath.ToSlash(fileName))
685+
t := proc.ValidTargets{Group: d.target}
677686
caseInsensitiveSearch:
678-
for _, t := range d.target.Targets() {
687+
for t.Next() {
679688
for _, symFile := range t.BinInfo().Sources {
680689
if fileNameNormalized == strings.ToLower(filepath.ToSlash(symFile)) {
681690
fileName = symFile
@@ -689,6 +698,7 @@ func (d *Debugger) CreateBreakpoint(requestedBp *api.Breakpoint) (*api.Breakpoin
689698
addrs, err = proc.FindFunctionLocation(d.target.Selected, requestedBp.FunctionName, requestedBp.Line)
690699
case len(requestedBp.Addrs) > 0:
691700
addrs = requestedBp.Addrs
701+
//TODO(aarzilli): read requestedBp.AddrPid
692702
default:
693703
addrs = []uint64{requestedBp.Addr}
694704
}
@@ -707,7 +717,18 @@ func (d *Debugger) CreateBreakpoint(requestedBp *api.Breakpoint) (*api.Breakpoin
707717

708718
func (d *Debugger) convertBreakpoint(lbp *proc.LogicalBreakpoint) *api.Breakpoint {
709719
abp := api.ConvertLogicalBreakpoint(lbp)
710-
api.ConvertPhysicalBreakpoints(abp, d.findBreakpoint(lbp.LogicalID))
720+
bps := []*proc.Breakpoint{}
721+
pids := []int{}
722+
t := proc.ValidTargets{Group: d.target}
723+
for t.Next() {
724+
for _, bp := range t.Breakpoints().M {
725+
if bp.LogicalID() == lbp.LogicalID {
726+
bps = append(bps, bp)
727+
pids = append(pids, t.Pid())
728+
}
729+
}
730+
}
731+
api.ConvertPhysicalBreakpoints(abp, pids, bps)
711732
return abp
712733
}
713734

@@ -846,7 +867,7 @@ func (d *Debugger) CreateEBPFTracepoint(fnName string) error {
846867
d.targetMutex.Lock()
847868
defer d.targetMutex.Unlock()
848869
if len(d.target.Targets()) != 1 {
849-
panic("multiple targets not implemented")
870+
return ErrNotImplementedWithMultitarget
850871
}
851872
p := d.target.Selected
852873
return p.SetEBPFTracepoint(fnName)
@@ -1045,23 +1066,22 @@ func isBpHitCondNotSatisfiable(bp *api.Breakpoint) bool {
10451066
func (d *Debugger) Breakpoints(all bool) []*api.Breakpoint {
10461067
d.targetMutex.Lock()
10471068
defer d.targetMutex.Unlock()
1048-
if len(d.target.Targets()) != 1 {
1049-
panic("multiple targets not implemented")
1050-
}
1051-
p := d.target.Selected
10521069

10531070
abps := []*api.Breakpoint{}
10541071
if all {
1055-
for _, bp := range p.Breakpoints().M {
1056-
var abp *api.Breakpoint
1057-
if bp.Logical != nil {
1058-
abp = api.ConvertLogicalBreakpoint(bp.Logical)
1059-
} else {
1060-
abp = &api.Breakpoint{}
1072+
t := proc.ValidTargets{Group: d.target}
1073+
for t.Next() {
1074+
for _, bp := range t.Breakpoints().M {
1075+
var abp *api.Breakpoint
1076+
if bp.Logical != nil {
1077+
abp = api.ConvertLogicalBreakpoint(bp.Logical)
1078+
} else {
1079+
abp = &api.Breakpoint{}
1080+
}
1081+
api.ConvertPhysicalBreakpoints(abp, []int{t.Pid()}, []*proc.Breakpoint{bp})
1082+
abp.VerboseDescr = bp.VerboseDescr()
1083+
abps = append(abps, abp)
10611084
}
1062-
api.ConvertPhysicalBreakpoints(abp, []*proc.Breakpoint{bp})
1063-
abp.VerboseDescr = bp.VerboseDescr()
1064-
abps = append(abps, abp)
10651085
}
10661086
} else {
10671087
for _, lbp := range d.target.LogicalBreakpoints {
@@ -1422,7 +1442,8 @@ func (d *Debugger) Sources(filter string) ([]string, error) {
14221442
}
14231443

14241444
files := []string{}
1425-
for _, t := range d.target.Targets() {
1445+
t := proc.ValidTargets{Group: d.target}
1446+
for t.Next() {
14261447
for _, f := range t.BinInfo().Sources {
14271448
if regex.Match([]byte(f)) {
14281449
files = append(files, f)
@@ -1460,7 +1481,8 @@ func (d *Debugger) Functions(filter string) ([]string, error) {
14601481
}
14611482

14621483
funcs := []string{}
1463-
for _, t := range d.target.Targets() {
1484+
t := proc.ValidTargets{Group: d.target}
1485+
for t.Next() {
14641486
for _, f := range t.BinInfo().Functions {
14651487
if regex.MatchString(f.Name) {
14661488
funcs = append(funcs, f.Name)
@@ -1484,7 +1506,8 @@ func (d *Debugger) Types(filter string) ([]string, error) {
14841506

14851507
r := []string{}
14861508

1487-
for _, t := range d.target.Targets() {
1509+
t := proc.ValidTargets{Group: d.target}
1510+
for t.Next() {
14881511
types, err := t.BinInfo().Types()
14891512
if err != nil {
14901513
return nil, err
@@ -1932,13 +1955,6 @@ func (d *Debugger) FindLocation(goid, frame, deferredCall int, locStr string, in
19321955
d.targetMutex.Lock()
19331956
defer d.targetMutex.Unlock()
19341957

1935-
if len(d.target.Targets()) != 1 {
1936-
//TODO(aarzilli): if there is more than one target process all must be
1937-
//searched and the addresses returned need to specify which target process
1938-
//they belong to.
1939-
panic("multiple targets not implemented")
1940-
}
1941-
19421958
if _, err := d.target.Valid(); err != nil {
19431959
return nil, err
19441960
}
@@ -1948,7 +1964,7 @@ func (d *Debugger) FindLocation(goid, frame, deferredCall int, locStr string, in
19481964
return nil, err
19491965
}
19501966

1951-
return d.findLocation(d.target.Selected, goid, frame, deferredCall, locStr, loc, includeNonExecutableLines, substitutePathRules)
1967+
return d.findLocation(goid, frame, deferredCall, locStr, loc, includeNonExecutableLines, substitutePathRules)
19521968
}
19531969

19541970
// FindLocationSpec will find the location specified by 'locStr' and 'locSpec'.
@@ -1959,34 +1975,39 @@ func (d *Debugger) FindLocationSpec(goid, frame, deferredCall int, locStr string
19591975
d.targetMutex.Lock()
19601976
defer d.targetMutex.Unlock()
19611977

1962-
if len(d.target.Targets()) != 1 {
1963-
//TODO(aarzilli): if there is more than one target process all must be
1964-
//searched and the addresses returned need to specify which target process
1965-
//they belong to.
1966-
panic("multiple targets not implemented")
1967-
}
1968-
19691978
if _, err := d.target.Valid(); err != nil {
19701979
return nil, err
19711980
}
19721981

1973-
return d.findLocation(d.target.Selected, goid, frame, deferredCall, locStr, locSpec, includeNonExecutableLines, substitutePathRules)
1982+
return d.findLocation(goid, frame, deferredCall, locStr, locSpec, includeNonExecutableLines, substitutePathRules)
19741983
}
19751984

1976-
func (d *Debugger) findLocation(p *proc.Target, goid, frame, deferredCall int, locStr string, locSpec locspec.LocationSpec, includeNonExecutableLines bool, substitutePathRules [][2]string) ([]api.Location, error) {
1977-
s, _ := proc.ConvertEvalScope(p, goid, frame, deferredCall)
1978-
1979-
locs, err := locSpec.Find(p, d.processArgs, s, locStr, includeNonExecutableLines, substitutePathRules)
1980-
for i := range locs {
1981-
if locs[i].PC == 0 {
1982-
continue
1985+
func (d *Debugger) findLocation(goid, frame, deferredCall int, locStr string, locSpec locspec.LocationSpec, includeNonExecutableLines bool, substitutePathRules [][2]string) ([]api.Location, error) {
1986+
locations := []api.Location{}
1987+
t := proc.ValidTargets{Group: d.target}
1988+
for t.Next() {
1989+
pid := t.Pid()
1990+
s, _ := proc.ConvertEvalScope(t.Target, goid, frame, deferredCall)
1991+
locs, err := locSpec.Find(t.Target, d.processArgs, s, locStr, includeNonExecutableLines, substitutePathRules)
1992+
if err != nil {
1993+
return nil, err
19831994
}
1984-
file, line, fn := p.BinInfo().PCToLine(locs[i].PC)
1985-
locs[i].File = file
1986-
locs[i].Line = line
1987-
locs[i].Function = api.ConvertFunction(fn)
1995+
for i := range locs {
1996+
if locs[i].PC == 0 {
1997+
continue
1998+
}
1999+
file, line, fn := t.BinInfo().PCToLine(locs[i].PC)
2000+
locs[i].File = file
2001+
locs[i].Line = line
2002+
locs[i].Function = api.ConvertFunction(fn)
2003+
locs[i].PCPids = make([]int, len(locs[i].PCs))
2004+
for j := range locs[i].PCs {
2005+
locs[i].PCPids[j] = pid
2006+
}
2007+
}
2008+
locations = append(locations, locs...)
19882009
}
1989-
return locs, err
2010+
return locations, nil
19902011
}
19912012

19922013
// Disassemble code between startPC and endPC.

0 commit comments

Comments
 (0)