草庐IT

linux - signalfd 和 sigaction 之间可以竞争吗?

coder 2023-06-18 原文

为特定信号指定处理程序的经典方法是通过 sigaction。 Linux 还提供了 signalfd 功能,我们可以在其中将信号连接到文件描述符,然后将 select/(e)poll 应用于该描述符,这非常适合许多事件循环驱动系统的概念。

我想知道当两种机制发生冲突时会发生/应该发生什么。可以有竞争条件吗?在 signalfd 联机帮助页 ( http://man7.org/linux/man-pages/man2/signalfd.2.html ) 上,我们读到:

Normally, the set of signals to be received via the file descriptor should be blocked using sigprocmask(2), to prevent the signals being handled according to their default dispositions.

因此,它说“通常”我们使用信号掩码以防止(默认)处理程序处理信号。它并没有说当我们有一个文件描述符连接到它时,我们必须阻止该信号。不幸的是,手册页没有具体说明当我们不阻止信号时会发生什么。

这看起来像是定义不明确的行为。我不相信这实际上没有明确定义,并且想知道这里是否有人知道i)我能找到关于系统应该如何运行的详细规范或ii ) 它的行为方式。

我特别感兴趣的是这个执行顺序:

  1. 针对某个信号的signalfd,包括对这个信号的阻塞
  2. 此信号的畅通
  3. 此信号的 sigaction(默认处理程序或自定义处理程序)

这是未定义的行为还是必须发生的事情有标准/规范?处理程序是否总是优先于文件描述符?处理程序是否调用了 文件描述符触发了一个事件?设置 sigaction 是否会更改信号掩码,从而不需要渲染步骤 (2)?

我可以尝试从涉及实际代码的系统测试中推导出实际行为。然而,我当然更愿意找到一份详细的文档,并认为我自己无法找到正确的引用资料。

最佳答案

signalfd 的行为与 sigwaitinfo 相同,只是您可以通过文件描述符访问信息。这意味着 signalfd 接收信号同步,并且信号处理程序(或默认配置)被调用首先

来源: TLPI 第 22.10 和 22.11 章(M. Kerrisk)。

行为定义明确,但未必符合预期,联机帮助页的措辞相当糟糕。说“通常...应该” 表明您实际上不必这样做,或者更糟的是作者不太确定。
如果你想让它“正常”工作,即你通常期望的方式(看,我也使用“正常”),你必须阻止信号。否则,信号将通过文件描述符可用,但仍将首先调用处理程序(这是完全合法的,但大多数人可能会认为这是“奇怪的行为”)。

因此,存在两种不同的竞争条件。一个条件是您询问的条件,但尽管该行为有点出乎意料,但它定义明确且(有点)记录在案,如果您考虑一下,那么它并不是严格意义上的竞争条件。而是一种“双重交付”。
另一个竞争条件是在您创建 signalfd 之后但在您阻止信号之前信号到达的可能性。这是不太可能的,但原则上,这可能会发生。幸运的是,解决方案很简单,您可以阻塞信号,然后创建文件描述符(如果中间有信号到达,文件描述符将立即准备就绪)。

您命名的命令序列(创建文件描述符、解锁,然后是 sigaction)具有类似的竞争条件。您应该先安装一个处理程序然后解除阻塞,否则可能会在处理程序到达之前传递信号。但这与 signalfd 无关。文件描述符在任何情况下仍可用于读取信号,但如果信号在解除阻塞和安装处理程序之间到达,则默认配置可能会终止进程。

关于linux - signalfd 和 sigaction 之间可以竞争吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20228092/

