mirror of
https://github.com/N64Recomp/N64ModernRuntime.git
synced 2025-12-18 22:12:34 +00:00
Refactor. State: OK
This commit is contained in:
parent
118ad933de
commit
b18f6b3ff8
1 changed files with 384 additions and 107 deletions
|
|
@ -6,12 +6,13 @@
|
||||||
#include "recomp.h"
|
#include "recomp.h"
|
||||||
#include "helpers.hpp"
|
#include "helpers.hpp"
|
||||||
|
|
||||||
|
#define MAX_FILES 16
|
||||||
typedef struct ControllerPak {
|
typedef struct ControllerPak {
|
||||||
OSPfsState state;
|
OSPfsState state;
|
||||||
std::fstream file;
|
std::fstream file;
|
||||||
|
std::fstream header;
|
||||||
} ControllerPak;
|
} ControllerPak;
|
||||||
|
|
||||||
static ControllerPak sControllerPak[16];
|
|
||||||
static int sFileSelect = 0;
|
static int sFileSelect = 0;
|
||||||
|
|
||||||
void ByteSwapFile(u8* buffer, size_t size) {
|
void ByteSwapFile(u8* buffer, size_t size) {
|
||||||
|
|
@ -30,6 +31,9 @@ void ByteSwapFile(u8* buffer, size_t size) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Pak_WriteHeader(OSPfsState state, s32 file_no) {
|
||||||
|
}
|
||||||
|
|
||||||
// s32 osPfsIsPlug(OSMesgQueue* mq, u8* pattern)
|
// s32 osPfsIsPlug(OSMesgQueue* mq, u8* pattern)
|
||||||
extern "C" void osPfsIsPlug_recomp(uint8_t* rdram, recomp_context* ctx) {
|
extern "C" void osPfsIsPlug_recomp(uint8_t* rdram, recomp_context* ctx) {
|
||||||
MEM_B(0, ctx->r5) = 1; // *pattern = 1;
|
MEM_B(0, ctx->r5) = 1; // *pattern = 1;
|
||||||
|
|
@ -46,6 +50,13 @@ extern "C" void osPfsInit_recomp(uint8_t* rdram, recomp_context* ctx) {
|
||||||
pfs->channel = channel;
|
pfs->channel = channel;
|
||||||
pfs->status = 0x1;
|
pfs->status = 0x1;
|
||||||
|
|
||||||
|
ControllerPak pak;
|
||||||
|
|
||||||
|
if (!std::filesystem::exists("controllerPak_header.sav")) {
|
||||||
|
pak.header.open("controllerPak_header.sav", std::ios::binary | std::ios::in | std::ios::out | std::ios::trunc);
|
||||||
|
pak.header.close();
|
||||||
|
}
|
||||||
|
|
||||||
ctx->r2 = 0; // PFS_NO_ERROR
|
ctx->r2 = 0; // PFS_NO_ERROR
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -66,7 +77,51 @@ extern "C" void osPfsInitPak_recomp(uint8_t* rdram, recomp_context* ctx) {
|
||||||
extern "C" void osPfsFreeBlocks_recomp(uint8_t* rdram, recomp_context* ctx) {
|
extern "C" void osPfsFreeBlocks_recomp(uint8_t* rdram, recomp_context* ctx) {
|
||||||
s32* bytes_not_used = _arg<1, s32*>(rdram, ctx);
|
s32* bytes_not_used = _arg<1, s32*>(rdram, ctx);
|
||||||
|
|
||||||
*bytes_not_used = 123 << 8;
|
// locals:
|
||||||
|
s32 usedSpace = 0;
|
||||||
|
|
||||||
|
ControllerPak pak;
|
||||||
|
|
||||||
|
// *bytes_not_used = 123 << 8;
|
||||||
|
|
||||||
|
pak.header.open("controllerPak_header.sav", std::ios::binary | std::ios::in | std::ios::out);
|
||||||
|
|
||||||
|
if (!pak.header.good()) {
|
||||||
|
printf("file is not good!\n");
|
||||||
|
assert(false);
|
||||||
|
}
|
||||||
|
if (!pak.header.is_open()) {
|
||||||
|
printf("file isn't open!\n");
|
||||||
|
assert(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (size_t i = 0; i < 16; i++) {
|
||||||
|
u32 seek = i * sizeof(OSPfsState);
|
||||||
|
s32 file_size_in_bytes;
|
||||||
|
u32 game_code;
|
||||||
|
u16 company_code;
|
||||||
|
|
||||||
|
// file_size
|
||||||
|
pak.header.seekg(seek + 0x0, std::ios::beg);
|
||||||
|
pak.header.read((char*) &file_size_in_bytes, sizeof(file_size_in_bytes));
|
||||||
|
// game_code
|
||||||
|
pak.header.seekg(seek + 0x4, std::ios::beg);
|
||||||
|
pak.header.read((char*) &game_code, sizeof(game_code));
|
||||||
|
// company_code
|
||||||
|
pak.header.seekg(seek + 0x08, std::ios::beg);
|
||||||
|
pak.header.read((char*) &company_code, sizeof(company_code));
|
||||||
|
|
||||||
|
if ((company_code == 0) || (game_code == 0)) {
|
||||||
|
continue;
|
||||||
|
} else {
|
||||||
|
usedSpace += file_size_in_bytes >> 8;
|
||||||
|
file_size_in_bytes = game_code = company_code = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pak.header.close();
|
||||||
|
|
||||||
|
*bytes_not_used = (123 - usedSpace) << 8;
|
||||||
|
|
||||||
ctx->r2 = 0; // PFS_NO_ERROR
|
ctx->r2 = 0; // PFS_NO_ERROR
|
||||||
}
|
}
|
||||||
|
|
@ -84,112 +139,177 @@ extern "C" void osPfsAllocateFile_recomp(uint8_t* rdram, recomp_context* ctx) {
|
||||||
s32 file_size_in_bytes = (s32) MEM_W(0x14, ctx->r29);
|
s32 file_size_in_bytes = (s32) MEM_W(0x14, ctx->r29);
|
||||||
s32* file_no = TO_PTR(s32, MEM_W(0x18, ctx->r29));
|
s32* file_no = TO_PTR(s32, MEM_W(0x18, ctx->r29));
|
||||||
|
|
||||||
printf("osPfsAllocateFile_recomp was called!\n");
|
if ((company_code == 0) || (game_code == 0)) {
|
||||||
|
ctx->r2 = 5; // PFS_ERR_INVALID
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ControllerPak pak;
|
||||||
|
|
||||||
|
pak.header.open("controllerPak_header.sav", std::ios::binary | std::ios::in | std::ios::out);
|
||||||
|
|
||||||
|
if (!pak.header.good()) {
|
||||||
|
printf("file is not good!\n");
|
||||||
assert(false);
|
assert(false);
|
||||||
|
}
|
||||||
|
if (!pak.header.is_open()) {
|
||||||
|
printf("file isn't open!\n");
|
||||||
|
assert(false);
|
||||||
|
}
|
||||||
|
u8 freeFileIndex = 0;
|
||||||
#if 0
|
#if 0
|
||||||
ControllerPak* pak = &sControllerPak[*file_no];
|
/* Search for a free slot */
|
||||||
|
for (size_t i = 0; i < 16; i++) {
|
||||||
|
u32 seek = i * sizeof(OSPfsState);
|
||||||
|
u32 game_code_;
|
||||||
|
u16 company_code_;
|
||||||
|
|
||||||
printf("osPfsAllocateFile_recomp:\n");
|
// game_code
|
||||||
|
pak.header.seekg(seek + 0x4, std::ios::beg);
|
||||||
|
pak.header.read((char*) &game_code_, sizeof(game_code_));
|
||||||
|
// company_code
|
||||||
|
pak.header.seekg(seek + 0x08, std::ios::beg);
|
||||||
|
pak.header.read((char*) &company_code_, sizeof(company_code_));
|
||||||
|
|
||||||
*file_no = sFileSelect++;
|
if ((company_code_ == 0) || (game_code_ == 0)) {
|
||||||
|
freeFileIndex = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
game_code_ = company_code_ = 0;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
/* Set file parameters to header */
|
||||||
|
freeFileIndex = 0;
|
||||||
|
u32 seek = freeFileIndex * 0x20;
|
||||||
|
|
||||||
|
// file_size
|
||||||
|
pak.header.seekp(seek + 0x0, std::ios::beg);
|
||||||
|
pak.header.write((char*) &file_size_in_bytes, sizeof(file_size_in_bytes));
|
||||||
|
// game_code
|
||||||
|
pak.header.seekp(seek + 0x4, std::ios::beg);
|
||||||
|
pak.header.write((char*) &game_code, sizeof(game_code));
|
||||||
|
// company_code
|
||||||
|
pak.header.seekp(seek + 0x08, std::ios::beg);
|
||||||
|
pak.header.write((char*) &company_code, sizeof(company_code));
|
||||||
|
// ext_name
|
||||||
|
pak.header.seekp(seek + 0x0C, std::ios::beg);
|
||||||
|
pak.header.write((char*) ext_name, 4);
|
||||||
|
// game_name
|
||||||
|
pak.header.seekp(seek + 0x10, std::ios::beg);
|
||||||
|
pak.header.write((char*) game_name, 16);
|
||||||
|
|
||||||
|
pak.header.close();
|
||||||
|
|
||||||
|
/* Create empty file */
|
||||||
|
|
||||||
char filename[100];
|
char filename[100];
|
||||||
sprintf(filename, "controllerPak_file_%d.sav", *file_no);
|
sprintf(filename, "controllerPak_file_%d.sav", freeFileIndex);
|
||||||
// pak->file = fopen(filename, "wb+");
|
pak.file.open(filename, std::ios::binary | std::ios::in | std::ios::out | std::ios::trunc);
|
||||||
//pak->file.open()
|
|
||||||
|
|
||||||
file_size_in_bytes = (file_size_in_bytes + 31) & ~31;
|
file_size_in_bytes = (file_size_in_bytes + 31) & ~31;
|
||||||
|
|
||||||
u8* zero_block = (u8*) malloc(file_size_in_bytes);
|
u8* zero_block = (u8*) malloc(file_size_in_bytes);
|
||||||
memset(zero_block, 0, file_size_in_bytes);
|
memset(zero_block, 0, file_size_in_bytes);
|
||||||
fwrite(zero_block, file_size_in_bytes, 1, pak->file);
|
|
||||||
|
|
||||||
pak->state.company_code = company_code;
|
pak.file.seekp(0, std::ios::beg);
|
||||||
pak->state.game_code = game_code;
|
pak.file.write((char*) zero_block, file_size_in_bytes);
|
||||||
strcpy(pak->state.game_name, (const char*) game_name);
|
|
||||||
strcpy(pak->state.ext_name, (const char*) ext_name);
|
|
||||||
#endif
|
|
||||||
ctx->r2 = 0; // PFS_NO_ERROR
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: VALIDATE
|
free(zero_block);
|
||||||
// Can't really be validated without a valid controller pak file system header, which is only used
|
|
||||||
// by the controller pak manager present in games by holding START on boot and not necessary for recomp.
|
|
||||||
// s32 osPfsDeleteFile(OSPfs* pfs, u16 company_code, u32 game_code, u8* game_name, u8* ext_name);
|
|
||||||
extern "C" void osPfsDeleteFile_recomp(uint8_t* rdram, recomp_context* ctx) {
|
|
||||||
u16 company_code = _arg<1, u16>(rdram, ctx);
|
|
||||||
u32 game_code = _arg<2, u32>(rdram, ctx);
|
|
||||||
u8* game_name = _arg<3, u8*>(rdram, ctx);
|
|
||||||
u8* ext_name = TO_PTR(u8, MEM_W(0x10, ctx->r29));
|
|
||||||
s32 file_no;
|
|
||||||
|
|
||||||
printf("osPfsDeleteFile_recomp:\n");
|
pak.file.close();
|
||||||
|
|
||||||
for (size_t i = 0; i < ARRAYSIZE(sControllerPak); i++) {
|
*file_no = freeFileIndex;
|
||||||
if ((sControllerPak[i].state.company_code = company_code) && (sControllerPak[i].state.game_code == game_code) &&
|
|
||||||
(strcmp(sControllerPak[i].state.game_name, (const char*) game_name) == 0) &&
|
|
||||||
strcmp(sControllerPak[i].state.ext_name, (const char*) ext_name) == 0) {
|
|
||||||
file_no = i;
|
|
||||||
|
|
||||||
char filename[100];
|
|
||||||
sprintf(filename, "controllerPak_file_%d.sav", file_no);
|
|
||||||
remove(filename);
|
|
||||||
|
|
||||||
ctx->r2 = 0; // PFS_NO_ERROR
|
ctx->r2 = 0; // PFS_NO_ERROR
|
||||||
printf("Returned 0: FILE DELETED\n\n");
|
|
||||||
return;
|
|
||||||
} else {
|
|
||||||
ctx->r2 = 1; // PFS_ERR_NOPACK
|
|
||||||
printf("Returned 1: FILE NOT FOUND\n\n");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsFileEmpty(std::fstream& file) {
|
bool IsFileEmpty(std::fstream& file) {
|
||||||
|
|
||||||
|
if (!file.good()) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
file.seekg(0, std::ios::end);
|
file.seekg(0, std::ios::end);
|
||||||
long size = file.tellg();
|
long size = file.tellg();
|
||||||
file.seekg(0, std::ios::beg);
|
file.seekg(0, std::ios::beg);
|
||||||
|
|
||||||
return size == 0;
|
return size == 0;
|
||||||
}
|
}
|
||||||
// f.read(ptr, size) and f.write(ptr, size);
|
|
||||||
// s32 osPfsFileState(OSPfs* pfs, s32 file_no, OSPfsState* state);
|
// s32 osPfsFileState(OSPfs* pfs, s32 file_no, OSPfsState* state);
|
||||||
extern "C" void osPfsFileState_recomp(uint8_t* rdram, recomp_context* ctx) {
|
extern "C" void osPfsFileState_recomp(uint8_t* rdram, recomp_context* ctx) {
|
||||||
s32 file_no = _arg<1, s32>(rdram, ctx);
|
s32 file_no = _arg<1, s32>(rdram, ctx);
|
||||||
OSPfsState* state = _arg<2, OSPfsState*>(rdram, ctx);
|
OSPfsState* state = _arg<2, OSPfsState*>(rdram, ctx);
|
||||||
|
|
||||||
ControllerPak& pak = sControllerPak[file_no];
|
// locals:
|
||||||
|
s32 file_size_in_bytes;
|
||||||
|
u32 game_code;
|
||||||
|
u16 company_code;
|
||||||
|
char ext_name[4];
|
||||||
|
char game_name[16];
|
||||||
|
|
||||||
|
// should pass the state of the requested file_no to the incoming state pointer,
|
||||||
|
// games call this function 16 times, once per bank
|
||||||
|
// fills the incoming state with the information inside the header of the pak.
|
||||||
|
|
||||||
|
/* Read game info from pak */
|
||||||
|
ControllerPak pak;
|
||||||
|
|
||||||
char filename[100];
|
char filename[100];
|
||||||
sprintf(filename, "controllerPak_file_%d.sav", file_no);
|
sprintf(filename, "controllerPak_file_%d.sav", file_no);
|
||||||
// pak.file = fopen(filename, "rb+");
|
|
||||||
pak.file.open(filename, std::ios::binary | std::ios::in | std::ios::out);
|
pak.file.open(filename, std::ios::binary | std::ios::in | std::ios::out);
|
||||||
if (!pak.file.good()) {
|
|
||||||
printf("File's not good\n");
|
if (IsFileEmpty(pak.file)) {
|
||||||
|
pak.file.close();
|
||||||
|
ctx->r2 = 1; // PFS_NO_ERROR
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
pak.file.close();
|
||||||
|
|
||||||
|
pak.header.open("controllerPak_header.sav", std::ios::binary | std::ios::in | std::ios::out);
|
||||||
|
|
||||||
|
if (!pak.header.good()) {
|
||||||
|
printf("file is not good!\n");
|
||||||
|
assert(false);
|
||||||
|
}
|
||||||
|
if (!pak.header.is_open()) {
|
||||||
|
printf("file isn't open!\n");
|
||||||
assert(false);
|
assert(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!IsFileEmpty(pak.file)) {
|
u32 seek = file_no * sizeof(OSPfsState);
|
||||||
state->file_size = pak.state.file_size;
|
|
||||||
state->company_code = pak.state.company_code;
|
|
||||||
state->game_code = pak.state.game_code;
|
|
||||||
|
|
||||||
for (size_t j = 0; j < ARRAYSIZE(pak.state.game_name); j++) {
|
// file_size
|
||||||
state->game_name[j] = pak.state.game_name[j];
|
pak.header.seekg(seek + 0x0, std::ios::beg);
|
||||||
|
pak.header.read((char*) &file_size_in_bytes, sizeof(file_size_in_bytes));
|
||||||
|
// game_code
|
||||||
|
pak.header.seekg(seek + 0x4, std::ios::beg);
|
||||||
|
pak.header.read((char*) &game_code, sizeof(game_code));
|
||||||
|
// company_code
|
||||||
|
pak.header.seekg(seek + 0x08, std::ios::beg);
|
||||||
|
pak.header.read((char*) &company_code, sizeof(company_code));
|
||||||
|
// ext_name
|
||||||
|
pak.header.seekg(seek + 0x0C, std::ios::beg);
|
||||||
|
pak.header.read((char*) ext_name, ARRAYSIZE(ext_name));
|
||||||
|
// game_name
|
||||||
|
pak.header.seekg(seek + 0x10, std::ios::beg);
|
||||||
|
pak.header.read((char*) game_name, ARRAYSIZE(game_name));
|
||||||
|
|
||||||
|
pak.header.close();
|
||||||
|
|
||||||
|
state->file_size = file_size_in_bytes;
|
||||||
|
state->company_code = game_code;
|
||||||
|
state->game_code = game_code;
|
||||||
|
|
||||||
|
for (size_t j = 0; j < ARRAYSIZE(game_name); j++) {
|
||||||
|
state->game_name[j] = game_name[j];
|
||||||
}
|
}
|
||||||
for (size_t j = 0; j < ARRAYSIZE(pak.state.ext_name); j++) {
|
for (size_t j = 0; j < ARRAYSIZE(ext_name); j++) {
|
||||||
state->ext_name[j] = pak.state.ext_name[j];
|
state->ext_name[j] = ext_name[j];
|
||||||
}
|
}
|
||||||
|
|
||||||
// fclose(pak.file);
|
|
||||||
pak.file.close();
|
|
||||||
ctx->r2 = 0; // PFS_NO_ERROR
|
ctx->r2 = 0; // PFS_NO_ERROR
|
||||||
} else {
|
|
||||||
// fclose(pak.file);
|
|
||||||
pak.file.close();
|
|
||||||
remove(filename);
|
|
||||||
ctx->r2 = 1; // PFS_ERR_NOPACK
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// s32 osPfsFindFile(OSPfs* pfs, u16 company_code, u32 game_code, u8* game_name, u8* ext_name, s32* file_no);
|
// s32 osPfsFindFile(OSPfs* pfs, u16 company_code, u32 game_code, u8* game_name, u8* ext_name, s32* file_no);
|
||||||
|
|
@ -198,40 +318,73 @@ extern "C" void osPfsFindFile_recomp(uint8_t* rdram, recomp_context* ctx) {
|
||||||
u32 game_code = _arg<2, u32>(rdram, ctx);
|
u32 game_code = _arg<2, u32>(rdram, ctx);
|
||||||
u8* game_name = _arg<3, u8*>(rdram, ctx);
|
u8* game_name = _arg<3, u8*>(rdram, ctx);
|
||||||
u8* ext_name = TO_PTR(u8, MEM_W(0x10, ctx->r29));
|
u8* ext_name = TO_PTR(u8, MEM_W(0x10, ctx->r29));
|
||||||
s32* file_no = TO_PTR(s32, MEM_W(0x14, ctx->r29));
|
s32* file_no = TO_PTR(s32, MEM_W(0x14, ctx->r29)); // we should return the index of the file found here
|
||||||
|
|
||||||
for (size_t i = 0; i < ARRAYSIZE(sControllerPak); i++) {
|
ControllerPak pak;
|
||||||
if ((sControllerPak[i].state.game_code == game_code) &&
|
|
||||||
(sControllerPak[i].state.company_code == company_code) &&
|
char filename[100];
|
||||||
(strcmp(sControllerPak[i].state.game_name, (const char*) game_name) == 0) &&
|
sprintf(filename, "controllerPak_file_%d.sav", *file_no);
|
||||||
strcmp(sControllerPak[i].state.ext_name, (const char*) ext_name) == 0) {
|
pak.file.open(filename, std::ios::binary | std::ios::in | std::ios::out);
|
||||||
|
|
||||||
|
if (!std::filesystem::exists(filename) || IsFileEmpty(pak.file)) {
|
||||||
|
pak.file.close();
|
||||||
|
ctx->r2 = 5; // PFS_ERR_INVALID
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
pak.file.close();
|
||||||
|
|
||||||
|
pak.header.open("controllerPak_header.sav", std::ios::binary | std::ios::in | std::ios::out);
|
||||||
|
|
||||||
|
if (!pak.header.good()) {
|
||||||
|
printf("file is not good!\n");
|
||||||
|
assert(false);
|
||||||
|
}
|
||||||
|
if (!pak.header.is_open()) {
|
||||||
|
printf("file isn't open!\n");
|
||||||
|
assert(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (size_t i = 0; i < 16; i++) {
|
||||||
|
// locals:
|
||||||
|
u32 game_code_;
|
||||||
|
u16 company_code_;
|
||||||
|
char ext_name_[4];
|
||||||
|
char game_name_[16];
|
||||||
|
u32 seek = i * sizeof(OSPfsState);
|
||||||
|
|
||||||
|
// game_code
|
||||||
|
pak.header.seekg(seek + 0x4, std::ios::beg);
|
||||||
|
pak.header.read((char*) &game_code_, sizeof(game_code_));
|
||||||
|
// company_code
|
||||||
|
pak.header.seekg(seek + 0x08, std::ios::beg);
|
||||||
|
pak.header.read((char*) &company_code_, sizeof(company_code_));
|
||||||
|
// ext_name
|
||||||
|
pak.header.seekg(seek + 0x0C, std::ios::beg);
|
||||||
|
pak.header.read((char*) ext_name_, 4);
|
||||||
|
// game_name
|
||||||
|
pak.header.seekg(seek + 0x10, std::ios::beg);
|
||||||
|
pak.header.read((char*) game_name_, 16);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
if ((company_code_ == 0) || (game_code_ == 0)) {
|
||||||
|
continue;
|
||||||
|
} else {
|
||||||
|
if ((game_code == game_code_) && (company_code == company_code_) &&
|
||||||
|
(strcmp((const char*) game_name, (const char*) game_name_) == 0) &&
|
||||||
|
strcmp((const char*) ext_name, (const char*) ext_name_) == 0) {
|
||||||
// File found
|
// File found
|
||||||
*file_no = i;
|
*file_no = i;
|
||||||
|
pak.header.close();
|
||||||
ctx->r2 = 0; // PFS_NO_ERROR
|
ctx->r2 = 0; // PFS_NO_ERROR
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Open file
|
|
||||||
*file_no = sFileSelect++;
|
|
||||||
|
|
||||||
char filename[100];
|
|
||||||
sprintf(filename, "controllerPak_file_%d.sav", *file_no);
|
|
||||||
// sControllerPak[*file_no].file = fopen(filename, "rb+");
|
|
||||||
// if (sControllerPak[*file_no].file == NULL) {
|
|
||||||
// sControllerPak[*file_no].file = fopen(filename, "wb+");
|
|
||||||
// }
|
|
||||||
sControllerPak[*file_no].file.open(filename, std::ios::binary | std::ios::in | std::ios::out);
|
|
||||||
if (!sControllerPak[*file_no].file.good()) {
|
|
||||||
assert(false);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
sControllerPak[*file_no].state.company_code = company_code;
|
pak.header.close();
|
||||||
sControllerPak[*file_no].state.game_code = game_code;
|
// File not found
|
||||||
strcpy(sControllerPak[*file_no].state.game_name, (const char*) game_name);
|
ctx->r2 = 5; // PFS_ERR_INVALID
|
||||||
strcpy(sControllerPak[*file_no].state.ext_name, (const char*) ext_name);
|
|
||||||
|
|
||||||
ctx->r2 = 0; // PFS_NO_ERROR
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// s32 osPfsReadWriteFile(OSPfs* pfs, s32 file_no, u8 flag, int offset, int size_in_bytes, u8* data_buffer);
|
// s32 osPfsReadWriteFile(OSPfs* pfs, s32 file_no, u8 flag, int offset, int size_in_bytes, u8* data_buffer);
|
||||||
|
|
@ -242,24 +395,39 @@ extern "C" void osPfsReadWriteFile_recomp(uint8_t* rdram, recomp_context* ctx) {
|
||||||
s32 size_in_bytes = (s32) MEM_W(0x10, ctx->r29);
|
s32 size_in_bytes = (s32) MEM_W(0x10, ctx->r29);
|
||||||
u8* data_buffer = TO_PTR(u8, MEM_W(0x14, ctx->r29));
|
u8* data_buffer = TO_PTR(u8, MEM_W(0x14, ctx->r29));
|
||||||
|
|
||||||
ControllerPak& pak = sControllerPak[file_no];
|
ControllerPak pak;
|
||||||
|
|
||||||
|
char filename[100];
|
||||||
|
sprintf(filename, "controllerPak_file_%d.sav", file_no);
|
||||||
|
pak.file.open(filename, std::ios::binary | std::ios::in | std::ios::out);
|
||||||
|
|
||||||
|
if (!pak.file.good()) {
|
||||||
|
printf("file is not good!\n");
|
||||||
|
ctx->r2 = 5; // PFS_ERR_INVALID // VALIDATE
|
||||||
|
return;
|
||||||
|
// assert(false);
|
||||||
|
}
|
||||||
|
if (!pak.file.is_open()) {
|
||||||
|
printf("file isn't open!\n");
|
||||||
|
ctx->r2 = 5; // PFS_ERR_INVALID // VALIDATE
|
||||||
|
return;
|
||||||
|
// assert(false);
|
||||||
|
}
|
||||||
|
|
||||||
if (flag == 0) {
|
if (flag == 0) {
|
||||||
// fseek(pak.file, offset, SEEK_SET);
|
|
||||||
pak.file.seekg(offset, std::ios::beg);
|
pak.file.seekg(offset, std::ios::beg);
|
||||||
// fread(data_buffer, size_in_bytes, 1, pak.file);
|
|
||||||
pak.file.read((char*) data_buffer, size_in_bytes);
|
pak.file.read((char*) data_buffer, size_in_bytes);
|
||||||
// TODO: use a separate buffer for holding the swapped memory
|
// TODO: use a separate buffer for holding the swapped memory
|
||||||
ByteSwapFile(data_buffer, size_in_bytes);
|
ByteSwapFile(data_buffer, size_in_bytes);
|
||||||
} else {
|
} else {
|
||||||
// TODO: use a separate buffer for holding the swapped memory
|
// TODO: use a separate buffer for holding the swapped memory
|
||||||
ByteSwapFile(data_buffer, size_in_bytes);
|
ByteSwapFile(data_buffer, size_in_bytes);
|
||||||
// fseek(pak.file, offset, SEEK_SET);
|
pak.file.seekp(offset, std::ios::beg);
|
||||||
pak.file.seekg(offset, std::ios::beg);
|
|
||||||
// fwrite(data_buffer, size_in_bytes, 1, pak.file);
|
|
||||||
pak.file.write((char*) data_buffer, size_in_bytes);
|
pak.file.write((char*) data_buffer, size_in_bytes);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pak.file.close();
|
||||||
|
|
||||||
ctx->r2 = 0; // PFS_NO_ERROR
|
ctx->r2 = 0; // PFS_NO_ERROR
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -272,19 +440,128 @@ extern "C" void osPfsNumFiles_recomp(uint8_t* rdram, recomp_context* ctx) {
|
||||||
s32* max_files = _arg<1, s32*>(rdram, ctx);
|
s32* max_files = _arg<1, s32*>(rdram, ctx);
|
||||||
s32* files_used = _arg<2, s32*>(rdram, ctx);
|
s32* files_used = _arg<2, s32*>(rdram, ctx);
|
||||||
|
|
||||||
|
ControllerPak pak;
|
||||||
|
|
||||||
|
pak.header.open("controllerPak_header.sav", std::ios::binary | std::ios::in | std::ios::out);
|
||||||
|
|
||||||
|
if (!pak.header.good()) {
|
||||||
|
printf("file is not good!\n");
|
||||||
|
assert(false);
|
||||||
|
}
|
||||||
|
if (!pak.header.is_open()) {
|
||||||
|
printf("file isn't open!\n");
|
||||||
|
assert(false);
|
||||||
|
}
|
||||||
|
|
||||||
u8 files = 0;
|
u8 files = 0;
|
||||||
for (size_t i = 0; i < ARRAYSIZE(sControllerPak); i++) {
|
for (size_t i = 0; i < 16; i++) {
|
||||||
if ((sControllerPak[i].state.company_code != 0) && (sControllerPak[i].state.game_code != 0)) {
|
u32 seek = i * sizeof(OSPfsState);
|
||||||
|
u32 game_code = 0;
|
||||||
|
u16 company_code = 0;
|
||||||
|
|
||||||
|
// game_code
|
||||||
|
pak.header.seekg(seek + 0x4, std::ios::beg);
|
||||||
|
pak.header.read((char*) &game_code, sizeof(game_code));
|
||||||
|
// company_code
|
||||||
|
pak.header.seekg(seek + 0x08, std::ios::beg);
|
||||||
|
pak.header.read((char*) &company_code, sizeof(company_code));
|
||||||
|
|
||||||
|
if ((company_code != 0) || (game_code != 0)) {
|
||||||
files++;
|
files++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pak.header.close();
|
||||||
|
|
||||||
*files_used = files;
|
*files_used = files;
|
||||||
*max_files = ARRAYSIZE(sControllerPak);
|
*max_files = MAX_FILES;
|
||||||
|
|
||||||
ctx->r2 = 0; // PFS_NO_ERROR
|
ctx->r2 = 0; // PFS_NO_ERROR
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: Doesn't work, the game is sending NULL game_name, ext_name and company_code for some reason?
|
||||||
|
// s32 osPfsDeleteFile(OSPfs* pfs, u16 company_code, u32 game_code, u8* game_name, u8* ext_name);
|
||||||
|
extern "C" void osPfsDeleteFile_recomp(uint8_t* rdram, recomp_context* ctx) {
|
||||||
|
u16 company_code = _arg<1, u16>(rdram, ctx);
|
||||||
|
u32 game_code = _arg<2, u32>(rdram, ctx);
|
||||||
|
u8* game_name = _arg<3, u8*>(rdram, ctx);
|
||||||
|
u8* ext_name = TO_PTR(u8, MEM_W(0x10, ctx->r29));
|
||||||
|
|
||||||
|
printf("osPfsDeleteFile_recomp:\n");
|
||||||
|
|
||||||
|
if (company_code == 0 || game_code == 0) {
|
||||||
|
ctx->r2 = 5; // PFS_ERR_INVALID
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ControllerPak pak;
|
||||||
|
|
||||||
|
pak.header.open("controllerPak_header.sav", std::ios::binary | std::ios::in | std::ios::out);
|
||||||
|
|
||||||
|
if (!pak.header.good()) {
|
||||||
|
printf("file is not good!\n");
|
||||||
|
assert(false);
|
||||||
|
}
|
||||||
|
if (!pak.header.is_open()) {
|
||||||
|
printf("file isn't open!\n");
|
||||||
|
assert(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int i = 0; i < MAX_FILES; i++) {
|
||||||
|
// locals:
|
||||||
|
u32 game_code_;
|
||||||
|
u16 company_code_;
|
||||||
|
char ext_name_[4];
|
||||||
|
char game_name_[16];
|
||||||
|
u32 seek = i * sizeof(OSPfsState);
|
||||||
|
|
||||||
|
// game_code
|
||||||
|
pak.header.seekg(seek + 0x4, std::ios::beg);
|
||||||
|
pak.header.read((char*) &game_code_, sizeof(game_code_));
|
||||||
|
// company_code
|
||||||
|
pak.header.seekg(seek + 0x08, std::ios::beg);
|
||||||
|
pak.header.read((char*) &company_code_, sizeof(company_code_));
|
||||||
|
// ext_name
|
||||||
|
pak.header.seekg(seek + 0x0C, std::ios::beg);
|
||||||
|
pak.header.read((char*) ext_name_, 4);
|
||||||
|
// game_name
|
||||||
|
pak.header.seekg(seek + 0x10, std::ios::beg);
|
||||||
|
pak.header.read((char*) game_name_, 16);
|
||||||
|
|
||||||
|
if ((company_code_ == 0) || (game_code_ == 0)) {
|
||||||
|
continue;
|
||||||
|
} else {
|
||||||
|
if ((game_code == game_code_) && (company_code == company_code_) &&
|
||||||
|
(strcmp((const char*) game_name, (const char*) game_name_) == 0) &&
|
||||||
|
strcmp((const char*) ext_name, (const char*) ext_name_) == 0) {
|
||||||
|
// File found
|
||||||
|
|
||||||
|
// Zero out the header for this file.
|
||||||
|
u8* zero_block = (u8*) malloc(sizeof(OSPfsState));
|
||||||
|
memset(zero_block, 0, sizeof(OSPfsState));
|
||||||
|
pak.header.seekp(seek + 0x0, std::ios::beg);
|
||||||
|
pak.header.write((char*) &zero_block, sizeof(OSPfsState));
|
||||||
|
free(zero_block);
|
||||||
|
|
||||||
|
pak.header.close();
|
||||||
|
|
||||||
|
char filename[100];
|
||||||
|
sprintf(filename, "controllerPak_file_%d.sav", i);
|
||||||
|
remove(filename);
|
||||||
|
|
||||||
|
ctx->r2 = 0; // PFS_NO_ERROR
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pak.header.close();
|
||||||
|
|
||||||
|
// File not found
|
||||||
|
ctx->r2 = 5; // PFS_ERR_INVALID
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
extern "C" void osPfsRepairId_recomp(uint8_t* rdram, recomp_context* ctx) {
|
extern "C" void osPfsRepairId_recomp(uint8_t* rdram, recomp_context* ctx) {
|
||||||
_return<s32>(ctx, 1); // PFS_ERR_NOPACK
|
_return<s32>(ctx, 1); // PFS_ERR_NOPACK
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Reference in a new issue