本文将通过一些简单的内核模块程序,显示一个进程的所有内存区域。通过此程序理解进程的整个地址空间与内存区域之间的关系。
打印内存区域
上文中,我们通过打印某个进程的maps文件来查看某个进程的内存区域。如果你理解了进程,进程的用户空间,内存区域三者之间的关系,那么就可以通过内核模块的方式打印指定进程的内存区域。
通过在内核模块加载函数中调用下述函数,来打印当前进程的内存区域。首先通过全局变量current获得当前进程的mm字段,该字段指向当前进程的用户空间(mm_struct);由于多个内存区域(vm_area_struct)是通过一个双链表(最新内核中)链接在一起的,所以在接下来的for循环当中,依次遍历各个内存区域,打印当前内存区域的起始地址和终止地址,并且打印内核对该区域的操作权限。完整代码在这里。
static void list_myvma(void) { struct mm_struct *mm = current->mm; struct vm_area_struct *vma; printk("list vma..\n"); //print the current process's name and pid printk("current:%s pid:%d\n",current->comm,current->pid); down_read(&mm->mmap_sem); //vma is a linklist for(vma = mm->mmap; vma; vma = vma->vm_next) { //from the begining to the ending of a virtual memory area printk("0x%lx-0x%lx ",vma->vm_start,vma->vm_end); //check the flags of this VMA if(vma->vm_flags & VM_READ) printk("r"); else printk("-"); if(vma->vm_flags & VM_WRITE) printk("w"); else printk("-"); if(vma->vm_flags & VM_EXEC) printk("x"); else printk("-"); if(vma->vm_flags & VM_SHARED) printk("s"); else printk("p"); printk("\n"); } up_read(&mm->mmap_sem); }
试一下吧!
源代码在http://www.kerneltravel.net/?p=450;
但是编译时报错:
error:dereferencing pointer to incomplete type
你有源文件么?
[回复一下]
edsionte 回复:
19 9 月, 2011 at 08:35
@ltang, 请问您是对代码有疑惑还是要源码的?源码我这里是没有的,不好意思。
[回复一下]
tony 回复:
19 9 月, 2011 at 11:32
@edsionte, 我是在ubuntu9.10上面编译,内核版本为2.6.31.4,用上面网页上的链接
处的源码,编译报错.问题原因也找到:缺少两个头文件:
#include
#include
[回复一下]
edsionte 回复:
19 9 月, 2011 at 11:45
@ltang, 好的。这个例子挺好的~
[回复一下]
学习了!
[回复一下]