Commit b8648184 authored by Russ Cox's avatar Russ Cox

cmd/api: read std package info once, not per goos-goarch-cgo

Cuts api test time from 12.7r 26.2u 14.2s to 7.5r 12.1u 2.2s.

After this change, all.bash runs in ~4:36 on my laptop.

For #26473.

Change-Id: I4211e6afcd7ab61a4ed2c9a2aa5ac1ea04982695
Reviewed-on: https://go-review.googlesource.com/c/go/+/177597
Run-TryBot: Russ Cox <rsc@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarBrad Fitzpatrick <bradfitz@golang.org>
parent 2d357d8d
...@@ -144,7 +144,7 @@ func main() { ...@@ -144,7 +144,7 @@ func main() {
} else { } else {
stds, err := exec.Command(goCmd(), "list", "std").Output() stds, err := exec.Command(goCmd(), "list", "std").Output()
if err != nil { if err != nil {
log.Fatal(err) log.Fatalf("go list std: %v\n%s", err, stds)
} }
for _, pkg := range strings.Fields(string(stds)) { for _, pkg := range strings.Fields(string(stds)) {
if !internalPkg.MatchString(pkg) { if !internalPkg.MatchString(pkg) {
...@@ -153,10 +153,25 @@ func main() { ...@@ -153,10 +153,25 @@ func main() {
} }
} }
importDir, importMap := loadImports()
// The code below assumes that the import map can vary
// by package, so that an import in one package (directory) might mean
// something different from the same import in another.
// While this can happen in GOPATH mode with vendoring,
// it is not possible in the standard library: the one importMap
// returned by loadImports applies to all packages.
// Construct a per-directory importMap that resolves to
// that single map for all packages.
importMapForDir := make(map[string]map[string]string)
for _, dir := range importDir {
importMapForDir[dir] = importMap
}
var featureCtx = make(map[string]map[string]bool) // feature -> context name -> true var featureCtx = make(map[string]map[string]bool) // feature -> context name -> true
for _, context := range contexts { for _, context := range contexts {
w := NewWalker(context, filepath.Join(build.Default.GOROOT, "src")) w := NewWalker(context, filepath.Join(build.Default.GOROOT, "src"))
w.loadImports(pkgNames, w.context) w.importDir = importDir
w.importMap = importMapForDir
for _, name := range pkgNames { for _, name := range pkgNames {
// Vendored packages do not contribute to our // Vendored packages do not contribute to our
...@@ -440,58 +455,58 @@ func tagKey(dir string, context *build.Context, tags []string) string { ...@@ -440,58 +455,58 @@ func tagKey(dir string, context *build.Context, tags []string) string {
return key return key
} }
func (w *Walker) loadImports(paths []string, context *build.Context) { // loadImports returns information about the packages in the standard library
if context == nil { // and the packages they themselves import.
context = &build.Default // importDir maps expanded import path to the directory containing that package.
} // importMap maps source import path to expanded import path.
// The source import path and expanded import path are identical except for vendored packages.
var ( // For example, on return:
tags = context.BuildTags //
cgoEnabled = "0" // importMap["math"] = "math"
) // importDir["math"] = "<goroot>/src/math"
if context.CgoEnabled { //
tags = append(tags[:len(tags):len(tags)], "cgo") // importMap["golang.org/x/net/route"] = "vendor/golang.org/x/net/route"
cgoEnabled = "1" // importDir["vendor/golang.org/x/net/route"] = "<goroot>/src/vendor/golang.org/x/net/route"
} //
// There are a few imports that only appear on certain platforms,
// TODO(golang.org/issue/29666): Request only the fields that we need. // including it turns out x/net/route, and we add those explicitly.
cmd := exec.Command(goCmd(), "list", "-e", "-deps", "-json") func loadImports() (importDir map[string]string, importMap map[string]string) {
if len(tags) > 0 { out, err := exec.Command(goCmd(), "list", "-e", "-deps", "-json", "std").CombinedOutput()
cmd.Args = append(cmd.Args, "-tags", strings.Join(tags, " "))
}
cmd.Args = append(cmd.Args, paths...)
cmd.Env = append(os.Environ(),
"GOOS="+context.GOOS,
"GOARCH="+context.GOARCH,
"CGO_ENABLED="+cgoEnabled,
)
stdout := new(bytes.Buffer)
cmd.Stdout = stdout
cmd.Stderr = new(strings.Builder)
err := cmd.Run()
if err != nil { if err != nil {
log.Fatalf("%s failed: %v\n%s", strings.Join(cmd.Args, " "), err, cmd.Stderr) log.Fatalf("loading imports: %v\n%s", err, out)
} }
w.importDir = make(map[string]string) importDir = make(map[string]string)
w.importMap = make(map[string]map[string]string) importMap = make(map[string]string)
dec := json.NewDecoder(stdout) dec := json.NewDecoder(bytes.NewReader(out))
for { for {
var pkg struct { var pkg struct {
ImportPath, Dir string ImportPath, Dir string
ImportMap map[string]string ImportMap map[string]string
} }
if err := dec.Decode(&pkg); err == io.EOF { err := dec.Decode(&pkg)
if err == io.EOF {
break break
} else if err != nil { }
log.Fatalf("%s: invalid output: %v", strings.Join(cmd.Args, " "), err) if err != nil {
log.Fatalf("go list: invalid output: %v", err)
} }
w.importDir[pkg.ImportPath] = pkg.Dir importDir[pkg.ImportPath] = pkg.Dir
w.importMap[pkg.Dir] = pkg.ImportMap for k, v := range pkg.ImportMap {
importMap[k] = v
}
} }
// Fixup for vendor packages listed in args above.
fixup := []string{
"vendor/golang.org/x/net/route",
}
for _, pkg := range fixup {
importDir[pkg] = filepath.Join(build.Default.GOROOT, "src", pkg)
importMap[strings.TrimPrefix(pkg, "vendor/")] = pkg
}
return
} }
// Importing is a sentinel taking the place in Walker.imported // Importing is a sentinel taking the place in Walker.imported
...@@ -523,7 +538,7 @@ func (w *Walker) ImportFrom(fromPath, fromDir string, mode types.ImportMode) (*t ...@@ -523,7 +538,7 @@ func (w *Walker) ImportFrom(fromPath, fromDir string, mode types.ImportMode) (*t
dir = filepath.Join(w.root, filepath.FromSlash(name)) dir = filepath.Join(w.root, filepath.FromSlash(name))
} }
if fi, err := os.Stat(dir); err != nil || !fi.IsDir() { if fi, err := os.Stat(dir); err != nil || !fi.IsDir() {
log.Fatalf("no source in tree for import %q: %v", name, err) log.Fatalf("no source in tree for import %q (from import %s in %s): %v", name, fromPath, fromDir, err)
} }
context := w.context context := w.context
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment