Linuxシステムコール、シグナルの使い方

UNIX / Linux システムコール・プログラミング

プロセス間通信(IPC)、シグナル

シグナルとは?


プロセスに対して非同期のイベント通知を行います。

プロセス側でシグナルハンドラを定義することで、非同期通知を受けることができます。

Ctrl+C で、動作中のコマンド(プロセス)を停止できますが、Ctrl+C は SIGINT というシグナルを発生させてプロセスの停止を行っています。

また、シグナルは SYSTEM V IPC です。


どのようにシグナルを使うか、下記にサンプルコードを示します。

このサンプルコードでは、 SIGINT と SIGTSTP の2つを受け取るシグナルハンドラを記述しています。

シグナルとシグナルハンドラは、sigaction()システムコールを使って設定されます。
同様のシステムコールとして、signal()が存在しますが、現代ではsigaction()の利用が推奨されています。

また、実装バージョンやアーキテクチャによっても動作や構造体が異なる場合がありますので注意してください。

実行した状態で、Ctrl + c(SIGINT) を押してもメッセージは表示されますがプログラムは停止しません。
Ctrl + z(SIGTSTP) を押すとプログラムが停止するようになっています。

$ ./signal
I'm working...
^CCatch SIGNAL[2]    <-- Ctrl + c key in.
Work harder!              <-- Signal handler message. Don’t stop.
I'm working...
I'm working...
^ZCatch SIGNAL[20]   <-- Ctrl + z key in.
We have a break.        <-- Signal handler message. It’s goiung stop.
I've had enough!
$

signal.c

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <signal.h>
  4. #include <string.h>
  5. #include <unistd.h>
  6.  
  7. static void signalHandler(int sig) ;
  8.  
  9. static int doit;
  10.  
  11. int main(int argc, char** argv) {
  12.  
  13. struct sigaction act;
  14.  
  15. act.sa_handler = &signalHandler;
  16. act.sa_flags = 0;
  17. sigfillset(&act.sa_mask);
  18.  
  19. if (sigaction(SIGINT, &act, NULL)<0) {
  20. perror("SIGINT sigaction()");
  21. exit(-1);
  22. }
  23. if (sigaction(SIGTSTP, &act, NULL)<0) {
  24. perror("SIGTSTP sigaction()");
  25. exit(-1);
  26. }
  27.  
  28. doit = 1;
  29.  
  30. while(doit) {
  31. printf("I'm working...\n");
  32. sleep(1);
  33. }
  34.  
  35. printf("I've had enough!\n");
  36. return(0);
  37. }
  38.  
  39. static void signalHandler(int sig) {
  40.  
  41. printf("Catch SIGNAL[%d]\n", sig);
  42.  
  43. switch(sig) {
  44. case SIGINT:
  45. printf("Work harder!\n");
  46. break;
  47. case SIGTSTP:
  48. printf("We have a break.\n");
  49. doit = 0; // Quit process working
  50. break;
  51. default:
  52. printf("I don't know.\n");
  53. }
  54. }




今回のサンプルでは、SIGINTとSIGTSTPと、役割が決まっているシグナルを使って処理しています。
しかも、通常ではSIGINTは強制終了ですが終了せず、SIGTSTPは通常は一時停止ですが強制終了してしまうという変則的な動作を定義しています。

では、システム設計上自由にシグナルを使うことができないんじゃないか?
と、思われるかもしれませんが、SIGUSR1とSIGUSR2という2つのシグナルがあり、プログラムで自由に使うことができます。

実際に使われている例としては、ddコマンドに実装されています。
ddコマンドで、HDDのゼロクリアなどを行うとき、
  $ dd if=/dev/zero of=/dev/sda
のようにしますが、
容量が大きなHDDを初期化するときなど非常に時間がかかります。
このとき、killコマンドで動作中のddコマンドに SIGUSR1 を送ると、動作状況を表示するようになっています。

このように、プログラムで自由な目的に使いたい場合はこの SUGUSR1とSIGUSR2を利用するべきですね。


シグナルの定義:
実際にシグナルはドコに定義されているのか?
今回も、BeagleBoneの開発環境で説明します。

gcc-linaro-7.4.1-2019.02-x86_64_arm-linux-gnueabihfの環境では下記の場所に定義されています。

