1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105
|
//go:build linux
package dmflakey
import (
"fmt"
"os"
"os/exec"
"time"
"unsafe"
"golang.org/x/sys/unix"
)
// newFlakeyDevice creates flakey device.
//
// REF: https://docs.kernel.org/admin-guide/device-mapper/dm-flakey.html
func newFlakeyDevice(flakeyDevice, loopDevice string, interval time.Duration) error {
loopSize, err := getBlkSize(loopDevice)
if err != nil {
return fmt.Errorf("failed to get the size of the loop device %s: %w", loopDevice, err)
}
// The flakey device will be available in interval.Seconds().
table := fmt.Sprintf("0 %d flakey %s 0 %d 0",
loopSize, loopDevice, int(interval.Seconds()))
args := []string{"create", flakeyDevice, "--table", table}
output, err := exec.Command("dmsetup", args...).CombinedOutput()
if err != nil {
return fmt.Errorf("failed to create flakey device %s with table %s (out: %s): %w",
flakeyDevice, table, string(output), err)
}
return nil
}
// reloadFlakeyDevice reloads the flakey device with feature table.
func reloadFlakeyDevice(flakeyDevice string, syncFS bool, table string) (retErr error) {
args := []string{"suspend", "--nolockfs", flakeyDevice}
if syncFS {
args[1] = flakeyDevice
args = args[:len(args)-1]
}
output, err := exec.Command("dmsetup", args...).CombinedOutput()
if err != nil {
return fmt.Errorf("failed to suspend flakey device %s (out: %s): %w",
flakeyDevice, string(output), err)
}
defer func() {
output, derr := exec.Command("dmsetup", "resume", flakeyDevice).CombinedOutput()
if derr != nil {
derr = fmt.Errorf("failed to resume flakey device %s (out: %s): %w",
flakeyDevice, string(output), derr)
}
if retErr == nil {
retErr = derr
}
}()
output, err = exec.Command("dmsetup", "load", flakeyDevice, "--table", table).CombinedOutput()
if err != nil {
return fmt.Errorf("failed to reload flakey device %s with table (%s) (out: %s): %w",
flakeyDevice, table, string(output), err)
}
return nil
}
// removeFlakeyDevice removes flakey device.
func deleteFlakeyDevice(flakeyDevice string) error {
output, err := exec.Command("dmsetup", "remove", flakeyDevice).CombinedOutput()
if err != nil {
return fmt.Errorf("failed to remove flakey device %s (out: %s): %w",
flakeyDevice, string(output), err)
}
return nil
}
// getBlkSize64 gets device size in bytes (BLKGETSIZE64).
//
// REF: https://man7.org/linux/man-pages/man8/blockdev.8.html
func getBlkSize64(device string) (int64, error) {
deviceFd, err := os.Open(device)
if err != nil {
return 0, fmt.Errorf("failed to open device %s: %w", device, err)
}
defer deviceFd.Close()
var size int64
if _, _, err := unix.Syscall(unix.SYS_IOCTL, deviceFd.Fd(), unix.BLKGETSIZE64, uintptr(unsafe.Pointer(&size))); err != 0 {
return 0, fmt.Errorf("failed to get block size: %w", err)
}
return size, nil
}
// getBlkSize gets size in 512-byte sectors (BLKGETSIZE64 / 512).
//
// REF: https://man7.org/linux/man-pages/man8/blockdev.8.html
func getBlkSize(device string) (int64, error) {
size, err := getBlkSize64(device)
return size / 512, err
}
|