Source file
src/go/types/decl.go
1
2
3
4
5 package types
6
7 import (
8 "fmt"
9 "go/ast"
10 "go/constant"
11 "go/token"
12 "internal/buildcfg"
13 . "internal/types/errors"
14 "slices"
15 )
16
17 func (check *Checker) declare(scope *Scope, id *ast.Ident, obj Object, pos token.Pos) {
18
19
20
21
22 if obj.Name() != "_" {
23 if alt := scope.Insert(obj); alt != nil {
24 err := check.newError(DuplicateDecl)
25 err.addf(obj, "%s redeclared in this block", obj.Name())
26 err.addAltDecl(alt)
27 err.report()
28 return
29 }
30 obj.setScopePos(pos)
31 }
32 if id != nil {
33 check.recordDef(id, obj)
34 }
35 }
36
37
38 func pathString(path []Object) string {
39 var s string
40 for i, p := range path {
41 if i > 0 {
42 s += "->"
43 }
44 s += p.Name()
45 }
46 return s
47 }
48
49
50
51 func (check *Checker) objDecl(obj Object, def *TypeName) {
52 if tracePos {
53 check.pushPos(atPos(obj.Pos()))
54 defer func() {
55
56 if p := recover(); p != nil {
57 panic(p)
58 }
59 check.popPos()
60 }()
61 }
62
63 if check.conf._Trace && obj.Type() == nil {
64 if check.indent == 0 {
65 fmt.Println()
66 }
67 check.trace(obj.Pos(), "-- checking %s (%s, objPath = %s)", obj, obj.color(), pathString(check.objPath))
68 check.indent++
69 defer func() {
70 check.indent--
71 check.trace(obj.Pos(), "=> %s (%s)", obj, obj.color())
72 }()
73 }
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102 if obj.color() == white && obj.Type() != nil {
103 obj.setColor(black)
104 return
105 }
106
107 switch obj.color() {
108 case white:
109 assert(obj.Type() == nil)
110
111
112
113 obj.setColor(grey + color(check.push(obj)))
114 defer func() {
115 check.pop().setColor(black)
116 }()
117
118 case black:
119 assert(obj.Type() != nil)
120 return
121
122 default:
123
124 fallthrough
125
126 case grey:
127
128
129
130
131
132
133
134
135
136
137 switch obj := obj.(type) {
138 case *Const:
139 if !check.validCycle(obj) || obj.typ == nil {
140 obj.typ = Typ[Invalid]
141 }
142
143 case *Var:
144 if !check.validCycle(obj) || obj.typ == nil {
145 obj.typ = Typ[Invalid]
146 }
147
148 case *TypeName:
149 if !check.validCycle(obj) {
150
151
152
153
154
155 obj.typ = Typ[Invalid]
156 }
157
158 case *Func:
159 if !check.validCycle(obj) {
160
161
162
163
164
165
166 }
167
168 default:
169 panic("unreachable")
170 }
171 assert(obj.Type() != nil)
172 return
173 }
174
175 d := check.objMap[obj]
176 if d == nil {
177 check.dump("%v: %s should have been declared", obj.Pos(), obj)
178 panic("unreachable")
179 }
180
181
182 defer func(env environment) {
183 check.environment = env
184 }(check.environment)
185 check.environment = environment{scope: d.file, version: d.version}
186
187
188
189
190
191
192 switch obj := obj.(type) {
193 case *Const:
194 check.decl = d
195 check.constDecl(obj, d.vtyp, d.init, d.inherited)
196 case *Var:
197 check.decl = d
198 check.varDecl(obj, d.lhs, d.vtyp, d.init)
199 case *TypeName:
200
201 check.typeDecl(obj, d.tdecl, def)
202 check.collectMethods(obj)
203 case *Func:
204
205 check.funcDecl(obj, d)
206 default:
207 panic("unreachable")
208 }
209 }
210
211
212
213 func (check *Checker) validCycle(obj Object) (valid bool) {
214
215 if debug {
216 info := check.objMap[obj]
217 inObjMap := info != nil && (info.fdecl == nil || info.fdecl.Recv == nil)
218 isPkgObj := obj.Parent() == check.pkg.scope
219 if isPkgObj != inObjMap {
220 check.dump("%v: inconsistent object map for %s (isPkgObj = %v, inObjMap = %v)", obj.Pos(), obj, isPkgObj, inObjMap)
221 panic("unreachable")
222 }
223 }
224
225
226 assert(obj.color() >= grey)
227 start := obj.color() - grey
228 cycle := check.objPath[start:]
229 tparCycle := false
230 nval := 0
231 ndef := 0
232 loop:
233 for _, obj := range cycle {
234 switch obj := obj.(type) {
235 case *Const, *Var:
236 nval++
237 case *TypeName:
238
239
240
241 if check.inTParamList && isGeneric(obj.typ) {
242 tparCycle = true
243 break loop
244 }
245
246
247
248
249
250
251
252
253
254
255 var alias bool
256 if check.conf._EnableAlias {
257 alias = obj.IsAlias()
258 } else {
259 if d := check.objMap[obj]; d != nil {
260 alias = d.tdecl.Assign.IsValid()
261 } else {
262 alias = obj.IsAlias()
263 }
264 }
265 if !alias {
266 ndef++
267 }
268 case *Func:
269
270 default:
271 panic("unreachable")
272 }
273 }
274
275 if check.conf._Trace {
276 check.trace(obj.Pos(), "## cycle detected: objPath = %s->%s (len = %d)", pathString(cycle), obj.Name(), len(cycle))
277 if tparCycle {
278 check.trace(obj.Pos(), "## cycle contains: generic type in a type parameter list")
279 } else {
280 check.trace(obj.Pos(), "## cycle contains: %d values, %d type definitions", nval, ndef)
281 }
282 defer func() {
283 if valid {
284 check.trace(obj.Pos(), "=> cycle is valid")
285 } else {
286 check.trace(obj.Pos(), "=> error: cycle is invalid")
287 }
288 }()
289 }
290
291 if !tparCycle {
292
293
294
295 if nval == len(cycle) {
296 return true
297 }
298
299
300
301
302 if nval == 0 && ndef > 0 {
303 return true
304 }
305 }
306
307 check.cycleError(cycle, firstInSrc(cycle))
308 return false
309 }
310
311
312 func (check *Checker) cycleError(cycle []Object, start int) {
313
314
315
316
317 name := func(obj Object) string {
318 return packagePrefix(obj.Pkg(), check.qualifier) + obj.Name()
319 }
320
321
322 obj := cycle[start]
323 tname, _ := obj.(*TypeName)
324 if tname != nil && tname.IsAlias() {
325
326
327 if !check.conf._EnableAlias {
328 check.validAlias(tname, Typ[Invalid])
329 }
330 }
331
332
333 if len(cycle) == 1 {
334 if tname != nil {
335 check.errorf(obj, InvalidDeclCycle, "invalid recursive type: %s refers to itself", name(obj))
336 } else {
337 check.errorf(obj, InvalidDeclCycle, "invalid cycle in declaration: %s refers to itself", name(obj))
338 }
339 return
340 }
341
342 err := check.newError(InvalidDeclCycle)
343 if tname != nil {
344 err.addf(obj, "invalid recursive type %s", name(obj))
345 } else {
346 err.addf(obj, "invalid cycle in declaration of %s", name(obj))
347 }
348
349 for i := range cycle {
350 next := cycle[(start+i+1)%len(cycle)]
351 err.addf(obj, "%s refers to %s", name(obj), name(next))
352 obj = next
353 }
354 err.report()
355 }
356
357
358
359 func firstInSrc(path []Object) int {
360 fst, pos := 0, path[0].Pos()
361 for i, t := range path[1:] {
362 if cmpPos(t.Pos(), pos) < 0 {
363 fst, pos = i+1, t.Pos()
364 }
365 }
366 return fst
367 }
368
369 type (
370 decl interface {
371 node() ast.Node
372 }
373
374 importDecl struct{ spec *ast.ImportSpec }
375 constDecl struct {
376 spec *ast.ValueSpec
377 iota int
378 typ ast.Expr
379 init []ast.Expr
380 inherited bool
381 }
382 varDecl struct{ spec *ast.ValueSpec }
383 typeDecl struct{ spec *ast.TypeSpec }
384 funcDecl struct{ decl *ast.FuncDecl }
385 )
386
387 func (d importDecl) node() ast.Node { return d.spec }
388 func (d constDecl) node() ast.Node { return d.spec }
389 func (d varDecl) node() ast.Node { return d.spec }
390 func (d typeDecl) node() ast.Node { return d.spec }
391 func (d funcDecl) node() ast.Node { return d.decl }
392
393 func (check *Checker) walkDecls(decls []ast.Decl, f func(decl)) {
394 for _, d := range decls {
395 check.walkDecl(d, f)
396 }
397 }
398
399 func (check *Checker) walkDecl(d ast.Decl, f func(decl)) {
400 switch d := d.(type) {
401 case *ast.BadDecl:
402
403 case *ast.GenDecl:
404 var last *ast.ValueSpec
405 for iota, s := range d.Specs {
406 switch s := s.(type) {
407 case *ast.ImportSpec:
408 f(importDecl{s})
409 case *ast.ValueSpec:
410 switch d.Tok {
411 case token.CONST:
412
413 inherited := true
414 switch {
415 case s.Type != nil || len(s.Values) > 0:
416 last = s
417 inherited = false
418 case last == nil:
419 last = new(ast.ValueSpec)
420 inherited = false
421 }
422 check.arityMatch(s, last)
423 f(constDecl{spec: s, iota: iota, typ: last.Type, init: last.Values, inherited: inherited})
424 case token.VAR:
425 check.arityMatch(s, nil)
426 f(varDecl{s})
427 default:
428 check.errorf(s, InvalidSyntaxTree, "invalid token %s", d.Tok)
429 }
430 case *ast.TypeSpec:
431 f(typeDecl{s})
432 default:
433 check.errorf(s, InvalidSyntaxTree, "unknown ast.Spec node %T", s)
434 }
435 }
436 case *ast.FuncDecl:
437 f(funcDecl{d})
438 default:
439 check.errorf(d, InvalidSyntaxTree, "unknown ast.Decl node %T", d)
440 }
441 }
442
443 func (check *Checker) constDecl(obj *Const, typ, init ast.Expr, inherited bool) {
444 assert(obj.typ == nil)
445
446
447 defer func(iota constant.Value, errpos positioner) {
448 check.iota = iota
449 check.errpos = errpos
450 }(check.iota, check.errpos)
451 check.iota = obj.val
452 check.errpos = nil
453
454
455 obj.val = constant.MakeUnknown()
456
457
458 if typ != nil {
459 t := check.typ(typ)
460 if !isConstType(t) {
461
462
463 if isValid(under(t)) {
464 check.errorf(typ, InvalidConstType, "invalid constant type %s", t)
465 }
466 obj.typ = Typ[Invalid]
467 return
468 }
469 obj.typ = t
470 }
471
472
473 var x operand
474 if init != nil {
475 if inherited {
476
477
478
479
480
481
482 check.errpos = atPos(obj.pos)
483 }
484 check.expr(nil, &x, init)
485 }
486 check.initConst(obj, &x)
487 }
488
489 func (check *Checker) varDecl(obj *Var, lhs []*Var, typ, init ast.Expr) {
490 assert(obj.typ == nil)
491
492
493 if typ != nil {
494 obj.typ = check.varType(typ)
495
496
497
498
499
500
501
502
503 }
504
505
506 if init == nil {
507 if typ == nil {
508
509 obj.typ = Typ[Invalid]
510 }
511 return
512 }
513
514 if lhs == nil || len(lhs) == 1 {
515 assert(lhs == nil || lhs[0] == obj)
516 var x operand
517 check.expr(newTarget(obj.typ, obj.name), &x, init)
518 check.initVar(obj, &x, "variable declaration")
519 return
520 }
521
522 if debug {
523
524 if !slices.Contains(lhs, obj) {
525 panic("inconsistent lhs")
526 }
527 }
528
529
530
531
532
533 if typ != nil {
534 for _, lhs := range lhs {
535 lhs.typ = obj.typ
536 }
537 }
538
539 check.initVars(lhs, []ast.Expr{init}, nil)
540 }
541
542
543 func (check *Checker) isImportedConstraint(typ Type) bool {
544 named := asNamed(typ)
545 if named == nil || named.obj.pkg == check.pkg || named.obj.pkg == nil {
546 return false
547 }
548 u, _ := named.under().(*Interface)
549 return u != nil && !u.IsMethodSet()
550 }
551
552 func (check *Checker) typeDecl(obj *TypeName, tdecl *ast.TypeSpec, def *TypeName) {
553 assert(obj.typ == nil)
554
555
556 versionErr := false
557
558 var rhs Type
559 check.later(func() {
560 if t := asNamed(obj.typ); t != nil {
561 check.validType(t)
562 }
563
564 _ = !versionErr && check.isImportedConstraint(rhs) && check.verifyVersionf(tdecl.Type, go1_18, "using type constraint %s", rhs)
565 }).describef(obj, "validType(%s)", obj.Name())
566
567
568 var tparam0 *ast.Field
569 if tdecl.TypeParams.NumFields() > 0 {
570 tparam0 = tdecl.TypeParams.List[0]
571 }
572
573
574 if tdecl.Assign.IsValid() {
575
576
577 if !versionErr && tparam0 != nil && !check.verifyVersionf(tparam0, go1_23, "generic type alias") {
578 versionErr = true
579 }
580 if !versionErr && !check.verifyVersionf(atPos(tdecl.Assign), go1_9, "type alias") {
581 versionErr = true
582 }
583
584 if check.conf._EnableAlias {
585
586
587
588
589
590
591
592
593 alias := check.newAlias(obj, Typ[Invalid])
594 setDefType(def, alias)
595
596
597 if tparam0 != nil {
598 if !versionErr && !buildcfg.Experiment.AliasTypeParams {
599 check.error(tdecl, UnsupportedFeature, "generic type alias requires GOEXPERIMENT=aliastypeparams")
600 versionErr = true
601 }
602 check.openScope(tdecl, "type parameters")
603 defer check.closeScope()
604 check.collectTypeParams(&alias.tparams, tdecl.TypeParams)
605 }
606
607 rhs = check.definedType(tdecl.Type, obj)
608 assert(rhs != nil)
609 alias.fromRHS = rhs
610 Unalias(alias)
611 } else {
612
613
614
615
616
617 gotypesalias.IncNonDefault()
618
619 if !versionErr && tparam0 != nil {
620 check.error(tdecl, UnsupportedFeature, "generic type alias requires GODEBUG=gotypesalias=1 or unset")
621 versionErr = true
622 }
623
624 check.brokenAlias(obj)
625 rhs = check.typ(tdecl.Type)
626 check.validAlias(obj, rhs)
627 }
628 return
629 }
630
631
632 if !versionErr && tparam0 != nil && !check.verifyVersionf(tparam0, go1_18, "type parameter") {
633 versionErr = true
634 }
635
636 named := check.newNamed(obj, nil, nil)
637 setDefType(def, named)
638
639 if tdecl.TypeParams != nil {
640 check.openScope(tdecl, "type parameters")
641 defer check.closeScope()
642 check.collectTypeParams(&named.tparams, tdecl.TypeParams)
643 }
644
645
646 rhs = check.definedType(tdecl.Type, obj)
647 assert(rhs != nil)
648 named.fromRHS = rhs
649
650
651
652 if named.underlying == nil {
653 named.underlying = Typ[Invalid]
654 }
655
656
657
658
659
660
661 if isTypeParam(rhs) {
662 check.error(tdecl.Type, MisplacedTypeParam, "cannot use a type parameter as RHS in type declaration")
663 named.underlying = Typ[Invalid]
664 }
665 }
666
667 func (check *Checker) collectTypeParams(dst **TypeParamList, list *ast.FieldList) {
668 var tparams []*TypeParam
669
670
671
672 scopePos := list.Pos()
673 for _, f := range list.List {
674 for _, name := range f.Names {
675 tparams = append(tparams, check.declareTypeParam(name, scopePos))
676 }
677 }
678
679
680
681
682 *dst = bindTParams(tparams)
683
684
685
686
687
688
689
690
691 assert(!check.inTParamList)
692 check.inTParamList = true
693 defer func() {
694 check.inTParamList = false
695 }()
696
697 index := 0
698 for _, f := range list.List {
699 var bound Type
700
701
702 if f.Type != nil {
703 bound = check.bound(f.Type)
704 if isTypeParam(bound) {
705
706
707
708
709 check.error(f.Type, MisplacedTypeParam, "cannot use a type parameter as constraint")
710 bound = Typ[Invalid]
711 }
712 } else {
713 bound = Typ[Invalid]
714 }
715 for i := range f.Names {
716 tparams[index+i].bound = bound
717 }
718 index += len(f.Names)
719 }
720 }
721
722 func (check *Checker) bound(x ast.Expr) Type {
723
724
725
726 wrap := false
727 switch op := x.(type) {
728 case *ast.UnaryExpr:
729 wrap = op.Op == token.TILDE
730 case *ast.BinaryExpr:
731 wrap = op.Op == token.OR
732 }
733 if wrap {
734 x = &ast.InterfaceType{Methods: &ast.FieldList{List: []*ast.Field{{Type: x}}}}
735 t := check.typ(x)
736
737 if t, _ := t.(*Interface); t != nil {
738 t.implicit = true
739 }
740 return t
741 }
742 return check.typ(x)
743 }
744
745 func (check *Checker) declareTypeParam(name *ast.Ident, scopePos token.Pos) *TypeParam {
746
747
748
749
750
751
752 tname := NewTypeName(name.Pos(), check.pkg, name.Name, nil)
753 tpar := check.newTypeParam(tname, Typ[Invalid])
754 check.declare(check.scope, name, tname, scopePos)
755 return tpar
756 }
757
758 func (check *Checker) collectMethods(obj *TypeName) {
759
760
761
762
763 methods := check.methods[obj]
764 if methods == nil {
765 return
766 }
767 delete(check.methods, obj)
768 assert(!check.objMap[obj].tdecl.Assign.IsValid())
769
770
771 var mset objset
772
773
774
775 base := asNamed(obj.typ)
776 if base != nil {
777 assert(base.TypeArgs().Len() == 0)
778
779
780
781 check.later(func() {
782 check.checkFieldUniqueness(base)
783 }).describef(obj, "verifying field uniqueness for %v", base)
784
785
786
787
788 for i := 0; i < base.NumMethods(); i++ {
789 m := base.Method(i)
790 assert(m.name != "_")
791 assert(mset.insert(m) == nil)
792 }
793 }
794
795
796 for _, m := range methods {
797
798
799 assert(m.name != "_")
800 if alt := mset.insert(m); alt != nil {
801 if alt.Pos().IsValid() {
802 check.errorf(m, DuplicateMethod, "method %s.%s already declared at %v", obj.Name(), m.name, alt.Pos())
803 } else {
804 check.errorf(m, DuplicateMethod, "method %s.%s already declared", obj.Name(), m.name)
805 }
806 continue
807 }
808
809 if base != nil {
810 base.AddMethod(m)
811 }
812 }
813 }
814
815 func (check *Checker) checkFieldUniqueness(base *Named) {
816 if t, _ := base.under().(*Struct); t != nil {
817 var mset objset
818 for i := 0; i < base.NumMethods(); i++ {
819 m := base.Method(i)
820 assert(m.name != "_")
821 assert(mset.insert(m) == nil)
822 }
823
824
825
826 for _, fld := range t.fields {
827 if fld.name != "_" {
828 if alt := mset.insert(fld); alt != nil {
829
830
831 _ = alt.(*Func)
832
833
834
835 err := check.newError(DuplicateFieldAndMethod)
836 err.addf(alt, "field and method with the same name %s", fld.name)
837 err.addAltDecl(fld)
838 err.report()
839 }
840 }
841 }
842 }
843 }
844
845 func (check *Checker) funcDecl(obj *Func, decl *declInfo) {
846 assert(obj.typ == nil)
847
848
849 assert(check.iota == nil)
850
851 sig := new(Signature)
852 obj.typ = sig
853
854
855
856
857
858
859
860 saved := obj.color_
861 obj.color_ = black
862 fdecl := decl.fdecl
863 check.funcType(sig, fdecl.Recv, fdecl.Type)
864 obj.color_ = saved
865
866
867
868 sig.scope.pos = fdecl.Pos()
869 sig.scope.end = fdecl.End()
870
871 if fdecl.Type.TypeParams.NumFields() > 0 && fdecl.Body == nil {
872 check.softErrorf(fdecl.Name, BadDecl, "generic function is missing function body")
873 }
874
875
876
877 if !check.conf.IgnoreFuncBodies && fdecl.Body != nil {
878 check.later(func() {
879 check.funcBody(decl, obj.name, sig, fdecl.Body, nil)
880 }).describef(obj, "func %s", obj.name)
881 }
882 }
883
884 func (check *Checker) declStmt(d ast.Decl) {
885 pkg := check.pkg
886
887 check.walkDecl(d, func(d decl) {
888 switch d := d.(type) {
889 case constDecl:
890 top := len(check.delayed)
891
892
893 lhs := make([]*Const, len(d.spec.Names))
894 for i, name := range d.spec.Names {
895 obj := NewConst(name.Pos(), pkg, name.Name, nil, constant.MakeInt64(int64(d.iota)))
896 lhs[i] = obj
897
898 var init ast.Expr
899 if i < len(d.init) {
900 init = d.init[i]
901 }
902
903 check.constDecl(obj, d.typ, init, d.inherited)
904 }
905
906
907 check.processDelayed(top)
908
909
910
911
912
913 scopePos := d.spec.End()
914 for i, name := range d.spec.Names {
915 check.declare(check.scope, name, lhs[i], scopePos)
916 }
917
918 case varDecl:
919 top := len(check.delayed)
920
921 lhs0 := make([]*Var, len(d.spec.Names))
922 for i, name := range d.spec.Names {
923 lhs0[i] = newVar(LocalVar, name.Pos(), pkg, name.Name, nil)
924 }
925
926
927 for i, obj := range lhs0 {
928 var lhs []*Var
929 var init ast.Expr
930 switch len(d.spec.Values) {
931 case len(d.spec.Names):
932
933 init = d.spec.Values[i]
934 case 1:
935
936 lhs = lhs0
937 init = d.spec.Values[0]
938 default:
939 if i < len(d.spec.Values) {
940 init = d.spec.Values[i]
941 }
942 }
943 check.varDecl(obj, lhs, d.spec.Type, init)
944 if len(d.spec.Values) == 1 {
945
946
947
948
949
950 if debug {
951 for _, obj := range lhs0 {
952 assert(obj.typ != nil)
953 }
954 }
955 break
956 }
957 }
958
959
960 check.processDelayed(top)
961
962
963
964 scopePos := d.spec.End()
965 for i, name := range d.spec.Names {
966
967 check.declare(check.scope, name, lhs0[i], scopePos)
968 }
969
970 case typeDecl:
971 obj := NewTypeName(d.spec.Name.Pos(), pkg, d.spec.Name.Name, nil)
972
973
974
975 scopePos := d.spec.Name.Pos()
976 check.declare(check.scope, d.spec.Name, obj, scopePos)
977
978 obj.setColor(grey + color(check.push(obj)))
979 check.typeDecl(obj, d.spec, nil)
980 check.pop().setColor(black)
981 default:
982 check.errorf(d.node(), InvalidSyntaxTree, "unknown ast.Decl node %T", d.node())
983 }
984 })
985 }
986
View as plain text