mirror of
https://github.com/autc04/Retro68.git
synced 2024-12-12 11:29:30 +00:00
140 lines
3.4 KiB
Go
140 lines
3.4 KiB
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.
|
|
|
|
// +build darwin dragonfly freebsd linux netbsd openbsd solaris
|
|
|
|
package net
|
|
|
|
import (
|
|
"os"
|
|
"syscall"
|
|
)
|
|
|
|
func newFileFD(f *os.File) (*netFD, error) {
|
|
fd, err := dupCloseOnExec(int(f.Fd()))
|
|
if err != nil {
|
|
return nil, os.NewSyscallError("dup", err)
|
|
}
|
|
|
|
if err = syscall.SetNonblock(fd, true); err != nil {
|
|
closesocket(fd)
|
|
return nil, err
|
|
}
|
|
|
|
sotype, err := syscall.GetsockoptInt(fd, syscall.SOL_SOCKET, syscall.SO_TYPE)
|
|
if err != nil {
|
|
closesocket(fd)
|
|
return nil, os.NewSyscallError("getsockopt", err)
|
|
}
|
|
|
|
family := syscall.AF_UNSPEC
|
|
toAddr := sockaddrToTCP
|
|
lsa, _ := syscall.Getsockname(fd)
|
|
switch lsa.(type) {
|
|
default:
|
|
closesocket(fd)
|
|
return nil, syscall.EINVAL
|
|
case *syscall.SockaddrInet4:
|
|
family = syscall.AF_INET
|
|
if sotype == syscall.SOCK_DGRAM {
|
|
toAddr = sockaddrToUDP
|
|
} else if sotype == syscall.SOCK_RAW {
|
|
toAddr = sockaddrToIP
|
|
}
|
|
case *syscall.SockaddrInet6:
|
|
family = syscall.AF_INET6
|
|
if sotype == syscall.SOCK_DGRAM {
|
|
toAddr = sockaddrToUDP
|
|
} else if sotype == syscall.SOCK_RAW {
|
|
toAddr = sockaddrToIP
|
|
}
|
|
case *syscall.SockaddrUnix:
|
|
family = syscall.AF_UNIX
|
|
toAddr = sockaddrToUnix
|
|
if sotype == syscall.SOCK_DGRAM {
|
|
toAddr = sockaddrToUnixgram
|
|
} else if sotype == syscall.SOCK_SEQPACKET {
|
|
toAddr = sockaddrToUnixpacket
|
|
}
|
|
}
|
|
laddr := toAddr(lsa)
|
|
rsa, _ := syscall.Getpeername(fd)
|
|
raddr := toAddr(rsa)
|
|
|
|
netfd, err := newFD(fd, family, sotype, laddr.Network())
|
|
if err != nil {
|
|
closesocket(fd)
|
|
return nil, err
|
|
}
|
|
if err := netfd.init(); err != nil {
|
|
netfd.Close()
|
|
return nil, err
|
|
}
|
|
netfd.setAddr(laddr, raddr)
|
|
return netfd, nil
|
|
}
|
|
|
|
// FileConn returns a copy of the network connection corresponding to
|
|
// the open file f. It is the caller's responsibility to close f when
|
|
// finished. Closing c does not affect f, and closing f does not
|
|
// affect c.
|
|
func FileConn(f *os.File) (c Conn, err error) {
|
|
fd, err := newFileFD(f)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
switch fd.laddr.(type) {
|
|
case *TCPAddr:
|
|
return newTCPConn(fd), nil
|
|
case *UDPAddr:
|
|
return newUDPConn(fd), nil
|
|
case *IPAddr:
|
|
return newIPConn(fd), nil
|
|
case *UnixAddr:
|
|
return newUnixConn(fd), nil
|
|
}
|
|
fd.Close()
|
|
return nil, syscall.EINVAL
|
|
}
|
|
|
|
// FileListener returns a copy of the network listener corresponding
|
|
// to the open file f. It is the caller's responsibility to close l
|
|
// when finished. Closing l does not affect f, and closing f does not
|
|
// affect l.
|
|
func FileListener(f *os.File) (l Listener, err error) {
|
|
fd, err := newFileFD(f)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
switch laddr := fd.laddr.(type) {
|
|
case *TCPAddr:
|
|
return &TCPListener{fd}, nil
|
|
case *UnixAddr:
|
|
return &UnixListener{fd, laddr.Name}, nil
|
|
}
|
|
fd.Close()
|
|
return nil, syscall.EINVAL
|
|
}
|
|
|
|
// FilePacketConn returns a copy of the packet network connection
|
|
// corresponding to the open file f. It is the caller's
|
|
// responsibility to close f when finished. Closing c does not affect
|
|
// f, and closing f does not affect c.
|
|
func FilePacketConn(f *os.File) (c PacketConn, err error) {
|
|
fd, err := newFileFD(f)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
switch fd.laddr.(type) {
|
|
case *UDPAddr:
|
|
return newUDPConn(fd), nil
|
|
case *IPAddr:
|
|
return newIPConn(fd), nil
|
|
case *UnixAddr:
|
|
return newUnixConn(fd), nil
|
|
}
|
|
fd.Close()
|
|
return nil, syscall.EINVAL
|
|
}
|