程序被加载器加载到内存后,通过/proc/$pid/maps文件,我们可以观测到程序被加载的内存位置。那么,通过打印进程内存的方式,让我们确认程序是不是真的加载到内存,以及加载到内存的程序和硬盘中的文件有没有区别。
编写测试程序:
#include <stdio.h>
#include <stdbool.h>
#include <unistd.h>
#include <stdlib.h>
int main() {
while(true) {
sleep(3);
printf("something\n");
}
return 0;
}
编译启动,然后打印程序的16进制
这里只展示部分16进制,用于与内存中的16进制进行对比。
编写内核代码:
#include <linux/module.h>
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/string.h>
#include <linux/netdevice.h>
#include <linux/printk.h>
static pte_t *get_pte(struct task_struct *task, unsigned long address) {
pgd_t *pgd;
p4d_t *p4d;
pud_t *pud;
pmd_t *pmd;
pte_t *pte;
struct mm_struct *mm = task->mm;
pgd = pgd_offset(mm,address);
if (pgd_none(*pgd) || pgd_bad(*pgd)) {
return NULL;
}
p4d = p4d_offset(pgd, address);
if (p4d_none(*p4d) || p4d_bad(*p4d))
return NULL;
pud = pud_offset(p4d, address);
if (pud_none(*pud) || pud_bad(*pud))
return NULL;
pmd = pmd_offset(pud, address);
if (pmd_none(*pmd) || pmd_bad(*pmd))
return NULL;
pte = pte_offset_kernel(pmd, address);
if (pte_none(*pte))
return NULL;
return pte;
}
static int hello_init(void) {
struct task_struct *p;
struct vm_area_struct *vma;
int len;
pte_t *pte;
struct page *page;
unsigned long addr;
printk(KERN_ALERT "init fishing\n");
for_each_process(p) {
if (strcmp(p->comm,"a.out") == 0) {
printk(KERN_ALERT "%s-->%p\n",p->comm, p->mm);
for(vma = p->mm->mmap;vma!=NULL;vma = vma->vm_next) {
printk(KERN_ALERT "%lx - %lx\n",vma->vm_start, vma->vm_end);
pte = get_pte(p, vma->vm_start);
if (pte == NULL)
break;
page = pte_page(*pte);
addr = page_address(page);
len = vma->vm_end - vma->vm_start;
print_hex_dump(KERN_NOTICE,"",0,16,1,addr,len,false);
break;
}
}
}
return 0;
}
static void hello_exit(void) {
printk(KERN_ALERT "exit fishing\n");
}
subsys_initcall(hello_init);
module_exit(hello_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("shakespeare");
此处只打印了进程第一个vm_area_struct表达的内存。
通过上述截图对比可知,程序的16进制被严格的加载到进程的内存空间,并且从进程的第一个vm_area_struct结构表达的内存处开始保存。