1
0
Fork 0

Add test-data logger

This commit is contained in:
James Turner 2020-04-25 18:03:39 +01:00
parent e76787fa62
commit e8bf4220a8
4 changed files with 264 additions and 15 deletions

View file

@ -6,6 +6,7 @@ set(TESTSUITE_SOURCES
${CMAKE_CURRENT_SOURCE_DIR}/scene_graph.cxx
${CMAKE_CURRENT_SOURCE_DIR}/TestPilot.cxx
${CMAKE_CURRENT_SOURCE_DIR}/NasalUnitTesting_TestSuite.cxx
${CMAKE_CURRENT_SOURCE_DIR}/TestDataLogger.cxx
PARENT_SCOPE
)
@ -17,5 +18,6 @@ set(TESTSUITE_HEADERS
${CMAKE_CURRENT_SOURCE_DIR}/PrivateAccessorFDM.hxx
${CMAKE_CURRENT_SOURCE_DIR}/scene_graph.hxx
${CMAKE_CURRENT_SOURCE_DIR}/TestPilot.hxx
${CMAKE_CURRENT_SOURCE_DIR}/TestDataLogger.hxx
PARENT_SCOPE
)

View file

@ -0,0 +1,170 @@
/*
* Copyright (C) 2020 James Turner
*
* This file is part of the program FlightGear.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include "TestDataLogger.hxx"
#include <algorithm>
#include <cmath>
#include <limits>
#include <vector>
#include <simgear/io/iostreams/sgstream.hxx>
#include "Main/globals.hxx"
namespace FGTestApi {
using DoubleVec = std::vector<double>;
static std::unique_ptr<DataLogger> static_instance;
class DataLogger::DataLoggerPrivate
{
public:
sg_ofstream _stream;
struct SampleInfo {
int column;
std::string name;
// range / units info, later
SGPropertyNode_ptr property;
};
double _currentTimeBase;
std::vector<SampleInfo> _samples;
DoubleVec _openRow;
bool _didHeader = false;
void writeCurrentRow()
{
if (!_didHeader) {
writeHeader();
_didHeader = true;
}
// capture property values into the open row data
std::for_each(_samples.begin(), _samples.end(), [this](const SampleInfo& info) {
if (info.property) {
_openRow[info.column] = info.property->getDoubleValue();
}
});
// write time base
_stream << globals->get_sim_time_sec() << ",";
for (const auto v : _openRow) {
if (std::isnan(v)) {
_stream << ","; // skip this data point
} else {
_stream << v << ",";
}
}
_stream << "\n";
std::fill(_openRow.begin(), _openRow.end(), std::numeric_limits<double>::quiet_NaN());
}
void writeHeader()
{
_stream << "sim-time, ";
std::for_each(_samples.begin(), _samples.end(), [this](const SampleInfo& info) {
_stream << info.name << ", ";
});
_stream << "\n";
}
};
DataLogger::DataLogger()
{
d.reset(new DataLoggerPrivate);
}
DataLogger::~DataLogger()
{
d->_stream.close();
}
bool DataLogger::isActive()
{
return static_instance != nullptr;
}
DataLogger* DataLogger::instance()
{
if (!static_instance) {
static_instance.reset(new DataLogger);
}
return static_instance.get();
}
void DataLogger::initTest(const std::string& testName)
{
d->_stream = sg_ofstream(testName + "_trace.csv");
}
void DataLogger::tearDown()
{
if (static_instance) {
static_instance.reset();
}
}
void DataLogger::writeRecord()
{
d->writeCurrentRow();
}
void DataLogger::recordProperty(const std::string& name, SGPropertyNode_ptr prop)
{
int index = static_cast<int>(d->_samples.size());
DataLoggerPrivate::SampleInfo info{index, name, prop};
d->_samples.push_back(info);
if (d->_openRow.size() <= index) {
d->_openRow.resize(index + 1, std::numeric_limits<double>::quiet_NaN());
}
}
void DataLogger::recordSamplePoint(const std::string& name, double value)
{
auto it = std::find_if(d->_samples.begin(), d->_samples.end(), [&name](const DataLoggerPrivate::SampleInfo& sample) {
return name == sample.name;
});
int index = 0;
if (it == d->_samples.end()) {
index = static_cast<int>(d->_samples.size());
DataLoggerPrivate::SampleInfo info{index, name};
d->_samples.push_back(info);
} else {
index = it->column;
}
// grow _openRow as required
if (d->_openRow.size() <= index) {
d->_openRow.resize(index + 1, std::numeric_limits<double>::quiet_NaN());
}
d->_openRow[index] = value;
}
} // namespace FGTestApi

View file

@ -0,0 +1,57 @@
/*
* Copyright (C) 2020 James Turner
*
* This file is part of the program FlightGear.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#pragma once
#include <memory>
#include <string>
#include <simgear/props/props.hxx>
namespace FGTestApi {
class DataLogger
{
public:
~DataLogger();
static DataLogger* instance();
static bool isActive();
void initTest(const std::string& testName);
void recordProperty(const std::string& name, SGPropertyNode_ptr prop);
void setUp();
void tearDown();
void recordSamplePoint(const std::string& name, double value);
void writeRecord();
private:
DataLogger();
class DataLoggerPrivate;
std::unique_ptr<DataLoggerPrivate> d;
};
} // namespace FGTestApi

View file

@ -2,6 +2,7 @@
#include "test_suite/dataStore.hxx"
#include "TestDataLogger.hxx"
#include "testGlobals.hxx"
#include <simgear/io/iostreams/sgstream.hxx>
@ -254,15 +255,30 @@ void setPositionAndStabilise(const SGGeod& g)
void runForTime(double t)
{
int ticks = t * 120.0;
const int tickHz = 30;
const double tickDuration = 1.0 / tickHz;
int ticks = static_cast<int>(t * tickHz);
assert(ticks > 0);
const double dt = 1 / 120.0;
const int logInterval = 0.5 * tickHz;
int nextLog = 0;
for (int t = 0; t < ticks; ++t) {
globals->inc_sim_time_sec(dt);
globals->get_subsystem_mgr()->update(dt);
if (global_loggingToKML) {
logCoordinate(globals->get_aircraft_position());
globals->inc_sim_time_sec(tickDuration);
globals->get_subsystem_mgr()->update(tickDuration);
if (nextLog == 0) {
if (global_loggingToKML) {
logCoordinate(globals->get_aircraft_position());
}
if (DataLogger::isActive()) {
DataLogger::instance()->writeRecord();
}
nextLog = logInterval;
} else {
nextLog--;
}
}
}
@ -274,20 +290,24 @@ bool runForTimeWithCheck(double t, RunCheck check)
int ticks = static_cast<int>(t * tickHz);
assert(ticks > 0);
const int logInterval = 2 * tickHz; // every two seconds
const int logInterval = 0.5 * tickHz;
int nextLog = 0;
for (int t = 0; t < ticks; ++t) {
globals->inc_sim_time_sec(tickDuration);
globals->get_subsystem_mgr()->update(tickDuration);
if (global_loggingToKML) {
if (nextLog == 0) {
if (nextLog == 0) {
if (global_loggingToKML) {
logCoordinate(globals->get_aircraft_position());
nextLog = logInterval;
} else {
nextLog--;
}
if (DataLogger::isActive()) {
DataLogger::instance()->writeRecord();
}
nextLog = logInterval;
} else {
nextLog--;
}
bool done = check();