40#include "FGActuator.h"
41#include "input_output/FGXMLElement.h"
42#include "math/FGParameterValue.h"
43#include "models/FGFCS.h"
59 PreviousHystOutput = 0.0;
60 PreviousRateLimOutput = 0.0;
61 PreviousLagInput = PreviousLagOutput = 0.0;
62 bias = hysteresis_width = deadband_width = 0.0;
64 rate_limit_incr = rate_limit_decr = 0;
65 fail_zero = fail_hardover = fail_stuck =
false;
70 CheckInputNodes(1, 1, element);
82 auto PropertyManager = fcs->GetPropertyManager();
84 while ( ratelim_el ) {
87 PropertyManager, ratelim_el);
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;
96 rate_limit_incr = rate_limit;
97 rate_limit_decr = rate_limit;
111 InitializeLagCoefficients();
114 bind(element, PropertyManager.get());
123 delete rate_limit_incr;
124 if (rate_limit_decr != rate_limit_incr)
125 delete rate_limit_decr;
134void FGActuator::ResetPastStates(
void)
136 FGFCSComponent::ResetPastStates();
138 PreviousOutput = PreviousHystOutput = PreviousRateLimOutput
139 = PreviousLagInput = PreviousLagOutput = Output = 0.0;
146 Input = InputNodes[0]->getDoubleValue();
148 if( fcs->GetTrimStatus() ) initialized = 0;
150 if (fail_zero) Input = 0;
151 if (fail_hardover) Input = Input < 0.0 ? ClipMin->GetValue() : ClipMax->GetValue();
161 Output = PreviousOutput;
164 if (rate_limit_incr != 0 || rate_limit_decr != 0) RateLimit();
165 if (deadband_width != 0.0) Deadband();
166 if (hysteresis_width != 0.0) Hysteresis();
167 if (bias != 0.0) Bias();
168 if (delay != 0) Delay();
171 PreviousOutput = Output;
178 double clipmax = ClipMax->GetValue();
181 if (Output >= clipmax && clipmax != 0)
184 double clipmin = ClipMin->GetValue();
185 if (Output <= clipmin && clipmin != 0)
197void FGActuator::Bias(
void)
204void FGActuator::Lag(
void)
208 double input = Output;
212 if (lagVal != lag->GetValue())
213 InitializeLagCoefficients();
214 Output = ca * (input + PreviousLagInput) + PreviousLagOutput * cb;
217 PreviousLagInput = input;
218 PreviousLagOutput = Output;
223void FGActuator::Hysteresis(
void)
228 double input = Output;
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);
237 PreviousHystOutput = Output;
242void FGActuator::RateLimit(
void)
247 double input = Output;
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;
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;
261 PreviousRateLimOutput = Output;
266void FGActuator::Deadband(
void)
271 double input = Output;
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);
284void FGActuator::bind(Element* el, FGPropertyManager* PropertyManager)
288 FGFCSComponent::bind(el, PropertyManager);
290 if (Name.find(
"/") == string::npos) {
291 tmp =
"fcs/" + PropertyManager->mkPropertyName(Name,
true);
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";
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);
306void FGActuator::InitializeLagCoefficients()
308 lagVal = lag->GetValue();
309 double denom = 2.00 + dt * lagVal;
310 ca = dt * lagVal / denom;
311 cb = (2.00 - dt * lagVal) / denom;
333void FGActuator::Debug(
int from)
335 if (debug_lvl <= 0)
return;
339 cout <<
" INPUT: " << InputNodes[0]->GetNameWithSign() << endl;
341 if (!OutputNodes.empty()) {
342 for (
auto node: OutputNodes)
343 cout <<
" OUTPUT: " << node->GetName() << endl;
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;
349 if (rate_limit_decr != 0) {
350 cout <<
" Decreasing rate limit: " << rate_limit_decr->GetName() << endl;
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;
357 if (debug_lvl & 2 ) {
358 if (from == 0) cout <<
"Instantiated: FGActuator" << endl;
359 if (from == 1) cout <<
"Destroyed: FGActuator" << endl;
361 if (debug_lvl & 4 ) {
363 if (debug_lvl & 8 ) {
365 if (debug_lvl & 16) {
367 if (debug_lvl & 64) {
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.
void SetFailZero(bool set)
This function fails the actuator to zero.
bool Run(void) override
This function processes the input.
Base class for JSBSim Flight Control System Components.
Encapsulates the Flight Control System (FCS) functionality.
Represents a either a real value or a property value.
Represents various types of parameters.