Exemplo n.º 1
0
Arquivo: main.c Projeto: huangxw/gx
int main(int argc, char *argv[])
{
    struct sigaction         act, old;
    struct addrinfo          ask, *res;
    struct sockaddr_storage  ss;
    int c, opt, rc, ss_len;
    char host[INET6_ADDRSTRLEN + 1];
    char serv[16];
    const char options[] = "hd" "p:";
    memset(&ask, 0, sizeof(ask));

    /* parse options */
    while(1) {
        if (-1 == (c = getopt(argc, argv, options))) {
            break;
        }

        switch (c) {
            case 'h':
                usage(argv[0]);
                break;
            case 'p':
                listen_port = optarg;
                break;
            default:
                exit(1);
        }
    }

    /* bind to socket */
    memset(&ask, 0, sizeof(ask));
    ask.ai_flags = AI_PASSIVE;

    if (listen_ip) {
        ask.ai_flags |= AI_CANONNAME;
    }

    ask.ai_socktype = SOCK_STREAM;
    ask.ai_family = PF_INET;

    if (0 != (rc = getaddrinfo(listen_ip, listen_port, &ask, &res))) {
        fprintf(stderr, "getaddrinfo (ipv4): %s\n", gai_strerror(rc));
        exit(1);
    }

    if (-1 == (slisten = socket(res->ai_family, res->ai_socktype, res->ai_protocol))) {
        exit(1);
    }

    if (-1 == slisten) {
        exit(1);
    }

    close_on_exec(slisten);
    memcpy(&ss, res->ai_addr, res->ai_addrlen);
    ss_len = res->ai_addrlen;

    if (0 != (rc = getnameinfo((struct sockaddr *)&ss, ss_len,
                               host, INET6_ADDRSTRLEN, serv, 15,
                               NI_NUMERICHOST | NI_NUMERICSERV))) {
        exit(1);
    }

    tcp_port = atoi(serv);
    opt = 1;
    setsockopt(slisten, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));
    fcntl(slisten, F_SETFL, O_NONBLOCK);

    if (-1 == bind(slisten, (struct sockaddr *) &ss, ss_len)) {
        exit(1);
    }

    if (-1 == listen(slisten, 2 * max_conn)) {
        exit(1);
    }

    init_quote();
    printf("gx start!\n\n");
#if defined(linux)
    printf("###############################\n");
    printf("HOST INFO:\n");
    host_info();
