#include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define MAX_WORKERS 128 pthread_t hworker[MAX_WORKERS]; #define MAX_EVENTS 512 struct epoll_event ev; struct epoll_event events[MAX_EVENTS]; int epfd; int sockfd; static int exit_flag = 0; char html[] = "HTTP/1.1 200 OK\r\n" "Server: F-Stack\r\n" "Date: Sat, 25 Feb 2017 09:26:33 GMT\r\n" "Content-Type: text/html\r\n" "Content-Length: 438\r\n" "Last-Modified: Tue, 21 Feb 2017 09:44:03 GMT\r\n" "Connection: keep-alive\r\n" "Accept-Ranges: bytes\r\n" "\r\n" "\r\n" "\r\n" "\r\n" "Welcome to F-Stack!\r\n" "\r\n" "\r\n" "\r\n" "

Welcome to F-Stack!

\r\n" "\r\n" "

For online documentation and support please refer to\r\n" "F-Stack.org.
\r\n" "\r\n" "

Thank you for using F-Stack.

\r\n" "\r\n" ""; void sig_term(int sig) { printf("we caught signal %d, to exit helloworld\n", sig); exit_flag = 1; //alarm_event_sem(); return; } void *loop(void *arg) { /* Wait for events to happen */ while (!exit_flag) { /* * If not call alarm_event_sem, and epoll_wait timeout is 0, * it can't exit normal, so timeout can't set to 0. */ int nevents = epoll_wait(epfd, events, MAX_EVENTS, 100); int i; if (nevents <= 0) { if (nevents) { printf("hello world epoll wait ret %d, errno:%d, %s\n", nevents, errno, strerror(errno)); break; } //usleep(100); sleep(1); } //printf("get nevents:%d\n", nevents); for (i = 0; i < nevents; ++i) { /* Handle new connect */ if (events[i].data.fd == sockfd) { while (1) { int nclientfd = accept(sockfd, NULL, NULL); printf("accept sockfd:%d, nclientfd:%d, errono:%d/%s\n", sockfd, nclientfd, errno, strerror(errno)); if (nclientfd < 0) { break; } /* Add to event list */ ev.data.fd = nclientfd; ev.events = EPOLLIN; if (epoll_ctl(epfd, EPOLL_CTL_ADD, nclientfd, &ev) != 0) { printf("ff_epoll_ctl failed:%d, %s\n", errno, strerror(errno)); close(nclientfd); break; } } } else { if (events[i].events & EPOLLERR ) { /* Simply close socket */ epoll_ctl(epfd, EPOLL_CTL_DEL, events[i].data.fd, NULL); close(events[i].data.fd); } else if (events[i].events & EPOLLIN) { char buf[256]; size_t readlen = read( events[i].data.fd, buf, sizeof(buf)); if(readlen > 0) { ssize_t writelen = write( events[i].data.fd, html, sizeof(html) - 1); if (writelen < 0){ printf("write failed, readlen:%lu, writelen:%lu, :%d, %s\n", readlen, writelen, errno, strerror(errno)); close(events[i].data.fd); } } else { epoll_ctl(epfd, EPOLL_CTL_DEL, events[i].data.fd, NULL); close(events[i].data.fd); } } else { printf("unknown event: %d:%8.8X\n", i, events[i].events); } } } } return NULL; } int main(int argc, char * argv[]) { int i, worker_num = 1; signal(SIGINT, sig_term); signal(SIGTERM, sig_term); sockfd = socket(AF_INET, SOCK_STREAM, 0); printf("sockfd:%d\n", sockfd); if (sockfd < 0) { printf("ff_socket failed\n"); return -1; } int on = 1; ioctl(sockfd, FIONBIO, &on); struct sockaddr_in my_addr; bzero(&my_addr, sizeof(my_addr)); my_addr.sin_family = AF_INET; my_addr.sin_port = htons(80); my_addr.sin_addr.s_addr = htonl(INADDR_ANY); int ret = bind(sockfd, (const struct sockaddr *)&my_addr, sizeof(my_addr)); if (ret < 0) { printf("ff_bind failed\n"); close(sockfd); return -1; } ret = listen(sockfd, MAX_EVENTS); if (ret < 0) { printf("ff_listen failed\n"); close(sockfd); return -1; } epfd = epoll_create(512); printf("epfd:%d\n", epfd); if (epfd <= 0) { printf("ff_epoll_create failed, errno:%d, %s\n", errno, strerror(errno)); close(sockfd); return -1; } ev.data.fd = sockfd; ev.events = EPOLLIN; ret = epoll_ctl(epfd, EPOLL_CTL_ADD, sockfd, &ev); if (ret < 0) { printf("ff_listen failed\n"); close(epfd); close(sockfd); return -1; } for (i = 0; i < worker_num; i++) { if(pthread_create(&hworker[i], NULL, loop, (void *)&i) < 0) { printf("create loop thread failed., errno:%d/%s\n", errno, strerror(errno)); close(epfd); close(sockfd); return -1; } } for (i = 0; i < worker_num; i++) { pthread_join(hworker[i], NULL); } close(epfd); close(sockfd); return 0; }