|
|
0e3136 |
diff --git a/docs/manual/mod/mpm_common.html.en b/docs/manual/mod/mpm_common.html.en
|
|
|
0e3136 |
index e7af21d..01d54b7 100644
|
|
|
0e3136 |
--- a/docs/manual/mod/mpm_common.html.en
|
|
|
0e3136 |
+++ b/docs/manual/mod/mpm_common.html.en
|
|
|
0e3136 |
@@ -42,6 +42,7 @@ more than one multi-processing module (MPM)
|
|
|
0e3136 |
EnableExceptionHook
|
|
|
0e3136 |
GracefulShutdownTimeout
|
|
|
0e3136 |
Listen
|
|
|
0e3136 |
+ ListenFree
|
|
|
0e3136 |
ListenBackLog
|
|
|
0e3136 |
ListenCoresBucketsRatio
|
|
|
0e3136 |
MaxConnectionsPerChild
|
|
|
0e3136 |
@@ -244,6 +245,31 @@ discussion of the Address already in use error message,
|
|
|
0e3136 |
including other causes.
|
|
|
0e3136 |
|
|
|
0e3136 |
|
|
|
0e3136 |
+
|
|
|
0e3136 |
+
|
|
|
0e3136 |
+
|
|
|
0e3136 |
+
|
|
|
0e3136 |
+Description:IP addresses and ports that the server
|
|
|
0e3136 |
+listens to. Doesn't require IP address to be up
|
|
|
0e3136 |
+Syntax:ListenFree [IP-address:]portnumber [protocol]
|
|
|
0e3136 |
+Context:server config
|
|
|
0e3136 |
+Status:MPM
|
|
|
0e3136 |
+Module:event , worker , prefork , mpm_winnt , mpm_netware , mpmt_os2
|
|
|
0e3136 |
+Compatibility:This directive is currently available only in Red Hat Enterprise Linux
|
|
|
0e3136 |
+
|
|
|
0e3136 |
+ The ListenFree directive is
|
|
|
0e3136 |
+ identical to the Listen directive.
|
|
|
0e3136 |
+ The only difference is in the usage of the IP_FREEBIND socket
|
|
|
0e3136 |
+ option, which is enabled by default with ListenFree .
|
|
|
0e3136 |
+ If IP_FREEBIND is enabled, it allows httpd to bind to an IP
|
|
|
0e3136 |
+ address that is nonlocal or does not (yet) exist. This allows httpd to
|
|
|
0e3136 |
+ listen on a socket without requiring the underlying network interface
|
|
|
0e3136 |
+ or the specified dynamic IP address to be up at the time when httpd
|
|
|
0e3136 |
+ is trying to bind to it.
|
|
|
0e3136 |
+
|
|
|
0e3136 |
+
|
|
|
0e3136 |
+
|
|
|
0e3136 |
+
|
|
|
0e3136 |
|
|
|
0e3136 |
|
|
|
0e3136 |
|
|
|
0e3136 |
diff --git a/include/ap_listen.h b/include/ap_listen.h
|
|
|
0e3136 |
index 58c2574..1a53292 100644
|
|
|
0e3136 |
--- a/include/ap_listen.h
|
|
|
0e3136 |
+++ b/include/ap_listen.h
|
|
|
0e3136 |
@@ -137,6 +137,9 @@ AP_DECLARE_NONSTD(const char *) ap_set_listenbacklog(cmd_parms *cmd, void *dummy
|
|
|
0e3136 |
AP_DECLARE_NONSTD(const char *) ap_set_listencbratio(cmd_parms *cmd, void *dummy, const char *arg);
|
|
|
0e3136 |
AP_DECLARE_NONSTD(const char *) ap_set_listener(cmd_parms *cmd, void *dummy,
|
|
|
0e3136 |
int argc, char *const argv[]);
|
|
|
0e3136 |
+AP_DECLARE_NONSTD(const char *) ap_set_freelistener(cmd_parms *cmd, void *dummy,
|
|
|
0e3136 |
+ int argc, char *const argv[]);
|
|
|
0e3136 |
+
|
|
|
0e3136 |
AP_DECLARE_NONSTD(const char *) ap_set_send_buffer_size(cmd_parms *cmd, void *dummy,
|
|
|
0e3136 |
const char *arg);
|
|
|
0e3136 |
AP_DECLARE_NONSTD(const char *) ap_set_receive_buffer_size(cmd_parms *cmd,
|
|
|
0e3136 |
@@ -150,6 +153,8 @@ AP_INIT_TAKE1("ListenCoresBucketsRatio", ap_set_listencbratio, NULL, RSRC_CONF,
|
|
|
0e3136 |
"Ratio between the number of CPU cores (online) and the number of listeners buckets"), \
|
|
|
0e3136 |
AP_INIT_TAKE_ARGV("Listen", ap_set_listener, NULL, RSRC_CONF, \
|
|
|
0e3136 |
"A port number or a numeric IP address and a port number, and an optional protocol"), \
|
|
|
0e3136 |
+AP_INIT_TAKE_ARGV("ListenFree", ap_set_freelistener, NULL, RSRC_CONF, \
|
|
|
0e3136 |
+ "A port number or a numeric IP address and a port number, and an optional protocol"), \
|
|
|
0e3136 |
AP_INIT_TAKE1("SendBufferSize", ap_set_send_buffer_size, NULL, RSRC_CONF, \
|
|
|
0e3136 |
"Send buffer size in bytes"), \
|
|
|
0e3136 |
AP_INIT_TAKE1("ReceiveBufferSize", ap_set_receive_buffer_size, NULL, \
|
|
|
0e3136 |
diff --git a/server/listen.c b/server/listen.c
|
|
|
0e3136 |
index e2e028a..6ef664b 100644
|
|
|
0e3136 |
--- a/server/listen.c
|
|
|
0e3136 |
+++ b/server/listen.c
|
|
|
0e3136 |
@@ -63,6 +63,7 @@ static int ap_listenbacklog;
|
|
|
0e3136 |
static int ap_listencbratio;
|
|
|
0e3136 |
static int send_buffer_size;
|
|
|
0e3136 |
static int receive_buffer_size;
|
|
|
0e3136 |
+static int ap_listenfreebind;
|
|
|
0e3136 |
#ifdef HAVE_SYSTEMD
|
|
|
0e3136 |
static int use_systemd = -1;
|
|
|
0e3136 |
#endif
|
|
|
0e3136 |
@@ -162,6 +163,21 @@ static apr_status_t make_sock(apr_pool_t *p, ap_listen_rec *server, int do_bind_
|
|
|
0e3136 |
}
|
|
|
0e3136 |
#endif
|
|
|
0e3136 |
|
|
|
0e3136 |
+
|
|
|
0e3136 |
+#if defined(APR_SO_FREEBIND)
|
|
|
0e3136 |
+ if (ap_listenfreebind) {
|
|
|
0e3136 |
+ if (apr_socket_opt_set(s, APR_SO_FREEBIND, one) < 0) {
|
|
|
0e3136 |
+ stat = apr_get_netos_error();
|
|
|
0e3136 |
+ ap_log_perror(APLOG_MARK, APLOG_CRIT, stat, p, APLOGNO(02182)
|
|
|
0e3136 |
+ "make_sock: apr_socket_opt_set: "
|
|
|
0e3136 |
+ "error setting APR_SO_FREEBIND");
|
|
|
0e3136 |
+ apr_socket_close(s);
|
|
|
0e3136 |
+ return stat;
|
|
|
0e3136 |
+ }
|
|
|
0e3136 |
+ }
|
|
|
0e3136 |
+#endif
|
|
|
0e3136 |
+
|
|
|
0e3136 |
+
|
|
|
0e3136 |
if (do_bind_listen) {
|
|
|
0e3136 |
#if APR_HAVE_IPV6
|
|
|
0e3136 |
if (server->bind_addr->family == APR_INET6) {
|
|
|
0e3136 |
@@ -956,6 +972,7 @@ AP_DECLARE(void) ap_listen_pre_config(void)
|
|
|
0e3136 |
}
|
|
|
0e3136 |
}
|
|
|
0e3136 |
|
|
|
0e3136 |
+
|
|
|
0e3136 |
AP_DECLARE_NONSTD(const char *) ap_set_listener(cmd_parms *cmd, void *dummy,
|
|
|
0e3136 |
int argc, char *const argv[])
|
|
|
0e3136 |
{
|
|
|
0e3136 |
@@ -1016,6 +1033,14 @@ AP_DECLARE_NONSTD(const char *) ap_set_listener(cmd_parms *cmd, void *dummy,
|
|
|
0e3136 |
return alloc_listener(cmd->server->process, host, port, proto, NULL);
|
|
|
0e3136 |
}
|
|
|
0e3136 |
|
|
|
0e3136 |
+AP_DECLARE_NONSTD(const char *) ap_set_freelistener(cmd_parms *cmd, void *dummy,
|
|
|
0e3136 |
+ int argc,
|
|
|
0e3136 |
+ char *const argv[])
|
|
|
0e3136 |
+{
|
|
|
0e3136 |
+ ap_listenfreebind = 1;
|
|
|
0e3136 |
+ return ap_set_listener(cmd, dummy, argc, argv);
|
|
|
0e3136 |
+}
|
|
|
0e3136 |
+
|
|
|
0e3136 |
AP_DECLARE_NONSTD(const char *) ap_set_listenbacklog(cmd_parms *cmd,
|
|
|
0e3136 |
void *dummy,
|
|
|
0e3136 |
const char *arg)
|