mirror of
				https://github.com/postgres/postgres.git
				synced 2025-10-24 01:29:19 +03:00 
			
		
		
		
	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 5acd417c8, so
back-patch to all supported branches.
Discussion: https://postgr.es/m/20170627163757.25161.528@wrigleys.postgresql.org
			
			
This commit is contained in:
		| @@ -1615,7 +1615,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; | ||||||
|  |  | ||||||
| @@ -1627,7 +1627,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) | ||||||
| @@ -1635,7 +1636,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) | ||||||
| @@ -1643,7 +1654,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; | ||||||
| @@ -1662,7 +1673,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; | ||||||
|  |  | ||||||
| @@ -1681,14 +1693,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) | ||||||
| 	{ | 	{ | ||||||
| @@ -1701,7 +1723,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"); | ||||||
| @@ -1751,7 +1773,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; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1280,7 +1280,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) | ||||||
| 	{ | 	{ | ||||||
| @@ -1291,9 +1292,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 */ | ||||||
| 	/* Darwin 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) | ||||||
| 	{ | 	{ | ||||||
| @@ -1304,7 +1316,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; | ||||||
| @@ -1372,7 +1383,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, | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user