Skip to content

Commit bbd15ff

Browse files
committed
go/types: report error when recognizing issue #18395.
The fix (CL 79575) for #18395 is too risky at this stage of the Go 1.10 release process. Since issue #18395 is easily recognized (but not easily fixed), report an error instead of silently continuing. This avoids inscrutable follow on errors. Also, make sure all empty interfaces are "completed", and adjust printing code to report incomplete interfaces. For #18395. Change-Id: I7fa5f97ff31ac9775c9a6d318fce9f526b0350cd Reviewed-on: https://go-review.googlesource.com/80455 Reviewed-by: Alan Donovan <[email protected]>
1 parent 98a8e52 commit bbd15ff

File tree

8 files changed

+71
-14
lines changed

8 files changed

+71
-14
lines changed

src/go/internal/gccgoimporter/parser_test.go

+5
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,11 @@ func TestTypeParser(t *testing.T) {
4545
t.Errorf("expected full parse, stopped at %q", p.lit)
4646
}
4747

48+
// interfaces must be explicitly completed
49+
if ityp, _ := typ.(*types.Interface); ityp != nil {
50+
ityp.Complete()
51+
}
52+
4853
got := typ.String()
4954
if got != test.want {
5055
t.Errorf("got type %q, expected %q", got, test.want)

src/go/types/builtins.go

+3-4
Original file line numberDiff line numberDiff line change
@@ -470,15 +470,14 @@ func (check *Checker) builtin(x *operand, call *ast.CallExpr, id builtinId) (_ b
470470

471471
case _Panic:
472472
// panic(x)
473-
T := new(Interface)
474-
check.assignment(x, T, "argument to panic")
473+
check.assignment(x, &emptyInterface, "argument to panic")
475474
if x.mode == invalid {
476475
return
477476
}
478477

479478
x.mode = novalue
480479
if check.Types != nil {
481-
check.recordBuiltinType(call.Fun, makeSig(nil, T))
480+
check.recordBuiltinType(call.Fun, makeSig(nil, &emptyInterface))
482481
}
483482

484483
case _Print, _Println:
@@ -508,7 +507,7 @@ func (check *Checker) builtin(x *operand, call *ast.CallExpr, id builtinId) (_ b
508507
case _Recover:
509508
// recover() interface{}
510509
x.mode = value
511-
x.typ = new(Interface)
510+
x.typ = &emptyInterface
512511
if check.Types != nil {
513512
check.recordBuiltinType(call.Fun, makeSig(x.typ))
514513
}

src/go/types/object_test.go

+6-6
Original file line numberDiff line numberDiff line change
@@ -32,12 +32,12 @@ func TestIsAlias(t *testing.T) {
3232
{NewTypeName(0, nil, "t0", nil), false}, // no type yet
3333
{NewTypeName(0, pkg, "t0", nil), false}, // no type yet
3434
{t1, false}, // type name refers to named type and vice versa
35-
{NewTypeName(0, nil, "t2", new(Interface)), true}, // type name refers to unnamed type
36-
{NewTypeName(0, pkg, "t3", n1), true}, // type name refers to named type with different type name
37-
{NewTypeName(0, nil, "t4", Typ[Int32]), true}, // type name refers to basic type with different name
38-
{NewTypeName(0, nil, "int32", Typ[Int32]), false}, // type name refers to basic type with same name
39-
{NewTypeName(0, pkg, "int32", Typ[Int32]), true}, // type name is declared in user-defined package (outside Universe)
40-
{NewTypeName(0, nil, "rune", Typ[Rune]), true}, // type name refers to basic type rune which is an alias already
35+
{NewTypeName(0, nil, "t2", &emptyInterface), true}, // type name refers to unnamed type
36+
{NewTypeName(0, pkg, "t3", n1), true}, // type name refers to named type with different type name
37+
{NewTypeName(0, nil, "t4", Typ[Int32]), true}, // type name refers to basic type with different name
38+
{NewTypeName(0, nil, "int32", Typ[Int32]), false}, // type name refers to basic type with same name
39+
{NewTypeName(0, pkg, "int32", Typ[Int32]), true}, // type name is declared in user-defined package (outside Universe)
40+
{NewTypeName(0, nil, "rune", Typ[Rune]), true}, // type name refers to basic type rune which is an alias already
4141
} {
4242
check(test.name, test.alias)
4343
}

src/go/types/testdata/cycles4.src

+12
Original file line numberDiff line numberDiff line change
@@ -108,3 +108,15 @@ type Element interface {
108108
type Event interface {
109109
Target() Element
110110
}
111+
112+
// Recognize issue #13895.
113+
114+
type (
115+
_ interface{ m(B1) }
116+
A1 interface{ a(D1) }
117+
B1 interface{ A1 }
118+
C1 interface{ B1 /* ERROR issue #18395 */ }
119+
D1 interface{ C1 }
120+
)
121+
122+
var _ A1 = C1 /* ERROR cannot use C1 */ (nil)

src/go/types/type.go

+13-1
Original file line numberDiff line numberDiff line change
@@ -246,10 +246,22 @@ type Interface struct {
246246
allMethods []*Func // ordered list of methods declared with or embedded in this interface (TODO(gri): replace with mset)
247247
}
248248

249-
// NewInterface returns a new interface for the given methods and embedded types.
249+
// emptyInterface represents the empty (completed) interface
250+
var emptyInterface = Interface{allMethods: markComplete}
251+
252+
// markComplete is used to mark an empty interface as completely
253+
// set up by setting the allMethods field to a non-nil empty slice.
254+
var markComplete = make([]*Func, 0)
255+
256+
// NewInterface returns a new (incomplete) interface for the given methods and embedded types.
257+
// To compute the method set of the interface, Complete must be called.
250258
func NewInterface(methods []*Func, embeddeds []*Named) *Interface {
251259
typ := new(Interface)
252260

261+
if len(methods) == 0 && len(embeddeds) == 0 {
262+
return typ
263+
}
264+
253265
var mset objset
254266
for _, m := range methods {
255267
if mset.insert(m) != nil {

src/go/types/typestring.go

+10
Original file line numberDiff line numberDiff line change
@@ -156,6 +156,7 @@ func writeType(buf *bytes.Buffer, typ Type, qf Qualifier, visited []Type) {
156156
// }
157157
//
158158
buf.WriteString("interface{")
159+
empty := true
159160
if gcCompatibilityMode {
160161
// print flattened interface
161162
// (useful to compare against gc-generated interfaces)
@@ -165,6 +166,7 @@ func writeType(buf *bytes.Buffer, typ Type, qf Qualifier, visited []Type) {
165166
}
166167
buf.WriteString(m.name)
167168
writeSignature(buf, m.typ.(*Signature), qf, visited)
169+
empty = false
168170
}
169171
} else {
170172
// print explicit interface methods and embedded types
@@ -174,14 +176,22 @@ func writeType(buf *bytes.Buffer, typ Type, qf Qualifier, visited []Type) {
174176
}
175177
buf.WriteString(m.name)
176178
writeSignature(buf, m.typ.(*Signature), qf, visited)
179+
empty = false
177180
}
178181
for i, typ := range t.embeddeds {
179182
if i > 0 || len(t.methods) > 0 {
180183
buf.WriteString("; ")
181184
}
182185
writeType(buf, typ, qf, visited)
186+
empty = false
183187
}
184188
}
189+
if t.allMethods == nil || len(t.methods) > len(t.allMethods) {
190+
if !empty {
191+
buf.WriteByte(' ')
192+
}
193+
buf.WriteString("/* incomplete */")
194+
}
185195
buf.WriteByte('}')
186196

187197
case *Map:

src/go/types/typestring_test.go

+20
Original file line numberDiff line numberDiff line change
@@ -138,6 +138,26 @@ func TestTypeString(t *testing.T) {
138138
}
139139
}
140140

141+
func TestIncompleteInterfaces(t *testing.T) {
142+
sig := NewSignature(nil, nil, nil, false)
143+
for _, test := range []struct {
144+
typ *Interface
145+
want string
146+
}{
147+
{new(Interface), "interface{/* incomplete */}"},
148+
{new(Interface).Complete(), "interface{}"},
149+
{NewInterface(nil, nil), "interface{/* incomplete */}"},
150+
{NewInterface(nil, nil).Complete(), "interface{}"},
151+
{NewInterface([]*Func{NewFunc(token.NoPos, nil, "m", sig)}, nil), "interface{m() /* incomplete */}"},
152+
{NewInterface([]*Func{NewFunc(token.NoPos, nil, "m", sig)}, nil).Complete(), "interface{m()}"},
153+
} {
154+
got := test.typ.String()
155+
if got != test.want {
156+
t.Errorf("got: %s, want: %s", got, test.want)
157+
}
158+
}
159+
}
160+
141161
func TestQualifiedTypeString(t *testing.T) {
142162
p, _ := pkgFor("p.go", "package p; type T int", nil)
143163
q, _ := pkgFor("q.go", "package q", nil)

src/go/types/typexpr.go

+2-3
Original file line numberDiff line numberDiff line change
@@ -540,9 +540,8 @@ func (check *Checker) interfaceType(iface *Interface, ityp *ast.InterfaceType, d
540540
}
541541
iface.embeddeds = append(iface.embeddeds, named)
542542
// collect embedded methods
543-
if debug && embed.allMethods == nil {
544-
check.dump("%s: incomplete embedded interface %s", pos, named)
545-
unreachable()
543+
if embed.allMethods == nil {
544+
check.errorf(pos, "internal error: incomplete embedded interface %s (issue #18395)", named)
546545
}
547546
for _, m := range embed.allMethods {
548547
if check.declareInSet(&mset, pos, m) {

0 commit comments

Comments
 (0)