Source file src/go/types/check.go
1 // Copyright 2011 The Go Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style 3 // license that can be found in the LICENSE file. 4 5 // This file implements the Check function, which drives type-checking. 6 7 package types 8 9 import ( 10 "fmt" 11 "go/ast" 12 "go/constant" 13 "go/token" 14 "internal/godebug" 15 . "internal/types/errors" 16 "sync/atomic" 17 ) 18 19 // nopos, noposn indicate an unknown position 20 var nopos token.Pos 21 var noposn = atPos(nopos) 22 23 // debugging/development support 24 const debug = false // leave on during development 25 26 // gotypesalias controls the use of Alias types. 27 // As of Apr 16 2024 they are used by default. 28 // To disable their use, set GODEBUG to gotypesalias=0. 29 // This GODEBUG flag will be removed in the near future (tentatively Go 1.24). 30 var gotypesalias = godebug.New("gotypesalias") 31 32 // _aliasAny changes the behavior of [Scope.Lookup] for "any" in the 33 // [Universe] scope. 34 // 35 // This is necessary because while Alias creation is controlled by 36 // [Config._EnableAlias], based on the gotypealias variable, the representation 37 // of "any" is a global. In [Scope.Lookup], we select this global 38 // representation based on the result of [aliasAny], but as a result need to 39 // guard against this behavior changing during the type checking pass. 40 // Therefore we implement the following rule: any number of goroutines can type 41 // check concurrently with the same EnableAlias value, but if any goroutine 42 // tries to type check concurrently with a different EnableAlias value, we 43 // panic. 44 // 45 // To achieve this, _aliasAny is a state machine: 46 // 47 // 0: no type checking is occurring 48 // negative: type checking is occurring without _EnableAlias set 49 // positive: type checking is occurring with _EnableAlias set 50 var _aliasAny int32 51 52 func aliasAny() bool { 53 v := gotypesalias.Value() 54 useAlias := v != "0" 55 inuse := atomic.LoadInt32(&_aliasAny) 56 if inuse != 0 && useAlias != (inuse > 0) { 57 panic(fmt.Sprintf("gotypealias mutated during type checking, gotypesalias=%s, inuse=%d", v, inuse)) 58 } 59 return useAlias 60 } 61 62 // exprInfo stores information about an untyped expression. 63 type exprInfo struct { 64 isLhs bool // expression is lhs operand of a shift with delayed type-check 65 mode operandMode 66 typ *Basic 67 val constant.Value // constant value; or nil (if not a constant) 68 } 69 70 // An environment represents the environment within which an object is 71 // type-checked. 72 type environment struct { 73 decl *declInfo // package-level declaration whose init expression/function body is checked 74 scope *Scope // top-most scope for lookups 75 version goVersion // current accepted language version; changes across files 76 iota constant.Value // value of iota in a constant declaration; nil otherwise 77 errpos positioner // if set, identifier position of a constant with inherited initializer 78 inTParamList bool // set if inside a type parameter list 79 sig *Signature // function signature if inside a function; nil otherwise 80 isPanic map[*ast.CallExpr]bool // set of panic call expressions (used for termination check) 81 hasLabel bool // set if a function makes use of labels (only ~1% of functions); unused outside functions 82 hasCallOrRecv bool // set if an expression contains a function call or channel receive operation 83 84 // go/types only 85 exprPos token.Pos // if valid, identifiers are looked up as if at position pos (used by CheckExpr, Eval) 86 } 87 88 // lookupScope looks up name in the current environment and if an object 89 // is found it returns the scope containing the object and the object. 90 // Otherwise it returns (nil, nil). 91 // 92 // Note that obj.Parent() may be different from the returned scope if the 93 // object was inserted into the scope and already had a parent at that 94 // time (see Scope.Insert). This can only happen for dot-imported objects 95 // whose parent is the scope of the package that exported them. 96 func (env *environment) lookupScope(name string) (*Scope, Object) { 97 for s := env.scope; s != nil; s = s.parent { 98 if obj := s.Lookup(name); obj != nil && (!env.exprPos.IsValid() || cmpPos(obj.scopePos(), env.exprPos) <= 0) { 99 return s, obj 100 } 101 } 102 return nil, nil 103 } 104 105 // lookup is like lookupScope but it only returns the object (or nil). 106 func (env *environment) lookup(name string) Object { 107 _, obj := env.lookupScope(name) 108 return obj 109 } 110 111 // An importKey identifies an imported package by import path and source directory 112 // (directory containing the file containing the import). In practice, the directory 113 // may always be the same, or may not matter. Given an (import path, directory), an 114 // importer must always return the same package (but given two different import paths, 115 // an importer may still return the same package by mapping them to the same package 116 // paths). 117 type importKey struct { 118 path, dir string 119 } 120 121 // A dotImportKey describes a dot-imported object in the given scope. 122 type dotImportKey struct { 123 scope *Scope 124 name string 125 } 126 127 // An action describes a (delayed) action. 128 type action struct { 129 version goVersion // applicable language version 130 f func() // action to be executed 131 desc *actionDesc // action description; may be nil, requires debug to be set 132 } 133 134 // If debug is set, describef sets a printf-formatted description for action a. 135 // Otherwise, it is a no-op. 136 func (a *action) describef(pos positioner, format string, args ...any) { 137 if debug { 138 a.desc = &actionDesc{pos, format, args} 139 } 140 } 141 142 // An actionDesc provides information on an action. 143 // For debugging only. 144 type actionDesc struct { 145 pos positioner 146 format string 147 args []any 148 } 149 150 // A Checker maintains the state of the type checker. 151 // It must be created with [NewChecker]. 152 type Checker struct { 153 // package information 154 // (initialized by NewChecker, valid for the life-time of checker) 155 conf *Config 156 ctxt *Context // context for de-duplicating instances 157 fset *token.FileSet 158 pkg *Package 159 *Info 160 nextID uint64 // unique Id for type parameters (first valid Id is 1) 161 objMap map[Object]*declInfo // maps package-level objects and (non-interface) methods to declaration info 162 impMap map[importKey]*Package // maps (import path, source directory) to (complete or fake) package 163 // see TODO in validtype.go 164 // valids instanceLookup // valid *Named (incl. instantiated) types per the validType check 165 166 // pkgPathMap maps package names to the set of distinct import paths we've 167 // seen for that name, anywhere in the import graph. It is used for 168 // disambiguating package names in error messages. 169 // 170 // pkgPathMap is allocated lazily, so that we don't pay the price of building 171 // it on the happy path. seenPkgMap tracks the packages that we've already 172 // walked. 173 pkgPathMap map[string]map[string]bool 174 seenPkgMap map[*Package]bool 175 176 // information collected during type-checking of a set of package files 177 // (initialized by Files, valid only for the duration of check.Files; 178 // maps and lists are allocated on demand) 179 files []*ast.File // package files 180 versions map[*ast.File]string // maps files to goVersion strings (each file has an entry); shared with Info.FileVersions if present; may be unaltered Config.GoVersion 181 imports []*PkgName // list of imported packages 182 dotImportMap map[dotImportKey]*PkgName // maps dot-imported objects to the package they were dot-imported through 183 brokenAliases map[*TypeName]bool // set of aliases with broken (not yet determined) types 184 unionTypeSets map[*Union]*_TypeSet // computed type sets for union types 185 usedVars map[*Var]bool // set of used variables 186 usedPkgNames map[*PkgName]bool // set of used package names 187 mono monoGraph // graph for detecting non-monomorphizable instantiation loops 188 189 firstErr error // first error encountered 190 methods map[*TypeName][]*Func // maps package scope type names to associated non-blank (non-interface) methods 191 untyped map[ast.Expr]exprInfo // map of expressions without final type 192 delayed []action // stack of delayed action segments; segments are processed in FIFO order 193 objPath []Object // path of object dependencies during type inference (for cycle reporting) 194 cleaners []cleaner // list of types that may need a final cleanup at the end of type-checking 195 196 // environment within which the current object is type-checked (valid only 197 // for the duration of type-checking a specific object) 198 environment 199 200 // debugging 201 indent int // indentation for tracing 202 } 203 204 // addDeclDep adds the dependency edge (check.decl -> to) if check.decl exists 205 func (check *Checker) addDeclDep(to Object) { 206 from := check.decl 207 if from == nil { 208 return // not in a package-level init expression 209 } 210 if _, found := check.objMap[to]; !found { 211 return // to is not a package-level object 212 } 213 from.addDep(to) 214 } 215 216 // Note: The following three alias-related functions are only used 217 // when Alias types are not enabled. 218 219 // brokenAlias records that alias doesn't have a determined type yet. 220 // It also sets alias.typ to Typ[Invalid]. 221 // Not used if check.conf._EnableAlias is set. 222 func (check *Checker) brokenAlias(alias *TypeName) { 223 assert(!check.conf._EnableAlias) 224 if check.brokenAliases == nil { 225 check.brokenAliases = make(map[*TypeName]bool) 226 } 227 check.brokenAliases[alias] = true 228 alias.typ = Typ[Invalid] 229 } 230 231 // validAlias records that alias has the valid type typ (possibly Typ[Invalid]). 232 func (check *Checker) validAlias(alias *TypeName, typ Type) { 233 assert(!check.conf._EnableAlias) 234 delete(check.brokenAliases, alias) 235 alias.typ = typ 236 } 237 238 // isBrokenAlias reports whether alias doesn't have a determined type yet. 239 func (check *Checker) isBrokenAlias(alias *TypeName) bool { 240 assert(!check.conf._EnableAlias) 241 return check.brokenAliases[alias] 242 } 243 244 func (check *Checker) rememberUntyped(e ast.Expr, lhs bool, mode operandMode, typ *Basic, val constant.Value) { 245 m := check.untyped 246 if m == nil { 247 m = make(map[ast.Expr]exprInfo) 248 check.untyped = m 249 } 250 m[e] = exprInfo{lhs, mode, typ, val} 251 } 252 253 // later pushes f on to the stack of actions that will be processed later; 254 // either at the end of the current statement, or in case of a local constant 255 // or variable declaration, before the constant or variable is in scope 256 // (so that f still sees the scope before any new declarations). 257 // later returns the pushed action so one can provide a description 258 // via action.describef for debugging, if desired. 259 func (check *Checker) later(f func()) *action { 260 i := len(check.delayed) 261 check.delayed = append(check.delayed, action{version: check.version, f: f}) 262 return &check.delayed[i] 263 } 264 265 // push pushes obj onto the object path and returns its index in the path. 266 func (check *Checker) push(obj Object) int { 267 check.objPath = append(check.objPath, obj) 268 return len(check.objPath) - 1 269 } 270 271 // pop pops and returns the topmost object from the object path. 272 func (check *Checker) pop() Object { 273 i := len(check.objPath) - 1 274 obj := check.objPath[i] 275 check.objPath[i] = nil 276 check.objPath = check.objPath[:i] 277 return obj 278 } 279 280 type cleaner interface { 281 cleanup() 282 } 283 284 // needsCleanup records objects/types that implement the cleanup method 285 // which will be called at the end of type-checking. 286 func (check *Checker) needsCleanup(c cleaner) { 287 check.cleaners = append(check.cleaners, c) 288 } 289 290 // NewChecker returns a new [Checker] instance for a given package. 291 // [Package] files may be added incrementally via checker.Files. 292 func NewChecker(conf *Config, fset *token.FileSet, pkg *Package, info *Info) *Checker { 293 // make sure we have a configuration 294 if conf == nil { 295 conf = new(Config) 296 } 297 298 // make sure we have an info struct 299 if info == nil { 300 info = new(Info) 301 } 302 303 // Note: clients may call NewChecker with the Unsafe package, which is 304 // globally shared and must not be mutated. Therefore NewChecker must not 305 // mutate *pkg. 306 // 307 // (previously, pkg.goVersion was mutated here: go.dev/issue/61212) 308 309 // In go/types, conf._EnableAlias is controlled by gotypesalias. 310 conf._EnableAlias = gotypesalias.Value() != "0" 311 312 return &Checker{ 313 conf: conf, 314 ctxt: conf.Context, 315 fset: fset, 316 pkg: pkg, 317 Info: info, 318 objMap: make(map[Object]*declInfo), 319 impMap: make(map[importKey]*Package), 320 usedVars: make(map[*Var]bool), 321 usedPkgNames: make(map[*PkgName]bool), 322 } 323 } 324 325 // initFiles initializes the files-specific portion of checker. 326 // The provided files must all belong to the same package. 327 func (check *Checker) initFiles(files []*ast.File) { 328 // start with a clean slate (check.Files may be called multiple times) 329 // TODO(gri): what determines which fields are zeroed out here, vs at the end 330 // of checkFiles? 331 check.files = nil 332 check.imports = nil 333 check.dotImportMap = nil 334 335 check.firstErr = nil 336 check.methods = nil 337 check.untyped = nil 338 check.delayed = nil 339 check.objPath = nil 340 check.cleaners = nil 341 342 // We must initialize usedVars and usedPkgNames both here and in NewChecker, 343 // because initFiles is not called in the CheckExpr or Eval codepaths, yet we 344 // want to free this memory at the end of Files ('used' predicates are 345 // only needed in the context of a given file). 346 check.usedVars = make(map[*Var]bool) 347 check.usedPkgNames = make(map[*PkgName]bool) 348 349 // determine package name and collect valid files 350 pkg := check.pkg 351 for _, file := range files { 352 switch name := file.Name.Name; pkg.name { 353 case "": 354 if name != "_" { 355 pkg.name = name 356 } else { 357 check.error(file.Name, BlankPkgName, "invalid package name _") 358 } 359 fallthrough 360 361 case name: 362 check.files = append(check.files, file) 363 364 default: 365 check.errorf(atPos(file.Package), MismatchedPkgName, "package %s; expected package %s", name, pkg.name) 366 // ignore this file 367 } 368 } 369 370 // reuse Info.FileVersions if provided 371 versions := check.Info.FileVersions 372 if versions == nil { 373 versions = make(map[*ast.File]string) 374 } 375 check.versions = versions 376 377 pkgVersion := asGoVersion(check.conf.GoVersion) 378 if pkgVersion.isValid() && len(files) > 0 && pkgVersion.cmp(go_current) > 0 { 379 check.errorf(files[0], TooNew, "package requires newer Go version %v (application built with %v)", 380 pkgVersion, go_current) 381 } 382 383 // determine Go version for each file 384 for _, file := range check.files { 385 // use unaltered Config.GoVersion by default 386 // (This version string may contain dot-release numbers as in go1.20.1, 387 // unlike file versions which are Go language versions only, if valid.) 388 v := check.conf.GoVersion 389 390 // If the file specifies a version, use max(fileVersion, go1.21). 391 if fileVersion := asGoVersion(file.GoVersion); fileVersion.isValid() { 392 // Go 1.21 introduced the feature of setting the go.mod 393 // go line to an early version of Go and allowing //go:build lines 394 // to set the Go version in a given file. Versions Go 1.21 and later 395 // can be set backwards compatibly as that was the first version 396 // files with go1.21 or later build tags could be built with. 397 // 398 // Set the version to max(fileVersion, go1.21): That will allow a 399 // downgrade to a version before go1.22, where the for loop semantics 400 // change was made, while being backwards compatible with versions of 401 // go before the new //go:build semantics were introduced. 402 v = string(versionMax(fileVersion, go1_21)) 403 404 // Report a specific error for each tagged file that's too new. 405 // (Normally the build system will have filtered files by version, 406 // but clients can present arbitrary files to the type checker.) 407 if fileVersion.cmp(go_current) > 0 { 408 // Use position of 'package [p]' for types/types2 consistency. 409 // (Ideally we would use the //build tag itself.) 410 check.errorf(file.Name, TooNew, "file requires newer Go version %v (application built with %v)", fileVersion, go_current) 411 } 412 } 413 versions[file] = v 414 } 415 } 416 417 func versionMax(a, b goVersion) goVersion { 418 if a.cmp(b) < 0 { 419 return b 420 } 421 return a 422 } 423 424 // A bailout panic is used for early termination. 425 type bailout struct{} 426 427 func (check *Checker) handleBailout(err *error) { 428 switch p := recover().(type) { 429 case nil, bailout: 430 // normal return or early exit 431 *err = check.firstErr 432 default: 433 // re-panic 434 panic(p) 435 } 436 } 437 438 // Files checks the provided files as part of the checker's package. 439 func (check *Checker) Files(files []*ast.File) (err error) { 440 if check.pkg == Unsafe { 441 // Defensive handling for Unsafe, which cannot be type checked, and must 442 // not be mutated. See https://go.dev/issue/61212 for an example of where 443 // Unsafe is passed to NewChecker. 444 return nil 445 } 446 447 // Avoid early returns here! Nearly all errors can be 448 // localized to a piece of syntax and needn't prevent 449 // type-checking of the rest of the package. 450 451 defer check.handleBailout(&err) 452 check.checkFiles(files) 453 return 454 } 455 456 // checkFiles type-checks the specified files. Errors are reported as 457 // a side effect, not by returning early, to ensure that well-formed 458 // syntax is properly type annotated even in a package containing 459 // errors. 460 func (check *Checker) checkFiles(files []*ast.File) { 461 // Ensure that _EnableAlias is consistent among concurrent type checking 462 // operations. See the documentation of [_aliasAny] for details. 463 if check.conf._EnableAlias { 464 if atomic.AddInt32(&_aliasAny, 1) <= 0 { 465 panic("EnableAlias set while !EnableAlias type checking is ongoing") 466 } 467 defer atomic.AddInt32(&_aliasAny, -1) 468 } else { 469 if atomic.AddInt32(&_aliasAny, -1) >= 0 { 470 panic("!EnableAlias set while EnableAlias type checking is ongoing") 471 } 472 defer atomic.AddInt32(&_aliasAny, 1) 473 } 474 475 print := func(msg string) { 476 if check.conf._Trace { 477 fmt.Println() 478 fmt.Println(msg) 479 } 480 } 481 482 print("== initFiles ==") 483 check.initFiles(files) 484 485 print("== collectObjects ==") 486 check.collectObjects() 487 488 print("== packageObjects ==") 489 check.packageObjects() 490 491 print("== processDelayed ==") 492 check.processDelayed(0) // incl. all functions 493 494 print("== cleanup ==") 495 check.cleanup() 496 497 print("== initOrder ==") 498 check.initOrder() 499 500 if !check.conf.DisableUnusedImportCheck { 501 print("== unusedImports ==") 502 check.unusedImports() 503 } 504 505 print("== recordUntyped ==") 506 check.recordUntyped() 507 508 if check.firstErr == nil { 509 // TODO(mdempsky): Ensure monomorph is safe when errors exist. 510 check.monomorph() 511 } 512 513 check.pkg.goVersion = check.conf.GoVersion 514 check.pkg.complete = true 515 516 // no longer needed - release memory 517 check.imports = nil 518 check.dotImportMap = nil 519 check.pkgPathMap = nil 520 check.seenPkgMap = nil 521 check.brokenAliases = nil 522 check.unionTypeSets = nil 523 check.usedVars = nil 524 check.usedPkgNames = nil 525 check.ctxt = nil 526 527 // TODO(gri): shouldn't the cleanup above occur after the bailout? 528 // TODO(gri) There's more memory we should release at this point. 529 } 530 531 // processDelayed processes all delayed actions pushed after top. 532 func (check *Checker) processDelayed(top int) { 533 // If each delayed action pushes a new action, the 534 // stack will continue to grow during this loop. 535 // However, it is only processing functions (which 536 // are processed in a delayed fashion) that may 537 // add more actions (such as nested functions), so 538 // this is a sufficiently bounded process. 539 savedVersion := check.version 540 for i := top; i < len(check.delayed); i++ { 541 a := &check.delayed[i] 542 if check.conf._Trace { 543 if a.desc != nil { 544 check.trace(a.desc.pos.Pos(), "-- "+a.desc.format, a.desc.args...) 545 } else { 546 check.trace(nopos, "-- delayed %p", a.f) 547 } 548 } 549 check.version = a.version // reestablish the effective Go version captured earlier 550 a.f() // may append to check.delayed 551 552 if check.conf._Trace { 553 fmt.Println() 554 } 555 } 556 assert(top <= len(check.delayed)) // stack must not have shrunk 557 check.delayed = check.delayed[:top] 558 check.version = savedVersion 559 } 560 561 // cleanup runs cleanup for all collected cleaners. 562 func (check *Checker) cleanup() { 563 // Don't use a range clause since Named.cleanup may add more cleaners. 564 for i := 0; i < len(check.cleaners); i++ { 565 check.cleaners[i].cleanup() 566 } 567 check.cleaners = nil 568 } 569 570 // go/types doesn't support recording of types directly in the AST. 571 // dummy function to match types2 code. 572 func (check *Checker) recordTypeAndValueInSyntax(x ast.Expr, mode operandMode, typ Type, val constant.Value) { 573 // nothing to do 574 } 575 576 // go/types doesn't support recording of types directly in the AST. 577 // dummy function to match types2 code. 578 func (check *Checker) recordCommaOkTypesInSyntax(x ast.Expr, t0, t1 Type) { 579 // nothing to do 580 } 581 582 // instantiatedIdent determines the identifier of the type instantiated in expr. 583 // Helper function for recordInstance in recording.go. 584 func instantiatedIdent(expr ast.Expr) *ast.Ident { 585 var selOrIdent ast.Expr 586 switch e := expr.(type) { 587 case *ast.IndexExpr: 588 selOrIdent = e.X 589 case *ast.IndexListExpr: // only exists in go/ast, not syntax 590 selOrIdent = e.X 591 case *ast.SelectorExpr, *ast.Ident: 592 selOrIdent = e 593 } 594 switch x := selOrIdent.(type) { 595 case *ast.Ident: 596 return x 597 case *ast.SelectorExpr: 598 return x.Sel 599 } 600 601 // extra debugging of go.dev/issue/63933 602 panic(sprintf(nil, nil, true, "instantiated ident not found; please report: %s", expr)) 603 } 604