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 Class Reference

Detailed Description

The FGfdmSocket class enables JSBSim to communicate via sockets.

This class provides functionality for sending and receiving data over a socket connection. It can behave as both a client and/or a server, depending on the constructor used. The socket can use either UDP or TCP protocol for communication.

Definition at line 74 of file FGfdmSocket.h.

#include <FGfdmSocket.h>

+ Inheritance diagram for FGfdmSocket:
+ Collaboration diagram for FGfdmSocket:

Public Types

enum  ProtocolType { ptUDP , ptTCP }
 
- Public Types inherited from FGJSBBase
enum  { eL = 1 , eM , eN }
 Moments L, M, N. More...
 
enum  { eP = 1 , eQ , eR }
 Rates P, Q, R. More...
 
enum  { eU = 1 , eV , eW }
 Velocities U, V, W. More...
 
enum  { eX = 1 , eY , eZ }
 Positions X, Y, Z. More...
 
enum  { ePhi = 1 , eTht , ePsi }
 Euler angles Phi, Theta, Psi. More...
 
enum  { eDrag = 1 , eSide , eLift }
 Stability axis forces, Drag, Side force, Lift. More...
 
enum  { eRoll = 1 , ePitch , eYaw }
 Local frame orientation Roll, Pitch, Yaw. More...
 
enum  { eNorth = 1 , eEast , eDown }
 Local frame position North, East, Down. More...
 
enum  { eLat = 1 , eLong , eRad }
 Locations Radius, Latitude, Longitude. More...
 
enum  {
  inNone = 0 , inDegrees , inRadians , inMeters ,
  inFeet
}
 Conversion specifiers. More...
 

Public Member Functions

 FGfdmSocket (const std::string &address, int port, int protocol, int precision=7)
 Construct a client socket.
 
 FGfdmSocket (int port, int protocol, int precision=7)
 Construct a server socket.
 
void Append (const char *)
 Append the specified C-style string to the internal buffer.
 
void Append (const std::string &s)
 Append the specified string to the internal buffer.
 
void Append (double value)
 Append the specified double value to the internal buffer.
 
void Append (long value)
 Append the specified long value to the internal buffer.
 
void Clear (const std::string &s)
 Clear the internal buffer and appends the specified string.
 
void Clear (void)
 Clear the internal buffer.
 
void Close (void)
 Close the socket connection if the protocol is TCP.
 
bool GetConnectStatus (void)
 Return the connection status of the socket.
 
std::string Receive (void)
 Receive data from the socket connection.
 
int Reply (const std::string &text)
 Send a reply to the client ending by a prompt "JSBSim>".
 
void Send (const char *data, int length)
 Send the specified data over the socket connection.
 
void Send (const std::string &data)
 
void Send (void)
 Send the internal buffer over the socket connection.
 
void WaitUntilReadable (void)
 Wait until the TCP socket is readable.
 
- Public Member Functions inherited from FGJSBBase
 FGJSBBase ()
 Constructor for FGJSBBase.
 
virtual ~FGJSBBase ()
 Destructor for FGJSBBase.
 
void disableHighLighting (void)
 Disables highlighting in the console output.
 

Additional Inherited Members

- Static Public Member Functions inherited from FGJSBBase
static const std::string & GetVersion (void)
 Returns the version number of JSBSim.
 
static constexpr double KelvinToFahrenheit (double kelvin)
 Converts from degrees Kelvin to degrees Fahrenheit.
 
static constexpr double CelsiusToRankine (double celsius)
 Converts from degrees Celsius to degrees Rankine.
 
static constexpr double RankineToCelsius (double rankine)
 Converts from degrees Rankine to degrees Celsius.
 
static constexpr double KelvinToRankine (double kelvin)
 Converts from degrees Kelvin to degrees Rankine.
 
static constexpr double RankineToKelvin (double rankine)
 Converts from degrees Rankine to degrees Kelvin.
 
static constexpr double FahrenheitToCelsius (double fahrenheit)
 Converts from degrees Fahrenheit to degrees Celsius.
 
static constexpr double CelsiusToFahrenheit (double celsius)
 Converts from degrees Celsius to degrees Fahrenheit.
 
static constexpr double CelsiusToKelvin (double celsius)
 Converts from degrees Celsius to degrees Kelvin.
 
