1*288bf522SAndroid Build Coastguard Worker /*
2*288bf522SAndroid Build Coastguard Worker ** Copyright 2010 The Android Open Source Project
3*288bf522SAndroid Build Coastguard Worker **
4*288bf522SAndroid Build Coastguard Worker ** Licensed under the Apache License, Version 2.0 (the "License");
5*288bf522SAndroid Build Coastguard Worker ** you may not use this file except in compliance with the License.
6*288bf522SAndroid Build Coastguard Worker ** You may obtain a copy of the License at
7*288bf522SAndroid Build Coastguard Worker **
8*288bf522SAndroid Build Coastguard Worker ** http://www.apache.org/licenses/LICENSE-2.0
9*288bf522SAndroid Build Coastguard Worker **
10*288bf522SAndroid Build Coastguard Worker ** Unless required by applicable law or agreed to in writing, software
11*288bf522SAndroid Build Coastguard Worker ** distributed under the License is distributed on an "AS IS" BASIS,
12*288bf522SAndroid Build Coastguard Worker ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*288bf522SAndroid Build Coastguard Worker ** See the License for the specific language governing permissions and
14*288bf522SAndroid Build Coastguard Worker ** limitations under the License.
15*288bf522SAndroid Build Coastguard Worker */
16*288bf522SAndroid Build Coastguard Worker
17*288bf522SAndroid Build Coastguard Worker
18*288bf522SAndroid Build Coastguard Worker /* Opens /proc/sched_stat and diff's the counters.
19*288bf522SAndroid Build Coastguard Worker Currently support version 15, modify parse() to support other
20*288bf522SAndroid Build Coastguard Worker versions
21*288bf522SAndroid Build Coastguard Worker */
22*288bf522SAndroid Build Coastguard Worker
23*288bf522SAndroid Build Coastguard Worker #include <stdlib.h>
24*288bf522SAndroid Build Coastguard Worker #include <stdio.h>
25*288bf522SAndroid Build Coastguard Worker #include <errno.h>
26*288bf522SAndroid Build Coastguard Worker #include <sys/uio.h>
27*288bf522SAndroid Build Coastguard Worker #include <unistd.h>
28*288bf522SAndroid Build Coastguard Worker #include <sys/time.h>
29*288bf522SAndroid Build Coastguard Worker #include <fcntl.h>
30*288bf522SAndroid Build Coastguard Worker
31*288bf522SAndroid Build Coastguard Worker #define MAX_CPU 2
32*288bf522SAndroid Build Coastguard Worker
33*288bf522SAndroid Build Coastguard Worker struct cpu_stat {
34*288bf522SAndroid Build Coastguard Worker /* sched_yield() stats */
35*288bf522SAndroid Build Coastguard Worker unsigned int yld_count; /* sched_yield() called */
36*288bf522SAndroid Build Coastguard Worker
37*288bf522SAndroid Build Coastguard Worker /* schedule() stats */
38*288bf522SAndroid Build Coastguard Worker unsigned int sched_switch; /* switched to expired queue and reused it */
39*288bf522SAndroid Build Coastguard Worker unsigned int sched_count; /* schedule() called */
40*288bf522SAndroid Build Coastguard Worker unsigned int sched_goidle; /* schedule() left the cpu idle */
41*288bf522SAndroid Build Coastguard Worker
42*288bf522SAndroid Build Coastguard Worker /* try_to_wake_up() stats */
43*288bf522SAndroid Build Coastguard Worker unsigned int ttwu_count; /* try_to_wake_up() called */
44*288bf522SAndroid Build Coastguard Worker /* try_to_wake_up() called and found the process being awakened last ran on
45*288bf522SAndroid Build Coastguard Worker * the waking cpu */
46*288bf522SAndroid Build Coastguard Worker unsigned int ttwu_local;
47*288bf522SAndroid Build Coastguard Worker
48*288bf522SAndroid Build Coastguard Worker /* latency stats */
49*288bf522SAndroid Build Coastguard Worker unsigned long long cpu_time; /* time spent running by tasks (ms) */
50*288bf522SAndroid Build Coastguard Worker unsigned long long run_delay; /* time spent waiting to run by tasks (ms) */
51*288bf522SAndroid Build Coastguard Worker unsigned long pcount; /* number of tasks (not necessarily unique) given */
52*288bf522SAndroid Build Coastguard Worker };
53*288bf522SAndroid Build Coastguard Worker
54*288bf522SAndroid Build Coastguard Worker struct cpu_stat cpu_prev[MAX_CPU];
55*288bf522SAndroid Build Coastguard Worker struct cpu_stat cpu_delta[MAX_CPU];
56*288bf522SAndroid Build Coastguard Worker struct cpu_stat tmp;
57*288bf522SAndroid Build Coastguard Worker
next_line(const char * b)58*288bf522SAndroid Build Coastguard Worker static const char *next_line(const char *b) {
59*288bf522SAndroid Build Coastguard Worker while (1) {
60*288bf522SAndroid Build Coastguard Worker switch (*b) {
61*288bf522SAndroid Build Coastguard Worker case '\n':
62*288bf522SAndroid Build Coastguard Worker return b + 1;
63*288bf522SAndroid Build Coastguard Worker case '\0':
64*288bf522SAndroid Build Coastguard Worker return NULL;
65*288bf522SAndroid Build Coastguard Worker }
66*288bf522SAndroid Build Coastguard Worker b++;
67*288bf522SAndroid Build Coastguard Worker }
68*288bf522SAndroid Build Coastguard Worker }
print()69*288bf522SAndroid Build Coastguard Worker static int print() {
70*288bf522SAndroid Build Coastguard Worker int i;
71*288bf522SAndroid Build Coastguard Worker
72*288bf522SAndroid Build Coastguard Worker printf("CPU yield() schedule() switch idle ttwu() local cpu_time wait_time timeslices\n");
73*288bf522SAndroid Build Coastguard Worker for (i=0; i<MAX_CPU; i++) {
74*288bf522SAndroid Build Coastguard Worker printf(" %2d %7u %10u %6u %4u %8u %5u %9llu %9llu %10lu\n",
75*288bf522SAndroid Build Coastguard Worker i,
76*288bf522SAndroid Build Coastguard Worker cpu_delta[i].yld_count,
77*288bf522SAndroid Build Coastguard Worker cpu_delta[i].sched_count, cpu_delta[i].sched_switch, cpu_delta[i].sched_goidle,
78*288bf522SAndroid Build Coastguard Worker cpu_delta[i].ttwu_count, cpu_delta[i].ttwu_local,
79*288bf522SAndroid Build Coastguard Worker cpu_delta[i].cpu_time / 1000000, cpu_delta[i].run_delay / 1000000, cpu_delta[i].pcount);
80*288bf522SAndroid Build Coastguard Worker }
81*288bf522SAndroid Build Coastguard Worker return 0;
82*288bf522SAndroid Build Coastguard Worker }
83*288bf522SAndroid Build Coastguard Worker
parse_cpu_v15(const char * b)84*288bf522SAndroid Build Coastguard Worker static int parse_cpu_v15(const char *b) {
85*288bf522SAndroid Build Coastguard Worker int cpu;
86*288bf522SAndroid Build Coastguard Worker
87*288bf522SAndroid Build Coastguard Worker if (sscanf(b, "cpu%d %u %u %u %u %u %u %llu %llu %lu\n",
88*288bf522SAndroid Build Coastguard Worker &cpu, &tmp.yld_count,
89*288bf522SAndroid Build Coastguard Worker &tmp.sched_switch, &tmp.sched_count, &tmp.sched_goidle,
90*288bf522SAndroid Build Coastguard Worker &tmp.ttwu_count, &tmp.ttwu_local,
91*288bf522SAndroid Build Coastguard Worker &tmp.cpu_time, &tmp.run_delay, &tmp.pcount) != 10) {
92*288bf522SAndroid Build Coastguard Worker printf("Could not parse %s\n", b);
93*288bf522SAndroid Build Coastguard Worker return -1;
94*288bf522SAndroid Build Coastguard Worker }
95*288bf522SAndroid Build Coastguard Worker
96*288bf522SAndroid Build Coastguard Worker cpu_delta[cpu].yld_count = tmp.yld_count - cpu_prev[cpu].yld_count;
97*288bf522SAndroid Build Coastguard Worker cpu_delta[cpu].sched_switch = tmp.sched_switch - cpu_prev[cpu].sched_switch;
98*288bf522SAndroid Build Coastguard Worker cpu_delta[cpu].sched_count = tmp.sched_count - cpu_prev[cpu].sched_count;
99*288bf522SAndroid Build Coastguard Worker cpu_delta[cpu].sched_goidle = tmp.sched_goidle - cpu_prev[cpu].sched_goidle;
100*288bf522SAndroid Build Coastguard Worker cpu_delta[cpu].ttwu_count = tmp.ttwu_count - cpu_prev[cpu].ttwu_count;
101*288bf522SAndroid Build Coastguard Worker cpu_delta[cpu].ttwu_local = tmp.ttwu_local - cpu_prev[cpu].ttwu_local;
102*288bf522SAndroid Build Coastguard Worker cpu_delta[cpu].cpu_time = tmp.cpu_time - cpu_prev[cpu].cpu_time;
103*288bf522SAndroid Build Coastguard Worker cpu_delta[cpu].run_delay = tmp.run_delay - cpu_prev[cpu].run_delay;
104*288bf522SAndroid Build Coastguard Worker cpu_delta[cpu].pcount = tmp.pcount - cpu_prev[cpu].pcount;
105*288bf522SAndroid Build Coastguard Worker
106*288bf522SAndroid Build Coastguard Worker cpu_prev[cpu] = tmp;
107*288bf522SAndroid Build Coastguard Worker return 0;
108*288bf522SAndroid Build Coastguard Worker }
109*288bf522SAndroid Build Coastguard Worker
110*288bf522SAndroid Build Coastguard Worker
parse(const char * b)111*288bf522SAndroid Build Coastguard Worker static int parse(const char *b) {
112*288bf522SAndroid Build Coastguard Worker unsigned int version;
113*288bf522SAndroid Build Coastguard Worker unsigned long long ts;
114*288bf522SAndroid Build Coastguard Worker
115*288bf522SAndroid Build Coastguard Worker if (sscanf(b, "version %u\n", &version) != 1) {
116*288bf522SAndroid Build Coastguard Worker printf("Could not parse version\n");
117*288bf522SAndroid Build Coastguard Worker return -1;
118*288bf522SAndroid Build Coastguard Worker }
119*288bf522SAndroid Build Coastguard Worker switch (version) {
120*288bf522SAndroid Build Coastguard Worker case 15:
121*288bf522SAndroid Build Coastguard Worker b = next_line(b);
122*288bf522SAndroid Build Coastguard Worker if (!b || sscanf(b, "timestamp %llu\n", &ts) != 1) {
123*288bf522SAndroid Build Coastguard Worker printf("Could not parse timestamp\n");
124*288bf522SAndroid Build Coastguard Worker return -1;
125*288bf522SAndroid Build Coastguard Worker }
126*288bf522SAndroid Build Coastguard Worker while (1) {
127*288bf522SAndroid Build Coastguard Worker b = next_line(b);
128*288bf522SAndroid Build Coastguard Worker if (!b) break;
129*288bf522SAndroid Build Coastguard Worker if (b[0] == 'c') {
130*288bf522SAndroid Build Coastguard Worker if (parse_cpu_v15(b)) return -1;
131*288bf522SAndroid Build Coastguard Worker }
132*288bf522SAndroid Build Coastguard Worker }
133*288bf522SAndroid Build Coastguard Worker break;
134*288bf522SAndroid Build Coastguard Worker default:
135*288bf522SAndroid Build Coastguard Worker printf("Can not handle version %u\n", version);
136*288bf522SAndroid Build Coastguard Worker return -1;
137*288bf522SAndroid Build Coastguard Worker }
138*288bf522SAndroid Build Coastguard Worker return 0;
139*288bf522SAndroid Build Coastguard Worker }
140*288bf522SAndroid Build Coastguard Worker
main(int argc,char ** argv)141*288bf522SAndroid Build Coastguard Worker int main(int argc, char **argv) {
142*288bf522SAndroid Build Coastguard Worker int i;
143*288bf522SAndroid Build Coastguard Worker int fd;
144*288bf522SAndroid Build Coastguard Worker char buf[4096];
145*288bf522SAndroid Build Coastguard Worker
146*288bf522SAndroid Build Coastguard Worker while (1) {
147*288bf522SAndroid Build Coastguard Worker fd = open("/proc/schedstat", O_RDONLY);
148*288bf522SAndroid Build Coastguard Worker if (fd < 0) return -1;
149*288bf522SAndroid Build Coastguard Worker i = read(fd, buf, sizeof(buf) - 1);
150*288bf522SAndroid Build Coastguard Worker close(fd);
151*288bf522SAndroid Build Coastguard Worker buf[i] = '\0';
152*288bf522SAndroid Build Coastguard Worker if (parse(buf)) return -1;
153*288bf522SAndroid Build Coastguard Worker print();
154*288bf522SAndroid Build Coastguard Worker sleep(1);
155*288bf522SAndroid Build Coastguard Worker }
156*288bf522SAndroid Build Coastguard Worker return 0;
157*288bf522SAndroid Build Coastguard Worker }
158