server_socket.cpp
Go to the documentation of this file.
1 /*
2  * @brief server_socket implements a server socket connection.
3  *
4  * Copyright (C) 2021 Ing.-Buero Dr. Michael Lehning, Hildesheim
5  * Copyright (C) 2021 SICK AG, Waldkirch
6  *
7  * Licensed under the Apache License, Version 2.0 (the "License");
8  * you may not use this file except in compliance with the License.
9  * You may obtain a copy of the License at
10  *
11  * http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing, software
14  * distributed under the License is distributed on an "AS IS" BASIS,
15  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16  * See the License for the specific language governing permissions and
17  * limitations under the License.
18  *
19  * All rights reserved.
20  *
21  * Redistribution and use in source and binary forms, with or without
22  * modification, are permitted provided that the following conditions are met:
23  *
24  * * Redistributions of source code must retain the above copyright
25  * notice, this list of conditions and the following disclaimer.
26  * * Redistributions in binary form must reproduce the above copyright
27  * notice, this list of conditions and the following disclaimer in the
28  * documentation and/or other materials provided with the distribution.
29  * * Neither the name of SICK AG nor the names of its
30  * contributors may be used to endorse or promote products derived from
31  * this software without specific prior written permission
32  * * Neither the name of Ing.-Buero Dr. Michael Lehning nor the names of its
33  * contributors may be used to endorse or promote products derived from
34  * this software without specific prior written permission
35  *
36  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
37  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
38  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
39  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
40  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
41  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
42  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
43  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
44  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
45  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
46  * POSSIBILITY OF SUCH DAMAGE.
47  *
48  * Authors:
49  * Michael Lehning <michael.lehning@lehning.de>
50  *
51  * Copyright 2021 SICK AG
52  * Copyright 2021 Ing.-Buero Dr. Michael Lehning
53  *
54  */
55 #include <chrono>
56 #include <thread>
57 
58 #include "sick_scan/ros_wrapper.h"
60 
61 #ifndef SOCKET_ERROR
62 #define SOCKET_ERROR (-1)
63 #endif
64 
65 #ifndef SD_BOTH
66 #define SD_BOTH 0x02
67 #endif
68 
69 #ifndef _MSC_VER
70  typedef struct sockaddr SOCKADDR;
71  typedef struct sockaddr_in SOCKADDR_IN;
72  static int closesocket ( int fd )
73  {
74  return close(fd);
75  }
76 #endif
77 
78 #define TCP_SELECT_BEFORE_ACCEPT 0
79 #define TCP_SELECT_BEFORE_READ 0
80 #define TCP_SELECT_BEFORE_WRITE 0
81 
82 #if defined _MSC_VER && defined min
83 #undef min
84 #endif
85 #if defined _MSC_VER && defined max
86 #undef max
87 #endif
88 
92 sick_scan_xd::ServerSocket::ServerSocket() : m_iListenPortNumber(-1), m_tListenSocket(INVALID_SOCKET), m_tConnectedSocket(INVALID_SOCKET)
93 {
94 }
95 
100 {
101 }
102 
107 bool sick_scan_xd::ServerSocket::open(int tcp_port, bool bTcpAnyHost)
108 {
109  SOCKADDR_IN tSockAddr;
110  int reuseaddr = 1;
111 
112  // Initialisation
113 
114  memset(&tSockAddr,0,sizeof(tSockAddr));
115  m_iListenPortNumber = tcp_port;
116  m_tListenSocket = INVALID_SOCKET;
117  m_tConnectedSocket = INVALID_SOCKET;
118 
119  // Create listening socket
120 
121  m_tListenSocket = ::socket( PF_INET, SOCK_STREAM, IPPROTO_TCP );
122  if( m_tListenSocket == INVALID_SOCKET )
123  {
124  ROS_ERROR_STREAM("## ERROR ServerSocket::open(" << m_iListenPortNumber << "," << bTcpAnyHost << "): can't open socket");
125  return false;
126  }
127 
128  tSockAddr.sin_family = AF_INET;
129  tSockAddr.sin_port = htons((unsigned short)m_iListenPortNumber);
130  if( bTcpAnyHost )
131  {
132  tSockAddr.sin_addr.s_addr = htonl(INADDR_ANY);
133  }
134  else
135  {
136  tSockAddr.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
137  }
138 
139  bool bOkay = true;
140  bOkay = bOkay && (setsockopt(m_tListenSocket,SOL_SOCKET,SO_REUSEADDR,(const char*)&reuseaddr,sizeof(reuseaddr)) != SOCKET_ERROR);
141  bOkay = bOkay && (bind(m_tListenSocket, (SOCKADDR*)&tSockAddr, sizeof(tSockAddr)) == 0);
142  bOkay = bOkay && (listen(m_tListenSocket, 1) == 0);
143 
144  if( !bOkay )
145  {
146  ROS_ERROR_STREAM("## ERROR ServerSocket::open(" << m_iListenPortNumber << "," << bTcpAnyHost << "): bind socket resp. listen failed.");
147  closesocket(m_tListenSocket);
148  m_tListenSocket = INVALID_SOCKET;
149  }
150 
151  ROS_INFO_STREAM("ServerSocket: listening to port " << m_iListenPortNumber);
152  return bOkay;
153 }
154 
160 {
161  // Socket okay?
162  if (m_tListenSocket == INVALID_SOCKET)
163  {
164  ROS_ERROR_STREAM("## ERROR ServerSocket::connect(port " << m_iListenPortNumber << "): socket not initialized, call open() before connect().");
165  return false;
166  }
167 
168  // Wait for and accept a new connection
169 # if UTL_TCP_SELECT_BEFORE_ACCEPT
170  {
171  fd_set readfds;
172  FD_ZERO( &readfds );
173  FD_SET(m_tListenSocket, &readfds);
174  if (select(0, &readfds, NULL, NULL, NULL) <= 0 || FD_ISSET(m_tListenSocket, &readfds) == 0)
175  {
176  ROS_ERROR_STREAM("## ERROR ServerSocket::connect(port " << m_iListenPortNumber << "): select failed.");
177  return false;
178  }
179  }
180 # endif // TCP_SELECT_BEFORE_ACCEPT
181 
182  m_tConnectedSocket = accept(m_tListenSocket, NULL, NULL);
183  if (m_tConnectedSocket == INVALID_SOCKET)
184  {
185  ROS_ERROR_STREAM("## ERROR ServerSocket::connect(port " << m_iListenPortNumber << "): accept failed.");
186  return false;
187  }
188 
189  ROS_INFO_STREAM("ServerSocket (port " << m_iListenPortNumber << "): connected to client.");
190  return true;
191 }
192 
197 int sick_scan_xd::ServerSocket::read(int num_bytes, std::vector<uint8_t>& out_buffer, bool read_blocking)
198 {
199  if( m_tListenSocket == INVALID_SOCKET || m_tConnectedSocket == INVALID_SOCKET )
200  {
201  ROS_ERROR_STREAM("## ERROR ServerSocket::read(port " << m_iListenPortNumber << "): socket not connected");
202  return -1;
203  }
204  if( num_bytes <= 0 )
205  {
206  return 0;
207  }
208 
209 # if TCP_SELECT_BEFORE_READ
210  {
211  fd_set readfds;
212  FD_ZERO(&readfds);
213  FD_SET(m_tConnectedSocket, &readfds);
214  if (select(0, &readfds, NULL, NULL, NULL) <= 0 || FD_ISSET(m_tConnectedSocket, &readfds ) == 0)
215  {
216  ROS_ERROR_STREAM("## ERROR ServerSocket::read(port " << m_iListenPortNumber << "): select failed.");
217  return -1;
218  }
219  }
220 # endif
221 
222  // set socket to nonblocking mode
223  int recv_flags = 0;
224 # ifdef _MSC_VER
225  u_long recv_mode = (read_blocking ? 0 : 1); // FIONBIO enables or disables the blocking mode for the socket. If iMode = 0, blocking is enabled, if iMode != 0, non-blocking mode is enabled.
226  ioctlsocket(m_tConnectedSocket, FIONBIO, &recv_mode);
227 # else
228  if(!read_blocking)
229  recv_flags |= MSG_DONTWAIT;
230 # endif
231 
232  std::vector<uint8_t> buffer(num_bytes);
233  int nrbytes = 0;
234  while (ROS::ok() && nrbytes < num_bytes && m_tListenSocket != INVALID_SOCKET && m_tConnectedSocket != INVALID_SOCKET)
235  {
236  int n = ::recv(m_tConnectedSocket, (char*)(buffer.data() + nrbytes), num_bytes - nrbytes, recv_flags);
237  if(n <= 0)
238  {
239  if(read_blocking)
240  {
241  std::this_thread::sleep_for(std::chrono::milliseconds(1));
242  continue;
243  }
244  else
245  {
246  break;
247  }
248  }
249  nrbytes += n;
250  }
251  out_buffer.insert(out_buffer.end(), buffer.begin(), buffer.end());
252 
253  if (read_blocking && nrbytes < num_bytes)
254  {
255  ROS_ERROR_STREAM("## ERROR ServerSocket::read(port " << m_iListenPortNumber << "): failed, " << nrbytes << " of " << num_bytes << " bytes read from socket.");
256  }
257  return nrbytes;
258 }
259 
264 bool sick_scan_xd::ServerSocket::read(int num_bytes, uint8_t* out_buffer, bool read_blocking)
265 {
266  std::vector<uint8_t> buffer;
267  int nrbytes = read(num_bytes, buffer, read_blocking);
268  if(nrbytes > 0)
269  memcpy(out_buffer, buffer.data(), std::min(nrbytes, num_bytes));
270  return nrbytes == num_bytes;
271 }
272 
277 bool sick_scan_xd::ServerSocket::write(const uint8_t* buffer, int num_bytes, int num_retries_on_error)
278 {
279 
280  if (m_tConnectedSocket == INVALID_SOCKET)
281  {
282  ROS_ERROR_STREAM("## ERROR ServerSocket::write(port " << m_iListenPortNumber << "): socket not connected");
283  return false;
284  }
285  if (buffer == NULL || num_bytes <= 0)
286  {
287  return false;
288  }
289 
290  int nrbytes = 0, nrErrors = 0;
291  num_retries_on_error = std::max(num_retries_on_error, 1);
292  while (ROS::ok() && num_bytes > 0 && nrErrors < num_retries_on_error && m_tListenSocket != INVALID_SOCKET && m_tConnectedSocket != INVALID_SOCKET)
293  {
294 # if TCP_SELECT_BEFORE_WRITE
295  fd_set writefds;
296  FD_ZERO( &writefds );
297  FD_SET( m_tConnectedSocket, &writefds );
298  if (select( 0, NULL, &writefds, NULL, NULL) <= 0 || FD_ISSET( ptServerSocket->tConnectedSocket, &writefds ) == 0)
299  {
300  ROS_ERROR_STREAM("## ERROR ServerSocket::write(port " << m_iListenPortNumber << "): select failed.");
301  return false;
302  }
303 # endif
304  nrbytes = send(m_tConnectedSocket, (char*)buffer, num_bytes, 0);
305  if( nrbytes > 0 )
306  {
307  num_bytes -= nrbytes;
308  }
309  else if(nrbytes < 0)
310  {
311  nrErrors++;
312  }
313  }
314  if( num_bytes > 0 )
315  {
316  ROS_ERROR_STREAM("## ERROR ServerSocket::write(port " << m_iListenPortNumber << "): failed to send " << num_bytes << " bytes.");
317  return false;
318  }
319  return true;
320 }
321 
326 {
327  if(m_tConnectedSocket != INVALID_SOCKET)
328  {
329  shutdown(m_tConnectedSocket,SD_BOTH);
330  closesocket(m_tConnectedSocket);
331  m_tConnectedSocket = INVALID_SOCKET;
332  }
333  if(m_tListenSocket != INVALID_SOCKET)
334  {
335  shutdown(m_tListenSocket,SD_BOTH);
336  closesocket(m_tListenSocket);
337  m_tListenSocket = INVALID_SOCKET;
338  }
339 }
340 
345 {
346  return (m_tConnectedSocket != INVALID_SOCKET && m_tListenSocket != INVALID_SOCKET);
347 }
SOCKET_ERROR
#define SOCKET_ERROR
Definition: server_socket.cpp:62
NULL
#define NULL
sick_scan_xd::ServerSocket::~ServerSocket
virtual ~ServerSocket()
Definition: server_socket.cpp:99
sick_scan_xd::ServerSocket::open
virtual bool open(int tcp_port, bool bTcpAnyHost=false)
Definition: server_socket.cpp:107
SOCKADDR_IN
struct sockaddr_in SOCKADDR_IN
Definition: server_socket.cpp:71
roswrap::console::shutdown
ROSCONSOLE_DECL void shutdown()
Definition: rossimu.cpp:269
sick_scan_xd::ServerSocket::is_open
virtual bool is_open(void)
Definition: server_socket.cpp:344
ROS_INFO_STREAM
#define ROS_INFO_STREAM(...)
Definition: sick_scan_ros2_example.cpp:71
sick_scan_xd::ServerSocket::write
virtual bool write(const uint8_t *buffer, int num_bytes, int num_retries_on_error=1)
Definition: server_socket.cpp:277
server_socket.h
closesocket
static int closesocket(int fd)
Definition: server_socket.cpp:72
test_server.tcp_port
int tcp_port
Definition: test_server.py:204
sick_scan_xd::ServerSocket::close
virtual void close(void)
Definition: server_socket.cpp:325
SD_BOTH
#define SD_BOTH
Definition: server_socket.cpp:66
colaa::detail::read
T read(const std::string &str)
General template which is unimplemented; implemented specializations follow below.
Definition: colaa.hpp:72
sick_scan_xd::ServerSocket::connect
virtual bool connect()
Definition: server_socket.cpp:159
ROS_ERROR_STREAM
#define ROS_ERROR_STREAM(...)
Definition: sick_scan_ros2_example.cpp:72
sick_scan_xd::ServerSocket::read
virtual int read(int num_bytes, std::vector< uint8_t > &out_buffer, bool read_blocking=true)
Definition: server_socket.cpp:197
roswrap::ok
ROSCPP_DECL bool ok()
Check whether it's time to exit.
Definition: rossimu.cpp:273
ros_wrapper.h
SOCKADDR
struct sockaddr SOCKADDR
Definition: udp_sockets.h:82
INVALID_SOCKET
#define INVALID_SOCKET
Definition: udp_sockets.h:83
sick_scan_xd::ServerSocket::ServerSocket
ServerSocket()
Definition: server_socket.cpp:92


sick_scan_xd
Author(s): Michael Lehning , Jochen Sprickerhof , Martin Günther
autogenerated on Fri Oct 25 2024 02:47:10