Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

go/types: panic in allowVersion if token.Pos is out of bounds #69477

Closed
adonovan opened this issue Sep 15, 2024 · 2 comments
Closed

go/types: panic in allowVersion if token.Pos is out of bounds #69477

adonovan opened this issue Sep 15, 2024 · 2 comments
Assignees
Labels
NeedsFix The path to resolution is known, but the work has not been done.

Comments

@adonovan
Copy link
Member

adonovan commented Sep 15, 2024

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

	// 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 {
		if lit, ok := n.(*ast.BasicLit); ok {
			lit.ValuePos = 99999
		}
		return true
	})

	// Type check
	pkg := types.NewPackage("p", "p")
	check := types.NewChecker(&types.Config{}, fset, pkg, nil)
	if err := check.Files([]*ast.File{f}); err != nil { // panic: file not found for pos = 99999
		log.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.

@gopherbot
Copy link
Contributor

Change https://go.dev/cl/613735 mentions this issue: go/types: compute effective Go version independent of token.Pos

mateusz834 pushed a commit to mateusz834/tgoast that referenced this issue Mar 14, 2025
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

Fixes golang/go#69477
Fixes golang/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]>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
NeedsFix The path to resolution is known, but the work has not been done.
Projects
None yet
Development

No branches or pull requests

4 participants