286 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			286 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			Go
		
	
	
	
// 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()
 | 
						|
}
 |