diff options
author | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2020-06-09 13:47:44 +0100 |
---|---|---|
committer | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2020-06-09 13:47:44 +0100 |
commit | 488f72bad4fab719861b4641ff832f60729a5950 (patch) | |
tree | efb6714cb551c7611dffe0f25116cbab20b36a78 /src/libutil | |
parent | 4f69d50e5ef648c09b042d9bbca9a1b9106288b2 (diff) | |
download | rspamd-488f72bad4fab719861b4641ff832f60729a5950.tar.gz rspamd-488f72bad4fab719861b4641ff832f60729a5950.zip |
[Minor] Add flags to listen socket creation
Diffstat (limited to 'src/libutil')
-rw-r--r-- | src/libutil/addr.c | 51 | ||||
-rw-r--r-- | src/libutil/addr.h | 10 |
2 files changed, 49 insertions, 12 deletions
diff --git a/src/libutil/addr.c b/src/libutil/addr.c index 52c0dbf97..b43f7cf03 100644 --- a/src/libutil/addr.c +++ b/src/libutil/addr.c @@ -1042,10 +1042,11 @@ rspamd_inet_address_connect (const rspamd_inet_addr_t *addr, gint type, int rspamd_inet_address_listen (const rspamd_inet_addr_t *addr, gint type, - gboolean async) + enum rspamd_inet_address_listen_opts opts, + gint listen_queue) { gint fd, r; - gint on = 1; + gint on = 1, serrno; const struct sockaddr *sa; const char *path; @@ -1053,7 +1054,8 @@ rspamd_inet_address_listen (const rspamd_inet_addr_t *addr, gint type, return -1; } - fd = rspamd_socket_create (addr->af, type, 0, async); + fd = rspamd_socket_create (addr->af, type, 0, + (opts & RSPAMD_INET_ADDRESS_LISTEN_ASYNC)); if (fd == -1) { return -1; } @@ -1070,7 +1072,23 @@ rspamd_inet_address_listen (const rspamd_inet_addr_t *addr, gint type, sa = &addr->u.in.addr.sa; } - (void)setsockopt (fd, SOL_SOCKET, SO_REUSEADDR, (const void *)&on, sizeof (gint)); +#if defined(SO_REUSEADDR) + if (setsockopt (fd, SOL_SOCKET, SO_REUSEADDR, (const void *)&on, sizeof (gint)) == -1) { + msg_err ("cannot set SO_REUSEADDR on %d: %d", fd, strerror (errno)); + goto err; + } +#endif + +#if defined(SO_REUSEPORT) + if (opts & RSPAMD_INET_ADDRESS_LISTEN_REUSEPORT) { + on = 1; + + if (setsockopt (fd, SOL_SOCKET, SO_REUSEPORT, (const void *)&on, sizeof (gint)) == -1) { + msg_err ("cannot set SO_REUSEPORT on %d: %d", fd, strerror (errno)); + goto err; + } + } +#endif #ifdef HAVE_IPV6_V6ONLY if (addr->af == AF_INET6) { @@ -1086,13 +1104,13 @@ rspamd_inet_address_listen (const rspamd_inet_addr_t *addr, gint type, r = bind (fd, sa, addr->slen); if (r == -1) { - if (!async || errno != EINPROGRESS) { - close (fd); + if (!(opts & RSPAMD_INET_ADDRESS_LISTEN_ASYNC) || errno != EINPROGRESS) { msg_warn ("bind %s failed: %d, '%s'", rspamd_inet_address_to_string_pretty (addr), errno, strerror (errno)); - return -1; + + goto err; } } @@ -1115,18 +1133,31 @@ rspamd_inet_address_listen (const rspamd_inet_addr_t *addr, gint type, path, addr->u.un->mode, strerror (errno)); } } - r = listen (fd, -1); + + r = listen (fd, listen_queue); if (r == -1) { msg_warn ("listen %s failed: %d, '%s'", rspamd_inet_address_to_string_pretty (addr), errno, strerror (errno)); - close (fd); - return -1; + + goto err; } } return fd; + +err: + /* Error path */ + serrno = errno; + + if (fd != -1) { + close (fd); + } + + errno = serrno; + + return -1; } gssize diff --git a/src/libutil/addr.h b/src/libutil/addr.h index e17a4031c..852e43b60 100644 --- a/src/libutil/addr.h +++ b/src/libutil/addr.h @@ -228,15 +228,21 @@ void rspamd_inet_address_set_port (rspamd_inet_addr_t *addr, uint16_t port); int rspamd_inet_address_connect (const rspamd_inet_addr_t *addr, gint type, gboolean async); +enum rspamd_inet_address_listen_opts { + RSPAMD_INET_ADDRESS_LISTEN_DEFAULT = 0, + RSPAMD_INET_ADDRESS_LISTEN_ASYNC = (1u << 0u), + RSPAMD_INET_ADDRESS_LISTEN_REUSEPORT = (1u << 1u), +}; /** * Listen on a specified inet address * @param addr * @param type - * @param async + * @param opts * @return */ int rspamd_inet_address_listen (const rspamd_inet_addr_t *addr, gint type, - gboolean async); + enum rspamd_inet_address_listen_opts opts, + gint listen_queue); /** * Check whether specified ip is valid (not INADDR_ANY or INADDR_NONE) for ipv4 or ipv6 |