From 4ad8bcf33ebaba4e6c2d04dadc10cd83c1f8a828 Mon Sep 17 00:00:00 2001 From: kyoh86 Date: Wed, 16 Nov 2022 23:34:03 +0900 Subject: [PATCH 1/2] reproduce #13 --- exportloopref_test.go | 5 +++++ testdata/src/reref/another_file.go | 4 ++++ testdata/src/reref/issue13.go | 13 +++++++++++++ 3 files changed, 22 insertions(+) create mode 100644 testdata/src/reref/another_file.go create mode 100644 testdata/src/reref/issue13.go diff --git a/exportloopref_test.go b/exportloopref_test.go index 444a80c..6633ffb 100644 --- a/exportloopref_test.go +++ b/exportloopref_test.go @@ -46,3 +46,8 @@ func TestDepPointer(t *testing.T) { testdata := analysistest.TestData() analysistest.Run(t, testdata, exportloopref.Analyzer, "deeppointer") } + +func TestReRef(t *testing.T) { + testdata := analysistest.TestData() + analysistest.Run(t, testdata, exportloopref.Analyzer, "reref") +} diff --git a/testdata/src/reref/another_file.go b/testdata/src/reref/another_file.go new file mode 100644 index 0000000..adac634 --- /dev/null +++ b/testdata/src/reref/another_file.go @@ -0,0 +1,4 @@ +package main + +// Moving this map to main.go fixes nil pointer deference +var globalMapInDifferentFile = map[int]*MyStruct{} diff --git a/testdata/src/reref/issue13.go b/testdata/src/reref/issue13.go new file mode 100644 index 0000000..ce9654b --- /dev/null +++ b/testdata/src/reref/issue13.go @@ -0,0 +1,13 @@ +package main + +type MyStruct struct { + MyStructPtrField *int +} + +func main() { + localVal := 0 + arr := []MyStruct{{&localVal}} + for _, p := range arr { + t := *p.MyStructPtrField + globalMapInDifferentFile[t] = &p // want "exporting a pointer for the loop variable p" + } From 19770962ce2d15c1c251d76e5ff725b505aac859 Mon Sep 17 00:00:00 2001 From: kyoh86 Date: Wed, 16 Nov 2022 23:41:14 +0900 Subject: [PATCH 2/2] fix #13: don't follow the postion of the global variable: it clearly is not a local variable. --- exportloopref.go | 3 +++ 1 file changed, 3 insertions(+) diff --git a/exportloopref.go b/exportloopref.go index b432512..d071d5c 100644 --- a/exportloopref.go +++ b/exportloopref.go @@ -298,6 +298,9 @@ func (s *Searcher) isVar(loop ast.Node, expr ast.Expr) bool { } switch typed := expr.(type) { case (*ast.Ident): + if typed.Obj == nil { + return false // global var in another file (ref: #13) + } _, isVar := vars[typed.Obj.Pos()] return isVar case (*ast.IndexExpr): // like X[Y], check X