197 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			197 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
| /*
 | |
|  * Test name: test05_srv.c
 | |
|  *
 | |
|  * Objective: Test an impatient UDP server with timeouts  
 | |
|  *
 | |
|  * Description: Implements an echo server using the UDP protocol. It is 
 | |
|  * based on test04_srv, but it has a timeout value. 
 | |
|  */
 | |
| 
 | |
| #include <sys/types.h>
 | |
| #include <sys/ioctl.h>
 | |
| #include <sys/select.h>
 | |
| #include <sys/asynchio.h>
 | |
| #include <stdio.h>
 | |
| #include <string.h>
 | |
| #include <stdlib.h>
 | |
| #include <stdarg.h>
 | |
| #include <fcntl.h>
 | |
| #include <errno.h>
 | |
| #include <unistd.h>
 | |
| #include <net/netlib.h>
 | |
| #include <net/gen/netdb.h>
 | |
| #include <net/gen/in.h>
 | |
| #include <net/gen/udp.h>
 | |
| #include <net/gen/udp_hdr.h>
 | |
| #include <net/gen/udp_io.h>
 | |
| #include <net/hton.h>
 | |
| 
 | |
| #define PORT 6000L
 | |
| 
 | |
| /* type for received data */
 | |
| typedef struct 
 | |
| {
 | |
|   udp_io_hdr_t header;
 | |
|   char data[1024];
 | |
| } udp_buffer_t;
 | |
| 
 | |
| int udp_conf(long port) {
 | |
| 
 | |
|   char *udp_device;
 | |
|   int netfd;
 | |
|   nwio_udpopt_t udpopt;
 | |
|                             
 | |
|   /* Get default UDP device */
 | |
|   if ((udp_device = getenv("UDP_DEVICE")) == NULL) 
 | |
|     udp_device = UDP_DEVICE;
 | |
|                                 
 | |
|   /* Open UDP connection */ 
 | |
|   if ((netfd = open(udp_device, O_RDWR)) < 0) 
 | |
|   {
 | |
|     fprintf(stderr,"Error opening UDP connection\n");
 | |
|     return -1;
 | |
|   }               
 | |
|                                 
 | |
|   /* Configure UDP connection */ 
 | |
|   udpopt.nwuo_flags = NWUO_COPY | NWUO_LP_SET | NWUO_EN_LOC | NWUO_DI_BROAD 
 | |
|                    | NWUO_RP_ANY | NWUO_RA_ANY | NWUO_RWDATALL | NWUO_DI_IPOPT;
 | |
| 
 | |
|   udpopt.nwuo_locport = (udpport_t) htons(port);
 | |
| 	                        
 | |
|   if ((ioctl(netfd, NWIOSUDPOPT, &udpopt))<0) 
 | |
|   {
 | |
|     fprintf(stderr,"Error configuring the connection\n");
 | |
|     close(netfd);
 | |
|     return -1;
 | |
|   }               
 | |
| 	                        
 | |
|   /* Get conf options */
 | |
|   if ((ioctl(netfd, NWIOGUDPOPT, &udpopt))<0) 
 | |
|   {
 | |
|     fprintf(stderr,"Error getting the conf\n");
 | |
|     close(netfd);
 | |
|     return -1;
 | |
|   }               
 | |
| 
 | |
|   return netfd;
 | |
| }
 | |
| 
 | |
