Materiale su splice, ed esempio non funzionante
[gapil.git] / sources / TCP_echo_fifth.c
1 /* TCP_echo.c
2  * 
3  * Copyright (C) 2001-2003 Simone Piccardi
4  * 
5  * This program is free software; you can redistribute it and/or modify
6  * it under the terms of the GNU General Public License as published by
7  * the Free Software Foundation; either version 2 of the License, or (at
8  * your option) any later version.
9  * 
10  * This program is distributed in the hope that it will be useful, but
11  * WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * General Public License for more details.
14  * 
15  * You should have received a copy of the GNU General Public License
16  * along with this program; if not, write to the Free Software
17  * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
18  */
19 /****************************************************************
20  *
21  * Program TCP_echo.c
22  * Simple TCP client for echo service (port 7)
23  *
24  * Author: Simone Piccardi
25  * Jun. 2001
26  *
27  * Usage: echo -h give all info's
28  *
29  ****************************************************************/
30 /* 
31  * Include needed headers
32  */
33 #include <sys/types.h>   /* primitive system data types */
34 #include <unistd.h>      /* unix standard library */
35 #include <arpa/inet.h>   /* IP addresses conversion utilities */
36 #include <sys/socket.h>  /* socket constants, types and functions */
37 #include <stdio.h>       /* standard I/O library */
38 #include <errno.h>       /* error definitions and routines */
39 #include <string.h>      /* C strings library */
40
41 #include "Gapil.h"
42 #include "macros.h"
43
44 #define MAXLINE 256
45 void usage(void);
46 void ClientEcho(FILE * filein, int socket);
47 void SigTERM_hand(int sig);
48
49 /* Program begin */
50 int main(int argc, char *argv[])
51 {
52 /* 
53  * Variables definition  
54  */
55     int sock, i;
56     int reset = 0;
57     struct linger ling;
58     /*
59      * Input section: decode parameters passed in the calling 
60      * Use getopt function
61      */
62     opterr = 0;  /* don't want writing to stderr */
63     while ( (i = getopt(argc, argv, "hr")) != -1) {
64         switch (i) {
65         /* 
66          * Handling options 
67          */ 
68         case 'h':  
69             printf("Wrong -h option use\n");
70             usage();
71             return(1);
72             break;
73         case 'r':
74             reset = 1;
75             break;
76         case '?':   /* unrecognized options */
77             printf("Unrecognized options -%c\n",optopt);
78             usage();
79         default:    /* should not reached */
80             usage();
81         }
82     }
83     /* ***********************************************************
84      * 
85      *           Options processing completed
86      *
87      *                Main code beginning
88      * 
89      * ***********************************************************/
90     /* call sockaddr to get a connected socket */
91     if ( (sock = sockconn(argv[optind], "echo", 6, SOCK_STREAM)) < 0) {
92         if (errno) perror("Socket creation error");
93         return 1;
94     }
95     /* check if resetting on close is required */
96     if (reset) {
97         printf("Setting reset on close \n");
98         ling.l_onoff = 1;
99         ling.l_linger = 0;      
100         if (setsockopt(sock, SOL_SOCKET, SO_LINGER, &ling, sizeof(ling))) {
101             perror("Cannot set linger");
102             exit(1);
103         }
104     }
105     /* do read/write operations */
106     ClientEcho(stdin, sock);
107     /* normal exit */
108     return 0;
109 }
110 /*
111  * routine to print usage info and exit 
112  */
113 void usage(void) {
114     printf("Take daytime from a remote host \n");
115     printf("Usage:\n");
116     printf("  daytime [-h] [-v] [host in dotted decimal form] \n");
117 //    printf("  -v         set verbosity on\n");
118     printf("  -r           require reset on closing\n");
119     printf("  -h           print this help\n");
120     exit(1);
121 }
122
123 void ClientEcho(FILE * filein, int socket) 
124 {
125     char sendbuff[MAXLINE+1], recvbuff[MAXLINE+1];
126     int nread, nwrite; 
127     int maxfd;
128     fd_set fset;
129     int eof = 0;
130     /* initialize file descriptor set */
131     FD_ZERO(&fset);
132     maxfd = max(fileno(filein), socket) + 1;
133     while (1) {
134         FD_SET(socket, &fset);         /* set for the socket */
135         if (eof == 0) {
136             FD_SET(fileno(filein), &fset); /* set for the standard input */
137         }
138         select(maxfd, &fset, NULL, NULL, NULL); /* wait for read ready */
139         if (FD_ISSET(fileno(filein), &fset)) {  /* if ready on stdin */
140             if (fgets(sendbuff, MAXLINE, filein) == NULL) { /* if no input */
141                 eof = 1;               /* EOF on input */
142                 shutdown(socket, SHUT_WR);      /* close write half */
143                 FD_CLR(fileno(filein), &fset);  /* no more interest on stdin */
144             } else {                   /* else we have to write to socket */
145                 nwrite = FullWrite(socket, sendbuff, strlen(sendbuff)); 
146                 if (nwrite < 0) {      /* on error stop */
147                     printf("Errore in scrittura: %s", strerror(errno));
148                     return;
149                 }
150             }
151         }
152         if (FD_ISSET(socket, &fset)) { /* if ready on socket */ 
153             nread = read(socket, recvbuff, strlen(sendbuff)); /* do read */
154             if (nread < 0) {  /* error condition, stop client */
155                 printf("Errore in lettura: %s\n", strerror(errno));
156                 return;
157             }
158             if (nread == 0) { /* server closed connection, stop */
159                 if (eof == 1) {
160                     return;
161                 } else {
162                     printf("EOF prematuro sul socket\n");
163                     return;
164                 }
165             }
166             recvbuff[nread] = 0;   /* else read is ok, write on stdout */
167             if (fputs(recvbuff, stdout) == EOF) {
168                 perror("Errore in scrittura su terminale");
169                 return;
170             }
171         }
172     }
173 }