arm-linux-gnueabihf/libc/usr/include/bits/signum.h

  1. /* Signal number definitions. Linux version.
  2. Copyright (C) 1995-2017 Free Software Foundation, Inc.
  3. This file is part of the GNU C Library.
  4.  
  5. The GNU C Library is free software; you can redistribute it and/or
  6. modify it under the terms of the GNU Lesser General Public
  7. License as published by the Free Software Foundation; either
  8. version 2.1 of the License, or (at your option) any later version.
  9.  
  10. The GNU C Library is distributed in the hope that it will be useful,
  11. but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  13. Lesser General Public License for more details.
  14.  
  15. You should have received a copy of the GNU Lesser General Public
  16. License along with the GNU C Library; if not, see
  17. . */
  18. #ifdef _SIGNAL_H
  19. /* Fake signal functions. */
  20. #define SIG_ERR ((__sighandler_t) -1) /* Error return. */
  21. #define SIG_DFL ((__sighandler_t) 0) /* Default action. */
  22. #define SIG_IGN ((__sighandler_t) 1) /* Ignore signal. */
  23. #ifdef __USE_UNIX98
  24. # define SIG_HOLD ((__sighandler_t) 2) /* Add signal to hold mask. */
  25. #endif
  26. /* Signals. */
  27. #define SIGHUP 1 /* Hangup (POSIX). */
  28. #define SIGINT 2 /* Interrupt (ANSI). */
  29. #define SIGQUIT 3 /* Quit (POSIX). */
  30. #define SIGILL 4 /* Illegal instruction (ANSI). */
  31. #define SIGTRAP 5 /* Trace trap (POSIX). */
  32. #define SIGABRT 6 /* Abort (ANSI). */
  33. #define SIGIOT 6 /* IOT trap (4.2 BSD). */
  34. #define SIGBUS 7 /* BUS error (4.2 BSD). */
  35. #define SIGFPE 8 /* Floating-point exception (ANSI). */
  36. #define SIGKILL 9 /* Kill, unblockable (POSIX). */
  37. #define SIGUSR1 10 /* User-defined signal 1 (POSIX). */
  38. #define SIGSEGV 11 /* Segmentation violation (ANSI). */
  39. #define SIGUSR2 12 /* User-defined signal 2 (POSIX). */
  40. #define SIGPIPE 13 /* Broken pipe (POSIX). */
  41. #define SIGALRM 14 /* Alarm clock (POSIX). */
  42. #define SIGTERM 15 /* Termination (ANSI). */
  43. #define SIGSTKFLT 16 /* Stack fault. */
  44. #define SIGCLD SIGCHLD /* Same as SIGCHLD (System V). */
  45. #define SIGCHLD 17 /* Child status has changed (POSIX). */
  46. #define SIGCONT 18 /* Continue (POSIX). */
  47. #define SIGSTOP 19 /* Stop, unblockable (POSIX). */
  48. #define SIGTSTP 20 /* Keyboard stop (POSIX). */
  49. #define SIGTTIN 21 /* Background read from tty (POSIX). */
  50. #define SIGTTOU 22 /* Background write to tty (POSIX). */
  51. #define SIGURG 23 /* Urgent condition on socket (4.2 BSD). */
  52. #define SIGXCPU 24 /* CPU limit exceeded (4.2 BSD). */
  53. #define SIGXFSZ 25 /* File size limit exceeded (4.2 BSD). */
  54. #define SIGVTALRM 26 /* Virtual alarm clock (4.2 BSD). */
  55. #define SIGPROF 27 /* Profiling alarm clock (4.2 BSD). */
  56. #define SIGWINCH 28 /* Window size change (4.3 BSD, Sun). */
  57. #define SIGPOLL SIGIO /* Pollable event occurred (System V). */
  58. #define SIGIO 29 /* I/O now possible (4.2 BSD). */
  59. #define SIGPWR 30 /* Power failure restart (System V). */
  60. #define SIGSYS 31 /* Bad system call. */
  61. #define SIGUNUSED 31
  62. #define _NSIG 65 /* Biggest signal number + 1
  63. (including real-time signals). */
  64. #define SIGRTMIN (__libc_current_sigrtmin ())
  65. #define SIGRTMAX (__libc_current_sigrtmax ())
  66. /* These are the hard limits of the kernel. These values should not be
  67. used directly at user level. */
  68. #define __SIGRTMIN 32
  69. #define __SIGRTMAX (_NSIG - 1)
  70. #endif /* included. */

Have a Happy Hucking!!



Lightning Brains

コメント

このブログの人気の投稿

Linuxシステムコール、メッセージキューの使い方

Linuxシステムコール、共有メモリの使い方

Linuxシステムコール、セマフォの使い方