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
FGLGear.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: FGLGear.cpp
4 Author: Jon S. Berndt
5 Norman H. Princen
6 Bertrand Coconnier
7 Date started: 11/18/99
8 Purpose: Encapsulates the landing gear elements
9 Called by: FGAircraft
10
11 ------------- Copyright (C) 1999 Jon S. Berndt (jon@jsbsim.org) -------------
12
13 This program is free software; you can redistribute it and/or modify it under
14 the terms of the GNU Lesser General Public License as published by the Free
15 Software Foundation; either version 2 of the License, or (at your option) any
16 later version.
17
18 This program is distributed in the hope that it will be useful, but WITHOUT
19 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
20 FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
21 details.
22
23 You should have received a copy of the GNU Lesser General Public License along
24 with this program; if not, write to the Free Software Foundation, Inc., 59
25 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
26
27 Further information about the GNU Lesser General Public License can also be
28 found on the world wide web at http://www.gnu.org.
29
30FUNCTIONAL DESCRIPTION
31--------------------------------------------------------------------------------
32
33HISTORY
34--------------------------------------------------------------------------------
3511/18/99 JSB Created
3601/30/01 NHP Extended gear model to properly simulate steering and braking
3707/08/09 BC Modified gear model to support large angles between aircraft
38 and ground
39
40/%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
41INCLUDES
42%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
43
44#include "FGLGear.h"
45#include "FGFDMExec.h"
46#include "models/FGGroundReactions.h"
47#include "math/FGTable.h"
48#include "input_output/FGXMLElement.h"
49#include "models/FGInertial.h"
50
51using namespace std;
52
53namespace JSBSim {
54
55/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
56DEFINITIONS
57%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
58
59/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
60GLOBAL DATA
61%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
62
63// Body To Structural (body frame is rotated 180 deg about Y and lengths are
64// given in ft instead of inches)
65const FGMatrix33 FGLGear::Tb2s(-1./inchtoft, 0., 0., 0., 1./inchtoft, 0., 0., 0., -1./inchtoft);
66const FGMatrix33 FGLGear::Ts2b(-inchtoft, 0., 0., 0., inchtoft, 0., 0., 0., -inchtoft);
67
68/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
69CLASS IMPLEMENTATION
70%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
71
72FGLGear::FGLGear(Element* el, FGFDMExec* fdmex, int number, const struct Inputs& inputs) :
73 FGForce(fdmex),
74 in(inputs),
75 GearNumber(number),
76 SteerAngle(0.0),
77 Castered(false),
78 StaticFriction(false),
79 eSteerType(stSteer)
80{
81 kSpring = bDamp = bDampRebound = dynamicFCoeff = staticFCoeff = rollingFCoeff = maxSteerAngle = 0;
82 isRetractable = false;
83 eDampType = dtLinear;
84 eDampTypeRebound = dtLinear;
85
86 name = el->GetAttributeValue("name");
87 string sContactType = el->GetAttributeValue("type");
88 if (sContactType == "BOGEY") {
89 eContactType = ctBOGEY;
90 } else if (sContactType == "STRUCTURE") {
91 eContactType = ctSTRUCTURE;
92 } else {
93 // Unknown contact point types will be treated as STRUCTURE.
94 eContactType = ctSTRUCTURE;
95 }
96
97 // Default values for structural contact points
98 if (eContactType == ctSTRUCTURE) {
99 kSpring = in.EmptyWeight;
100 bDamp = kSpring;
101 bDampRebound = kSpring * 10;
102 staticFCoeff = 1.0;
103 dynamicFCoeff = 1.0;
104 }
105
106 auto PropertyManager = fdmex->GetPropertyManager();
107
108 fStrutForce = 0;
109 Element* strutForce = el->FindElement("strut_force");
110 if (strutForce) {
111 Element* springFunc = strutForce->FindElement("function");
112 fStrutForce = new FGFunction(fdmex, springFunc);
113 }
114 else {
115 if (el->FindElement("spring_coeff"))
116 kSpring = el->FindElementValueAsNumberConvertTo("spring_coeff", "LBS/FT");
117 if (el->FindElement("damping_coeff")) {
118 Element* dampCoeff = el->FindElement("damping_coeff");
119 if (dampCoeff->GetAttributeValue("type") == "SQUARE") {
120 eDampType = dtSquare;
121 bDamp = el->FindElementValueAsNumberConvertTo("damping_coeff", "LBS/FT2/SEC2");
122 } else {
123 bDamp = el->FindElementValueAsNumberConvertTo("damping_coeff", "LBS/FT/SEC");
124 }
125 }
126
127 if (el->FindElement("damping_coeff_rebound")) {
128 Element* dampCoeffRebound = el->FindElement("damping_coeff_rebound");
129 if (dampCoeffRebound->GetAttributeValue("type") == "SQUARE") {
130 eDampTypeRebound = dtSquare;
131 bDampRebound = el->FindElementValueAsNumberConvertTo("damping_coeff_rebound", "LBS/FT2/SEC2");
132 } else {
133 bDampRebound = el->FindElementValueAsNumberConvertTo("damping_coeff_rebound", "LBS/FT/SEC");
134 }
135 } else {
136 bDampRebound = bDamp;
137 eDampTypeRebound = eDampType;
138 }
139 }
140
141 if (el->FindElement("dynamic_friction"))
142 dynamicFCoeff = el->FindElementValueAsNumber("dynamic_friction");
143 if (el->FindElement("static_friction"))
144 staticFCoeff = el->FindElementValueAsNumber("static_friction");
145 if (el->FindElement("rolling_friction"))
146 rollingFCoeff = el->FindElementValueAsNumber("rolling_friction");
147 if (el->FindElement("retractable"))
148 isRetractable = ((unsigned int)el->FindElementValueAsNumber("retractable"))>0.0?true:false;
149
150 if (el->FindElement("max_steer"))
151 maxSteerAngle = el->FindElementValueAsNumberConvertTo("max_steer", "DEG");
152
153 Element* castered_el = el->FindElement("castered");
154
155 if ((maxSteerAngle == 360 && !castered_el)
156 || (castered_el && castered_el->GetDataAsNumber() != 0.0)) {
157 eSteerType = stCaster;
158 Castered = true;
159 }
160 else if (maxSteerAngle == 0.0) {
161 eSteerType = stFixed;
162 }
163 else
164 eSteerType = stSteer;
165
166 GroundReactions = fdmex->GetGroundReactions().get();
167
168 ForceY_Table = 0;
169 Element* force_table = el->FindElement("table");
170 while (force_table) {
171 string force_type = force_table->GetAttributeValue("name");
172 if (force_type == "CORNERING_COEFF") {
173 ForceY_Table = new FGTable(PropertyManager, force_table);
174 break;
175 } else {
176 cerr << "Undefined force table for " << name << " contact point" << endl;
177 }
178 force_table = el->FindNextElement("table");
179 }
180
181 Element* element = el->FindElement("location");
182 if (element) vXYZn = element->FindElementTripletConvertTo("IN");
183 else {
184 stringstream s;
185 s << "No location given for contact " << name;
186 cerr << endl << s.str() << endl;
187 throw BaseException(s.str());
188 }
189 SetTransformType(FGForce::tCustom);
190
191 element = el->FindElement("orientation");
192 if (element && (eContactType == ctBOGEY)) {
193 FGQuaternion quatFromEuler(element->FindElementTripletConvertTo("RAD"));
194
195 mTGear = quatFromEuler.GetT();
196 }
197 else {
198 mTGear(1,1) = 1.;
199 mTGear(2,2) = 1.;
200 mTGear(3,3) = 1.;
201 }
202
203 string sBrakeGroup = el->FindElementValue("brake_group");
204
205 if (sBrakeGroup == "LEFT" ) eBrakeGrp = bgLeft;
206 else if (sBrakeGroup == "RIGHT" ) eBrakeGrp = bgRight;
207 else if (sBrakeGroup == "CENTER") eBrakeGrp = bgCenter;
208 else if (sBrakeGroup == "NOSE" ) eBrakeGrp = bgCenter; // Nose brake is not supported by FGFCS
209 else if (sBrakeGroup == "TAIL" ) eBrakeGrp = bgCenter; // Tail brake is not supported by FGFCS
210 else if (sBrakeGroup == "NONE" ) eBrakeGrp = bgNone;
211 else if (sBrakeGroup.empty() ) eBrakeGrp = bgNone;
212 else {
213 cerr << "Improper braking group specification in config file: "
214 << sBrakeGroup << " is undefined." << endl;
215 }
216
217// Add some AI here to determine if gear is located properly according to its
218// brake group type ??
219
220 useFCSGearPos = false;
221 ReportEnable = true;
222 TakeoffReported = LandingReported = false;
223
224 // Set Pacejka terms
225
226 Stiffness = 0.06;
227 Shape = 2.8;
228 Peak = staticFCoeff;
229 Curvature = 1.03;
230
231 ResetToIC();
232
233 bind(PropertyManager.get());
234
235 Debug(0);
236}
237
238//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
239
241{
242 delete ForceY_Table;
243 delete fStrutForce;
244 Debug(1);
245}
246
247//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
248
249void FGLGear::ResetToIC(void)
250{
251 GearPos = 1.0;
252
253 WOW = lastWOW = false;
254 FirstContact = false;
255 StartedGroundRun = false;
256 LandingDistanceTraveled = TakeoffDistanceTraveled = TakeoffDistanceTraveled50ft = 0.0;
257 MaximumStrutForce = MaximumStrutTravel = 0.0;
258 SinkRate = GroundSpeed = 0.0;
259 SteerAngle = 0.0;
260
261 vWhlVelVec.InitMatrix();
262
263 compressLength = 0.0;
264 compressSpeed = 0.0;
265 maxCompLen = 0.0;
266
267 WheelSlip = 0.0;
268
269 // Initialize Lagrange multipliers
270 for (int i=0; i < 3; i++) {
271 LMultiplier[i].ForceJacobian.InitMatrix();
272 LMultiplier[i].LeverArm.InitMatrix();
273 LMultiplier[i].Min = 0.0;
274 LMultiplier[i].Max = 0.0;
275 LMultiplier[i].value = 0.0;
276 }
277}
278
279//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
280
282{
283 double gearPos = 1.0;
284
285 vFn.InitMatrix();
286
287 // Compute AGL
288 FGColumnVector3 normal, terrainVel, dummy;
289 FGLocation gearLoc, contact;
290 FGColumnVector3 vWhlBodyVec = Ts2b * (vXYZn - in.vXYZcg);
291
292 vLocalGear = in.Tb2l * vWhlBodyVec; // Get local frame wheel location
293 gearLoc = in.Location.LocalToLocation(vLocalGear);
294
295 // Compute the height of the theoretical location of the wheel (if strut is
296 // not compressed) with respect to the ground level (AGL)
297 double height = fdmex->GetInertial()->GetContactPoint(gearLoc, contact,
298 normal, terrainVel, dummy);
299
300 // Don't want strut compression when in contact with the ground to return
301 // a negative AGL
302 AGL = max(height, 0.0);
303
304 if (isRetractable) gearPos = GetGearUnitPos();
305
306 if (gearPos > 0.99) { // Gear DOWN
307
308 if (!fdmex->GetTrimStatus())
309 height -= GroundReactions->GetBumpHeight();
310 staticFFactor = GroundReactions->GetStaticFFactor();
311 rollingFFactor = GroundReactions->GetRollingFFactor();
312 maximumForce = GroundReactions->GetMaximumForce();
313 bumpiness = GroundReactions->GetBumpiness();
314 isSolid = GroundReactions->GetSolid();
315
316 FGColumnVector3 vWhlDisplVec;
317 double LGearProj = 1.0;
318
319 if (height < 0.0) {
320 WOW = true;
321 vGroundNormal = in.Tec2b * normal;
322
323 // The height returned by GetGroundCallback() is the AGL and is expressed
324 // in the Z direction of the local coordinate frame. We now need to
325 // transform this height in actual compression of the strut (BOGEY) or in
326 // the normal direction to the ground (STRUCTURE)
327 double normalZ = (in.Tec2l*normal)(eZ);
328 LGearProj = -(mTGear.Transposed() * vGroundNormal)(eZ);
329
330 // The following equations use the vector to the tire contact patch
331 // including the strut compression.
332 switch(eContactType) {
333 case ctBOGEY:
334 if (isSolid) {
335 compressLength = LGearProj > 0.0 ? height * normalZ / LGearProj : 0.0;
336 vWhlDisplVec = mTGear * FGColumnVector3(0., 0., -compressLength);
337 } else {
338 // Gears don't (or hardly) compress in liquids
339 WOW = false;
340 }
341 break;
342 case ctSTRUCTURE:
343 compressLength = height * normalZ / DotProduct(normal, normal);
344 vWhlDisplVec = compressLength * vGroundNormal;
345 break;
346 }
347 }
348 else
349 WOW = false;
350
351 if (WOW) {
352 FGColumnVector3 vWhlContactVec = vWhlBodyVec + vWhlDisplVec;
353 vActingXYZn = vXYZn + Tb2s * vWhlDisplVec;
354 FGColumnVector3 vBodyWhlVel = in.PQR * vWhlContactVec;
355 vBodyWhlVel += in.UVW - in.Tec2b * terrainVel;
356 vWhlVelVec = mTGear.Transposed() * vBodyWhlVel;
357
358 InitializeReporting();
359 ComputeSteeringAngle();
360 ComputeGroundFrame();
361
362 vGroundWhlVel = mT.Transposed() * vBodyWhlVel;
363
364 if (fdmex->GetTrimStatus() || in.TotalDeltaT == 0.0)
365 compressSpeed = 0.0; // Steady state is sought during trimming
366 else {
367 compressSpeed = -vGroundWhlVel(eZ);
368 if (eContactType == ctBOGEY)
369 compressSpeed /= LGearProj;
370
371 // If the gear is entering in contact with the ground during the current
372 // time step, the compression speed might actually be lower than the
373 // aircraft velocity projected along the gear leg (compressSpeed).
374 double maxCompressSpeed = compressLength/in.TotalDeltaT;
375 if (fabs(compressSpeed) > maxCompressSpeed)
376 compressSpeed = sign(compressSpeed)*maxCompressSpeed;
377 }
378
379 ComputeVerticalStrutForce();
380
381 // Compute the friction coefficients in the wheel ground plane.
382 if (eContactType == ctBOGEY) {
383 ComputeSlipAngle();
384 ComputeBrakeForceCoefficient();
385 ComputeSideForceCoefficient();
386 }
387
388 // Prepare the Jacobians and the Lagrange multipliers for later friction
389 // forces calculations.
390 ComputeJacobian(vWhlContactVec);
391 } else { // Gear is NOT compressed
392 compressLength = 0.0;
393 compressSpeed = 0.0;
394 WheelSlip = 0.0;
395 StrutForce = 0.0;
396 vWhlDisplVec.InitMatrix();
397
398 LMultiplier[ftRoll].value = 0.0;
399 LMultiplier[ftSide].value = 0.0;
400 LMultiplier[ftDynamic].value = 0.0;
401
402 // Return to neutral position between 1.0 and 0.8 gear pos.
403 SteerAngle *= max(gearPos-0.8, 0.0)/0.2;
404
405 ResetReporting();
406 }
407 }
408
409 if (!WOW) {
410 // Let wheel spin down slowly
411 vWhlVelVec(eX) -= 13.0 * in.TotalDeltaT;
412 if (vWhlVelVec(eX) < 0.0) vWhlVelVec(eX) = 0.0;
413 }
414
415 if (!fdmex->GetTrimStatus()) {
416 ReportTakeoffOrLanding();
417
418 // Require both WOW and LastWOW to be true before checking crash conditions
419 // to allow the WOW flag to be used in terminating a scripted run.
420 if (WOW && lastWOW) CrashDetect();
421
422 lastWOW = WOW;
423 }
424
425 return FGForce::GetBodyForces();
426}
427
428//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
429// Build a local "ground" coordinate system defined by
430// eX : projection of the rolling direction on the ground
431// eY : projection of the sliping direction on the ground
432// eZ : normal to the ground
433
434void FGLGear::ComputeGroundFrame(void)
435{
436 FGColumnVector3 roll = mTGear * FGColumnVector3(cos(SteerAngle), sin(SteerAngle), 0.);
437 FGColumnVector3 side = vGroundNormal * roll;
438
439 roll -= DotProduct(roll, vGroundNormal) * vGroundNormal;
440 roll.Normalize();
441 side.Normalize();
442
443 mT(eX,eX) = roll(eX);
444 mT(eY,eX) = roll(eY);
445 mT(eZ,eX) = roll(eZ);
446 mT(eX,eY) = side(eX);
447 mT(eY,eY) = side(eY);
448 mT(eZ,eY) = side(eZ);
449 mT(eX,eZ) = vGroundNormal(eX);
450 mT(eY,eZ) = vGroundNormal(eY);
451 mT(eZ,eZ) = vGroundNormal(eZ);
452}
453
454//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
455// Calculate tire slip angle.
456
457void FGLGear::ComputeSlipAngle(void)
458{
459// Check that the speed is non-null otherwise keep the current angle
460 if (vGroundWhlVel.Magnitude(eX,eY) > 1E-3)
461 WheelSlip = -atan2(vGroundWhlVel(eY), fabs(vGroundWhlVel(eX)))*radtodeg;
462}
463
464//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
465// Compute the steering angle in any case.
466// This will also make sure that animations will look right.
467
468void FGLGear::ComputeSteeringAngle(void)
469{
470 if (Castered) {
471 // Check that the speed is non-null otherwise keep the current angle
472 if (vWhlVelVec.Magnitude(eX,eY) > 0.1)
473 SteerAngle = atan2(vWhlVelVec(eY), fabs(vWhlVelVec(eX)));
474 }
475}
476
477//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
478// Reset reporting functionality after takeoff
479
480void FGLGear::ResetReporting(void)
481{
482 if (in.DistanceAGL > 200.0) {
483 FirstContact = false;
484 StartedGroundRun = false;
485 LandingReported = false;
486 TakeoffReported = true;
487 LandingDistanceTraveled = 0.0;
488 MaximumStrutForce = MaximumStrutTravel = 0.0;
489 }
490}
491
492//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
493
494void FGLGear::InitializeReporting(void)
495{
496 // If this is the first time the wheel has made contact, remember some values
497 // for later printout.
498
499 if (!FirstContact) {
500 FirstContact = true;
501 SinkRate = compressSpeed;
502 GroundSpeed = in.Vground;
503 TakeoffReported = false;
504 }
505
506 // If the takeoff run is starting, initialize.
507
508 if ((in.Vground > 0.1) &&
509 (in.BrakePos[bgLeft] == 0) &&
510 (in.BrakePos[bgRight] == 0) &&
511 (in.TakeoffThrottle && !StartedGroundRun))
512 {
513 TakeoffDistanceTraveled = 0;
514 TakeoffDistanceTraveled50ft = 0;
515 StartedGroundRun = true;
516 }
517}
518
519//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
520// Takeoff and landing reporting functionality
521
522void FGLGear::ReportTakeoffOrLanding(void)
523{
524 if (FirstContact)
525 LandingDistanceTraveled += in.Vground * in.TotalDeltaT;
526
527 if (StartedGroundRun) {
528 TakeoffDistanceTraveled50ft += in.Vground * in.TotalDeltaT;
529 if (WOW) TakeoffDistanceTraveled += in.Vground * in.TotalDeltaT;
530 }
531
532 if ( ReportEnable
533 && in.Vground <= 0.05
534 && !LandingReported
535 && in.WOW)
536 {
537 if (debug_lvl > 0) Report(erLand);
538 }
539
540 if ( ReportEnable
541 && !TakeoffReported
542 && (in.DistanceAGL - vLocalGear(eZ)) > 50.0
543 && !in.WOW)
544 {
545 if (debug_lvl > 0) Report(erTakeoff);
546 }
547
548 if (lastWOW != WOW)
549 {
550 if (debug_lvl > 0) {
551 cout << "GEAR_CONTACT: " << fdmex->GetSimTime() << " seconds: " << name
552 << " " << WOW << endl;
553 }
554 }
555}
556
557//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
558// Crash detection logic (really out-of-bounds detection)
559
560void FGLGear::CrashDetect(void)
561{
562 if ( (compressLength > 500.0 ||
563 vFn.Magnitude() > 100000000.0 ||
564 GetMoments().Magnitude() > 5000000000.0 ||
565 SinkRate > 1.4666*30 ) && !fdmex->IntegrationSuspended())
566 {
567 if (debug_lvl > 0) {
568 cout << "*CRASH DETECTED* " << fdmex->GetSimTime() << " seconds: " << name;
569 }
570
571 // fdmex->SuspendIntegration();
572 }
573}
574
575//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
576// The following needs work regarding friction coefficients and braking and
577// steering The BrakeFCoeff formula assumes that an anti-skid system is used.
578// It also assumes that we won't be turning and braking at the same time.
579// Will fix this later.
580// [JSB] The braking force coefficients include normal rolling coefficient +
581// a percentage of the static friction coefficient based on braking applied.
582
583void FGLGear::ComputeBrakeForceCoefficient(void)
584{
585 BrakeFCoeff = rollingFFactor * rollingFCoeff;
586
587 if (eBrakeGrp != bgNone)
588 BrakeFCoeff += in.BrakePos[eBrakeGrp] * staticFFactor * (staticFCoeff - rollingFCoeff);
589}
590
591//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
592// Compute the sideforce coefficients using Pacejka's Magic Formula.
593//
594// y(x) = D sin {C arctan [Bx - E(Bx - arctan Bx)]}
595//
596// Where: B = Stiffness Factor (0.06, here)
597// C = Shape Factor (2.8, here)
598// D = Peak Factor (0.8, here)
599// E = Curvature Factor (1.03, here)
600
601void FGLGear::ComputeSideForceCoefficient(void)
602{
603 if (ForceY_Table) {
604 FCoeff = ForceY_Table->GetValue(WheelSlip);
605 } else {
606 double StiffSlip = Stiffness*WheelSlip;
607 FCoeff = Peak * sin(Shape*atan(StiffSlip - Curvature*(StiffSlip - atan(StiffSlip))));
608 }
609 FCoeff *= staticFFactor;
610}
611
612//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
613// Compute the vertical force on the wheel using square-law damping (per comment
614// in paper AIAA-2000-4303 - see header prologue comments). We might consider
615// allowing for both square and linear damping force calculation. Also need to
616// possibly give a "rebound damping factor" that differs from the compression
617// case.
618
619void FGLGear::ComputeVerticalStrutForce()
620{
621 if (fStrutForce)
622 StrutForce = min(fStrutForce->GetValue(), (double)0.0);
623 else {
624 double springForce = -compressLength * kSpring;
625 double dampForce = 0;
626
627 if (compressSpeed >= 0.0) {
628
629 if (eDampType == dtLinear)
630 dampForce = -compressSpeed * bDamp;
631 else
632 dampForce = -compressSpeed * compressSpeed * bDamp;
633
634 } else {
635
636 if (eDampTypeRebound == dtLinear)
637 dampForce = -compressSpeed * bDampRebound;
638 else
639 dampForce = compressSpeed * compressSpeed * bDampRebound;
640
641 }
642
643 StrutForce = min(springForce + dampForce, (double)0.0);
644 if (StrutForce > maximumForce) {
645 StrutForce = maximumForce;
646 compressLength = -StrutForce / kSpring;
647 }
648 }
649
650 // The reaction force of the wheel is always normal to the ground
651 switch (eContactType) {
652 case ctBOGEY:
653 // Project back the strut force in the local coordinate frame of the ground
654 vFn(eZ) = StrutForce / (mTGear.Transposed()*vGroundNormal)(eZ);
655 break;
656 case ctSTRUCTURE:
657 vFn(eZ) = -StrutForce;
658 break;
659 }
660
661 // Remember these values for reporting
662 MaximumStrutForce = max(MaximumStrutForce, fabs(StrutForce));
663 MaximumStrutTravel = max(MaximumStrutTravel, fabs(compressLength));
664}
665
666//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
667
668double FGLGear::GetGearUnitPos(void) const
669{
670 // hack to provide backward compatibility to gear/gear-pos-norm property
671 if( useFCSGearPos || in.FCSGearPos != 1.0 ) {
672 useFCSGearPos = true;
673 return in.FCSGearPos;
674 }
675 return GearPos;
676}
677
678//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
679// Compute the jacobian entries for the friction forces resolution later
680// in FGPropagate
681
682void FGLGear::ComputeJacobian(const FGColumnVector3& vWhlContactVec)
683{
684 // When the point of contact is moving, dynamic friction is used
685 // This type of friction is limited to ctSTRUCTURE elements because their
686 // friction coefficient is the same in every directions
687 if ((eContactType == ctSTRUCTURE) && (vGroundWhlVel.Magnitude(eX,eY) > 1E-3)) {
688
689 FGColumnVector3 velocityDirection = vGroundWhlVel;
690
691 StaticFriction = false;
692
693 velocityDirection(eZ) = 0.;
694 velocityDirection.Normalize();
695
696 LMultiplier[ftDynamic].ForceJacobian = mT * velocityDirection;
697 LMultiplier[ftDynamic].Max = 0.;
698 LMultiplier[ftDynamic].Min = -fabs(staticFFactor * dynamicFCoeff * vFn(eZ));
699 LMultiplier[ftDynamic].LeverArm = vWhlContactVec;
700
701 // The Lagrange multiplier value obtained from the previous iteration is
702 // kept. This is supposed to accelerate the convergence of the projected
703 // Gauss-Seidel algorithm. The code just below is to make sure that the
704 // initial value is consistent with the current friction coefficient and
705 // normal reaction.
706 LMultiplier[ftDynamic].value = Constrain(LMultiplier[ftDynamic].Min, LMultiplier[ftDynamic].value, LMultiplier[ftDynamic].Max);
707
708 GroundReactions->RegisterLagrangeMultiplier(&LMultiplier[ftDynamic]);
709 }
710 else {
711 // Static friction is used for ctSTRUCTURE when the contact point is not
712 // moving. It is always used for ctBOGEY elements because the friction
713 // coefficients of a tyre depend on the direction of the movement (roll &
714 // side directions). This cannot be handled properly by the so-called
715 // "dynamic friction".
716 StaticFriction = true;
717
718 LMultiplier[ftRoll].ForceJacobian = mT * FGColumnVector3(1.,0.,0.);
719 LMultiplier[ftSide].ForceJacobian = mT * FGColumnVector3(0.,1.,0.);
720 LMultiplier[ftRoll].LeverArm = vWhlContactVec;
721 LMultiplier[ftSide].LeverArm = vWhlContactVec;
722
723 switch(eContactType) {
724 case ctBOGEY:
725 LMultiplier[ftRoll].Max = fabs(BrakeFCoeff * vFn(eZ));
726 LMultiplier[ftSide].Max = fabs(FCoeff * vFn(eZ));
727 break;
728 case ctSTRUCTURE:
729 LMultiplier[ftRoll].Max = fabs(staticFFactor * staticFCoeff * vFn(eZ));
730 LMultiplier[ftSide].Max = LMultiplier[ftRoll].Max;
731 break;
732 }
733
734 LMultiplier[ftRoll].Min = -LMultiplier[ftRoll].Max;
735 LMultiplier[ftSide].Min = -LMultiplier[ftSide].Max;
736
737 // The Lagrange multiplier value obtained from the previous iteration is
738 // kept. This is supposed to accelerate the convergence of the projected
739 // Gauss-Seidel algorithm. The code just below is to make sure that the
740 // initial value is consistent with the current friction coefficient and
741 // normal reaction.
742 LMultiplier[ftRoll].value = Constrain(LMultiplier[ftRoll].Min, LMultiplier[ftRoll].value, LMultiplier[ftRoll].Max);
743 LMultiplier[ftSide].value = Constrain(LMultiplier[ftSide].Min, LMultiplier[ftSide].value, LMultiplier[ftSide].Max);
744
745 GroundReactions->RegisterLagrangeMultiplier(&LMultiplier[ftRoll]);
746 GroundReactions->RegisterLagrangeMultiplier(&LMultiplier[ftSide]);
747 }
748}
749
750//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
751// This routine is called after the Lagrange multiplier has been computed in
752// the FGAccelerations class. The friction forces of the landing gear are then
753// updated accordingly.
754void FGLGear::UpdateForces(void)
755{
756 if (StaticFriction) {
757 vFn(eX) = LMultiplier[ftRoll].value;
758 vFn(eY) = LMultiplier[ftSide].value;
759 }
760 else {
761 FGColumnVector3 forceDir = mT.Transposed() * LMultiplier[ftDynamic].ForceJacobian;
762 vFn(eX) = LMultiplier[ftDynamic].value * forceDir(eX);
763 vFn(eY) = LMultiplier[ftDynamic].value * forceDir(eY);
764 }
765}
766
767//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
768
769void FGLGear::SetstaticFCoeff(double coeff)
770{
771 staticFCoeff = coeff;
772 Peak = coeff;
773}
774
775//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
776
777void FGLGear::bind(FGPropertyManager* PropertyManager)
778{
779 string property_name;
780 string base_property_name;
781
782 switch(eContactType) {
783 case ctBOGEY:
784 base_property_name = CreateIndexedPropertyName("gear/unit", GearNumber);
785 break;
786 case ctSTRUCTURE:
787 base_property_name = CreateIndexedPropertyName("contact/unit", GearNumber);
788 break;
789 default:
790 return;
791 }
792
793 property_name = base_property_name + "/AGL-ft";
794 PropertyManager->Tie(property_name.c_str(), &AGL);
795 property_name = base_property_name + "/WOW";
796 PropertyManager->Tie( property_name.c_str(), &WOW );
797 property_name = base_property_name + "/x-position";
798 PropertyManager->Tie( property_name.c_str(), (FGForce*)this,
799 &FGForce::GetLocationX, &FGForce::SetLocationX);
800 property_name = base_property_name + "/y-position";
801 PropertyManager->Tie( property_name.c_str(), (FGForce*)this,
802 &FGForce::GetLocationY, &FGForce::SetLocationY);
803 property_name = base_property_name + "/z-position";
804 PropertyManager->Tie( property_name.c_str(), (FGForce*)this,
805 &FGForce::GetLocationZ, &FGForce::SetLocationZ);
806 property_name = base_property_name + "/compression-ft";
807 PropertyManager->Tie( property_name.c_str(), &compressLength );
808 property_name = base_property_name + "/compression-velocity-fps";
809 PropertyManager->Tie( property_name.c_str(), &compressSpeed );
810 property_name = base_property_name + "/static_friction_coeff";
811 PropertyManager->Tie( property_name.c_str(), (FGLGear*)this,
812 &FGLGear::GetstaticFCoeff, &FGLGear::SetstaticFCoeff);
813 property_name = base_property_name + "/dynamic_friction_coeff";
814 PropertyManager->Tie( property_name.c_str(), &dynamicFCoeff );
815
816 if (eContactType == ctBOGEY) {
817 property_name = base_property_name + "/slip-angle-deg";
818 PropertyManager->Tie( property_name.c_str(), &WheelSlip );
819 property_name = base_property_name + "/wheel-speed-fps";
820 PropertyManager->Tie( property_name.c_str(), (FGLGear*)this,
821 &FGLGear::GetWheelRollVel);
822 property_name = base_property_name + "/side_friction_coeff";
823 PropertyManager->Tie( property_name.c_str(), &FCoeff );
824 property_name = base_property_name + "/rolling_friction_coeff";
825 PropertyManager->Tie( property_name.c_str(), &rollingFCoeff );
826
827 if (eSteerType == stCaster) {
828 property_name = base_property_name + "/steering-angle-deg";
829 PropertyManager->Tie( property_name.c_str(), this, &FGLGear::GetSteerAngleDeg );
830 property_name = base_property_name + "/castered";
831 PropertyManager->Tie( property_name.c_str(), &Castered);
832 }
833 }
834
835 if( isRetractable ) {
836 property_name = base_property_name + "/pos-norm";
837 PropertyManager->Tie( property_name.c_str(), &GearPos );
838 }
839
840 if (eSteerType != stFixed) {
841 // This property allows the FCS to override the steering position angle that
842 // is set by the property fcs/steer-cmd-norm. The prefix fcs/ has been kept
843 // for backward compatibility.
844 string tmp = CreateIndexedPropertyName("fcs/steer-pos-deg", GearNumber);
845 PropertyManager->Tie(tmp.c_str(), this, &FGLGear::GetSteerAngleDeg, &FGLGear::SetSteerAngleDeg);
846 }
847
848 property_name = base_property_name + "/solid";
849 PropertyManager->Tie( property_name.c_str(), &isSolid);
850 property_name = base_property_name + "/bumpiness";
851 PropertyManager->Tie( property_name.c_str(), &bumpiness);
852 property_name = base_property_name + "/maximum-force-lbs";
853 PropertyManager->Tie( property_name.c_str(), &maximumForce);
854 property_name = base_property_name + "/rolling_friction-factor";
855 PropertyManager->Tie( property_name.c_str(), &rollingFFactor);
856 property_name = base_property_name + "/static-friction-factor";
857 PropertyManager->Tie( property_name.c_str(), &staticFFactor);
858}
859
860//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
861
862void FGLGear::Report(ReportType repType)
863{
864 if (fabs(TakeoffDistanceTraveled) < 0.001) return; // Don't print superfluous reports
865
866 switch(repType) {
867 case erLand:
868 cout << endl << "Touchdown report for " << name << " (WOW at time: "
869 << fdmex->GetSimTime() << " seconds)" << endl;
870 cout << " Sink rate at contact: " << SinkRate << " fps, "
871 << SinkRate*0.3048 << " mps" << endl;
872 cout << " Contact ground speed: " << GroundSpeed*.5925 << " knots, "
873 << GroundSpeed*0.3048 << " mps" << endl;
874 cout << " Maximum contact force: " << MaximumStrutForce << " lbs, "
875 << MaximumStrutForce*4.448 << " Newtons" << endl;
876 cout << " Maximum strut travel: " << MaximumStrutTravel*12.0 << " inches, "
877 << MaximumStrutTravel*30.48 << " cm" << endl;
878 cout << " Distance traveled: " << LandingDistanceTraveled << " ft, "
879 << LandingDistanceTraveled*0.3048 << " meters" << endl;
880 LandingReported = true;
881 break;
882 case erTakeoff:
883 cout << endl << "Takeoff report for " << name << " (Liftoff at time: "
884 << fdmex->GetSimTime() << " seconds)" << endl;
885 cout << " Distance traveled: " << TakeoffDistanceTraveled
886 << " ft, " << TakeoffDistanceTraveled*0.3048 << " meters" << endl;
887 cout << " Distance traveled (over 50'): " << TakeoffDistanceTraveled50ft
888 << " ft, " << TakeoffDistanceTraveled50ft*0.3048 << " meters" << endl;
889 cout << " [Altitude (ASL): " << in.DistanceASL << " ft. / "
890 << in.DistanceASL*FGJSBBase::fttom << " m | Temperature: "
891 << in.Temperature - 459.67 << " F / "
892 << RankineToCelsius(in.Temperature) << " C]" << endl;
893 cout << " [Velocity (KCAS): " << in.VcalibratedKts << "]" << endl;
894 TakeoffReported = true;
895 break;
896 case erNone:
897 break;
898 }
899}
900
901//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
902// The bitmasked value choices are as follows:
903// unset: In this case (the default) JSBSim would only print
904// out the normally expected messages, essentially echoing
905// the config files as they are read. If the environment
906// variable is not set, debug_lvl is set to 1 internally
907// 0: This requests JSBSim not to output any messages
908// whatsoever.
909// 1: This value explicity requests the normal JSBSim
910// startup messages
911// 2: This value asks for a message to be printed out when
912// a class is instantiated
913// 4: When this value is set, a message is displayed when a
914// FGModel object executes its Run() method
915// 8: When this value is set, various runtime state variables
916// are printed out periodically
917// 16: When set various parameters are sanity checked and
918// a message is printed out when they go out of bounds
919
920void FGLGear::Debug(int from)
921{
922 static const char* sSteerType[] = {"STEERABLE", "FIXED", "CASTERED" };
923 static const char* sBrakeGroup[] = {"NONE", "LEFT", "RIGHT", "CENTER", "NOSE", "TAIL"};
924 static const char* sContactType[] = {"BOGEY", "STRUCTURE" };
925
926 if (debug_lvl <= 0) return;
927
928 if (debug_lvl & 1) { // Standard console startup message output
929 if (from == 0) { // Constructor - loading and initialization
930 cout << " " << sContactType[eContactType] << " " << name << endl;
931 cout << " Location: " << vXYZn << endl;
932 cout << " Spring Constant: " << kSpring << endl;
933
934 if (eDampType == dtLinear)
935 cout << " Damping Constant: " << bDamp << " (linear)" << endl;
936 else
937 cout << " Damping Constant: " << bDamp << " (square law)" << endl;
938
939 if (eDampTypeRebound == dtLinear)
940 cout << " Rebound Damping Constant: " << bDampRebound << " (linear)" << endl;
941 else
942 cout << " Rebound Damping Constant: " << bDampRebound << " (square law)" << endl;
943
944 cout << " Dynamic Friction: " << dynamicFCoeff << endl;
945 cout << " Static Friction: " << staticFCoeff << endl;
946 if (eContactType == ctBOGEY) {
947 cout << " Rolling Friction: " << rollingFCoeff << endl;
948 cout << " Steering Type: " << sSteerType[eSteerType] << endl;
949 cout << " Grouping: " << sBrakeGroup[eBrakeGrp] << endl;
950 cout << " Max Steer Angle: " << maxSteerAngle << endl;
951 cout << " Retractable: " << isRetractable << endl;
952 }
953 }
954 }
955 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
956 if (from == 0) cout << "Instantiated: FGLGear" << endl;
957 if (from == 1) cout << "Destroyed: FGLGear" << endl;
958 }
959 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
960 }
961 if (debug_lvl & 8 ) { // Runtime state variables
962 }
963 if (debug_lvl & 16) { // Sanity checking
964 }
965 if (debug_lvl & 64) {
966 if (from == 0) { // Constructor
967 }
968 }
969}
970
971} // namespace JSBSim
Element * FindElement(const std::string &el="")
Searches for a specified element.
FGColumnVector3 FindElementTripletConvertTo(const std::string &target_units)
Composes a 3-element column vector for the supplied location or orientation.
std::string GetAttributeValue(const std::string &key)
Retrieves an attribute.
std::string FindElementValue(const std::string &el="")
Searches for the named element and returns the string data belonging to it.
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.
double FindElementValueAsNumber(const std::string &el="")
Searches for the named element and returns the data belonging to it as a number.
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< FGGroundReactions > GetGroundReactions(void) const
Returns the FGGroundReactions pointer.
bool IntegrationSuspended(void) const
Returns the simulation suspension state.
Definition FGFDMExec.h:562
double GetSimTime(void) const
Returns the cumulative simulation time in seconds.
Definition FGFDMExec.h:549
std::shared_ptr< FGInertial > GetInertial(void) const
Returns the FGInertial pointer.
std::shared_ptr< FGPropertyManager > GetPropertyManager(void) const
Returns a pointer to the property manager object.
Definition FGFDMExec.h:421
Utility class that aids in the conversion of forces between coordinate systems and calculation of mom...
Definition FGForce.h:222
Represents a mathematical function.
Definition FGFunction.h:765
double GetValue(void) const override
Retrieves the value of the function object.
static constexpr double Constrain(double min, double value, double max)
Constrain a value between a minimum and a maximum value.
Definition FGJSBBase.h:288
static constexpr double RankineToCelsius(double rankine)
Converts from degrees Rankine to degrees Celsius.
Definition FGJSBBase.h:199
const FGColumnVector3 & GetBodyForces(void) override
The Force vector for this gear.
Definition FGLGear.cpp:281
FGLGear(Element *el, FGFDMExec *Executive, int number, const struct Inputs &input)
Constructor.
Definition FGLGear.cpp:72
~FGLGear()
Destructor.
Definition FGLGear.cpp:240
FGLocation holds an arbitrary location in the Earth centered Earth fixed reference frame (ECEF).
Definition FGLocation.h:152
FGLocation LocalToLocation(const FGColumnVector3 &lvec) const
Conversion from Local frame coordinates to a location in the earth centered and fixed frame.
Definition FGLocation.h:326
FGMatrix33 Transposed(void) const
Transposed matrix.
Definition FGMatrix33.h:221
Models the Quaternion representation of rotations.
const FGMatrix33 & GetT(void) const
Transformation matrix.
bool GetSolid(void)
Gets the surface is a solid flag value.
Definition FGSurface.h:107
double GetBumpiness(void)
Gets the normalized bumpiness factor associated with the surface.
Definition FGSurface.h:104
double GetRollingFFactor(void)
Gets the rolling friction factor of the surface area.
Definition FGSurface.h:98
double GetMaximumForce(void)
Gets the maximum force of the surface area.
Definition FGSurface.h:101
double GetBumpHeight()
Returns the height of the bump at the provided offset.
Definition FGSurface.cpp:92
double GetStaticFFactor(void)
Gets the static friction factor of the surface area.
Definition FGSurface.h:95
Lookup table class.
Definition FGTable.h:234
double GetValue(void) const
Get the current table value.
Definition FGTable.cpp:465