linux进程map,linux内存优化一文中 查看进程mem_map 的实现

最近有同事在读<>一文,我也看了下,写得不错,满通俗易懂的,但是其中关于

进程mem map 显示的东西,好象没提供,我搜了下只找到个显示全部mem map 的,于是自己写了一个,

虽然代码很小,但写+调试也花了半天时间,废话少说 ,具体代码如下:

mem_map.c :

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#define liuprintk printk

unsigned short *proc_memmap;

int msize;

struct page *vm_get_page(struct vm_area_struct *vma, unsigned long addr, pte_t pte)

{

unsigned long pfn = pte_pfn(pte);

if (unlikely(vma->vm_flags & (VM_PFNMAP|VM_MIXEDMAP))) {

liuprintk("\nliutest: has pure pfn \n");

#if 0

if (vma->vm_flags & VM_MIXEDMAP) {

if (!pfn_valid(pfn))

return NULL; /* invalid page */

goto out;

} else {

unsigned long off;

off = (addr - vma->vm_start) >> PAGE_SHIFT; /* VM_PFNMAP 的处理*/

if (pfn == vma->vm_pgoff + off)

return NULL;

if (!is_cow_mapping(vma->vm_flags))

return NULL;

}

#endif

}

//out:

return pfn_to_page(pfn);

}

static inline void

show_one_pte(struct mm_struct *mm, pte_t *src_pte, struct vm_area_struct *vma,

unsigned long addr) /* liunote linux pte */

{

unsigned long vm_flags = vma->vm_flags;

pte_t pte = *src_pte; /* get pte val */

struct page *page;

int pm_index;

if (unlikely(!pte_present(pte))) {

if (!pte_file(pte)) { /* liunote anymous page */

/* liunote can do it best */

}

goto out_show_pte;

}

page = vm_get_page(vma, addr, pte); /* 是normal page */

if (page) {

/* liunote get share proc num */

pm_index = (addr - vma->vm_start) /PAGE_SIZE;

proc_memmap[pm_index] = atomic_read(&(page)->_mapcount);

//printk("liutest pmd_index=0x%X,addr= 0x%X,vmstart=0x%X:,mapcount=%d", pm_index ,addr , vma->vm_start,atomic_read(&(page)->_mapcount));

if (proc_memmap[pm_index]==0){

proc_memmap[pm_index] =1;

}

}

out_show_pte:

return;

}

static int show_ptes(struct mm_struct *mm, pmd_t *pmd, struct vm_area_struct *vma,

unsigned long addr, unsigned long end)

{

pte_t *pte;

spinlock_t *ptl;

/* liunote below dst_pte is linux pte ??? */

pte = pte_offset_map_nested(pmd, addr);

ptl = pte_lockptr(mm, pmd);

/* liunote need lock to read it ??? */

/* spin_lock_nested(ptl, SINGLE_DEPTH_NESTING);*/

do {

if (pte_none(*pte)) { /* src_pte = 0 */

//liuprintk("\nliutest: bad pte=0x%X,addr=0x%X\n",pte,addr);

continue;

}

show_one_pte(mm, pte, vma, addr);

} while (pte++, addr += PAGE_SIZE, addr != end);

/* spin_unlock(ptl); */

return 0;

}

static inline int show_pmds(struct mm_struct *mm, pud_t *pud, struct vm_area_struct *vma,

unsigned long addr, unsigned long end)

{

pmd_t * pmd;

unsigned long next;

pmd = pmd_offset(pud, addr);

do {

next = pmd_addr_end(addr, end);

if (pmd_none_or_clear_bad(pmd)){

liuprintk("\nliutest: bad pmd=0x%X,addr=0x%X\n",pmd,addr);

continue;

}

if (show_ptes(mm, pmd,

vma, addr, next))

return -ENOMEM;

} while (pmd++, addr = next, addr != end);

return 0;

}

static inline int show_puds(struct mm_struct *mm, pgd_t *pgd, struct vm_area_struct *vma,

unsigned long addr, unsigned long end)

{

pud_t *pud ;

unsigned long next;

pud = pud_offset(pgd, addr);

do {

next = pud_addr_end(addr, end);

if (pud_none_or_clear_bad(pud)){

liuprintk("\n liutest: bad pud=0x%X,addr=0x%X\n",pud,addr);

continue;

}

if (show_pmds(mm, pud,

vma, addr, next))

return -ENOMEM;

} while (pud++, addr = next, addr != end);

return 0;

}

// liuadd for print pte

int show_pages(struct mm_struct * mm, struct vm_area_struct *vma)

