2020-09-21 17:17:34 +00:00
|
|
|
// SPDX-License-Identifier: MIT
|
2020-06-21 12:25:28 +00:00
|
|
|
//
|
2020-09-21 17:17:34 +00:00
|
|
|
// EmulationStation Desktop Edition
|
2020-06-21 12:25:28 +00:00
|
|
|
// TextureDataManager.cpp
|
|
|
|
//
|
|
|
|
// Loading and unloading of texture data.
|
|
|
|
//
|
|
|
|
|
2017-01-22 23:28:06 +00:00
|
|
|
#include "resources/TextureDataManager.h"
|
2017-11-01 22:21:10 +00:00
|
|
|
|
|
|
|
#include "resources/TextureData.h"
|
2017-01-22 23:28:06 +00:00
|
|
|
#include "resources/TextureResource.h"
|
2020-08-08 13:14:33 +00:00
|
|
|
#include "Log.h"
|
2017-01-22 23:28:06 +00:00
|
|
|
#include "Settings.h"
|
|
|
|
|
|
|
|
TextureDataManager::TextureDataManager()
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
unsigned char data[5 * 5 * 4];
|
|
|
|
mBlank = std::shared_ptr<TextureData>(new TextureData(false));
|
|
|
|
for (int i = 0; i < (5 * 5); ++i) {
|
|
|
|
data[i*4] = (i % 2) * 255;
|
|
|
|
data[i*4+1] = (i % 2) * 255;
|
|
|
|
data[i*4+2] = (i % 2) * 255;
|
|
|
|
data[i*4+3] = 0;
|
|
|
|
}
|
|
|
|
mBlank->initFromRGBA(data, 5, 5);
|
|
|
|
mLoader = new TextureLoader;
|
2017-01-22 23:28:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TextureDataManager::~TextureDataManager()
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
delete mLoader;
|
2017-01-22 23:28:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::shared_ptr<TextureData> TextureDataManager::add(const TextureResource* key, bool tiled)
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
remove(key);
|
|
|
|
std::shared_ptr<TextureData> data(new TextureData(tiled));
|
|
|
|
mTextures.push_front(data);
|
|
|
|
mTextureLookup[key] = mTextures.cbegin();
|
|
|
|
return data;
|
2017-01-22 23:28:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void TextureDataManager::remove(const TextureResource* key)
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
// Find the entry in the list.
|
|
|
|
auto it = mTextureLookup.find(key);
|
|
|
|
if (it != mTextureLookup.cend()) {
|
|
|
|
// Remove the list entry.
|
|
|
|
mTextures.erase((*it).second);
|
|
|
|
// And the lookup.
|
|
|
|
mTextureLookup.erase(it);
|
|
|
|
}
|
2017-01-22 23:28:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
std::shared_ptr<TextureData> TextureDataManager::get(const TextureResource* key)
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
// If it's in the cache then we want to remove it from it's current location and
|
|
|
|
// move it to the top.
|
|
|
|
std::shared_ptr<TextureData> tex;
|
|
|
|
auto it = mTextureLookup.find(key);
|
|
|
|
if (it != mTextureLookup.cend()) {
|
|
|
|
tex = *(*it).second;
|
|
|
|
// Remove the list entry.
|
|
|
|
mTextures.erase((*it).second);
|
|
|
|
// Put it at the top.
|
|
|
|
mTextures.push_front(tex);
|
|
|
|
// Store it back in the lookup.
|
|
|
|
mTextureLookup[key] = mTextures.cbegin();
|
|
|
|
|
|
|
|
// Make sure it's loaded or queued for loading.
|
|
|
|
load(tex);
|
|
|
|
}
|
|
|
|
return tex;
|
2017-01-22 23:28:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool TextureDataManager::bind(const TextureResource* key)
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
std::shared_ptr<TextureData> tex = get(key);
|
|
|
|
bool bound = false;
|
|
|
|
if (tex != nullptr)
|
|
|
|
bound = tex->uploadAndBind();
|
|
|
|
if (!bound)
|
|
|
|
mBlank->uploadAndBind();
|
|
|
|
return bound;
|
2017-01-22 23:28:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t TextureDataManager::getTotalSize()
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
size_t total = 0;
|
|
|
|
for (auto tex : mTextures)
|
|
|
|
total += tex->width() * tex->height() * 4;
|
|
|
|
return total;
|
2017-01-22 23:28:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t TextureDataManager::getCommittedSize()
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
size_t total = 0;
|
|
|
|
for (auto tex : mTextures)
|
|
|
|
total += tex->getVRAMUsage();
|
|
|
|
return total;
|
2017-01-22 23:28:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t TextureDataManager::getQueueSize()
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
return mLoader->getQueueSize();
|
2017-01-22 23:28:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void TextureDataManager::load(std::shared_ptr<TextureData> tex, bool block)
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
// See if it's already loaded.
|
|
|
|
if (tex->isLoaded())
|
|
|
|
return;
|
|
|
|
// Not loaded. Make sure there is room.
|
|
|
|
size_t size = TextureResource::getTotalMemUsage();
|
2020-08-08 13:14:33 +00:00
|
|
|
size_t settingVRAM = (size_t)Settings::getInstance()->getInt("MaxVRAM");
|
|
|
|
|
|
|
|
if (settingVRAM < 80) {
|
2020-08-08 20:33:27 +00:00
|
|
|
LOG(LogWarning) << "MaxVRAM is too low at " << settingVRAM <<
|
|
|
|
" MiB, setting it to the minimum allowed value of 80 MiB.";
|
2020-08-08 13:14:33 +00:00
|
|
|
Settings::getInstance()->setInt("MaxVRAM", 80);
|
|
|
|
settingVRAM = 80;
|
|
|
|
}
|
2020-08-08 20:33:27 +00:00
|
|
|
else if (settingVRAM > 1024) {
|
|
|
|
LOG(LogWarning) << "MaxVRAM is too high at " << settingVRAM <<
|
|
|
|
" MiB, setting it to the maximum allowed value of 1024 MiB.";
|
|
|
|
Settings::getInstance()->setInt("MaxVRAM", 1024);
|
|
|
|
settingVRAM = 1024;
|
2020-08-08 13:14:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t max_texture = settingVRAM * 1024 * 1024;
|
2020-06-21 12:25:28 +00:00
|
|
|
|
|
|
|
for (auto it = mTextures.crbegin(); it != mTextures.crend(); ++it) {
|
|
|
|
if (size < max_texture)
|
|
|
|
break;
|
|
|
|
//size -= (*it)->getVRAMUsage();
|
|
|
|
(*it)->releaseVRAM();
|
|
|
|
(*it)->releaseRAM();
|
|
|
|
// It may be already in the loader queue. In this case it wouldn't have been using
|
|
|
|
// any VRAM yet but it will be. Remove it from the loader queue.
|
|
|
|
mLoader->remove(*it);
|
|
|
|
size = TextureResource::getTotalMemUsage();
|
|
|
|
}
|
|
|
|
if (!block)
|
|
|
|
mLoader->load(tex);
|
|
|
|
else
|
|
|
|
tex->load();
|
2017-01-22 23:28:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TextureLoader::TextureLoader() : mExit(false)
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
mThread = new std::thread(&TextureLoader::threadProc, this);
|
2017-01-22 23:28:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TextureLoader::~TextureLoader()
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
// Just abort any waiting texture.
|
|
|
|
mTextureDataQ.clear();
|
|
|
|
mTextureDataLookup.clear();
|
2017-01-22 23:28:06 +00:00
|
|
|
|
2020-06-21 12:25:28 +00:00
|
|
|
// Exit the thread.
|
|
|
|
mExit = true;
|
|
|
|
mEvent.notify_one();
|
|
|
|
mThread->join();
|
|
|
|
delete mThread;
|
2017-01-22 23:28:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void TextureLoader::threadProc()
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
while (!mExit) {
|
|
|
|
std::shared_ptr<TextureData> textureData;
|
|
|
|
{
|
|
|
|
// Wait for an event to say there is something in the queue.
|
|
|
|
std::unique_lock<std::mutex> lock(mMutex);
|
|
|
|
mEvent.wait(lock);
|
|
|
|
if (!mTextureDataQ.empty()) {
|
|
|
|
textureData = mTextureDataQ.front();
|
|
|
|
mTextureDataQ.pop_front();
|
|
|
|
mTextureDataLookup.erase(mTextureDataLookup.find(textureData.get()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Queue has been released here but we might have a texture to process.
|
|
|
|
while (textureData) {
|
|
|
|
textureData->load();
|
|
|
|
|
|
|
|
// See if there is another item in the queue.
|
|
|
|
textureData = nullptr;
|
|
|
|
std::unique_lock<std::mutex> lock(mMutex);
|
|
|
|
if (!mTextureDataQ.empty()) {
|
|
|
|
textureData = mTextureDataQ.front();
|
|
|
|
mTextureDataQ.pop_front();
|
|
|
|
mTextureDataLookup.erase(mTextureDataLookup.find(textureData.get()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-01-22 23:28:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void TextureLoader::load(std::shared_ptr<TextureData> textureData)
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
// Make sure it's not already loaded.
|
|
|
|
if (!textureData->isLoaded()) {
|
|
|
|
std::unique_lock<std::mutex> lock(mMutex);
|
|
|
|
// Remove it from the queue if it is already there.
|
|
|
|
auto td = mTextureDataLookup.find(textureData.get());
|
|
|
|
if (td != mTextureDataLookup.cend()) {
|
|
|
|
mTextureDataQ.erase((*td).second);
|
|
|
|
mTextureDataLookup.erase(td);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Put it on the start of the queue as we want the newly requested textures to load first.
|
|
|
|
mTextureDataQ.push_front(textureData);
|
|
|
|
mTextureDataLookup[textureData.get()] = mTextureDataQ.cbegin();
|
|
|
|
mEvent.notify_one();
|
|
|
|
}
|
2017-01-22 23:28:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void TextureLoader::remove(std::shared_ptr<TextureData> textureData)
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
// Just remove it from the queue so we don't attempt to load it.
|
|
|
|
std::unique_lock<std::mutex> lock(mMutex);
|
|
|
|
auto td = mTextureDataLookup.find(textureData.get());
|
|
|
|
if (td != mTextureDataLookup.cend()) {
|
|
|
|
mTextureDataQ.erase((*td).second);
|
|
|
|
mTextureDataLookup.erase(td);
|
|
|
|
}
|
2017-01-22 23:28:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
size_t TextureLoader::getQueueSize()
|
|
|
|
{
|
2020-06-21 12:25:28 +00:00
|
|
|
// Get the amount of video memory that will be used once all textures in
|
|
|
|
// the queue are loaded.
|
|
|
|
size_t mem = 0;
|
|
|
|
std::unique_lock<std::mutex> lock(mMutex);
|
|
|
|
for (auto tex : mTextureDataQ)
|
|
|
|
mem += tex->width() * tex->height() * 4;
|
|
|
|
|
|
|
|
return mem;
|
2017-01-22 23:28:06 +00:00
|
|
|
}
|