2012-04-20 20:04:42 +00:00
|
|
|
/*
|
|
|
|
Copyright 2012 Google 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 localdisk
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
dirLockMu sync.Mutex // guards rest:
|
|
|
|
locksOut int64
|
2012-04-23 00:17:51 +00:00
|
|
|
dirLocks = map[string]*sync.RWMutex{}
|
2012-04-20 20:04:42 +00:00
|
|
|
)
|
|
|
|
|
2012-04-23 00:17:51 +00:00
|
|
|
func getDirLock(dir string) *sync.RWMutex {
|
|
|
|
dirLockMu.Lock()
|
|
|
|
defer dirLockMu.Unlock()
|
|
|
|
locksOut++
|
|
|
|
l, ok := dirLocks[dir]
|
|
|
|
if !ok {
|
|
|
|
l = new(sync.RWMutex)
|
|
|
|
dirLocks[dir] = l
|
|
|
|
}
|
|
|
|
return l
|
2012-04-20 20:04:42 +00:00
|
|
|
}
|
|
|
|
|
2012-04-23 00:17:51 +00:00
|
|
|
func unlockDirLock() {
|
2012-04-20 20:04:42 +00:00
|
|
|
dirLockMu.Lock()
|
2012-04-23 00:17:51 +00:00
|
|
|
defer dirLockMu.Unlock()
|
2012-04-20 20:04:42 +00:00
|
|
|
locksOut--
|
|
|
|
if locksOut == 0 {
|
2012-04-23 00:17:51 +00:00
|
|
|
dirLocks = map[string]*sync.RWMutex{}
|
2012-04-20 20:04:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-23 00:17:51 +00:00
|
|
|
type unlocker interface {
|
|
|
|
Unlock()
|
|
|
|
}
|
|
|
|
|
|
|
|
// keepDirectoryLock locks directory and returns the locked object.
|
2012-04-20 20:04:42 +00:00
|
|
|
// Holding the lock prevents the directory from being deleted.
|
|
|
|
// The caller must Unlock it when finished.
|
2012-04-23 00:17:51 +00:00
|
|
|
func keepDirectoryLock(dir string) unlocker {
|
|
|
|
mu := getDirLock(dir)
|
|
|
|
mu.RLock()
|
|
|
|
return keepLock{mu}
|
|
|
|
}
|
|
|
|
|
|
|
|
type keepLock struct {
|
|
|
|
mu *sync.RWMutex
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l keepLock) Unlock() {
|
|
|
|
l.mu.RUnlock()
|
|
|
|
unlockDirLock()
|
|
|
|
}
|
|
|
|
|
|
|
|
// deleteDirectoryLock locks directory and returns the locked object.
|
|
|
|
// Holding the lock is necessary while deleting the directory.
|
|
|
|
// The caller must Unlock it when finished.
|
|
|
|
func deleteDirectoryLock(dir string) unlocker {
|
|
|
|
mu := getDirLock(dir)
|
|
|
|
mu.Lock()
|
|
|
|
return deleteLock{mu}
|
|
|
|
}
|
|
|
|
|
|
|
|
type deleteLock struct {
|
|
|
|
mu *sync.RWMutex
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l deleteLock) Unlock() {
|
|
|
|
l.mu.Unlock()
|
|
|
|
unlockDirLock()
|
2012-04-20 20:04:42 +00:00
|
|
|
}
|