Skip to content
Snippets Groups Projects
syscall_windows.go 49.4 KiB
Newer Older
  • Learn to ignore specific revisions
  • // Copyright 2009 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.
    
    // Windows system calls.
    
    package syscall
    
    
    	"internal/bytealg"
    
    	"internal/itoa"
    
    	"internal/oserror"
    
    	"internal/race"
    
    	"runtime"
    
    	"unicode/utf16"
    
    type Handle uintptr
    
    const InvalidHandle = ^Handle(0)
    
    
    // StringToUTF16 returns the UTF-16 encoding of the UTF-8 string s,
    // with a terminating NUL added. If s contains a NUL byte this
    // function panics instead of returning an error.
    //
    // Deprecated: Use UTF16FromString instead.
    
    func StringToUTF16(s string) []uint16 {
    	a, err := UTF16FromString(s)
    	if err != nil {
    		panic("syscall: string with NUL passed to StringToUTF16")
    	}
    	return a
    }
    
    // UTF16FromString returns the UTF-16 encoding of the UTF-8 string
    // s, with a terminating NUL added. If s contains a NUL byte at any
    // location, it returns (nil, EINVAL).
    func UTF16FromString(s string) ([]uint16, error) {
    
    	if bytealg.IndexByteString(s, 0) != -1 {
    		return nil, EINVAL
    
    	// In the worst case all characters require two uint16.
    	// Also account for the terminating NULL character.
    	buf := make([]uint16, 0, len(s)*2+1)
    	for _, r := range s {
    		buf = utf16.AppendRune(buf, r)
    	}
    	return utf16.AppendRune(buf, '\x00'), nil
    
    // UTF16ToString returns the UTF-8 encoding of the UTF-16 sequence s,
    // with a terminating NUL removed.
    func UTF16ToString(s []uint16) string {
    
    	for i, v := range s {
    		if v == 0 {
    			s = s[0:i]
    			break
    		}
    
    	return string(utf16.Decode(s))
    
    // utf16PtrToString is like UTF16ToString, but takes *uint16
    // as a parameter instead of []uint16.
    
    func utf16PtrToString(p *uint16) string {
    
    	if p == nil {
    		return ""
    	}
    	// Find NUL terminator.
    	end := unsafe.Pointer(p)
    	n := 0
    
    		end = unsafe.Pointer(uintptr(end) + unsafe.Sizeof(*p))
    		n++
    	}
    
    	s := unsafe.Slice(p, n)
    
    // StringToUTF16Ptr returns pointer to the UTF-16 encoding of
    // the UTF-8 string s, with a terminating NUL added. If s
    
    // contains a NUL byte this function panics instead of
    
    //
    // Deprecated: Use UTF16PtrFromString instead.
    
    func StringToUTF16Ptr(s string) *uint16 { return &StringToUTF16(s)[0] }
    
    // UTF16PtrFromString returns pointer to the UTF-16 encoding of
    // the UTF-8 string s, with a terminating NUL added. If s
    // contains a NUL byte at any location, it returns (nil, EINVAL).
    func UTF16PtrFromString(s string) (*uint16, error) {
    	a, err := UTF16FromString(s)
    	if err != nil {
    		return nil, err
    	}
    	return &a[0], nil
    }
    
    
    // Errno is the Windows error number.
    
    // Errno values can be tested against error values using errors.Is.
    // For example:
    
    //
    //	_, _, err := syscall.Syscall(...)
    
    //	if errors.Is(err, fs.ErrNotExist) ...
    
    func langid(pri, sub uint16) uint32 { return uint32(sub)<<10 | uint32(pri) }
    
    
    // FormatMessage is deprecated (msgsrc should be uintptr, not uint32, but can
    // not be changed due to the Go 1 compatibility guarantee).
    //
    // Deprecated: Use FormatMessage from golang.org/x/sys/windows instead.
    func FormatMessage(flags uint32, msgsrc uint32, msgid uint32, langid uint32, buf []uint16, args *byte) (n uint32, err error) {
    	return formatMessage(flags, uintptr(msgsrc), msgid, langid, buf, args)
    }
    
    
    func (e Errno) Error() string {
    	// deal with special go errors
    	idx := int(e - APPLICATION_ERROR)
    	if 0 <= idx && idx < len(errors) {
    		return errors[idx]
    	}
    	// ask windows for the remaining errors
    	var flags uint32 = FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_ARGUMENT_ARRAY | FORMAT_MESSAGE_IGNORE_INSERTS
    	b := make([]uint16, 300)
    
    	n, err := formatMessage(flags, 0, uint32(e), langid(LANG_ENGLISH, SUBLANG_ENGLISH_US), b, nil)
    
    		n, err = formatMessage(flags, 0, uint32(e), 0, b, nil)
    
    			return "winapi error #" + itoa.Itoa(int(e))
    
    	}
    	// trim terminating \r and \n
    	for ; n > 0 && (b[n-1] == '\n' || b[n-1] == '\r'); n-- {
    	}
    	return string(utf16.Decode(b[:n]))
    }
    
    
    const _ERROR_BAD_NETPATH = Errno(53)
    
    func (e Errno) Is(target error) bool {
    	switch target {
    	case oserror.ErrPermission:
    
    		return e == ERROR_ACCESS_DENIED ||
    			e == EACCES ||
    			e == EPERM
    
    	case oserror.ErrExist:
    		return e == ERROR_ALREADY_EXISTS ||
    			e == ERROR_DIR_NOT_EMPTY ||
    
    			e == ERROR_FILE_EXISTS ||
    			e == EEXIST ||
    			e == ENOTEMPTY
    
    	case oserror.ErrNotExist:
    		return e == ERROR_FILE_NOT_FOUND ||
    			e == _ERROR_BAD_NETPATH ||
    
    		return e == ENOSYS ||
    			e == ENOTSUP ||
    			e == EOPNOTSUPP ||
    			e == EWINDOWS
    
    func (e Errno) Temporary() bool {
    
    	return e == EINTR || e == EMFILE || e.Timeout()
    
    }
    
    func (e Errno) Timeout() bool {
    	return e == EAGAIN || e == EWOULDBLOCK || e == ETIMEDOUT
    }
    
    
    // Implemented in runtime/syscall_windows.go.
    
    func compileCallback(fn any, cleanstack bool) uintptr
    
    // NewCallback converts a Go function to a function pointer conforming to the stdcall calling convention.
    // This is useful when interoperating with Windows code requiring callbacks.
    
    // The argument is expected to be a function with one uintptr-sized result. The function must not have arguments with size larger than the size of uintptr.
    
    // Only a limited number of callbacks may be created in a single Go process, and any memory allocated
    // for these callbacks is never released.
    // Between NewCallback and NewCallbackCDecl, at least 1024 callbacks can always be created.
    
    func NewCallback(fn any) uintptr {
    
    	return compileCallback(fn, true)
    }
    
    
    // NewCallbackCDecl converts a Go function to a function pointer conforming to the cdecl calling convention.
    // This is useful when interoperating with Windows code requiring callbacks.
    
    // The argument is expected to be a function with one uintptr-sized result. The function must not have arguments with size larger than the size of uintptr.
    
    // Only a limited number of callbacks may be created in a single Go process, and any memory allocated
    // for these callbacks is never released.
    // Between NewCallback and NewCallbackCDecl, at least 1024 callbacks can always be created.
    
    func NewCallbackCDecl(fn any) uintptr {
    
    	return compileCallback(fn, false)
    }
    
    // windows api calls
    
    
    //sys	GetLastError() (lasterr error)
    
    Russ Cox's avatar
    Russ Cox committed
    //sys	LoadLibrary(libname string) (handle Handle, err error) = LoadLibraryW
    //sys	FreeLibrary(handle Handle) (err error)
    //sys	GetProcAddress(module Handle, procname string) (proc uintptr, err error)
    //sys	GetVersion() (ver uint32, err error)
    
    //sys	rtlGetNtVersionNumbers(majorVersion *uint32, minorVersion *uint32, buildNumber *uint32) = ntdll.RtlGetNtVersionNumbers
    
    //sys	formatMessage(flags uint32, msgsrc uintptr, msgid uint32, langid uint32, buf []uint16, args *byte) (n uint32, err error) = FormatMessageW
    
    //sys	ExitProcess(exitcode uint32)
    
    Russ Cox's avatar
    Russ Cox committed
    //sys	CreateFile(name *uint16, access uint32, mode uint32, sa *SecurityAttributes, createmode uint32, attrs uint32, templatefile int32) (handle Handle, err error) [failretval==InvalidHandle] = CreateFileW
    
    //sys	readFile(handle Handle, buf []byte, done *uint32, overlapped *Overlapped) (err error) = ReadFile
    //sys	writeFile(handle Handle, buf []byte, done *uint32, overlapped *Overlapped) (err error) = WriteFile
    
    Russ Cox's avatar
    Russ Cox committed
    //sys	SetFilePointer(handle Handle, lowoffset int32, highoffsetptr *int32, whence uint32) (newlowoffset uint32, err error) [failretval==0xffffffff]
    //sys	CloseHandle(handle Handle) (err error)
    //sys	GetStdHandle(stdhandle int) (handle Handle, err error) [failretval==InvalidHandle]
    
    //sys	findFirstFile1(name *uint16, data *win32finddata1) (handle Handle, err error) [failretval==InvalidHandle] = FindFirstFileW
    //sys	findNextFile1(handle Handle, data *win32finddata1) (err error) = FindNextFileW
    
    Russ Cox's avatar
    Russ Cox committed
    //sys	FindClose(handle Handle) (err error)
    //sys	GetFileInformationByHandle(handle Handle, data *ByHandleFileInformation) (err error)
    //sys	GetCurrentDirectory(buflen uint32, buf *uint16) (n uint32, err error) = GetCurrentDirectoryW
    //sys	SetCurrentDirectory(path *uint16) (err error) = SetCurrentDirectoryW
    //sys	CreateDirectory(path *uint16, sa *SecurityAttributes) (err error) = CreateDirectoryW
    //sys	RemoveDirectory(path *uint16) (err error) = RemoveDirectoryW
    //sys	DeleteFile(path *uint16) (err error) = DeleteFileW
    //sys	MoveFile(from *uint16, to *uint16) (err error) = MoveFileW
    //sys	GetComputerName(buf *uint16, n *uint32) (err error) = GetComputerNameW
    //sys	SetEndOfFile(handle Handle) (err error)
    
    Alex Brainman's avatar
    Alex Brainman committed
    //sys	GetSystemTimeAsFileTime(time *Filetime)
    
    Russ Cox's avatar
    Russ Cox committed
    //sys	GetTimeZoneInformation(tzi *Timezoneinformation) (rc uint32, err error) [failretval==0xffffffff]
    
    //sys	createIoCompletionPort(filehandle Handle, cphandle Handle, key uintptr, threadcnt uint32) (handle Handle, err error) = CreateIoCompletionPort
    //sys	getQueuedCompletionStatus(cphandle Handle, qty *uint32, key *uintptr, overlapped **Overlapped, timeout uint32) (err error) = GetQueuedCompletionStatus
    //sys	postQueuedCompletionStatus(cphandle Handle, qty uint32, key uintptr, overlapped *Overlapped) (err error) = PostQueuedCompletionStatus
    
    Russ Cox's avatar
    Russ Cox committed
    //sys	CancelIo(s Handle) (err error)
    
    //sys	CancelIoEx(s Handle, o *Overlapped) (err error)
    
    Russ Cox's avatar
    Russ Cox committed
    //sys	CreateProcess(appName *uint16, commandLine *uint16, procSecurity *SecurityAttributes, threadSecurity *SecurityAttributes, inheritHandles bool, creationFlags uint32, env *uint16, currentDir *uint16, startupInfo *StartupInfo, outProcInfo *ProcessInformation) (err error) = CreateProcessW
    
    //sys	CreateProcessAsUser(token Token, appName *uint16, commandLine *uint16, procSecurity *SecurityAttributes, threadSecurity *SecurityAttributes, inheritHandles bool, creationFlags uint32, env *uint16, currentDir *uint16, startupInfo *StartupInfo, outProcInfo *ProcessInformation) (err error) = advapi32.CreateProcessAsUserW
    
    Russ Cox's avatar
    Russ Cox committed
    //sys	OpenProcess(da uint32, inheritHandle bool, pid uint32) (handle Handle, err error)
    //sys	TerminateProcess(handle Handle, exitcode uint32) (err error)
    //sys	GetExitCodeProcess(handle Handle, exitcode *uint32) (err error)
    //sys	GetStartupInfo(startupInfo *StartupInfo) (err error) = GetStartupInfoW
    //sys	GetCurrentProcess() (pseudoHandle Handle, err error)
    
    //sys	GetProcessTimes(handle Handle, creationTime *Filetime, exitTime *Filetime, kernelTime *Filetime, userTime *Filetime) (err error)
    
    Russ Cox's avatar
    Russ Cox committed
    //sys	DuplicateHandle(hSourceProcessHandle Handle, hSourceHandle Handle, hTargetProcessHandle Handle, lpTargetHandle *Handle, dwDesiredAccess uint32, bInheritHandle bool, dwOptions uint32) (err error)
    //sys	WaitForSingleObject(handle Handle, waitMilliseconds uint32) (event uint32, err error) [failretval==0xffffffff]
    //sys	GetTempPath(buflen uint32, buf *uint16) (n uint32, err error) = GetTempPathW
    //sys	CreatePipe(readhandle *Handle, writehandle *Handle, sa *SecurityAttributes, size uint32) (err error)
    //sys	GetFileType(filehandle Handle) (n uint32, err error)
    //sys	CryptAcquireContext(provhandle *Handle, container *uint16, provider *uint16, provtype uint32, flags uint32) (err error) = advapi32.CryptAcquireContextW
    //sys	CryptReleaseContext(provhandle Handle, flags uint32) (err error) = advapi32.CryptReleaseContext
    //sys	CryptGenRandom(provhandle Handle, buflen uint32, buf *byte) (err error) = advapi32.CryptGenRandom
    //sys	GetEnvironmentStrings() (envs *uint16, err error) [failretval==nil] = kernel32.GetEnvironmentStringsW
    //sys	FreeEnvironmentStrings(envs *uint16) (err error) = kernel32.FreeEnvironmentStringsW
    //sys	GetEnvironmentVariable(name *uint16, buffer *uint16, size uint32) (n uint32, err error) = kernel32.GetEnvironmentVariableW
    //sys	SetEnvironmentVariable(name *uint16, value *uint16) (err error) = kernel32.SetEnvironmentVariableW
    //sys	SetFileTime(handle Handle, ctime *Filetime, atime *Filetime, wtime *Filetime) (err error)
    //sys	GetFileAttributes(name *uint16) (attrs uint32, err error) [failretval==INVALID_FILE_ATTRIBUTES] = kernel32.GetFileAttributesW
    //sys	SetFileAttributes(name *uint16, attrs uint32) (err error) = kernel32.SetFileAttributesW
    //sys	GetFileAttributesEx(name *uint16, level uint32, info *byte) (err error) = kernel32.GetFileAttributesExW
    
    //sys	GetCommandLine() (cmd *uint16) = kernel32.GetCommandLineW
    
    Russ Cox's avatar
    Russ Cox committed
    //sys	CommandLineToArgv(cmd *uint16, argc *int32) (argv *[8192]*[8192]uint16, err error) [failretval==nil] = shell32.CommandLineToArgvW
    //sys	LocalFree(hmem Handle) (handle Handle, err error) [failretval!=0]
    //sys	SetHandleInformation(handle Handle, mask uint32, flags uint32) (err error)
    //sys	FlushFileBuffers(handle Handle) (err error)
    //sys	GetFullPathName(path *uint16, buflen uint32, buf *uint16, fname **uint16) (n uint32, err error) = kernel32.GetFullPathNameW
    
    //sys	GetLongPathName(path *uint16, buf *uint16, buflen uint32) (n uint32, err error) = kernel32.GetLongPathNameW
    
    //sys	GetShortPathName(longpath *uint16, shortpath *uint16, buflen uint32) (n uint32, err error) = kernel32.GetShortPathNameW
    
    Russ Cox's avatar
    Russ Cox committed
    //sys	CreateFileMapping(fhandle Handle, sa *SecurityAttributes, prot uint32, maxSizeHigh uint32, maxSizeLow uint32, name *uint16) (handle Handle, err error) = kernel32.CreateFileMappingW
    //sys	MapViewOfFile(handle Handle, access uint32, offsetHigh uint32, offsetLow uint32, length uintptr) (addr uintptr, err error)
    //sys	UnmapViewOfFile(addr uintptr) (err error)
    //sys	FlushViewOfFile(addr uintptr, length uintptr) (err error)
    //sys	VirtualLock(addr uintptr, length uintptr) (err error)
    //sys	VirtualUnlock(addr uintptr, length uintptr) (err error)
    //sys	TransmitFile(s Handle, handle Handle, bytesToWrite uint32, bytsPerSend uint32, overlapped *Overlapped, transmitFileBuf *TransmitFileBuffers, flags uint32) (err error) = mswsock.TransmitFile
    //sys	ReadDirectoryChanges(handle Handle, buf *byte, buflen uint32, watchSubTree bool, mask uint32, retlen *uint32, overlapped *Overlapped, completionRoutine uintptr) (err error) = kernel32.ReadDirectoryChangesW
    //sys	CertOpenSystemStore(hprov Handle, name *uint16) (store Handle, err error) = crypt32.CertOpenSystemStoreW
    
    //sys   CertOpenStore(storeProvider uintptr, msgAndCertEncodingType uint32, cryptProv uintptr, flags uint32, para uintptr) (handle Handle, err error) = crypt32.CertOpenStore
    
    //sys	CertEnumCertificatesInStore(store Handle, prevContext *CertContext) (context *CertContext, err error) [failretval==nil] = crypt32.CertEnumCertificatesInStore
    
    //sys   CertAddCertificateContextToStore(store Handle, certContext *CertContext, addDisposition uint32, storeContext **CertContext) (err error) = crypt32.CertAddCertificateContextToStore
    
    Russ Cox's avatar
    Russ Cox committed
    //sys	CertCloseStore(store Handle, flags uint32) (err error) = crypt32.CertCloseStore
    
    //sys   CertGetCertificateChain(engine Handle, leaf *CertContext, time *Filetime, additionalStore Handle, para *CertChainPara, flags uint32, reserved uintptr, chainCtx **CertChainContext) (err error) = crypt32.CertGetCertificateChain
    //sys   CertFreeCertificateChain(ctx *CertChainContext) = crypt32.CertFreeCertificateChain
    //sys   CertCreateCertificateContext(certEncodingType uint32, certEncoded *byte, encodedLen uint32) (context *CertContext, err error) [failretval==nil] = crypt32.CertCreateCertificateContext
    //sys   CertFreeCertificateContext(ctx *CertContext) (err error) = crypt32.CertFreeCertificateContext
    //sys   CertVerifyCertificateChainPolicy(policyOID uintptr, chain *CertChainContext, para *CertChainPolicyPara, status *CertChainPolicyStatus) (err error) = crypt32.CertVerifyCertificateChainPolicy
    
    //sys	RegOpenKeyEx(key Handle, subkey *uint16, options uint32, desiredAccess uint32, result *Handle) (regerrno error) = advapi32.RegOpenKeyExW
    //sys	RegCloseKey(key Handle) (regerrno error) = advapi32.RegCloseKey
    //sys	RegQueryInfoKey(key Handle, class *uint16, classLen *uint32, reserved *uint32, subkeysLen *uint32, maxSubkeyLen *uint32, maxClassLen *uint32, valuesLen *uint32, maxValueNameLen *uint32, maxValueLen *uint32, saLen *uint32, lastWriteTime *Filetime) (regerrno error) = advapi32.RegQueryInfoKeyW
    
    //sys	regEnumKeyEx(key Handle, index uint32, name *uint16, nameLen *uint32, reserved *uint32, class *uint16, classLen *uint32, lastWriteTime *Filetime) (regerrno error) = advapi32.RegEnumKeyExW
    
    //sys	RegQueryValueEx(key Handle, name *uint16, reserved *uint32, valtype *uint32, buf *byte, buflen *uint32) (regerrno error) = advapi32.RegQueryValueExW
    
    //sys	getCurrentProcessId() (pid uint32) = kernel32.GetCurrentProcessId
    
    //sys	GetConsoleMode(console Handle, mode *uint32) (err error) = kernel32.GetConsoleMode
    //sys	WriteConsole(console Handle, buf *uint16, towrite uint32, written *uint32, reserved *byte) (err error) = kernel32.WriteConsoleW
    
    //sys	ReadConsole(console Handle, buf *uint16, toread uint32, read *uint32, inputControl *byte) (err error) = kernel32.ReadConsoleW
    
    //sys	CreateToolhelp32Snapshot(flags uint32, processId uint32) (handle Handle, err error) [failretval==InvalidHandle] = kernel32.CreateToolhelp32Snapshot
    //sys	Process32First(snapshot Handle, procEntry *ProcessEntry32) (err error) = kernel32.Process32FirstW
    //sys	Process32Next(snapshot Handle, procEntry *ProcessEntry32) (err error) = kernel32.Process32NextW
    
    //sys	DeviceIoControl(handle Handle, ioControlCode uint32, inBuffer *byte, inBufferSize uint32, outBuffer *byte, outBufferSize uint32, bytesReturned *uint32, overlapped *Overlapped) (err error)
    // This function returns 1 byte BOOLEAN rather than the 4 byte BOOL.
    //sys	CreateSymbolicLink(symlinkfilename *uint16, targetfilename *uint16, flags uint32) (err error) [failretval&0xff==0] = CreateSymbolicLinkW
    //sys	CreateHardLink(filename *uint16, existingfilename *uint16, reserved uintptr) (err error) [failretval&0xff==0] = CreateHardLinkW
    
    //sys	initializeProcThreadAttributeList(attrlist *_PROC_THREAD_ATTRIBUTE_LIST, attrcount uint32, flags uint32, size *uintptr) (err error) = InitializeProcThreadAttributeList
    //sys	deleteProcThreadAttributeList(attrlist *_PROC_THREAD_ATTRIBUTE_LIST) = DeleteProcThreadAttributeList
    
    //sys	updateProcThreadAttribute(attrlist *_PROC_THREAD_ATTRIBUTE_LIST, flags uint32, attr uintptr, value unsafe.Pointer, size uintptr, prevvalue unsafe.Pointer, returnedsize *uintptr) (err error) = UpdateProcThreadAttribute
    
    
    // syscall interface implementation for other packages
    
    func makeInheritSa() *SecurityAttributes {
    	var sa SecurityAttributes
    	sa.Length = uint32(unsafe.Sizeof(sa))
    	sa.InheritHandle = 1
    	return &sa
    }
    
    
    Russ Cox's avatar
    Russ Cox committed
    func Open(path string, mode int, perm uint32) (fd Handle, err error) {
    
    		return InvalidHandle, ERROR_FILE_NOT_FOUND
    
    	pathp, err := UTF16PtrFromString(path)
    	if err != nil {
    		return InvalidHandle, err
    	}
    
    	var access uint32
    	switch mode & (O_RDONLY | O_WRONLY | O_RDWR) {
    	case O_RDONLY:
    
    	case O_WRONLY:
    
    	case O_RDWR:
    
    		access = GENERIC_READ | GENERIC_WRITE
    	}
    
    	if mode&O_CREAT != 0 {
    		access |= GENERIC_WRITE
    	}
    
    	if mode&O_APPEND != 0 {
    		access &^= GENERIC_WRITE
    		access |= FILE_APPEND_DATA
    	}
    
    	sharemode := uint32(FILE_SHARE_READ | FILE_SHARE_WRITE)
    
    	var sa *SecurityAttributes
    	if mode&O_CLOEXEC == 0 {
    		sa = makeInheritSa()
    	}
    
    	case mode&(O_CREAT|O_EXCL) == (O_CREAT | O_EXCL):
    		createmode = CREATE_NEW
    	case mode&(O_CREAT|O_TRUNC) == (O_CREAT | O_TRUNC):
    		createmode = CREATE_ALWAYS
    	case mode&O_CREAT == O_CREAT:
    		createmode = OPEN_ALWAYS
    	case mode&O_TRUNC == O_TRUNC:
    
    		createmode = TRUNCATE_EXISTING
    	default:
    		createmode = OPEN_EXISTING
    	}
    
    	var attrs uint32 = FILE_ATTRIBUTE_NORMAL
    	if perm&S_IWRITE == 0 {
    		attrs = FILE_ATTRIBUTE_READONLY
    
    		if createmode == CREATE_ALWAYS {
    			// We have been asked to create a read-only file.
    			// If the file already exists, the semantics of
    			// the Unix open system call is to preserve the
    			// existing permissions. If we pass CREATE_ALWAYS
    			// and FILE_ATTRIBUTE_READONLY to CreateFile,
    			// and the file already exists, CreateFile will
    			// change the file permissions.
    			// Avoid that to preserve the Unix semantics.
    			h, e := CreateFile(pathp, access, sharemode, sa, TRUNCATE_EXISTING, FILE_ATTRIBUTE_NORMAL, 0)
    			switch e {
    			case ERROR_FILE_NOT_FOUND, _ERROR_BAD_NETPATH, ERROR_PATH_NOT_FOUND:
    				// File does not exist. These are the same
    				// errors as Errno.Is checks for ErrNotExist.
    				// Carry on to create the file.
    			default:
    				// Success or some different error.
    				return h, e
    			}
    		}
    
    	if createmode == OPEN_EXISTING && access == GENERIC_READ {
    		// Necessary for opening directory handles.
    		attrs |= FILE_FLAG_BACKUP_SEMANTICS
    	}
    	return CreateFile(pathp, access, sharemode, sa, createmode, attrs, 0)
    
    Russ Cox's avatar
    Russ Cox committed
    func Read(fd Handle, p []byte) (n int, err error) {
    
    	e := ReadFile(fd, p, &done, nil)
    
    Russ Cox's avatar
    Russ Cox committed
    	if e != nil {
    
    		if e == ERROR_BROKEN_PIPE {
    
    			// NOTE(brainman): work around ERROR_BROKEN_PIPE is returned on reading EOF from stdin
    
    Russ Cox's avatar
    Russ Cox committed
    			return 0, nil
    
    	return int(done), nil
    }
    
    func Write(fd Handle, p []byte) (n int, err error) {
    	var done uint32
    	e := WriteFile(fd, p, &done, nil)
    	if e != nil {
    		return 0, e
    	}
    	return int(done), nil
    }
    
    func ReadFile(fd Handle, p []byte, done *uint32, overlapped *Overlapped) error {
    	err := readFile(fd, p, done, overlapped)
    
    	if race.Enabled {
    
    		if *done > 0 {
    			race.WriteRange(unsafe.Pointer(&p[0]), int(*done))
    
    		race.Acquire(unsafe.Pointer(&ioSync))
    
    Dmitriy Vyukov's avatar
    Dmitriy Vyukov committed
    	}
    
    	if msanenabled && *done > 0 {
    		msanWrite(unsafe.Pointer(&p[0]), int(*done))
    
    	if asanenabled && *done > 0 {
    		asanWrite(unsafe.Pointer(&p[0]), int(*done))
    
    func WriteFile(fd Handle, p []byte, done *uint32, overlapped *Overlapped) error {
    
    	if race.Enabled {
    		race.ReleaseMerge(unsafe.Pointer(&ioSync))
    
    Dmitriy Vyukov's avatar
    Dmitriy Vyukov committed
    	}
    
    	err := writeFile(fd, p, done, overlapped)
    	if race.Enabled && *done > 0 {
    		race.ReadRange(unsafe.Pointer(&p[0]), int(*done))
    
    	if msanenabled && *done > 0 {
    		msanRead(unsafe.Pointer(&p[0]), int(*done))
    
    	if asanenabled && *done > 0 {
    		asanRead(unsafe.Pointer(&p[0]), int(*done))
    
    Dmitriy Vyukov's avatar
    Dmitriy Vyukov committed
    var ioSync int64
    
    
    var procSetFilePointerEx = modkernel32.NewProc("SetFilePointerEx")
    
    const ptrSize = unsafe.Sizeof(uintptr(0))
    
    // setFilePointerEx calls SetFilePointerEx.
    // See https://msdn.microsoft.com/en-us/library/windows/desktop/aa365542(v=vs.85).aspx
    func setFilePointerEx(handle Handle, distToMove int64, newFilePointer *int64, whence uint32) error {
    	var e1 Errno
    
    	if unsafe.Sizeof(uintptr(0)) == 8 {
    
    		_, _, e1 = Syscall6(procSetFilePointerEx.Addr(), 4, uintptr(handle), uintptr(distToMove), uintptr(unsafe.Pointer(newFilePointer)), uintptr(whence), 0, 0)
    
    	} else {
    		// Different 32-bit systems disgaree about whether distToMove starts 8-byte aligned.
    		switch runtime.GOARCH {
    		default:
    			panic("unsupported 32-bit architecture")
    		case "386":
    
    			// distToMove is a LARGE_INTEGER, which is 64 bits.
    
    			_, _, e1 = Syscall6(procSetFilePointerEx.Addr(), 5, uintptr(handle), uintptr(distToMove), uintptr(distToMove>>32), uintptr(unsafe.Pointer(newFilePointer)), uintptr(whence), 0)
    		case "arm":
    			// distToMove must be 8-byte aligned per ARM calling convention
    
    			// https://docs.microsoft.com/en-us/cpp/build/overview-of-arm-abi-conventions#stage-c-assignment-of-arguments-to-registers-and-stack
    
    			_, _, e1 = Syscall6(procSetFilePointerEx.Addr(), 6, uintptr(handle), 0, uintptr(distToMove), uintptr(distToMove>>32), uintptr(unsafe.Pointer(newFilePointer)), uintptr(whence))
    		}
    
    Russ Cox's avatar
    Russ Cox committed
    func Seek(fd Handle, offset int64, whence int) (newoffset int64, err error) {
    
    	var w uint32
    	switch whence {
    	case 0:
    		w = FILE_BEGIN
    	case 1:
    		w = FILE_CURRENT
    	case 2:
    		w = FILE_END
    	}
    
    	// use GetFileType to check pipe, pipe can't do seek
    
    	ft, _ := GetFileType(fd)
    
    	if ft == FILE_TYPE_PIPE {
    
    		return 0, ESPIPE
    
    	err = setFilePointerEx(fd, offset, &newoffset, w)
    	return
    
    Russ Cox's avatar
    Russ Cox committed
    func Close(fd Handle) (err error) {
    
    	return CloseHandle(fd)
    
    }
    
    var (
    	Stdin  = getStdHandle(STD_INPUT_HANDLE)
    	Stdout = getStdHandle(STD_OUTPUT_HANDLE)
    	Stderr = getStdHandle(STD_ERROR_HANDLE)
    )
    
    
    func getStdHandle(h int) (fd Handle) {
    
    Alex Brainman's avatar
    Alex Brainman committed
    const ImplementsGetwd = true
    
    
    Russ Cox's avatar
    Russ Cox committed
    func Getwd() (wd string, err error) {
    
    Alex Brainman's avatar
    Alex Brainman committed
    	b := make([]uint16, 300)
    	n, e := GetCurrentDirectory(uint32(len(b)), &b[0])
    
    Russ Cox's avatar
    Russ Cox committed
    	if e != nil {
    
    Alex Brainman's avatar
    Alex Brainman committed
    		return "", e
    	}
    
    Russ Cox's avatar
    Russ Cox committed
    	return string(utf16.Decode(b[0:n])), nil
    
    Russ Cox's avatar
    Russ Cox committed
    func Chdir(path string) (err error) {
    
    	pathp, err := UTF16PtrFromString(path)
    	if err != nil {
    		return err
    	}
    	return SetCurrentDirectory(pathp)
    
    Russ Cox's avatar
    Russ Cox committed
    func Mkdir(path string, mode uint32) (err error) {
    
    	pathp, err := UTF16PtrFromString(path)
    	if err != nil {
    		return err
    	}
    	return CreateDirectory(pathp, nil)
    
    Russ Cox's avatar
    Russ Cox committed
    func Rmdir(path string) (err error) {
    
    	pathp, err := UTF16PtrFromString(path)
    	if err != nil {
    		return err
    	}
    	return RemoveDirectory(pathp)
    
    Russ Cox's avatar
    Russ Cox committed
    func Unlink(path string) (err error) {
    
    	pathp, err := UTF16PtrFromString(path)
    	if err != nil {
    		return err
    	}
    	return DeleteFile(pathp)
    
    Russ Cox's avatar
    Russ Cox committed
    func Rename(oldpath, newpath string) (err error) {
    
    	from, err := UTF16PtrFromString(oldpath)
    	if err != nil {
    		return err
    	}
    	to, err := UTF16PtrFromString(newpath)
    	if err != nil {
    		return err
    	}
    
    Russ Cox's avatar
    Russ Cox committed
    func ComputerName() (name string, err error) {
    
    Alex Brainman's avatar
    Alex Brainman committed
    	var n uint32 = MAX_COMPUTERNAME_LENGTH + 1
    	b := make([]uint16, n)
    
    Russ Cox's avatar
    Russ Cox committed
    	if e != nil {
    
    Alex Brainman's avatar
    Alex Brainman committed
    		return "", e
    	}
    
    Russ Cox's avatar
    Russ Cox committed
    	return string(utf16.Decode(b[0:n])), nil
    
    Russ Cox's avatar
    Russ Cox committed
    func Ftruncate(fd Handle, length int64) (err error) {
    
    Alex Brainman's avatar
    Alex Brainman committed
    	curoffset, e := Seek(fd, 0, 1)
    
    Russ Cox's avatar
    Russ Cox committed
    	if e != nil {
    
    Alex Brainman's avatar
    Alex Brainman committed
    		return e
    	}
    	defer Seek(fd, curoffset, 0)
    
    Russ Cox's avatar
    Russ Cox committed
    	if e != nil {
    
    Alex Brainman's avatar
    Alex Brainman committed
    		return e
    	}
    
    	e = SetEndOfFile(fd)
    
    Russ Cox's avatar
    Russ Cox committed
    	if e != nil {
    
    Alex Brainman's avatar
    Alex Brainman committed
    		return e
    	}
    
    Russ Cox's avatar
    Russ Cox committed
    	return nil
    
    Russ Cox's avatar
    Russ Cox committed
    func Gettimeofday(tv *Timeval) (err error) {
    
    Alex Brainman's avatar
    Alex Brainman committed
    	var ft Filetime
    	GetSystemTimeAsFileTime(&ft)
    
    	*tv = NsecToTimeval(ft.Nanoseconds())
    
    Russ Cox's avatar
    Russ Cox committed
    	return nil
    
    Russ Cox's avatar
    Russ Cox committed
    func Pipe(p []Handle) (err error) {
    
    	if len(p) != 2 {
    		return EINVAL
    	}
    
    	var r, w Handle
    
    	e := CreatePipe(&r, &w, makeInheritSa(), 0)
    
    Russ Cox's avatar
    Russ Cox committed
    	if e != nil {
    
    	p[0] = r
    	p[1] = w
    
    Russ Cox's avatar
    Russ Cox committed
    	return nil
    
    Russ Cox's avatar
    Russ Cox committed
    func Utimes(path string, tv []Timeval) (err error) {
    
    	if len(tv) != 2 {
    		return EINVAL
    	}
    
    	pathp, e := UTF16PtrFromString(path)
    	if e != nil {
    		return e
    	}
    	h, e := CreateFile(pathp,
    
    		FILE_WRITE_ATTRIBUTES, FILE_SHARE_WRITE, nil,
    
    		OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, 0)
    
    Russ Cox's avatar
    Russ Cox committed
    	if e != nil {
    
    	defer Close(h)
    
    	a := NsecToFiletime(tv[0].Nanoseconds())
    	w := NsecToFiletime(tv[1].Nanoseconds())
    
    func UtimesNano(path string, ts []Timespec) (err error) {
    	if len(ts) != 2 {
    		return EINVAL
    	}
    	pathp, e := UTF16PtrFromString(path)
    	if e != nil {
    		return e
    	}
    	h, e := CreateFile(pathp,
    		FILE_WRITE_ATTRIBUTES, FILE_SHARE_WRITE, nil,
    
    		OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, 0)
    
    	if e != nil {
    		return e
    	}
    	defer Close(h)
    	a := NsecToFiletime(TimespecToNsec(ts[0]))
    	w := NsecToFiletime(TimespecToNsec(ts[1]))
    	return SetFileTime(h, nil, &a, &w)
    }
    
    
    Russ Cox's avatar
    Russ Cox committed
    func Fsync(fd Handle) (err error) {
    
    	return FlushFileBuffers(fd)
    
    Russ Cox's avatar
    Russ Cox committed
    func Chmod(path string, mode uint32) (err error) {
    
    	p, e := UTF16PtrFromString(path)
    	if e != nil {
    		return e
    	}
    
    	attrs, e := GetFileAttributes(p)
    
    Russ Cox's avatar
    Russ Cox committed
    	if e != nil {
    
    		return e
    	}
    	if mode&S_IWRITE != 0 {
    		attrs &^= FILE_ATTRIBUTE_READONLY
    	} else {
    		attrs |= FILE_ATTRIBUTE_READONLY
    	}
    	return SetFileAttributes(p, attrs)
    }
    
    
    func LoadCancelIoEx() error {
    	return procCancelIoEx.Find()
    }
    
    
    func LoadSetFileCompletionNotificationModes() error {
    	return procSetFileCompletionNotificationModes.Find()
    }
    
    
    const socket_error = uintptr(^uint32(0))
    
    
    //sys	WSAStartup(verreq uint32, data *WSAData) (sockerr error) = ws2_32.WSAStartup
    
    //sys	WSACleanup() (err error) [failretval==socket_error] = ws2_32.WSACleanup
    //sys	WSAIoctl(s Handle, iocc uint32, inbuf *byte, cbif uint32, outbuf *byte, cbob uint32, cbbr *uint32, overlapped *Overlapped, completionRoutine uintptr) (err error) [failretval==socket_error] = ws2_32.WSAIoctl
    
    Russ Cox's avatar
    Russ Cox committed
    //sys	socket(af int32, typ int32, protocol int32) (handle Handle, err error) [failretval==InvalidHandle] = ws2_32.socket
    
    //sys	Setsockopt(s Handle, level int32, optname int32, optval *byte, optlen int32) (err error) [failretval==socket_error] = ws2_32.setsockopt
    //sys	Getsockopt(s Handle, level int32, optname int32, optval *byte, optlen *int32) (err error) [failretval==socket_error] = ws2_32.getsockopt
    
    //sys	bind(s Handle, name unsafe.Pointer, namelen int32) (err error) [failretval==socket_error] = ws2_32.bind
    //sys	connect(s Handle, name unsafe.Pointer, namelen int32) (err error) [failretval==socket_error] = ws2_32.connect
    
    //sys	getsockname(s Handle, rsa *RawSockaddrAny, addrlen *int32) (err error) [failretval==socket_error] = ws2_32.getsockname
    //sys	getpeername(s Handle, rsa *RawSockaddrAny, addrlen *int32) (err error) [failretval==socket_error] = ws2_32.getpeername
    //sys	listen(s Handle, backlog int32) (err error) [failretval==socket_error] = ws2_32.listen
    //sys	shutdown(s Handle, how int32) (err error) [failretval==socket_error] = ws2_32.shutdown
    //sys	Closesocket(s Handle) (err error) [failretval==socket_error] = ws2_32.closesocket
    
    Russ Cox's avatar
    Russ Cox committed
    //sys	AcceptEx(ls Handle, as Handle, buf *byte, rxdatalen uint32, laddrlen uint32, raddrlen uint32, recvd *uint32, overlapped *Overlapped) (err error) = mswsock.AcceptEx
    
    //sys	GetAcceptExSockaddrs(buf *byte, rxdatalen uint32, laddrlen uint32, raddrlen uint32, lrsa **RawSockaddrAny, lrsalen *int32, rrsa **RawSockaddrAny, rrsalen *int32) = mswsock.GetAcceptExSockaddrs
    
    //sys	WSARecv(s Handle, bufs *WSABuf, bufcnt uint32, recvd *uint32, flags *uint32, overlapped *Overlapped, croutine *byte) (err error) [failretval==socket_error] = ws2_32.WSARecv
    //sys	WSASend(s Handle, bufs *WSABuf, bufcnt uint32, sent *uint32, flags uint32, overlapped *Overlapped, croutine *byte) (err error) [failretval==socket_error] = ws2_32.WSASend
    //sys	WSARecvFrom(s Handle, bufs *WSABuf, bufcnt uint32, recvd *uint32, flags *uint32,  from *RawSockaddrAny, fromlen *int32, overlapped *Overlapped, croutine *byte) (err error) [failretval==socket_error] = ws2_32.WSARecvFrom
    //sys	WSASendTo(s Handle, bufs *WSABuf, bufcnt uint32, sent *uint32, flags uint32, to *RawSockaddrAny, tolen int32,  overlapped *Overlapped, croutine *byte) (err error) [failretval==socket_error] = ws2_32.WSASendTo
    
    Russ Cox's avatar
    Russ Cox committed
    //sys	GetHostByName(name string) (h *Hostent, err error) [failretval==nil] = ws2_32.gethostbyname
    //sys	GetServByName(name string, proto string) (s *Servent, err error) [failretval==nil] = ws2_32.getservbyname
    
    //sys	Ntohs(netshort uint16) (u uint16) = ws2_32.ntohs
    
    Russ Cox's avatar
    Russ Cox committed
    //sys	GetProtoByName(name string) (p *Protoent, err error) [failretval==nil] = ws2_32.getprotobyname
    
    //sys	DnsQuery(name string, qtype uint16, options uint32, extra *byte, qrs **DNSRecord, pr *byte) (status error) = dnsapi.DnsQuery_W
    
    //sys	DnsRecordListFree(rl *DNSRecord, freetype uint32) = dnsapi.DnsRecordListFree
    
    //sys	DnsNameCompare(name1 *uint16, name2 *uint16) (same bool) = dnsapi.DnsNameCompare_W
    
    //sys	GetAddrInfoW(nodename *uint16, servicename *uint16, hints *AddrinfoW, result **AddrinfoW) (sockerr error) = ws2_32.GetAddrInfoW
    //sys	FreeAddrInfoW(addrinfo *AddrinfoW) = ws2_32.FreeAddrInfoW
    
    //sys	GetIfEntry(pIfRow *MibIfRow) (errcode error) = iphlpapi.GetIfEntry
    //sys	GetAdaptersInfo(ai *IpAdapterInfo, ol *uint32) (errcode error) = iphlpapi.GetAdaptersInfo
    
    //sys	SetFileCompletionNotificationModes(handle Handle, flags uint8) (err error) = kernel32.SetFileCompletionNotificationModes
    //sys	WSAEnumProtocols(protocols *int32, protocolBuffer *WSAProtocolInfo, bufferLength *uint32) (n int32, err error) [failretval==-1] = ws2_32.WSAEnumProtocolsW
    
    // For testing: clients can set this flag to force
    // creation of IPv6 sockets to return EAFNOSUPPORT.
    var SocketDisableIPv6 bool
    
    
    type RawSockaddrInet4 struct {
    	Family uint16
    	Port   uint16
    	Addr   [4]byte /* in_addr */
    	Zero   [8]uint8
    }
    
    
    type RawSockaddrInet6 struct {
    	Family   uint16
    	Port     uint16
    	Flowinfo uint32
    	Addr     [16]byte /* in6_addr */
    	Scope_id uint32
    }
    
    
    type RawSockaddr struct {
    	Family uint16
    	Data   [14]int8
    }
    
    type RawSockaddrAny struct {
    	Addr RawSockaddr
    
    	sockaddr() (ptr unsafe.Pointer, len int32, err error) // lowercase; only we can define Sockaddrs
    
    }
    
    type SockaddrInet4 struct {
    	Port int
    	Addr [4]byte
    	raw  RawSockaddrInet4
    }
    
    
    func (sa *SockaddrInet4) sockaddr() (unsafe.Pointer, int32, error) {
    
    	if sa.Port < 0 || sa.Port > 0xFFFF {
    
    	}
    	sa.raw.Family = AF_INET
    	p := (*[2]byte)(unsafe.Pointer(&sa.raw.Port))
    	p[0] = byte(sa.Port >> 8)
    	p[1] = byte(sa.Port)
    
    	return unsafe.Pointer(&sa.raw), int32(unsafe.Sizeof(sa.raw)), nil
    
    }
    
    type SockaddrInet6 struct {
    
    	Port   int
    	ZoneId uint32
    	Addr   [16]byte
    
    	raw    RawSockaddrInet6
    
    func (sa *SockaddrInet6) sockaddr() (unsafe.Pointer, int32, error) {
    
    	if sa.Port < 0 || sa.Port > 0xFFFF {
    
    	}
    	sa.raw.Family = AF_INET6
    	p := (*[2]byte)(unsafe.Pointer(&sa.raw.Port))
    	p[0] = byte(sa.Port >> 8)
    	p[1] = byte(sa.Port)
    	sa.raw.Scope_id = sa.ZoneId
    
    	return unsafe.Pointer(&sa.raw), int32(unsafe.Sizeof(sa.raw)), nil
    
    type RawSockaddrUnix struct {
    	Family uint16
    	Path   [UNIX_PATH_MAX]int8
    }
    
    
    type SockaddrUnix struct {
    	Name string
    
    	raw  RawSockaddrUnix
    
    func (sa *SockaddrUnix) sockaddr() (unsafe.Pointer, int32, error) {
    
    	name := sa.Name
    	n := len(name)
    	if n > len(sa.raw.Path) {
    		return nil, 0, EINVAL
    	}
    	if n == len(sa.raw.Path) && name[0] != '@' {
    		return nil, 0, EINVAL
    	}
    	sa.raw.Family = AF_UNIX
    	for i := 0; i < n; i++ {
    		sa.raw.Path[i] = int8(name[i])
    	}
    	// length is family (uint16), name, NUL.
    	sl := int32(2)
    	if n > 0 {
    		sl += int32(n) + 1
    	}
    	if sa.raw.Path[0] == '@' {
    		sa.raw.Path[0] = 0
    		// Don't count trailing NUL for abstract address.
    		sl--
    	}
    
    	return unsafe.Pointer(&sa.raw), sl, nil
    
    Russ Cox's avatar
    Russ Cox committed
    func (rsa *RawSockaddrAny) Sockaddr() (Sockaddr, error) {
    
    	switch rsa.Addr.Family {
    	case AF_UNIX:
    
    		pp := (*RawSockaddrUnix)(unsafe.Pointer(rsa))
    		sa := new(SockaddrUnix)
    		if pp.Path[0] == 0 {
    			// "Abstract" Unix domain socket.
    			// Rewrite leading NUL as @ for textual display.
    			// (This is the standard convention.)
    			// Not friendly to overwrite in place,
    			// but the callers below don't care.
    			pp.Path[0] = '@'
    		}
    
    		// Assume path ends at NUL.
    		// This is not technically the Linux semantics for
    		// abstract Unix domain sockets--they are supposed
    		// to be uninterpreted fixed-size binary blobs--but
    		// everyone uses this convention.
    		n := 0
    		for n < len(pp.Path) && pp.Path[n] != 0 {
    			n++
    		}
    
    		sa.Name = string(unsafe.Slice((*byte)(unsafe.Pointer(&pp.Path[0])), n))
    
    
    	case AF_INET:
    		pp := (*RawSockaddrInet4)(unsafe.Pointer(rsa))
    		sa := new(SockaddrInet4)
    		p := (*[2]byte)(unsafe.Pointer(&pp.Port))
    		sa.Port = int(p[0])<<8 + int(p[1])
    
    Russ Cox's avatar
    Russ Cox committed
    		return sa, nil
    
    		pp := (*RawSockaddrInet6)(unsafe.Pointer(rsa))
    		sa := new(SockaddrInet6)
    		p := (*[2]byte)(unsafe.Pointer(&pp.Port))
    		sa.Port = int(p[0])<<8 + int(p[1])
    		sa.ZoneId = pp.Scope_id
    
    		return sa, nil
    
    Russ Cox's avatar
    Russ Cox committed
    func Socket(domain, typ, proto int) (fd Handle, err error) {
    
    	if domain == AF_INET6 && SocketDisableIPv6 {
    
    		return InvalidHandle, EAFNOSUPPORT
    
    Russ Cox's avatar
    Russ Cox committed
    	return socket(int32(domain), int32(typ), int32(proto))
    
    Russ Cox's avatar
    Russ Cox committed
    func SetsockoptInt(fd Handle, level, opt int, value int) (err error) {
    
    Russ Cox's avatar
    Russ Cox committed
    	return Setsockopt(fd, int32(level), int32(opt), (*byte)(unsafe.Pointer(&v)), int32(unsafe.Sizeof(v)))
    
    Russ Cox's avatar
    Russ Cox committed
    func Bind(fd Handle, sa Sockaddr) (err error) {
    
    	ptr, n, err := sa.sockaddr()
    
    Russ Cox's avatar
    Russ Cox committed
    	if err != nil {
    
    	return bind(fd, ptr, n)
    
    Russ Cox's avatar
    Russ Cox committed
    func Connect(fd Handle, sa Sockaddr) (err error) {
    
    	ptr, n, err := sa.sockaddr()
    
    Russ Cox's avatar
    Russ Cox committed
    	if err != nil {
    
    	return connect(fd, ptr, n)
    
    Russ Cox's avatar
    Russ Cox committed
    func Getsockname(fd Handle) (sa Sockaddr, err error) {
    
    	var rsa RawSockaddrAny
    	l := int32(unsafe.Sizeof(rsa))
    
    Russ Cox's avatar
    Russ Cox committed
    	if err = getsockname(fd, &rsa, &l); err != nil {
    
    Russ Cox's avatar
    Russ Cox committed
    func Getpeername(fd Handle) (sa Sockaddr, err error) {
    
    	var rsa RawSockaddrAny
    	l := int32(unsafe.Sizeof(rsa))
    
    Russ Cox's avatar
    Russ Cox committed
    	if err = getpeername(fd, &rsa, &l); err != nil {
    
    Russ Cox's avatar
    Russ Cox committed
    func Listen(s Handle, n int) (err error) {
    	return listen(s, int32(n))
    
    Russ Cox's avatar
    Russ Cox committed
    func Shutdown(fd Handle, how int) (err error) {
    	return shutdown(fd, int32(how))
    
    Russ Cox's avatar
    Russ Cox committed
    func WSASendto(s Handle, bufs *WSABuf, bufcnt uint32, sent *uint32, flags uint32, to Sockaddr, overlapped *Overlapped, croutine *byte) (err error) {
    
    	var rsa unsafe.Pointer
    	var len int32
    	if to != nil {
    		rsa, len, err = to.sockaddr()
    		if err != nil {
    			return err
    		}
    
    	r1, _, e1 := Syscall9(procWSASendTo.Addr(), 9, uintptr(s), uintptr(unsafe.Pointer(bufs)), uintptr(bufcnt), uintptr(unsafe.Pointer(sent)), uintptr(flags), uintptr(unsafe.Pointer(rsa)), uintptr(len), uintptr(unsafe.Pointer(overlapped)), uintptr(unsafe.Pointer(croutine)))
    
    	if r1 == socket_error {
    		if e1 != 0 {
    			err = errnoErr(e1)
    		} else {
    			err = EINVAL
    		}
    	}
    	return err
    }
    
    
    func wsaSendtoInet4(s Handle, bufs *WSABuf, bufcnt uint32, sent *uint32, flags uint32, to *SockaddrInet4, overlapped *Overlapped, croutine *byte) (err error) {
    
    	rsa, len, err := to.sockaddr()
    	if err != nil {
    		return err
    	}
    	r1, _, e1 := Syscall9(procWSASendTo.Addr(), 9, uintptr(s), uintptr(unsafe.Pointer(bufs)), uintptr(bufcnt), uintptr(unsafe.Pointer(sent)), uintptr(flags), uintptr(unsafe.Pointer(rsa)), uintptr(len), uintptr(unsafe.Pointer(overlapped)), uintptr(unsafe.Pointer(croutine)))
    	if r1 == socket_error {
    		if e1 != 0 {
    			err = errnoErr(e1)
    		} else {
    			err = EINVAL
    		}
    	}
    	return err
    }
    
    
    func wsaSendtoInet6(s Handle, bufs *WSABuf, bufcnt uint32, sent *uint32, flags uint32, to *SockaddrInet6, overlapped *Overlapped, croutine *byte) (err error) {
    
    	rsa, len, err := to.sockaddr()
    	if err != nil {
    		return err
    	}
    	r1, _, e1 := Syscall9(procWSASendTo.Addr(), 9, uintptr(s), uintptr(unsafe.Pointer(bufs)), uintptr(bufcnt), uintptr(unsafe.Pointer(sent)), uintptr(flags), uintptr(unsafe.Pointer(rsa)), uintptr(len), uintptr(unsafe.Pointer(overlapped)), uintptr(unsafe.Pointer(croutine)))
    
    	if r1 == socket_error {
    		if e1 != 0 {
    			err = errnoErr(e1)
    		} else {
    			err = EINVAL
    		}
    	}
    	return err
    
    func LoadGetAddrInfo() error {
    	return procGetAddrInfoW.Find()
    }
    
    
    var connectExFunc struct {
    	once sync.Once
    	addr uintptr
    	err  error
    }
    
    func LoadConnectEx() error {
    	connectExFunc.once.Do(func() {
    		var s Handle
    		s, connectExFunc.err = Socket(AF_INET, SOCK_STREAM, IPPROTO_TCP)
    		if connectExFunc.err != nil {
    			return
    		}
    		defer CloseHandle(s)
    		var n uint32