UnixシグナルハンドラからQt関数を呼び出す

Unix のシグナルハンドラから Qt の関数を呼び出すことはできません。標準の POSIX ルールが適用されます:シグナルハンドラから呼び出せるのは非同期シグナルセーフな関数だけです。Unix のシグナルハンドラから呼び出せる関数の完全なリストはSignal Actions を参照してください。

しかし、QtでUnixシグナルハンドラを使う方法はあります。その方法とは、UnixシグナルハンドラにQtシグナルを発生させるような処理をさせ、Unixシグナルハンドラからリターンすることです。Qt プログラムに戻ると、Qt シグナルが発信され、Qt スロット関数で受信されます。

これを実現する1つの簡単な方法は、扱いたいUnixシグナルごとにクラス内でソケットペアを宣言することです。ソケット・ペアは静的データ・メンバとして宣言する。また、各ソケットペアのリードエンドを監視するためにQSocketNotifier を作成し、Unix シグナルハンドラを静的クラスメソッドとして宣言し、Unix シグナルハンドラそれぞれに対応するスロット関数を宣言します。この例では、SIGHUPシグナルとSIGTERMシグナルの両方を処理する。注意: 以下のコードを読み進める前に、socketpair(2) と sigaction(2) のマニュアルページを読む必要があります。

class MyDaemon : public QObject
{
    Q_OBJECT

  public:
    MyDaemon(QObject *parent = 0);
    ~MyDaemon();

    // Unix signal handlers.
    static void hupSignalHandler(int unused);
    static void termSignalHandler(int unused);

  public slots:
    // Qt signal handlers.
    void handleSigHup();
    void handleSigTerm();

  private:
    static int sighupFd[2];
    static int sigtermFd[2];

    QSocketNotifier *snHup;
    QSocketNotifier *snTerm;
};

MyDaemon コンストラクタで、各ファイル記述子のペアを初期化するために socketpair(2) 関数を使用し、各ペアの読み取り終了を監視するためにQSocketNotifier を作成する。各QSocketNotifier のactivated() シグナルは、効果的にUnixシグナルをQSocketNotifier::activated() シグナルに変換する適切なスロット関数に接続されます。

MyDaemon::MyDaemon(QObject *parent)
        : QObject(parent)
{
    if (::socketpair(AF_UNIX, SOCK_STREAM, 0, sighupFd))
       qFatal("Couldn't create HUP socketpair");

    if (::socketpair(AF_UNIX, SOCK_STREAM, 0, sigtermFd))
       qFatal("Couldn't create TERM socketpair");
    snHup = new QSocketNotifier(sighupFd[1], QSocketNotifier::Read, this);
    connect(snHup, SIGNAL(activated(QSocketDescriptor)), this, SLOT(handleSigHup()));
    snTerm = new QSocketNotifier(sigtermFd[1], QSocketNotifier::Read, this);
    connect(snTerm, SIGNAL(activated(QSocketDescriptor)), this, SLOT(handleSigTerm()));

    ...
}

起動コードの他のどこかで、sigaction(2) を使って Unix シグナルハンドラをインストールします。

static int setup_unix_signal_handlers()
{
    struct sigaction hup, term;

    hup.sa_handler = MyDaemon::hupSignalHandler;
    sigemptyset(&hup.sa_mask);
    hup.sa_flags = 0;
    hup.sa_flags |= SA_RESTART;

    if (sigaction(SIGHUP, &hup, 0))
       return 1;

    term.sa_handler = MyDaemon::termSignalHandler;
    sigemptyset(&term.sa_mask);
    term.sa_flags = 0;
    term.sa_flags |= SA_RESTART;

    if (sigaction(SIGTERM, &term, 0))
       return 2;

    return 0;
}

Unix シグナルハンドラでは、ソケットペアのライトエンドにバイトを書き込んでリターンする。これにより、対応するQSocketNotifier が activated() シグナルを発し、適切な Qt スロット関数が実行されます。

void MyDaemon::hupSignalHandler(int)
{
    char a = 1;
    ::write(sighupFd[0], &a, sizeof(a));
}

void MyDaemon::termSignalHandler(int)
{
    char a = 1;
    ::write(sigtermFd[0], &a, sizeof(a));
}

QSocketNotifier::activated()シグナルに接続されたスロット関数で、バイトを読み込みます。これで、安全にQtのシグナルに戻り、UnixのシグナルハンドラではできなかったQtの処理をすべて行うことができます。

void MyDaemon::handleSigTerm()
{
    snTerm->setEnabled(false);
    char tmp;
    ::read(sigtermFd[1], &tmp, sizeof(tmp));

    // do Qt stuff

    snTerm->setEnabled(true);
}

void MyDaemon::handleSigHup()
{
    snHup->setEnabled(false);
    char tmp;
    ::read(sighupFd[1], &tmp, sizeof(tmp));

    // do Qt stuff

    snHup->setEnabled(true);
}

ここに含まれる文書の著作権は、それぞれの所有者に帰属します 本書で提供されるドキュメントは、Free Software Foundation が発行したGNU Free Documentation License version 1.3に基づいてライセンスされています。 Qtおよびそれぞれのロゴは、フィンランドおよびその他の国におけるThe Qt Company Ltd.の 商標です。その他すべての商標は、それぞれの所有者に帰属します。