当前位置: 首页 > 面试题库 >

如何将Linux内核缓冲区映射到用户空间?

董飞
2023-03-14
问题内容

假设使用基于页面的方案分配缓冲区。一种实现mmap的方法是使用remap_pfn_range,但LDD3表示这不适用于常规内存。看来我们可以通过使用SetPageReserved标记保留的页面来解决此问题,以便将其锁定在内存中。但是,不是所有内核内存都已经不可交换,即已经保留了吗?为什么需要显式设置保留位?

这与从HIGH_MEM分配的页面有关吗?


问题答案:

使用mmap方法从内核映射一组页面的最简单方法是使用故障处理程序来映射这些页面。基本上,您最终得到如下结果:

static int my_mmap(struct file *filp, struct vm_area_struct *vma)
{
    vma->vm_ops = &my_vm_ops;
    return 0;
}

static const struct file_operations my_fops = {
    .owner  = THIS_MODULE,
    .open   = nonseekable_open,
    .mmap   = my_mmap,
    .llseek = no_llseek,
};

(其中其他文件操作是您的模块需要的)。另外,my_mmap您还可以执行任何范围检查等操作来验证mmap参数。

然后vm_ops看起来像:

static int my_fault(struct vm_area_struct *vma, struct vm_fault *vmf)
{
    vmf->page = my_page_at_index(vmf->pgoff);
    get_page(vmf->page);

    return 0;
}

static const struct vm_operations_struct my_vm_ops = {
    .fault      = my_fault
}

您只需要找出传递给您的故障功能的给定vma / vmf,即可将哪个页面映射到用户空间。这完全取决于模块的工作方式。例如,如果您做了

my_buf = vmalloc_user(MY_BUF_SIZE);

那么您使用的页面将类似于

vmalloc_to_page(my_buf + (vmf->pgoff << PAGE_SHIFT));

但是您可以轻松创建一个数组并为每个条目分配一个页面,无论使用什么,都可以使用kmalloc。

[只是注意到这my_fault是一个稍微有趣的功能名称]



 类似资料:
  • 问题内容: 似乎在Windows 32位上,内核将从全部4G用户虚拟内存空间中保留1G虚拟内存,并将 某些 内核空间映射到此1G空间中。 所以我的问题是: 在32位Linux上是否有类似情况? 如果是这样,我们如何看到整个内存布局? 我认为 只能看到某些过程的用户空间布局。 谢谢! 问题答案: 实际上,在32位Windows上,如果没有引导选项,内核将映射到线性地址空间的前2GB,剩下2GB的空间

  • 问题内容: 我在linux上有一个Java应用程序,它可以打开UDP套接字并等待消息。 在高负载下运行了几个小时之后,有一个数据包丢失,即数据包被内核接收,但不是由我的应用程序接收(我们在嗅探器中看到丢失的数据包,在netstat中看到UDP数据包丢失,我们没有看到这些数据包)在我们的应用日志中)。 我们尝试扩大套接字缓冲区,但这并没有帮助-我们早些时候就开始丢失数据包,仅此而已。 对于调试,我想

  • 我正在尝试使用RxJava创建一个顺序下载服务。用户可以批量添加项目(20、30等)或单个项目。这些项目将被添加到队列中,然后以10个批次的顺序下载。为此,我使用PublishSubject: 它发出用户添加的项(ID),然后将缓冲区操作符应用于批处理项。使用这些ID,可以在flatMap中下载项目,并在订阅的onNext中返回。 代码大部分按预期工作。项目已成功批处理并下载,但即使在发出所有项目

  • 我正在为Raspbian(Raspberry Pi 2)中的Python编写视频捕获脚本,我在使用v4l2的Python绑定时遇到了麻烦,因为我在内存映射缓冲区方面没有成功。 我所需要的: 从高清网络摄像头捕获视频(稍后将同时捕获其中两个) 我所尝试的: 使用OpenCV(cv2)。它很容易使用,但是它增加了很多处理负载,因为它将网络摄像头的JPEG帧转换为原始图像,然后我必须在通过WLAN发送它

  • 问题内容: Linux的默认套接字缓冲区大小是多少?有什么命令可以看到吗? 问题答案: 如果要在终端中查看缓冲区大小,可以看一下: (供阅读) (写) 它们包含三个数字,分别是最小,默认和最大内存大小值(以字节为单位)。

  • 问题内容: 我写了一个非常简单的函数,可以从给定的URL下载图像,调整图像大小并上传到S3(使用’gm’和’knox’),我不知道我是否在正确地将流读取到缓冲区中。(一切正常,但这是正确的方法吗?) 另外,我想了解一些有关事件循环的知识,如何知道该函数的一次调用不会泄漏任何内容,也不会将’buf’变量更改为另一个已经运行的调用(否则这种情况是不可能的,因为回调是匿名的功能?) 问题答案: 总的来说