2020-02-26 21:48:05 +01:00
|
|
|
//go:build !windows
|
|
|
|
// +build !windows
|
|
|
|
|
|
|
|
package restorer
|
|
|
|
|
2022-09-04 10:39:16 +02:00
|
|
|
import (
|
|
|
|
"github.com/restic/restic/internal/restic"
|
|
|
|
)
|
2020-02-26 21:48:05 +01:00
|
|
|
|
|
|
|
// WriteAt writes p to f.File at offset. It tries to do a sparse write
|
|
|
|
// and updates f.size.
|
|
|
|
func (f *partialFile) WriteAt(p []byte, offset int64) (n int, err error) {
|
2022-08-07 17:26:46 +02:00
|
|
|
if !f.sparse {
|
|
|
|
return f.File.WriteAt(p, offset)
|
|
|
|
}
|
|
|
|
|
2020-02-26 21:48:05 +01:00
|
|
|
n = len(p)
|
|
|
|
|
|
|
|
// Skip the longest all-zero prefix of p.
|
|
|
|
// If it's long enough, we can punch a hole in the file.
|
2022-09-04 10:39:16 +02:00
|
|
|
skipped := restic.ZeroPrefixLen(p)
|
2020-02-26 21:48:05 +01:00
|
|
|
p = p[skipped:]
|
|
|
|
offset += int64(skipped)
|
|
|
|
|
|
|
|
switch {
|
|
|
|
case len(p) == 0:
|
|
|
|
// All zeros, file already big enough. A previous WriteAt or
|
|
|
|
// Truncate will have produced the zeros in f.File.
|
|
|
|
|
|
|
|
default:
|
2022-08-07 17:56:14 +02:00
|
|
|
var n2 int
|
|
|
|
n2, err = f.File.WriteAt(p, offset)
|
|
|
|
n = skipped + n2
|
2020-02-26 21:48:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return n, err
|
|
|
|
}
|