From 17ffab02aff3193dedf544c0916ef69cae224455 Mon Sep 17 00:00:00 2001 From: noahdietz Date: Mon, 23 Sep 2024 10:46:48 -0700 Subject: [PATCH 1/3] fix(locations): make source info access concurrent safe --- locations/locations.go | 45 ++++++++++++++++++++++++++++++++---------- 1 file changed, 35 insertions(+), 10 deletions(-) diff --git a/locations/locations.go b/locations/locations.go index 27e5dd965..56ee80576 100644 --- a/locations/locations.go +++ b/locations/locations.go @@ -23,6 +23,8 @@ package locations import ( + "sync" + "github.com/jhump/protoreflect/desc" dpb "google.golang.org/protobuf/types/descriptorpb" ) @@ -37,12 +39,24 @@ func pathLocation(d desc.Descriptor, path ...int) *dpb.SourceCodeInfo_Location { return sourceInfoRegistry.sourceInfo(d.GetFile()).findLocation(fullPath) } -type sourceInfo map[string]*dpb.SourceCodeInfo_Location +type sourceInfo struct { + info map[string]*dpb.SourceCodeInfo_Location + lock sync.Mutex +} + +func newSourceInfo() *sourceInfo { + return &sourceInfo{ + info: map[string]*dpb.SourceCodeInfo_Location{}, + } +} // findLocation returns the Location for a given path. -func (si sourceInfo) findLocation(path []int32) *dpb.SourceCodeInfo_Location { +func (si *sourceInfo) findLocation(path []int32) *dpb.SourceCodeInfo_Location { + si.lock.Lock() + defer si.lock.Unlock() + // If the path exists in the source info registry, return that object. - if loc, ok := si[strPath(path)]; ok { + if loc, ok := si.info[strPath(path)]; ok { return loc } @@ -53,7 +67,16 @@ func (si sourceInfo) findLocation(path []int32) *dpb.SourceCodeInfo_Location { // The source map registry is a singleton that computes a source map for // any file descriptor that it is given, but then caches it to avoid computing // the source map for the same file descriptors over and over. -type sourceInfoRegistryType map[*desc.FileDescriptor]sourceInfo +type sourceInfoRegistryType struct { + registry map[*desc.FileDescriptor]*sourceInfo + lock sync.Mutex +} + +func newSourceInfoRegistryType() *sourceInfoRegistryType { + return &sourceInfoRegistryType{ + registry: map[*desc.FileDescriptor]*sourceInfo{}, + } +} // Each location has a path defined as an []int32, but we can not // use slices as keys, so compile them into a string. @@ -70,22 +93,24 @@ func strPath(segments []int32) (p string) { // sourceInfo compiles the source info object for a given file descriptor. // It also caches this into a registry, so subsequent calls using the same // descriptor will return the same object. -func (sir sourceInfoRegistryType) sourceInfo(fd *desc.FileDescriptor) sourceInfo { - answer, ok := sir[fd] +func (sir *sourceInfoRegistryType) sourceInfo(fd *desc.FileDescriptor) *sourceInfo { + sir.lock.Lock() + defer sir.lock.Unlock() + answer, ok := sir.registry[fd] if !ok { - answer = sourceInfo{} + answer = newSourceInfo() // This file descriptor does not yet have a source info map. // Compile one. for _, loc := range fd.AsFileDescriptorProto().GetSourceCodeInfo().GetLocation() { - answer[strPath(loc.Path)] = loc + answer.info[strPath(loc.Path)] = loc } // Now that we calculated all of this, cache it on the registry so it // does not need to be calculated again. - sir[fd] = answer + sir.registry[fd] = answer } return answer } -var sourceInfoRegistry = sourceInfoRegistryType{} +var sourceInfoRegistry = newSourceInfoRegistryType() From 3265ade3d97d20ca5449c7f9b80dbfffec836968 Mon Sep 17 00:00:00 2001 From: noahdietz Date: Mon, 23 Sep 2024 11:42:07 -0700 Subject: [PATCH 2/3] follow mutex hat pattern --- locations/locations.go | 18 ++++++++++-------- 1 file changed, 10 insertions(+), 8 deletions(-) diff --git a/locations/locations.go b/locations/locations.go index 56ee80576..7d8fadcd8 100644 --- a/locations/locations.go +++ b/locations/locations.go @@ -40,8 +40,9 @@ func pathLocation(d desc.Descriptor, path ...int) *dpb.SourceCodeInfo_Location { } type sourceInfo struct { - info map[string]*dpb.SourceCodeInfo_Location - lock sync.Mutex + // infoMu protects the info map + infoMu sync.Mutex + info map[string]*dpb.SourceCodeInfo_Location } func newSourceInfo() *sourceInfo { @@ -52,8 +53,8 @@ func newSourceInfo() *sourceInfo { // findLocation returns the Location for a given path. func (si *sourceInfo) findLocation(path []int32) *dpb.SourceCodeInfo_Location { - si.lock.Lock() - defer si.lock.Unlock() + si.infoMu.Lock() + defer si.infoMu.Unlock() // If the path exists in the source info registry, return that object. if loc, ok := si.info[strPath(path)]; ok { @@ -68,8 +69,9 @@ func (si *sourceInfo) findLocation(path []int32) *dpb.SourceCodeInfo_Location { // any file descriptor that it is given, but then caches it to avoid computing // the source map for the same file descriptors over and over. type sourceInfoRegistryType struct { - registry map[*desc.FileDescriptor]*sourceInfo - lock sync.Mutex + // registryMu protects the registry map + registryMu sync.Mutex + registry map[*desc.FileDescriptor]*sourceInfo } func newSourceInfoRegistryType() *sourceInfoRegistryType { @@ -94,8 +96,8 @@ func strPath(segments []int32) (p string) { // It also caches this into a registry, so subsequent calls using the same // descriptor will return the same object. func (sir *sourceInfoRegistryType) sourceInfo(fd *desc.FileDescriptor) *sourceInfo { - sir.lock.Lock() - defer sir.lock.Unlock() + sir.registryMu.Lock() + defer sir.registryMu.Unlock() answer, ok := sir.registry[fd] if !ok { answer = newSourceInfo() From 5f1cf389da4b41918a6c4b63720f6126705be39d Mon Sep 17 00:00:00 2001 From: noahdietz Date: Mon, 23 Sep 2024 12:03:19 -0700 Subject: [PATCH 3/3] add test and run unit tests with -race --- .github/workflows/ci.yaml | 2 +- locations/locations_test.go | 28 ++++++++++++++++++++++++++++ 2 files changed, 29 insertions(+), 1 deletion(-) diff --git a/.github/workflows/ci.yaml b/.github/workflows/ci.yaml index 9a7b67f4e..399db11a4 100644 --- a/.github/workflows/ci.yaml +++ b/.github/workflows/ci.yaml @@ -12,7 +12,7 @@ jobs: - uses: actions/setup-go@v5 with: go-version: "1.20" - - run: go test -p 1 ./... + - run: go test -race ./... lint: runs-on: ubuntu-latest steps: diff --git a/locations/locations_test.go b/locations/locations_test.go index 2ada999a4..e4afe555f 100644 --- a/locations/locations_test.go +++ b/locations/locations_test.go @@ -16,6 +16,7 @@ package locations import ( "strings" + "sync" "testing" "github.com/jhump/protoreflect/desc" @@ -47,3 +48,30 @@ func parse(t *testing.T, s string) *desc.FileDescriptor { } return fds[0] } + +func TestSourceInfo_Concurrency(t *testing.T) { + fd := parse(t, ` + syntax = "proto3"; + package foo.bar; + `) + + var wg sync.WaitGroup + wg.Add(1) + go func() { + defer wg.Done() + FileSyntax(fd) + }() + + wg.Add(1) + go func() { + defer wg.Done() + FilePackage(fd) + }() + + wg.Add(1) + go func() { + defer wg.Done() + FileImport(fd, 0) + }() + wg.Wait() +}