Files
kernel_common_drivers/drivers/memory_ext/mem_debug.c
T
Jianxiong Pan 300eb86ec4 mm: profile default cma pool usage. [1/1]
PD#SWPL-252747

Problem:
profile default cma pool usage.

Solution:
cat /proc/profile_default_cma
==============================
count(KB)            kaddr, function
------------------------------
   10632,         bf5e0364, amdv_init_receiver [aml_media]
     512,         c03cd278, atomic_pool_expand
     272,         bf1af5ec, meson_mmc_probe [amlogic_mmc]
     256,         c1307960, atomic_pool_init
     176,         bf27a628, rdma_ins_setting [aml_media]
     136,         bfd33fd0, xhci_segment_alloc [amlogic_usb]
      96,         c0c92ac4, snd_dma_dev_alloc
------------------------------
total pages:  3055,     12220 kB, type:CMA

Verify:
local.

Change-Id: Icebc024415c204ca787ac2fdefd50f643b246d01
Signed-off-by: Jianxiong Pan <jianxiong.pan@amlogic.com>
2026-01-21 22:23:49 -08:00

543 lines
14 KiB
C

// SPDX-License-Identifier: (GPL-2.0+ OR MIT)
/*
* Copyright (c) 2019 Amlogic, Inc. All rights reserved.
*/
#include <linux/slab.h>
#include <linux/kernel.h>
#include <linux/export.h>
#include <linux/errno.h>
#include <asm/sections.h>
#include <linux/sizes.h>
#ifdef CONFIG_RISCV
#include <asm/pgtable.h>
#else
#include <asm/memory.h>
#endif
#include <linux/init.h>
#include <linux/initrd.h>
#include <linux/gfp.h>
#include <linux/memblock.h>
#include <linux/dma-mapping.h>
#include <linux/vmalloc.h>
#include <linux/mm.h>
#include <linux/amlogic/mem_debug.h>
#include <linux/crash_dump.h>
#ifdef CONFIG_ARM64
#include <asm/boot.h>
#endif
#include <asm/fixmap.h>
#include <linux/kasan.h>
#include <linux/seq_file.h>
#ifdef CONFIG_HIGHMEM
#include <asm/highmem.h>
#endif
#ifdef CONFIG_AMLOGIC_CMA
#include <linux/amlogic/aml_cma.h>
#endif
#ifdef CONFIG_AMLOGIC_VMAP
#include <linux/amlogic/vmap_stack.h>
#endif
#if IS_MODULE(CONFIG_AMLOGIC_MEM_DEBUG)
#include <linux/module.h>
#endif
#include <linux/page_owner.h>
#include <linux/cma.h>
#include <cma.h>
static int pagemap_en;
unsigned long mlock_fault_size;
void dump_mem_layout(char *buf)
{
#define MLK(b, t) b, t, ((t) - (b)) >> 10
#define MLM(b, t) b, t, ((t) - (b)) >> 20
#define MLG(b, t) b, t, ((t) - (b)) >> 30
#define MLK_ROUNDUP(b, t) b, t, DIV_ROUND_UP(((t) - (b)), SZ_1K)
#ifdef CONFIG_ARM64
int pos = 0;
pos += sprintf(buf + pos, "Virtual kernel memory layout:\n");
#ifdef CONFIG_KASAN_GENERIC
pos += sprintf(buf + pos, " kasan : 0x%16lx - 0x%16lx (%6ld GB)\n",
MLG(KASAN_SHADOW_START, KASAN_SHADOW_END));
#endif
pos += sprintf(buf + pos, " modules : 0x%16lx - 0x%16lx (%6ld MB)\n",
MLM(MODULES_VADDR, MODULES_END));
pos += sprintf(buf + pos, " vmalloc : 0x%16lx - 0x%16lx (%6ld GB)\n",
MLG(VMALLOC_START, VMALLOC_END));
#if IS_MODULE(CONFIG_AMLOGIC_MEM_DEBUG)
pos += sprintf(buf + pos, " fixed : 0x%16lx - 0x%16lx (%6ld KB)\n",
MLK(FIXADDR_START, FIXADDR_TOP));
pos += sprintf(buf + pos, " PCI I/O : 0x%16lx - 0x%16lx (%6ld MB)\n",
MLM(PCI_IO_START, PCI_IO_END));
#ifdef CONFIG_SPARSEMEM_VMEMMAP
pos += sprintf(buf + pos, " vmemmap : 0x%16lx - 0x%16lx (%6ld GB maximum)\n",
MLG(VMEMMAP_START, VMEMMAP_START + VMEMMAP_SIZE));
#endif
#else
pos += sprintf(buf + pos, " .text : 0x%px" " - 0x%px" " (%6ld KB) 0x%lx\n",
MLK_ROUNDUP(_text, _etext), (unsigned long)__pa_symbol(_text));
pos += sprintf(buf + pos, " .rodata : 0x%px" " - 0x%px" " (%6ld KB) 0x%lx\n",
MLK_ROUNDUP(__start_rodata, __init_begin),
(unsigned long)__pa_symbol(__start_rodata));
pos += sprintf(buf + pos, " .init : 0x%px" " - 0x%px" " (%6ld KB) 0x%lx\n",
MLK_ROUNDUP(__init_begin, __init_end),
(unsigned long)__pa_symbol(__start_rodata));
pos += sprintf(buf + pos, " .data : 0x%px" " - 0x%px" " (%6ld KB) 0x%lx\n",
MLK_ROUNDUP(_sdata, _edata),
(unsigned long)__pa_symbol(_sdata));
pos += sprintf(buf + pos, " .bss : 0x%px" " - 0x%px" " (%6ld KB) 0x%lx\n",
MLK_ROUNDUP(__bss_start, __bss_stop),
(unsigned long)__pa_symbol(__bss_start));
pos += sprintf(buf + pos, " fixed : 0x%16lx - 0x%16lx (%6ld KB)\n",
MLK(FIXADDR_START, FIXADDR_TOP));
pos += sprintf(buf + pos, " PCI I/O : 0x%16lx - 0x%16lx (%6ld MB)\n",
MLM(PCI_IO_START, PCI_IO_END));
#ifdef CONFIG_SPARSEMEM_VMEMMAP
pos += sprintf(buf + pos, " vmemmap : 0x%16lx - 0x%16lx (%6ld GB maximum)\n",
MLG(VMEMMAP_START, VMEMMAP_START + VMEMMAP_SIZE));
pos += sprintf(buf + pos, " 0x%16lx - 0x%16lx (%6ld MB actual)\n",
MLM((unsigned long)phys_to_page(memblock_start_of_DRAM()),
(unsigned long)virt_to_page(high_memory)));
#endif
pos += sprintf(buf + pos, " memory : 0x%16lx - 0x%16lx (%6ld MB) 0x%lx\n",
MLM(__phys_to_virt(memblock_start_of_DRAM()),
(unsigned long)high_memory), (unsigned long)memblock_start_of_DRAM());
pos += sprintf(buf + pos, " offset : 0x%lx\n", kaslr_offset());
#endif
#elif defined CONFIG_ARM
sprintf(buf, "Virtual kernel memory layout:\n"
#ifdef CONFIG_KASAN
" kasan : 0x%08lx - 0x%08lx (%4ld MB)\n"
#endif
#ifdef CONFIG_HAVE_TCM
" DTCM : 0x%08lx - 0x%08lx (%4ld kB)\n"
" ITCM : 0x%08lx - 0x%08lx (%4ld kB)\n"
#endif
" fixmap : 0x%08lx - 0x%08lx (%4ld kB)\n"
" vmalloc : 0x%08lx - 0x%08lx (%4ld MB)\n"
" lowmem : 0x%08lx - 0x%08lx (%4ld MB)\n"
#ifdef CONFIG_HIGHMEM
" pkmap : 0x%08lx - 0x%08lx (%4ld MB)\n"
#endif
#ifdef CONFIG_MODULES
" modules : 0x%08lx - 0x%08lx (%4ld MB)\n"
#endif
" .text : 0x%px" " - 0x%px" " (%4td kB)\n"
" .init : 0x%px" " - 0x%px" " (%4td kB)\n"
" .data : 0x%px" " - 0x%px" " (%4td kB)\n"
" .bss : 0x%px" " - 0x%px" " (%4td kB)\n",
#ifdef CONFIG_KASAN
MLM(KASAN_SHADOW_START, KASAN_SHADOW_END),
#endif
#ifdef CONFIG_HAVE_TCM
MLK(DTCM_OFFSET, (unsigned long) dtcm_end),
MLK(ITCM_OFFSET, (unsigned long) itcm_end),
#endif
MLK(FIXADDR_START, FIXADDR_END),
MLM(VMALLOC_START, VMALLOC_END),
MLM(PAGE_OFFSET, (unsigned long)high_memory),
#ifdef CONFIG_HIGHMEM
MLM(PKMAP_BASE, (PKMAP_BASE) + (LAST_PKMAP) *
(PAGE_SIZE)),
#endif
#ifdef CONFIG_MODULES
MLM(MODULES_VADDR, MODULES_END),
#endif
MLK_ROUNDUP(_text, _etext),
MLK_ROUNDUP(__init_begin, __init_end),
MLK_ROUNDUP(_sdata, _edata),
MLK_ROUNDUP(__bss_start, __bss_stop));
#elif defined CONFIG_RISCV
int pos = 0;
pos += sprintf(buf + pos, "Virtual kernel memory layout:\n");
pos += sprintf(buf + pos, " fixed : 0x%16lx - 0x%16lx (%6ld KB)\n",
MLK(FIXADDR_START, FIXADDR_TOP));
pos += sprintf(buf + pos, " PCI I/O : 0x%16lx - 0x%16lx (%6ld MB)\n",
MLM(PCI_IO_START, PCI_IO_END));
#ifdef CONFIG_SPARSEMEM_VMEMMAP
pos += sprintf(buf + pos, " vmemmap : 0x%16lx - 0x%16lx (%6ld GB maximum)\n",
MLG(VMEMMAP_START, VMEMMAP_START + VMEMMAP_SIZE));
pos += sprintf(buf + pos, " 0x%16lx - 0x%16lx (%6ld MB actual)\n",
MLM((unsigned long)phys_to_page(memblock_start_of_DRAM()),
(unsigned long)virt_to_page(high_memory)));
#endif
pos += sprintf(buf + pos, " vmalloc : 0x%16lx - 0x%16lx (%6ld GB)\n",
MLG(VMALLOC_START, VMALLOC_START + VMALLOC_SIZE));
pos += sprintf(buf + pos, " memory : 0x%16lx - 0x%16lx (%6ld MB) 0x%lx\n",
MLM(PAGE_OFFSET, (unsigned long)high_memory),
(unsigned long)memblock_start_of_DRAM());
pos += sprintf(buf + pos, " modules : 0x%16lx - 0x%16lx (%6ld MB)\n",
MLM(MODULES_VADDR, MODULES_END));
pos += sprintf(buf + pos, " kernel : 0x%16lx - 0x%16lx (%6ld MB)\n",
MLM(KERNEL_LINK_ADDR, ADDRESS_SPACE_END));
pos += sprintf(buf + pos, " .text : 0x%px" " - 0x%px" " (%6ld KB) 0x%lx\n",
MLK_ROUNDUP(_text, _etext), (unsigned long)__pa_symbol(_text));
pos += sprintf(buf + pos, " .init : 0x%px" " - 0x%px" " (%6ld KB) 0x%lx\n",
MLK_ROUNDUP(__init_begin, __init_end),
(unsigned long)__pa_symbol(__init_begin));
pos += sprintf(buf + pos, " .rodata : 0x%px" " - 0x%px" " (%6ld KB) 0x%lx\n",
MLK_ROUNDUP(__start_rodata, __end_rodata),
(unsigned long)__pa_symbol(__start_rodata));
pos += sprintf(buf + pos, " .data : 0x%px" " - 0x%px" " (%6ld KB) 0x%lx\n",
MLK_ROUNDUP(_data, _edata),
(unsigned long)__pa_symbol(_data));
pos += sprintf(buf + pos, " .bss : 0x%px" " - 0x%px" " (%6ld KB) 0x%lx\n",
MLK_ROUNDUP(__bss_start, __bss_stop),
(unsigned long)__pa_symbol(__bss_start));
#endif
}
static int mdebug_show(struct seq_file *m, void *arg)
{
char *buf = kmalloc(4096, GFP_KERNEL);
if (!buf) {
pr_err("%s failed\n", __func__);
return -1;
}
/* update only once */
dump_mem_layout(buf);
seq_printf(m, "%s\n", buf);
pr_info("pagemap_en:%d\n", pagemap_en);
memset(buf, 0, 4096);
sprintf(buf, "mlock_fault_size: %ldkb\n", mlock_fault_size * 4);
seq_printf(m, "%s\n", buf);
kfree(buf);
return 0;
}
static int mdebug_open(struct inode *inode, struct file *file)
{
return single_open(file, mdebug_show, NULL);
}
int pagemap_enabled(void)
{
return pagemap_en;
}
#if !IS_MODULE(CONFIG_AMLOGIC_MEM_DEBUG)
/*
* These information will auto inject to /proc/meminfo sysfs
*/
void arch_report_meminfo(struct seq_file *m)
{
#ifdef CONFIG_AMLOGIC_CMA
seq_printf(m, "DriverCma: %8ld kB\n",
get_cma_allocated() * (1 << (PAGE_SHIFT - 10)));
#endif
#ifdef CONFIG_AMLOGIC_VMAP
vmap_report_meminfo(m);
#endif
}
static int early_pagemap(char *buf)
{
if (!buf)
return -EINVAL;
if (!strncmp(buf, "eng", 3) ||
!strncmp(buf, "userdebug", 9))
pagemap_en = 1;
pr_info("%s pagemap for %s build\n",
pagemap_en ? "enable" : "disable", buf);
return 1;
}
__setup("buildvariant=", early_pagemap);
#endif
static ssize_t mdebug_write(struct file *file, const char __user *buffer,
size_t count, loff_t *ppos)
{
char *buf;
unsigned long arg = 0;
buf = kmalloc(count, GFP_KERNEL);
if (!buf)
return -ENOMEM;
if (copy_from_user(buf, buffer, count)) {
kfree(buf);
return -EINVAL;
}
if (!strncmp(buf, "pagemap_en=", 6)) { /* option for 'pagemap_en=' */
if (sscanf(buf, "pagemap_en=%ld", &arg) < 0) {
kfree(buf);
return -EINVAL;
}
pagemap_en = arg ? 1 : 0;
pr_info("set pagemap_en to %d\n", pagemap_en);
}
kfree(buf);
return count;
}
static const struct proc_ops mdebug_ops = {
.proc_open = mdebug_open,
.proc_read = seq_read,
.proc_write = mdebug_write,
.proc_lseek = seq_lseek,
.proc_release = single_release,
};
static size_t str_end_char(const char *s, size_t count, char c)
{
size_t len = count, offset = count;
while (count--) {
if (*s == c || *s++ == '\0') {
offset = len - count;
break;
}
}
return offset;
}
void dump_mem_layout_boot_phase(void)
{
char *buf = (void *)__get_free_page(GFP_KERNEL);
int i = 0, len = 0, offset = 0;
if (!buf) {
pr_err("%s alloc buffer failed\n", __func__);
} else {
dump_mem_layout(buf);
len = strlen(buf);
while (i < len) {
offset = str_end_char(buf + i, 512, '\n');
pr_crit("%.*s", offset, buf + i);
i += offset;
}
free_page((unsigned long)buf);
}
}
static inline unsigned long aml_cma_bitmap_maxno(struct cma *cma)
{
return cma->count >> cma->order_per_bit;
}
static int update_cma_stat(struct cma *cma, struct cma_stat *stat)
{
unsigned long start, pfn;
struct page *page;
unsigned long *bitmap;
unsigned long nbits;
int pos = 0;
char *buf = NULL;
if (!cma) {
pr_err("not find mm_cma pool\n");
return -1;
}
if (!stat->buffer) {
pr_err("not alloc stat buffer, suggest: vzalloc(cma->count * 3)\n");
return -EINVAL;
}
buf = stat->buffer;
nbits = aml_cma_bitmap_maxno(cma);
bitmap = bitmap_zalloc(aml_cma_bitmap_maxno(cma), GFP_KERNEL);
if (!bitmap) {
pr_info("codec cma bitmap alloc failed.\n");
return -ENOMEM;
}
spin_lock_irq(&cma->lock);
bitmap_copy(bitmap, cma->bitmap, nbits);
spin_unlock_irq(&cma->lock);
start = cma->base_pfn;
pos += sprintf(buf + pos, "cma base: %lx, count: %lx\n", start, cma->count);
pos += sprintf(buf + pos, "D: driver, B: buddy free, A: anon, F: file\n");
for (pfn = start; pfn < start + cma->count; pfn++) {
if (pfn % 64 == 0)
pos += sprintf(buf + pos, "\npfn: %lx", pfn);
if (test_bit(pfn - start, bitmap)) {
stat->driver_c++;
pos += sprintf(buf + pos, " D");
continue;
}
page = pfn_to_page(pfn);
if (PageBuddy(page) || (!folio_mapcount(page_folio(page)) &&
!folio_ref_count(page_folio(page)))) {
stat->free_c++;
pos += sprintf(buf + pos, " B");
continue;
}
if (PageAnon(page)) {
stat->anon_c++;
pos += sprintf(buf + pos, " A");
} else {
stat->file_c++;
pos += sprintf(buf + pos, " F");
}
}
pos += sprintf(buf + pos, "\ncount: %ld, driver: %d, free: %d, anon: %d, file: %d\n",
cma->count, stat->driver_c, stat->free_c, stat->anon_c, stat->file_c);
bitmap_free(bitmap);
return 0;
}
int get_cma_stat(struct cma *cma, struct cma_stat *stat, bool print_log)
{
int ret = update_cma_stat(cma, stat);
if (ret)
return ret;
if (print_log)
pr_info("%s\n", stat->buffer);
return 0;
}
EXPORT_SYMBOL(get_cma_stat);
static int offset_show;
static struct cma *g_cma;
static int list_cma_pool(struct cma *cma, void *data)
{
const char *cma_name = cma_get_name(cma);
char *buf = (char *)data;
offset_show += sprintf(buf + offset_show, "\t%s\n", cma_name);
return 0;
}
static int cma_stat_show(struct seq_file *m, void *arg)
{
char *buf;
struct cma_stat stat = {0};
if (!g_cma) {
buf = kzalloc(512, GFP_KERNEL);
if (!buf)
return -ENOMEM;
offset_show = 0;
cma_for_each_area(list_cma_pool, buf);
seq_puts(m, "please echo cma_name, then cat /proc/cma_stat.\n");
seq_printf(m, "cma list:\n%s\n", buf);
kfree(buf);
} else {
stat.buffer = vzalloc(g_cma->count * 3);
if (!stat.buffer)
return -ENOMEM;
update_cma_stat(g_cma, &stat);
seq_printf(m, "%s\n", stat.buffer);
vfree(stat.buffer);
}
return 0;
}
static int cma_stat_open(struct inode *inode, struct file *file)
{
return single_open(file, cma_stat_show, NULL);
}
static int get_cma_pool(struct cma *cma, void *data)
{
const char *cma_name = cma_get_name(cma);
char *buf = (char *)data;
if (strstr(cma_name, buf))
g_cma = cma;
return 0;
}
static ssize_t cma_stat_write(struct file *file, const char __user *buffer,
size_t count, loff_t *ppos)
{
char *buf;
buf = kzalloc(count, GFP_KERNEL);
if (!buf)
return -ENOMEM;
/*remove line break at the end*/
if (copy_from_user(buf, buffer, count - 1)) {
kfree(buf);
return -EINVAL;
}
if (!strncmp(buf, "none", 4))
g_cma = NULL;
else
cma_for_each_area(get_cma_pool, buf);
kfree(buf);
return count;
}
static const struct proc_ops cma_stat_ops = {
.proc_open = cma_stat_open,
.proc_read = seq_read,
.proc_write = cma_stat_write,
.proc_lseek = seq_lseek,
.proc_release = single_release,
};
static int __init memory_debug_init(void)
{
struct proc_dir_entry *d_mdebug;
d_mdebug = proc_create("mem_debug", 0444, NULL, &mdebug_ops);
if (IS_ERR_OR_NULL(d_mdebug)) {
pr_err("%s, create proc failed\n", __func__);
return -1;
}
d_mdebug = proc_create("cma_stat", 0444, NULL, &cma_stat_ops);
if (IS_ERR_OR_NULL(d_mdebug)) {
pr_err("create /proc/cma_stat failed\n");
return -1;
}
return 0;
}
#if IS_MODULE(CONFIG_AMLOGIC_MEM_DEBUG)
static int __init mem_debug_module_init(void)
{
memory_debug_init();
dump_mem_layout_boot_phase();
return 0;
}
static void __exit mem_debug_module_exit(void)
{
}
module_init(mem_debug_module_init);
module_exit(mem_debug_module_exit);
MODULE_LICENSE("GPL v2");
#else
rootfs_initcall(memory_debug_init);
#endif