| |
| |
| |
|
|
| package poll |
|
|
| import ( |
| "io" |
| "syscall" |
| ) |
|
|
| |
| func SendFile(fd *FD, src uintptr, size int64) (written int64, err error, handled bool) { |
| defer func() { |
| TestHookDidSendFile(fd, 0, written, err, written > 0) |
| }() |
| if fd.kind == kindPipe { |
| |
| return 0, syscall.ESPIPE, false |
| } |
| hsrc := syscall.Handle(src) |
| if ft, _ := syscall.GetFileType(hsrc); ft == syscall.FILE_TYPE_PIPE { |
| return 0, syscall.ESPIPE, false |
| } |
|
|
| if err := fd.writeLock(); err != nil { |
| return 0, err, false |
| } |
| defer fd.writeUnlock() |
|
|
| |
| var fi syscall.ByHandleFileInformation |
| if err := syscall.GetFileInformationByHandle(hsrc, &fi); err != nil { |
| return 0, err, false |
| } |
| fileSize := int64(fi.FileSizeHigh)<<32 + int64(fi.FileSizeLow) |
| startpos, err := syscall.Seek(hsrc, 0, io.SeekCurrent) |
| if err != nil { |
| return 0, err, false |
| } |
| maxSize := fileSize - startpos |
| if size <= 0 { |
| size = maxSize |
| } else { |
| size = min(size, maxSize) |
| } |
|
|
| defer func() { |
| if written > 0 { |
| |
| |
| |
| _, serr := syscall.Seek(hsrc, startpos+written, io.SeekStart) |
| if err != nil { |
| err = serr |
| } |
| } |
| }() |
|
|
| |
| |
| |
| const maxChunkSizePerCall = int64(0x7fffffff - 1) |
|
|
| for size > 0 { |
| chunkSize := maxChunkSizePerCall |
| if chunkSize > size { |
| chunkSize = size |
| } |
|
|
| fd.setOffset(startpos + written) |
| n, err := fd.execIO('w', func(o *operation) (uint32, error) { |
| err := syscall.TransmitFile(fd.Sysfd, hsrc, uint32(chunkSize), 0, &o.o, nil, syscall.TF_WRITE_BEHIND) |
| if err != nil { |
| return 0, err |
| } |
| return uint32(chunkSize), nil |
| }) |
| if err != nil { |
| return written, err, written > 0 |
| } |
|
|
| size -= int64(n) |
| written += int64(n) |
| } |
|
|
| return written, nil, written > 0 |
| } |
|
|