611 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			611 lines
		
	
	
		
			25 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 implements Andersen's analysis, an inclusion-based
 | |
| pointer analysis algorithm first described in (Andersen, 1994).
 | |
| 
 | |
| A pointer analysis relates every pointer expression in a whole program
 | |
| to the set of memory locations to which it might point.  This
 | |
| information can be used to construct a call graph of the program that
 | |
| precisely represents the destinations of dynamic function and method
 | |
| calls.  It can also be used to determine, for example, which pairs of
 | |
| channel operations operate on the same channel.
 | |
| 
 | |
| The package allows the client to request a set of expressions of
 | |
| interest for which the points-to information will be returned once the
 | |
| analysis is complete.  In addition, the client may request that a
 | |
| callgraph is constructed.  The example program in example_test.go
 | |
| demonstrates both of these features.  Clients should not request more
 | |
| information than they need since it may increase the cost of the
 | |
| analysis significantly.
 | |
| 
 | |
| 
 | |
| CLASSIFICATION
 | |
| 
 | |
| Our algorithm is INCLUSION-BASED: the points-to sets for x and y will
 | |
| be related by pts(y) ⊇ pts(x) if the program contains the statement
 | |
| y = x.
 | |
| 
 | |
| It is FLOW-INSENSITIVE: it ignores all control flow constructs and the
 | |
| order of statements in a program.  It is therefore a "MAY ALIAS"
 | |
| analysis: its facts are of the form "P may/may not point to L",
 | |
| not "P must point to L".
 | |
| 
 | |
| It is FIELD-SENSITIVE: it builds separate points-to sets for distinct
 | |
| fields, such as x and y in struct { x, y *int }.
 | |
| 
 | |
| It is mostly CONTEXT-INSENSITIVE: most functions are analyzed once,
 | |
| so values can flow in at one call to the function and return out at
 | |
| another.  Only some smaller functions are analyzed with consideration
 | |
| of their calling context.
 | |
| 
 | |
| It has a CONTEXT-SENSITIVE HEAP: objects are named by both allocation
 | |
| site and context, so the objects returned by two distinct calls to f:
 | |
|    func f() *T { return new(T) }
 | |
| are distinguished up to the limits of the calling context.
 | |
| 
 | |
| It is a WHOLE PROGRAM analysis: it requires SSA-form IR for the
 | |
| complete Go program and summaries for native code.
 | |
| 
 | |
