mirror of
https://github.com/versity/scoutfs.git
synced 2026-04-28 00:46:57 +00:00
Add percpu counters that will let us track all manner of things.
To report them we add a sysfs directory full of attribute files in a
sysfs dir for each mount:
# (cd /sys/fs/scoutfs/loop0/counters && grep . *)
skip_delete:0
skip_insert:3218
skip_lookup:8439
skip_next:1190
skip_search:156
The implementation is careful to define each counter in only one place.
We don't have to make sure that a bunch of defintions and arrays are in
sync.
This builds off of Ben's initial patches that added sysfs dirs.
Signed-off-by: Zach Brown <zab@versity.com>
Signed-off-by: Ben McClelland <ben.mcclelland@versity.com>
132 lines
3.5 KiB
C
132 lines
3.5 KiB
C
/*
|
|
* Copyright (C) 2016 Versity Software, Inc. All rights reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public
|
|
* License v2 as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* General Public License for more details.
|
|
*/
|
|
#include <linux/kernel.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/fs.h>
|
|
#include <linux/percpu_counter.h>
|
|
|
|
#include "super.h"
|
|
#include "counters.h"
|
|
|
|
/*
|
|
* Maintain simple percpu counters which are always ticking. sysfs
|
|
* makes this a whole lot more noisy than it needs to be.
|
|
*/
|
|
|
|
#undef EXPAND_COUNTER
|
|
#define EXPAND_COUNTER(which) { .name = __stringify(which), .mode = 0644 },
|
|
static struct attribute scoutfs_counter_attrs[] = {
|
|
EXPAND_EACH_COUNTER
|
|
};
|
|
|
|
/* zero BSS and + 1 makes this null terminated */
|
|
#define NR_ATTRS ARRAY_SIZE(scoutfs_counter_attrs)
|
|
static struct attribute *scoutfs_counter_attr_ptrs[NR_ATTRS + 1];
|
|
|
|
static ssize_t scoutfs_counter_attr_show(struct kobject *kobj,
|
|
struct attribute *attr, char *buf)
|
|
{
|
|
struct scoutfs_counters *counters;
|
|
struct percpu_counter *pcpu;
|
|
size_t index;
|
|
|
|
/* use the index in the _attrs array to discover the pcpu pointer */
|
|
counters = container_of(kobj, struct scoutfs_counters, kobj);
|
|
index = attr - scoutfs_counter_attrs;
|
|
pcpu = &counters->FIRST_COUNTER + index;
|
|
|
|
return snprintf(buf, PAGE_SIZE, "%lld\n", percpu_counter_sum(pcpu));
|
|
}
|
|
|
|
static void scoutfs_counters_kobj_release(struct kobject *kobj)
|
|
{
|
|
struct scoutfs_counters *counters;
|
|
|
|
counters = container_of(kobj, struct scoutfs_counters, kobj);
|
|
|
|
complete(&counters->comp);
|
|
}
|
|
|
|
static const struct sysfs_ops scoutfs_counter_attr_ops = {
|
|
.show = scoutfs_counter_attr_show,
|
|
};
|
|
|
|
static struct kobj_type scoutfs_counters_ktype = {
|
|
.default_attrs = scoutfs_counter_attr_ptrs,
|
|
.sysfs_ops = &scoutfs_counter_attr_ops,
|
|
.release = scoutfs_counters_kobj_release,
|
|
};
|
|
|
|
int scoutfs_setup_counters(struct super_block *sb)
|
|
{
|
|
struct scoutfs_sb_info *sbi = SCOUTFS_SB(sb);
|
|
struct scoutfs_counters *counters;
|
|
struct percpu_counter *pcpu;
|
|
int ret;
|
|
|
|
counters = kzalloc(sizeof(struct scoutfs_counters), GFP_KERNEL);
|
|
if (!counters)
|
|
return -ENOMEM;
|
|
sbi->counters = counters;
|
|
|
|
scoutfs_foreach_counter(sb, pcpu) {
|
|
ret = percpu_counter_init(pcpu, 0, GFP_KERNEL);
|
|
if (ret)
|
|
return ret;
|
|
}
|
|
|
|
counters->kobj.kset = sbi->kset;
|
|
init_completion(&counters->comp);
|
|
ret = kobject_init_and_add(&counters->kobj, &scoutfs_counters_ktype,
|
|
NULL, "counters");
|
|
if (ret) {
|
|
/* tear down partial to avoid destroying null kobjs */
|
|
scoutfs_foreach_counter(sb, pcpu)
|
|
percpu_counter_destroy(pcpu);
|
|
kfree(counters);
|
|
sbi->counters = NULL;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
void scoutfs_destroy_counters(struct super_block *sb)
|
|
{
|
|
struct scoutfs_sb_info *sbi = SCOUTFS_SB(sb);
|
|
struct scoutfs_counters *counters = sbi->counters;
|
|
struct percpu_counter *pcpu;
|
|
|
|
/* this only destroys fully initialized counters */
|
|
if (!counters)
|
|
return;
|
|
|
|
kobject_del(&counters->kobj);
|
|
kobject_put(&counters->kobj);
|
|
wait_for_completion(&counters->comp);
|
|
|
|
scoutfs_foreach_counter(sb, pcpu)
|
|
percpu_counter_destroy(pcpu);
|
|
|
|
kfree(counters);
|
|
sbi->counters = NULL;
|
|
}
|
|
|
|
void __init scoutfs_init_counters(void)
|
|
{
|
|
int i;
|
|
|
|
/* not ARRAY_SIZE because that would clobber null term */
|
|
for (i = 0; i < NR_ATTRS; i++)
|
|
scoutfs_counter_attr_ptrs[i] = &scoutfs_counter_attrs[i];
|
|
}
|