mirror of
https://gitee.com/anod/open_agb_firm.git
synced 2025-05-06 05:44:11 +08:00
275 lines
5.8 KiB
C
275 lines
5.8 KiB
C
/*
|
|
* This file is part of libn3ds
|
|
* Copyright (C) 2024 derrek, profi200
|
|
*
|
|
* 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 3 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, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "types.h"
|
|
#include "error_codes.h"
|
|
#include "fs.h"
|
|
#include "ff.h"
|
|
|
|
|
|
static const char *const g_fsPathTable[FS_MAX_DRIVES] = {FS_DRIVE_NAMES};
|
|
static struct
|
|
{
|
|
FATFS fsTable[FS_MAX_DRIVES];
|
|
|
|
FIL fTable[FS_MAX_FILES];
|
|
u32 fBitmap;
|
|
u32 fHandles;
|
|
|
|
DIR dTable[FS_MAX_DIRS];
|
|
u32 dBitmap;
|
|
u32 dHandles;
|
|
} g_fsState = {0};
|
|
|
|
|
|
|
|
static Result fres2Res(FRESULT fr)
|
|
{
|
|
if(fr != FR_OK) return fr + RES_FR_DISK_ERR - 1;
|
|
else return RES_OK;
|
|
}
|
|
|
|
static u32 findUnusedFileSlot(void)
|
|
{
|
|
if(g_fsState.fHandles >= FS_MAX_FILES) return (u32)-1;
|
|
|
|
u32 i = 0;
|
|
do
|
|
{
|
|
if((g_fsState.fBitmap & BIT(i)) == 0) break;
|
|
} while(++i < FS_MAX_FILES);
|
|
|
|
return i;
|
|
}
|
|
|
|
static bool isFileHandleValid(FHandle h)
|
|
{
|
|
if(h > g_fsState.fHandles) return false;
|
|
else return true;
|
|
}
|
|
|
|
static u32 findUnusedDirSlot(void)
|
|
{
|
|
if(g_fsState.dHandles >= FS_MAX_DIRS) return (u32)-1;
|
|
|
|
u32 i = 0;
|
|
do
|
|
{
|
|
if((g_fsState.dBitmap & BIT(i)) == 0) break;
|
|
} while(++i < FS_MAX_DIRS);
|
|
|
|
return i;
|
|
}
|
|
|
|
static bool isDirHandleValid(DHandle h)
|
|
{
|
|
if(h > g_fsState.dHandles) return false;
|
|
else return true;
|
|
}
|
|
|
|
Result fMount(FsDrive drive)
|
|
{
|
|
if(drive >= FS_MAX_DRIVES) return RES_FR_INVALID_DRIVE;
|
|
|
|
return fres2Res(f_mount(&g_fsState.fsTable[drive], g_fsPathTable[drive], 1));
|
|
}
|
|
|
|
Result fUnmount(FsDrive drive)
|
|
{
|
|
if(drive >= FS_MAX_DRIVES) return RES_FR_INVALID_DRIVE;
|
|
|
|
return fres2Res(f_mount(NULL, g_fsPathTable[drive], 0));
|
|
}
|
|
|
|
Result fGetFree(FsDrive drive, u64 *const size)
|
|
{
|
|
if(drive >= FS_MAX_DRIVES) return RES_FR_INVALID_DRIVE;
|
|
|
|
DWORD freeClusters;
|
|
FATFS *fs;
|
|
Result res = fres2Res(f_getfree(g_fsPathTable[drive], &freeClusters, &fs));
|
|
if(res == RES_OK)
|
|
{
|
|
if(size) *size = (u64)(freeClusters * fs->csize) * 512u;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
Result fOpen(FHandle *const hOut, const char *const path, u8 mode)
|
|
{
|
|
if(hOut == NULL) return RES_INVALID_ARG;
|
|
const u32 slot = findUnusedFileSlot();
|
|
if(slot == (u32)-1) return RES_FR_TOO_MANY_OPEN_FILES;
|
|
|
|
Result res = fres2Res(f_open(&g_fsState.fTable[slot], path, mode));
|
|
if(res == RES_OK)
|
|
{
|
|
g_fsState.fBitmap |= BIT(slot);
|
|
g_fsState.fHandles++;
|
|
*hOut = (FHandle)slot;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
Result fRead(FHandle h, void *const buf, u32 size, u32 *const bytesRead)
|
|
{
|
|
if(!isFileHandleValid(h)) return RES_FR_INVALID_OBJECT;
|
|
|
|
UINT tmpBytesRead;
|
|
Result res = fres2Res(f_read(&g_fsState.fTable[h], buf, size, &tmpBytesRead));
|
|
|
|
if(bytesRead != NULL) *bytesRead = tmpBytesRead;
|
|
|
|
return res;
|
|
}
|
|
|
|
Result fWrite(FHandle h, const void *const buf, u32 size, u32 *const bytesWritten)
|
|
{
|
|
if(!isFileHandleValid(h)) return RES_FR_INVALID_OBJECT;
|
|
|
|
UINT tmpBytesWritten;
|
|
Result res = fres2Res(f_write(&g_fsState.fTable[h], buf, size, &tmpBytesWritten));
|
|
|
|
if(bytesWritten != NULL) *bytesWritten = tmpBytesWritten;
|
|
|
|
return res;
|
|
}
|
|
|
|
Result fSync(FHandle h)
|
|
{
|
|
if(!isFileHandleValid(h)) return RES_FR_INVALID_OBJECT;
|
|
|
|
return fres2Res(f_sync(&g_fsState.fTable[h]));
|
|
}
|
|
|
|
Result fLseek(FHandle h, u32 off)
|
|
{
|
|
if(!isFileHandleValid(h)) return RES_FR_INVALID_OBJECT;
|
|
|
|
return fres2Res(f_lseek(&g_fsState.fTable[h], off));
|
|
}
|
|
|
|
u32 fTell(FHandle h)
|
|
{
|
|
if(!isFileHandleValid(h)) return 0;
|
|
|
|
return f_tell(&g_fsState.fTable[h]);
|
|
}
|
|
|
|
u32 fSize(FHandle h)
|
|
{
|
|
if(!isFileHandleValid(h)) return 0;
|
|
|
|
return f_size(&g_fsState.fTable[h]);
|
|
}
|
|
|
|
Result fClose(FHandle h)
|
|
{
|
|
if(g_fsState.fHandles == 0 || !isFileHandleValid(h))
|
|
return RES_FR_INVALID_OBJECT;
|
|
|
|
Result res = fres2Res(f_close(&g_fsState.fTable[h]));
|
|
g_fsState.fBitmap &= ~BIT(h);
|
|
g_fsState.fHandles--;
|
|
|
|
return res;
|
|
}
|
|
|
|
Result fStat(const char *const path, FILINFO *const fi)
|
|
{
|
|
return fres2Res(f_stat(path, fi));
|
|
}
|
|
|
|
Result fChdir(const char *const path)
|
|
{
|
|
return fres2Res(f_chdir(path));
|
|
}
|
|
|
|
Result fOpenDir(DHandle *const hOut, const char *const path)
|
|
{
|
|
if(hOut == NULL) return RES_INVALID_ARG;
|
|
const u32 slot = findUnusedDirSlot();
|
|
if(slot == (u32)-1) return RES_FR_TOO_MANY_OPEN_FILES;
|
|
|
|
Result res = fres2Res(f_opendir(&g_fsState.dTable[slot], path));
|
|
if(res == RES_OK)
|
|
{
|
|
g_fsState.dBitmap |= BIT(slot);
|
|
g_fsState.dHandles++;
|
|
*hOut = (DHandle)slot;
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
Result fReadDir(DHandle h, FILINFO *const fi, u32 num, u32 *const entriesRead)
|
|
{
|
|
if(!isDirHandleValid(h)) return RES_FR_INVALID_OBJECT;
|
|
// TODO: Check for insanely high num?
|
|
|
|
u32 i;
|
|
DIR *const dir = &g_fsState.dTable[h];
|
|
Result res = RES_OK;
|
|
for(i = 0; i < num; i++)
|
|
{
|
|
res = fres2Res(f_readdir(dir, &fi[i]));
|
|
|
|
if(res != RES_OK || fi[i].fname[0] == 0) break;
|
|
}
|
|
|
|
if(entriesRead != NULL) *entriesRead = i;
|
|
|
|
return res;
|
|
}
|
|
|
|
Result fCloseDir(DHandle h)
|
|
{
|
|
if(g_fsState.dHandles == 0 || !isDirHandleValid(h))
|
|
return RES_FR_INVALID_OBJECT;
|
|
|
|
Result res = fres2Res(f_closedir(&g_fsState.dTable[h]));
|
|
g_fsState.dBitmap &= ~BIT(h);
|
|
g_fsState.dHandles--;
|
|
|
|
return res;
|
|
}
|
|
|
|
Result fMkdir(const char *const path)
|
|
{
|
|
return fres2Res(f_mkdir(path));
|
|
}
|
|
|
|
Result fRename(const char *const old, const char *const _new)
|
|
{
|
|
return fres2Res(f_rename(old, _new));
|
|
}
|
|
|
|
Result fUnlink(const char *const path)
|
|
{
|
|
return fres2Res(f_unlink(path));
|
|
}
|
|
|
|
void fsDeinit(void)
|
|
{
|
|
for(u32 i = 0; i < FS_MAX_FILES; i++) fClose(i);
|
|
for(u32 i = 0; i < FS_MAX_DRIVES; i++) fUnmount(i);
|
|
|
|
// TODO: Deinit drives.
|
|
} |