JSBSim Flight Dynamics Model 1.2.2 (22 Mar 2025)
An Open Source Flight Dynamics and Control Software Library in C++
Loading...
Searching...
No Matches
FGfdmSocket.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: FGfdmSocket.cpp
4 Author: Jon S. Berndt
5 Date started: 11/08/99
6 Purpose: Encapsulates a socket
7 Called by: FGOutput, et. al.
8
9 ------------- Copyright (C) 1999 Jon S. Berndt (jon@jsbsim.org) -------------
10
11 This program is free software; you can redistribute it and/or modify it under
12 the terms of the GNU Lesser General Public License as published by the Free
13 Software Foundation; either version 2 of the License, or (at your option) any
14 later version.
15
16 This program is distributed in the hope that it will be useful, but WITHOUT
17 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
18 FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
19 details.
20
21 You should have received a copy of the GNU Lesser General Public License along
22 with this program; if not, write to the Free Software Foundation, Inc., 59
23 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
24
25 Further information about the GNU Lesser General Public License can also be
26 found on the world wide web at http://www.gnu.org.
27
28FUNCTIONAL DESCRIPTION
29--------------------------------------------------------------------------------
30This class excapsulates a socket for simple data writing
31
32HISTORY
33--------------------------------------------------------------------------------
3411/08/99 JSB Created
3511/08/07 HDW Added Generic Socket Send
36
37%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
38INCLUDES
39%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
40
41#if defined(_MSC_VER) || defined(__MINGW32__)
42#include <ws2tcpip.h>
43#elif defined(__OpenBSD__)
44#include <sys/types.h>
45#include <sys/socket.h>
46#include <netdb.h>
47#include <unistd.h>
48#include <fcntl.h>
49#else
50#include <fcntl.h>
51#include <unistd.h>
52#endif
53#include <iomanip>
54#include <iostream>
55#include <sstream>
56#include <cstring>
57#include <assert.h>
58
59#include "FGfdmSocket.h"
60#include "input_output/string_utilities.h"
61
62using std::cout;
63using std::cerr;
64using std::endl;
65using std::string;
66
67// Defines that make BSD/Unix sockets and Windows sockets syntax look alike.
68#ifndef _WIN32
69#define closesocket close
70#define INVALID_SOCKET -1
71#define SOCKET_ERROR -1
72#define SD_BOTH SHUT_RDWR
73#endif
74
75namespace JSBSim {
76
77/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
78CLASS IMPLEMENTATION
79%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
80
81#ifdef _WIN32
82static bool LoadWinSockDLL(int debug_lvl)
83{
84 WSADATA wsaData;
85 if (WSAStartup(MAKEWORD(1, 1), &wsaData)) {
86 cerr << "Winsock DLL not initialized ..." << endl;
87 return false;
88 }
89
90 if (debug_lvl > 0)
91 cout << "Winsock DLL loaded ..." << endl;
92
93 return true;
94}
95#endif
96
97FGfdmSocket::FGfdmSocket(const string& address, int port, int protocol, int precision)
98{
99 sckt = sckt_in = INVALID_SOCKET;
100 Protocol = (ProtocolType)protocol;
101 connected = false;
102 struct addrinfo *addr = nullptr;
103 this->precision = precision;
104
105#ifdef _WIN32
106 if (!LoadWinSockDLL(debug_lvl)) return;
107#endif
108
109 struct addrinfo hints;
110 memset(&hints, 0, sizeof(struct addrinfo));
111 hints.ai_family = AF_INET;
112 if (protocol == ptUDP)
113 hints.ai_socktype = SOCK_DGRAM;
114 else
115 hints.ai_socktype = SOCK_STREAM;
116 hints.ai_protocol = 0;
117 if (!is_number(address))
118 hints.ai_flags = AI_ADDRCONFIG;
119 else
120 hints.ai_flags = AI_NUMERICHOST;
121
122 int failure = getaddrinfo(address.c_str(), NULL, &hints, &addr);
123 if (failure || !addr) {
124 cerr << "Could not get host net address " << address;
125
126 if (hints.ai_flags == AI_NUMERICHOST)
127 cerr << " by number..." << endl;
128 else
129 cerr << " by name..." << endl;
130
131 cerr << gai_strerror(failure) << endl;
132
133 freeaddrinfo(addr);
134 return;
135 }
136
137 sckt = socket(addr->ai_family, addr->ai_socktype, addr->ai_protocol);
138
139 if (debug_lvl > 0) {
140 if (protocol == ptUDP) //use udp protocol
141 cout << "Creating UDP socket on port " << port << endl;
142 else //use tcp protocol
143 cout << "Creating TCP socket on port " << port << endl;
144 }
145
146 if (sckt != INVALID_SOCKET) { // successful
147 int len = sizeof(struct sockaddr_in);
148 memcpy(&scktName, addr->ai_addr, len);
149 scktName.sin_port = htons(port);
150
151 if (connect(sckt, (struct sockaddr*)&scktName, len) == 0) { // successful
152 if (debug_lvl > 0)
153 cout << "Successfully connected to socket for output ..." << endl;
154 connected = true;
155 } else // unsuccessful
156 cerr << "Could not connect to socket for output ..." << endl;
157 } else // unsuccessful
158 cerr << "Could not create socket for FDM output, error = " << errno << endl;
159
160 freeaddrinfo(addr);
161
162 Debug(0);
163}
164
165//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
166// assumes TCP or UDP socket on localhost, for inbound datagrams
167FGfdmSocket::FGfdmSocket(int port, int protocol, int precision)
168{
169 sckt = INVALID_SOCKET;
170 connected = false;
171 Protocol = (ProtocolType)protocol;
172 string ProtocolName;
173 this->precision = precision;
174
175#ifdef _WIN32
176 if (!LoadWinSockDLL(debug_lvl)) return;
177#endif
178
179 if (Protocol == ptUDP) { //use udp protocol
180 ProtocolName = "UDP";
181 sckt = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
182#ifdef _WIN32
183 u_long NonBlock = 1; // True
184 ioctlsocket(sckt, FIONBIO, &NonBlock);
185#else
186 int flags = fcntl(sckt, F_GETFL, 0);
187 fcntl(sckt, F_SETFL, flags | O_NONBLOCK);
188#endif
189 }
190 else {
191 ProtocolName = "TCP";
192 sckt = socket(AF_INET, SOCK_STREAM, 0);
193 }
194
195 if (debug_lvl > 0)
196 cout << "Creating input " << ProtocolName << " socket on port " << port
197 << endl;
198
199 if (sckt != INVALID_SOCKET) {
200 memset(&scktName, 0, sizeof(struct sockaddr_in));
201 scktName.sin_family = AF_INET;
202 scktName.sin_port = htons(port);
203
204 if (Protocol == ptUDP)
205 scktName.sin_addr.s_addr = htonl(INADDR_ANY);
206
207 socklen_t len = sizeof(struct sockaddr_in);
208 if (bind(sckt, (struct sockaddr*)&scktName, len) != SOCKET_ERROR) {
209 if (debug_lvl > 0)
210 cout << "Successfully bound to " << ProtocolName
211 << " input socket on port " << port << endl << endl;
212
213 if (Protocol == ptTCP) {
214 if (listen(sckt, 5) != SOCKET_ERROR) { // successful listen()
215 connected = true;
216#ifdef _WIN32
217 u_long NoBlock = 1;
218 ioctlsocket(sckt, FIONBIO, &NoBlock);
219#else
220 int flags = fcntl(sckt, F_GETFL, 0);
221 fcntl(sckt, F_SETFL, flags | O_NONBLOCK);
222#endif
223 sckt_in = accept(sckt, (struct sockaddr*)&scktName, &len);
224 } else {
225 closesocket(sckt);
226 sckt = INVALID_SOCKET;
227 cerr << "Could not listen ..." << endl;
228 }
229 } else
230 connected = true;
231 } else { // unsuccessful
232 closesocket(sckt);
233 sckt = INVALID_SOCKET;
234 cerr << "Could not bind to " << ProtocolName << " input socket, error = "
235 << errno << endl;
236 }
237 } else // unsuccessful
238 cerr << "Could not create " << ProtocolName << " socket for input, error = "
239 << errno << endl;
240
241 Debug(0);
242}
243
244//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
245
246FGfdmSocket::~FGfdmSocket()
247{
248 // Release the file descriptors to the OS.
249 if (sckt_in != INVALID_SOCKET) shutdown(sckt_in, SD_BOTH);
250 if (sckt != INVALID_SOCKET) closesocket(sckt);
251 Debug(1);
252}
253
254//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
255
257{
258 char buf[1024];
259 string data; // TODO should allocate this with a standard size as a
260 // class attribute and pass as a reference?
261
262 if (Protocol == ptTCP){
263 if (sckt_in == INVALID_SOCKET) {
264 socklen_t len = sizeof(struct sockaddr_in);
265 sckt_in = accept(sckt, (struct sockaddr*)&scktName, &len);
266 if (sckt_in != INVALID_SOCKET) {
267#ifdef _WIN32
268 u_long NoBlock = 1;
269 ioctlsocket(sckt_in, FIONBIO, &NoBlock);
270#else
271 int flags = fcntl(sckt_in, F_GETFL, 0);
272 fcntl(sckt_in, F_SETFL, flags | O_NONBLOCK);
273#endif
274 if (send(sckt_in, "Connected to JSBSim server\r\nJSBSim> ", 36, 0) == SOCKET_ERROR)
275 LogSocketError("Receive - TCP connection acknowledgement");
276 }
277 }
278
279 if (sckt_in != INVALID_SOCKET) {
280 int num_chars;
281
282 while ((num_chars = recv(sckt_in, buf, sizeof buf, 0)) > 0)
283 data.append(buf, num_chars);
284
285 if (num_chars == SOCKET_ERROR || num_chars == 0) {
286#ifdef _WIN32
287 if (WSAGetLastError() != WSAEWOULDBLOCK)
288#else
289 if (errno != EWOULDBLOCK)
290#endif
291 {
292 LogSocketError("Receive - TCP data reception");
293 // when nothing received and the error isn't "would block"
294 // then assume that the client has closed the socket.
295 cout << "Socket Closed. Back to listening" << endl;
296 closesocket(sckt_in);
297 sckt_in = INVALID_SOCKET;
298 }
299 }
300 }
301 }
302
303 // this is for FGUDPInputSocket
304 if (sckt != INVALID_SOCKET && Protocol == ptUDP) {
305 struct sockaddr addr;
306 socklen_t fromlen = sizeof addr;
307 int num_chars = recvfrom(sckt, buf, sizeof buf, 0, (struct sockaddr*)&addr, &fromlen);
308 if (num_chars > 0) data.append(buf, num_chars);
309 if (num_chars == SOCKET_ERROR) {
310#ifdef _WIN32
311 if (WSAGetLastError() != WSAEWOULDBLOCK)
312#else
313 if (errno != EWOULDBLOCK)
314#endif
315 LogSocketError("Receive - UDP data reception");
316 }
317 }
318
319 return data;
320}
321
322//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
323
324int FGfdmSocket::Reply(const string& text)
325{
326 int num_chars_sent=0;
327 assert(Protocol == ptTCP);
328
329 if (sckt_in != INVALID_SOCKET) {
330 num_chars_sent = send(sckt_in, text.c_str(), text.size(), 0);
331 if (num_chars_sent == SOCKET_ERROR) LogSocketError("Reply - Send data");
332 if (send(sckt_in, "JSBSim> ", 8, 0) == SOCKET_ERROR) LogSocketError("Reply - Prompt");
333 } else {
334 cerr << "Socket reply must be to a valid socket" << endl;
335 return -1;
336 }
337 return num_chars_sent;
338}
339
340//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
341
343{
344 assert(Protocol == ptTCP);
345 closesocket(sckt_in);
346 sckt_in = INVALID_SOCKET;
347}
348
349//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
350
352{
353 buffer.str("");
354}
355
356//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
357
358void FGfdmSocket::Clear(const string& s)
359{
360 Clear();
361 buffer << s << ' ';
362}
363
364//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
365
366void FGfdmSocket::Append(const char* item)
367{
368 if (buffer.tellp() > 0) buffer << ',';
369 buffer << item;
370}
371
372//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
373
374void FGfdmSocket::Append(double item)
375{
376 if (buffer.tellp() > 0) buffer << ',';
377 buffer << std::setw(12) << std::setprecision(precision) << item;
378}
379
380//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
381
382void FGfdmSocket::Append(long item)
383{
384 if (buffer.tellp() > 0) buffer << ',';
385 buffer << std::setw(12) << item;
386}
387
388//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
389
391{
392 buffer << '\n';
393 string str = buffer.str();
394 Send(str.c_str(), str.size());
395}
396
397//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
398
399void FGfdmSocket::Send(const char *data, int length)
400{
401 if (Protocol == ptTCP && sckt_in != INVALID_SOCKET) {
402 if ((send(sckt_in, data, length, 0)) == SOCKET_ERROR) LogSocketError("Send - TCP data sending");
403 return;
404 }
405
406 if (Protocol == ptUDP && sckt != INVALID_SOCKET) {
407 if ((send(sckt, data, length, 0)) == SOCKET_ERROR) LogSocketError("Send - UDP data sending");
408 return;
409 }
410
411 cerr << "Data sending must be to a valid socket" << endl;
412}
413
414//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
415
417{
418 assert(Protocol == ptTCP);
419 if (sckt_in == INVALID_SOCKET) return;
420
421 fd_set fds;
422 FD_ZERO(&fds);
423 FD_SET(sckt_in, &fds);
424
425 int result = select(FD_SETSIZE, &fds, nullptr, nullptr, nullptr);
426
427 if (result == 0) {
428 cerr << "Socket timeout." << endl;
429 return;
430 } else if (result != SOCKET_ERROR)
431 return;
432
433 LogSocketError("WaitUntilReadable");
434}
435
436//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
437
438void FGfdmSocket::LogSocketError(const std::string& msg)
439{
440 // An error has occurred, display the error message.
441 cerr << "Socket error in " << msg << ": ";
442#ifdef _WIN32
443 LPSTR errorMessage = nullptr;
444 DWORD errorCode = WSAGetLastError();
445 FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS,
446 nullptr, errorCode, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPSTR)&errorMessage, 0, nullptr);
447 cerr << errorMessage << endl;
448 LocalFree(errorMessage);
449#else
450 cerr << strerror(errno) << endl;
451#endif
452}
453
454//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
455// The bitmasked value choices are as follows:
456// unset: In this case (the default) JSBSim would only print
457// out the normally expected messages, essentially echoing
458// the config files as they are read. If the environment
459// variable is not set, debug_lvl is set to 1 internally
460// 0: This requests JSBSim not to output any messages
461// whatsoever.
462// 1: This value explicity requests the normal JSBSim
463// startup messages
464// 2: This value asks for a message to be printed out when
465// a class is instantiated
466// 4: When this value is set, a message is displayed when a
467// FGModel object executes its Run() method
468// 8: When this value is set, various runtime state variables
469// are printed out periodically
470// 16: When set various parameters are sanity checked and
471// a message is printed out when they go out of bounds
472
473void FGfdmSocket::Debug(int from)
474{
475 if (debug_lvl <= 0) return;
476
477 if (debug_lvl & 1) { // Standard console startup message output
478 if (from == 0) { // Constructor
479 }
480 }
481 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
482 if (from == 0) cout << "Instantiated: FGfdmSocket" << endl;
483 if (from == 1) cout << "Destroyed: FGfdmSocket" << endl;
484 }
485 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
486 }
487 if (debug_lvl & 8 ) { // Runtime state variables
488 }
489 if (debug_lvl & 16) { // Sanity checking
490 }
491 if (debug_lvl & 64) {
492 if (from == 0) { // Constructor
493 }
494 }
495}
496}
std::string Receive(void)
Receive data from the socket connection.
void Clear(void)
Clear the internal buffer.
void Append(const std::string &s)
Append the specified string to the internal buffer.
void Close(void)
Close the socket connection if the protocol is TCP.
void WaitUntilReadable(void)
Wait until the TCP socket is readable.
FGfdmSocket(const std::string &address, int port, int protocol, int precision=7)
Construct a client socket.
int Reply(const std::string &text)
Send a reply to the client ending by a prompt "JSBSim>".
void Send(void)
Send the internal buffer over the socket connection.