2022-06-01 20:53:18 +00:00
|
|
|
#include "readerscache.hpp"
|
|
|
|
|
|
|
|
#include <stdexcept>
|
|
|
|
|
|
|
|
namespace ESM
|
|
|
|
{
|
|
|
|
ReadersCache::BusyItem::BusyItem(ReadersCache& owner, std::list<Item>::iterator item) noexcept
|
|
|
|
: mOwner(owner)
|
|
|
|
, mItem(item)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
|
|
|
}
|
2022-06-01 20:53:18 +00:00
|
|
|
|
|
|
|
ReadersCache::BusyItem::~BusyItem() noexcept
|
|
|
|
{
|
|
|
|
mOwner.releaseItem(mItem);
|
|
|
|
}
|
|
|
|
|
|
|
|
ReadersCache::ReadersCache(std::size_t capacity)
|
|
|
|
: mCapacity(capacity)
|
2022-09-22 18:26:05 +00:00
|
|
|
{
|
|
|
|
}
|
2022-06-01 20:53:18 +00:00
|
|
|
|
|
|
|
ReadersCache::BusyItem ReadersCache::get(std::size_t index)
|
|
|
|
{
|
|
|
|
const auto indexIt = mIndex.find(index);
|
|
|
|
std::list<Item>::iterator it;
|
|
|
|
if (indexIt == mIndex.end())
|
|
|
|
{
|
|
|
|
closeExtraReaders();
|
|
|
|
it = mBusyItems.emplace(mBusyItems.end());
|
|
|
|
mIndex.emplace(index, it);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
switch (indexIt->second->mState)
|
|
|
|
{
|
|
|
|
case State::Busy:
|
|
|
|
throw std::logic_error("ESMReader at index " + std::to_string(index) + " is busy");
|
|
|
|
case State::Free:
|
|
|
|
it = indexIt->second;
|
|
|
|
mBusyItems.splice(mBusyItems.end(), mFreeItems, it);
|
|
|
|
break;
|
|
|
|
case State::Closed:
|
|
|
|
closeExtraReaders();
|
|
|
|
it = indexIt->second;
|
|
|
|
if (it->mName.has_value())
|
|
|
|
{
|
|
|
|
it->mReader.open(*it->mName);
|
|
|
|
it->mName.reset();
|
|
|
|
}
|
|
|
|
mBusyItems.splice(mBusyItems.end(), mClosedItems, it);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
it->mState = State::Busy;
|
|
|
|
}
|
|
|
|
|
|
|
|
return BusyItem(*this, it);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ReadersCache::closeExtraReaders()
|
|
|
|
{
|
|
|
|
while (!mFreeItems.empty() && mBusyItems.size() + mFreeItems.size() + 1 > mCapacity)
|
|
|
|
{
|
|
|
|
const auto it = mFreeItems.begin();
|
|
|
|
if (it->mReader.isOpen())
|
|
|
|
{
|
|
|
|
it->mName = it->mReader.getName();
|
|
|
|
it->mReader.close();
|
|
|
|
}
|
|
|
|
mClosedItems.splice(mClosedItems.end(), mFreeItems, it);
|
|
|
|
it->mState = State::Closed;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ReadersCache::releaseItem(std::list<Item>::iterator it) noexcept
|
|
|
|
{
|
|
|
|
assert(it->mState == State::Busy);
|
|
|
|
if (it->mReader.isOpen())
|
|
|
|
{
|
|
|
|
mFreeItems.splice(mFreeItems.end(), mBusyItems, it);
|
|
|
|
it->mState = State::Free;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
mClosedItems.splice(mClosedItems.end(), mBusyItems, it);
|
|
|
|
it->mState = State::Closed;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|