503 lines
14 KiB
C
503 lines
14 KiB
C
/*---------------------------------------------------------------------------*\
|
|
========= |
|
|
\\ / F ield | OpenFOAM: The Open Source CFD Toolbox
|
|
\\ / O peration |
|
|
\\ / A nd | www.openfoam.com
|
|
\\/ M anipulation |
|
|
-------------------------------------------------------------------------------
|
|
Copyright (C) 2011-2016 OpenFOAM Foundation
|
|
Copyright (C) 2015-2025 OpenCFD Ltd.
|
|
-------------------------------------------------------------------------------
|
|
License
|
|
This file is part of OpenFOAM.
|
|
|
|
OpenFOAM 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 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
OpenFOAM 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 OpenFOAM. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
\*---------------------------------------------------------------------------*/
|
|
|
|
#include "fieldStatistics.H"
|
|
#include "fieldTypes.H"
|
|
#include "addToRunTimeSelectionTable.H"
|
|
|
|
// * * * * * * * * * * * * * * Static Data Members * * * * * * * * * * * * * //
|
|
|
|
namespace Foam
|
|
{
|
|
namespace functionObjects
|
|
{
|
|
defineTypeNameAndDebug(fieldStatistics, 0);
|
|
addToRunTimeSelectionTable(functionObject, fieldStatistics, dictionary);
|
|
}
|
|
}
|
|
|
|
const Foam::Enum
|
|
<
|
|
Foam::functionObjects::fieldStatistics::modeType
|
|
>
|
|
Foam::functionObjects::fieldStatistics::modeTypeNames_
|
|
({
|
|
{ modeType::mdMag, "magnitude" },
|
|
{ modeType::mdCmpt, "component" },
|
|
});
|
|
|
|
const Foam::Enum
|
|
<
|
|
Foam::functionObjects::fieldStatistics::meanType
|
|
>
|
|
Foam::functionObjects::fieldStatistics::meanTypeNames_
|
|
({
|
|
{ meanType::ARITHMETIC, "arithmetic" },
|
|
{ meanType::VOLUMETRIC, "volumetric" },
|
|
});
|
|
|
|
|
|
// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * //
|
|
|
|
// Implementation
|
|
#include "fieldStatisticsImpl.cxx"
|
|
|
|
|
|
// * * * * * * * * * * * * * Private Member Functions * * * * * * * * * * * //
|
|
|
|
Foam::functionObjects::fieldStatistics::statistic
|
|
Foam::functionObjects::fieldStatistics::createStatistic
|
|
(
|
|
const word& statName,
|
|
const modeType mode
|
|
)
|
|
{
|
|
statistic m;
|
|
m.name_ = statName;
|
|
|
|
m.calc = [this, statName, mode](variantInput input) -> variantOutput
|
|
{
|
|
return std::visit
|
|
(
|
|
[this, statName, mode](auto&& arg) -> variantOutput
|
|
{
|
|
using T = std::decay_t<decltype(arg)>;
|
|
using BaseType = typename T::value_type;
|
|
|
|
if (statName == "min")
|
|
{
|
|
if (mode == mdMag) return mag(calcMin<BaseType>(arg));
|
|
else return calcMin<BaseType>(arg);
|
|
}
|
|
else if (statName == "max")
|
|
{
|
|
if (mode == mdMag) return mag(calcMax<BaseType>(arg));
|
|
else return calcMax<BaseType>(arg);
|
|
}
|
|
else if (statName == "mean")
|
|
{
|
|
if (mode == mdMag) return mag(calcMean<BaseType>(arg));
|
|
else return calcMean<BaseType>(arg);
|
|
}
|
|
else if (statName == "variance")
|
|
{
|
|
if (mode == mdMag) return mag(calcVariance<BaseType>(arg));
|
|
else return calcVariance<BaseType>(arg);
|
|
}
|
|
else return scalar{}; // Default case (for compiler)
|
|
},
|
|
input
|
|
);
|
|
};
|
|
|
|
return m;
|
|
}
|
|
|
|
|
|
void Foam::functionObjects::fieldStatistics::writeFileHeader
|
|
(
|
|
Ostream& os,
|
|
const word& fieldName
|
|
)
|
|
{
|
|
writeHeader(os, word("Field Statistics: " + fieldName));
|
|
writeCommented(os, "Time");
|
|
|
|
// Number of input statistics (i.e., statistics_) should be the same with
|
|
// that of output statistics (i.e., results_). However, for consistency,
|
|
// the output file columns are based on output statistics.
|
|
const auto& result = results_(fieldName);
|
|
|
|
for (const auto& iter : result.csorted())
|
|
{
|
|
const word& name = iter.key();
|
|
writeTabbed(os, name);
|
|
}
|
|
os << endl;
|
|
}
|
|
|
|
|
|
void Foam::functionObjects::fieldStatistics::writeStatData()
|
|
{
|
|
for (const word& fieldName : fieldSet_.selectionNames())
|
|
{
|
|
const auto& results = results_(fieldName);
|
|
|
|
if (!results.size()) break;
|
|
|
|
OFstream& file = *filePtrs_(fieldName);
|
|
|
|
writeCurrentTime(file);
|
|
|
|
for (const auto& iter : results.csorted())
|
|
{
|
|
const variantOutput& value = iter.val();
|
|
|
|
std::visit
|
|
(
|
|
[&file](const auto& v)
|
|
{
|
|
if constexpr
|
|
(
|
|
is_vectorspace_v<std::decay_t<decltype(v)>>
|
|
)
|
|
{
|
|
for (const auto& val : v) file<< token::TAB << val;
|
|
}
|
|
else
|
|
{
|
|
file<< token::TAB << v;
|
|
}
|
|
},
|
|
value
|
|
);
|
|
}
|
|
file<< nl;
|
|
}
|
|
}
|
|
|
|
|
|
void Foam::functionObjects::fieldStatistics::logStatData()
|
|
{
|
|
for (const word& fieldName : fieldSet_.selectionNames())
|
|
{
|
|
const auto& results = results_(fieldName);
|
|
|
|
if (!results.size()) break;
|
|
|
|
Info<< nl << " Field " << fieldName << nl;
|
|
|
|
for (const auto& iter : results.csorted())
|
|
{
|
|
const word& name = iter.key();
|
|
const variantOutput& value = iter.val();
|
|
|
|
Info<< " " << name << " ";
|
|
std::visit
|
|
(
|
|
[](const auto& v)
|
|
{
|
|
if constexpr
|
|
(
|
|
is_vectorspace_v<std::decay_t<decltype(v)>>
|
|
)
|
|
{
|
|
for (const auto& val : v) Info<< val << " ";
|
|
}
|
|
else
|
|
{
|
|
Info<< v;
|
|
}
|
|
},
|
|
value
|
|
);
|
|
Info<< nl;
|
|
}
|
|
}
|
|
Info<< endl;
|
|
}
|
|
|
|
|
|
void Foam::functionObjects::fieldStatistics::writeExtremaFileHeader
|
|
(
|
|
Ostream& os,
|
|
const word& fieldName
|
|
)
|
|
{
|
|
writeHeader(os, word("Field Extrema Data: " + fieldName));
|
|
writeCommented(os, "Time");
|
|
writeTabbed(os, "min");
|
|
writeTabbed(os, "min_procID");
|
|
writeTabbed(os, "min_cellID");
|
|
writeTabbed(os, "min_position");
|
|
writeTabbed(os, "max");
|
|
writeTabbed(os, "max_procID");
|
|
writeTabbed(os, "max_cellID");
|
|
writeTabbed(os, "max_position");
|
|
os << endl;
|
|
}
|
|
|
|
|
|
void Foam::functionObjects::fieldStatistics::writeExtremaData()
|
|
{
|
|
for (const word& fieldName : fieldSet_.selectionNames())
|
|
{
|
|
const auto& min = extremaMetaData_(fieldName).first();
|
|
const auto& max = extremaMetaData_(fieldName).second();
|
|
|
|
OFstream& efile = *extremaFilePtrs_(fieldName);
|
|
|
|
writeCurrentTime(efile);
|
|
|
|
efile<< token::TAB;
|
|
|
|
std::visit([&efile](const auto& v){ efile<< v; }, min.value_);
|
|
|
|
efile<< token::TAB
|
|
<< min.procID_ << token::TAB
|
|
<< min.cellID_ << token::TAB
|
|
<< min.position_ << token::TAB;
|
|
|
|
std::visit([&efile](const auto& v){ efile<< v; }, max.value_);
|
|
|
|
efile<< token::TAB
|
|
<< max.procID_ << token::TAB
|
|
<< max.cellID_ << token::TAB
|
|
<< max.position_ << nl;
|
|
}
|
|
}
|
|
|
|
|
|
void Foam::functionObjects::fieldStatistics::logExtremaData()
|
|
{
|
|
for (const word& fieldName : fieldSet_.selectionNames())
|
|
{
|
|
const auto& min = extremaMetaData_(fieldName).first();
|
|
const auto& max = extremaMetaData_(fieldName).second();
|
|
|
|
const word outputName =
|
|
(mode_ == mdMag)
|
|
? word("mag(" + fieldName + ")")
|
|
: fieldName;
|
|
|
|
std::visit
|
|
(
|
|
[outputName](const auto& v)
|
|
{
|
|
Info<< " min(" << outputName << ") = " << v;
|
|
},
|
|
min.value_
|
|
);
|
|
|
|
Info<< " in cell " << min.cellID_
|
|
<< " at location " << min.position_
|
|
<< " on processor " << min.procID_;
|
|
|
|
std::visit
|
|
(
|
|
[outputName](const auto& v)
|
|
{
|
|
Info<< nl << " max(" << outputName << ") = " << v;
|
|
},
|
|
max.value_
|
|
);
|
|
|
|
Info<< " in cell " << max.cellID_
|
|
<< " at location " << max.position_
|
|
<< " on processor " << max.procID_ << endl;
|
|
}
|
|
}
|
|
|
|
|
|
// * * * * * * * * * * * * * * * * Constructors * * * * * * * * * * * * * * //
|
|
|
|
Foam::functionObjects::fieldStatistics::fieldStatistics
|
|
(
|
|
const word& name,
|
|
const Time& runTime,
|
|
const dictionary& dict
|
|
)
|
|
:
|
|
fvMeshFunctionObject(name, runTime, dict),
|
|
writeFile(mesh_, name, typeName, dict),
|
|
fieldSet_(mesh_)
|
|
{
|
|
read(dict);
|
|
}
|
|
|
|
|
|
// * * * * * * * * * * * * * * * Member Functions * * * * * * * * * * * * * //
|
|
|
|
bool Foam::functionObjects::fieldStatistics::read(const dictionary& dict)
|
|
{
|
|
if (!(fvMeshFunctionObject::read(dict) && writeFile::read(dict)))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
internal_ = dict.getOrDefault("internal", false);
|
|
extrema_ = dict.getOrDefault("extrema", false);
|
|
|
|
mode_ = modeTypeNames_.getOrDefault("mode", dict, modeType::mdMag);
|
|
mean_ = meanTypeNames_.getOrDefault("mean", dict, meanType::ARITHMETIC);
|
|
|
|
// Reset and reprepare the input field names
|
|
fieldSet_.clear();
|
|
fieldSet_.read(dict);
|
|
fieldSet_.updateSelection();
|
|
|
|
// Reset and reprepare the input statistics
|
|
statistics_.clear();
|
|
const wordHashSet stats(dict.get<wordHashSet>("statistics"));
|
|
for (const word& m : stats)
|
|
{
|
|
statistics_.insert(m, createStatistic(m, mode_));
|
|
}
|
|
|
|
// Reset the output-statistics container
|
|
results_.clear();
|
|
|
|
// Reset and reprepare the output statistics
|
|
extremaMetaData_.clear();
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool Foam::functionObjects::fieldStatistics::execute()
|
|
{
|
|
// Calculate the specified statistics for the specified fields
|
|
for (const word& fieldName : fieldSet_.selectionNames())
|
|
{
|
|
const bool ok =
|
|
(
|
|
calcStat<scalar>(fieldName)
|
|
|| calcStat<vector>(fieldName)
|
|
|| calcStat<symmTensor>(fieldName)
|
|
|| calcStat<tensor>(fieldName)
|
|
);
|
|
|
|
if (!ok)
|
|
{
|
|
WarningInFunction
|
|
<< "Unable to find field " << fieldName << endl;
|
|
}
|
|
}
|
|
|
|
// Store the statistical results into the state containers
|
|
for (const word& fieldName : fieldSet_.selectionNames())
|
|
{
|
|
const auto& results = results_(fieldName);
|
|
|
|
for (const auto& iter : results.csorted())
|
|
{
|
|
const word& name = iter.key();
|
|
const variantOutput& value = iter.val();
|
|
|
|
const word variableName(fieldName + "_" + name);
|
|
|
|
std::visit
|
|
(
|
|
[this, variableName](const auto& v)
|
|
{
|
|
this->setResult(variableName, v);
|
|
},
|
|
value
|
|
);
|
|
}
|
|
|
|
if (extrema_)
|
|
{
|
|
const auto& min = extremaMetaData_(fieldName).first();
|
|
std::visit
|
|
(
|
|
[this, fieldName](const auto& v)
|
|
{
|
|
this->setResult(word(fieldName + "_min"), v);
|
|
},
|
|
min.value_
|
|
);
|
|
this->setResult(word(fieldName + "_min_procID"), min.procID_);
|
|
this->setResult(word(fieldName + "_min_cellID"), min.cellID_);
|
|
this->setResult(word(fieldName + "_min_position"), min.position_);
|
|
|
|
const auto& max = extremaMetaData_(fieldName).second();
|
|
std::visit
|
|
(
|
|
[this, fieldName](const auto& v)
|
|
{
|
|
this->setResult(word(fieldName + "_max"), v);
|
|
},
|
|
max.value_
|
|
);
|
|
this->setResult(word(fieldName + "_max_procID"), max.procID_);
|
|
this->setResult(word(fieldName + "_max_cellID"), max.cellID_);
|
|
this->setResult(word(fieldName + "_max_position"), max.position_);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool Foam::functionObjects::fieldStatistics::write()
|
|
{
|
|
Log << type() << " " << name() << " write:" << endl;
|
|
|
|
// Create an output file per field
|
|
if (writeToFile() && !writtenHeader_)
|
|
{
|
|
for (const word& fieldName : fieldSet_.selectionNames())
|
|
{
|
|
filePtrs_.set(fieldName, newFileAtStartTime(fieldName));
|
|
|
|
OFstream& file = *filePtrs_(fieldName);
|
|
|
|
writeFileHeader(file, fieldName);
|
|
}
|
|
|
|
if (extrema_)
|
|
{
|
|
for (const word& fieldName : fieldSet_.selectionNames())
|
|
{
|
|
extremaFilePtrs_.set
|
|
(
|
|
fieldName,
|
|
newFileAtStartTime(word("fieldMinMax_" + fieldName))
|
|
);
|
|
|
|
OFstream& file = *extremaFilePtrs_(fieldName);
|
|
writeExtremaFileHeader(file, fieldName);
|
|
}
|
|
}
|
|
|
|
writtenHeader_ = true;
|
|
}
|
|
|
|
// Write the statistical results to the output file if requested
|
|
if (writeToFile())
|
|
{
|
|
if (extrema_) writeExtremaData();
|
|
|
|
writeStatData();
|
|
}
|
|
|
|
// Print the statistical results to the standard stream if requested
|
|
if (log)
|
|
{
|
|
if (extrema_) logExtremaData();
|
|
|
|
logStatData();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
// ************************************************************************* //
|