信号处理程序在我的C http服务器上不工作

时间:2016-11-04 00:14:00

标签: c linux signals httpserver

问题:当我按下ctrl C时,我希望我的单个处理程序按预期工作并打印出“EXITED NICELY”。这是一个赋值,我们必须使用信号处理程序。正如你所看到的,我也尝试了sigaction,但结果相同。

当前行为:“工作”正在打印出来,表明信号处理程序正在运行,但它必须卡在某处,因为它不会取消程序。虽然如果按ctrl c然后向服务器发送http请求,例如curl http:/ localhost:port / file.name,它将正常退出并打印出我想要的消息。但是,如果没有我发送请求,我希望它能做到这一点。

通过进一步研究进行编辑。我接受电话之前和之后都打印了一份。打印之前将打印一次然后接受将只保持它直到它收到连接。那么问题在哪里,但我们如何解决?

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<sys/socket.h>
#include<arpa/inet.h>
#include<netdb.h>
#include<signal.h>
#include<fcntl.h>
#include "http_common.h"
#define CONNMAX 1000
#define BYTES 1024

char *ROOT;
int verbose;
int signalReceived = 1;
int listenfd, clients[CONNMAX];
void error(char *);

static void clean(int arg)
{
        if(arg == SIGINT) {
                printf("work\n");
                signalReceived = 0;
                //signal(SIGINT, clean);
        }
        else if(arg == SIGHUP) {
                signalReceived = 0;
        }
}

int main(int argc, char *argv[])
{
        //struct sigaction act;
        //memset (&act, '\0', sizeof(act));
        //act.sa_handler = clean;
        //sigemptyset(&act.sa_mask);
        //act.sa_flags = SA_RESTART;
        signal(SIGINT, clean);
        //signal(SIGHUP, clean);
        //if (sigaction(SIGINT, &act, NULL) == -1)
        //      printf("doing something\n");
        struct sockaddr_in clientaddr;
        socklen_t addrlen;
        char c;
        char PORT[6];
        ROOT = getenv("PWD");
        strcpy(PORT, "8888");
        int slot;
        while ((c = getopt (argc, argv, "p:v")) != -1)

                switch (c) {
                case'v':
                        verbose = 1;
                        break;
                case'p':
                        strcpy(PORT, optarg);
                        break;
                case'?':
                        fprintf(stderr, "Wrong arguments given\n");
                        exit(1);
                default:
                        exit(1);
                }
        printf("Listening on port %s%s%s, root is %s%s%s\n", "\033[92m", PORT, "\033[0m", "\033[92m", ROOT, "\033[0m");
        int i = 0;
        for (i = 0; i < CONNMAX; i++)
                clients[i] = -1;
        startServer(PORT, &listenfd);
        while (signalReceived == 1) {
                addrlen = sizeof(clientaddr);
                clients[slot] = accept (listenfd, (struct sockaddr *) &clientaddr, &addrlen);
                if (clients[slot] < 0)
                        exit(0);
                else {
                        if (fork() == 0) {
                                respond(slot, verbose, ROOT, clients);
                             exit(0);
                        }
                }
                while (clients[slot] != -1)
                        slot = (slot + 1) % CONNMAX;

        }
        printf("EXITED NICLEY\n");//ients[slot] = accept (listenfd, (struct sockaddr *) &clientaddr, &addrlen);
        return 0;
}

1 个答案:

答案 0 :(得分:1)

您必须使用sigaction()并且没有SA_RESTART标记来注册信号。

当您使用signal()注册信号处理程序时,它将设置SA_RESTART标志。见the glibc manual

  

在GNU C库中,使用信号建立处理程序会将所有标志设置为零,但SA_RESTART除外,其值取决于您使用siginterrupt进行的设置。请参阅Interrupted Primitives,了解这是什么。

设置SA_RESTART时,信号将中断(大多数)系统调用,但它会重新启动它们。见the signal man page

  

如果在系统调用或库函数调用被阻止时调用信号处理程序,则:

     
      
  • 在信号处理程序返回后自动重新启动调用;或
  •   
  • 呼叫失败并显示错误EINTR。
  •   
     

这两种行为中的哪一种取决于接口以及是否使用SA_RESTART标志建立了信号处理程序(参见sigaction(2))。