Files
openfoam/src/OpenFOAM/db/objectRegistry/objectRegistry.C
Mark Olesen 129b738136 ENH: define IOobjectList::csorted(), deprecate some sorted() const methods
- prefer csorted() method for const access since it ensures that the
  return values are also const pointers (for example) even if
  the object itself can be accessed as a non-const.

- the csorted() method already existed for HashTable and
  objectRegistry, but now added to IOobjectList for method name
  consistency (even although the IOobjectList only has a const-access
  version)

ENH: objectRegistry with templated strict lookup

- for lookupClass and csorted/sorted. Allows isType restriction as a
  compile-time specification.
2023-07-31 20:11:32 +02:00

557 lines
13 KiB
C

/*---------------------------------------------------------------------------*\
========= |
\\ / F ield | OpenFOAM: The Open Source CFD Toolbox
\\ / O peration |
\\ / A nd | www.openfoam.com
\\/ M anipulation |
-------------------------------------------------------------------------------
Copyright (C) 2011-2019 OpenFOAM Foundation
Copyright (C) 2015-2023 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 "objectRegistry.H"
#include "Time.H"
#include "predicates.H"
// * * * * * * * * * * * * * * Static Data Members * * * * * * * * * * * * * //
namespace Foam
{
defineTypeNameAndDebug(objectRegistry, 0);
}
// * * * * * * * * * * * * * * * Local Functions * * * * * * * * * * * * * * //
namespace Foam
{
// Templated implementation for erase() with iterator range.
// Prefer not to expose directly.
template<class InputIter>
static label eraseImpl(objectRegistry& obr, InputIter first, InputIter last)
{
label changed = 0;
for
(
const label nTotal = obr.size();
changed < nTotal && first != last; // Terminate early
++first
)
{
if (obr.erase(*first))
{
++changed;
}
}
return changed;
}
} // End namespace Foam
// * * * * * * * * * * * * * Private Member Functions * * * * * * * * * * * //
bool Foam::objectRegistry::parentNotTime() const noexcept
{
return (&parent_ != static_cast<const objectRegistry*>(&time_));
}
// * * * * * * * * * * * * * * * * Constructors * * * * * * * * * * * * * * //
Foam::objectRegistry::objectRegistry
(
const Time& t,
const label initialCapacity
)
:
regIOobject
(
IOobject
(
word::validate(t.caseName()),
t.path(),
t,
IOobjectOption::NO_READ,
IOobjectOption::AUTO_WRITE,
IOobjectOption::NO_REGISTER
),
true // to flag that this is the top-level regIOobject
),
HashTable<regIOobject*>(initialCapacity),
time_(t),
parent_(t),
dbDir_(name()),
event_(1),
cacheTemporaryObjectsActive_(false),
cacheTemporaryObjects_(0),
temporaryObjects_(0)
{}
Foam::objectRegistry::objectRegistry
(
const IOobject& io,
const label initialCapacity
)
:
regIOobject(io),
HashTable<regIOobject*>(initialCapacity),
time_(io.time()),
parent_(io.db()),
dbDir_(parent_.dbDir()/local()/name()),
event_(1),
cacheTemporaryObjectsActive_(false),
cacheTemporaryObjects_(0),
temporaryObjects_(0)
{
writeOpt(IOobjectOption::AUTO_WRITE);
}
// * * * * * * * * * * * * * * * * Destructor * * * * * * * * * * * * * * * //
Foam::objectRegistry::~objectRegistry()
{
objectRegistry::clear();
}
// * * * * * * * * * * * * * * * Member Functions * * * * * * * * * * * * * //
bool Foam::objectRegistry::isTimeDb() const noexcept
{
return (this == &static_cast<const objectRegistry&>(time_));
}
Foam::HashTable<Foam::wordHashSet> Foam::objectRegistry::classes() const
{
return classesImpl(*this, predicates::always());
}
Foam::label Foam::objectRegistry::count(const char* clsName) const
{
// No nullptr check - only called with string literals
return count(static_cast<word>(clsName));
}
Foam::wordList Foam::objectRegistry::names() const
{
return HashTable<regIOobject*>::toc();
}
Foam::wordList Foam::objectRegistry::sortedNames() const
{
return HashTable<regIOobject*>::sortedToc();
}
Foam::wordList Foam::objectRegistry::names(const char* clsName) const
{
// No nullptr check - only called with string literals
return names(static_cast<word>(clsName));
}
Foam::wordList Foam::objectRegistry::sortedNames(const char* clsName) const
{
// No nullptr check - only called with string literals
return sortedNames(static_cast<word>(clsName));
}
const Foam::objectRegistry& Foam::objectRegistry::subRegistry
(
const word& name,
const bool forceCreate,
const bool recursive
) const
{
if (forceCreate && !foundObject<objectRegistry>(name, recursive))
{
objectRegistry* subObr = new objectRegistry
(
IOobject
(
name,
time().constant(),
*this,
IOobjectOption::NO_READ,
IOobjectOption::NO_WRITE,
IOobjectOption::REGISTER
)
);
subObr->store();
}
return lookupObject<objectRegistry>(name, recursive);
}
Foam::label Foam::objectRegistry::getEvent() const
{
label curEvent = event_++;
if (event_ == labelMax)
{
if (objectRegistry::debug)
{
WarningInFunction
<< "Event counter has overflowed. "
<< "Resetting counter on all dependent objects." << nl
<< "This might cause extra evaluations." << endl;
}
// Reset event counter
curEvent = 1;
event_ = 2;
// No need to reset dependent objects; overflow is now handled
// in regIOobject::upToDate
}
return curEvent;
}
bool Foam::objectRegistry::checkIn(regIOobject* io) const
{
if (!io) return false;
objectRegistry& obr = const_cast<objectRegistry&>(*this);
if (objectRegistry::debug)
{
Pout<< "objectRegistry::checkIn : "
<< name() << " : checking in " << io->name()
<< " of type " << io->type()
<< endl;
}
// Delete cached object if it has the same name as io, and in the
// cacheTemporaryObjects list
if (!cacheTemporaryObjects_.empty())
{
auto cacheIter = cacheTemporaryObjects_.find(io->name());
if (cacheIter.good())
{
iterator iter = obr.find(io->name());
if
(
iter.good()
&& iter.val() != io
&& iter.val()->ownedByRegistry()
)
{
if (objectRegistry::debug)
{
Pout<< "objectRegistry::checkIn : "
<< name() << " : deleting cached object "
<< io->name() << endl;
}
cacheIter.val().first() = false;
deleteCachedObject(iter.val());
}
}
}
bool ok = obr.insert(io->name(), io);
if (!ok && objectRegistry::debug)
{
WarningInFunction
<< name() << " : Attempt to checkIn object with name "
<< io->name() << " which was already checked in"
<< endl;
}
return ok;
}
bool Foam::objectRegistry::checkOut(regIOobject* io) const
{
if (!io) return false;
objectRegistry& obr = const_cast<objectRegistry&>(*this);
iterator iter = obr.find(io->name());
if (iter.good())
{
if (objectRegistry::debug)
{
Pout<< "objectRegistry::checkOut : "
<< name() << " : checking out " << io->name()
<< " of type " << io->type()
<< endl;
}
if (iter.val() != io)
{
if (objectRegistry::debug)
{
WarningInFunction
<< name() << " : Attempt to checkOut copy of "
<< iter.key()
<< endl;
}
return false;
}
return obr.erase(iter);
}
if (objectRegistry::debug)
{
Pout<< "objectRegistry::checkOut : "
<< name() << " : could not find " << io->name() << " in registry"
<< endl;
}
return false;
}
bool Foam::objectRegistry::checkIn(regIOobject& io) const
{
return checkIn(&io);
}
bool Foam::objectRegistry::checkOut(regIOobject& io) const
{
return checkOut(&io);
}
bool Foam::objectRegistry::checkOut(const word& key) const
{
return const_cast<objectRegistry&>(*this).erase(key);
}
void Foam::objectRegistry::clear()
{
// Free anything owned by the registry, but first unset both
// 'ownedByRegistry' and 'registered' flags to ensure that the
// regIOobject destructor will not affect the registry
for (iterator iter = begin(); iter != end(); ++iter)
{
regIOobject* ptr = iter.val();
if (ptr && ptr->ownedByRegistry())
{
if (objectRegistry::debug)
{
Pout<< "objectRegistry::clear : " << ptr->name() << nl;
}
ptr->release(true); // Relinquish ownership and registration
delete ptr; // Delete also clears fileHandler watches
}
}
HashTable<regIOobject*>::clear();
}
void Foam::objectRegistry::clearStorage()
{
objectRegistry::clear();
HashTable<regIOobject*>::clearStorage();
}
bool Foam::objectRegistry::erase(const iterator& iter)
{
// Remove from registry - see notes in objectRegistry::clear()
if (iter.good())
{
regIOobject* ptr = const_cast<iterator&>(iter).val();
const bool ok = HashTable<regIOobject*>::erase(iter);
if (ptr && ptr->ownedByRegistry())
{
ptr->release(true); // Relinquish ownership and registration
delete ptr; // Delete also clears fileHandler watches
}
return ok;
}
return false;
}
bool Foam::objectRegistry::erase(const word& key)
{
return erase(find(key));
}
Foam::label Foam::objectRegistry::erase(std::initializer_list<word> keys)
{
return eraseImpl(*this, keys.begin(), keys.end());
}
Foam::label Foam::objectRegistry::erase(const UList<word>& keys)
{
return eraseImpl(*this, keys.begin(), keys.end());
}
void Foam::objectRegistry::rename(const word& newName)
{
regIOobject::rename(newName);
// Adjust dbDir_ as well
const auto i = dbDir_.rfind('/');
if (i == string::npos)
{
dbDir_ = newName;
}
else
{
dbDir_.replace(i+1, string::npos, newName);
}
}
const Foam::regIOobject* Foam::objectRegistry::cfindIOobject
(
const word& name,
const bool recursive
) const
{
const_iterator iter = cfind(name);
if (iter.good())
{
return iter.val();
}
else if (recursive && this->parentNotTime())
{
return parent_.cfindIOobject(name, recursive);
}
return nullptr;
}
bool Foam::objectRegistry::contains
(
const word& name,
const bool recursive
) const
{
return cfindIOobject(name, recursive);
}
bool Foam::objectRegistry::modified() const
{
for (const_iterator iter = cbegin(); iter != cend(); ++iter)
{
if (iter.val()->modified())
{
return true;
}
}
return false;
}
void Foam::objectRegistry::readModifiedObjects()
{
for (iterator iter = begin(); iter != end(); ++iter)
{
if (objectRegistry::debug)
{
Pout<< "objectRegistry::readModifiedObjects() : "
<< name() << " : Considering reading object "
<< iter.key() << endl;
}
iter.val()->readIfModified();
}
}
bool Foam::objectRegistry::readIfModified()
{
readModifiedObjects();
return true;
}
bool Foam::objectRegistry::writeObject
(
IOstreamOption streamOpt,
const bool writeOnProc
) const
{
bool ok = true;
for (const_iterator iter = cbegin(); iter != cend(); ++iter)
{
if (objectRegistry::debug)
{
const regIOobject& obj = *iter.val();
Pout<< "objectRegistry::write() : "
<< name() << " : Considering writing object "
<< iter.key() << " of type "
<< obj.type() << " with writeOpt "
<< static_cast<int>(obj.writeOpt())
<< " to file " << obj.objectRelPath() << endl;
}
if (iter.val()->writeOpt() != IOobjectOption::NO_WRITE)
{
ok = iter.val()->writeObject(streamOpt, writeOnProc) && ok;
}
}
return ok;
}
// ************************************************************************* //