#include <sys/types.h>
#include <sys/socket.h>
#include <sys/time.h>
#include <sys/un.h>
#include <stdio.h>
#include <err.h>
#include <errno.h>
#include <fcntl.h>
#include <string.h>
#include <time.h>
#include <unistd.h>
union msg_control{
struct cmsghdr cmsgh;
char control[CMSG_SPACE(sizeof(int) * 2)];
};
int main(int argc, char *argv[])
{
struct timespec ts_start, ts_now, ts_time;
union msg_control msg_control;
int iov_buf;
struct iovec iov;
struct msghdr msgh;
struct cmsghdr *cmsgh;
int sp[2], sl[2], ts;
int infinite = 0;
if (argc > 1 && !strcmp(argv[1], "--infinite"))
infinite = 1;
if (!infinite)
if (clock_gettime(CLOCK_BOOTTIME, &ts_start) <0)
err(1, "clock_gettime");
while (1) {
if (socketpair(AF_UNIX, SOCK_STREAM|O_NONBLOCK, 0, sp) < 0)
err(1, "socketpair");
iov_buf = 0;
iov.iov_base = &iov_buf;
iov.iov_len = sizeof(iov_buf);
msgh.msg_control = msg_control.control;
msgh.msg_controllen = CMSG_SPACE(sizeof(int));
msgh.msg_iov = &iov;
msgh.msg_iovlen = 1;
msgh.msg_name = NULL;
msgh.msg_namelen = 0;
cmsgh = CMSG_FIRSTHDR(&msgh);
cmsgh->cmsg_len = CMSG_LEN(sizeof(int));
cmsgh->cmsg_level = SOL_SOCKET;
cmsgh->cmsg_type = SCM_RIGHTS;
*((int *)CMSG_DATA(cmsgh)) = sp[0];
if (sendmsg(sp[0], &msgh, 0) < 0) {
if (errno == EMFILE) {
close(sp[0]);
goto skip;
}
err(1, "sendmsg sp0");
}
*((int *)CMSG_DATA(cmsgh)) = sp[1];
if (sendmsg(sp[1], &msgh, 0) < 0) {
if (errno == EMFILE) {
close(sp[0]);
goto skip;
}
err(1, "sendmsg sp1");
}
close(sp[0]);
if (socketpair(AF_UNIX, SOCK_STREAM|O_NONBLOCK, 0, sl) < 0)
err(1, "socketpair");
iov_buf = 0;
iov.iov_base = &iov_buf;
iov.iov_len = sizeof(iov_buf);
msgh.msg_control = msg_control.control;
msgh.msg_controllen = CMSG_SPACE(sizeof(int) * 2);
msgh.msg_iov = &iov;
msgh.msg_iovlen = 1;
msgh.msg_name = NULL;
msgh.msg_namelen = 0;
cmsgh = CMSG_FIRSTHDR(&msgh);
cmsgh->cmsg_len = CMSG_LEN(sizeof(int) * 2);
cmsgh->cmsg_level = SOL_SOCKET;
cmsgh->cmsg_type = SCM_RIGHTS;
*((int *)CMSG_DATA(cmsgh) + 0) = sl[0];
*((int *)CMSG_DATA(cmsgh) + 1) = sl[1];
if (sendmsg(sl[0], &msgh, 0) < 0) {
if (errno != EMFILE)
err(1, "sendmsg sl0");
}
close(sl[0]);
close(sl[1]);
if (recvmsg(sp[1], &msgh, 0) < 0) {
if (errno == EMSGSIZE)
goto skip;
err(1, "recvmsg sp1");
}
if (!(cmsgh = CMSG_FIRSTHDR(&msgh)))
errx(1, "bad cmsg header");
if (cmsgh->cmsg_level != SOL_SOCKET)
errx(1, "bad cmsg level");
if (cmsgh->cmsg_type != SCM_RIGHTS)
errx(1, "bad cmsg type");
if (cmsgh->cmsg_len != CMSG_LEN(sizeof(ts)))
errx(1, "bad cmsg length");
ts = *((int *)CMSG_DATA(cmsgh));
if (recvmsg(ts, &msgh, 0) < 0) {
if (errno == EMSGSIZE)
goto skip;
err(1, "recvmsg ts");
}
close(ts);
if (!(cmsgh = CMSG_FIRSTHDR(&msgh)))
errx(1, "bad cmsg header");
if (cmsgh->cmsg_level != SOL_SOCKET)
errx(1, "bad cmsg level");
if (cmsgh->cmsg_type != SCM_RIGHTS)
errx(1, "bad cmsg type");
if (cmsgh->cmsg_len != CMSG_LEN(sizeof(ts)))
errx(1, "bad cmsg length");
ts = *((int *)CMSG_DATA(cmsgh));
close(ts);
skip:
close(sp[1]);
if (!infinite) {
if (clock_gettime(CLOCK_BOOTTIME, &ts_now) <0)
err(1, "clock_gettime");
timespecsub(&ts_now, &ts_start, &ts_time);
if (ts_time.tv_sec >= 20)
break;
}
}
return 0;
}