| int main(int argc,char *argv[]) {
 | |
|   int fd;
 | |
|   ssize_t data_read;
 | |
|   udp_buffer_t buffer;
 | |
|   ipaddr_t tmp_addr;
 | |
|   udpport_t tmp_port;
 | |
|   int ret;
 | |
|   fd_set fds_read;
 | |
|   struct timeval timeout;
 | |
|   ipaddr_t client_addr, this_addr;
 | |
|   udpport_t client_port; 
 | |
| 
 | |
|   if ((fd = udp_conf(PORT)) < 0) {
 | |
|   	fprintf(stderr, "Error configuring UDP connection\n");
 | |
|     	exit(-1);	
 | |
|   }
 | |
|   printf("Waiting for messages on port: %ld\n", PORT);
 | |
|   fflush(stdout);
 | |
| 
 | |
|   /* get a first message so we know who is the client and we can harass it
 | |
|      afterwards */
 | |
| 
 | |
|   /* Initialize fd_set */
 | |
|   FD_ZERO(&fds_read);
 | |
|   FD_SET(fd, &fds_read);
 | |
|   /* Set timeout structure */
 | |
|   timeout.tv_sec = 3;
 | |
|   timeout.tv_usec = 0;
 | |
|   ret = select(4, &fds_read, NULL, NULL, NULL);
 | |
| 
 | |
|   if (ret < 0) {
 | |
|   	fprintf(stderr, "Error in select\n");
 | |
|   	exit(-1);
 | |
|   }
 | |
|   if (!FD_ISSET(fd, &fds_read)) {
 | |
|   	fprintf(stderr, "Error: Should be receiving some data from network(?)\n");
 | |
|   	exit(-1);
 | |
|   }
 | |
|   printf("Ready to receive...\n");
 | |
|   /* Read received data */
 | |
|   data_read = read(fd, &buffer, sizeof(udp_buffer_t));
 | |
|   printf("Received data: %s\n", buffer.data);
 | |
|                                 
 | |
|   /* Can exit if the received string == exit */
 | |
|   if (!strcmp(buffer.data,"exit")) 
 | |
|       	exit(0); 
 | |
| 
 | |
|   /* Send data back, swap addresses */
 | |
|   tmp_addr = buffer.header.uih_src_addr;
 | |
|   buffer.header.uih_src_addr = buffer.header.uih_dst_addr;
 | |
|   buffer.header.uih_dst_addr = tmp_addr;
 | |
|   /* save address of both ends */
 | |
|   client_addr = tmp_addr;
 | |
|   this_addr = buffer.header.uih_src_addr;  
 | |
| 
 | |
|   /* Swap ports */
 | |
|   tmp_port = buffer.header.uih_src_port;
 | |
|   buffer.header.uih_src_port = buffer.header.uih_dst_port;
 | |
|   buffer.header.uih_dst_port = tmp_port;
 | |
|   /* save client port */
 | |
|   client_port = tmp_port;
 | |
| 
 | |
|   /* Write the same back  */
 | |
|   write(fd, &buffer, data_read);             
 | |
| 
 | |
|   while (1) 
 | |
|   {
 | |
| 
 | |
|     /* Initialize fd_set */
 | |
|     FD_ZERO(&fds_read);
 | |
|     FD_SET(fd, &fds_read);
 | |
|     /* Set timeout structure */
 | |
|     timeout.tv_sec = 3;
 | |
|     timeout.tv_usec = 0;
 | |
|     /* Wait for data available to be read (timeout) */
 | |
|     ret = select(4, &fds_read, NULL, NULL, &timeout);
 | |
|     if (ret < 0) {
 | |
|     	fprintf(stderr, "Error on select: %d", errno);
 | |
|     	exit(-1);
 | |
|     }
 | |
|     /* if timeout */
 | |
|     if (ret == 0) {
 | |
|     	/* Send angry msg to client asking for more */
 | |
|      	printf("Tired of waiting, send client an angry message\n");
 | |
|      	buffer.header.uih_src_addr = this_addr;
 | |
|      	buffer.header.uih_dst_addr = client_addr;
 | |
|      	buffer.header.uih_src_port = PORT;
 | |
|      	buffer.header.uih_dst_port = client_port;
 | |
|     	strcpy(buffer.data, "Hey! I want to receive some data!\n");
 | |
|     	write(fd, &buffer, sizeof(udp_buffer_t));
 | |
|     }
 | |
|     /* If receive data from network */
 | |
|     if (FD_ISSET(fd, &fds_read)) {
 | |
| 	printf("Ready to receive...\n");
 | |
|     	/* Read received data */
 | |
|     	data_read = read(fd, &buffer, sizeof(udp_buffer_t));
 | |
|     	printf("Received data: %s\n", buffer.data);
 | |
|                                 
 | |
|     	/* Can exit if the received string == exit */
 | |
|     	if (!strcmp(buffer.data,"exit")) 
 | |
|       		break; 
 | |
| 
 | |
|     	/* Send data back, swap addresses */
 | |
|     	tmp_addr = buffer.header.uih_src_addr;
 | |
|     	buffer.header.uih_src_addr = buffer.header.uih_dst_addr;
 | |
|     	buffer.header.uih_dst_addr = tmp_addr;
 | |
|   
 | |
|     	/* Swap ports */
 | |
|     	tmp_port = buffer.header.uih_src_port;
 | |
|     	buffer.header.uih_src_port = buffer.header.uih_dst_port;
 | |
|     	buffer.header.uih_dst_port = tmp_port;
 | |
|   
 | |
|     	/* Write the same back  */
 | |
|     	write(fd, &buffer, data_read);             
 | |
|     }
 | |
|   }               
 | |
|   close(fd);
 | |
| }                       
 | 
