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