static constexpr double KelvinToCelsius (double kelvin)
 Converts from degrees Kelvin to degrees Celsius.
 
static constexpr double FeetToMeters (double measure)
 Converts from feet to meters.
 
static bool EqualToRoundoff (double a, double b)
 Finite precision comparison.
 
static bool EqualToRoundoff (float a, float b)
 Finite precision comparison.
 
static bool EqualToRoundoff (float a, double b)
 Finite precision comparison.
 
static bool EqualToRoundoff (double a, float b)
 Finite precision comparison.
 
static constexpr double Constrain (double min, double value, double max)
 Constrain a value between a minimum and a maximum value.
 
static constexpr double sign (double num)
 
- Static Public Attributes inherited from FGJSBBase
static char highint [5] = {27, '[', '1', 'm', '\0' }
 highlights text
 
static char halfint [5] = {27, '[', '2', 'm', '\0' }
 low intensity text
 
static char normint [6] = {27, '[', '2', '2', 'm', '\0' }
 normal intensity text
 
static char reset [5] = {27, '[', '0', 'm', '\0' }
 resets text properties
 
static char underon [5] = {27, '[', '4', 'm', '\0' }
 underlines text
 
static char underoff [6] = {27, '[', '2', '4', 'm', '\0' }
 underline off
 
static char fgblue [6] = {27, '[', '3', '4', 'm', '\0' }
 blue text
 
static char fgcyan [6] = {27, '[', '3', '6', 'm', '\0' }
 cyan text
 
static char fgred [6] = {27, '[', '3', '1', 'm', '\0' }
 red text
 
static char fggreen [6] = {27, '[', '3', '2', 'm', '\0' }
 green text
 
static char fgdef [6] = {27, '[', '3', '9', 'm', '\0' }
 default text
 
static short debug_lvl = 1
 
- Static Protected Member Functions inherited from FGJSBBase
static std::string CreateIndexedPropertyName (const std::string &Property, int index)
 
- Static Protected Attributes inherited from FGJSBBase
static constexpr double radtodeg = 180. / M_PI
 
static constexpr double degtorad = M_PI / 180.
 
static constexpr double hptoftlbssec = 550.0
 
static constexpr double psftoinhg = 0.014138
 
static constexpr double psftopa = 47.88
 
static constexpr double fttom = 0.3048
 
static constexpr double ktstofps = 1852./(3600*fttom)
 
static constexpr double fpstokts = 1.0 / ktstofps
 
static constexpr double inchtoft = 1.0/12.0
 
static constexpr double m3toft3 = 1.0/(fttom*fttom*fttom)
 
static constexpr double in3tom3 = inchtoft*inchtoft*inchtoft/m3toft3
 
static constexpr double inhgtopa = 3386.38
 
static constexpr double slugtolb = 32.174049
 Note that definition of lbtoslug by the inverse of slugtolb and not to a different constant you can also get from some tables will make lbtoslug*slugtolb == 1 up to the magnitude of roundoff.
 
static constexpr double lbtoslug = 1.0/slugtolb
 
static constexpr double kgtolb = 2.20462
 
static constexpr double kgtoslug = 0.06852168
 
static const std::string needed_cfg_version = "2.0"
 
static const std::string JSBSim_version = JSBSIM_VERSION " " __DATE__ " " __TIME__
 

Member Enumeration Documentation

◆ ProtocolType

enum ProtocolType

Definition at line 176 of file FGfdmSocket.h.

176{ptUDP, ptTCP};

Constructor & Destructor Documentation

◆ FGfdmSocket() [1/2]

FGfdmSocket ( const std::string &  address,
int  port,
int  protocol,
int  precision = 7 
)

Construct a client socket.

Parameters
addressThe IP address or hostname of the server to connect to.
portThe port number to connect to.
protocolThe protocol to use for communication (ptUDP or ptTCP).
precisionThe precision to use for floating-point numbers (default is 7).

Definition at line 97 of file FGfdmSocket.cpp.

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}

◆ FGfdmSocket() [2/2]

FGfdmSocket ( int  port,
int  protocol,
int  precision = 7 
)

Construct a server socket.

Parameters
portThe port number to listen on.
protocolThe protocol to use for communication (ptUDP or ptTCP).
precisionThe precision to use for floating-point numbers (default is 7).

