Skip to content

Commit 851c747

Browse files
committed
gopls/internal/golang: fix crash when hovering over implicit
After months of intermittent investigation, I was finally able to reproduce the telemetry crash of golang/go#69362: if the operant of the type switch is undefined, the selectedType will be nil, and therefore logic may proceed to the point where it reaches a nil entry in types.Info.Defs. The fix is of course straightforward, now that we understand it: we cannot rely on types.Info.Defs not containing nil entries. A follow-up CL will introduce an analyzer to detect such problematic uses of the go/types API. Fixes golang/go#69362 Change-Id: I8f75c24710dbb2e78c79d8b9d721f45d9a040cd7 Reviewed-on: https://go-review.googlesource.com/c/tools/+/652015 Reviewed-by: Alan Donovan <[email protected]> LUCI-TryBot-Result: Go LUCI <[email protected]>
1 parent 3e76cae commit 851c747

File tree

2 files changed

+16
-9
lines changed

2 files changed

+16
-9
lines changed

gopls/internal/golang/hover.go

+4-9
Original file line numberDiff line numberDiff line change
@@ -375,15 +375,10 @@ func hover(ctx context.Context, snapshot *cache.Snapshot, fh file.Handle, pp pro
375375
// use the default build config for all other types, even
376376
// if they embed platform-variant types.
377377
//
378-
var sizeOffset string // optional size/offset description
379-
// debugging #69362: unexpected nil Defs[ident] value (?)
380-
_ = ident.Pos() // (can't be nil due to check after referencedObject)
381-
_ = pkg.TypesInfo() // (can't be nil due to check in call to inferredSignature)
382-
_ = pkg.TypesInfo().Defs // (can't be nil due to nature of cache.Package)
383-
if def, ok := pkg.TypesInfo().Defs[ident]; ok {
384-
_ = def.Pos() // can't be nil due to reasoning in #69362.
385-
}
386-
if def, ok := pkg.TypesInfo().Defs[ident]; ok && ident.Pos() == def.Pos() {
378+
var sizeOffset string
379+
380+
// As painfully learned in golang/go#69362, Defs can contain nil entries.
381+
if def, _ := pkg.TypesInfo().Defs[ident]; def != nil && ident.Pos() == def.Pos() {
387382
// This is the declaring identifier.
388383
// (We can't simply use ident.Pos() == obj.Pos() because
389384
// referencedObject prefers the TypeName for an embedded field).

gopls/internal/test/marker/testdata/hover/issues.txt

+12
Original file line numberDiff line numberDiff line change
@@ -20,3 +20,15 @@ package issue64237
2020
import "golang.org/lsptests/nonexistant" //@diag("\"golang", re"could not import")
2121

2222
var _ = nonexistant.Value //@hovererr("nonexistant", "no package data")
23+
24+
-- issue69362/p.go --
25+
package issue69362
26+
27+
// golang/go#69362: hover panics over undefined implicits.
28+
29+
func _() {
30+
switch x := y.(type) { //@diag("y", re"undefined"), hover("x", "x", "")
31+
case int:
32+
_ = x
33+
}
34+
}

0 commit comments

Comments
 (0)