| // Copyright 2015 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| package poll |
| |
| import "syscall" |
| |
| // Not strictly needed, but very helpful for debugging, see issue #10221. |
| //go:cgo_import_dynamic _ _ "libsendfile.so" |
| //go:cgo_import_dynamic _ _ "libsocket.so" |
| |
| // maxSendfileSize is the largest chunk size we ask the kernel to copy |
| // at a time. |
| const maxSendfileSize int = 4 << 20 |
| |
| // SendFile wraps the sendfile system call. |
| func SendFile(dstFD *FD, src int, pos, remain int64) (int64, error) { |
| if err := dstFD.writeLock(); err != nil { |
| return 0, err |
| } |
| defer dstFD.writeUnlock() |
| |
| dst := int(dstFD.Sysfd) |
| var written int64 |
| var err error |
| for remain > 0 { |
| n := maxSendfileSize |
| if int64(n) > remain { |
| n = int(remain) |
| } |
| pos1 := pos |
| n, err1 := syscall.Sendfile(dst, src, &pos1, n) |
| if err1 == syscall.EAGAIN || err1 == syscall.EINTR { |
| // partial write may have occurred |
| if n = int(pos1 - pos); n == 0 { |
| // nothing more to write |
| err1 = nil |
| } |
| } |
| if n > 0 { |
| pos += int64(n) |
| written += int64(n) |
| remain -= int64(n) |
| } |
| if n == 0 && err1 == nil { |
| break |
| } |
| if err1 == syscall.EAGAIN { |
| if err1 = dstFD.pd.waitWrite(dstFD.isFile); err1 == nil { |
| continue |
| } |
| } |
| if err1 == syscall.EINTR { |
| continue |
| } |
| if err1 != nil { |
| // This includes syscall.ENOSYS (no kernel |
| // support) and syscall.EINVAL (fd types which |
| // don't implement sendfile) |
| err = err1 |
| break |
| } |
| } |
| return written, err |
| } |