| // 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 |
| } |