[9] | 1 | /*
|
---|
| 2 | * Test name: test05_srv.c
|
---|
| 3 | *
|
---|
| 4 | * Objective: Test an impatient UDP server with timeouts
|
---|
| 5 | *
|
---|
| 6 | * Description: Implements an echo server using the UDP protocol. It is
|
---|
| 7 | * based on test04_srv, but it has a timeout value.
|
---|
| 8 | */
|
---|
| 9 |
|
---|
| 10 | #include <sys/types.h>
|
---|
| 11 | #include <sys/ioctl.h>
|
---|
| 12 | #include <sys/select.h>
|
---|
| 13 | #include <sys/asynchio.h>
|
---|
| 14 | #include <stdio.h>
|
---|
| 15 | #include <string.h>
|
---|
| 16 | #include <stdlib.h>
|
---|
| 17 | #include <stdarg.h>
|
---|
| 18 | #include <fcntl.h>
|
---|
| 19 | #include <errno.h>
|
---|
| 20 | #include <unistd.h>
|
---|
| 21 | #include <net/netlib.h>
|
---|
| 22 | #include <net/gen/netdb.h>
|
---|
| 23 | #include <net/gen/in.h>
|
---|
| 24 | #include <net/gen/udp.h>
|
---|
| 25 | #include <net/gen/udp_hdr.h>
|
---|
| 26 | #include <net/gen/udp_io.h>
|
---|
| 27 | #include <net/hton.h>
|
---|
| 28 |
|
---|
| 29 | #define PORT 6000L
|
---|
| 30 |
|
---|
| 31 | /* type for received data */
|
---|
| 32 | typedef struct
|
---|
| 33 | {
|
---|
| 34 | udp_io_hdr_t header;
|
---|
| 35 | char data[1024];
|
---|
| 36 | } udp_buffer_t;
|
---|
| 37 |
|
---|
| 38 | int udp_conf(long port) {
|
---|
| 39 |
|
---|
| 40 | char *udp_device;
|
---|
| 41 | int netfd;
|
---|
| 42 | nwio_udpopt_t udpopt;
|
---|
| 43 |
|
---|
| 44 | /* Get default UDP device */
|
---|
| 45 | if ((udp_device = getenv("UDP_DEVICE")) == NULL)
|
---|
| 46 | udp_device = UDP_DEVICE;
|
---|
| 47 |
|
---|
| 48 | /* Open UDP connection */
|
---|
| 49 | if ((netfd = open(udp_device, O_RDWR)) < 0)
|
---|
| 50 | {
|
---|
| 51 | fprintf(stderr,"Error opening UDP connection\n");
|
---|
| 52 | return -1;
|
---|
| 53 | }
|
---|
| 54 |
|
---|
| 55 | /* Configure UDP connection */
|
---|
| 56 | udpopt.nwuo_flags = NWUO_COPY | NWUO_LP_SET | NWUO_EN_LOC | NWUO_DI_BROAD
|
---|
| 57 | | NWUO_RP_ANY | NWUO_RA_ANY | NWUO_RWDATALL | NWUO_DI_IPOPT;
|
---|
| 58 |
|
---|
| 59 | udpopt.nwuo_locport = (udpport_t) htons(port);
|
---|
| 60 |
|
---|
| 61 | if ((ioctl(netfd, NWIOSUDPOPT, &udpopt))<0)
|
---|
| 62 | {
|
---|
| 63 | fprintf(stderr,"Error configuring the connection\n");
|
---|
| 64 | close(netfd);
|
---|
| 65 | return -1;
|
---|
| 66 | }
|
---|
| 67 |
|
---|
| 68 | /* Get conf options */
|
---|
| 69 | if ((ioctl(netfd, NWIOGUDPOPT, &udpopt))<0)
|
---|
| 70 | {
|
---|
| 71 | fprintf(stderr,"Error getting the conf\n");
|
---|
| 72 | close(netfd);
|
---|
| 73 | return -1;
|
---|
| 74 | }
|
---|
| 75 |
|
---|
| 76 | return netfd;
|
---|
| 77 | }
|
---|
| 78 |
|
---|
| 79 | int main(int argc,char *argv[]) {
|
---|
| 80 | int fd;
|
---|
| 81 | ssize_t data_read;
|
---|
| 82 | udp_buffer_t buffer;
|
---|
| 83 | ipaddr_t tmp_addr;
|
---|
| 84 | udpport_t tmp_port;
|
---|
| 85 | int ret;
|
---|
| 86 | fd_set fds_read;
|
---|
| 87 | struct timeval timeout;
|
---|
| 88 | ipaddr_t client_addr, this_addr;
|
---|
| 89 | udpport_t client_port;
|
---|
| 90 |
|
---|
| 91 | if ((fd = udp_conf(PORT)) < 0) {
|
---|
| 92 | fprintf(stderr, "Error configuring UDP connection\n");
|
---|
| 93 | exit(-1);
|
---|
| 94 | }
|
---|
| 95 | printf("Waiting for messages on port: %ld\n", PORT);
|
---|
| 96 | fflush(stdout);
|
---|
| 97 |
|
---|
| 98 | /* get a first message so we know who is the client and we can harass it
|
---|
| 99 | afterwards */
|
---|
| 100 |
|
---|
| 101 | /* Initialize fd_set */
|
---|
| 102 | FD_ZERO(&fds_read);
|
---|
| 103 | FD_SET(fd, &fds_read);
|
---|
| 104 | /* Set timeout structure */
|
---|
| 105 | timeout.tv_sec = 3;
|
---|
| 106 | timeout.tv_usec = 0;
|
---|
| 107 | ret = select(4, &fds_read, NULL, NULL, NULL);
|
---|
| 108 |
|
---|
| 109 | if (ret < 0) {
|
---|
| 110 | fprintf(stderr, "Error in select\n");
|
---|
| 111 | exit(-1);
|
---|
| 112 | }
|
---|
| 113 | if (!FD_ISSET(fd, &fds_read)) {
|
---|
| 114 | fprintf(stderr, "Error: Should be receiving some data from network(?)\n");
|
---|
| 115 | exit(-1);
|
---|
| 116 | }
|
---|
| 117 | printf("Ready to receive...\n");
|
---|
| 118 | /* Read received data */
|
---|
| 119 | data_read = read(fd, &buffer, sizeof(udp_buffer_t));
|
---|
| 120 | printf("Received data: %s\n", buffer.data);
|
---|
| 121 |
|
---|
| 122 | /* Can exit if the received string == exit */
|
---|
| 123 | if (!strcmp(buffer.data,"exit"))
|
---|
| 124 | exit(0);
|
---|
| 125 |
|
---|
| 126 | /* Send data back, swap addresses */
|
---|
| 127 | tmp_addr = buffer.header.uih_src_addr;
|
---|
| 128 | buffer.header.uih_src_addr = buffer.header.uih_dst_addr;
|
---|
| 129 | buffer.header.uih_dst_addr = tmp_addr;
|
---|
| 130 | /* save address of both ends */
|
---|
| 131 | client_addr = tmp_addr;
|
---|
| 132 | this_addr = buffer.header.uih_src_addr;
|
---|
| 133 |
|
---|
| 134 | /* Swap ports */
|
---|
| 135 | tmp_port = buffer.header.uih_src_port;
|
---|
| 136 | buffer.header.uih_src_port = buffer.header.uih_dst_port;
|
---|
| 137 | buffer.header.uih_dst_port = tmp_port;
|
---|
| 138 | /* save client port */
|
---|
| 139 | client_port = tmp_port;
|
---|
| 140 |
|
---|
| 141 | /* Write the same back */
|
---|
| 142 | write(fd, &buffer, data_read);
|
---|
| 143 |
|
---|
| 144 | while (1)
|
---|
| 145 | {
|
---|
| 146 |
|
---|
| 147 | /* Initialize fd_set */
|
---|
| 148 | FD_ZERO(&fds_read);
|
---|
| 149 | FD_SET(fd, &fds_read);
|
---|
| 150 | /* Set timeout structure */
|
---|
| 151 | timeout.tv_sec = 3;
|
---|
| 152 | timeout.tv_usec = 0;
|
---|
| 153 | /* Wait for data available to be read (timeout) */
|
---|
| 154 | ret = select(4, &fds_read, NULL, NULL, &timeout);
|
---|
| 155 | if (ret < 0) {
|
---|
| 156 | fprintf(stderr, "Error on select: %d", errno);
|
---|
| 157 | exit(-1);
|
---|
| 158 | }
|
---|
| 159 | /* if timeout */
|
---|
| 160 | if (ret == 0) {
|
---|
| 161 | /* Send angry msg to client asking for more */
|
---|
| 162 | printf("Tired of waiting, send client an angry message\n");
|
---|
| 163 | buffer.header.uih_src_addr = this_addr;
|
---|
| 164 | buffer.header.uih_dst_addr = client_addr;
|
---|
| 165 | buffer.header.uih_src_port = PORT;
|
---|
| 166 | buffer.header.uih_dst_port = client_port;
|
---|
| 167 | strcpy(buffer.data, "Hey! I want to receive some data!\n");
|
---|
| 168 | write(fd, &buffer, sizeof(udp_buffer_t));
|
---|
| 169 | }
|
---|
| 170 | /* If receive data from network */
|
---|
| 171 | if (FD_ISSET(fd, &fds_read)) {
|
---|
| 172 | printf("Ready to receive...\n");
|
---|
| 173 | /* Read received data */
|
---|
| 174 | data_read = read(fd, &buffer, sizeof(udp_buffer_t));
|
---|
| 175 | printf("Received data: %s\n", buffer.data);
|
---|
| 176 |
|
---|
| 177 | /* Can exit if the received string == exit */
|
---|
| 178 | if (!strcmp(buffer.data,"exit"))
|
---|
| 179 | break;
|
---|
| 180 |
|
---|
| 181 | /* Send data back, swap addresses */
|
---|
| 182 | tmp_addr = buffer.header.uih_src_addr;
|
---|
| 183 | buffer.header.uih_src_addr = buffer.header.uih_dst_addr;
|
---|
| 184 | buffer.header.uih_dst_addr = tmp_addr;
|
---|
| 185 |
|
---|
| 186 | /* Swap ports */
|
---|
| 187 | tmp_port = buffer.header.uih_src_port;
|
---|
| 188 | buffer.header.uih_src_port = buffer.header.uih_dst_port;
|
---|
| 189 | buffer.header.uih_dst_port = tmp_port;
|
---|
| 190 |
|
---|
| 191 | /* Write the same back */
|
---|
| 192 | write(fd, &buffer, data_read);
|
---|
| 193 | }
|
---|
| 194 | }
|
---|
| 195 | close(fd);
|
---|
| 196 | }
|
---|