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
    +
    top
    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
     
    top
    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)