// Copyright (c) 2012- PPSSPP Project. // This program 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, version 2.0 or later versions. // This program 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 2.0 for more details. // A copy of the GPL 2.0 should have been included with the program. // If not, see http://www.gnu.org/licenses/ // Official git repository and contact information can be found at // https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/. #pragma once #include #include "base/mutex.h" #include "Common/CommonTypes.h" #include "Core/Loaders.h" class CachingFileLoader : public FileLoader { public: CachingFileLoader(FileLoader *backend); virtual ~CachingFileLoader() override; virtual bool Exists() override; virtual bool IsDirectory() override; virtual s64 FileSize() override; virtual std::string Path() const override; virtual void Seek(s64 absolutePos) override; virtual size_t Read(size_t bytes, size_t count, void *data) override { return ReadAt(filepos_, bytes, count, data); } virtual size_t Read(size_t bytes, void *data) override { return ReadAt(filepos_, bytes, data); } virtual size_t ReadAt(s64 absolutePos, size_t bytes, size_t count, void *data) override { return ReadAt(absolutePos, bytes * count, data) / bytes; } virtual size_t ReadAt(s64 absolutePos, size_t bytes, void *data) override; private: void InitCache(); void ShutdownCache(); size_t ReadFromCache(s64 pos, size_t bytes, void *data); // Guaranteed to read at least one block into the cache. void SaveIntoCache(s64 pos, size_t bytes, 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, // 256 MB BLOCK_READAHEAD = 4, }; s64 filesize_; s64 filepos_; FileLoader *backend_; int exists_; int isDirectory_; 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 blocks_; recursive_mutex blocksMutex_; mutable recursive_mutex backendMutex_; bool aheadThread_; };