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
|
// Copyright 2013 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
//lint:file-ignore SA1019 go/ssa's test suite is built around the deprecated go/loader. We'll leave fixing that to upstream.
// Incomplete source tree on Android.
//go:build !android
// +build !android
package ir_test
// This file runs the IR builder in sanity-checking mode on all
// packages beneath $GOROOT and prints some summary information.
//
// Run with "go test -cpu=8 to" set GOMAXPROCS.
import (
"go/ast"
"go/token"
"runtime"
"testing"
"time"
"honnef.co/go/tools/go/ir"
"honnef.co/go/tools/go/ir/irutil"
"golang.org/x/tools/go/packages"
)
func bytesAllocated() uint64 {
runtime.GC()
var stats runtime.MemStats
runtime.ReadMemStats(&stats)
return stats.TotalAlloc
}
func TestStdlib(t *testing.T) {
if testing.Short() {
t.Skip("skipping in short mode; too slow (golang.org/issue/14113)")
}
var (
numFuncs int
numInstrs int
dLoad time.Duration
dCreate time.Duration
dBuild time.Duration
allocLoad uint64
allocBuild uint64
)
// Load, parse and type-check the program.
t0 := time.Now()
alloc0 := bytesAllocated()
cfg := &packages.Config{
Mode: packages.NeedName | packages.NeedFiles | packages.NeedCompiledGoFiles | packages.NeedImports | packages.NeedDeps | packages.NeedTypes | packages.NeedSyntax | packages.NeedTypesInfo | packages.NeedTypesSizes,
}
pkgs, err := packages.Load(cfg, "std")
if err != nil {
t.Fatalf("Load failed: %v", err)
}
allocLoad = bytesAllocated() - alloc0
dLoad = time.Since(t0)
alloc0 = bytesAllocated()
for _, pkg := range pkgs {
if len(pkg.Errors) != 0 {
t.Fatalf("Load failed: %v", pkg.Errors)
}
var mode ir.BuilderMode
// Comment out these lines during benchmarking. Approx IR build costs are noted.
mode |= ir.SanityCheckFunctions // + 2% space, + 4% time
mode |= ir.GlobalDebug // +30% space, +18% time
prog := ir.NewProgram(pkg.Fset, mode)
t0 := time.Now()
var irpkg *ir.Package
for _, pkg2 := range pkgs {
r := prog.CreatePackage(pkg2.Types, pkg2.Syntax, pkg2.TypesInfo, true)
if pkg2 == pkg {
irpkg = r
}
}
dCreate += time.Since(t0)
t0 = time.Now()
irpkg.Build()
dBuild += time.Since(t0)
allFuncs := irutil.AllFunctions(prog)
numFuncs += len(allFuncs)
// Check that all non-synthetic functions have distinct names.
// Synthetic wrappers for exported methods should be distinct too,
// except for unexported ones (explained at (*Function).RelString).
byName := make(map[string]*ir.Function)
for fn := range allFuncs {
if fn.Synthetic == 0 || (ast.IsExported(fn.Name()) && fn.Synthetic != ir.SyntheticGeneric) {
str := fn.String()
prev := byName[str]
byName[str] = fn
if prev != nil {
t.Errorf("%s: duplicate function named %s",
prog.Fset.Position(fn.Pos()), str)
t.Errorf("%s: (previously defined here)",
prog.Fset.Position(prev.Pos()))
}
}
}
// Dump some statistics.
var numInstrs int
for fn := range allFuncs {
for _, b := range fn.Blocks {
numInstrs += len(b.Instrs)
}
}
}
allocBuild = bytesAllocated() - alloc0
// determine line count
var lineCount int
pkgs[0].Fset.Iterate(func(f *token.File) bool {
lineCount += f.LineCount()
return true
})
// NB: when benchmarking, don't forget to clear the debug +
// sanity builder flags for better performance.
t.Log("GOMAXPROCS: ", runtime.GOMAXPROCS(0))
t.Log("#Source lines: ", lineCount)
t.Log("Load/parse/typecheck: ", dLoad)
t.Log("IR create: ", dCreate)
t.Log("IR build: ", dBuild)
// IR stats:
t.Log("#Packages: ", len(pkgs))
t.Log("#Functions: ", numFuncs)
t.Log("#Instructions: ", numInstrs)
t.Log("#MB AST+types: ", allocLoad/1e6)
t.Log("#MB IR: ", allocBuild/1e6)
}
|