xref: /aosp_15_r20/external/blktrace/blkrawverify.c (revision 1a3d31e37cc95e9919fd86900a2b6a555f55952c)
1*1a3d31e3SAndroid Build Coastguard Worker /*
2*1a3d31e3SAndroid Build Coastguard Worker  * block queue tracing application
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 #include <stdio.h>
22*1a3d31e3SAndroid Build Coastguard Worker #include <stdlib.h>
23*1a3d31e3SAndroid Build Coastguard Worker #include <string.h>
24*1a3d31e3SAndroid Build Coastguard Worker #include <errno.h>
25*1a3d31e3SAndroid Build Coastguard Worker #include <sys/types.h>
26*1a3d31e3SAndroid Build Coastguard Worker #include <sys/stat.h>
27*1a3d31e3SAndroid Build Coastguard Worker #include <unistd.h>
28*1a3d31e3SAndroid Build Coastguard Worker 
29*1a3d31e3SAndroid Build Coastguard Worker #include "blktrace.h"
30*1a3d31e3SAndroid Build Coastguard Worker 
31*1a3d31e3SAndroid Build Coastguard Worker struct trace_info {
32*1a3d31e3SAndroid Build Coastguard Worker 	int bit_field;
33*1a3d31e3SAndroid Build Coastguard Worker 	char *string;
34*1a3d31e3SAndroid Build Coastguard Worker };
35*1a3d31e3SAndroid Build Coastguard Worker 
36*1a3d31e3SAndroid Build Coastguard Worker int data_is_native = -1;
37*1a3d31e3SAndroid Build Coastguard Worker 
38*1a3d31e3SAndroid Build Coastguard Worker #define TRACE_TO_STRING(f)	{.bit_field = f, .string = #f}
39*1a3d31e3SAndroid Build Coastguard Worker static struct trace_info traces[] = {
40*1a3d31e3SAndroid Build Coastguard Worker 	TRACE_TO_STRING( BLK_TC_READ ),
41*1a3d31e3SAndroid Build Coastguard Worker 	TRACE_TO_STRING( BLK_TC_WRITE ),
42*1a3d31e3SAndroid Build Coastguard Worker 	TRACE_TO_STRING( BLK_TC_FLUSH ),
43*1a3d31e3SAndroid Build Coastguard Worker 	TRACE_TO_STRING( BLK_TC_SYNC ),
44*1a3d31e3SAndroid Build Coastguard Worker 	TRACE_TO_STRING( BLK_TC_QUEUE ),
45*1a3d31e3SAndroid Build Coastguard Worker 	TRACE_TO_STRING( BLK_TC_REQUEUE ),
46*1a3d31e3SAndroid Build Coastguard Worker 	TRACE_TO_STRING( BLK_TC_ISSUE ),
47*1a3d31e3SAndroid Build Coastguard Worker 	TRACE_TO_STRING( BLK_TC_COMPLETE ),
48*1a3d31e3SAndroid Build Coastguard Worker 	TRACE_TO_STRING( BLK_TC_FS ),
49*1a3d31e3SAndroid Build Coastguard Worker 	TRACE_TO_STRING( BLK_TC_PC ),
50*1a3d31e3SAndroid Build Coastguard Worker 	TRACE_TO_STRING( BLK_TC_AHEAD ),
51*1a3d31e3SAndroid Build Coastguard Worker 	TRACE_TO_STRING( BLK_TC_META ),
52*1a3d31e3SAndroid Build Coastguard Worker 	TRACE_TO_STRING( BLK_TC_DISCARD ),
53*1a3d31e3SAndroid Build Coastguard Worker 	TRACE_TO_STRING( BLK_TC_FUA ),
54*1a3d31e3SAndroid Build Coastguard Worker };
55*1a3d31e3SAndroid Build Coastguard Worker #define N_TRACES (sizeof(traces) / sizeof(struct trace_info))
56*1a3d31e3SAndroid Build Coastguard Worker 
57*1a3d31e3SAndroid Build Coastguard Worker struct act_info {
58*1a3d31e3SAndroid Build Coastguard Worker 	__u32 val;
59*1a3d31e3SAndroid Build Coastguard Worker 	char *string;
60*1a3d31e3SAndroid Build Coastguard Worker };
61*1a3d31e3SAndroid Build Coastguard Worker 
62*1a3d31e3SAndroid Build Coastguard Worker #define ACT_TO_STRING(f)	{.val = f, .string = #f}
63*1a3d31e3SAndroid Build Coastguard Worker static struct act_info acts[] = {
64*1a3d31e3SAndroid Build Coastguard Worker 	ACT_TO_STRING( __BLK_TA_QUEUE ),
65*1a3d31e3SAndroid Build Coastguard Worker 	ACT_TO_STRING( __BLK_TA_QUEUE ),
66*1a3d31e3SAndroid Build Coastguard Worker 	ACT_TO_STRING( __BLK_TA_BACKMERGE ),
67*1a3d31e3SAndroid Build Coastguard Worker 	ACT_TO_STRING( __BLK_TA_FRONTMERGE ),
68*1a3d31e3SAndroid Build Coastguard Worker 	ACT_TO_STRING( __BLK_TA_GETRQ ),
69*1a3d31e3SAndroid Build Coastguard Worker 	ACT_TO_STRING( __BLK_TA_SLEEPRQ ),
70*1a3d31e3SAndroid Build Coastguard Worker 	ACT_TO_STRING( __BLK_TA_REQUEUE ),
71*1a3d31e3SAndroid Build Coastguard Worker 	ACT_TO_STRING( __BLK_TA_ISSUE ),
72*1a3d31e3SAndroid Build Coastguard Worker 	ACT_TO_STRING( __BLK_TA_COMPLETE ),
73*1a3d31e3SAndroid Build Coastguard Worker 	ACT_TO_STRING( __BLK_TA_PLUG ),
74*1a3d31e3SAndroid Build Coastguard Worker 	ACT_TO_STRING( __BLK_TA_UNPLUG_IO ),
75*1a3d31e3SAndroid Build Coastguard Worker 	ACT_TO_STRING( __BLK_TA_UNPLUG_TIMER ),
76*1a3d31e3SAndroid Build Coastguard Worker 	ACT_TO_STRING( __BLK_TA_INSERT ),
77*1a3d31e3SAndroid Build Coastguard Worker 	ACT_TO_STRING( __BLK_TA_SPLIT ),
78*1a3d31e3SAndroid Build Coastguard Worker 	ACT_TO_STRING( __BLK_TA_BOUNCE ),
79*1a3d31e3SAndroid Build Coastguard Worker 	ACT_TO_STRING( __BLK_TA_REMAP )
80*1a3d31e3SAndroid Build Coastguard Worker };
81*1a3d31e3SAndroid Build Coastguard Worker #define N_ACTS (sizeof(acts) / sizeof(struct act_info))
82*1a3d31e3SAndroid Build Coastguard Worker 
act_to_str(__u32 action)83*1a3d31e3SAndroid Build Coastguard Worker static char *act_to_str(__u32 action)
84*1a3d31e3SAndroid Build Coastguard Worker {
85*1a3d31e3SAndroid Build Coastguard Worker 	static char buf[1024];
86*1a3d31e3SAndroid Build Coastguard Worker 	unsigned int i;
87*1a3d31e3SAndroid Build Coastguard Worker 	unsigned int act = action & 0xffff;
88*1a3d31e3SAndroid Build Coastguard Worker 	unsigned int trace = (action >> BLK_TC_SHIFT) & 0xffff;
89*1a3d31e3SAndroid Build Coastguard Worker 
90*1a3d31e3SAndroid Build Coastguard Worker 	if (act < N_ACTS) {
91*1a3d31e3SAndroid Build Coastguard Worker 		sprintf(buf, "%s ", acts[act].string);
92*1a3d31e3SAndroid Build Coastguard Worker 		for (i = 0; i < N_TRACES; i++)
93*1a3d31e3SAndroid Build Coastguard Worker 			if (trace & (1 << i)) {
94*1a3d31e3SAndroid Build Coastguard Worker 				char buf2[1024];
95*1a3d31e3SAndroid Build Coastguard Worker 				sprintf(buf2, "| %s ", traces[i].string);
96*1a3d31e3SAndroid Build Coastguard Worker 				strcat(buf, buf2);
97*1a3d31e3SAndroid Build Coastguard Worker 			}
98*1a3d31e3SAndroid Build Coastguard Worker 	}
99*1a3d31e3SAndroid Build Coastguard Worker 	else
100*1a3d31e3SAndroid Build Coastguard Worker 		sprintf(buf, "Invalid action=%08x", action);
101*1a3d31e3SAndroid Build Coastguard Worker 
102*1a3d31e3SAndroid Build Coastguard Worker 	return buf;
103*1a3d31e3SAndroid Build Coastguard Worker }
104*1a3d31e3SAndroid Build Coastguard Worker 
dump_trace(FILE * ofp,char * prefix,struct blk_io_trace * bit)105*1a3d31e3SAndroid Build Coastguard Worker static void dump_trace(FILE *ofp, char *prefix, struct blk_io_trace *bit)
106*1a3d31e3SAndroid Build Coastguard Worker {
107*1a3d31e3SAndroid Build Coastguard Worker 	fprintf(ofp, "    Dump %s\n", prefix);
108*1a3d31e3SAndroid Build Coastguard Worker 	fprintf(ofp, "        %8s: %08x\n", "magic", bit->magic);
109*1a3d31e3SAndroid Build Coastguard Worker 	fprintf(ofp, "        %8s: %u\n", "sequence", bit->sequence);
110*1a3d31e3SAndroid Build Coastguard Worker 	fprintf(ofp, "        %8s: %llu\n", "time", (unsigned long long) bit->time);
111*1a3d31e3SAndroid Build Coastguard Worker 	fprintf(ofp, "        %8s: %llu\n", "sector", (unsigned long long) bit->sector);
112*1a3d31e3SAndroid Build Coastguard Worker 	fprintf(ofp, "        %8s: %u\n", "bytes", bit->bytes);
113*1a3d31e3SAndroid Build Coastguard Worker 	fprintf(ofp, "        %8s: %s\n", "action", act_to_str(bit->action));
114*1a3d31e3SAndroid Build Coastguard Worker 	fprintf(ofp, "        %8s: %u\n", "bytes", bit->bytes);
115*1a3d31e3SAndroid Build Coastguard Worker 	fprintf(ofp, "        %8s: %u\n", "cpu", bit->cpu);
116*1a3d31e3SAndroid Build Coastguard Worker 	fprintf(ofp, "        %8s: %u\n", "error", bit->error);
117*1a3d31e3SAndroid Build Coastguard Worker 	fprintf(ofp, "        %8s: %u\n", "pdu_len", bit->pdu_len);
118*1a3d31e3SAndroid Build Coastguard Worker 	fprintf(ofp, "        %8s: (%u,%u)\n\n", "device", MAJOR(bit->device),
119*1a3d31e3SAndroid Build Coastguard Worker 						           MINOR(bit->device));
120*1a3d31e3SAndroid Build Coastguard Worker }
121*1a3d31e3SAndroid Build Coastguard Worker 
process(FILE ** fp,char * devname,char * file,unsigned int cpu)122*1a3d31e3SAndroid Build Coastguard Worker static int process(FILE **fp, char *devname, char *file, unsigned int cpu)
123*1a3d31e3SAndroid Build Coastguard Worker {
124*1a3d31e3SAndroid Build Coastguard Worker #	define SWAP_BITS() do {						\
125*1a3d31e3SAndroid Build Coastguard Worker 		if (bit_save) {						\
126*1a3d31e3SAndroid Build Coastguard Worker 			struct blk_io_trace *tmp = bit_save;		\
127*1a3d31e3SAndroid Build Coastguard Worker 			bit_save = bit;					\
128*1a3d31e3SAndroid Build Coastguard Worker 			bit = tmp;					\
129*1a3d31e3SAndroid Build Coastguard Worker 		}							\
130*1a3d31e3SAndroid Build Coastguard Worker 		else {							\
131*1a3d31e3SAndroid Build Coastguard Worker 			bit_save = bit;					\
132*1a3d31e3SAndroid Build Coastguard Worker 			bit = malloc(sizeof(struct blk_io_trace));	\
133*1a3d31e3SAndroid Build Coastguard Worker 		}							\
134*1a3d31e3SAndroid Build Coastguard Worker 	} while (0)
135*1a3d31e3SAndroid Build Coastguard Worker 
136*1a3d31e3SAndroid Build Coastguard Worker #	define INC_BAD(str) do {					\
137*1a3d31e3SAndroid Build Coastguard Worker 		nbad++;							\
138*1a3d31e3SAndroid Build Coastguard Worker 		fprintf(ofp, "    ----------------\n");			\
139*1a3d31e3SAndroid Build Coastguard Worker 		if (bit_save) dump_trace(ofp,"seq-1",bit_save);		\
140*1a3d31e3SAndroid Build Coastguard Worker 		dump_trace(ofp, str, bit);				\
141*1a3d31e3SAndroid Build Coastguard Worker 		SWAP_BITS();						\
142*1a3d31e3SAndroid Build Coastguard Worker 	} while (0)
143*1a3d31e3SAndroid Build Coastguard Worker 
144*1a3d31e3SAndroid Build Coastguard Worker 	size_t n;
145*1a3d31e3SAndroid Build Coastguard Worker 	FILE *ifp, *ofp;
146*1a3d31e3SAndroid Build Coastguard Worker 	__u32 save_device = 0, save_sequence = 0;
147*1a3d31e3SAndroid Build Coastguard Worker 	__u64 save_time = 0;
148*1a3d31e3SAndroid Build Coastguard Worker 	struct blk_io_trace *bit_save = NULL;
149*1a3d31e3SAndroid Build Coastguard Worker 	struct blk_io_trace *bit = malloc(sizeof(struct blk_io_trace));
150*1a3d31e3SAndroid Build Coastguard Worker 	unsigned int ngood = 0;
151*1a3d31e3SAndroid Build Coastguard Worker 	unsigned int nbad = 0;
152*1a3d31e3SAndroid Build Coastguard Worker 	unsigned int nbad_trace = 0, nbad_pdu = 0, nbad_cpu = 0;
153*1a3d31e3SAndroid Build Coastguard Worker 	unsigned int nbad_seq = 0, nbad_dev = 0, nbad_time = 0;
154*1a3d31e3SAndroid Build Coastguard Worker 	char ofname[1024];
155*1a3d31e3SAndroid Build Coastguard Worker 
156*1a3d31e3SAndroid Build Coastguard Worker 	ifp = fopen(file, "r");
157*1a3d31e3SAndroid Build Coastguard Worker 	if (!ifp)
158*1a3d31e3SAndroid Build Coastguard Worker 		return 0;
159*1a3d31e3SAndroid Build Coastguard Worker 
160*1a3d31e3SAndroid Build Coastguard Worker 	sprintf(ofname, "%s.verify.out", devname);
161*1a3d31e3SAndroid Build Coastguard Worker 
162*1a3d31e3SAndroid Build Coastguard Worker 	if (!*fp) {
163*1a3d31e3SAndroid Build Coastguard Worker 		*fp = fopen(ofname, "w");
164*1a3d31e3SAndroid Build Coastguard Worker 		if (*fp == NULL) {
165*1a3d31e3SAndroid Build Coastguard Worker 			fprintf(stderr,"Failed to open %s (%s), skipping\n",
166*1a3d31e3SAndroid Build Coastguard Worker 				ofname, strerror(errno));
167*1a3d31e3SAndroid Build Coastguard Worker 			fclose(ifp);
168*1a3d31e3SAndroid Build Coastguard Worker 			return 0;
169*1a3d31e3SAndroid Build Coastguard Worker 		}
170*1a3d31e3SAndroid Build Coastguard Worker 		fprintf(*fp, "\n---------------\n" );
171*1a3d31e3SAndroid Build Coastguard Worker 		fprintf(*fp, "Verifying %s\n", devname);
172*1a3d31e3SAndroid Build Coastguard Worker 	}
173*1a3d31e3SAndroid Build Coastguard Worker 
174*1a3d31e3SAndroid Build Coastguard Worker 	ofp = *fp;
175*1a3d31e3SAndroid Build Coastguard Worker 	while ((n = fread(bit, sizeof(struct blk_io_trace), 1, ifp)) == 1) {
176*1a3d31e3SAndroid Build Coastguard Worker 		if (ferror(ifp)) {
177*1a3d31e3SAndroid Build Coastguard Worker 			clearerr(ifp);
178*1a3d31e3SAndroid Build Coastguard Worker 			perror("fread");
179*1a3d31e3SAndroid Build Coastguard Worker 			break;
180*1a3d31e3SAndroid Build Coastguard Worker 		}
181*1a3d31e3SAndroid Build Coastguard Worker 		if (data_is_native == -1)
182*1a3d31e3SAndroid Build Coastguard Worker 			check_data_endianness(bit->magic);
183*1a3d31e3SAndroid Build Coastguard Worker 
184*1a3d31e3SAndroid Build Coastguard Worker 		trace_to_cpu(bit);
185*1a3d31e3SAndroid Build Coastguard Worker 
186*1a3d31e3SAndroid Build Coastguard Worker 		if (!CHECK_MAGIC(bit)) {
187*1a3d31e3SAndroid Build Coastguard Worker 			INC_BAD("bad trace");
188*1a3d31e3SAndroid Build Coastguard Worker 			continue;
189*1a3d31e3SAndroid Build Coastguard Worker 		}
190*1a3d31e3SAndroid Build Coastguard Worker 
191*1a3d31e3SAndroid Build Coastguard Worker 		if ((bit->magic & 0xff) != SUPPORTED_VERSION) {
192*1a3d31e3SAndroid Build Coastguard Worker 			fprintf(stderr, "unsupported trace version\n");
193*1a3d31e3SAndroid Build Coastguard Worker 			break;
194*1a3d31e3SAndroid Build Coastguard Worker 		}
195*1a3d31e3SAndroid Build Coastguard Worker 
196*1a3d31e3SAndroid Build Coastguard Worker 		if (bit->pdu_len) {
197*1a3d31e3SAndroid Build Coastguard Worker 			char *pdu_buf;
198*1a3d31e3SAndroid Build Coastguard Worker 
199*1a3d31e3SAndroid Build Coastguard Worker 			pdu_buf = malloc(bit->pdu_len);
200*1a3d31e3SAndroid Build Coastguard Worker 			n = fread(pdu_buf, bit->pdu_len, 1, ifp);
201*1a3d31e3SAndroid Build Coastguard Worker 			if (n == 0) {
202*1a3d31e3SAndroid Build Coastguard Worker 				INC_BAD("bad pdu");
203*1a3d31e3SAndroid Build Coastguard Worker 				nbad_seq++;
204*1a3d31e3SAndroid Build Coastguard Worker 				free(pdu_buf);
205*1a3d31e3SAndroid Build Coastguard Worker 				break;
206*1a3d31e3SAndroid Build Coastguard Worker 			}
207*1a3d31e3SAndroid Build Coastguard Worker 			free(pdu_buf);
208*1a3d31e3SAndroid Build Coastguard Worker 		}
209*1a3d31e3SAndroid Build Coastguard Worker 
210*1a3d31e3SAndroid Build Coastguard Worker 		if (bit->cpu != cpu) {
211*1a3d31e3SAndroid Build Coastguard Worker 			INC_BAD("bad cpu");
212*1a3d31e3SAndroid Build Coastguard Worker 			nbad_cpu++;
213*1a3d31e3SAndroid Build Coastguard Worker 			continue;
214*1a3d31e3SAndroid Build Coastguard Worker 		}
215*1a3d31e3SAndroid Build Coastguard Worker 
216*1a3d31e3SAndroid Build Coastguard Worker 		/*
217*1a3d31e3SAndroid Build Coastguard Worker 		 * skip notify traces, they don't have valid sequences
218*1a3d31e3SAndroid Build Coastguard Worker 		 */
219*1a3d31e3SAndroid Build Coastguard Worker 		if (bit->action & BLK_TC_ACT(BLK_TC_NOTIFY))
220*1a3d31e3SAndroid Build Coastguard Worker 			continue;
221*1a3d31e3SAndroid Build Coastguard Worker 
222*1a3d31e3SAndroid Build Coastguard Worker 		if (ngood) {
223*1a3d31e3SAndroid Build Coastguard Worker 			if (bit->sequence <= save_sequence) {
224*1a3d31e3SAndroid Build Coastguard Worker 				INC_BAD("bad seq");
225*1a3d31e3SAndroid Build Coastguard Worker 				nbad_seq++;
226*1a3d31e3SAndroid Build Coastguard Worker 				continue;
227*1a3d31e3SAndroid Build Coastguard Worker 			}
228*1a3d31e3SAndroid Build Coastguard Worker 			else if (bit->time <= save_time) {
229*1a3d31e3SAndroid Build Coastguard Worker 				INC_BAD("time regression");
230*1a3d31e3SAndroid Build Coastguard Worker 				nbad_time++;
231*1a3d31e3SAndroid Build Coastguard Worker 				continue;
232*1a3d31e3SAndroid Build Coastguard Worker 			}
233*1a3d31e3SAndroid Build Coastguard Worker 			else if (bit->device != save_device) {
234*1a3d31e3SAndroid Build Coastguard Worker 				INC_BAD("bad dev");
235*1a3d31e3SAndroid Build Coastguard Worker 				nbad_dev++;
236*1a3d31e3SAndroid Build Coastguard Worker 				continue;
237*1a3d31e3SAndroid Build Coastguard Worker 			}
238*1a3d31e3SAndroid Build Coastguard Worker 		}
239*1a3d31e3SAndroid Build Coastguard Worker 
240*1a3d31e3SAndroid Build Coastguard Worker 		save_sequence = bit->sequence;
241*1a3d31e3SAndroid Build Coastguard Worker 		save_time = bit->time;
242*1a3d31e3SAndroid Build Coastguard Worker 		save_device = bit->device;
243*1a3d31e3SAndroid Build Coastguard Worker 
244*1a3d31e3SAndroid Build Coastguard Worker 		ngood++;
245*1a3d31e3SAndroid Build Coastguard Worker 		SWAP_BITS();
246*1a3d31e3SAndroid Build Coastguard Worker 	}
247*1a3d31e3SAndroid Build Coastguard Worker 
248*1a3d31e3SAndroid Build Coastguard Worker 	if (n == 0 && !feof(ifp))
249*1a3d31e3SAndroid Build Coastguard Worker 		fprintf(stderr,"%s: fread failed %d/%s\n",
250*1a3d31e3SAndroid Build Coastguard Worker 		        file, errno, strerror(errno));
251*1a3d31e3SAndroid Build Coastguard Worker 	fclose(ifp);
252*1a3d31e3SAndroid Build Coastguard Worker 
253*1a3d31e3SAndroid Build Coastguard Worker 	fprintf(ofp, "    ---------------------\n");
254*1a3d31e3SAndroid Build Coastguard Worker 	fprintf(ofp, "    Summary for cpu %d:\n", cpu);
255*1a3d31e3SAndroid Build Coastguard Worker 	fprintf(ofp, "    %10d valid + %10d invalid (%5.1f%%) processed\n\n",
256*1a3d31e3SAndroid Build Coastguard Worker 		ngood, nbad,
257*1a3d31e3SAndroid Build Coastguard Worker 		ngood ? 100.0 * (float)ngood / (float)(ngood + nbad) : 0.0);
258*1a3d31e3SAndroid Build Coastguard Worker 
259*1a3d31e3SAndroid Build Coastguard Worker 	if (nbad) {
260*1a3d31e3SAndroid Build Coastguard Worker 		if (nbad_trace)
261*1a3d31e3SAndroid Build Coastguard Worker 			fprintf(ofp, "%8s %d traces\n", "", nbad_trace);
262*1a3d31e3SAndroid Build Coastguard Worker 		if (nbad_trace)
263*1a3d31e3SAndroid Build Coastguard Worker 			fprintf(ofp, "%8s %d pdu\n", "", nbad_pdu);
264*1a3d31e3SAndroid Build Coastguard Worker 		if (nbad_cpu)
265*1a3d31e3SAndroid Build Coastguard Worker 			fprintf(ofp, "%8s %d cpu\n", "", nbad_cpu);
266*1a3d31e3SAndroid Build Coastguard Worker 		if (nbad_seq)
267*1a3d31e3SAndroid Build Coastguard Worker 			fprintf(ofp, "%8s %d seq\n", "", nbad_seq);
268*1a3d31e3SAndroid Build Coastguard Worker 		if (nbad_dev)
269*1a3d31e3SAndroid Build Coastguard Worker 			fprintf(ofp, "%8s %d dev\n", "", nbad_dev);
270*1a3d31e3SAndroid Build Coastguard Worker 		if (nbad_time)
271*1a3d31e3SAndroid Build Coastguard Worker 			fprintf(ofp, "%8s %d time\n", "", nbad_time);
272*1a3d31e3SAndroid Build Coastguard Worker 		fprintf(ofp,"\n");
273*1a3d31e3SAndroid Build Coastguard Worker 	}
274*1a3d31e3SAndroid Build Coastguard Worker 
275*1a3d31e3SAndroid Build Coastguard Worker 	return nbad;
276*1a3d31e3SAndroid Build Coastguard Worker }
277*1a3d31e3SAndroid Build Coastguard Worker 
main(int argc,char * argv[])278*1a3d31e3SAndroid Build Coastguard Worker int main(int argc, char *argv[])
279*1a3d31e3SAndroid Build Coastguard Worker {
280*1a3d31e3SAndroid Build Coastguard Worker 	char *devname;
281*1a3d31e3SAndroid Build Coastguard Worker 	struct stat st;
282*1a3d31e3SAndroid Build Coastguard Worker 	int i, cpu, nbad, rval = 0;
283*1a3d31e3SAndroid Build Coastguard Worker 	FILE *ofp;
284*1a3d31e3SAndroid Build Coastguard Worker 	char *ofname = malloc(1024);
285*1a3d31e3SAndroid Build Coastguard Worker 	char *fname = malloc(1024);
286*1a3d31e3SAndroid Build Coastguard Worker 
287*1a3d31e3SAndroid Build Coastguard Worker 	if (argc < 2) {
288*1a3d31e3SAndroid Build Coastguard Worker 		fprintf(stderr,"FATAL: Need device name(s)\n");
289*1a3d31e3SAndroid Build Coastguard Worker 		fprintf(stderr,"Usage: blkrawverify <dev> [<dev>...]\n");
290*1a3d31e3SAndroid Build Coastguard Worker 		exit(1);
291*1a3d31e3SAndroid Build Coastguard Worker 	}
292*1a3d31e3SAndroid Build Coastguard Worker 
293*1a3d31e3SAndroid Build Coastguard Worker 	for (i = 1; i < argc; i++) {
294*1a3d31e3SAndroid Build Coastguard Worker 		devname = argv[i];
295*1a3d31e3SAndroid Build Coastguard Worker 		sprintf(ofname, "%s.verify.out", devname);
296*1a3d31e3SAndroid Build Coastguard Worker 		ofp = NULL;
297*1a3d31e3SAndroid Build Coastguard Worker 
298*1a3d31e3SAndroid Build Coastguard Worker 		printf("Verifying %s\n", devname); fflush(stdout);
299*1a3d31e3SAndroid Build Coastguard Worker 		for (cpu = 0; ; cpu++) {
300*1a3d31e3SAndroid Build Coastguard Worker 			sprintf(fname, "%s.blktrace.%d", devname, cpu);
301*1a3d31e3SAndroid Build Coastguard Worker 			if (stat(fname, &st) < 0) {
302*1a3d31e3SAndroid Build Coastguard Worker 				if (cpu == 0) {
303*1a3d31e3SAndroid Build Coastguard Worker 					fprintf(stderr, "No tracefiles found for %s\n",
304*1a3d31e3SAndroid Build Coastguard Worker 						devname);
305*1a3d31e3SAndroid Build Coastguard Worker 					rval = 1;
306*1a3d31e3SAndroid Build Coastguard Worker 				}
307*1a3d31e3SAndroid Build Coastguard Worker 				break;
308*1a3d31e3SAndroid Build Coastguard Worker 			}
309*1a3d31e3SAndroid Build Coastguard Worker 			printf("    CPU %d ", cpu); fflush(stdout);
310*1a3d31e3SAndroid Build Coastguard Worker 			nbad = process(&ofp, devname, fname, cpu);
311*1a3d31e3SAndroid Build Coastguard Worker 			if (nbad) {
312*1a3d31e3SAndroid Build Coastguard Worker 				printf("-- %d bad", nbad);
313*1a3d31e3SAndroid Build Coastguard Worker 				rval = 1;
314*1a3d31e3SAndroid Build Coastguard Worker 			}
315*1a3d31e3SAndroid Build Coastguard Worker 			printf("\n");
316*1a3d31e3SAndroid Build Coastguard Worker 		}
317*1a3d31e3SAndroid Build Coastguard Worker 		if (ofp) {
318*1a3d31e3SAndroid Build Coastguard Worker 			fclose(ofp);
319*1a3d31e3SAndroid Build Coastguard Worker 			fprintf(stdout, "Wrote output to %s\n", ofname);
320*1a3d31e3SAndroid Build Coastguard Worker 		}
321*1a3d31e3SAndroid Build Coastguard Worker 	}
322*1a3d31e3SAndroid Build Coastguard Worker 
323*1a3d31e3SAndroid Build Coastguard Worker 	return rval;
324*1a3d31e3SAndroid Build Coastguard Worker }
325