X-Git-Url: https://gapil.gnulinux.it/gitweb/?p=gapil.git;a=blobdiff_plain;f=sources%2FTCP_echo.c;h=2d41e936f247cece2222a1795570be13b1bfd7d3;hp=86b78c2762a1806840bb757680d6ee0c49d7b317;hb=fa15a3f1ecd64efd8440e46d398fd9976abc3d25;hpb=6c1923829690f7edc97225c27707d3924803c2d4 diff --git a/sources/TCP_echo.c b/sources/TCP_echo.c index 86b78c2..2d41e93 100644 --- a/sources/TCP_echo.c +++ b/sources/TCP_echo.c @@ -26,19 +26,27 @@ * * Usage: echo -h give all info's * - * $Id: TCP_echo.c,v 1.8 2003/08/17 23:03:44 piccardi Exp $ - * ****************************************************************/ /* * Include needed headers */ -#include /* predefined types */ -#include /* include unix standard library */ -#include /* IP addresses conversion utiliites */ -#include /* socket library */ -#include /* include standard I/O library */ -#include /* include error codes */ -#include /* include erroro strings definitions */ +#include /* primitive system data types */ +#include /* unix standard library */ +#include /* IP addresses conversion utilities */ +#include /* socket constants, types and functions */ +#include /* standard I/O library */ +#include /* error definitions and routines */ +#include /* C strings library */ +#include /* C standard library */ +#include /* TCP constants and types */ + +/* still not defined in some include, because too new ... */ +#ifndef TCP_CONGESTION +#define TCP_CONGESTION 13 +#endif + +#include "Gapil.h" +#include "macros.h" #define MAXLINE 256 void usage(void); @@ -52,15 +60,17 @@ int main(int argc, char *argv[]) * Variables definition */ int sock, i; + socklen_t len; int reset = 0; - struct sockaddr_in serv_add; + int verbosity = 0; + char buffer[MAXLINE]; struct linger ling; /* * Input section: decode parameters passed in the calling * Use getopt function */ opterr = 0; /* don't want writing to stderr */ - while ( (i = getopt(argc, argv, "hr")) != -1) { + while ( (i = getopt(argc, argv, "hrv")) != -1) { switch (i) { /* * Handling options @@ -70,6 +80,9 @@ int main(int argc, char *argv[]) usage(); return(1); break; + case 'v': + verbosity = 1; + break; case 'r': reset = 1; break; @@ -87,24 +100,20 @@ int main(int argc, char *argv[]) * Main code beginning * * ***********************************************************/ - /* create socket */ - if ( (sock = socket(AF_INET, SOCK_STREAM, 0)) < 0) { - perror("Socket creation error"); + /* call sockconn to get a connected socket */ + if ( (sock = sockconn(argv[optind], "echo", 6, SOCK_STREAM)) < 0) { + if (errno) perror("Socket creation error"); return 1; } - /* initialize address */ - memset((void *) &serv_add, 0, sizeof(serv_add)); /* clear server address */ - serv_add.sin_family = AF_INET; /* address type is INET */ - serv_add.sin_port = htons(7); /* echo port is 7 */ - /* build address using inet_pton */ - if ( (inet_pton(AF_INET, argv[optind], &serv_add.sin_addr)) <= 0) { - perror("Address creation error"); - return 1; - } - /* extablish connection */ - if (connect(sock, (struct sockaddr *)&serv_add, sizeof(serv_add)) < 0) { - perror("Connection error"); - return 1; + /* print some info about the socket, used to test some TCP_* options */ + if (verbosity) { + len = sizeof(buffer); + if (getsockopt(sock, SOL_TCP, TCP_CONGESTION, buffer, &len) < 0) { + perror("Cannot read congestion algorithm"); + } else { + buffer[len]=0; + printf("Congestion algorithm %s\n", buffer); + } } /* check if resetting on close is required */ if (reset) { @@ -138,26 +147,50 @@ void ClientEcho(FILE * filein, int socket) { char sendbuff[MAXLINE+1], recvbuff[MAXLINE+1]; int nread, nwrite; - while (fgets(sendbuff, MAXLINE, filein) != NULL) { - nwrite = FullWrite(socket, sendbuff, strlen(sendbuff)); - if (nwrite < 0) { - printf("Errore in scrittura: %s", strerror(errno)); - return; - } - nread = read(socket, recvbuff, strlen(sendbuff)); - if (nread < 0) { - printf("Errore in lettura: %s\n", strerror(errno)); - return; + int maxfd; + fd_set fset; + int eof = 0; + /* initialize file descriptor set */ + FD_ZERO(&fset); + maxfd = max(fileno(filein), socket) + 1; + while (1) { + FD_SET(socket, &fset); /* set for the socket */ + if (eof == 0) { + FD_SET(fileno(filein), &fset); /* set for the standard input */ } - if (nread == 0) { - printf("EOF sul socket\n"); - return; + select(maxfd, &fset, NULL, NULL, NULL); /* wait for read ready */ + if (FD_ISSET(fileno(filein), &fset)) { /* if ready on stdin */ + if (fgets(sendbuff, MAXLINE, filein) == NULL) { /* if no input */ + eof = 1; /* EOF on input */ + shutdown(socket, SHUT_WR); /* close write half */ + FD_CLR(fileno(filein), &fset); /* no more interest on stdin */ + } else { /* else we have to write to socket */ + nwrite = FullWrite(socket, sendbuff, strlen(sendbuff)); + if (nwrite < 0) { /* on error stop */ + printf("Errore in scrittura: %s", strerror(errno)); + return; + } + } } - recvbuff[nread] = 0; - if (fputs(recvbuff, stdout) == EOF) { - perror("Errore in scrittura su terminale"); - return; + if (FD_ISSET(socket, &fset)) { /* if ready on socket */ + nread = read(socket, recvbuff, strlen(sendbuff)); /* do read */ + if (nread < 0) { /* error condition, stop client */ + printf("Errore in lettura: %s\n", strerror(errno)); + return; + } + if (nread == 0) { /* server closed connection, stop */ + if (eof == 1) { + return; + } else { + printf("EOF prematuro sul socket\n"); + return; + } + } + recvbuff[nread] = 0; /* else read is ok, write on stdout */ + if (fputs(recvbuff, stdout) == EOF) { + perror("Errore in scrittura su terminale"); + return; + } } } - return; }