| See the (Hind, PASTE'01) survey paper for an explanation of these terms.
 | |
| 
 | |
| 
 | |
| SOUNDNESS
 | |
| 
 | |
| The analysis is fully sound when invoked on pure Go programs that do not
 | |
| use reflection or unsafe.Pointer conversions.  In other words, if there
 | |
| is any possible execution of the program in which pointer P may point to
 | |
| object O, the analysis will report that fact.
 | |
| 
 | |
| 
 | |
| REFLECTION
 | |
| 
 | |
| By default, the "reflect" library is ignored by the analysis, as if all
 | |
| its functions were no-ops, but if the client enables the Reflection flag,
 | |
| the analysis will make a reasonable attempt to model the effects of
 | |
| calls into this library.  However, this comes at a significant
 | |
| performance cost, and not all features of that library are yet
 | |
| implemented.  In addition, some simplifying approximations must be made
 | |
| to ensure that the analysis terminates; for example, reflection can be
 | |
| used to construct an infinite set of types and values of those types,
 | |
| but the analysis arbitrarily bounds the depth of such types.
 | |
| 
 | |
| Most but not all reflection operations are supported.
 | |
| In particular, addressable reflect.Values are not yet implemented, so
 | |
| operations such as (reflect.Value).Set have no analytic effect.
 | |
| 
 | |
| 
 | |
| UNSAFE POINTER CONVERSIONS
 | |
| 
 | |
| The pointer analysis makes no attempt to understand aliasing between the
 | |
| operand x and result y of an unsafe.Pointer conversion:
 | |
|    y = (*T)(unsafe.Pointer(x))
 | |
| It is as if the conversion allocated an entirely new object:
 | |
|    y = new(T)
 | |
| 
 | |
| 
 | |
| NATIVE CODE
 | |
| 
 | |
| The analysis cannot model the aliasing effects of functions written in
 | |
| languages other than Go, such as runtime intrinsics in C or assembly, or
 | |
| code accessed via cgo.  The result is as if such functions are no-ops.
 | |
| However, various important intrinsics are understood by the analysis,
 | |
| along with built-ins such as append.
 | |
| 
 | |
| The analysis currently provides no way for users to specify the aliasing
 | |
| effects of native code.
 | |
| 
 | |
| ------------------------------------------------------------------------
 | |
| 
 | |
| IMPLEMENTATION
 | |
| 
 | |
| The remaining documentation is intended for package maintainers and
 | |
| pointer analysis specialists.  Maintainers should have a solid
 | |
| understanding of the referenced papers (especially those by H&L and PKH)
 | |
| before making making significant changes.
 | |
| 
 | |
| The implementation is similar to that described in (Pearce et al,
 | |
| PASTE'04).  Unlike many algorithms which interleave constraint
 | |
| generation and solving, constructing the callgraph as they go, this
 | |
| implementation for the most part observes a phase ordering (generation
 | |
| before solving), with only simple (copy) constraints being generated
 | |
| during solving.  (The exception is reflection, which creates various
 | |
| constraints during solving as new types flow to reflect.Value
 | |
| operations.)  This improves the traction of presolver optimisations,
 | |
| but imposes certain restrictions, e.g. potential context sensitivity
 | |
| is limited since all variants must be created a priori.
 | |
| 
 | |
| 
 | |
| TERMINOLOGY
 | |
| 
 | |
| A type is said to be "pointer-like" if it is a reference to an object.
 | |
| Pointer-like types include pointers and also interfaces, maps, channels,
 | |
| functions and slices.
 | |
| 
 | |
| We occasionally use C's x->f notation to distinguish the case where x
 | |
| is a struct pointer from x.f where is a struct value.
 | |
| 
 | |
| Pointer analysis literature (and our comments) often uses the notation
 | |
| dst=*src+offset to mean something different than what it means in Go.
 | |
| It means: for each node index p in pts(src), the node index p+offset is
 | |
| in pts(dst).  Similarly *dst+offset=src is used for store constraints
 | |
| and dst=src+offset for offset-address constraints.
 | |
| 
 | |
| 
 | |
| NODES
 | |
| 
 | |
| Nodes are the key datastructure of the analysis, and have a dual role:
 | |
| they represent both constraint variables (equivalence classes of
 | |
| pointers) and members of points-to sets (things that can be pointed
 | |
| at, i.e. "labels").
 | |
| 
 | |
| Nodes are naturally numbered.  The numbering enables compact
 | |
| representations of sets of nodes such as bitvectors (or BDDs); and the
 | |
| ordering enables a very cheap way to group related nodes together.  For
 | |
| example, passing n parameters consists of generating n parallel
 | |
| constraints from caller+i to callee+i for 0<=i<n.
 | |
| 
 | |
| The zero nodeid means "not a pointer".  For simplicity, we generate flow
 | |
| constraints even for non-pointer types such as int.  The pointer
 | |
| equivalence (PE) presolver optimization detects which variables cannot
 | |
| point to anything; this includes not only all variables of non-pointer
 | |
| types (such as int) but also variables of pointer-like types if they are
 | |
| always nil, or are parameters to a function that is never called.
 | |
| 
 | |
| Each node represents a scalar part of a value or object.
 | |
| Aggregate types (structs, tuples, arrays) are recursively flattened
 | |
| out into a sequential list of scalar component types, and all the
 | |
| elements of an array are represented by a single node.  (The
 | |
| flattening of a basic type is a list containing a single node.)
 | |
| 
 | |
| Nodes are connected into a graph with various kinds of labelled edges:
 | |
| simple edges (or copy constraints) represent value flow.  Complex
 | |
| edges (load, store, etc) trigger the creation of new simple edges
 | |
| during the solving phase.
 | |
| 
 | |
| 
 | |
| OBJECTS
 | |
| 
 | |
| Conceptually, an "object" is a contiguous sequence of nodes denoting
 | |
| an addressable location: something that a pointer can point to.  The
 | |
| first node of an object has a non-nil obj field containing information
 | |
| about the allocation: its size, context, and ssa.Value.
 | |
| 
 | |
| Objects include:
 | |
|    - functions and globals;
 | |
|    - variable allocations in the stack frame or heap;
 | |
|    - maps, channels and slices created by calls to make();
 | |
|    - allocations to construct an interface;
 | |
|    - allocations caused by conversions, e.g. []byte(str).
 | |
|    - arrays allocated by calls to append();
 | |
| 
 | |
| Many objects have no Go types.  For example, the func, map and chan type
 | |
| kinds in Go are all varieties of pointers, but their respective objects
 | |
| are actual functions (executable code), maps (hash tables), and channels
 | |
| (synchronized queues).  Given the way we model interfaces, they too are
 | |
| pointers to "tagged" objects with no Go type.  And an *ssa.Global denotes
 | |
| the address of a global variable, but the object for a Global is the
 | |
| actual data.  So, the types of an ssa.Value that creates an object is
 | |
| "off by one indirection": a pointer to the object.
 | |
| 
 | |
| The individual nodes of an object are sometimes referred to as "labels".
 | |
| 
 | |
| For uniformity, all objects have a non-zero number of fields, even those
 | |
| of the empty type struct{}.  (All arrays are treated as if of length 1,
 | |
| so there are no empty arrays.  The empty tuple is never address-taken,
 | |
| so is never an object.)
 | |
| 
 | |
| 
 | |
| TAGGED OBJECTS
 | |
| 
 | |
| An tagged object has the following layout:
 | |
| 
 | |
|     T          -- obj.flags ⊇ {otTagged}
 | |
|     v
 | |
|     ...
 | |
| 
 | |
| The T node's typ field is the dynamic type of the "payload": the value
 | |
| v which follows, flattened out.  The T node's obj has the otTagged
 | |
| flag.
 | |
| 
 | |
| Tagged objects are needed when generalizing across types: interfaces,
 | |
| reflect.Values, reflect.Types.  Each of these three types is modelled
 | |
| as a pointer that exclusively points to tagged objects.
 | |
| 
 | |
| Tagged objects may be indirect (obj.flags ⊇ {otIndirect}) meaning that
 | |
| the value v is not of type T but *T; this is used only for
 | |
| reflect.Values that represent lvalues.  (These are not implemented yet.)
 | |
| 
 | |
| 
 | |
| ANALYSIS ABSTRACTION OF EACH TYPE
 | |
| 
 | |
| Variables of the following "scalar" types may be represented by a
 | |
| single node: basic types, pointers, channels, maps, slices, 'func'
 | |
| pointers, interfaces.
 | |
| 
 | |
| Pointers
 | |
|   Nothing to say here, oddly.
 | |
| 
 | |
| Basic types (bool, string, numbers, unsafe.Pointer)
 | |
|   Currently all fields in the flattening of a type, including
 | |
|   non-pointer basic types such as int, are represented in objects and
 | |
|   values.  Though non-pointer nodes within values are uninteresting,
 | |
|   non-pointer nodes in objects may be useful (if address-taken)
 | |
|   because they permit the analysis to deduce, in this example,
 | |
| 
 | |
|      var s struct{ ...; x int; ... }
 | |
|      p := &s.x
 | |
| 
 | |
|   that p points to s.x.  If we ignored such object fields, we could only
 | |
|   say that p points somewhere within s.
 | |
| 
 | |
|   All other basic types are ignored.  Expressions of these types have
 | |
|   zero nodeid, and fields of these types within aggregate other types
 | |
|   are omitted.
 | |
| 
 | |
|   unsafe.Pointers are not modelled as pointers, so a conversion of an
 | |
|   unsafe.Pointer to *T is (unsoundly) treated equivalent to new(T).
 | |
| 
 | |
| Channels
 | |
|   An expression of type 'chan T' is a kind of pointer that points
 | |
|   exclusively to channel objects, i.e. objects created by MakeChan (or
 | |
|   reflection).
 | |
| 
 | |
|   'chan T' is treated like *T.
 | |
|   *ssa.MakeChan is treated as equivalent to new(T).
 | |
|   *ssa.Send and receive (*ssa.UnOp(ARROW)) and are equivalent to store
 | |
|    and load.
 | |
| 
 | |
| Maps
 | |
|   An expression of type 'map[K]V' is a kind of pointer that points
 | |
|   exclusively to map objects, i.e. objects created by MakeMap (or
 | |
|   reflection).
 | |
| 
 | |
|   map K[V] is treated like *M where M = struct{k K; v V}.
 | |
|   *ssa.MakeMap is equivalent to new(M).
 | |
|   *ssa.MapUpdate is equivalent to *y=x where *y and x have type M.
 | |
|   *ssa.Lookup is equivalent to y=x.v where x has type *M.
 | |
| 
 | |
| Slices
 | |
|   A slice []T, which dynamically resembles a struct{array *T, len, cap int},
 | |
|   is treated as if it were just a *T pointer; the len and cap fields are
 | |
|   ignored.
 | |
| 
 | |
|   *ssa.MakeSlice is treated like new([1]T): an allocation of a
 | |
|    singleton array.
 | |
|   *ssa.Index on a slice is equivalent to a load.
 | |
|   *ssa.IndexAddr on a slice returns the address of the sole element of the
 | |
|   slice, i.e. the same address.
 | |
|   *ssa.Slice is treated as a simple copy.
 | |
| 
 | |
| Functions
 | |
|   An expression of type 'func...' is a kind of pointer that points
 | |
|   exclusively to function objects.
 | |
| 
 | |
|   A function object has the following layout:
 | |
| 
 | |
|      identity         -- typ:*types.Signature; obj.flags ⊇ {otFunction}
 | |
|      params_0         -- (the receiver, if a method)
 | |
|      ...
 | |
|      params_n-1
 | |
|      results_0
 | |
|      ...
 | |
|      results_m-1
 | |
| 
 | |
|   There may be multiple function objects for the same *ssa.Function
 | |
|   due to context-sensitive treatment of some functions.
 | |
| 
 | |
|   The first node is the function's identity node.
 | |
|   Associated with every callsite is a special "targets" variable,
 | |
|   whose pts() contains the identity node of each function to which
 | |
|   the call may dispatch.  Identity words are not otherwise used during
 | |
|   the analysis, but we construct the call graph from the pts()
 | |
|   solution for such nodes.
 | |
| 
 | |
|   The following block of contiguous nodes represents the flattened-out
 | |
|   types of the parameters ("P-block") and results ("R-block") of the
 | |
|   function object.
 | |
| 
 | |
|   The treatment of free variables of closures (*ssa.FreeVar) is like
 | |
|   that of global variables; it is not context-sensitive.
 | |
|   *ssa.MakeClosure instructions create copy edges to Captures.
 | |
| 
 | |
|   A Go value of type 'func' (i.e. a pointer to one or more functions)
 | |
|   is a pointer whose pts() contains function objects.  The valueNode()
 | |
|   for an *ssa.Function returns a singleton for that function.
 | |
| 
 | |
| Interfaces
 | |
|   An expression of type 'interface{...}' is a kind of pointer that
 | |
|   points exclusively to tagged objects.  All tagged objects pointed to
 | |
|   by an interface are direct (the otIndirect flag is clear) and
 | |
|   concrete (the tag type T is not itself an interface type).  The
 | |
|   associated ssa.Value for an interface's tagged objects may be an
 | |
|   *ssa.MakeInterface instruction, or nil if the tagged object was
 | |
|   created by an instrinsic (e.g. reflection).
 | |
| 
 | |
|   Constructing an interface value causes generation of constraints for
 | |
|   all of the concrete type's methods; we can't tell a priori which
 | |
|   ones may be called.
 | |
| 
 | |
|   TypeAssert y = x.(T) is implemented by a dynamic constraint
 | |
|   triggered by each tagged object O added to pts(x): a typeFilter
 | |
|   constraint if T is an interface type, or an untag constraint if T is
 | |
|   a concrete type.  A typeFilter tests whether O.typ implements T; if
 | |
|   so, O is added to pts(y).  An untagFilter tests whether O.typ is
 | |
|   assignable to T,and if so, a copy edge O.v -> y is added.
 | |
| 
 | |
|   ChangeInterface is a simple copy because the representation of
 | |
|   tagged objects is independent of the interface type (in contrast
 | |
|   to the "method tables" approach used by the gc runtime).
 | |
| 
 | |
|   y := Invoke x.m(...) is implemented by allocating contiguous P/R
 | |
|   blocks for the callsite and adding a dynamic rule triggered by each
 | |
|   tagged object added to pts(x).  The rule adds param/results copy
 | |
|   edges to/from each discovered concrete method.
 | |
| 
 | |
|   (Q. Why do we model an interface as a pointer to a pair of type and
 | |
|   value, rather than as a pair of a pointer to type and a pointer to
 | |
|   value?
 | |
|   A. Control-flow joins would merge interfaces ({T1}, {V1}) and ({T2},
 | |
|   {V2}) to make ({T1,T2}, {V1,V2}), leading to the infeasible and
 | |
|   type-unsafe combination (T1,V2).  Treating the value and its concrete
 | |
|   type as inseparable makes the analysis type-safe.)
 | |
| 
 | |
| reflect.Value
 | |
|   A reflect.Value is modelled very similar to an interface{}, i.e. as
 | |
|   a pointer exclusively to tagged objects, but with two generalizations.
 | |
| 
 | |
|   1) a reflect.Value that represents an lvalue points to an indirect
 | |
|      (obj.flags ⊇ {otIndirect}) tagged object, which has a similar
 | |
|      layout to an tagged object except that the value is a pointer to
 | |
|      the dynamic type.  Indirect tagged objects preserve the correct
 | |
|      aliasing so that mutations made by (reflect.Value).Set can be
 | |
|      observed.
 | |
| 
 | |
|      Indirect objects only arise when an lvalue is derived from an
 | |
|      rvalue by indirection, e.g. the following code:
 | |
| 
 | |
|         type S struct { X T }
 | |
|         var s S
 | |
|         var i interface{} = &s    // i points to a *S-tagged object (from MakeInterface)
 | |
|         v1 := reflect.ValueOf(i)  // v1 points to same *S-tagged object as i
 | |
|         v2 := v1.Elem()           // v2 points to an indirect S-tagged object, pointing to s
 | |
|         v3 := v2.FieldByName("X") // v3 points to an indirect int-tagged object, pointing to s.X
 | |
|         v3.Set(y)                 // pts(s.X) ⊇ pts(y)
 | |
| 
 | |
|      Whether indirect or not, the concrete type of the tagged object
 | |
|      corresponds to the user-visible dynamic type, and the existence
 | |
|      of a pointer is an implementation detail.
 | |
| 
 | |
|      (NB: indirect tagged objects are not yet implemented)
 | |
| 
 | |
|   2) The dynamic type tag of a tagged object pointed to by a
 | |
