aboutsummaryrefslogtreecommitdiffstats
path: root/src/libutil
diff options
context:
space:
mode:
authorVsevolod Stakhov <vsevolod@highsecure.ru>2020-06-09 13:47:44 +0100
committerVsevolod Stakhov <vsevolod@highsecure.ru>2020-06-09 13:47:44 +0100
commit488f72bad4fab719861b4641ff832f60729a5950 (patch)
treeefb6714cb551c7611dffe0f25116cbab20b36a78 /src/libutil
parent4f69d50e5ef648c09b042d9bbca9a1b9106288b2 (diff)
downloadrspamd-488f72bad4fab719861b4641ff832f60729a5950.tar.gz
rspamd-488f72bad4fab719861b4641ff832f60729a5950.zip
[Minor] Add flags to listen socket creation
Diffstat (limited to 'src/libutil')
-rw-r--r--src/libutil/addr.c51
-rw-r--r--src/libutil/addr.h10
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