[<prev] [next>] [day] [month] [year] [list]
Message-Id: <1359554742-4705-1-git-send-email-chenggang.qin@gmail.com>
Date: Wed, 30 Jan 2013 22:05:42 +0800
From: chenggang <chenggang.qin@...il.com>
To: linux-kernel@...r.kernel.org
Cc: chenggang <chenggang.qin@...il.com>,
Peter Zijlstra <peterz@...radead.org>,
Paul Mackerras <paulus@...ba.org>,
Ingo Molnar <mingo@...hat.com>,
Arnaldo Carvalho de Melo <acme@...stprotocols.net>,
David Ahern <dsahern@...il.com>,
Steven Rostedt <rostedt@...dmis.org>,
Frederic Weisbecker <fweisbec@...il.com>,
Arjan van de Ven <arjan@...ux.intel.com>,
Namhyung Kim <namhyung@...il.com>,
Yanmin Zhang <yanmin.zhang@...el.com>,
Wu Fengguang <fengguang.wu@...el.com>,
Mike Galbraith <efault@....de>,
Andrew Morton <akpm@...ux-foundation.org>,
Chenggang Qin <chenggang.qcg@...baba-inc.com>
Subject: [PATCH 2/2] perf script: add python script to show system's file r/w behavior
From: chenggang.qin@...il.com
This patch depends on the other patch: https://lkml.org/lkml/2013/1/29/47
Because this patch uses 2 tracepoint events are introduced by the patch of the
above mentioned.
If the engineers want to analyze the file access behavior of some applications
without source code, perf script mechanism with some appropriate tracepoints
events in the VFS subsystem are excellent choice.
The system engineers or developers of server software require to know what files
are accessed by the target processes with in a period of time. Then they can
find the hot applications and the hot files. Based on the two tracepoint events,
vfs:generic_file_aio_read and vfs:generic_file_aio_write (introduced by the patch:
https://lkml.org/lkml/2013/1/29/47), the python script are introduced by this patch
can record the system context and other related infomation while any process access
a file. Then, this patch can show the details of the file access behavior of every
processes. The detail information include: process's pid, comm, the number of file
read/write, and the related file's name.
The usage of this script is:
perf script record filerw
perf script report filerw [comm|pid]
Cc: Peter Zijlstra <peterz@...radead.org>
Cc: Paul Mackerras <paulus@...ba.org>
Cc: Ingo Molnar <mingo@...hat.com>
Cc: Arnaldo Carvalho de Melo <acme@...stprotocols.net>
Cc: David Ahern <dsahern@...il.com>
Cc: Steven Rostedt <rostedt@...dmis.org>
Cc: Frederic Weisbecker <fweisbec@...il.com>
Cc: Arjan van de Ven <arjan@...ux.intel.com>
Cc: Namhyung Kim <namhyung@...il.com>
Cc: Yanmin Zhang <yanmin.zhang@...el.com>
Cc: Wu Fengguang <fengguang.wu@...el.com>
Cc: Mike Galbraith <efault@....de>
Cc: Andrew Morton <akpm@...ux-foundation.org>
Signed-off-by: Chenggang Qin <chenggang.qcg@...baba-inc.com>
---
tools/perf/scripts/python/bin/filerw-record | 2 +
tools/perf/scripts/python/bin/filerw-report | 21 +++
tools/perf/scripts/python/filerw.py | 189 +++++++++++++++++++++++++++
3 files changed, 212 insertions(+)
create mode 100755 tools/perf/scripts/python/bin/filerw-record
create mode 100644 tools/perf/scripts/python/bin/filerw-report
create mode 100644 tools/perf/scripts/python/filerw.py
diff --git a/tools/perf/scripts/python/bin/filerw-record b/tools/perf/scripts/python/bin/filerw-record
new file mode 100755
index 0000000..80f358c
--- /dev/null
+++ b/tools/perf/scripts/python/bin/filerw-record
@@ -0,0 +1,2 @@
+#!/bin/bash
+perf record -e vfs:generic_file_aio_read -e vfs:generic_file_aio_write $@
diff --git a/tools/perf/scripts/python/bin/filerw-report b/tools/perf/scripts/python/bin/filerw-report
new file mode 100644
index 0000000..5a4dac9
--- /dev/null
+++ b/tools/perf/scripts/python/bin/filerw-report
@@ -0,0 +1,21 @@
+#!/bin/bash
+# description: file read/write operations statistic
+# args: [comm]
+n_args=0
+for i in "$@"
+do
+ if expr match "$i" "-" > /dev/null ; then
+ break
+ fi
+ n_args=$(( $n_args + 1 ))
+done
+if [ "$n_args" -gt 1 ] ; then
+ echo "usage: perf script report filerw [comm|pid]"
+ exit
+fi
+
+if [ "$n_args" -gt 0 ] ; then
+ comm=$1
+ shift
+fi
+perf script $@ -s "$PERF_EXEC_PATH"/scripts/python/filerw.py $comm
diff --git a/tools/perf/scripts/python/filerw.py b/tools/perf/scripts/python/filerw.py
new file mode 100644
index 0000000..f5bd820
--- /dev/null
+++ b/tools/perf/scripts/python/filerw.py
@@ -0,0 +1,189 @@
+# file read/write counts
+# (c) 2013, Chenggang Qin <chenggang.qcg@...il.com>
+# Licensed under the terms of the GNU GPL License version 2
+
+# Displays system-wide file aio read/write behavior.
+# It helps us to investigate what files are accessed by all
+# processes or a special process.
+#
+# options
+# comm: show details of the file r/w behavior of a special process.
+
+import os, sys
+
+sys.path.append(os.environ['PERF_EXEC_PATH'] + \
+ '/scripts/python/Perf-Trace-Util/lib/Perf/Trace')
+
+from perf_trace_context import *
+from Core import *
+from Util import *
+
+usage = "perf script record filerw\n" \
+ "perf script report filerw [comm|pid]\n"
+
+for_comm = None
+for_pid = None
+pid_2_comm = None
+
+if len(sys.argv) > 2:
+ sys.exit(usage)
+
+if len(sys.argv) > 1:
+ try:
+ for_pid = int(sys.argv[1])
+ except:
+ for_comm = sys.argv[1]
+
+file_write = autodict()
+file_read = autodict()
+
+file_write_bytes = autodict()
+file_read_bytes = autodict()
+
+comm_read_info = autodict()
+comm_write_info = autodict()
+
+wevent_count = 0
+revent_count = 0
+
+comm_revent_count = 0;
+comm_wevent_count = 0;
+
+def trace_begin():
+ print "Press control+C to stop and show the summary"
+
+def trace_end():
+ if (for_comm is not None) or (for_pid is not None):
+ print_file_event_for_comm()
+ else:
+ print_file_event_totals()
+
+def vfs__generic_file_aio_write(event_name, context, common_cpu,
+ common_secs, common_nsecs, common_pid, common_comm,
+ pos, bytes, fname):
+ global wevent_count
+ global comm_wevent_count
+ global pid_2_comm
+
+ if (for_comm is not None) or (for_pid is not None):
+ if (common_comm != for_comm) and (common_pid != for_pid):
+ return
+ else:
+ if (pid_2_comm is None) and (for_pid is not None):
+ pid_2_comm = common_comm
+ comm_write_info[comm_wevent_count] = (common_pid, \
+ common_cpu, common_secs, common_nsecs, \
+ fname, pos, bytes)
+ comm_wevent_count += 1
+ return
+
+ wevent_count += 1
+ try:
+ file_write[(common_pid, common_comm, fname)] += 1
+ file_write_bytes[(common_pid, common_comm, fname)] += bytes
+ except TypeError:
+ file_write[(common_pid, common_comm, fname)] = 1
+ file_write_bytes[(common_pid, common_comm, fname)] = bytes
+
+def vfs__generic_file_aio_read(event_name, context, common_cpu,
+ common_secs, common_nsecs, common_pid, common_comm,
+ pos, bytes, fname):
+ global revent_count
+ global comm_revent_count
+ global pid_2_comm
+
+ if (for_comm is not None) or (for_pid is not None):
+ if (common_comm != for_comm) and (common_pid != for_pid):
+ return
+ else:
+ if (pid_2_comm is None) and (for_pid is not None):
+ pid_2_comm = common_comm
+ comm_read_info[comm_revent_count] = (common_pid, \
+ common_cpu, common_secs, common_nsecs, \
+ fname, pos, bytes)
+ comm_revent_count += 1
+ return
+
+ try:
+ revent_count += 1
+ file_read[(common_pid, common_comm, fname)] += 1
+ file_read_bytes[(common_pid, common_comm, fname)] += bytes
+ except TypeError:
+ file_read[(common_pid, common_comm, fname)] = 1
+ file_read_bytes[(common_pid, common_comm, fname)] = bytes
+
+def print_file_event_totals():
+ clear_term()
+ if for_comm is not None:
+ print "\nfile accesses for %s:\n\n" % (for_comm),
+ else:
+ print "\nfile accesses:\n\n",
+
+ print "All read events: %d\n" % (revent_count)
+ print "%5s %20s %40s %10s %10s\n" % ("pid", "comm", "read file", "count", "Bytes"),
+ print "%5s %20s %40s %10s %10s\n" % ("-----", "----------",\
+ "----------", "----------", \
+ "----------"),
+
+ for tuple, val in sorted(file_read.iteritems(), key = lambda(k, v): (v, k), \
+ reverse = True):
+ try:
+ print "%5d %20s %40s %10d %10d\n" % \
+ (tuple[0], tuple[1], tuple[2][:40], val, \
+ file_read_bytes[tuple[0], tuple[1], tuple[2]])
+ except TypeError:
+ pass
+
+ print "\n\n"
+
+ print "All write events: %d\n" % (wevent_count)
+ print "%5s %20s %40s %10s %10s\n" % ("pid", "comm", "write file", "count", "Bytes"),
+ print "%5s %20s %40s %10s %10s\n" % ("-----", "----------",\
+ "----------", "----------", \
+ "----------"),
+
+ for tuple, val in sorted(file_write.iteritems(), key = lambda(k, v): (v, k), \
+ reverse = True):
+ try:
+ print "%5d %20s %40s %10d %10d\n" % \
+ (tuple[0], tuple[1], tuple[2][:40], val, \
+ file_write_bytes[tuple[0], tuple[1], tuple[2]])
+ except TypeError:
+ pass
+
+def print_file_event_for_comm():
+ if for_comm is not None:
+ print "File r/w record for %s:\n" % (for_comm)
+ else:
+ print "File r/w record for %s (pid: %d):\n" % (pid_2_comm, for_pid)
+ print "Number of read: %d\n" % (comm_revent_count)
+ print "%5s %3s %16s %40s %10s %10s\n" % ("pid", "cpu", "timestamp", "file name", \
+ "position", "Bytes")
+ print "%5s %3s %16s %40s %10s %10s\n" % ("-----", "---", "----------", \
+ "----------", "----------", \
+ "----------"),
+
+ for i in range(0, comm_revent_count):
+ try:
+ print "%5d %3d %6d.%9d %40s %10s %10s\n" % (comm_read_info[i][0], \
+ comm_read_info[i][1], comm_read_info[i][2], \
+ comm_read_info[i][3], comm_read_info[i][4], \
+ comm_read_info[i][5], comm_read_info[i][6])
+ except TypeError:
+ pass
+
+ print "\nNumber of write: %d\n" % (comm_wevent_count)
+ print "%5s %3s %16s %40s %10s %10s\n" % ("pid", "cpu", "timestamp", "file name", \
+ "position", "Bytes")
+ print "%5s %3s %16s %40s %10s %10s\n" % ("-----", "---", "----------", \
+ "----------", "----------", \
+ "----------"),
+ for i in range(0, comm_wevent_count):
+ try:
+ print "%5d %3d %6d.%9d %40s %10s %10s\n" % (comm_write_info[i][0], \
+ comm_write_info[i][1], comm_write_info[i][2], \
+ comm_write_info[i][3], comm_write_info[i][4], \
+ comm_write_info[i][5], comm_write_info[i][6])
+ except TypeError:
+ pass
+
--
1.7.9.5
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/
Powered by blists - more mailing lists