/* ScummVM - Scumm Interpreter * Copyright (C) 2005 The ScummVM 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; either version 2 * of the License, or (at your option) any later version. * 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 for more details. * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * * $Header$ * */ #include "backends/ps2/fileio.h" #include #include #include #include #include #include #include "backends/ps2/asyncfio.h" #include "base/engine.h" #include "common/file.h" #define CACHE_SIZE (2048 * 32) #define MAX_READ_STEP (2048 * 16) #define MAX_CACHED_FILES 6 #define CACHE_READ_THRESHOLD (16 * 2048) #define CACHE_FILL_MIN (2048 * 24) extern void sioprintf(const char *zFormat, ...); AsyncFio fio; Ps2File::Ps2File(int64 cacheId) { _cacheId = cacheId; } Ps2File::~Ps2File(void) { } class Ps2ReadFile : public Ps2File { public: Ps2ReadFile(int64 cacheId); virtual ~Ps2ReadFile(void); virtual bool open(const char *name); virtual uint32 read(void *dest, uint32 len); virtual uint32 write(const void *src, uint32 len); virtual uint32 tell(void); virtual uint32 size(void); virtual int seek(int32 offset, int origin); virtual bool eof(void); private: void cacheReadAhead(void); void cacheReadSync(void); int _fd, _sema; uint8 *_cacheBuf; bool _cacheOpRunning; uint32 _filePos, _physFilePos, _cachePos; uint32 _fileSize, _bytesInCache, _cacheOfs; uint32 _readBytesBlock; }; Ps2ReadFile::Ps2ReadFile(int64 cacheId) : Ps2File(cacheId) { _fd = -1; _cacheBuf = (uint8*)memalign(64, CACHE_SIZE); _cacheOpRunning = 0; _filePos = _physFilePos = _cachePos = 0; _fileSize = _bytesInCache = _cacheOfs = 0; _cacheOpRunning = false; _readBytesBlock = 0; ee_sema_t newSema; newSema.init_count = 1; newSema.max_count = 1; _sema = CreateSema(&newSema); assert(_sema >= 0); } Ps2ReadFile::~Ps2ReadFile(void) { if (_cacheOpRunning) cacheReadSync(); free(_cacheBuf); if (_fd >= 0) fio.close(_fd); DeleteSema(_sema); } bool Ps2ReadFile::open(const char *name) { assert(_fd < 0); _fd = fio.open(name, O_RDONLY); if (_fd >= 0) { _fileSize = fio.seek(_fd, 0, SEEK_END); fio.seek(_fd, 0, SEEK_SET); return true; } else return false; } uint32 Ps2ReadFile::tell(void) { WaitSema(_sema); uint32 res = _filePos; SignalSema(_sema); return res; } uint32 Ps2ReadFile::size(void) { WaitSema(_sema); uint32 res = _fileSize; SignalSema(_sema); return res; } bool Ps2ReadFile::eof(void) { WaitSema(_sema); bool res = (_filePos == _fileSize); SignalSema(_sema); return res; } int Ps2ReadFile::seek(int32 offset, int origin) { WaitSema(_sema); int seekDest; int res = -1; switch (origin) { case SEEK_SET: seekDest = offset; break; case SEEK_CUR: seekDest = _filePos + offset; break; case SEEK_END: seekDest = _fileSize + offset; break; default: seekDest = -1; break; } if ((seekDest >= 0) && (seekDest <= _fileSize)) { _filePos = seekDest; res = 0; } SignalSema(_sema); return res; } void Ps2ReadFile::cacheReadAhead(void) { if (_cacheOpRunning) { // there's already some cache read running if (fio.poll(_fd)) // did it finish? cacheReadSync(); // yes. } if ((!_cacheOpRunning) && (_readBytesBlock >= CACHE_READ_THRESHOLD) && fio.fioAvail()) { // the engine seems to do sequential reads and there are no other I/Os going on. read ahead. uint32 cachePosEnd = _cachePos + _bytesInCache; if (_cachePos > _filePos) return; // there was a seek in the meantime, don't cache. if (cachePosEnd - _filePos >= CACHE_FILL_MIN) return; // cache is full enough. if (cachePosEnd == _fileSize) return; // can't read beyond EOF. assert(cachePosEnd < _fileSize); if (_cachePos + _bytesInCache <= _filePos) { _cacheOfs = _bytesInCache = 0; _cachePos = cachePosEnd = _filePos; assert(_filePos == _physFilePos); } else { uint32 cacheDiff = _filePos - _cachePos; assert(_bytesInCache >= cacheDiff); _bytesInCache -= cacheDiff; _cachePos += cacheDiff; _cacheOfs = (_cacheOfs + cacheDiff) % CACHE_SIZE; } if (_physFilePos != cachePosEnd) { sioprintf("unexpected _physFilePos %d cache %d %d", _physFilePos, _cacheOfs, _bytesInCache); _physFilePos = fio.seek(_fd, cachePosEnd, SEEK_SET); if (_physFilePos != cachePosEnd) { sioprintf("cache seek error: seek to %d instead of %d, fs = %d", _physFilePos, cachePosEnd, _fileSize); return; } } uint32 cacheDest = (_cacheOfs + _bytesInCache) % CACHE_SIZE; uint32 cacheRead = CACHE_SIZE - _bytesInCache; if (cacheDest + cacheRead > CACHE_SIZE) cacheRead = CACHE_SIZE - cacheDest; if (cacheRead > MAX_READ_STEP) cacheRead = MAX_READ_STEP; assert(cacheRead); _cacheOpRunning = true; fio.read(_fd, _cacheBuf + cacheDest, cacheRead); } } void Ps2ReadFile::cacheReadSync(void) { if (_cacheOpRunning) { int res = fio.sync(_fd); assert(res >= 0); _bytesInCache += res; _physFilePos += res; _cacheOpRunning = false; } } uint32 Ps2ReadFile::read(void *dest, uint32 len) { WaitSema(_sema); uint8 *destBuf = (uint8*)dest; if ((_filePos < _cachePos) || (_filePos + len > _cachePos + _bytesInCache)) cacheReadSync(); // we have to read from CD, sync cache. while (len) { if ((_filePos >= _cachePos) && (_filePos < _cachePos + _bytesInCache)) { // read from cache uint32 staPos = (_cacheOfs + (_filePos - _cachePos)) % CACHE_SIZE; uint32 cpyLen = _bytesInCache - (_filePos - _cachePos); if (cpyLen > len) cpyLen = len; if (staPos + cpyLen > CACHE_SIZE) cpyLen = CACHE_SIZE - staPos; assert(cpyLen); memcpy(destBuf, _cacheBuf + staPos, cpyLen); _filePos += cpyLen; destBuf += cpyLen; _readBytesBlock += len; len -= cpyLen; } else { // cache miss assert(!_cacheOpRunning); if (_physFilePos != _filePos) { if ((_filePos < _physFilePos) || (_filePos > _physFilePos + (CACHE_SIZE / 2))) _readBytesBlock = 0; // reset cache hit count if (fio.seek(_fd, _filePos, SEEK_SET) == _filePos) _physFilePos = _filePos; else break; // read beyond EOF } assert(_physFilePos == _filePos); int doRead = (len > MAX_READ_STEP) ? MAX_READ_STEP : len; if (doRead < 2048) doRead = 2048; fio.read(_fd, _cacheBuf, doRead); _cachePos = _filePos; _cacheOfs = 0; _bytesInCache = fio.sync(_fd); _physFilePos = _filePos + _bytesInCache; if (!_bytesInCache) break; // EOF } } cacheReadAhead(); SignalSema(_sema); return destBuf - (uint8*)dest; } uint32 Ps2ReadFile::write(const void *src, uint32 len) { sioprintf("write request on Ps2ReadFile!"); SleepThread(); return 0; } class Ps2WriteFile : public Ps2File { public: Ps2WriteFile(int64 cacheId); virtual ~Ps2WriteFile(void); virtual bool open(const char *name); virtual uint32 read(void *dest, uint32 len); virtual uint32 write(const void *src, uint32 len); virtual uint32 tell(void); virtual uint32 size(void); virtual int seek(int32 offset, int origin); virtual bool eof(void); private: int _fd; uint8 *_cacheBuf; uint32 _filePos, _bytesInCache; }; Ps2WriteFile::Ps2WriteFile(int64 cacheId) : Ps2File(cacheId) { _fd = -1; _cacheBuf = (uint8*)malloc(CACHE_SIZE); _filePos = _bytesInCache = 0; } Ps2WriteFile::~Ps2WriteFile(void) { if ((_fd >= 0) && (_bytesInCache)) { fio.write(_fd, _cacheBuf, _bytesInCache); int wrRes = fio.sync(_fd); if (wrRes != _bytesInCache) // too late to return an error printf("Cache flush on fclose(): Unable to write %d cached bytes to mc, only %d bytes written\n", _bytesInCache, wrRes); } if (_fd >= 0) fio.close(_fd); free(_cacheBuf); } bool Ps2WriteFile::open(const char *name) { _fd = fio.open(name, O_WRONLY | O_CREAT | O_TRUNC); return (_fd >= 0); } uint32 Ps2WriteFile::read(void *dest, uint32 len) { printf("ERROR: Read request on Ps2WriteFile\n"); SleepThread(); return 0; } uint32 Ps2WriteFile::write(const void *src, uint32 len) { uint32 size = len; uint8 *srcBuf = (uint8*)src; while (size) { uint32 doCpy = (len > CACHE_SIZE - _bytesInCache) ? (CACHE_SIZE - _bytesInCache) : len; if (doCpy) { memcpy(_cacheBuf + _bytesInCache, srcBuf, doCpy); _bytesInCache += doCpy; srcBuf += doCpy; size -= doCpy; } if (_bytesInCache == CACHE_SIZE) { fio.write(_fd, _cacheBuf, _bytesInCache); int wrRes = fio.sync(_fd); if (wrRes != _bytesInCache) { printf("Unable to flush %d cached bytes to memory card!\n", _bytesInCache); return 0; } _filePos += _bytesInCache; _bytesInCache = 0; } } return len; } uint32 Ps2WriteFile::tell(void) { return _bytesInCache + _filePos; } uint32 Ps2WriteFile::size(void) { return tell(); } int Ps2WriteFile::seek(int32 offset, int origin) { printf("Seek(%d/%d) request on Ps2WriteFile\n", offset, origin); SleepThread(); return 0; } bool Ps2WriteFile::eof(void) { return true; } struct TocNode { char name[64]; TocNode *next, *sub; bool isDir; uint8 nameLen; }; class TocManager { public: TocManager(void); ~TocManager(void); void readEntries(const char *root); int64 fileExists(const char *name); bool haveEntries(void); private: void readDir(const char *path, TocNode **node, int level); TocNode *_rootNode; char _root[256]; uint8 _rootLen; }; TocManager tocManager; struct FioHandleCache { Ps2File *file; FioHandleCache *next, *prev; }; static FioHandleCache *cacheListStart = NULL; static FioHandleCache *cacheListEnd = NULL; static int cacheListLen = 0; static int openFileCount = 0; static int cacheListSema = -1; Ps2File *findInCache(int64 id); FILE *ps2_fopen(const char *fname, const char *mode) { if (cacheListSema == -1) { ee_sema_t newSema; newSema.init_count = 1; newSema.max_count = 1; cacheListSema = CreateSema(&newSema); assert(cacheListSema >= 0); } if (!tocManager.haveEntries() && g_engine) // read the TOC the first time the engine opens a file tocManager.readEntries(g_engine->getGameDataPath()); if (((mode[0] != 'r') && (mode[0] != 'w')) || ((mode[1] != '\0') && (mode[1] != 'b'))) { printf("unsupported mode \"%s\" for file \"%s\"\n", mode, fname); return NULL; } bool rdOnly = (mode[0] == 'r'); int64 cacheId = -1; if (rdOnly && tocManager.haveEntries()) cacheId = tocManager.fileExists(fname); if (cacheId != 0) { Ps2File *file = findInCache(cacheId); if (file) { //sioprintf("open from cache: %s (%d) [%d]\n", fname, cacheId, file->_handle->_handle); return (FILE*)file; } if (rdOnly) { // smush files need a quite different caching behaviour than normal data files if (strstr(fname, ".san") || strstr(fname, ".SAN") || strstr(fname, ".San")) file = new Ps2SmushFile(cacheId); else file = new Ps2ReadFile(cacheId); } else file = new Ps2WriteFile(cacheId); if (file->open(fname)) { openFileCount++; return (FILE*)file; } else delete file; } return NULL; } void checkCacheListLen(void) { while ((cacheListLen > MAX_CACHED_FILES) || ((openFileCount > 13) && cacheListLen)) { assert(cacheListEnd && cacheListStart); delete cacheListEnd->file; cacheListEnd->prev->next = NULL; FioHandleCache *temp = cacheListEnd; cacheListEnd = cacheListEnd->prev; delete temp; cacheListLen--; openFileCount--; } } int ps2_fclose(FILE *stream) { Ps2File *file = (Ps2File*)stream; if (file->_cacheId > 0) { // this is a file on the CD, could be smart to cache it FioHandleCache *newHandle = new FioHandleCache; newHandle->file = file; file->seek(0, SEEK_SET); WaitSema(cacheListSema); if (!cacheListEnd) { assert(!cacheListStart); newHandle->prev = newHandle->next = NULL; cacheListEnd = cacheListStart = newHandle; } else { assert(cacheListStart); newHandle->prev = NULL; newHandle->next = cacheListStart; cacheListStart->prev = newHandle; cacheListStart = newHandle; } cacheListLen++; checkCacheListLen(); SignalSema(cacheListSema); } else { openFileCount--; delete file; } return 0; } Ps2File *findInCache(int64 id) { if (id <= 0) return NULL; WaitSema(cacheListSema); FioHandleCache *node = cacheListStart; while (node) { if (node->file->_cacheId == id) { if (node == cacheListStart) cacheListStart = node->next; if (node == cacheListEnd) cacheListEnd = node->prev; if (node->prev) node->prev->next = node->next; if (node->next) node->next->prev = node->prev; Ps2File *ret = node->file; delete node; cacheListLen--; SignalSema(cacheListSema); return ret; } else node = node->next; } SignalSema(cacheListSema); return NULL; } int ps2_fseek(FILE *stream, long offset, int origin) { return ((Ps2File*)stream)->seek(offset, origin); } uint32 ps2_ftell(FILE *stream) { return ((Ps2File*)stream)->tell(); } uint32 ps2_fsize(FILE *stream) { return ((Ps2File*)stream)->size(); } int ps2_feof(FILE *stream) { return ((Ps2File*)stream)->eof(); } size_t ps2_fread(void *buf, size_t r, size_t n, FILE *stream) { assert(r != 0); return ((Ps2File*)stream)->read(buf, r * n) / r; } int ps2_fgetc(FILE *stream) { uint8 temp; if (((Ps2File*)stream)->read(&temp, 1)) return temp; else return EOF; } char *ps2_fgets(char *buf, int n, FILE *stream) { char *retVal = buf; while (n--) { if (n == 0) *buf = '\0'; else { char c = ps2_fgetc(stream); if (c == EOF) return NULL; if ((c == '\r') || (c == '\n')) { *buf++ = '\0'; return retVal; } *buf++ = c; } } return retVal; } int ps2_fprintf(FILE *pOut, const char *zFormat, ...) { va_list ap; char resStr[2048]; va_start(ap,zFormat); int res = vsnprintf(resStr, 2048, zFormat, ap); va_end(ap); if ((pOut == stderr) || (pOut == stdout)) { printf("%s", resStr); sioprintf("%s", resStr); } else res = ps2_fwrite(resStr, 1, res, pOut); return res; } size_t ps2_fwrite(const void *buf, size_t r, size_t n, FILE *stream) { assert(r != 0); return ((Ps2File*)stream)->write(buf, r * n) / r; } int ps2_fputc(int c, FILE *stream) { if (((Ps2File*)stream)->write(&c, 1) == 1) return c; else return -1; } int ps2_fputs(const char *s, FILE *stream) { int len = strlen(s); if (ps2_fwrite(s, 1, len, stream) == len) return len; else return EOF; } int ps2_fflush(FILE *stream) { printf("fflush not implemented\n"); return 0; } TocManager::TocManager(void) { _rootNode = NULL; } TocManager::~TocManager(void) { // todo: write this... } bool TocManager::haveEntries(void) { return _rootNode != NULL; } void TocManager::readEntries(const char *root) { _rootLen = strlen(root); strcpy(_root, root); if (_root[_rootLen - 1] == '/') { _rootLen--; _root[_rootLen] = '\0'; } readDir(_root, &_rootNode, 0); } #define MAX_DIR_ENTRIES 512 void TocManager::readDir(const char *path, TocNode **node, int level) { if (level <= 2) { // we don't scan deeper than that struct TocEntry tocEntries[MAX_DIR_ENTRIES]; int files = CDVD_GetDir(path + 5, NULL, CDVD_GET_FILES_AND_DIRS, tocEntries, MAX_DIR_ENTRIES, NULL); for (int cnt = 0; cnt < files; cnt++) { if (tocEntries[cnt].filename[0] != '.') { // skip '.' and '..' *node = new TocNode; (*node)->sub = (*node)->next = NULL; (*node)->nameLen = strlen(tocEntries[cnt].filename); memcpy((*node)->name, tocEntries[cnt].filename, (*node)->nameLen + 1); if (tocEntries[cnt].fileProperties & 2) { // directory (*node)->isDir = true; char nextPath[256]; sprintf(nextPath, "%s/%s", path, tocEntries[cnt].filename); readDir(nextPath, &((*node)->sub), level + 1); } else (*node)->isDir = false; node = &((*node)->next); } } } } int64 TocManager::fileExists(const char *name) { const char *tmpName = name; if (((name[_rootLen] != '/') && (name[_rootLen] != '\0')) || (strnicmp(name, _root, _rootLen) != 0)) { for (int i = 0; i < 8; i++) if (name[i] == ':') // we don't know the content of other drives, return -1; // assume file exists else if ((name[i] == '/') || (name[i] == '\\')) return 0; // does not exists (this is probably ScummVM trying the 'current directory'.) return 0; } uint8 nameLen = strlen(name); name += _rootLen + 1; nameLen -= _rootLen + 1; TocNode *node = _rootNode; int64 retId = 1; while (node) { if (((name[node->nameLen] == '/') || (name[node->nameLen] == '\0')) && (strnicmp(name, node->name, node->nameLen) == 0)) { name += node->nameLen; nameLen -= node->nameLen; if (node->isDir) { if (nameLen) { name++; // skip '/' nameLen--; node = node->sub; retId <<= 10; } else return 0; // can't open a directory with fopen() } else { if (nameLen == 0) return retId; // ok, found else return 0; // here's a file, but there's something left in the path } } else { node = node->next; retId++; } } return 0; // does not exist }