Source file
src/go/types/object.go
1
2
3
4
5
6
7
8 package types
9
10 import (
11 "bytes"
12 "fmt"
13 "go/constant"
14 "go/token"
15 "strings"
16 "unicode"
17 "unicode/utf8"
18 )
19
20
21
22
23
24
25
26
27
28
29 type Object interface {
30 Parent() *Scope
31 Pos() token.Pos
32 Pkg() *Package
33 Name() string
34 Type() Type
35 Exported() bool
36 Id() string
37
38
39
40 String() string
41
42
43
44
45
46 order() uint32
47
48
49 color() color
50
51
52 setType(Type)
53
54
55 setOrder(uint32)
56
57
58 setColor(color color)
59
60
61 setParent(*Scope)
62
63
64
65
66 sameId(pkg *Package, name string, foldCase bool) bool
67
68
69 scopePos() token.Pos
70
71
72 setScopePos(pos token.Pos)
73 }
74
75 func isExported(name string) bool {
76 ch, _ := utf8.DecodeRuneInString(name)
77 return unicode.IsUpper(ch)
78 }
79
80
81
82 func Id(pkg *Package, name string) string {
83 if isExported(name) {
84 return name
85 }
86
87
88
89
90
91 path := "_"
92
93
94 if pkg != nil && pkg.path != "" {
95 path = pkg.path
96 }
97 return path + "." + name
98 }
99
100
101 type object struct {
102 parent *Scope
103 pos token.Pos
104 pkg *Package
105 name string
106 typ Type
107 order_ uint32
108 color_ color
109 scopePos_ token.Pos
110 }
111
112
113 type color uint32
114
115
116
117 const (
118 white color = iota
119 black
120 grey
121 )
122
123 func (c color) String() string {
124 switch c {
125 case white:
126 return "white"
127 case black:
128 return "black"
129 default:
130 return "grey"
131 }
132 }
133
134
135
136 func colorFor(t Type) color {
137 if t != nil {
138 return black
139 }
140 return white
141 }
142
143
144
145 func (obj *object) Parent() *Scope { return obj.parent }
146
147
148 func (obj *object) Pos() token.Pos { return obj.pos }
149
150
151
152 func (obj *object) Pkg() *Package { return obj.pkg }
153
154
155 func (obj *object) Name() string { return obj.name }
156
157
158 func (obj *object) Type() Type { return obj.typ }
159
160
161
162
163 func (obj *object) Exported() bool { return isExported(obj.name) }
164
165
166 func (obj *object) Id() string { return Id(obj.pkg, obj.name) }
167
168 func (obj *object) String() string { panic("abstract") }
169 func (obj *object) order() uint32 { return obj.order_ }
170 func (obj *object) color() color { return obj.color_ }
171 func (obj *object) scopePos() token.Pos { return obj.scopePos_ }
172
173 func (obj *object) setParent(parent *Scope) { obj.parent = parent }
174 func (obj *object) setType(typ Type) { obj.typ = typ }
175 func (obj *object) setOrder(order uint32) { assert(order > 0); obj.order_ = order }
176 func (obj *object) setColor(color color) { assert(color != white); obj.color_ = color }
177 func (obj *object) setScopePos(pos token.Pos) { obj.scopePos_ = pos }
178
179 func (obj *object) sameId(pkg *Package, name string, foldCase bool) bool {
180
181 if foldCase && strings.EqualFold(obj.name, name) {
182 return true
183 }
184
185
186
187
188 if obj.name != name {
189 return false
190 }
191
192 if obj.Exported() {
193 return true
194 }
195
196 return samePkg(obj.pkg, pkg)
197 }
198
199
200
201
202
203
204
205
206
207
208
209 func (a *object) cmp(b *object) int {
210 if a == b {
211 return 0
212 }
213
214
215 if a == nil {
216 return -1
217 }
218 if b == nil {
219 return +1
220 }
221
222
223 ea := isExported(a.name)
224 eb := isExported(b.name)
225 if ea != eb {
226 if ea {
227 return -1
228 }
229 return +1
230 }
231
232
233 if a.name != b.name {
234 return strings.Compare(a.name, b.name)
235 }
236 if !ea {
237 return strings.Compare(a.pkg.path, b.pkg.path)
238 }
239
240 return 0
241 }
242
243
244
245 type PkgName struct {
246 object
247 imported *Package
248 }
249
250
251
252 func NewPkgName(pos token.Pos, pkg *Package, name string, imported *Package) *PkgName {
253 return &PkgName{object{nil, pos, pkg, name, Typ[Invalid], 0, black, nopos}, imported}
254 }
255
256
257
258 func (obj *PkgName) Imported() *Package { return obj.imported }
259
260
261 type Const struct {
262 object
263 val constant.Value
264 }
265
266
267
268 func NewConst(pos token.Pos, pkg *Package, name string, typ Type, val constant.Value) *Const {
269 return &Const{object{nil, pos, pkg, name, typ, 0, colorFor(typ), nopos}, val}
270 }
271
272
273 func (obj *Const) Val() constant.Value { return obj.val }
274
275 func (*Const) isDependency() {}
276
277
278
279
280
281
282 type TypeName struct {
283 object
284 }
285
286
287
288
289
290
291
292
293 func NewTypeName(pos token.Pos, pkg *Package, name string, typ Type) *TypeName {
294 return &TypeName{object{nil, pos, pkg, name, typ, 0, colorFor(typ), nopos}}
295 }
296
297
298
299 func _NewTypeNameLazy(pos token.Pos, pkg *Package, name string, load func(named *Named) (tparams []*TypeParam, underlying Type, methods []*Func)) *TypeName {
300 obj := NewTypeName(pos, pkg, name, nil)
301 NewNamed(obj, nil, nil).loader = load
302 return obj
303 }
304
305
306 func (obj *TypeName) IsAlias() bool {
307 switch t := obj.typ.(type) {
308 case nil:
309 return false
310
311
312 case *Basic:
313
314 if obj.pkg == Unsafe {
315 return false
316 }
317
318
319
320
321
322
323 return obj.pkg != nil || t.name != obj.name || t == universeByte || t == universeRune
324 case *Named:
325 return obj != t.obj
326 case *TypeParam:
327 return obj != t.obj
328 default:
329 return true
330 }
331 }
332
333
334 type Var struct {
335 object
336 origin *Var
337 embedded bool
338 isField bool
339 isParam bool
340 }
341
342
343
344 func NewVar(pos token.Pos, pkg *Package, name string, typ Type) *Var {
345 return &Var{object: object{nil, pos, pkg, name, typ, 0, colorFor(typ), nopos}}
346 }
347
348
349 func NewParam(pos token.Pos, pkg *Package, name string, typ Type) *Var {
350 return &Var{object: object{nil, pos, pkg, name, typ, 0, colorFor(typ), nopos}, isParam: true}
351 }
352
353
354
355
356 func NewField(pos token.Pos, pkg *Package, name string, typ Type, embedded bool) *Var {
357 return &Var{object: object{nil, pos, pkg, name, typ, 0, colorFor(typ), nopos}, embedded: embedded, isField: true}
358 }
359
360
361
362 func (obj *Var) Anonymous() bool { return obj.embedded }
363
364
365 func (obj *Var) Embedded() bool { return obj.embedded }
366
367
368 func (obj *Var) IsField() bool { return obj.isField }
369
370
371
372
373
374
375
376
377 func (obj *Var) Origin() *Var {
378 if obj.origin != nil {
379 return obj.origin
380 }
381 return obj
382 }
383
384 func (*Var) isDependency() {}
385
386
387
388
389 type Func struct {
390 object
391 hasPtrRecv_ bool
392 origin *Func
393 }
394
395
396
397 func NewFunc(pos token.Pos, pkg *Package, name string, sig *Signature) *Func {
398 var typ Type
399 if sig != nil {
400 typ = sig
401 } else {
402
403
404
405
406 }
407 return &Func{object{nil, pos, pkg, name, typ, 0, colorFor(typ), nopos}, false, nil}
408 }
409
410
411 func (obj *Func) Signature() *Signature {
412 if obj.typ != nil {
413 return obj.typ.(*Signature)
414 }
415
416
417
418
419
420
421
422
423 return new(Signature)
424 }
425
426
427
428 func (obj *Func) FullName() string {
429 var buf bytes.Buffer
430 writeFuncName(&buf, obj, nil)
431 return buf.String()
432 }
433
434
435
436
437 func (obj *Func) Scope() *Scope { return obj.typ.(*Signature).scope }
438
439
440
441
442
443
444
445
446 func (obj *Func) Origin() *Func {
447 if obj.origin != nil {
448 return obj.origin
449 }
450 return obj
451 }
452
453
454
455
456
457 func (obj *Func) Pkg() *Package { return obj.object.Pkg() }
458
459
460 func (obj *Func) hasPtrRecv() bool {
461
462
463
464
465 if sig, _ := obj.typ.(*Signature); sig != nil && sig.recv != nil {
466 _, isPtr := deref(sig.recv.typ)
467 return isPtr
468 }
469
470
471
472
473
474
475 return obj.hasPtrRecv_
476 }
477
478 func (*Func) isDependency() {}
479
480
481
482 type Label struct {
483 object
484 used bool
485 }
486
487
488 func NewLabel(pos token.Pos, pkg *Package, name string) *Label {
489 return &Label{object{pos: pos, pkg: pkg, name: name, typ: Typ[Invalid], color_: black}, false}
490 }
491
492
493
494 type Builtin struct {
495 object
496 id builtinId
497 }
498
499 func newBuiltin(id builtinId) *Builtin {
500 return &Builtin{object{name: predeclaredFuncs[id].name, typ: Typ[Invalid], color_: black}, id}
501 }
502
503
504 type Nil struct {
505 object
506 }
507
508 func writeObject(buf *bytes.Buffer, obj Object, qf Qualifier) {
509 var tname *TypeName
510 typ := obj.Type()
511
512 switch obj := obj.(type) {
513 case *PkgName:
514 fmt.Fprintf(buf, "package %s", obj.Name())
515 if path := obj.imported.path; path != "" && path != obj.name {
516 fmt.Fprintf(buf, " (%q)", path)
517 }
518 return
519
520 case *Const:
521 buf.WriteString("const")
522
523 case *TypeName:
524 tname = obj
525 buf.WriteString("type")
526 if isTypeParam(typ) {
527 buf.WriteString(" parameter")
528 }
529
530 case *Var:
531 if obj.isField {
532 buf.WriteString("field")
533 } else {
534 buf.WriteString("var")
535 }
536
537 case *Func:
538 buf.WriteString("func ")
539 writeFuncName(buf, obj, qf)
540 if typ != nil {
541 WriteSignature(buf, typ.(*Signature), qf)
542 }
543 return
544
545 case *Label:
546 buf.WriteString("label")
547 typ = nil
548
549 case *Builtin:
550 buf.WriteString("builtin")
551 typ = nil
552
553 case *Nil:
554 buf.WriteString("nil")
555 return
556
557 default:
558 panic(fmt.Sprintf("writeObject(%T)", obj))
559 }
560
561 buf.WriteByte(' ')
562
563
564 if obj.Pkg() != nil && obj.Pkg().scope.Lookup(obj.Name()) == obj {
565 buf.WriteString(packagePrefix(obj.Pkg(), qf))
566 }
567 buf.WriteString(obj.Name())
568
569 if typ == nil {
570 return
571 }
572
573 if tname != nil {
574 switch t := typ.(type) {
575 case *Basic:
576
577
578 return
579 case genericType:
580 if t.TypeParams().Len() > 0 {
581 newTypeWriter(buf, qf).tParamList(t.TypeParams().list())
582 }
583 }
584 if tname.IsAlias() {
585 buf.WriteString(" =")
586 if alias, ok := typ.(*Alias); ok {
587 typ = alias.fromRHS
588 }
589 } else if t, _ := typ.(*TypeParam); t != nil {
590 typ = t.bound
591 } else {
592
593
594 typ = under(typ)
595 }
596 }
597
598
599
600
601 if obj.Name() == "any" && obj.Parent() == Universe {
602 assert(Identical(typ, &emptyInterface))
603 typ = &emptyInterface
604 }
605
606 buf.WriteByte(' ')
607 WriteType(buf, typ, qf)
608 }
609
610 func packagePrefix(pkg *Package, qf Qualifier) string {
611 if pkg == nil {
612 return ""
613 }
614 var s string
615 if qf != nil {
616 s = qf(pkg)
617 } else {
618 s = pkg.Path()
619 }
620 if s != "" {
621 s += "."
622 }
623 return s
624 }
625
626
627
628
629 func ObjectString(obj Object, qf Qualifier) string {
630 var buf bytes.Buffer
631 writeObject(&buf, obj, qf)
632 return buf.String()
633 }
634
635 func (obj *PkgName) String() string { return ObjectString(obj, nil) }
636 func (obj *Const) String() string { return ObjectString(obj, nil) }
637 func (obj *TypeName) String() string { return ObjectString(obj, nil) }
638 func (obj *Var) String() string { return ObjectString(obj, nil) }
639 func (obj *Func) String() string { return ObjectString(obj, nil) }
640 func (obj *Label) String() string { return ObjectString(obj, nil) }
641 func (obj *Builtin) String() string { return ObjectString(obj, nil) }
642 func (obj *Nil) String() string { return ObjectString(obj, nil) }
643
644 func writeFuncName(buf *bytes.Buffer, f *Func, qf Qualifier) {
645 if f.typ != nil {
646 sig := f.typ.(*Signature)
647 if recv := sig.Recv(); recv != nil {
648 buf.WriteByte('(')
649 if _, ok := recv.Type().(*Interface); ok {
650
651
652
653
654 buf.WriteString("interface")
655 } else {
656 WriteType(buf, recv.Type(), qf)
657 }
658 buf.WriteByte(')')
659 buf.WriteByte('.')
660 } else if f.pkg != nil {
661 buf.WriteString(packagePrefix(f.pkg, qf))
662 }
663 }
664 buf.WriteString(f.name)
665 }
666
View as plain text