// 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 // This file defines the main datatypes and Analyze function of the pointer analysis. import ( "fmt" "go/token" "io" "os" "code.google.com/p/go.tools/go/types" "code.google.com/p/go.tools/go/types/typemap" "code.google.com/p/go.tools/ssa" ) // object.flags bitmask values. const ( otTagged = 1 << iota // type-tagged object otIndirect // type-tagged object with indirect payload otFunction // function object ) // An object represents a contiguous block of memory to which some // (generalized) pointer may point. // // (Note: most variables called 'obj' are not *objects but nodeids // such that a.nodes[obj].obj != nil.) // type object struct { // flags is a bitset of the node type (ot*) flags defined above. flags uint32 // Number of following nodes belonging to the same "object" // allocation. Zero for all other nodes. size uint32 // The SSA operation that caused this object to be allocated. // May be nil for (e.g.) intrinsic allocations. val ssa.Value // The call-graph node (=context) in which this object was allocated. // May be nil for global objects: Global, Const, some Functions. cgn *cgnode // If this is an rtype instance object, or a *rtype-tagged // object, this is its type. rtype types.Type } // nodeid denotes a node. // It is an index within analysis.nodes. // We use small integers, not *node pointers, for many reasons: // - they are smaller on 64-bit systems. // - sets of them can be represented compactly in bitvectors or BDDs. // - order matters; a field offset can be computed by simple addition. type nodeid uint32 // A node is an equivalence class of memory locations. // Nodes may be pointers, pointed-to locations, neither, or both. // // Nodes that are pointed-to locations ("labels") have an enclosing // object (see analysis.enclosingObject). // type node struct { // If non-nil, this node is the start of an object // (addressable memory location). // The following obj.size words implicitly belong to the object; // they locate their object by scanning back. obj *object // The type of the field denoted by this node. Non-aggregate, // unless this is an tagged.T node (i.e. the thing // pointed to by an interface) in which case typ is that type. typ types.Type // subelement indicates which directly embedded subelement of // an object of aggregate type (struct, tuple, array) this is. subelement *fieldInfo // e.g. ".a.b[*].c" // Points-to sets. pts nodeset // points-to set of this node prevPts nodeset // pts(n) in previous iteration (for difference propagation) // Graph edges copyTo nodeset // simple copy constraint edges // Complex constraints attached to this node (x). // - *loadConstraint y=*x // - *offsetAddrConstraint y=&x.f or y=&x[0] // - *storeConstraint *x=z // - *typeAssertConstraint y=x.(T) // - *invokeConstraint y=x.f(params...) complex constraintset } type constraint interface { String() string // For a complex constraint, returns the nodeid of the pointer // to which it is attached. ptr() nodeid // solve is called for complex constraints when the pts for // the node to which they are attached has changed. solve(a *analysis, n *node, delta nodeset) } // dst = &src // pts(dst) ⊇ {src} // A base constraint used to initialize the solver's pt sets type addrConstraint struct { dst nodeid // (ptr) src nodeid } // dst = src // A simple constraint represented directly as a copyTo graph edge. type copyConstraint struct { dst nodeid src nodeid // (ptr) } // dst = src[offset] // A complex constraint attached to src (the pointer) type loadConstraint struct { offset uint32 dst nodeid src nodeid // (ptr) } // dst[offset] = src // A complex constraint attached to dst (the pointer) type storeConstraint struct { offset uint32 dst nodeid // (ptr) src nodeid } // dst = &src.f or dst = &src[0] // A complex constraint attached to dst (the pointer) type offsetAddrConstraint struct { offset uint32 dst nodeid src nodeid // (ptr) } // dst = src.(typ) // A complex constraint attached to src (the interface). type typeAssertConstraint struct { typ types.Type dst nodeid src nodeid // (ptr) } // src.method(params...) // A complex constraint attached to iface. type invokeConstraint struct { method *types.Func // the abstract method iface nodeid // (ptr) the interface params nodeid // the first parameter in the params/results block } // An analysis instance holds the state of a single pointer analysis problem. type analysis struct { config *Config // the client's control/observer interface prog *ssa.Program // the program being analyzed log io.Writer // log stream; nil to disable panicNode nodeid // sink for panic, source for recover nodes []*node // indexed by nodeid flattenMemo map[types.Type][]*fieldInfo // memoization of flatten() constraints []constraint // set of constraints callsites []*callsite // all callsites genq []*cgnode // queue of functions to generate constraints for intrinsics map[*ssa.Function]intrinsic // non-nil values are summaries for intrinsic fns funcObj map[*ssa.Function]nodeid // default function object for each func probes map[*ssa.CallCommon]nodeid // maps call to print() to argument variable valNode map[ssa.Value]nodeid // node for each ssa.Value work worklist // solver's worklist // Reflection: hasher typemap.Hasher // cache of type hashes reflectValueObj types.Object // type symbol for reflect.Value (if present) reflectRtypeObj types.Object // *types.TypeName for reflect.rtype (if present) reflectRtype *types.Pointer // *reflect.rtype rtypes typemap.M // nodeid of canonical *rtype-tagged object for type T reflectZeros typemap.M // nodeid of canonical T-tagged object for zero value } // enclosingObj returns the object (addressible memory object) that encloses node id. // Panic ensues if that node does not belong to any object. func (a *analysis) enclosingObj(id nodeid) *object { // Find previous node with obj != nil. for i := id; i >= 0; i-- { n := a.nodes[i] if obj := n.obj; obj != nil { if i+nodeid(obj.size) <= id { break // out of bounds } return obj } } panic("node has no enclosing object") } // labelFor returns the Label for node id. // Panic ensues if that node is not addressable. func (a *analysis) labelFor(id nodeid) *Label { return &Label{ obj: a.enclosingObj(id), subelement: a.nodes[id].subelement, } } func (a *analysis) warnf(pos token.Pos, format string, args ...interface{}) { if Warn := a.config.Warn; Warn != nil { Warn(pos, format, args...) } else { fmt.Fprintf(os.Stderr, "%s: warning: ", a.prog.Fset.Position(pos)) fmt.Fprintf(os.Stderr, format, args...) fmt.Fprintln(os.Stderr) } } // Analyze runs the pointer analysis with the scope and options // specified by config, and returns the (synthetic) root of the callgraph. // func Analyze(config *Config) CallGraphNode { a := &analysis{ config: config, log: config.Log, prog: config.prog(), valNode: make(map[ssa.Value]nodeid), flattenMemo: make(map[types.Type][]*fieldInfo), hasher: typemap.MakeHasher(), intrinsics: make(map[*ssa.Function]intrinsic), funcObj: make(map[*ssa.Function]nodeid), probes: make(map[*ssa.CallCommon]nodeid), work: makeMapWorklist(), } if reflect := a.prog.ImportedPackage("reflect"); reflect != nil { a.reflectValueObj = reflect.Object.Scope().Lookup("Value") a.reflectRtypeObj = reflect.Object.Scope().Lookup("rtype") a.reflectRtype = types.NewPointer(a.reflectRtypeObj.Type()) // Override flattening of reflect.Value, treating it like a basic type. tReflectValue := a.reflectValueObj.Type() a.flattenMemo[tReflectValue] = []*fieldInfo{{typ: tReflectValue}} a.rtypes.SetHasher(a.hasher) a.reflectZeros.SetHasher(a.hasher) } if false { a.log = os.Stderr // for debugging crashes; extremely verbose } if a.log != nil { fmt.Fprintln(a.log, "======== NEW ANALYSIS ========") } root := a.generate() // ---------- Presolver ---------- // TODO(adonovan): opt: presolver optimisations. // ---------- Solver ---------- a.solve() if a.log != nil { // Dump solution. for i, n := range a.nodes { if n.pts != nil { fmt.Fprintf(a.log, "pts(n%d) = %s : %s\n", i, n.pts, n.typ) } } } // Notify the client of the callsites if they're interested. if CallSite := a.config.CallSite; CallSite != nil { for _, site := range a.callsites { CallSite(site) } } Call := a.config.Call for _, site := range a.callsites { for nid := range a.nodes[site.targets].pts { cgn := a.nodes[nid].obj.cgn // Notify the client of the call graph, if // they're interested. if Call != nil { Call(site, cgn) } // Warn about calls to non-intrinsic external functions. if fn := cgn.fn; fn.Blocks == nil && a.findIntrinsic(fn) == nil { a.warnf(site.Pos(), "unsound call to unknown intrinsic: %s", fn) a.warnf(fn.Pos(), " (declared here)") } } } return root }