181 lines
5.1 KiB
Go
181 lines
5.1 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 ssa
|
||
|
|
|
||
|
|
// This file defines utilities for population of method sets.
|
||
|
|
|
||
|
|
import (
|
||
|
|
"fmt"
|
||
|
|
"go/types"
|
||
|
|
|
||
|
|
"golang.org/x/tools/go/types/typeutil"
|
||
|
|
"golang.org/x/tools/internal/typesinternal"
|
||
|
|
)
|
||
|
|
|
||
|
|
// MethodValue returns the Function implementing method sel, building
|
||
|
|
// wrapper methods on demand. It returns nil if sel denotes an
|
||
|
|
// interface or generic method.
|
||
|
|
//
|
||
|
|
// Precondition: sel.Kind() == MethodVal.
|
||
|
|
//
|
||
|
|
// Thread-safe.
|
||
|
|
//
|
||
|
|
// Acquires prog.methodsMu.
|
||
|
|
func (prog *Program) MethodValue(sel *types.Selection) *Function {
|
||
|
|
if sel.Kind() != types.MethodVal {
|
||
|
|
panic(fmt.Sprintf("MethodValue(%s) kind != MethodVal", sel))
|
||
|
|
}
|
||
|
|
T := sel.Recv()
|
||
|
|
if types.IsInterface(T) {
|
||
|
|
return nil // interface method or type parameter
|
||
|
|
}
|
||
|
|
|
||
|
|
if prog.isParameterized(T) {
|
||
|
|
return nil // generic method
|
||
|
|
}
|
||
|
|
|
||
|
|
if prog.mode&LogSource != 0 {
|
||
|
|
defer logStack("MethodValue %s %v", T, sel)()
|
||
|
|
}
|
||
|
|
|
||
|
|
var b builder
|
||
|
|
|
||
|
|
m := func() *Function {
|
||
|
|
prog.methodsMu.Lock()
|
||
|
|
defer prog.methodsMu.Unlock()
|
||
|
|
|
||
|
|
// Get or create SSA method set.
|
||
|
|
mset, ok := prog.methodSets.At(T).(*methodSet)
|
||
|
|
if !ok {
|
||
|
|
mset = &methodSet{mapping: make(map[string]*Function)}
|
||
|
|
prog.methodSets.Set(T, mset)
|
||
|
|
}
|
||
|
|
|
||
|
|
// Get or create SSA method.
|
||
|
|
id := sel.Obj().Id()
|
||
|
|
fn, ok := mset.mapping[id]
|
||
|
|
if !ok {
|
||
|
|
obj := sel.Obj().(*types.Func)
|
||
|
|
needsPromotion := len(sel.Index()) > 1
|
||
|
|
needsIndirection := !isPointer(recvType(obj)) && isPointer(T)
|
||
|
|
if needsPromotion || needsIndirection {
|
||
|
|
fn = createWrapper(prog, toSelection(sel))
|
||
|
|
fn.buildshared = b.shared()
|
||
|
|
b.enqueue(fn)
|
||
|
|
} else {
|
||
|
|
fn = prog.objectMethod(obj, &b)
|
||
|
|
}
|
||
|
|
if fn.Signature.Recv() == nil {
|
||
|
|
panic(fn)
|
||
|
|
}
|
||
|
|
mset.mapping[id] = fn
|
||
|
|
} else {
|
||
|
|
b.waitForSharedFunction(fn)
|
||
|
|
}
|
||
|
|
|
||
|
|
return fn
|
||
|
|
}()
|
||
|
|
|
||
|
|
b.iterate()
|
||
|
|
|
||
|
|
return m
|
||
|
|
}
|
||
|
|
|
||
|
|
// objectMethod returns the Function for a given method symbol.
|
||
|
|
// The symbol may be an instance of a generic function. It need not
|
||
|
|
// belong to an existing SSA package created by a call to
|
||
|
|
// prog.CreatePackage.
|
||
|
|
//
|
||
|
|
// objectMethod panics if the function is not a method.
|
||
|
|
//
|
||
|
|
// Acquires prog.objectMethodsMu.
|
||
|
|
func (prog *Program) objectMethod(obj *types.Func, b *builder) *Function {
|
||
|
|
sig := obj.Type().(*types.Signature)
|
||
|
|
if sig.Recv() == nil {
|
||
|
|
panic("not a method: " + obj.String())
|
||
|
|
}
|
||
|
|
|
||
|
|
// Belongs to a created package?
|
||
|
|
if fn := prog.FuncValue(obj); fn != nil {
|
||
|
|
return fn
|
||
|
|
}
|
||
|
|
|
||
|
|
// Instantiation of generic?
|
||
|
|
if originObj := obj.Origin(); originObj != obj {
|
||
|
|
origin := prog.objectMethod(originObj, b)
|
||
|
|
assert(origin.typeparams.Len() > 0, "origin is not generic")
|
||
|
|
targs := receiverTypeArgs(obj)
|
||
|
|
return origin.instance(targs, b)
|
||
|
|
}
|
||
|
|
|
||
|
|
// Consult/update cache of methods created from types.Func.
|
||
|
|
prog.objectMethodsMu.Lock()
|
||
|
|
defer prog.objectMethodsMu.Unlock()
|
||
|
|
fn, ok := prog.objectMethods[obj]
|
||
|
|
if !ok {
|
||
|
|
fn = createFunction(prog, obj, obj.Name(), nil, nil, "")
|
||
|
|
fn.Synthetic = "from type information (on demand)"
|
||
|
|
fn.buildshared = b.shared()
|
||
|
|
b.enqueue(fn)
|
||
|
|
|
||
|
|
if prog.objectMethods == nil {
|
||
|
|
prog.objectMethods = make(map[*types.Func]*Function)
|
||
|
|
}
|
||
|
|
prog.objectMethods[obj] = fn
|
||
|
|
} else {
|
||
|
|
b.waitForSharedFunction(fn)
|
||
|
|
}
|
||
|
|
return fn
|
||
|
|
}
|
||
|
|
|
||
|
|
// LookupMethod returns the implementation of the method of type T
|
||
|
|
// identified by (pkg, name). It returns nil if the method exists but
|
||
|
|
// is an interface method or generic method, and panics if T has no such method.
|
||
|
|
func (prog *Program) LookupMethod(T types.Type, pkg *types.Package, name string) *Function {
|
||
|
|
sel := prog.MethodSets.MethodSet(T).Lookup(pkg, name)
|
||
|
|
if sel == nil {
|
||
|
|
panic(fmt.Sprintf("%s has no method %s", T, types.Id(pkg, name)))
|
||
|
|
}
|
||
|
|
return prog.MethodValue(sel)
|
||
|
|
}
|
||
|
|
|
||
|
|
// methodSet contains the (concrete) methods of a concrete type (non-interface, non-parameterized).
|
||
|
|
type methodSet struct {
|
||
|
|
mapping map[string]*Function // populated lazily
|
||
|
|
}
|
||
|
|
|
||
|
|
// RuntimeTypes returns a new unordered slice containing all types in
|
||
|
|
// the program for which a runtime type is required.
|
||
|
|
//
|
||
|
|
// A runtime type is required for any non-parameterized, non-interface
|
||
|
|
// type that is converted to an interface, or for any type (including
|
||
|
|
// interface types) derivable from one through reflection.
|
||
|
|
//
|
||
|
|
// The methods of such types may be reachable through reflection or
|
||
|
|
// interface calls even if they are never called directly.
|
||
|
|
//
|
||
|
|
// Thread-safe.
|
||
|
|
//
|
||
|
|
// Acquires prog.makeInterfaceTypesMu.
|
||
|
|
func (prog *Program) RuntimeTypes() []types.Type {
|
||
|
|
prog.makeInterfaceTypesMu.Lock()
|
||
|
|
defer prog.makeInterfaceTypesMu.Unlock()
|
||
|
|
|
||
|
|
// Compute the derived types on demand, since many SSA clients
|
||
|
|
// never call RuntimeTypes, and those that do typically call
|
||
|
|
// it once (often within ssautil.AllFunctions, which will
|
||
|
|
// eventually not use it; see Go issue #69291.) This
|
||
|
|
// eliminates the need to eagerly compute all the element
|
||
|
|
// types during SSA building.
|
||
|
|
var runtimeTypes []types.Type
|
||
|
|
add := func(t types.Type) { runtimeTypes = append(runtimeTypes, t) }
|
||
|
|
var set typeutil.Map // for de-duping identical types
|
||
|
|
for t := range prog.makeInterfaceTypes {
|
||
|
|
typesinternal.ForEachElement(&set, &prog.MethodSets, t, add)
|
||
|
|
}
|
||
|
|
|
||
|
|
return runtimeTypes
|
||
|
|
}
|