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
FGFilter.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: FGFilter.cpp
4 Author: Jon S. Berndt
5 Date started: 11/2000
6
7 ------------- Copyright (C) 2000 -------------
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 "FGFilter.h"
41#include "models/FGFCS.h"
42#include "math/FGParameterValue.h"
43
44using namespace std;
45
46namespace JSBSim {
47
48/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
49CLASS IMPLEMENTATION
50%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
51
52FGFilter::FGFilter(FGFCS* fcs, Element* element)
53 : FGFCSComponent(fcs, element), DynamicFilter(false), Initialize(true)
54{
55 C[1] = C[2] = C[3] = C[4] = C[5] = C[6] = nullptr;
56
57 CheckInputNodes(1, 1, element);
58
59 auto PropertyManager = fcs->GetPropertyManager();
60 for (int i=1; i<7; i++)
61 ReadFilterCoefficients(element, i, PropertyManager);
62
63 if (Type == "LAG_FILTER") FilterType = eLag ;
64 else if (Type == "LEAD_LAG_FILTER") FilterType = eLeadLag ;
65 else if (Type == "SECOND_ORDER_FILTER") FilterType = eOrder2 ;
66 else if (Type == "WASHOUT_FILTER") FilterType = eWashout ;
67 else FilterType = eUnknown ;
68
69 CalculateDynamicFilters();
70
71 bind(element, PropertyManager.get());
72
73 Debug(0);
74}
75
76//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
77
78FGFilter::~FGFilter()
79{
80 Debug(1);
81}
82
83//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
84
85void FGFilter::ResetPastStates(void)
86{
87 FGFCSComponent::ResetPastStates();
88
89 Input = 0.0; Initialize = true;
90}
91
92//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
93
94void FGFilter::ReadFilterCoefficients(Element* element, int index,
95 std::shared_ptr<FGPropertyManager> PropertyManager)
96{
97 // index is known to be 1-7.
98 // A stringstream would be overkill, but also trying to avoid sprintf
99 string coefficient = "c0";
100 coefficient[1] += index;
101
102 if ( element->FindElement(coefficient) ) {
103 C[index] = new FGParameterValue(element->FindElement(coefficient),
104 PropertyManager);
105 DynamicFilter |= !C[index]->IsConstant();
106 }
107}
108
109//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
110
111void FGFilter::CalculateDynamicFilters(void)
112{
113 double denom;
114
115 switch (FilterType) {
116 case eLag:
117 denom = 2.0 + dt*C[1];
118 ca = dt*C[1] / denom;
119 cb = (2.0 - dt*C[1]) / denom;
120
121 break;
122 case eLeadLag:
123 denom = 2.0*C[3] + dt*C[4];
124 ca = (2.0*C[1] + dt*C[2]) / denom;
125 cb = (dt*C[2] - 2.0*C[1]) / denom;
126 cc = (2.0*C[3] - dt*C[4]) / denom;
127 break;
128 case eOrder2:
129 denom = 4.0*C[4] + 2.0*C[5]*dt + C[6]*dt*dt;
130 ca = (4.0*C[1] + 2.0*C[2]*dt + C[3]*dt*dt) / denom;
131 cb = (2.0*C[3]*dt*dt - 8.0*C[1]) / denom;
132 cc = (4.0*C[1] - 2.0*C[2]*dt + C[3]*dt*dt) / denom;
133 cd = (2.0*C[6]*dt*dt - 8.0*C[4]) / denom;
134 ce = (4.0*C[4] - 2.0*C[5]*dt + C[6]*dt*dt) / denom;
135 break;
136 case eWashout:
137 denom = 2.0 + dt*C[1];
138 ca = 2.0 / denom;
139 cb = (2.0 - dt*C[1]) / denom;
140 break;
141 case eUnknown:
142 cerr << "Unknown filter type" << endl;
143 break;
144 }
145
146}
147
148//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
149
150bool FGFilter::Run(void)
151{
152 if (Initialize) {
153
154 PreviousOutput2 = PreviousInput2 = PreviousOutput1 = PreviousInput1 = Output = Input;
155 Initialize = false;
156
157 } else {
158
159 Input = InputNodes[0]->getDoubleValue();
160
161 if (DynamicFilter) CalculateDynamicFilters();
162
163 switch (FilterType) {
164 case eLag:
165 Output = (Input + PreviousInput1) * ca + PreviousOutput1 * cb;
166 break;
167 case eLeadLag:
168 Output = Input * ca + PreviousInput1 * cb + PreviousOutput1 * cc;
169 break;
170 case eOrder2:
171 Output = Input * ca + PreviousInput1 * cb + PreviousInput2 * cc
172 - PreviousOutput1 * cd - PreviousOutput2 * ce;
173 break;
174 case eWashout:
175 Output = Input * ca - PreviousInput1 * ca + PreviousOutput1 * cb;
176 break;
177 case eUnknown:
178 break;
179 }
180
181 }
182
183 PreviousOutput2 = PreviousOutput1;
184 PreviousOutput1 = Output;
185 PreviousInput2 = PreviousInput1;
186 PreviousInput1 = Input;
187
188 Clip();
189 SetOutput();
190
191 return true;
192}
193
194//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
195// The bitmasked value choices are as follows:
196// unset: In this case (the default) JSBSim would only print
197// out the normally expected messages, essentially echoing
198// the config files as they are read. If the environment
199// variable is not set, debug_lvl is set to 1 internally
200// 0: This requests JSBSim not to output any messages
201// whatsoever.
202// 1: This value explicity requests the normal JSBSim
203// startup messages
204// 2: This value asks for a message to be printed out when
205// a class is instantiated
206// 4: When this value is set, a message is displayed when a
207// FGModel object executes its Run() method
208// 8: When this value is set, various runtime state variables
209// are printed out periodically
210// 16: When set various parameters are sanity checked and
211// a message is printed out when they go out of bounds
212
213void FGFilter::Debug(int from)
214{
215 if (debug_lvl <= 0) return;
216
217 if (debug_lvl & 1) { // Standard console startup message output
218 if (from == 0) { // Constructor
219 cout << " INPUT: " << InputNodes[0]->GetName() << endl;
220
221 for (int i=1; i < 7; i++) {
222 if (!C[i]) break;
223
224 cout << " C[" << i << "]";
225 if (!C[i]->IsConstant()) cout << " is the value of property";
226 cout << ": "<< C[i]->GetName() << endl;
227 }
228
229 for (auto node: OutputNodes)
230 cout << " OUTPUT: " << node->getNameString() << endl;
231 }
232 }
233 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
234 if (from == 0) cout << "Instantiated: FGFilter" << endl;
235 if (from == 1) cout << "Destroyed: FGFilter" << endl;
236 }
237 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
238 }
239 if (debug_lvl & 8 ) { // Runtime state variables
240 }
241 if (debug_lvl & 16) { // Sanity checking
242 }
243 if (debug_lvl & 64) {
244 if (from == 0) { // Constructor
245 }
246 }
247}
248}