{

pgd_t *pgd;

unsigned long next;

unsigned long addr = vma->vm_start;

unsigned long end = vma->vm_end;

int ret,i;

int pages =0;

// according vm size ,malloc proc memmap

msize = end-addr;

msize = (msize & (PAGE_SIZE-1)) ? (msize+PAGE_SIZE)/PAGE_SIZE : msize/PAGE_SIZE;

liuprintk("\nliutest: msize =0x%X, PAGE_SIZE=0x%X\n",msize,PAGE_SIZE);

proc_memmap = kmalloc( msize*(sizeof(unsigned short)),GFP_KERNEL);

if(proc_memmap == NULL){

printk("\n liutest : proc mem map alloc fail \n");

return 1;

}

for(i=0; i

proc_memmap[i] = 0;

/*

if (!(vma->vm_flags & (VM_HUGETLB|VM_NONLINEAR|VM_PFNMAP|VM_INSERTPAGE))) {

if (!vma->anon_vma)

return 0;

}

*/

ret = 0;

pgd = pgd_offset(mm, addr);

do {

next = pgd_addr_end(addr, end);

if (pgd_none_or_clear_bad(pgd)){

liuprintk("\nliutest: bad pgd=0x%X,addr=0x%X\n",pgd,addr);

continue;

}

if (unlikely(show_puds(mm, pgd, vma, addr, next))) {

ret = -ENOMEM;

break;

}

} while (pgd++, addr = next, addr != end);

#if 1

for(i=0; i

if(proc_memmap[i])

pages++;

printk("\nliutest: this vm area has pages =0x%X\n",pages);

// free memmap

kfree(proc_memmap);

#endif

return ret;

}

#undef liuprintk

上面mem_map.c 文件加到proc 目录下

然后在task_mmu.c show_map_vma 函数末尾加上:

show_pages(vma->vm_mm, vma);

proc 下面Makefile 加上 mem_map.o

代码有点乱,不过测试过还行,内容全在proc_memmap 数组里,大小为msize,

可根据自己喜好显示从proc 输出

下面是测试busybox内存情况

Cd proc

Cd 1

Cat maps显示如下:

00008000-00100000 r-xp 00000000 00:01 60         /bin/busybox

Pages:0xF8-0x4D      ( F8页,实际占用4D页,下面是实际页内存引用情况)

0x0002 0x0002 0x0000 0x0000 0x0002 0x0002 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000

0x0000 0x0000 0x0000 0x0000 0x0000 0x0001 0x0001 0x0001 0x0001 0x0001 0x0001 0x0001 0x0001 0x0001 0x0001 0x0001 0x0001 0x0001 0x0001 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0001 0x0000 0x0001 0x0001 0x0000 0x0000 0x0000

0x0000 0x0000 0x0000 0x0000 0x0002 0x0000 0x0002 0x0000 0x0000 0x0002 0x0002 0x0002 0x0002 0x0000 0x0000 0x0000 0x0000 0x0001 0x0001 0x0001 0x0001 0x0000 0x0000 0x0000 0x0001 0x0001 0x0000 0x0000 0x0001 0x0001 0x0001 0x0001

0x0001 0x0001 0x0002 0x0002 0x0002 0x0002 0x0002 0x0001 0x0002 0x0002 0x0001 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0001 0x0002 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000

0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0001 0x0002 0x0000 0x0000 0x0002 0x0002 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0002 0x0002

0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0002 0x0001 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0001 0x0000 0x0000 0x0000 0x0002 0x0000 0x0000 0x0000 0x0000 0x0000

0x0000 0x0002 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0002 0x0001 0x0002 0x0000 0x0002 0x0000 0x0001 0x0001 0x0000 0x0000 0x0000

0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0001 0x0000 0x0000 0x0002 0x0002 0x0000 0x0002 0x0002 0x0000 0x0000 0x0002 0x0000 0x0001 0x0001 0x0001 0x0001 0x0000 0x0000

00107000-00108000 rw-p 000f7000 00:01 60         /bin/busybox

Pages:0x1-0x1

0x0001

00108000-0012e000 rw-p 00108000 00:00 0          [heap]

Pages:0x26-0x6

0x0001 0x0001 0x0000 0x0001 0x0001 0x0001 0x0001 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000

0x0000 0x0000 0x0000 0x0000 0x0000 0x0000

40000000-40001000 rw-p 40000000 00:00 0

Pages:0x1-0x1

0x0001

be844000-be859000 rw-p befeb000 00:00 0          [stack]

Pages:0x15-0x2     (这是栈,有15页,实际占用2页,下面也可以看出,占了倒数的两页符合栈特性,)

0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0000 0x0001 0x0001

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值