1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151
|
/*++
/* NAME
/* master_listen 3
/* SUMMARY
/* Postfix master - start/stop listeners
/* SYNOPSIS
/* #include "master.h"
/*
/* void master_listen_init(serv)
/* MASTER_SERV *serv;
/*
/* void master_listen_cleanup(serv)
/* MASTER_SERV *serv;
/* DESCRIPTION
/* master_listen_init() turns on the listener implemented by the
/* named process. FIFOs and UNIX-domain sockets are created with
/* mode 0622 and with ownership mail_owner.
/*
/* master_listen_cleanup() turns off the listener implemented by the
/* named process.
/* DIAGNOSTICS
/* BUGS
/* SEE ALSO
/* inet_listen(3), internet-domain listener
/* unix_listen(3), unix-domain listener
/* fifo_listen(3), named-pipe listener
/* set_eugid(3), set effective user/group attributes
/* LICENSE
/* .ad
/* .fi
/* The Secure Mailer license must be distributed with this software.
/* AUTHOR(S)
/* Wietse Venema
/* IBM T.J. Watson Research
/* P.O. Box 704
/* Yorktown Heights, NY 10598, USA
/*--*/
/* System library. */
#include <sys_defs.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <string.h>
/* Utility library. */
#include <msg.h>
#include <listen.h>
#include <mymalloc.h>
#include <stringops.h>
#include <inet_addr_list.h>
#include <set_eugid.h>
#include <iostuff.h>
/* Global library. */
#include <mail_params.h>
/* Application-specific. */
#include "master.h"
/* master_listen_init - enable connection requests */
void master_listen_init(MASTER_SERV *serv)
{
char *myname = "master_listen_init";
char *end_point;
int n;
/*
* Find out what transport we should use, then create one or more
* listener sockets. Make the listener sockets non-blocking, so that
* child processes don't block in accept() when multiple processes are
* selecting on the same socket and only one of them gets the connection.
*/
switch (serv->type) {
/*
* UNIX-domain or stream listener endpoints always come as singlets.
*/
case MASTER_SERV_TYPE_UNIX:
set_eugid(var_owner_uid, var_owner_gid);
serv->listen_fd[0] =
LOCAL_LISTEN(serv->name, serv->max_proc > var_proc_limit ?
serv->max_proc : var_proc_limit, NON_BLOCKING);
close_on_exec(serv->listen_fd[0], CLOSE_ON_EXEC);
set_eugid(getuid(), getgid());
break;
/*
* FIFO listener endpoints always come as singlets.
*/
case MASTER_SERV_TYPE_FIFO:
set_eugid(var_owner_uid, var_owner_gid);
serv->listen_fd[0] = fifo_listen(serv->name, 0622, NON_BLOCKING);
close_on_exec(serv->listen_fd[0], CLOSE_ON_EXEC);
set_eugid(getuid(), getgid());
break;
/*
* INET-domain listener endpoints can be wildcarded (the default) or
* bound to specific interface addresses.
*/
case MASTER_SERV_TYPE_INET:
if (serv->addr_list.inet == 0) { /* wild-card */
serv->listen_fd[0] =
inet_listen(serv->name, serv->max_proc > var_proc_limit ?
serv->max_proc : var_proc_limit, NON_BLOCKING);
close_on_exec(serv->listen_fd[0], CLOSE_ON_EXEC);
} else { /* virtual */
for (n = 0; n < serv->listen_fd_count; n++) {
end_point = concatenate(inet_ntoa(serv->addr_list.inet->addrs[n]),
":", serv->name, (char *) 0);
serv->listen_fd[n]
= inet_listen(end_point, serv->max_proc > var_proc_limit ?
serv->max_proc : var_proc_limit, NON_BLOCKING);
close_on_exec(serv->listen_fd[n], CLOSE_ON_EXEC);
myfree(end_point);
}
}
break;
default:
msg_panic("%s: unknown service type: %d", myname, serv->type);
}
}
/* master_listen_cleanup - disable connection requests */
void master_listen_cleanup(MASTER_SERV *serv)
{
char *myname = "master_listen_cleanup";
int n;
/*
* XXX The listen socket is shared with child processes. Closing the
* socket in the master process does not really disable listeners in
* child processes. There seems to be no documented way to turn off a
* listener. The 4.4BSD shutdown(2) man page promises an ENOTCONN error
* when shutdown(2) is applied to a socket that is not connected.
*/
for (n = 0; n < serv->listen_fd_count; n++) {
if (close(serv->listen_fd[n]) < 0)
msg_warn("%s: close listener socket %d: %m",
myname, serv->listen_fd[n]);
serv->listen_fd[n] = -1;
}
}
|