blob: df2ddbd1131102b66c1a58026dbf38dcf41d1650 [file] [log] [blame] [edit]
// Copyright 2015 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.
//go:build !illumos
package net
import (
"internal/syscall/unix"
"runtime"
"syscall"
"time"
)
// Some macros of TCP Keep-Alive options on Solaris 11.4 may
// differ from those on OpenSolaris-based derivatives.
const (
sysTCP_KEEPIDLE = 0x1D
sysTCP_KEEPINTVL = 0x1E
sysTCP_KEEPCNT = 0x1F
)
func setKeepAliveIdle(fd *netFD, d time.Duration) error {
if !unix.SupportTCPKeepAliveIdleIntvlCNT() {
return setKeepAliveIdleAndIntervalAndCount(fd, d, -1, -1)
}
if d == 0 {
d = defaultTCPKeepAliveIdle
} else if d < 0 {
return nil
}
// The kernel expects seconds so round to next highest second.
secs := int(roundDurationUp(d, time.Second))
err := fd.pfd.SetsockoptInt(syscall.IPPROTO_TCP, sysTCP_KEEPIDLE, secs)
runtime.KeepAlive(fd)
return wrapSyscallError("setsockopt", err)
}
func setKeepAliveInterval(fd *netFD, d time.Duration) error {
if !unix.SupportTCPKeepAliveIdleIntvlCNT() {
return syscall.EPROTOTYPE
}
if d == 0 {
d = defaultTCPKeepAliveInterval
} else if d < 0 {
return nil
}
// The kernel expects seconds so round to next highest second.
secs := int(roundDurationUp(d, time.Second))
err := fd.pfd.SetsockoptInt(syscall.IPPROTO_TCP, sysTCP_KEEPINTVL, secs)
runtime.KeepAlive(fd)
return wrapSyscallError("setsockopt", err)
}
func setKeepAliveCount(fd *netFD, n int) error {
if !unix.SupportTCPKeepAliveIdleIntvlCNT() {
return syscall.EPROTOTYPE
}
if n == 0 {
n = defaultTCPKeepAliveCount
} else if n < 0 {
return nil
}
err := fd.pfd.SetsockoptInt(syscall.IPPROTO_TCP, sysTCP_KEEPCNT, n)
runtime.KeepAlive(fd)
return wrapSyscallError("setsockopt", err)
}
// setKeepAliveIdleAndIntervalAndCount serves for Solaris prior to 11.4 by simulating
// the TCP_KEEPIDLE, TCP_KEEPINTVL, and TCP_KEEPCNT with `TCP_KEEPALIVE_THRESHOLD` + `TCP_KEEPALIVE_ABORT_THRESHOLD`.
func setKeepAliveIdleAndIntervalAndCount(fd *netFD, idle, interval time.Duration, count int) error {
if idle == 0 {
idle = defaultTCPKeepAliveIdle
}
// The kernel expects milliseconds so round to next highest
// millisecond.
if idle > 0 {
msecs := int(roundDurationUp(idle, time.Millisecond))
err := fd.pfd.SetsockoptInt(syscall.IPPROTO_TCP, syscall.TCP_KEEPALIVE_THRESHOLD, msecs)
runtime.KeepAlive(fd)
if err != nil {
return wrapSyscallError("setsockopt", err)
}
}
if interval == 0 {
interval = defaultTCPKeepAliveInterval
}
if count == 0 {
count = defaultTCPKeepAliveCount
}
// TCP_KEEPINTVL and TCP_KEEPCNT are not available on Solaris
// prior to 11.4, so it's pointless to "leave it unchanged"
// with negative value for only one of them. On the other hand,
// setting both to negative values should pragmatically leave the
// TCP_KEEPALIVE_ABORT_THRESHOLD unchanged.
abortIdle := int(roundDurationUp(interval, time.Millisecond)) * count
if abortIdle < 0 {
return syscall.ENOPROTOOPT
}
if interval < 0 && count < 0 {
abortIdle = -1
}
if abortIdle > 0 {
// Note that the consequent probes will not be sent at equal intervals on Solaris,
// but will be sent using the exponential backoff algorithm.
err := fd.pfd.SetsockoptInt(syscall.IPPROTO_TCP, syscall.TCP_KEEPALIVE_ABORT_THRESHOLD, abortIdle)
runtime.KeepAlive(fd)
return wrapSyscallError("setsockopt", err)
}
return nil
}