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
FGActuator.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: FGActuator.cpp
4 Author: Jon Berndt
5 Date started: 21 February 2006
6
7 ------------- Copyright (C) 2007 Jon S. Berndt (jon@jsbsim.org) -------------
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
26FUNCTIONAL DESCRIPTION
27--------------------------------------------------------------------------------
28
29HISTORY
30--------------------------------------------------------------------------------
31
32%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
33COMMENTS, REFERENCES, and NOTES
34%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
35
36%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
37INCLUDES
38%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
39
40#include "FGActuator.h"
41#include "input_output/FGXMLElement.h"
42#include "math/FGParameterValue.h"
43#include "models/FGFCS.h"
44
45using namespace std;
46
47namespace JSBSim {
48
49/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
50CLASS IMPLEMENTATION
51%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
52
54 : FGFCSComponent(fcs, element)
55{
56 // inputs are read from the base class constructor
57
58 PreviousOutput = 0.0;
59 PreviousHystOutput = 0.0;
60 PreviousRateLimOutput = 0.0;
61 PreviousLagInput = PreviousLagOutput = 0.0;
62 bias = hysteresis_width = deadband_width = 0.0;
63 lag = nullptr;
64 rate_limit_incr = rate_limit_decr = 0; // no limit
65 fail_zero = fail_hardover = fail_stuck = false;
66 ca = cb = 0.0;
67 initialized = 0;
68 saturated = false;
69
70 CheckInputNodes(1, 1, element);
71
72 if ( element->FindElement("deadband_width") ) {
73 deadband_width = element->FindElementValueAsNumber("deadband_width");
74 }
75 if ( element->FindElement("hysteresis_width") ) {
76 hysteresis_width = element->FindElementValueAsNumber("hysteresis_width");
77 }
78
79 // There can be a single rate limit specified, or increasing and
80 // decreasing rate limits specified, and rate limits can be numeric, or
81 // a property.
82 auto PropertyManager = fcs->GetPropertyManager();
83 Element* ratelim_el = element->FindElement("rate_limit");
84 while ( ratelim_el ) {
85 string rate_limit_str = ratelim_el->GetDataLine();
86 FGParameter* rate_limit = new FGParameterValue(rate_limit_str,
87 PropertyManager, ratelim_el);
88
89 if (ratelim_el->HasAttribute("sense")) {
90 string sense = ratelim_el->GetAttributeValue("sense");
91 if (sense.substr(0,4) == "incr")
92 rate_limit_incr = rate_limit;
93 else if (sense.substr(0,4) == "decr")
94 rate_limit_decr = rate_limit;
95 } else {
96 rate_limit_incr = rate_limit;
97 rate_limit_decr = rate_limit;
98 }
99 ratelim_el = element->FindNextElement("rate_limit");
100 }
101
102 if ( element->FindElement("bias") ) {
103 bias = element->FindElementValueAsNumber("bias");
104 }
105
106 // Lag if specified can be numeric or a property
107 Element* lag_el = element->FindElement("lag");
108 if ( lag_el ) {
109 string lag_str = lag_el->GetDataLine();
110 lag = new FGParameterValue(lag_str, PropertyManager, lag_el);
111 InitializeLagCoefficients();
112 }
113
114 bind(element, PropertyManager.get());
115
116 Debug(0);
117}
118
119//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
120
122{
123 delete rate_limit_incr;
124 if (rate_limit_decr != rate_limit_incr)
125 delete rate_limit_decr;
126
127 delete lag;
128
129 Debug(1);
130}
131
132//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
133
134void FGActuator::ResetPastStates(void)
135{
136 FGFCSComponent::ResetPastStates();
137
138 PreviousOutput = PreviousHystOutput = PreviousRateLimOutput
139 = PreviousLagInput = PreviousLagOutput = Output = 0.0;
140}
141
142//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
143
145{
146 Input = InputNodes[0]->getDoubleValue();
147
148 if( fcs->GetTrimStatus() ) initialized = 0;
149
150 if (fail_zero) Input = 0;
151 if (fail_hardover) Input = Input < 0.0 ? ClipMin->GetValue() : ClipMax->GetValue();
152
153 Output = Input; // Perfect actuator. At this point, if no failures are present
154 // and no subsequent lag, limiting, etc. is done, the output
155 // is simply the input. If any further processing is done
156 // (below) such as lag, rate limiting, hysteresis, etc., then
157 // the Input will be further processed and the eventual Output
158 // will be overwritten from this perfect value.
159
160 if (fail_stuck) {
161 Output = PreviousOutput;
162 } else {
163 if (lag) Lag(); // models actuator lag
164 if (rate_limit_incr != 0 || rate_limit_decr != 0) RateLimit(); // limit the actuator rate
165 if (deadband_width != 0.0) Deadband();
166 if (hysteresis_width != 0.0) Hysteresis();
167 if (bias != 0.0) Bias(); // models a finite bias
168 if (delay != 0) Delay(); // Model transport latency
169 }
170
171 PreviousOutput = Output; // previous value needed for "stuck" malfunction
172
173 initialized = 1;
174
175 Clip();
176
177 if (clip) {
178 double clipmax = ClipMax->GetValue();
179 saturated = false;
180
181 if (Output >= clipmax && clipmax != 0)
182 saturated = true;
183 else{
184 double clipmin = ClipMin->GetValue();
185 if (Output <= clipmin && clipmin != 0)
186 saturated = true;
187 }
188 }
189
190 SetOutput();
191
192 return true;
193}
194
195//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
196
197void FGActuator::Bias(void)
198{
199 Output += bias;
200}
201
202//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
203
204void FGActuator::Lag(void)
205{
206 // "Output" on the right side of the "=" is the current frame input
207 // for this Lag filter
208 double input = Output;
209
210 if (initialized) {
211 // Check if lag value has changed via dynamic property
212 if (lagVal != lag->GetValue())
213 InitializeLagCoefficients();
214 Output = ca * (input + PreviousLagInput) + PreviousLagOutput * cb;
215 }
216
217 PreviousLagInput = input;
218 PreviousLagOutput = Output;
219}
220
221//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
222
223void FGActuator::Hysteresis(void)
224{
225 // Note: this function acts cumulatively on the "Output" parameter. So,
226 // "Output" is - for the purposes of this Hysteresis method - really the input
227 // to the method.
228 double input = Output;
229
230 if ( initialized ) {
231 if (input > PreviousHystOutput)
232 Output = max(PreviousHystOutput, input-0.5*hysteresis_width);
233 else if (input < PreviousHystOutput)
234 Output = min(PreviousHystOutput, input+0.5*hysteresis_width);
235 }
236
237 PreviousHystOutput = Output;
238}
239
240//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
241
242void FGActuator::RateLimit(void)
243{
244 // Note: this function acts cumulatively on the "Output" parameter. So,
245 // "Output" is - for the purposes of this RateLimit method - really the input
246 // to the method.
247 double input = Output;
248 if ( initialized ) {
249 double delta = input - PreviousRateLimOutput;
250 if (rate_limit_incr) {
251 double rate_limit = rate_limit_incr->GetValue();
252 if (delta > dt * rate_limit)
253 Output = PreviousRateLimOutput + rate_limit * dt;
254 }
255 if (rate_limit_decr) {
256 double rate_limit = -rate_limit_decr->GetValue();
257 if (delta < dt * rate_limit)
258 Output = PreviousRateLimOutput + rate_limit * dt;
259 }
260 }
261 PreviousRateLimOutput = Output;
262}
263
264//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
265
266void FGActuator::Deadband(void)
267{
268 // Note: this function acts cumulatively on the "Output" parameter. So,
269 // "Output" is - for the purposes of this Deadband method - really the input
270 // to the method.
271 double input = Output;
272
273 if (input < -deadband_width/2.0) {
274 Output = (input + deadband_width/2.0);
275 } else if (input > deadband_width/2.0) {
276 Output = (input - deadband_width/2.0);
277 } else {
278 Output = 0.0;
279 }
280}
281
282//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
283
284void FGActuator::bind(Element* el, FGPropertyManager* PropertyManager)
285{
286 string tmp = Name;
287
288 FGFCSComponent::bind(el, PropertyManager);
289
290 if (Name.find("/") == string::npos) {
291 tmp = "fcs/" + PropertyManager->mkPropertyName(Name, true);
292 }
293 const string tmp_zero = tmp + "/malfunction/fail_zero";
294 const string tmp_hardover = tmp + "/malfunction/fail_hardover";
295 const string tmp_stuck = tmp + "/malfunction/fail_stuck";
296 const string tmp_sat = tmp + "/saturated";
297
298 PropertyManager->Tie( tmp_zero, this, &FGActuator::GetFailZero, &FGActuator::SetFailZero);
299 PropertyManager->Tie( tmp_hardover, this, &FGActuator::GetFailHardover, &FGActuator::SetFailHardover);
300 PropertyManager->Tie( tmp_stuck, this, &FGActuator::GetFailStuck, &FGActuator::SetFailStuck);
301 PropertyManager->Tie( tmp_sat, this, &FGActuator::IsSaturated);
302}
303
304//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
305
306void FGActuator::InitializeLagCoefficients()
307{
308 lagVal = lag->GetValue();
309 double denom = 2.00 + dt * lagVal;
310 ca = dt * lagVal / denom;
311 cb = (2.00 - dt * lagVal) / denom;
312}
313
314//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
315// The bitmasked value choices are as follows:
316// unset: In this case (the default) JSBSim would only print
317// out the normally expected messages, essentially echoing
318// the config files as they are read. If the environment
319// variable is not set, debug_lvl is set to 1 internally
320// 0: This requests JSBSim not to output any messages
321// whatsoever.
322// 1: This value explicity requests the normal JSBSim
323// startup messages
324// 2: This value asks for a message to be printed out when
325// a class is instantiated
326// 4: When this value is set, a message is displayed when a
327// FGModel object executes its Run() method
328// 8: When this value is set, various runtime state variables
329// are printed out periodically
330// 16: When set various parameters are sanity checked and
331// a message is printed out when they go out of bounds
332
333void FGActuator::Debug(int from)
334{
335 if (debug_lvl <= 0) return;
336
337 if (debug_lvl & 1) { // Standard console startup message output
338 if (from == 0) { // Constructor
339 cout << " INPUT: " << InputNodes[0]->GetNameWithSign() << endl;
340
341 if (!OutputNodes.empty()) {
342 for (auto node: OutputNodes)
343 cout << " OUTPUT: " << node->GetName() << endl;
344 }
345 if (bias != 0.0) cout << " Bias: " << bias << endl;
346 if (rate_limit_incr != 0) {
347 cout << " Increasing rate limit: " << rate_limit_incr->GetName() << endl;
348 }
349 if (rate_limit_decr != 0) {
350 cout << " Decreasing rate limit: " << rate_limit_decr->GetName() << endl;
351 }
352 if (lag != 0) cout << " Actuator lag: " << lag->GetName() << endl;
353 if (hysteresis_width != 0) cout << " Hysteresis width: " << hysteresis_width << endl;
354 if (deadband_width != 0) cout << " Deadband width: " << deadband_width << endl;
355 }
356 }
357 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
358 if (from == 0) cout << "Instantiated: FGActuator" << endl;
359 if (from == 1) cout << "Destroyed: FGActuator" << endl;
360 }
361 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
362 }
363 if (debug_lvl & 8 ) { // Runtime state variables
364 }
365 if (debug_lvl & 16) { // Sanity checking
366 }
367 if (debug_lvl & 64) {
368 if (from == 0) { // Constructor
369 }
370 }
371}
372}
Element * FindElement(const std::string &el="")
Searches for a specified element.
std::string GetAttributeValue(const std::string &key)
Retrieves an attribute.
std::string GetDataLine(unsigned int i=0)
Gets a line of data belonging to an element.
Element * FindNextElement(const std::string &el="")
Searches for the next element as specified.
bool HasAttribute(const std::string &key)
Determines if an element has the supplied attribute.
double FindElementValueAsNumber(const std::string &el="")
Searches for the named element and returns the data belonging to it as a number.
FGActuator(FGFCS *fcs, Element *element)
Constructor.
~FGActuator()
Destructor.
void SetFailZero(bool set)
This function fails the actuator to zero.
Definition FGActuator.h:145
bool Run(void) override
This function processes the input.
Base class for JSBSim Flight Control System Components.
Encapsulates the Flight Control System (FCS) functionality.
Definition FGFCS.h:189
Represents a either a real value or a property value.
Represents various types of parameters.
Definition FGParameter.h:61