Skip to content

Commit 98f1808

Browse files
committed
mm: Make mem_dump_obj() handle vmalloc() memory
This commit adds vmalloc() support to mem_dump_obj(). Note that the vmalloc_dump_obj() function combines the checking and dumping, in contrast with the split between kmem_valid_obj() and kmem_dump_obj(). The reason for the difference is that the checking in the vmalloc() case involves acquiring a global lock, and redundant acquisitions of global locks should be avoided, even on not-so-fast paths. Note that this change causes on-stack variables to be reported as vmalloc() storage from kernel_clone() or similar, depending on the degree of inlining that your compiler does. This is likely more helpful than the earlier "non-paged (local) memory". Cc: Andrew Morton <[email protected]> Cc: Joonsoo Kim <[email protected]> Cc: <[email protected]> Reported-by: Andrii Nakryiko <[email protected]> Acked-by: Vlastimil Babka <[email protected]> Tested-by: Naresh Kamboju <[email protected]> Signed-off-by: Paul E. McKenney <[email protected]>
1 parent b70fa3b commit 98f1808

File tree

3 files changed

+26
-6
lines changed

3 files changed

+26
-6
lines changed

include/linux/vmalloc.h

+6
Original file line numberDiff line numberDiff line change
@@ -246,4 +246,10 @@ pcpu_free_vm_areas(struct vm_struct **vms, int nr_vms)
246246
int register_vmap_purge_notifier(struct notifier_block *nb);
247247
int unregister_vmap_purge_notifier(struct notifier_block *nb);
248248

249+
#ifdef CONFIG_MMU
250+
bool vmalloc_dump_obj(void *object);
251+
#else
252+
static inline bool vmalloc_dump_obj(void *object) { return false; }
253+
#endif
254+
249255
#endif /* _LINUX_VMALLOC_H */

mm/util.c

+8-6
Original file line numberDiff line numberDiff line change
@@ -996,18 +996,20 @@ int __weak memcmp_pages(struct page *page1, struct page *page2)
996996
*/
997997
void mem_dump_obj(void *object)
998998
{
999+
if (kmem_valid_obj(object)) {
1000+
kmem_dump_obj(object);
1001+
return;
1002+
}
1003+
if (vmalloc_dump_obj(object))
1004+
return;
9991005
if (!virt_addr_valid(object)) {
10001006
if (object == NULL)
10011007
pr_cont(" NULL pointer.\n");
10021008
else if (object == ZERO_SIZE_PTR)
10031009
pr_cont(" zero-size pointer.\n");
10041010
else
1005-
pr_cont(" non-paged (local) memory.\n");
1006-
return;
1007-
}
1008-
if (kmem_valid_obj(object)) {
1009-
kmem_dump_obj(object);
1011+
pr_cont(" non-paged memory.\n");
10101012
return;
10111013
}
1012-
pr_cont(" non-slab memory.\n");
1014+
pr_cont(" non-slab/vmalloc memory.\n");
10131015
}

mm/vmalloc.c

+12
Original file line numberDiff line numberDiff line change
@@ -3448,6 +3448,18 @@ void pcpu_free_vm_areas(struct vm_struct **vms, int nr_vms)
34483448
}
34493449
#endif /* CONFIG_SMP */
34503450

3451+
bool vmalloc_dump_obj(void *object)
3452+
{
3453+
struct vm_struct *vm;
3454+
void *objp = (void *)PAGE_ALIGN((unsigned long)object);
3455+
3456+
vm = find_vm_area(objp);
3457+
if (!vm)
3458+
return false;
3459+
pr_cont(" vmalloc allocated at %pS\n", vm->caller);
3460+
return true;
3461+
}
3462+
34513463
#ifdef CONFIG_PROC_FS
34523464
static void *s_start(struct seq_file *m, loff_t *pos)
34533465
__acquires(&vmap_purge_lock)

0 commit comments

Comments
 (0)