5

如何确定设备内存的页框号?来自 LDD3/ Ch. 15/“使用 remap_pfn_range”和“一个简单的实现”部分,pfn 已等同于 vm_pgoff 字段。我对此感到困惑。怎么会这样?

请注意,vm_pgoff 被描述为:

文件中区域的偏移量,以页为单位。映射文件或设备时,这是该区域中映射的第一页的文件位置。

因此,如果映射的第一页也对应于文件的第一页(我认为这很常见),vm_pgoff 将为 0。对吗?如果是这样,这似乎不是 remap_pfn_range( ) 的 pfn 参数的正确值。我在这里想念什么?什么是正确的值?为了便于参考,我从下面的LDD3中复制了相关代码(第426页)

static int simple_remap_mmap(struct file *filp, struct vm_area_struct *vma)
{
if (remap_pfn_range(vma, vma->vm_start, vm->vm_pgoff,
                    vma->vm_end - vma->vm_start,
                    vma->vm_page_prot))
    return -EAGAIN;
...
}
4

1 回答 1

5

venk,我只是一个新手。我对你的问题很好奇,而且我已经阅读了你在 Linux 设备驱动程序中指出的部分。

我应该推荐另一本书,Professional Linux kernel architecture。那里描述了有关 mmap 的一些细节。

在上面的书中,似乎在调用函数 simple_remap_mmap() 作为示例之前对 vm->vm_pgoff 进行了一些转换,这在 Linux 内核源代码中不存在。

因此,即使 vm_pgoff 为 0,在对其进行一些转换后,该成员变量 vm_pgoff 中也可能存在正确的值。

以下代码存在于 Linux 内核源代码 3.3.5 中

#ifdef CONFIG_DEVKMEM

static int mmap_kmem(struct file *file, struct vm_area_struct *vma)
{

    unsigned long pfn;

    /* Turn a kernel-virtual address into a physical page frame */

    pfn = __pa((u64)vma->vm_pgoff << PAGE_SHIFT) >> PAGE_SHIFT;   <----- Here

    /*
     * RED-PEN: on some architectures there is more mapped memory than
     * available in mem_map which pfn_valid checks for. Perhaps should add a
     * new macro here.
     *
     * RED-PEN: vmalloc is not supported right now.
     */
    if (!pfn_valid(pfn))
        return -EIO;

    vma->vm_pgoff = pfn;
    return mmap_mem(file, vma);
}
#endif
于 2012-07-17T09:02:36.243 回答