プロセスグループにSIGINTを送信することが時々無視されることがあります。

プロセスグループにSIGINTを送信することが時々無視されることがあります。

Bashでプロセスグループを起動します。その後、プロセスグループ全体に SIGINT を送信します。時々SIGINTはプロセスを終了しますが、時にはそうではありません。 SIGINTが時々無視されるのはなぜですか?

プロセスグループがバックグラウンドで起動されたか、bashシェルのネスト、Mac / Linuxオペレーティングシステムによって異なる動作が表示されます。誰もがこれについて明らかにすることができれば非常に感謝します。

次の例では、Python実行可能ファイルを使用しています。sleep_in_pgrp.py

#!/usr/bin/env python2.7
import os;
import subprocess
os.setpgrp();
subprocess.check_call(["sleep","10000"]);

プロセスグループを作成し、スリープモードを開始します。観察された現象はPythonとは関係ありません。私はbashにコマンドや組み込み機能がないので、Pythonを使用していますsetpgrp修正する:明らかに対話型シェルを実行して新しいプロセスグループ

1) バックグラウンドでプロセスグループを起動し、リーダーを待ちます。 SIGINT は無視されます。

次のコマンドを実行します。

$ bash -c '  { sleep_in_pgrp.py; } & wait $!  '

BashはバックグラウンドでPythonを起動して待ちます。他の端末から:

$ ps -Heo pid,ppid,tpgid,pgid,sid,user,args
   PID   PPID  TPGID   PGID    SID     COMMAND
  2507   1574   2963   2507   2507     -bash
  2963   2507   2963   2963   2507       bash -c   { sleep_in_pgrp.py; } & wait $!
  2964   2963   2963   2963   2507         bash -c   { sleep_in_pgrp.py; } & wait $!
  2965   2964   2963   2965   2507           python2.7 ./sleep_in_pgrp.py
  2966   2965   2963   2965   2507             sleep 10000

SIGINT'ing Pythonのプロセスグループプロセスを終了しないでください。なぜですか?

$ sudo kill -s SIGINT -- -2965

2) フォアグラウンドでプロセスグループを起動します。信号知能が機能します。

を削除すると、& wait $!SIGINTは期待どおりにプロセスグループを終了します。理由はわかりませんが、この場合、SIGINTがプロセスを終了したという事実は驚くべきことではありません。

$ bash -c '  { sleep_in_pgrp.py; }  '

他の端末から:

$ ps -Heo pid,ppid,tpgid,pgid,sid,user,args
   PID   PPID  TPGID   PGID    SID     COMMAND
  2507   1574   3352   2507   2507     -bash
  3352   2507   3352   3352   2507       bash -c   { sleep_in_pgrp.py; }
  3353   3352   3352   3353   2507         python2.7 ./sleep_in_pgrp.py
  3354   3353   3352   3353   2507           sleep 10000

SIGINT はプロセスグループを終了します。

$ sudo kill -s SIGINT -- -3353

3)バックグラウンドでPythonを実行している間にサブシェルを削除します。信号知能が機能します。

シェルの入れ子がここでの動作に影響を与えるという事実に非常に驚きました。私は説明を考えることができません。

bash -c最初に削除しました。

$ { sleep_in_pgrp.py; } & wait $!

他の端末から:

$ ps -Heo pid,ppid,tpgid,pgid,sid,user,args
   PID   PPID  TPGID   PGID    SID     COMMAND
  2507   1574   2507   2507   2507     -bash
  3488   2507   2507   3488   2507       -bash
  3489   3488   2507   3489   2507         python2.7 ./sleep_in_pgrp.py
  3490   3489   2507   3489   2507           sleep 10000

SIGINT はプロセスグループを終了します。

$ sudo kill -s SIGINT -- -2507

4)Macで最初のコマンドを実行します。 SIGINTが動作します。

最初の2つのコマンドはCentOs7仮想マシンで実行されます。

$ uname -a
Linux ip-10-229-193-124 3.10.0-693.5.2.el7.x86_64 #1 SMP Fri Oct 13 10:46:25 EDT 2017 x86_64 x86_64 x86_64 GNU/Linux

これで、Macのサブシェルで背景Pythonを使用して最初のコマンドを実行します。

$ uname -a
Darwin mbp-005063 15.6.0 Darwin Kernel Version 15.6.0: Sun Jun  4 21:43:07 PDT 2017; root:xnu-3248.70.3~1/RELEASE_X86_64 x86_64

