Avionics
Dropship Simulator
Xplane.cpp
Go to the documentation of this file.
1 #include "Xplane.h"
2 
3 namespace Devices
4 {
5  void Xplane::Initialize(Logger* prmLogger, XplaneConfig* prmConfig, Bus* prmBus)
6  {
7  logger = prmLogger;
8  config = prmConfig;
9  bus = prmBus;
10 
11  if (!config->enabled) return;
12 
13  WSADATA data;
14  if (WSAStartup(MAKEWORD(2, 2), &data) != 0)
15  {
16  logger->Log("Xplane::Initialize WSAStartup failed!", Logger::Error);
17  config->enabled = false;
18  return;
19  }
20 
21  fd = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
22  if (fd == SOCKET_ERROR)
23  {
24  char msg[99];
25  sprintf_s(msg, 99, "Xplane::Initialize cannot create socket (error %i)", WSAGetLastError());
26  logger->Log(msg, Logger::Error);
27  config->enabled = false;
28  return;
29  }
30 
31  struct sockaddr_in myaddr;
32  memset(reinterpret_cast<char *>(&myaddr), 0, sizeof(myaddr));
33  myaddr.sin_family = AF_INET ;
34  myaddr.sin_addr.s_addr = htonl(INADDR_ANY);
35  myaddr.sin_port = htons(config->port);
36  if (bind(fd, reinterpret_cast<struct sockaddr *>(&myaddr), sizeof(myaddr)) == -1)
37  {
38  logger->Log("Xplane::Initialize bind failed!", Logger::Error, WSAGetLastError());
39  config->enabled = false;
40  return;
41  }
42  }
43 
44  void Xplane::FrameMove() const
45  {
46  unsigned long bytesWaiting = 0;
47  if (ioctlsocket(fd, FIONREAD, &bytesWaiting))
48  {
49  logger->Log("Xplane::FrameMove ioctlsocket failed!", Logger::Error, WSAGetLastError());
50  config->enabled = false;
51  return;
52  }
53 
54  if (bytesWaiting == 0) return;
55 
56  char buf[199];
57 
58  int recvlen = recvfrom(fd, buf, sizeof(buf), 0, nullptr, nullptr);
59  if (recvlen > 5)
60  {
61  if (buf[0] == 'D' && buf[1] == 'A' && buf[2] == 'T' && buf[3] == 'A' && buf[4] == '@')
62  {
63  int b = 5;
64  int bytesLeft = recvlen - b;
65  int sentences = bytesLeft / 36;
66  for (int i = 0; i < sentences; i++)
67  {
68  sentence sent;
69  memcpy(&sent, &buf[b], sizeof(sent));
70  switch (sent.group)
71  {
72  case 3:
73  // group 3, 0 is _Vind,_kias, 1 is _Vind,_keas, 2 is Vtrue,_ktas, 3 is Vtrue,_ktgs, 4 is unused, 5 is _Vind,__mph, 6 is Vtrue,mphas, 7 is Vtrue,mphgs
74  bus->IndicatedAirspeed = sent.data[0].floatData * 0.514444f; // nautical per hour to meters per second
75  break;
76  case 4:
77  // 0 is _Mach,ratio, 1 is unused, 2 is _WI,__fpm, 3 is unused, 4 is Gload,norml, 5 is Gload,axial, 6 is Gload,_side, 7 is unused
78  // norml = updown (positive is down/gravity), axial = frontback (positive is back), side = leftright (positive is left)
79  bus->NormalAcceleration = D3DXVECTOR3(sent.data[6].floatData, sent.data[5].floatData, sent.data[4].floatData);
80  break;
81  case 8:
82  // 0 is elev, 1 is aileron, 2 is rudder, 3 is unused, 4 is unused, 5 is unused, 6 is unused, 7 is unused
83  bus->PitchControl = sent.data[0].floatData;
84  bus->RollControl = sent.data[1].floatData;
85  bus->YawControl = sent.data[2].floatData;
86  break;
87  case 17:
88  // group 17, 0 is pitch, 1 is roll, 2 is true heading, 3 is magnetic heading, 4 is unused, 5 is unused, 6 is unused, 7 is unused
89  bus->PitchAttitude = sent.data[0].floatData;
90  bus->RollAttitude = sent.data[1].floatData;
91  bus->HeadingTrue = sent.data[2].floatData;
92  //bus->HeadingMagnetic = sent.data[3].floatData;
93  break;
94  case 20:
95  // group 20, 0 is latitude, 1 is longitude, 2 is MSL altitude, 3 is AGL altitude, 4 is squat switch, 5 is indicated altitude (baro adjusted?), 6 is rounded latitude?, 7 is rounded longitude?
96  bus->Latitude = sent.data[0].floatData;
97  bus->Longitude = sent.data[1].floatData;
99  bus->RadioAltitude = sent.data[3].floatData;
100  bus->OnGround = sent.data[4].floatData == 1.0f;
101  break;
102  }
103  b += 36;
104  }
105  //printf("\n");
106  }
107  else
108  {
109  char msg[199];
110  sprintf_s(msg, 199, "Xplane::FrameMove Received %i bytes NOT DATA header", recvlen);
111  logger->Log(msg, Logger::Warn);
112  }
113  }
114  else
115  {
116  char msg[199];
117  sprintf_s(msg, 199, "Xplane::FrameMove Received %i bytes NOT DATA header", recvlen);
118  logger->Log(msg, Logger::Warn);
119  }
120  }
121 
123  {
124  if (!config || !config->enabled) return;
125 
126  WSACleanup();
127  }
128 }
SOCKET fd
Definition: Xplane.h:32
float PressureAltitude
Definition: Bus.h:40
float YawControl
Definition: Bus.h:57
Definition: Logger.h:5
float RollControl
Definition: Bus.h:57
float Latitude
Definition: Bus.h:204
float IndicatedAirspeed
(3) Indicated Airspeed in km per second?
Definition: Bus.h:42
float PitchAttitude
(6) Pitch attitude;
Definition: Bus.h:49
okay, the portable keyboard numbers don&#39;t work like the outside keypad because the outside keypad is ...
Definition: Analog.cpp:3
float PitchControl
Definition: Bus.h:57
float Longitude
Definition: Bus.h:204
float floatData
Definition: Xplane.h:21
float HeadingTrue
(4) Heading–primary flight crew reference (if selectable, record discrete, true or magnetic); ...
Definition: Bus.h:44
Bus * bus
Definition: Xplane.h:31
XplaneConfig * config
Definition: Xplane.h:30
Definition: Bus.h:12
D3DXVECTOR3 NormalAcceleration
(5) vertical (11) longitudinal (18) lateral
Definition: Bus.h:47
float RollAttitude
(7) Roll attitude;
Definition: Bus.h:51
unsigned char group
Definition: Xplane.h:15
float RadioAltitude
Definition: Bus.h:134
bool OnGround
Definition: Bus.h:149
void FrameMove() const
Definition: Xplane.cpp:44
void Log(const char *msg, Level level=Info, int errorCode=0)
These have to be in this order.
Definition: Logger.cpp:16
union Devices::sentence::@5 data[8]
Logger * logger
Definition: Xplane.h:29
void Initialize(Logger *logger, XplaneConfig *config, Bus *prmBus)
Definition: Xplane.cpp:5