diff options
Diffstat (limited to 'src/conn-tools/s6-tcpserver4d.c')
-rw-r--r-- | src/conn-tools/s6-tcpserver4d.c | 245 |
1 files changed, 139 insertions, 106 deletions
diff --git a/src/conn-tools/s6-tcpserver4d.c b/src/conn-tools/s6-tcpserver4d.c index 4c1edf2..8dd6614 100644 --- a/src/conn-tools/s6-tcpserver4d.c +++ b/src/conn-tools/s6-tcpserver4d.c @@ -9,9 +9,11 @@ #include <fcntl.h> #include <signal.h> -#include <skalibs/gccattributes.h> -#include <skalibs/allreadwrite.h> +#include <skalibs/posixplz.h> +#include <skalibs/uint16.h> +#include <skalibs/uint32.h> #include <skalibs/types.h> +#include <skalibs/allreadwrite.h> #include <skalibs/sgetopt.h> #include <skalibs/strerr.h> #include <skalibs/fmtscan.h> @@ -22,33 +24,38 @@ #include <skalibs/selfpipe.h> #include <skalibs/iopause.h> #include <skalibs/socket.h> -#include <skalibs/exec.h> +#include <skalibs/genset.h> +#include <skalibs/avltreen.h> + +#include <s6/ucspiserver.h> -#define ABSOLUTE_MAXCONN 1000 +#define ABSOLUTE_MAXCONN 16384 #define USAGE "s6-tcpserver4d [ -v verbosity ] [ -1 ] [ -c maxconn ] [ -C localmaxconn ] prog..." -typedef struct pidip_s pidip_t, *pidip_t_ref ; -struct pidip_s +typedef struct pidi_s pidi, *pidi_ref ; +struct pidi_s { - pid_t left ; - uint32_t right ; + pid_t pid ; + uint32_t i ; } ; -static unsigned int maxconn = 40 ; -static unsigned int localmaxconn = 40 ; -static unsigned int verbosity = 1 ; +static uint32_t maxconn = 40 ; +static uint32_t localmaxconn = 40 ; +static uint32_t verbosity = 1 ; static int cont = 1 ; -static pidip_t *pidip = 0 ; -static unsigned int numconn = 0 ; -static diuint32 *ipnum = 0 ; -static unsigned int iplen = 0 ; -static char fmtmaxconn[UINT_FMT+1] = "/" ; -static char fmtlocalmaxconn[UINT_FMT+1] = "/" ; +static genset *pidis ; +#define PIDI(i) genset_p(pidi, pidis, (i)) +#define numconn genset_n(pidis) +static genset *ipnums ; +#define IPNUM(i) genset_p(diuint32, ipnums, (i)) +static avltreen *by_ip ; +static avltreen *by_pid ; +static char fmtmaxconn[UINT32_FMT+1] = "/" ; +static char fmtlocalmaxconn[UINT32_FMT+1] = "/" ; - /* Utility functions */ static inline void dieusage () { @@ -60,25 +67,33 @@ static inline void X (void) strerr_dief1x(101, "internal inconsistency. Please submit a bug-report.") ; } - - /* Lookup primitives */ - -static inline unsigned int lookup_pid (pid_t pid) +static void *bypid_dtok (uint32_t d, void *aux) { - unsigned int i = 0 ; - for (; i < numconn ; i++) if (pid == pidip[i].left) break ; - return i ; + genset *g = aux ; + return &genset_p(pidi, g, d)->pid ; } -static inline unsigned int lookup_ip (uint32_t ip) +static int bypid_cmp (void const *a, void const *b, void *aux) { - unsigned int i = 0 ; - for (; i < iplen ; i++) if (ip == ipnum[i].left) break ; - return i ; + (void)aux ; + pid_t aa = *(pid_t const *)a ; + pid_t bb = *(pid_t const *)b ; + return aa < bb ? -1 : aa > bb ; } +static void *byip_dtok (uint32_t d, void *aux) +{ + genset *g = aux ; + return &genset_p(diuint32, g, d)->left ; +} - /* Logging */ +static int byip_cmp (void const *a, void const *b, void *aux) +{ + (void)aux ; + uint32_t aa = *(uint32_t const *)a ; + uint32_t bb = *(uint32_t const *)b ; + return aa < bb ? -1 : aa > bb ; +} static inline void log_start (void) { @@ -97,28 +112,28 @@ static void log_status (void) strerr_warni3x("status: ", fmt, fmtmaxconn) ; } -static inline void log_deny (uint32_t ip, uint16_t port, unsigned int num) +static inline void log_deny (uint32_t ip, uint16_t port, uint32_t num) { char fmtip[UINT32_FMT] ; char fmtport[UINT16_FMT] ; - char fmtnum[UINT_FMT] ; + char fmtnum[UINT32_FMT] ; fmtip[ip4_fmtu32(fmtip, ip)] = 0 ; fmtport[uint16_fmt(fmtport, port)] = 0 ; - fmtnum[uint_fmt(fmtnum, num)] = 0 ; + fmtnum[uint32_fmt(fmtnum, num)] = 0 ; strerr_warni7sys("deny ", fmtip, ":", fmtport, " count ", fmtnum, fmtlocalmaxconn) ; } -static inline void log_accept (pid_t pid, uint32_t ip, uint16_t port, unsigned int num) +static inline void log_accept (pid_t pid, uint32_t ip, uint16_t port, uint32_t num) { char fmtipport[IP4_FMT + UINT16_FMT + 1] ; char fmtpid[PID_FMT] ; - char fmtnum[UINT_FMT] ; + char fmtnum[UINT32_FMT] ; size_t n ; n = ip4_fmtu32(fmtipport, ip) ; fmtipport[n++] = ':' ; n += uint16_fmt(fmtipport + n, port) ; fmtipport[n] = 0 ; - fmtnum[uint_fmt(fmtnum, num)] = 0 ; + fmtnum[uint32_fmt(fmtnum, num)] = 0 ; fmtpid[pid_fmt(fmtpid, pid)] = 0 ; strerr_warni7x("allow ", fmtipport, " pid ", fmtpid, " count ", fmtnum, fmtlocalmaxconn) ; } @@ -134,37 +149,42 @@ static inline void log_close (pid_t pid, uint32_t ip, int w) strerr_warni6x("end pid ", fmtpid, " ip ", fmtip, WIFSIGNALED(w) ? " signal " : " exitcode ", fmtw) ; } - - /* Signal handling */ +static int killthem_iter (void *data, void *aux) +{ + kill(((pidi *)data)->pid, *(int *)aux) ; + return 1 ; +} static void killthem (int sig) { - unsigned int i = 0 ; - for (; i < numconn ; i++) kill(pidip[i].left, sig) ; + genset_iter(pidis, &killthem_iter, &sig) ; } static inline void wait_children (void) { for (;;) { - unsigned int i ; - int w ; - pid_t pid = wait_nohang(&w) ; + uint32_t d ; + int wstat ; + pid_t pid = wait_nohang(&wstat) ; if (pid < 0) if (errno != ECHILD) strerr_diefu1sys(111, "wait_nohang") ; else break ; else if (!pid) break ; - i = lookup_pid(pid) ; - if (i < numconn) /* it's one of ours ! */ + if (avltreen_search(by_pid, &pid, &d)) { - uint32_t ip = pidip[i].right ; - unsigned int j = lookup_ip(ip) ; - if (j >= iplen) X() ; - if (!--ipnum[j].right) ipnum[j] = ipnum[--iplen] ; - pidip[i] = pidip[--numconn] ; + uint32_t i = PIDI(d)->i ; + uint32_t ip = IPNUM(i)->left ; + avltreen_delete(by_pid, &pid) ; + genset_delete(pidis, d) ; + if (!--IPNUM(i)->right) + { + avltreen_delete(by_ip, &ip) ; + genset_delete(ipnums, i) ; + } if (verbosity >= 2) { - log_close(pid, ip, w) ; + log_close(pid, ip, wstat) ; log_status() ; } } @@ -214,73 +234,63 @@ static inline void handle_signals (void) } } - - /* New connection handling */ - -static inline void run_child (int, uint32_t, uint16_t, unsigned int, char const *const *) gccattr_noreturn ; -static inline void run_child (int s, uint32_t ip, uint16_t port, unsigned int num, char const *const *argv) -{ - char fmt[74] ; - size_t n = 0 ; - PROG = "s6-tcpserver4d (child)" ; - if ((fd_move(1, s) < 0) || (fd_copy(0, 1) < 0)) - strerr_diefu1sys(111, "move fds") ; - memcpy(fmt+n, "PROTO=TCP\0TCPREMOTEIP=", 22) ; n += 22 ; - n += ip4_fmtu32(fmt+n, ip) ; fmt[n++] = 0 ; - memcpy(fmt+n, "TCPREMOTEPORT=", 14) ; n += 14 ; - n += uint16_fmt(fmt+n, port) ; fmt[n++] = 0 ; - memcpy(fmt+n, "TCPCONNNUM=", 11) ; n += 11 ; - n += uint_fmt(fmt+n, num) ; fmt[n++] = 0 ; - xmexec_n(argv, fmt, n, 4) ; -} - -static inline void new_connection (int s, uint32_t ip, uint16_t port, char const *const *argv) +static inline void new_connection (int s, uint32_t ip, uint16_t port, char const *const *argv, char const *const *envp, size_t envlen) { - unsigned int i = lookup_ip(ip) ; - unsigned int num = (i < iplen) ? ipnum[i].right : 0 ; + size_t m = 0 ; pid_t pid ; + uint32_t d ; + uint32_t num = 0 ; + char fmt[47 + IP4_FMT + UINT16_FMT + UINT_FMT] ; + + if (avltreen_search(by_ip, &ip, &d)) num = IPNUM(d)->right ; if (num >= localmaxconn) { log_deny(ip, port, num) ; return ; } - pid = fork() ; - if (pid < 0) + + memcpy(fmt + m, "PROTO=TCP\0TCPREMOTEIP=", 22) ; m += 22 ; + m += ip4_fmtu32(fmt + m, ip) ; + fmt[m++] = 0 ; + memcpy(fmt + m, "TCPREMOTEPORT=", 14) ; m += 14 ; + m += uint16_fmt(fmt + m, port) ; fmt[m++] = 0 ; + memcpy(fmt + m, "TCPCONNNUM=", 11) ; m += 11 ; + m += uint_fmt(fmt + m, num) ; fmt[m++] = 0 ; + + pid = s6_ucspiserver_spawn(s, argv, envp, envlen, fmt, m, 4) ; + if (!pid) { - if (verbosity) strerr_warnwu1sys("fork") ; + if (verbosity) strerr_warnwu2sys("spawn ", argv[0]) ; return ; } - else if (!pid) - { - selfpipe_finish() ; - run_child(s, ip, port, num+1, argv) ; - } - if (i < iplen) ipnum[i].right = num + 1 ; + if (num) IPNUM(d)->right++ ; else { - ipnum[iplen].left = ip ; - ipnum[iplen++].right = 1 ; + d = genset_new(ipnums) ; + IPNUM(d)->left = ip ; + IPNUM(d)->right = 1 ; + avltreen_insert(by_ip, d) ; } - pidip[numconn].left = pid ; - pidip[numconn++].right = ip ; + + num = genset_new(pidis) ; + PIDI(num)->pid = pid ; + PIDI(num)->i = d ; + avltreen_insert(by_pid, num) ; if (verbosity >= 2) { - log_accept(pid, ip, port, ipnum[i].right) ; + log_accept(pid, ip, port, IPNUM(d)->right) ; log_status() ; } } - - /* And the main */ - int main (int argc, char const *const *argv) { iopause_fd x[2] = { { .events = IOPAUSE_READ }, { .fd = 0, .events = IOPAUSE_READ | IOPAUSE_EXCEPT } } ; + int flag1 = 0 ; PROG = "s6-tcpserver4d" ; { subgetopt l = SUBGETOPT_ZERO ; - int flag1 = 0 ; for (;;) { int opt = subgetopt_r(argc, argv, "1c:C:v:", &l) ; @@ -288,9 +298,9 @@ int main (int argc, char const *const *argv) switch (opt) { case '1' : flag1 = 1 ; break ; - case 'c' : if (!uint0_scan(l.arg, &maxconn)) dieusage() ; break ; - case 'C' : if (!uint0_scan(l.arg, &localmaxconn)) dieusage() ; break ; - case 'v' : if (!uint0_scan(l.arg, &verbosity)) dieusage() ; break ; + case 'c' : if (!uint320_scan(l.arg, &maxconn)) dieusage() ; break ; + case 'C' : if (!uint320_scan(l.arg, &localmaxconn)) dieusage() ; break ; + case 'v' : if (!uint320_scan(l.arg, &verbosity)) dieusage() ; break ; default : dieusage() ; } } @@ -301,7 +311,8 @@ int main (int argc, char const *const *argv) if (fstat(0, &st) < 0) strerr_diefu1sys(111, "fstat stdin") ; if (!S_ISSOCK(st.st_mode)) strerr_dief1x(100, "stdin is not a socket") ; } - if (coe(0) < 0) strerr_diefu1sys(111, "make socket close-on-exec") ; + if (coe(0) == -1 || ndelay_on(0) == -1) + strerr_diefu1sys(111, "set socket flags") ; if (flag1) { if (fcntl(1, F_GETFD) < 0) @@ -325,13 +336,40 @@ int main (int argc, char const *const *argv) sigaddset(&set, SIGABRT) ; if (!selfpipe_trapset(&set)) strerr_diefu1sys(111, "trap signals") ; } - fmtlocalmaxconn[1+uint_fmt(fmtlocalmaxconn+1, localmaxconn)] = 0 ; + fmtlocalmaxconn[1+uint32_fmt(fmtlocalmaxconn+1, localmaxconn)] = 0 ; + } + + { + diuint32 ipnum_storage[maxconn] ; + uint32_t ipnum_freelist[maxconn] ; + avlnode byip_storage[maxconn] ; + uint32_t byip_freelist[maxconn] ; + pidi pidi_storage[maxconn] ; + uint32_t pidi_freelist[maxconn] ; + avlnode bypid_storage[maxconn] ; + uint32_t bypid_freelist[maxconn] ; + genset ipnum_info ; + genset pidi_info ; + avltreen byip_info ; + avltreen bypid_info ; + size_t envlen = env_len((char const *const *)environ) ; + + GENSET_init(&ipnum_info, diuint32, ipnum_storage, ipnum_freelist, maxconn) ; + GENSET_init(&pidi_info, pidi, pidi_storage, pidi_freelist, maxconn) ; + avltreen_init(&byip_info, byip_storage, byip_freelist, maxconn, &byip_dtok, &byip_cmp, &ipnum_info) ; + avltreen_init(&bypid_info, bypid_storage, bypid_freelist, maxconn, &bypid_dtok, &bypid_cmp, &pidi_info) ; + ipnums = &ipnum_info ; + pidis = &pidi_info ; + by_ip = &byip_info ; + by_pid = &bypid_info ; + if (verbosity >= 2) { - fmtmaxconn[1+uint_fmt(fmtmaxconn+1, maxconn)] = 0 ; + fmtmaxconn[1+uint32_fmt(fmtmaxconn+1, maxconn)] = 0 ; log_start() ; log_status() ; } + if (flag1) { uint16_t port ; @@ -345,14 +383,9 @@ int main (int argc, char const *const *argv) else m = uint16_fmt(fmtport, port) ; fmtport[m++] = '\n' ; allwrite(1, fmtport, m) ; - fd_close(1) ; + close(1) ; } - } - { - pidip_t pidip_inyostack[maxconn] ; - diuint32 ipnum_inyostack[maxconn] ; - pidip = pidip_inyostack ; ipnum = ipnum_inyostack ; while (cont) { if (iopause_g(x, 1 + (numconn < maxconn), 0) < 0) strerr_diefu1sys(111, "iopause") ; @@ -366,7 +399,7 @@ int main (int argc, char const *const *argv) char packedip[4] ; uint16_t port ; int fd = socket_accept4(x[1].fd, packedip, &port) ; - if (fd < 0) + if (fd == -1) { if (verbosity) strerr_warnwu1sys("accept") ; } @@ -374,7 +407,7 @@ int main (int argc, char const *const *argv) { uint32_t ip ; uint32_unpack_big(packedip, &ip) ; - new_connection(fd, ip, port, argv) ; + new_connection(fd, ip, port, argv, (char const *const *)environ, envlen) ; fd_close(fd) ; } } |