1*1a3d31e3SAndroid Build Coastguard Worker /*
2*1a3d31e3SAndroid Build Coastguard Worker * blktrace output analysis: generate a timeline & gather statistics
3*1a3d31e3SAndroid Build Coastguard Worker *
4*1a3d31e3SAndroid Build Coastguard Worker * Copyright (C) 2006 Alan D. Brunelle <[email protected]>
5*1a3d31e3SAndroid Build Coastguard Worker *
6*1a3d31e3SAndroid Build Coastguard Worker * This program is free software; you can redistribute it and/or modify
7*1a3d31e3SAndroid Build Coastguard Worker * it under the terms of the GNU General Public License as published by
8*1a3d31e3SAndroid Build Coastguard Worker * the Free Software Foundation; either version 2 of the License, or
9*1a3d31e3SAndroid Build Coastguard Worker * (at your option) any later version.
10*1a3d31e3SAndroid Build Coastguard Worker *
11*1a3d31e3SAndroid Build Coastguard Worker * This program is distributed in the hope that it will be useful,
12*1a3d31e3SAndroid Build Coastguard Worker * but WITHOUT ANY WARRANTY; without even the implied warranty of
13*1a3d31e3SAndroid Build Coastguard Worker * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14*1a3d31e3SAndroid Build Coastguard Worker * GNU General Public License for more details.
15*1a3d31e3SAndroid Build Coastguard Worker *
16*1a3d31e3SAndroid Build Coastguard Worker * You should have received a copy of the GNU General Public License
17*1a3d31e3SAndroid Build Coastguard Worker * along with this program; if not, write to the Free Software
18*1a3d31e3SAndroid Build Coastguard Worker * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
19*1a3d31e3SAndroid Build Coastguard Worker *
20*1a3d31e3SAndroid Build Coastguard Worker */
21*1a3d31e3SAndroid Build Coastguard Worker
22*1a3d31e3SAndroid Build Coastguard Worker #include <stdio.h>
23*1a3d31e3SAndroid Build Coastguard Worker #include <unistd.h>
24*1a3d31e3SAndroid Build Coastguard Worker #include <sys/types.h>
25*1a3d31e3SAndroid Build Coastguard Worker #include <sys/stat.h>
26*1a3d31e3SAndroid Build Coastguard Worker #include <fcntl.h>
27*1a3d31e3SAndroid Build Coastguard Worker #include <sys/mman.h>
28*1a3d31e3SAndroid Build Coastguard Worker #include <string.h>
29*1a3d31e3SAndroid Build Coastguard Worker
30*1a3d31e3SAndroid Build Coastguard Worker #include "blktrace.h"
31*1a3d31e3SAndroid Build Coastguard Worker #include "globals.h"
32*1a3d31e3SAndroid Build Coastguard Worker
33*1a3d31e3SAndroid Build Coastguard Worker #define DEF_LEN (16 * 1024 * 1024)
34*1a3d31e3SAndroid Build Coastguard Worker
35*1a3d31e3SAndroid Build Coastguard Worker static int fd;
36*1a3d31e3SAndroid Build Coastguard Worker static void *cur_map = MAP_FAILED;
37*1a3d31e3SAndroid Build Coastguard Worker static off_t cur_min, cur, cur_max, total_size;
38*1a3d31e3SAndroid Build Coastguard Worker static size_t len;
39*1a3d31e3SAndroid Build Coastguard Worker static struct blk_io_trace *next_t;
40*1a3d31e3SAndroid Build Coastguard Worker static long pgsz;
41*1a3d31e3SAndroid Build Coastguard Worker
42*1a3d31e3SAndroid Build Coastguard Worker int data_is_native = -1;
43*1a3d31e3SAndroid Build Coastguard Worker
min_len(size_t a,size_t b)44*1a3d31e3SAndroid Build Coastguard Worker static inline size_t min_len(size_t a, size_t b)
45*1a3d31e3SAndroid Build Coastguard Worker {
46*1a3d31e3SAndroid Build Coastguard Worker return a < b ? a : b;
47*1a3d31e3SAndroid Build Coastguard Worker }
48*1a3d31e3SAndroid Build Coastguard Worker
convert_to_cpu(struct blk_io_trace * t,struct blk_io_trace * tp,void ** pdu)49*1a3d31e3SAndroid Build Coastguard Worker static inline size_t convert_to_cpu(struct blk_io_trace *t,
50*1a3d31e3SAndroid Build Coastguard Worker struct blk_io_trace *tp,
51*1a3d31e3SAndroid Build Coastguard Worker void **pdu)
52*1a3d31e3SAndroid Build Coastguard Worker {
53*1a3d31e3SAndroid Build Coastguard Worker if (data_is_native == -1)
54*1a3d31e3SAndroid Build Coastguard Worker check_data_endianness(t->magic);
55*1a3d31e3SAndroid Build Coastguard Worker
56*1a3d31e3SAndroid Build Coastguard Worker if (data_is_native)
57*1a3d31e3SAndroid Build Coastguard Worker memcpy(tp, t, sizeof(*tp));
58*1a3d31e3SAndroid Build Coastguard Worker else {
59*1a3d31e3SAndroid Build Coastguard Worker tp->magic = be32_to_cpu(t->magic);
60*1a3d31e3SAndroid Build Coastguard Worker tp->sequence = be32_to_cpu(t->sequence);
61*1a3d31e3SAndroid Build Coastguard Worker tp->time = be64_to_cpu(t->time);
62*1a3d31e3SAndroid Build Coastguard Worker tp->sector = be64_to_cpu(t->sector);
63*1a3d31e3SAndroid Build Coastguard Worker tp->bytes = be32_to_cpu(t->bytes);
64*1a3d31e3SAndroid Build Coastguard Worker tp->action = be32_to_cpu(t->action);
65*1a3d31e3SAndroid Build Coastguard Worker tp->pid = be32_to_cpu(t->pid);
66*1a3d31e3SAndroid Build Coastguard Worker tp->device = be32_to_cpu(t->device);
67*1a3d31e3SAndroid Build Coastguard Worker tp->cpu = be16_to_cpu(t->cpu);
68*1a3d31e3SAndroid Build Coastguard Worker tp->error = be16_to_cpu(t->error);
69*1a3d31e3SAndroid Build Coastguard Worker tp->pdu_len = be16_to_cpu(t->pdu_len);
70*1a3d31e3SAndroid Build Coastguard Worker }
71*1a3d31e3SAndroid Build Coastguard Worker
72*1a3d31e3SAndroid Build Coastguard Worker if (tp->pdu_len) {
73*1a3d31e3SAndroid Build Coastguard Worker *pdu = malloc(tp->pdu_len);
74*1a3d31e3SAndroid Build Coastguard Worker memcpy(*pdu, t+1, tp->pdu_len);
75*1a3d31e3SAndroid Build Coastguard Worker } else
76*1a3d31e3SAndroid Build Coastguard Worker *pdu = NULL;
77*1a3d31e3SAndroid Build Coastguard Worker
78*1a3d31e3SAndroid Build Coastguard Worker return sizeof(*tp) + tp->pdu_len;
79*1a3d31e3SAndroid Build Coastguard Worker }
80*1a3d31e3SAndroid Build Coastguard Worker
move_map(void)81*1a3d31e3SAndroid Build Coastguard Worker static int move_map(void)
82*1a3d31e3SAndroid Build Coastguard Worker {
83*1a3d31e3SAndroid Build Coastguard Worker if (cur_map != MAP_FAILED)
84*1a3d31e3SAndroid Build Coastguard Worker munmap(cur_map, len);
85*1a3d31e3SAndroid Build Coastguard Worker
86*1a3d31e3SAndroid Build Coastguard Worker cur_min = (cur & ~(pgsz-1));
87*1a3d31e3SAndroid Build Coastguard Worker len = min_len(DEF_LEN, total_size - cur_min);
88*1a3d31e3SAndroid Build Coastguard Worker if (len < sizeof(*next_t))
89*1a3d31e3SAndroid Build Coastguard Worker return 0;
90*1a3d31e3SAndroid Build Coastguard Worker
91*1a3d31e3SAndroid Build Coastguard Worker cur_map = mmap(NULL, len, PROT_READ, MAP_SHARED, fd,
92*1a3d31e3SAndroid Build Coastguard Worker cur_min);
93*1a3d31e3SAndroid Build Coastguard Worker if (cur_map == MAP_FAILED) {
94*1a3d31e3SAndroid Build Coastguard Worker perror("mmap");
95*1a3d31e3SAndroid Build Coastguard Worker exit(1);
96*1a3d31e3SAndroid Build Coastguard Worker }
97*1a3d31e3SAndroid Build Coastguard Worker
98*1a3d31e3SAndroid Build Coastguard Worker cur_max = cur_min + len;
99*1a3d31e3SAndroid Build Coastguard Worker return (cur < cur_max);
100*1a3d31e3SAndroid Build Coastguard Worker }
101*1a3d31e3SAndroid Build Coastguard Worker
setup_ifile(char * fname)102*1a3d31e3SAndroid Build Coastguard Worker void setup_ifile(char *fname)
103*1a3d31e3SAndroid Build Coastguard Worker {
104*1a3d31e3SAndroid Build Coastguard Worker struct stat buf;
105*1a3d31e3SAndroid Build Coastguard Worker
106*1a3d31e3SAndroid Build Coastguard Worker pgsz = sysconf(_SC_PAGESIZE);
107*1a3d31e3SAndroid Build Coastguard Worker
108*1a3d31e3SAndroid Build Coastguard Worker fd = my_open(fname, O_RDONLY);
109*1a3d31e3SAndroid Build Coastguard Worker if (fd < 0) {
110*1a3d31e3SAndroid Build Coastguard Worker perror(fname);
111*1a3d31e3SAndroid Build Coastguard Worker exit(1);
112*1a3d31e3SAndroid Build Coastguard Worker }
113*1a3d31e3SAndroid Build Coastguard Worker if (fstat(fd, &buf) < 0) {
114*1a3d31e3SAndroid Build Coastguard Worker perror(fname);
115*1a3d31e3SAndroid Build Coastguard Worker exit(1);
116*1a3d31e3SAndroid Build Coastguard Worker }
117*1a3d31e3SAndroid Build Coastguard Worker total_size = buf.st_size;
118*1a3d31e3SAndroid Build Coastguard Worker
119*1a3d31e3SAndroid Build Coastguard Worker if (!move_map())
120*1a3d31e3SAndroid Build Coastguard Worker exit(0);
121*1a3d31e3SAndroid Build Coastguard Worker }
122*1a3d31e3SAndroid Build Coastguard Worker
cleanup_ifile(void)123*1a3d31e3SAndroid Build Coastguard Worker void cleanup_ifile(void)
124*1a3d31e3SAndroid Build Coastguard Worker {
125*1a3d31e3SAndroid Build Coastguard Worker if (cur_map != MAP_FAILED)
126*1a3d31e3SAndroid Build Coastguard Worker munmap(cur_map, len);
127*1a3d31e3SAndroid Build Coastguard Worker close(fd);
128*1a3d31e3SAndroid Build Coastguard Worker }
129*1a3d31e3SAndroid Build Coastguard Worker
next_trace(struct blk_io_trace * t,void ** pdu)130*1a3d31e3SAndroid Build Coastguard Worker int next_trace(struct blk_io_trace *t, void **pdu)
131*1a3d31e3SAndroid Build Coastguard Worker {
132*1a3d31e3SAndroid Build Coastguard Worker size_t this_len;
133*1a3d31e3SAndroid Build Coastguard Worker
134*1a3d31e3SAndroid Build Coastguard Worker if ((cur + 512) > cur_max)
135*1a3d31e3SAndroid Build Coastguard Worker if (!move_map()) {
136*1a3d31e3SAndroid Build Coastguard Worker cleanup_ifile();
137*1a3d31e3SAndroid Build Coastguard Worker return 0;
138*1a3d31e3SAndroid Build Coastguard Worker }
139*1a3d31e3SAndroid Build Coastguard Worker
140*1a3d31e3SAndroid Build Coastguard Worker next_t = cur_map + (cur - cur_min);
141*1a3d31e3SAndroid Build Coastguard Worker this_len = convert_to_cpu(next_t, t, pdu);
142*1a3d31e3SAndroid Build Coastguard Worker cur += this_len;
143*1a3d31e3SAndroid Build Coastguard Worker
144*1a3d31e3SAndroid Build Coastguard Worker return 1;
145*1a3d31e3SAndroid Build Coastguard Worker }
146*1a3d31e3SAndroid Build Coastguard Worker
pct_done(void)147*1a3d31e3SAndroid Build Coastguard Worker double pct_done(void)
148*1a3d31e3SAndroid Build Coastguard Worker {
149*1a3d31e3SAndroid Build Coastguard Worker return 100.0 * ((double)cur / (double)total_size);
150*1a3d31e3SAndroid Build Coastguard Worker }
151