2022-11-16 07:32:07 +00:00
|
|
|
/**
|
|
|
|
* @file mempak.c
|
|
|
|
*
|
|
|
|
* This file implements an interface for a controller memory pak filesystem (pfs), with operations to create, delete,
|
|
|
|
* read, write and check the size of files, as well as obtain a listing of which files currently exist.
|
|
|
|
*
|
|
|
|
* Each file is assigned an uppercase ASCII letter as an identifier, the game name for each is marked as
|
|
|
|
* 'ZELDA DEMO TOOL', encoded according to the N64 Font Code described in section 26.3 of the N64 Programming Manual.
|
|
|
|
*/
|
2020-10-03 15:22:44 +00:00
|
|
|
#include "global.h"
|
2020-06-07 23:38:33 +00:00
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
#define MEMPAK_MAX_FILES 11
|
|
|
|
|
2020-06-07 23:38:33 +00:00
|
|
|
OSPfs sMempakPfsHandle;
|
|
|
|
s32 sMempakFreeBytes;
|
2022-11-16 07:32:07 +00:00
|
|
|
s32 sMempakFiles[MEMPAK_MAX_FILES];
|
2020-06-07 23:38:33 +00:00
|
|
|
|
|
|
|
u16 sMempakCompanyCode = 1;
|
|
|
|
u32 sMempakGameCode = 1;
|
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
// NCH is a heavily abbreviated "N64 font code CHaracter"
|
|
|
|
// Conversion from A-Z to N64 Font Code
|
|
|
|
#define NCH(c) ((c)-0x27)
|
|
|
|
// Conversion from spaces to N64 Font Code
|
|
|
|
#define NCH_SPC (0x0F)
|
|
|
|
|
|
|
|
u8 sMempakGameName[PFS_FILE_NAME_LEN] = {
|
|
|
|
NCH('Z'), NCH('E'), NCH('L'), NCH('D'), NCH('A'), NCH_SPC, NCH('D'), NCH('E'),
|
|
|
|
NCH('M'), NCH('O'), NCH_SPC, NCH('T'), NCH('O'), NCH('O'), NCH('L'), NCH_SPC,
|
|
|
|
};
|
|
|
|
|
|
|
|
u8 sMempakExtName[PFS_FILE_EXT_LEN] = { 0 };
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initializes the memory pak filesystem for a memory pak in the controller plugged into the specified port. Subsequent
|
|
|
|
* memory pak operations will use the same controller port.
|
|
|
|
*
|
|
|
|
* @return true if the operation completed successfully, false otherwise
|
|
|
|
*/
|
|
|
|
s32 Mempak_Init(s32 controllerNum) {
|
|
|
|
OSMesgQueue* serialEventQueue;
|
2020-07-30 19:50:18 +00:00
|
|
|
s32 pad;
|
2021-02-14 00:49:40 +00:00
|
|
|
s32 ret = false;
|
2020-06-07 23:38:33 +00:00
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
serialEventQueue = PadMgr_AcquireSerialEventQueue(&gPadMgr);
|
2020-06-07 23:38:33 +00:00
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
if (osPfsInitPak(serialEventQueue, &sMempakPfsHandle, controllerNum) == 0) {
|
2020-06-07 23:38:33 +00:00
|
|
|
ret = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
osPfsFreeBlocks(&sMempakPfsHandle, &sMempakFreeBytes);
|
2022-11-16 07:32:07 +00:00
|
|
|
PadMgr_ReleaseSerialEventQueue(&gPadMgr, serialEventQueue);
|
2020-06-07 23:38:33 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
s32 Mempak_GetFreeBytes(s32 controllerNum) {
|
2020-06-07 23:38:33 +00:00
|
|
|
return sMempakFreeBytes;
|
|
|
|
}
|
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
/**
|
|
|
|
* Checks if the files identified by letters between `start` and `end` (inclusive) exist on the memory pak.
|
|
|
|
*
|
|
|
|
* This must be called before performing any individual file operations.
|
|
|
|
*
|
|
|
|
* @param controllerNum Unused, the controller used is that which was last passed to `Mempak_Init`
|
|
|
|
* @param start Start file letter
|
|
|
|
* @param end End file letter (inclusive)
|
|
|
|
* @return a bitfield where set bits indicate that the file exists
|
|
|
|
*/
|
|
|
|
s32 Mempak_FindFiles(s32 controllerNum, char start, char end) {
|
|
|
|
OSMesgQueue* serialEventQueue;
|
2020-06-07 23:38:33 +00:00
|
|
|
s32 error;
|
2022-11-16 07:32:07 +00:00
|
|
|
char letter;
|
2021-02-14 00:49:40 +00:00
|
|
|
u32 bit = 1;
|
2022-11-16 07:32:07 +00:00
|
|
|
s32 bits = 0;
|
2020-06-07 23:38:33 +00:00
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
serialEventQueue = PadMgr_AcquireSerialEventQueue(&gPadMgr);
|
2021-02-14 00:49:40 +00:00
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
for (letter = start; letter <= end; letter++) {
|
|
|
|
sMempakExtName[0] = NCH(letter);
|
2020-06-07 23:38:33 +00:00
|
|
|
|
|
|
|
error = osPfsFindFile(&sMempakPfsHandle, sMempakCompanyCode, sMempakGameCode, sMempakGameName, sMempakExtName,
|
2022-11-16 07:32:07 +00:00
|
|
|
&sMempakFiles[MEMPAK_LETTER_TO_INDEX(letter)]);
|
2020-06-07 23:38:33 +00:00
|
|
|
if (error == 0) {
|
2022-11-16 07:32:07 +00:00
|
|
|
bits |= bit;
|
2020-06-07 23:38:33 +00:00
|
|
|
} else {
|
2022-11-16 07:32:07 +00:00
|
|
|
sMempakFiles[MEMPAK_LETTER_TO_INDEX(letter)] = -1;
|
2020-06-07 23:38:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bit <<= 1;
|
2024-01-12 15:38:13 +00:00
|
|
|
PRINTF("mempak: find '%c' (%d)\n", letter, error);
|
2020-06-07 23:38:33 +00:00
|
|
|
}
|
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
PadMgr_ReleaseSerialEventQueue(&gPadMgr, serialEventQueue);
|
2024-01-12 15:38:13 +00:00
|
|
|
PRINTF("mempak: find '%c' - '%c' %02x\n", start, end, bits);
|
2022-11-16 07:32:07 +00:00
|
|
|
return bits;
|
2020-06-07 23:38:33 +00:00
|
|
|
}
|
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
/**
|
|
|
|
* Writes data to the file identified with `letter`.
|
|
|
|
*
|
|
|
|
* @param controllerNum Unused, the controller used is that which was last passed to `Mempak_Init`
|
|
|
|
* @param letter Memory pak file letter, in the range 'A' to ('A' + MEMPAK_MAX_FILES)
|
|
|
|
* @param buffer Buffer containing data to write
|
|
|
|
* @param offset Offset into the file to write to
|
|
|
|
* @param size Size in bytes
|
|
|
|
* @return true if the operation completed successfully, false otherwise
|
|
|
|
*/
|
|
|
|
s32 Mempak_Write(s32 controllerNum, char letter, void* buffer, s32 offset, s32 size) {
|
|
|
|
OSMesgQueue* serialEventQueue;
|
2020-06-07 23:38:33 +00:00
|
|
|
s32 error;
|
2021-02-14 00:49:40 +00:00
|
|
|
s32 ret = false;
|
2020-07-30 19:50:18 +00:00
|
|
|
s32 pad;
|
2020-06-07 23:38:33 +00:00
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
serialEventQueue = PadMgr_AcquireSerialEventQueue(&gPadMgr);
|
2021-02-14 00:49:40 +00:00
|
|
|
|
2020-06-07 23:38:33 +00:00
|
|
|
if (size < sMempakFreeBytes) {
|
2022-11-16 07:32:07 +00:00
|
|
|
error = osPfsReadWriteFile(&sMempakPfsHandle, sMempakFiles[MEMPAK_LETTER_TO_INDEX(letter)], PFS_WRITE, offset,
|
|
|
|
size, buffer);
|
2020-06-07 23:38:33 +00:00
|
|
|
if (error == 0) {
|
|
|
|
ret = true;
|
|
|
|
}
|
2024-01-12 15:38:13 +00:00
|
|
|
PRINTF("mempak: write %d byte '%c' (%d)->%d\n", size, letter, sMempakFiles[MEMPAK_LETTER_TO_INDEX(letter)],
|
|
|
|
error);
|
2020-06-07 23:38:33 +00:00
|
|
|
}
|
2022-11-16 07:32:07 +00:00
|
|
|
PadMgr_ReleaseSerialEventQueue(&gPadMgr, serialEventQueue);
|
2020-06-07 23:38:33 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
/**
|
|
|
|
* Reads data from the file identified with `letter`.
|
|
|
|
*
|
|
|
|
* @param controllerNum Unused, the controller used is that which was last passed to `Mempak_Init`
|
|
|
|
* @param letter Memory pak file letter, in the range 'A' to ('A' + MEMPAK_MAX_FILES)
|
|
|
|
* @param buffer Buffer to read data into
|
|
|
|
* @param offset Offset into the file to read from
|
|
|
|
* @param size Size in bytes
|
|
|
|
* @return true if the operation completed successfully, false otherwise
|
|
|
|
*/
|
|
|
|
s32 Mempak_Read(s32 controllerNum, char letter, void* buffer, s32 offset, s32 size) {
|
|
|
|
OSMesgQueue* serialEventQueue;
|
2020-06-07 23:38:33 +00:00
|
|
|
s32 error;
|
2021-02-14 00:49:40 +00:00
|
|
|
s32 ret = false;
|
2020-07-30 19:50:18 +00:00
|
|
|
s32 pad;
|
2020-06-07 23:38:33 +00:00
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
serialEventQueue = PadMgr_AcquireSerialEventQueue(&gPadMgr);
|
2021-02-14 00:49:40 +00:00
|
|
|
|
2020-06-07 23:38:33 +00:00
|
|
|
if (size < sMempakFreeBytes) {
|
2022-11-16 07:32:07 +00:00
|
|
|
error = osPfsReadWriteFile(&sMempakPfsHandle, sMempakFiles[MEMPAK_LETTER_TO_INDEX(letter)], PFS_READ, offset,
|
|
|
|
size, buffer);
|
2020-06-07 23:38:33 +00:00
|
|
|
if (error == 0) {
|
|
|
|
ret = true;
|
|
|
|
}
|
2024-01-12 15:38:13 +00:00
|
|
|
PRINTF("mempak: read %d byte '%c' (%d)<-%d\n", size, letter, sMempakFiles[MEMPAK_LETTER_TO_INDEX(letter)],
|
|
|
|
error);
|
2020-06-07 23:38:33 +00:00
|
|
|
}
|
2022-11-16 07:32:07 +00:00
|
|
|
PadMgr_ReleaseSerialEventQueue(&gPadMgr, serialEventQueue);
|
2020-06-07 23:38:33 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
/**
|
|
|
|
* Creates a new file on the memory pak.
|
|
|
|
*
|
|
|
|
* @param controllerNum Unused, the controller used is that which was last passed to `Mempak_Init`
|
|
|
|
* @param letter Memory pak file letter, in the range 'A' to ('A' + MEMPAK_MAX_FILES).
|
|
|
|
* If this points to a valid file letter the new file will be created using that letter, otherwise it will create
|
|
|
|
* a file using the first free letter and return it through this argument. If no letters are free, the last letter
|
|
|
|
* ('A' + MEMPAK_MAX_FILES - 1) is used.
|
|
|
|
* @param size File size
|
|
|
|
* @return true if the operation completed successfully, false otherwise
|
|
|
|
*/
|
|
|
|
s32 Mempak_CreateFile(s32 controllerNum, char* letter, s32 size) {
|
|
|
|
OSMesgQueue* serialEventQueue;
|
2020-06-07 23:38:33 +00:00
|
|
|
s32 error;
|
2022-11-16 07:32:07 +00:00
|
|
|
s32 ret = false;
|
2020-06-07 23:38:33 +00:00
|
|
|
s32 i;
|
2020-07-30 19:50:18 +00:00
|
|
|
s32 pad;
|
2020-06-07 23:38:33 +00:00
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
serialEventQueue = PadMgr_AcquireSerialEventQueue(&gPadMgr);
|
|
|
|
|
|
|
|
if (*letter >= MEMPAK_INDEX_TO_LETTER(0) && *letter < MEMPAK_INDEX_TO_LETTER(MEMPAK_MAX_FILES)) {
|
|
|
|
// Create file with specific letter
|
|
|
|
|
|
|
|
sMempakExtName[0] = NCH(*letter);
|
|
|
|
if (-1 == sMempakFiles[MEMPAK_LETTER_TO_INDEX(*letter)]) {
|
|
|
|
// File does not already exist
|
2021-02-14 00:49:40 +00:00
|
|
|
|
2020-06-07 23:38:33 +00:00
|
|
|
error = osPfsAllocateFile(&sMempakPfsHandle, sMempakCompanyCode, sMempakGameCode, sMempakGameName,
|
2022-11-16 07:32:07 +00:00
|
|
|
sMempakExtName, size, &sMempakFiles[MEMPAK_LETTER_TO_INDEX(*letter)]);
|
2020-06-07 23:38:33 +00:00
|
|
|
if (error == 0) {
|
2022-11-16 07:32:07 +00:00
|
|
|
ret = true;
|
2020-06-07 23:38:33 +00:00
|
|
|
}
|
2024-01-12 15:38:13 +00:00
|
|
|
PRINTF("mempak: alloc %d byte '%c' (%d)\n", size, *letter, error);
|
2020-06-07 23:38:33 +00:00
|
|
|
} else {
|
2022-11-16 07:32:07 +00:00
|
|
|
// File already exists, delete then alloc
|
|
|
|
|
|
|
|
sMempakExtName[0] = NCH(*letter);
|
2020-06-07 23:38:33 +00:00
|
|
|
if (osPfsDeleteFile(&sMempakPfsHandle, sMempakCompanyCode, sMempakGameCode, sMempakGameName,
|
|
|
|
sMempakExtName) == 0) {
|
2022-11-16 07:32:07 +00:00
|
|
|
ret = true;
|
2020-06-07 23:38:33 +00:00
|
|
|
}
|
|
|
|
error = osPfsAllocateFile(&sMempakPfsHandle, sMempakCompanyCode, sMempakGameCode, sMempakGameName,
|
2022-11-16 07:32:07 +00:00
|
|
|
sMempakExtName, size, &sMempakFiles[MEMPAK_LETTER_TO_INDEX(*letter)]);
|
2020-06-07 23:38:33 +00:00
|
|
|
if (error == 0) {
|
2022-11-16 07:32:07 +00:00
|
|
|
ret |= true;
|
2020-06-07 23:38:33 +00:00
|
|
|
}
|
2024-01-12 15:38:13 +00:00
|
|
|
PRINTF("mempak: resize %d byte '%c' (%d)\n", size, *letter, error);
|
2020-06-07 23:38:33 +00:00
|
|
|
}
|
|
|
|
} else {
|
2022-11-16 07:32:07 +00:00
|
|
|
// Find first free letter and create a file identified by it
|
|
|
|
for (i = 0; i < MEMPAK_MAX_FILES - 1; i++) {
|
2020-06-07 23:38:33 +00:00
|
|
|
if (sMempakFiles[i] == -1) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2022-11-16 07:32:07 +00:00
|
|
|
*letter = MEMPAK_INDEX_TO_LETTER(i);
|
2020-06-07 23:38:33 +00:00
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
sMempakExtName[0] = NCH(*letter);
|
2020-06-07 23:38:33 +00:00
|
|
|
error = osPfsAllocateFile(&sMempakPfsHandle, sMempakCompanyCode, sMempakGameCode, sMempakGameName,
|
|
|
|
sMempakExtName, size, &sMempakFiles[i]);
|
2024-01-12 15:38:13 +00:00
|
|
|
PRINTF("mempak: alloc %d byte '%c' (%d) with search\n", size, *letter, error);
|
2020-06-07 23:38:33 +00:00
|
|
|
if (error == 0) {
|
2022-11-16 07:32:07 +00:00
|
|
|
ret = true;
|
2020-06-07 23:38:33 +00:00
|
|
|
}
|
|
|
|
}
|
2022-11-16 07:32:07 +00:00
|
|
|
PadMgr_ReleaseSerialEventQueue(&gPadMgr, serialEventQueue);
|
2020-06-07 23:38:33 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
/**
|
|
|
|
* Deletes the file identified with `letter`.
|
|
|
|
*
|
|
|
|
* @param controllerNum Unused, the controller used is that which was last passed to `Mempak_Init`
|
|
|
|
* @param letter Memory pak file letter, in the range 'A' to ('A' + MEMPAK_MAX_FILES)
|
|
|
|
* @return true if the operation completed successfully, false otherwise
|
|
|
|
*/
|
|
|
|
s32 Mempak_DeleteFile(s32 controllerNum, char letter) {
|
|
|
|
OSMesgQueue* serialEventQueue;
|
2020-06-07 23:38:33 +00:00
|
|
|
s32 error;
|
2021-02-14 00:49:40 +00:00
|
|
|
s32 ret = false;
|
2020-06-07 23:38:33 +00:00
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
serialEventQueue = PadMgr_AcquireSerialEventQueue(&gPadMgr);
|
2021-02-14 00:49:40 +00:00
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
sMempakExtName[0] = NCH(letter);
|
2020-06-07 23:38:33 +00:00
|
|
|
error = osPfsDeleteFile(&sMempakPfsHandle, sMempakCompanyCode, sMempakGameCode, sMempakGameName, sMempakExtName);
|
|
|
|
if (error == 0) {
|
|
|
|
ret = true;
|
|
|
|
}
|
2024-01-12 15:38:13 +00:00
|
|
|
PRINTF("mempak: delete '%c' (%d)\n", letter, error);
|
2021-02-14 00:49:40 +00:00
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
PadMgr_ReleaseSerialEventQueue(&gPadMgr, serialEventQueue);
|
2020-06-07 23:38:33 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
/**
|
|
|
|
* Gets the size of the file identified with `letter`.
|
|
|
|
*
|
|
|
|
* @param controllerNum Unused, the controller used is that which was last passed to `Mempak_Init`
|
|
|
|
* @param letter Memory pak file letter, in the range 'A' to ('A' + MEMPAK_MAX_FILES)
|
|
|
|
* @return the size of the file, or 0 if the operation failed for any reason
|
|
|
|
*/
|
|
|
|
s32 Mempak_GetFileSize(s32 controllerNum, char letter) {
|
|
|
|
OSMesgQueue* serialEventQueue = PadMgr_AcquireSerialEventQueue(&gPadMgr);
|
2020-06-07 23:38:33 +00:00
|
|
|
OSPfsState state;
|
2022-11-16 07:32:07 +00:00
|
|
|
s32 error = osPfsFileState(&sMempakPfsHandle, sMempakFiles[MEMPAK_LETTER_TO_INDEX(letter)], &state);
|
2020-07-30 19:50:18 +00:00
|
|
|
s32 pad;
|
2020-06-07 23:38:33 +00:00
|
|
|
|
2022-11-16 07:32:07 +00:00
|
|
|
PadMgr_ReleaseSerialEventQueue(&gPadMgr, serialEventQueue);
|
2021-02-14 00:49:40 +00:00
|
|
|
|
2020-06-07 23:38:33 +00:00
|
|
|
if (error != 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return state.file_size;
|
|
|
|
}
|