From 180b2e56ac33fd74b2271e68d995270a3d2ee3af Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Andr=C3=A9=20Duffeck?= Date: Fri, 16 Jun 2023 09:11:02 +0200 Subject: [PATCH 1/2] Fix concurrent access to the lock map Co-authored-by: Christian Richter --- .../jsoncs3/providercache/providercache.go | 18 +++++------------- .../receivedsharecache/receivedsharecache.go | 18 +++++------------- .../manager/jsoncs3/sharecache/sharecache.go | 18 +++++------------- 3 files changed, 15 insertions(+), 39 deletions(-) diff --git a/pkg/share/manager/jsoncs3/providercache/providercache.go b/pkg/share/manager/jsoncs3/providercache/providercache.go index a91b517fd5..08cbb06000 100644 --- a/pkg/share/manager/jsoncs3/providercache/providercache.go +++ b/pkg/share/manager/jsoncs3/providercache/providercache.go @@ -43,8 +43,7 @@ const tracerName = "providercache" // Cache holds share information structured by provider and space type Cache struct { - lockMapLock sync.Mutex - lockMap map[string]*sync.Mutex + lockMap sync.Map Providers map[string]*Spaces @@ -106,16 +105,9 @@ func (s *Shares) UnmarshalJSON(data []byte) error { // LockSpace locks the cache for a given space and returns an unlock function func (c *Cache) LockSpace(spaceID string) func() { - lock := c.lockMap[spaceID] - if lock == nil { - c.lockMapLock.Lock() - lock = c.lockMap[spaceID] - if lock == nil { - c.lockMap[spaceID] = &sync.Mutex{} - lock = c.lockMap[spaceID] - } - c.lockMapLock.Unlock() - } + v, _ := c.lockMap.LoadOrStore(spaceID, &sync.Mutex{}) + lock := v.(*sync.Mutex) + lock.Lock() return func() { lock.Unlock() } } @@ -126,7 +118,7 @@ func New(s metadata.Storage, ttl time.Duration) Cache { Providers: map[string]*Spaces{}, storage: s, ttl: ttl, - lockMap: map[string]*sync.Mutex{}, + lockMap: sync.Map{}, } } diff --git a/pkg/share/manager/jsoncs3/receivedsharecache/receivedsharecache.go b/pkg/share/manager/jsoncs3/receivedsharecache/receivedsharecache.go index faf6d13f49..4810f1ff13 100644 --- a/pkg/share/manager/jsoncs3/receivedsharecache/receivedsharecache.go +++ b/pkg/share/manager/jsoncs3/receivedsharecache/receivedsharecache.go @@ -44,8 +44,7 @@ const tracerName = "receivedsharecache" // It functions as an in-memory cache with a persistence layer // The storage is sharded by user type Cache struct { - lockMapLock sync.Mutex - lockMap map[string]*sync.Mutex + lockMap sync.Map ReceivedSpaces map[string]*Spaces @@ -79,21 +78,14 @@ func New(s metadata.Storage, ttl time.Duration) Cache { ReceivedSpaces: map[string]*Spaces{}, storage: s, ttl: ttl, - lockMap: map[string]*sync.Mutex{}, + lockMap: sync.Map{}, } } func (c *Cache) lockUser(userID string) func() { - lock := c.lockMap[userID] - if lock == nil { - c.lockMapLock.Lock() - lock = c.lockMap[userID] - if lock == nil { - c.lockMap[userID] = &sync.Mutex{} - lock = c.lockMap[userID] - } - c.lockMapLock.Unlock() - } + v, _ := c.lockMap.LoadOrStore(userID, &sync.Mutex{}) + lock := v.(*sync.Mutex) + lock.Lock() return func() { lock.Unlock() } } diff --git a/pkg/share/manager/jsoncs3/sharecache/sharecache.go b/pkg/share/manager/jsoncs3/sharecache/sharecache.go index 649f8dee6f..f4058d4594 100644 --- a/pkg/share/manager/jsoncs3/sharecache/sharecache.go +++ b/pkg/share/manager/jsoncs3/sharecache/sharecache.go @@ -43,8 +43,7 @@ const tracerName = "sharecache" // It functions as an in-memory cache with a persistence layer // The storage is sharded by user/group type Cache struct { - lockMapLock sync.Mutex - lockMap map[string]*sync.Mutex + lockMap sync.Map UserShares map[string]*UserShareCache @@ -69,16 +68,9 @@ type SpaceShareIDs struct { } func (c *Cache) lockUser(userID string) func() { - lock := c.lockMap[userID] - if lock == nil { - c.lockMapLock.Lock() - lock = c.lockMap[userID] - if lock == nil { - c.lockMap[userID] = &sync.Mutex{} - lock = c.lockMap[userID] - } - c.lockMapLock.Unlock() - } + v, _ := c.lockMap.LoadOrStore(userID, &sync.Mutex{}) + lock := v.(*sync.Mutex) + lock.Lock() return func() { lock.Unlock() } } @@ -91,7 +83,7 @@ func New(s metadata.Storage, namespace, filename string, ttl time.Duration) Cach namespace: namespace, filename: filename, ttl: ttl, - lockMap: map[string]*sync.Mutex{}, + lockMap: sync.Map{}, } } From a2fec67c0314408a05c277e78c588136eb58cb7c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Andr=C3=A9=20Duffeck?= Date: Fri, 16 Jun 2023 09:33:39 +0200 Subject: [PATCH 2/2] Add changelog --- changelog/unreleased/jsoncs3-lock-congestion.md | 1 + 1 file changed, 1 insertion(+) diff --git a/changelog/unreleased/jsoncs3-lock-congestion.md b/changelog/unreleased/jsoncs3-lock-congestion.md index 21149b1be2..2254531c07 100644 --- a/changelog/unreleased/jsoncs3-lock-congestion.md +++ b/changelog/unreleased/jsoncs3-lock-congestion.md @@ -3,4 +3,5 @@ Bugfix: Reduce jsoncs3 lock congestion We changed the locking strategy in the jsoncs3 share manager to cause less lock congestion increasing the performance in certain scenarios. +https://github.com/cs3org/reva/pull/3985 https://github.com/cs3org/reva/pull/3964