0

方案 1

假设存在以下情况。

我的程序调用了第 3 方库,后者又调用了另一个第 3 方库,后者转身并进行操作系统调用。

   MyProgram
        |
   some3rdPartyFunction() ---> 3rd party library
                                 |
                             another3rdPartyFunction()----> another 3rd party library
                                                                      |
                                                                    OS call()

第一组问题:

  • 如果此 OS 调用对已关闭的套接字进行套接字调用,会发生什么?
  • 错误是否返回到进行操作系统调用的进程?
  • 系统在什么时候返回 SIGPIPE 消息?
  • SIGPIPE 是只发送到调用函数还是一直传播到 MyProgram?

方案 2

我碰巧看到一个进程的线程转储的以下输出。

rt_sigprocmask(SIG_SETMASK, ...)
rt_sigreturn({mask=[PIPE]})
rt_sigprocmask(SIG_SETMASK, [], NULL, 8) = 0
rt_sigaction(SIGINT, [], [], 8) = 0
rt_sigaction(SIGTERM, [], [], 8) = 0
rt_sigaction(SIGHUP, [], [], 8) = 0
rt_sigaction(SIGABRT, [], [], 8) = 0
rt_sigprocmask(SIG_SETMASK, [], [], 8) = 0
rt_sigreturn({mask=[PIPE]})

第二组问题:

  • 为什么要发送这么多信号消息?
  • 这些是直接发送到 MyProgram 还是从较低的操作系统调用向上传播?
  • 随着发送的信号种类繁多,如何确定究竟是哪一个导致了问题?
4

1 回答 1

2

如果此 OS 调用对已关闭的套接字进行套接字调用,会发生什么?

好吧,在谈论文件描述符(可能已关闭)和流套接字(如 TCP,可能部分或完全关闭)时,“关闭”可能会有点误导。我认为您要问的是POSIX记录为的write()调用错误

[EPIPE] 尝试在已关闭以进行写入或不再连接的套接字上进行写入。在后一种情况下,如果套接字是 SOCK_STREAM 类型,则还应向线程发送 SIGPIPE 信号。

错误是否返回到进行操作系统调用的进程?

错误会返回给任何被调用的代码write,并为最终调用write.

系统在什么时候返回 SIGPIPE 消息?

尝试写入不可写的套接字或管道时会生成 SIGPIPE。

SIGPIPE 是只发送到调用函数还是一直传播到 MyProgram?

SIGPIPE 是为调用线程生成的。它在调用链中的位置——您的程序、第 3 方库等——与信号传递无关。

为什么要发送这么多信号消息?

您所显示的内容均未发送。这些电话中的大多数都是关于屏蔽和信号处理,而不是信号生成。这些rt_sigreturn调用告诉您Linux 正在处理 SIGPIPE 的传递

这些是直接发送到 MyProgram 还是从较低的操作系统调用向上传播?

我不明白这是什么意思。SIGPIPE 是为您进程中的特定线程生成的,无论您在 MyProgram 中的代码是否知道该线程的存在,或者第 3 方库是否“秘密”创建了该线程。

随着发送的信号种类繁多,如何确定究竟是哪一个导致了问题?

在您展示的内容中只生成一个信号,那就是 SIGPIPE。

于 2021-05-05T16:01:00.367 回答