mirror of
https://github.com/golang/go.git
synced 2025-05-05 15:43:04 +00:00
The freebsd/riscv64 port was added in go1.20, make cmd/api aware of it and backfill API files. For #58582 Change-Id: I242b161cdb6221c8f60a1868c6435be26fca0078 Reviewed-on: https://go-review.googlesource.com/c/go/+/469135 Run-TryBot: Dmitri Goutnik <dgoutnik@gmail.com> TryBot-Result: Gopher Robot <gobot@golang.org> Auto-Submit: Ian Lance Taylor <iant@google.com> Run-TryBot: Ian Lance Taylor <iant@google.com> Reviewed-by: Ian Lance Taylor <iant@google.com> Reviewed-by: Bryan Mills <bcmills@google.com>
1269 lines
32 KiB
Go
1269 lines
32 KiB
Go
// Copyright 2011 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 api computes the exported API of a set of Go packages.
|
|
// It is only a test, not a command, nor a usefully importable package.
|
|
package api
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"encoding/json"
|
|
"flag"
|
|
"fmt"
|
|
"go/ast"
|
|
"go/build"
|
|
"go/parser"
|
|
"go/token"
|
|
"go/types"
|
|
"internal/testenv"
|
|
"io"
|
|
"log"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"regexp"
|
|
"runtime"
|
|
"sort"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
"testing"
|
|
)
|
|
|
|
const verbose = false
|
|
|
|
func goCmd() string {
|
|
var exeSuffix string
|
|
if runtime.GOOS == "windows" {
|
|
exeSuffix = ".exe"
|
|
}
|
|
path := filepath.Join(testenv.GOROOT(nil), "bin", "go"+exeSuffix)
|
|
if _, err := os.Stat(path); err == nil {
|
|
return path
|
|
}
|
|
return "go"
|
|
}
|
|
|
|
// contexts are the default contexts which are scanned, unless
|
|
// overridden by the -contexts flag.
|
|
var contexts = []*build.Context{
|
|
{GOOS: "linux", GOARCH: "386", CgoEnabled: true},
|
|
{GOOS: "linux", GOARCH: "386"},
|
|
{GOOS: "linux", GOARCH: "amd64", CgoEnabled: true},
|
|
{GOOS: "linux", GOARCH: "amd64"},
|
|
{GOOS: "linux", GOARCH: "arm", CgoEnabled: true},
|
|
{GOOS: "linux", GOARCH: "arm"},
|
|
{GOOS: "darwin", GOARCH: "amd64", CgoEnabled: true},
|
|
{GOOS: "darwin", GOARCH: "amd64"},
|
|
{GOOS: "darwin", GOARCH: "arm64", CgoEnabled: true},
|
|
{GOOS: "darwin", GOARCH: "arm64"},
|
|
{GOOS: "windows", GOARCH: "amd64"},
|
|
{GOOS: "windows", GOARCH: "386"},
|
|
{GOOS: "freebsd", GOARCH: "386", CgoEnabled: true},
|
|
{GOOS: "freebsd", GOARCH: "386"},
|
|
{GOOS: "freebsd", GOARCH: "amd64", CgoEnabled: true},
|
|
{GOOS: "freebsd", GOARCH: "amd64"},
|
|
{GOOS: "freebsd", GOARCH: "arm", CgoEnabled: true},
|
|
{GOOS: "freebsd", GOARCH: "arm"},
|
|
{GOOS: "freebsd", GOARCH: "arm64", CgoEnabled: true},
|
|
{GOOS: "freebsd", GOARCH: "arm64"},
|
|
{GOOS: "freebsd", GOARCH: "riscv64", CgoEnabled: true},
|
|
{GOOS: "freebsd", GOARCH: "riscv64"},
|
|
{GOOS: "netbsd", GOARCH: "386", CgoEnabled: true},
|
|
{GOOS: "netbsd", GOARCH: "386"},
|
|
{GOOS: "netbsd", GOARCH: "amd64", CgoEnabled: true},
|
|
{GOOS: "netbsd", GOARCH: "amd64"},
|
|
{GOOS: "netbsd", GOARCH: "arm", CgoEnabled: true},
|
|
{GOOS: "netbsd", GOARCH: "arm"},
|
|
{GOOS: "netbsd", GOARCH: "arm64", CgoEnabled: true},
|
|
{GOOS: "netbsd", GOARCH: "arm64"},
|
|
{GOOS: "openbsd", GOARCH: "386", CgoEnabled: true},
|
|
{GOOS: "openbsd", GOARCH: "386"},
|
|
{GOOS: "openbsd", GOARCH: "amd64", CgoEnabled: true},
|
|
{GOOS: "openbsd", GOARCH: "amd64"},
|
|
}
|
|
|
|
func contextName(c *build.Context) string {
|
|
s := c.GOOS + "-" + c.GOARCH
|
|
if c.CgoEnabled {
|
|
s += "-cgo"
|
|
}
|
|
if c.Dir != "" {
|
|
s += fmt.Sprintf(" [%s]", c.Dir)
|
|
}
|
|
return s
|
|
}
|
|
|
|
func parseContext(c string) *build.Context {
|
|
parts := strings.Split(c, "-")
|
|
if len(parts) < 2 {
|
|
log.Fatalf("bad context: %q", c)
|
|
}
|
|
bc := &build.Context{
|
|
GOOS: parts[0],
|
|
GOARCH: parts[1],
|
|
}
|
|
if len(parts) == 3 {
|
|
if parts[2] == "cgo" {
|
|
bc.CgoEnabled = true
|
|
} else {
|
|
log.Fatalf("bad context: %q", c)
|
|
}
|
|
}
|
|
return bc
|
|
}
|
|
|
|
var internalPkg = regexp.MustCompile(`(^|/)internal($|/)`)
|
|
|
|
var exitCode = 0
|
|
|
|
func Check(t *testing.T) {
|
|
checkFiles, err := filepath.Glob(filepath.Join(testenv.GOROOT(t), "api/go1*.txt"))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
var nextFiles []string
|
|
if strings.Contains(runtime.Version(), "devel") {
|
|
next, err := filepath.Glob(filepath.Join(testenv.GOROOT(t), "api/next/*.txt"))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
nextFiles = next
|
|
}
|
|
|
|
for _, c := range contexts {
|
|
c.Compiler = build.Default.Compiler
|
|
}
|
|
|
|
walkers := make([]*Walker, len(contexts))
|
|
var wg sync.WaitGroup
|
|
for i, context := range contexts {
|
|
i, context := i, context
|
|
wg.Add(1)
|
|
go func() {
|
|
defer wg.Done()
|
|
walkers[i] = NewWalker(context, filepath.Join(testenv.GOROOT(t), "src"))
|
|
}()
|
|
}
|
|
wg.Wait()
|
|
|
|
var featureCtx = make(map[string]map[string]bool) // feature -> context name -> true
|
|
for _, w := range walkers {
|
|
pkgNames := w.stdPackages
|
|
if flag.NArg() > 0 {
|
|
pkgNames = flag.Args()
|
|
}
|
|
|
|
for _, name := range pkgNames {
|
|
pkg, err := w.import_(name)
|
|
if _, nogo := err.(*build.NoGoError); nogo {
|
|
continue
|
|
}
|
|
if err != nil {
|
|
log.Fatalf("Import(%q): %v", name, err)
|
|
}
|
|
w.export(pkg)
|
|
}
|
|
|
|
ctxName := contextName(w.context)
|
|
for _, f := range w.Features() {
|
|
if featureCtx[f] == nil {
|
|
featureCtx[f] = make(map[string]bool)
|
|
}
|
|
featureCtx[f][ctxName] = true
|
|
}
|
|
}
|
|
|
|
var features []string
|
|
for f, cmap := range featureCtx {
|
|
if len(cmap) == len(contexts) {
|
|
features = append(features, f)
|
|
continue
|
|
}
|
|
comma := strings.Index(f, ",")
|
|
for cname := range cmap {
|
|
f2 := fmt.Sprintf("%s (%s)%s", f[:comma], cname, f[comma:])
|
|
features = append(features, f2)
|
|
}
|
|
}
|
|
|
|
bw := bufio.NewWriter(os.Stdout)
|
|
defer bw.Flush()
|
|
|
|
var required []string
|
|
for _, file := range checkFiles {
|
|
required = append(required, fileFeatures(file, needApproval(file))...)
|
|
}
|
|
var optional []string
|
|
for _, file := range nextFiles {
|
|
optional = append(optional, fileFeatures(file, true)...)
|
|
}
|
|
exception := fileFeatures(filepath.Join(testenv.GOROOT(t), "api/except.txt"), false)
|
|
|
|
if exitCode == 1 {
|
|
t.Errorf("API database problems found")
|
|
}
|
|
if !compareAPI(bw, features, required, optional, exception, false) {
|
|
t.Errorf("API differences found")
|
|
}
|
|
}
|
|
|
|
// export emits the exported package features.
|
|
func (w *Walker) export(pkg *apiPackage) {
|
|
if verbose {
|
|
log.Println(pkg)
|
|
}
|
|
pop := w.pushScope("pkg " + pkg.Path())
|
|
w.current = pkg
|
|
w.collectDeprecated()
|
|
scope := pkg.Scope()
|
|
for _, name := range scope.Names() {
|
|
if token.IsExported(name) {
|
|
w.emitObj(scope.Lookup(name))
|
|
}
|
|
}
|
|
pop()
|
|
}
|
|
|
|
func set(items []string) map[string]bool {
|
|
s := make(map[string]bool)
|
|
for _, v := range items {
|
|
s[v] = true
|
|
}
|
|
return s
|
|
}
|
|
|
|
var spaceParensRx = regexp.MustCompile(` \(\S+?\)`)
|
|
|
|
func featureWithoutContext(f string) string {
|
|
if !strings.Contains(f, "(") {
|
|
return f
|
|
}
|
|
return spaceParensRx.ReplaceAllString(f, "")
|
|
}
|
|
|
|
// portRemoved reports whether the given port-specific API feature is
|
|
// okay to no longer exist because its port was removed.
|
|
func portRemoved(feature string) bool {
|
|
return strings.Contains(feature, "(darwin-386)") ||
|
|
strings.Contains(feature, "(darwin-386-cgo)")
|
|
}
|
|
|
|
func compareAPI(w io.Writer, features, required, optional, exception []string, allowAdd bool) (ok bool) {
|
|
ok = true
|
|
|
|
optionalSet := set(optional)
|
|
exceptionSet := set(exception)
|
|
featureSet := set(features)
|
|
|
|
sort.Strings(features)
|
|
sort.Strings(required)
|
|
|
|
take := func(sl *[]string) string {
|
|
s := (*sl)[0]
|
|
*sl = (*sl)[1:]
|
|
return s
|
|
}
|
|
|
|
for len(required) > 0 || len(features) > 0 {
|
|
switch {
|
|
case len(features) == 0 || (len(required) > 0 && required[0] < features[0]):
|
|
feature := take(&required)
|
|
if exceptionSet[feature] {
|
|
// An "unfortunate" case: the feature was once
|
|
// included in the API (e.g. go1.txt), but was
|
|
// subsequently removed. These are already
|
|
// acknowledged by being in the file
|
|
// "api/except.txt". No need to print them out
|
|
// here.
|
|
} else if portRemoved(feature) {
|
|
// okay.
|
|
} else if featureSet[featureWithoutContext(feature)] {
|
|
// okay.
|
|
} else {
|
|
fmt.Fprintf(w, "-%s\n", feature)
|
|
ok = false // broke compatibility
|
|
}
|
|
case len(required) == 0 || (len(features) > 0 && required[0] > features[0]):
|
|
newFeature := take(&features)
|
|
if optionalSet[newFeature] {
|
|
// Known added feature to the upcoming release.
|
|
// Delete it from the map so we can detect any upcoming features
|
|
// which were never seen. (so we can clean up the nextFile)
|
|
delete(optionalSet, newFeature)
|
|
} else {
|
|
fmt.Fprintf(w, "+%s\n", newFeature)
|
|
if !allowAdd {
|
|
ok = false // we're in lock-down mode for next release
|
|
}
|
|
}
|
|
default:
|
|
take(&required)
|
|
take(&features)
|
|
}
|
|
}
|
|
|
|
// In next file, but not in API.
|
|
var missing []string
|
|
for feature := range optionalSet {
|
|
missing = append(missing, feature)
|
|
}
|
|
sort.Strings(missing)
|
|
for _, feature := range missing {
|
|
fmt.Fprintf(w, "±%s\n", feature)
|
|
}
|
|
return
|
|
}
|
|
|
|
// aliasReplacer applies type aliases to earlier API files,
|
|
// to avoid misleading negative results.
|
|
// This makes all the references to os.FileInfo in go1.txt
|
|
// be read as if they said fs.FileInfo, since os.FileInfo is now an alias.
|
|
// If there are many of these, we could do a more general solution,
|
|
// but for now the replacer is fine.
|
|
var aliasReplacer = strings.NewReplacer(
|
|
"os.FileInfo", "fs.FileInfo",
|
|
"os.FileMode", "fs.FileMode",
|
|
"os.PathError", "fs.PathError",
|
|
)
|
|
|
|
func fileFeatures(filename string, needApproval bool) []string {
|
|
bs, err := os.ReadFile(filename)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
s := string(bs)
|
|
|
|
// Diagnose common mistakes people make,
|
|
// since there is no apifmt to format these files.
|
|
// The missing final newline is important for the
|
|
// final release step of cat next/*.txt >go1.X.txt.
|
|
// If the files don't end in full lines, the concatenation goes awry.
|
|
if strings.Contains(s, "\r") {
|
|
log.Printf("%s: contains CRLFs", filename)
|
|
exitCode = 1
|
|
}
|
|
if s == "" {
|
|
log.Printf("%s: empty file", filename)
|
|
exitCode = 1
|
|
} else if s[len(s)-1] != '\n' {
|
|
log.Printf("%s: missing final newline", filename)
|
|
exitCode = 1
|
|
}
|
|
s = aliasReplacer.Replace(s)
|
|
lines := strings.Split(s, "\n")
|
|
var nonblank []string
|
|
for i, line := range lines {
|
|
line = strings.TrimSpace(line)
|
|
if line == "" || strings.HasPrefix(line, "#") {
|
|
continue
|
|
}
|
|
if needApproval {
|
|
feature, approval, ok := strings.Cut(line, "#")
|
|
if !ok {
|
|
log.Printf("%s:%d: missing proposal approval\n", filename, i+1)
|
|
exitCode = 1
|
|
} else {
|
|
_, err := strconv.Atoi(approval)
|
|
if err != nil {
|
|
log.Printf("%s:%d: malformed proposal approval #%s\n", filename, i+1, approval)
|
|
exitCode = 1
|
|
}
|
|
}
|
|
line = strings.TrimSpace(feature)
|
|
} else {
|
|
if strings.Contains(line, " #") {
|
|
log.Printf("%s:%d: unexpected approval\n", filename, i+1)
|
|
exitCode = 1
|
|
}
|
|
}
|
|
nonblank = append(nonblank, line)
|
|
}
|
|
return nonblank
|
|
}
|
|
|
|
var fset = token.NewFileSet()
|
|
|
|
type Walker struct {
|
|
context *build.Context
|
|
root string
|
|
scope []string
|
|
current *apiPackage
|
|
deprecated map[token.Pos]bool
|
|
features map[string]bool // set
|
|
imported map[string]*apiPackage // packages already imported
|
|
stdPackages []string // names, omitting "unsafe", internal, and vendored packages
|
|
importMap map[string]map[string]string // importer dir -> import path -> canonical path
|
|
importDir map[string]string // canonical import path -> dir
|
|
|
|
}
|
|
|
|
func NewWalker(context *build.Context, root string) *Walker {
|
|
w := &Walker{
|
|
context: context,
|
|
root: root,
|
|
features: map[string]bool{},
|
|
imported: map[string]*apiPackage{"unsafe": &apiPackage{Package: types.Unsafe}},
|
|
}
|
|
w.loadImports()
|
|
return w
|
|
}
|
|
|
|
func (w *Walker) Features() (fs []string) {
|
|
for f := range w.features {
|
|
fs = append(fs, f)
|
|
}
|
|
sort.Strings(fs)
|
|
return
|
|
}
|
|
|
|
var parsedFileCache = make(map[string]*ast.File)
|
|
|
|
func (w *Walker) parseFile(dir, file string) (*ast.File, error) {
|
|
filename := filepath.Join(dir, file)
|
|
if f := parsedFileCache[filename]; f != nil {
|
|
return f, nil
|
|
}
|
|
|
|
f, err := parser.ParseFile(fset, filename, nil, parser.ParseComments)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
parsedFileCache[filename] = f
|
|
|
|
return f, nil
|
|
}
|
|
|
|
// Disable before debugging non-obvious errors from the type-checker.
|
|
const usePkgCache = true
|
|
|
|
var (
|
|
pkgCache = map[string]*apiPackage{} // map tagKey to package
|
|
pkgTags = map[string][]string{} // map import dir to list of relevant tags
|
|
)
|
|
|
|
// tagKey returns the tag-based key to use in the pkgCache.
|
|
// It is a comma-separated string; the first part is dir, the rest tags.
|
|
// The satisfied tags are derived from context but only those that
|
|
// matter (the ones listed in the tags argument plus GOOS and GOARCH) are used.
|
|
// The tags list, which came from go/build's Package.AllTags,
|
|
// is known to be sorted.
|
|
func tagKey(dir string, context *build.Context, tags []string) string {
|
|
ctags := map[string]bool{
|
|
context.GOOS: true,
|
|
context.GOARCH: true,
|
|
}
|
|
if context.CgoEnabled {
|
|
ctags["cgo"] = true
|
|
}
|
|
for _, tag := range context.BuildTags {
|
|
ctags[tag] = true
|
|
}
|
|
// TODO: ReleaseTags (need to load default)
|
|
key := dir
|
|
|
|
// explicit on GOOS and GOARCH as global cache will use "all" cached packages for
|
|
// an indirect imported package. See https://github.com/golang/go/issues/21181
|
|
// for more detail.
|
|
tags = append(tags, context.GOOS, context.GOARCH)
|
|
sort.Strings(tags)
|
|
|
|
for _, tag := range tags {
|
|
if ctags[tag] {
|
|
key += "," + tag
|
|
ctags[tag] = false
|
|
}
|
|
}
|
|
return key
|
|
}
|
|
|
|
type listImports struct {
|
|
stdPackages []string // names, omitting "unsafe", internal, and vendored packages
|
|
importDir map[string]string // canonical import path → directory
|
|
importMap map[string]map[string]string // import path → canonical import path
|
|
}
|
|
|
|
var listCache sync.Map // map[string]listImports, keyed by contextName
|
|
|
|
// listSem is a semaphore restricting concurrent invocations of 'go list'. 'go
|
|
// list' has its own internal concurrency, so we use a hard-coded constant (to
|
|
// allow the I/O-intensive phases of 'go list' to overlap) instead of scaling
|
|
// all the way up to GOMAXPROCS.
|
|
var listSem = make(chan semToken, 2)
|
|
|
|
type semToken struct{}
|
|
|
|
// loadImports populates w with information about the packages in the standard
|
|
// library and the packages they themselves import in w's build context.
|
|
//
|
|
// The source import path and expanded import path are identical except for vendored packages.
|
|
// For example, on return:
|
|
//
|
|
// w.importMap["math"] = "math"
|
|
// w.importDir["math"] = "<goroot>/src/math"
|
|
//
|
|
// w.importMap["golang.org/x/net/route"] = "vendor/golang.org/x/net/route"
|
|
// w.importDir["vendor/golang.org/x/net/route"] = "<goroot>/src/vendor/golang.org/x/net/route"
|
|
//
|
|
// Since the set of packages that exist depends on context, the result of
|
|
// loadImports also depends on context. However, to improve test running time
|
|
// the configuration for each environment is cached across runs.
|
|
func (w *Walker) loadImports() {
|
|
if w.context == nil {
|
|
return // test-only Walker; does not use the import map
|
|
}
|
|
|
|
name := contextName(w.context)
|
|
|
|
imports, ok := listCache.Load(name)
|
|
if !ok {
|
|
listSem <- semToken{}
|
|
defer func() { <-listSem }()
|
|
|
|
cmd := exec.Command(goCmd(), "list", "-e", "-deps", "-json", "std")
|
|
cmd.Env = listEnv(w.context)
|
|
if w.context.Dir != "" {
|
|
cmd.Dir = w.context.Dir
|
|
}
|
|
out, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
log.Fatalf("loading imports: %v\n%s", err, out)
|
|
}
|
|
|
|
var stdPackages []string
|
|
importMap := make(map[string]map[string]string)
|
|
importDir := make(map[string]string)
|
|
dec := json.NewDecoder(bytes.NewReader(out))
|
|
for {
|
|
var pkg struct {
|
|
ImportPath, Dir string
|
|
ImportMap map[string]string
|
|
Standard bool
|
|
}
|
|
err := dec.Decode(&pkg)
|
|
if err == io.EOF {
|
|
break
|
|
}
|
|
if err != nil {
|
|
log.Fatalf("go list: invalid output: %v", err)
|
|
}
|
|
|
|
// - Package "unsafe" contains special signatures requiring
|
|
// extra care when printing them - ignore since it is not
|
|
// going to change w/o a language change.
|
|
// - Internal and vendored packages do not contribute to our
|
|
// API surface. (If we are running within the "std" module,
|
|
// vendored dependencies appear as themselves instead of
|
|
// their "vendor/" standard-library copies.)
|
|
// - 'go list std' does not include commands, which cannot be
|
|
// imported anyway.
|
|
if ip := pkg.ImportPath; pkg.Standard && ip != "unsafe" && !strings.HasPrefix(ip, "vendor/") && !internalPkg.MatchString(ip) {
|
|
stdPackages = append(stdPackages, ip)
|
|
}
|
|
importDir[pkg.ImportPath] = pkg.Dir
|
|
if len(pkg.ImportMap) > 0 {
|
|
importMap[pkg.Dir] = make(map[string]string, len(pkg.ImportMap))
|
|
}
|
|
for k, v := range pkg.ImportMap {
|
|
importMap[pkg.Dir][k] = v
|
|
}
|
|
}
|
|
|
|
sort.Strings(stdPackages)
|
|
imports = listImports{
|
|
stdPackages: stdPackages,
|
|
importMap: importMap,
|
|
importDir: importDir,
|
|
}
|
|
imports, _ = listCache.LoadOrStore(name, imports)
|
|
}
|
|
|
|
li := imports.(listImports)
|
|
w.stdPackages = li.stdPackages
|
|
w.importDir = li.importDir
|
|
w.importMap = li.importMap
|
|
}
|
|
|
|
// listEnv returns the process environment to use when invoking 'go list' for
|
|
// the given context.
|
|
func listEnv(c *build.Context) []string {
|
|
if c == nil {
|
|
return os.Environ()
|
|
}
|
|
|
|
environ := append(os.Environ(),
|
|
"GOOS="+c.GOOS,
|
|
"GOARCH="+c.GOARCH)
|
|
if c.CgoEnabled {
|
|
environ = append(environ, "CGO_ENABLED=1")
|
|
} else {
|
|
environ = append(environ, "CGO_ENABLED=0")
|
|
}
|
|
return environ
|
|
}
|
|
|
|
type apiPackage struct {
|
|
*types.Package
|
|
Files []*ast.File
|
|
}
|
|
|
|
// Importing is a sentinel taking the place in Walker.imported
|
|
// for a package that is in the process of being imported.
|
|
var importing apiPackage
|
|
|
|
// Import implements types.Importer.
|
|
func (w *Walker) Import(name string) (*types.Package, error) {
|
|
return w.ImportFrom(name, "", 0)
|
|
}
|
|
|
|
// ImportFrom implements types.ImporterFrom.
|
|
func (w *Walker) ImportFrom(fromPath, fromDir string, mode types.ImportMode) (*types.Package, error) {
|
|
pkg, err := w.importFrom(fromPath, fromDir, mode)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return pkg.Package, nil
|
|
}
|
|
|
|
func (w *Walker) import_(name string) (*apiPackage, error) {
|
|
return w.importFrom(name, "", 0)
|
|
}
|
|
|
|
func (w *Walker) importFrom(fromPath, fromDir string, mode types.ImportMode) (*apiPackage, error) {
|
|
name := fromPath
|
|
if canonical, ok := w.importMap[fromDir][fromPath]; ok {
|
|
name = canonical
|
|
}
|
|
|
|
pkg := w.imported[name]
|
|
if pkg != nil {
|
|
if pkg == &importing {
|
|
log.Fatalf("cycle importing package %q", name)
|
|
}
|
|
return pkg, nil
|
|
}
|
|
w.imported[name] = &importing
|
|
|
|
// Determine package files.
|
|
dir := w.importDir[name]
|
|
if dir == "" {
|
|
dir = filepath.Join(w.root, filepath.FromSlash(name))
|
|
}
|
|
if fi, err := os.Stat(dir); err != nil || !fi.IsDir() {
|
|
log.Panicf("no source in tree for import %q (from import %s in %s): %v", name, fromPath, fromDir, err)
|
|
}
|
|
|
|
context := w.context
|
|
if context == nil {
|
|
context = &build.Default
|
|
}
|
|
|
|
// Look in cache.
|
|
// If we've already done an import with the same set
|
|
// of relevant tags, reuse the result.
|
|
var key string
|
|
if usePkgCache {
|
|
if tags, ok := pkgTags[dir]; ok {
|
|
key = tagKey(dir, context, tags)
|
|
if pkg := pkgCache[key]; pkg != nil {
|
|
w.imported[name] = pkg
|
|
return pkg, nil
|
|
}
|
|
}
|
|
}
|
|
|
|
info, err := context.ImportDir(dir, 0)
|
|
if err != nil {
|
|
if _, nogo := err.(*build.NoGoError); nogo {
|
|
return nil, err
|
|
}
|
|
log.Fatalf("pkg %q, dir %q: ScanDir: %v", name, dir, err)
|
|
}
|
|
|
|
// Save tags list first time we see a directory.
|
|
if usePkgCache {
|
|
if _, ok := pkgTags[dir]; !ok {
|
|
pkgTags[dir] = info.AllTags
|
|
key = tagKey(dir, context, info.AllTags)
|
|
}
|
|
}
|
|
|
|
filenames := append(append([]string{}, info.GoFiles...), info.CgoFiles...)
|
|
|
|
// Parse package files.
|
|
var files []*ast.File
|
|
for _, file := range filenames {
|
|
f, err := w.parseFile(dir, file)
|
|
if err != nil {
|
|
log.Fatalf("error parsing package %s: %s", name, err)
|
|
}
|
|
files = append(files, f)
|
|
}
|
|
|
|
// Type-check package files.
|
|
var sizes types.Sizes
|
|
if w.context != nil {
|
|
sizes = types.SizesFor(w.context.Compiler, w.context.GOARCH)
|
|
}
|
|
conf := types.Config{
|
|
IgnoreFuncBodies: true,
|
|
FakeImportC: true,
|
|
Importer: w,
|
|
Sizes: sizes,
|
|
}
|
|
tpkg, err := conf.Check(name, fset, files, nil)
|
|
if err != nil {
|
|
ctxt := "<no context>"
|
|
if w.context != nil {
|
|
ctxt = fmt.Sprintf("%s-%s", w.context.GOOS, w.context.GOARCH)
|
|
}
|
|
log.Fatalf("error typechecking package %s: %s (%s)", name, err, ctxt)
|
|
}
|
|
pkg = &apiPackage{tpkg, files}
|
|
|
|
if usePkgCache {
|
|
pkgCache[key] = pkg
|
|
}
|
|
|
|
w.imported[name] = pkg
|
|
return pkg, nil
|
|
}
|
|
|
|
// pushScope enters a new scope (walking a package, type, node, etc)
|
|
// and returns a function that will leave the scope (with sanity checking
|
|
// for mismatched pushes & pops)
|
|
func (w *Walker) pushScope(name string) (popFunc func()) {
|
|
w.scope = append(w.scope, name)
|
|
return func() {
|
|
if len(w.scope) == 0 {
|
|
log.Fatalf("attempt to leave scope %q with empty scope list", name)
|
|
}
|
|
if w.scope[len(w.scope)-1] != name {
|
|
log.Fatalf("attempt to leave scope %q, but scope is currently %#v", name, w.scope)
|
|
}
|
|
w.scope = w.scope[:len(w.scope)-1]
|
|
}
|
|
}
|
|
|
|
func sortedMethodNames(typ *types.Interface) []string {
|
|
n := typ.NumMethods()
|
|
list := make([]string, n)
|
|
for i := range list {
|
|
list[i] = typ.Method(i).Name()
|
|
}
|
|
sort.Strings(list)
|
|
return list
|
|
}
|
|
|
|
// sortedEmbeddeds returns constraint types embedded in an
|
|
// interface. It does not include embedded interface types or methods.
|
|
func (w *Walker) sortedEmbeddeds(typ *types.Interface) []string {
|
|
n := typ.NumEmbeddeds()
|
|
list := make([]string, 0, n)
|
|
for i := 0; i < n; i++ {
|
|
emb := typ.EmbeddedType(i)
|
|
switch emb := emb.(type) {
|
|
case *types.Interface:
|
|
list = append(list, w.sortedEmbeddeds(emb)...)
|
|
case *types.Union:
|
|
var buf bytes.Buffer
|
|
nu := emb.Len()
|
|
for i := 0; i < nu; i++ {
|
|
if i > 0 {
|
|
buf.WriteString(" | ")
|
|
}
|
|
term := emb.Term(i)
|
|
if term.Tilde() {
|
|
buf.WriteByte('~')
|
|
}
|
|
w.writeType(&buf, term.Type())
|
|
}
|
|
list = append(list, buf.String())
|
|
}
|
|
}
|
|
sort.Strings(list)
|
|
return list
|
|
}
|
|
|
|
func (w *Walker) writeType(buf *bytes.Buffer, typ types.Type) {
|
|
switch typ := typ.(type) {
|
|
case *types.Basic:
|
|
s := typ.Name()
|
|
switch typ.Kind() {
|
|
case types.UnsafePointer:
|
|
s = "unsafe.Pointer"
|
|
case types.UntypedBool:
|
|
s = "ideal-bool"
|
|
case types.UntypedInt:
|
|
s = "ideal-int"
|
|
case types.UntypedRune:
|
|
// "ideal-char" for compatibility with old tool
|
|
// TODO(gri) change to "ideal-rune"
|
|
s = "ideal-char"
|
|
case types.UntypedFloat:
|
|
s = "ideal-float"
|
|
case types.UntypedComplex:
|
|
s = "ideal-complex"
|
|
case types.UntypedString:
|
|
s = "ideal-string"
|
|
case types.UntypedNil:
|
|
panic("should never see untyped nil type")
|
|
default:
|
|
switch s {
|
|
case "byte":
|
|
s = "uint8"
|
|
case "rune":
|
|
s = "int32"
|
|
}
|
|
}
|
|
buf.WriteString(s)
|
|
|
|
case *types.Array:
|
|
fmt.Fprintf(buf, "[%d]", typ.Len())
|
|
w.writeType(buf, typ.Elem())
|
|
|
|
case *types.Slice:
|
|
buf.WriteString("[]")
|
|
w.writeType(buf, typ.Elem())
|
|
|
|
case *types.Struct:
|
|
buf.WriteString("struct")
|
|
|
|
case *types.Pointer:
|
|
buf.WriteByte('*')
|
|
w.writeType(buf, typ.Elem())
|
|
|
|
case *types.Tuple:
|
|
panic("should never see a tuple type")
|
|
|
|
case *types.Signature:
|
|
buf.WriteString("func")
|
|
w.writeSignature(buf, typ)
|
|
|
|
case *types.Interface:
|
|
buf.WriteString("interface{")
|
|
if typ.NumMethods() > 0 || typ.NumEmbeddeds() > 0 {
|
|
buf.WriteByte(' ')
|
|
}
|
|
if typ.NumMethods() > 0 {
|
|
buf.WriteString(strings.Join(sortedMethodNames(typ), ", "))
|
|
}
|
|
if typ.NumEmbeddeds() > 0 {
|
|
buf.WriteString(strings.Join(w.sortedEmbeddeds(typ), ", "))
|
|
}
|
|
if typ.NumMethods() > 0 || typ.NumEmbeddeds() > 0 {
|
|
buf.WriteByte(' ')
|
|
}
|
|
buf.WriteString("}")
|
|
|
|
case *types.Map:
|
|
buf.WriteString("map[")
|
|
w.writeType(buf, typ.Key())
|
|
buf.WriteByte(']')
|
|
w.writeType(buf, typ.Elem())
|
|
|
|
case *types.Chan:
|
|
var s string
|
|
switch typ.Dir() {
|
|
case types.SendOnly:
|
|
s = "chan<- "
|
|
case types.RecvOnly:
|
|
s = "<-chan "
|
|
case types.SendRecv:
|
|
s = "chan "
|
|
default:
|
|
panic("unreachable")
|
|
}
|
|
buf.WriteString(s)
|
|
w.writeType(buf, typ.Elem())
|
|
|
|
case *types.Named:
|
|
obj := typ.Obj()
|
|
pkg := obj.Pkg()
|
|
if pkg != nil && pkg != w.current.Package {
|
|
buf.WriteString(pkg.Name())
|
|
buf.WriteByte('.')
|
|
}
|
|
buf.WriteString(typ.Obj().Name())
|
|
|
|
case *types.TypeParam:
|
|
// Type parameter names may change, so use a placeholder instead.
|
|
fmt.Fprintf(buf, "$%d", typ.Index())
|
|
|
|
default:
|
|
panic(fmt.Sprintf("unknown type %T", typ))
|
|
}
|
|
}
|
|
|
|
func (w *Walker) writeSignature(buf *bytes.Buffer, sig *types.Signature) {
|
|
if tparams := sig.TypeParams(); tparams != nil {
|
|
w.writeTypeParams(buf, tparams, true)
|
|
}
|
|
w.writeParams(buf, sig.Params(), sig.Variadic())
|
|
switch res := sig.Results(); res.Len() {
|
|
case 0:
|
|
// nothing to do
|
|
case 1:
|
|
buf.WriteByte(' ')
|
|
w.writeType(buf, res.At(0).Type())
|
|
default:
|
|
buf.WriteByte(' ')
|
|
w.writeParams(buf, res, false)
|
|
}
|
|
}
|
|
|
|
func (w *Walker) writeTypeParams(buf *bytes.Buffer, tparams *types.TypeParamList, withConstraints bool) {
|
|
buf.WriteByte('[')
|
|
c := tparams.Len()
|
|
for i := 0; i < c; i++ {
|
|
if i > 0 {
|
|
buf.WriteString(", ")
|
|
}
|
|
tp := tparams.At(i)
|
|
w.writeType(buf, tp)
|
|
if withConstraints {
|
|
buf.WriteByte(' ')
|
|
w.writeType(buf, tp.Constraint())
|
|
}
|
|
}
|
|
buf.WriteByte(']')
|
|
}
|
|
|
|
func (w *Walker) writeParams(buf *bytes.Buffer, t *types.Tuple, variadic bool) {
|
|
buf.WriteByte('(')
|
|
for i, n := 0, t.Len(); i < n; i++ {
|
|
if i > 0 {
|
|
buf.WriteString(", ")
|
|
}
|
|
typ := t.At(i).Type()
|
|
if variadic && i+1 == n {
|
|
buf.WriteString("...")
|
|
typ = typ.(*types.Slice).Elem()
|
|
}
|
|
w.writeType(buf, typ)
|
|
}
|
|
buf.WriteByte(')')
|
|
}
|
|
|
|
func (w *Walker) typeString(typ types.Type) string {
|
|
var buf bytes.Buffer
|
|
w.writeType(&buf, typ)
|
|
return buf.String()
|
|
}
|
|
|
|
func (w *Walker) signatureString(sig *types.Signature) string {
|
|
var buf bytes.Buffer
|
|
w.writeSignature(&buf, sig)
|
|
return buf.String()
|
|
}
|
|
|
|
func (w *Walker) emitObj(obj types.Object) {
|
|
switch obj := obj.(type) {
|
|
case *types.Const:
|
|
if w.isDeprecated(obj) {
|
|
w.emitf("const %s //deprecated", obj.Name())
|
|
}
|
|
w.emitf("const %s %s", obj.Name(), w.typeString(obj.Type()))
|
|
x := obj.Val()
|
|
short := x.String()
|
|
exact := x.ExactString()
|
|
if short == exact {
|
|
w.emitf("const %s = %s", obj.Name(), short)
|
|
} else {
|
|
w.emitf("const %s = %s // %s", obj.Name(), short, exact)
|
|
}
|
|
case *types.Var:
|
|
if w.isDeprecated(obj) {
|
|
w.emitf("var %s //deprecated", obj.Name())
|
|
}
|
|
w.emitf("var %s %s", obj.Name(), w.typeString(obj.Type()))
|
|
case *types.TypeName:
|
|
w.emitType(obj)
|
|
case *types.Func:
|
|
w.emitFunc(obj)
|
|
default:
|
|
panic("unknown object: " + obj.String())
|
|
}
|
|
}
|
|
|
|
func (w *Walker) emitType(obj *types.TypeName) {
|
|
name := obj.Name()
|
|
if w.isDeprecated(obj) {
|
|
w.emitf("type %s //deprecated", name)
|
|
}
|
|
if tparams := obj.Type().(*types.Named).TypeParams(); tparams != nil {
|
|
var buf bytes.Buffer
|
|
buf.WriteString(name)
|
|
w.writeTypeParams(&buf, tparams, true)
|
|
name = buf.String()
|
|
}
|
|
typ := obj.Type()
|
|
if obj.IsAlias() {
|
|
w.emitf("type %s = %s", name, w.typeString(typ))
|
|
return
|
|
}
|
|
switch typ := typ.Underlying().(type) {
|
|
case *types.Struct:
|
|
w.emitStructType(name, typ)
|
|
case *types.Interface:
|
|
w.emitIfaceType(name, typ)
|
|
return // methods are handled by emitIfaceType
|
|
default:
|
|
w.emitf("type %s %s", name, w.typeString(typ.Underlying()))
|
|
}
|
|
|
|
// emit methods with value receiver
|
|
var methodNames map[string]bool
|
|
vset := types.NewMethodSet(typ)
|
|
for i, n := 0, vset.Len(); i < n; i++ {
|
|
m := vset.At(i)
|
|
if m.Obj().Exported() {
|
|
w.emitMethod(m)
|
|
if methodNames == nil {
|
|
methodNames = make(map[string]bool)
|
|
}
|
|
methodNames[m.Obj().Name()] = true
|
|
}
|
|
}
|
|
|
|
// emit methods with pointer receiver; exclude
|
|
// methods that we have emitted already
|
|
// (the method set of *T includes the methods of T)
|
|
pset := types.NewMethodSet(types.NewPointer(typ))
|
|
for i, n := 0, pset.Len(); i < n; i++ {
|
|
m := pset.At(i)
|
|
if m.Obj().Exported() && !methodNames[m.Obj().Name()] {
|
|
w.emitMethod(m)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (w *Walker) emitStructType(name string, typ *types.Struct) {
|
|
typeStruct := fmt.Sprintf("type %s struct", name)
|
|
w.emitf(typeStruct)
|
|
defer w.pushScope(typeStruct)()
|
|
|
|
for i := 0; i < typ.NumFields(); i++ {
|
|
f := typ.Field(i)
|
|
if !f.Exported() {
|
|
continue
|
|
}
|
|
typ := f.Type()
|
|
if f.Anonymous() {
|
|
if w.isDeprecated(f) {
|
|
w.emitf("embedded %s //deprecated", w.typeString(typ))
|
|
}
|
|
w.emitf("embedded %s", w.typeString(typ))
|
|
continue
|
|
}
|
|
if w.isDeprecated(f) {
|
|
w.emitf("%s //deprecated", f.Name())
|
|
}
|
|
w.emitf("%s %s", f.Name(), w.typeString(typ))
|
|
}
|
|
}
|
|
|
|
func (w *Walker) emitIfaceType(name string, typ *types.Interface) {
|
|
pop := w.pushScope("type " + name + " interface")
|
|
|
|
var methodNames []string
|
|
complete := true
|
|
mset := types.NewMethodSet(typ)
|
|
for i, n := 0, mset.Len(); i < n; i++ {
|
|
m := mset.At(i).Obj().(*types.Func)
|
|
if !m.Exported() {
|
|
complete = false
|
|
continue
|
|
}
|
|
methodNames = append(methodNames, m.Name())
|
|
if w.isDeprecated(m) {
|
|
w.emitf("%s //deprecated", m.Name())
|
|
}
|
|
w.emitf("%s%s", m.Name(), w.signatureString(m.Type().(*types.Signature)))
|
|
}
|
|
|
|
if !complete {
|
|
// The method set has unexported methods, so all the
|
|
// implementations are provided by the same package,
|
|
// so the method set can be extended. Instead of recording
|
|
// the full set of names (below), record only that there were
|
|
// unexported methods. (If the interface shrinks, we will notice
|
|
// because a method signature emitted during the last loop
|
|
// will disappear.)
|
|
w.emitf("unexported methods")
|
|
}
|
|
|
|
pop()
|
|
|
|
if !complete {
|
|
return
|
|
}
|
|
|
|
if len(methodNames) == 0 {
|
|
w.emitf("type %s interface {}", name)
|
|
return
|
|
}
|
|
|
|
sort.Strings(methodNames)
|
|
w.emitf("type %s interface { %s }", name, strings.Join(methodNames, ", "))
|
|
}
|
|
|
|
func (w *Walker) emitFunc(f *types.Func) {
|
|
sig := f.Type().(*types.Signature)
|
|
if sig.Recv() != nil {
|
|
panic("method considered a regular function: " + f.String())
|
|
}
|
|
if w.isDeprecated(f) {
|
|
w.emitf("func %s //deprecated", f.Name())
|
|
}
|
|
w.emitf("func %s%s", f.Name(), w.signatureString(sig))
|
|
}
|
|
|
|
func (w *Walker) emitMethod(m *types.Selection) {
|
|
sig := m.Type().(*types.Signature)
|
|
recv := sig.Recv().Type()
|
|
// report exported methods with unexported receiver base type
|
|
if true {
|
|
base := recv
|
|
if p, _ := recv.(*types.Pointer); p != nil {
|
|
base = p.Elem()
|
|
}
|
|
if obj := base.(*types.Named).Obj(); !obj.Exported() {
|
|
log.Fatalf("exported method with unexported receiver base type: %s", m)
|
|
}
|
|
}
|
|
tps := ""
|
|
if rtp := sig.RecvTypeParams(); rtp != nil {
|
|
var buf bytes.Buffer
|
|
w.writeTypeParams(&buf, rtp, false)
|
|
tps = buf.String()
|
|
}
|
|
if w.isDeprecated(m.Obj()) {
|
|
w.emitf("method (%s%s) %s //deprecated", w.typeString(recv), tps, m.Obj().Name())
|
|
}
|
|
w.emitf("method (%s%s) %s%s", w.typeString(recv), tps, m.Obj().Name(), w.signatureString(sig))
|
|
}
|
|
|
|
func (w *Walker) emitf(format string, args ...any) {
|
|
f := strings.Join(w.scope, ", ") + ", " + fmt.Sprintf(format, args...)
|
|
if strings.Contains(f, "\n") {
|
|
panic("feature contains newlines: " + f)
|
|
}
|
|
|
|
if _, dup := w.features[f]; dup {
|
|
panic("duplicate feature inserted: " + f)
|
|
}
|
|
w.features[f] = true
|
|
|
|
if verbose {
|
|
log.Printf("feature: %s", f)
|
|
}
|
|
}
|
|
|
|
func needApproval(filename string) bool {
|
|
name := filepath.Base(filename)
|
|
if name == "go1.txt" {
|
|
return false
|
|
}
|
|
minor := strings.TrimSuffix(strings.TrimPrefix(name, "go1."), ".txt")
|
|
n, err := strconv.Atoi(minor)
|
|
if err != nil {
|
|
log.Fatalf("unexpected api file: %v", name)
|
|
}
|
|
return n >= 19 // started tracking approvals in Go 1.19
|
|
}
|
|
|
|
func (w *Walker) collectDeprecated() {
|
|
isDeprecated := func(doc *ast.CommentGroup) bool {
|
|
if doc != nil {
|
|
for _, c := range doc.List {
|
|
if strings.HasPrefix(c.Text, "// Deprecated:") {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
w.deprecated = make(map[token.Pos]bool)
|
|
mark := func(id *ast.Ident) {
|
|
if id != nil {
|
|
w.deprecated[id.Pos()] = true
|
|
}
|
|
}
|
|
for _, file := range w.current.Files {
|
|
ast.Inspect(file, func(n ast.Node) bool {
|
|
switch n := n.(type) {
|
|
case *ast.File:
|
|
if isDeprecated(n.Doc) {
|
|
mark(n.Name)
|
|
}
|
|
return true
|
|
case *ast.GenDecl:
|
|
if isDeprecated(n.Doc) {
|
|
for _, spec := range n.Specs {
|
|
switch spec := spec.(type) {
|
|
case *ast.ValueSpec:
|
|
for _, id := range spec.Names {
|
|
mark(id)
|
|
}
|
|
case *ast.TypeSpec:
|
|
mark(spec.Name)
|
|
}
|
|
}
|
|
}
|
|
return true // look at specs
|
|
case *ast.FuncDecl:
|
|
if isDeprecated(n.Doc) {
|
|
mark(n.Name)
|
|
}
|
|
return false
|
|
case *ast.TypeSpec:
|
|
if isDeprecated(n.Doc) {
|
|
mark(n.Name)
|
|
}
|
|
return true // recurse into struct or interface type
|
|
case *ast.StructType:
|
|
return true // recurse into fields
|
|
case *ast.InterfaceType:
|
|
return true // recurse into methods
|
|
case *ast.FieldList:
|
|
return true // recurse into fields
|
|
case *ast.ValueSpec:
|
|
if isDeprecated(n.Doc) {
|
|
for _, id := range n.Names {
|
|
mark(id)
|
|
}
|
|
}
|
|
return false
|
|
case *ast.Field:
|
|
if isDeprecated(n.Doc) {
|
|
for _, id := range n.Names {
|
|
mark(id)
|
|
}
|
|
if len(n.Names) == 0 {
|
|
// embedded field T or *T?
|
|
typ := n.Type
|
|
if ptr, ok := typ.(*ast.StarExpr); ok {
|
|
typ = ptr.X
|
|
}
|
|
if id, ok := typ.(*ast.Ident); ok {
|
|
mark(id)
|
|
}
|
|
}
|
|
}
|
|
return false
|
|
default:
|
|
return false
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func (w *Walker) isDeprecated(obj types.Object) bool {
|
|
return w.deprecated[obj.Pos()]
|
|
}
|