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