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
FGCondition.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: FGCondition.cpp
4 Author: Jon S. Berndt
5 Date started: 1/2/2003
6
7 -------------- Copyright (C) 2003 Jon S. Berndt (jon@jsbsim.org) --------------
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
26HISTORY
27--------------------------------------------------------------------------------
28
29%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
30COMMENTS, REFERENCES, and NOTES
31%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
32
33%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
34INCLUDES
35%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
36
37#include <iostream>
38#include <cstdlib>
39#include <stdexcept>
40#include <assert.h>
41#include <array>
42#include <utility>
43
44#include "FGCondition.h"
45#include "FGPropertyValue.h"
46#include "input_output/FGXMLElement.h"
47#include "input_output/FGPropertyManager.h"
48#include "FGParameterValue.h"
49
50using namespace std;
51
52namespace JSBSim {
53
54/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
55CLASS IMPLEMENTATION
56%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
57
58// This constructor is called when tests are inside an element
59FGCondition::FGCondition(Element* element, std::shared_ptr<FGPropertyManager> PropertyManager)
60 : Logic(elUndef), TestParam1(nullptr), TestParam2(nullptr),
61 Comparison(ecUndef)
62{
63 string logic = element->GetAttributeValue("logic");
64 if (!logic.empty()) {
65 if (logic == "OR") Logic = eOR;
66 else if (logic == "AND") Logic = eAND;
67 else { // error
68 throw BaseException("FGCondition: unrecognized LOGIC token:'" + logic + "'");
69 }
70 } else {
71 Logic = eAND; // default
72 }
73
74 assert(Logic != elUndef);
75
76 for (unsigned int i=0; i<element->GetNumDataLines(); i++) {
77 string data = element->GetDataLine(i);
78 conditions.push_back(make_shared<FGCondition>(data, PropertyManager, element));
79 }
80
81 Element* condition_element = element->GetElement();
82 const string& elName = element->GetName();
83
84 while (condition_element) {
85 string tagName = condition_element->GetName();
86
87 if (tagName != elName) {
88 throw BaseException("FGCondition: unrecognized TAG:'" + tagName + "' in the condition statement.");
89 }
90
91 conditions.push_back(make_shared<FGCondition>(condition_element, PropertyManager));
92 condition_element = element->GetNextElement();
93 }
94
95 if (conditions.empty()) throw BaseException("Empty conditional");
96
97 Debug(0);
98}
99
100//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
101// This constructor is called when there are no nested test groups inside the
102// condition
103
104FGCondition::FGCondition(const string& test, std::shared_ptr<FGPropertyManager> PropertyManager,
105 Element* el)
106 : Logic(elUndef), TestParam1(nullptr), TestParam2(nullptr),
107 Comparison(ecUndef)
108{
109 static constexpr array<pair<const char*, enum eComparison>, 18> mComparison {{
110 {"!=", eNE},
111 {"<", eLT},
112 {"<=", eLE},
113 {"==", eEQ},
114 {">", eGT},
115 {">=", eGE},
116 {"EQ", eEQ},
117 {"GE", eGE},
118 {"GT", eGT},
119 {"LE", eLE},
120 {"LT", eLT},
121 {"NE", eNE},
122 {"eq", eEQ},
123 {"ge", eGE},
124 {"gt", eGT},
125 {"le", eLE},
126 {"lt", eLT},
127 {"ne", eNE},
128 }};
129
130 vector<string> test_strings = split(test, ' ');
131
132 if (test_strings.size() == 3) {
133 TestParam1 = new FGPropertyValue(test_strings[0], PropertyManager, el);
134 conditional = test_strings[1];
135 TestParam2 = new FGParameterValue(test_strings[2], PropertyManager, el);
136 } else {
137 ostringstream s;
138 s << " Conditional test is invalid: \"" << test
139 << "\" has " << test_strings.size() << " elements in the "
140 << "test condition.\n";
141 throw BaseException(s.str());
142 }
143
144 assert(Comparison == ecUndef);
145 for(auto& elm: mComparison) {
146 if (conditional == elm.first) {
147 Comparison = elm.second;
148 break;
149 }
150 }
151
152 if (Comparison == ecUndef) {
153 throw BaseException("FGCondition: Comparison operator: \""+conditional
154 +"\" does not exist. Please check the conditional.");
155 }
156}
157
158//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
159
160bool FGCondition::Evaluate(void) const
161{
162 bool pass = false;
163
164 if (!TestParam1) {
165
166 if (Logic == eAND) {
167
168 pass = true;
169 for (auto& cond: conditions) {
170 if (!cond->Evaluate()) pass = false;
171 }
172
173 } else { // Logic must be eOR
174
175 pass = false;
176 for (auto& cond: conditions) {
177 if (cond->Evaluate()) pass = true;
178 }
179
180 }
181
182 } else {
183 double value1 = TestParam1->GetValue();
184 double value2 = TestParam2->GetValue();
185
186 switch (Comparison) {
187 case eEQ:
188 pass = value1 == value2;
189 break;
190 case eNE:
191 pass = value1 != value2;
192 break;
193 case eGT:
194 pass = value1 > value2;
195 break;
196 case eGE:
197 pass = value1 >= value2;
198 break;
199 case eLT:
200 pass = value1 < value2;
201 break;
202 case eLE:
203 pass = value1 <= value2;
204 break;
205 default:
206 assert(false); // Should not be reached
207 break;
208 }
209 }
210
211 return pass;
212}
213
214//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
215
216void FGCondition::PrintCondition(string indent) const
217{
218 string scratch;
219
220 if (!conditions.empty()) {
221
222 switch(Logic) {
223 case (elUndef):
224 scratch = " UNSET";
225 cerr << "unset logic for test condition" << endl;
226 break;
227 case (eAND):
228 scratch = indent + "if all of the following are true: {";
229 break;
230 case (eOR):
231 scratch = indent + "if any of the following are true: {";
232 break;
233 default:
234 scratch = " UNKNOWN";
235 cerr << "Unknown logic for test condition" << endl;
236 }
237 cout << scratch << endl;
238
239 for (auto& cond: conditions) {
240 cond->PrintCondition(indent + " ");
241 cout << endl;
242 }
243
244 cout << indent << "}";
245
246 } else {
247 cout << indent << TestParam1->GetName() << " " << conditional
248 << " " << TestParam2->GetName();
249 }
250}
251
252//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
253// The bitmasked value choices are as follows:
254// unset: In this case (the default) JSBSim would only print
255// out the normally expected messages, essentially echoing
256// the config files as they are read. If the environment
257// variable is not set, debug_lvl is set to 1 internally
258// 0: This requests JSBSim not to output any messages
259// whatsoever.
260// 1: This value explicity requests the normal JSBSim
261// startup messages
262// 2: This value asks for a message to be printed out when
263// a class is instantiated
264// 4: When this value is set, a message is displayed when a
265// FGModel object executes its Run() method
266// 8: When this value is set, various runtime state variables
267// are printed out periodically
268// 16: When set various parameters are sanity checked and
269// a message is printed out when they go out of bounds
270
271void FGCondition::Debug(int from)
272{
273 if (debug_lvl <= 0) return;
274
275 if (debug_lvl & 1) { // Standard console startup message output
276 if (from == 0) { // Constructor
277
278 }
279 }
280 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
281 if (from == 0) cout << "Instantiated: FGCondition" << endl;
282 if (from == 1) cout << "Destroyed: FGCondition" << endl;
283 }
284 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
285 }
286 if (debug_lvl & 8 ) { // Runtime state variables
287 }
288 if (debug_lvl & 16) { // Sanity checking
289 }
290 if (debug_lvl & 64) {
291 if (from == 0) { // Constructor
292 }
293 }
294}
295
296} //namespace JSBSim