mirror of
https://github.com/golang/go
synced 2024-11-19 04:04:47 -07:00
3371b79a96
(reflect.Value).Send (reflect.Value).TrySend (reflect.Value).Recv (reflect.Value).TryRecv (reflect.Type).ChanOf (reflect.Type).In (reflect.Type).Out reflect.Indirect reflect.MakeChan Also: - specialize genInvoke when the receiver is a reflect.Type under the assumption that there's only one possible concrete type. This makes all reflect.Type operations context-sensitive since the calls are no longer dynamic. - Rename all variables to match the actual parameter names used in the reflect API. - Add pointer.Config.Reflection flag (exposed in oracle as --reflect, default false) to enable reflection. It currently adds about 20% running time. I'll make it true after the presolver is implemented. - Simplified worklist datatype and solver main loop slightly (~10% speed improvement). - Use addLabel() utility to add a label to a PTS. (Working on my 3 yr old 2x2GHz+4GB Mac vs 8x4GHz+24GB workstation, one really notices the cost of pointer analysis. Note to self: time to implement presolver.) R=crawshaw CC=golang-dev https://golang.org/cl/13242062
349 lines
8.5 KiB
Go
349 lines
8.5 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
|
|
|
|
// This file defines a naive Andersen-style solver for the inclusion
|
|
// constraint system.
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
"code.google.com/p/go.tools/go/types"
|
|
)
|
|
|
|
func (a *analysis) solve() {
|
|
// Solver main loop.
|
|
for round := 1; ; round++ {
|
|
if a.log != nil {
|
|
fmt.Fprintf(a.log, "Solving, round %d\n", round)
|
|
}
|
|
|
|
// Add new constraints to the graph:
|
|
// static constraints from SSA on round 1,
|
|
// dynamic constraints from reflection thereafter.
|
|
a.processNewConstraints()
|
|
|
|
id := a.work.take()
|
|
if id == empty {
|
|
break
|
|
}
|
|
if a.log != nil {
|
|
fmt.Fprintf(a.log, "\tnode n%d\n", id)
|
|
}
|
|
|
|
n := a.nodes[id]
|
|
|
|
// Difference propagation.
|
|
delta := n.pts.diff(n.prevPts)
|
|
if delta == nil {
|
|
continue
|
|
}
|
|
n.prevPts = n.pts.clone()
|
|
|
|
// Apply all resolution rules attached to n.
|
|
a.solveConstraints(n, delta)
|
|
|
|
if a.log != nil {
|
|
fmt.Fprintf(a.log, "\t\tpts(n%d) = %s\n", id, n.pts)
|
|
}
|
|
}
|
|
|
|
if a.log != nil {
|
|
fmt.Fprintf(a.log, "Solver done\n")
|
|
}
|
|
}
|
|
|
|
// processNewConstraints takes the new constraints from a.constraints
|
|
// and adds them to the graph, ensuring
|
|
// that new constraints are applied to pre-existing labels and
|
|
// that pre-existing constraints are applied to new labels.
|
|
//
|
|
func (a *analysis) processNewConstraints() {
|
|
// Take the slice of new constraints.
|
|
// (May grow during call to solveConstraints.)
|
|
constraints := a.constraints
|
|
a.constraints = nil
|
|
|
|
// Initialize points-to sets from addr-of (base) constraints.
|
|
for _, c := range constraints {
|
|
if c, ok := c.(*addrConstraint); ok {
|
|
dst := a.nodes[c.dst]
|
|
dst.pts.add(c.src)
|
|
|
|
// Populate the worklist with nodes that point to
|
|
// something initially (due to addrConstraints) and
|
|
// have other constraints attached.
|
|
// (A no-op in round 1.)
|
|
if dst.copyTo != nil || dst.complex != nil {
|
|
a.addWork(c.dst)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Attach simple (copy) and complex constraints to nodes.
|
|
var stale nodeset
|
|
for _, c := range constraints {
|
|
var id nodeid
|
|
switch c := c.(type) {
|
|
case *addrConstraint:
|
|
// base constraints handled in previous loop
|
|
continue
|
|
case *copyConstraint:
|
|
// simple (copy) constraint
|
|
id = c.src
|
|
a.nodes[id].copyTo.add(c.dst)
|
|
default:
|
|
// complex constraint
|
|
id = c.ptr()
|
|
a.nodes[id].complex.add(c)
|
|
}
|
|
|
|
if n := a.nodes[id]; len(n.pts) > 0 {
|
|
if len(n.prevPts) > 0 {
|
|
stale.add(id)
|
|
}
|
|
a.addWork(id)
|
|
}
|
|
}
|
|
// Apply new constraints to pre-existing PTS labels.
|
|
for id := range stale {
|
|
n := a.nodes[id]
|
|
a.solveConstraints(n, n.prevPts)
|
|
}
|
|
}
|
|
|
|
// solveConstraints applies each resolution rule attached to node n to
|
|
// the set of labels delta. It may generate new constraints in
|
|
// a.constraints.
|
|
//
|
|
func (a *analysis) solveConstraints(n *node, delta nodeset) {
|
|
if delta == nil {
|
|
return
|
|
}
|
|
|
|
// Process complex constraints dependent on n.
|
|
for c := range n.complex {
|
|
if a.log != nil {
|
|
fmt.Fprintf(a.log, "\t\tconstraint %s\n", c)
|
|
}
|
|
// TODO(adonovan): parameter n is never used. Remove?
|
|
c.solve(a, n, delta)
|
|
}
|
|
|
|
// Process copy constraints.
|
|
var copySeen nodeset
|
|
for mid := range n.copyTo {
|
|
if copySeen.add(mid) {
|
|
if a.nodes[mid].pts.addAll(delta) {
|
|
a.addWork(mid)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// addLabel adds label to the points-to set of ptr and reports whether the set grew.
|
|
func (a *analysis) addLabel(ptr, label nodeid) bool {
|
|
return a.nodes[ptr].pts.add(label)
|
|
}
|
|
|
|
func (a *analysis) addWork(id nodeid) {
|
|
a.work.add(id)
|
|
if a.log != nil {
|
|
fmt.Fprintf(a.log, "\t\twork: n%d\n", id)
|
|
}
|
|
}
|
|
|
|
func (c *addrConstraint) ptr() nodeid {
|
|
panic("addrConstraint: not a complex constraint")
|
|
}
|
|
func (c *copyConstraint) ptr() nodeid {
|
|
panic("addrConstraint: not a complex constraint")
|
|
}
|
|
|
|
// Complex constraints attach themselves to the relevant pointer node.
|
|
|
|
func (c *storeConstraint) ptr() nodeid {
|
|
return c.dst
|
|
}
|
|
func (c *loadConstraint) ptr() nodeid {
|
|
return c.src
|
|
}
|
|
func (c *offsetAddrConstraint) ptr() nodeid {
|
|
return c.src
|
|
}
|
|
func (c *typeAssertConstraint) ptr() nodeid {
|
|
return c.src
|
|
}
|
|
func (c *invokeConstraint) ptr() nodeid {
|
|
return c.iface
|
|
}
|
|
|
|
// onlineCopy adds a copy edge. It is called online, i.e. during
|
|
// solving, so it adds edges and pts members directly rather than by
|
|
// instantiating a 'constraint'.
|
|
//
|
|
// The size of the copy is implicitly 1.
|
|
// It returns true if pts(dst) changed.
|
|
//
|
|
func (a *analysis) onlineCopy(dst, src nodeid) bool {
|
|
if dst != src {
|
|
if nsrc := a.nodes[src]; nsrc.copyTo.add(dst) {
|
|
if a.log != nil {
|
|
fmt.Fprintf(a.log, "\t\t\tdynamic copy n%d <- n%d\n", dst, src)
|
|
}
|
|
return a.nodes[dst].pts.addAll(nsrc.pts)
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
// Returns sizeof.
|
|
// Implicitly adds nodes to worklist.
|
|
//
|
|
// TODO(adonovan): now that we support a.copy() during solving, we
|
|
// could eliminate onlineCopyN, but it's much slower. Investigate.
|
|
//
|
|
func (a *analysis) onlineCopyN(dst, src nodeid, sizeof uint32) uint32 {
|
|
for i := uint32(0); i < sizeof; i++ {
|
|
if a.onlineCopy(dst, src) {
|
|
a.addWork(dst)
|
|
}
|
|
src++
|
|
dst++
|
|
}
|
|
return sizeof
|
|
}
|
|
|
|
func (c *loadConstraint) solve(a *analysis, n *node, delta nodeset) {
|
|
var changed bool
|
|
for k := range delta {
|
|
koff := k + nodeid(c.offset)
|
|
if a.onlineCopy(c.dst, koff) {
|
|
changed = true
|
|
}
|
|
}
|
|
if changed {
|
|
a.addWork(c.dst)
|
|
}
|
|
}
|
|
|
|
func (c *storeConstraint) solve(a *analysis, n *node, delta nodeset) {
|
|
for k := range delta {
|
|
koff := k + nodeid(c.offset)
|
|
if a.onlineCopy(koff, c.src) {
|
|
a.addWork(koff)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (c *offsetAddrConstraint) solve(a *analysis, n *node, delta nodeset) {
|
|
dst := a.nodes[c.dst]
|
|
for k := range delta {
|
|
if dst.pts.add(k + nodeid(c.offset)) {
|
|
a.addWork(c.dst)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (c *typeAssertConstraint) solve(a *analysis, n *node, delta nodeset) {
|
|
tIface, _ := c.typ.Underlying().(*types.Interface)
|
|
|
|
for ifaceObj := range delta {
|
|
tDyn, v, indirect := a.taggedValue(ifaceObj)
|
|
if tDyn == nil {
|
|
panic("not a tagged value")
|
|
}
|
|
if indirect {
|
|
// TODO(adonovan): we'll need to implement this
|
|
// when we start creating indirect tagged objects.
|
|
panic("indirect tagged object")
|
|
}
|
|
|
|
if tIface != nil {
|
|
if types.IsAssignableTo(tDyn, tIface) {
|
|
if a.addLabel(c.dst, ifaceObj) {
|
|
a.addWork(c.dst)
|
|
}
|
|
}
|
|
} else {
|
|
if types.IsIdentical(tDyn, c.typ) {
|
|
// Copy entire payload to dst.
|
|
//
|
|
// TODO(adonovan): opt: if tConc is
|
|
// nonpointerlike we can skip this
|
|
// entire constraint, perhaps. We
|
|
// only care about pointers among the
|
|
// fields.
|
|
a.onlineCopyN(c.dst, v, a.sizeof(tDyn))
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (c *invokeConstraint) solve(a *analysis, n *node, delta nodeset) {
|
|
for ifaceObj := range delta {
|
|
tDyn, v, indirect := a.taggedValue(ifaceObj)
|
|
if tDyn == nil {
|
|
panic("not a tagged value")
|
|
}
|
|
if indirect {
|
|
// TODO(adonovan): we may need to implement this if
|
|
// we ever apply invokeConstraints to reflect.Value PTSs,
|
|
// e.g. for (reflect.Value).Call.
|
|
panic("indirect tagged object")
|
|
}
|
|
|
|
// Look up the concrete method.
|
|
meth := tDyn.MethodSet().Lookup(c.method.Pkg(), c.method.Name())
|
|
if meth == nil {
|
|
panic(fmt.Sprintf("n%d: type %s has no method %s (iface=n%d)",
|
|
c.iface, tDyn, c.method, ifaceObj))
|
|
}
|
|
fn := a.prog.Method(meth)
|
|
if fn == nil {
|
|
panic(fmt.Sprintf("n%d: no ssa.Function for %s", c.iface, meth))
|
|
}
|
|
sig := fn.Signature
|
|
|
|
fnObj := a.funcObj[fn] // dynamic calls use shared contour
|
|
if fnObj == 0 {
|
|
// a.valueNode(fn) was not called during gen phase.
|
|
panic(fmt.Sprintf("a.funcObj(%s)==nil", fn))
|
|
}
|
|
|
|
// Make callsite's fn variable point to identity of
|
|
// concrete method. (There's no need to add it to
|
|
// worklist since it never has attached constraints.)
|
|
a.addLabel(c.params, fnObj)
|
|
|
|
// Extract value and connect to method's receiver.
|
|
// Copy payload to method's receiver param (arg0).
|
|
arg0 := a.funcParams(fnObj)
|
|
recvSize := a.sizeof(sig.Recv().Type())
|
|
a.onlineCopyN(arg0, v, recvSize)
|
|
|
|
src := c.params + 1 // skip past identity
|
|
dst := arg0 + nodeid(recvSize)
|
|
|
|
// Copy caller's argument block to method formal parameters.
|
|
paramsSize := a.sizeof(sig.Params())
|
|
a.onlineCopyN(dst, src, paramsSize)
|
|
src += nodeid(paramsSize)
|
|
dst += nodeid(paramsSize)
|
|
|
|
// Copy method results to caller's result block.
|
|
resultsSize := a.sizeof(sig.Results())
|
|
a.onlineCopyN(src, dst, resultsSize)
|
|
}
|
|
}
|
|
|
|
func (c *addrConstraint) solve(a *analysis, n *node, delta nodeset) {
|
|
panic("addr is not a complex constraint")
|
|
}
|
|
|
|
func (c *copyConstraint) solve(a *analysis, n *node, delta nodeset) {
|
|
panic("copy is not a complex constraint")
|
|
}
|