Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
B
bcc
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
bcc
Commits
4a2f2214
Commit
4a2f2214
authored
Dec 18, 2017
by
yonghong-song
Committed by
GitHub
Dec 18, 2017
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #1480 from htbegin/bucket_sort_fn
histogram: support sorting bucket before dumping table
parents
e1a743e6
33a5a4bd
Changes
2
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
63 additions
and
9 deletions
+63
-9
src/python/bcc/table.py
src/python/bcc/table.py
+30
-9
tests/python/test_histogram.py
tests/python/test_histogram.py
+33
-0
No files found.
src/python/bcc/table.py
View file @
4a2f2214
...
@@ -275,9 +275,11 @@ class TableBase(MutableMapping):
...
@@ -275,9 +275,11 @@ class TableBase(MutableMapping):
return
next_key
return
next_key
def
print_log2_hist
(
self
,
val_type
=
"value"
,
section_header
=
"Bucket ptr"
,
def
print_log2_hist
(
self
,
val_type
=
"value"
,
section_header
=
"Bucket ptr"
,
section_print_fn
=
None
,
bucket_fn
=
None
,
strip_leading_zero
=
None
):
section_print_fn
=
None
,
bucket_fn
=
None
,
strip_leading_zero
=
None
,
bucket_sort_fn
=
None
):
"""print_log2_hist(val_type="value", section_header="Bucket ptr",
"""print_log2_hist(val_type="value", section_header="Bucket ptr",
section_print_fn=None, bucket_fn=None)
section_print_fn=None, bucket_fn=None,
strip_leading_zero=None, bucket_sort_fn=None):
Prints a table as a log2 histogram. The table must be stored as
Prints a table as a log2 histogram. The table must be stored as
log2. The val_type argument is optional, and is a column header.
log2. The val_type argument is optional, and is a column header.
...
@@ -287,9 +289,12 @@ class TableBase(MutableMapping):
...
@@ -287,9 +289,12 @@ class TableBase(MutableMapping):
to format into a string as it sees fit. If bucket_fn is not None,
to format into a string as it sees fit. If bucket_fn is not None,
it will be used to produce a bucket value for the histogram keys.
it will be used to produce a bucket value for the histogram keys.
If the value of strip_leading_zero is not False, prints a histogram
If the value of strip_leading_zero is not False, prints a histogram
that is omitted leading zeros from the beginning. The maximum index
that is omitted leading zeros from the beginning.
allowed is log2_index_max (65), which will accommodate any 64-bit
If bucket_sort_fn is not None, it will be used to sort the buckets
integer in the histogram.
before iterating them, and it is useful when there are multiple fields
in the secondary key.
The maximum index allowed is log2_index_max (65), which will
accommodate any 64-bit integer in the histogram.
"""
"""
if
isinstance
(
self
.
Key
(),
ct
.
Structure
):
if
isinstance
(
self
.
Key
(),
ct
.
Structure
):
tmp
=
{}
tmp
=
{}
...
@@ -302,7 +307,13 @@ class TableBase(MutableMapping):
...
@@ -302,7 +307,13 @@ class TableBase(MutableMapping):
vals
=
tmp
[
bucket
]
=
tmp
.
get
(
bucket
,
[
0
]
*
log2_index_max
)
vals
=
tmp
[
bucket
]
=
tmp
.
get
(
bucket
,
[
0
]
*
log2_index_max
)
slot
=
getattr
(
k
,
f2
)
slot
=
getattr
(
k
,
f2
)
vals
[
slot
]
=
v
.
value
vals
[
slot
]
=
v
.
value
for
bucket
,
vals
in
tmp
.
items
():
buckets
=
tmp
.
keys
()
if
bucket_sort_fn
:
buckets
=
bucket_sort_fn
(
buckets
)
for
bucket
in
buckets
:
vals
=
tmp
[
bucket
]
if
section_print_fn
:
if
section_print_fn
:
print
(
"
\
n
%s = %s"
%
(
section_header
,
print
(
"
\
n
%s = %s"
%
(
section_header
,
section_print_fn
(
bucket
)))
section_print_fn
(
bucket
)))
...
@@ -316,9 +327,10 @@ class TableBase(MutableMapping):
...
@@ -316,9 +327,10 @@ class TableBase(MutableMapping):
_print_log2_hist
(
vals
,
val_type
,
strip_leading_zero
)
_print_log2_hist
(
vals
,
val_type
,
strip_leading_zero
)
def
print_linear_hist
(
self
,
val_type
=
"value"
,
section_header
=
"Bucket ptr"
,
def
print_linear_hist
(
self
,
val_type
=
"value"
,
section_header
=
"Bucket ptr"
,
section_print_fn
=
None
,
bucket_fn
=
None
):
section_print_fn
=
None
,
bucket_fn
=
None
,
bucket_sort_fn
=
None
):
"""print_linear_hist(val_type="value", section_header="Bucket ptr",
"""print_linear_hist(val_type="value", section_header="Bucket ptr",
section_print_fn=None, bucket_fn=None)
section_print_fn=None, bucket_fn=None,
bucket_sort_fn=None)
Prints a table as a linear histogram. This is intended to span integer
Prints a table as a linear histogram. This is intended to span integer
ranges, eg, from 0 to 100. The val_type argument is optional, and is a
ranges, eg, from 0 to 100. The val_type argument is optional, and is a
...
@@ -327,6 +339,9 @@ class TableBase(MutableMapping):
...
@@ -327,6 +339,9 @@ class TableBase(MutableMapping):
each. If section_print_fn is not None, it will be passed the bucket
each. If section_print_fn is not None, it will be passed the bucket
value to format into a string as it sees fit. If bucket_fn is not None,
value to format into a string as it sees fit. If bucket_fn is not None,
it will be used to produce a bucket value for the histogram keys.
it will be used to produce a bucket value for the histogram keys.
If bucket_sort_fn is not None, it will be used to sort the buckets
before iterating them, and it is useful when there are multiple fields
in the secondary key.
The maximum index allowed is linear_index_max (1025), which is hoped
The maximum index allowed is linear_index_max (1025), which is hoped
to be sufficient for integer ranges spanned.
to be sufficient for integer ranges spanned.
"""
"""
...
@@ -341,7 +356,13 @@ class TableBase(MutableMapping):
...
@@ -341,7 +356,13 @@ class TableBase(MutableMapping):
vals
=
tmp
[
bucket
]
=
tmp
.
get
(
bucket
,
[
0
]
*
linear_index_max
)
vals
=
tmp
[
bucket
]
=
tmp
.
get
(
bucket
,
[
0
]
*
linear_index_max
)
slot
=
getattr
(
k
,
f2
)
slot
=
getattr
(
k
,
f2
)
vals
[
slot
]
=
v
.
value
vals
[
slot
]
=
v
.
value
for
bucket
,
vals
in
tmp
.
items
():
buckets
=
tmp
.
keys
()
if
bucket_sort_fn
:
buckets
=
bucket_sort_fn
(
buckets
)
for
bucket
in
buckets
:
vals
=
tmp
[
bucket
]
if
section_print_fn
:
if
section_print_fn
:
print
(
"
\
n
%s = %s"
%
(
section_header
,
print
(
"
\
n
%s = %s"
%
(
section_header
,
section_print_fn
(
bucket
)))
section_print_fn
(
bucket
)))
...
...
tests/python/test_histogram.py
View file @
4a2f2214
...
@@ -72,6 +72,39 @@ int kprobe__finish_task_switch(struct pt_regs *ctx, struct task_struct *prev) {
...
@@ -72,6 +72,39 @@ int kprobe__finish_task_switch(struct pt_regs *ctx, struct task_struct *prev) {
b
[
"hist1"
].
print_log2_hist
()
b
[
"hist1"
].
print_log2_hist
()
b
.
cleanup
()
b
.
cleanup
()
def
test_multiple_key
(
self
):
b
=
BPF
(
text
=
"""
#include <uapi/linux/ptrace.h>
#include <uapi/linux/fs.h>
struct hist_s_key {
u64 key_1;
u64 key_2;
};
struct hist_key {
struct hist_s_key s_key;
u64 slot;
};
BPF_HISTOGRAM(mk_hist, struct hist_key, 1024);
int kprobe__vfs_read(struct pt_regs *ctx, struct file *file,
char __user *buf, size_t count) {
struct hist_key key = {.slot = bpf_log2l(count)};
key.s_key.key_1 = (unsigned long)buf & 0x70;
key.s_key.key_2 = (unsigned long)buf & 0x7;
mk_hist.increment(key);
return 0;
}
"""
)
def
bucket_sort
(
buckets
):
buckets
.
sort
()
return
buckets
for
i
in
range
(
0
,
100
):
time
.
sleep
(
0.01
)
b
[
"mk_hist"
].
print_log2_hist
(
"size"
,
"k_1 & k_2"
,
section_print_fn
=
lambda
bucket
:
"%3d %d"
%
(
bucket
[
0
],
bucket
[
1
]),
bucket_fn
=
lambda
bucket
:
(
bucket
.
key_1
,
bucket
.
key_2
),
strip_leading_zero
=
True
,
bucket_sort_fn
=
bucket_sort
)
b
.
cleanup
()
if
__name__
==
"__main__"
:
if
__name__
==
"__main__"
:
main
()
main
()
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment