一:kmap()和kunmap()函数
永久内核映射允许内核建立高端页框到内核地址空间的长期映射。他使用主内核页表中一个专门的页表,其页表地址存放在pkmap_page_table中,页表包含512项或1024项,
因此,内核一次最多访问2M或4M的高端内存(地址范围是 4G-8M 到 4G-4M 之间,这个地址空间起叫“内核永久映射空间”或者“永久内核映射空间”)。
kmap()和kunmap()函数不能用于中断处理程序,因为其可能会睡眠。
pkmap_count是一个容量为LAST_PKMAP的整数数组,其中每个元素都对应一个持久映射页。
宏定义与关键变量定义:
pkmap_page_table:高端内存主内核页表中,一个用于永久内核映射的专用页表锁在的地址
LAST_PKMAP: 上述页表所含有的表项(512或者1024)
PKMAP_BASE:该页表所映射线性地址的start地址
pkmap_count:对页表项提供计数器的数组
page_address_htable:散列表,用于记录高端页框与永久内核映射的线性地址之间的关系
page_address_map:一个数据结构,包含指向页描述符的指针和分配给页框的线性地址;用于为高端内存的每
个页框提供当前映射,它被包含在page_address_htable这个hansh表中
内核主要通过以下数据结构来建立物理页page与其在虚拟内存区位置的关联
- struct page_address_map { //页地址映射
- struct page *page; //页的描述结构
- void *virtual; //页的虚拟地址
- struct list_head list; //通过list字段链接到页表池全局链表page_address_pool中或page_address_htable[hash_ptr(page,PA_HASH_ORDER)].lh
- };
- 1 void *kmap(struct page* page)
- 2 {
- 3 if(!PageHighMem(page))//判断是否是高端内存
- 4 return page_address(page);
- 5 return kmap_high(page);
- 6 }
- 7
- 8 void *kmap_high(struct page *page)
- 9 {
- 10 unsigned long vaddr;
- 11 spin_lock(&kmap_lock);//自旋锁
- 12 vaddr = (unsigned long) page_address(page);
- 13 if(!vaddr) //如果该page在虚拟内存中部对应有效的地址,那么调用map_new_virtual()函数进行映射该页,
- 14 vaddr = map_new_virtual(page);
- 15 pkmap_count[(vaddr - PKMAP_BASE) >> PAGE_SHIFT]++;//pkamp_count数组包含LAST_PKAMP个计数器
- 16 spin_unlock(&kmap_lock);
- 17 return (void *) vaddr;
- 18 }
二:kmap_atomic()临时内存函数
kmap_atomic()函数主要用于建立临时内存映射,它不能用于可能建立睡眠的代码。