// 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. package pointer import ( "bytes" "fmt" "go/token" "io" "golang.org/x/tools/container/intsets" "golang.org/x/tools/go/callgraph" "golang.org/x/tools/go/ssa" "golang.org/x/tools/go/types/typeutil" ) // A Config formulates a pointer analysis problem for Analyze. It is // only usable for a single invocation of Analyze and must not be // reused. type Config struct { // Mains contains the set of 'main' packages to analyze // Clients must provide the analysis with at least one // package defining a main() function. // // Non-main packages in the ssa.Program that are not // dependencies of any main package may still affect the // analysis result, because they contribute runtime types and // thus methods. // TODO(adonovan): investigate whether this is desirable. Mains []*ssa.Package // Reflection determines whether to handle reflection // operators soundly, which is currently rather slow since it // causes constraint to be generated during solving // proportional to the number of constraint variables, which // has not yet been reduced by presolver optimisation. Reflection bool // BuildCallGraph determines whether to construct a callgraph. // If enabled, the graph will be available in Result.CallGraph. BuildCallGraph bool // The client populates Queries[v] or IndirectQueries[v] // for each ssa.Value v of interest, to request that the // points-to sets pts(v) or pts(*v) be computed. If the // client needs both points-to sets, v may appear in both // maps. // // (IndirectQueries is typically used for Values corresponding // to source-level lvalues, e.g. an *ssa.Global.) // // The analysis populates the corresponding // Result.{Indirect,}Queries map when it creates the pointer // variable for v or *v. Upon completion the client can // inspect that map for the results. // // TODO(adonovan): this API doesn't scale well for batch tools // that want to dump the entire solution. Perhaps optionally // populate a map[*ssa.DebugRef]Pointer in the Result, one // entry per source expression. // Queries map[ssa.Value]struct{} IndirectQueries map[ssa.Value]struct{} extendedQueries map[ssa.Value][]*extendedQuery // If Log is non-nil, log messages are written to it. // Logging is extremely verbose. Log io.Writer } type track uint32 const ( trackChan track = 1 << iota // track 'chan' references trackMap // track 'map' references trackPtr // track regular pointers trackSlice // track slice references trackAll = ^track(0) ) // AddQuery adds v to Config.Queries. // Precondition: CanPoint(v.Type()). func (c *Config) AddQuery(v ssa.Value) { if !CanPoint(v.Type()) { panic(fmt.Sprintf("%s is not a pointer-like value: %s", v, v.Type())) } if c.Queries == nil { c.Queries = make(map[ssa.Value]struct{}) } c.Queries[v] = struct{}{} } // AddQuery adds v to Config.IndirectQueries. // Precondition: CanPoint(v.Type().Underlying().(*types.Pointer).Elem()). func (c *Config) AddIndirectQuery(v ssa.Value) { if c.IndirectQueries == nil { c.IndirectQueries = make(map[ssa.Value]struct{}) } if !CanPoint(mustDeref(v.Type())) { panic(fmt.Sprintf("%s is not the address of a pointer-like value: %s", v, v.Type())) } c.IndirectQueries[v] = struct{}{} } // AddExtendedQuery adds an extended, AST-based query on v to the // analysis. The query, which must be a single Go expression, allows // destructuring the value. // // The query must operate on a variable named 'x', which represents // the value, and result in a pointer-like object. Only a subset of // Go expressions are permitted in queries, namely channel receives, // pointer dereferences, field selectors, array/slice/map/tuple // indexing and grouping with parentheses. The specific indices when // indexing arrays, slices and maps have no significance. Indices used // on tuples must be numeric and within bounds. // // All field selectors must be explicit, even ones usually elided // due to promotion of embedded fields. // // The query 'x' is identical to using AddQuery. The query '*x' is // identical to using AddIndirectQuery. // // On success, AddExtendedQuery returns a Pointer to the queried // value. This Pointer will be initialized during analysis. Using it // before analysis has finished has undefined behavior. // // Example: // // given v, which represents a function call to 'fn() (int, []*T)', and // // 'type T struct { F *int }', the following query will access the field F. // c.AddExtendedQuery(v, "x[1][0].F") func (c *Config) AddExtendedQuery(v ssa.Value, query string) (*Pointer, error) { ops, _, err := parseExtendedQuery(v.Type().Underlying(), query) if err != nil { return nil, fmt.Errorf("invalid query %q: %s", query, err) } if c.extendedQueries == nil { c.extendedQueries = make(map[ssa.Value][]*extendedQuery) } ptr := &Pointer{} c.extendedQueries[v] = append(c.extendedQueries[v], &extendedQuery{ops: ops, ptr: ptr}) return ptr, nil } func (c *Config) prog() *ssa.Program { for _, main := range c.Mains { return main.Prog } panic("empty scope") } type Warning struct { Pos token.Pos Message string } // A Result contains the results of a pointer analysis. // // See Config for how to request the various Result components. // type Result struct { CallGraph *callgraph.Graph // discovered call graph Queries map[ssa.Value]Pointer // pts(v) for each v in Config.Queries. IndirectQueries map[ssa.Value]Pointer // pts(*v) for each v in Config.IndirectQueries. Warnings []Warning // warnings of unsoundness } // A Pointer is an equivalence class of pointer-like values. // // A Pointer doesn't have a unique type because pointers of distinct // types may alias the same object. // type Pointer struct { a *analysis n nodeid } // A PointsToSet is a set of labels (locations or allocations). type PointsToSet struct { a *analysis // may be nil if pts is nil pts *nodeset } func (s PointsToSet) String() string { var buf bytes.Buffer buf.WriteByte('[') if s.pts != nil { var space [50]int for i, l := range s.pts.AppendTo(space[:0]) { if i > 0 { buf.WriteString(", ") } buf.WriteString(s.a.labelFor(nodeid(l)).String()) } } buf.WriteByte(']') return buf.String() } // PointsTo returns the set of labels that this points-to set // contains. func (s PointsToSet) Labels() []*Label { var labels []*Label if s.pts != nil { var space [50]int for _, l := range s.pts.AppendTo(space[:0]) { labels = append(labels, s.a.labelFor(nodeid(l))) } } return labels } // If this PointsToSet came from a Pointer of interface kind // or a reflect.Value, DynamicTypes returns the set of dynamic // types that it may contain. (For an interface, they will // always be concrete types.) // // The result is a mapping whose keys are the dynamic types to which // it may point. For each pointer-like key type, the corresponding // map value is the PointsToSet for pointers of that type. // // The result is empty unless CanHaveDynamicTypes(T). // func (s PointsToSet) DynamicTypes() *typeutil.Map { var tmap typeutil.Map tmap.SetHasher(s.a.hasher) if s.pts != nil { var space [50]int for _, x := range s.pts.AppendTo(space[:0]) { ifaceObjId := nodeid(x) if !s.a.isTaggedObject(ifaceObjId) { continue // !CanHaveDynamicTypes(tDyn) } tDyn, v, indirect := s.a.taggedValue(ifaceObjId) if indirect { panic("indirect tagged object") // implement later } pts, ok := tmap.At(tDyn).(PointsToSet) if !ok { pts = PointsToSet{s.a, new(nodeset)} tmap.Set(tDyn, pts) } pts.pts.addAll(&s.a.nodes[v].solve.pts) } } return &tmap } // Intersects reports whether this points-to set and the // argument points-to set contain common members. func (x PointsToSet) Intersects(y PointsToSet) bool { if x.pts == nil || y.pts == nil { return false } // This takes Θ(|x|+|y|) time. var z intsets.Sparse z.Intersection(&x.pts.Sparse, &y.pts.Sparse) return !z.IsEmpty() } func (p Pointer) String() string { return fmt.Sprintf("n%d", p.n) } // PointsTo returns the points-to set of this pointer. func (p Pointer) PointsTo() PointsToSet { if p.n == 0 { return PointsToSet{} } return PointsToSet{p.a, &p.a.nodes[p.n].solve.pts} } // MayAlias reports whether the receiver pointer may alias // the argument pointer. func (p Pointer) MayAlias(q Pointer) bool { return p.PointsTo().Intersects(q.PointsTo()) } // DynamicTypes returns p.PointsTo().DynamicTypes(). func (p Pointer) DynamicTypes() *typeutil.Map { return p.PointsTo().DynamicTypes() }