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