#include <sys/event.h>
#include <sys/procdesc.h>
#include <sys/stat.h>
#include <sys/user.h>
#include <sys/wait.h>
#include <err.h>
#include <fcntl.h>
#include <signal.h>
#include <unistd.h>
#include <atf-c.h>
ATF_TC_WITHOUT_HEAD(shared_table_filt_sig);
ATF_TC_BODY(shared_table_filt_sig, tc)
{
struct sigaction sa;
pid_t pid;
int error, status;
sa.sa_handler = SIG_IGN;
sigemptyset(&sa.sa_mask);
sa.sa_flags = 0;
error = sigaction(SIGINT, &sa, NULL);
ATF_REQUIRE(error == 0);
pid = rfork(RFPROC);
ATF_REQUIRE(pid != -1);
if (pid == 0) {
struct kevent ev;
int kq;
kq = kqueue();
if (kq < 0)
err(1, "kqueue");
EV_SET(&ev, SIGINT, EVFILT_SIGNAL, EV_ADD | EV_ENABLE, 0, 0,
NULL);
if (kevent(kq, &ev, 1, NULL, 0, NULL) < 0)
err(2, "kevent");
if (kevent(kq, NULL, 0, &ev, 1, NULL) < 0)
err(3, "kevent");
_exit(0);
}
usleep(100000);
error = kill(pid, SIGINT);
ATF_REQUIRE(error == 0);
error = waitpid(pid, &status, 0);
ATF_REQUIRE(error != -1);
ATF_REQUIRE(WIFEXITED(status));
ATF_REQUIRE_EQ(WEXITSTATUS(status), 0);
}
#define TIMER_FORKED 0
#define TIMER_TIMEOUT 1
#define RECV_TIMER 0x01
#define RECV_VNODE 0x02
#define RECV_CLOREAD 0x04
#define RECV_ERROR 0x80
#define RECV_ALL (RECV_TIMER | RECV_VNODE)
static int
cponfork_notes_check(int kq, int clofd)
{
struct kevent ev;
int error, received = 0;
EV_SET(&ev, TIMER_TIMEOUT, EVFILT_TIMER,
EV_ADD | EV_ENABLE | EV_ONESHOT, NOTE_SECONDS, 4, NULL);
error = kevent(kq, &ev, 1, NULL, 0, NULL);
if (error == -1)
return (RECV_ERROR);
while ((received & RECV_ALL) != RECV_ALL) {
error = kevent(kq, NULL, 0, &ev, 1, NULL);
if (error < 0)
return (RECV_ERROR);
else if (error == 0)
break;
switch (ev.filter) {
case EVFILT_TIMER:
if (ev.ident == TIMER_TIMEOUT)
return (received | RECV_ERROR);
received |= RECV_TIMER;
break;
case EVFILT_VNODE:
received |= RECV_VNODE;
break;
case EVFILT_READ:
if ((int)ev.ident != clofd)
return (received | RECV_ERROR);
received |= RECV_CLOREAD;
break;
}
}
return (received);
}
ATF_TC_WITHOUT_HEAD(cponfork_notes);
ATF_TC_BODY(cponfork_notes, tc)
{
struct kevent ev[3];
int clofd, dfd, error, kq, pdfd, pmask, status;
pid_t pid;
kq = kqueuex(KQUEUE_CPONFORK);
ATF_REQUIRE(kq >= 0);
dfd = open(".", O_DIRECTORY);
ATF_REQUIRE(dfd >= 0);
clofd = kqueue();
ATF_REQUIRE(clofd >= 0);
EV_SET(&ev[0], 0, EVFILT_USER, EV_ADD | EV_ENABLE, 0, 0, NULL);
error = kevent(clofd, &ev[0], 1, NULL, 0, NULL);
ATF_REQUIRE(error != -1);
EV_SET(&ev[0], dfd, EVFILT_VNODE, EV_ADD | EV_ENABLE | EV_ONESHOT,
NOTE_WRITE, 0, NULL);
EV_SET(&ev[1], TIMER_FORKED, EVFILT_TIMER, EV_ADD | EV_ENABLE | EV_ONESHOT,
NOTE_SECONDS, 3, NULL);
EV_SET(&ev[2], clofd, EVFILT_READ, EV_ADD | EV_ENABLE | EV_ONESHOT, 0,
0, NULL);
error = kevent(kq, &ev[0], 3, NULL, 0, NULL);
ATF_REQUIRE(error != -1);
EV_SET(&ev[0], 0, EVFILT_USER, 0, NOTE_TRIGGER, 0, NULL);
error = kevent(clofd, &ev[0], 1, NULL, 0, NULL);
pid = pdfork(&pdfd, 0);
ATF_REQUIRE(pid != -1);
if (pid == 0) {
struct kinfo_file kf = { .kf_structsize = sizeof(kf) };
if (fcntl(kq, F_KINFO, &kf) != 0)
_exit(RECV_ERROR);
else if (kf.kf_type != KF_TYPE_KQUEUE)
_exit(RECV_ERROR);
_exit(cponfork_notes_check(kq, clofd));
}
error = mkdir("canary", 0755);
ATF_REQUIRE(error == 0);
pmask = cponfork_notes_check(kq, clofd);
ATF_REQUIRE_EQ(pmask, RECV_ALL | RECV_CLOREAD);
_Static_assert(RECV_ALL <= UCHAR_MAX,
"Too many events to observe -- switch from waitpid -> waitid");
error = waitpid(pid, &status, 0);
ATF_REQUIRE(error != -1);
ATF_REQUIRE(WIFEXITED(status));
ATF_REQUIRE_EQ(WEXITSTATUS(status), RECV_ALL);
}
ATF_TP_ADD_TCS(tp)
{
ATF_TP_ADD_TC(tp, shared_table_filt_sig);
ATF_TP_ADD_TC(tp, cponfork_notes);
return (atf_no_error());
}