ViSP  2.8.0
vpServer.cpp
1 /****************************************************************************
2  *
3  * $Id: vpServer.cpp 4056 2013-01-05 13:04:42Z fspindle $
4  *
5  * This file is part of the ViSP software.
6  * Copyright (C) 2005 - 2013 by INRIA. All rights reserved.
7  *
8  * This software is free software; you can redistribute it and/or
9  * modify it under the terms of the GNU General Public License
10  * ("GPL") version 2 as published by the Free Software Foundation.
11  * See the file LICENSE.txt at the root directory of this source
12  * distribution for additional information about the GNU GPL.
13  *
14  * For using ViSP with software that can not be combined with the GNU
15  * GPL, please contact INRIA about acquiring a ViSP Professional
16  * Edition License.
17  *
18  * See http://www.irisa.fr/lagadic/visp/visp.html for more information.
19  *
20  * This software was developed at:
21  * INRIA Rennes - Bretagne Atlantique
22  * Campus Universitaire de Beaulieu
23  * 35042 Rennes Cedex
24  * France
25  * http://www.irisa.fr/lagadic
26  *
27  * If you have questions regarding the use of this file, please contact
28  * INRIA at visp@inria.fr
29  *
30  * This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE
31  * WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
32  *
33  *
34  * Description:
35  * TCP Server
36  *
37  * Authors:
38  * Aurelien Yol
39  *
40  *****************************************************************************/
41 
42 #include <visp/vpServer.h>
43 
44 
48 vpServer::vpServer( ) : vpNetwork(), started(false)
49 {
50  int protocol = 0;
51  emitter.socketFileDescriptorEmitter = socket(AF_INET, SOCK_STREAM, protocol);
52 #ifdef UNIX
54 #else
55  if (emitter.socketFileDescriptorEmitter == INVALID_SOCKET)
56 #endif
57  {
58  vpERROR_TRACE( "vpServer::vpServer(), cannot open socket." );
59  }
60  emitter.emitterAdress.sin_family = AF_INET;
61  emitter.emitterAdress.sin_addr.s_addr = INADDR_ANY;
62  emitter.emitterAdress.sin_port = 0;
63 
64  adress = inet_ntoa(emitter.emitterAdress.sin_addr);
65  port = emitter.emitterAdress.sin_port;
66 }
67 
73 vpServer::vpServer( const int &port_serv ) : vpNetwork(), started(false)
74 {
75  int protocol = 0;
76  emitter.socketFileDescriptorEmitter = socket(AF_INET, SOCK_STREAM, protocol);
77 #ifdef UNIX
79 #else
80  if (emitter.socketFileDescriptorEmitter == INVALID_SOCKET)
81 #endif
82  {
83  vpERROR_TRACE( "vpServer::vpServer(const int &port_serv), cannot open socket." );
84  }
85  emitter.emitterAdress.sin_family = AF_INET;
86  emitter.emitterAdress.sin_addr.s_addr = INADDR_ANY; //inet_addr("127.0.0.1");;
87  emitter.emitterAdress.sin_port = htons( (unsigned short)port_serv );
88 
89  adress = inet_ntoa(emitter.emitterAdress.sin_addr);
90  port = port_serv;
91 }
92 
99 vpServer::vpServer( const std::string &adress_serv,const int &port_serv ) : vpNetwork(), started(false)
100 {
101  int protocol = 0;
102  emitter.socketFileDescriptorEmitter = socket(AF_INET, SOCK_STREAM, protocol);
103 #ifdef UNIX
105 #else
106  if (emitter.socketFileDescriptorEmitter == INVALID_SOCKET)
107 #endif
108  {
109  vpERROR_TRACE( "vpServer::vpServer(const std::string &adress_serv,const int &port_serv), cannot open socket." );
110  }
111  emitter.emitterAdress.sin_family = AF_INET;
112  emitter.emitterAdress.sin_addr.s_addr = inet_addr(adress_serv.c_str());
113  emitter.emitterAdress.sin_port = htons( (unsigned short)port_serv );
114 
115  adress = adress_serv;
116  port = port_serv;
117 }
118 
123 {
124 #ifdef UNIX
126 #else //Win32
127  closesocket( (unsigned)emitter.socketFileDescriptorEmitter );
128 #endif
129 
130  for(unsigned int i = 0 ; i < receptor_list.size() ; i++)
131 #ifdef UNIX
132  close( receptor_list[i].socketFileDescriptorReceptor );
133 #else //Win32
134  closesocket( (unsigned)receptor_list[i].socketFileDescriptorReceptor );
135 #endif
136 
137 }
138 
145 {
146  int serverStructLength = sizeof(emitter.emitterAdress);
147 #ifdef UNIX
148  int bindResult = bind( emitter.socketFileDescriptorEmitter, (struct sockaddr *) &emitter.emitterAdress, (unsigned)serverStructLength );
149 #else //Win32
150  int bindResult = bind( (unsigned)emitter.socketFileDescriptorEmitter, (struct sockaddr *) &emitter.emitterAdress, serverStructLength );
151 #endif
152 
153 
154  if( bindResult < 0 )
155  {
156  std::cout << "Error id : " << bindResult << std::endl;
157  std::string errorMessage( "vpServer::vpServer(), cannot bind to port" );
158  char posrtNumberString[32];
159  sprintf( posrtNumberString, "%d", port );
160  errorMessage += " ";
161  errorMessage += posrtNumberString;
162  errorMessage += " The port may be already used.";
163  vpERROR_TRACE( errorMessage.c_str() );
164  return false;
165  }
166 
167 #ifdef SO_NOSIGPIPE
168  // Mac OS X does not have the MSG_NOSIGNAL flag. It does have this
169  // connections based version, however.
171  int set_option = 1;
172  if (0 == setsockopt(emitter.socketFileDescriptorEmitter, SOL_SOCKET, SO_NOSIGPIPE, &set_option, sizeof(set_option))) {
173  } else {
174  std::cout << "Failed to set socket signal option" << std::endl;
175  }
176  }
177 #endif // SO_NOSIGPIPE
178 
179 #ifdef UNIX
180  listen( emitter.socketFileDescriptorEmitter, (int)max_clients );
181 #else //Win32
182  listen( (unsigned)emitter.socketFileDescriptorEmitter, (int)max_clients );
183 #endif
184 
185  std::cout << "Server ready" << std::endl;
186 
187  started = true;
188 
189  return true;
190 }
191 
198 {
199  if(!started)
200  if(!start()){
201  return false;
202  }
203 
204  tv.tv_sec = tv_sec;
205  tv.tv_usec = tv_usec;
206 
207  FD_ZERO(&readFileDescriptor);
208 
211 
212  for(unsigned int i=0; i<receptor_list.size(); i++){
213  FD_SET((unsigned)receptor_list[i].socketFileDescriptorReceptor,&readFileDescriptor);
214 
215  if(i == 0)
216  socketMax = receptor_list[i].socketFileDescriptorReceptor;
217 
218  if(socketMax < receptor_list[i].socketFileDescriptorReceptor) socketMax = receptor_list[i].socketFileDescriptorReceptor;
219  }
220 
221  int value = select((int)socketMax+1,&readFileDescriptor,NULL,NULL,&tv);
222  if(value == -1){
223  //vpERROR_TRACE( "vpServer::run(), select()" );
224  return false;
225  }
226  else if(value == 0){
227  return false;
228  }
229  else{
230  if(FD_ISSET((unsigned int)emitter.socketFileDescriptorEmitter,&readFileDescriptor)){
231  vpNetwork::vpReceptor client;
232  client.receptorAddressSize = sizeof(client.receptorAddress);
233 #ifdef UNIX
234  client.socketFileDescriptorReceptor = accept(emitter.socketFileDescriptorEmitter,(struct sockaddr*) &client.receptorAddress, &client.receptorAddressSize);
235 #else //Win32
236  client.socketFileDescriptorReceptor = accept((unsigned int)emitter.socketFileDescriptorEmitter,(struct sockaddr*) &client.receptorAddress, &client.receptorAddressSize);
237 #endif
238 
239 #ifdef UNIX
240  if((client.socketFileDescriptorReceptor) == -1)
241 #else
242  if((client.socketFileDescriptorReceptor) == INVALID_SOCKET)
243 #endif
244  vpERROR_TRACE( "vpServer::run(), accept()" );
245 
246  client.receptorIP = inet_ntoa(client.receptorAddress.sin_addr);
247  printf("New client connected : %s\n", inet_ntoa(client.receptorAddress.sin_addr));
248  receptor_list.push_back(client);
249 
250  return true;
251  }
252  else{
253  for(unsigned int i=0; i<receptor_list.size(); i++){
254  if(FD_ISSET((unsigned int)receptor_list[i].socketFileDescriptorReceptor,&readFileDescriptor)){
255  char deco;
256 #ifdef UNIX
257  int numbytes = recv(receptor_list[i].socketFileDescriptorReceptor, &deco, 1, MSG_PEEK);
258 #else //Win32
259  int numbytes = recv((unsigned int)receptor_list[i].socketFileDescriptorReceptor, &deco, 1, MSG_PEEK);
260 #endif
261 
262 
263  if(numbytes == 0)
264  {
265  std::cout << "Disconnected : " << inet_ntoa(receptor_list[i].receptorAddress.sin_addr) << std::endl;
266  receptor_list.erase(receptor_list.begin()+(int)i);
267  return 0;
268  }
269  }
270  }
271  }
272  }
273 
274  return false;
275 }
276 
281 {
282  vpNetwork::print("Client");
283 }
284 
socklen_t receptorAddressSize
Definition: vpNetwork.h:90
fd_set readFileDescriptor
Definition: vpNetwork.h:114
int socketFileDescriptorEmitter
Definition: vpNetwork.h:102
#define vpERROR_TRACE
Definition: vpDebug.h:379
void print(const char *id="")
Definition: vpNetwork.cpp:125
bool checkForConnections()
Definition: vpServer.cpp:197
This class represents a Transmission Control Protocol (TCP) network.
Definition: vpNetwork.h:83
bool start()
Definition: vpServer.cpp:144
int socketFileDescriptorReceptor
Definition: vpNetwork.h:89
vpServer()
Definition: vpServer.cpp:48
int socketMax
Definition: vpNetwork.h:116
vpEmitter emitter
Definition: vpNetwork.h:112
long tv_sec
Definition: vpNetwork.h:133
virtual ~vpServer()
Definition: vpServer.cpp:122
struct timeval tv
Definition: vpNetwork.h:132
std::vector< vpReceptor > receptor_list
Definition: vpNetwork.h:113
long tv_usec
Definition: vpNetwork.h:134
void print()
Definition: vpServer.cpp:280
struct sockaddr_in receptorAddress
Definition: vpNetwork.h:95
std::string receptorIP
Definition: vpNetwork.h:96
struct sockaddr_in emitterAdress
Definition: vpNetwork.h:100