From 328bcba4f01ed4b7a0381b72c858f10e6cb56c2e Mon Sep 17 00:00:00 2001 From: Jesper Dangaard Brouer Date: Tue, 6 Aug 2024 14:35:39 +0200 Subject: [PATCH] cgroup-rstat-flushing: also encode yield in lock_key_t This allow us to remove the other maps. What a nice cleanup :-) Signed-off-by: Jesper Dangaard Brouer --- examples/cgroup-rstat-flushing.bpf.c | 62 ++++++++-------------------- examples/cgroup-rstat-flushing.yaml | 8 +++- 2 files changed, 24 insertions(+), 46 deletions(-) diff --git a/examples/cgroup-rstat-flushing.bpf.c b/examples/cgroup-rstat-flushing.bpf.c index f026fcd6..12a4ea67 100644 --- a/examples/cgroup-rstat-flushing.bpf.c +++ b/examples/cgroup-rstat-flushing.bpf.c @@ -22,45 +22,31 @@ struct { /* Complex key for encoding lock properties */ struct lock_key_t { - u8 contended; - u8 level; + u8 contended; + u8 yield; + u16 level; }; -#define MAX_LOCK_KEY_ENTRIES 64 +#define MAX_LOCK_KEY_ENTRIES 128 -/* Total counter for obtaining lock together with contended state +/* Total counter for obtaining lock together with state (prometheus labels) * - * This counter also contains "yield" case. To determine "normal" lock - * case subtract "yield" counter in prometheus query. + * State for cgroup level, contended and yield case. + * + * The problematic/interesting case is when the lock was contended (prior to + * obtaining the lock). This "contended" label is key in analyzing locking + * issues. + * + * Kernel can yield the rstat lock when walking individial CPU stats. This + * leads to "interresting" concurrency issues. Thus, having a label "yield" + * can help diagnose. */ struct { __uint(type, BPF_MAP_TYPE_PERCPU_HASH); __uint(max_entries, MAX_LOCK_KEY_ENTRIES); __type(key, struct lock_key_t); - //__type(key, u16); __type(value, u64); } cgroup_rstat_locked_total SEC(".maps"); -/* Counter for obtaining lock again after yield (and contended state). - * - * Kernel can yield the rstat lock when walking individial CPU stats. - * This leads to "interresting" concurrency issues. Thus, keep - * seperate counter for "yield" cases to help diagnose. - */ -struct { - __uint(type, BPF_MAP_TYPE_PERCPU_ARRAY); - __uint(max_entries, 2); /* contended state used as key */ - __type(key, u32); - __type(value, u64); -} cgroup_rstat_locked_yield SEC(".maps"); - -/* Counter for lock contended case, recorded per cgroup level */ -struct { - __uint(type, BPF_MAP_TYPE_PERCPU_ARRAY); - __uint(max_entries, MAX_CGROUP_LEVELS + 1); - __type(key, u32); - __type(value, u64); -} cgroup_rstat_lock_contended SEC(".maps"); - /** Measurement#1: lock rates * ========================= For locks, the problematic/interesting case is when the lock was contended. @@ -82,29 +68,17 @@ int BPF_PROG(rstat_locked, struct cgroup *cgrp, int cpu, bool contended) { struct lock_key_t lock_key = { 0 }; u32 level = cgrp->level; - u32 key; if (level > MAX_CGROUP_LEVELS) level = MAX_CGROUP_LEVELS; - lock_key.contended = contended; - lock_key.level = (level & 0xF); - increment_map_nosync(&cgroup_rstat_locked_total, &lock_key, 1); - - key = contended; if (cpu >= 0) - increment_map_nosync(&cgroup_rstat_locked_yield, &key, 1); - - /* What cgroup level is interesting, but I didn't manage to encode it - * in above counters. As contended case is the most interesting, have - * level counter for contended. - */ - if (contended) { - u32 lvl = cgrp->level; + lock_key.yield = 1; - increment_map_nosync(&cgroup_rstat_lock_contended, &lvl, 1); - } + lock_key.contended = contended; + lock_key.level = (level & 0xFF); + increment_map_nosync(&cgroup_rstat_locked_total, &lock_key, 1); return 0; } diff --git a/examples/cgroup-rstat-flushing.yaml b/examples/cgroup-rstat-flushing.yaml index 379e5cd7..a357c3c4 100644 --- a/examples/cgroup-rstat-flushing.yaml +++ b/examples/cgroup-rstat-flushing.yaml @@ -8,14 +8,18 @@ metrics: decoders: - name: uint - name: cgroup_rstat_locked_total - help: Times rstat lock was obtainted plus contended state and cgroup level + help: Times rstat lock was obtainted with state for cgroup level, contended and yield labels: - name: contended size: 1 decoders: # contended boolean converted to 0 and 1 - name: uint - - name: level + - name: yield size: 1 + decoders: # was this a yielded lock case + - name: uint + - name: level + size: 2 decoders: - name: uint - name: cgroup_rstat_locked_yield