You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
The type checker performs various checks on the Go language level applicable to certain constructs, such as whether underscores are permitted in integer literals, which requires go1.13+. The logic to extract the file version is fragile, and panics if the position information is out of bounds, as is common when type checking synthesized or modified code.
// Parse the file.fset:=token.NewFileSet()
f, _:=parser.ParseFile(fset, "a.go", "package p; const k = 123", 0)
format.Node(os.Stderr, fset, f)
// Set an invalid Pos on the BasicLit.ast.Inspect(f, func(n ast.Node) bool {
iflit, ok:=n.(*ast.BasicLit); ok {
lit.ValuePos=99999
}
returntrue
})
// Type checkpkg:=types.NewPackage("p", "p")
check:=types.NewChecker(&types.Config{}, fset, pkg, nil)
iferr:=check.Files([]*ast.File{f}); err!=nil { // panic: file not found for pos = 99999log.Fatal(err)
}
The type checker should not panic due to missing or inaccurate position information. (Ideally it wouldn't use such information for type checking judgements, only for symbol positions and error messages, but that ship has sailed.)
A more robust solution would be to propagate the file version information top-down when traversing over the syntax trees, instead of trying to figure out which file a given node belongs to based on its Pos.
This is the root cause of the gopls crashes #69338, and I have encountered it in other refactoring work.
The text was updated successfully, but these errors were encountered:
Previously, the Checker.allowVersion method would use a token.Pos
to try to infer which file of the current package the checker
was "in". This proved fragile when type-checking syntax that
had been modified or synthesized and whose positions were invalid.
This change records the effective version in the checker state
(checker.environment.version). Just like other aspects of the
environment, the version changes from one file to the next
and must be saved and restored with each check.later closure.
Similarly, declInfo captures and temporarily reinstates
the effective version when checking each object.
+ Test of position independence in go/types and types2
+ Test of panic avoidance in go/types
Fixesgolang/go#69477Fixesgolang/go#69338
Change-Id: Ic06f9d88151c64a4f7848f8942d08e3c312cdd6f
Reviewed-on: https://go-review.googlesource.com/c/go/+/613735
Reviewed-by: Robert Griesemer <[email protected]>
Auto-Submit: Alan Donovan <[email protected]>
LUCI-TryBot-Result: Go LUCI <[email protected]>
The type checker performs various checks on the Go language level applicable to certain constructs, such as whether underscores are permitted in integer literals, which requires go1.13+. The logic to extract the file version is fragile, and panics if the position information is out of bounds, as is common when type checking synthesized or modified code.
Here's a reproducible test case:
https://go.dev/play/p/1GF5wfNIBYw
The type checker should not panic due to missing or inaccurate position information. (Ideally it wouldn't use such information for type checking judgements, only for symbol positions and error messages, but that ship has sailed.)
A more robust solution would be to propagate the file version information top-down when traversing over the syntax trees, instead of trying to figure out which file a given node belongs to based on its Pos.
This is the root cause of the gopls crashes #69338, and I have encountered it in other refactoring work.
The text was updated successfully, but these errors were encountered: