1*8b18a8bfSjruoho /* $NetBSD: t_nice.c,v 1.8 2012/03/18 07:00:51 jruoho Exp $ */
2bed8d3f0Sjruoho
3bed8d3f0Sjruoho /*-
4bed8d3f0Sjruoho * Copyright (c) 2011 The NetBSD Foundation, Inc.
5bed8d3f0Sjruoho * All rights reserved.
6bed8d3f0Sjruoho *
7bed8d3f0Sjruoho * This code is derived from software contributed to The NetBSD Foundation
8bed8d3f0Sjruoho * by Jukka Ruohonen.
9bed8d3f0Sjruoho *
10bed8d3f0Sjruoho * Redistribution and use in source and binary forms, with or without
11bed8d3f0Sjruoho * modification, are permitted provided that the following conditions
12bed8d3f0Sjruoho * are met:
13bed8d3f0Sjruoho * 1. Redistributions of source code must retain the above copyright
14bed8d3f0Sjruoho * notice, this list of conditions and the following disclaimer.
15bed8d3f0Sjruoho * 2. Redistributions in binary form must reproduce the above copyright
16bed8d3f0Sjruoho * notice, this list of conditions and the following disclaimer in the
17bed8d3f0Sjruoho * documentation and/or other materials provided with the distribution.
18bed8d3f0Sjruoho *
19bed8d3f0Sjruoho * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20bed8d3f0Sjruoho * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21bed8d3f0Sjruoho * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22bed8d3f0Sjruoho * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23bed8d3f0Sjruoho * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24bed8d3f0Sjruoho * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25bed8d3f0Sjruoho * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26bed8d3f0Sjruoho * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27bed8d3f0Sjruoho * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28bed8d3f0Sjruoho * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29bed8d3f0Sjruoho * POSSIBILITY OF SUCH DAMAGE.
30bed8d3f0Sjruoho */
31bed8d3f0Sjruoho #include <sys/cdefs.h>
32*8b18a8bfSjruoho __RCSID("$NetBSD: t_nice.c,v 1.8 2012/03/18 07:00:51 jruoho Exp $");
33bed8d3f0Sjruoho
34bed8d3f0Sjruoho #include <sys/resource.h>
35bed8d3f0Sjruoho #include <sys/wait.h>
36bed8d3f0Sjruoho
37bed8d3f0Sjruoho #include <atf-c.h>
38bed8d3f0Sjruoho #include <errno.h>
39bed8d3f0Sjruoho #include <limits.h>
4038b0bf3bSjruoho #include <pthread.h>
41bed8d3f0Sjruoho #include <stdlib.h>
42bed8d3f0Sjruoho #include <unistd.h>
43bed8d3f0Sjruoho
4438b0bf3bSjruoho static void *threadfunc(void *);
4538b0bf3bSjruoho
4638b0bf3bSjruoho static void *
threadfunc(void * arg)4738b0bf3bSjruoho threadfunc(void *arg)
4838b0bf3bSjruoho {
4938b0bf3bSjruoho int pri, val;
5038b0bf3bSjruoho
5138b0bf3bSjruoho val = *(int *)arg;
5238b0bf3bSjruoho
5338b0bf3bSjruoho errno = 0;
5438b0bf3bSjruoho pri = getpriority(PRIO_PROCESS, 0);
5538b0bf3bSjruoho ATF_REQUIRE(errno == 0);
5638b0bf3bSjruoho
5738b0bf3bSjruoho if (pri != val)
5838b0bf3bSjruoho atf_tc_fail("nice(3) value was not propagated to threads");
5938b0bf3bSjruoho
6038b0bf3bSjruoho return NULL;
6138b0bf3bSjruoho }
6238b0bf3bSjruoho
63bed8d3f0Sjruoho ATF_TC(nice_err);
ATF_TC_HEAD(nice_err,tc)64bed8d3f0Sjruoho ATF_TC_HEAD(nice_err, tc)
65bed8d3f0Sjruoho {
66*8b18a8bfSjruoho atf_tc_set_md_var(tc, "descr",
67*8b18a8bfSjruoho "Test nice(3) for invalid parameters (PR lib/42587)");
68bed8d3f0Sjruoho atf_tc_set_md_var(tc, "require.user", "unprivileged");
69bed8d3f0Sjruoho }
70bed8d3f0Sjruoho
ATF_TC_BODY(nice_err,tc)71bed8d3f0Sjruoho ATF_TC_BODY(nice_err, tc)
72bed8d3f0Sjruoho {
73bed8d3f0Sjruoho int i;
74bed8d3f0Sjruoho
75bed8d3f0Sjruoho /*
76bed8d3f0Sjruoho * The call should fail with EPERM if the
77bed8d3f0Sjruoho * supplied parameter is negative and the
78*8b18a8bfSjruoho * caller does not have privileges.
79bed8d3f0Sjruoho */
80bed8d3f0Sjruoho for (i = -20; i < 0; i++) {
81bed8d3f0Sjruoho
82bed8d3f0Sjruoho errno = 0;
83bed8d3f0Sjruoho
8453714e0dSnjoly ATF_REQUIRE_ERRNO(EPERM, nice(i) == -1);
85bed8d3f0Sjruoho }
86bed8d3f0Sjruoho }
87bed8d3f0Sjruoho
88bed8d3f0Sjruoho ATF_TC(nice_priority);
ATF_TC_HEAD(nice_priority,tc)89bed8d3f0Sjruoho ATF_TC_HEAD(nice_priority, tc)
90bed8d3f0Sjruoho {
91bed8d3f0Sjruoho atf_tc_set_md_var(tc, "descr", "Test nice(3) vs. getpriority(2)");
92bed8d3f0Sjruoho }
93bed8d3f0Sjruoho
ATF_TC_BODY(nice_priority,tc)94bed8d3f0Sjruoho ATF_TC_BODY(nice_priority, tc)
95bed8d3f0Sjruoho {
96bed8d3f0Sjruoho int i, pri, nic;
97bed8d3f0Sjruoho pid_t pid;
98bed8d3f0Sjruoho int sta;
99bed8d3f0Sjruoho
100bed8d3f0Sjruoho for (i = 0; i <= 20; i++) {
101bed8d3f0Sjruoho
102bed8d3f0Sjruoho nic = nice(i);
103bed8d3f0Sjruoho ATF_REQUIRE(nic != -1);
104bed8d3f0Sjruoho
105bed8d3f0Sjruoho errno = 0;
106bed8d3f0Sjruoho pri = getpriority(PRIO_PROCESS, 0);
107bed8d3f0Sjruoho ATF_REQUIRE(errno == 0);
108bed8d3f0Sjruoho
109bed8d3f0Sjruoho if (nic != pri)
110bed8d3f0Sjruoho atf_tc_fail("nice(3) and getpriority(2) conflict");
111bed8d3f0Sjruoho
112bed8d3f0Sjruoho /*
113bed8d3f0Sjruoho * Also verify that the nice(3) values
114bed8d3f0Sjruoho * are inherited by child processes.
115bed8d3f0Sjruoho */
116bed8d3f0Sjruoho pid = fork();
117bed8d3f0Sjruoho ATF_REQUIRE(pid >= 0);
118bed8d3f0Sjruoho
119bed8d3f0Sjruoho if (pid == 0) {
120bed8d3f0Sjruoho
121bed8d3f0Sjruoho errno = 0;
122bed8d3f0Sjruoho pri = getpriority(PRIO_PROCESS, 0);
123bed8d3f0Sjruoho ATF_REQUIRE(errno == 0);
124bed8d3f0Sjruoho
125bed8d3f0Sjruoho if (nic != pri)
126724a8843Sjruoho _exit(EXIT_FAILURE);
127bed8d3f0Sjruoho
128724a8843Sjruoho _exit(EXIT_SUCCESS);
129bed8d3f0Sjruoho }
130bed8d3f0Sjruoho
131bed8d3f0Sjruoho (void)wait(&sta);
132bed8d3f0Sjruoho
133bed8d3f0Sjruoho if (WIFEXITED(sta) == 0 || WEXITSTATUS(sta) != EXIT_SUCCESS)
134bed8d3f0Sjruoho atf_tc_fail("nice(3) value was not inherited");
135bed8d3f0Sjruoho }
136bed8d3f0Sjruoho }
137bed8d3f0Sjruoho
138bed8d3f0Sjruoho ATF_TC(nice_root);
ATF_TC_HEAD(nice_root,tc)139bed8d3f0Sjruoho ATF_TC_HEAD(nice_root, tc)
140bed8d3f0Sjruoho {
141bed8d3f0Sjruoho atf_tc_set_md_var(tc, "descr", "Test that nice(3) works");
142bed8d3f0Sjruoho atf_tc_set_md_var(tc, "require.user", "root");
143bed8d3f0Sjruoho }
144bed8d3f0Sjruoho
ATF_TC_BODY(nice_root,tc)145bed8d3f0Sjruoho ATF_TC_BODY(nice_root, tc)
146bed8d3f0Sjruoho {
147bed8d3f0Sjruoho int i;
148bed8d3f0Sjruoho
149bed8d3f0Sjruoho for (i = -20; i <= 20; i++) {
150bed8d3f0Sjruoho
151bed8d3f0Sjruoho ATF_REQUIRE(nice(i) != -1);
152bed8d3f0Sjruoho }
153bed8d3f0Sjruoho }
154bed8d3f0Sjruoho
15538b0bf3bSjruoho ATF_TC(nice_thread);
ATF_TC_HEAD(nice_thread,tc)15638b0bf3bSjruoho ATF_TC_HEAD(nice_thread, tc)
15738b0bf3bSjruoho {
15838b0bf3bSjruoho atf_tc_set_md_var(tc, "descr", "Test nice(3) with threads");
15938b0bf3bSjruoho }
16038b0bf3bSjruoho
ATF_TC_BODY(nice_thread,tc)16138b0bf3bSjruoho ATF_TC_BODY(nice_thread, tc)
16238b0bf3bSjruoho {
16338b0bf3bSjruoho pthread_t tid[5];
16438b0bf3bSjruoho int rv, val;
16538b0bf3bSjruoho size_t i;
16638b0bf3bSjruoho
16738b0bf3bSjruoho /*
16838b0bf3bSjruoho * Test that the scheduling priority is
16938b0bf3bSjruoho * propagated to all system scope threads.
17038b0bf3bSjruoho */
17138b0bf3bSjruoho for (i = 0; i < __arraycount(tid); i++) {
17238b0bf3bSjruoho
17338b0bf3bSjruoho val = nice(i);
17438b0bf3bSjruoho ATF_REQUIRE(val != -1);
17538b0bf3bSjruoho
17638b0bf3bSjruoho rv = pthread_create(&tid[i], NULL, threadfunc, &val);
17738b0bf3bSjruoho ATF_REQUIRE(rv == 0);
17838b0bf3bSjruoho
17938b0bf3bSjruoho rv = pthread_join(tid[i], NULL);
18038b0bf3bSjruoho ATF_REQUIRE(rv == 0);
18138b0bf3bSjruoho }
18238b0bf3bSjruoho }
18338b0bf3bSjruoho
ATF_TP_ADD_TCS(tp)184bed8d3f0Sjruoho ATF_TP_ADD_TCS(tp)
185bed8d3f0Sjruoho {
186bed8d3f0Sjruoho
187bed8d3f0Sjruoho ATF_TP_ADD_TC(tp, nice_err);
188bed8d3f0Sjruoho ATF_TP_ADD_TC(tp, nice_priority);
189bed8d3f0Sjruoho ATF_TP_ADD_TC(tp, nice_root);
19038b0bf3bSjruoho ATF_TP_ADD_TC(tp, nice_thread);
191bed8d3f0Sjruoho
192bed8d3f0Sjruoho return atf_no_error();
193bed8d3f0Sjruoho }
194