有关linux - signalfd 和 sigaction 之间可以竞争吗?的更多相关文章

  1. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

    类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

  2. ruby - 使用 Vim Rails,您可以创建一个新的迁移文件并一次性打开它吗? - 2

    使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta

  3. ruby - 我可以使用 Ruby 从 CSV 中删除列吗? - 2

    查看Ruby的CSV库的文档,我非常确定这是可能且简单的。我只需要使用Ruby删除CSV文件的前三列,但我没有成功运行它。 最佳答案 csv_table=CSV.read(file_path_in,:headers=>true)csv_table.delete("header_name")csv_table.to_csv#=>ThenewCSVinstringformat检查CSV::Table文档:http://ruby-doc.org/stdlib-1.9.2/libdoc/csv/rdoc/CSV/Table.html

  4. ruby-on-rails - Rails 应用程序之间的通信 - 2

    我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此

  5. ruby - 我可以使用 aws-sdk-ruby 在 AWS S3 上使用事务性文件删除/上传吗? - 2

    我发现ActiveRecord::Base.transaction在复杂方法中非常有效。我想知道是否可以在如下事务中从AWSS3上传/删除文件:S3Object.transactiondo#writeintofiles#raiseanexceptionend引发异常后,每个操作都应在S3上回滚。S3Object这可能吗?? 最佳答案 虽然S3API具有批量删除功能,但它不支持事务,因为每个删除操作都可以独立于其他操作成功/失败。该API不提供任何批量上传功能(通过PUT或POST),因此每个上传操作都是通过一个独立的API调用完成的

  6. ruby - #之间? Cooper 的 *Beginning Ruby* 中的错误或异常 - 2

    在Cooper的书BeginningRuby中,第166页有一个我无法重现的示例。classSongincludeComparableattr_accessor:lengthdef(other)@lengthother.lengthenddefinitialize(song_name,length)@song_name=song_name@length=lengthendenda=Song.new('Rockaroundtheclock',143)b=Song.new('BohemianRhapsody',544)c=Song.new('MinuteWaltz',60)a.betwee

  7. ruby - 有人可以帮助解释类创建的 post_initialize 回调吗 (Sandi Metz) - 2

    我正在阅读SandiMetz的POODR,并且遇到了一个我不太了解的编码原则。这是代码:classBicycleattr_reader:size,:chain,:tire_sizedefinitialize(args={})@size=args[:size]||1@chain=args[:chain]||2@tire_size=args[:tire_size]||3post_initialize(args)endendclassMountainBike此代码将为其各自的属性输出1,2,3,4,5。我不明白的是查找方法。当一辆山地自行车被实例化时,因为它没有自己的initialize方法

  8. ruby - 是否可以覆盖 gemfile 进行本地开发? - 2

    我们的git存储库中目前有一个Gemfile。但是,有一个gem我只在我的环境中本地使用(我的团队不使用它)。为了使用它,我必须将它添加到我们的Gemfile中,但每次我checkout到我们的master/dev主分支时,由于与跟踪的gemfile冲突,我必须删除它。我想要的是类似Gemfile.local的东西,它将继承从Gemfile导入的gems,但也允许在那里导入新的gems以供使用只有我的机器。此文件将在.gitignore中被忽略。这可能吗? 最佳答案 设置BUNDLE_GEMFILE环境变量:BUNDLE_GEMFI

  9. ruby-on-rails - `a ||= b` 和 `a = b if a.nil 之间的区别? - 2

    我正在检查一个Rails项目。在ERubyHTML模板页面上,我看到了这样几行:我不明白为什么不这样写:在这种情况下,||=和ifnil?有什么区别? 最佳答案 在这种特殊情况下没有区别,但可能是出于习惯。每当我看到nil?被使用时,它几乎总是使用不当。在Ruby中,很少有东西在逻辑上是假的,只有文字false和nil是。这意味着像if(!x.nil?)这样的代码几乎总是更好地表示为if(x)除非期望x可能是文字false。我会将其切换为||=false,因为它具有相同的结果,但这在很大程度上取决于偏好。唯一的缺点是赋值会在每次运行

  10. ruby - 我可以将我的 README.textile 以正确的格式放入我的 RDoc 中吗? - 2

    我喜欢使用Textile或Markdown为我的项目编写自述文件,但是当我生成RDoc时,自述文件被解释为RDoc并且看起来非常糟糕。有没有办法让RDoc通过RedCloth或BlueCloth而不是它自己的格式化程序运行文件?它可以配置为自动检测文件后缀的格式吗?(例如README.textile通过RedCloth运行,但README.mdown通过BlueCloth运行) 最佳答案 使用YARD直接代替RDoc将允许您包含Textile或Markdown文件,只要它们的文件后缀是合理的。我经常使用类似于以下Rake任务的东西:

随机推荐