1*df57947fSPedro F. Giffuni /*- 2*df57947fSPedro F. Giffuni * SPDX-License-Identifier: BSD-4-Clause 3*df57947fSPedro F. Giffuni * 4e5f1b1b1SPeter Dufault * Copyright (c) 1996-1999 5e5f1b1b1SPeter Dufault * HD Associates, Inc. All rights reserved. 6e5f1b1b1SPeter Dufault * 7e5f1b1b1SPeter Dufault * Redistribution and use in source and binary forms, with or without 8e5f1b1b1SPeter Dufault * modification, are permitted provided that the following conditions 9e5f1b1b1SPeter Dufault * are met: 10e5f1b1b1SPeter Dufault * 1. Redistributions of source code must retain the above copyright 11e5f1b1b1SPeter Dufault * notice, this list of conditions and the following disclaimer. 12e5f1b1b1SPeter Dufault * 2. Redistributions in binary form must reproduce the above copyright 13e5f1b1b1SPeter Dufault * notice, this list of conditions and the following disclaimer in the 14e5f1b1b1SPeter Dufault * documentation and/or other materials provided with the distribution. 15e5f1b1b1SPeter Dufault * 3. All advertising materials mentioning features or use of this software 16e5f1b1b1SPeter Dufault * must display the following acknowledgement: 17e5f1b1b1SPeter Dufault * This product includes software developed by HD Associates, Inc 18e5f1b1b1SPeter Dufault * 4. Neither the name of the author nor the names of any co-contributors 19e5f1b1b1SPeter Dufault * may be used to endorse or promote products derived from this software 20e5f1b1b1SPeter Dufault * without specific prior written permission. 21e5f1b1b1SPeter Dufault * 22e5f1b1b1SPeter Dufault * THIS SOFTWARE IS PROVIDED BY HD ASSOCIATES AND CONTRIBUTORS ``AS IS'' AND 23e5f1b1b1SPeter Dufault * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 24e5f1b1b1SPeter Dufault * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 25e5f1b1b1SPeter Dufault * ARE DISCLAIMED. IN NO EVENT SHALL HD ASSOCIATES OR CONTRIBUTORS BE LIABLE 26e5f1b1b1SPeter Dufault * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 27e5f1b1b1SPeter Dufault * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 28e5f1b1b1SPeter Dufault * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 29e5f1b1b1SPeter Dufault * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 30e5f1b1b1SPeter Dufault * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 31e5f1b1b1SPeter Dufault * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 32e5f1b1b1SPeter Dufault * SUCH DAMAGE. 33e5f1b1b1SPeter Dufault * $FreeBSD$ 34e5f1b1b1SPeter Dufault * 35e5f1b1b1SPeter Dufault */ 36e5f1b1b1SPeter Dufault 37e5f1b1b1SPeter Dufault /* XXX: The spec says that if _POSIX_C_SOURCE is defined then 38e5f1b1b1SPeter Dufault * _POSIX_SOURCE is ignored. However, this is similar to 39e5f1b1b1SPeter Dufault * the code in the O'Reilly "POSIX.4" book 40e5f1b1b1SPeter Dufault */ 41e5f1b1b1SPeter Dufault 42e5f1b1b1SPeter Dufault #define _POSIX_VERSION 199309L 43e5f1b1b1SPeter Dufault #define _POSIX_SOURCE 44e5f1b1b1SPeter Dufault #define _POSIX_C_SOURCE 199309L 45e5f1b1b1SPeter Dufault 46a39b8b86SEnji Cooper #include <sys/mman.h> 47e5f1b1b1SPeter Dufault #include <errno.h> 48e5f1b1b1SPeter Dufault #include <fcntl.h> 49a39b8b86SEnji Cooper #include <limits.h> 50e5f1b1b1SPeter Dufault #include <sched.h> 51a39b8b86SEnji Cooper #include <stdio.h> 52a39b8b86SEnji Cooper #define __XSI_VISIBLE 1 53a39b8b86SEnji Cooper #include <stdlib.h> 54a39b8b86SEnji Cooper #undef __XSI_VISIBLE 55a39b8b86SEnji Cooper #include <string.h> 56a39b8b86SEnji Cooper #include <unistd.h> 57e5f1b1b1SPeter Dufault 58e5f1b1b1SPeter Dufault #include "prutil.h" 59e5f1b1b1SPeter Dufault 60e5f1b1b1SPeter Dufault static FILE *verbose; 61e5f1b1b1SPeter Dufault 62e5f1b1b1SPeter Dufault static void 63e5f1b1b1SPeter Dufault checkpris(int sched) 64e5f1b1b1SPeter Dufault { 65e5f1b1b1SPeter Dufault int smin; 66e5f1b1b1SPeter Dufault int smax; 67e5f1b1b1SPeter Dufault 68e5f1b1b1SPeter Dufault errno = 0; 69e5f1b1b1SPeter Dufault 70e5f1b1b1SPeter Dufault if ( (smin = sched_get_priority_min(sched)) == -1 && errno) 71e5f1b1b1SPeter Dufault quit("sched_get_priority_min"); 72e5f1b1b1SPeter Dufault 73e5f1b1b1SPeter Dufault if ( (smax = sched_get_priority_max(sched)) == -1 && errno) 74e5f1b1b1SPeter Dufault quit("sched_get_priority_max"); 75e5f1b1b1SPeter Dufault 76e5f1b1b1SPeter Dufault if (smax - smin + 1 < 32 || smax < smin) { 77e5f1b1b1SPeter Dufault fprintf(stderr, "Illegal priority range for %s: %d to %d\n", 78e5f1b1b1SPeter Dufault sched_text(sched), smin, smax); 79e5f1b1b1SPeter Dufault exit(-1); 80e5f1b1b1SPeter Dufault } 81e5f1b1b1SPeter Dufault 82e5f1b1b1SPeter Dufault if (verbose) 83e5f1b1b1SPeter Dufault fprintf(verbose, "%12s: sched_min %2d sched_max %2d\n", 84e5f1b1b1SPeter Dufault sched_text(sched), smin, smax); 85e5f1b1b1SPeter Dufault } 86e5f1b1b1SPeter Dufault 87e5f1b1b1SPeter Dufault /* Set "try_anyway" to quit if you don't want to go on when 88e5f1b1b1SPeter Dufault * it doesn't look like something should work. 89e5f1b1b1SPeter Dufault */ 90e5f1b1b1SPeter Dufault static void try_anyway(const char *s) 91e5f1b1b1SPeter Dufault { 92e5f1b1b1SPeter Dufault fputs(s, stderr); 93e5f1b1b1SPeter Dufault fprintf(stderr, "(trying anyway)\n"); 94e5f1b1b1SPeter Dufault errno = 0; 95e5f1b1b1SPeter Dufault } 96e5f1b1b1SPeter Dufault 97e5f1b1b1SPeter Dufault static void q(int line, int code, const char *text) 98e5f1b1b1SPeter Dufault { 99e5f1b1b1SPeter Dufault if (code == -1) 100e5f1b1b1SPeter Dufault { 101e5f1b1b1SPeter Dufault fprintf(stderr, "Error at line %d:\n", line); 102e5f1b1b1SPeter Dufault perror(text); 103e5f1b1b1SPeter Dufault exit(errno); 104e5f1b1b1SPeter Dufault } 105e5f1b1b1SPeter Dufault } 106e5f1b1b1SPeter Dufault 107e5f1b1b1SPeter Dufault int sched(int ac, char *av[]) 108e5f1b1b1SPeter Dufault { 109e5f1b1b1SPeter Dufault int fifo_schedmin, fifo_schedmax; 110e5f1b1b1SPeter Dufault int i; 111e5f1b1b1SPeter Dufault struct sched_param rt_param; 112e5f1b1b1SPeter Dufault int n_instances = 10; 113e5f1b1b1SPeter Dufault int sched; 114e5f1b1b1SPeter Dufault 115e5f1b1b1SPeter Dufault verbose = 0; 116e5f1b1b1SPeter Dufault 117e5f1b1b1SPeter Dufault #if _POSIX_VERSION < 199309 118e5f1b1b1SPeter Dufault try_anyway("The _POSIX_VERSION predates P1003.1B\n"); 119e5f1b1b1SPeter Dufault #endif 120e5f1b1b1SPeter Dufault 121e5f1b1b1SPeter Dufault #if !defined(_POSIX_PRIORITY_SCHEDULING) 122e5f1b1b1SPeter Dufault try_anyway( 123e5f1b1b1SPeter Dufault "The environment does not claim to support Posix scheduling.\n"); 124e5f1b1b1SPeter Dufault #endif 125e5f1b1b1SPeter Dufault 126e5f1b1b1SPeter Dufault /* Is priority scheduling configured? 127e5f1b1b1SPeter Dufault */ 128e5f1b1b1SPeter Dufault errno = 0; 129e5f1b1b1SPeter Dufault if (sysconf(_SC_PRIORITY_SCHEDULING) == -1) { 130e5f1b1b1SPeter Dufault if (errno != 0) { 131e5f1b1b1SPeter Dufault /* This isn't valid - may be a standard violation 132e5f1b1b1SPeter Dufault */ 133e5f1b1b1SPeter Dufault quit("(should not happen) sysconf(_SC_PRIORITY_SCHEDULING)"); 134e5f1b1b1SPeter Dufault } 135e5f1b1b1SPeter Dufault else { 136e5f1b1b1SPeter Dufault try_anyway( 137e5f1b1b1SPeter Dufault "The environment does not have run-time " 138e5f1b1b1SPeter Dufault "support for Posix scheduling.\n"); 139e5f1b1b1SPeter Dufault } 140e5f1b1b1SPeter Dufault } 141e5f1b1b1SPeter Dufault 142e5f1b1b1SPeter Dufault /* Check that the priorities seem reasonable. 143e5f1b1b1SPeter Dufault */ 144e5f1b1b1SPeter Dufault 145e5f1b1b1SPeter Dufault checkpris(SCHED_FIFO); 146e5f1b1b1SPeter Dufault checkpris(SCHED_RR); 147e5f1b1b1SPeter Dufault checkpris(SCHED_OTHER); 148e5f1b1b1SPeter Dufault 149e5f1b1b1SPeter Dufault /* BSD extensions? 150e5f1b1b1SPeter Dufault */ 151e5f1b1b1SPeter Dufault #if defined(SCHED_IDLE) 152e5f1b1b1SPeter Dufault checkpris(SCHED_IDLE); 153e5f1b1b1SPeter Dufault #endif 154e5f1b1b1SPeter Dufault 155e5f1b1b1SPeter Dufault fifo_schedmin = sched_get_priority_min(SCHED_FIFO); 156e5f1b1b1SPeter Dufault fifo_schedmax = sched_get_priority_max(SCHED_FIFO); 157e5f1b1b1SPeter Dufault 158e5f1b1b1SPeter Dufault /* Make sure we can do some basic schedule switching: 159e5f1b1b1SPeter Dufault */ 160e5f1b1b1SPeter Dufault { 161e5f1b1b1SPeter Dufault struct sched_param orig_param, shouldbe; 162e5f1b1b1SPeter Dufault int orig_scheduler = sched_is(__LINE__, &orig_param, -1); 163e5f1b1b1SPeter Dufault 164e5f1b1b1SPeter Dufault if (verbose) 165e5f1b1b1SPeter Dufault fprintf(verbose, 166e5f1b1b1SPeter Dufault "The original scheduler is %s and the priority is %d.\n", 167e5f1b1b1SPeter Dufault sched_text(orig_scheduler), orig_param.sched_priority); 168e5f1b1b1SPeter Dufault 169e5f1b1b1SPeter Dufault /* Basic check: Try to set current settings: 170e5f1b1b1SPeter Dufault */ 171e5f1b1b1SPeter Dufault q(__LINE__, sched_setscheduler(0, orig_scheduler, &orig_param), 172e5f1b1b1SPeter Dufault "sched_setscheduler: Can't set original scheduler"); 173e5f1b1b1SPeter Dufault 174e5f1b1b1SPeter Dufault rt_param.sched_priority = fifo_schedmin; 175e5f1b1b1SPeter Dufault 176e5f1b1b1SPeter Dufault q(__LINE__, sched_setscheduler(0, SCHED_FIFO, &rt_param), 177e5f1b1b1SPeter Dufault "sched_setscheduler SCHED_FIFO"); 178e5f1b1b1SPeter Dufault 179e5f1b1b1SPeter Dufault (void)sched_is(__LINE__, 0, SCHED_FIFO); 180e5f1b1b1SPeter Dufault 181e5f1b1b1SPeter Dufault q(__LINE__, sched_getparam(0, &shouldbe), "sched_getparam"); 182e5f1b1b1SPeter Dufault 183e5f1b1b1SPeter Dufault if (shouldbe.sched_priority != fifo_schedmin) 184e5f1b1b1SPeter Dufault quit("sched_setscheduler wrong priority (min)"); 185e5f1b1b1SPeter Dufault 186e5f1b1b1SPeter Dufault rt_param.sched_priority = fifo_schedmin; 187e5f1b1b1SPeter Dufault 188e5f1b1b1SPeter Dufault q(__LINE__, sched_setparam(0, &rt_param), 189e5f1b1b1SPeter Dufault "sched_setparam to fifo_schedmin"); 190e5f1b1b1SPeter Dufault 191e5f1b1b1SPeter Dufault rt_param.sched_priority = fifo_schedmin + 1; 192e5f1b1b1SPeter Dufault 193e5f1b1b1SPeter Dufault q(__LINE__, sched_setparam(0, &rt_param), 194e5f1b1b1SPeter Dufault "sched_setparam to fifo_schedmin + 1"); 195e5f1b1b1SPeter Dufault 196e5f1b1b1SPeter Dufault q(__LINE__, sched_getparam(0, &shouldbe), 197e5f1b1b1SPeter Dufault "sched_getparam"); 198e5f1b1b1SPeter Dufault 199e5f1b1b1SPeter Dufault if (shouldbe.sched_priority != fifo_schedmin + 1) 200e5f1b1b1SPeter Dufault quit("sched_setscheduler wrong priority (min + 1)"); 201e5f1b1b1SPeter Dufault 202e5f1b1b1SPeter Dufault q(__LINE__, sched_setscheduler(0, SCHED_RR, &rt_param), 203e5f1b1b1SPeter Dufault "sched_setscheduler SCHED_RR"); 204e5f1b1b1SPeter Dufault 205e5f1b1b1SPeter Dufault (void)sched_is(__LINE__, 0, SCHED_RR); 206e5f1b1b1SPeter Dufault 207e5f1b1b1SPeter Dufault q(__LINE__, sched_setscheduler(0, orig_scheduler, &orig_param), 208e5f1b1b1SPeter Dufault "sched_setscheduler restoring original scheduler"); 209e5f1b1b1SPeter Dufault 210e5f1b1b1SPeter Dufault (void)sched_is(__LINE__, 0, orig_scheduler); 211e5f1b1b1SPeter Dufault } 212e5f1b1b1SPeter Dufault 213e5f1b1b1SPeter Dufault 214e5f1b1b1SPeter Dufault { 215a39b8b86SEnji Cooper char nam[] = "P1003_1b_schedXXXXXX"; 216e5f1b1b1SPeter Dufault int fd; 217e5f1b1b1SPeter Dufault pid_t p; 218e5f1b1b1SPeter Dufault pid_t *lastrun; 219e5f1b1b1SPeter Dufault 220a39b8b86SEnji Cooper fd = mkstemp(nam); 221a39b8b86SEnji Cooper if (fd == -1) 222a39b8b86SEnji Cooper q(__LINE__, errno, "mkstemp failed"); 223e5f1b1b1SPeter Dufault 224e5f1b1b1SPeter Dufault (void)unlink(nam); 225e5f1b1b1SPeter Dufault 226e5f1b1b1SPeter Dufault p = (pid_t)0; 227e5f1b1b1SPeter Dufault 228e5f1b1b1SPeter Dufault write(fd, &p, sizeof(p)); 229e5f1b1b1SPeter Dufault 230e5f1b1b1SPeter Dufault q(__LINE__, (int)(lastrun = mmap(0, sizeof(*lastrun), PROT_READ|PROT_WRITE, 231e5f1b1b1SPeter Dufault MAP_SHARED, fd, 0)), "mmap"); 232e5f1b1b1SPeter Dufault 233e5f1b1b1SPeter Dufault /* Set our priority at the highest: 234e5f1b1b1SPeter Dufault */ 235e5f1b1b1SPeter Dufault sched = SCHED_FIFO; 236e5f1b1b1SPeter Dufault rt_param.sched_priority = fifo_schedmax; 237e5f1b1b1SPeter Dufault q(__LINE__, sched_setscheduler(0, sched, &rt_param), 238e5f1b1b1SPeter Dufault "sched_setscheduler sched"); 239e5f1b1b1SPeter Dufault 240e5f1b1b1SPeter Dufault for (i = 0; i < n_instances; i++) 241e5f1b1b1SPeter Dufault { 242e5f1b1b1SPeter Dufault pid_t me; 243e5f1b1b1SPeter Dufault 244e5f1b1b1SPeter Dufault /* XXX This is completely bogus. The children never run. 245e5f1b1b1SPeter Dufault */ 246e5f1b1b1SPeter Dufault if ((me = fork()) != 0) 247e5f1b1b1SPeter Dufault { 248e5f1b1b1SPeter Dufault /* Parent. 249e5f1b1b1SPeter Dufault */ 250e5f1b1b1SPeter Dufault (void)sched_is(__LINE__, 0, sched); 251e5f1b1b1SPeter Dufault 252e5f1b1b1SPeter Dufault /* Lower our priority: 253e5f1b1b1SPeter Dufault */ 254e5f1b1b1SPeter Dufault rt_param.sched_priority--; 255e5f1b1b1SPeter Dufault 256e5f1b1b1SPeter Dufault q(__LINE__, sched_setscheduler(0, sched, &rt_param), 257e5f1b1b1SPeter Dufault "sched_setscheduler sched"); 258e5f1b1b1SPeter Dufault 259e5f1b1b1SPeter Dufault while (1) 260e5f1b1b1SPeter Dufault { 261e5f1b1b1SPeter Dufault q(__LINE__, sched_getparam(0, &rt_param), "sched_getparam"); 262e5f1b1b1SPeter Dufault 263e5f1b1b1SPeter Dufault rt_param.sched_priority--; 264e5f1b1b1SPeter Dufault 265e5f1b1b1SPeter Dufault 266e5f1b1b1SPeter Dufault if (rt_param.sched_priority < fifo_schedmin) 267e5f1b1b1SPeter Dufault exit(0); 268e5f1b1b1SPeter Dufault 269e5f1b1b1SPeter Dufault *lastrun = me; 270e5f1b1b1SPeter Dufault q(__LINE__, sched_setparam(0, &rt_param), "sched_setparam"); 271e5f1b1b1SPeter Dufault 272e5f1b1b1SPeter Dufault if (*lastrun == me) 273e5f1b1b1SPeter Dufault { 274e5f1b1b1SPeter Dufault /* The child will run twice 275e5f1b1b1SPeter Dufault * at the end: 276e5f1b1b1SPeter Dufault */ 277e5f1b1b1SPeter Dufault if (!me || rt_param.sched_priority != 0) 278e5f1b1b1SPeter Dufault { 279e5f1b1b1SPeter Dufault fprintf(stderr, 280e5f1b1b1SPeter Dufault "ran process %ld twice at priority %d\n", 281e5f1b1b1SPeter Dufault (long)me, rt_param.sched_priority + 1); 282e5f1b1b1SPeter Dufault exit(-1); 283e5f1b1b1SPeter Dufault } 284e5f1b1b1SPeter Dufault } 285e5f1b1b1SPeter Dufault } 286e5f1b1b1SPeter Dufault } 287e5f1b1b1SPeter Dufault } 288e5f1b1b1SPeter Dufault } 289e5f1b1b1SPeter Dufault 290e5f1b1b1SPeter Dufault return 0; 291e5f1b1b1SPeter Dufault } 292e5f1b1b1SPeter Dufault #ifdef STANDALONE_TESTS 293e5f1b1b1SPeter Dufault int main(int argc, char *argv[]) { return sched(argc, argv); } 294e5f1b1b1SPeter Dufault #endif 295