AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 734931
Accepted
drahnoel
drahnoel
Asked: 2023-02-10 08:22:17 +0800 CST2023-02-10 08:22:17 +0800 CST 2023-02-10 08:22:17 +0800 CST

为什么没有实现复合系统调用之类的东西?

  • 772

由于内核空间和用户空间之间的隔离,Syscalls(系统调用)会导致一些性能损失。因此,减少系统调用听起来是个好主意。

所以我的想法是,我们可以将系统调用打包成一个。因此,我们的想法是将系统调用和参数放在内存中的简单数据结构中。然后我们可以引入一个新的系统调用,我们给这个数据结构。如果一个(或所有)系统调用完成,内核然后可以并行触发所有功能并恢复线程。

我认为这种方法将成为并发编程(异步 I/O)的良好基础,并将通过允许在任何系统调用上并发并减少整体上下文切换来改进现有的 select/poll/epoll 解决方案。

为什么不这样做?

system-calls
  • 3 3 个回答
  • 2012 Views

3 个回答

  • Voted
  1. Best Answer
    Stephen Kitt
    2023-02-10T08:25:37+08:002023-02-10T08:25:37+08:00

    这已经存在。在 Linux 上,它由io_uring实现,自内核版本 5.1(2019 年 5 月)起可用:操作被放置在一个队列(或者更确切地说,环)上并且在没有系统调用的情况下进行处理,它们的结果将进入另一个队列。

    • 25
  2. Austin Hemmelgarn
    2023-02-10T19:22:30+08:002023-02-10T19:22:30+08:00

    一般概念已经完成并且确实存在。正如 Stephen Kitt 的回答所指出的,最接近的示例是 Linux 上的 io_uring,但它远非此类接口的唯一示例。Windows、Solaris、AmigaOS 和少数其他操作系统都具有类似的面向 IO 的完成队列机制,其工作方式与 io_uring 类似(Linux 实际上有点晚了)。

    此外,类 UNIX 系统上实际上有很多系统调用,虽然它们不像您建议的那样工作,但通过将一些通常在用户空间中完成的任务推入内核来避免大量潜在的上下文切换。系统sendfile()调用可能是此类系统调用的最佳示例,它执行一项非常常见的任务(将大量数据从一个文件描述符复制到另一个文件描述符)并将其完全推送到内核模式,从而完全避免循环和大量上下文切换(和额外的缓冲区)在用户空间中执行此操作所需的。

    不过,这里要理解的一个关键问题是,为了使它有意义,像这样批量设置与相关操作集相关的所有内容的成本必须低于仅以“正常”方式进行的成本。只有在处理大量IO 时使用 io_uring 才有意义,例如在为 VM 模拟块存储设备时(QEMU 支持为此使用它,即使在快速主机硬件上的性能差异也是疯狂的),或者读取数千每秒处理一次文件(我工作的公司最近开始在内部讨论可能对此类工作负载使用 io_uring)。相似地,sendfile()仅当您需要多次读/写迭代来通过用户空间复制数据时才有意义(尽管这通常是无法负担用户空间中的缓冲区空间的功能,而不是运行读/写迭代更快) .

    此外,系统调用实际上必须在批处理上下文中有意义。如果处理保留调用的顺序,IO 通常在这里有意义,但很多事情都没有。例如,尝试使用这种类型的接口是愚蠢的(可能exec()是 fork 和 exec 的组合,但不是普通的 exec)。同样,某些类型的系统调用只有在单独处理时才有用。操纵进程的信号掩码就是一个很好的例子,除了初始设置之外,您几乎总是这样做是为了保护代码中的关键部分,并且您通常需要为此目的进行及时、可预测的处理。

    • 21
  3. Andrew Henle
    2023-02-11T11:50:29+08:002023-02-11T11:50:29+08:00

    这些功能已经存在了很长时间。

    1997 年的 Solaris 2.6添加了一个内核异步 IO 系统调用,它正是这样做的 - kaio()。

    访问它的一种方法是通过`lio_listio() 函数:

    lio_listio

    • 列出定向 I/O

    概要

    cc [ flag... ] file... -lrt [ library... ]
    #include <aio.h>
    
    int lio_listio(int mode, struct aiocb *restrict const list[], int nent,
         struct sigevent *restrict sig);
    

    描述

    该lio_listio()函数允许调用进程、LWP 或线程在单个函数调用中启动 I/O 请求列表。

    可以在https://github.com/illumos/illumos-gate/blob/470204d3561e07978b63600336e8d47cc75387fa/usr/src/lib/libc/port/aio中找到Illumoslibc源代码,该源代码是开源的并且源自原始的 Solaris 实现/posix_aio.c#L121lio_listio()

    此类功能不常见的原因之一是,除非整个软件和硬件系统旨在利用它,否则它们实际上不会提高性能。

    必须配置存储以提供正确对齐的块,必须构建文件系统以便它们与存储系统提供的块正确对齐,并且需要编写整个软件堆栈以免搞砸 IO - 这一切都有做正确对齐的 IO。

    对于旋转磁盘,对同一磁盘的一批 IO 操作很容易相互干扰,并且实际上会随着磁头花费更多时间寻找而减慢一切。

    根据我的经验,只需要其中一层做错事情,批处理系统调用的性能优势就会消失在开销中。因为与最糟糕的系统调用开销相比,IO 是缓慢的。

    创建和维护组合的硬件/软件系统以利用批处理 IO 系统调用提供的性能改进的成本是巨大的。

    我见过的最好的数字是将许多 IO 调用批处理到一个系统调用中可以提高大约 25-30% 的性能。

    如果您全天候连续处理数百 GB 的数据,那很重要。

    构建和维护这样的整个系统只是为了将观看猫视频的延迟从 8 毫秒降低到 6 毫秒?没那么多。

    • 7

相关问题

  • 从 strace 日志中过滤掉失败的系统调用

  • 系统调用读取的缓冲区使用的地址格式是什么?

  • 在 C++ 代码中使用 system() 函数与使用源代码一样快速吗?[关闭]

  • 内核模式位

  • 需要一些系统调用

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve