xref: /f-stack/example/main.c (revision 2dfcd880)
1a9643ea8Slogwang #include <stdio.h>
2a9643ea8Slogwang #include <stdlib.h>
3a9643ea8Slogwang #include <stdint.h>
4a9643ea8Slogwang #include <strings.h>
5a9643ea8Slogwang #include <sys/types.h>
6a9643ea8Slogwang #include <sys/socket.h>
7a9643ea8Slogwang #include <arpa/inet.h>
8a9643ea8Slogwang #include <errno.h>
9a9643ea8Slogwang #include <assert.h>
10a9643ea8Slogwang 
11a9643ea8Slogwang #include "ff_config.h"
12a9643ea8Slogwang #include "ff_api.h"
13a9643ea8Slogwang 
14a9643ea8Slogwang #define MAX_EVENTS 512
15a9643ea8Slogwang 
16a9643ea8Slogwang /* kevent set */
17a9643ea8Slogwang struct kevent kevSet;
18a9643ea8Slogwang /* events */
19a9643ea8Slogwang struct kevent events[MAX_EVENTS];
20a9643ea8Slogwang /* kq */
21a9643ea8Slogwang int kq;
22a9643ea8Slogwang int sockfd;
23a9643ea8Slogwang 
24a9643ea8Slogwang char html[] =
25a9643ea8Slogwang "HTTP/1.1 200 OK\r\n"
26a9643ea8Slogwang "Server: F-Stack\r\n"
27a9643ea8Slogwang "Date: Sat, 25 Feb 2017 09:26:33 GMT\r\n"
28a9643ea8Slogwang "Content-Type: text/html\r\n"
29a9643ea8Slogwang "Content-Length: 439\r\n"
30a9643ea8Slogwang "Last-Modified: Tue, 21 Feb 2017 09:44:03 GMT\r\n"
31a9643ea8Slogwang "Connection: keep-alive\r\n"
32a9643ea8Slogwang "Accept-Ranges: bytes\r\n"
33a9643ea8Slogwang "\r\n"
34a9643ea8Slogwang "<!DOCTYPE html>\r\n"
35a9643ea8Slogwang "<html>\r\n"
36a9643ea8Slogwang "<head>\r\n"
37a9643ea8Slogwang "<title>Welcome to F-Stack!</title>\r\n"
38a9643ea8Slogwang "<style>\r\n"
39a9643ea8Slogwang "    body {  \r\n"
40a9643ea8Slogwang "        width: 35em;\r\n"
41a9643ea8Slogwang "        margin: 0 auto; \r\n"
42a9643ea8Slogwang "        font-family: Tahoma, Verdana, Arial, sans-serif;\r\n"
43a9643ea8Slogwang "    }\r\n"
44a9643ea8Slogwang "</style>\r\n"
45a9643ea8Slogwang "</head>\r\n"
46a9643ea8Slogwang "<body>\r\n"
47a9643ea8Slogwang "<h1>Welcome to F-Stack!</h1>\r\n"
48a9643ea8Slogwang "\r\n"
49a9643ea8Slogwang "<p>For online documentation and support please refer to\r\n"
50a9643ea8Slogwang "<a href=\"http://F-Stack.org/\">F-Stack.org</a>.<br/>\r\n"
51a9643ea8Slogwang "\r\n"
52a9643ea8Slogwang "<p><em>Thank you for using F-Stack.</em></p>\r\n"
53a9643ea8Slogwang "</body>\r\n"
54a9643ea8Slogwang "</html>";
55a9643ea8Slogwang 
56a9643ea8Slogwang int loop(void *arg)
57a9643ea8Slogwang {
58a9643ea8Slogwang     /* Wait for events to happen */
59a9643ea8Slogwang     unsigned nevents = ff_kevent(kq, NULL, 0, events, MAX_EVENTS, NULL);
60a9643ea8Slogwang     unsigned i;
61a9643ea8Slogwang 
62a9643ea8Slogwang     for (i = 0; i < nevents; ++i) {
63a9643ea8Slogwang         struct kevent event = events[i];
64a9643ea8Slogwang         int clientfd = (int)event.ident;
65a9643ea8Slogwang 
66a9643ea8Slogwang         /* Handle disconnect */
67a9643ea8Slogwang         if (event.flags & EV_EOF) {
68a9643ea8Slogwang 
69a9643ea8Slogwang             /* Simply close socket */
70a9643ea8Slogwang             ff_close(clientfd);
71a9643ea8Slogwang 
72*2dfcd880Swhl739             //printf("A client has left the server...,fd:%d\n", clientfd);
73a9643ea8Slogwang         } else if (clientfd == sockfd) {
74a9643ea8Slogwang             int nclientfd = ff_accept(sockfd, NULL, NULL);
75a9643ea8Slogwang 
76a9643ea8Slogwang             assert(nclientfd > 0);
77a9643ea8Slogwang 
78a9643ea8Slogwang             /* Add to event list */
79a9643ea8Slogwang             kevSet.data     = 0;
80a9643ea8Slogwang             kevSet.fflags   = 0;
81a9643ea8Slogwang             kevSet.filter   = EVFILT_READ;
82a9643ea8Slogwang             kevSet.flags    = EV_ADD;
83a9643ea8Slogwang             kevSet.ident    = nclientfd;
84a9643ea8Slogwang             kevSet.udata    = NULL;
85a9643ea8Slogwang 
86a9643ea8Slogwang             assert(ff_kevent(kq, &kevSet, 1, NULL, 0, NULL) == 0);
87a9643ea8Slogwang 
88*2dfcd880Swhl739             //printf("A new client connected to the server..., fd:%d\n", nclientfd);
89a9643ea8Slogwang 
90a9643ea8Slogwang         } else if (event.filter == EVFILT_READ) {
91a9643ea8Slogwang             char buf[256];
92a9643ea8Slogwang             size_t readlen = ff_read(clientfd, buf, sizeof(buf));
93a9643ea8Slogwang 
94*2dfcd880Swhl739             //printf("bytes %zu are available to read...,fd:%d\n", (size_t)event.data, clientfd);
95a9643ea8Slogwang 
96a9643ea8Slogwang             ff_write(clientfd, html, sizeof(html));
97a9643ea8Slogwang 
98a9643ea8Slogwang         } else {
99*2dfcd880Swhl739             printf("unknown event: %8.8X\n", event.flags);
100a9643ea8Slogwang         }
101a9643ea8Slogwang     }
102a9643ea8Slogwang }
103a9643ea8Slogwang 
104a9643ea8Slogwang int main(int argc, char * argv[])
105a9643ea8Slogwang {
106a9643ea8Slogwang     char *conf;
107a9643ea8Slogwang     if (argc < 2) {
108a9643ea8Slogwang         conf = "./config.ini";
109a9643ea8Slogwang     } else {
110a9643ea8Slogwang         conf = argv[1];
111a9643ea8Slogwang     }
112a9643ea8Slogwang 
113a9643ea8Slogwang     ff_init(conf, argc, argv);
114a9643ea8Slogwang 
115a9643ea8Slogwang     int sockfd = ff_socket(AF_INET, SOCK_STREAM, 0);
116a9643ea8Slogwang     printf("sockfd:%d\n", sockfd);
117a9643ea8Slogwang     if (sockfd < 0)
118a9643ea8Slogwang         printf("ff_socket failed\n");
119a9643ea8Slogwang 
120a9643ea8Slogwang     struct sockaddr_in my_addr;
121a9643ea8Slogwang     bzero(&my_addr, sizeof(my_addr));
122a9643ea8Slogwang     my_addr.sin_family = AF_INET;
123a9643ea8Slogwang     my_addr.sin_port = htons(80);
124a9643ea8Slogwang     my_addr.sin_addr.s_addr = htonl(INADDR_ANY);
125a9643ea8Slogwang 
126a9643ea8Slogwang     int ret = ff_bind(sockfd, (struct linux_sockaddr *)&my_addr, sizeof(my_addr));
127a9643ea8Slogwang     if (ret < 0) {
128a9643ea8Slogwang         printf("ff_bind failed\n");
129a9643ea8Slogwang     }
130a9643ea8Slogwang 
131a9643ea8Slogwang     ret = ff_listen(sockfd, MAX_EVENTS);
132a9643ea8Slogwang     if (ret < 0) {
133a9643ea8Slogwang         printf("ff_listen failed\n");
134a9643ea8Slogwang     }
135a9643ea8Slogwang 
136a9643ea8Slogwang     kevSet.data     = MAX_EVENTS;
137a9643ea8Slogwang     kevSet.fflags   = 0;
138a9643ea8Slogwang     kevSet.filter   = EVFILT_READ;
139a9643ea8Slogwang     kevSet.flags    = EV_ADD;
140a9643ea8Slogwang     kevSet.ident    = sockfd;
141a9643ea8Slogwang     kevSet.udata    = NULL;
142a9643ea8Slogwang 
143a9643ea8Slogwang     assert((kq = ff_kqueue()) > 0);
144a9643ea8Slogwang 
145a9643ea8Slogwang     /* Update kqueue */
146a9643ea8Slogwang     ff_kevent(kq, &kevSet, 1, NULL, 0, NULL);
147a9643ea8Slogwang 
148a9643ea8Slogwang     ff_run(loop, NULL);
149a9643ea8Slogwang     return 0;
150a9643ea8Slogwang }
151a9643ea8Slogwang 
152a9643ea8Slogwang 
153