mirror of
https://github.com/rclone/rclone.git
synced 2025-01-26 16:20:23 +01:00
backend/local: fadvise run syscall on a dedicated go routine
Before we issued an additional syscall periodically on a hot path. This patch offloads the fadvise syscall to a dedicated go routine.
This commit is contained in:
parent
24161d12ab
commit
712f7e38f7
@ -45,6 +45,35 @@ type fadvise struct {
|
|||||||
lastPos int64
|
lastPos int64
|
||||||
curPos int64
|
curPos int64
|
||||||
windowSize int64
|
windowSize int64
|
||||||
|
|
||||||
|
freePagesCh chan offsetLength
|
||||||
|
doneCh chan struct{}
|
||||||
|
}
|
||||||
|
|
||||||
|
type offsetLength struct {
|
||||||
|
offset int64
|
||||||
|
length int64
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
defaultAllowPages = 32
|
||||||
|
defaultWorkerQueueSize = 64
|
||||||
|
)
|
||||||
|
|
||||||
|
func newFadvise(o *Object, fd int, offset int64) *fadvise {
|
||||||
|
f := &fadvise{
|
||||||
|
o: o,
|
||||||
|
fd: fd,
|
||||||
|
lastPos: offset,
|
||||||
|
curPos: offset,
|
||||||
|
windowSize: int64(os.Getpagesize()) * defaultAllowPages,
|
||||||
|
|
||||||
|
freePagesCh: make(chan offsetLength, defaultWorkerQueueSize),
|
||||||
|
doneCh: make(chan struct{}),
|
||||||
|
}
|
||||||
|
go f.worker()
|
||||||
|
|
||||||
|
return f
|
||||||
}
|
}
|
||||||
|
|
||||||
// sequential configures readahead strategy in Linux kernel.
|
// sequential configures readahead strategy in Linux kernel.
|
||||||
@ -77,10 +106,23 @@ func (f *fadvise) freePagesIfNeeded() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (f *fadvise) freePages() {
|
func (f *fadvise) freePages() {
|
||||||
if err := unix.Fadvise(f.fd, f.lastPos, f.curPos-f.lastPos, unix.FADV_DONTNEED); err != nil {
|
f.freePagesCh <- offsetLength{f.lastPos, f.curPos - f.lastPos}
|
||||||
|
f.lastPos = f.curPos
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *fadvise) worker() {
|
||||||
|
for p := range f.freePagesCh {
|
||||||
|
if err := unix.Fadvise(f.fd, p.offset, p.length, unix.FADV_DONTNEED); err != nil {
|
||||||
fs.Debugf(f.o, "fadvise dontneed failed on file descriptor %d: %s", f.fd, err)
|
fs.Debugf(f.o, "fadvise dontneed failed on file descriptor %d: %s", f.fd, err)
|
||||||
}
|
}
|
||||||
f.lastPos = f.curPos
|
}
|
||||||
|
|
||||||
|
close(f.doneCh)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (f *fadvise) wait() {
|
||||||
|
close(f.freePagesCh)
|
||||||
|
<-f.doneCh
|
||||||
}
|
}
|
||||||
|
|
||||||
type fadviseReadCloser struct {
|
type fadviseReadCloser struct {
|
||||||
@ -88,8 +130,6 @@ type fadviseReadCloser struct {
|
|||||||
inner io.ReadCloser
|
inner io.ReadCloser
|
||||||
}
|
}
|
||||||
|
|
||||||
var defaultWindowSize = int64(32 * os.Getpagesize())
|
|
||||||
|
|
||||||
// newFadviseReadCloser wraps os.File so that reading from that file would
|
// newFadviseReadCloser wraps os.File so that reading from that file would
|
||||||
// remove already consumed pages from kernel page cache.
|
// remove already consumed pages from kernel page cache.
|
||||||
// In addition to that it instructs kernel to double the readahead window to
|
// In addition to that it instructs kernel to double the readahead window to
|
||||||
@ -97,13 +137,7 @@ var defaultWindowSize = int64(32 * os.Getpagesize())
|
|||||||
// See also fadvise.
|
// See also fadvise.
|
||||||
func newFadviseReadCloser(o *Object, f *os.File, offset, limit int64) io.ReadCloser {
|
func newFadviseReadCloser(o *Object, f *os.File, offset, limit int64) io.ReadCloser {
|
||||||
r := fadviseReadCloser{
|
r := fadviseReadCloser{
|
||||||
fadvise: &fadvise{
|
fadvise: newFadvise(o, int(f.Fd()), offset),
|
||||||
o: o,
|
|
||||||
fd: int(f.Fd()),
|
|
||||||
lastPos: offset,
|
|
||||||
curPos: offset,
|
|
||||||
windowSize: defaultWindowSize,
|
|
||||||
},
|
|
||||||
inner: f,
|
inner: f,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -111,6 +145,7 @@ func newFadviseReadCloser(o *Object, f *os.File, offset, limit int64) io.ReadClo
|
|||||||
// file descriptor would also fail. In that case return the provided os.File
|
// file descriptor would also fail. In that case return the provided os.File
|
||||||
// pointer.
|
// pointer.
|
||||||
if !r.sequential(limit) {
|
if !r.sequential(limit) {
|
||||||
|
r.wait()
|
||||||
return f
|
return f
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -125,5 +160,6 @@ func (f fadviseReadCloser) Read(p []byte) (n int, err error) {
|
|||||||
|
|
||||||
func (f fadviseReadCloser) Close() error {
|
func (f fadviseReadCloser) Close() error {
|
||||||
f.freePages()
|
f.freePages()
|
||||||
|
f.wait()
|
||||||
return f.inner.Close()
|
return f.inner.Close()
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user