2020-06-18 06:15:47 +00:00
|
|
|
// Copyright (C) 2019 The Syncthing Authors.
|
|
|
|
//
|
|
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
|
|
|
2020-08-19 05:56:35 +00:00
|
|
|
// +build linux,!ppc,!ppc64,!ppc64le
|
2020-06-18 06:15:47 +00:00
|
|
|
|
|
|
|
package fs
|
|
|
|
|
|
|
|
import (
|
2020-08-07 05:47:48 +00:00
|
|
|
"io"
|
2020-06-18 06:15:47 +00:00
|
|
|
"syscall"
|
|
|
|
"unsafe"
|
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
registerCopyRangeImplementation(CopyRangeMethodIoctl, copyRangeImplementationForBasicFile(copyRangeIoctl))
|
|
|
|
}
|
|
|
|
|
|
|
|
const FICLONE = 0x40049409
|
|
|
|
const FICLONERANGE = 0x4020940d
|
|
|
|
|
|
|
|
/*
|
|
|
|
http://man7.org/linux/man-pages/man2/ioctl_ficlonerange.2.html
|
|
|
|
|
|
|
|
struct file_clone_range {
|
|
|
|
__s64 src_fd;
|
|
|
|
__u64 src_offset;
|
|
|
|
__u64 src_length;
|
|
|
|
__u64 dest_offset;
|
|
|
|
};
|
|
|
|
*/
|
|
|
|
type fileCloneRange struct {
|
|
|
|
srcFd int64
|
|
|
|
srcOffset uint64
|
|
|
|
srcLength uint64
|
|
|
|
dstOffset uint64
|
|
|
|
}
|
|
|
|
|
|
|
|
func copyRangeIoctl(src, dst basicFile, srcOffset, dstOffset, size int64) error {
|
|
|
|
fi, err := src.Stat()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-08-07 05:47:48 +00:00
|
|
|
if srcOffset+size > fi.Size() {
|
|
|
|
return io.ErrUnexpectedEOF
|
|
|
|
}
|
|
|
|
|
2020-06-18 06:15:47 +00:00
|
|
|
// https://www.man7.org/linux/man-pages/man2/ioctl_ficlonerange.2.html
|
|
|
|
// If src_length is zero, the ioctl reflinks to the end of the source file.
|
|
|
|
if srcOffset+size == fi.Size() {
|
|
|
|
size = 0
|
|
|
|
}
|
|
|
|
|
|
|
|
if srcOffset == 0 && dstOffset == 0 && size == 0 {
|
|
|
|
// Optimization for whole file copies.
|
2020-08-07 05:47:48 +00:00
|
|
|
var errNo syscall.Errno
|
|
|
|
_, err := withFileDescriptors(src, dst, func(srcFd, dstFd uintptr) (int, error) {
|
|
|
|
_, _, errNo = syscall.Syscall(syscall.SYS_IOCTL, dstFd, FICLONE, srcFd)
|
|
|
|
return 0, nil
|
|
|
|
})
|
|
|
|
// Failure in withFileDescriptors
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-06-18 06:15:47 +00:00
|
|
|
if errNo != 0 {
|
|
|
|
return errNo
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-08-07 05:47:48 +00:00
|
|
|
var errNo syscall.Errno
|
|
|
|
_, err = withFileDescriptors(src, dst, func(srcFd, dstFd uintptr) (int, error) {
|
|
|
|
params := fileCloneRange{
|
|
|
|
srcFd: int64(srcFd),
|
|
|
|
srcOffset: uint64(srcOffset),
|
|
|
|
srcLength: uint64(size),
|
|
|
|
dstOffset: uint64(dstOffset),
|
|
|
|
}
|
|
|
|
_, _, errNo = syscall.Syscall(syscall.SYS_IOCTL, dstFd, FICLONERANGE, uintptr(unsafe.Pointer(¶ms)))
|
|
|
|
return 0, nil
|
|
|
|
})
|
|
|
|
// Failure in withFileDescriptors
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2020-06-18 06:15:47 +00:00
|
|
|
}
|
|
|
|
if errNo != 0 {
|
|
|
|
return errNo
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|