40#include "FGFCSComponent.h"
41#include "models/FGFCS.h"
42#include "math/FGParameterValue.h"
54 Input = Output = delay_time = 0.0;
57 clip = cyclic_clip =
false;
58 dt = fcs->GetChannelDeltaT();
60 auto PropertyManager = fcs->GetPropertyManager();
61 if (element->
GetName() ==
string(
"lag_filter")) {
63 }
else if (element->
GetName() ==
string(
"lead_lag_filter")) {
64 Type =
"LEAD_LAG_FILTER";
65 }
else if (element->
GetName() ==
string(
"washout_filter")) {
66 Type =
"WASHOUT_FILTER";
67 }
else if (element->
GetName() ==
string(
"second_order_filter")) {
68 Type =
"SECOND_ORDER_FILTER";
69 }
else if (element->
GetName() ==
string(
"integrator")) {
71 }
else if (element->
GetName() ==
string(
"summer")) {
73 }
else if (element->
GetName() ==
string(
"pure_gain")) {
75 }
else if (element->
GetName() ==
string(
"scheduled_gain")) {
76 Type =
"SCHEDULED_GAIN";
77 }
else if (element->
GetName() ==
string(
"aerosurface_scale")) {
78 Type =
"AEROSURFACE_SCALE";
79 }
else if (element->
GetName() ==
string(
"switch")) {
81 }
else if (element->
GetName() ==
string(
"kinematic")) {
83 }
else if (element->
GetName() ==
string(
"deadband")) {
85 }
else if (element->
GetName() ==
string(
"fcs_function")) {
86 Type =
"FCS_FUNCTION";
87 }
else if (element->
GetName() ==
string(
"pid")) {
89 }
else if (element->
GetName() ==
string(
"sensor")) {
91 }
else if (element->
GetName() ==
string(
"accelerometer")) {
92 Type =
"ACCELEROMETER";
93 }
else if (element->
GetName() ==
string(
"magnetometer")) {
94 Type =
"MAGNETOMETER";
95 }
else if (element->
GetName() ==
string(
"gyro")) {
97 }
else if (element->
GetName() ==
string(
"actuator")) {
99 }
else if (element->
GetName() ==
string(
"waypoint_heading")) {
100 Type =
"WAYPOINT_HEADING";
101 }
else if (element->
GetName() ==
string(
"waypoint_distance")) {
102 Type =
"WAYPOINT_DISTANCE";
103 }
else if (element->
GetName() ==
string(
"angle")) {
105 }
else if (element->
GetName() ==
string(
"distributor")) {
106 Type =
"DISTRIBUTOR";
114 while (init_element) {
116 PropertyManager, init_element));
121 while (input_element) {
123 PropertyManager, input_element));
131 bool node_exists = PropertyManager->HasNode(output_node_name);
134 cerr << out_elem->
ReadFrom() <<
" Unable to process property: "
135 << output_node_name << endl;
136 throw(
string(
"Invalid output property name in flight control definition"));
138 OutputNodes.push_back(OutputNode);
144 OutputNode->setDoubleValue(Output);
152 delay_time = delayParam.GetValue();
154 if (delayType.length() > 0) {
155 if (delayType ==
"time") {
156 delay = (
unsigned int)(delay_time / dt);
157 }
else if (delayType ==
"frames") {
158 delay = (
unsigned int)delay_time;
160 cerr <<
"Unallowed delay type" << endl;
163 delay = (
unsigned int)(delay_time / dt);
165 output_array.resize(delay);
166 for (
unsigned int i=0; i<delay; i++) output_array[i] = 0.0;
174 <<
"Element <min> is missing, <clipto> is ignored." << endl;
183 <<
"Element <max> is missing, <clipto> is ignored." << endl;
208void FGFCSComponent::ResetPastStates(
void)
211 for (
auto &elm: output_array)
217void FGFCSComponent::CheckInputNodes(
size_t MinNodes,
size_t MaxNodes, Element* el)
219 size_t num = InputNodes.size();
221 if (num < MinNodes) {
222 cerr << el->ReadFrom()
223 <<
" Not enough <input> nodes are provided" << endl
224 <<
" Expecting " << MinNodes <<
" while " << num
225 <<
" are provided." << endl;
226 throw(
"Some inputs are missing.");
229 if (num > MaxNodes) {
230 cerr << el->ReadFrom()
231 <<
" Too many <input> nodes are provided" << endl
232 <<
" Expecting " << MaxNodes <<
" while " << num
233 <<
" are provided." << endl
234 <<
" The last " << num-MaxNodes <<
" input nodes will be ignored."
241void FGFCSComponent::SetOutput(
void)
243 for (
auto node: OutputNodes)
244 node->setDoubleValue(Output);
249void FGFCSComponent::Delay(
void)
251 if (fcs->GetTrimStatus()) {
254 std::fill(output_array.begin(), output_array.end(), Output);
257 output_array[index] = Output;
258 if ((
unsigned int)index == delay-1) index = 0;
260 Output = output_array[index];
266void FGFCSComponent::Clip(
void)
269 double vmin = ClipMin->GetValue();
270 double vmax = ClipMax->GetValue();
271 double range = vmax - vmin;
274 cerr <<
"Trying to clip with a max value (" << vmax <<
") from "
275 << ClipMax->GetName() <<
" lower than the min value (" << vmin
276 <<
") from " << ClipMin->GetName() <<
"." << endl
277 <<
"Clipping is ignored." << endl;
281 if (cyclic_clip && range != 0.0) {
282 double value = Output - vmin;
283 Output = fmod(value, range) + vmin;
301void FGFCSComponent::bind(Element* el, FGPropertyManager* PropertyManager)
304 if (Name.find(
"/") == string::npos)
305 tmp =
"fcs/" + PropertyManager->mkPropertyName(Name,
true);
309 bool node_exists = PropertyManager->HasNode(tmp);
310 FGPropertyNode* node = PropertyManager->GetNode(tmp,
true);
313 OutputNodes.push_back(node);
319 node->setDoubleValue(Output);
322 cerr << el->ReadFrom()
323 <<
"Could not get or create property " << tmp << endl;
346void FGFCSComponent::Debug(
int from)
348 if (debug_lvl <= 0)
return;
352 cout << endl <<
" Loading Component \"" << Name
353 <<
"\" of type: " << Type << endl;
356 cout <<
" Minimum limit: " << ClipMin->GetName() << endl;
357 cout <<
" Maximum limit: " << ClipMax->GetName() << endl;
359 if (delay > 0) cout <<
" Frame delay: " << delay
360 <<
" frames (" << delay*dt <<
" sec)" << endl;
363 if (debug_lvl & 2 ) {
364 if (from == 0) cout <<
"Instantiated: FGFCSComponent" << endl;
365 if (from == 1) cout <<
"Destroyed: FGFCSComponent" << endl;
367 if (debug_lvl & 4 ) {
369 if (debug_lvl & 8 ) {
371 if (debug_lvl & 16) {
373 if (debug_lvl & 64) {
Element * FindElement(const std::string &el="")
Searches for a specified element.
const std::string & GetName(void) const
Retrieves the element name.
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.
std::string ReadFrom(void) const
Return a string that contains a description of the location where the current XML element was read fr...
Element * FindNextElement(const std::string &el="")
Searches for the next element as specified.
virtual ~FGFCSComponent()
Destructor.
FGFCSComponent(FGFCS *fcs, Element *el)
Constructor.
Encapsulates the Flight Control System (FCS) functionality.
static constexpr double Constrain(double min, double value, double max)
Constrain a value between a minimum and a maximum value.
Represents a either a real value or a property value.
Class wrapper for property handling.
FGPropertyNode * GetNode(const std::string &path, bool create=false)
Get a property node.
Represents a property value which can use late binding.