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
FGLinearActuator.cpp
1/* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2 *
3Module: FGLinearActuator.cpp
4Author: Adriano Bassignana
5Date started: 2019-01-03
6
7------------- Copyright (C) 2019 Adriano Bassignana -------------
8
9This program is free software; you can redistribute it and/or modify it under
10the terms of the GNU Lesser General Public License as published by the Free
11Software Foundation; either version 2 of the License, or (at your option) any
12later version.
13
14This program is distributed in the hope that it will be useful, but WITHOUT
15ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
16FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
17details.
18
19You should have received a copy of the GNU Lesser General Public License along
20with this program; if not, write to the Free Software Foundation, Inc., 59
21Temple Place - Suite 330, Boston, MA 02111-1307, USA.
22
23Further information about the GNU Lesser General Public License can also be
24found 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 "FGLinearActuator.h"
41#include "models/FGFCS.h"
42#include "math/FGParameterValue.h"
43
44using namespace std;
45
46namespace JSBSim {
47
48/* %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
49CLASS IMPLEMENTATION
50%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
51
53 : FGFCSComponent(fcs, element)
54{
55 CheckInputNodes(1, 1, element);
56
57 ptrSet = nullptr;
58 auto PropertyManager = fcs->GetPropertyManager();
59 if (element->FindElement("set")) {
60 string property_string = element->FindElementValue("set");
61 ptrSet = new FGParameterValue(property_string, PropertyManager, element);
62 if (ptrSet && ptrSet->IsConstant()) {
63 set = ptrSet->GetValue() >= 0.5;
64 }
65 }
66
67 ptrReset = nullptr;
68 if (element->FindElement("reset")) {
69 string property_string = element->FindElementValue("reset");
70 ptrReset = new FGParameterValue(property_string, PropertyManager, element);
71 if (ptrReset && ptrReset->IsConstant()) {
72 reset = ptrReset->GetValue() >= 0.5;
73 }
74 }
75
76 ptrVersus = nullptr;
77 if (element->FindElement("versus")) {
78 string property_string = element->FindElementValue("versus");
79 ptrVersus = new FGParameterValue(property_string, PropertyManager, element);
80 if (ptrVersus && ptrVersus->IsConstant()) {
81 versus = ptrVersus->GetValue();
82 }
83 }
84
85 ptrBias = nullptr;
86 if (element->FindElement("bias")) {
87 string property_string = element->FindElementValue("bias");
88 ptrBias = new FGParameterValue(property_string, PropertyManager, element);
89 if (ptrBias && ptrBias->IsConstant()) {
90 bias = ptrBias->GetValue();
91 }
92 }
93
94 if (element->FindElement("module")) {
95 module = element->FindElementValueAsNumber("module");
96 if (module < 0) {
97 cout << "FGLinearActuator::Run " << InputNodes[0]->GetNameWithSign()
98 << " <module> parameter is forced from " << module
99 << " value to 1.0 value" << endl;
100 module = 1.0;
101 }
102 }
103
104 if (element->FindElement("hysteresis")) {
105 hysteresis = element->FindElementValueAsNumber("hysteresis");
106 if (hysteresis < 0) {
107 cout << "FGLinearActuator::Run " << InputNodes[0]->GetNameWithSign()
108 << " <hysteresis> parameter is forced from " << hysteresis
109 << " value to 0.0 value" << endl;
110 hysteresis = 0.0;
111 }
112 }
113
114 if (element->FindElement("lag")) {
115 lag = element->FindElementValueAsNumber("lag");
116 if (lag > 0.0) {
117 double denom = 2.00 + dt*lag;
118 ca = dt * lag / denom;
119 cb = (2.00 - dt * lag) / denom;
120 previousLagInput = previousLagOutput = 0.0;
121 } else {
122 if (lag < 0) {
123 cout << "FGLinearActuator::Run " << InputNodes[0]->GetNameWithSign()
124 << " <lag> parameter is forced from "
125 << lag << " value to 0.0 value" << endl;
126 lag = 0;
127 }
128 }
129 }
130
131 if (element->FindElement("rate")) {
132 rate = element->FindElementValueAsNumber("rate");
133 if (rate <= 0 || rate > 1.0) {
134 cout << "FGLinearActuator::Run " << InputNodes[0]->GetNameWithSign()
135 << " <rate> parameter is forced from " << rate
136 << " value to 0.5 value" << endl;
137 rate = 0.5;
138 }
139 }
140
141 if (element->FindElement("gain"))
142 gain = element->FindElementValueAsNumber("gain");
143
144 bind(element, PropertyManager.get());
145
146 Debug(0);
147}
148
149// %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
150
152{
153 Debug(1);
154}
155
156// %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
157
159{
160 if (ptrSet && !ptrSet->IsConstant()) set = ptrSet->GetValue() >= 0.5;
161 if (ptrReset && !ptrReset->IsConstant()) reset = ptrReset->GetValue() >= 0.5;
162
163 if (reset) {
164 inputMem = 0.0;
165 countSpin = 0;
166 direction = 0;
167 Output = 0.0;
168 inputLast = 0.0;
169 } else {
170 if (set) {
171 Input = InputNodes[0]->getDoubleValue() - inputLast;
172 double inputDelta = Input - inputMem;
173 if (abs(inputDelta) >= hysteresis) {
174 if (ptrVersus && !ptrVersus->IsConstant()) {
175 versus = ptrVersus->GetValue();
176 if (versus >= 0.5) {
177 versus = 1;
178 } else if (versus <= -0.5) {
179 versus = -1;
180 } else versus = 0;
181 }
182 if (abs(inputDelta) <= (module * rate)) {
183 if (inputDelta > 0.0) {
184 direction = 1;
185 } else if (inputDelta < 0.0) {
186 direction = -1;
187 }
188 }
189 if ((versus == 0) || (versus == direction)) {
190 inputMem = Input;
191 if (abs(inputDelta) >= (module*rate)) {
192 if (inputDelta < 0)
193 countSpin++;
194 else
195 countSpin--;
196 }
197 } else if ((versus != 0) && (direction != 0) && (versus != direction)) {
198 inputLast += inputDelta;
199 }
200 }
201 }
202 if (ptrBias && !ptrBias->IsConstant()) {
203 bias = ptrBias->GetValue();
204 }
205 Output = gain * (bias + inputMem + module*countSpin);
206 }
207
208 if (lag > 0.0) {
209 double input = Output;
210 Output = ca * (input + previousLagInput) + previousLagOutput * cb;
211 previousLagInput = input;
212 previousLagOutput = Output;
213 }
214
215 SetOutput();
216
217 return true;
218}
219
220// %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
221// The bitmasked value choices are as follows:
222// unset: In this case (the default) JSBSim would only print
223// out the normally expected messages, essentially echoing
224// the config files as they are read. If the environment
225// variable is not set, debug_lvl is set to 1 internally
226// 0: This requests JSBSim not to output any messages
227// whatsoever.
228// 1: This value explicity requests the normal JSBSim
229// startup messages
230// 2: This value asks for a message to be printed out when
231// a class is instantiated
232// 4: When this value is set, a message is displayed when a
233// FGModel object executes its Run() method
234// 8: When this value is set, various runtime state variables
235// are printed out periodically
236// 16: When set various parameters are sanity checked and
237// a message is printed out when they go out of bounds
238void FGLinearActuator::Debug(int from)
239{
240 if (debug_lvl <= 0) return;
241
242 if (debug_lvl & 1) { // Standard console startup message output
243 if (from == 0) { // Constructor
244 cout << " INPUT: " << InputNodes[0]->GetNameWithSign() << endl;
245 cout << " inputMem: " << inputMem << endl;
246 cout << " bias: " << bias << endl;
247 cout << " module: " << module << endl;
248 cout << " hysteresis: " << hysteresis << endl;
249 cout << " rate: " << rate << endl;
250 cout << " versus: " << versus << endl;
251 cout << " direction: " << direction << endl;
252 cout << " countSpin: " << countSpin << endl;
253 cout << " Lag: " << lag << endl;
254 cout << " Gain: " << gain << endl;
255 cout << " set: " << set << endl;
256 cout << " reset: " << reset << endl;
257 for (auto node: OutputNodes)
258 cout << " OUTPUT: " << node->GetName() << endl;
259 }
260 }
261 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
262 if (from == 0) cout << "Instantiated: FGLinearActuator" << endl;
263 if (from == 1) cout << "Destroyed: FGLinearActuator" << endl;
264 }
265 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
266 }
267 if (debug_lvl & 8 ) { // Runtime state variables
268 }
269 if (debug_lvl & 16) { // Sanity checking
270 }
271 if (debug_lvl & 64) {
272 if (from == 0) { // Constructor
273 }
274 }
275}
276}
Element * FindElement(const std::string &el="")
Searches for a specified element.
std::string FindElementValue(const std::string &el="")
Searches for the named element and returns the string data belonging to it.
double FindElementValueAsNumber(const std::string &el="")
Searches for the named element and returns the data belonging to it as a number.
Base class for JSBSim Flight Control System Components.
Encapsulates the Flight Control System (FCS) functionality.
Definition FGFCS.h:189
bool Run(void) override
The execution method for this FCS component.
FGLinearActuator(FGFCS *fcs, Element *element)
Constructor.
Represents a either a real value or a property value.