Linux 是否允许从信号处理程序进行任何系统调用?

2024-11-04 08:43:00
admin
原创
30
摘要:问题描述:我的理解是,一般来说,如果从信号处理程序调用非异步信号安全函数,则行为是未定义的,但我听说 Linux 允许您安全地调用任何系统调用。这是真的吗?此外,SIGSEGV 处理程序的唯一可移植行为是中止或退出,但我知道如果您返回,Linux 实际上会恢复执行,对吗?解决方案 1:根据第 2 节signa...

问题描述:

我的理解是,一般来说,如果从信号处理程序调用非异步信号安全函数,则行为是未定义的,但我听说 Linux 允许您安全地调用任何系统调用。这是真的吗?此外,SIGSEGV 处理程序的唯一可移植行为是中止或退出,但我知道如果您返回,Linux 实际上会恢复执行,对吗?


解决方案 1:

根据第 2 节signal手册:

请参阅 signal(7) 以获取可以从信号处理程序内部安全调用的异步信号安全函数列表。

手册第 7 节signals列出了以下函数和/或系统调用以及非常清晰的描述:

异步信号安全函数

   A signal handler function must be very careful, since processing elsewhere may
   be interrupted at some arbitrary point in the execution of the program.  POSIX
   has the concept of "safe function".  If a signal interrupts the execution of
   an unsafe function, and handler calls an unsafe function, then the behavior of
   the program is undefined.

   POSIX.1-2004 (also known as POSIX.1-2001 Technical Corrigendum 2) requires an
   implementation to guarantee that the following functions can be safely called
   inside a signal handler:

       _Exit()
       _exit()
       abort()
       accept()
       access()
       aio_error()
       aio_return()
       aio_suspend()
       alarm()
       bind()
       cfgetispeed()
       cfgetospeed()
       cfsetispeed()
       cfsetospeed()
       chdir()
       chmod()
       chown()
       clock_gettime()
       close()
       connect()
       creat()
       dup()
       dup2()
       execle()
       execve()
       fchmod()
       fchown()
       fcntl()
       fdatasync()
       fork()
       fpathconf()
       fstat()
       fsync()
       ftruncate()
       getegid()
       geteuid()
       getgid()
       getgroups()
       getpeername()
       getpgrp()
       getpid()
       getppid()
       getsockname()
       getsockopt()
       getuid()
       kill()
       link()
       listen()
       lseek()
       lstat()
       mkdir()
       mkfifo()
       open()
       pathconf()
       pause()
       pipe()
       poll()
       posix_trace_event()
       pselect()
       raise()
       read()
       readlink()
       recv()
       recvfrom()
       recvmsg()
       rename()
       rmdir()
       select()
       sem_post()
       send()
       sendmsg()
       sendto()
       setgid()
       setpgid()
       setsid()
       setsockopt()
       setuid()
       shutdown()
       sigaction()
       sigaddset()
       sigdelset()
       sigemptyset()
       sigfillset()
       sigismember()
       signal()
       sigpause()
       sigpending()
       sigprocmask()
       sigqueue()
       sigset()
       sigsuspend()
       sleep()
       sockatmark()
       socket()
       socketpair()
       stat()
       symlink()
       sysconf()
       tcdrain()
       tcflow()
       tcflush()
       tcgetattr()
       tcgetpgrp()
       tcsendbreak()
       tcsetattr()
       tcsetpgrp()
       time()
       timer_getoverrun()
       timer_gettime()
       timer_settime()
       times()
       umask()
       uname()
       unlink()
       utime()
       wait()
       waitpid()
       write()

   POSIX.1-2008 removes fpathconf(), pathconf(), and sysconf() from the above
   list, and adds the following functions:

       execl()
       execv()
       faccessat()
       fchmodat()
       fchownat()
       fexecve()
       fstatat()
       futimens()
       linkat()
       mkdirat()
       mkfifoat()
       mknod()
       mknodat()
       openat()
       readlinkat()
       renameat()
       symlinkat()
       unlinkat()
       utimensat()
       utimes()

我相信这些信息比我们有时在某处听到的东西更可靠。因此,Linux 确实只允许某些系统调用,但不是全部。所以你的问题的答案很简单——不。

解决方案 2:

是与否

是的:

您可以在信号处理程序中调用任何真实/原始系统调用。内核有责任确保其安全性(从内核的角度来看)。

1) 内核不知道用户空间的上下文,或者说内核在传递信号时将状态保存到用户空间后故意忘记了它。(注意:执行恢复是由用户在保存的状态的帮助下通过系统调用完成的,而不是由内核完成的,内核已经忘记了)

