root/tools/regression/p1003_1b/sched.c
/*-
 * SPDX-License-Identifier: BSD-4-Clause
 *
 * Copyright (c) 1996-1999
 *      HD Associates, 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.
 * 3. All advertising materials mentioning features or use of this software
 *    must display the following acknowledgement:
 *      This product includes software developed by HD Associates, Inc
 * 4. Neither the name of the author nor the names of any co-contributors
 *    may be used to endorse or promote products derived from this software
 *    without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY HD ASSOCIATES 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 HD ASSOCIATES 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.
 *
 */

/* XXX: The spec says that if _POSIX_C_SOURCE is defined then
 *      _POSIX_SOURCE is ignored.  However, this is similar to
 *      the code in the O'Reilly "POSIX.4" book
 */

#define _POSIX_VERSION 199309L
#define _POSIX_SOURCE
#define _POSIX_C_SOURCE 199309L

#include <sys/mman.h>
#include <errno.h>
#include <fcntl.h>
#include <limits.h>
#include <sched.h>
#include <stdio.h>
#define __XSI_VISIBLE 1
#include <stdlib.h>
#undef __XSI_VISIBLE
#include <string.h>
#include <unistd.h>

#include "prutil.h"

static FILE *verbose;

static void
checkpris(int sched)
{
        int smin;
        int smax;

        errno = 0;

        if ( (smin = sched_get_priority_min(sched)) == -1 && errno)
                quit("sched_get_priority_min");

        if ( (smax = sched_get_priority_max(sched)) == -1 && errno)
                quit("sched_get_priority_max");

        if (smax - smin + 1 < 32 || smax < smin) {
                fprintf(stderr, "Illegal priority range for %s: %d to %d\n",
                sched_text(sched), smin, smax);
                exit(-1);
        }

        if (verbose)
                fprintf(verbose, "%12s: sched_min %2d sched_max %2d\n",
                sched_text(sched), smin, smax);
}

/* Set "try_anyway" to quit if you don't want to go on when
 * it doesn't look like something should work.
 */
static void try_anyway(const char *s)
{
        fputs(s, stderr);
        fprintf(stderr, "(trying anyway)\n");
        errno = 0;
}

static void q(int line, int code, const char *text)
{
        if (code == -1)
        {
                fprintf(stderr, "Error at line %d:\n", line);
                perror(text);
                exit(errno);
        }
}

