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"
55#include "input_output/string_utilities.h"
71 cout <<
"FGInitialCondition: This class requires a pointer to a valid FGFDMExec object" << endl;
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,
93 double calpha = cos(alpha0), cbeta = cos(beta0);
94 double salpha = sin(alpha0), sbeta = sin(beta0);
98 vPQR_body = {p0, q0, r0};
99 alpha = alpha0; beta = beta0;
103 fdmex->
GetInertial()->SetAltitudeAGL(position, altAGLFt0);
104 lastLatitudeSet = setgeoc;
105 lastAltitudeSet = setagl;
112 lastSpeedSet = setuvw;
114 Tw2b = { calpha*cbeta, -calpha*sbeta, -salpha,
116 salpha*cbeta, -salpha*sbeta, calpha };
137 vUVW_NED.InitMatrix();
138 vPQR_body.InitMatrix();
143 Tw2b = { 1., 0., 0., 0., 1., 0., 0., 0., 1. };
144 Tb2w = { 1., 0., 0., 0., 1., 0., 0., 0., 1. };
146 lastSpeedSet = setvt;
147 lastAltitudeSet = setasl;
148 lastLatitudeSet = setgeoc;
150 trimRequested = TrimMode::tNone;
157 const auto Atmosphere = fdmex->GetAtmosphere();
159 double rho = Atmosphere->GetDensity(altitudeASL);
160 SetVtrueFpsIC(ve*ktstofps*sqrt(FGAtmosphere::StdDaySLdensity/rho));
161 lastSpeedSet = setve;
168 const auto Atmosphere = fdmex->GetAtmosphere();
170 double soundSpeed = Atmosphere->GetSoundSpeed(altitudeASL);
172 lastSpeedSet = setmach;
179 const auto Atmosphere = fdmex->GetAtmosphere();
181 double pressure = Atmosphere->GetPressure(altitudeASL);
182 double mach = Auxiliary->MachFromVcalibrated(fabs(vcas)*ktstofps, pressure);
183 double soundSpeed = Atmosphere->GetSoundSpeed(altitudeASL);
186 lastSpeedSet = setvc;
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;
205 calpha = cbeta = 1.0;
206 salpha = sbeta = 0.0;
209 alpha = atan2( wa, ua );
218 beta = atan2( va, uwa );
230 Tw2b = { calpha*cbeta, -calpha*sbeta, -salpha,
232 salpha*cbeta, -salpha*sbeta, calpha };
249 _vt_NED = vUVW_NED + _vWIND_NED;
252 calcAeroAngles(_vt_NED);
254 lastSpeedSet = setvg;
271 _vt_NED *= vtrue / vt;
276 vUVW_NED = _vt_NED - _vWIND_NED;
278 calcAeroAngles(_vt_NED);
280 lastSpeedSet = setvt;
290 if (fabs(hdot) > vt) {
291 cerr <<
"The climb rate cannot be higher than the true speed." << endl;
298 double hdot0 = -_vt_NED(eW);
300 if (fabs(hdot0) < vt) {
301 double scale = sqrt((vt*vt-hdot*hdot)/(vt*vt-hdot0*hdot0));
302 _vt_NED(eU) *= scale;
303 _vt_NED(eV) *= scale;
306 vUVW_NED = _vt_NED - _WIND_NED;
309 calcThetaBeta(alpha, _vt_NED);
321 calcThetaBeta(alfa, _vt_NED);
329void FGInitialCondition::calcThetaBeta(
double alfa,
const FGColumnVector3& _vt_NED)
332 double calpha = cos(alfa), salpha = sin(alfa);
343 -salpha, 0., calpha);
348 FGColumnVector3 u = y - DotProduct(y, n) * n;
349 FGColumnVector3 p = y * n;
351 if (DotProduct(p, v0) < 0) p *= -1.0;
354 u *= DotProduct(v0, y) / DotProduct(u, y);
364 if (DotProduct(v0, v0) < DotProduct(u, u)) {
365 cerr <<
"Cannot modify angle 'alpha' from " << alpha <<
" to " << alfa << endl;
369 FGColumnVector3 v1 = u + sqrt(DotProduct(v0, v0) - DotProduct(u, u))*p;
371 FGColumnVector3 v0xz(v0(eU), 0., v0(eW));
372 FGColumnVector3 v1xz(v1(eU), 0., v1(eW));
375 double sinTheta = (v1xz * v0xz)(eY);
376 vOrient(eTht) = asin(sinTheta);
378 orientation = FGQuaternion(vOrient);
380 const FGMatrix33& Tl2b = orientation.
GetT();
381 FGColumnVector3 v2 = Talpha * Tl2b * _vt_NED;
384 beta = atan2(v2(eV), v2(eU));
385 double cbeta=1.0, sbeta=0.0;
390 Tw2b = { calpha*cbeta, -calpha*sbeta, -salpha,
392 salpha*cbeta, -salpha*sbeta, calpha };
409 double calpha = cos(alpha), salpha = sin(alpha);
410 double cbeta = cos(beta), sbeta = sin(beta);
416 Tw2b = { calpha*cbeta, -calpha*sbeta, -salpha,
418 salpha*cbeta, -salpha*sbeta, calpha };
423 FGColumnVector3 v1xy(sqrt(v0xy(eX)*v0xy(eX)+v0xy(eY)*v0xy(eY)-vf(eY)*vf(eY)),vf(eY),0.);
427 if (vf(eX) < 0.) v0xy(eX) *= -1.0;
429 double sinPsi = (v1xy * v0xy)(eZ);
430 double cosPsi = DotProduct(v0xy, v1xy);
431 vOrient(ePsi) = atan2(sinPsi, cosPsi);
438 v2xz(eV) = vfxz(eV) = 0.0;
441 double sinTheta = (v2xz * vfxz)(eY);
442 vOrient(eTht) = -asin(sinTheta);
450void FGInitialCondition::SetEulerAngleRadIC(
int idx,
double angle)
459 vOrient(idx) = angle;
462 if ((lastSpeedSet != setned) && (lastSpeedSet != setvg)) {
464 vUVW_NED = newTb2l * _vUVW_BODY;
465 _vt_NED = vUVW_NED + _vWIND_NED;
469 calcAeroAngles(_vt_NED);
477void FGInitialCondition::SetBodyVelFpsIC(
int idx,
double vel)
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;
485 _vUVW_BODY(idx) = vel;
486 vUVW_NED = Tb2l * _vUVW_BODY;
487 _vt_NED = vUVW_NED + _vWIND_NED;
488 vt = _vt_NED.Magnitude();
490 calcAeroAngles(_vt_NED);
492 lastSpeedSet = setuvw;
500void FGInitialCondition::SetNEDVelFpsIC(
int idx,
double vel)
502 const FGMatrix33& Tb2l = orientation.
GetTInv();
503 FGColumnVector3 _vt_NED = Tb2l * Tw2b * FGColumnVector3(vt, 0., 0.);
504 FGColumnVector3 _vWIND_NED = _vt_NED - vUVW_NED;
507 _vt_NED = vUVW_NED + _vWIND_NED;
508 vt = _vt_NED.Magnitude();
510 calcAeroAngles(_vt_NED);
512 lastSpeedSet = setned;
524 calcAeroAngles(_vt_NED);
540 _vWIND_NED -= DotProduct(_vWIND_NED, _vCROSS) * _vCROSS;
543 _vWIND_NED += (cross * ktstofps) * _vCROSS;
544 _vt_NED = vUVW_NED + _vWIND_NED;
547 calcAeroAngles(_vt_NED);
567 _vWIND_NED -= DotProduct(_vWIND_NED, _vHEAD) * _vHEAD;
570 _vWIND_NED += (head * ktstofps) * _vHEAD;
571 _vt_NED = vUVW_NED + _vWIND_NED;
575 calcAeroAngles(_vt_NED);
588 _vt_NED(eW) = vUVW_NED(eW) + wD * ktstofps;
591 calcAeroAngles(_vt_NED);
608 _vHEAD *= (mag*ktstofps) / windMag;
610 _vHEAD = {mag*ktstofps, 0., 0.};
612 _vWIND_NED(eU) = _vHEAD(eU);
613 _vWIND_NED(eV) = _vHEAD(eV);
614 _vt_NED = vUVW_NED + _vWIND_NED;
617 calcAeroAngles(_vt_NED);
630 double mag = _vWIND_NED.
Magnitude(eU, eV);
631 FGColumnVector3 _vHEAD(mag*cos(dir*degtorad), mag*sin(dir*degtorad), 0.);
633 _vWIND_NED(eU) = _vHEAD(eU);
634 _vWIND_NED(eV) = _vHEAD(eV);
635 _vt_NED = vUVW_NED + _vWIND_NED;
638 calcAeroAngles(_vt_NED);
648 if (lastAltitudeSet == setagl)
663 return fdmex->
GetInertial()->GetAltitudeAGL(position);
675 fdmex->
GetInertial()->GetContactPoint(position, contact, normal, v, w);
683 const auto Atmosphere = fdmex->GetAtmosphere();
685 double pressure = Atmosphere->GetPressure(altitudeASL);
686 double soundSpeed = Atmosphere->GetSoundSpeed(altitudeASL);
687 double rho = Atmosphere->GetDensity(altitudeASL);
689 double mach0 = vt / soundSpeed;
690 double vc0 = Auxiliary->VcalibratedFromMach(mach0, pressure);
691 double ve0 = vt * sqrt(rho/FGAtmosphere::StdDaySLdensity);
693 switch(lastLatitudeSet) {
695 fdmex->
GetInertial()->SetAltitudeAGL(position, agl);
701 double e2 = 1.0-b*b/(a*a);
708 double h = -2.0*max(a,b);
710 while ((fabs(n-prev_n) > 1E-15 || fabs(h-agl) > 1E-10) && iter < 10) {
711 geodLat = atan(tanlat/(1-n));
715 double sinGeodLat = sin(geodLat);
716 double N = a/sqrt(1-e2*sinGeodLat*sinGeodLat);
726 soundSpeed = Atmosphere->GetSoundSpeed(altitudeASL);
727 rho = Atmosphere->GetDensity(altitudeASL);
728 pressure = Atmosphere->GetPressure(altitudeASL);
730 switch(lastSpeedSet) {
732 mach0 = Auxiliary->MachFromVcalibrated(vc0, pressure);
739 SetVtrueFpsIC(ve0 * sqrt(FGAtmosphere::StdDaySLdensity/rho));
745 lastAltitudeSet = setagl;
755 const auto Atmosphere = fdmex->GetAtmosphere();
757 double pressure = Atmosphere->GetPressure(altitudeASL);
758 double soundSpeed = Atmosphere->GetSoundSpeed(altitudeASL);
759 double rho = Atmosphere->GetDensity(altitudeASL);
761 double mach0 = vt / soundSpeed;
762 double vc0 = Auxiliary->VcalibratedFromMach(mach0, pressure);
763 double ve0 = vt * sqrt(rho/FGAtmosphere::StdDaySLdensity);
765 switch(lastLatitudeSet) {
772 double e2 = 1.0-b*b/(a*a);
774 double cosGeodLat = cos(geodLatitude);
775 double sinGeodLat = sin(geodLatitude);
776 double N = a/sqrt(1-e2*sinGeodLat*sinGeodLat);
777 double geodAlt = 0.0;
783 if (cosGeodLat > fabs(sinGeodLat)) {
784 double tanGeodLat = sinGeodLat/cosGeodLat;
785 double x0 = N*e2*cosGeodLat;
787 while (fabs(n-prev_n) > 1E-15 && iter < 10) {
788 double tanLat = (1-n)*tanGeodLat;
789 double cos2Lat = 1./(1.+tanLat*tanLat);
790 double slr = b/sqrt(1.-e2*cos2Lat);
791 double R = slr + alt;
797 geodAlt = x/cosGeodLat-N;
800 double cotanGeodLat = cosGeodLat/sinGeodLat;
801 double z0 = N*e2*sinGeodLat;
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);
814 geodAlt = z/sinGeodLat-N*(1-e2);
830 soundSpeed = Atmosphere->GetSoundSpeed(altitudeASL);
831 rho = Atmosphere->GetDensity(altitudeASL);
832 pressure = Atmosphere->GetPressure(altitudeASL);
834 switch(lastSpeedSet) {
836 mach0 = Auxiliary->MachFromVcalibrated(vc0, pressure);
843 SetVtrueFpsIC(ve0 * sqrt(FGAtmosphere::StdDaySLdensity/rho));
849 lastAltitudeSet = setasl;
857 lastLatitudeSet = setgeod;
859 switch (lastAltitudeSet)
865 fdmex->
GetInertial()->SetAltitudeAGL(position, agl);
884 lastLatitudeSet = setgeoc;
886 switch(lastAltitudeSet) {
906 switch(lastAltitudeSet) {
926 return _vWIND_NED.
Magnitude(eU, eV) == 0.0 ? 0.0
927 : atan2(_vWIND_NED(eV), _vWIND_NED(eU))*radtodeg;
935 return _vt_NED - vUVW_NED;
951double FGInitialCondition::GetBodyWindFpsIC(
int idx)
const
958 return _vWIND_BODY(idx);
965 const auto Atmosphere = fdmex->GetAtmosphere();
967 double pressure = Atmosphere->GetPressure(altitudeASL);
968 double soundSpeed = Atmosphere->GetSoundSpeed(altitudeASL);
969 double mach = vt / soundSpeed;
971 return fpstokts * Auxiliary->VcalibratedFromMach(mach, pressure);
978 const auto Atmosphere = fdmex->GetAtmosphere();
980 double rho = Atmosphere->GetDensity(altitudeASL);
981 return fpstokts * vt * sqrt(rho/FGAtmosphere::StdDaySLdensity);
988 const auto Atmosphere = fdmex->GetAtmosphere();
990 double soundSpeed = Atmosphere->GetSoundSpeed(altitudeASL);
991 return vt / soundSpeed;
996double FGInitialCondition::GetBodyVelFpsIC(
int idx)
const
1001 return _vUVW_BODY(idx);
1008 SGPath init_file_name;
1009 if(useAircraftPath && rstfile.isRelative()) {
1012 init_file_name = rstfile;
1016 Element* document = XMLFileRead.LoadXMLDocument(init_file_name);
1021 s <<
"File: " << init_file_name <<
" could not be read.";
1022 cerr << s.str() << endl;
1026 if (document->
GetName() !=
"initialize") {
1028 s <<
"File: " << init_file_name <<
" is not a reset file.";
1029 cerr << s.str() << endl;
1033 bool result =
false;
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;
1043 }
else if (version >= 2.0) {
1044 result = Load_v2(document);
1045 }
else if (version >= 1.0) {
1046 result = Load_v1(document);
1050 result = Load_v1(document);
1055 while (running_elements) {
1057 enginesRunning |= engineNumber == -1 ? engineNumber : 1 << engineNumber;
1066bool FGInitialCondition::LoadLatitude(
Element* position_el)
1073 if (fabs(latitude) > 0.5*M_PI) {
1075 if (unit_type.empty()) unit_type=
"RAD";
1077 cerr << latitude_el->
ReadFrom() <<
"The latitude value "
1079 <<
" is outside the range [";
1080 if (unit_type ==
"DEG")
1081 cerr <<
"-90 DEG ; +90 DEG]" << endl;
1083 cerr <<
"-PI/2 RAD; +PI/2 RAD]" << endl;
1090 if (lat_type ==
"geod" || lat_type ==
"geodetic") {
1092 lastLatitudeSet = setgeod;
1096 lastLatitudeSet = setgeoc;
1105void FGInitialCondition::SetTrimRequest(std::string trim)
1107 std::string& trimOption = to_lower(trim);
1108 if (trimOption ==
"1")
1109 trimRequested = TrimMode::tGround;
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;
1126bool FGInitialCondition::Load_v1(Element* document)
1130 if (document->FindElement(
"longitude"))
1131 SetLongitudeRadIC(document->FindElementValueAsNumberConvertTo(
"longitude",
"RAD"));
1132 if (document->FindElement(
"elevation"))
1135 if (document->FindElement(
"altitude"))
1137 else if (document->FindElement(
"altitudeAGL"))
1138 SetAltitudeAGLFtIC(document->FindElementValueAsNumberConvertTo(
"altitudeAGL",
"FT"));
1139 else if (document->FindElement(
"altitudeMSL"))
1140 SetAltitudeASLFtIC(document->FindElementValueAsNumberConvertTo(
"altitudeMSL",
"FT"));
1142 result = LoadLatitude(document);
1144 FGColumnVector3 vOrient = orientation.
GetEuler();
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");
1153 orientation = FGQuaternion(vOrient);
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"))
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"))
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"))
1191 if (document->FindElement(
"targetNlf"))
1193 if (document->FindElement(
"trim"))
1194 SetTrimRequest(document->FindElementValue(
"trim"));
1196 vPQR_body.InitMatrix();
1203bool FGInitialCondition::Load_v2(Element* document)
1205 FGColumnVector3 vOrient;
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");
1215 FGMatrix33 Ti2ec(cos(epa), sin(epa), 0.0,
1216 -sin(epa), cos(epa), 0.0,
1218 FGMatrix33 Tec2i = Ti2ec.Transposed();
1220 if (document->FindElement(
"planet_rotation_rate")) {
1221 fdmex->
GetInertial()->SetOmegaPlanet(document->FindElementValueAsNumberConvertTo(
"planet_rotation_rate",
"RAD"));
1225 FGColumnVector3 vOmegaEarth = fdmex->
GetInertial()->GetOmegaPlanet();
1227 if (document->FindElement(
"elevation")) {
1228 fdmex->
GetInertial()->SetTerrainElevation(document->FindElementValueAsNumberConvertTo(
"elevation",
"FT"));
1237 Element* position_el = document->
FindElement(
"position");
1240 frame = to_lower(frame);
1241 if (frame ==
"eci") {
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"));
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"));
1255 cerr << endl <<
" No altitude or radius initial condition is given." << endl;
1260 result = LoadLatitude(position_el);
1263 position = position_el->FindElementTripletConvertTo(
"FT");
1266 cerr << endl <<
" Neither ECI nor ECEF frame is specified for initial position." << endl;
1270 cerr << endl <<
" Initial position not specified in this initialization file." << endl;
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") {
1316 FGQuaternion QuatI2Body = FGQuaternion(vOrient);
1317 QuatI2Body.Normalize();
1318 FGQuaternion QuatLocal2I = Tec2i * position.
GetTl2ec();
1319 QuatLocal2I.Normalize();
1320 orientation = QuatLocal2I * QuatI2Body;
1322 }
else if (frame ==
"ecef") {
1334 FGQuaternion QuatEC2Body(vOrient);
1335 QuatEC2Body.Normalize();
1336 FGQuaternion QuatLocal2EC = position.
GetTl2ec();
1337 QuatLocal2EC.Normalize();
1338 orientation = QuatLocal2EC * QuatEC2Body;
1340 }
else if (frame ==
"local") {
1342 orientation = FGQuaternion(vOrient);
1346 cerr << endl <<
fgred <<
" Orientation frame type: \"" << frame
1347 <<
"\" is not supported!" <<
reset << endl << endl;
1361 Element* velocity_el = document->FindElement(
"velocity");
1362 FGMatrix33 mTec2l = position.
GetTec2l();
1363 const FGMatrix33& Tb2l = orientation.
GetTInv();
1367 string frame = velocity_el->GetAttributeValue(
"frame");
1368 frame = to_lower(frame);
1369 FGColumnVector3 vInitVelocity = velocity_el->FindElementTripletConvertTo(
"FT/SEC");
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;
1386 cerr << endl <<
fgred <<
" Velocity frame type: \"" << frame
1387 <<
"\" is not supported!" <<
reset << endl << endl;
1394 vUVW_NED.InitMatrix();
1398 vt = vUVW_NED.Magnitude();
1400 calcAeroAngles(vUVW_NED);
1408 Element* attrate_el = document->FindElement(
"attitude_rate");
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");
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") {
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()) {
1434 cerr << endl <<
fgred <<
" Attitude rate frame type: \"" << frame
1435 <<
"\" is not supported!" <<
reset << endl << endl;
1438 }
else if (frame.empty()) {
1439 vPQR_body.InitMatrix();
1443 vPQR_body.InitMatrix();
1451void FGInitialCondition::bind(FGPropertyManager* PropertyManager)
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,
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,
1592 PropertyManager->Tie(
"ic/targetNlf",
this,
1616void FGInitialCondition::Debug(
int from)
1618 if (debug_lvl <= 0)
return;
1620 if (debug_lvl & 1) {
1622 if (debug_lvl & 2 ) {
1623 if (from == 0) cout <<
"Instantiated: FGInitialCondition" << endl;
1624 if (from == 1) cout <<
"Destroyed: FGInitialCondition" << endl;
1626 if (debug_lvl & 4 ) {
1628 if (debug_lvl & 8 ) {
1630 if (debug_lvl & 16) {
1632 if (debug_lvl & 64) {
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.
std::shared_ptr< FGAircraft > GetAircraft(void) const
Returns the FGAircraft pointer.
const SGPath & GetFullAircraftPath(void)
Retrieves the full aircraft path name.
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.
~FGInitialCondition()
Destructor.
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
static char reset[5]
resets text properties
FGLocation holds an arbitrary location in the Earth centered Earth fixed reference frame (ECEF).
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.
double GetLatitude() const
Get the GEOCENTRIC latitude in radians.
const FGMatrix33 & GetTec2l(void) const
Transform matrix from the earth centered to local horizontal frame.
double GetGeodLatitudeRad(void) const
Get the GEODETIC latitude in radians.
double GetLongitude() const
Get the longitude.
void SetLatitude(double latitude)
Set the GEOCENTRIC latitude.
double GetRadius() const
Get the distance from the center of the earth in feet.
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.
void SetPositionGeodetic(double lon, double lat, double height)
Sets the longitude, latitude and the distance above the reference spheroid.
Handles matrix math operations.
FGMatrix33 Transposed(void) const
Transposed matrix.
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.