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
FGInitialCondition.cpp
1/*******************************************************************************
2
3 Header: FGInitialCondition.cpp
4 Author: Tony Peden, Bertrand Coconnier
5 Date started: 7/1/99
6
7 --------- Copyright (C) 1999 Anthony K. Peden (apeden@earthlink.net) ---------
8
9 This program is free software; you can redistribute it and/or modify it under
10 the terms of the GNU Lesser General Public License as published by the Free
11 Software Foundation; either version 2 of the License, or (at your option) any
12 later version.
13
14 This program is distributed in the hope that it will be useful, but WITHOUT
15 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
16 FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
17 details.
18
19 You should have received a copy of the GNU Lesser General Public License along
20 with this program; if not, write to the Free Software Foundation, Inc., 59
21 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
22
23 Further information about the GNU Lesser General Public License can also be
24 found on the world wide web at http://www.gnu.org.
25
26
27 HISTORY
28--------------------------------------------------------------------------------
297/1/99 TP Created
3011/25/10 BC Complete revision - Use minimal set of variables to prevent
31 inconsistent states. Wind is correctly handled.
32
33
34FUNCTIONAL DESCRIPTION
35--------------------------------------------------------------------------------
36
37The purpose of this class is to take a set of initial conditions and provide a
38kinematically consistent set of body axis velocity components, euler angles, and
39altitude. This class does not attempt to trim the model i.e. the sim will most
40likely start in a very dynamic state (unless, of course, you have chosen your
41IC's wisely) even after setting it up with this class.
42
43********************************************************************************
44INCLUDES
45*******************************************************************************/
46
47#include "FGInitialCondition.h"
48#include "models/FGInertial.h"
49#include "models/FGAtmosphere.h"
50#include "models/FGAccelerations.h"
51#include "models/FGAuxiliary.h"
52#include "input_output/FGXMLFileRead.h"
53#include "FGTrim.h"
54#include "FGFDMExec.h"
55#include "input_output/string_utilities.h"
56
57using namespace std;
58
59namespace JSBSim {
60
61//******************************************************************************
62
64{
66
67 if(FDMExec) {
68 Aircraft=fdmex->GetAircraft();
69 Auxiliary=fdmex->GetAuxiliary();
70 } else {
71 cout << "FGInitialCondition: This class requires a pointer to a valid FGFDMExec object" << endl;
72 }
73
74 Debug(0);
75}
76
77//******************************************************************************
78
83
84//******************************************************************************
85
86void FGInitialCondition::ResetIC(double u0, double v0, double w0,
87 double p0, double q0, double r0,
88 double alpha0, double beta0,
89 double phi0, double theta0, double psi0,
90 double latRad0, double lonRad0, double altAGLFt0,
91 double gamma0)
92{
93 double calpha = cos(alpha0), cbeta = cos(beta0);
94 double salpha = sin(alpha0), sbeta = sin(beta0);
95
97
98 vPQR_body = {p0, q0, r0};
99 alpha = alpha0; beta = beta0;
100
101 position.SetLongitude(lonRad0);
102 position.SetLatitude(latRad0);
103 fdmex->GetInertial()->SetAltitudeAGL(position, altAGLFt0);
104 lastLatitudeSet = setgeoc;
105 lastAltitudeSet = setagl;
106
107 orientation = FGQuaternion(phi0, theta0, psi0);
108 const FGMatrix33& Tb2l = orientation.GetTInv();
109
110 vUVW_NED = Tb2l * FGColumnVector3(u0, v0, w0);
111 vt = vUVW_NED.Magnitude();
112 lastSpeedSet = setuvw;
113
114 Tw2b = { calpha*cbeta, -calpha*sbeta, -salpha,
115 sbeta, cbeta, 0.0,
116 salpha*cbeta, -salpha*sbeta, calpha };
117 Tb2w = Tw2b.Transposed();
118
120}
121
122//******************************************************************************
123
125{
126 alpha = beta = 0.0;
127 epa = 0.0;
128
129 double a = fdmex->GetInertial()->GetSemimajor();
130 double b = fdmex->GetInertial()->GetSemiminor();
131
132 position.SetEllipse(a, b);
133
134 position.SetPositionGeodetic(0.0, 0.0, 0.0);
135
136 orientation = FGQuaternion(0.0, 0.0, 0.0);
137 vUVW_NED.InitMatrix();
138 vPQR_body.InitMatrix();
139 vt=0;
140
141 targetNlfIC = 1.0;
142
143 Tw2b = { 1., 0., 0., 0., 1., 0., 0., 0., 1. };
144 Tb2w = { 1., 0., 0., 0., 1., 0., 0., 0., 1. };
145
146 lastSpeedSet = setvt;
147 lastAltitudeSet = setasl;
148 lastLatitudeSet = setgeoc;
149 enginesRunning = 0;
150 trimRequested = TrimMode::tNone;
151}
152
153//******************************************************************************
154
156{
157 const auto Atmosphere = fdmex->GetAtmosphere();
158 double altitudeASL = GetAltitudeASLFtIC();
159 double rho = Atmosphere->GetDensity(altitudeASL);
160 SetVtrueFpsIC(ve*ktstofps*sqrt(FGAtmosphere::StdDaySLdensity/rho));
161 lastSpeedSet = setve;
162}
163
164//******************************************************************************
165
167{
168 const auto Atmosphere = fdmex->GetAtmosphere();
169 double altitudeASL = GetAltitudeASLFtIC();
170 double soundSpeed = Atmosphere->GetSoundSpeed(altitudeASL);
171 SetVtrueFpsIC(mach*soundSpeed);
172 lastSpeedSet = setmach;
173}
174
175//******************************************************************************
176
178{
179 const auto Atmosphere = fdmex->GetAtmosphere();
180 double altitudeASL = GetAltitudeASLFtIC();
181 double pressure = Atmosphere->GetPressure(altitudeASL);
182 double mach = Auxiliary->MachFromVcalibrated(fabs(vcas)*ktstofps, pressure);
183 double soundSpeed = Atmosphere->GetSoundSpeed(altitudeASL);
184
185 SetVtrueFpsIC(mach * soundSpeed);
186 lastSpeedSet = setvc;
187}
188
189//******************************************************************************
190// Updates alpha and beta according to the aircraft true airspeed in the local
191// NED frame.
192
193void FGInitialCondition::calcAeroAngles(const FGColumnVector3& _vt_NED)
194{
195 const FGMatrix33& Tl2b = orientation.GetT();
196 FGColumnVector3 _vt_BODY = Tl2b * _vt_NED;
197 double ua = _vt_BODY(eX);
198 double va = _vt_BODY(eY);
199 double wa = _vt_BODY(eZ);
200 double uwa = sqrt(ua*ua + wa*wa);
201 double calpha, cbeta;
202 double salpha, sbeta;
203
204 alpha = beta = 0.0;
205 calpha = cbeta = 1.0;
206 salpha = sbeta = 0.0;
207
208 if( wa != 0 )
209 alpha = atan2( wa, ua );
210
211 // alpha cannot be constrained without updating other informations like the
212 // true speed or the Euler angles. Otherwise we might end up with an
213 // inconsistent state of the aircraft.
214 /*alpha = Constrain(fdmex->GetAerodynamics()->GetAlphaCLMin(), alpha,
215 fdmex->GetAerodynamics()->GetAlphaCLMax());*/
216
217 if( va != 0 )
218 beta = atan2( va, uwa );
219
220 if (uwa != 0) {
221 calpha = ua / uwa;
222 salpha = wa / uwa;
223 }
224
225 if (vt != 0) {
226 cbeta = uwa / vt;
227 sbeta = va / vt;
228 }
229
230 Tw2b = { calpha*cbeta, -calpha*sbeta, -salpha,
231 sbeta, cbeta, 0.0,
232 salpha*cbeta, -salpha*sbeta, calpha };
233 Tb2w = Tw2b.Transposed();
234}
235
236//******************************************************************************
237// Set the ground velocity. Caution it sets the vertical velocity to zero to
238// keep backward compatibility.
239
241{
242 const FGMatrix33& Tb2l = orientation.GetTInv();
243 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
244 FGColumnVector3 _vWIND_NED = _vt_NED - vUVW_NED;
245
246 vUVW_NED(eU) = vg * orientation.GetCosEuler(ePsi);
247 vUVW_NED(eV) = vg * orientation.GetSinEuler(ePsi);
248 vUVW_NED(eW) = 0.;
249 _vt_NED = vUVW_NED + _vWIND_NED;
250 vt = _vt_NED.Magnitude();
251
252 calcAeroAngles(_vt_NED);
253
254 lastSpeedSet = setvg;
255}
256
257//******************************************************************************
258// Sets the true airspeed. The amplitude of the airspeed is modified but its
259// direction is kept unchanged. If there is no wind, the same is true for the
260// ground velocity. If there is some wind, the airspeed direction is unchanged
261// but this may result in the ground velocity direction being altered. This is
262// for backward compatibility.
263
265{
266 const FGMatrix33& Tb2l = orientation.GetTInv();
267 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
268 FGColumnVector3 _vWIND_NED = _vt_NED - vUVW_NED;
269
270 if (vt > 0.1)
271 _vt_NED *= vtrue / vt;
272 else
273 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vtrue, 0., 0.);
274
275 vt = vtrue;
276 vUVW_NED = _vt_NED - _vWIND_NED;
277
278 calcAeroAngles(_vt_NED);
279
280 lastSpeedSet = setvt;
281}
282
283//******************************************************************************
284// When the climb rate is modified, we need to update the angles theta and beta
285// to keep the true airspeed amplitude, the AoA and the heading unchanged.
286// Beta will be modified if the aircraft roll angle is not null.
287
289{
290 if (fabs(hdot) > vt) {
291 cerr << "The climb rate cannot be higher than the true speed." << endl;
292 return;
293 }
294
295 const FGMatrix33& Tb2l = orientation.GetTInv();
296 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
297 FGColumnVector3 _WIND_NED = _vt_NED - vUVW_NED;
298 double hdot0 = -_vt_NED(eW);
299
300 if (fabs(hdot0) < vt) { // Is this check really needed ?
301 double scale = sqrt((vt*vt-hdot*hdot)/(vt*vt-hdot0*hdot0));
302 _vt_NED(eU) *= scale;
303 _vt_NED(eV) *= scale;
304 }
305 _vt_NED(eW) = -hdot;
306 vUVW_NED = _vt_NED - _WIND_NED;
307
308 // Updating the angles theta and beta to keep the true airspeed amplitude
309 calcThetaBeta(alpha, _vt_NED);
310}
311
312//******************************************************************************
313// When the AoA is modified, we need to update the angles theta and beta to
314// keep the true airspeed amplitude, the climb rate and the heading unchanged.
315// Beta will be modified if the aircraft roll angle is not null.
316
318{
319 const FGMatrix33& Tb2l = orientation.GetTInv();
320 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
321 calcThetaBeta(alfa, _vt_NED);
322}
323
324//******************************************************************************
325// When the AoA is modified, we need to update the angles theta and beta to
326// keep the true airspeed amplitude, the climb rate and the heading unchanged.
327// Beta will be modified if the aircraft roll angle is not null.
328
329void FGInitialCondition::calcThetaBeta(double alfa, const FGColumnVector3& _vt_NED)
330{
331 FGColumnVector3 vOrient = orientation.GetEuler();
332 double calpha = cos(alfa), salpha = sin(alfa);
333 double cpsi = orientation.GetCosEuler(ePsi), spsi = orientation.GetSinEuler(ePsi);
334 double cphi = orientation.GetCosEuler(ePhi), sphi = orientation.GetSinEuler(ePhi);
335 FGMatrix33 Tpsi( cpsi, spsi, 0.,
336 -spsi, cpsi, 0.,
337 0., 0., 1.);
338 FGMatrix33 Tphi(1., 0., 0.,
339 0., cphi, sphi,
340 0.,-sphi, cphi);
341 FGMatrix33 Talpha( calpha, 0., salpha,
342 0., 1., 0.,
343 -salpha, 0., calpha);
344
345 FGColumnVector3 v0 = Tpsi * _vt_NED;
346 FGColumnVector3 n = (Talpha * Tphi).Transposed() * FGColumnVector3(0., 0., 1.);
347 FGColumnVector3 y = {0., 1., 0.};
348 FGColumnVector3 u = y - DotProduct(y, n) * n;
349 FGColumnVector3 p = y * n;
350
351 if (DotProduct(p, v0) < 0) p *= -1.0;
352 p.Normalize();
353
354 u *= DotProduct(v0, y) / DotProduct(u, y);
355
356 // There are situations where the desired alpha angle cannot be obtained. This
357 // is not a limitation of the algorithm but is due to the mathematical problem
358 // not having a solution. This can only be cured by limiting the alpha angle
359 // or by modifying an additional angle (psi ?). Since this is anticipated to
360 // be a pathological case (mainly when a high roll angle is required) this
361 // situation is not addressed below. However if there are complaints about the
362 // following error being raised too often, we might need to reconsider this
363 // position.
364 if (DotProduct(v0, v0) < DotProduct(u, u)) {
365 cerr << "Cannot modify angle 'alpha' from " << alpha << " to " << alfa << endl;
366 return;
367 }
368
369 FGColumnVector3 v1 = u + sqrt(DotProduct(v0, v0) - DotProduct(u, u))*p;
370
371 FGColumnVector3 v0xz(v0(eU), 0., v0(eW));
372 FGColumnVector3 v1xz(v1(eU), 0., v1(eW));
373 v0xz.Normalize();
374 v1xz.Normalize();
375 double sinTheta = (v1xz * v0xz)(eY);
376 vOrient(eTht) = asin(sinTheta);
377
378 orientation = FGQuaternion(vOrient);
379
380 const FGMatrix33& Tl2b = orientation.GetT();
381 FGColumnVector3 v2 = Talpha * Tl2b * _vt_NED;
382
383 alpha = alfa;
384 beta = atan2(v2(eV), v2(eU));
385 double cbeta=1.0, sbeta=0.0;
386 if (vt != 0.0) {
387 cbeta = v2(eU) / vt;
388 sbeta = v2(eV) / vt;
389 }
390 Tw2b = { calpha*cbeta, -calpha*sbeta, -salpha,
391 sbeta, cbeta, 0.0,
392 salpha*cbeta, -salpha*sbeta, calpha };
393 Tb2w = Tw2b.Transposed();
394}
395
396//******************************************************************************
397// When the beta angle is modified, we need to update the angles theta and psi
398// to keep the true airspeed (amplitude and direction - including the climb rate)
399// and the alpha angle unchanged. This may result in the aircraft heading (psi)
400// being altered especially if there is cross wind.
401
403{
404 const FGMatrix33& Tb2l = orientation.GetTInv();
405 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
406 FGColumnVector3 vOrient = orientation.GetEuler();
407
408 beta = bta;
409 double calpha = cos(alpha), salpha = sin(alpha);
410 double cbeta = cos(beta), sbeta = sin(beta);
411 double cphi = orientation.GetCosEuler(ePhi), sphi = orientation.GetSinEuler(ePhi);
412 FGMatrix33 TphiInv(1., 0., 0.,
413 0., cphi,-sphi,
414 0., sphi, cphi);
415
416 Tw2b = { calpha*cbeta, -calpha*sbeta, -salpha,
417 sbeta, cbeta, 0.0,
418 salpha*cbeta, -salpha*sbeta, calpha };
419 Tb2w = Tw2b.Transposed();
420
421 FGColumnVector3 vf = TphiInv * Tw2b * FGColumnVector3(vt, 0., 0.);
422 FGColumnVector3 v0xy(_vt_NED(eX), _vt_NED(eY), 0.);
423 FGColumnVector3 v1xy(sqrt(v0xy(eX)*v0xy(eX)+v0xy(eY)*v0xy(eY)-vf(eY)*vf(eY)),vf(eY),0.);
424 v0xy.Normalize();
425 v1xy.Normalize();
426
427 if (vf(eX) < 0.) v0xy(eX) *= -1.0;
428
429 double sinPsi = (v1xy * v0xy)(eZ);
430 double cosPsi = DotProduct(v0xy, v1xy);
431 vOrient(ePsi) = atan2(sinPsi, cosPsi);
432 FGMatrix33 Tpsi( cosPsi, sinPsi, 0.,
433 -sinPsi, cosPsi, 0.,
434 0., 0., 1.);
435
436 FGColumnVector3 v2xz = Tpsi * _vt_NED;
437 FGColumnVector3 vfxz = vf;
438 v2xz(eV) = vfxz(eV) = 0.0;
439 v2xz.Normalize();
440 vfxz.Normalize();
441 double sinTheta = (v2xz * vfxz)(eY);
442 vOrient(eTht) = -asin(sinTheta);
443
444 orientation = FGQuaternion(vOrient);
445}
446
447//******************************************************************************
448// Modifies the body frame orientation.
449
450void FGInitialCondition::SetEulerAngleRadIC(int idx, double angle)
451{
452 const FGMatrix33& Tb2l = orientation.GetTInv();
453 const FGMatrix33& Tl2b = orientation.GetT();
454 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
455 FGColumnVector3 _vWIND_NED = _vt_NED - vUVW_NED;
456 FGColumnVector3 _vUVW_BODY = Tl2b * vUVW_NED;
457 FGColumnVector3 vOrient = orientation.GetEuler();
458
459 vOrient(idx) = angle;
460 orientation = FGQuaternion(vOrient);
461
462 if ((lastSpeedSet != setned) && (lastSpeedSet != setvg)) {
463 const FGMatrix33& newTb2l = orientation.GetTInv();
464 vUVW_NED = newTb2l * _vUVW_BODY;
465 _vt_NED = vUVW_NED + _vWIND_NED;
466 vt = _vt_NED.Magnitude();
467 }
468
469 calcAeroAngles(_vt_NED);
470}
471
472//******************************************************************************
473// Modifies an aircraft velocity component (eU, eV or eW) in the body frame. The
474// true airspeed is modified accordingly. If there is some wind, the airspeed
475// direction modification may differ from the body velocity modification.
476
477void FGInitialCondition::SetBodyVelFpsIC(int idx, double vel)
478{
479 const FGMatrix33& Tb2l = orientation.GetTInv();
480 const FGMatrix33& Tl2b = orientation.GetT();
481 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
482 FGColumnVector3 _vUVW_BODY = Tl2b * vUVW_NED;
483 FGColumnVector3 _vWIND_NED = _vt_NED - vUVW_NED;
484
485 _vUVW_BODY(idx) = vel;
486 vUVW_NED = Tb2l * _vUVW_BODY;
487 _vt_NED = vUVW_NED + _vWIND_NED;
488 vt = _vt_NED.Magnitude();
489
490 calcAeroAngles(_vt_NED);
491
492 lastSpeedSet = setuvw;
493}
494
495//******************************************************************************
496// Modifies an aircraft velocity component (eX, eY or eZ) in the local NED frame.
497// The true airspeed is modified accordingly. If there is some wind, the airspeed
498// direction modification may differ from the local velocity modification.
499
500void FGInitialCondition::SetNEDVelFpsIC(int idx, double vel)
501{
502 const FGMatrix33& Tb2l = orientation.GetTInv();
503 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
504 FGColumnVector3 _vWIND_NED = _vt_NED - vUVW_NED;
505
506 vUVW_NED(idx) = vel;
507 _vt_NED = vUVW_NED + _vWIND_NED;
508 vt = _vt_NED.Magnitude();
509
510 calcAeroAngles(_vt_NED);
511
512 lastSpeedSet = setned;
513}
514
515//******************************************************************************
516// Set wind amplitude and direction in the local NED frame. The aircraft velocity
517// with respect to the ground is not changed but the true airspeed is.
518
519void FGInitialCondition::SetWindNEDFpsIC(double wN, double wE, double wD )
520{
521 FGColumnVector3 _vt_NED = vUVW_NED + FGColumnVector3(wN, wE, wD);
522 vt = _vt_NED.Magnitude();
523
524 calcAeroAngles(_vt_NED);
525}
526
527//******************************************************************************
528// Set the cross wind velocity (in knots). Here, 'cross wind' means perpendicular
529// to the aircraft heading and parallel to the ground. The aircraft velocity
530// with respect to the ground is not changed but the true airspeed is.
531
533{
534 const FGMatrix33& Tb2l = orientation.GetTInv();
535 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
536 FGColumnVector3 _vWIND_NED = _vt_NED - vUVW_NED;
537 FGColumnVector3 _vCROSS(-orientation.GetSinEuler(ePsi), orientation.GetCosEuler(ePsi), 0.);
538
539 // Gram-Schmidt process is used to remove the existing cross wind component
540 _vWIND_NED -= DotProduct(_vWIND_NED, _vCROSS) * _vCROSS;
541 // Which is now replaced by the new value. The input cross wind is expected
542 // in knots, so first convert to fps, which is the internal unit used.
543 _vWIND_NED += (cross * ktstofps) * _vCROSS;
544 _vt_NED = vUVW_NED + _vWIND_NED;
545 vt = _vt_NED.Magnitude();
546
547 calcAeroAngles(_vt_NED);
548}
549
550//******************************************************************************
551// Set the head wind velocity (in knots). Here, 'head wind' means parallel
552// to the aircraft heading and to the ground. The aircraft velocity
553// with respect to the ground is not changed but the true airspeed is.
554
556{
557 const FGMatrix33& Tb2l = orientation.GetTInv();
558 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
559 FGColumnVector3 _vWIND_NED = _vt_NED - vUVW_NED;
560 // This is a head wind, so the direction vector for the wind
561 // needs to be set opposite to the heading the aircraft
562 // is taking. So, the cos and sin of the heading (psi)
563 // are negated in the line below.
564 FGColumnVector3 _vHEAD(-orientation.GetCosEuler(ePsi), -orientation.GetSinEuler(ePsi), 0.);
565
566 // Gram-Schmidt process is used to remove the existing head wind component
567 _vWIND_NED -= DotProduct(_vWIND_NED, _vHEAD) * _vHEAD;
568 // Which is now replaced by the new value. The input head wind is expected
569 // in knots, so first convert to fps, which is the internal unit used.
570 _vWIND_NED += (head * ktstofps) * _vHEAD;
571 _vt_NED = vUVW_NED + _vWIND_NED;
572
573 vt = _vt_NED.Magnitude();
574
575 calcAeroAngles(_vt_NED);
576}
577
578//******************************************************************************
579// Set the vertical wind velocity (in knots). The 'vertical' is taken in the
580// local NED frame. The aircraft velocity with respect to the ground is not
581// changed but the true airspeed is.
582
584{
585 const FGMatrix33& Tb2l = orientation.GetTInv();
586 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
587
588 _vt_NED(eW) = vUVW_NED(eW) + wD * ktstofps;
589 vt = _vt_NED.Magnitude();
590
591 calcAeroAngles(_vt_NED);
592}
593
594//******************************************************************************
595// Modifies the wind velocity (in knots) while keeping its direction unchanged.
596// The vertical component (in local NED frame) is unmodified. The aircraft
597// velocity with respect to the ground is not changed but the true airspeed is.
598
600{
601 const FGMatrix33& Tb2l = orientation.GetTInv();
602 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
603 FGColumnVector3 _vWIND_NED = _vt_NED - vUVW_NED;
604 FGColumnVector3 _vHEAD(_vWIND_NED(eU), _vWIND_NED(eV), 0.);
605 double windMag = _vHEAD.Magnitude();
606
607 if (windMag > 0.001)
608 _vHEAD *= (mag*ktstofps) / windMag;
609 else
610 _vHEAD = {mag*ktstofps, 0., 0.};
611
612 _vWIND_NED(eU) = _vHEAD(eU);
613 _vWIND_NED(eV) = _vHEAD(eV);
614 _vt_NED = vUVW_NED + _vWIND_NED;
615 vt = _vt_NED.Magnitude();
616
617 calcAeroAngles(_vt_NED);
618}
619
620//******************************************************************************
621// Modifies the wind direction while keeping its velocity unchanged. The vertical
622// component (in local NED frame) is unmodified. The aircraft velocity with
623// respect to the ground is not changed but the true airspeed is.
624
626{
627 const FGMatrix33& Tb2l = orientation.GetTInv();
628 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
629 FGColumnVector3 _vWIND_NED = _vt_NED - vUVW_NED;
630 double mag = _vWIND_NED.Magnitude(eU, eV);
631 FGColumnVector3 _vHEAD(mag*cos(dir*degtorad), mag*sin(dir*degtorad), 0.);
632
633 _vWIND_NED(eU) = _vHEAD(eU);
634 _vWIND_NED(eV) = _vHEAD(eV);
635 _vt_NED = vUVW_NED + _vWIND_NED;
636 vt = _vt_NED.Magnitude();
637
638 calcAeroAngles(_vt_NED);
639}
640
641//******************************************************************************
642
644{
645 double agl = GetAltitudeAGLFtIC();
646 fdmex->GetInertial()->SetTerrainElevation(elev);
647
648 if (lastAltitudeSet == setagl)
650}
651
652//******************************************************************************
653
655{
656 return position.GetRadius() - position.GetSeaLevelRadius();
657}
658
659//******************************************************************************
660
662{
663 return fdmex->GetInertial()->GetAltitudeAGL(position);
664}
665
666//******************************************************************************
667
669{
670 FGColumnVector3 normal, v, w;
671 FGLocation contact;
672 double a = fdmex->GetInertial()->GetSemimajor();
673 double b = fdmex->GetInertial()->GetSemiminor();
674 contact.SetEllipse(a, b);
675 fdmex->GetInertial()->GetContactPoint(position, contact, normal, v, w);
676 return contact.GetGeodAltitude();
677}
678
679//******************************************************************************
680
682{
683 const auto Atmosphere = fdmex->GetAtmosphere();
684 double altitudeASL = GetAltitudeASLFtIC();
685 double pressure = Atmosphere->GetPressure(altitudeASL);
686 double soundSpeed = Atmosphere->GetSoundSpeed(altitudeASL);
687 double rho = Atmosphere->GetDensity(altitudeASL);
688
689 double mach0 = vt / soundSpeed;
690 double vc0 = Auxiliary->VcalibratedFromMach(mach0, pressure);
691 double ve0 = vt * sqrt(rho/FGAtmosphere::StdDaySLdensity);
692
693 switch(lastLatitudeSet) {
694 case setgeod:
695 fdmex->GetInertial()->SetAltitudeAGL(position, agl);
696 break;
697 case setgeoc:
698 {
699 double a = fdmex->GetInertial()->GetSemimajor();
700 double b = fdmex->GetInertial()->GetSemiminor();
701 double e2 = 1.0-b*b/(a*a);
702 double tanlat = tan(position.GetLatitude());
703 double n = e2;
704 double prev_n = 1.0;
705 int iter = 0;
706 double longitude = position.GetLongitude();
707 double alt = position.GetGeodAltitude();
708 double h = -2.0*max(a,b);
709 double geodLat;
710 while ((fabs(n-prev_n) > 1E-15 || fabs(h-agl) > 1E-10) && iter < 10) {
711 geodLat = atan(tanlat/(1-n));
712 position.SetPositionGeodetic(longitude, geodLat, alt);
713 h = GetAltitudeAGLFtIC();
714 alt += agl-h;
715 double sinGeodLat = sin(geodLat);
716 double N = a/sqrt(1-e2*sinGeodLat*sinGeodLat);
717 prev_n = n;
718 n = e2*N/(N+alt);
719 iter++;
720 }
721 }
722 break;
723 }
724
725 altitudeASL = GetAltitudeASLFtIC();
726 soundSpeed = Atmosphere->GetSoundSpeed(altitudeASL);
727 rho = Atmosphere->GetDensity(altitudeASL);
728 pressure = Atmosphere->GetPressure(altitudeASL);
729
730 switch(lastSpeedSet) {
731 case setvc:
732 mach0 = Auxiliary->MachFromVcalibrated(vc0, pressure);
733 SetVtrueFpsIC(mach0 * soundSpeed);
734 break;
735 case setmach:
736 SetVtrueFpsIC(mach0 * soundSpeed);
737 break;
738 case setve:
739 SetVtrueFpsIC(ve0 * sqrt(FGAtmosphere::StdDaySLdensity/rho));
740 break;
741 default: // Make the compiler stop complaining about missing enums
742 break;
743 }
744
745 lastAltitudeSet = setagl;
746}
747
748//******************************************************************************
749// Set the altitude SL. If the airspeed has been previously set with parameters
750// that are atmosphere dependent (Mach, VCAS, VEAS) then the true airspeed is
751// modified to keep the last set speed to its previous value.
752
754{
755 const auto Atmosphere = fdmex->GetAtmosphere();
756 double altitudeASL = GetAltitudeASLFtIC();
757 double pressure = Atmosphere->GetPressure(altitudeASL);
758 double soundSpeed = Atmosphere->GetSoundSpeed(altitudeASL);
759 double rho = Atmosphere->GetDensity(altitudeASL);
760
761 double mach0 = vt / soundSpeed;
762 double vc0 = Auxiliary->VcalibratedFromMach(mach0, pressure);
763 double ve0 = vt * sqrt(rho/FGAtmosphere::StdDaySLdensity);
764
765 switch(lastLatitudeSet) {
766 case setgeod:
767 {
768 // Given an altitude above the mean sea level (or a position radius which
769 // is the same) and a geodetic latitude, compute the geodetic altitude.
770 double a = fdmex->GetInertial()->GetSemimajor();
771 double b = fdmex->GetInertial()->GetSemiminor();
772 double e2 = 1.0-b*b/(a*a);
773 double geodLatitude = position.GetGeodLatitudeRad();
774 double cosGeodLat = cos(geodLatitude);
775 double sinGeodLat = sin(geodLatitude);
776 double N = a/sqrt(1-e2*sinGeodLat*sinGeodLat);
777 double geodAlt = 0.0;
778 double n = e2;
779 double prev_n = 1.0;
780 int iter = 0;
781 // Use tan or cotan to solve the geodetic altitude to avoid floating point
782 // exceptions.
783 if (cosGeodLat > fabs(sinGeodLat)) { // tan() can safely be used.
784 double tanGeodLat = sinGeodLat/cosGeodLat;
785 double x0 = N*e2*cosGeodLat;
786 double x = 0.0;
787 while (fabs(n-prev_n) > 1E-15 && iter < 10) {
788 double tanLat = (1-n)*tanGeodLat; // See Stevens & Lewis 1.6-14
789 double cos2Lat = 1./(1.+tanLat*tanLat);
790 double slr = b/sqrt(1.-e2*cos2Lat);
791 double R = slr + alt;
792 x = R*sqrt(cos2Lat); // OK, cos(latitude) is always positive.
793 prev_n = n;
794 n = x0/x;
795 iter++;
796 }
797 geodAlt = x/cosGeodLat-N;
798 }
799 else { // better use cotan (i.e. 1./tan())
800 double cotanGeodLat = cosGeodLat/sinGeodLat;
801 double z0 = N*e2*sinGeodLat;
802 double z = 0.0;
803 while (fabs(n-prev_n) > 1E-15 && iter < 10) {
804 double cotanLat = cotanGeodLat/(1-n);
805 double sin2Lat = 1./(1.+cotanLat*cotanLat);
806 double cos2Lat = 1.-sin2Lat;
807 double slr = b/sqrt(1.-e2*cos2Lat);
808 double R = slr + alt;
809 z = R*sign(cotanLat)*sqrt(sin2Lat);
810 prev_n = n;
811 n = z0/(z0+z);
812 iter++;
813 }
814 geodAlt = z/sinGeodLat-N*(1-e2);
815 }
816
817 double longitude = position.GetLongitude();
818 position.SetPositionGeodetic(longitude, geodLatitude, geodAlt);
819 }
820 break;
821 case setgeoc:
822 {
823 double slr = position.GetSeaLevelRadius();
824 position.SetRadius(slr+alt);
825 }
826 break;
827 }
828
829 altitudeASL = position.GetGeodAltitude();
830 soundSpeed = Atmosphere->GetSoundSpeed(altitudeASL);
831 rho = Atmosphere->GetDensity(altitudeASL);
832 pressure = Atmosphere->GetPressure(altitudeASL);
833
834 switch(lastSpeedSet) {
835 case setvc:
836 mach0 = Auxiliary->MachFromVcalibrated(vc0, pressure);
837 SetVtrueFpsIC(mach0 * soundSpeed);
838 break;
839 case setmach:
840 SetVtrueFpsIC(mach0 * soundSpeed);
841 break;
842 case setve:
843 SetVtrueFpsIC(ve0 * sqrt(FGAtmosphere::StdDaySLdensity/rho));
844 break;
845 default: // Make the compiler stop complaining about missing enums
846 break;
847 }
848
849 lastAltitudeSet = setasl;
850}
851
852//******************************************************************************
853
855{
856 double lon = position.GetLongitude();
857 lastLatitudeSet = setgeod;
858
859 switch (lastAltitudeSet)
860 {
861 case setagl:
862 {
863 double agl = GetAltitudeAGLFtIC();
864 position.SetPositionGeodetic(lon, geodLatitude, 0.);
865 fdmex->GetInertial()->SetAltitudeAGL(position, agl);
866 }
867 break;
868 case setasl:
869 {
870 double asl = GetAltitudeASLFtIC();
871 position.SetPositionGeodetic(lon, geodLatitude, 0.);
873 }
874 break;
875 }
876}
877
878//******************************************************************************
879
881{
882 double altitude;
883
884 lastLatitudeSet = setgeoc;
885
886 switch(lastAltitudeSet) {
887 case setagl:
888 altitude = GetAltitudeAGLFtIC();
889 position.SetLatitude(lat);
890 SetAltitudeAGLFtIC(altitude);
891 break;
892 default:
893 altitude = GetAltitudeASLFtIC();
894 position.SetLatitude(lat);
895 SetAltitudeASLFtIC(altitude);
896 break;
897 }
898}
899
900//******************************************************************************
901
903{
904 double altitude;
905
906 switch(lastAltitudeSet) {
907 case setagl:
908 altitude = GetAltitudeAGLFtIC();
909 position.SetLongitude(lon);
910 SetAltitudeAGLFtIC(altitude);
911 break;
912 default:
913 position.SetLongitude(lon);
914 break;
915 }
916}
917
918//******************************************************************************
919
921{
922 const FGMatrix33& Tb2l = orientation.GetTInv();
923 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
924 FGColumnVector3 _vWIND_NED = _vt_NED - vUVW_NED;
925
926 return _vWIND_NED.Magnitude(eU, eV) == 0.0 ? 0.0
927 : atan2(_vWIND_NED(eV), _vWIND_NED(eU))*radtodeg;
928}
929
930//******************************************************************************
931
933 const FGMatrix33& Tb2l = orientation.GetTInv();
934 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
935 return _vt_NED - vUVW_NED;
936}
937
938//******************************************************************************
939
941{
942 const FGMatrix33& Tb2l = orientation.GetTInv();
943 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
944 FGColumnVector3 _vWIND_NED = _vt_NED - vUVW_NED;
945
946 return _vWIND_NED.Magnitude(eU, eV);
947}
948
949//******************************************************************************
950
951double FGInitialCondition::GetBodyWindFpsIC(int idx) const
952{
953 const FGMatrix33& Tl2b = orientation.GetT();
954 FGColumnVector3 _vt_BODY = Tw2b * FGColumnVector3(vt, 0., 0.);
955 FGColumnVector3 _vUVW_BODY = Tl2b * vUVW_NED;
956 FGColumnVector3 _vWIND_BODY = _vt_BODY - _vUVW_BODY;
957
958 return _vWIND_BODY(idx);
959}
960
961//******************************************************************************
962
964{
965 const auto Atmosphere = fdmex->GetAtmosphere();
966 double altitudeASL = GetAltitudeASLFtIC();
967 double pressure = Atmosphere->GetPressure(altitudeASL);
968 double soundSpeed = Atmosphere->GetSoundSpeed(altitudeASL);
969 double mach = vt / soundSpeed;
970
971 return fpstokts * Auxiliary->VcalibratedFromMach(mach, pressure);
972}
973
974//******************************************************************************
975
977{
978 const auto Atmosphere = fdmex->GetAtmosphere();
979 double altitudeASL = GetAltitudeASLFtIC();
980 double rho = Atmosphere->GetDensity(altitudeASL);
981 return fpstokts * vt * sqrt(rho/FGAtmosphere::StdDaySLdensity);
982}
983
984//******************************************************************************
985
987{
988 const auto Atmosphere = fdmex->GetAtmosphere();
989 double altitudeASL = GetAltitudeASLFtIC();
990 double soundSpeed = Atmosphere->GetSoundSpeed(altitudeASL);
991 return vt / soundSpeed;
992}
993
994//******************************************************************************
995
996double FGInitialCondition::GetBodyVelFpsIC(int idx) const
997{
998 const FGMatrix33& Tl2b = orientation.GetT();
999 FGColumnVector3 _vUVW_BODY = Tl2b * vUVW_NED;
1000
1001 return _vUVW_BODY(idx);
1002}
1003
1004//******************************************************************************
1005
1006bool FGInitialCondition::Load(const SGPath& rstfile, bool useAircraftPath)
1007{
1008 SGPath init_file_name;
1009 if(useAircraftPath && rstfile.isRelative()) {
1010 init_file_name = fdmex->GetFullAircraftPath()/rstfile.utf8Str();
1011 } else {
1012 init_file_name = rstfile;
1013 }
1014
1015 FGXMLFileRead XMLFileRead;
1016 Element* document = XMLFileRead.LoadXMLDocument(init_file_name);
1017
1018 // Make sure that the document is valid
1019 if (!document) {
1020 stringstream s;
1021 s << "File: " << init_file_name << " could not be read.";
1022 cerr << s.str() << endl;
1023 throw BaseException(s.str());
1024 }
1025
1026 if (document->GetName() != "initialize") {
1027 stringstream s;
1028 s << "File: " << init_file_name << " is not a reset file.";
1029 cerr << s.str() << endl;
1030 throw BaseException(s.str());
1031 }
1032
1033 bool result = false;
1034
1035 // If doc has an version, check it. Otherwise fall back to legacy.
1036 if (document->HasAttribute("version")) {
1037 double version = document->GetAttributeValueAsNumber("version");
1038
1039 if (version >= 3.0) {
1040 const string s("Only initialization file formats 1 and 2 are currently supported");
1041 cerr << document->ReadFrom() << endl << s << endl;
1042 throw BaseException(s);
1043 } else if (version >= 2.0) {
1044 result = Load_v2(document);
1045 } else if (version >= 1.0) {
1046 result = Load_v1(document);
1047 }
1048
1049 } else {
1050 result = Load_v1(document);
1051 }
1052
1053 // Check to see if any engines are specified to be initialized in a running state
1054 Element* running_elements = document->FindElement("running");
1055 while (running_elements) {
1056 int engineNumber = int(running_elements->GetDataAsNumber());
1057 enginesRunning |= engineNumber == -1 ? engineNumber : 1 << engineNumber;
1058 running_elements = document->FindNextElement("running");
1059 }
1060
1061 return result;
1062}
1063
1064//******************************************************************************
1065
1066bool FGInitialCondition::LoadLatitude(Element* position_el)
1067{
1068 Element* latitude_el = position_el->FindElement("latitude");
1069
1070 if (latitude_el) {
1071 double latitude = position_el->FindElementValueAsNumberConvertTo("latitude", "RAD");
1072
1073 if (fabs(latitude) > 0.5*M_PI) {
1074 string unit_type = latitude_el->GetAttributeValue("unit");
1075 if (unit_type.empty()) unit_type="RAD";
1076
1077 cerr << latitude_el->ReadFrom() << "The latitude value "
1078 << latitude_el->GetDataAsNumber() << " " << unit_type
1079 << " is outside the range [";
1080 if (unit_type == "DEG")
1081 cerr << "-90 DEG ; +90 DEG]" << endl;
1082 else
1083 cerr << "-PI/2 RAD; +PI/2 RAD]" << endl;
1084
1085 return false;
1086 }
1087
1088 string lat_type = latitude_el->GetAttributeValue("type");
1089
1090 if (lat_type == "geod" || lat_type == "geodetic") {
1091 SetGeodLatitudeRadIC(latitude);
1092 lastLatitudeSet = setgeod;
1093 }
1094 else {
1095 SetLatitudeRadIC(latitude);
1096 lastLatitudeSet = setgeoc;
1097 }
1098 }
1099
1100 return true;
1101}
1102
1103//******************************************************************************
1104
1105void FGInitialCondition::SetTrimRequest(std::string trim)
1106{
1107 std::string& trimOption = to_lower(trim);
1108 if (trimOption == "1")
1109 trimRequested = TrimMode::tGround; // For backwards compatabiity
1110 else if (trimOption == "longitudinal")
1111 trimRequested = TrimMode::tLongitudinal;
1112 else if (trimOption == "full")
1113 trimRequested = TrimMode::tFull;
1114 else if (trimOption == "ground")
1115 trimRequested = TrimMode::tGround;
1116 else if (trimOption == "pullup")
1117 trimRequested = TrimMode::tPullup;
1118 else if (trimOption == "custom")
1119 trimRequested = TrimMode::tCustom;
1120 else if (trimOption == "turn")
1121 trimRequested = TrimMode::tTurn;
1122}
1123
1124//******************************************************************************
1125
1126bool FGInitialCondition::Load_v1(Element* document)
1127{
1128 bool result = true;
1129
1130 if (document->FindElement("longitude"))
1131 SetLongitudeRadIC(document->FindElementValueAsNumberConvertTo("longitude", "RAD"));
1132 if (document->FindElement("elevation"))
1133 SetTerrainElevationFtIC(document->FindElementValueAsNumberConvertTo("elevation", "FT"));
1134
1135 if (document->FindElement("altitude")) // This is feet above ground level
1136 SetAltitudeAGLFtIC(document->FindElementValueAsNumberConvertTo("altitude", "FT"));
1137 else if (document->FindElement("altitudeAGL")) // This is feet above ground level
1138 SetAltitudeAGLFtIC(document->FindElementValueAsNumberConvertTo("altitudeAGL", "FT"));
1139 else if (document->FindElement("altitudeMSL")) // This is feet above sea level
1140 SetAltitudeASLFtIC(document->FindElementValueAsNumberConvertTo("altitudeMSL", "FT"));
1141
1142 result = LoadLatitude(document);
1143
1144 FGColumnVector3 vOrient = orientation.GetEuler();
1145
1146 if (document->FindElement("phi"))
1147 vOrient(ePhi) = document->FindElementValueAsNumberConvertTo("phi", "RAD");
1148 if (document->FindElement("theta"))
1149 vOrient(eTht) = document->FindElementValueAsNumberConvertTo("theta", "RAD");
1150 if (document->FindElement("psi"))
1151 vOrient(ePsi) = document->FindElementValueAsNumberConvertTo("psi", "RAD");
1152
1153 orientation = FGQuaternion(vOrient);
1154
1155 if (document->FindElement("ubody"))
1156 SetUBodyFpsIC(document->FindElementValueAsNumberConvertTo("ubody", "FT/SEC"));
1157 if (document->FindElement("vbody"))
1158 SetVBodyFpsIC(document->FindElementValueAsNumberConvertTo("vbody", "FT/SEC"));
1159 if (document->FindElement("wbody"))
1160 SetWBodyFpsIC(document->FindElementValueAsNumberConvertTo("wbody", "FT/SEC"));
1161 if (document->FindElement("vnorth"))
1162 SetVNorthFpsIC(document->FindElementValueAsNumberConvertTo("vnorth", "FT/SEC"));
1163 if (document->FindElement("veast"))
1164 SetVEastFpsIC(document->FindElementValueAsNumberConvertTo("veast", "FT/SEC"));
1165 if (document->FindElement("vdown"))
1166 SetVDownFpsIC(document->FindElementValueAsNumberConvertTo("vdown", "FT/SEC"));
1167 if (document->FindElement("vc"))
1168 SetVcalibratedKtsIC(document->FindElementValueAsNumberConvertTo("vc", "KTS"));
1169 if (document->FindElement("vt"))
1170 SetVtrueKtsIC(document->FindElementValueAsNumberConvertTo("vt", "KTS"));
1171 if (document->FindElement("mach"))
1172 SetMachIC(document->FindElementValueAsNumber("mach"));
1173 if (document->FindElement("gamma"))
1174 SetFlightPathAngleDegIC(document->FindElementValueAsNumberConvertTo("gamma", "DEG"));
1175 if (document->FindElement("roc"))
1176 SetClimbRateFpsIC(document->FindElementValueAsNumberConvertTo("roc", "FT/SEC"));
1177 if (document->FindElement("vground"))
1178 SetVgroundKtsIC(document->FindElementValueAsNumberConvertTo("vground", "KTS"));
1179 if (document->FindElement("alpha"))
1180 SetAlphaDegIC(document->FindElementValueAsNumberConvertTo("alpha", "DEG"));
1181 if (document->FindElement("beta"))
1182 SetBetaDegIC(document->FindElementValueAsNumberConvertTo("beta", "DEG"));
1183 if (document->FindElement("vwind"))
1184 SetWindMagKtsIC(document->FindElementValueAsNumberConvertTo("vwind", "KTS"));
1185 if (document->FindElement("winddir"))
1186 SetWindDirDegIC(document->FindElementValueAsNumberConvertTo("winddir", "DEG"));
1187 if (document->FindElement("hwind"))
1188 SetHeadWindKtsIC(document->FindElementValueAsNumberConvertTo("hwind", "KTS"));
1189 if (document->FindElement("xwind"))
1190 SetCrossWindKtsIC(document->FindElementValueAsNumberConvertTo("xwind", "KTS"));
1191 if (document->FindElement("targetNlf"))
1192 SetTargetNlfIC(document->FindElementValueAsNumber("targetNlf"));
1193 if (document->FindElement("trim"))
1194 SetTrimRequest(document->FindElementValue("trim"));
1195
1196 vPQR_body.InitMatrix();
1197
1198 return result;
1199}
1200
1201//******************************************************************************
1202
1203bool FGInitialCondition::Load_v2(Element* document)
1204{
1205 FGColumnVector3 vOrient;
1206 bool result = true;
1207
1208 // support both earth_position_angle and planet_position_angle, for now.
1209 if (document->FindElement("earth_position_angle"))
1210 epa = document->FindElementValueAsNumberConvertTo("earth_position_angle", "RAD");
1211 if (document->FindElement("planet_position_angle"))
1212 epa = document->FindElementValueAsNumberConvertTo("planet_position_angle", "RAD");
1213
1214 // Calculate the inertial to ECEF matrices
1215 FGMatrix33 Ti2ec(cos(epa), sin(epa), 0.0,
1216 -sin(epa), cos(epa), 0.0,
1217 0.0, 0.0, 1.0);
1218 FGMatrix33 Tec2i = Ti2ec.Transposed();
1219
1220 if (document->FindElement("planet_rotation_rate")) {
1221 fdmex->GetInertial()->SetOmegaPlanet(document->FindElementValueAsNumberConvertTo("planet_rotation_rate", "RAD"));
1222 fdmex->GetPropagate()->in.vOmegaPlanet = fdmex->GetInertial()->GetOmegaPlanet();
1223 fdmex->GetAccelerations()->in.vOmegaPlanet = fdmex->GetInertial()->GetOmegaPlanet();
1224 }
1225 FGColumnVector3 vOmegaEarth = fdmex->GetInertial()->GetOmegaPlanet();
1226
1227 if (document->FindElement("elevation")) {
1228 fdmex->GetInertial()->SetTerrainElevation(document->FindElementValueAsNumberConvertTo("elevation", "FT"));
1229 }
1230
1231 // Initialize vehicle position
1232 //
1233 // Allowable frames:
1234 // - ECI (Earth Centered Inertial)
1235 // - ECEF (Earth Centered, Earth Fixed)
1236
1237 Element* position_el = document->FindElement("position");
1238 if (position_el) {
1239 string frame = position_el->GetAttributeValue("frame");
1240 frame = to_lower(frame);
1241 if (frame == "eci") { // Need to transform vLoc to ECEF for storage and use in FGLocation.
1242 position = Ti2ec * position_el->FindElementTripletConvertTo("FT");
1243 } else if (frame == "ecef") {
1244 if (!position_el->FindElement("x") && !position_el->FindElement("y") && !position_el->FindElement("z")) {
1245 if (position_el->FindElement("longitude")) {
1246 SetLongitudeRadIC(position_el->FindElementValueAsNumberConvertTo("longitude", "RAD"));
1247 }
1248 if (position_el->FindElement("radius")) {
1249 position.SetRadius(position_el->FindElementValueAsNumberConvertTo("radius", "FT"));
1250 } else if (position_el->FindElement("altitudeAGL")) {
1251 SetAltitudeAGLFtIC(position_el->FindElementValueAsNumberConvertTo("altitudeAGL", "FT"));
1252 } else if (position_el->FindElement("altitudeMSL")) {
1253 SetAltitudeASLFtIC(position_el->FindElementValueAsNumberConvertTo("altitudeMSL", "FT"));
1254 } else {
1255 cerr << endl << " No altitude or radius initial condition is given." << endl;
1256 result = false;
1257 }
1258
1259 if (result)
1260 result = LoadLatitude(position_el);
1261
1262 } else {
1263 position = position_el->FindElementTripletConvertTo("FT");
1264 }
1265 } else {
1266 cerr << endl << " Neither ECI nor ECEF frame is specified for initial position." << endl;
1267 result = false;
1268 }
1269 } else {
1270 cerr << endl << " Initial position not specified in this initialization file." << endl;
1271 result = false;
1272 }
1273
1274 // End of position initialization
1275
1276 // Initialize vehicle orientation
1277 // Allowable frames
1278 // - ECI (Earth Centered Inertial)
1279 // - ECEF (Earth Centered, Earth Fixed)
1280 // - Local
1281 //
1282 // Need to convert the provided orientation to a local orientation, using
1283 // the given orientation and knowledge of the Earth position angle.
1284 // This could be done using matrices (where in the subscript "b/a",
1285 // it is meant "b with respect to a", and where b=body frame,
1286 // i=inertial frame, l=local NED frame and e=ecef frame) as:
1287 //
1288 // C_b/l = C_b/e * C_e/l
1289 //
1290 // Using quaternions (note reverse ordering compared to matrix representation):
1291 //
1292 // Q_b/l = Q_e/l * Q_b/e
1293 //
1294 // Use the specific matrices as needed. The above example of course is for the whole
1295 // body to local orientation.
1296 // The new orientation angles can be extracted from the matrix or the quaternion.
1297 // ToDo: Do we need to deal with normalization of the quaternions here?
1298
1299 Element* orientation_el = document->FindElement("orientation");
1300 if (orientation_el) {
1301 string frame = orientation_el->GetAttributeValue("frame");
1302 frame = to_lower(frame);
1303 vOrient = orientation_el->FindElementTripletConvertTo("RAD");
1304 if (frame == "eci") {
1305
1306 // In this case, we are supplying the Euler angles for the vehicle with
1307 // respect to the inertial system, represented by the C_b/i Matrix.
1308 // We want the body orientation with respect to the local (NED frame):
1309 //
1310 // C_b/l = C_b/i * C_i/l
1311 //
1312 // Or, using quaternions (note reverse ordering compared to matrix representation):
1313 //
1314 // Q_b/l = Q_i/l * Q_b/i
1315
1316 FGQuaternion QuatI2Body = FGQuaternion(vOrient);
1317 QuatI2Body.Normalize();
1318 FGQuaternion QuatLocal2I = Tec2i * position.GetTl2ec();
1319 QuatLocal2I.Normalize();
1320 orientation = QuatLocal2I * QuatI2Body;
1321
1322 } else if (frame == "ecef") {
1323
1324 // In this case we are given the Euler angles representing the orientation of
1325 // the body with respect to the ECEF system, represented by the C_b/e Matrix.
1326 // We want the body orientation with respect to the local (NED frame):
1327 //
1328 // C_b/l = C_b/e * C_e/l
1329 //
1330 // Using quaternions (note reverse ordering compared to matrix representation):
1331 //
1332 // Q_b/l = Q_e/l * Q_b/e
1333
1334 FGQuaternion QuatEC2Body(vOrient); // Store relationship of Body frame wrt ECEF frame, Q_b/e
1335 QuatEC2Body.Normalize();
1336 FGQuaternion QuatLocal2EC = position.GetTl2ec(); // Get Q_e/l from matrix
1337 QuatLocal2EC.Normalize();
1338 orientation = QuatLocal2EC * QuatEC2Body; // Q_b/l = Q_e/l * Q_b/e
1339
1340 } else if (frame == "local") {
1341
1342 orientation = FGQuaternion(vOrient);
1343
1344 } else {
1345
1346 cerr << endl << fgred << " Orientation frame type: \"" << frame
1347 << "\" is not supported!" << reset << endl << endl;
1348 result = false;
1349
1350 }
1351 }
1352
1353 // Initialize vehicle velocity
1354 // Allowable frames
1355 // - ECI (Earth Centered Inertial)
1356 // - ECEF (Earth Centered, Earth Fixed)
1357 // - Local
1358 // - Body
1359 // The vehicle will be defaulted to (0,0,0) in the Body frame if nothing is provided.
1360
1361 Element* velocity_el = document->FindElement("velocity");
1362 FGMatrix33 mTec2l = position.GetTec2l();
1363 const FGMatrix33& Tb2l = orientation.GetTInv();
1364
1365 if (velocity_el) {
1366
1367 string frame = velocity_el->GetAttributeValue("frame");
1368 frame = to_lower(frame);
1369 FGColumnVector3 vInitVelocity = velocity_el->FindElementTripletConvertTo("FT/SEC");
1370
1371 if (frame == "eci") {
1372 FGColumnVector3 omega_cross_r = vOmegaEarth * (Tec2i * position);
1373 vUVW_NED = mTec2l * (vInitVelocity - omega_cross_r);
1374 lastSpeedSet = setned;
1375 } else if (frame == "ecef") {
1376 vUVW_NED = mTec2l * vInitVelocity;
1377 lastSpeedSet = setned;
1378 } else if (frame == "local") {
1379 vUVW_NED = vInitVelocity;
1380 lastSpeedSet = setned;
1381 } else if (frame == "body") {
1382 vUVW_NED = Tb2l * vInitVelocity;
1383 lastSpeedSet = setuvw;
1384 } else {
1385
1386 cerr << endl << fgred << " Velocity frame type: \"" << frame
1387 << "\" is not supported!" << reset << endl << endl;
1388 result = false;
1389
1390 }
1391
1392 } else {
1393
1394 vUVW_NED.InitMatrix();
1395
1396 }
1397
1398 vt = vUVW_NED.Magnitude();
1399
1400 calcAeroAngles(vUVW_NED);
1401
1402 // Initialize vehicle body rates
1403 // Allowable frames
1404 // - ECI (Earth Centered Inertial)
1405 // - ECEF (Earth Centered, Earth Fixed)
1406 // - Body
1407
1408 Element* attrate_el = document->FindElement("attitude_rate");
1409
1410 if (attrate_el) {
1411
1412 string frame = attrate_el->GetAttributeValue("frame");
1413 frame = to_lower(frame);
1414 const FGMatrix33& Tl2b = orientation.GetT();
1415 FGColumnVector3 vAttRate = attrate_el->FindElementTripletConvertTo("RAD/SEC");
1416
1417 if (frame == "eci") {
1418 FGMatrix33 Ti2l = position.GetTec2l() * Ti2ec;
1419 vPQR_body = Tl2b * Ti2l * (vAttRate - vOmegaEarth);
1420 } else if (frame == "ecef") {
1421 vPQR_body = Tl2b * position.GetTec2l() * vAttRate;
1422 } else if (frame == "local") {
1423 // Refer to Stevens and Lewis, 1.5-14a, pg. 49.
1424 // This is the rotation rate of the "Local" frame, expressed in the local frame.
1425 double radInv = 1.0 / position.GetRadius();
1426 FGColumnVector3 vOmegaLocal = {radInv*vUVW_NED(eEast),
1427 -radInv*vUVW_NED(eNorth),
1428 -radInv*vUVW_NED(eEast)*tan(position.GetLatitude())};
1429 vPQR_body = Tl2b * (vAttRate + vOmegaLocal);
1430 } else if (frame == "body") {
1431 vPQR_body = vAttRate;
1432 } else if (!frame.empty()) { // misspelling of frame
1433
1434 cerr << endl << fgred << " Attitude rate frame type: \"" << frame
1435 << "\" is not supported!" << reset << endl << endl;
1436 result = false;
1437
1438 } else if (frame.empty()) {
1439 vPQR_body.InitMatrix();
1440 }
1441
1442 } else { // Body frame attitude rate assumed 0 relative to local.
1443 vPQR_body.InitMatrix();
1444 }
1445
1446 return result;
1447}
1448
1449//******************************************************************************
1450
1451void FGInitialCondition::bind(FGPropertyManager* PropertyManager)
1452{
1453 PropertyManager->Tie("ic/vc-kts", this,
1456 PropertyManager->Tie("ic/ve-kts", this,
1459 PropertyManager->Tie("ic/vg-kts", this,
1462 PropertyManager->Tie("ic/vt-kts", this,
1465 PropertyManager->Tie("ic/mach", this,
1468 PropertyManager->Tie("ic/roc-fpm", this,
1471 PropertyManager->Tie("ic/gamma-deg", this,
1474 PropertyManager->Tie("ic/alpha-deg", this,
1477 PropertyManager->Tie("ic/beta-deg", this,
1480 PropertyManager->Tie("ic/theta-deg", this,
1483 PropertyManager->Tie("ic/phi-deg", this,
1486 PropertyManager->Tie("ic/psi-true-deg", this,
1489 PropertyManager->Tie("ic/lat-gc-deg", this,
1492 PropertyManager->Tie("ic/long-gc-deg", this,
1495 PropertyManager->Tie("ic/h-sl-ft", this,
1498 PropertyManager->Tie("ic/h-agl-ft", this,
1501 PropertyManager->Tie("ic/terrain-elevation-ft", this,
1504 PropertyManager->Tie("ic/vg-fps", this,
1507 PropertyManager->Tie("ic/vt-fps", this,
1510 PropertyManager->Tie("ic/vw-bx-fps", this,
1512 PropertyManager->Tie("ic/vw-by-fps", this,
1514 PropertyManager->Tie("ic/vw-bz-fps", this,
1516 PropertyManager->Tie("ic/vw-north-fps", this,
1518 PropertyManager->Tie("ic/vw-east-fps", this,
1520 PropertyManager->Tie("ic/vw-down-fps", this,
1522 PropertyManager->Tie("ic/vw-mag-fps", this,
1525 PropertyManager->Tie("ic/vw-dir-deg", this,
1528
1529 PropertyManager->Tie("ic/roc-fps", this,
1532 PropertyManager->Tie("ic/u-fps", this,
1535 PropertyManager->Tie("ic/v-fps", this,
1538 PropertyManager->Tie("ic/w-fps", this,
1541 PropertyManager->Tie("ic/vn-fps", this,
1544 PropertyManager->Tie("ic/ve-fps", this,
1547 PropertyManager->Tie("ic/vd-fps", this,
1550 PropertyManager->Tie("ic/gamma-rad", this,
1553 PropertyManager->Tie("ic/alpha-rad", this,
1556 PropertyManager->Tie("ic/theta-rad", this,
1559 PropertyManager->Tie("ic/beta-rad", this,
1562 PropertyManager->Tie("ic/phi-rad", this,
1565 PropertyManager->Tie("ic/psi-true-rad", this,
1568 PropertyManager->Tie("ic/lat-gc-rad", this,
1571 PropertyManager->Tie("ic/long-gc-rad", this,
1574 PropertyManager->Tie("ic/p-rad_sec", this,
1577 PropertyManager->Tie("ic/q-rad_sec", this,
1580 PropertyManager->Tie("ic/r-rad_sec", this,
1583 PropertyManager->Tie("ic/lat-geod-rad", this,
1586 PropertyManager->Tie("ic/lat-geod-deg", this,
1589 PropertyManager->Tie("ic/geod-alt-ft", &position,
1591
1592 PropertyManager->Tie("ic/targetNlf", this,
1595}
1596
1597//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
1598// The bitmasked value choices are as follows:
1599// unset: In this case (the default) JSBSim would only print
1600// out the normally expected messages, essentially echoing
1601// the config files as they are read. If the environment
1602// variable is not set, debug_lvl is set to 1 internally
1603// 0: This requests JSBSim not to output any messages
1604// whatsoever.
1605// 1: This value explicity requests the normal JSBSim
1606// startup messages
1607// 2: This value asks for a message to be printed out when
1608// a class is instantiated
1609// 4: When this value is set, a message is displayed when a
1610// FGModel object executes its Run() method
1611// 8: When this value is set, various runtime state variables
1612// are printed out periodically
1613// 16: When set various parameters are sanity checked and
1614// a message is printed out when they go out of bounds
1615
1616void FGInitialCondition::Debug(int from)
1617{
1618 if (debug_lvl <= 0) return;
1619
1620 if (debug_lvl & 1) { // Standard console startup message output
1621 }
1622 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
1623 if (from == 0) cout << "Instantiated: FGInitialCondition" << endl;
1624 if (from == 1) cout << "Destroyed: FGInitialCondition" << endl;
1625 }
1626 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
1627 }
1628 if (debug_lvl & 8 ) { // Runtime state variables
1629 }
1630 if (debug_lvl & 16) { // Sanity checking
1631 }
1632 if (debug_lvl & 64) {
1633 if (from == 0) { // Constructor
1634 }
1635 }
1636}
1637}
Element * FindElement(const std::string &el="")
Searches for a specified element.
const std::string & GetName(void) const
Retrieves the element name.
double GetAttributeValueAsNumber(const std::string &key)
Retrieves an attribute value as a double precision real number.
std::string GetAttributeValue(const std::string &key)
Retrieves an attribute.
std::string ReadFrom(void) const
Return a string that contains a description of the location where the current XML element was read fr...
double FindElementValueAsNumberConvertTo(const std::string &el, const std::string &target_units)
Searches for the named element and converts and returns the data belonging to it.
Element * FindNextElement(const std::string &el="")
Searches for the next element as specified.
double GetDataAsNumber(void)
Converts the element data to a number.
bool HasAttribute(const std::string &key)
Determines if an element has the supplied attribute.
This class implements a 3 element column vector.
double Magnitude(void) const
Length of the vector.
FGColumnVector3 & Normalize(void)
Normalize.
Encapsulates the JSBSim simulation executive.
Definition FGFDMExec.h:184
std::shared_ptr< FGAircraft > GetAircraft(void) const
Returns the FGAircraft pointer.
const SGPath & GetFullAircraftPath(void)
Retrieves the full aircraft path name.
Definition FGFDMExec.h:401
std::shared_ptr< FGPropagate > GetPropagate(void) const
Returns the FGPropagate pointer.
std::shared_ptr< FGInertial > GetInertial(void) const
Returns the FGInertial pointer.
std::shared_ptr< FGAuxiliary > GetAuxiliary(void) const
Returns the FGAuxiliary pointer.
std::shared_ptr< FGAccelerations > GetAccelerations(void) const
Returns the FGAccelerations pointer.
void SetPsiRadIC(double psi)
Sets the initial heading angle.
double GetPRadpsIC() const
Gets the initial body axis roll rate.
double GetVNorthFpsIC(void) const
Gets the initial local frame X (North) velocity.
void SetThetaDegIC(double theta)
Sets pitch angle initial condition in degrees.
void SetVequivalentKtsIC(double ve)
Set equivalent airspeed initial condition in knots.
double GetPhiRadIC(void) const
Gets the initial roll angle.
double GetVBodyFpsIC(void) const
Gets the initial body axis Y velocity.
double GetVEastFpsIC(void) const
Gets the initial local frame Y (East) velocity.
double GetLatitudeDegIC(void) const
Gets the initial latitude.
void SetWindDownKtsIC(double wD)
Sets the initial wind downward speed.
void ResetIC(double u0, double v0, double w0, double p0, double q0, double r0, double alpha0, double beta0, double phi0, double theta0, double psi0, double latitudeRad0, double longitudeRad0, double altitudeAGL0, double gamma0)
Resets the IC data structure to new values.
double GetVgroundFpsIC(void) const
Gets the initial ground velocity.
double GetLongitudeRadIC(void) const
Gets the initial longitude.
double GetFlightPathAngleDegIC(void) const
Gets the initial flight path angle.
double GetThetaDegIC(void) const
Gets the initial pitch angle.
double GetVgroundKtsIC(void) const
Gets the initial ground speed.
void SetUBodyFpsIC(double ubody)
Sets the initial body axis X velocity.
void SetWindMagKtsIC(double mag)
Sets the initial total wind speed.
double GetPhiDegIC(void) const
Gets the initial roll angle.
void InitializeIC(void)
Initialize the initial conditions to default values.
double GetAltitudeASLFtIC(void) const
Gets the initial altitude above sea level.
void SetClimbRateFpmIC(double roc)
Sets the climb rate initial condition in feet/minute.
double GetWindNFpsIC(void) const
Gets the initial wind velocity in local frame.
void SetFlightPathAngleRadIC(double gamma)
Sets the initial flight path angle.
void SetWBodyFpsIC(double wbody)
Sets the initial body axis Z velocity.
double GetTargetNlfIC(void) const
Gets the target normal load factor set from IC.
void SetMachIC(double mach)
Set mach initial condition.
void SetVtrueKtsIC(double vtrue)
Set true airspeed initial condition in knots.
void SetBetaRadIC(double beta)
Sets the initial sideslip angle.
double GetWindEFpsIC(void) const
Gets the initial wind velocity in local frame.
double GetTerrainElevationFtIC(void) const
Gets the initial terrain elevation.
void SetWindNEDFpsIC(double wN, double wE, double wD)
Sets the initial wind velocity.
void SetCrossWindKtsIC(double cross)
Sets the initial crosswind speed.
void SetVBodyFpsIC(double vbody)
Sets the initial body axis Y velocity.
void SetLongitudeRadIC(double lon)
Sets the initial longitude.
void SetAlphaDegIC(double a)
Sets angle of attack initial condition in degrees.
void SetPRadpsIC(double P)
Sets the initial body axis roll rate.
double GetVDownFpsIC(void) const
Gets the initial local frame Z (Down) velocity.
double GetVcalibratedKtsIC(void) const
Gets the initial calibrated airspeed.
double GetWindDirDegIC(void) const
Gets the initial wind direction.
void SetFlightPathAngleDegIC(double gamma)
Sets the flight path angle initial condition in degrees.
double GetWBodyFpsIC(void) const
Gets the initial body axis Z velocity.
double GetBetaDegIC(void) const
Gets the initial sideslip angle.
FGColumnVector3 GetWindNEDFpsIC(void) const
Gets the initial wind velocity in the NED local frame.
void SetVgroundKtsIC(double vg)
Set ground speed initial condition in knots.
void SetLongitudeDegIC(double lon)
Sets the initial longitude.
double GetClimbRateFpsIC(void) const
Gets the initial climb rate.
void SetGeodLatitudeRadIC(double glat)
Sets the initial geodetic latitude.
void SetAltitudeAGLFtIC(double agl)
Sets the initial Altitude above ground level.
FGInitialCondition(FGFDMExec *fdmex)
Constructor.
double GetWindMagFpsIC(void) const
Gets the initial total wind velocity in feet/sec.
double GetAltitudeAGLFtIC(void) const
Gets the initial altitude above ground level.
double GetMachIC(void) const
Gets the initial mach.
void SetThetaRadIC(double theta)
Sets the initial pitch angle.
double GetVtrueFpsIC(void) const
Gets the initial true velocity.
double GetGeodLatitudeDegIC(void) const
Gets the initial geodetic latitude.
void SetTerrainElevationFtIC(double elev)
Sets the initial terrain elevation.
double GetRRadpsIC() const
Gets the initial body axis yaw rate.
double GetAlphaRadIC(void) const
Gets the initial angle of attack.
double GetAlphaDegIC(void) const
Gets the initial angle of attack.
void SetClimbRateFpsIC(double roc)
Sets the initial climb rate.
void SetTargetNlfIC(double nlf)
Sets the target normal load factor.
void SetVtrueFpsIC(double vt)
Sets the initial true airspeed.
void SetPsiDegIC(double psi)
Sets the heading angle initial condition in degrees.
void SetAlphaRadIC(double alpha)
Sets the initial angle of attack.
double GetGeodLatitudeRadIC(void) const
Gets the initial geodetic latitude.
void SetHeadWindKtsIC(double head)
Sets the initial headwind velocity.
void SetRRadpsIC(double R)
Sets the initial body axis yaw rate.
void SetQRadpsIC(double Q)
Sets the initial body axis pitch rate.
double GetUBodyFpsIC(void) const
Gets the initial body axis X velocity.
double GetQRadpsIC() const
Gets the initial body axis pitch rate.
void SetVDownFpsIC(double vd)
Sets the initial local axis down velocity.
void SetLatitudeRadIC(double lat)
Sets the initial latitude.
void SetGeodLatitudeDegIC(double glat)
Sets the initial geodetic latitude.
void SetVcalibratedKtsIC(double vc)
Set calibrated airspeed initial condition in knots.
double GetVequivalentKtsIC(void) const
Gets the initial equivalent airspeed.
void SetPhiDegIC(double phi)
Sets the roll angle initial condition in degrees.
void SetBetaDegIC(double b)
Sets angle of sideslip initial condition in degrees.
double GetFlightPathAngleRadIC(void) const
Gets the initial flight path angle.
double GetPsiRadIC(void) const
Gets the initial heading angle.
bool Load(const SGPath &rstname, bool useAircraftPath=true)
Loads the initial conditions.
void SetWindDirDegIC(double dir)
Sets the initial wind direction.
double GetClimbRateFpmIC(void) const
Gets the initial climb rate.
double GetThetaRadIC(void) const
Gets the initial pitch angle.
void SetVNorthFpsIC(double vn)
Sets the initial local axis north velocity.
void SetPhiRadIC(double phi)
Sets the initial roll angle.
double GetWindUFpsIC(void) const
Gets the initial body axis X wind velocity.
void SetLatitudeDegIC(double lat)
Sets the initial latitude.
double GetWindVFpsIC(void) const
Gets the initial body axis Y wind velocity.
void SetAltitudeASLFtIC(double altitudeASL)
Sets the altitude above sea level initial condition in feet.
void SetWindMagFpsIC(double mag)
Sets the initial total wind speed.
double GetVtrueKtsIC(void) const
Gets the initial true velocity.
double GetPsiDegIC(void) const
Gets the initial heading angle.
double GetLatitudeRadIC(void) const
Gets the initial latitude.
double GetWindDFpsIC(void) const
Gets the initial wind velocity in local frame.
void SetVEastFpsIC(double ve)
Sets the initial local axis east velocity.
void SetVgroundFpsIC(double vg)
Sets the initial ground speed.
double GetWindWFpsIC(void) const
Gets the initial body axis Z wind velocity.
double GetBetaRadIC(void) const
Gets the initial angle of sideslip.
double GetLongitudeDegIC(void) const
Gets the initial longitude.
static char fgred[6]
red text
Definition FGJSBBase.h:166
static char reset[5]
resets text properties
Definition FGJSBBase.h:156
FGLocation holds an arbitrary location in the Earth centered Earth fixed reference frame (ECEF).
Definition FGLocation.h:152
void SetEllipse(double semimajor, double semiminor)
Sets the semimajor and semiminor axis lengths for this planet.
void SetLongitude(double longitude)
Set the longitude.
const FGMatrix33 & GetTl2ec(void) const
Transform matrix from local horizontal to earth centered frame.
Definition FGLocation.h:296
double GetLatitude() const
Get the GEOCENTRIC latitude in radians.
Definition FGLocation.h:252
const FGMatrix33 & GetTec2l(void) const
Transform matrix from the earth centered to local horizontal frame.
Definition FGLocation.h:301
double GetGeodLatitudeRad(void) const
Get the GEODETIC latitude in radians.
Definition FGLocation.h:258
double GetLongitude() const
Get the longitude.
Definition FGLocation.h:234
void SetLatitude(double latitude)
Set the GEOCENTRIC latitude.
double GetRadius() const
Get the distance from the center of the earth in feet.
Definition FGLocation.h:291
double GetSeaLevelRadius(void) const
Get the sea level radius in feet below the current location.
void SetRadius(double radius)
Set the distance from the center of the earth.
double GetGeodAltitude(void) const
Gets the geodetic altitude in feet.
Definition FGLocation.h:279
void SetPositionGeodetic(double lon, double lat, double height)
Sets the longitude, latitude and the distance above the reference spheroid.
Handles matrix math operations.
Definition FGMatrix33.h:70
FGMatrix33 Transposed(void) const
Transposed matrix.
Definition FGMatrix33.h:221
Models the Quaternion representation of rotations.
double GetSinEuler(int i) const
Retrieves sine of the given euler angle.
double GetCosEuler(int i) const
Retrieves cosine of the given euler angle.
const FGMatrix33 & GetTInv(void) const
Backward transformation matrix.
const FGColumnVector3 & GetEuler(void) const
Retrieves the Euler angles.
const FGMatrix33 & GetT(void) const
Transformation matrix.