Explorar o código

devmapper: wait for devices to be effectively removed before returning a successful remove

Solomon Hykes %!s(int64=11) %!d(string=hai) anos
pai
achega
ea04f3de72
Modificáronse 1 ficheiros con 55 adicións e 7 borrados
  1. 55 7
      devmapper/deviceset_devmapper.go

+ 55 - 7
devmapper/deviceset_devmapper.go

@@ -527,26 +527,58 @@ func (devices *DeviceSetDM) RemoveDevice(hash string) error {
 }
 }
 
 
 func (devices *DeviceSetDM) deactivateDevice(hash string) error {
 func (devices *DeviceSetDM) deactivateDevice(hash string) error {
-	info := devices.Devices[hash]
-	if info == nil {
-		return fmt.Errorf("hash %s doesn't exists", hash)
+	utils.Debugf("[devmapper] deactivateDevice(%s)", hash)
+	defer utils.Debugf("[devmapper] deactivateDevice END")
+	var devname string
+	// FIXME: shouldn't we just register the pool into devices?
+	devname, err := devices.byHash(hash)
+	if err != nil {
+		return err
 	}
 	}
-
-	devinfo, err := getInfo(info.Name())
+	devinfo, err := getInfo(devname)
 	if err != nil {
 	if err != nil {
 		utils.Debugf("\n--->Err: %s\n", err)
 		utils.Debugf("\n--->Err: %s\n", err)
 		return err
 		return err
 	}
 	}
 	if devinfo.Exists != 0 {
 	if devinfo.Exists != 0 {
-		if err := removeDevice(info.Name()); err != nil {
+		if err := removeDevice(devname); err != nil {
 			utils.Debugf("\n--->Err: %s\n", err)
 			utils.Debugf("\n--->Err: %s\n", err)
 			return err
 			return err
 		}
 		}
+		if err := devices.waitRemove(hash); err != nil {
+			return err
+		}
 	}
 	}
 
 
 	return nil
 	return nil
 }
 }
 
 
+// waitRemove blocks until either:
+// a) the device registered at <device_set_prefix>-<hash> is removed,
+// or b) the 1 second timeout expires.
+func (devices *DeviceSetDM) waitRemove(hash string) error {
+	devname, err := devices.byHash(hash)
+	if err != nil {
+		return err
+	}
+	i := 0
+	for ; i<1000; i+=1 {
+		devinfo, err := getInfo(devname)
+		if err != nil {
+			return err
+		}
+		utils.Debugf("Waiting for removal of %s: exists=%d", devname, devinfo.Exists)
+		if devinfo.Exists == 0 {
+			break
+		}
+		time.Sleep(1 * time.Millisecond)
+	}
+	if i == 1000 {
+		return fmt.Errorf("Timeout while waiting for device %s to be removed", devname)
+	}
+	return nil
+}
+
 // waitClose blocks until either:
 // waitClose blocks until either:
 // a) the device registered at <device_set_prefix>-<hash> is closed,
 // a) the device registered at <device_set_prefix>-<hash> is closed,
 // or b) the 1 second timeout expires.
 // or b) the 1 second timeout expires.
@@ -573,6 +605,22 @@ func (devices *DeviceSetDM) waitClose(hash string) error {
 	return nil
 	return nil
 }
 }
 
 
+// byHash is a hack to allow looking up the deviceset's pool by the hash "pool".
+// FIXME: it seems probably cleaner to register the pool in devices.Devices,
+// but I am afraid of arcane implications deep in the devicemapper code,
+// so this will do.
+func (devices *DeviceSetDM) byHash(hash string) (devname string, err error) {
+	if hash == "pool" {
+		return devices.getPoolDevName(), nil
+	}
+	info := devices.Devices[hash]
+	if info == nil {
+		return "", fmt.Errorf("hash %s doesn't exists", hash)
+	}
+	return info.Name(), nil
+}
+
+
 func (devices *DeviceSetDM) Shutdown() error {
 func (devices *DeviceSetDM) Shutdown() error {
 	devices.Lock()
 	devices.Lock()
 	utils.Debugf("[devmapper] Shutting down DeviceSet: %s", devices.root)
 	utils.Debugf("[devmapper] Shutting down DeviceSet: %s", devices.root)
@@ -602,7 +650,7 @@ func (devices *DeviceSetDM) Shutdown() error {
 
 
 	pool := devices.getPoolDevName()
 	pool := devices.getPoolDevName()
 	if devinfo, err := getInfo(pool); err == nil && devinfo.Exists != 0 {
 	if devinfo, err := getInfo(pool); err == nil && devinfo.Exists != 0 {
-		if err := removeDevice(pool); err != nil {
+		if err := devices.deactivateDevice("pool"); err != nil {
 			utils.Debugf("Shutdown deactivate %s , error: %s\n", pool, err)
 			utils.Debugf("Shutdown deactivate %s , error: %s\n", pool, err)
 		}
 		}
 	}
 	}