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
FGOutput.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: FGOutput.cpp
4 Author: Jon Berndt
5 Date started: 12/02/98
6 Purpose: Manage output of sim parameters to file, stdout or socket
7 Called by: FGSimExec
8
9 ------------- Copyright (C) 1999 Jon S. Berndt (jon@jsbsim.org) -------------
10
11 This program is free software; you can redistribute it and/or modify it under
12 the terms of the GNU Lesser General Public License as published by the Free
13 Software Foundation; either version 2 of the License, or (at your option) any
14 later version.
15
16 This program is distributed in the hope that it will be useful, but WITHOUT
17 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
18 FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
19 details.
20
21 You should have received a copy of the GNU Lesser General Public License along
22 with this program; if not, write to the Free Software Foundation, Inc., 59
23 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
24
25 Further information about the GNU Lesser General Public License can also be
26 found on the world wide web at http://www.gnu.org.
27
28FUNCTIONAL DESCRIPTION
29--------------------------------------------------------------------------------
30This is the place where you create output routines to dump data for perusal
31later.
32
33HISTORY
34--------------------------------------------------------------------------------
3512/02/98 JSB Created
3611/09/07 HDW Added FlightGear Socket Interface
37
38%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
39INCLUDES
40%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
41
42#include "FGOutput.h"
43#include "input_output/FGOutputTextFile.h"
44#include "input_output/FGOutputFG.h"
45#include "input_output/FGXMLFileRead.h"
46#include "input_output/FGModelLoader.h"
47#include "input_output/string_utilities.h"
48
49using namespace std;
50
51namespace JSBSim {
52
53/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
54CLASS IMPLEMENTATION
55%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
56
57FGOutput::FGOutput(FGFDMExec* fdmex) : FGModel(fdmex)
58{
59 typedef int (FGOutput::*iOPV)(void) const;
60
61 Name = "FGOutput";
62 enabled = true;
63
64 PropertyManager->Tie("simulation/force-output", this, (iOPV)0, &FGOutput::ForceOutput);
65
66 Debug(0);
67}
68
69//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
70
71FGOutput::~FGOutput()
72{
73 for (auto output: OutputTypes)
74 delete output;
75
76 Debug(1);
77}
78
79//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
80
81bool FGOutput::InitModel(void)
82{
83 bool ret = false;
84
85 if (!FGModel::InitModel()) return false;
86
87 for (auto output: OutputTypes)
88 ret &= output->InitModel();
89
90 return ret;
91}
92
93//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
94
95bool FGOutput::Run(bool Holding)
96{
97 if (FDMExec->GetTrimStatus()) return true;
98 if (FGModel::Run(Holding)) return true;
99 if (Holding) return false;
100 if (!enabled) return true;
101
102 for (auto output: OutputTypes)
103 output->Run(Holding);
104
105 return false;
106}
107
108//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
109
110void FGOutput::Print(void)
111{
112 for (auto output: OutputTypes)
113 output->Print();
114}
115
116//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
117
118void FGOutput::SetStartNewOutput(void)
119{
120 for (auto output: OutputTypes)
121 output->SetStartNewOutput();
122}
123
124//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
125
126bool FGOutput::Toggle(int idx)
127{
128 if (idx >= (int)0 && idx < (int)OutputTypes.size())
129 return OutputTypes[idx]->Toggle();
130
131 return false;
132}
133
134//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
135
136void FGOutput::SetRateHz(double rate)
137{
138 for (auto output: OutputTypes)
139 output->SetRateHz(rate);
140}
141
142//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
143
144void FGOutput::ForceOutput(int idx)
145{
146 if (idx >= (int)0 && idx < (int)OutputTypes.size())
147 OutputTypes[idx]->Print();
148}
149
150//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
151
152bool FGOutput::SetOutputName(unsigned int idx, const std::string& name)
153{
154 if (idx >= OutputTypes.size()) return false;
155
156 OutputTypes[idx]->SetOutputName(name);
157 return true;
158}
159
160//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
161
162string FGOutput::GetOutputName(unsigned int idx) const
163{
164 string name;
165
166 if (idx < OutputTypes.size())
167 name = OutputTypes[idx]->GetOutputName();
168 return name;
169}
170
171//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
172
173bool FGOutput::SetDirectivesFile(const SGPath& fname)
174{
176 Element* document = XMLFile.LoadXMLDocument(fname);
177 if (!document) {
178 stringstream s;
179 s << "Could not read directive file: " << fname;
180 throw BaseException(s.str());
181 }
182
183 bool result = Load(document);
184 if (!result)
185 cerr << endl << "Aircraft output element has problems in file " << fname << endl;
186
187 return result;
188}
189
190//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
191
192bool FGOutput::Load(int subSystems, std::string protocol, std::string type,
193 std::string port, std::string name, double outRate,
194 std::vector<FGPropertyNode_ptr> & outputProperties)
195{
196 size_t idx = OutputTypes.size();
197 FGOutputType* Output = 0;
198
199 if (debug_lvl > 0) cout << endl << " Output data set: " << idx << endl;
200
201 type = to_upper(type);
202
203 if (type == "CSV") {
204 FGOutputTextFile* OutputTextFile = new FGOutputTextFile(FDMExec);
205 OutputTextFile->SetDelimiter(",");
206 Output = OutputTextFile;
207 } else if (type == "TABULAR") {
208 FGOutputTextFile* OutputTextFile = new FGOutputTextFile(FDMExec);
209 OutputTextFile->SetDelimiter("\t");
210 Output = OutputTextFile;
211 } else if (type == "SOCKET") {
212 Output = new FGOutputSocket(FDMExec);
213 name += ":" + port + "/" + protocol;
214 } else if (type == "FLIGHTGEAR") {
215 Output = new FGOutputFG(FDMExec);
216 name += ":" + port + "/" + protocol;
217 } else if (type == "TERMINAL") {
218 // Not done yet
219 } else if (type != string("NONE")) {
220 cerr << "Unknown type of output specified in config file" << endl;
221 }
222
223 if (!Output) return false;
224
225 Output->SetIdx(idx);
226 Output->SetOutputName(name);
227 Output->SetRateHz(outRate);
228 Output->SetSubSystems(subSystems);
229 Output->SetOutputProperties(outputProperties);
230
231 OutputTypes.push_back(Output);
232
233 Debug(2);
234 return true;
235}
236
237//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
238
239bool FGOutput::Load(Element* document, const SGPath& dir)
240{
241 // Optional path to use for included files
242 includePath = dir;
243
244 // Perform base class Pre-Load
245 if (!FGModel::Upload(document, false))
246 return false;
247
248 size_t idx = OutputTypes.size();
249 string type = document->GetAttributeValue("type");
250 FGOutputType* Output = 0;
251
252 if (debug_lvl > 0) cout << endl << " Output data set: " << idx << " " << endl;
253
254 type = to_upper(type);
255
256 if (type == "CSV") {
257 Output = new FGOutputTextFile(FDMExec);
258 } else if (type == "TABULAR") {
259 Output = new FGOutputTextFile(FDMExec);
260 } else if (type == "SOCKET") {
261 Output = new FGOutputSocket(FDMExec);
262 } else if (type == "FLIGHTGEAR") {
263 Output = new FGOutputFG(FDMExec);
264 } else if (type == "TERMINAL") {
265 // Not done yet
266 } else if (type != string("NONE")) {
267 cerr << "Unknown type of output specified in config file" << endl;
268 }
269
270 if (!Output) return false;
271
272 Output->SetIdx(idx);
273 Output->PreLoad(document, FDMExec);
274 Output->Load(document);
275 Output->PostLoad(document, FDMExec);
276
277 OutputTypes.push_back(Output);
278
279 Debug(2);
280 return true;
281}
282
283//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
284
285SGPath FGOutput::FindFullPathName(const SGPath& path) const
286{
287 // Check optional include path if set
288 if (!includePath.isNull()) {
289 SGPath name = CheckPathName(includePath, path);
290 if (!name.isNull()) return name;
291 }
292
293 return FGModel::FindFullPathName(path);
294}
295
296//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
297// The bitmasked value choices are as follows:
298// unset: In this case (the default) JSBSim would only print
299// out the normally expected messages, essentially echoing
300// the config files as they are read. If the environment
301// variable is not set, debug_lvl is set to 1 internally
302// 0: This requests JSBSim not to output any messages
303// whatsoever.
304// 1: This value explicity requests the normal JSBSim
305// startup messages
306// 2: This value asks for a message to be printed out when
307// a class is instantiated
308// 4: When this value is set, a message is displayed when a
309// FGModel object executes its Run() method
310// 8: When this value is set, various runtime state variables
311// are printed out periodically
312// 16: When set various parameters are sanity checked and
313// a message is printed out when they go out of bounds
314
315void FGOutput::Debug(int from)
316{
317 if (debug_lvl <= 0) return;
318
319 if (debug_lvl & 1) { // Standard console startup message output
320 if (from == 0) { // Constructor
321
322 }
323 if (from == 2) {
324 }
325 }
326 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
327 if (from == 0) cout << "Instantiated: FGOutput" << endl;
328 if (from == 1) cout << "Destroyed: FGOutput" << endl;
329 }
330 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
331 }
332 if (debug_lvl & 8 ) { // Runtime state variables
333 }
334 if (debug_lvl & 16) { // Sanity checking
335 }
336 if (debug_lvl & 64) {
337 if (from == 0) { // Constructor
338 }
339 }
340}
341}
std::string GetAttributeValue(const std::string &key)
Retrieves an attribute.
Implements the output to a FlightGear socket.
Definition FGOutputFG.h:64
Implements the output to a socket.
Implements the output to a human readable text file.
void SetDelimiter(const std::string &delim)
Set the delimiter.
Abstract class to provide functions generic to all the output directives.
bool Load(Element *el) override
Init the output directives from an XML file (implement the FGModel interface).
void SetRateHz(double rtHz)
Set the output rate for this output instances.
void SetIdx(unsigned int idx)
Set the idx for this output instance.
virtual void SetOutputName(const std::string &name)
Overwrites the name identifier under which the output will be logged.
void SetSubSystems(int subSystems)
Set the activated subsystems for this output instance.
void SetOutputProperties(std::vector< FGPropertyNode_ptr > &outputProperties)
Set the list of properties that should be output for this output instance.
This class is solely for the purpose of determining what type of file is given on the command line.
Definition JSBSim.cpp:155