1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
|
package main
import (
"io/ioutil"
"path"
"sort"
"strings"
)
// GlobalData contains data describing pkgsrc as a whole.
type GlobalData struct {
Pkgsrcdir string // Relative to the current working directory.
MasterSiteURLToVar map[string]string // "https://github.com/" => "MASTER_SITE_GITHUB"
MasterSiteVarToURL map[string]string // "MASTER_SITE_GITHUB" => "https://github.com/"
PkgOptions map[string]string // "x11" => "Provides X11 support"
Tools ToolRegistry //
SystemBuildDefs map[string]bool // The set of user-defined variables that are added to BUILD_DEFS within the bsd.pkg.mk file.
suggestedUpdates []SuggestedUpdate //
suggestedWipUpdates []SuggestedUpdate //
LastChange map[string]*Change //
UserDefinedVars map[string]*MkLine // varname => line
Deprecated map[string]string //
vartypes map[string]*Vartype // varcanon => type
}
// Change is a change entry from the `doc/CHANGES-*` files.
type Change struct {
Line *Line
Action string
Pkgpath string
Version string
Author string
Date string
}
// SuggestedUpdate is from the `doc/TODO` file.
type SuggestedUpdate struct {
Line *Line
Pkgname string
Version string
Comment string
}
func (gd *GlobalData) Initialize() {
firstArg := G.Todo[0]
if fileExists(firstArg) {
firstArg = path.Dir(firstArg)
}
if relTopdir := findPkgsrcTopdir(firstArg); relTopdir != "" {
gd.Pkgsrcdir = firstArg + "/" + relTopdir
} else {
dummyLine.Fatalf("%q is not inside a pkgsrc tree.", firstArg)
}
gd.vartypes = make(map[string]*Vartype)
gd.InitVartypes()
gd.loadDistSites()
gd.loadPkgOptions()
gd.loadDocChanges()
gd.loadSuggestedUpdates()
gd.loadUserDefinedVars()
gd.loadTools()
gd.loadDeprecatedVars()
}
func (gd *GlobalData) loadDistSites() {
fname := gd.Pkgsrcdir + "/mk/fetch/sites.mk"
lines := LoadExistingLines(fname, true)
name2url := make(map[string]string)
url2name := make(map[string]string)
for _, line := range lines {
if m, varname, _, _, _, urls, _ := MatchVarassign(line.Text); m {
if hasPrefix(varname, "MASTER_SITE_") && varname != "MASTER_SITE_BACKUP" {
for _, url := range splitOnSpace(urls) {
if matches(url, `^(?:http://|https://|ftp://)`) {
if name2url[varname] == "" {
name2url[varname] = url
}
url2name[url] = varname
}
}
}
}
}
// Explicitly allowed, although not defined in mk/fetch/sites.mk.
name2url["MASTER_SITE_LOCAL"] = "ftp://ftp.NetBSD.org/pub/pkgsrc/distfiles/LOCAL_PORTS/"
if G.opts.Debug {
traceStep("Loaded %d MASTER_SITE_* URLs.", len(url2name))
}
gd.MasterSiteURLToVar = url2name
gd.MasterSiteVarToURL = name2url
}
func (gd *GlobalData) loadPkgOptions() {
fname := gd.Pkgsrcdir + "/mk/defaults/options.description"
lines := LoadExistingLines(fname, false)
gd.PkgOptions = make(map[string]string)
for _, line := range lines {
if m, optname, optdescr := match2(line.Text, `^([-0-9a-z_+]+)(?:\s+(.*))?$`); m {
gd.PkgOptions[optname] = optdescr
} else {
line.Fatalf("Unknown line format.")
}
}
}
func (gd *GlobalData) loadTools() {
toolFiles := []string{"defaults.mk"}
{
fname := G.globalData.Pkgsrcdir + "/mk/tools/bsd.tools.mk"
lines := LoadExistingLines(fname, true)
for _, line := range lines {
if m, _, _, includefile := match3(line.Text, reMkInclude); m {
if !contains(includefile, "/") {
toolFiles = append(toolFiles, includefile)
}
}
}
if len(toolFiles) <= 1 {
NewLineWhole(fname).Fatalf("Too few tool files.")
}
}
reg := NewToolRegistry()
reg.RegisterTool(&Tool{"echo", "ECHO", true, true, true})
reg.RegisterTool(&Tool{"echo -n", "ECHO_N", true, true, true})
reg.RegisterTool(&Tool{"false", "FALSE", true /*why?*/, true, false})
reg.RegisterTool(&Tool{"test", "TEST", true, true, true})
reg.RegisterTool(&Tool{"true", "TRUE", true /*why?*/, true, true})
systemBuildDefs := make(map[string]bool)
for _, basename := range toolFiles {
fname := G.globalData.Pkgsrcdir + "/mk/tools/" + basename
lines := LoadExistingLines(fname, true)
for _, line := range lines {
if m, varname, _, _, _, value, _ := MatchVarassign(line.Text); m {
if varname == "TOOLS_CREATE" && (value == "[" || matches(value, `^?[-\w.]+$`)) {
reg.Register(value)
} else if m, toolname := match1(varname, `^_TOOLS_VARNAME\.([-\w.]+|\[)$`); m {
reg.RegisterVarname(toolname, value)
} else if m, toolname := match1(varname, `^(?:TOOLS_PATH|_TOOLS_DEPMETHOD)\.([-\w.]+|\[)$`); m {
reg.Register(toolname)
} else if m, toolname := match1(varname, `_TOOLS\.(.*)`); m {
reg.Register(toolname)
for _, tool := range splitOnSpace(value) {
reg.Register(tool)
}
}
}
}
}
for _, basename := range []string{"bsd.prefs.mk", "bsd.pkg.mk"} {
fname := G.globalData.Pkgsrcdir + "/mk/" + basename
condDepth := 0
lines := LoadExistingLines(fname, true)
for _, line := range lines {
text := line.Text
if m, varname, _, _, _, value, _ := MatchVarassign(text); m {
if varname == "USE_TOOLS" {
if G.opts.Debug {
traceStep("[condDepth=%d] %s", condDepth, value)
}
if condDepth == 0 || condDepth == 1 && basename == "bsd.prefs.mk" {
for _, toolname := range splitOnSpace(value) {
if !containsVarRef(toolname) {
for _, tool := range []*Tool{reg.Register(toolname), reg.Register("TOOLS_" + toolname)} {
tool.Predefined = true
if basename == "bsd.prefs.mk" {
tool.UsableAtLoadtime = true
}
}
}
}
}
} else if varname == "_BUILD_DEFS" {
for _, bdvar := range splitOnSpace(value) {
systemBuildDefs[bdvar] = true
}
}
} else if m, _, cond, _ := matchMkCond(text); m {
switch cond {
case "if", "ifdef", "ifndef", "for":
condDepth++
case "endif", "endfor":
condDepth--
}
}
}
}
if G.opts.Debug {
reg.Trace()
}
if G.opts.Debug {
traceStep("systemBuildDefs: %v", systemBuildDefs)
}
// Some user-defined variables do not influence the binary
// package at all and therefore do not have to be added to
// BUILD_DEFS; therefore they are marked as “already added”.
systemBuildDefs["DISTDIR"] = true
systemBuildDefs["FETCH_CMD"] = true
systemBuildDefs["FETCH_OUTPUT_ARGS"] = true
systemBuildDefs["GAMES_USER"] = true
systemBuildDefs["GAMES_GROUP"] = true
systemBuildDefs["GAMEDATAMODE"] = true
systemBuildDefs["GAMEDIRMODE"] = true
systemBuildDefs["GAMEMODE"] = true
systemBuildDefs["GAMEOWN"] = true
systemBuildDefs["GAMEGRP"] = true
gd.Tools = reg
gd.SystemBuildDefs = systemBuildDefs
}
func loadSuggestedUpdates(fname string) []SuggestedUpdate {
lines := LoadExistingLines(fname, false)
return parselinesSuggestedUpdates(lines)
}
func parselinesSuggestedUpdates(lines []*Line) []SuggestedUpdate {
var updates []SuggestedUpdate
state := 0
for _, line := range lines {
text := line.Text
if state == 0 && text == "Suggested package updates" {
state = 1
} else if state == 1 && text == "" {
state = 2
} else if state == 2 {
state = 3
} else if state == 3 && text == "" {
state = 4
}
if state == 3 {
if m, pkgname, comment := match2(text, `^\to\s(\S+)(?:\s*(.+))?$`); m {
if m, pkgbase, pkgversion := match2(pkgname, rePkgname); m {
updates = append(updates, SuggestedUpdate{line, pkgbase, pkgversion, comment})
} else {
line.Warn1("Invalid package name %q", pkgname)
}
} else {
line.Warn1("Invalid line format %q", text)
}
}
}
return updates
}
func (gd *GlobalData) loadSuggestedUpdates() {
gd.suggestedUpdates = loadSuggestedUpdates(G.globalData.Pkgsrcdir + "/doc/TODO")
if wipFilename := G.globalData.Pkgsrcdir + "/wip/TODO"; fileExists(wipFilename) {
gd.suggestedWipUpdates = loadSuggestedUpdates(wipFilename)
}
}
func (gd *GlobalData) loadDocChangesFromFile(fname string) []*Change {
lines := LoadExistingLines(fname, false)
parseChange := func(line *Line) *Change {
text := line.Text
if !hasPrefix(line.Text, "\t") {
return nil
}
f := strings.Fields(text)
n := len(f)
if n != 4 && n != 6 {
return nil
}
action, pkgpath, author, date := f[0], f[1], f[len(f)-2], f[len(f)-1]
if !hasPrefix(author, "[") || !hasSuffix(date, "]") {
return nil
}
author, date = author[1:], date[:len(date)-1]
switch {
case action == "Added" && f[2] == "version" && n == 6:
return &Change{line, action, pkgpath, f[3], author, date}
case (action == "Updated" || action == "Downgraded") && f[2] == "to" && n == 6:
return &Change{line, action, pkgpath, f[3], author, date}
case action == "Removed" && (n == 6 && f[2] == "successor" || n == 4):
return &Change{line, action, pkgpath, "", author, date}
case (action == "Renamed" || action == "Moved") && f[2] == "to" && n == 6:
return &Change{line, action, pkgpath, "", author, date}
}
return nil
}
var changes []*Change
for _, line := range lines {
if change := parseChange(line); change != nil {
changes = append(changes, change)
} else if len(line.Text) >= 2 && line.Text[0] == '\t' && 'A' <= line.Text[1] && line.Text[1] <= 'Z' {
line.Warn1("Unknown doc/CHANGES line: %q", line.Text)
Explain1("See mk/misc/developer.mk for the rules.")
}
}
return changes
}
func (gd *GlobalData) GetSuggestedPackageUpdates() []SuggestedUpdate {
if G.Wip {
return gd.suggestedWipUpdates
} else {
return gd.suggestedUpdates
}
}
func (gd *GlobalData) loadDocChanges() {
docdir := G.globalData.Pkgsrcdir + "/doc"
files, err := ioutil.ReadDir(docdir)
if err != nil {
NewLineWhole(docdir).Fatalf("Cannot be read.")
}
var fnames []string
for _, file := range files {
fname := file.Name()
if matches(fname, `^CHANGES-20\d\d$`) && fname >= "CHANGES-2011" {
fnames = append(fnames, fname)
}
}
sort.Strings(fnames)
gd.LastChange = make(map[string]*Change)
for _, fname := range fnames {
changes := gd.loadDocChangesFromFile(docdir + "/" + fname)
for _, change := range changes {
gd.LastChange[change.Pkgpath] = change
}
}
}
func (gd *GlobalData) loadUserDefinedVars() {
lines := LoadExistingLines(G.globalData.Pkgsrcdir+"/mk/defaults/mk.conf", true)
mklines := NewMkLines(lines)
gd.UserDefinedVars = make(map[string]*MkLine)
for _, mkline := range mklines.mklines {
if mkline.IsVarassign() {
gd.UserDefinedVars[mkline.Varname()] = mkline
}
}
}
func (gd *GlobalData) loadDeprecatedVars() {
gd.Deprecated = map[string]string{
// December 2003
"FIX_RPATH": "It has been removed from pkgsrc in 2003.",
// February 2005
"LIB_DEPENDS": "Use DEPENDS instead.",
"ONLY_FOR_ARCHS": "Use ONLY_FOR_PLATFORM instead.",
"NOT_FOR_ARCHS": "Use NOT_FOR_PLATFORM instead.",
"ONLY_FOR_OPSYS": "Use ONLY_FOR_PLATFORM instead.",
"NOT_FOR_OPSYS": "Use NOT_FOR_PLATFORM instead.",
// May 2005
"ALL_TARGET": "Use BUILD_TARGET instead.",
"DIGEST_FILE": "Use DISTINFO_FILE instead.",
"IGNORE": "Use PKG_FAIL_REASON or PKG_SKIP_REASON instead.",
"IS_INTERACTIVE": "Use INTERACTIVE_STAGE instead.",
"KERBEROS": "Use the PKG_OPTIONS framework instead.",
"MASTER_SITE_SUBDIR": "Use some form of MASTER_SITES instead.",
"MD5_FILE": "Use DISTINFO_FILE instead.",
"MIRROR_DISTFILE": "Use NO_BIN_ON_FTP and/or NO_SRC_ON_FTP instead.",
"NO_CDROM": "Use NO_BIN_ON_CDROM and/or NO_SRC_ON_CDROM instead.",
"NO_PATCH": "You can just remove it.",
"NO_WRKSUBDIR": "Use WRKSRC=${WRKDIR} instead.",
"PATCH_SITE_SUBDIR": "Use some form of PATCHES_SITES instead.",
"PATCH_SUM_FILE": "Use DISTINFO_FILE instead.",
"PKG_JVM": "Use PKG_DEFAULT_JVM instead.",
"USE_BUILDLINK2": "You can just remove it.",
"USE_BUILDLINK3": "You can just remove it.",
"USE_CANNA": "Use the PKG_OPTIONS framework instead.",
"USE_DB4": "Use the PKG_OPTIONS framework instead.",
"USE_DIRS": "You can just remove it.",
"USE_ESOUND": "Use the PKG_OPTIONS framework instead.",
"USE_GIF": "Use the PKG_OPTIONS framework instead.",
"USE_GMAKE": "Use USE_TOOLS+=gmake instead.",
"USE_GNU_TOOLS": "Use USE_TOOLS instead.",
"USE_IDEA": "Use the PKG_OPTIONS framework instead.",
"USE_LIBCRACK": "Use the PKG_OPTIONS framework instead.",
"USE_MMX": "Use the PKG_OPTIONS framework instead.",
"USE_PKGLIBTOOL": "Use USE_LIBTOOL instead.",
"USE_SSL": "Include \"../../security/openssl/buildlink3.mk\" instead.",
// July 2005
"USE_PERL5": "Use USE_TOOLS+=perl or USE_TOOLS+=perl:run instead.",
// October 2005
"NO_TOOLS": "You can just remove it.",
"NO_WRAPPER": "You can just remove it.",
// November 2005
"ALLFILES": "Use CKSUMFILES instead.",
"DEPENDS_TARGET": "Use DEPENDS instead.",
"FETCH_DEPENDS": "Use DEPENDS instead.",
"RUN_DEPENDS": "Use DEPENDS instead.",
// December 2005
"USE_CUPS": "Use the PKG_OPTIONS framework (option cups) instead.",
"USE_I586": "Use the PKG_OPTIONS framework (option i586) instead.",
"USE_INN": "Use the PKG_OPTIONS framework instead.",
"USE_OPENLDAP": "Use the PKG_OPTIONS framework (option openldap) instead.",
"USE_OSS": "Use the PKG_OPTIONS framework (option oss) instead.",
"USE_RSAREF2": "Use the PKG_OPTIONS framework (option rsaref) instead.",
"USE_SASL": "Use the PKG_OPTIONS framework (option sasl) instead.",
"USE_SASL2": "Use the PKG_OPTIONS framework (option sasl) instead.",
"USE_SJ3": "Use the PKG_OPTIONS framework (option sj3) instead.",
"USE_SOCKS": "Use the PKG_OPTIONS framework (socks4 and socks5 options) instead.",
"USE_WNN4": "Use the PKG_OPTIONS framework (option wnn4) instead.",
"USE_XFACE": "Use the PKG_OPTIONS framework instead.",
// February 2006
"TOOLS_DEPMETHOD": "Use the :build or :run modifiers in USE_TOOLS instead.",
"MANDIR": "Please use ${PREFIX}/${PKGMANDIR} instead.",
"DOWNLOADED_DISTFILE": "Use the shell variable $$extract_file instead.",
"DECOMPRESS_CMD": "Use EXTRACT_CMD instead.",
// March 2006
"INSTALL_EXTRA_TMPL": "Use INSTALL_TEMPLATE instead.",
"DEINSTALL_EXTRA_TMPL": "Use DEINSTALL_TEMPLATE instead.",
// April 2006
"RECOMMENDED": "Use ABI_DEPENDS instead.",
"BUILD_USES_MSGFMT": "Use USE_TOOLS+=msgfmt instead.",
"USE_MSGFMT_PLURALS": "Use USE_TOOLS+=msgfmt instead.",
// May 2006
"EXTRACT_USING_PAX": "Use \"EXTRACT_OPTS=-t pax\" instead.",
"NO_EXTRACT": "It doesn't exist anymore.",
"_FETCH_MESSAGE": "Use FETCH_MESSAGE (different format) instead.",
"BUILDLINK_DEPENDS.*": "Use BUILDLINK_API_DEPENDS.* instead.",
"BUILDLINK_RECOMMENDED.*": "Use BUILDLINK_ABI_DEPENDS.* instead.",
"SHLIB_HANDLING": "Use CHECK_SHLIBS_SUPPORTED instead.",
"USE_RMAN": "It has been removed.",
// June 2006
"DEINSTALL_SRC": "Use the pkginstall framework instead.",
"INSTALL_SRC": "Use the pkginstall framework instead.",
"DEINSTALL_TEMPLATE": "Use DEINSTALL_TEMPLATES instead.",
"INSTALL_TEMPLATE": "Use INSTALL_TEMPLATES instead.",
"HEADER_TEMPLATE": "Use HEADER_TEMPLATES instead.",
"_REPLACE.*": "Use REPLACE.* instead.",
"_REPLACE_FILES.*": "Use REPLACE_FILES.* instead.",
"MESSAGE": "Use MESSAGE_SRC instead.",
"INSTALL_FILE": "It may only be used internally by pkgsrc.",
"DEINSTALL_FILE": "It may only be used internally by pkgsrc.",
// July 2006
"USE_DIGEST": "You can just remove it.",
"LTCONFIG_OVERRIDE": "You can just remove it.",
"USE_GNU_GETTEXT": "You can just remove it.",
"BUILD_ENV": "Use PKGSRC_MAKE_ENV instead.",
"DYNAMIC_MASTER_SITES": "You can just remove it.",
// September 2006
"MAKEFILE": "Use MAKE_FILE instead.",
// November 2006
"SKIP_PORTABILITY_CHECK": "Use CHECK_PORTABILITY_SKIP (a list of patterns) instead.",
// January 2007
"BUILDLINK_TRANSFORM.*": "Use BUILDLINK_FNAME_TRANSFORM.* instead.",
// March 2007
"SCRIPTDIR": "You can just remove it.",
"NO_PKG_REGISTER": "You can just remove it.",
"NO_DEPENDS": "You can just remove it.",
// October 2007
"_PKG_SILENT": "Use RUN (with more error checking) instead.",
"_PKG_DEBUG": "Use RUN (with more error checking) instead.",
"LICENCE": "Use LICENSE instead.",
// November 2007
//USE_NCURSES Include "../../devel/ncurses/buildlink3.mk" instead.
// December 2007
"INSTALLATION_DIRS_FROM_PLIST": "Use AUTO_MKDIRS instead.",
// April 2009
"NO_PACKAGE": "It doesn't exist anymore.",
"NO_MTREE": "You can just remove it.",
// July 2012
"SETGIDGAME": "Use USE_GAMESGROUP instead.",
"GAMEGRP": "Use GAMES_GROUP instead.",
"GAMEOWN": "Use GAMES_USER instead.",
// July 2013
"USE_GNU_READLINE": "Include \"../../devel/readline/buildlink3.mk\" instead.",
// October 2014
"SVR4_PKGNAME": "Just remove it.",
"PKG_INSTALLATION_TYPES": "Just remove it.",
// January 2016
"SUBST_POSTCMD.*": "Has been removed, as it seemed unused.",
// June 2016
"USE_CROSSBASE": "Has been removed.",
}
}
// See `mk/tools/`.
type Tool struct {
Name string // e.g. "sed", "gzip"
Varname string // e.g. "SED", "GZIP_CMD"
MustUseVarForm bool // True for `echo`, because of many differing implementations.
Predefined bool // This tool is used by the pkgsrc infrastructure, therefore the package does not need to add it to `USE_TOOLS` explicitly.
UsableAtLoadtime bool // May be used after including `bsd.prefs.mk`.
}
type ToolRegistry struct {
byName map[string]*Tool
byVarname map[string]*Tool
}
func NewToolRegistry() ToolRegistry {
return ToolRegistry{make(map[string]*Tool), make(map[string]*Tool)}
}
func (tr *ToolRegistry) Register(toolname string) *Tool {
tool := tr.byName[toolname]
if tool == nil {
tool = &Tool{Name: toolname}
tr.byName[toolname] = tool
}
return tool
}
func (tr *ToolRegistry) RegisterVarname(toolname, varname string) *Tool {
tool := tr.Register(toolname)
tool.Varname = varname
tr.byVarname[varname] = tool
return tool
}
func (tr *ToolRegistry) RegisterTool(tool *Tool) {
if tool.Name != "" && tr.byName[tool.Name] == nil {
tr.byName[tool.Name] = tool
}
if tool.Varname != "" && tr.byVarname[tool.Varname] == nil {
tr.byVarname[tool.Varname] = tool
}
}
func (tr *ToolRegistry) Trace() {
if G.opts.Debug {
defer tracecall0()()
}
var keys []string
for k := range tr.byName {
keys = append(keys, k)
}
sort.Strings(keys)
for _, toolname := range keys {
traceStep("tool %+v", tr.byName[toolname])
}
}
|