xref: /aosp_15_r20/system/extras/ioshark/ioshark_bench_subr.c (revision 288bf5226967eb3dac5cce6c939ccc2a7f2b4fe5)
1*288bf522SAndroid Build Coastguard Worker /*
2*288bf522SAndroid Build Coastguard Worker  * Copyright (C) 2016 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 #include <stdio.h>
18*288bf522SAndroid Build Coastguard Worker #include <sys/time.h>
19*288bf522SAndroid Build Coastguard Worker #include <sys/types.h>
20*288bf522SAndroid Build Coastguard Worker #include <unistd.h>
21*288bf522SAndroid Build Coastguard Worker #include <stdlib.h>
22*288bf522SAndroid Build Coastguard Worker #include <signal.h>
23*288bf522SAndroid Build Coastguard Worker #include <string.h>
24*288bf522SAndroid Build Coastguard Worker #include <sys/stat.h>
25*288bf522SAndroid Build Coastguard Worker #include <errno.h>
26*288bf522SAndroid Build Coastguard Worker #include <fcntl.h>
27*288bf522SAndroid Build Coastguard Worker #include <string.h>
28*288bf522SAndroid Build Coastguard Worker #include <assert.h>
29*288bf522SAndroid Build Coastguard Worker #include <sys/vfs.h>
30*288bf522SAndroid Build Coastguard Worker #include <sys/statvfs.h>
31*288bf522SAndroid Build Coastguard Worker #include <sys/mman.h>
32*288bf522SAndroid Build Coastguard Worker #include <inttypes.h>
33*288bf522SAndroid Build Coastguard Worker #include "ioshark.h"
34*288bf522SAndroid Build Coastguard Worker #include "ioshark_bench.h"
35*288bf522SAndroid Build Coastguard Worker #define _BSD_SOURCE
36*288bf522SAndroid Build Coastguard Worker #include <endian.h>
37*288bf522SAndroid Build Coastguard Worker 
38*288bf522SAndroid Build Coastguard Worker extern char *progname;
39*288bf522SAndroid Build Coastguard Worker extern int verbose, summary_mode;
40*288bf522SAndroid Build Coastguard Worker 
41*288bf522SAndroid Build Coastguard Worker void *
files_db_create_handle(void)42*288bf522SAndroid Build Coastguard Worker files_db_create_handle(void)
43*288bf522SAndroid Build Coastguard Worker {
44*288bf522SAndroid Build Coastguard Worker 	struct files_db_handle *h;
45*288bf522SAndroid Build Coastguard Worker 	int i;
46*288bf522SAndroid Build Coastguard Worker 
47*288bf522SAndroid Build Coastguard Worker 	h = malloc(sizeof(struct files_db_handle));
48*288bf522SAndroid Build Coastguard Worker 	for (i = 0 ; i < FILE_DB_HASHSIZE ; i++)
49*288bf522SAndroid Build Coastguard Worker 		h->files_db_buckets[i] = NULL;
50*288bf522SAndroid Build Coastguard Worker 	return h;
51*288bf522SAndroid Build Coastguard Worker }
52*288bf522SAndroid Build Coastguard Worker 
files_db_lookup_byfileno(void * handle,int fileno)53*288bf522SAndroid Build Coastguard Worker void *files_db_lookup_byfileno(void *handle, int fileno)
54*288bf522SAndroid Build Coastguard Worker {
55*288bf522SAndroid Build Coastguard Worker 	u_int32_t	hash;
56*288bf522SAndroid Build Coastguard Worker 	struct files_db_handle *h = (struct files_db_handle *)handle;
57*288bf522SAndroid Build Coastguard Worker 	struct files_db_s *db_node;
58*288bf522SAndroid Build Coastguard Worker 
59*288bf522SAndroid Build Coastguard Worker 	hash = fileno % FILE_DB_HASHSIZE;
60*288bf522SAndroid Build Coastguard Worker 	db_node = h->files_db_buckets[hash];
61*288bf522SAndroid Build Coastguard Worker 	while (db_node != NULL) {
62*288bf522SAndroid Build Coastguard Worker 		if (db_node->fileno == fileno)
63*288bf522SAndroid Build Coastguard Worker 			break;
64*288bf522SAndroid Build Coastguard Worker 		db_node = db_node->next;
65*288bf522SAndroid Build Coastguard Worker 	}
66*288bf522SAndroid Build Coastguard Worker 	return db_node;
67*288bf522SAndroid Build Coastguard Worker }
68*288bf522SAndroid Build Coastguard Worker 
files_db_add_byfileno(void * handle,int fileno,int readonly)69*288bf522SAndroid Build Coastguard Worker void *files_db_add_byfileno(void *handle, int fileno, int readonly)
70*288bf522SAndroid Build Coastguard Worker {
71*288bf522SAndroid Build Coastguard Worker 	u_int32_t	hash = fileno % FILE_DB_HASHSIZE;
72*288bf522SAndroid Build Coastguard Worker 	struct files_db_handle *h = (struct files_db_handle *)handle;
73*288bf522SAndroid Build Coastguard Worker 	struct files_db_s *db_node;
74*288bf522SAndroid Build Coastguard Worker 
75*288bf522SAndroid Build Coastguard Worker 	db_node = (struct files_db_s *)
76*288bf522SAndroid Build Coastguard Worker 		files_db_lookup_byfileno(handle, fileno);
77*288bf522SAndroid Build Coastguard Worker 	if (db_node == NULL) {
78*288bf522SAndroid Build Coastguard Worker 		db_node = malloc(sizeof(struct files_db_s));
79*288bf522SAndroid Build Coastguard Worker 		db_node->fileno = fileno;
80*288bf522SAndroid Build Coastguard Worker 		db_node->filename = NULL;
81*288bf522SAndroid Build Coastguard Worker 		db_node->readonly = readonly;
82*288bf522SAndroid Build Coastguard Worker 		db_node->size = 0;
83*288bf522SAndroid Build Coastguard Worker 		db_node->fd = -1;
84*288bf522SAndroid Build Coastguard Worker 		db_node->next = h->files_db_buckets[hash];
85*288bf522SAndroid Build Coastguard Worker 		h->files_db_buckets[hash] = db_node;
86*288bf522SAndroid Build Coastguard Worker 	} else {
87*288bf522SAndroid Build Coastguard Worker 		fprintf(stderr,
88*288bf522SAndroid Build Coastguard Worker 			"%s: Node to be added already exists fileno = %d\n\n",
89*288bf522SAndroid Build Coastguard Worker 			__func__, fileno);
90*288bf522SAndroid Build Coastguard Worker 		exit(EXIT_FAILURE);
91*288bf522SAndroid Build Coastguard Worker 	}
92*288bf522SAndroid Build Coastguard Worker 	return db_node;
93*288bf522SAndroid Build Coastguard Worker }
94*288bf522SAndroid Build Coastguard Worker 
95*288bf522SAndroid Build Coastguard Worker void
files_db_fsync_discard_files(void * handle)96*288bf522SAndroid Build Coastguard Worker files_db_fsync_discard_files(void *handle)
97*288bf522SAndroid Build Coastguard Worker {
98*288bf522SAndroid Build Coastguard Worker 	struct files_db_handle *h = (struct files_db_handle *)handle;
99*288bf522SAndroid Build Coastguard Worker 	struct files_db_s *db_node;
100*288bf522SAndroid Build Coastguard Worker 	int i;
101*288bf522SAndroid Build Coastguard Worker 
102*288bf522SAndroid Build Coastguard Worker 	for (i = 0 ; i < FILE_DB_HASHSIZE ; i++) {
103*288bf522SAndroid Build Coastguard Worker 		db_node = h->files_db_buckets[i];
104*288bf522SAndroid Build Coastguard Worker 		while (db_node != NULL) {
105*288bf522SAndroid Build Coastguard Worker 			int do_close = 0;
106*288bf522SAndroid Build Coastguard Worker 
107*288bf522SAndroid Build Coastguard Worker 			if (db_node->fd == -1) {
108*288bf522SAndroid Build Coastguard Worker 				int fd;
109*288bf522SAndroid Build Coastguard Worker 				int openflags;
110*288bf522SAndroid Build Coastguard Worker 
111*288bf522SAndroid Build Coastguard Worker 				/*n
112*288bf522SAndroid Build Coastguard Worker 				 * File was closed, let's open it so we can
113*288bf522SAndroid Build Coastguard Worker 				 * fsync and fadvise(DONTNEED) it.
114*288bf522SAndroid Build Coastguard Worker 				 */
115*288bf522SAndroid Build Coastguard Worker 				do_close = 1;
116*288bf522SAndroid Build Coastguard Worker 				if (files_db_readonly(db_node))
117*288bf522SAndroid Build Coastguard Worker 					openflags = O_RDONLY;
118*288bf522SAndroid Build Coastguard Worker 				else
119*288bf522SAndroid Build Coastguard Worker 					openflags = O_RDWR;
120*288bf522SAndroid Build Coastguard Worker 				fd = open(files_db_get_filename(db_node),
121*288bf522SAndroid Build Coastguard Worker 					  openflags);
122*288bf522SAndroid Build Coastguard Worker 				if (fd < 0) {
123*288bf522SAndroid Build Coastguard Worker 					fprintf(stderr,
124*288bf522SAndroid Build Coastguard Worker 						"%s: open(%s %x): %m\n",
125*288bf522SAndroid Build Coastguard Worker 						progname, db_node->filename,
126*288bf522SAndroid Build Coastguard Worker 						openflags);
127*288bf522SAndroid Build Coastguard Worker 					exit(EXIT_FAILURE);
128*288bf522SAndroid Build Coastguard Worker 				}
129*288bf522SAndroid Build Coastguard Worker 				db_node->fd = fd;
130*288bf522SAndroid Build Coastguard Worker 			}
131*288bf522SAndroid Build Coastguard Worker 			if (!db_node->readonly && fsync(db_node->fd) < 0) {
132*288bf522SAndroid Build Coastguard Worker 				fprintf(stderr, "%s: Cannot fsync %s\n",
133*288bf522SAndroid Build Coastguard Worker 					__func__, db_node->filename);
134*288bf522SAndroid Build Coastguard Worker 				exit(1);
135*288bf522SAndroid Build Coastguard Worker 			}
136*288bf522SAndroid Build Coastguard Worker 			if (posix_fadvise(db_node->fd, 0, 0,
137*288bf522SAndroid Build Coastguard Worker 					  POSIX_FADV_DONTNEED) < 0) {
138*288bf522SAndroid Build Coastguard Worker 				fprintf(stderr,
139*288bf522SAndroid Build Coastguard Worker 					"%s: Cannot fadvise(DONTNEED) %s\n",
140*288bf522SAndroid Build Coastguard Worker 					__func__, db_node->filename);
141*288bf522SAndroid Build Coastguard Worker 				exit(1);
142*288bf522SAndroid Build Coastguard Worker 			}
143*288bf522SAndroid Build Coastguard Worker 			if (do_close) {
144*288bf522SAndroid Build Coastguard Worker 				close(db_node->fd);
145*288bf522SAndroid Build Coastguard Worker 				db_node->fd = -1;
146*288bf522SAndroid Build Coastguard Worker 			}
147*288bf522SAndroid Build Coastguard Worker 			db_node = db_node->next;
148*288bf522SAndroid Build Coastguard Worker 		}
149*288bf522SAndroid Build Coastguard Worker 	}
150*288bf522SAndroid Build Coastguard Worker }
151*288bf522SAndroid Build Coastguard Worker 
152*288bf522SAndroid Build Coastguard Worker void
files_db_update_fd(void * node,int fd)153*288bf522SAndroid Build Coastguard Worker files_db_update_fd(void *node, int fd)
154*288bf522SAndroid Build Coastguard Worker {
155*288bf522SAndroid Build Coastguard Worker 	struct files_db_s *db_node = (struct files_db_s *)node;
156*288bf522SAndroid Build Coastguard Worker 
157*288bf522SAndroid Build Coastguard Worker 	db_node->fd = fd;
158*288bf522SAndroid Build Coastguard Worker }
159*288bf522SAndroid Build Coastguard Worker 
160*288bf522SAndroid Build Coastguard Worker void
files_db_close_fd(void * node)161*288bf522SAndroid Build Coastguard Worker files_db_close_fd(void *node)
162*288bf522SAndroid Build Coastguard Worker {
163*288bf522SAndroid Build Coastguard Worker 	struct files_db_s *db_node = (struct files_db_s *)node;
164*288bf522SAndroid Build Coastguard Worker 
165*288bf522SAndroid Build Coastguard Worker 	if (db_node->fd != -1)
166*288bf522SAndroid Build Coastguard Worker 		close(db_node->fd);
167*288bf522SAndroid Build Coastguard Worker 	db_node->fd = -1;
168*288bf522SAndroid Build Coastguard Worker }
169*288bf522SAndroid Build Coastguard Worker 
170*288bf522SAndroid Build Coastguard Worker void
files_db_close_files(void * handle)171*288bf522SAndroid Build Coastguard Worker files_db_close_files(void *handle)
172*288bf522SAndroid Build Coastguard Worker {
173*288bf522SAndroid Build Coastguard Worker 	struct files_db_handle *h = (struct files_db_handle *)handle;
174*288bf522SAndroid Build Coastguard Worker 	struct files_db_s *db_node;
175*288bf522SAndroid Build Coastguard Worker 	int i;
176*288bf522SAndroid Build Coastguard Worker 
177*288bf522SAndroid Build Coastguard Worker 	for (i = 0 ; i < FILE_DB_HASHSIZE ; i++) {
178*288bf522SAndroid Build Coastguard Worker 		db_node = h->files_db_buckets[i];
179*288bf522SAndroid Build Coastguard Worker 		while (db_node != NULL) {
180*288bf522SAndroid Build Coastguard Worker 			if ((db_node->fd != -1) && close(db_node->fd) < 0) {
181*288bf522SAndroid Build Coastguard Worker 				fprintf(stderr, "%s: Cannot close %s\n",
182*288bf522SAndroid Build Coastguard Worker 					__func__, db_node->filename);
183*288bf522SAndroid Build Coastguard Worker 				exit(1);
184*288bf522SAndroid Build Coastguard Worker 			}
185*288bf522SAndroid Build Coastguard Worker 			db_node->fd = -1;
186*288bf522SAndroid Build Coastguard Worker 			db_node = db_node->next;
187*288bf522SAndroid Build Coastguard Worker 		}
188*288bf522SAndroid Build Coastguard Worker 	}
189*288bf522SAndroid Build Coastguard Worker }
190*288bf522SAndroid Build Coastguard Worker 
191*288bf522SAndroid Build Coastguard Worker void
files_db_unlink_files(void * handle)192*288bf522SAndroid Build Coastguard Worker files_db_unlink_files(void *handle)
193*288bf522SAndroid Build Coastguard Worker {
194*288bf522SAndroid Build Coastguard Worker 	struct files_db_handle *h = (struct files_db_handle *)handle;
195*288bf522SAndroid Build Coastguard Worker 	struct files_db_s *db_node;
196*288bf522SAndroid Build Coastguard Worker 	int i;
197*288bf522SAndroid Build Coastguard Worker 
198*288bf522SAndroid Build Coastguard Worker 	for (i = 0 ; i < FILE_DB_HASHSIZE ; i++) {
199*288bf522SAndroid Build Coastguard Worker 		db_node = h->files_db_buckets[i];
200*288bf522SAndroid Build Coastguard Worker 		while (db_node != NULL) {
201*288bf522SAndroid Build Coastguard Worker 			if ((db_node->fd != -1) && close(db_node->fd) < 0) {
202*288bf522SAndroid Build Coastguard Worker 				fprintf(stderr, "%s: Cannot close %s\n",
203*288bf522SAndroid Build Coastguard Worker 					__func__, db_node->filename);
204*288bf522SAndroid Build Coastguard Worker 				exit(1);
205*288bf522SAndroid Build Coastguard Worker 			}
206*288bf522SAndroid Build Coastguard Worker 			db_node->fd = -1;
207*288bf522SAndroid Build Coastguard Worker 			if (is_readonly_mount(db_node->filename, db_node->size) == 0) {
208*288bf522SAndroid Build Coastguard Worker 				if (unlink(db_node->filename) < 0) {
209*288bf522SAndroid Build Coastguard Worker 					fprintf(stderr, "%s: Cannot unlink %s: %m\n",
210*288bf522SAndroid Build Coastguard Worker 						__func__, db_node->filename);
211*288bf522SAndroid Build Coastguard Worker 					exit(EXIT_FAILURE);
212*288bf522SAndroid Build Coastguard Worker 				}
213*288bf522SAndroid Build Coastguard Worker 			}
214*288bf522SAndroid Build Coastguard Worker 			db_node = db_node->next;
215*288bf522SAndroid Build Coastguard Worker 		}
216*288bf522SAndroid Build Coastguard Worker 	}
217*288bf522SAndroid Build Coastguard Worker }
218*288bf522SAndroid Build Coastguard Worker 
219*288bf522SAndroid Build Coastguard Worker void
files_db_free_memory(void * handle)220*288bf522SAndroid Build Coastguard Worker files_db_free_memory(void *handle)
221*288bf522SAndroid Build Coastguard Worker {
222*288bf522SAndroid Build Coastguard Worker 	struct files_db_handle *h = (struct files_db_handle *)handle;
223*288bf522SAndroid Build Coastguard Worker 	struct files_db_s *db_node, *tmp;
224*288bf522SAndroid Build Coastguard Worker 	int i;
225*288bf522SAndroid Build Coastguard Worker 
226*288bf522SAndroid Build Coastguard Worker 	for (i = 0 ; i < FILE_DB_HASHSIZE ; i++) {
227*288bf522SAndroid Build Coastguard Worker 		db_node = h->files_db_buckets[i];
228*288bf522SAndroid Build Coastguard Worker 		while (db_node != NULL) {
229*288bf522SAndroid Build Coastguard Worker 			tmp = db_node;
230*288bf522SAndroid Build Coastguard Worker 			db_node = db_node->next;
231*288bf522SAndroid Build Coastguard Worker 			free(tmp->filename);
232*288bf522SAndroid Build Coastguard Worker 			free(tmp);
233*288bf522SAndroid Build Coastguard Worker 		}
234*288bf522SAndroid Build Coastguard Worker 	}
235*288bf522SAndroid Build Coastguard Worker 	free(h);
236*288bf522SAndroid Build Coastguard Worker }
237*288bf522SAndroid Build Coastguard Worker 
238*288bf522SAndroid Build Coastguard Worker char *
get_buf(char ** buf,int * buflen,int len,int do_fill)239*288bf522SAndroid Build Coastguard Worker get_buf(char **buf, int *buflen, int len, int do_fill __attribute__((unused)))
240*288bf522SAndroid Build Coastguard Worker {
241*288bf522SAndroid Build Coastguard Worker 	if (len == 0 && *buf == NULL) {
242*288bf522SAndroid Build Coastguard Worker 		/*
243*288bf522SAndroid Build Coastguard Worker 		 * If we ever get a zero len
244*288bf522SAndroid Build Coastguard Worker 		 * request, start with MINBUFLEN
245*288bf522SAndroid Build Coastguard Worker 		 */
246*288bf522SAndroid Build Coastguard Worker 		if (*buf == NULL)
247*288bf522SAndroid Build Coastguard Worker 			len = MINBUFLEN / 2;
248*288bf522SAndroid Build Coastguard Worker 	}
249*288bf522SAndroid Build Coastguard Worker 	if (*buflen < len) {
250*288bf522SAndroid Build Coastguard Worker 		*buflen = MAX(MINBUFLEN, len * 2);
251*288bf522SAndroid Build Coastguard Worker 		if (*buf)
252*288bf522SAndroid Build Coastguard Worker 			free(*buf);
253*288bf522SAndroid Build Coastguard Worker 		*buf = malloc(*buflen);
254*288bf522SAndroid Build Coastguard Worker 		if (do_fill) {
255*288bf522SAndroid Build Coastguard Worker 			u_int32_t *s;
256*288bf522SAndroid Build Coastguard Worker 			int count;
257*288bf522SAndroid Build Coastguard Worker 
258*288bf522SAndroid Build Coastguard Worker 			s = (u_int32_t *)*buf;
259*288bf522SAndroid Build Coastguard Worker 			count = *buflen / sizeof(u_int32_t);
260*288bf522SAndroid Build Coastguard Worker 			while (count > 0) {
261*288bf522SAndroid Build Coastguard Worker 				*s++ = rand();
262*288bf522SAndroid Build Coastguard Worker 				count--;
263*288bf522SAndroid Build Coastguard Worker 			}
264*288bf522SAndroid Build Coastguard Worker 		}
265*288bf522SAndroid Build Coastguard Worker 	}
266*288bf522SAndroid Build Coastguard Worker 	assert(*buf != NULL);
267*288bf522SAndroid Build Coastguard Worker 	return *buf;
268*288bf522SAndroid Build Coastguard Worker }
269*288bf522SAndroid Build Coastguard Worker 
270*288bf522SAndroid Build Coastguard Worker void
create_file(char * path,size_t size,struct rw_bytes_s * rw_bytes)271*288bf522SAndroid Build Coastguard Worker create_file(char *path, size_t size, struct rw_bytes_s *rw_bytes)
272*288bf522SAndroid Build Coastguard Worker {
273*288bf522SAndroid Build Coastguard Worker 	int fd, n;
274*288bf522SAndroid Build Coastguard Worker 	char *buf = NULL;
275*288bf522SAndroid Build Coastguard Worker 	int buflen = 0;
276*288bf522SAndroid Build Coastguard Worker 
277*288bf522SAndroid Build Coastguard Worker 	fd = open(path, O_WRONLY|O_CREAT|O_TRUNC, 0644);
278*288bf522SAndroid Build Coastguard Worker 	if (fd < 0) {
279*288bf522SAndroid Build Coastguard Worker 		fprintf(stderr, "%s Cannot create file %s: %m\n", progname, path);
280*288bf522SAndroid Build Coastguard Worker 		exit(EXIT_FAILURE);
281*288bf522SAndroid Build Coastguard Worker 	}
282*288bf522SAndroid Build Coastguard Worker 	while (size > 0) {
283*288bf522SAndroid Build Coastguard Worker 		n = MIN(size, MINBUFLEN);
284*288bf522SAndroid Build Coastguard Worker 		buf = get_buf(&buf, &buflen, n, 1);
285*288bf522SAndroid Build Coastguard Worker 		if (write(fd, buf, n) < n) {
286*288bf522SAndroid Build Coastguard Worker 			fprintf(stderr,
287*288bf522SAndroid Build Coastguard Worker 				"%s Cannot write file %s: %m\n", progname, path);
288*288bf522SAndroid Build Coastguard Worker 			free(buf);
289*288bf522SAndroid Build Coastguard Worker 			exit(EXIT_FAILURE);
290*288bf522SAndroid Build Coastguard Worker 		}
291*288bf522SAndroid Build Coastguard Worker 		rw_bytes->bytes_written += n;
292*288bf522SAndroid Build Coastguard Worker 		size -= n;
293*288bf522SAndroid Build Coastguard Worker 	}
294*288bf522SAndroid Build Coastguard Worker 	free(buf);
295*288bf522SAndroid Build Coastguard Worker 	if (fsync(fd) < 0) {
296*288bf522SAndroid Build Coastguard Worker 		fprintf(stderr, "%s Cannot fsync file %s: %m\n", progname, path);
297*288bf522SAndroid Build Coastguard Worker 		exit(EXIT_FAILURE);
298*288bf522SAndroid Build Coastguard Worker 	}
299*288bf522SAndroid Build Coastguard Worker 	if ((errno = posix_fadvise(fd, 0, 0, POSIX_FADV_DONTNEED)) != 0) {
300*288bf522SAndroid Build Coastguard Worker 		fprintf(stderr,
301*288bf522SAndroid Build Coastguard Worker 			"%s Cannot fadvise(DONTNEED) file %s: %m\n", progname, path);
302*288bf522SAndroid Build Coastguard Worker 		exit(EXIT_FAILURE);
303*288bf522SAndroid Build Coastguard Worker 	}
304*288bf522SAndroid Build Coastguard Worker 	close(fd);
305*288bf522SAndroid Build Coastguard Worker }
306*288bf522SAndroid Build Coastguard Worker 
307*288bf522SAndroid Build Coastguard Worker void
print_op_stats(u_int64_t * op_counts)308*288bf522SAndroid Build Coastguard Worker print_op_stats(u_int64_t *op_counts)
309*288bf522SAndroid Build Coastguard Worker {
310*288bf522SAndroid Build Coastguard Worker 	int i;
311*288bf522SAndroid Build Coastguard Worker 	extern char *IO_op[];
312*288bf522SAndroid Build Coastguard Worker 
313*288bf522SAndroid Build Coastguard Worker 	printf("IO Operation counts :\n");
314*288bf522SAndroid Build Coastguard Worker 	for (i = IOSHARK_LSEEK ; i < IOSHARK_MAX_FILE_OP ; i++) {
315*288bf522SAndroid Build Coastguard Worker 		printf("%s: %ju\n",
316*288bf522SAndroid Build Coastguard Worker 		       IO_op[i], op_counts[i]);
317*288bf522SAndroid Build Coastguard Worker 	}
318*288bf522SAndroid Build Coastguard Worker }
319*288bf522SAndroid Build Coastguard Worker 
320*288bf522SAndroid Build Coastguard Worker void
print_bytes(char * desc,struct rw_bytes_s * rw_bytes)321*288bf522SAndroid Build Coastguard Worker print_bytes(char *desc, struct rw_bytes_s *rw_bytes)
322*288bf522SAndroid Build Coastguard Worker {
323*288bf522SAndroid Build Coastguard Worker 	if (!summary_mode)
324*288bf522SAndroid Build Coastguard Worker 		printf("%s: Reads = %dMB, Writes = %dMB\n",
325*288bf522SAndroid Build Coastguard Worker 		       desc,
326*288bf522SAndroid Build Coastguard Worker 		       (int)(rw_bytes->bytes_read / (1024 * 1024)),
327*288bf522SAndroid Build Coastguard Worker 		       (int)(rw_bytes->bytes_written / (1024 * 1024)));
328*288bf522SAndroid Build Coastguard Worker 	else
329*288bf522SAndroid Build Coastguard Worker 		printf("%d %d ",
330*288bf522SAndroid Build Coastguard Worker 		       (int)(rw_bytes->bytes_read / (1024 * 1024)),
331*288bf522SAndroid Build Coastguard Worker 		       (int)(rw_bytes->bytes_written / (1024 * 1024)));
332*288bf522SAndroid Build Coastguard Worker }
333*288bf522SAndroid Build Coastguard Worker 
334*288bf522SAndroid Build Coastguard Worker struct cpu_disk_util_stats {
335*288bf522SAndroid Build Coastguard Worker 	/* CPU util */
336*288bf522SAndroid Build Coastguard Worker 	u_int64_t user_cpu_ticks;
337*288bf522SAndroid Build Coastguard Worker 	u_int64_t nice_cpu_ticks;
338*288bf522SAndroid Build Coastguard Worker 	u_int64_t system_cpu_ticks;
339*288bf522SAndroid Build Coastguard Worker 	u_int64_t idle_cpu_ticks;
340*288bf522SAndroid Build Coastguard Worker 	u_int64_t iowait_cpu_ticks;
341*288bf522SAndroid Build Coastguard Worker 	u_int64_t hardirq_cpu_ticks;
342*288bf522SAndroid Build Coastguard Worker 	u_int64_t softirq_cpu_ticks;
343*288bf522SAndroid Build Coastguard Worker 	/* disk util */
344*288bf522SAndroid Build Coastguard Worker 	unsigned long long uptime;
345*288bf522SAndroid Build Coastguard Worker 	unsigned int tot_ticks;
346*288bf522SAndroid Build Coastguard Worker 	unsigned long rd_ios;
347*288bf522SAndroid Build Coastguard Worker 	unsigned long wr_ios;
348*288bf522SAndroid Build Coastguard Worker 	unsigned long rd_sec;
349*288bf522SAndroid Build Coastguard Worker 	unsigned long wr_sec;
350*288bf522SAndroid Build Coastguard Worker };
351*288bf522SAndroid Build Coastguard Worker 
352*288bf522SAndroid Build Coastguard Worker static struct cpu_disk_util_stats before;
353*288bf522SAndroid Build Coastguard Worker static struct cpu_disk_util_stats after;
354*288bf522SAndroid Build Coastguard Worker 
355*288bf522SAndroid Build Coastguard Worker #define BUFSIZE		8192
356*288bf522SAndroid Build Coastguard Worker 
357*288bf522SAndroid Build Coastguard Worker static int hz;
358*288bf522SAndroid Build Coastguard Worker 
359*288bf522SAndroid Build Coastguard Worker static void
get_HZ(void)360*288bf522SAndroid Build Coastguard Worker get_HZ(void)
361*288bf522SAndroid Build Coastguard Worker {
362*288bf522SAndroid Build Coastguard Worker 	if ((hz = sysconf(_SC_CLK_TCK)) == -1)
363*288bf522SAndroid Build Coastguard Worker 		exit(1);
364*288bf522SAndroid Build Coastguard Worker }
365*288bf522SAndroid Build Coastguard Worker 
366*288bf522SAndroid Build Coastguard Worker #if 0
367*288bf522SAndroid Build Coastguard Worker static int num_cores;
368*288bf522SAndroid Build Coastguard Worker 
369*288bf522SAndroid Build Coastguard Worker static void
370*288bf522SAndroid Build Coastguard Worker get_cores(void)
371*288bf522SAndroid Build Coastguard Worker {
372*288bf522SAndroid Build Coastguard Worker 	if ((num_cores = sysconf(_SC_NPROCESSORS_ONLN)) == -1)
373*288bf522SAndroid Build Coastguard Worker 		exit(1);
374*288bf522SAndroid Build Coastguard Worker }
375*288bf522SAndroid Build Coastguard Worker #endif
376*288bf522SAndroid Build Coastguard Worker 
377*288bf522SAndroid Build Coastguard Worker extern char *blockdev_name;
378*288bf522SAndroid Build Coastguard Worker 
379*288bf522SAndroid Build Coastguard Worker static void
get_blockdev_name(char * bdev)380*288bf522SAndroid Build Coastguard Worker get_blockdev_name(char *bdev)
381*288bf522SAndroid Build Coastguard Worker {
382*288bf522SAndroid Build Coastguard Worker 	char dev_name[BUFSIZE];
383*288bf522SAndroid Build Coastguard Worker 	FILE *cmd;
384*288bf522SAndroid Build Coastguard Worker 
385*288bf522SAndroid Build Coastguard Worker 	cmd = popen("getprop ro.product.name", "r");
386*288bf522SAndroid Build Coastguard Worker 	if (cmd == NULL) {
387*288bf522SAndroid Build Coastguard Worker 		fprintf(stderr, "%s: Cannot popen getprop\n",
388*288bf522SAndroid Build Coastguard Worker 			progname);
389*288bf522SAndroid Build Coastguard Worker 		exit(1);
390*288bf522SAndroid Build Coastguard Worker 	}
391*288bf522SAndroid Build Coastguard Worker 	if (fgets(dev_name, BUFSIZE, cmd) == NULL) {
392*288bf522SAndroid Build Coastguard Worker 		fprintf(stderr,
393*288bf522SAndroid Build Coastguard Worker 			"%s: Bad output from getprop ro.product.name\n",
394*288bf522SAndroid Build Coastguard Worker 			progname);
395*288bf522SAndroid Build Coastguard Worker 		exit(1);
396*288bf522SAndroid Build Coastguard Worker 	}
397*288bf522SAndroid Build Coastguard Worker 	pclose(cmd);
398*288bf522SAndroid Build Coastguard Worker 	/* strncmp needed because of the trailing '\n' */
399*288bf522SAndroid Build Coastguard Worker 	if (strncmp(dev_name, "bullhead", strlen("bullhead")) == 0 ||
400*288bf522SAndroid Build Coastguard Worker 	    strncmp(dev_name, "angler", strlen("angler")) == 0 ||
401*288bf522SAndroid Build Coastguard Worker 	    strncmp(dev_name, "shamu", strlen("shamu")) == 0 ||
402*288bf522SAndroid Build Coastguard Worker 	    strncmp(dev_name, "aosp_gobo", strlen("aosp_gobo")) == 0 ||
403*288bf522SAndroid Build Coastguard Worker 	    strncmp(dev_name, "full_k37_y33_gms", strlen("full_k37_y33_gms")) == 0 ||
404*288bf522SAndroid Build Coastguard Worker 	    strncmp(dev_name, "fugu", strlen("fugu")) == 0) {
405*288bf522SAndroid Build Coastguard Worker 		strcpy(bdev, "mmcblk0");
406*288bf522SAndroid Build Coastguard Worker 	} else if (strncmp(dev_name, "marlin", strlen("marlin")) == 0 ||
407*288bf522SAndroid Build Coastguard Worker 		   strncmp(dev_name, "sailfish", strlen("sailfish")) == 0 ||
408*288bf522SAndroid Build Coastguard Worker 		   strncmp(dev_name, "taimen", strlen("taimen")) == 0 ||
409*288bf522SAndroid Build Coastguard Worker 		   strncmp(dev_name, "walleye", strlen("walleye")) == 0) {
410*288bf522SAndroid Build Coastguard Worker 		strcpy(bdev, "sda");
411*288bf522SAndroid Build Coastguard Worker 	} else if (blockdev_name != NULL) {
412*288bf522SAndroid Build Coastguard Worker 		strcpy(bdev, blockdev_name);
413*288bf522SAndroid Build Coastguard Worker 	} else {
414*288bf522SAndroid Build Coastguard Worker 		fprintf(stderr,
415*288bf522SAndroid Build Coastguard Worker 			"%s: Unknown device %s, please specify block device name with -b\n",
416*288bf522SAndroid Build Coastguard Worker 			progname, dev_name);
417*288bf522SAndroid Build Coastguard Worker 		exit(1);
418*288bf522SAndroid Build Coastguard Worker 	}
419*288bf522SAndroid Build Coastguard Worker }
420*288bf522SAndroid Build Coastguard Worker 
421*288bf522SAndroid Build Coastguard Worker static void
read_disk_util_state(struct cpu_disk_util_stats * state)422*288bf522SAndroid Build Coastguard Worker read_disk_util_state(struct cpu_disk_util_stats *state)
423*288bf522SAndroid Build Coastguard Worker {
424*288bf522SAndroid Build Coastguard Worker 	FILE *fp;
425*288bf522SAndroid Build Coastguard Worker         char line[BUFSIZE], dev_name[BUFSIZE];
426*288bf522SAndroid Build Coastguard Worker         unsigned int major, minor;
427*288bf522SAndroid Build Coastguard Worker 	unsigned int ios_pgr;
428*288bf522SAndroid Build Coastguard Worker 	unsigned int rq_ticks;
429*288bf522SAndroid Build Coastguard Worker 	unsigned int wr_ticks;
430*288bf522SAndroid Build Coastguard Worker 	unsigned long rd_ticks;
431*288bf522SAndroid Build Coastguard Worker 	unsigned long rd_merges;
432*288bf522SAndroid Build Coastguard Worker 	unsigned long wr_merges;
433*288bf522SAndroid Build Coastguard Worker 	unsigned long up_sec, up_cent;
434*288bf522SAndroid Build Coastguard Worker 	char blockdev_name[BUFSIZE];
435*288bf522SAndroid Build Coastguard Worker 
436*288bf522SAndroid Build Coastguard Worker 	/* Read and parse /proc/uptime */
437*288bf522SAndroid Build Coastguard Worker 	fp = fopen("/proc/uptime", "r");
438*288bf522SAndroid Build Coastguard Worker 	if (fgets(line, sizeof(line), fp) == NULL) {
439*288bf522SAndroid Build Coastguard Worker 		fprintf(stderr, "%s: Cannot read /proc/uptime\n",
440*288bf522SAndroid Build Coastguard Worker 			progname);
441*288bf522SAndroid Build Coastguard Worker 		exit(1);
442*288bf522SAndroid Build Coastguard Worker 	}
443*288bf522SAndroid Build Coastguard Worker 	fclose(fp);
444*288bf522SAndroid Build Coastguard Worker 	sscanf(line, "%lu.%lu", &up_sec, &up_cent);
445*288bf522SAndroid Build Coastguard Worker 	state->uptime = (unsigned long long) up_sec * hz +
446*288bf522SAndroid Build Coastguard Worker 		(unsigned long long) up_cent * hz / 100;
447*288bf522SAndroid Build Coastguard Worker 
448*288bf522SAndroid Build Coastguard Worker 	/* Read and parse /proc/diskstats */
449*288bf522SAndroid Build Coastguard Worker 	get_blockdev_name(blockdev_name);
450*288bf522SAndroid Build Coastguard Worker 	fp = fopen("/proc/diskstats", "r");
451*288bf522SAndroid Build Coastguard Worker 	while (fgets(line, sizeof(line), fp)) {
452*288bf522SAndroid Build Coastguard Worker 		sscanf(line,
453*288bf522SAndroid Build Coastguard Worker 		       "%u %u %s %lu %lu %lu %lu %lu %lu %lu %u %u %u %u",
454*288bf522SAndroid Build Coastguard Worker 		       &major, &minor, dev_name,
455*288bf522SAndroid Build Coastguard Worker 		       &state->rd_ios, &rd_merges, &state->rd_sec,
456*288bf522SAndroid Build Coastguard Worker 		       &rd_ticks, &state->wr_ios, &wr_merges,
457*288bf522SAndroid Build Coastguard Worker 		       &state->wr_sec, &wr_ticks,
458*288bf522SAndroid Build Coastguard Worker 		       &ios_pgr, &state->tot_ticks, &rq_ticks);
459*288bf522SAndroid Build Coastguard Worker                 if (strcmp(dev_name, blockdev_name) == 0) {
460*288bf522SAndroid Build Coastguard Worker 			/*
461*288bf522SAndroid Build Coastguard Worker 			 * tot_ticks is "number of milliseconds spent
462*288bf522SAndroid Build Coastguard Worker 			 * doing I/Os". Look at Documentation/iostats.txt.
463*288bf522SAndroid Build Coastguard Worker 			 * Or at genhd.c:diskstats_show(), which calls
464*288bf522SAndroid Build Coastguard Worker 			 * jiffies_to_msecs() on this field before printing
465*288bf522SAndroid Build Coastguard Worker 			 * it. Convert this to hz, so we can do all our math
466*288bf522SAndroid Build Coastguard Worker 			 * in ticks.
467*288bf522SAndroid Build Coastguard Worker 			 */
468*288bf522SAndroid Build Coastguard Worker 			state->tot_ticks /= 1000; /* to seconds */
469*288bf522SAndroid Build Coastguard Worker 			state->tot_ticks *= hz;   /* to hz	*/
470*288bf522SAndroid Build Coastguard Worker 			fclose(fp);
471*288bf522SAndroid Build Coastguard Worker 			return;
472*288bf522SAndroid Build Coastguard Worker 		}
473*288bf522SAndroid Build Coastguard Worker 	}
474*288bf522SAndroid Build Coastguard Worker         fprintf(stderr, "%s: Did not find device sda in /proc/diskstats\n",
475*288bf522SAndroid Build Coastguard Worker 		progname);
476*288bf522SAndroid Build Coastguard Worker 	exit(1);
477*288bf522SAndroid Build Coastguard Worker }
478*288bf522SAndroid Build Coastguard Worker 
479*288bf522SAndroid Build Coastguard Worker static void
read_cpu_util_state(struct cpu_disk_util_stats * state)480*288bf522SAndroid Build Coastguard Worker read_cpu_util_state(struct cpu_disk_util_stats *state)
481*288bf522SAndroid Build Coastguard Worker {
482*288bf522SAndroid Build Coastguard Worker 	FILE *fp;
483*288bf522SAndroid Build Coastguard Worker 	char line[BUFSIZE], cpu[BUFSIZE];
484*288bf522SAndroid Build Coastguard Worker 
485*288bf522SAndroid Build Coastguard Worker 	/* Read and parse /proc/stat */
486*288bf522SAndroid Build Coastguard Worker 	fp = fopen("/proc/stat", "r");
487*288bf522SAndroid Build Coastguard Worker 	if (fgets(line, sizeof(line), fp) == NULL) {
488*288bf522SAndroid Build Coastguard Worker 		fprintf(stderr, "%s: Cannot read /proc/stat\n",
489*288bf522SAndroid Build Coastguard Worker 			progname);
490*288bf522SAndroid Build Coastguard Worker 		exit(1);
491*288bf522SAndroid Build Coastguard Worker 	}
492*288bf522SAndroid Build Coastguard Worker 	fclose(fp);
493*288bf522SAndroid Build Coastguard Worker 	sscanf(line, "%s %ju %ju %ju %ju %ju %ju %ju",
494*288bf522SAndroid Build Coastguard Worker 	       cpu,
495*288bf522SAndroid Build Coastguard Worker 	       &state->user_cpu_ticks,
496*288bf522SAndroid Build Coastguard Worker 	       &state->nice_cpu_ticks,
497*288bf522SAndroid Build Coastguard Worker 	       &state->system_cpu_ticks,
498*288bf522SAndroid Build Coastguard Worker 	       &state->idle_cpu_ticks,
499*288bf522SAndroid Build Coastguard Worker 	       &state->iowait_cpu_ticks,
500*288bf522SAndroid Build Coastguard Worker 	       &state->hardirq_cpu_ticks,
501*288bf522SAndroid Build Coastguard Worker 	       &state->softirq_cpu_ticks);
502*288bf522SAndroid Build Coastguard Worker }
503*288bf522SAndroid Build Coastguard Worker 
504*288bf522SAndroid Build Coastguard Worker void
capture_util_state_before(void)505*288bf522SAndroid Build Coastguard Worker capture_util_state_before(void)
506*288bf522SAndroid Build Coastguard Worker {
507*288bf522SAndroid Build Coastguard Worker 	get_HZ();
508*288bf522SAndroid Build Coastguard Worker 	read_disk_util_state(&before);
509*288bf522SAndroid Build Coastguard Worker 	read_cpu_util_state(&before);
510*288bf522SAndroid Build Coastguard Worker }
511*288bf522SAndroid Build Coastguard Worker 
512*288bf522SAndroid Build Coastguard Worker void
report_cpu_disk_util(void)513*288bf522SAndroid Build Coastguard Worker report_cpu_disk_util(void)
514*288bf522SAndroid Build Coastguard Worker {
515*288bf522SAndroid Build Coastguard Worker         double disk_util, cpu_util;
516*288bf522SAndroid Build Coastguard Worker 	u_int64_t tot1, tot2, delta1, delta2;
517*288bf522SAndroid Build Coastguard Worker 
518*288bf522SAndroid Build Coastguard Worker 	read_disk_util_state(&after);
519*288bf522SAndroid Build Coastguard Worker 	read_cpu_util_state(&after);
520*288bf522SAndroid Build Coastguard Worker 	/* CPU Util */
521*288bf522SAndroid Build Coastguard Worker 	tot2 = after.user_cpu_ticks + after.nice_cpu_ticks +
522*288bf522SAndroid Build Coastguard Worker 		after.system_cpu_ticks + after.hardirq_cpu_ticks +
523*288bf522SAndroid Build Coastguard Worker 		after.softirq_cpu_ticks;
524*288bf522SAndroid Build Coastguard Worker 	tot1 = before.user_cpu_ticks + before.nice_cpu_ticks +
525*288bf522SAndroid Build Coastguard Worker 		before.system_cpu_ticks + before.hardirq_cpu_ticks +
526*288bf522SAndroid Build Coastguard Worker 		before.softirq_cpu_ticks;
527*288bf522SAndroid Build Coastguard Worker 	delta1 = tot2 - tot1;
528*288bf522SAndroid Build Coastguard Worker 	tot2 += after.iowait_cpu_ticks + after.idle_cpu_ticks;
529*288bf522SAndroid Build Coastguard Worker 	tot1 += before.iowait_cpu_ticks + before.idle_cpu_ticks;
530*288bf522SAndroid Build Coastguard Worker 	delta2 = tot2 - tot1;
531*288bf522SAndroid Build Coastguard Worker 	cpu_util = delta1 * 100.0 / delta2;
532*288bf522SAndroid Build Coastguard Worker 	if (!summary_mode)
533*288bf522SAndroid Build Coastguard Worker 		printf("CPU util = %.2f%%\n", cpu_util);
534*288bf522SAndroid Build Coastguard Worker 	else
535*288bf522SAndroid Build Coastguard Worker 		printf("%.2f ", cpu_util);
536*288bf522SAndroid Build Coastguard Worker 	/* Next compute system (incl irq/softirq) and user cpu util */
537*288bf522SAndroid Build Coastguard Worker 	delta1 = (after.user_cpu_ticks + after.nice_cpu_ticks) -
538*288bf522SAndroid Build Coastguard Worker 		(before.user_cpu_ticks + before.nice_cpu_ticks);
539*288bf522SAndroid Build Coastguard Worker 	cpu_util = delta1 * 100.0 / delta2;
540*288bf522SAndroid Build Coastguard Worker 	if (!summary_mode)
541*288bf522SAndroid Build Coastguard Worker 		printf("User CPU util = %.2f%%\n", cpu_util);
542*288bf522SAndroid Build Coastguard Worker 	else
543*288bf522SAndroid Build Coastguard Worker 		printf("%.2f ", cpu_util);
544*288bf522SAndroid Build Coastguard Worker 	delta1 = (after.system_cpu_ticks + after.hardirq_cpu_ticks +
545*288bf522SAndroid Build Coastguard Worker 		  after.softirq_cpu_ticks) -
546*288bf522SAndroid Build Coastguard Worker 		(before.system_cpu_ticks + before.hardirq_cpu_ticks +
547*288bf522SAndroid Build Coastguard Worker 		 before.softirq_cpu_ticks);
548*288bf522SAndroid Build Coastguard Worker 	cpu_util = delta1 * 100.0 / delta2;
549*288bf522SAndroid Build Coastguard Worker 	if (!summary_mode)
550*288bf522SAndroid Build Coastguard Worker 		printf("System CPU util = %.2f%%\n", cpu_util);
551*288bf522SAndroid Build Coastguard Worker 	else
552*288bf522SAndroid Build Coastguard Worker 		printf("%.2f ", cpu_util);
553*288bf522SAndroid Build Coastguard Worker 	/* Disk Util */
554*288bf522SAndroid Build Coastguard Worker 	disk_util = (after.tot_ticks - before.tot_ticks) * 100.0 /
555*288bf522SAndroid Build Coastguard Worker 		(after.uptime - before.uptime);
556*288bf522SAndroid Build Coastguard Worker 	if (verbose) {
557*288bf522SAndroid Build Coastguard Worker 		printf("Reads : nr_ios %lu, MB read %lu\n",
558*288bf522SAndroid Build Coastguard Worker 	       (after.rd_ios - before.rd_ios),
559*288bf522SAndroid Build Coastguard Worker 	       (after.rd_sec - before.rd_sec) / 2048);
560*288bf522SAndroid Build Coastguard Worker 		printf("Writes : nr_ios %lu, MB written %lu\n",
561*288bf522SAndroid Build Coastguard Worker 	       (after.wr_ios - before.wr_ios),
562*288bf522SAndroid Build Coastguard Worker 		       (after.wr_sec - before.wr_sec) / 2048);
563*288bf522SAndroid Build Coastguard Worker 	}
564*288bf522SAndroid Build Coastguard Worker 	if (!summary_mode)
565*288bf522SAndroid Build Coastguard Worker 		printf("Disk util = %.2f%%\n", disk_util);
566*288bf522SAndroid Build Coastguard Worker 	else
567*288bf522SAndroid Build Coastguard Worker 		printf("%.2f", disk_util);
568*288bf522SAndroid Build Coastguard Worker }
569*288bf522SAndroid Build Coastguard Worker 
570*288bf522SAndroid Build Coastguard Worker 
571*288bf522SAndroid Build Coastguard Worker static struct ioshark_filename_struct *filename_cache;
572*288bf522SAndroid Build Coastguard Worker static int filename_cache_num_entries;
573*288bf522SAndroid Build Coastguard Worker 
574*288bf522SAndroid Build Coastguard Worker char *
get_ro_filename(int ix)575*288bf522SAndroid Build Coastguard Worker get_ro_filename(int ix)
576*288bf522SAndroid Build Coastguard Worker {
577*288bf522SAndroid Build Coastguard Worker 	if (ix >= filename_cache_num_entries)
578*288bf522SAndroid Build Coastguard Worker 		return NULL;
579*288bf522SAndroid Build Coastguard Worker 	return filename_cache[ix].path;
580*288bf522SAndroid Build Coastguard Worker }
581*288bf522SAndroid Build Coastguard Worker 
582*288bf522SAndroid Build Coastguard Worker void
init_filename_cache(void)583*288bf522SAndroid Build Coastguard Worker init_filename_cache(void)
584*288bf522SAndroid Build Coastguard Worker {
585*288bf522SAndroid Build Coastguard Worker 	int fd;
586*288bf522SAndroid Build Coastguard Worker 	struct stat st;
587*288bf522SAndroid Build Coastguard Worker 
588*288bf522SAndroid Build Coastguard Worker 	fd = open("ioshark_filenames", O_RDONLY);
589*288bf522SAndroid Build Coastguard Worker 	if (fd < 0) {
590*288bf522SAndroid Build Coastguard Worker 		fprintf(stderr, "%s Can't open ioshark_filenames file\n",
591*288bf522SAndroid Build Coastguard Worker 			progname);
592*288bf522SAndroid Build Coastguard Worker 		exit(EXIT_FAILURE);
593*288bf522SAndroid Build Coastguard Worker 	}
594*288bf522SAndroid Build Coastguard Worker 	if (fstat(fd, &st) < 0) {
595*288bf522SAndroid Build Coastguard Worker 		fprintf(stderr, "%s Can't fstat ioshark_filenames file\n",
596*288bf522SAndroid Build Coastguard Worker 			progname);
597*288bf522SAndroid Build Coastguard Worker 		exit(EXIT_FAILURE);
598*288bf522SAndroid Build Coastguard Worker 	}
599*288bf522SAndroid Build Coastguard Worker 	filename_cache_num_entries = st.st_size /
600*288bf522SAndroid Build Coastguard Worker 		sizeof(struct ioshark_filename_struct);
601*288bf522SAndroid Build Coastguard Worker 	filename_cache = mmap(NULL, st.st_size, PROT_READ,
602*288bf522SAndroid Build Coastguard Worker 			      MAP_SHARED | MAP_LOCKED | MAP_POPULATE,
603*288bf522SAndroid Build Coastguard Worker 			      fd, 0);
604*288bf522SAndroid Build Coastguard Worker 	if (filename_cache == MAP_FAILED) {
605*288bf522SAndroid Build Coastguard Worker 		fprintf(stderr, "%s Can't fstat ioshark_filenames file: %m\n", progname);
606*288bf522SAndroid Build Coastguard Worker 		exit(EXIT_FAILURE);
607*288bf522SAndroid Build Coastguard Worker 	}
608*288bf522SAndroid Build Coastguard Worker 	close(fd);
609*288bf522SAndroid Build Coastguard Worker }
610*288bf522SAndroid Build Coastguard Worker 
611*288bf522SAndroid Build Coastguard Worker void
free_filename_cache(void)612*288bf522SAndroid Build Coastguard Worker free_filename_cache(void)
613*288bf522SAndroid Build Coastguard Worker {
614*288bf522SAndroid Build Coastguard Worker 	size_t mmap_size;
615*288bf522SAndroid Build Coastguard Worker 
616*288bf522SAndroid Build Coastguard Worker 	mmap_size = filename_cache_num_entries *
617*288bf522SAndroid Build Coastguard Worker 		sizeof(struct ioshark_filename_struct);
618*288bf522SAndroid Build Coastguard Worker 	munmap(filename_cache, mmap_size);
619*288bf522SAndroid Build Coastguard Worker }
620*288bf522SAndroid Build Coastguard Worker 
621*288bf522SAndroid Build Coastguard Worker /*
622*288bf522SAndroid Build Coastguard Worker  * Is the passed in filename a regular file ? (eg. not a directory).
623*288bf522SAndroid Build Coastguard Worker  * Second, is it in a read-only partition ?
624*288bf522SAndroid Build Coastguard Worker  */
625*288bf522SAndroid Build Coastguard Worker int
is_readonly_mount(char * filename,size_t size)626*288bf522SAndroid Build Coastguard Worker is_readonly_mount(char *filename, size_t size)
627*288bf522SAndroid Build Coastguard Worker {
628*288bf522SAndroid Build Coastguard Worker 	struct statfs statfsbuf;
629*288bf522SAndroid Build Coastguard Worker 	struct stat statbuf;
630*288bf522SAndroid Build Coastguard Worker 
631*288bf522SAndroid Build Coastguard Worker 	if (stat(filename, &statbuf) < 0) {
632*288bf522SAndroid Build Coastguard Worker 		/* File possibly deleted */
633*288bf522SAndroid Build Coastguard Worker 		return 0;
634*288bf522SAndroid Build Coastguard Worker 	}
635*288bf522SAndroid Build Coastguard Worker 	if (!S_ISREG(statbuf.st_mode)) {
636*288bf522SAndroid Build Coastguard Worker 		/* Is it a regular file ? */
637*288bf522SAndroid Build Coastguard Worker 		return 0;
638*288bf522SAndroid Build Coastguard Worker 	}
639*288bf522SAndroid Build Coastguard Worker 	if ((size_t)statbuf.st_size < size) {
640*288bf522SAndroid Build Coastguard Worker 		/* Size of existing file is smaller than we expect */
641*288bf522SAndroid Build Coastguard Worker 		return 0;
642*288bf522SAndroid Build Coastguard Worker 	}
643*288bf522SAndroid Build Coastguard Worker 	if (statfs(filename, &statfsbuf) < 0) {
644*288bf522SAndroid Build Coastguard Worker 		/* This shouldn't happen */
645*288bf522SAndroid Build Coastguard Worker 		return 0;
646*288bf522SAndroid Build Coastguard Worker 	}
647*288bf522SAndroid Build Coastguard Worker 	if ((statfsbuf.f_flags & ST_RDONLY) == 0)
648*288bf522SAndroid Build Coastguard Worker 		return 0;
649*288bf522SAndroid Build Coastguard Worker 	else
650*288bf522SAndroid Build Coastguard Worker 		return 1;
651*288bf522SAndroid Build Coastguard Worker }
652*288bf522SAndroid Build Coastguard Worker 
653*288bf522SAndroid Build Coastguard Worker int
ioshark_read_header(FILE * fp,struct ioshark_header * header)654*288bf522SAndroid Build Coastguard Worker ioshark_read_header(FILE *fp, struct ioshark_header *header)
655*288bf522SAndroid Build Coastguard Worker {
656*288bf522SAndroid Build Coastguard Worker 	if (fread(header, sizeof(struct ioshark_header), 1, fp) != 1)
657*288bf522SAndroid Build Coastguard Worker 		return -1;
658*288bf522SAndroid Build Coastguard Worker 	header->version = be64toh(header->version);
659*288bf522SAndroid Build Coastguard Worker 	header->num_files = be64toh(header->num_files);
660*288bf522SAndroid Build Coastguard Worker 	header->num_io_operations = be64toh(header->num_io_operations);
661*288bf522SAndroid Build Coastguard Worker 	return 1;
662*288bf522SAndroid Build Coastguard Worker }
663*288bf522SAndroid Build Coastguard Worker 
664*288bf522SAndroid Build Coastguard Worker int
ioshark_read_file_state(FILE * fp,struct ioshark_file_state * state)665*288bf522SAndroid Build Coastguard Worker ioshark_read_file_state(FILE *fp, struct ioshark_file_state *state)
666*288bf522SAndroid Build Coastguard Worker {
667*288bf522SAndroid Build Coastguard Worker 	if (fread(state, sizeof(struct ioshark_file_state), 1, fp) != 1)
668*288bf522SAndroid Build Coastguard Worker 		return -1;
669*288bf522SAndroid Build Coastguard Worker 	state->fileno = be64toh(state->fileno);
670*288bf522SAndroid Build Coastguard Worker 	state->size = be64toh(state->size);
671*288bf522SAndroid Build Coastguard Worker 	state->global_filename_ix = be64toh(state->global_filename_ix);
672*288bf522SAndroid Build Coastguard Worker 	return 1;
673*288bf522SAndroid Build Coastguard Worker }
674*288bf522SAndroid Build Coastguard Worker 
675*288bf522SAndroid Build Coastguard Worker int
ioshark_read_file_op(FILE * fp,struct ioshark_file_operation * file_op)676*288bf522SAndroid Build Coastguard Worker ioshark_read_file_op(FILE *fp, struct ioshark_file_operation *file_op)
677*288bf522SAndroid Build Coastguard Worker {
678*288bf522SAndroid Build Coastguard Worker 	if (fread(file_op, sizeof(struct ioshark_file_operation), 1, fp) != 1)
679*288bf522SAndroid Build Coastguard Worker 		return -1;
680*288bf522SAndroid Build Coastguard Worker 	file_op->delta_us = be64toh(file_op->delta_us);
681*288bf522SAndroid Build Coastguard Worker 	file_op->op_union.enum_size = be32toh(file_op->op_union.enum_size);
682*288bf522SAndroid Build Coastguard Worker 	file_op->fileno = be64toh(file_op->fileno);
683*288bf522SAndroid Build Coastguard Worker 	switch (file_op->ioshark_io_op) {
684*288bf522SAndroid Build Coastguard Worker 	case IOSHARK_LSEEK:
685*288bf522SAndroid Build Coastguard Worker 	case IOSHARK_LLSEEK:
686*288bf522SAndroid Build Coastguard Worker 		file_op->lseek_offset = be64toh(file_op->lseek_offset);
687*288bf522SAndroid Build Coastguard Worker 		file_op->lseek_action = be32toh(file_op->lseek_action);
688*288bf522SAndroid Build Coastguard Worker 		break;
689*288bf522SAndroid Build Coastguard Worker 	case IOSHARK_PREAD64:
690*288bf522SAndroid Build Coastguard Worker 	case IOSHARK_PWRITE64:
691*288bf522SAndroid Build Coastguard Worker 		file_op->prw_offset = be64toh(file_op->prw_offset);
692*288bf522SAndroid Build Coastguard Worker 		file_op->prw_len = be64toh(file_op->prw_len);
693*288bf522SAndroid Build Coastguard Worker 		break;
694*288bf522SAndroid Build Coastguard Worker 	case IOSHARK_READ:
695*288bf522SAndroid Build Coastguard Worker 	case IOSHARK_WRITE:
696*288bf522SAndroid Build Coastguard Worker 		file_op->rw_len = be64toh(file_op->rw_len);
697*288bf522SAndroid Build Coastguard Worker 		break;
698*288bf522SAndroid Build Coastguard Worker 	case IOSHARK_MMAP:
699*288bf522SAndroid Build Coastguard Worker 	case IOSHARK_MMAP2:
700*288bf522SAndroid Build Coastguard Worker 		file_op->mmap_offset = be64toh(file_op->mmap_offset);
701*288bf522SAndroid Build Coastguard Worker 		file_op->mmap_len = be64toh(file_op->mmap_len);
702*288bf522SAndroid Build Coastguard Worker 		file_op->mmap_prot = be32toh(file_op->mmap_prot);
703*288bf522SAndroid Build Coastguard Worker 		break;
704*288bf522SAndroid Build Coastguard Worker 	case IOSHARK_OPEN:
705*288bf522SAndroid Build Coastguard Worker 		file_op->open_flags = be32toh(file_op->open_flags);
706*288bf522SAndroid Build Coastguard Worker 		file_op->open_mode = be32toh(file_op->open_mode);
707*288bf522SAndroid Build Coastguard Worker 		break;
708*288bf522SAndroid Build Coastguard Worker 	case IOSHARK_FSYNC:
709*288bf522SAndroid Build Coastguard Worker 	case IOSHARK_FDATASYNC:
710*288bf522SAndroid Build Coastguard Worker 		break;
711*288bf522SAndroid Build Coastguard Worker 	case IOSHARK_CLOSE:
712*288bf522SAndroid Build Coastguard Worker 		break;
713*288bf522SAndroid Build Coastguard Worker 	default:
714*288bf522SAndroid Build Coastguard Worker 		fprintf(stderr, "%s: unknown FILE_OP %d\n",
715*288bf522SAndroid Build Coastguard Worker 			progname, file_op->ioshark_io_op);
716*288bf522SAndroid Build Coastguard Worker 		exit(EXIT_FAILURE);
717*288bf522SAndroid Build Coastguard Worker 		break;
718*288bf522SAndroid Build Coastguard Worker 	}
719*288bf522SAndroid Build Coastguard Worker 	return 1;
720*288bf522SAndroid Build Coastguard Worker }
721