#endif
    printf("###############################\n");
    printf("SERVER INFO:\n");
    /* setup signal handler */
    memset(&act, 0, sizeof(act));
    sigemptyset(&act.sa_mask);
    act.sa_handler = SIG_IGN;
    sigaction(SIGPIPE, &act, &old);
    sigaction(SIGCHLD, &act, &old);
    act.sa_handler = catchsig;
    sigaction(SIGHUP, &act, &old);
    sigaction(SIGTERM, &act, &old);
    sigaction(SIGINT, &act, &old);

    /* go! */
    if (nthreads > 1) {
        int i;
        threads = malloc(sizeof(pthread_t) * nthreads);

        for (i = 1; i < nthreads; i++) {
            pthread_create(threads + i, NULL, mainloop, threads + i);
            pthread_detach(threads[i]);
        }
    }

    mainloop(NULL);
    fprintf(stderr, "bye...\n");
    exit(0);
}
Exemplo n.º 2
0
/* return file descriptor if success, <0 otherwise */
int httpd_init(int sport, char *sname, int use_ssl,  const char *certificate, 
               const char *password, int use_v6)
{
    struct addrinfo ask,*res = NULL;
    struct sockaddr_storage  ss;
    int opt, rc, ss_len, v4 = 1, v6 = 0;
    char host[INET6_ADDRSTRLEN+1];
    char serv[16];
    char listen_port[6];  
    char *listen_ip = NULL;
 
    /* set config */
    sprintf(listen_port, "%d", sport);
    server_name = sname;
    if (use_v6) {
        v6 = 1;
    } 

#ifdef USE_SSL
    with_ssl = use_ssl;
#endif
    /* FIXME nthreads */

    /* get server name */
    gethostname(server_host,255);
    memset(&ask,0,sizeof(ask));
    ask.ai_flags = AI_CANONNAME;
    if ((rc = getaddrinfo(server_host, NULL, &ask, &res)) == 0) {
        if (res->ai_canonname) {
            strcpy(server_host,res->ai_canonname);
        }
	
        if (res != NULL) {
            freeaddrinfo(res);
            res = NULL;
        }
    }
    
    /* bind to socket */
    slisten = -1;
    memset(&ask, 0, sizeof(ask));
    ask.ai_flags = AI_PASSIVE;
    if (listen_ip) {
        ask.ai_flags |= AI_CANONNAME;
    }
    ask.ai_socktype = SOCK_STREAM;

    /* try ipv6 first ... */
    if (slisten == -1 &&  v6) {
        ask.ai_family = PF_INET6;

        g_timeout = 0;
        alarm(2);
        rc = getaddrinfo(listen_ip, listen_port, &ask, &res);
        if (g_timeout) {
            log_error_func(1, LOG_ERR,"getaddrinfo (ipv6): DNS timeout",NULL);  
        }
        alarm(0);

        if (rc == 0) {
            if ((slisten = socket(res->ai_family, res->ai_socktype,
                                  res->ai_protocol)) == -1) {
                log_error_func(1, LOG_ERR,"socket (ipv6)",NULL);
            }
        } else {
            log_error_func(1, LOG_ERR, "getaddrinfo (ipv6)", NULL);
        }
    }
	
    g_timeout = 0;
    alarm(2);
    
    /* ... failing that try ipv4 */
    if (slisten == -1 &&  v4) {
        ask.ai_family = PF_INET;

        g_timeout = 0;
        alarm(1);
        rc = getaddrinfo(listen_ip, listen_port, &ask, &res);
        if (g_timeout) {
            log_error_func(1, LOG_ERR,"getaddrinfo (ipv4): DNS timeout",NULL);  
            return -1;
        }
        alarm(0);

        if (rc == 0) {
            if ((slisten = socket(res->ai_family, res->ai_socktype,
                                  res->ai_protocol)) == -1) {
                log_error_func(1, LOG_ERR,"socket (ipv4)",NULL);
                return -1;
            }
        } else {
            log_error_func(1, LOG_ERR, "getaddrinfo (ipv4)", NULL);
            return -1;
        }
    }

    if (slisten == -1) {
        return -1;
    }

    memcpy(&ss,res->ai_addr,res->ai_addrlen);
    ss_len = res->ai_addrlen;
    if (res->ai_canonname) {
        strcpy(server_host,res->ai_canonname);
    }
    if ((rc = getnameinfo((struct sockaddr*)&ss,ss_len,
                          host,INET6_ADDRSTRLEN,serv,15,
                          NI_NUMERICHOST | NI_NUMERICSERV)) != 0) {
        log_error_func(1, LOG_ERR, "getnameinfo", NULL);
        return -1;
    }


    tcp_port = atoi(serv);
    opt = 1;
    setsockopt(slisten,SOL_SOCKET,SO_REUSEADDR,&opt,sizeof(opt));
    fcntl(slisten,F_SETFL,O_NONBLOCK);

    /* Use accept filtering, if available. */
#ifdef SO_ACCEPTFILTER
    {
        struct accept_filter_arg af;
        memset(&af,0,sizeof(af));
        strcpy(af.af_name,"httpready");
        setsockopt(slisten, SOL_SOCKET, SO_ACCEPTFILTER, (char*)&af, sizeof(af));
    }
#endif /* SO_ACCEPTFILTER */

   
    if (bind(slisten, (struct sockaddr*) &ss, ss_len) == -1) {
        log_error_func(1, LOG_ERR,"bind",NULL);
        return -1;
    }
    if (listen(slisten, 2*max_conn) == -1) {
        log_error_func(1, LOG_ERR,"listen",NULL);
        return -1;
    }


    /* init misc stuff */
    init_mime(mimetypes,"text/plain");
    init_quote();
   
#ifdef USE_SSL
    if (with_ssl) {
        init_ssl(certificate, password);
    }
#endif

#ifdef DEBUG  
	fprintf(stderr,
            "http server started\n"
            "  ipv6  : %s\n"
#ifdef USE_SSL
	        "  ssl   : %s\n"
#endif
            "  node  : %s\n"
            "  ipaddr: %s\n"
            "  port  : %d\n"
            ,
            res->ai_family == PF_INET6 ? "yes" : "no",
#ifdef USE_SSL
            with_ssl ? "yes" : "no",
#endif
            server_host,host,tcp_port);
#endif

	    	
	if (res != NULL) {
        freeaddrinfo(res);
	}

    /* go! */
#ifdef HTTPD_USE_THREADS
    if (nthreads > 1) {
        int i;
        threads = malloc(sizeof(pthread_t) * nthreads);
        for (i = 1; i < nthreads; i++) {
            pthread_create(threads+i,NULL,mainloop,threads+i);
            pthread_detach(threads[i]);
        }
    }
#endif
  
    return slisten;
}