在C中处理信号时调用进程的pid不正确

时间:2016-03-27 21:08:21

标签: c linux signals pid

我真的不知道我到底做错了什么。我想从输入信号中提取呼叫者的pid,但我获得的值完全不正确。

这是我的“捕手”代码:

#include <stdio.h>
#include <stdlib.h>
#include <signal.h>
#include <unistd.h>

int SIGNALS_RECEIVED = 0;
pid_t CALLING_PID;

void signal_received(int sig, siginfo_t *info, void *context) {
    SIGNALS_RECEIVED++;

    if(SIGNALS_RECEIVED == 1) {
        CALLING_PID = info->si_pid;
        printf("%ld\n", (long) CALLING_PID);
    }
}

int main() {
    struct sigaction act;
    act.sa_sigaction = &signal_received;

    sigaction(SIGUSR1, &act, NULL);

    while(1) {

    }

    return 0;
}

和“发件人”:

#include <stdio.h>
#include <stdlib.h>
#include <signal.h>
#include <unistd.h>

int main(int argc, char **argv) {

    char line[10];
    FILE *cmd = popen("pidof -s catcher", "r");

    fgets(line, 10, cmd);
    pid_t pid = strtoul(line, NULL, 10);

    pclose(cmd);

    int i;
    for(i = 0; i < 400; i++) {
        kill(pid, SIGUSR1);
    }

    kill(pid, SIGUSR2);

    return 0;
}

因此,当第一个捕手,然后是发件人,我得到:

./catcher 
398533948
SIGNALS_RECEIVED: 24

发件人的pid是:

ps aux | grep *sender
maciej    4704  100  0.0   4328  1268 pts/13   R+   22:46   0:15 ./sender

我的Linux版本:

Linux version 4.2.0-34-generic (buildd@lgw01-55) (gcc version 4.8.2 (Ubuntu 4.8.2-19ubuntu1) ) #39~14.04.1-Ubuntu SMP Fri Mar 11 11:38:02 UTC 2016

1 个答案:

答案 0 :(得分:0)

阅读sigaction()的手册页:

  

sa_handler指定与signum关联的操作...此函数接收信号编号作为其唯一参数。

这不是你想要的。你想要这个:

  

如果SA_SIGINFO中指定了sa_flags,则sa_sigaction(而不是sa_handler)指定signum的信号处理函数。此函数接收信号编号作为其第一个参数,指向siginfo_t作为其第二个参数...

安装处理程序时未设置SA_SIGINFO标志。

在调用sigaction之前,在代码中添加以下行:

act.sa_flags = SA_SIGINFO;