Commit 4a2f2214 authored by yonghong-song's avatar yonghong-song Committed by GitHub

Merge pull request #1480 from htbegin/bucket_sort_fn

histogram: support sorting bucket before dumping table
parents e1a743e6 33a5a4bd
...@@ -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)))
......
...@@ -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()
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