Definition at line 167 of file FGfdmSocket.cpp.

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}

◆ ~FGfdmSocket()

Definition at line 246 of file FGfdmSocket.cpp.

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}

Member Function Documentation

◆ Append() [1/4]

void Append ( const char *  item)

Append the specified C-style string to the internal buffer.

Parameters
sThe C-style string to append.

Definition at line 366 of file FGfdmSocket.cpp.

367{
368 if (buffer.tellp() > 0) buffer << ',';
369 buffer << item;
370}

◆ Append() [2/4]

void Append ( const std::string &  s)
inline

Append the specified string to the internal buffer.

Parameters
sThe string to append.

Definition at line 130 of file FGfdmSocket.h.

130{Append(s.c_str());}
void Append(const std::string &s)
Append the specified string to the internal buffer.
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ Append() [3/4]

void Append ( double  value)

Append the specified double value to the internal buffer.

Parameters
valueThe double value to append.

Definition at line 374 of file FGfdmSocket.cpp.

375{
376 if (buffer.tellp() > 0) buffer << ',';
377 buffer << std::setw(12) << std::setprecision(precision) << item;
378}

◆ Append() [4/4]

void Append ( long  value)

Append the specified long value to the internal buffer.

Parameters
valueThe long value to append.

Definition at line 382 of file FGfdmSocket.cpp.

383{
384 if (buffer.tellp() > 0) buffer << ',';
385 buffer << std::setw(12) << item;
386}

◆ Clear() [1/2]

void Clear ( const std::string &  s)

Clear the internal buffer and appends the specified string.

Parameters
sThe string to append after clearing the buffer.

Definition at line 358 of file FGfdmSocket.cpp.

359{
360 Clear();
361 buffer << s << ' ';
362}
void Clear(void)
Clear the internal buffer.
+ Here is the call graph for this function:

◆ Clear() [2/2]

void Clear ( void  )

Clear the internal buffer.

Definition at line 351 of file FGfdmSocket.cpp.

352{
353 buffer.str("");
354}
+ Here is the caller graph for this function:

◆ Close()

void Close ( void  )

Close the socket connection if the protocol is TCP.

Definition at line 342 of file FGfdmSocket.cpp.

343{
344 assert(Protocol == ptTCP);
345 closesocket(sckt_in);
346 sckt_in = INVALID_SOCKET;
347}
+ Here is the caller graph for this function:

◆ GetConnectStatus()

bool GetConnectStatus ( void  )
inline

Return the connection status of the socket.

Returns
True if the socket is connected, false otherwise.

Definition at line 171 of file FGfdmSocket.h.

171{return connected;}
+ Here is the caller graph for this function:

◆ Receive()

string Receive ( void  )

Receive data from the socket connection.

Returns
The received data as a string.

Definition at line 256 of file FGfdmSocket.cpp.

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}
+ Here is the caller graph for this function:

◆ Reply()

int Reply ( const std::string &  text)

Send a reply to the client ending by a prompt "JSBSim>".

Parameters
textThe reply text to send.
Returns
The number of bytes sent.

Definition at line 324 of file FGfdmSocket.cpp.

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}
+ Here is the caller graph for this function:

◆ Send() [1/3]

void Send ( const char *  data,
int  length 
)

Send the specified data over the socket connection.

Parameters
dataThe data to send.
lengthThe length of the data.

Definition at line 399 of file FGfdmSocket.cpp.

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}

◆ Send() [2/3]

void Send ( const std::string &  data)
inline

Definition at line 108 of file FGfdmSocket.h.

108{ Send(data.c_str(), data.length()); }
void Send(void)
Send the internal buffer over the socket connection.

◆ Send() [3/3]

void Send ( void  )

Send the internal buffer over the socket connection.

Definition at line 390 of file FGfdmSocket.cpp.

391{
392 buffer << '\n';
393 string str = buffer.str();
394 Send(str.c_str(), str.size());
395}
+ Here is the call graph for this function:
+ Here is the caller graph for this function:

◆ WaitUntilReadable()

void WaitUntilReadable ( void  )

Wait until the TCP socket is readable.

Definition at line 416 of file FGfdmSocket.cpp.

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}
+ Here is the caller graph for this function:

The documentation for this class was generated from the following files: