1
2
3
4
5 package types2
6
7 import (
8 "cmd/compile/internal/syntax"
9 . "internal/types/errors"
10 "sort"
11 "strings"
12 )
13
14
15
16
17
18
19
20
21
22
23
24
25
26 type _TypeSet struct {
27 methods []*Func
28 terms termlist
29 comparable bool
30 }
31
32
33 func (s *_TypeSet) IsEmpty() bool { return s.terms.isEmpty() }
34
35
36 func (s *_TypeSet) IsAll() bool { return s.IsMethodSet() && len(s.methods) == 0 }
37
38
39 func (s *_TypeSet) IsMethodSet() bool { return !s.comparable && s.terms.isAll() }
40
41
42 func (s *_TypeSet) IsComparable(seen map[Type]bool) bool {
43 if s.terms.isAll() {
44 return s.comparable
45 }
46 return s.is(func(t *term) bool {
47 return t != nil && comparable(t.typ, false, seen, nil)
48 })
49 }
50
51
52 func (s *_TypeSet) NumMethods() int { return len(s.methods) }
53
54
55
56 func (s *_TypeSet) Method(i int) *Func { return s.methods[i] }
57
58
59 func (s *_TypeSet) LookupMethod(pkg *Package, name string, foldCase bool) (int, *Func) {
60 return lookupMethod(s.methods, pkg, name, foldCase)
61 }
62
63 func (s *_TypeSet) String() string {
64 switch {
65 case s.IsEmpty():
66 return "∅"
67 case s.IsAll():
68 return "𝓤"
69 }
70
71 hasMethods := len(s.methods) > 0
72 hasTerms := s.hasTerms()
73
74 var buf strings.Builder
75 buf.WriteByte('{')
76 if s.comparable {
77 buf.WriteString("comparable")
78 if hasMethods || hasTerms {
79 buf.WriteString("; ")
80 }
81 }
82 for i, m := range s.methods {
83 if i > 0 {
84 buf.WriteString("; ")
85 }
86 buf.WriteString(m.String())
87 }
88 if hasMethods && hasTerms {
89 buf.WriteString("; ")
90 }
91 if hasTerms {
92 buf.WriteString(s.terms.String())
93 }
94 buf.WriteString("}")
95 return buf.String()
96 }
97
98
99
100
101
102 func (s *_TypeSet) hasTerms() bool { return !s.terms.isEmpty() && !s.terms.isAll() }
103
104
105 func (s1 *_TypeSet) subsetOf(s2 *_TypeSet) bool { return s1.terms.subsetOf(s2.terms) }
106
107
108
109
110
111
112 func (s *_TypeSet) is(f func(*term) bool) bool {
113 if !s.hasTerms() {
114 return f(nil)
115 }
116 for _, t := range s.terms {
117 assert(t.typ != nil)
118 if !f(t) {
119 return false
120 }
121 }
122 return true
123 }
124
125
126
127
128 func (s *_TypeSet) underIs(f func(Type) bool) bool {
129 if !s.hasTerms() {
130 return f(nil)
131 }
132 for _, t := range s.terms {
133 assert(t.typ != nil)
134
135 u := t.typ
136 if !t.tilde {
137 u = under(u)
138 }
139 if debug {
140 assert(Identical(u, under(u)))
141 }
142 if !f(u) {
143 return false
144 }
145 }
146 return true
147 }
148
149
150 var topTypeSet = _TypeSet{terms: allTermlist}
151
152
153 func computeInterfaceTypeSet(check *Checker, pos syntax.Pos, ityp *Interface) *_TypeSet {
154 if ityp.tset != nil {
155 return ityp.tset
156 }
157
158
159
160
161
162
163
164 if !ityp.complete {
165 return &topTypeSet
166 }
167
168 if check != nil && check.conf.Trace {
169
170
171
172 if !pos.IsKnown() && len(ityp.methods) > 0 {
173 pos = ityp.methods[0].pos
174 }
175
176 check.trace(pos, "-- type set for %s", ityp)
177 check.indent++
178 defer func() {
179 check.indent--
180 check.trace(pos, "=> %s ", ityp.typeSet())
181 }()
182 }
183
184
185
186
187
188
189 ityp.tset = &_TypeSet{terms: allTermlist}
190
191 var unionSets map[*Union]*_TypeSet
192 if check != nil {
193 if check.unionTypeSets == nil {
194 check.unionTypeSets = make(map[*Union]*_TypeSet)
195 }
196 unionSets = check.unionTypeSets
197 } else {
198 unionSets = make(map[*Union]*_TypeSet)
199 }
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214 var seen objset
215 var allMethods []*Func
216 mpos := make(map[*Func]syntax.Pos)
217 addMethod := func(pos syntax.Pos, m *Func, explicit bool) {
218 switch other := seen.insert(m); {
219 case other == nil:
220 allMethods = append(allMethods, m)
221 mpos[m] = pos
222 case explicit:
223 if check != nil {
224 var err error_
225 err.code = DuplicateDecl
226 err.errorf(pos, "duplicate method %s", m.name)
227 err.errorf(mpos[other.(*Func)], "other declaration of %s", m.name)
228 check.report(&err)
229 }
230 default:
231
232
233
234
235
236 if check != nil {
237 check.later(func() {
238 if !check.allowVersion(m.pkg, pos, go1_14) || !Identical(m.typ, other.Type()) {
239 var err error_
240 err.code = DuplicateDecl
241 err.errorf(pos, "duplicate method %s", m.name)
242 err.errorf(mpos[other.(*Func)], "other declaration of %s", m.name)
243 check.report(&err)
244 }
245 }).describef(pos, "duplicate method check for %s", m.name)
246 }
247 }
248 }
249
250 for _, m := range ityp.methods {
251 addMethod(m.pos, m, true)
252 }
253
254
255 allTerms := allTermlist
256 allComparable := false
257 for i, typ := range ityp.embeddeds {
258
259
260
261 var pos syntax.Pos
262 if ityp.embedPos != nil {
263 pos = (*ityp.embedPos)[i]
264 }
265 var comparable bool
266 var terms termlist
267 switch u := under(typ).(type) {
268 case *Interface:
269
270 assert(!isTypeParam(typ))
271 tset := computeInterfaceTypeSet(check, pos, u)
272
273 if check != nil && check.isImportedConstraint(typ) && !check.verifyVersionf(pos, go1_18, "embedding constraint interface %s", typ) {
274 continue
275 }
276 comparable = tset.comparable
277 for _, m := range tset.methods {
278 addMethod(pos, m, false)
279 }
280 terms = tset.terms
281 case *Union:
282 if check != nil && !check.verifyVersionf(pos, go1_18, "embedding interface element %s", u) {
283 continue
284 }
285 tset := computeUnionTypeSet(check, unionSets, pos, u)
286 if tset == &invalidTypeSet {
287 continue
288 }
289 assert(!tset.comparable)
290 assert(len(tset.methods) == 0)
291 terms = tset.terms
292 default:
293 if !isValid(u) {
294 continue
295 }
296 if check != nil && !check.verifyVersionf(pos, go1_18, "embedding non-interface type %s", typ) {
297 continue
298 }
299 terms = termlist{{false, typ}}
300 }
301
302
303
304
305 allTerms, allComparable = intersectTermLists(allTerms, allComparable, terms, comparable)
306 }
307
308 ityp.tset.comparable = allComparable
309 if len(allMethods) != 0 {
310 sortMethods(allMethods)
311 ityp.tset.methods = allMethods
312 }
313 ityp.tset.terms = allTerms
314
315 return ityp.tset
316 }
317
318
319
320
321
322
323
324 func intersectTermLists(xterms termlist, xcomp bool, yterms termlist, ycomp bool) (termlist, bool) {
325 terms := xterms.intersect(yterms)
326
327
328 comp := xcomp || ycomp
329 if comp && !terms.isAll() {
330
331 i := 0
332 for _, t := range terms {
333 assert(t.typ != nil)
334 if comparable(t.typ, false , nil, nil) {
335 terms[i] = t
336 i++
337 }
338 }
339 terms = terms[:i]
340 if !terms.isAll() {
341 comp = false
342 }
343 }
344 assert(!comp || terms.isAll())
345 return terms, comp
346 }
347
348 func sortMethods(list []*Func) {
349 sort.Sort(byUniqueMethodName(list))
350 }
351
352 func assertSortedMethods(list []*Func) {
353 if !debug {
354 panic("assertSortedMethods called outside debug mode")
355 }
356 if !sort.IsSorted(byUniqueMethodName(list)) {
357 panic("methods not sorted")
358 }
359 }
360
361
362 type byUniqueMethodName []*Func
363
364 func (a byUniqueMethodName) Len() int { return len(a) }
365 func (a byUniqueMethodName) Less(i, j int) bool { return a[i].less(&a[j].object) }
366 func (a byUniqueMethodName) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
367
368
369
370
371 var invalidTypeSet _TypeSet
372
373
374
375 func computeUnionTypeSet(check *Checker, unionSets map[*Union]*_TypeSet, pos syntax.Pos, utyp *Union) *_TypeSet {
376 if tset, _ := unionSets[utyp]; tset != nil {
377 return tset
378 }
379
380
381 unionSets[utyp] = new(_TypeSet)
382
383 var allTerms termlist
384 for _, t := range utyp.terms {
385 var terms termlist
386 u := under(t.typ)
387 if ui, _ := u.(*Interface); ui != nil {
388
389 assert(!isTypeParam(t.typ))
390 terms = computeInterfaceTypeSet(check, pos, ui).terms
391 } else if !isValid(u) {
392 continue
393 } else {
394 if t.tilde && !Identical(t.typ, u) {
395
396
397 t = nil
398 }
399 terms = termlist{(*term)(t)}
400 }
401
402
403 allTerms = allTerms.union(terms)
404 if len(allTerms) > maxTermCount {
405 if check != nil {
406 check.errorf(pos, InvalidUnion, "cannot handle more than %d union terms (implementation limitation)", maxTermCount)
407 }
408 unionSets[utyp] = &invalidTypeSet
409 return unionSets[utyp]
410 }
411 }
412 unionSets[utyp].terms = allTerms
413
414 return unionSets[utyp]
415 }
416
View as plain text