高端内存的映射 -- 永久映射和临时映射

高端内存(Highmem)中的页不能永久地映射到内核地址空间,因此,通过alloc_pages()函数,以__GFP_HIGHMEM标志分配的页不可能有虚拟地址。

X86体系结构中,高于896MB的所有物理内存都是高端内存,这些物理内存页不会永久的或自动的映射到内核地址空间。这些页需要被分配之后,才会映射到内核的虚拟地址空间上。

X86体系结构中,高端内存中的页,通常被映射到3G-4G虚拟地址空间上。

1. 永久映射:

要永久映射一个给定的page结构到内核地址空间,可以使用: void *kmap(struct page *page);

static inline void *kmap(struct page *page)
{
	might_sleep();
	return page_address(page);
}

这个函数可以睡眠,因此kmap()只能用在进程上下文中,不能用于中断上下文中。

这里,高端页的映射函数page_address()的定义如下:

/**
 * page_address - get the mapped virtual address of a page
 * @page: &struct page to get the virtual address of
 *
 * Returns the page's virtual address.
 */
void *page_address(struct page *page)
{
	unsigned long flags;
	void *ret;
	struct page_address_slot *pas;

	if (!PageHighMem(page))
		return lowmem_page_address(page);

	pas = page_slot(page);
	ret = NULL;
	spin_lock_irqsave(&pas->lock, flags);
	if (!list_empty(&pas->lh)) {
		struct page_address_map *pam;

		list_for_each_entry(pam, &pas->lh, list) {
			if (pam->page == page) {
				ret = pam->virtual;
				goto done;
			}
		}
	}
done:
	spin_unlock_irqrestore(&pas->lock, flags);
	return ret;
}
低端页的映射行数:由于低端内存能自动的映射到内核空间,所以只要直接返回其虚拟地址即可。

#define page_address(page) lowmem_page_address(page)
或:

#define page_address(page) ((page)->virtual)
所以,高端页的地址映射的函数 , 跟低端页的地址映射函数定义不一样:

当不再需要映射时,应该解除映射: void kunmap(struct page *page);

static inline void kunmap(struct page *page)
{
}
奇怪,怎么是个空函数。

2. 临时映射

当需要创建一个不允许睡眠的映射时,内核提供了一个临时映射(也叫原子映射)函数。

即内核可以原子的把高端内存中的一个页映射到某个保留的映射中。

临时映射可以在不能睡眠的地方,如中断处理程序中,因为获取映射时,绝对不会阻塞。

临时映射函数如下: void *kmap_atomic(struct *page, enum kmtype type);

/*
 * Make both: kmap_atomic(page, idx) and kmap_atomic(page) work.
 */
#define kmap_atomic(page, args...) __kmap_atomic(page)
static inline void *__kmap_atomic(struct page *page)
{
	pagefault_disable();
	return page_address(page);
}
这个函数不会睡眠,因此可以用在中断上下文和其他不能重新调度的地方;它也禁止内核抢占,因为映射对每个处理器都是唯一的。

取消映射函数: void kunmap_atomic(void *kvaddr, enum km_type type);

#define kunmap_atomic(addr, args...)				\
do {								\
	BUILD_BUG_ON(__same_type((addr), struct page *));	\
	__kunmap_atomic(addr);					\
} while (0)
static inline void __kunmap_atomic(void *addr)
{
	pagefault_enable();
}


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值