|      reflect.Value may be an interface type; it need not be concrete.
 | |
| 
 | |
|      This arises in code such as this:
 | |
|         tEface := reflect.TypeOf(new(interface{}).Elem() // interface{}
 | |
|         eface := reflect.Zero(tEface)
 | |
|      pts(eface) is a singleton containing an interface{}-tagged
 | |
|      object.  That tagged object's payload is an interface{} value,
 | |
|      i.e. the pts of the payload contains only concrete-tagged
 | |
|      objects, although in this example it's the zero interface{} value,
 | |
|      so its pts is empty.
 | |
| 
 | |
| reflect.Type
 | |
|   Just as in the real "reflect" library, we represent a reflect.Type
 | |
|   as an interface whose sole implementation is the concrete type,
 | |
|   *reflect.rtype.  (This choice is forced on us by go/types: clients
 | |
|   cannot fabricate types with arbitrary method sets.)
 | |
| 
 | |
|   rtype instances are canonical: there is at most one per dynamic
 | |
|   type.  (rtypes are in fact large structs but since identity is all
 | |
|   that matters, we represent them by a single node.)
 | |
| 
 | |
|   The payload of each *rtype-tagged object is an *rtype pointer that
 | |
|   points to exactly one such canonical rtype object.  We exploit this
 | |
|   by setting the node.typ of the payload to the dynamic type, not
 | |
|   '*rtype'.  This saves us an indirection in each resolution rule.  As
 | |
|   an optimisation, *rtype-tagged objects are canonicalized too.
 | |
| 
 | |
| 
 | |
| Aggregate types:
 | |
| 
 | |
| Aggregate types are treated as if all directly contained
 | |
| aggregates are recursively flattened out.
 | |
| 
 | |
| Structs
 | |
|   *ssa.Field y = x.f creates a simple edge to y from x's node at f's offset.
 | |
| 
 | |
|   *ssa.FieldAddr y = &x->f requires a dynamic closure rule to create
 | |
|    simple edges for each struct discovered in pts(x).
 | |
| 
 | |
|   The nodes of a struct consist of a special 'identity' node (whose
 | |
|   type is that of the struct itself), followed by the nodes for all
 | |
|   the struct's fields, recursively flattened out.  A pointer to the
 | |
|   struct is a pointer to its identity node.  That node allows us to
 | |
|   distinguish a pointer to a struct from a pointer to its first field.
 | |
| 
 | |
|   Field offsets are logical field offsets (plus one for the identity
 | |
|   node), so the sizes of the fields can be ignored by the analysis.
 | |
| 
 | |
|   (The identity node is non-traditional but enables the distiction
 | |
|   described above, which is valuable for code comprehension tools.
 | |
|   Typical pointer analyses for C, whose purpose is compiler
 | |
|   optimization, must soundly model unsafe.Pointer (void*) conversions,
 | |
|   and this requires fidelity to the actual memory layout using physical
 | |
|   field offsets.)
 | |
| 
 | |
|   *ssa.Field y = x.f creates a simple edge to y from x's node at f's offset.
 | |
| 
 | |
|   *ssa.FieldAddr y = &x->f requires a dynamic closure rule to create
 | |
|    simple edges for each struct discovered in pts(x).
 | |
| 
 | |
| Arrays
 | |
|   We model an array by an identity node (whose type is that of the
 | |
|   array itself) followed by a node representing all the elements of
 | |
|   the array; the analysis does not distinguish elements with different
 | |
|   indices.  Effectively, an array is treated like struct{elem T}, a
 | |
|   load y=x[i] like y=x.elem, and a store x[i]=y like x.elem=y; the
 | |
|   index i is ignored.
 | |
| 
 | |
|   A pointer to an array is pointer to its identity node.  (A slice is
 | |
|   also a pointer to an array's identity node.)  The identity node
 | |
|   allows us to distinguish a pointer to an array from a pointer to one
 | |
|   of its elements, but it is rather costly because it introduces more
 | |
|   offset constraints into the system.  Furthermore, sound treatment of
 | |
|   unsafe.Pointer would require us to dispense with this node.
 | |
| 
 | |
|   Arrays may be allocated by Alloc, by make([]T), by calls to append,
 | |
|   and via reflection.
 | |
| 
 | |
| Tuples (T, ...)
 | |
|   Tuples are treated like structs with naturally numbered fields.
 | |
|   *ssa.Extract is analogous to *ssa.Field.
 | |
| 
 | |
|   However, tuples have no identity field since by construction, they
 | |
|   cannot be address-taken.
 | |
| 
 | |
| 
 | |
| FUNCTION CALLS
 | |
| 
 | |
|   There are three kinds of function call:
 | |
|   (1) static "call"-mode calls of functions.
 | |
|   (2) dynamic "call"-mode calls of functions.
 | |
|   (3) dynamic "invoke"-mode calls of interface methods.
 | |
|   Cases 1 and 2 apply equally to methods and standalone functions.
 | |
| 
 | |
|   Static calls.
 | |
|     A static call consists three steps:
 | |
|     - finding the function object of the callee;
 | |
|     - creating copy edges from the actual parameter value nodes to the
 | |
|       P-block in the function object (this includes the receiver if
 | |
|       the callee is a method);
 | |
|     - creating copy edges from the R-block in the function object to
 | |
|       the value nodes for the result of the call.
 | |
| 
 | |
|     A static function call is little more than two struct value copies
 | |
|     between the P/R blocks of caller and callee:
 | |
| 
 | |
|        callee.P = caller.P
 | |
|        caller.R = callee.R
 | |
| 
 | |
|     Context sensitivity
 | |
| 
 | |
|       Static calls (alone) may be treated context sensitively,
 | |
|       i.e. each callsite may cause a distinct re-analysis of the
 | |
|       callee, improving precision.  Our current context-sensitivity
 | |
|       policy treats all intrinsics and getter/setter methods in this
 | |
|       manner since such functions are small and seem like an obvious
 | |
|       source of spurious confluences, though this has not yet been
 | |
|       evaluated.
 | |
| 
 | |
|   Dynamic function calls
 | |
| 
 | |
|     Dynamic calls work in a similar manner except that the creation of
 | |
|     copy edges occurs dynamically, in a similar fashion to a pair of
 | |
|     struct copies in which the callee is indirect:
 | |
| 
 | |
|        callee->P = caller.P
 | |
|        caller.R = callee->R
 | |
| 
 | |
|     (Recall that the function object's P- and R-blocks are contiguous.)
 | |
| 
 | |
|   Interface method invocation
 | |
| 
 | |
|     For invoke-mode calls, we create a params/results block for the
 | |
|     callsite and attach a dynamic closure rule to the interface.  For
 | |
|     each new tagged object that flows to the interface, we look up
 | |
|     the concrete method, find its function object, and connect its P/R
 | |
|     blocks to the callsite's P/R blocks, adding copy edges to the graph
 | |
|     during solving.
 | |
| 
 | |
|   Recording call targets
 | |
| 
 | |
|     The analysis notifies its clients of each callsite it encounters,
 | |
|     passing a CallSite interface.  Among other things, the CallSite
 | |
|     contains a synthetic constraint variable ("targets") whose
 | |
|     points-to solution includes the set of all function objects to
 | |
|     which the call may dispatch.
 | |
| 
 | |
|     It is via this mechanism that the callgraph is made available.
 | |
|     Clients may also elect to be notified of callgraph edges directly;
 | |
|     internally this just iterates all "targets" variables' pts(·)s.
 | |
| 
 | |
| 
 | |
| PRESOLVER
 | |
| 
 | |
| We implement Hash-Value Numbering (HVN), a pre-solver constraint
 | |
| optimization described in Hardekopf & Lin, SAS'07.  This is documented
 | |
| in more detail in hvn.go.  We intend to add its cousins HR and HU in
 | |
| future.
 | |
| 
 | |
| 
 | |
| SOLVER
 | |
| 
 | |
| The solver is currently a naive Andersen-style implementation; it does
 | |
| not perform online cycle detection, though we plan to add solver
 | |
| optimisations such as Hybrid- and Lazy- Cycle Detection from (Hardekopf
 | |
| & Lin, PLDI'07).
 | |
| 
 | |
| It uses difference propagation (Pearce et al, SQC'04) to avoid
 | |
| redundant re-triggering of closure rules for values already seen.
 | |
| 
 | |
| Points-to sets are represented using sparse bit vectors (similar to
 | |
| those used in LLVM and gcc), which are more space- and time-efficient
 | |
| than sets based on Go's built-in map type or dense bit vectors.
 | |
| 
 | |
| Nodes are permuted prior to solving so that object nodes (which may
 | |
| appear in points-to sets) are lower numbered than non-object (var)
 | |
| nodes.  This improves the density of the set over which the PTSs
 | |
| range, and thus the efficiency of the representation.
 | |
| 
 | |
| Partly thanks to avoiding map iteration, the execution of the solver is
 | |
| 100% deterministic, a great help during debugging.
 | |
| 
 | |
| 
 | |
| FURTHER READING
 | |
| 
 | |
| Andersen, L. O. 1994. Program analysis and specialization for the C
 | |
| programming language. Ph.D. dissertation. DIKU, University of
 | |
| Copenhagen.
 | |
| 
 | |
| David J. Pearce, Paul H. J. Kelly, and Chris Hankin. 2004.  Efficient
 | |
| field-sensitive pointer analysis for C. In Proceedings of the 5th ACM
 | |
| SIGPLAN-SIGSOFT workshop on Program analysis for software tools and
 | |
| engineering (PASTE '04). ACM, New York, NY, USA, 37-42.
 | |
| http://doi.acm.org/10.1145/996821.996835
 | |
| 
 | |
| David J. Pearce, Paul H. J. Kelly, and Chris Hankin. 2004. Online
 | |
| Cycle Detection and Difference Propagation: Applications to Pointer
 | |
| Analysis. Software Quality Control 12, 4 (December 2004), 311-337.
 | |
| http://dx.doi.org/10.1023/B:SQJO.0000039791.93071.a2
 | |
| 
 | |
| David Grove and Craig Chambers. 2001. A framework for call graph
 | |
| construction algorithms. ACM Trans. Program. Lang. Syst. 23, 6
 | |
| (November 2001), 685-746.
 | |
| http://doi.acm.org/10.1145/506315.506316
 | |
| 
 | |
| Ben Hardekopf and Calvin Lin. 2007. The ant and the grasshopper: fast
 | |
| and accurate pointer analysis for millions of lines of code. In
 | |
| Proceedings of the 2007 ACM SIGPLAN conference on Programming language
 | |
| design and implementation (PLDI '07). ACM, New York, NY, USA, 290-299.
 | |
| http://doi.acm.org/10.1145/1250734.1250767
 | |
| 
 | |
| Ben Hardekopf and Calvin Lin. 2007. Exploiting pointer and location
 | |
| equivalence to optimize pointer analysis. In Proceedings of the 14th
 | |
| international conference on Static Analysis (SAS'07), Hanne Riis
 | |
| Nielson and Gilberto Filé (Eds.). Springer-Verlag, Berlin, Heidelberg,
 | |
| 265-280.
 | |
| 
 | |
| Atanas Rountev and Satish Chandra. 2000. Off-line variable substitution
 | |
| for scaling points-to analysis. In Proceedings of the ACM SIGPLAN 2000
 | |
| conference on Programming language design and implementation (PLDI '00).
 | |
| ACM, New York, NY, USA, 47-56. DOI=10.1145/349299.349310
 | |
| http://doi.acm.org/10.1145/349299.349310
 | |
| 
 | |
| */
 | |
| package pointer // import "golang.org/x/tools/go/pointer"
 |