Commit f0256c77 authored by Tom Lane's avatar Tom Lane

Support tcp_keepalives_idle option on Solaris.

Turns out that the socket option for this is named TCP_KEEPALIVE_THRESHOLD,
at least according to the tcp(7P) man page for Solaris 11.  (But since that
text refers to "SunOS", it's likely pretty ancient.)  It appears that the
symbol TCP_KEEPALIVE does get defined on that platform, but it doesn't
seem to represent a valid protocol-level socket option.  This leads to
bleats in the postmaster log, and no tcp_keepalives_idle functionality.

Per bug #14720 from Andrey Lizenko, as well as an earlier report from
Dhiraj Chawla that nobody had followed up on.  The issue's been there
since we added the TCP_KEEPALIVE code path in commit 5acd417c, so
back-patch to all supported branches.

Discussion: https://postgr.es/m/20170627163757.25161.528@wrigleys.postgresql.org
parent 9c7dc892
...@@ -1676,7 +1676,7 @@ pq_setkeepaliveswin32(Port *port, int idle, int interval) ...@@ -1676,7 +1676,7 @@ pq_setkeepaliveswin32(Port *port, int idle, int interval)
int int
pq_getkeepalivesidle(Port *port) pq_getkeepalivesidle(Port *port)
{ {
#if defined(TCP_KEEPIDLE) || defined(TCP_KEEPALIVE) || defined(WIN32) #if defined(TCP_KEEPIDLE) || defined(TCP_KEEPALIVE_THRESHOLD) || defined(TCP_KEEPALIVE) || defined(WIN32)
if (port == NULL || IS_AF_UNIX(port->laddr.addr.ss_family)) if (port == NULL || IS_AF_UNIX(port->laddr.addr.ss_family))
return 0; return 0;
...@@ -1688,7 +1688,8 @@ pq_getkeepalivesidle(Port *port) ...@@ -1688,7 +1688,8 @@ pq_getkeepalivesidle(Port *port)
#ifndef WIN32 #ifndef WIN32
ACCEPT_TYPE_ARG3 size = sizeof(port->default_keepalives_idle); ACCEPT_TYPE_ARG3 size = sizeof(port->default_keepalives_idle);
#ifdef TCP_KEEPIDLE #if defined(TCP_KEEPIDLE)
/* TCP_KEEPIDLE is the name of this option on Linux and *BSD */
if (getsockopt(port->sock, IPPROTO_TCP, TCP_KEEPIDLE, if (getsockopt(port->sock, IPPROTO_TCP, TCP_KEEPIDLE,
(char *) &port->default_keepalives_idle, (char *) &port->default_keepalives_idle,
&size) < 0) &size) < 0)
...@@ -1696,7 +1697,17 @@ pq_getkeepalivesidle(Port *port) ...@@ -1696,7 +1697,17 @@ pq_getkeepalivesidle(Port *port)
elog(LOG, "getsockopt(TCP_KEEPIDLE) failed: %m"); elog(LOG, "getsockopt(TCP_KEEPIDLE) failed: %m");
port->default_keepalives_idle = -1; /* don't know */ port->default_keepalives_idle = -1; /* don't know */
} }
#else #elif defined(TCP_KEEPALIVE_THRESHOLD)
/* TCP_KEEPALIVE_THRESHOLD is the name of this option on Solaris */
if (getsockopt(port->sock, IPPROTO_TCP, TCP_KEEPALIVE_THRESHOLD,
(char *) &port->default_keepalives_idle,
&size) < 0)
{
elog(LOG, "getsockopt(TCP_KEEPALIVE_THRESHOLD) failed: %m");
port->default_keepalives_idle = -1; /* don't know */
}
#else /* must have TCP_KEEPALIVE */
/* TCP_KEEPALIVE is the name of this option on macOS */
if (getsockopt(port->sock, IPPROTO_TCP, TCP_KEEPALIVE, if (getsockopt(port->sock, IPPROTO_TCP, TCP_KEEPALIVE,
(char *) &port->default_keepalives_idle, (char *) &port->default_keepalives_idle,
&size) < 0) &size) < 0)
...@@ -1704,7 +1715,7 @@ pq_getkeepalivesidle(Port *port) ...@@ -1704,7 +1715,7 @@ pq_getkeepalivesidle(Port *port)
elog(LOG, "getsockopt(TCP_KEEPALIVE) failed: %m"); elog(LOG, "getsockopt(TCP_KEEPALIVE) failed: %m");
port->default_keepalives_idle = -1; /* don't know */ port->default_keepalives_idle = -1; /* don't know */
} }
#endif /* TCP_KEEPIDLE */ #endif /* KEEPIDLE/KEEPALIVE_THRESHOLD/KEEPALIVE */
#else /* WIN32 */ #else /* WIN32 */
/* We can't get the defaults on Windows, so return "don't know" */ /* We can't get the defaults on Windows, so return "don't know" */
port->default_keepalives_idle = -1; port->default_keepalives_idle = -1;
...@@ -1723,7 +1734,8 @@ pq_setkeepalivesidle(int idle, Port *port) ...@@ -1723,7 +1734,8 @@ pq_setkeepalivesidle(int idle, Port *port)
if (port == NULL || IS_AF_UNIX(port->laddr.addr.ss_family)) if (port == NULL || IS_AF_UNIX(port->laddr.addr.ss_family))
return STATUS_OK; return STATUS_OK;
#if defined(TCP_KEEPIDLE) || defined(TCP_KEEPALIVE) || defined(SIO_KEEPALIVE_VALS) /* check SIO_KEEPALIVE_VALS here, not just WIN32, as some toolchains lack it */
#if defined(TCP_KEEPIDLE) || defined(TCP_KEEPALIVE_THRESHOLD) || defined(TCP_KEEPALIVE) || defined(SIO_KEEPALIVE_VALS)
if (idle == port->keepalives_idle) if (idle == port->keepalives_idle)
return STATUS_OK; return STATUS_OK;
...@@ -1742,14 +1754,24 @@ pq_setkeepalivesidle(int idle, Port *port) ...@@ -1742,14 +1754,24 @@ pq_setkeepalivesidle(int idle, Port *port)
if (idle == 0) if (idle == 0)
idle = port->default_keepalives_idle; idle = port->default_keepalives_idle;
#ifdef TCP_KEEPIDLE #if defined(TCP_KEEPIDLE)
/* TCP_KEEPIDLE is the name of this option on Linux and *BSD */
if (setsockopt(port->sock, IPPROTO_TCP, TCP_KEEPIDLE, if (setsockopt(port->sock, IPPROTO_TCP, TCP_KEEPIDLE,
(char *) &idle, sizeof(idle)) < 0) (char *) &idle, sizeof(idle)) < 0)
{ {
elog(LOG, "setsockopt(TCP_KEEPIDLE) failed: %m"); elog(LOG, "setsockopt(TCP_KEEPIDLE) failed: %m");
return STATUS_ERROR; return STATUS_ERROR;
} }
#else #elif defined(TCP_KEEPALIVE_THRESHOLD)
/* TCP_KEEPALIVE_THRESHOLD is the name of this option on Solaris */
if (setsockopt(port->sock, IPPROTO_TCP, TCP_KEEPALIVE_THRESHOLD,
(char *) &idle, sizeof(idle)) < 0)
{
elog(LOG, "setsockopt(TCP_KEEPALIVE_THRESHOLD) failed: %m");
return STATUS_ERROR;
}
#else /* must have TCP_KEEPALIVE */
/* TCP_KEEPALIVE is the name of this option on macOS */
if (setsockopt(port->sock, IPPROTO_TCP, TCP_KEEPALIVE, if (setsockopt(port->sock, IPPROTO_TCP, TCP_KEEPALIVE,
(char *) &idle, sizeof(idle)) < 0) (char *) &idle, sizeof(idle)) < 0)
{ {
...@@ -1762,7 +1784,7 @@ pq_setkeepalivesidle(int idle, Port *port) ...@@ -1762,7 +1784,7 @@ pq_setkeepalivesidle(int idle, Port *port)
#else /* WIN32 */ #else /* WIN32 */
return pq_setkeepaliveswin32(port, idle, port->keepalives_interval); return pq_setkeepaliveswin32(port, idle, port->keepalives_interval);
#endif #endif
#else /* TCP_KEEPIDLE || SIO_KEEPALIVE_VALS */ #else /* no way to set it */
if (idle != 0) if (idle != 0)
{ {
elog(LOG, "setting the keepalive idle time is not supported"); elog(LOG, "setting the keepalive idle time is not supported");
...@@ -1812,7 +1834,7 @@ pq_setkeepalivesinterval(int interval, Port *port) ...@@ -1812,7 +1834,7 @@ pq_setkeepalivesinterval(int interval, Port *port)
if (port == NULL || IS_AF_UNIX(port->laddr.addr.ss_family)) if (port == NULL || IS_AF_UNIX(port->laddr.addr.ss_family))
return STATUS_OK; return STATUS_OK;
#if defined(TCP_KEEPINTVL) || defined (SIO_KEEPALIVE_VALS) #if defined(TCP_KEEPINTVL) || defined(SIO_KEEPALIVE_VALS)
if (interval == port->keepalives_interval) if (interval == port->keepalives_interval)
return STATUS_OK; return STATUS_OK;
......
...@@ -1470,7 +1470,8 @@ setKeepalivesIdle(PGconn *conn) ...@@ -1470,7 +1470,8 @@ setKeepalivesIdle(PGconn *conn)
if (idle < 0) if (idle < 0)
idle = 0; idle = 0;
#ifdef TCP_KEEPIDLE #if defined(TCP_KEEPIDLE)
/* TCP_KEEPIDLE is the name of this option on Linux and *BSD */
if (setsockopt(conn->sock, IPPROTO_TCP, TCP_KEEPIDLE, if (setsockopt(conn->sock, IPPROTO_TCP, TCP_KEEPIDLE,
(char *) &idle, sizeof(idle)) < 0) (char *) &idle, sizeof(idle)) < 0)
{ {
...@@ -1481,9 +1482,20 @@ setKeepalivesIdle(PGconn *conn) ...@@ -1481,9 +1482,20 @@ setKeepalivesIdle(PGconn *conn)
SOCK_STRERROR(SOCK_ERRNO, sebuf, sizeof(sebuf))); SOCK_STRERROR(SOCK_ERRNO, sebuf, sizeof(sebuf)));
return 0; return 0;
} }
#else #elif defined(TCP_KEEPALIVE_THRESHOLD)
#ifdef TCP_KEEPALIVE /* TCP_KEEPALIVE_THRESHOLD is the name of this option on Solaris */
/* macOS uses TCP_KEEPALIVE rather than TCP_KEEPIDLE */ if (setsockopt(conn->sock, IPPROTO_TCP, TCP_KEEPALIVE_THRESHOLD,
(char *) &idle, sizeof(idle)) < 0)
{
char sebuf[256];
appendPQExpBuffer(&conn->errorMessage,
libpq_gettext("setsockopt(TCP_KEEPALIVE_THRESHOLD) failed: %s\n"),
SOCK_STRERROR(SOCK_ERRNO, sebuf, sizeof(sebuf)));
return 0;
}
#elif defined(TCP_KEEPALIVE)
/* TCP_KEEPALIVE is the name of this option on macOS */
if (setsockopt(conn->sock, IPPROTO_TCP, TCP_KEEPALIVE, if (setsockopt(conn->sock, IPPROTO_TCP, TCP_KEEPALIVE,
(char *) &idle, sizeof(idle)) < 0) (char *) &idle, sizeof(idle)) < 0)
{ {
...@@ -1494,7 +1506,6 @@ setKeepalivesIdle(PGconn *conn) ...@@ -1494,7 +1506,6 @@ setKeepalivesIdle(PGconn *conn)
SOCK_STRERROR(SOCK_ERRNO, sebuf, sizeof(sebuf))); SOCK_STRERROR(SOCK_ERRNO, sebuf, sizeof(sebuf)));
return 0; return 0;
} }
#endif
#endif #endif
return 1; return 1;
...@@ -1562,7 +1573,7 @@ setKeepalivesCount(PGconn *conn) ...@@ -1562,7 +1573,7 @@ setKeepalivesCount(PGconn *conn)
return 1; return 1;
} }
#else /* Win32 */ #else /* WIN32 */
#ifdef SIO_KEEPALIVE_VALS #ifdef SIO_KEEPALIVE_VALS
/* /*
* Enable keepalives and set the keepalive values on Win32, * Enable keepalives and set the keepalive values on Win32,
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment