#include <sys/types.h>
#include <sys/event.h>
#include <sys/socket.h>
#include <sys/un.h>
#include <stdio.h>
#include <err.h>
#include <errno.h>
#include <fcntl.h>
#include <pthread.h>
#include <string.h>
#include <time.h>
#include <unistd.h>
union msg_control{
struct cmsghdr cmsgh;
char control[CMSG_SPACE(sizeof(int)*2)];
};
static void *thr_close(void *arg)
{
close(*(int *)arg);
return NULL;
}
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;
pthread_t thr;
int s[2], fd, kqfd;
int infinite = 0, error;
if (argc > 1 && !strcmp(argv[1], "--infinite"))
infinite = 1;
if ((kqfd = kqueue()) < 0)
err(1, "kqueue");
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, s) < 0)
err(1, "socketpair");
if ((fd = socket(AF_UNIX, SOCK_DGRAM, 0)) < 0)
err(1, "open");
iov_buf = 0;
iov.iov_base = &iov_buf;
iov.iov_len = sizeof(iov_buf);
msgh.msg_control = msg_control.control;
msgh.msg_controllen = sizeof(msg_control.control);
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) = fd;
*((int *)CMSG_DATA(cmsgh) + 1) = kqfd;
error = pthread_create(&thr, NULL, thr_close, &fd);
if (error)
errc(1, error, "pthread_create");
if (sendmsg(s[0], &msgh, 0) < 0) {
if (errno != EINVAL)
err(1, "sendmsg");
}
error = pthread_join(thr, NULL);
if (error)
errc(1, error, "pthread_join");
close(s[0]);
close(s[1]);
close(fd);
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;
}
}
close(kqfd);
return 0;
}