/* $NetBSD: t_nanosleep.c,v 1.1 2024/10/09 13:02:53 kre Exp $ */

/*-
 * Copyright (c) 2024 The NetBSD Foundation, Inc.
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
 * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
 * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 */

#include <sys/cdefs.h>
__COPYRIGHT("@(#) Copyright (c) 2024\
 The NetBSD Foundation, inc. All rights reserved.");
__RCSID("$NetBSD: t_nanosleep.c,v 1.1 2024/10/09 13:02:53 kre Exp $");

#include <sys/types.h>
#include <sys/wait.h>

#include <atf-c.h>

#include <errno.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include <unistd.h>

static void
sacrifice(void)
{
	pause();
}

static void
tester(pid_t victim, clockid_t clock, int flags)
{
	/*
	 * we need this sleep to be long enough that we
	 * can accurately detect when the sleep finishes
	 * early, but not so long that when there's no
	 * bug and things actually sleep this long, that
	 * the execution of a sleep this long, several
	 * times, won't slow down the overall testing
	 * process too much.    Trial and error...
	 */
	struct timespec to_sleep = { 4, 0 };

	struct timespec before, after;
	struct timespec *ts;
	int e;

	if (clock_gettime(clock, &before) != 0)
		exit(1);

	if (flags & TIMER_ABSTIME) {
		timespecadd(&to_sleep, &before, &after);
		ts = &after;
	} else
		ts = &to_sleep;

	printf("Test: Clock=%d Flags=%x, starting at %jd.%.9ld\n",
		(int)clock, flags, (intmax_t)before.tv_sec, before.tv_nsec);
	if (flags & TIMER_ABSTIME)
		printf("Sleeping until %jd.%.9ld\n",
		    (intmax_t)ts->tv_sec, ts->tv_nsec);
	else
		printf("Sleeping for %jd.%.9ld\n",
		    (intmax_t)ts->tv_sec, ts->tv_nsec);

	/* OK, we're ready */

	/* these next two steps need to be as close together as possible */
	if (kill(victim, SIGKILL) == -1)
		exit(2);
	if ((e = clock_nanosleep(clock, flags, ts, &after)) != 0)
		exit(20 + e);

	if (!(flags & TIMER_ABSTIME)) {
		printf("Remaining to sleep: %jd.%.9ld\n",
		    (intmax_t)after.tv_sec, after.tv_nsec);

		if (after.tv_sec != 0 || after.tv_nsec != 0)
			exit(3);
	}

	if (clock_gettime(clock, &after) != 0)
		exit(4);

	printf("Sleep ended at: %jd.%.9ld\n",
		(intmax_t)after.tv_sec, after.tv_nsec);

	timespecadd(&before, &to_sleep, &before);
	if (timespeccmp(&before, &after, >))
		exit(5);

	exit(0);
}

/*
 * The parent of the masochist/victim above, controls everything.
 */
static void
runit(clockid_t clock, int flags)
{
	pid_t v, m, x;
	int status;
	struct timespec brief = { 0, 3 * 100 * 1000 * 1000 };  /* 300 ms */

	ATF_REQUIRE((v = fork()) != -1);
	if (v == 0)
		sacrifice();

	ATF_REQUIRE((m = fork()) != -1);
	if (m == 0)
		tester(v, clock, flags);

	ATF_REQUIRE((x = wait(&status)) != -1);

	if (x == m) {
		/*
		 * This is bad, the murderer shouldn't die first
		 */
		fprintf(stderr, "M exited first, status %#x\n", status);
		(void)kill(v, SIGKILL);	/* just in case */
		atf_tc_fail("2nd child predeceased first");
	}
	if (x != v) {
		fprintf(stderr, "Unknown exit from %d (status: %#x)"
		    "(M=%d V=%d)\n", x, status, m, v);
		(void)kill(m, SIGKILL);
		(void)kill(v, SIGKILL);
		atf_tc_fail("Strange child died");
	}

	/*
	 * OK, the victim died, we don't really care why,
	 * (it should have been because of a SIGKILL, maybe
	 * test for that someday).
	 *
	 * Now we get to proceed to the real test.
	 *
	 * But we want to wait a short whle to try and be sure
	 * that m (the child still running) has a chance to
	 * fall asleep.
	 */
	(void) clock_nanosleep(CLOCK_MONOTONIC, TIMER_RELTIME, &brief, NULL);

	/*
	 * This is the test, for PR kern/58733
	 *   -  stop a process while in clock_nanosleep()
	 *   -  resume it again
	 *   -  see if it still sleeps as long as was requested (or longer)
	 */
	ATF_REQUIRE(kill(m, SIGSTOP) == 0);
	(void) clock_nanosleep(CLOCK_MONOTONIC, TIMER_RELTIME, &brief, NULL);
	ATF_REQUIRE(kill(m, SIGCONT) == 0);

	ATF_REQUIRE((x = wait(&status)) != -1);

	if (x != m) {
		fprintf(stderr, "Unknown exit from %d (status: %#x)"
		    "(M=%d V=%d)\n", x, status, m, v);
		(void) kill(m, SIGKILL);
		atf_tc_fail("Strange child died");
	}

	if (status == 0)
		atf_tc_pass();

	/*
	 * Here we should decode the status, and give a better
	 * clue what really went wrong.   Later...
	 */
	fprintf(stderr, "Test failed: status from M: %#x\n", status);
	atf_tc_fail("M exited with non-zero status.  PR kern/58733");
}


ATF_TC(nanosleep_monotonic_absolute);
ATF_TC_HEAD(nanosleep_monotonic_absolute, tc)
{
	atf_tc_set_md_var(tc, "descr", "Checks clock_nanosleep(MONO, ABS)");
}
ATF_TC_BODY(nanosleep_monotonic_absolute, tc)
{
	runit(CLOCK_MONOTONIC, TIMER_ABSTIME);
}

ATF_TC(nanosleep_monotonic_relative);
ATF_TC_HEAD(nanosleep_monotonic_relative, tc)
{
	atf_tc_set_md_var(tc, "descr", "Checks clock_nanosleep(MONO, REL)");
}
ATF_TC_BODY(nanosleep_monotonic_relative, tc)
{
	runit(CLOCK_MONOTONIC, TIMER_RELTIME);
}

ATF_TC(nanosleep_realtime_absolute);
ATF_TC_HEAD(nanosleep_realtime_absolute, tc)
{
	atf_tc_set_md_var(tc, "descr", "Checks clock_nanosleep(REAL, ABS)");
}
ATF_TC_BODY(nanosleep_realtime_absolute, tc)
{
	runit(CLOCK_REALTIME, TIMER_ABSTIME);
}

ATF_TC(nanosleep_realtime_relative);
ATF_TC_HEAD(nanosleep_realtime_relative, tc)
{
	atf_tc_set_md_var(tc, "descr", "Checks clock_nanosleep(REAL, REL)");
}
ATF_TC_BODY(nanosleep_realtime_relative, tc)
{
	runit(CLOCK_REALTIME, TIMER_RELTIME);
}

ATF_TP_ADD_TCS(tp)
{

	ATF_TP_ADD_TC(tp, nanosleep_monotonic_absolute);
	ATF_TP_ADD_TC(tp, nanosleep_monotonic_relative);
	ATF_TP_ADD_TC(tp, nanosleep_realtime_absolute);
	ATF_TP_ADD_TC(tp, nanosleep_realtime_relative);

	return atf_no_error();
}