Commit 2d713b80 authored by Namhyung Kim's avatar Namhyung Kim Committed by Arnaldo Carvalho de Melo

perf callchain: Add enum match_result for match_chain()

The append_chain() might return either result of match_chain() or other
(error) code.  But match_chain() can return any value in s64 type so
it's hard to check the error case.  Add new enum match_result and make
match_chain() return non-negative values only so that we can check the
error cases.
Signed-off-by: default avatarNamhyung Kim <namhyung@kernel.org>
Acked-by: default avatarJiri Olsa <jolsa@kernel.org>
Cc: Andi Kleen <andi@firstfloor.org>
Cc: David Ahern <dsahern@gmail.com>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Stephane Eranian <eranian@google.com>
Cc: Wang Nan <wangnan0@huawei.com>
Link: http://lkml.kernel.org/r/1455631723-17345-5-git-send-email-namhyung@kernel.orgSigned-off-by: default avatarArnaldo Carvalho de Melo <acme@redhat.com>
parent 8451cbb9
...@@ -475,16 +475,32 @@ add_child(struct callchain_node *parent, ...@@ -475,16 +475,32 @@ add_child(struct callchain_node *parent,
return new; return new;
} }
static s64 match_chain(struct callchain_cursor_node *node, enum match_result {
struct callchain_list *cnode) MATCH_ERROR = -1,
MATCH_EQ,
MATCH_LT,
MATCH_GT,
};
static enum match_result match_chain(struct callchain_cursor_node *node,
struct callchain_list *cnode)
{ {
struct symbol *sym = node->sym; struct symbol *sym = node->sym;
u64 left, right;
if (cnode->ms.sym && sym && if (cnode->ms.sym && sym &&
callchain_param.key == CCKEY_FUNCTION) callchain_param.key == CCKEY_FUNCTION) {
return cnode->ms.sym->start - sym->start; left = cnode->ms.sym->start;
else right = sym->start;
return cnode->ip - node->ip; } else {
left = cnode->ip;
right = node->ip;
}
if (left == right)
return MATCH_EQ;
return left > right ? MATCH_GT : MATCH_LT;
} }
/* /*
...@@ -549,7 +565,7 @@ split_add_child(struct callchain_node *parent, ...@@ -549,7 +565,7 @@ split_add_child(struct callchain_node *parent,
cnode = list_first_entry(&first->val, struct callchain_list, cnode = list_first_entry(&first->val, struct callchain_list,
list); list);
if (match_chain(node, cnode) < 0) if (match_chain(node, cnode) == MATCH_LT)
pp = &p->rb_left; pp = &p->rb_left;
else else
pp = &p->rb_right; pp = &p->rb_right;
...@@ -562,7 +578,7 @@ split_add_child(struct callchain_node *parent, ...@@ -562,7 +578,7 @@ split_add_child(struct callchain_node *parent,
} }
} }
static int static enum match_result
append_chain(struct callchain_node *root, append_chain(struct callchain_node *root,
struct callchain_cursor *cursor, struct callchain_cursor *cursor,
u64 period); u64 period);
...@@ -583,17 +599,17 @@ append_chain_children(struct callchain_node *root, ...@@ -583,17 +599,17 @@ append_chain_children(struct callchain_node *root,
/* lookup in childrens */ /* lookup in childrens */
while (*p) { while (*p) {
s64 ret; enum match_result ret;
parent = *p; parent = *p;
rnode = rb_entry(parent, struct callchain_node, rb_node_in); rnode = rb_entry(parent, struct callchain_node, rb_node_in);
/* If at least first entry matches, rely to children */ /* If at least first entry matches, rely to children */
ret = append_chain(rnode, cursor, period); ret = append_chain(rnode, cursor, period);
if (ret == 0) if (ret == MATCH_EQ)
goto inc_children_hit; goto inc_children_hit;
if (ret < 0) if (ret == MATCH_LT)
p = &parent->rb_left; p = &parent->rb_left;
else else
p = &parent->rb_right; p = &parent->rb_right;
...@@ -611,7 +627,7 @@ append_chain_children(struct callchain_node *root, ...@@ -611,7 +627,7 @@ append_chain_children(struct callchain_node *root,
root->children_count++; root->children_count++;
} }
static int static enum match_result
append_chain(struct callchain_node *root, append_chain(struct callchain_node *root,
struct callchain_cursor *cursor, struct callchain_cursor *cursor,
u64 period) u64 period)
...@@ -620,7 +636,7 @@ append_chain(struct callchain_node *root, ...@@ -620,7 +636,7 @@ append_chain(struct callchain_node *root,
u64 start = cursor->pos; u64 start = cursor->pos;
bool found = false; bool found = false;
u64 matches; u64 matches;
int cmp = 0; enum match_result cmp = MATCH_ERROR;
/* /*
* Lookup in the current node * Lookup in the current node
...@@ -636,7 +652,7 @@ append_chain(struct callchain_node *root, ...@@ -636,7 +652,7 @@ append_chain(struct callchain_node *root,
break; break;
cmp = match_chain(node, cnode); cmp = match_chain(node, cnode);
if (cmp) if (cmp != MATCH_EQ)
break; break;
found = true; found = true;
...@@ -646,7 +662,7 @@ append_chain(struct callchain_node *root, ...@@ -646,7 +662,7 @@ append_chain(struct callchain_node *root,
/* matches not, relay no the parent */ /* matches not, relay no the parent */
if (!found) { if (!found) {
WARN_ONCE(!cmp, "Chain comparison error\n"); WARN_ONCE(cmp == MATCH_ERROR, "Chain comparison error\n");
return cmp; return cmp;
} }
...@@ -655,20 +671,20 @@ append_chain(struct callchain_node *root, ...@@ -655,20 +671,20 @@ append_chain(struct callchain_node *root,
/* we match only a part of the node. Split it and add the new chain */ /* we match only a part of the node. Split it and add the new chain */
if (matches < root->val_nr) { if (matches < root->val_nr) {
split_add_child(root, cursor, cnode, start, matches, period); split_add_child(root, cursor, cnode, start, matches, period);
return 0; return MATCH_EQ;
} }
/* we match 100% of the path, increment the hit */ /* we match 100% of the path, increment the hit */
if (matches == root->val_nr && cursor->pos == cursor->nr) { if (matches == root->val_nr && cursor->pos == cursor->nr) {
root->hit += period; root->hit += period;
root->count++; root->count++;
return 0; return MATCH_EQ;
} }
/* We match the node and still have a part remaining */ /* We match the node and still have a part remaining */
append_chain_children(root, cursor, period); append_chain_children(root, cursor, period);
return 0; return MATCH_EQ;
} }
int callchain_append(struct callchain_root *root, int callchain_append(struct callchain_root *root,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment