|
Packit |
c4abd9 |
/*
|
|
Packit |
c4abd9 |
* block queue tracing application
|
|
Packit |
c4abd9 |
*
|
|
Packit |
c4abd9 |
* Copyright (C) 2006 Alan D. Brunelle <Alan.Brunelle@hp.com>
|
|
Packit |
c4abd9 |
*
|
|
Packit |
c4abd9 |
* This program is free software; you can redistribute it and/or modify
|
|
Packit |
c4abd9 |
* it under the terms of the GNU General Public License as published by
|
|
Packit |
c4abd9 |
* the Free Software Foundation; either version 2 of the License, or
|
|
Packit |
c4abd9 |
* (at your option) any later version.
|
|
Packit |
c4abd9 |
*
|
|
Packit |
c4abd9 |
* This program is distributed in the hope that it will be useful,
|
|
Packit |
c4abd9 |
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
Packit |
c4abd9 |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
Packit |
c4abd9 |
* GNU General Public License for more details.
|
|
Packit |
c4abd9 |
*
|
|
Packit |
c4abd9 |
* You should have received a copy of the GNU General Public License
|
|
Packit |
c4abd9 |
* along with this program; if not, write to the Free Software
|
|
Packit |
c4abd9 |
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
Packit |
c4abd9 |
*
|
|
Packit |
c4abd9 |
*/
|
|
Packit |
c4abd9 |
#include <stdio.h>
|
|
Packit |
c4abd9 |
#include <stdlib.h>
|
|
Packit |
c4abd9 |
#include <string.h>
|
|
Packit |
c4abd9 |
#include <errno.h>
|
|
Packit |
c4abd9 |
#include <sys/types.h>
|
|
Packit |
c4abd9 |
#include <sys/stat.h>
|
|
Packit |
c4abd9 |
#include <unistd.h>
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
#include "blktrace.h"
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
struct trace_info {
|
|
Packit |
c4abd9 |
int bit_field;
|
|
Packit |
c4abd9 |
char *string;
|
|
Packit |
c4abd9 |
};
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
int data_is_native = -1;
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
#define TRACE_TO_STRING(f) {.bit_field = f, .string = #f}
|
|
Packit |
c4abd9 |
static struct trace_info traces[] = {
|
|
Packit |
c4abd9 |
TRACE_TO_STRING( BLK_TC_READ ),
|
|
Packit |
c4abd9 |
TRACE_TO_STRING( BLK_TC_WRITE ),
|
|
Packit |
c4abd9 |
TRACE_TO_STRING( BLK_TC_FLUSH ),
|
|
Packit |
c4abd9 |
TRACE_TO_STRING( BLK_TC_SYNC ),
|
|
Packit |
c4abd9 |
TRACE_TO_STRING( BLK_TC_QUEUE ),
|
|
Packit |
c4abd9 |
TRACE_TO_STRING( BLK_TC_REQUEUE ),
|
|
Packit |
c4abd9 |
TRACE_TO_STRING( BLK_TC_ISSUE ),
|
|
Packit |
c4abd9 |
TRACE_TO_STRING( BLK_TC_COMPLETE ),
|
|
Packit |
c4abd9 |
TRACE_TO_STRING( BLK_TC_FS ),
|
|
Packit |
c4abd9 |
TRACE_TO_STRING( BLK_TC_PC ),
|
|
Packit |
c4abd9 |
TRACE_TO_STRING( BLK_TC_AHEAD ),
|
|
Packit |
c4abd9 |
TRACE_TO_STRING( BLK_TC_META ),
|
|
Packit |
c4abd9 |
TRACE_TO_STRING( BLK_TC_DISCARD ),
|
|
Packit |
c4abd9 |
TRACE_TO_STRING( BLK_TC_FUA ),
|
|
Packit |
c4abd9 |
};
|
|
Packit |
c4abd9 |
#define N_TRACES (sizeof(traces) / sizeof(struct trace_info))
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
struct act_info {
|
|
Packit |
c4abd9 |
__u32 val;
|
|
Packit |
c4abd9 |
char *string;
|
|
Packit |
c4abd9 |
};
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
#define ACT_TO_STRING(f) {.val = f, .string = #f}
|
|
Packit |
c4abd9 |
static struct act_info acts[] = {
|
|
Packit |
c4abd9 |
ACT_TO_STRING( __BLK_TA_QUEUE ),
|
|
Packit |
c4abd9 |
ACT_TO_STRING( __BLK_TA_QUEUE ),
|
|
Packit |
c4abd9 |
ACT_TO_STRING( __BLK_TA_BACKMERGE ),
|
|
Packit |
c4abd9 |
ACT_TO_STRING( __BLK_TA_FRONTMERGE ),
|
|
Packit |
c4abd9 |
ACT_TO_STRING( __BLK_TA_GETRQ ),
|
|
Packit |
c4abd9 |
ACT_TO_STRING( __BLK_TA_SLEEPRQ ),
|
|
Packit |
c4abd9 |
ACT_TO_STRING( __BLK_TA_REQUEUE ),
|
|
Packit |
c4abd9 |
ACT_TO_STRING( __BLK_TA_ISSUE ),
|
|
Packit |
c4abd9 |
ACT_TO_STRING( __BLK_TA_COMPLETE ),
|
|
Packit |
c4abd9 |
ACT_TO_STRING( __BLK_TA_PLUG ),
|
|
Packit |
c4abd9 |
ACT_TO_STRING( __BLK_TA_UNPLUG_IO ),
|
|
Packit |
c4abd9 |
ACT_TO_STRING( __BLK_TA_UNPLUG_TIMER ),
|
|
Packit |
c4abd9 |
ACT_TO_STRING( __BLK_TA_INSERT ),
|
|
Packit |
c4abd9 |
ACT_TO_STRING( __BLK_TA_SPLIT ),
|
|
Packit |
c4abd9 |
ACT_TO_STRING( __BLK_TA_BOUNCE ),
|
|
Packit |
c4abd9 |
ACT_TO_STRING( __BLK_TA_REMAP )
|
|
Packit |
c4abd9 |
};
|
|
Packit |
c4abd9 |
#define N_ACTS (sizeof(acts) / sizeof(struct act_info))
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
static char *act_to_str(__u32 action)
|
|
Packit |
c4abd9 |
{
|
|
Packit |
c4abd9 |
static char buf[1024];
|
|
Packit |
c4abd9 |
unsigned int i;
|
|
Packit |
c4abd9 |
unsigned int act = action & 0xffff;
|
|
Packit |
c4abd9 |
unsigned int trace = (action >> BLK_TC_SHIFT) & 0xffff;
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
if (act < N_ACTS) {
|
|
Packit |
c4abd9 |
sprintf(buf, "%s ", acts[act].string);
|
|
Packit |
c4abd9 |
for (i = 0; i < N_TRACES; i++)
|
|
Packit |
c4abd9 |
if (trace & (1 << i)) {
|
|
Packit |
c4abd9 |
char buf2[1024];
|
|
Packit |
c4abd9 |
sprintf(buf2, "| %s ", traces[i].string);
|
|
Packit |
c4abd9 |
strcat(buf, buf2);
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
else
|
|
Packit |
c4abd9 |
sprintf(buf, "Invalid action=%08x", action);
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
return buf;
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
static void dump_trace(FILE *ofp, char *prefix, struct blk_io_trace *bit)
|
|
Packit |
c4abd9 |
{
|
|
Packit |
c4abd9 |
fprintf(ofp, " Dump %s\n", prefix);
|
|
Packit |
c4abd9 |
fprintf(ofp, " %8s: %08x\n", "magic", bit->magic);
|
|
Packit |
c4abd9 |
fprintf(ofp, " %8s: %u\n", "sequence", bit->sequence);
|
|
Packit |
c4abd9 |
fprintf(ofp, " %8s: %llu\n", "time", (unsigned long long) bit->time);
|
|
Packit |
c4abd9 |
fprintf(ofp, " %8s: %llu\n", "sector", (unsigned long long) bit->sector);
|
|
Packit |
c4abd9 |
fprintf(ofp, " %8s: %u\n", "bytes", bit->bytes);
|
|
Packit |
c4abd9 |
fprintf(ofp, " %8s: %s\n", "action", act_to_str(bit->action));
|
|
Packit |
c4abd9 |
fprintf(ofp, " %8s: %u\n", "bytes", bit->bytes);
|
|
Packit |
c4abd9 |
fprintf(ofp, " %8s: %u\n", "cpu", bit->cpu);
|
|
Packit |
c4abd9 |
fprintf(ofp, " %8s: %u\n", "error", bit->error);
|
|
Packit |
c4abd9 |
fprintf(ofp, " %8s: %u\n", "pdu_len", bit->pdu_len);
|
|
Packit |
c4abd9 |
fprintf(ofp, " %8s: (%u,%u)\n\n", "device", MAJOR(bit->device),
|
|
Packit |
c4abd9 |
MINOR(bit->device));
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
static int process(FILE **fp, char *devname, char *file, unsigned int cpu)
|
|
Packit |
c4abd9 |
{
|
|
Packit |
c4abd9 |
# define SWAP_BITS() do { \
|
|
Packit |
c4abd9 |
if (bit_save) { \
|
|
Packit |
c4abd9 |
struct blk_io_trace *tmp = bit_save; \
|
|
Packit |
c4abd9 |
bit_save = bit; \
|
|
Packit |
c4abd9 |
bit = tmp; \
|
|
Packit |
c4abd9 |
} \
|
|
Packit |
c4abd9 |
else { \
|
|
Packit |
c4abd9 |
bit_save = bit; \
|
|
Packit |
c4abd9 |
bit = malloc(sizeof(struct blk_io_trace)); \
|
|
Packit |
c4abd9 |
} \
|
|
Packit |
c4abd9 |
} while (0)
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
# define INC_BAD(str) do { \
|
|
Packit |
c4abd9 |
nbad++; \
|
|
Packit |
c4abd9 |
fprintf(ofp, " ----------------\n"); \
|
|
Packit |
c4abd9 |
if (bit_save) dump_trace(ofp,"seq-1",bit_save); \
|
|
Packit |
c4abd9 |
dump_trace(ofp, str, bit); \
|
|
Packit |
c4abd9 |
SWAP_BITS(); \
|
|
Packit |
c4abd9 |
} while (0)
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
size_t n;
|
|
Packit |
c4abd9 |
FILE *ifp, *ofp;
|
|
Packit |
c4abd9 |
__u32 save_device = 0, save_sequence = 0;
|
|
Packit |
c4abd9 |
__u64 save_time = 0;
|
|
Packit |
c4abd9 |
struct blk_io_trace *bit_save = NULL;
|
|
Packit |
c4abd9 |
struct blk_io_trace *bit = malloc(sizeof(struct blk_io_trace));
|
|
Packit |
c4abd9 |
unsigned int ngood = 0;
|
|
Packit |
c4abd9 |
unsigned int nbad = 0;
|
|
Packit |
c4abd9 |
unsigned int nbad_trace = 0, nbad_pdu = 0, nbad_cpu = 0;
|
|
Packit |
c4abd9 |
unsigned int nbad_seq = 0, nbad_dev = 0, nbad_time = 0;
|
|
Packit |
c4abd9 |
char ofname[1024];
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
ifp = fopen(file, "r");
|
|
Packit |
c4abd9 |
if (!ifp)
|
|
Packit |
c4abd9 |
return 0;
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
sprintf(ofname, "%s.verify.out", devname);
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
if (!*fp) {
|
|
Packit |
c4abd9 |
*fp = fopen(ofname, "w");
|
|
Packit |
c4abd9 |
if (*fp == NULL) {
|
|
Packit |
c4abd9 |
fprintf(stderr,"Failed to open %s (%s), skipping\n",
|
|
Packit |
c4abd9 |
ofname, strerror(errno));
|
|
Packit |
c4abd9 |
fclose(ifp);
|
|
Packit |
c4abd9 |
return 0;
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
fprintf(*fp, "\n---------------\n" );
|
|
Packit |
c4abd9 |
fprintf(*fp, "Verifying %s\n", devname);
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
ofp = *fp;
|
|
Packit |
c4abd9 |
while ((n = fread(bit, sizeof(struct blk_io_trace), 1, ifp)) == 1) {
|
|
Packit |
c4abd9 |
if (ferror(ifp)) {
|
|
Packit |
c4abd9 |
clearerr(ifp);
|
|
Packit |
c4abd9 |
perror("fread");
|
|
Packit |
c4abd9 |
break;
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
if (data_is_native == -1)
|
|
Packit |
c4abd9 |
check_data_endianness(bit->magic);
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
trace_to_cpu(bit);
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
if (!CHECK_MAGIC(bit)) {
|
|
Packit |
c4abd9 |
INC_BAD("bad trace");
|
|
Packit |
c4abd9 |
continue;
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
if ((bit->magic & 0xff) != SUPPORTED_VERSION) {
|
|
Packit |
c4abd9 |
fprintf(stderr, "unsupported trace version\n");
|
|
Packit |
c4abd9 |
break;
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
if (bit->pdu_len) {
|
|
Packit |
c4abd9 |
char *pdu_buf;
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
pdu_buf = malloc(bit->pdu_len);
|
|
Packit |
c4abd9 |
n = fread(pdu_buf, bit->pdu_len, 1, ifp);
|
|
Packit |
c4abd9 |
if (n == 0) {
|
|
Packit |
c4abd9 |
INC_BAD("bad pdu");
|
|
Packit |
c4abd9 |
nbad_seq++;
|
|
Packit |
c4abd9 |
free(pdu_buf);
|
|
Packit |
c4abd9 |
break;
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
free(pdu_buf);
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
if (bit->cpu != cpu) {
|
|
Packit |
c4abd9 |
INC_BAD("bad cpu");
|
|
Packit |
c4abd9 |
nbad_cpu++;
|
|
Packit |
c4abd9 |
continue;
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
/*
|
|
Packit |
c4abd9 |
* skip notify traces, they don't have valid sequences
|
|
Packit |
c4abd9 |
*/
|
|
Packit |
c4abd9 |
if (bit->action & BLK_TC_ACT(BLK_TC_NOTIFY))
|
|
Packit |
c4abd9 |
continue;
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
if (ngood) {
|
|
Packit |
c4abd9 |
if (bit->sequence <= save_sequence) {
|
|
Packit |
c4abd9 |
INC_BAD("bad seq");
|
|
Packit |
c4abd9 |
nbad_seq++;
|
|
Packit |
c4abd9 |
continue;
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
else if (bit->time <= save_time) {
|
|
Packit |
c4abd9 |
INC_BAD("time regression");
|
|
Packit |
c4abd9 |
nbad_time++;
|
|
Packit |
c4abd9 |
continue;
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
else if (bit->device != save_device) {
|
|
Packit |
c4abd9 |
INC_BAD("bad dev");
|
|
Packit |
c4abd9 |
nbad_dev++;
|
|
Packit |
c4abd9 |
continue;
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
save_sequence = bit->sequence;
|
|
Packit |
c4abd9 |
save_time = bit->time;
|
|
Packit |
c4abd9 |
save_device = bit->device;
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
ngood++;
|
|
Packit |
c4abd9 |
SWAP_BITS();
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
if (n == 0 && !feof(ifp))
|
|
Packit |
c4abd9 |
fprintf(stderr,"%s: fread failed %d/%s\n",
|
|
Packit |
c4abd9 |
file, errno, strerror(errno));
|
|
Packit |
c4abd9 |
fclose(ifp);
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
fprintf(ofp, " ---------------------\n");
|
|
Packit |
c4abd9 |
fprintf(ofp, " Summary for cpu %d:\n", cpu);
|
|
Packit |
c4abd9 |
fprintf(ofp, " %10d valid + %10d invalid (%5.1f%%) processed\n\n",
|
|
Packit |
c4abd9 |
ngood, nbad,
|
|
Packit |
c4abd9 |
ngood ? 100.0 * (float)ngood / (float)(ngood + nbad) : 0.0);
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
if (nbad) {
|
|
Packit |
c4abd9 |
if (nbad_trace)
|
|
Packit |
c4abd9 |
fprintf(ofp, "%8s %d traces\n", "", nbad_trace);
|
|
Packit |
c4abd9 |
if (nbad_trace)
|
|
Packit |
c4abd9 |
fprintf(ofp, "%8s %d pdu\n", "", nbad_pdu);
|
|
Packit |
c4abd9 |
if (nbad_cpu)
|
|
Packit |
c4abd9 |
fprintf(ofp, "%8s %d cpu\n", "", nbad_cpu);
|
|
Packit |
c4abd9 |
if (nbad_seq)
|
|
Packit |
c4abd9 |
fprintf(ofp, "%8s %d seq\n", "", nbad_seq);
|
|
Packit |
c4abd9 |
if (nbad_dev)
|
|
Packit |
c4abd9 |
fprintf(ofp, "%8s %d dev\n", "", nbad_dev);
|
|
Packit |
c4abd9 |
if (nbad_time)
|
|
Packit |
c4abd9 |
fprintf(ofp, "%8s %d time\n", "", nbad_time);
|
|
Packit |
c4abd9 |
fprintf(ofp,"\n");
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
return nbad;
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
int main(int argc, char *argv[])
|
|
Packit |
c4abd9 |
{
|
|
Packit |
c4abd9 |
char *devname;
|
|
Packit |
c4abd9 |
struct stat st;
|
|
Packit |
c4abd9 |
int i, cpu, nbad, rval = 0;
|
|
Packit |
c4abd9 |
FILE *ofp;
|
|
Packit |
c4abd9 |
char *ofname = malloc(1024);
|
|
Packit |
c4abd9 |
char *fname = malloc(1024);
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
if (argc < 2) {
|
|
Packit |
c4abd9 |
fprintf(stderr,"FATAL: Need device name(s)\n");
|
|
Packit |
c4abd9 |
fprintf(stderr,"Usage: blkrawverify <dev> [<dev>...]\n");
|
|
Packit |
c4abd9 |
exit(1);
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
for (i = 1; i < argc; i++) {
|
|
Packit |
c4abd9 |
devname = argv[i];
|
|
Packit |
c4abd9 |
sprintf(ofname, "%s.verify.out", devname);
|
|
Packit |
c4abd9 |
ofp = NULL;
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
printf("Verifying %s\n", devname); fflush(stdout);
|
|
Packit |
c4abd9 |
for (cpu = 0; ; cpu++) {
|
|
Packit |
c4abd9 |
sprintf(fname, "%s.blktrace.%d", devname, cpu);
|
|
Packit |
c4abd9 |
if (stat(fname, &st) < 0) {
|
|
Packit |
c4abd9 |
if (cpu == 0) {
|
|
Packit |
c4abd9 |
fprintf(stderr, "No tracefiles found for %s\n",
|
|
Packit |
c4abd9 |
devname);
|
|
Packit |
c4abd9 |
rval = 1;
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
break;
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
printf(" CPU %d ", cpu); fflush(stdout);
|
|
Packit |
c4abd9 |
nbad = process(&ofp, devname, fname, cpu);
|
|
Packit |
c4abd9 |
if (nbad) {
|
|
Packit |
c4abd9 |
printf("-- %d bad", nbad);
|
|
Packit |
c4abd9 |
rval = 1;
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
printf("\n");
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
if (ofp) {
|
|
Packit |
c4abd9 |
fclose(ofp);
|
|
Packit |
c4abd9 |
fprintf(stdout, "Wrote output to %s\n", ofname);
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
}
|
|
Packit |
c4abd9 |
|
|
Packit |
c4abd9 |
return rval;
|
|
Packit |
c4abd9 |
}
|