2017-10-07 00:16:47 +00:00
|
|
|
// Copyright 2009 The Go Authors. All rights reserved.
|
2012-03-27 23:13:14 +00:00
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
2019-06-02 15:48:37 +00:00
|
|
|
// +build aix darwin dragonfly freebsd hurd linux netbsd openbsd solaris windows
|
2012-03-27 23:13:14 +00:00
|
|
|
|
|
|
|
package net
|
|
|
|
|
|
|
|
import (
|
2019-06-02 15:48:37 +00:00
|
|
|
"internal/bytealg"
|
2018-12-28 15:30:48 +00:00
|
|
|
"runtime"
|
2012-03-27 23:13:14 +00:00
|
|
|
"syscall"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Boolean to int.
|
|
|
|
func boolint(b bool) int {
|
|
|
|
if b {
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func ipv4AddrToInterface(ip IP) (*Interface, error) {
|
|
|
|
ift, err := Interfaces()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for _, ifi := range ift {
|
|
|
|
ifat, err := ifi.Addrs()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for _, ifa := range ifat {
|
|
|
|
switch v := ifa.(type) {
|
|
|
|
case *IPAddr:
|
|
|
|
if ip.Equal(v.IP) {
|
|
|
|
return &ifi, nil
|
|
|
|
}
|
|
|
|
case *IPNet:
|
|
|
|
if ip.Equal(v.IP) {
|
|
|
|
return &ifi, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ip.Equal(IPv4zero) {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
return nil, errNoSuchInterface
|
|
|
|
}
|
|
|
|
|
|
|
|
func interfaceToIPv4Addr(ifi *Interface) (IP, error) {
|
|
|
|
if ifi == nil {
|
|
|
|
return IPv4zero, nil
|
|
|
|
}
|
|
|
|
ifat, err := ifi.Addrs()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for _, ifa := range ifat {
|
|
|
|
switch v := ifa.(type) {
|
|
|
|
case *IPAddr:
|
|
|
|
if v.IP.To4() != nil {
|
|
|
|
return v.IP, nil
|
|
|
|
}
|
|
|
|
case *IPNet:
|
|
|
|
if v.IP.To4() != nil {
|
|
|
|
return v.IP, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil, errNoSuchInterface
|
|
|
|
}
|
|
|
|
|
|
|
|
func setIPv4MreqToInterface(mreq *syscall.IPMreq, ifi *Interface) error {
|
|
|
|
if ifi == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
ifat, err := ifi.Addrs()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
for _, ifa := range ifat {
|
|
|
|
switch v := ifa.(type) {
|
|
|
|
case *IPAddr:
|
|
|
|
if a := v.IP.To4(); a != nil {
|
|
|
|
copy(mreq.Interface[:], a)
|
|
|
|
goto done
|
|
|
|
}
|
|
|
|
case *IPNet:
|
|
|
|
if a := v.IP.To4(); a != nil {
|
|
|
|
copy(mreq.Interface[:], a)
|
|
|
|
goto done
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
done:
|
2019-06-02 15:48:37 +00:00
|
|
|
if bytealg.Equal(mreq.Multiaddr[:], IPv4zero.To4()) {
|
2012-03-27 23:13:14 +00:00
|
|
|
return errNoSuchMulticastInterface
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func setReadBuffer(fd *netFD, bytes int) error {
|
2018-12-28 15:30:48 +00:00
|
|
|
err := fd.pfd.SetsockoptInt(syscall.SOL_SOCKET, syscall.SO_RCVBUF, bytes)
|
|
|
|
runtime.KeepAlive(fd)
|
|
|
|
return wrapSyscallError("setsockopt", err)
|
2012-03-27 23:13:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func setWriteBuffer(fd *netFD, bytes int) error {
|
2018-12-28 15:30:48 +00:00
|
|
|
err := fd.pfd.SetsockoptInt(syscall.SOL_SOCKET, syscall.SO_SNDBUF, bytes)
|
|
|
|
runtime.KeepAlive(fd)
|
|
|
|
return wrapSyscallError("setsockopt", err)
|
2012-03-27 23:13:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func setKeepAlive(fd *netFD, keepalive bool) error {
|
2018-12-28 15:30:48 +00:00
|
|
|
err := fd.pfd.SetsockoptInt(syscall.SOL_SOCKET, syscall.SO_KEEPALIVE, boolint(keepalive))
|
|
|
|
runtime.KeepAlive(fd)
|
|
|
|
return wrapSyscallError("setsockopt", err)
|
2012-03-27 23:13:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func setLinger(fd *netFD, sec int) error {
|
|
|
|
var l syscall.Linger
|
|
|
|
if sec >= 0 {
|
|
|
|
l.Onoff = 1
|
|
|
|
l.Linger = int32(sec)
|
|
|
|
} else {
|
|
|
|
l.Onoff = 0
|
|
|
|
l.Linger = 0
|
|
|
|
}
|
2018-12-28 15:30:48 +00:00
|
|
|
err := fd.pfd.SetsockoptLinger(syscall.SOL_SOCKET, syscall.SO_LINGER, &l)
|
|
|
|
runtime.KeepAlive(fd)
|
|
|
|
return wrapSyscallError("setsockopt", err)
|
2012-03-27 23:13:14 +00:00
|
|
|
}
|