#pragma once
#include <map>
#include <mutex>
#include <thread>
#include "Common/CommonTypes.h"
#include "Core/Loaders.h"
class CachingFileLoader : public ProxiedFileLoader {
public:
CachingFileLoader(FileLoader *backend);
~CachingFileLoader();
bool Exists() override;
bool ExistsFast() override;
bool IsDirectory() override;
s64 FileSize() override;
size_t ReadAt(s64 absolutePos, size_t bytes, size_t count, void *data, Flags flags = Flags::NONE) override {
return ReadAt(absolutePos, bytes * count, data, flags) / bytes;
}
size_t ReadAt(s64 absolutePos, size_t bytes, void *data, Flags flags = Flags::NONE) override;
private:
void Prepare();
void InitCache();
void ShutdownCache();
size_t ReadFromCache(s64 pos, size_t bytes, void *data);
void SaveIntoCache(s64 pos, size_t bytes, Flags flags, bool readingAhead = false);
bool MakeCacheSpaceFor(size_t blocks, bool readingAhead);
void StartReadAhead(s64 pos);
enum {
BLOCK_SIZE = 65536,
BLOCK_SHIFT = 16,
MAX_BLOCKS_PER_READ = 16,
MAX_BLOCKS_CACHED = 4096,
BLOCK_READAHEAD = 4,
};
s64 filesize_ = 0;
int exists_ = -1;
int isDirectory_ = -1;
u64 generation_;
u64 oldestGeneration_;
size_t cacheSize_;
struct BlockInfo {
u8 *ptr;
u64 generation;
BlockInfo() : ptr(nullptr), generation(0) {
}
BlockInfo(u8 *p) : ptr(p), generation(0) {
}
};
std::map<s64, BlockInfo> blocks_;
std::recursive_mutex blocksMutex_;
bool aheadThreadRunning_ = false;
std::thread aheadThread_;
std::once_flag preparedFlag_;
};