minix/test/select/test05_srv.c

199 lines
5.3 KiB
C
Raw Normal View History

/*
* 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);
}