blob: 44eb9cd09e7558ecebe152175c5eea238a7c25cc [file] [log] [blame]
// 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.
package net
import (
"runtime"
"syscall"
"time"
)
func setKeepAliveIdle(fd *netFD, d time.Duration) error {
if d == 0 {
d = defaultTCPKeepAliveIdle
} else if d < 0 {
return nil
}
// The kernel expects milliseconds so round to next highest
// millisecond.
msecs := int(roundDurationUp(d, time.Millisecond))
// TODO(panjf2000): the system call here always returns an error of invalid argument,
// this was never discovered due to the lack of tests for TCP keep-alive on various
// platforms in Go's test suite. Try to dive deep and figure out the reason later.
// Check out https://go.dev/issue/64251 for more details.
err := fd.pfd.SetsockoptInt(syscall.IPPROTO_TCP, syscall.TCP_KEEPALIVE_THRESHOLD, msecs)
runtime.KeepAlive(fd)
return wrapSyscallError("setsockopt", err)
}
func setKeepAliveInterval(_ *netFD, d time.Duration) error {
if d < 0 {
return nil
}
// Normally we'd do
// syscall.SetsockoptInt(fd.sysfd, syscall.IPPROTO_TCP, syscall.TCP_KEEPINTVL, secs)
// here, but we can't because Solaris does not have TCP_KEEPINTVL.
// Solaris has TCP_KEEPALIVE_ABORT_THRESHOLD, but it's not the same
// thing, it refers to the total time until aborting (not between
// probes), and it uses an exponential backoff algorithm instead of
// waiting the same time between probes. We can't hope for the best
// and do it anyway, like on Darwin, because Solaris might eventually
// allocate a constant with a different meaning for the value of
// TCP_KEEPINTVL on illumos.
return syscall.ENOPROTOOPT
}
func setKeepAliveCount(_ *netFD, n int) error {
if n < 0 {
return nil
}
return syscall.ENOPROTOOPT
}