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