[PATCH v11 32/39] perf probe: Attach trace_probe_event with perf_probe_event

2015-07-08 Thread Wang Nan
This patch drops struct __event_package structure. Instead, it adds
trace_probe_event into 'struct perf_probe_event'.

trace_probe_event information gives further patches a chance to access
actual probe points and actual arguments. Using them, bpf_loader will
be able to attach one bpf program to different probing points of a
inline functions (which has multiple probing points) and glob
functions. Moreover, by reading arguments information, bpf code for
reading those arguments can be generated.

Signed-off-by: Wang Nan 
---
 tools/perf/builtin-probe.c|  4 ++-
 tools/perf/util/probe-event.c | 60 +--
 tools/perf/util/probe-event.h |  6 -
 3 files changed, 38 insertions(+), 32 deletions(-)

diff --git a/tools/perf/builtin-probe.c b/tools/perf/builtin-probe.c
index b81cec3..826d452 100644
--- a/tools/perf/builtin-probe.c
+++ b/tools/perf/builtin-probe.c
@@ -496,7 +496,9 @@ __cmd_probe(int argc, const char **argv, const char *prefix 
__maybe_unused)
usage_with_options(probe_usage, options);
}
 
-   ret = add_perf_probe_events(params.events, params.nevents);
+   ret = add_perf_probe_events(params.events,
+   params.nevents,
+   true);
if (ret < 0) {
pr_err_with_code("  Error: Failed to add events.", ret);
return ret;
diff --git a/tools/perf/util/probe-event.c b/tools/perf/util/probe-event.c
index 381f23a..083e8b4 100644
--- a/tools/perf/util/probe-event.c
+++ b/tools/perf/util/probe-event.c
@@ -1930,6 +1930,9 @@ void clear_perf_probe_event(struct perf_probe_event *pev)
struct perf_probe_arg_field *field, *next;
int i;
 
+   if (pev->ntevs)
+   cleanup_perf_probe_event(pev);
+
free(pev->event);
free(pev->group);
free(pev->target);
@@ -2778,61 +2781,58 @@ static int convert_to_probe_trace_events(struct 
perf_probe_event *pev,
return find_probe_trace_events_from_map(pev, tevs);
 }
 
-struct __event_package {
-   struct perf_probe_event *pev;
-   struct probe_trace_event*tevs;
-   int ntevs;
-};
-
-int add_perf_probe_events(struct perf_probe_event *pevs, int npevs)
+int cleanup_perf_probe_event(struct perf_probe_event *pev)
 {
-   int i, j, ret;
-   struct __event_package *pkgs;
+   int i;
 
-   ret = 0;
-   pkgs = zalloc(sizeof(struct __event_package) * npevs);
+   if (!pev || !pev->ntevs)
+   return 0;
 
-   if (pkgs == NULL)
-   return -ENOMEM;
+   for (i = 0; i < pev->ntevs; i++)
+   clear_probe_trace_event(>tevs[i]);
+
+   zfree(>tevs);
+   pev->ntevs = 0;
+   return 0;
+}
+
+int add_perf_probe_events(struct perf_probe_event *pevs, int npevs,
+ bool cleanup)
+{
+   int i, ret;
 
ret = init_symbol_maps(pevs->uprobes);
-   if (ret < 0) {
-   free(pkgs);
+   if (ret < 0)
return ret;
-   }
 
/* Loop 1: convert all events */
for (i = 0; i < npevs; i++) {
-   pkgs[i].pev = [i];
/* Init kprobe blacklist if needed */
-   if (!pkgs[i].pev->uprobes)
+   if (pevs[i].uprobes)
kprobe_blacklist__init();
/* Convert with or without debuginfo */
-   ret  = convert_to_probe_trace_events(pkgs[i].pev,
-[i].tevs);
-   if (ret < 0)
+   ret  = convert_to_probe_trace_events([i], [i].tevs);
+   if (ret < 0) {
+   cleanup = true;
goto end;
-   pkgs[i].ntevs = ret;
+   }
+   pevs[i].ntevs = ret;
}
/* This just release blacklist only if allocated */
kprobe_blacklist__release();
 
/* Loop 2: add all events */
for (i = 0; i < npevs; i++) {
-   ret = __add_probe_trace_events(pkgs[i].pev, pkgs[i].tevs,
-  pkgs[i].ntevs,
+   ret = __add_probe_trace_events([i], pevs[i].tevs,
+  pevs[i].ntevs,
   probe_conf.force_add);
if (ret < 0)
break;
}
 end:
/* Loop 3: cleanup and free trace events  */
-   for (i = 0; i < npevs; i++) {
-   for (j = 0; j < pkgs[i].ntevs; j++)
-   clear_probe_trace_event([i].tevs[j]);
-   zfree([i].tevs);
-   }
-   free(pkgs);
+   for (i = 0; cleanup && (i < npevs); i++)
+   cleanup_perf_probe_event([i]);
exit_symbol_maps();
 
return ret;
diff --git a/tools/perf/util/probe-event.h 

[PATCH v11 32/39] perf probe: Attach trace_probe_event with perf_probe_event

2015-07-08 Thread Wang Nan
This patch drops struct __event_package structure. Instead, it adds
trace_probe_event into 'struct perf_probe_event'.

trace_probe_event information gives further patches a chance to access
actual probe points and actual arguments. Using them, bpf_loader will
be able to attach one bpf program to different probing points of a
inline functions (which has multiple probing points) and glob
functions. Moreover, by reading arguments information, bpf code for
reading those arguments can be generated.

Signed-off-by: Wang Nan wangn...@huawei.com
---
 tools/perf/builtin-probe.c|  4 ++-
 tools/perf/util/probe-event.c | 60 +--
 tools/perf/util/probe-event.h |  6 -
 3 files changed, 38 insertions(+), 32 deletions(-)

diff --git a/tools/perf/builtin-probe.c b/tools/perf/builtin-probe.c
index b81cec3..826d452 100644
--- a/tools/perf/builtin-probe.c
+++ b/tools/perf/builtin-probe.c
@@ -496,7 +496,9 @@ __cmd_probe(int argc, const char **argv, const char *prefix 
__maybe_unused)
usage_with_options(probe_usage, options);
}
 
-   ret = add_perf_probe_events(params.events, params.nevents);
+   ret = add_perf_probe_events(params.events,
+   params.nevents,
+   true);
if (ret  0) {
pr_err_with_code(  Error: Failed to add events., ret);
return ret;
diff --git a/tools/perf/util/probe-event.c b/tools/perf/util/probe-event.c
index 381f23a..083e8b4 100644
--- a/tools/perf/util/probe-event.c
+++ b/tools/perf/util/probe-event.c
@@ -1930,6 +1930,9 @@ void clear_perf_probe_event(struct perf_probe_event *pev)
struct perf_probe_arg_field *field, *next;
int i;
 
+   if (pev-ntevs)
+   cleanup_perf_probe_event(pev);
+
free(pev-event);
free(pev-group);
free(pev-target);
@@ -2778,61 +2781,58 @@ static int convert_to_probe_trace_events(struct 
perf_probe_event *pev,
return find_probe_trace_events_from_map(pev, tevs);
 }
 
-struct __event_package {
-   struct perf_probe_event *pev;
-   struct probe_trace_event*tevs;
-   int ntevs;
-};
-
-int add_perf_probe_events(struct perf_probe_event *pevs, int npevs)
+int cleanup_perf_probe_event(struct perf_probe_event *pev)
 {
-   int i, j, ret;
-   struct __event_package *pkgs;
+   int i;
 
-   ret = 0;
-   pkgs = zalloc(sizeof(struct __event_package) * npevs);
+   if (!pev || !pev-ntevs)
+   return 0;
 
-   if (pkgs == NULL)
-   return -ENOMEM;
+   for (i = 0; i  pev-ntevs; i++)
+   clear_probe_trace_event(pev-tevs[i]);
+
+   zfree(pev-tevs);
+   pev-ntevs = 0;
+   return 0;
+}
+
+int add_perf_probe_events(struct perf_probe_event *pevs, int npevs,
+ bool cleanup)
+{
+   int i, ret;
 
ret = init_symbol_maps(pevs-uprobes);
-   if (ret  0) {
-   free(pkgs);
+   if (ret  0)
return ret;
-   }
 
/* Loop 1: convert all events */
for (i = 0; i  npevs; i++) {
-   pkgs[i].pev = pevs[i];
/* Init kprobe blacklist if needed */
-   if (!pkgs[i].pev-uprobes)
+   if (pevs[i].uprobes)
kprobe_blacklist__init();
/* Convert with or without debuginfo */
-   ret  = convert_to_probe_trace_events(pkgs[i].pev,
-pkgs[i].tevs);
-   if (ret  0)
+   ret  = convert_to_probe_trace_events(pevs[i], pevs[i].tevs);
+   if (ret  0) {
+   cleanup = true;
goto end;
-   pkgs[i].ntevs = ret;
+   }
+   pevs[i].ntevs = ret;
}
/* This just release blacklist only if allocated */
kprobe_blacklist__release();
 
/* Loop 2: add all events */
for (i = 0; i  npevs; i++) {
-   ret = __add_probe_trace_events(pkgs[i].pev, pkgs[i].tevs,
-  pkgs[i].ntevs,
+   ret = __add_probe_trace_events(pevs[i], pevs[i].tevs,
+  pevs[i].ntevs,
   probe_conf.force_add);
if (ret  0)
break;
}
 end:
/* Loop 3: cleanup and free trace events  */
-   for (i = 0; i  npevs; i++) {
-   for (j = 0; j  pkgs[i].ntevs; j++)
-   clear_probe_trace_event(pkgs[i].tevs[j]);
-   zfree(pkgs[i].tevs);
-   }
-   free(pkgs);
+   for (i = 0; cleanup  (i  npevs); i++)
+   cleanup_perf_probe_event(pevs[i]);
exit_symbol_maps();
 
return ret;
diff --git