Commit bf8f8b20 authored by Daniel Hill's avatar Daniel Hill Committed by Kent Overstreet

bcachefs: time stats now uses the mean_and_variance module.

Signed-off-by: default avatarDaniel Hill <daniel@gluo.nz>
Signed-off-by: default avatarKent Overstreet <kent.overstreet@linux.dev>
parent 92095781
...@@ -22,6 +22,7 @@ config BCACHEFS_FS ...@@ -22,6 +22,7 @@ config BCACHEFS_FS
select XXHASH select XXHASH
select SRCU select SRCU
select SYMBOLIC_ERRNAME select SYMBOLIC_ERRNAME
select MEAN_AND_VARIANCE
help help
The bcachefs filesystem - a modern, copy on write filesystem, with The bcachefs filesystem - a modern, copy on write filesystem, with
support for multiple devices, compression, checksumming, etc. support for multiple devices, compression, checksumming, etc.
......
...@@ -893,6 +893,12 @@ int bch2_fs_start(struct bch_fs *c) ...@@ -893,6 +893,12 @@ int bch2_fs_start(struct bch_fs *c)
bch2_dev_allocator_add(c, ca); bch2_dev_allocator_add(c, ca);
bch2_recalc_capacity(c); bch2_recalc_capacity(c);
for (i = 0; i < BCH_TRANSACTIONS_NR; i++) {
mutex_lock(&c->btree_transaction_stats[i].lock);
bch2_time_stats_init(&c->btree_transaction_stats[i].lock_hold_times);
mutex_unlock(&c->btree_transaction_stats[i].lock);
}
ret = BCH_SB_INITIALIZED(c->disk_sb.sb) ret = BCH_SB_INITIALIZED(c->disk_sb.sb)
? bch2_fs_recovery(c) ? bch2_fs_recovery(c)
: bch2_fs_initialize(c); : bch2_fs_initialize(c);
......
...@@ -24,6 +24,7 @@ ...@@ -24,6 +24,7 @@
#include <linux/sched/clock.h> #include <linux/sched/clock.h>
#include "eytzinger.h" #include "eytzinger.h"
#include "mean_and_variance.h"
#include "util.h" #include "util.h"
static const char si_units[] = "?kMGTPEZY"; static const char si_units[] = "?kMGTPEZY";
...@@ -323,38 +324,39 @@ static void bch2_time_stats_update_one(struct bch2_time_stats *stats, ...@@ -323,38 +324,39 @@ static void bch2_time_stats_update_one(struct bch2_time_stats *stats,
{ {
u64 duration, freq; u64 duration, freq;
duration = time_after64(end, start) if (time_after64(end, start)) {
? end - start : 0; duration = end - start;
freq = time_after64(end, stats->last_event) stats->duration_stats = mean_and_variance_update(stats->duration_stats, duration);
? end - stats->last_event : 0; mean_and_variance_weighted_update(&stats->duration_stats_weighted, duration);
stats->max_duration = max(stats->max_duration, duration);
stats->count++; stats->min_duration = min(stats->min_duration, duration);
bch2_quantiles_update(&stats->quantiles, duration);
stats->average_duration = stats->average_duration }
? ewma_add(stats->average_duration, duration, 6)
: duration;
stats->average_frequency = stats->average_frequency
? ewma_add(stats->average_frequency, freq, 6)
: freq;
stats->max_duration = max(stats->max_duration, duration);
stats->last_event = end;
bch2_quantiles_update(&stats->quantiles, duration); if (time_after64(end, stats->last_event)) {
freq = end - stats->last_event;
stats->freq_stats = mean_and_variance_update(stats->freq_stats, freq);
mean_and_variance_weighted_update(&stats->freq_stats_weighted, freq);
stats->max_freq = max(stats->max_freq, freq);
stats->min_freq = min(stats->min_freq, freq);
stats->last_event = end;
}
} }
void __bch2_time_stats_update(struct bch2_time_stats *stats, u64 start, u64 end) void __bch2_time_stats_update(struct bch2_time_stats *stats, u64 start, u64 end)
{ {
unsigned long flags; unsigned long flags;
WARN_RATELIMIT(!stats->min_duration || !stats->min_freq,
"time_stats: min_duration = %llu, min_freq = %llu",
stats->min_duration, stats->min_freq);
if (!stats->buffer) { if (!stats->buffer) {
spin_lock_irqsave(&stats->lock, flags); spin_lock_irqsave(&stats->lock, flags);
bch2_time_stats_update_one(stats, start, end); bch2_time_stats_update_one(stats, start, end);
if (stats->average_frequency < 32 && if (mean_and_variance_weighted_get_mean(stats->freq_stats_weighted) < 32 &&
stats->count > 1024) stats->duration_stats.n > 1024)
stats->buffer = stats->buffer =
alloc_percpu_gfp(struct bch2_time_stat_buffer, alloc_percpu_gfp(struct bch2_time_stat_buffer,
GFP_ATOMIC); GFP_ATOMIC);
...@@ -390,12 +392,15 @@ void __bch2_time_stats_update(struct bch2_time_stats *stats, u64 start, u64 end) ...@@ -390,12 +392,15 @@ void __bch2_time_stats_update(struct bch2_time_stats *stats, u64 start, u64 end)
static const struct time_unit { static const struct time_unit {
const char *name; const char *name;
u32 nsecs; u64 nsecs;
} time_units[] = { } time_units[] = {
{ "ns", 1 }, { "ns", 1 },
{ "us", NSEC_PER_USEC }, { "us", NSEC_PER_USEC },
{ "ms", NSEC_PER_MSEC }, { "ms", NSEC_PER_MSEC },
{ "sec", NSEC_PER_SEC }, { "s", NSEC_PER_SEC },
{ "m", NSEC_PER_SEC * 60},
{ "h", NSEC_PER_SEC * 3600},
{ "eon", U64_MAX },
}; };
static const struct time_unit *pick_time_units(u64 ns) static const struct time_unit *pick_time_units(u64 ns)
...@@ -418,35 +423,121 @@ void bch2_pr_time_units(struct printbuf *out, u64 ns) ...@@ -418,35 +423,121 @@ void bch2_pr_time_units(struct printbuf *out, u64 ns)
prt_printf(out, "%llu %s", div_u64(ns, u->nsecs), u->name); prt_printf(out, "%llu %s", div_u64(ns, u->nsecs), u->name);
} }
static void bch2_pr_time_units_aligned(struct printbuf *out, u64 ns)
{
const struct time_unit *u = pick_time_units(ns);
prt_printf(out, "%llu ", div64_u64(ns, u->nsecs));
prt_tab_rjust(out);
prt_printf(out, "%s", u->name);
}
#define TABSTOP_SIZE 12
static inline void pr_name_and_units(struct printbuf *out, const char *name, u64 ns)
{
prt_printf(out, name);
prt_tab(out);
bch2_pr_time_units_aligned(out, ns);
prt_newline(out);
}
void bch2_time_stats_to_text(struct printbuf *out, struct bch2_time_stats *stats) void bch2_time_stats_to_text(struct printbuf *out, struct bch2_time_stats *stats)
{ {
const struct time_unit *u; const struct time_unit *u;
u64 freq = READ_ONCE(stats->average_frequency); s64 f_mean = 0, d_mean = 0;
u64 q, last_q = 0; u64 q, last_q = 0, f_stddev = 0, d_stddev = 0;
int i; int i;
/*
* avoid divide by zero
*/
if (stats->freq_stats.n) {
f_mean = mean_and_variance_get_mean(stats->freq_stats);
f_stddev = mean_and_variance_get_stddev(stats->freq_stats);
d_mean = mean_and_variance_get_mean(stats->duration_stats);
d_stddev = mean_and_variance_get_stddev(stats->duration_stats);
}
printbuf_tabstop_push(out, out->indent + TABSTOP_SIZE);
prt_printf(out, "count:");
prt_tab(out);
prt_printf(out, "%llu ",
stats->duration_stats.n);
printbuf_tabstop_pop(out);
prt_newline(out);
printbuf_tabstops_reset(out);
prt_printf(out, "count:\t\t%llu", printbuf_tabstop_push(out, out->indent + 20);
stats->count); printbuf_tabstop_push(out, TABSTOP_SIZE + 2);
printbuf_tabstop_push(out, 0);
printbuf_tabstop_push(out, TABSTOP_SIZE + 2);
prt_tab(out);
prt_printf(out, "since mount");
prt_tab_rjust(out);
prt_tab(out);
prt_printf(out, "recent");
prt_tab_rjust(out);
prt_newline(out);
printbuf_tabstops_reset(out);
printbuf_tabstop_push(out, out->indent + 20);
printbuf_tabstop_push(out, TABSTOP_SIZE);
printbuf_tabstop_push(out, 2);
printbuf_tabstop_push(out, TABSTOP_SIZE);
prt_printf(out, "duration of events");
prt_newline(out);
printbuf_indent_add(out, 2);
pr_name_and_units(out, "min:", stats->min_duration);
pr_name_and_units(out, "max:", stats->max_duration);
prt_printf(out, "mean:");
prt_tab(out);
bch2_pr_time_units_aligned(out, d_mean);
prt_tab(out);
bch2_pr_time_units_aligned(out, mean_and_variance_weighted_get_mean(stats->duration_stats_weighted));
prt_newline(out);
prt_printf(out, "stddev:");
prt_tab(out);
bch2_pr_time_units_aligned(out, d_stddev);
prt_tab(out);
bch2_pr_time_units_aligned(out, mean_and_variance_weighted_get_stddev(stats->duration_stats_weighted));
printbuf_indent_sub(out, 2);
prt_newline(out); prt_newline(out);
prt_printf(out, "rate:\t\t%llu/sec",
freq ? div64_u64(NSEC_PER_SEC, freq) : 0); prt_printf(out, "time between events");
prt_newline(out); prt_newline(out);
printbuf_indent_add(out, 2);
prt_printf(out, "frequency:\t"); pr_name_and_units(out, "min:", stats->min_freq);
bch2_pr_time_units(out, freq); pr_name_and_units(out, "max:", stats->max_freq);
prt_printf(out, "mean:");
prt_tab(out);
bch2_pr_time_units_aligned(out, f_mean);
prt_tab(out);
bch2_pr_time_units_aligned(out, mean_and_variance_weighted_get_mean(stats->freq_stats_weighted));
prt_newline(out); prt_newline(out);
prt_printf(out, "avg duration:\t");
bch2_pr_time_units(out, stats->average_duration);
prt_printf(out, "stddev:");
prt_tab(out);
bch2_pr_time_units_aligned(out, f_stddev);
prt_tab(out);
bch2_pr_time_units_aligned(out, mean_and_variance_weighted_get_stddev(stats->freq_stats_weighted));
printbuf_indent_sub(out, 2);
prt_newline(out); prt_newline(out);
prt_printf(out, "max duration:\t");
bch2_pr_time_units(out, stats->max_duration); printbuf_tabstops_reset(out);
i = eytzinger0_first(NR_QUANTILES); i = eytzinger0_first(NR_QUANTILES);
u = pick_time_units(stats->quantiles.entries[i].m); u = pick_time_units(stats->quantiles.entries[i].m);
prt_newline(out);
prt_printf(out, "quantiles (%s):\t", u->name); prt_printf(out, "quantiles (%s):\t", u->name);
eytzinger0_for_each(i, NR_QUANTILES) { eytzinger0_for_each(i, NR_QUANTILES) {
bool is_last = eytzinger0_next(i, NR_QUANTILES) == -1; bool is_last = eytzinger0_next(i, NR_QUANTILES) == -1;
...@@ -468,6 +559,10 @@ void bch2_time_stats_exit(struct bch2_time_stats *stats) ...@@ -468,6 +559,10 @@ void bch2_time_stats_exit(struct bch2_time_stats *stats)
void bch2_time_stats_init(struct bch2_time_stats *stats) void bch2_time_stats_init(struct bch2_time_stats *stats)
{ {
memset(stats, 0, sizeof(*stats)); memset(stats, 0, sizeof(*stats));
stats->duration_stats_weighted.weight = 8;
stats->freq_stats_weighted.weight = 8;
stats->min_duration = U64_MAX;
stats->min_freq = U64_MAX;
spin_lock_init(&stats->lock); spin_lock_init(&stats->lock);
} }
......
...@@ -18,6 +18,8 @@ ...@@ -18,6 +18,8 @@
#include <linux/vmalloc.h> #include <linux/vmalloc.h>
#include <linux/workqueue.h> #include <linux/workqueue.h>
#include "mean_and_variance.h"
struct closure; struct closure;
#ifdef CONFIG_BCACHEFS_DEBUG #ifdef CONFIG_BCACHEFS_DEBUG
...@@ -407,14 +409,18 @@ struct bch2_time_stat_buffer { ...@@ -407,14 +409,18 @@ struct bch2_time_stat_buffer {
struct bch2_time_stats { struct bch2_time_stats {
spinlock_t lock; spinlock_t lock;
u64 count;
/* all fields are in nanoseconds */ /* all fields are in nanoseconds */
u64 average_duration;
u64 average_frequency;
u64 max_duration; u64 max_duration;
u64 min_duration;
u64 max_freq;
u64 min_freq;
u64 last_event; u64 last_event;
struct bch2_quantiles quantiles; struct bch2_quantiles quantiles;
struct mean_and_variance duration_stats;
struct mean_and_variance_weighted duration_stats_weighted;
struct mean_and_variance freq_stats;
struct mean_and_variance_weighted freq_stats_weighted;
struct bch2_time_stat_buffer __percpu *buffer; struct bch2_time_stat_buffer __percpu *buffer;
}; };
......
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