2)一些线程库是通过单线程实现的,因此线程已经在“信号处理程序”中,但这些线程可以调用任何系统调用。

不:

但是用户空间函数有自己的目的和副作用。有些不是可重入安全的,这些函数不能从信号处理程序调用。man 7 signal将帮助您找出哪些是可重入安全的。

举个例子,您可以在任何地方调用sys_futex(),包括信号处理程序,但如果您使用它sys_futex()来实现互斥锁,sys_futex()则当信号中断互斥锁的临界部分时,内部的信号处理程序可能会永远被阻塞。

此外,SIGSEGV 处理程序的唯一可移植行为是中止或退出,但我知道如果您返回,linux 实际上会恢复执行,对吗?

是的,如果你找不到原因。有些用户可能会将 SIGSEGV 用于他们自己的按需映射目的(例如,在 JIT 中,你可以翻译 SIGSEGV 信号处理程序中的代码,并将翻译后的代码 mmap 到内存然后返回),他们可以调用 mmap() 或 mprotect() 等。

解决方案 3:

我相信任何真正的系统调用都可以从信号处理程序中调用。真正的系统调用在<asm/unistd.h>(或<asm/unistd_64.h>) 中有一个数字。

手册页第 2 部分中的一些 posix 函数是通过“多路复用”系统调用实现的,因此在我看来它们不是“真正的系统调用”

从应用程序的角度来看,系统调用是一个原子操作;它几乎就像一条机器指令(从应用程序内部来看)。请参阅此答案。

如果您的问题是:处理程序*是否可以通过或SIGSEGV更改错误的地址映射?mprotect`mmap*那么我相信答案是*肯定的*(至少在 x86-64 和 x86-32 架构上),正如您引用的问题中所述,但我没有尝试。我读到这样做效率很低(SIGSEGV处理速度不是很快,并且mprotectmmap`也有点慢)。特别是,以这种方式模仿Hurd/Mach 外部寻呼机可能效率低下。

相关推荐
  为什么项目管理通常仍然耗时且低效?您是否还在反复更新电子表格、淹没在便利贴中并参加每周更新会议?这确实是耗费时间和精力。借助软件工具的帮助,您可以一目了然地全面了解您的项目。如今,国内外有足够多优秀的项目管理软件可以帮助您掌控每个项目。什么是项目管理软件?项目管理软件是广泛行业用于项目规划、资源分配和调度的软件。它使项...
项目管理软件   601  
  华为IPD与传统研发模式的8大差异在快速变化的商业环境中,产品研发模式的选择直接决定了企业的市场响应速度和竞争力。华为作为全球领先的通信技术解决方案供应商,其成功在很大程度上得益于对产品研发模式的持续创新。华为引入并深度定制的集成产品开发(IPD)体系,相较于传统的研发模式,展现出了显著的差异和优势。本文将详细探讨华为...
IPD流程是谁发明的   7  
  如何通过IPD流程缩短产品上市时间?在快速变化的市场环境中,产品上市时间成为企业竞争力的关键因素之一。集成产品开发(IPD, Integrated Product Development)作为一种先进的产品研发管理方法,通过其结构化的流程设计和跨部门协作机制,显著缩短了产品上市时间,提高了市场响应速度。本文将深入探讨如...
华为IPD流程   9  
  在项目管理领域,IPD(Integrated Product Development,集成产品开发)流程图是连接创意、设计与市场成功的桥梁。它不仅是一个视觉工具,更是一种战略思维方式的体现,帮助团队高效协同,确保产品按时、按质、按量推向市场。尽管IPD流程图可能初看之下显得错综复杂,但只需掌握几个关键点,你便能轻松驾驭...
IPD开发流程管理   8  
  在项目管理领域,集成产品开发(IPD)流程被视为提升产品上市速度、增强团队协作与创新能力的重要工具。然而,尽管IPD流程拥有诸多优势,其实施过程中仍可能遭遇多种挑战,导致项目失败。本文旨在深入探讨八个常见的IPD流程失败原因,并提出相应的解决方法,以帮助项目管理者规避风险,确保项目成功。缺乏明确的项目目标与战略对齐IP...
IPD流程图   8  
热门文章
项目管理软件有哪些?
云禅道AD
禅道项目管理软件

云端的项目管理软件

尊享禅道项目软件收费版功能

无需维护,随时随地协同办公

内置subversion和git源码管理

每天备份,随时转为私有部署

免费试用