* [PATCH][GIT PULL] tracing: add function profiler
@ 2009-03-21 4:37 Steven Rostedt
2009-03-21 11:26 ` Andrew Morton
0 siblings, 1 reply; 7+ messages in thread
From: Steven Rostedt @ 2009-03-21 4:37 UTC (permalink / raw)
To: LKML
Cc: Ingo Molnar, Peter Zijlstra, Thomas Gleixner,
Frédéric Weisbecker, Arjan van de Ven
Ingo,
Please pull the latest tip/tracing/ftrace tree, which can be found at:
git://git.kernel.org/pub/scm/linux/kernel/git/rostedt/linux-2.6-trace.git
tip/tracing/ftrace
Steven Rostedt (1):
tracing: add function profiler
----
include/linux/ftrace.h | 4 +
kernel/trace/Kconfig | 19 +++
kernel/trace/ftrace.c | 313 +++++++++++++++++++++++++++++++++++++++++++++++-
3 files changed, 334 insertions(+), 2 deletions(-)
---------------------------
commit b64b1cd804a0261e0052e67897c8f634414e94ce
Author: Steven Rostedt <srostedt@redhat.com>
Date: Fri Mar 20 12:50:56 2009 -0400
tracing: add function profiler
Impact: new profiling feature
This patch adds a function profiler. In debugfs/tracing/ two new
files are created.
function_profile_enabled - to enable or disable profiling
trace_stat/functions - the profiled functions.
For example:
echo 1 > /debugfs/tracing/function_profile_enabled
./hackbench 50
echo 0 > /debugfs/tracing/function_profile_enabled
yields:
cat /debugfs/tracing/trace_stat/functions
Function Hit
-------- ---
_spin_lock 10106442
_spin_unlock 10097492
kfree 6013704
_spin_unlock_irqrestore 4423941
_spin_lock_irqsave 4406825
__phys_addr 4181686
__slab_free 4038222
dput 4030130
path_put 4023387
unroll_tree_refs 4019532
[...]
The most hit functions are listed first. Functions that are not
hit are not listed.
This feature depends on and uses dynamic function tracing. When the
function profiling is disabled, no overhead occurs. But it still
takes up around 300KB to hold the data, thus it is not recomended
to keep it enabled for systems low on memory.
When a '1' is echoed into the function_profile_enabled file, the
counters for is function is reset back to zero. Thus you can see what
functions are hit most by different programs.
Signed-off-by: Steven Rostedt <srostedt@redhat.com>
diff --git a/include/linux/ftrace.h b/include/linux/ftrace.h
index db3fed6..33a71ba 100644
--- a/include/linux/ftrace.h
+++ b/include/linux/ftrace.h
@@ -147,6 +147,10 @@ enum {
struct dyn_ftrace {
unsigned long ip; /* address of mcount call-site */
unsigned long flags;
+#ifdef CONFIG_FUNCTION_PROFILER
+ unsigned long counter;
+ struct hlist_node node;
+#endif
struct dyn_arch_ftrace arch;
};
diff --git a/kernel/trace/Kconfig b/kernel/trace/Kconfig
index b0a46f8..cc81b96 100644
--- a/kernel/trace/Kconfig
+++ b/kernel/trace/Kconfig
@@ -101,6 +101,7 @@ config FUNCTION_GRAPH_TRACER
This is done by setting the current return address on the current
task structure into a stack of calls.
+
config IRQSOFF_TRACER
bool "Interrupts-off Latency Tracer"
default n
@@ -372,6 +373,24 @@ config DYNAMIC_FTRACE
were made. If so, it runs stop_machine (stops all CPUS)
and modifies the code to jump over the call to ftrace.
+config FUNCTION_PROFILER
+ bool "Kernel function profiler"
+ depends on DYNAMIC_FTRACE
+ default n
+ help
+ This option enables the kernel function profiler. When the dynamic
+ function tracing is enabled, a counter is added into the function
+ records used by the dynamic function tracer. A file is created in
+ debugfs called function_profile_enabled which defaults to zero.
+ When a 1 is echoed into this file profiling begins, and when a
+ zero is entered, profiling stops. A file in the trace_stats
+ directory called functions, that show the list of functions that
+ have been hit and their counters.
+
+ This takes up around 320K more memory.
+
+ If in doubt, say N
+
config FTRACE_MCOUNT_RECORD
def_bool y
depends on DYNAMIC_FTRACE
diff --git a/kernel/trace/ftrace.c b/kernel/trace/ftrace.c
index 7847806..ce74547 100644
--- a/kernel/trace/ftrace.c
+++ b/kernel/trace/ftrace.c
@@ -32,6 +32,7 @@
#include <asm/ftrace.h>
#include "trace.h"
+#include "trace_stat.h"
#define FTRACE_WARN_ON(cond) \
do { \
@@ -259,7 +260,6 @@ struct ftrace_func_probe {
struct rcu_head rcu;
};
-
enum {
FTRACE_ENABLE_CALLS = (1 << 0),
FTRACE_DISABLE_CALLS = (1 << 1),
@@ -307,6 +307,307 @@ static struct dyn_ftrace *ftrace_free_records;
} \
}
+#ifdef CONFIG_FUNCTION_PROFILER
+static struct hlist_head *ftrace_profile_hash;
+static int ftrace_profile_bits;
+static int ftrace_profile_enabled;
+static DEFINE_MUTEX(ftrace_profile_lock);
+
+static void *
+function_stat_next(void *v, int idx)
+{
+ struct dyn_ftrace *rec = v;
+ struct ftrace_page *pg;
+
+ pg = (struct ftrace_page *)((unsigned long)rec & PAGE_MASK);
+
+ again:
+ rec++;
+ if ((void *)rec >= (void *)&pg->records[pg->index]) {
+ pg = pg->next;
+ if (!pg)
+ return NULL;
+ rec = &pg->records[0];
+ }
+
+ if (rec->flags & FTRACE_FL_FREE ||
+ rec->flags & FTRACE_FL_FAILED ||
+ !(rec->flags & FTRACE_FL_CONVERTED) ||
+ /* ignore non hit functions */
+ !rec->counter)
+ goto again;
+
+ return rec;
+}
+
+static void *function_stat_start(void)
+{
+ return function_stat_next(&ftrace_pages_start->records[0], 0);
+}
+
+static int function_stat_cmp(void *p1, void *p2)
+{
+ struct dyn_ftrace *a = p1;
+ struct dyn_ftrace *b = p2;
+
+ if (a->counter < b->counter)
+ return -1;
+ if (a->counter > b->counter)
+ return 1;
+ else
+ return 0;
+}
+
+static int function_stat_headers(struct seq_file *m)
+{
+ seq_printf(m, " Function Hit\n"
+ " -------- ---\n");
+ return 0;
+}
+
+static int function_stat_show(struct seq_file *m, void *v)
+{
+ struct dyn_ftrace *rec = v;
+ char str[KSYM_SYMBOL_LEN];
+
+ kallsyms_lookup(rec->ip, NULL, NULL, NULL, str);
+
+ seq_printf(m, " %-30.30s %10lu\n", str, rec->counter);
+ return 0;
+}
+
+static struct tracer_stat function_stats = {
+ .name = "functions",
+ .stat_start = function_stat_start,
+ .stat_next = function_stat_next,
+ .stat_cmp = function_stat_cmp,
+ .stat_headers = function_stat_headers,
+ .stat_show = function_stat_show
+};
+
+static void ftrace_profile_init(int nr_funcs)
+{
+ unsigned long addr;
+ int order;
+ int size;
+
+ /*
+ * We are profiling all functions, lets make it 1/4th of the
+ * number of functions that are in core kernel. So we have to
+ * iterate 4 times.
+ */
+ order = (sizeof(struct hlist_head) * nr_funcs) / 4;
+ order = get_order(order);
+ size = 1 << (PAGE_SHIFT + order);
+
+ pr_info("Allocating %d KB for profiler hash\n", size >> 10);
+
+ addr = __get_free_pages(GFP_KERNEL | __GFP_ZERO, order);
+ if (!addr) {
+ pr_warning("Could not allocate function profiler hash\n");
+ return;
+ }
+
+ ftrace_profile_hash = (void *)addr;
+
+ /*
+ * struct hlist_head should be a pointer of 4 or 8 bytes.
+ * And a simple bit manipulation can be done, but if for
+ * some reason struct hlist_head is not a mulitple of 2,
+ * then we play it safe, and simply count. This function
+ * is done once at boot up, so it is not that critical in
+ * performance.
+ */
+
+ size--;
+ size /= sizeof(struct hlist_head);
+
+ for (; size; size >>= 1)
+ ftrace_profile_bits++;
+
+ pr_info("Function profiler has %d hash buckets\n",
+ 1 << ftrace_profile_bits);
+
+ return;
+}
+
+static ssize_t
+ftrace_profile_read(struct file *filp, char __user *ubuf,
+ size_t cnt, loff_t *ppos)
+{
+ char buf[64];
+ int r;
+
+ r = sprintf(buf, "%u\n", ftrace_profile_enabled);
+ return simple_read_from_buffer(ubuf, cnt, ppos, buf, r);
+}
+
+static void ftrace_profile_reset(void)
+{
+ struct dyn_ftrace *rec;
+ struct ftrace_page *pg;
+
+ do_for_each_ftrace_rec(pg, rec) {
+ rec->counter = 0;
+ } while_for_each_ftrace_rec();
+}
+
+static struct dyn_ftrace *ftrace_find_profiled_func(unsigned long ip)
+{
+ struct dyn_ftrace *rec;
+ struct hlist_head *hhd;
+ struct hlist_node *n;
+ unsigned long flags;
+ unsigned long key;
+
+ if (!ftrace_profile_hash)
+ return NULL;
+
+ key = hash_long(ip, ftrace_profile_bits);
+ hhd = &ftrace_profile_hash[key];
+
+ if (hlist_empty(hhd))
+ return NULL;
+
+ local_irq_save(flags);
+ hlist_for_each_entry_rcu(rec, n, hhd, node) {
+ if (rec->ip == ip)
+ goto out;
+ }
+ rec = NULL;
+ out:
+ local_irq_restore(flags);
+
+ return rec;
+}
+
+static void
+function_profile_call(unsigned long ip, unsigned long parent_ip)
+{
+ struct dyn_ftrace *rec;
+ unsigned long flags;
+
+ if (!ftrace_profile_enabled)
+ return;
+
+ local_irq_save(flags);
+ rec = ftrace_find_profiled_func(ip);
+ if (!rec)
+ goto out;
+
+ rec->counter++;
+ out:
+ local_irq_restore(flags);
+}
+
+static struct ftrace_ops ftrace_profile_ops __read_mostly =
+{
+ .func = function_profile_call,
+};
+
+static ssize_t
+ftrace_profile_write(struct file *filp, const char __user *ubuf,
+ size_t cnt, loff_t *ppos)
+{
+ unsigned long val;
+ char buf[64];
+ int ret;
+
+ if (!ftrace_profile_hash) {
+ pr_info("Can not enable hash due to earlier problems\n");
+ return -ENODEV;
+ }
+
+ if (cnt >= sizeof(buf))
+ return -EINVAL;
+
+ if (copy_from_user(&buf, ubuf, cnt))
+ return -EFAULT;
+
+ buf[cnt] = 0;
+
+ ret = strict_strtoul(buf, 10, &val);
+ if (ret < 0)
+ return ret;
+
+ val = !!val;
+
+ mutex_lock(&ftrace_profile_lock);
+ if (ftrace_profile_enabled ^ val) {
+ if (val) {
+ ftrace_profile_reset();
+ register_ftrace_function(&ftrace_profile_ops);
+ ftrace_profile_enabled = 1;
+ } else {
+ ftrace_profile_enabled = 0;
+ unregister_ftrace_function(&ftrace_profile_ops);
+ }
+ }
+ mutex_unlock(&ftrace_profile_lock);
+
+ filp->f_pos += cnt;
+
+ return cnt;
+}
+
+static const struct file_operations ftrace_profile_fops = {
+ .open = tracing_open_generic,
+ .read = ftrace_profile_read,
+ .write = ftrace_profile_write,
+};
+
+static void ftrace_profile_debugfs(struct dentry *d_tracer)
+{
+ struct dentry *entry;
+ int ret;
+
+ ret = register_stat_tracer(&function_stats);
+ if (ret) {
+ pr_warning("Warning: could not register "
+ "function stats\n");
+ return;
+ }
+
+ entry = debugfs_create_file("function_profile_enabled", 0644,
+ d_tracer, NULL, &ftrace_profile_fops);
+ if (!entry)
+ pr_warning("Could not create debugfs "
+ "'function_profile_enabled' entry\n");
+}
+
+static void ftrace_add_profile(struct dyn_ftrace *rec)
+{
+ unsigned long key;
+
+ if (!ftrace_profile_hash)
+ return;
+
+ key = hash_long(rec->ip, ftrace_profile_bits);
+ hlist_add_head_rcu(&rec->node, &ftrace_profile_hash[key]);
+}
+
+static void ftrace_profile_release(struct dyn_ftrace *rec)
+{
+ mutex_lock(&ftrace_profile_lock);
+ hlist_del(&rec->node);
+ mutex_unlock(&ftrace_profile_lock);
+}
+
+#else /* CONFIG_FUNCTION_PROFILER */
+static void ftrace_profile_init(int nr_funcs)
+{
+}
+static void ftrace_add_profile(struct dyn_ftrace *rec)
+{
+}
+static void ftrace_profile_debugfs(struct dentry *d_tracer)
+{
+}
+static void ftrace_profile_release(struct dyn_ftrace *rec)
+{
+}
+#endif /* CONFIG_FUNCTION_PROFILER */
+
#ifdef CONFIG_KPROBES
static int frozen_record_count;
@@ -357,8 +658,10 @@ void ftrace_release(void *start, unsigned long size)
mutex_lock(&ftrace_lock);
do_for_each_ftrace_rec(pg, rec) {
if ((rec->ip >= s) && (rec->ip < e) &&
- !(rec->flags & FTRACE_FL_FREE))
+ !(rec->flags & FTRACE_FL_FREE)) {
ftrace_free_rec(rec);
+ ftrace_profile_release(rec);
+ }
} while_for_each_ftrace_rec();
mutex_unlock(&ftrace_lock);
}
@@ -412,6 +715,8 @@ ftrace_record_ip(unsigned long ip)
rec->flags = (unsigned long)ftrace_new_addrs;
ftrace_new_addrs = rec;
+ ftrace_add_profile(rec);
+
return rec;
}
@@ -2155,6 +2460,8 @@ static __init int ftrace_init_dyn_debugfs(struct dentry *d_tracer)
"'set_graph_function' entry\n");
#endif /* CONFIG_FUNCTION_GRAPH_TRACER */
+ ftrace_profile_debugfs(d_tracer);
+
return 0;
}
@@ -2223,6 +2530,8 @@ void __init ftrace_init(void)
if (ret)
goto failed;
+ ftrace_profile_init(count);
+
last_ftrace_enabled = ftrace_enabled = 1;
ret = ftrace_convert_nops(NULL,
^ permalink raw reply related [flat|nested] 7+ messages in thread
* Re: [PATCH][GIT PULL] tracing: add function profiler
2009-03-21 4:37 [PATCH][GIT PULL] tracing: add function profiler Steven Rostedt
@ 2009-03-21 11:26 ` Andrew Morton
2009-03-21 11:57 ` Peter Zijlstra
2009-03-21 15:46 ` Steven Rostedt
0 siblings, 2 replies; 7+ messages in thread
From: Andrew Morton @ 2009-03-21 11:26 UTC (permalink / raw)
To: Steven Rostedt
Cc: LKML, Ingo Molnar, Peter Zijlstra, Thomas Gleixner,
Frédéric Weisbecker, Arjan van de Ven
On Sat, 21 Mar 2009 00:37:59 -0400 (EDT) Steven Rostedt <rostedt@goodmis.org> wrote:
> This patch adds a function profiler. In debugfs/tracing/ two new
> files are created.
>
> function_profile_enabled - to enable or disable profiling
>
> trace_stat/functions - the profiled functions.
>
> For example:
>
> echo 1 > /debugfs/tracing/function_profile_enabled
> ./hackbench 50
> echo 0 > /debugfs/tracing/function_profile_enabled
>
> yields:
>
> cat /debugfs/tracing/trace_stat/functions
>
> Function Hit
> -------- ---
> _spin_lock 10106442
> _spin_unlock 10097492
> kfree 6013704
> _spin_unlock_irqrestore 4423941
> _spin_lock_irqsave 4406825
> __phys_addr 4181686
> __slab_free 4038222
> dput 4030130
> path_put 4023387
> unroll_tree_refs 4019532
> [...]
>
> The most hit functions are listed first. Functions that are not
> hit are not listed.
Why is this useful?
Can we think of any scenarios where kernel developers would get
useful-to-them results from this? Results which couldn't be
obtained by other similarly-accessible means?
<strains a bit>
I guess that one could run workload A, look at
/debugfs/tracing/trace_stat/functions changes, then run worklaod B, then
look at its /debugfs/tracing/trace_stat/functions changes, then somehow
glean some information about the differences between the effects of the two
workloads on the kernel. Or something.
But in this rather fake example and, I suspect, in many others, the result
will be less useful than using oprofile/etc in the same fashion.
^ permalink raw reply [flat|nested] 7+ messages in thread
* Re: [PATCH][GIT PULL] tracing: add function profiler
2009-03-21 11:26 ` Andrew Morton
@ 2009-03-21 11:57 ` Peter Zijlstra
2009-03-21 14:12 ` Ingo Molnar
2009-03-21 15:50 ` Steven Rostedt
2009-03-21 15:46 ` Steven Rostedt
1 sibling, 2 replies; 7+ messages in thread
From: Peter Zijlstra @ 2009-03-21 11:57 UTC (permalink / raw)
To: Andrew Morton
Cc: Steven Rostedt, LKML, Ingo Molnar, Thomas Gleixner,
Frédéric Weisbecker, Arjan van de Ven, Paul Mackerras
On Sat, 2009-03-21 at 04:26 -0700, Andrew Morton wrote:
> On Sat, 21 Mar 2009 00:37:59 -0400 (EDT) Steven Rostedt <rostedt@goodmis.org> wrote:
>
> > This patch adds a function profiler. In debugfs/tracing/ two new
> > files are created.
> >
> > function_profile_enabled - to enable or disable profiling
> >
> > trace_stat/functions - the profiled functions.
> >
> > For example:
> >
> > echo 1 > /debugfs/tracing/function_profile_enabled
> > ./hackbench 50
> > echo 0 > /debugfs/tracing/function_profile_enabled
> >
> > yields:
> >
> > cat /debugfs/tracing/trace_stat/functions
> >
> > Function Hit
> > -------- ---
> > _spin_lock 10106442
> > _spin_unlock 10097492
> > kfree 6013704
> > _spin_unlock_irqrestore 4423941
> > _spin_lock_irqsave 4406825
> > __phys_addr 4181686
> > __slab_free 4038222
> > dput 4030130
> > path_put 4023387
> > unroll_tree_refs 4019532
> > [...]
> >
> > The most hit functions are listed first. Functions that are not
> > hit are not listed.
>
> Why is this useful?
>
> Can we think of any scenarios where kernel developers would get
> useful-to-them results from this? Results which couldn't be
> obtained by other similarly-accessible means?
>
> <strains a bit>
>
> I guess that one could run workload A, look at
> /debugfs/tracing/trace_stat/functions changes, then run worklaod B, then
> look at its /debugfs/tracing/trace_stat/functions changes, then somehow
> glean some information about the differences between the effects of the two
> workloads on the kernel. Or something.
>
> But in this rather fake example and, I suspect, in many others, the result
> will be less useful than using oprofile/etc in the same fashion.
I have to agree with Andrew here, my plan is to remove all the
profiling stuff from kernel/trace in favour of perf counters.
If you want exact function count profiling we could try to do something
perf counter based, eg. stick a software counter in the mcount thingy.
After that you'd need to get something like
this_pt_regs()/caller_pt_regs() which would provide the current kernel
stack information to generate profile information from.
Current software events use get_irq_regs() ?: task_pt_regs() for lack of
anything better.
^ permalink raw reply [flat|nested] 7+ messages in thread
* Re: [PATCH][GIT PULL] tracing: add function profiler
2009-03-21 11:57 ` Peter Zijlstra
@ 2009-03-21 14:12 ` Ingo Molnar
2009-03-21 15:50 ` Steven Rostedt
1 sibling, 0 replies; 7+ messages in thread
From: Ingo Molnar @ 2009-03-21 14:12 UTC (permalink / raw)
To: Peter Zijlstra
Cc: Andrew Morton, Steven Rostedt, LKML, Thomas Gleixner,
Frédéric Weisbecker, Arjan van de Ven, Paul Mackerras
* Peter Zijlstra <peterz@infradead.org> wrote:
> On Sat, 2009-03-21 at 04:26 -0700, Andrew Morton wrote:
> > On Sat, 21 Mar 2009 00:37:59 -0400 (EDT) Steven Rostedt <rostedt@goodmis.org> wrote:
> >
> > > This patch adds a function profiler. In debugfs/tracing/ two new
> > > files are created.
> > >
> > > function_profile_enabled - to enable or disable profiling
> > >
> > > trace_stat/functions - the profiled functions.
> > >
> > > For example:
> > >
> > > echo 1 > /debugfs/tracing/function_profile_enabled
> > > ./hackbench 50
> > > echo 0 > /debugfs/tracing/function_profile_enabled
> > >
> > > yields:
> > >
> > > cat /debugfs/tracing/trace_stat/functions
> > >
> > > Function Hit
> > > -------- ---
> > > _spin_lock 10106442
> > > _spin_unlock 10097492
> > > kfree 6013704
> > > _spin_unlock_irqrestore 4423941
> > > _spin_lock_irqsave 4406825
> > > __phys_addr 4181686
> > > __slab_free 4038222
> > > dput 4030130
> > > path_put 4023387
> > > unroll_tree_refs 4019532
> > > [...]
> > >
> > > The most hit functions are listed first. Functions that are not
> > > hit are not listed.
> >
> > Why is this useful?
> >
> > Can we think of any scenarios where kernel developers would get
> > useful-to-them results from this? Results which couldn't be
> > obtained by other similarly-accessible means?
> >
> > <strains a bit>
> >
> > I guess that one could run workload A, look at
> > /debugfs/tracing/trace_stat/functions changes, then run worklaod B, then
> > look at its /debugfs/tracing/trace_stat/functions changes, then somehow
> > glean some information about the differences between the effects of the two
> > workloads on the kernel. Or something.
> >
> > But in this rather fake example and, I suspect, in many others,
> > the result will be less useful than using oprofile/etc in the
> > same fashion.
>
> I have to agree with Andrew here, my plan is to remove all the
> profiling stuff from kernel/trace in favour of perf counters.
i agree with that - but still it would be useful to also have the
ability to do in-kernel histograms and the likely/unlikely profiler
(which is using the same histogram code) is already using that
facility.
But instead of the single-purpose likely/unlikely and now function
histogram code, we should allow individual counters/events to be
linked up with an in-kernel histogram - or route it to user-space
via perfcounters.
> If you want exact function count profiling we could try to do
> something perf counter based, eg. stick a software counter in the
> mcount thingy.
while i agree in general - lets realize that function (and branch)
software events _are_ special due to their sheer mass. I dont think
it's feasible to route that much information to user-space - just to
collapse it into a histogram there. Even doing it in the kernel is
already straining the performance envelope quite a bit.
Ingo
^ permalink raw reply [flat|nested] 7+ messages in thread
* Re: [PATCH][GIT PULL] tracing: add function profiler
2009-03-21 11:26 ` Andrew Morton
2009-03-21 11:57 ` Peter Zijlstra
@ 2009-03-21 15:46 ` Steven Rostedt
1 sibling, 0 replies; 7+ messages in thread
From: Steven Rostedt @ 2009-03-21 15:46 UTC (permalink / raw)
To: Andrew Morton
Cc: LKML, Ingo Molnar, Peter Zijlstra, Thomas Gleixner,
Frédéric Weisbecker, Arjan van de Ven
On Sat, 21 Mar 2009, Andrew Morton wrote:
> On Sat, 21 Mar 2009 00:37:59 -0400 (EDT) Steven Rostedt <rostedt@goodmis.org> wrote:
>
> > This patch adds a function profiler. In debugfs/tracing/ two new
> > files are created.
> >
> > function_profile_enabled - to enable or disable profiling
> >
> > trace_stat/functions - the profiled functions.
> >
> > For example:
> >
> > echo 1 > /debugfs/tracing/function_profile_enabled
> > ./hackbench 50
> > echo 0 > /debugfs/tracing/function_profile_enabled
> >
> > yields:
> >
> > cat /debugfs/tracing/trace_stat/functions
> >
> > Function Hit
> > -------- ---
> > _spin_lock 10106442
> > _spin_unlock 10097492
> > kfree 6013704
> > _spin_unlock_irqrestore 4423941
> > _spin_lock_irqsave 4406825
> > __phys_addr 4181686
> > __slab_free 4038222
> > dput 4030130
> > path_put 4023387
> > unroll_tree_refs 4019532
> > [...]
> >
> > The most hit functions are listed first. Functions that are not
> > hit are not listed.
>
> Why is this useful?
>
> Can we think of any scenarios where kernel developers would get
> useful-to-them results from this? Results which couldn't be
> obtained by other similarly-accessible means?
>
> <strains a bit>
>
> I guess that one could run workload A, look at
> /debugfs/tracing/trace_stat/functions changes, then run worklaod B, then
> look at its /debugfs/tracing/trace_stat/functions changes, then somehow
> glean some information about the differences between the effects of the two
> workloads on the kernel. Or something.
>
> But in this rather fake example and, I suspect, in many others, the result
> will be less useful than using oprofile/etc in the same fashion.
Actually, it would not be too hard to hook this to the function graph
tracer and make the list not only # of hits, but also the time spent in
each function.
-- Steve
^ permalink raw reply [flat|nested] 7+ messages in thread
* Re: [PATCH][GIT PULL] tracing: add function profiler
2009-03-21 11:57 ` Peter Zijlstra
2009-03-21 14:12 ` Ingo Molnar
@ 2009-03-21 15:50 ` Steven Rostedt
2009-03-21 16:26 ` Ingo Molnar
1 sibling, 1 reply; 7+ messages in thread
From: Steven Rostedt @ 2009-03-21 15:50 UTC (permalink / raw)
To: Peter Zijlstra
Cc: Andrew Morton, LKML, Ingo Molnar, Thomas Gleixner,
Frédéric Weisbecker, Arjan van de Ven, Paul Mackerras
On Sat, 21 Mar 2009, Peter Zijlstra wrote:
> On Sat, 2009-03-21 at 04:26 -0700, Andrew Morton wrote:
> > On Sat, 21 Mar 2009 00:37:59 -0400 (EDT) Steven Rostedt <rostedt@goodmis.org> wrote:
> >
> > > This patch adds a function profiler. In debugfs/tracing/ two new
> > > files are created.
> > >
> > > function_profile_enabled - to enable or disable profiling
> > >
> > > trace_stat/functions - the profiled functions.
> > >
> > > For example:
> > >
> > > echo 1 > /debugfs/tracing/function_profile_enabled
> > > ./hackbench 50
> > > echo 0 > /debugfs/tracing/function_profile_enabled
> > >
> > > yields:
> > >
> > > cat /debugfs/tracing/trace_stat/functions
> > >
> > > Function Hit
> > > -------- ---
> > > _spin_lock 10106442
> > > _spin_unlock 10097492
> > > kfree 6013704
> > > _spin_unlock_irqrestore 4423941
> > > _spin_lock_irqsave 4406825
> > > __phys_addr 4181686
> > > __slab_free 4038222
> > > dput 4030130
> > > path_put 4023387
> > > unroll_tree_refs 4019532
> > > [...]
> > >
> > > The most hit functions are listed first. Functions that are not
> > > hit are not listed.
> >
> > Why is this useful?
> >
> > Can we think of any scenarios where kernel developers would get
> > useful-to-them results from this? Results which couldn't be
> > obtained by other similarly-accessible means?
> >
> > <strains a bit>
> >
> > I guess that one could run workload A, look at
> > /debugfs/tracing/trace_stat/functions changes, then run worklaod B, then
> > look at its /debugfs/tracing/trace_stat/functions changes, then somehow
> > glean some information about the differences between the effects of the two
> > workloads on the kernel. Or something.
> >
> > But in this rather fake example and, I suspect, in many others, the result
> > will be less useful than using oprofile/etc in the same fashion.
>
> I have to agree with Andrew here, my plan is to remove all the
> profiling stuff from kernel/trace in favour of perf counters.
Well, the branch profilers, I would think, are too heavy for perf
counters.
>
> If you want exact function count profiling we could try to do something
> perf counter based, eg. stick a software counter in the mcount thingy.
I'd like to get the function graph version working, in order to calculate
the times spent in each funtion, then I can see how we can (if possible)
convert it over to perf counters. :-/
>
> After that you'd need to get something like
> this_pt_regs()/caller_pt_regs() which would provide the current kernel
> stack information to generate profile information from.
>
> Current software events use get_irq_regs() ?: task_pt_regs() for lack of
> anything better.
I need to take a deeper look at perf counters. Not sure why pt_regs would
be needed for the function counting. All the information needed is passed
via the mcount call.
-- Steve
^ permalink raw reply [flat|nested] 7+ messages in thread
* Re: [PATCH][GIT PULL] tracing: add function profiler
2009-03-21 15:50 ` Steven Rostedt
@ 2009-03-21 16:26 ` Ingo Molnar
0 siblings, 0 replies; 7+ messages in thread
From: Ingo Molnar @ 2009-03-21 16:26 UTC (permalink / raw)
To: Steven Rostedt
Cc: Peter Zijlstra, Andrew Morton, LKML, Thomas Gleixner,
Frédéric Weisbecker, Arjan van de Ven, Paul Mackerras
* Steven Rostedt <rostedt@goodmis.org> wrote:
> > If you want exact function count profiling we could try to do
> > something perf counter based, eg. stick a software counter in
> > the mcount thingy.
>
> I'd like to get the function graph version working, in order to
> calculate the times spent in each funtion, then I can see how we
> can (if possible) convert it over to perf counters. :-/
the perf counters angle is definitely worth pursuing (for
completeness of the scheme Peter is working on), but please lets not
throw the baby out with the bath water: in-kernel histogram
collection has real value.
It's _much_ faster than any user-space counter-part, especially for
such things as branch tracing or function tracing.
Ingo
^ permalink raw reply [flat|nested] 7+ messages in thread
end of thread, other threads:[~2009-03-21 16:27 UTC | newest]
Thread overview: 7+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2009-03-21 4:37 [PATCH][GIT PULL] tracing: add function profiler Steven Rostedt
2009-03-21 11:26 ` Andrew Morton
2009-03-21 11:57 ` Peter Zijlstra
2009-03-21 14:12 ` Ingo Molnar
2009-03-21 15:50 ` Steven Rostedt
2009-03-21 16:26 ` Ingo Molnar
2009-03-21 15:46 ` Steven Rostedt
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.