1*49cdfc7eSAndroid Build Coastguard Worker /*
2*49cdfc7eSAndroid Build Coastguard Worker * getrusage04 - accuracy of getrusage() with RUSAGE_THREAD
3*49cdfc7eSAndroid Build Coastguard Worker *
4*49cdfc7eSAndroid Build Coastguard Worker * This program is used for testing the following upstream commit:
5*49cdfc7eSAndroid Build Coastguard Worker * 761b1d26df542fd5eb348837351e4d2f3bc7bffe.
6*49cdfc7eSAndroid Build Coastguard Worker *
7*49cdfc7eSAndroid Build Coastguard Worker * getrusage() returns cpu resource usage with accuracy of 10ms
8*49cdfc7eSAndroid Build Coastguard Worker * when RUSAGE_THREAD is specified to the argument who. Meanwhile,
9*49cdfc7eSAndroid Build Coastguard Worker * accuracy is 1ms when RUSAGE_SELF is specified. This bad accuracy
10*49cdfc7eSAndroid Build Coastguard Worker * of getrusage() caused a big impact on some application which is
11*49cdfc7eSAndroid Build Coastguard Worker * critical to accuracy of cpu usage. The upstream fix removed
12*49cdfc7eSAndroid Build Coastguard Worker * casts to clock_t in task_u/stime(), to keep granularity of
13*49cdfc7eSAndroid Build Coastguard Worker * cputime_t over the calculation.
14*49cdfc7eSAndroid Build Coastguard Worker *
15*49cdfc7eSAndroid Build Coastguard Worker * Note that the accuracy on powerpc and s390x systems is always
16*49cdfc7eSAndroid Build Coastguard Worker * 10ms when either RUSAGE_THREAD or RUSAGE_SELF specified, so
17*49cdfc7eSAndroid Build Coastguard Worker * this test won't be executed on those platforms.
18*49cdfc7eSAndroid Build Coastguard Worker *
19*49cdfc7eSAndroid Build Coastguard Worker * Copyright (C) 2011 Red Hat, Inc.
20*49cdfc7eSAndroid Build Coastguard Worker * Copyright (C) 2013 Cyril Hrubis <[email protected]>
21*49cdfc7eSAndroid Build Coastguard Worker *
22*49cdfc7eSAndroid Build Coastguard Worker * This program is free software; you can redistribute it and/or
23*49cdfc7eSAndroid Build Coastguard Worker * modify it under the terms of version 2 of the GNU General Public
24*49cdfc7eSAndroid Build Coastguard Worker * License as published by the Free Software Foundation.
25*49cdfc7eSAndroid Build Coastguard Worker *
26*49cdfc7eSAndroid Build Coastguard Worker * This program is distributed in the hope that it would be useful,
27*49cdfc7eSAndroid Build Coastguard Worker * but WITHOUT ANY WARRANTY; without even the implied warranty of
28*49cdfc7eSAndroid Build Coastguard Worker * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
29*49cdfc7eSAndroid Build Coastguard Worker *
30*49cdfc7eSAndroid Build Coastguard Worker * Further, this software is distributed without any warranty that it
31*49cdfc7eSAndroid Build Coastguard Worker * is free of the rightful claim of any third person regarding
32*49cdfc7eSAndroid Build Coastguard Worker * infringement or the like. Any license provided herein, whether
33*49cdfc7eSAndroid Build Coastguard Worker * implied or otherwise, applies only to this software file. Patent
34*49cdfc7eSAndroid Build Coastguard Worker * licenses, if any, provided herein do not apply to combinations of
35*49cdfc7eSAndroid Build Coastguard Worker * this program with other software, or any other product whatsoever.
36*49cdfc7eSAndroid Build Coastguard Worker *
37*49cdfc7eSAndroid Build Coastguard Worker * You should have received a copy of the GNU General Public License
38*49cdfc7eSAndroid Build Coastguard Worker * along with this program; if not, write the Free Software
39*49cdfc7eSAndroid Build Coastguard Worker * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
40*49cdfc7eSAndroid Build Coastguard Worker * 02110-1301, USA.
41*49cdfc7eSAndroid Build Coastguard Worker */
42*49cdfc7eSAndroid Build Coastguard Worker
43*49cdfc7eSAndroid Build Coastguard Worker #define _GNU_SOURCE
44*49cdfc7eSAndroid Build Coastguard Worker #include <sys/types.h>
45*49cdfc7eSAndroid Build Coastguard Worker #include <sys/resource.h>
46*49cdfc7eSAndroid Build Coastguard Worker #include <sys/time.h>
47*49cdfc7eSAndroid Build Coastguard Worker #include <errno.h>
48*49cdfc7eSAndroid Build Coastguard Worker #include <stdio.h>
49*49cdfc7eSAndroid Build Coastguard Worker #include <stdlib.h>
50*49cdfc7eSAndroid Build Coastguard Worker #include <time.h>
51*49cdfc7eSAndroid Build Coastguard Worker
52*49cdfc7eSAndroid Build Coastguard Worker #include "test.h"
53*49cdfc7eSAndroid Build Coastguard Worker #include "safe_macros.h"
54*49cdfc7eSAndroid Build Coastguard Worker #include "lapi/posix_clocks.h"
55*49cdfc7eSAndroid Build Coastguard Worker
56*49cdfc7eSAndroid Build Coastguard Worker char *TCID = "getrusage04";
57*49cdfc7eSAndroid Build Coastguard Worker int TST_TOTAL = 1;
58*49cdfc7eSAndroid Build Coastguard Worker
59*49cdfc7eSAndroid Build Coastguard Worker #define RECORD_MAX 20
60*49cdfc7eSAndroid Build Coastguard Worker #define FACTOR_MAX 10
61*49cdfc7eSAndroid Build Coastguard Worker
62*49cdfc7eSAndroid Build Coastguard Worker #ifndef RUSAGE_THREAD
63*49cdfc7eSAndroid Build Coastguard Worker #define RUSAGE_THREAD 1
64*49cdfc7eSAndroid Build Coastguard Worker #endif
65*49cdfc7eSAndroid Build Coastguard Worker
66*49cdfc7eSAndroid Build Coastguard Worker static long BIAS_MAX;
67*49cdfc7eSAndroid Build Coastguard Worker
68*49cdfc7eSAndroid Build Coastguard Worker static int opt_factor;
69*49cdfc7eSAndroid Build Coastguard Worker static char *factor_str;
70*49cdfc7eSAndroid Build Coastguard Worker static long factor_nr = 1;
71*49cdfc7eSAndroid Build Coastguard Worker
72*49cdfc7eSAndroid Build Coastguard Worker option_t child_options[] = {
73*49cdfc7eSAndroid Build Coastguard Worker {"m:", &opt_factor, &factor_str},
74*49cdfc7eSAndroid Build Coastguard Worker {NULL, NULL, NULL}
75*49cdfc7eSAndroid Build Coastguard Worker };
76*49cdfc7eSAndroid Build Coastguard Worker
77*49cdfc7eSAndroid Build Coastguard Worker static void fusage(void);
78*49cdfc7eSAndroid Build Coastguard Worker static void busyloop(long wait);
79*49cdfc7eSAndroid Build Coastguard Worker static void setup(void);
80*49cdfc7eSAndroid Build Coastguard Worker static void cleanup(void);
81*49cdfc7eSAndroid Build Coastguard Worker
main(int argc,char * argv[])82*49cdfc7eSAndroid Build Coastguard Worker int main(int argc, char *argv[])
83*49cdfc7eSAndroid Build Coastguard Worker {
84*49cdfc7eSAndroid Build Coastguard Worker struct rusage usage;
85*49cdfc7eSAndroid Build Coastguard Worker unsigned long ulast, udelta, slast, sdelta;
86*49cdfc7eSAndroid Build Coastguard Worker int i, lc;
87*49cdfc7eSAndroid Build Coastguard Worker char msg_string[BUFSIZ];
88*49cdfc7eSAndroid Build Coastguard Worker
89*49cdfc7eSAndroid Build Coastguard Worker tst_parse_opts(argc, argv, child_options, fusage);
90*49cdfc7eSAndroid Build Coastguard Worker
91*49cdfc7eSAndroid Build Coastguard Worker #if (__powerpc__) || (__powerpc64__) || (__s390__) || (__s390x__)
92*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TCONF, NULL, "This test is not designed for current system");
93*49cdfc7eSAndroid Build Coastguard Worker #endif
94*49cdfc7eSAndroid Build Coastguard Worker
95*49cdfc7eSAndroid Build Coastguard Worker setup();
96*49cdfc7eSAndroid Build Coastguard Worker
97*49cdfc7eSAndroid Build Coastguard Worker if (opt_factor)
98*49cdfc7eSAndroid Build Coastguard Worker factor_nr = SAFE_STRTOL(cleanup, factor_str, 0, FACTOR_MAX);
99*49cdfc7eSAndroid Build Coastguard Worker
100*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO, "Using %ld as multiply factor for max [us]time "
101*49cdfc7eSAndroid Build Coastguard Worker "increment (1000+%ldus)!", factor_nr, BIAS_MAX * factor_nr);
102*49cdfc7eSAndroid Build Coastguard Worker
103*49cdfc7eSAndroid Build Coastguard Worker for (lc = 0; TEST_LOOPING(lc); lc++) {
104*49cdfc7eSAndroid Build Coastguard Worker tst_count = 0;
105*49cdfc7eSAndroid Build Coastguard Worker i = 0;
106*49cdfc7eSAndroid Build Coastguard Worker SAFE_GETRUSAGE(cleanup, RUSAGE_THREAD, &usage);
107*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO, "utime:%12lldus; stime:%12lldus",
108*49cdfc7eSAndroid Build Coastguard Worker (long long)usage.ru_utime.tv_usec,
109*49cdfc7eSAndroid Build Coastguard Worker (long long)usage.ru_stime.tv_usec);
110*49cdfc7eSAndroid Build Coastguard Worker ulast = usage.ru_utime.tv_usec;
111*49cdfc7eSAndroid Build Coastguard Worker slast = usage.ru_stime.tv_usec;
112*49cdfc7eSAndroid Build Coastguard Worker
113*49cdfc7eSAndroid Build Coastguard Worker while (i < RECORD_MAX) {
114*49cdfc7eSAndroid Build Coastguard Worker SAFE_GETRUSAGE(cleanup, RUSAGE_THREAD, &usage);
115*49cdfc7eSAndroid Build Coastguard Worker udelta = usage.ru_utime.tv_usec - ulast;
116*49cdfc7eSAndroid Build Coastguard Worker sdelta = usage.ru_stime.tv_usec - slast;
117*49cdfc7eSAndroid Build Coastguard Worker if (udelta > 0 || sdelta > 0) {
118*49cdfc7eSAndroid Build Coastguard Worker i++;
119*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO, "utime:%12lldus; stime:%12lldus",
120*49cdfc7eSAndroid Build Coastguard Worker (long long)usage.ru_utime.tv_usec,
121*49cdfc7eSAndroid Build Coastguard Worker (long long)usage.ru_stime.tv_usec);
122*49cdfc7eSAndroid Build Coastguard Worker if ((long)udelta > 1000 + (BIAS_MAX * factor_nr)) {
123*49cdfc7eSAndroid Build Coastguard Worker sprintf(msg_string,
124*49cdfc7eSAndroid Build Coastguard Worker "utime increased > %ldus:",
125*49cdfc7eSAndroid Build Coastguard Worker 1000 + BIAS_MAX * factor_nr);
126*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TFAIL, cleanup, msg_string,
127*49cdfc7eSAndroid Build Coastguard Worker " delta = %luus", udelta);
128*49cdfc7eSAndroid Build Coastguard Worker }
129*49cdfc7eSAndroid Build Coastguard Worker if ((long)sdelta > 1000 + (BIAS_MAX * factor_nr)) {
130*49cdfc7eSAndroid Build Coastguard Worker sprintf(msg_string,
131*49cdfc7eSAndroid Build Coastguard Worker "stime increased > %ldus:",
132*49cdfc7eSAndroid Build Coastguard Worker 1000 + BIAS_MAX * factor_nr);
133*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TFAIL, cleanup, msg_string,
134*49cdfc7eSAndroid Build Coastguard Worker " delta = %luus", sdelta);
135*49cdfc7eSAndroid Build Coastguard Worker }
136*49cdfc7eSAndroid Build Coastguard Worker }
137*49cdfc7eSAndroid Build Coastguard Worker ulast = usage.ru_utime.tv_usec;
138*49cdfc7eSAndroid Build Coastguard Worker slast = usage.ru_stime.tv_usec;
139*49cdfc7eSAndroid Build Coastguard Worker busyloop(100000);
140*49cdfc7eSAndroid Build Coastguard Worker }
141*49cdfc7eSAndroid Build Coastguard Worker }
142*49cdfc7eSAndroid Build Coastguard Worker
143*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TPASS, "Test Passed");
144*49cdfc7eSAndroid Build Coastguard Worker
145*49cdfc7eSAndroid Build Coastguard Worker cleanup();
146*49cdfc7eSAndroid Build Coastguard Worker tst_exit();
147*49cdfc7eSAndroid Build Coastguard Worker }
148*49cdfc7eSAndroid Build Coastguard Worker
fusage(void)149*49cdfc7eSAndroid Build Coastguard Worker static void fusage(void)
150*49cdfc7eSAndroid Build Coastguard Worker {
151*49cdfc7eSAndroid Build Coastguard Worker printf(" -m n use n as multiply factor for max [us]time "
152*49cdfc7eSAndroid Build Coastguard Worker "increment (1000+(1000*n)us),\n default value is 1\n");
153*49cdfc7eSAndroid Build Coastguard Worker }
154*49cdfc7eSAndroid Build Coastguard Worker
busyloop(long wait)155*49cdfc7eSAndroid Build Coastguard Worker static void busyloop(long wait)
156*49cdfc7eSAndroid Build Coastguard Worker {
157*49cdfc7eSAndroid Build Coastguard Worker while (wait--) ;
158*49cdfc7eSAndroid Build Coastguard Worker }
159*49cdfc7eSAndroid Build Coastguard Worker
160*49cdfc7eSAndroid Build Coastguard Worker /*
161*49cdfc7eSAndroid Build Coastguard Worker * The resolution of getrusage timers currently depends on CONFIG_HZ settings,
162*49cdfc7eSAndroid Build Coastguard Worker * as they are measured in jiffies.
163*49cdfc7eSAndroid Build Coastguard Worker *
164*49cdfc7eSAndroid Build Coastguard Worker * The problem is that there is no reasonable API to get either getrusage
165*49cdfc7eSAndroid Build Coastguard Worker * timers resolution or duration of jiffie.
166*49cdfc7eSAndroid Build Coastguard Worker *
167*49cdfc7eSAndroid Build Coastguard Worker * Here we use clock_getres() with linux specific CLOCK_REALTIME_COARSE (added
168*49cdfc7eSAndroid Build Coastguard Worker * in 2.6.32) which is also based on jiffies. This timer has the same
169*49cdfc7eSAndroid Build Coastguard Worker * granularity as getrusage but it's not guaranteed and it may change in the
170*49cdfc7eSAndroid Build Coastguard Worker * future.
171*49cdfc7eSAndroid Build Coastguard Worker *
172*49cdfc7eSAndroid Build Coastguard Worker * The default value for resolution was choosen to be 4ms as it corresponds to
173*49cdfc7eSAndroid Build Coastguard Worker * CONFIG_HZ=250 which seems to be default value.
174*49cdfc7eSAndroid Build Coastguard Worker */
guess_timer_resolution(void)175*49cdfc7eSAndroid Build Coastguard Worker static unsigned long guess_timer_resolution(void)
176*49cdfc7eSAndroid Build Coastguard Worker {
177*49cdfc7eSAndroid Build Coastguard Worker struct timespec res;
178*49cdfc7eSAndroid Build Coastguard Worker
179*49cdfc7eSAndroid Build Coastguard Worker if (clock_getres(CLOCK_REALTIME_COARSE, &res)) {
180*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO,
181*49cdfc7eSAndroid Build Coastguard Worker "CLOCK_REALTIME_COARSE not supported, using 4000 us");
182*49cdfc7eSAndroid Build Coastguard Worker return 4000;
183*49cdfc7eSAndroid Build Coastguard Worker }
184*49cdfc7eSAndroid Build Coastguard Worker
185*49cdfc7eSAndroid Build Coastguard Worker if (res.tv_nsec < 1000000 || res.tv_nsec > 10000000) {
186*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO, "Unexpected CLOCK_REALTIME_COARSE resolution,"
187*49cdfc7eSAndroid Build Coastguard Worker " using 4000 us");
188*49cdfc7eSAndroid Build Coastguard Worker return 4000;
189*49cdfc7eSAndroid Build Coastguard Worker }
190*49cdfc7eSAndroid Build Coastguard Worker
191*49cdfc7eSAndroid Build Coastguard Worker tst_resm(TINFO, "Expected timers granularity is %li us",
192*49cdfc7eSAndroid Build Coastguard Worker res.tv_nsec / 1000);
193*49cdfc7eSAndroid Build Coastguard Worker
194*49cdfc7eSAndroid Build Coastguard Worker return res.tv_nsec / 1000;
195*49cdfc7eSAndroid Build Coastguard Worker }
196*49cdfc7eSAndroid Build Coastguard Worker
setup(void)197*49cdfc7eSAndroid Build Coastguard Worker static void setup(void)
198*49cdfc7eSAndroid Build Coastguard Worker {
199*49cdfc7eSAndroid Build Coastguard Worker tst_sig(NOFORK, DEF_HANDLER, cleanup);
200*49cdfc7eSAndroid Build Coastguard Worker
201*49cdfc7eSAndroid Build Coastguard Worker if (tst_is_virt(VIRT_XEN) || tst_is_virt(VIRT_KVM) || tst_is_virt(VIRT_HYPERV))
202*49cdfc7eSAndroid Build Coastguard Worker tst_brkm(TCONF, NULL, "This testcase is not supported on this"
203*49cdfc7eSAndroid Build Coastguard Worker " virtual machine.");
204*49cdfc7eSAndroid Build Coastguard Worker
205*49cdfc7eSAndroid Build Coastguard Worker BIAS_MAX = guess_timer_resolution();
206*49cdfc7eSAndroid Build Coastguard Worker
207*49cdfc7eSAndroid Build Coastguard Worker TEST_PAUSE;
208*49cdfc7eSAndroid Build Coastguard Worker }
209*49cdfc7eSAndroid Build Coastguard Worker
cleanup(void)210*49cdfc7eSAndroid Build Coastguard Worker static void cleanup(void)
211*49cdfc7eSAndroid Build Coastguard Worker {
212*49cdfc7eSAndroid Build Coastguard Worker }
213