Core: Adding to library is now recursive

This commit is contained in:
Vicki Pfau 2021-01-05 00:23:52 -08:00
parent bda4316839
commit 527f235934
5 changed files with 50 additions and 38 deletions

View File

@ -94,6 +94,7 @@ Misc:
- Core: Add shutdown callback
- Core: Rework thread state synchronization
- Core: Improve support for ROM patch cheats, supporting disabling overlapping patches
- Core: Adding to library is now recursive
- GB: Allow pausing event loop while CPU is blocked
- GB: Add support for sleep and shutdown callbacks
- GB: Redo double speed emulation (closes mgba.io/i/1515)

View File

@ -35,7 +35,7 @@ void mLibraryDestroy(struct mLibrary*);
struct VDir;
struct VFile;
void mLibraryLoadDirectory(struct mLibrary* library, const char* base);
void mLibraryLoadDirectory(struct mLibrary* library, const char* base, bool recursive);
void mLibraryClear(struct mLibrary* library);
size_t mLibraryCount(struct mLibrary* library, const struct mLibraryEntry* constraints);

View File

@ -42,7 +42,7 @@ struct mLibrary {
static void _mLibraryDeleteEntry(struct mLibrary* library, struct mLibraryEntry* entry);
static void _mLibraryInsertEntry(struct mLibrary* library, struct mLibraryEntry* entry);
static void _mLibraryAddEntry(struct mLibrary* library, const char* filename, const char* base, struct VFile* vf);
static bool _mLibraryAddEntry(struct mLibrary* library, const char* filename, const char* base, struct VFile* vf);
static void _bindConstraints(sqlite3_stmt* statement, const struct mLibraryEntry* constraints) {
if (!constraints) {
@ -212,7 +212,7 @@ void mLibraryDestroy(struct mLibrary* library) {
free(library);
}
void mLibraryLoadDirectory(struct mLibrary* library, const char* base) {
void mLibraryLoadDirectory(struct mLibrary* library, const char* base, bool recursive) {
struct VDir* dir = VDirOpenArchive(base);
if (!dir) {
dir = VDirOpen(base);
@ -248,44 +248,55 @@ void mLibraryLoadDirectory(struct mLibrary* library, const char* base) {
dir->rewind(dir);
struct VDirEntry* dirent = dir->listNext(dir);
while (dirent) {
struct VFile* vf = dir->openFile(dir, dirent->name(dirent), O_RDONLY);
if (!vf) {
dirent = dir->listNext(dir);
continue;
const char* name = dirent->name(dirent);
struct VFile* vf = dir->openFile(dir, name, O_RDONLY);
bool wasAdded = false;
if (vf) {
wasAdded = _mLibraryAddEntry(library, name, base, vf);
}
if (!wasAdded && name[0] != '.') {
char newBase[PATH_MAX];
snprintf(newBase, sizeof(newBase), "%s" PATH_SEP "%s", base, name);
if (recursive) {
mLibraryLoadDirectory(library, newBase, recursive);
} else if (dirent->type(dirent) == VFS_FILE) {
mLibraryLoadDirectory(library, newBase, true); // This will add as an archive
}
}
_mLibraryAddEntry(library, dirent->name(dirent), base, vf);
dirent = dir->listNext(dir);
}
dir->close(dir);
sqlite3_exec(library->db, "COMMIT;", NULL, NULL, NULL);
}
void _mLibraryAddEntry(struct mLibrary* library, const char* filename, const char* base, struct VFile* vf) {
struct mCore* core;
bool _mLibraryAddEntry(struct mLibrary* library, const char* filename, const char* base, struct VFile* vf) {
if (!vf) {
return;
return false;
}
core = mCoreFindVF(vf);
if (core) {
struct mLibraryEntry entry;
memset(&entry, 0, sizeof(entry));
core->init(core);
core->loadROM(core, vf);
core->getGameTitle(core, entry.internalTitle);
core->getGameCode(core, entry.internalCode);
core->checksum(core, &entry.crc32, mCHECKSUM_CRC32);
entry.platform = core->platform(core);
entry.title = NULL;
entry.base = base;
entry.filename = filename;
entry.filesize = vf->size(vf);
_mLibraryInsertEntry(library, &entry);
// Note: this destroys the VFile
core->deinit(core);
} else {
struct mCore* core = mCoreFindVF(vf);
if (!core) {
vf->close(vf);
return false;
}
struct mLibraryEntry entry;
memset(&entry, 0, sizeof(entry));
core->init(core);
core->loadROM(core, vf);
core->getGameTitle(core, entry.internalTitle);
core->getGameCode(core, entry.internalCode);
core->checksum(core, &entry.crc32, mCHECKSUM_CRC32);
entry.platform = core->platform(core);
entry.title = NULL;
entry.base = base;
entry.filename = filename;
entry.filesize = vf->size(vf);
_mLibraryInsertEntry(library, &entry);
// Note: this destroys the VFile
core->deinit(core);
return true;
}
static void _mLibraryInsertEntry(struct mLibrary* library, struct mLibraryEntry* entry) {

View File

@ -108,10 +108,10 @@ QPair<QString, QString> LibraryController::selectedPath() {
return e ? qMakePair(e->base(), e->filename()) : qMakePair<QString, QString>("", "");
}
void LibraryController::addDirectory(const QString& dir) {
void LibraryController::addDirectory(const QString& dir, bool recursive) {
// The worker thread temporarily owns the library
std::shared_ptr<mLibrary> library = m_library;
m_libraryJob = GBAApp::app()->submitWorkerJob(std::bind(&LibraryController::loadDirectory, this, dir), this, [this, library]() {
m_libraryJob = GBAApp::app()->submitWorkerJob(std::bind(&LibraryController::loadDirectory, this, dir, recursive), this, [this, library]() {
m_libraryJob = -1;
refresh();
});
@ -181,10 +181,10 @@ void LibraryController::selectLastBootedGame() {
}
}
void LibraryController::loadDirectory(const QString& dir) {
// This class can get delted during this function (sigh) so we need to hold onto this
void LibraryController::loadDirectory(const QString& dir, bool recursive) {
// This class can get deleted during this function (sigh) so we need to hold onto this
std::shared_ptr<mLibrary> library = m_library;
mLibraryLoadDirectory(library.get(), dir.toUtf8().constData());
mLibraryLoadDirectory(library.get(), dir.toUtf8().constData(), recursive);
}
void LibraryController::freeLibrary() {
@ -192,4 +192,4 @@ void LibraryController::freeLibrary() {
mLibraryEntryFree(mLibraryListingGetPointer(&m_listing, i));
}
mLibraryListingClear(&m_listing);
}
}

View File

@ -84,7 +84,7 @@ public:
void selectLastBootedGame();
void addDirectory(const QString& dir);
void addDirectory(const QString& dir, bool recursive = true);
public slots:
void clear();
@ -97,7 +97,7 @@ private slots:
void refresh();
private:
void loadDirectory(const QString&); // Called on separate thread
void loadDirectory(const QString&, bool recursive = true); // Called on separate thread
void freeLibrary();
ConfigController* m_config = nullptr;