1 | /*
|
---|
2 | * Test name: test07_srv.c
|
---|
3 | *
|
---|
4 | * Objective: Test an impatient TCP server with a timeout.
|
---|
5 | *
|
---|
6 | * Description: Implements an echo server using the TCP protocol. It is
|
---|
7 | * based on test06_srv.c but has a timeout for receiving data from a client
|
---|
8 | * and if the time is up it sends an angry message to the client requesting
|
---|
9 | * for more information.
|
---|
10 | */
|
---|
11 |
|
---|
12 | #include <sys/types.h>
|
---|
13 | #include <sys/ioctl.h>
|
---|
14 | #include <sys/select.h>
|
---|
15 | #include <sys/wait.h>
|
---|
16 | #include <sys/asynchio.h>
|
---|
17 | #include <stdio.h>
|
---|
18 | #include <string.h>
|
---|
19 | #include <stdlib.h>
|
---|
20 | #include <stdarg.h>
|
---|
21 | #include <fcntl.h>
|
---|
22 | #include <errno.h>
|
---|
23 | #include <unistd.h>
|
---|
24 | #include <net/netlib.h>
|
---|
25 | #include <net/gen/netdb.h>
|
---|
26 | #include <net/gen/in.h>
|
---|
27 | #include <net/gen/tcp.h>
|
---|
28 | #include <net/gen/tcp_io.h>
|
---|
29 | #include <net/hton.h>
|
---|
30 | #include <net/gen/inet.h>
|
---|
31 |
|
---|
32 | #define PORT 6060L
|
---|
33 |
|
---|
34 | int listen(long port) {
|
---|
35 |
|
---|
36 | char *tcp_device;
|
---|
37 | int netfd;
|
---|
38 | nwio_tcpconf_t tcpconf;
|
---|
39 | nwio_tcpcl_t tcplistenopt;
|
---|
40 |
|
---|
41 | /* Get default UDP device */
|
---|
42 | if ((tcp_device = getenv("TCP_DEVICE")) == NULL)
|
---|
43 | tcp_device = TCP_DEVICE;
|
---|
44 |
|
---|
45 | /* Open TCP connection */
|
---|
46 | if ((netfd = open(tcp_device, O_RDWR)) < 0)
|
---|
47 | {
|
---|
48 | fprintf(stderr,"Error opening TCP connection\n");
|
---|
49 | return -1;
|
---|
50 | }
|
---|
51 |
|
---|
52 | /* Configure TCP connection */
|
---|
53 | tcpconf.nwtc_flags = NWTC_LP_SET | NWTC_UNSET_RA | NWTC_UNSET_RP;
|
---|
54 | tcpconf.nwtc_locport = (tcpport_t) htons(port);
|
---|
55 |
|
---|
56 | if ((ioctl(netfd, NWIOSTCPCONF, &tcpconf))<0)
|
---|
57 | {
|
---|
58 | fprintf(stderr,"Error configuring the connection\n");
|
---|
59 | close(netfd);
|
---|
60 | return -1;
|
---|
61 | }
|
---|
62 |
|
---|
63 | /* Get communication options */
|
---|
64 | if ((ioctl(netfd, NWIOGTCPCONF, &tcpconf)) < 0) {
|
---|
65 | fprintf(stderr, "Error getting configuration\n");
|
---|
66 | close(netfd);
|
---|
67 | return -1;
|
---|
68 | }
|
---|
69 |
|
---|
70 | /* Set conf options */
|
---|
71 | tcplistenopt.nwtcl_flags = 0;
|
---|
72 | printf("Waiting for connections...\n");
|
---|
73 | while ((ioctl(netfd, NWIOTCPLISTEN, &tcplistenopt)) == -1)
|
---|
74 | {
|
---|
75 | if (errno != EAGAIN)
|
---|
76 | {
|
---|
77 | fprintf(stderr,"Unable to listen for connections\n");
|
---|
78 | close(netfd);
|
---|
79 | }
|
---|
80 | sleep(-1);
|
---|
81 | }
|
---|
82 | return netfd;
|
---|
83 | }
|
---|
84 |
|
---|
85 | int main(int argc,char *argv[]) {
|
---|
86 | int fd;
|
---|
87 | ssize_t data_read;
|
---|
88 | char buffer[1024];
|
---|
89 | int ret;
|
---|
90 | fd_set fds_read;
|
---|
91 | struct timeval timeout;
|
---|
92 |
|
---|
93 | if ((fd = listen(PORT)) < 0) {
|
---|
94 | exit(-1);
|
---|
95 | }
|
---|
96 | printf("Waiting for messages on port: %ld\n", PORT);
|
---|
97 | fflush(stdout);
|
---|
98 | while (1)
|
---|
99 | {
|
---|
100 |
|
---|
101 | /* Initialize fd_set */
|
---|
102 | FD_ZERO(&fds_read);
|
---|
103 | FD_SET(fd, &fds_read);
|
---|
104 | /* set timeout */
|
---|
105 | timeout.tv_sec = 3;
|
---|
106 | timeout.tv_usec = 0;
|
---|
107 | /* Wait for data available to be read (no timeout) */
|
---|
108 | ret = select(4, &fds_read, NULL, NULL, &timeout);
|
---|
109 | if (ret < 0) {
|
---|
110 | fprintf(stderr, "Error on select: %d\n", errno);
|
---|
111 | exit(-1);
|
---|
112 | }
|
---|
113 | /* timeout */
|
---|
114 | if (ret == 0) {
|
---|
115 | strcpy(buffer, "I want to get some data!!\n");
|
---|
116 | write(fd, &buffer, 1024);
|
---|
117 | continue;
|
---|
118 | }
|
---|
119 |
|
---|
120 | /* data received from client */
|
---|
121 | if (FD_ISSET(fd, &fds_read)) {
|
---|
122 | printf("Ready to receive...\n");
|
---|
123 | /* Read received data */
|
---|
124 | data_read = read(fd, &buffer, 1024);
|
---|
125 | printf("Received data: %s\n", buffer);
|
---|
126 |
|
---|
127 | /* Can exit if the received string == exit */
|
---|
128 | if (!strcmp(buffer,"exit"))
|
---|
129 | break;
|
---|
130 |
|
---|
131 | /* Write the same back */
|
---|
132 | write(fd, &buffer, data_read);
|
---|
133 | }
|
---|
134 | }
|
---|
135 | printf("Connection finished\n");
|
---|
136 | close(fd);
|
---|
137 | }
|
---|