Commit a17c4616 authored by Alan Donovan's avatar Alan Donovan

go/types: include package import path in NamedType.String().

This avoids ambiguity and makes the diagnostics closer to
those issued by gc, but it is more verbose since it qualifies
intra-package references.

Without extra context---e.g. a 'from *Package' parameter to
Type.String()---we are forced to err on one side or the other.

Also, cosmetic changes to exp/ssa:
- Remove package-qualification workaround in Function.FullName.
- Always set go/types.Package.Path field to the import path,
  since we know the correct path at this point.
- In Function.DumpTo, show variadic '...' and result type info,
  and delete now-redundant "# Type: " line.

R=gri
CC=golang-dev
https://golang.org/cl/7325051
parent 5c3fb96b
...@@ -2570,11 +2570,13 @@ func (b *Builder) CreatePackage(importPath string, files []*ast.File) (*Package, ...@@ -2570,11 +2570,13 @@ func (b *Builder) CreatePackage(importPath string, files []*ast.File) (*Package,
// from the gc compiler's object files; no code will be available. // from the gc compiler's object files; no code will be available.
// //
func (b *Builder) createPackageImpl(typkg *types.Package, importPath string, files []*ast.File) *Package { func (b *Builder) createPackageImpl(typkg *types.Package, importPath string, files []*ast.File) *Package {
// TODO(gri): make this an invariant and eliminate importPath // The typechecker sets types.Package.Path only for GcImported
// param and Package field. // packages, since it doesn't know import path until after typechecking is done.
// if importPath != p.Types.Path { // Here we ensure it is always set, since we know the correct path.
// panic(importPath + " != " + p.Types.Path) // TODO(adonovan): eliminate redundant ssa.Package.ImportPath field.
// } if typkg.Path == "" {
typkg.Path = importPath
}
p := &Package{ p := &Package{
Prog: b.Prog, Prog: b.Prog,
......
...@@ -255,6 +255,7 @@ func (f *Function) finish() { ...@@ -255,6 +255,7 @@ func (f *Function) finish() {
} }
} }
} }
optimizeBlocks(f) optimizeBlocks(f)
// Build immediate-use (referrers) graph. // Build immediate-use (referrers) graph.
...@@ -351,8 +352,11 @@ func (f *Function) emit(instr Instruction) Value { ...@@ -351,8 +352,11 @@ func (f *Function) emit(instr Instruction) Value {
// FullName returns the full name of this function, qualified by // FullName returns the full name of this function, qualified by
// package name, receiver type, etc. // package name, receiver type, etc.
// //
// The specific formatting rules are not guaranteed and may change.
//
// Examples: // Examples:
// "math.IsNaN" // a package-level function // "math.IsNaN" // a package-level function
// "IsNaN" // intra-package reference to same
// "(*sync.WaitGroup).Add" // a declared method // "(*sync.WaitGroup).Add" // a declared method
// "(*exp/ssa.Ret).Block" // a bridge method // "(*exp/ssa.Ret).Block" // a bridge method
// "(ssa.Instruction).Block" // an interface method thunk // "(ssa.Instruction).Block" // an interface method thunk
...@@ -379,27 +383,20 @@ func (f *Function) fullName(from *Package) string { ...@@ -379,27 +383,20 @@ func (f *Function) fullName(from *Package) string {
} else { } else {
recvType = f.Params[0].Type() // interface method thunk recvType = f.Params[0].Type() // interface method thunk
} }
// TODO(adonovan): print type package-qualified, if NamedType.
return fmt.Sprintf("(%s).%s", recvType, f.Name_) return fmt.Sprintf("(%s).%s", recvType, f.Name_)
} }
// "pkg." prefix for cross-package references only.
var pkgQual string
if from != f.Pkg {
pkgQual = f.Pkg.ImportPath + "."
}
// Declared method? // Declared method?
if recv != nil { if recv != nil {
star := "" return fmt.Sprintf("(%s).%s", recv.Type, f.Name_)
if isPointer(recv.Type) {
star = "*"
}
return fmt.Sprintf("(%s%s%s).%s", star, pkgQual, deref(recv.Type), f.Name_)
} }
// Package-level function. // Package-level function.
return pkgQual + f.Name_ // Prefix with package name for cross-package references only.
if from != f.Pkg {
return fmt.Sprintf("%s.%s", f.Pkg.ImportPath, f.Name_)
}
return f.Name_
} }
// DumpTo prints to w a human readable "disassembly" of the SSA code of // DumpTo prints to w a human readable "disassembly" of the SSA code of
...@@ -408,7 +405,6 @@ func (f *Function) fullName(from *Package) string { ...@@ -408,7 +405,6 @@ func (f *Function) fullName(from *Package) string {
func (f *Function) DumpTo(w io.Writer) { func (f *Function) DumpTo(w io.Writer) {
fmt.Fprintf(w, "# Name: %s\n", f.FullName()) fmt.Fprintf(w, "# Name: %s\n", f.FullName())
fmt.Fprintf(w, "# Declared at %s\n", f.Prog.Files.Position(f.Pos)) fmt.Fprintf(w, "# Declared at %s\n", f.Prog.Files.Position(f.Pos))
fmt.Fprintf(w, "# Type: %s\n", f.Signature)
if f.Enclosing != nil { if f.Enclosing != nil {
fmt.Fprintf(w, "# Parent: %s\n", f.Enclosing.Name()) fmt.Fprintf(w, "# Parent: %s\n", f.Enclosing.Name())
...@@ -421,6 +417,7 @@ func (f *Function) DumpTo(w io.Writer) { ...@@ -421,6 +417,7 @@ func (f *Function) DumpTo(w io.Writer) {
} }
} }
// Function Signature in declaration syntax; derived from types.Signature.String().
io.WriteString(w, "func ") io.WriteString(w, "func ")
params := f.Params params := f.Params
if f.Signature.Recv != nil { if f.Signature.Recv != nil {
...@@ -435,9 +432,23 @@ func (f *Function) DumpTo(w io.Writer) { ...@@ -435,9 +432,23 @@ func (f *Function) DumpTo(w io.Writer) {
} }
io.WriteString(w, v.Name()) io.WriteString(w, v.Name())
io.WriteString(w, " ") io.WriteString(w, " ")
if f.Signature.IsVariadic && i == len(params)-1 {
io.WriteString(w, "...")
}
io.WriteString(w, v.Type().String()) io.WriteString(w, v.Type().String())
} }
io.WriteString(w, "):\n") io.WriteString(w, ")")
if res := f.Signature.Results; res != nil {
io.WriteString(w, " ")
var t types.Type
if len(res) == 1 && res[0].Name == "" {
t = res[0].Type
} else {
t = &types.Result{Values: res}
}
io.WriteString(w, t.String())
}
io.WriteString(w, ":\n")
for _, b := range f.Blocks { for _, b := range f.Blocks {
if b == nil { if b == nil {
......
...@@ -307,7 +307,11 @@ func writeType(buf *bytes.Buffer, typ Type) { ...@@ -307,7 +307,11 @@ func writeType(buf *bytes.Buffer, typ Type) {
case *NamedType: case *NamedType:
s := "<NamedType w/o object>" s := "<NamedType w/o object>"
if t.Obj != nil { if obj := t.Obj; obj != nil {
if obj.Pkg != nil && obj.Pkg.Path != "" {
buf.WriteString(obj.Pkg.Path)
buf.WriteString(".")
}
s = t.Obj.GetName() s = t.Obj.GetName()
} }
buf.WriteString(s) buf.WriteString(s)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment