Merge pull request #17176 from rhvgoyal/fix-locking-issue

devmapper: Drop devices lock before returning from function
This commit is contained in:
Brian Goff 2015-10-20 13:08:29 -04:00
Родитель ad861876e8 2f16895ee9
Коммит 7777c1be9b
2 изменённых файлов: 30 добавлений и 0 удалений

Просмотреть файл

@ -599,6 +599,7 @@ func (devices *DeviceSet) cleanupDeletedDevices() error {
// If there are no deleted devices, there is nothing to do. // If there are no deleted devices, there is nothing to do.
if devices.nrDeletedDevices == 0 { if devices.nrDeletedDevices == 0 {
devices.Unlock()
return nil return nil
} }

Просмотреть файл

@ -5,6 +5,7 @@ package devmapper
import ( import (
"fmt" "fmt"
"testing" "testing"
"time"
"github.com/docker/docker/daemon/graphdriver" "github.com/docker/docker/daemon/graphdriver"
"github.com/docker/docker/daemon/graphdriver/graphtest" "github.com/docker/docker/daemon/graphdriver/graphtest"
@ -79,3 +80,31 @@ func testChangeLoopBackSize(t *testing.T, delta, expectDataSize, expectMetaDataS
t.Fatal(err) t.Fatal(err)
} }
} }
// Make sure devices.Lock() has been release upon return from cleanupDeletedDevices() function
func TestDevmapperLockReleasedDeviceDeletion(t *testing.T) {
driver := graphtest.GetDriver(t, "devicemapper").(*graphtest.Driver).Driver.(*graphdriver.NaiveDiffDriver).ProtoDriver.(*Driver)
defer graphtest.PutDriver(t)
// Call cleanupDeletedDevices() and after the call take and release
// DeviceSet Lock. If lock has not been released, this will hang.
driver.DeviceSet.cleanupDeletedDevices()
doneChan := make(chan bool)
go func() {
driver.DeviceSet.Lock()
defer driver.DeviceSet.Unlock()
doneChan <- true
}()
select {
case <-time.After(time.Second * 5):
// Timer expired. That means lock was not released upon
// function return and we are deadlocked. Release lock
// here so that cleanup could succeed and fail the test.
driver.DeviceSet.Unlock()
t.Fatalf("Could not acquire devices lock after call to cleanupDeletedDevices()")
case <-doneChan:
}
}