forked from Minki/linux
mm: slub: move sysfs slab alloc/free interfaces to debugfs
alloc_calls and free_calls implementation in sysfs have two issues, one is PAGE_SIZE limitation of sysfs and other is it does not adhere to "one value per file" rule. To overcome this issues, move the alloc_calls and free_calls implementation to debugfs. Debugfs cache will be created if SLAB_STORE_USER flag is set. Rename the alloc_calls/free_calls to alloc_traces/free_traces, to be inline with what it does. [faiyazm@codeaurora.org: fix the leak of alloc/free traces debugfs interface] Link: https://lkml.kernel.org/r/1624248060-30286-1-git-send-email-faiyazm@codeaurora.org Link: https://lkml.kernel.org/r/1623438200-19361-1-git-send-email-faiyazm@codeaurora.org Signed-off-by: Faiyaz Mohammed <faiyazm@codeaurora.org> Reviewed-by: Vlastimil Babka <vbabka@suse.cz> Reviewed-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org> Cc: Christoph Lameter <cl@linux.com> Cc: Pekka Enberg <penberg@kernel.org> Cc: David Rientjes <rientjes@google.com> Cc: Joonsoo Kim <iamjoonsoo.kim@lge.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
parent
792702911f
commit
64dd68497b
@ -631,6 +631,12 @@ static inline bool slab_want_init_on_free(struct kmem_cache *c)
|
||||
return false;
|
||||
}
|
||||
|
||||
#if defined(CONFIG_DEBUG_FS) && defined(CONFIG_SLUB_DEBUG)
|
||||
void debugfs_slab_release(struct kmem_cache *);
|
||||
#else
|
||||
static inline void debugfs_slab_release(struct kmem_cache *s) { }
|
||||
#endif
|
||||
|
||||
#ifdef CONFIG_PRINTK
|
||||
#define KS_ADDRS_COUNT 16
|
||||
struct kmem_obj_info {
|
||||
|
@ -448,6 +448,7 @@ static void slab_caches_to_rcu_destroy_workfn(struct work_struct *work)
|
||||
rcu_barrier();
|
||||
|
||||
list_for_each_entry_safe(s, s2, &to_destroy, list) {
|
||||
debugfs_slab_release(s);
|
||||
kfence_shutdown_cache(s);
|
||||
#ifdef SLAB_SUPPORTS_SYSFS
|
||||
sysfs_slab_release(s);
|
||||
@ -475,6 +476,7 @@ static int shutdown_cache(struct kmem_cache *s)
|
||||
schedule_work(&slab_caches_to_rcu_destroy_work);
|
||||
} else {
|
||||
kfence_shutdown_cache(s);
|
||||
debugfs_slab_release(s);
|
||||
#ifdef SLAB_SUPPORTS_SYSFS
|
||||
sysfs_slab_unlink(s);
|
||||
sysfs_slab_release(s);
|
||||
|
274
mm/slub.c
274
mm/slub.c
@ -38,6 +38,7 @@
|
||||
#include <linux/random.h>
|
||||
#include <kunit/test.h>
|
||||
|
||||
#include <linux/debugfs.h>
|
||||
#include <trace/events/kmem.h>
|
||||
|
||||
#include "internal.h"
|
||||
@ -238,6 +239,12 @@ static inline int sysfs_slab_alias(struct kmem_cache *s, const char *p)
|
||||
{ return 0; }
|
||||
#endif
|
||||
|
||||
#if defined(CONFIG_DEBUG_FS) && defined(CONFIG_SLUB_DEBUG)
|
||||
static void debugfs_slab_add(struct kmem_cache *);
|
||||
#else
|
||||
static inline void debugfs_slab_add(struct kmem_cache *s) { }
|
||||
#endif
|
||||
|
||||
static inline void stat(const struct kmem_cache *s, enum stat_item si)
|
||||
{
|
||||
#ifdef CONFIG_SLUB_STATS
|
||||
@ -4593,6 +4600,9 @@ int __kmem_cache_create(struct kmem_cache *s, slab_flags_t flags)
|
||||
if (err)
|
||||
__kmem_cache_release(s);
|
||||
|
||||
if (s->flags & SLAB_STORE_USER)
|
||||
debugfs_slab_add(s);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
@ -4739,6 +4749,7 @@ long validate_slab_cache(struct kmem_cache *s)
|
||||
}
|
||||
EXPORT_SYMBOL(validate_slab_cache);
|
||||
|
||||
#ifdef CONFIG_DEBUG_FS
|
||||
/*
|
||||
* Generate lists of code addresses where slabcache objects are allocated
|
||||
* and freed.
|
||||
@ -4762,6 +4773,8 @@ struct loc_track {
|
||||
struct location *loc;
|
||||
};
|
||||
|
||||
static struct dentry *slab_debugfs_root;
|
||||
|
||||
static void free_loc_track(struct loc_track *t)
|
||||
{
|
||||
if (t->max)
|
||||
@ -4878,82 +4891,7 @@ static void process_slab(struct loc_track *t, struct kmem_cache *s,
|
||||
add_location(t, s, get_track(s, p, alloc));
|
||||
put_map(map);
|
||||
}
|
||||
|
||||
static int list_locations(struct kmem_cache *s, char *buf,
|
||||
enum track_item alloc)
|
||||
{
|
||||
int len = 0;
|
||||
unsigned long i;
|
||||
struct loc_track t = { 0, 0, NULL };
|
||||
int node;
|
||||
struct kmem_cache_node *n;
|
||||
|
||||
if (!alloc_loc_track(&t, PAGE_SIZE / sizeof(struct location),
|
||||
GFP_KERNEL)) {
|
||||
return sysfs_emit(buf, "Out of memory\n");
|
||||
}
|
||||
/* Push back cpu slabs */
|
||||
flush_all(s);
|
||||
|
||||
for_each_kmem_cache_node(s, node, n) {
|
||||
unsigned long flags;
|
||||
struct page *page;
|
||||
|
||||
if (!atomic_long_read(&n->nr_slabs))
|
||||
continue;
|
||||
|
||||
spin_lock_irqsave(&n->list_lock, flags);
|
||||
list_for_each_entry(page, &n->partial, slab_list)
|
||||
process_slab(&t, s, page, alloc);
|
||||
list_for_each_entry(page, &n->full, slab_list)
|
||||
process_slab(&t, s, page, alloc);
|
||||
spin_unlock_irqrestore(&n->list_lock, flags);
|
||||
}
|
||||
|
||||
for (i = 0; i < t.count; i++) {
|
||||
struct location *l = &t.loc[i];
|
||||
|
||||
len += sysfs_emit_at(buf, len, "%7ld ", l->count);
|
||||
|
||||
if (l->addr)
|
||||
len += sysfs_emit_at(buf, len, "%pS", (void *)l->addr);
|
||||
else
|
||||
len += sysfs_emit_at(buf, len, "<not-available>");
|
||||
|
||||
if (l->sum_time != l->min_time)
|
||||
len += sysfs_emit_at(buf, len, " age=%ld/%ld/%ld",
|
||||
l->min_time,
|
||||
(long)div_u64(l->sum_time,
|
||||
l->count),
|
||||
l->max_time);
|
||||
else
|
||||
len += sysfs_emit_at(buf, len, " age=%ld", l->min_time);
|
||||
|
||||
if (l->min_pid != l->max_pid)
|
||||
len += sysfs_emit_at(buf, len, " pid=%ld-%ld",
|
||||
l->min_pid, l->max_pid);
|
||||
else
|
||||
len += sysfs_emit_at(buf, len, " pid=%ld",
|
||||
l->min_pid);
|
||||
|
||||
if (num_online_cpus() > 1 &&
|
||||
!cpumask_empty(to_cpumask(l->cpus)))
|
||||
len += sysfs_emit_at(buf, len, " cpus=%*pbl",
|
||||
cpumask_pr_args(to_cpumask(l->cpus)));
|
||||
|
||||
if (nr_online_nodes > 1 && !nodes_empty(l->nodes))
|
||||
len += sysfs_emit_at(buf, len, " nodes=%*pbl",
|
||||
nodemask_pr_args(&l->nodes));
|
||||
|
||||
len += sysfs_emit_at(buf, len, "\n");
|
||||
}
|
||||
|
||||
free_loc_track(&t);
|
||||
if (!t.count)
|
||||
len += sysfs_emit_at(buf, len, "No data\n");
|
||||
|
||||
return len;
|
||||
}
|
||||
#endif /* CONFIG_DEBUG_FS */
|
||||
#endif /* CONFIG_SLUB_DEBUG */
|
||||
|
||||
#ifdef CONFIG_SYSFS
|
||||
@ -5343,21 +5281,6 @@ static ssize_t validate_store(struct kmem_cache *s,
|
||||
}
|
||||
SLAB_ATTR(validate);
|
||||
|
||||
static ssize_t alloc_calls_show(struct kmem_cache *s, char *buf)
|
||||
{
|
||||
if (!(s->flags & SLAB_STORE_USER))
|
||||
return -ENOSYS;
|
||||
return list_locations(s, buf, TRACK_ALLOC);
|
||||
}
|
||||
SLAB_ATTR_RO(alloc_calls);
|
||||
|
||||
static ssize_t free_calls_show(struct kmem_cache *s, char *buf)
|
||||
{
|
||||
if (!(s->flags & SLAB_STORE_USER))
|
||||
return -ENOSYS;
|
||||
return list_locations(s, buf, TRACK_FREE);
|
||||
}
|
||||
SLAB_ATTR_RO(free_calls);
|
||||
#endif /* CONFIG_SLUB_DEBUG */
|
||||
|
||||
#ifdef CONFIG_FAILSLAB
|
||||
@ -5521,8 +5444,6 @@ static struct attribute *slab_attrs[] = {
|
||||
&poison_attr.attr,
|
||||
&store_user_attr.attr,
|
||||
&validate_attr.attr,
|
||||
&alloc_calls_attr.attr,
|
||||
&free_calls_attr.attr,
|
||||
#endif
|
||||
#ifdef CONFIG_ZONE_DMA
|
||||
&cache_dma_attr.attr,
|
||||
@ -5810,6 +5731,173 @@ static int __init slab_sysfs_init(void)
|
||||
__initcall(slab_sysfs_init);
|
||||
#endif /* CONFIG_SYSFS */
|
||||
|
||||
#if defined(CONFIG_SLUB_DEBUG) && defined(CONFIG_DEBUG_FS)
|
||||
static int slab_debugfs_show(struct seq_file *seq, void *v)
|
||||
{
|
||||
|
||||
struct location *l;
|
||||
unsigned int idx = *(unsigned int *)v;
|
||||
struct loc_track *t = seq->private;
|
||||
|
||||
if (idx < t->count) {
|
||||
l = &t->loc[idx];
|
||||
|
||||
seq_printf(seq, "%7ld ", l->count);
|
||||
|
||||
if (l->addr)
|
||||
seq_printf(seq, "%pS", (void *)l->addr);
|
||||
else
|
||||
seq_puts(seq, "<not-available>");
|
||||
|
||||
if (l->sum_time != l->min_time) {
|
||||
seq_printf(seq, " age=%ld/%llu/%ld",
|
||||
l->min_time, div_u64(l->sum_time, l->count),
|
||||
l->max_time);
|
||||
} else
|
||||
seq_printf(seq, " age=%ld", l->min_time);
|
||||
|
||||
if (l->min_pid != l->max_pid)
|
||||
seq_printf(seq, " pid=%ld-%ld", l->min_pid, l->max_pid);
|
||||
else
|
||||
seq_printf(seq, " pid=%ld",
|
||||
l->min_pid);
|
||||
|
||||
if (num_online_cpus() > 1 && !cpumask_empty(to_cpumask(l->cpus)))
|
||||
seq_printf(seq, " cpus=%*pbl",
|
||||
cpumask_pr_args(to_cpumask(l->cpus)));
|
||||
|
||||
if (nr_online_nodes > 1 && !nodes_empty(l->nodes))
|
||||
seq_printf(seq, " nodes=%*pbl",
|
||||
nodemask_pr_args(&l->nodes));
|
||||
|
||||
seq_puts(seq, "\n");
|
||||
}
|
||||
|
||||
if (!idx && !t->count)
|
||||
seq_puts(seq, "No data\n");
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static void slab_debugfs_stop(struct seq_file *seq, void *v)
|
||||
{
|
||||
}
|
||||
|
||||
static void *slab_debugfs_next(struct seq_file *seq, void *v, loff_t *ppos)
|
||||
{
|
||||
struct loc_track *t = seq->private;
|
||||
|
||||
v = ppos;
|
||||
++*ppos;
|
||||
if (*ppos <= t->count)
|
||||
return v;
|
||||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static void *slab_debugfs_start(struct seq_file *seq, loff_t *ppos)
|
||||
{
|
||||
return ppos;
|
||||
}
|
||||
|
||||
static const struct seq_operations slab_debugfs_sops = {
|
||||
.start = slab_debugfs_start,
|
||||
.next = slab_debugfs_next,
|
||||
.stop = slab_debugfs_stop,
|
||||
.show = slab_debugfs_show,
|
||||
};
|
||||
|
||||
static int slab_debug_trace_open(struct inode *inode, struct file *filep)
|
||||
{
|
||||
|
||||
struct kmem_cache_node *n;
|
||||
enum track_item alloc;
|
||||
int node;
|
||||
struct loc_track *t = __seq_open_private(filep, &slab_debugfs_sops,
|
||||
sizeof(struct loc_track));
|
||||
struct kmem_cache *s = file_inode(filep)->i_private;
|
||||
|
||||
if (strcmp(filep->f_path.dentry->d_name.name, "alloc_traces") == 0)
|
||||
alloc = TRACK_ALLOC;
|
||||
else
|
||||
alloc = TRACK_FREE;
|
||||
|
||||
if (!alloc_loc_track(t, PAGE_SIZE / sizeof(struct location), GFP_KERNEL))
|
||||
return -ENOMEM;
|
||||
|
||||
/* Push back cpu slabs */
|
||||
flush_all(s);
|
||||
|
||||
for_each_kmem_cache_node(s, node, n) {
|
||||
unsigned long flags;
|
||||
struct page *page;
|
||||
|
||||
if (!atomic_long_read(&n->nr_slabs))
|
||||
continue;
|
||||
|
||||
spin_lock_irqsave(&n->list_lock, flags);
|
||||
list_for_each_entry(page, &n->partial, slab_list)
|
||||
process_slab(t, s, page, alloc);
|
||||
list_for_each_entry(page, &n->full, slab_list)
|
||||
process_slab(t, s, page, alloc);
|
||||
spin_unlock_irqrestore(&n->list_lock, flags);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int slab_debug_trace_release(struct inode *inode, struct file *file)
|
||||
{
|
||||
struct seq_file *seq = file->private_data;
|
||||
struct loc_track *t = seq->private;
|
||||
|
||||
free_loc_track(t);
|
||||
return seq_release_private(inode, file);
|
||||
}
|
||||
|
||||
static const struct file_operations slab_debugfs_fops = {
|
||||
.open = slab_debug_trace_open,
|
||||
.read = seq_read,
|
||||
.llseek = seq_lseek,
|
||||
.release = slab_debug_trace_release,
|
||||
};
|
||||
|
||||
static void debugfs_slab_add(struct kmem_cache *s)
|
||||
{
|
||||
struct dentry *slab_cache_dir;
|
||||
|
||||
if (unlikely(!slab_debugfs_root))
|
||||
return;
|
||||
|
||||
slab_cache_dir = debugfs_create_dir(s->name, slab_debugfs_root);
|
||||
|
||||
debugfs_create_file("alloc_traces", 0400,
|
||||
slab_cache_dir, s, &slab_debugfs_fops);
|
||||
|
||||
debugfs_create_file("free_traces", 0400,
|
||||
slab_cache_dir, s, &slab_debugfs_fops);
|
||||
}
|
||||
|
||||
void debugfs_slab_release(struct kmem_cache *s)
|
||||
{
|
||||
debugfs_remove_recursive(debugfs_lookup(s->name, slab_debugfs_root));
|
||||
}
|
||||
|
||||
static int __init slab_debugfs_init(void)
|
||||
{
|
||||
struct kmem_cache *s;
|
||||
|
||||
slab_debugfs_root = debugfs_create_dir("slab", NULL);
|
||||
|
||||
list_for_each_entry(s, &slab_caches, list)
|
||||
if (s->flags & SLAB_STORE_USER)
|
||||
debugfs_slab_add(s);
|
||||
|
||||
return 0;
|
||||
|
||||
}
|
||||
__initcall(slab_debugfs_init);
|
||||
#endif
|
||||
/*
|
||||
* The /proc/slabinfo ABI
|
||||
*/
|
||||
|
Loading…
Reference in New Issue
Block a user