mirror of
https://github.com/golang/go
synced 2024-11-22 03:14:41 -07:00
cmd/api: follow constant references
For gccgo. Also removes bunch of special cases. Fixes #2906 R=golang-dev, remyoudompheng CC=golang-dev https://golang.org/cl/5644050
This commit is contained in:
parent
4539d1f307
commit
c15a42ed76
@ -3,6 +3,11 @@
|
||||
// license that can be found in the LICENSE file.
|
||||
|
||||
// Api computes the exported API of a set of Go packages.
|
||||
//
|
||||
// BUG(bradfitz): Note that this tool is only currently suitable
|
||||
// for use on the Go standard library, not arbitrary packages.
|
||||
// Once the Go AST has type information, this tool will be more
|
||||
// reliable without hard-coded hacks throughout.
|
||||
package main
|
||||
|
||||
import (
|
||||
@ -167,7 +172,8 @@ type Walker struct {
|
||||
lastConstType string
|
||||
curPackageName string
|
||||
curPackage *ast.Package
|
||||
prevConstType map[string]string // identifier -> "ideal-int"
|
||||
prevConstType map[pkgSymbol]string
|
||||
constDep map[string]string // key's const identifier has type of future value const identifier
|
||||
packageState map[string]loadState
|
||||
interfaces map[pkgSymbol]*ast.InterfaceType
|
||||
selectorFullPkg map[string]string // "http" => "net/http", updated by imports
|
||||
@ -182,6 +188,7 @@ func NewWalker() *Walker {
|
||||
interfaces: make(map[pkgSymbol]*ast.InterfaceType),
|
||||
selectorFullPkg: make(map[string]string),
|
||||
wantedPkg: make(map[string]bool),
|
||||
prevConstType: make(map[pkgSymbol]string),
|
||||
}
|
||||
}
|
||||
|
||||
@ -199,34 +206,10 @@ const (
|
||||
// the cases we can't handle yet.
|
||||
func (w *Walker) hardCodedConstantType(name string) (typ string, ok bool) {
|
||||
switch w.scope[0] {
|
||||
case "pkg compress/gzip", "pkg compress/zlib":
|
||||
case "pkg syscall":
|
||||
switch name {
|
||||
case "NoCompression", "BestSpeed", "BestCompression", "DefaultCompression":
|
||||
return "ideal-int", true
|
||||
}
|
||||
case "pkg os":
|
||||
switch name {
|
||||
case "WNOHANG", "WSTOPPED", "WUNTRACED":
|
||||
return "ideal-int", true
|
||||
}
|
||||
case "pkg path/filepath":
|
||||
switch name {
|
||||
case "Separator", "ListSeparator":
|
||||
return "char", true
|
||||
}
|
||||
case "pkg unicode/utf8":
|
||||
switch name {
|
||||
case "RuneError":
|
||||
return "char", true
|
||||
}
|
||||
case "pkg text/scanner":
|
||||
// TODO: currently this tool only resolves const types
|
||||
// that reference other constant types if they appear
|
||||
// in the right order. the scanner package has
|
||||
// ScanIdents and such coming before the Ident/Int/etc
|
||||
// tokens, hence this hack.
|
||||
if strings.HasPrefix(name, "Scan") || name == "SkipComments" {
|
||||
return "ideal-int", true
|
||||
case "darwinAMD64":
|
||||
return "ideal-bool", true
|
||||
}
|
||||
}
|
||||
return "", false
|
||||
@ -306,7 +289,7 @@ func (w *Walker) WalkPackage(name string) {
|
||||
|
||||
w.curPackageName = name
|
||||
w.curPackage = apkg
|
||||
w.prevConstType = map[string]string{}
|
||||
w.constDep = map[string]string{}
|
||||
|
||||
for _, afile := range apkg.Files {
|
||||
w.recordTypes(afile)
|
||||
@ -316,6 +299,8 @@ func (w *Walker) WalkPackage(name string) {
|
||||
w.walkFile(afile)
|
||||
}
|
||||
|
||||
w.resolveConstantDeps()
|
||||
|
||||
// Now that we're done walking types, vars and consts
|
||||
// in the *ast.Package, use go/doc to do the rest
|
||||
// (functions and methods). This is done here because
|
||||
@ -447,8 +432,16 @@ func (w *Walker) constValueType(vi interface{}) (string, error) {
|
||||
case *ast.UnaryExpr:
|
||||
return w.constValueType(v.X)
|
||||
case *ast.SelectorExpr:
|
||||
// e.g. compress/gzip's BestSpeed == flate.BestSpeed
|
||||
return "", errTODO
|
||||
lhs := w.nodeString(v.X)
|
||||
rhs := w.nodeString(v.Sel)
|
||||
pkg, ok := w.selectorFullPkg[lhs]
|
||||
if !ok {
|
||||
return "", fmt.Errorf("unknown constant reference; unknown package in expression %s.%s", lhs, rhs)
|
||||
}
|
||||
if t, ok := w.prevConstType[pkgSymbol{pkg, rhs}]; ok {
|
||||
return t, nil
|
||||
}
|
||||
return "", fmt.Errorf("unknown constant reference to %s.%s", lhs, rhs)
|
||||
case *ast.Ident:
|
||||
if v.Name == "iota" {
|
||||
return "ideal-int", nil // hack.
|
||||
@ -460,10 +453,10 @@ func (w *Walker) constValueType(vi interface{}) (string, error) {
|
||||
// Hack.
|
||||
return "ideal-int", nil
|
||||
}
|
||||
if t, ok := w.prevConstType[v.Name]; ok {
|
||||
if t, ok := w.prevConstType[pkgSymbol{w.curPackageName, v.Name}]; ok {
|
||||
return t, nil
|
||||
}
|
||||
return "", fmt.Errorf("can't resolve existing constant %q", v.Name)
|
||||
return constDepPrefix + v.Name, nil
|
||||
case *ast.BinaryExpr:
|
||||
left, err := w.constValueType(v.X)
|
||||
if err != nil {
|
||||
@ -474,6 +467,8 @@ func (w *Walker) constValueType(vi interface{}) (string, error) {
|
||||
return "", err
|
||||
}
|
||||
if left != right {
|
||||
// TODO(bradfitz): encode the real rules here,
|
||||
// rather than this mess.
|
||||
if left == "ideal-int" && right == "ideal-float" {
|
||||
return "ideal-float", nil // math.Log2E
|
||||
}
|
||||
@ -487,6 +482,17 @@ func (w *Walker) constValueType(vi interface{}) (string, error) {
|
||||
// Hack, for package time.
|
||||
return "Duration", nil
|
||||
}
|
||||
if left == "ideal-int" && !strings.HasPrefix(right, "ideal-") {
|
||||
return right, nil
|
||||
}
|
||||
if right == "ideal-int" && !strings.HasPrefix(left, "ideal-") {
|
||||
return left, nil
|
||||
}
|
||||
if strings.HasPrefix(left, constDepPrefix) && strings.HasPrefix(right, constDepPrefix) {
|
||||
// Just pick one.
|
||||
// e.g. text/scanner GoTokens const-dependency:ScanIdents, const-dependency:ScanFloats
|
||||
return left, nil
|
||||
}
|
||||
return "", fmt.Errorf("in BinaryExpr, unhandled type mismatch; left=%q, right=%q", left, right)
|
||||
}
|
||||
return left, nil
|
||||
@ -601,11 +607,13 @@ func (w *Walker) resolveName(name string) (v interface{}, t interface{}, ok bool
|
||||
return nil, nil, false
|
||||
}
|
||||
|
||||
// constDepPrefix is a magic prefix that is used by constValueType
|
||||
// and walkConst to signal that a type isn't known yet. These are
|
||||
// resolved at the end of walking of a package's files.
|
||||
const constDepPrefix = "const-dependency:"
|
||||
|
||||
func (w *Walker) walkConst(vs *ast.ValueSpec) {
|
||||
for _, ident := range vs.Names {
|
||||
if !ast.IsExported(ident.Name) {
|
||||
continue
|
||||
}
|
||||
litType := ""
|
||||
if vs.Type != nil {
|
||||
litType = w.nodeString(vs.Type)
|
||||
@ -627,13 +635,44 @@ func (w *Walker) walkConst(vs *ast.ValueSpec) {
|
||||
}
|
||||
}
|
||||
}
|
||||
if strings.HasPrefix(litType, constDepPrefix) {
|
||||
dep := litType[len(constDepPrefix):]
|
||||
w.constDep[ident.Name] = dep
|
||||
continue
|
||||
}
|
||||
if litType == "" {
|
||||
log.Fatalf("unknown kind in const %q", ident.Name)
|
||||
}
|
||||
w.lastConstType = litType
|
||||
|
||||
w.prevConstType[pkgSymbol{w.curPackageName, ident.Name}] = litType
|
||||
|
||||
if ast.IsExported(ident.Name) {
|
||||
w.emitFeature(fmt.Sprintf("const %s %s", ident, litType))
|
||||
w.prevConstType[ident.Name] = litType
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func (w *Walker) resolveConstantDeps() {
|
||||
var findConstType func(string) string
|
||||
findConstType = func(ident string) string {
|
||||
if dep, ok := w.constDep[ident]; ok {
|
||||
return findConstType(dep)
|
||||
}
|
||||
if t, ok := w.prevConstType[pkgSymbol{w.curPackageName, ident}]; ok {
|
||||
return t
|
||||
}
|
||||
return ""
|
||||
}
|
||||
for ident := range w.constDep {
|
||||
if !ast.IsExported(ident) {
|
||||
continue
|
||||
}
|
||||
t := findConstType(ident)
|
||||
if t == "" {
|
||||
log.Fatalf("failed to resolve constant %q", ident)
|
||||
}
|
||||
w.emitFeature(fmt.Sprintf("const %s %s", ident, t))
|
||||
}
|
||||
}
|
||||
|
||||
|
2
src/cmd/api/testdata/src/pkg/p1/golden.txt
vendored
2
src/cmd/api/testdata/src/pkg/p1/golden.txt
vendored
@ -1,6 +1,8 @@
|
||||
pkg p1, const A ideal-int
|
||||
pkg p1, const A64 int64
|
||||
pkg p1, const AIsLowerA ideal-int
|
||||
pkg p1, const B ideal-int
|
||||
pkg p1, const ConstChase2 ideal-int
|
||||
pkg p1, const ConversionConst MyInt
|
||||
pkg p1, const FloatConst ideal-float
|
||||
pkg p1, const StrConst ideal-string
|
||||
|
5
src/cmd/api/testdata/src/pkg/p1/p1.go
vendored
5
src/cmd/api/testdata/src/pkg/p1/p1.go
vendored
@ -5,9 +5,14 @@ import (
|
||||
)
|
||||
|
||||
const (
|
||||
ConstChase2 = constChase // forward declaration to unexported ident
|
||||
constChase = AIsLowerA // forward declaration to exported ident
|
||||
|
||||
A = 1
|
||||
a = 11
|
||||
A64 int64 = 1
|
||||
|
||||
AIsLowerA = a // previously declared
|
||||
)
|
||||
|
||||
const (
|
||||
|
Loading…
Reference in New Issue
Block a user