41#if defined(_MSC_VER) || defined(__MINGW32__)
43#elif defined(__OpenBSD__)
45#include <sys/socket.h>
59#include "FGfdmSocket.h"
60#include "input_output/string_utilities.h"
69#define closesocket close
70#define INVALID_SOCKET -1
71#define SOCKET_ERROR -1
72#define SD_BOTH SHUT_RDWR
82static bool LoadWinSockDLL(
int debug_lvl)
85 if (WSAStartup(MAKEWORD(1, 1), &wsaData)) {
86 cerr <<
"Winsock DLL not initialized ..." << endl;
91 cout <<
"Winsock DLL loaded ..." << endl;
99 sckt = sckt_in = INVALID_SOCKET;
100 Protocol = (ProtocolType)protocol;
102 struct addrinfo *addr =
nullptr;
103 this->precision = precision;
106 if (!LoadWinSockDLL(debug_lvl))
return;
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;
115 hints.ai_socktype = SOCK_STREAM;
116 hints.ai_protocol = 0;
117 if (!is_number(address))
118 hints.ai_flags = AI_ADDRCONFIG;
120 hints.ai_flags = AI_NUMERICHOST;
122 int failure = getaddrinfo(address.c_str(), NULL, &hints, &addr);
123 if (failure || !addr) {
124 cerr <<
"Could not get host net address " << address;
126 if (hints.ai_flags == AI_NUMERICHOST)
127 cerr <<
" by number..." << endl;
129 cerr <<
" by name..." << endl;
131 cerr << gai_strerror(failure) << endl;
137 sckt = socket(addr->ai_family, addr->ai_socktype, addr->ai_protocol);
140 if (protocol == ptUDP)
141 cout <<
"Creating UDP socket on port " << port << endl;
143 cout <<
"Creating TCP socket on port " << port << endl;
146 if (sckt != INVALID_SOCKET) {
147 int len =
sizeof(
struct sockaddr_in);
148 memcpy(&scktName, addr->ai_addr, len);
149 scktName.sin_port = htons(port);
151 if (connect(sckt, (
struct sockaddr*)&scktName, len) == 0) {
153 cout <<
"Successfully connected to socket for output ..." << endl;
156 cerr <<
"Could not connect to socket for output ..." << endl;
158 cerr <<
"Could not create socket for FDM output, error = " << errno << endl;
169 sckt = INVALID_SOCKET;
171 Protocol = (ProtocolType)protocol;
173 this->precision = precision;
176 if (!LoadWinSockDLL(debug_lvl))
return;
179 if (Protocol == ptUDP) {
180 ProtocolName =
"UDP";
181 sckt = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
184 ioctlsocket(sckt, FIONBIO, &NonBlock);
186 int flags = fcntl(sckt, F_GETFL, 0);
187 fcntl(sckt, F_SETFL, flags | O_NONBLOCK);
191 ProtocolName =
"TCP";
192 sckt = socket(AF_INET, SOCK_STREAM, 0);
196 cout <<
"Creating input " << ProtocolName <<
" socket on port " << port
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);
204 if (Protocol == ptUDP)
205 scktName.sin_addr.s_addr = htonl(INADDR_ANY);
207 socklen_t len =
sizeof(
struct sockaddr_in);
208 if (bind(sckt, (
struct sockaddr*)&scktName, len) != SOCKET_ERROR) {
210 cout <<
"Successfully bound to " << ProtocolName
211 <<
" input socket on port " << port << endl << endl;
213 if (Protocol == ptTCP) {
214 if (listen(sckt, 5) != SOCKET_ERROR) {
218 ioctlsocket(sckt, FIONBIO, &NoBlock);
220 int flags = fcntl(sckt, F_GETFL, 0);
221 fcntl(sckt, F_SETFL, flags | O_NONBLOCK);
223 sckt_in = accept(sckt, (
struct sockaddr*)&scktName, &len);
226 sckt = INVALID_SOCKET;
227 cerr <<
"Could not listen ..." << endl;
233 sckt = INVALID_SOCKET;
234 cerr <<
"Could not bind to " << ProtocolName <<
" input socket, error = "
238 cerr <<
"Could not create " << ProtocolName <<
" socket for input, error = "
246FGfdmSocket::~FGfdmSocket()
249 if (sckt_in != INVALID_SOCKET) shutdown(sckt_in, SD_BOTH);
250 if (sckt != INVALID_SOCKET) closesocket(sckt);
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) {
269 ioctlsocket(sckt_in, FIONBIO, &NoBlock);
271 int flags = fcntl(sckt_in, F_GETFL, 0);
272 fcntl(sckt_in, F_SETFL, flags | O_NONBLOCK);
274 if (send(sckt_in,
"Connected to JSBSim server\r\nJSBSim> ", 36, 0) == SOCKET_ERROR)
275 LogSocketError(
"Receive - TCP connection acknowledgement");
279 if (sckt_in != INVALID_SOCKET) {
282 while ((num_chars = recv(sckt_in, buf,
sizeof buf, 0)) > 0)
283 data.append(buf, num_chars);
285 if (num_chars == SOCKET_ERROR || num_chars == 0) {
287 if (WSAGetLastError() != WSAEWOULDBLOCK)
289 if (errno != EWOULDBLOCK)
292 LogSocketError(
"Receive - TCP data reception");
295 cout <<
"Socket Closed. Back to listening" << endl;
296 closesocket(sckt_in);
297 sckt_in = INVALID_SOCKET;
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) {
311 if (WSAGetLastError() != WSAEWOULDBLOCK)
313 if (errno != EWOULDBLOCK)
315 LogSocketError(
"Receive - UDP data reception");
326 int num_chars_sent=0;
327 assert(Protocol == ptTCP);
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");
334 cerr <<
"Socket reply must be to a valid socket" << endl;
337 return num_chars_sent;
344 assert(Protocol == ptTCP);
345 closesocket(sckt_in);
346 sckt_in = INVALID_SOCKET;
368 if (buffer.tellp() > 0) buffer <<
',';
376 if (buffer.tellp() > 0) buffer <<
',';
377 buffer << std::setw(12) << std::setprecision(precision) << item;
384 if (buffer.tellp() > 0) buffer <<
',';
385 buffer << std::setw(12) << item;
393 string str = buffer.str();
394 Send(str.c_str(), str.size());
401 if (Protocol == ptTCP && sckt_in != INVALID_SOCKET) {
402 if ((send(sckt_in, data, length, 0)) == SOCKET_ERROR) LogSocketError(
"Send - TCP data sending");
406 if (Protocol == ptUDP && sckt != INVALID_SOCKET) {
407 if ((send(sckt, data, length, 0)) == SOCKET_ERROR) LogSocketError(
"Send - UDP data sending");
411 cerr <<
"Data sending must be to a valid socket" << endl;
418 assert(Protocol == ptTCP);
419 if (sckt_in == INVALID_SOCKET)
return;
423 FD_SET(sckt_in, &fds);
425 int result = select(FD_SETSIZE, &fds,
nullptr,
nullptr,
nullptr);
428 cerr <<
"Socket timeout." << endl;
430 }
else if (result != SOCKET_ERROR)
433 LogSocketError(
"WaitUntilReadable");
438void FGfdmSocket::LogSocketError(
const std::string& msg)
441 cerr <<
"Socket error in " << msg <<
": ";
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);
450 cerr << strerror(errno) << endl;
473void FGfdmSocket::Debug(
int from)
475 if (debug_lvl <= 0)
return;
481 if (debug_lvl & 2 ) {
482 if (from == 0) cout <<
"Instantiated: FGfdmSocket" << endl;
483 if (from == 1) cout <<
"Destroyed: FGfdmSocket" << endl;
485 if (debug_lvl & 4 ) {
487 if (debug_lvl & 8 ) {
489 if (debug_lvl & 16) {
491 if (debug_lvl & 64) {
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.