本文将通过一些简单的内核模块程序,显示一个进程的所有内存区域。通过此程序理解进程的整个地址空间与内存区域之间的关系。
打印内存区域
上文中,我们通过打印某个进程的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, 好的。这个例子挺好的~
[回复一下]
学习了!
[回复一下]