41#include "models/FGFCS.h"
42#include "math/FGParameterValue.h"
52FGFilter::FGFilter(FGFCS* fcs, Element* element)
53 : FGFCSComponent(fcs, element), DynamicFilter(false), Initialize(true)
55 C[1] = C[2] = C[3] = C[4] = C[5] = C[6] =
nullptr;
57 CheckInputNodes(1, 1, element);
59 auto PropertyManager = fcs->GetPropertyManager();
60 for (
int i=1; i<7; i++)
61 ReadFilterCoefficients(element, i, PropertyManager);
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 ;
69 CalculateDynamicFilters();
71 bind(element, PropertyManager.get());
85void FGFilter::ResetPastStates(
void)
87 FGFCSComponent::ResetPastStates();
89 Input = 0.0; Initialize =
true;
94void FGFilter::ReadFilterCoefficients(Element* element,
int index,
95 std::shared_ptr<FGPropertyManager> PropertyManager)
99 string coefficient =
"c0";
100 coefficient[1] += index;
102 if ( element->FindElement(coefficient) ) {
103 C[index] =
new FGParameterValue(element->FindElement(coefficient),
105 DynamicFilter |= !C[index]->IsConstant();
111void FGFilter::CalculateDynamicFilters(
void)
115 switch (FilterType) {
117 denom = 2.0 + dt*C[1];
118 ca = dt*C[1] / denom;
119 cb = (2.0 - dt*C[1]) / denom;
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;
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;
137 denom = 2.0 + dt*C[1];
139 cb = (2.0 - dt*C[1]) / denom;
142 cerr <<
"Unknown filter type" << endl;
150bool FGFilter::Run(
void)
154 PreviousOutput2 = PreviousInput2 = PreviousOutput1 = PreviousInput1 = Output = Input;
159 Input = InputNodes[0]->getDoubleValue();
161 if (DynamicFilter) CalculateDynamicFilters();
163 switch (FilterType) {
165 Output = (Input + PreviousInput1) * ca + PreviousOutput1 * cb;
168 Output = Input * ca + PreviousInput1 * cb + PreviousOutput1 * cc;
171 Output = Input * ca + PreviousInput1 * cb + PreviousInput2 * cc
172 - PreviousOutput1 * cd - PreviousOutput2 * ce;
175 Output = Input * ca - PreviousInput1 * ca + PreviousOutput1 * cb;
183 PreviousOutput2 = PreviousOutput1;
184 PreviousOutput1 = Output;
185 PreviousInput2 = PreviousInput1;
186 PreviousInput1 = Input;
213void FGFilter::Debug(
int from)
215 if (debug_lvl <= 0)
return;
219 cout <<
" INPUT: " << InputNodes[0]->GetName() << endl;
221 for (
int i=1; i < 7; i++) {
224 cout <<
" C[" << i <<
"]";
225 if (!C[i]->IsConstant()) cout <<
" is the value of property";
226 cout <<
": "<< C[i]->GetName() << endl;
229 for (
auto node: OutputNodes)
230 cout <<
" OUTPUT: " << node->getNameString() << endl;
233 if (debug_lvl & 2 ) {
234 if (from == 0) cout <<
"Instantiated: FGFilter" << endl;
235 if (from == 1) cout <<
"Destroyed: FGFilter" << endl;
237 if (debug_lvl & 4 ) {
239 if (debug_lvl & 8 ) {
241 if (debug_lvl & 16) {
243 if (debug_lvl & 64) {