ppsspp/Core/Util/RecentFiles.cpp
2025-03-26 17:59:30 +01:00

194 lines
5 KiB
C++

#include <mutex>
#include <algorithm>
#include <condition_variable>
#include "Common/File/FileUtil.h"
#include "Common/Thread/ThreadUtil.h"
#include "Common/Log.h"
#include "Common/TimeUtil.h"
#include "Core/Loaders.h"
#include "Core/Util/RecentFiles.h"
#include "Core/Config.h"
RecentFilesManager g_recentFiles;
RecentFilesManager::RecentFilesManager() {
}
RecentFilesManager::~RecentFilesManager() {
ResetThread();
}
std::vector<std::string> RecentFilesManager::GetRecentFiles() const {
std::lock_guard<std::mutex> guard(recentIsosLock);
return recentIsos;
}
bool RecentFilesManager::ContainsFile(const std::string &filename) {
if (g_Config.iMaxRecent <= 0)
return false;
// Unfortunately this resolve needs to be done synchonously.
std::string resolvedFilename = File::ResolvePath(filename);
std::lock_guard<std::mutex> guard(recentIsosLock);
for (const auto & file : recentIsos) {
if (file == resolvedFilename) {
return true;
}
}
return false;
}
bool RecentFilesManager::HasAny() const {
std::lock_guard<std::mutex> guard(recentIsosLock);
return !recentIsos.empty();
}
void RecentFilesManager::Clear() {
ResetThread();
std::lock_guard<std::mutex> guard(recentIsosLock);
recentIsos.clear();
}
void RecentFilesManager::ResetThread() {
std::lock_guard<std::mutex> guard(recentIsosThreadLock);
if (recentIsosThreadPending && recentIsosThread.joinable())
recentIsosThread.join();
}
void RecentFilesManager::SetThread(std::function<void()> f) {
std::lock_guard<std::mutex> guard(recentIsosThreadLock);
if (recentIsosThreadPending && recentIsosThread.joinable())
recentIsosThread.join();
recentIsosThread = std::thread(f);
recentIsosThreadPending = true;
}
void RecentFilesManager::Load(const Section *recent, int maxRecent) {
ResetThread();
std::vector<std::string> newRecent;
for (int i = 0; i < maxRecent; i++) {
char keyName[64];
std::string fileName;
snprintf(keyName, sizeof(keyName), "FileName%d", i);
if (recent->Get(keyName, &fileName, "") && !fileName.empty()) {
newRecent.push_back(fileName);
}
}
std::lock_guard<std::mutex> guard(recentIsosLock);
recentIsos = newRecent;
}
void RecentFilesManager::Save(Section *recent, int maxRecent) {
ResetThread();
std::vector<std::string> recentCopy;
{
std::lock_guard<std::mutex> guard(recentIsosLock);
recentCopy = recentIsos;
}
for (int i = 0; i < maxRecent; i++) {
char keyName[64];
snprintf(keyName, sizeof(keyName), "FileName%d", i);
if (i < (int)recentCopy.size()) {
recent->Set(keyName, recentCopy[i]);
} else {
recent->Delete(keyName); // delete the nonexisting FileName
}
}
}
void RecentFilesManager::RemoveResolved(const std::string &resolvedFilename) {
ResetThread();
std::lock_guard<std::mutex> guard(recentIsosLock);
auto iter = std::remove_if(recentIsos.begin(), recentIsos.end(), [resolvedFilename](const auto &str) {
return str == resolvedFilename;
});
// remove_if is weird.
recentIsos.erase(iter, recentIsos.end());
}
void RecentFilesManager::Add(const std::string &filename) {
if (g_Config.iMaxRecent <= 0) {
return;
}
std::string resolvedFilename = File::ResolvePath(filename);
RemoveResolved(resolvedFilename);
ResetThread();
std::lock_guard<std::mutex> guard(recentIsosLock);
recentIsos.insert(recentIsos.begin(), resolvedFilename);
if ((int)recentIsos.size() > g_Config.iMaxRecent)
recentIsos.resize(g_Config.iMaxRecent);
}
void RecentFilesManager::Remove(const std::string &filename) {
if (g_Config.iMaxRecent <= 0) {
return;
}
std::string resolvedFilename = File::ResolvePath(filename);
RemoveResolved(resolvedFilename);
}
void RecentFilesManager::Clean() {
SetThread([this] {
SetCurrentThreadName("RecentISOs");
AndroidJNIThreadContext jniContext; // destructor detaches
double startTime = time_now_d();
std::lock_guard<std::mutex> guard(recentIsosLock);
std::vector<std::string> cleanedRecent;
if (recentIsos.empty()) {
INFO_LOG(Log::Loader, "No recents list found.");
}
for (size_t i = 0; i < recentIsos.size(); i++) {
bool exists = false;
Path path = Path(recentIsos[i]);
switch (path.Type()) {
case PathType::CONTENT_URI:
case PathType::NATIVE:
exists = File::Exists(path);
if (!exists) {
if (TryUpdateSavedPath(&path)) {
exists = File::Exists(path);
INFO_LOG(Log::Loader, "Exists=%d when checking updated path: %s", exists, path.c_str());
}
}
break;
default:
FileLoader *loader = ConstructFileLoader(path);
exists = loader->ExistsFast();
delete loader;
break;
}
if (exists) {
std::string pathStr = path.ToString();
// Make sure we don't have any redundant items.
auto duplicate = std::find(cleanedRecent.begin(), cleanedRecent.end(), pathStr);
if (duplicate == cleanedRecent.end()) {
cleanedRecent.push_back(pathStr);
}
} else {
DEBUG_LOG(Log::Loader, "Removed %s from recent. errno=%d", path.c_str(), errno);
}
}
double recentTime = time_now_d() - startTime;
if (recentTime > 0.1) {
INFO_LOG(Log::System, "CleanRecent took %0.2f", recentTime);
}
recentIsos = cleanedRecent;
});
}