// Copyright 2019 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. // This file is largely based on go/analysis/internal/checker/checker.go. package source import ( "context" "fmt" "go/token" "go/types" "log" "reflect" "sort" "strings" "sync" "time" "golang.org/x/tools/go/analysis" ) func analyze(ctx context.Context, v View, pkgs []Package, analyzers []*analysis.Analyzer) []*Action { // Build nodes for initial packages. var roots []*Action for _, a := range analyzers { for _, pkg := range pkgs { root, err := pkg.GetActionGraph(ctx, a) if err != nil { continue } root.isroot = true roots = append(roots, root) } } // Execute the graph in parallel. execAll(v.FileSet(), roots) return roots } // An action represents one unit of analysis work: the application of // one analysis to one package. Actions form a DAG, both within a // package (as different analyzers are applied, either in sequence or // parallel), and across packages (as dependencies are analyzed). type Action struct { once sync.Once Analyzer *analysis.Analyzer Pkg Package Deps []*Action pass *analysis.Pass isroot bool objectFacts map[objectFactKey]analysis.Fact packageFacts map[packageFactKey]analysis.Fact inputs map[*analysis.Analyzer]interface{} result interface{} diagnostics []analysis.Diagnostic err error duration time.Duration } type objectFactKey struct { obj types.Object typ reflect.Type } type packageFactKey struct { pkg *types.Package typ reflect.Type } func (act *Action) String() string { return fmt.Sprintf("%s@%s", act.Analyzer, act.Pkg) } func execAll(fset *token.FileSet, actions []*Action) { var wg sync.WaitGroup for _, act := range actions { wg.Add(1) work := func(act *Action) { act.exec(fset) wg.Done() } go work(act) } wg.Wait() } func (act *Action) exec(fset *token.FileSet) { act.once.Do(func() { act.execOnce(fset) }) } func (act *Action) execOnce(fset *token.FileSet) { // Analyze dependencies. execAll(fset, act.Deps) // Report an error if any dependency failed. var failed []string for _, dep := range act.Deps { if dep.err != nil { failed = append(failed, dep.String()) } } if failed != nil { sort.Strings(failed) act.err = fmt.Errorf("failed prerequisites: %s", strings.Join(failed, ", ")) return } // Plumb the output values of the dependencies // into the inputs of this action. Also facts. inputs := make(map[*analysis.Analyzer]interface{}) act.objectFacts = make(map[objectFactKey]analysis.Fact) act.packageFacts = make(map[packageFactKey]analysis.Fact) for _, dep := range act.Deps { if dep.Pkg == act.Pkg { // Same package, different analysis (horizontal edge): // in-memory outputs of prerequisite analyzers // become inputs to this analysis pass. inputs[dep.Analyzer] = dep.result } else if dep.Analyzer == act.Analyzer { // (always true) // Same analysis, different package (vertical edge): // serialized facts produced by prerequisite analysis // become available to this analysis pass. inheritFacts(act, dep) } } // Run the analysis. pass := &analysis.Pass{ Analyzer: act.Analyzer, Fset: fset, Files: act.Pkg.GetSyntax(), Pkg: act.Pkg.GetTypes(), TypesInfo: act.Pkg.GetTypesInfo(), // TODO(rstambler): Get real TypeSizes from go/packages (golang.org/issues/30139). TypesSizes: &types.StdSizes{}, ResultOf: inputs, Report: func(d analysis.Diagnostic) { act.diagnostics = append(act.diagnostics, d) }, ImportObjectFact: act.importObjectFact, ExportObjectFact: act.exportObjectFact, ImportPackageFact: act.importPackageFact, ExportPackageFact: act.exportPackageFact, } act.pass = pass var err error if len(act.Pkg.GetErrors()) > 0 && !pass.Analyzer.RunDespiteErrors { err = fmt.Errorf("analysis skipped due to errors in package") } else { act.result, err = pass.Analyzer.Run(pass) if err == nil { if got, want := reflect.TypeOf(act.result), pass.Analyzer.ResultType; got != want { err = fmt.Errorf( "internal error: on package %s, analyzer %s returned a result of type %v, but declared ResultType %v", pass.Pkg.Path(), pass.Analyzer, got, want) } } } act.err = err // disallow calls after Run pass.ExportObjectFact = nil pass.ExportPackageFact = nil } // inheritFacts populates act.facts with // those it obtains from its dependency, dep. func inheritFacts(act, dep *Action) { for key, fact := range dep.objectFacts { // Filter out facts related to objects // that are irrelevant downstream // (equivalently: not in the compiler export data). if !exportedFrom(key.obj, dep.Pkg.GetTypes()) { continue } act.objectFacts[key] = fact } for key, fact := range dep.packageFacts { // TODO: filter out facts that belong to // packages not mentioned in the export data // to prevent side channels. act.packageFacts[key] = fact } } // exportedFrom reports whether obj may be visible to a package that imports pkg. // This includes not just the exported members of pkg, but also unexported // constants, types, fields, and methods, perhaps belonging to oether packages, // that find there way into the API. // This is an overapproximation of the more accurate approach used by // gc export data, which walks the type graph, but it's much simpler. // // TODO(adonovan): do more accurate filtering by walking the type graph. func exportedFrom(obj types.Object, pkg *types.Package) bool { switch obj := obj.(type) { case *types.Func: return obj.Exported() && obj.Pkg() == pkg || obj.Type().(*types.Signature).Recv() != nil case *types.Var: return obj.Exported() && obj.Pkg() == pkg || obj.IsField() case *types.TypeName, *types.Const: return true } return false // Nil, Builtin, Label, or PkgName } // importObjectFact implements Pass.ImportObjectFact. // Given a non-nil pointer ptr of type *T, where *T satisfies Fact, // importObjectFact copies the fact value to *ptr. func (act *Action) importObjectFact(obj types.Object, ptr analysis.Fact) bool { if obj == nil { panic("nil object") } key := objectFactKey{obj, factType(ptr)} if v, ok := act.objectFacts[key]; ok { reflect.ValueOf(ptr).Elem().Set(reflect.ValueOf(v).Elem()) return true } return false } // exportObjectFact implements Pass.ExportObjectFact. func (act *Action) exportObjectFact(obj types.Object, fact analysis.Fact) { if act.pass.ExportObjectFact == nil { log.Panicf("%s: Pass.ExportObjectFact(%s, %T) called after Run", act, obj, fact) } if obj.Pkg() != act.Pkg.GetTypes() { log.Panicf("internal error: in analysis %s of package %s: Fact.Set(%s, %T): can't set facts on objects belonging another package", act.Analyzer, act.Pkg, obj, fact) } key := objectFactKey{obj, factType(fact)} act.objectFacts[key] = fact // clobber any existing entry } // importPackageFact implements Pass.ImportPackageFact. // Given a non-nil pointer ptr of type *T, where *T satisfies Fact, // fact copies the fact value to *ptr. func (act *Action) importPackageFact(pkg *types.Package, ptr analysis.Fact) bool { if pkg == nil { panic("nil package") } key := packageFactKey{pkg, factType(ptr)} if v, ok := act.packageFacts[key]; ok { reflect.ValueOf(ptr).Elem().Set(reflect.ValueOf(v).Elem()) return true } return false } // exportPackageFact implements Pass.ExportPackageFact. func (act *Action) exportPackageFact(fact analysis.Fact) { if act.pass.ExportPackageFact == nil { log.Panicf("%s: Pass.ExportPackageFact(%T) called after Run", act, fact) } key := packageFactKey{act.pass.Pkg, factType(fact)} act.packageFacts[key] = fact // clobber any existing entry } func factType(fact analysis.Fact) reflect.Type { t := reflect.TypeOf(fact) if t.Kind() != reflect.Ptr { log.Fatalf("invalid Fact type: got %T, want pointer", t) } return t }