2013-03-21 20:47:23 -04:00
|
|
|
package docker
|
2013-03-21 01:41:03 -04:00
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2013-06-04 14:30:47 -04:00
|
|
|
"github.com/dotcloud/docker/utils"
|
2013-03-21 01:41:03 -04:00
|
|
|
"os"
|
2013-06-04 14:30:47 -04:00
|
|
|
"os/exec"
|
2013-03-21 01:41:03 -04:00
|
|
|
"path/filepath"
|
|
|
|
"syscall"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
func Unmount(target string) error {
|
2013-06-04 14:30:47 -04:00
|
|
|
if err := exec.Command("auplink", target, "flush").Run(); err != nil {
|
2013-10-08 03:54:47 -04:00
|
|
|
utils.Errorf("[warning]: couldn't run auplink before unmount: %s", err)
|
2013-06-04 14:30:47 -04:00
|
|
|
}
|
2013-03-21 01:41:03 -04:00
|
|
|
if err := syscall.Unmount(target, 0); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// Even though we just unmounted the filesystem, AUFS will prevent deleting the mntpoint
|
|
|
|
// for some time. We'll just keep retrying until it succeeds.
|
|
|
|
for retries := 0; retries < 1000; retries++ {
|
|
|
|
err := os.Remove(target)
|
|
|
|
if err == nil {
|
|
|
|
// rm mntpoint succeeded
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
// mntpoint doesn't exist anymore. Success.
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
// fmt.Printf("(%v) Remove %v returned: %v\n", retries, target, err)
|
|
|
|
time.Sleep(10 * time.Millisecond)
|
|
|
|
}
|
|
|
|
return fmt.Errorf("Umount: Failed to umount %v", target)
|
|
|
|
}
|
|
|
|
|
|
|
|
func Mounted(mountpoint string) (bool, error) {
|
|
|
|
mntpoint, err := os.Stat(mountpoint)
|
|
|
|
if err != nil {
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
parent, err := os.Stat(filepath.Join(mountpoint, ".."))
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
mntpointSt := mntpoint.Sys().(*syscall.Stat_t)
|
|
|
|
parentSt := parent.Sys().(*syscall.Stat_t)
|
|
|
|
return mntpointSt.Dev != parentSt.Dev, nil
|
|
|
|
}
|