From 984903cce1f54574d810d091819c4a045d11a173 Mon Sep 17 00:00:00 2001 From: Harshavardhana Date: Fri, 29 Apr 2016 01:29:09 -0700 Subject: [PATCH] server: Add global namespace lock. (#1398) Fixes #1393 --- main.go | 3 ++ namespace-lock.go | 128 ++++++++++++++++++++++++++++++++++++++++++++ object-api_test.go | 4 ++ server_test.go | 3 ++ xl-v1-createfile.go | 10 ++-- xl-v1-healfile.go | 7 ++- xl-v1-namespace.go | 71 ------------------------ xl-v1-readfile.go | 9 ++-- xl-v1.go | 64 +++------------------- 9 files changed, 157 insertions(+), 142 deletions(-) create mode 100644 namespace-lock.go delete mode 100644 xl-v1-namespace.go diff --git a/main.go b/main.go index 7e0909a18..e732e6c95 100644 --- a/main.go +++ b/main.go @@ -202,6 +202,9 @@ func main() { // Enable all loggers by now. enableLoggers() + // Initialize name space lock. + initNSLock() + // Do not print update messages, if quiet flag is set. if !globalQuiet { // Do not print any errors in release update function. diff --git a/namespace-lock.go b/namespace-lock.go new file mode 100644 index 000000000..018747a6e --- /dev/null +++ b/namespace-lock.go @@ -0,0 +1,128 @@ +/* + * Minio Cloud Storage, (C) 2016 Minio, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package main + +import "sync" + +// nsParam - carries name space resource. +type nsParam struct { + volume string + path string +} + +// nsLock - provides primitives for locking critical namespace regions. +type nsLock struct { + *sync.RWMutex + ref uint +} + +// nsLockMap - namespace lock map, provides primitives to Lock, +// Unlock, RLock and RUnlock. +type nsLockMap struct { + lockMap map[nsParam]*nsLock + mutex *sync.Mutex +} + +// Global name space lock. +var nsMutex *nsLockMap + +// initNSLock - initialize name space lock map. +func initNSLock() { + nsMutex = &nsLockMap{ + lockMap: make(map[nsParam]*nsLock), + mutex: &sync.Mutex{}, + } +} + +// Lock - locks the given resource for writes, using a previously +// allocated name space lock or initializing a new one. +func (n *nsLockMap) Lock(volume, path string) { + n.mutex.Lock() + defer n.mutex.Unlock() + + param := nsParam{volume, path} + nsLk, found := n.lockMap[param] + if !found { + nsLk = &nsLock{ + RWMutex: &sync.RWMutex{}, + ref: 0, + } + } + + // Acquire a write lock and update reference counter. + nsLk.Lock() + nsLk.ref++ + + n.lockMap[param] = nsLk +} + +// Unlock - unlocks any previously acquired write locks. +func (n *nsLockMap) Unlock(volume, path string) { + n.mutex.Lock() + defer n.mutex.Unlock() + + param := nsParam{volume, path} + if nsLk, found := n.lockMap[param]; found { + // Unlock a write lock and update reference counter. + nsLk.Unlock() + if nsLk.ref != 0 { + nsLk.ref-- + } + if nsLk.ref != 0 { + n.lockMap[param] = nsLk + } + } +} + +// RLock - locks any previously acquired read locks. +func (n *nsLockMap) RLock(volume, path string) { + n.mutex.Lock() + defer n.mutex.Unlock() + + param := nsParam{volume, path} + nsLk, found := n.lockMap[param] + if !found { + nsLk = &nsLock{ + RWMutex: &sync.RWMutex{}, + ref: 0, + } + } + + // Acquire a read lock and update reference counter. + nsLk.RLock() + nsLk.ref++ + + n.lockMap[param] = nsLk +} + +// RUnlock - unlocks any previously acquired read locks. +func (n *nsLockMap) RUnlock(volume, path string) { + n.mutex.Lock() + defer n.mutex.Unlock() + + param := nsParam{volume, path} + if nsLk, found := n.lockMap[param]; found { + // Unlock a read lock and update reference counter. + nsLk.RUnlock() + if nsLk.ref != 0 { + nsLk.ref-- + } + if nsLk.ref != 0 { + n.lockMap[param] = nsLk + } + } +} diff --git a/object-api_test.go b/object-api_test.go index 590ac73ab..6f4a97756 100644 --- a/object-api_test.go +++ b/object-api_test.go @@ -44,6 +44,10 @@ func (s *MySuite) TestFSAPISuite(c *C) { func (s *MySuite) TestXLAPISuite(c *C) { var storageList []string + + // Initialize name space lock. + initNSLock() + create := func() objectAPI { var nDisks = 16 // Maximum disks. var erasureDisks []string diff --git a/server_test.go b/server_test.go index 2f5d9c044..587352a87 100644 --- a/server_test.go +++ b/server_test.go @@ -81,6 +81,9 @@ func (s *MyAPISuite) SetUpSuite(c *C) { // Initialize server config. initConfig() + // Initialize name space lock. + initNSLock() + // Set port. addr := ":" + strconv.Itoa(getFreePort()) diff --git a/xl-v1-createfile.go b/xl-v1-createfile.go index 2996471f2..c7a43d09e 100644 --- a/xl-v1-createfile.go +++ b/xl-v1-createfile.go @@ -58,10 +58,9 @@ func (xl XL) writeErasure(volume, path string, reader *io.PipeReader, wcloser *w defer wcloser.release() // Lock right before reading from disk. - readLock := true - xl.lockNS(volume, path, readLock) + nsMutex.RLock(volume, path) partsMetadata, errs := xl.getPartsMetadata(volume, path) - xl.unlockNS(volume, path, readLock) + nsMutex.RUnlock(volume, path) // Count errors other than fileNotFound, bigger than the allowed // readQuorum, if yes throw an error. @@ -263,9 +262,8 @@ func (xl XL) writeErasure(volume, path string, reader *io.PipeReader, wcloser *w } // Lock right before commit to disk. - readLock = false // false means writeLock. - xl.lockNS(volume, path, readLock) - defer xl.unlockNS(volume, path, readLock) + nsMutex.Lock(volume, path) + defer nsMutex.Unlock(volume, path) // Close all writers and metadata writers in routines. for index, writer := range writers { diff --git a/xl-v1-healfile.go b/xl-v1-healfile.go index 319b2908e..31419d386 100644 --- a/xl-v1-healfile.go +++ b/xl-v1-healfile.go @@ -33,11 +33,10 @@ func (xl XL) healFile(volume string, path string) error { var writers = make([]io.WriteCloser, totalBlocks) // Acquire a read lock. - readLock := true - xl.lockNS(volume, path, readLock) - defer xl.unlockNS(volume, path, readLock) + nsMutex.RLock(volume, path) + defer nsMutex.RUnlock(volume, path) - // Fetch all online disks. + // List all online disks to verify if we need to heal. onlineDisks, metadata, heal, err := xl.listOnlineDisks(volume, path) if err != nil { log.WithFields(logrus.Fields{ diff --git a/xl-v1-namespace.go b/xl-v1-namespace.go deleted file mode 100644 index 2f276af6b..000000000 --- a/xl-v1-namespace.go +++ /dev/null @@ -1,71 +0,0 @@ -/* - * Minio Cloud Storage, (C) 2016 Minio, Inc. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -package main - -import "sync" - -// nameSpaceParam - carries name space resource. -type nameSpaceParam struct { - volume string - path string -} - -// nameSpaceLock - provides primitives for locking critical namespace regions. -type nameSpaceLock struct { - rwMutex *sync.RWMutex - count uint -} - -func (nsLock *nameSpaceLock) InUse() bool { - return nsLock.count != 0 -} - -// Lock acquires write lock and increments the namespace counter. -func (nsLock *nameSpaceLock) Lock() { - nsLock.rwMutex.Lock() - nsLock.count++ -} - -// Unlock releases write lock and decrements the namespace counter. -func (nsLock *nameSpaceLock) Unlock() { - nsLock.rwMutex.Unlock() - if nsLock.count != 0 { - nsLock.count-- - } -} - -// RLock acquires read lock and increments the namespace counter. -func (nsLock *nameSpaceLock) RLock() { - nsLock.rwMutex.RLock() - nsLock.count++ -} - -// RUnlock release read lock and decrements the namespace counter. -func (nsLock *nameSpaceLock) RUnlock() { - nsLock.rwMutex.RUnlock() - if nsLock.count != 0 { - nsLock.count-- - } -} - -// newNSLock - provides a new instance of namespace locking primitives. -func newNSLock() *nameSpaceLock { - return &nameSpaceLock{ - rwMutex: &sync.RWMutex{}, - count: 0, - } -} diff --git a/xl-v1-readfile.go b/xl-v1-readfile.go index 3022d909f..46022943f 100644 --- a/xl-v1-readfile.go +++ b/xl-v1-readfile.go @@ -36,10 +36,9 @@ func (xl XL) ReadFile(volume, path string, offset int64) (io.ReadCloser, error) } // Acquire a read lock. - readLock := true - xl.lockNS(volume, path, readLock) + nsMutex.RLock(volume, path) onlineDisks, metadata, heal, err := xl.listOnlineDisks(volume, path) - xl.unlockNS(volume, path, readLock) + nsMutex.RUnlock(volume, path) if err != nil { log.WithFields(logrus.Fields{ "volume": volume, @@ -63,7 +62,7 @@ func (xl XL) ReadFile(volume, path string, offset int64) (io.ReadCloser, error) } // Acquire read lock again. - xl.lockNS(volume, path, readLock) + nsMutex.RLock(volume, path) readers := make([]io.ReadCloser, len(xl.storageDisks)) for index, disk := range onlineDisks { if disk == nil { @@ -77,7 +76,7 @@ func (xl XL) ReadFile(volume, path string, offset int64) (io.ReadCloser, error) readers[index] = reader } } - xl.unlockNS(volume, path, readLock) + nsMutex.RUnlock(volume, path) // Initialize pipe. pipeReader, pipeWriter := io.Pipe() diff --git a/xl-v1.go b/xl-v1.go index 0e9dd3915..d724c046c 100644 --- a/xl-v1.go +++ b/xl-v1.go @@ -22,7 +22,6 @@ import ( slashpath "path" "sort" "strings" - "sync" "github.com/Sirupsen/logrus" "github.com/klauspost/reedsolomon" @@ -37,54 +36,12 @@ const ( // XL layer structure. type XL struct { - ReedSolomon reedsolomon.Encoder // Erasure encoder/decoder. - DataBlocks int - ParityBlocks int - storageDisks []StorageAPI - nameSpaceLockMap map[nameSpaceParam]*nameSpaceLock - nameSpaceLockMapMutex *sync.Mutex - readQuorum int - writeQuorum int -} - -// lockNS - locks the given resource, using a previously allocated -// name space lock or initializing a new one. -func (xl XL) lockNS(volume, path string, readLock bool) { - xl.nameSpaceLockMapMutex.Lock() - defer xl.nameSpaceLockMapMutex.Unlock() - - param := nameSpaceParam{volume, path} - nsLock, found := xl.nameSpaceLockMap[param] - if !found { - nsLock = newNSLock() - } - - if readLock { - nsLock.RLock() - } else { - nsLock.Lock() - } - - xl.nameSpaceLockMap[param] = nsLock -} - -// unlockNS - unlocks any previously acquired read or write locks. -func (xl XL) unlockNS(volume, path string, readLock bool) { - xl.nameSpaceLockMapMutex.Lock() - defer xl.nameSpaceLockMapMutex.Unlock() - - param := nameSpaceParam{volume, path} - if nsLock, found := xl.nameSpaceLockMap[param]; found { - if readLock { - nsLock.RUnlock() - } else { - nsLock.Unlock() - } - - if nsLock.InUse() { - xl.nameSpaceLockMap[param] = nsLock - } - } + ReedSolomon reedsolomon.Encoder // Erasure encoder/decoder. + DataBlocks int + ParityBlocks int + storageDisks []StorageAPI + readQuorum int + writeQuorum int } // newXL instantiate a new XL. @@ -135,10 +92,6 @@ func newXL(disks ...string) (StorageAPI, error) { // Save all the initialized storage disks. xl.storageDisks = storageDisks - // Initialize name space lock map. - xl.nameSpaceLockMap = make(map[nameSpaceParam]*nameSpaceLock) - xl.nameSpaceLockMapMutex = &sync.Mutex{} - // Figure out read and write quorum based on number of storage disks. // Read quorum should be always N/2 + 1 (due to Vandermonde matrix // erasure requirements) @@ -552,10 +505,9 @@ func (xl XL) StatFile(volume, path string) (FileInfo, error) { } // Acquire read lock. - readLock := true - xl.lockNS(volume, path, readLock) + nsMutex.RLock(volume, path) _, metadata, heal, err := xl.listOnlineDisks(volume, path) - xl.unlockNS(volume, path, readLock) + nsMutex.RUnlock(volume, path) if err != nil { log.WithFields(logrus.Fields{ "volume": volume,