|
@@ -25,6 +25,133 @@
|
|
|
|
|
|
__thread struct callchain_cursor callchain_cursor;
|
|
__thread struct callchain_cursor callchain_cursor;
|
|
|
|
|
|
|
|
+#ifdef HAVE_DWARF_UNWIND_SUPPORT
|
|
|
|
+static int get_stack_size(const char *str, unsigned long *_size)
|
|
|
|
+{
|
|
|
|
+ char *endptr;
|
|
|
|
+ unsigned long size;
|
|
|
|
+ unsigned long max_size = round_down(USHRT_MAX, sizeof(u64));
|
|
|
|
+
|
|
|
|
+ size = strtoul(str, &endptr, 0);
|
|
|
|
+
|
|
|
|
+ do {
|
|
|
|
+ if (*endptr)
|
|
|
|
+ break;
|
|
|
|
+
|
|
|
|
+ size = round_up(size, sizeof(u64));
|
|
|
|
+ if (!size || size > max_size)
|
|
|
|
+ break;
|
|
|
|
+
|
|
|
|
+ *_size = size;
|
|
|
|
+ return 0;
|
|
|
|
+
|
|
|
|
+ } while (0);
|
|
|
|
+
|
|
|
|
+ pr_err("callchain: Incorrect stack dump size (max %ld): %s\n",
|
|
|
|
+ max_size, str);
|
|
|
|
+ return -1;
|
|
|
|
+}
|
|
|
|
+#endif /* HAVE_DWARF_UNWIND_SUPPORT */
|
|
|
|
+
|
|
|
|
+int parse_callchain_record_opt(const char *arg)
|
|
|
|
+{
|
|
|
|
+ char *tok, *name, *saveptr = NULL;
|
|
|
|
+ char *buf;
|
|
|
|
+ int ret = -1;
|
|
|
|
+
|
|
|
|
+ /* We need buffer that we know we can write to. */
|
|
|
|
+ buf = malloc(strlen(arg) + 1);
|
|
|
|
+ if (!buf)
|
|
|
|
+ return -ENOMEM;
|
|
|
|
+
|
|
|
|
+ strcpy(buf, arg);
|
|
|
|
+
|
|
|
|
+ tok = strtok_r((char *)buf, ",", &saveptr);
|
|
|
|
+ name = tok ? : (char *)buf;
|
|
|
|
+
|
|
|
|
+ do {
|
|
|
|
+ /* Framepointer style */
|
|
|
|
+ if (!strncmp(name, "fp", sizeof("fp"))) {
|
|
|
|
+ if (!strtok_r(NULL, ",", &saveptr)) {
|
|
|
|
+ callchain_param.record_mode = CALLCHAIN_FP;
|
|
|
|
+ ret = 0;
|
|
|
|
+ } else
|
|
|
|
+ pr_err("callchain: No more arguments "
|
|
|
|
+ "needed for -g fp\n");
|
|
|
|
+ break;
|
|
|
|
+
|
|
|
|
+#ifdef HAVE_DWARF_UNWIND_SUPPORT
|
|
|
|
+ /* Dwarf style */
|
|
|
|
+ } else if (!strncmp(name, "dwarf", sizeof("dwarf"))) {
|
|
|
|
+ const unsigned long default_stack_dump_size = 8192;
|
|
|
|
+
|
|
|
|
+ ret = 0;
|
|
|
|
+ callchain_param.record_mode = CALLCHAIN_DWARF;
|
|
|
|
+ callchain_param.dump_size = default_stack_dump_size;
|
|
|
|
+
|
|
|
|
+ tok = strtok_r(NULL, ",", &saveptr);
|
|
|
|
+ if (tok) {
|
|
|
|
+ unsigned long size = 0;
|
|
|
|
+
|
|
|
|
+ ret = get_stack_size(tok, &size);
|
|
|
|
+ callchain_param.dump_size = size;
|
|
|
|
+ }
|
|
|
|
+#endif /* HAVE_DWARF_UNWIND_SUPPORT */
|
|
|
|
+ } else {
|
|
|
|
+ pr_err("callchain: Unknown --call-graph option "
|
|
|
|
+ "value: %s\n", arg);
|
|
|
|
+ break;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ } while (0);
|
|
|
|
+
|
|
|
|
+ free(buf);
|
|
|
|
+ return ret;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static int parse_callchain_mode(const char *value)
|
|
|
|
+{
|
|
|
|
+ if (!strncmp(value, "graph", strlen(value))) {
|
|
|
|
+ callchain_param.mode = CHAIN_GRAPH_ABS;
|
|
|
|
+ return 0;
|
|
|
|
+ }
|
|
|
|
+ if (!strncmp(value, "flat", strlen(value))) {
|
|
|
|
+ callchain_param.mode = CHAIN_FLAT;
|
|
|
|
+ return 0;
|
|
|
|
+ }
|
|
|
|
+ if (!strncmp(value, "fractal", strlen(value))) {
|
|
|
|
+ callchain_param.mode = CHAIN_GRAPH_REL;
|
|
|
|
+ return 0;
|
|
|
|
+ }
|
|
|
|
+ return -1;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static int parse_callchain_order(const char *value)
|
|
|
|
+{
|
|
|
|
+ if (!strncmp(value, "caller", strlen(value))) {
|
|
|
|
+ callchain_param.order = ORDER_CALLER;
|
|
|
|
+ return 0;
|
|
|
|
+ }
|
|
|
|
+ if (!strncmp(value, "callee", strlen(value))) {
|
|
|
|
+ callchain_param.order = ORDER_CALLEE;
|
|
|
|
+ return 0;
|
|
|
|
+ }
|
|
|
|
+ return -1;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static int parse_callchain_sort_key(const char *value)
|
|
|
|
+{
|
|
|
|
+ if (!strncmp(value, "function", strlen(value))) {
|
|
|
|
+ callchain_param.key = CCKEY_FUNCTION;
|
|
|
|
+ return 0;
|
|
|
|
+ }
|
|
|
|
+ if (!strncmp(value, "address", strlen(value))) {
|
|
|
|
+ callchain_param.key = CCKEY_ADDRESS;
|
|
|
|
+ return 0;
|
|
|
|
+ }
|
|
|
|
+ return -1;
|
|
|
|
+}
|
|
|
|
+
|
|
int
|
|
int
|
|
parse_callchain_report_opt(const char *arg)
|
|
parse_callchain_report_opt(const char *arg)
|
|
{
|
|
{
|
|
@@ -44,25 +171,12 @@ parse_callchain_report_opt(const char *arg)
|
|
return 0;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
- /* try to get the output mode */
|
|
|
|
- if (!strncmp(tok, "graph", strlen(tok)))
|
|
|
|
- callchain_param.mode = CHAIN_GRAPH_ABS;
|
|
|
|
- else if (!strncmp(tok, "flat", strlen(tok)))
|
|
|
|
- callchain_param.mode = CHAIN_FLAT;
|
|
|
|
- else if (!strncmp(tok, "fractal", strlen(tok)))
|
|
|
|
- callchain_param.mode = CHAIN_GRAPH_REL;
|
|
|
|
- /* try to get the call chain order */
|
|
|
|
- else if (!strncmp(tok, "caller", strlen(tok)))
|
|
|
|
- callchain_param.order = ORDER_CALLER;
|
|
|
|
- else if (!strncmp(tok, "callee", strlen(tok)))
|
|
|
|
- callchain_param.order = ORDER_CALLEE;
|
|
|
|
- /* try to get the sort key */
|
|
|
|
- else if (!strncmp(tok, "function", strlen(tok)))
|
|
|
|
- callchain_param.key = CCKEY_FUNCTION;
|
|
|
|
- else if (!strncmp(tok, "address", strlen(tok)))
|
|
|
|
- callchain_param.key = CCKEY_ADDRESS;
|
|
|
|
- /* try to get the min percent */
|
|
|
|
- else if (!minpcnt_set) {
|
|
|
|
|
|
+ if (!parse_callchain_mode(tok) ||
|
|
|
|
+ !parse_callchain_order(tok) ||
|
|
|
|
+ !parse_callchain_sort_key(tok)) {
|
|
|
|
+ /* parsing ok - move on to the next */
|
|
|
|
+ } else if (!minpcnt_set) {
|
|
|
|
+ /* try to get the min percent */
|
|
callchain_param.min_percent = strtod(tok, &endptr);
|
|
callchain_param.min_percent = strtod(tok, &endptr);
|
|
if (tok == endptr)
|
|
if (tok == endptr)
|
|
return -1;
|
|
return -1;
|
|
@@ -84,6 +198,47 @@ parse_callchain_report_opt(const char *arg)
|
|
return 0;
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+int perf_callchain_config(const char *var, const char *value)
|
|
|
|
+{
|
|
|
|
+ char *endptr;
|
|
|
|
+
|
|
|
|
+ if (prefixcmp(var, "call-graph."))
|
|
|
|
+ return 0;
|
|
|
|
+ var += sizeof("call-graph.") - 1;
|
|
|
|
+
|
|
|
|
+ if (!strcmp(var, "record-mode"))
|
|
|
|
+ return parse_callchain_record_opt(value);
|
|
|
|
+#ifdef HAVE_DWARF_UNWIND_SUPPORT
|
|
|
|
+ if (!strcmp(var, "dump-size")) {
|
|
|
|
+ unsigned long size = 0;
|
|
|
|
+ int ret;
|
|
|
|
+
|
|
|
|
+ ret = get_stack_size(value, &size);
|
|
|
|
+ callchain_param.dump_size = size;
|
|
|
|
+
|
|
|
|
+ return ret;
|
|
|
|
+ }
|
|
|
|
+#endif
|
|
|
|
+ if (!strcmp(var, "print-type"))
|
|
|
|
+ return parse_callchain_mode(value);
|
|
|
|
+ if (!strcmp(var, "order"))
|
|
|
|
+ return parse_callchain_order(value);
|
|
|
|
+ if (!strcmp(var, "sort-key"))
|
|
|
|
+ return parse_callchain_sort_key(value);
|
|
|
|
+ if (!strcmp(var, "threshold")) {
|
|
|
|
+ callchain_param.min_percent = strtod(value, &endptr);
|
|
|
|
+ if (value == endptr)
|
|
|
|
+ return -1;
|
|
|
|
+ }
|
|
|
|
+ if (!strcmp(var, "print-limit")) {
|
|
|
|
+ callchain_param.print_limit = strtod(value, &endptr);
|
|
|
|
+ if (value == endptr)
|
|
|
|
+ return -1;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return 0;
|
|
|
|
+}
|
|
|
|
+
|
|
static void
|
|
static void
|
|
rb_insert_callchain(struct rb_root *root, struct callchain_node *chain,
|
|
rb_insert_callchain(struct rb_root *root, struct callchain_node *chain,
|
|
enum chain_mode mode)
|
|
enum chain_mode mode)
|