Macの場合:

$ bash -c '  { sleep_in_pgrp.py; } & wait $!  '

他の端末から:

$   PID  PPID TPGID  PGID   SESS COMMAND
18741 40096 18741 18741      0 bash -c   { sleep_in_pgrp.py; } & wait $!
18742 18741 18741 18741      0 bash -c   { sleep_in_pgrp.py; } & wait $!
18743 18742 18741 18743      0 /usr/local/Cellar/python/2.7.12_2/Frameworks/Python.framework/Versions/2.7/Resources/Python.app/Contents/MacOS/Python ./sleep_in_pgrp.py
18744 18743 18741 18743      0 sleep 10000
40094  2423 18741 40094      0 /Applications/iTerm.app/Contents/MacOS/iTerm2 --server /usr/bin/login -fpl hbaba /Applications/iTerm.app/Contents/MacOS/iTerm2 --launch_shell
40095 40094 18741 40095      0 /usr/bin/login -fpl hbaba /Applications/iTerm.app/Contents/MacOS/iTerm2 --launch_shell
40096 40095 18741 40096      0 -bash
-+= 00001 root /sbin/launchd
 \-+= 02423 hbaba /Applications/iTerm.app/Contents/MacOS/iTerm2
   \-+= 40094 hbaba /Applications/iTerm.app/Contents/MacOS/iTerm2 --server /usr/bin/login -fpl hbaba /Applications/iTerm.app/Contents/MacOS/iTerm2 --launch_shell
     \-+= 40095 root /usr/bin/login -fpl hbaba /Applications/iTerm.app/Contents/MacOS/iTerm2 --launch_shell
       \-+= 40096 hbaba -bash
         \-+= 18741 hbaba bash -c   { sleep_in_pgrp.py; } & wait $!
           \-+- 18742 hbaba bash -c   { sleep_in_pgrp.py; } & wait $!
             \-+= 18743 hbaba /usr/local/Cellar/python/2.7.12_2/Frameworks/Python.framework/Versions/2.7/Resources/Python.app/Contents/MacOS/Python ./sleep_in_pgrp.py
               \--- 18744 hbaba sleep 10000

この場合、SIGINT はプロセスグループも終了します。

$ sudo kill -s INT -18743

CentOs7のBashバージョンは次のとおりです。

$ echo $BASH_VERSION
4.2.46(2)-release

Macでは、bashのバージョン

$ echo $BASH_VERSION
4.4.12(1)-release

これ回答説明+プロセスグループにSIGINTを送信するCtrl方法。CこれがプロセスグループにSIGINTを送信し、ここでやりたいことです。これ回答非対話型タスクにはSIGINTハンドラが必要であると述べました。私が見ている他の行動を説明することは不明です。また、バックグラウンドプロセスを待つことがそのプロセスのSIGINT処理に影響を与えるかどうかを知りたいです。

答え1

このプロセスは、他の機能で使用するためにSIGINT信号を取り込む可能性が高いです。

受信した信号が不明な場合にのみプロセスは終了します。ただし、プロセスが信号に接続されている機能を設定すると、信号に接続された機能はプログラムが完了するまで死ぬことはありません。

たとえば、単純なCプログラムは次のようになります。

 #include <signal.h>

 int sigreceived = 0;

 void mysignal();

 int main(){
    signal(2, mysignal); //SIGINT corresponds to 2 signal

    while(1);
    return 0;
  }

 void mysignal(){
  sigreceived=1;
 }

このプログラムでシグナル2をキャッチすることは、mysignal関数または他のものを呼び出すことです。プロセスを終了するのではなく、変数の値を変更するだけです。

これにより、このプロセスはSIGINTによって終了しません。

答え2

kill -INT pid""または" kill -INT -- -pgid"または-によってCtrl送信されたSIGINTは、C常にフォアグラウンドプロセス(フォアグラウンドプロセスグループ)に影響します。 &で非同期的に開始されたすべてのエントリはINTの影響を受けません(コマンドの置き換えなど、バックグラウンドで実行されるサブシェルを除く)(cmd;cmd, ...etc ) &$(cmd) &)。

忘れないでください:bashはWCE(Wait and Cooperative Exit)を実装します。つまり、親プロセスはSIGINTを介して子プロセスが完了するのを待ち、子プロセスがSIGINTを介して終了したCtrl場合Cにのみ終了することを決定します。

私より良い説明はhttps://www.cons.org/cracauer/sigint.html

関連情報