2017-03-29 20:55:42 -04:00
|
|
|
// +build linux,cgo
|
2013-11-27 22:12:51 -05:00
|
|
|
|
2015-12-14 17:16:34 -05:00
|
|
|
package loopback
|
2013-11-27 20:12:57 -05:00
|
|
|
|
|
|
|
import (
|
2015-12-14 17:16:34 -05:00
|
|
|
"errors"
|
2013-11-27 20:12:57 -05:00
|
|
|
"fmt"
|
2014-05-16 08:10:02 -04:00
|
|
|
"os"
|
|
|
|
|
2017-07-26 17:42:13 -04:00
|
|
|
"github.com/sirupsen/logrus"
|
2017-05-23 10:22:32 -04:00
|
|
|
"golang.org/x/sys/unix"
|
2013-11-27 20:12:57 -05:00
|
|
|
)
|
|
|
|
|
2015-12-14 17:16:34 -05:00
|
|
|
// Loopback related errors
|
|
|
|
var (
|
|
|
|
ErrAttachLoopbackDevice = errors.New("loopback attach failed")
|
|
|
|
ErrGetLoopbackBackingFile = errors.New("Unable to get loopback backing file")
|
|
|
|
ErrSetCapacity = errors.New("Unable set loopback capacity")
|
|
|
|
)
|
|
|
|
|
2013-11-27 20:44:54 -05:00
|
|
|
func stringToLoopName(src string) [LoNameSize]uint8 {
|
|
|
|
var dst [LoNameSize]uint8
|
|
|
|
copy(dst[:], src[:])
|
|
|
|
return dst
|
2013-11-27 20:12:57 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func getNextFreeLoopbackIndex() (int, error) {
|
2014-05-16 08:10:02 -04:00
|
|
|
f, err := os.OpenFile("/dev/loop-control", os.O_RDONLY, 0644)
|
2013-11-27 20:12:57 -05:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
index, err := ioctlLoopCtlGetFree(f.Fd())
|
|
|
|
if index < 0 {
|
|
|
|
index = 0
|
|
|
|
}
|
|
|
|
return index, err
|
|
|
|
}
|
|
|
|
|
2014-05-16 08:10:02 -04:00
|
|
|
func openNextAvailableLoopback(index int, sparseFile *os.File) (loopFile *os.File, err error) {
|
2013-11-27 20:12:57 -05:00
|
|
|
// Start looking for a free /dev/loop
|
|
|
|
for {
|
|
|
|
target := fmt.Sprintf("/dev/loop%d", index)
|
|
|
|
index++
|
|
|
|
|
2014-05-16 08:10:02 -04:00
|
|
|
fi, err := os.Stat(target)
|
2013-11-27 20:12:57 -05:00
|
|
|
if err != nil {
|
2014-05-16 08:10:02 -04:00
|
|
|
if os.IsNotExist(err) {
|
2016-06-11 16:16:55 -04:00
|
|
|
logrus.Error("There are no more loopback devices available.")
|
2013-11-27 20:12:57 -05:00
|
|
|
}
|
|
|
|
return nil, ErrAttachLoopbackDevice
|
|
|
|
}
|
|
|
|
|
2014-05-16 08:10:02 -04:00
|
|
|
if fi.Mode()&os.ModeDevice != os.ModeDevice {
|
2015-03-26 18:22:04 -04:00
|
|
|
logrus.Errorf("Loopback device %s is not a block device.", target)
|
2013-11-27 21:21:17 -05:00
|
|
|
continue
|
2013-11-27 20:12:57 -05:00
|
|
|
}
|
|
|
|
|
2013-11-28 14:02:53 -05:00
|
|
|
// OpenFile adds O_CLOEXEC
|
2014-05-16 08:10:02 -04:00
|
|
|
loopFile, err = os.OpenFile(target, os.O_RDWR, 0644)
|
2013-11-27 20:12:57 -05:00
|
|
|
if err != nil {
|
2015-03-26 18:22:04 -04:00
|
|
|
logrus.Errorf("Error opening loopback device: %s", err)
|
2013-11-27 20:12:57 -05:00
|
|
|
return nil, ErrAttachLoopbackDevice
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to attach to the loop file
|
|
|
|
if err := ioctlLoopSetFd(loopFile.Fd(), sparseFile.Fd()); err != nil {
|
|
|
|
loopFile.Close()
|
|
|
|
|
|
|
|
// If the error is EBUSY, then try the next loopback
|
2017-05-23 10:22:32 -04:00
|
|
|
if err != unix.EBUSY {
|
2015-03-26 18:22:04 -04:00
|
|
|
logrus.Errorf("Cannot set up loopback device %s: %s", target, err)
|
2013-11-27 20:12:57 -05:00
|
|
|
return nil, ErrAttachLoopbackDevice
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, we keep going with the loop
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
// In case of success, we finished. Break the loop.
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
// This can't happen, but let's be sure
|
|
|
|
if loopFile == nil {
|
2015-03-26 18:22:04 -04:00
|
|
|
logrus.Errorf("Unreachable code reached! Error attaching %s to a loopback device.", sparseFile.Name())
|
2013-11-27 20:12:57 -05:00
|
|
|
return nil, ErrAttachLoopbackDevice
|
|
|
|
}
|
|
|
|
|
|
|
|
return loopFile, nil
|
|
|
|
}
|
|
|
|
|
2015-09-04 17:02:29 -04:00
|
|
|
// AttachLoopDevice attaches the given sparse file to the next
|
2014-05-16 08:10:02 -04:00
|
|
|
// available loopback device. It returns an opened *os.File.
|
2014-11-05 18:10:38 -05:00
|
|
|
func AttachLoopDevice(sparseName string) (loop *os.File, err error) {
|
2013-11-27 20:12:57 -05:00
|
|
|
|
|
|
|
// Try to retrieve the next available loopback device via syscall.
|
2015-12-13 11:00:39 -05:00
|
|
|
// If it fails, we discard error and start looping for a
|
2013-11-27 20:12:57 -05:00
|
|
|
// loopback from index 0.
|
|
|
|
startIndex, err := getNextFreeLoopbackIndex()
|
|
|
|
if err != nil {
|
2015-03-26 18:22:04 -04:00
|
|
|
logrus.Debugf("Error retrieving the next available loopback: %s", err)
|
2013-11-27 20:12:57 -05:00
|
|
|
}
|
|
|
|
|
2013-11-28 14:02:53 -05:00
|
|
|
// OpenFile adds O_CLOEXEC
|
2014-05-16 08:10:02 -04:00
|
|
|
sparseFile, err := os.OpenFile(sparseName, os.O_RDWR, 0644)
|
2013-11-27 20:12:57 -05:00
|
|
|
if err != nil {
|
2015-03-26 18:22:04 -04:00
|
|
|
logrus.Errorf("Error opening sparse file %s: %s", sparseName, err)
|
2013-11-27 20:12:57 -05:00
|
|
|
return nil, ErrAttachLoopbackDevice
|
|
|
|
}
|
|
|
|
defer sparseFile.Close()
|
|
|
|
|
|
|
|
loopFile, err := openNextAvailableLoopback(startIndex, sparseFile)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the status of the loopback device
|
2015-09-04 17:02:29 -04:00
|
|
|
loopInfo := &loopInfo64{
|
2013-11-27 20:12:57 -05:00
|
|
|
loFileName: stringToLoopName(loopFile.Name()),
|
|
|
|
loOffset: 0,
|
|
|
|
loFlags: LoFlagsAutoClear,
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := ioctlLoopSetStatus64(loopFile.Fd(), loopInfo); err != nil {
|
2015-03-26 18:22:04 -04:00
|
|
|
logrus.Errorf("Cannot set up loopback device info: %s", err)
|
2013-11-27 20:12:57 -05:00
|
|
|
|
|
|
|
// If the call failed, then free the loopback device
|
|
|
|
if err := ioctlLoopClrFd(loopFile.Fd()); err != nil {
|
2016-06-11 16:16:55 -04:00
|
|
|
logrus.Error("Error while cleaning up the loopback device")
|
2013-11-27 20:12:57 -05:00
|
|
|
}
|
|
|
|
loopFile.Close()
|
|
|
|
return nil, ErrAttachLoopbackDevice
|
|
|
|
}
|
|
|
|
|
|
|
|
return loopFile, nil
|
|
|
|
}
|