int sched(int ac, char *av[])
{
        int fifo_schedmin, fifo_schedmax;
        int i;
        struct sched_param rt_param;
        int n_instances = 10;
        int sched;

        verbose = 0;

#if _POSIX_VERSION < 199309
        try_anyway("The _POSIX_VERSION predates P1003.1B\n");
#endif

#if !defined(_POSIX_PRIORITY_SCHEDULING)
        try_anyway(
        "The environment does not claim to support Posix scheduling.\n");
#endif

        /* Is priority scheduling configured?
         */
        errno = 0;
        if (sysconf(_SC_PRIORITY_SCHEDULING) == -1) {
                if (errno != 0) {
                        /* This isn't valid - may be a standard violation
                         */
                        quit("(should not happen) sysconf(_SC_PRIORITY_SCHEDULING)");
                }
                else {
                        try_anyway(
                        "The environment does not have run-time "
                        "support for Posix scheduling.\n");
                }
        }

        /* Check that the priorities seem reasonable.
         */

        checkpris(SCHED_FIFO);
        checkpris(SCHED_RR);
        checkpris(SCHED_OTHER);

/* BSD extensions?
 */
#if defined(SCHED_IDLE)
        checkpris(SCHED_IDLE);
#endif

        fifo_schedmin = sched_get_priority_min(SCHED_FIFO);
        fifo_schedmax = sched_get_priority_max(SCHED_FIFO);

        /* Make sure we can do some basic schedule switching:
         */
        {
                struct sched_param orig_param, shouldbe;
                int orig_scheduler = sched_is(__LINE__, &orig_param, -1);

                if (verbose)
                        fprintf(verbose,
                        "The original scheduler is %s and the priority is %d.\n",
                        sched_text(orig_scheduler), orig_param.sched_priority);

                /* Basic check: Try to set current settings:
                 */
                q(__LINE__, sched_setscheduler(0, orig_scheduler, &orig_param),
                        "sched_setscheduler: Can't set original scheduler");

                rt_param.sched_priority = fifo_schedmin;

                q(__LINE__, sched_setscheduler(0, SCHED_FIFO, &rt_param),
                "sched_setscheduler SCHED_FIFO");

                (void)sched_is(__LINE__, 0, SCHED_FIFO);

                q(__LINE__, sched_getparam(0, &shouldbe), "sched_getparam");

                if (shouldbe.sched_priority != fifo_schedmin)
                        quit("sched_setscheduler wrong priority (min)");

                rt_param.sched_priority = fifo_schedmin;

                q(__LINE__, sched_setparam(0, &rt_param),
                        "sched_setparam to fifo_schedmin");

                rt_param.sched_priority = fifo_schedmin + 1;

                q(__LINE__, sched_setparam(0, &rt_param),
                        "sched_setparam to fifo_schedmin + 1");

                q(__LINE__, sched_getparam(0, &shouldbe),
                        "sched_getparam");

                if (shouldbe.sched_priority != fifo_schedmin + 1)
                        quit("sched_setscheduler wrong priority (min + 1)");

                q(__LINE__, sched_setscheduler(0, SCHED_RR, &rt_param),
                        "sched_setscheduler SCHED_RR");

                (void)sched_is(__LINE__, 0, SCHED_RR);

                q(__LINE__, sched_setscheduler(0, orig_scheduler, &orig_param),
                        "sched_setscheduler restoring original scheduler");

                (void)sched_is(__LINE__, 0, orig_scheduler);
        }


        {
                char nam[] = "P1003_1b_schedXXXXXX";
                int fd;
                pid_t p;
                pid_t *lastrun;

                fd = mkstemp(nam);
                if (fd == -1)
                        q(__LINE__, errno, "mkstemp failed");

                (void)unlink(nam);

                p = (pid_t)0;

                write(fd, &p, sizeof(p));

                q(__LINE__,  (int)(lastrun = mmap(0, sizeof(*lastrun), PROT_READ|PROT_WRITE,
                MAP_SHARED, fd, 0)), "mmap");

                /* Set our priority at the highest:
                 */
                sched = SCHED_FIFO;
                rt_param.sched_priority = fifo_schedmax;
                q(__LINE__, sched_setscheduler(0, sched, &rt_param),
                "sched_setscheduler sched");

                for (i = 0; i < n_instances; i++)
                {
                        pid_t me;

                        /* XXX This is completely bogus.  The children never run.
                         */
                        if ((me = fork()) != 0)
                        {
                                /* Parent.
                                 */
                                (void)sched_is(__LINE__, 0, sched);

                                /* Lower our priority:
                                 */
                                rt_param.sched_priority--;

                                q(__LINE__, sched_setscheduler(0, sched, &rt_param),
                                "sched_setscheduler sched");

                                while (1)
                                {
                                        q(__LINE__, sched_getparam(0, &rt_param), "sched_getparam");

                                        rt_param.sched_priority--;


                                        if (rt_param.sched_priority < fifo_schedmin)
                                                exit(0);

                                        *lastrun = me;
                                        q(__LINE__, sched_setparam(0, &rt_param), "sched_setparam");

                                        if (*lastrun == me)
                                        {
                                                /* The child will run twice
                                                 * at  the end:
                                                 */
                                                if (!me || rt_param.sched_priority != 0)
                                                {
                                                        fprintf(stderr,
                                                        "ran process %ld twice at priority %d\n",
                                                        (long)me, rt_param.sched_priority + 1);
                                                        exit(-1);
                                                }
                                        }
                                }
                        }
                }
        }

        return 0;
}
#ifdef STANDALONE_TESTS
int main(int argc, char *argv[]) { return sched(argc, argv); }
#endif