2008-08-06 01:09:59 +00:00
|
|
|
/****************************************************************************
|
2008-09-12 05:28:40 +00:00
|
|
|
* Snes9x 1.51 Nintendo Wii/Gamecube Port
|
2008-08-13 22:44:59 +00:00
|
|
|
*
|
2008-08-06 01:09:59 +00:00
|
|
|
* softdev July 2006
|
|
|
|
* crunchy2 May 2007
|
2008-10-14 09:21:34 +00:00
|
|
|
* Michniewski 2008
|
2008-08-10 03:14:39 +00:00
|
|
|
* Tantric August 2008
|
2008-08-06 01:09:59 +00:00
|
|
|
*
|
2008-08-07 05:19:17 +00:00
|
|
|
* fileop.cpp
|
2008-08-06 01:09:59 +00:00
|
|
|
*
|
2008-09-12 05:28:40 +00:00
|
|
|
* FAT File operations
|
|
|
|
***************************************************************************/
|
|
|
|
|
2008-08-06 01:09:59 +00:00
|
|
|
#include <gccore.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <ogcsys.h>
|
2008-08-10 03:14:39 +00:00
|
|
|
#include <zlib.h>
|
|
|
|
#include "memmap.h"
|
|
|
|
|
2008-08-07 05:19:17 +00:00
|
|
|
#include "fileop.h"
|
2008-08-06 01:55:59 +00:00
|
|
|
#include "unzip.h"
|
2008-08-06 01:09:59 +00:00
|
|
|
#include "video.h"
|
2008-08-07 05:19:17 +00:00
|
|
|
#include "menudraw.h"
|
2008-08-06 01:09:59 +00:00
|
|
|
#include "filesel.h"
|
|
|
|
#include "sram.h"
|
|
|
|
#include "preferences.h"
|
2008-09-09 17:36:48 +00:00
|
|
|
#include "snes9xGX.h"
|
2008-08-06 01:09:59 +00:00
|
|
|
|
2008-10-14 09:21:34 +00:00
|
|
|
// FAT file pointer - the only one we should ever use!
|
|
|
|
FILE * fatfile;
|
2008-08-06 01:55:59 +00:00
|
|
|
|
2008-08-07 03:25:02 +00:00
|
|
|
/****************************************************************************
|
2008-10-28 06:52:38 +00:00
|
|
|
* MountFAT
|
|
|
|
* Attempts to mount the FAT device specified
|
|
|
|
* Sets libfat to use the device by default
|
|
|
|
* Enables read-ahead cache for SD/USB
|
2008-09-12 05:28:40 +00:00
|
|
|
***************************************************************************/
|
2008-10-28 06:52:38 +00:00
|
|
|
bool MountFAT(PARTITION_INTERFACE part)
|
|
|
|
{
|
|
|
|
bool mounted = fatMountNormalInterface(part, 8);
|
|
|
|
|
|
|
|
if(mounted)
|
|
|
|
{
|
|
|
|
fatSetDefaultInterface(part);
|
|
|
|
#ifdef HW_RVL
|
|
|
|
if(part == PI_INTERNAL_SD || part == PI_USBSTORAGE)
|
|
|
|
fatEnableReadAhead (part, 6, 64);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
return mounted;
|
|
|
|
}
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-10-28 06:52:38 +00:00
|
|
|
/****************************************************************************
|
|
|
|
* UnmountFAT
|
|
|
|
* Unmounts the FAT device specified
|
|
|
|
***************************************************************************/
|
|
|
|
void UnmountFAT(PARTITION_INTERFACE part)
|
|
|
|
{
|
|
|
|
if(!fatUnmount(part))
|
|
|
|
fatUnsafeUnmount(part);
|
2008-08-07 03:25:02 +00:00
|
|
|
}
|
|
|
|
|
2008-08-08 07:57:01 +00:00
|
|
|
/****************************************************************************
|
2008-10-28 06:52:38 +00:00
|
|
|
* UnmountAllFAT
|
|
|
|
* Unmounts all FAT devices
|
|
|
|
***************************************************************************/
|
|
|
|
void UnmountAllFAT()
|
|
|
|
{
|
|
|
|
#ifdef HW_RVL
|
|
|
|
UnmountFAT(PI_INTERNAL_SD);
|
|
|
|
UnmountFAT(PI_USBSTORAGE);
|
|
|
|
#endif
|
|
|
|
UnmountFAT(PI_SDGECKO_A);
|
|
|
|
UnmountFAT(PI_SDGECKO_B);
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* ChangeFATInterface
|
|
|
|
* Unmounts all devices and attempts to mount/configure the device specified
|
2008-09-12 05:28:40 +00:00
|
|
|
***************************************************************************/
|
2008-08-12 03:25:16 +00:00
|
|
|
bool ChangeFATInterface(int method, bool silent)
|
2008-08-08 07:57:01 +00:00
|
|
|
{
|
2008-10-28 06:52:38 +00:00
|
|
|
bool mounted = false;
|
|
|
|
|
|
|
|
// unmount all FAT devices
|
|
|
|
UnmountAllFAT();
|
2008-08-08 07:57:01 +00:00
|
|
|
|
|
|
|
if(method == METHOD_SD)
|
|
|
|
{
|
|
|
|
#ifdef HW_RVL
|
2008-10-28 06:52:38 +00:00
|
|
|
mounted = MountFAT(PI_INTERNAL_SD); // try Wii internal SD
|
2008-08-08 07:57:01 +00:00
|
|
|
#endif
|
|
|
|
|
2008-10-28 06:52:38 +00:00
|
|
|
if(!mounted) // internal SD not found
|
|
|
|
mounted = MountFAT(PI_SDGECKO_A); // try SD Gecko on slot A
|
|
|
|
if(!mounted) // internal SD and SD Gecko (on slot A) not found
|
|
|
|
mounted = MountFAT(PI_SDGECKO_B); // try SD Gecko on slot B
|
|
|
|
if(!mounted && !silent) // no SD device found
|
|
|
|
WaitPrompt ((char *)"SD card not found!");
|
2008-08-08 07:57:01 +00:00
|
|
|
}
|
|
|
|
else if(method == METHOD_USB)
|
|
|
|
{
|
|
|
|
#ifdef HW_RVL
|
2008-10-28 06:52:38 +00:00
|
|
|
mounted = MountFAT(PI_USBSTORAGE);
|
|
|
|
if(!mounted && !silent)
|
|
|
|
WaitPrompt ((char *)"USB drive not found!");
|
2008-08-08 07:57:01 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2008-10-28 06:52:38 +00:00
|
|
|
return mounted;
|
2008-08-08 07:57:01 +00:00
|
|
|
}
|
|
|
|
|
2008-08-06 01:55:59 +00:00
|
|
|
/***************************************************************************
|
2008-08-10 03:14:39 +00:00
|
|
|
* Browse FAT subdirectories
|
2008-09-12 05:28:40 +00:00
|
|
|
**************************************************************************/
|
2008-08-10 03:14:39 +00:00
|
|
|
int
|
2008-08-12 03:25:16 +00:00
|
|
|
ParseFATdirectory(int method)
|
2008-08-06 01:55:59 +00:00
|
|
|
{
|
2008-08-10 03:14:39 +00:00
|
|
|
int nbfiles = 0;
|
|
|
|
DIR_ITER *fatdir;
|
|
|
|
char filename[MAXPATHLEN];
|
|
|
|
struct stat filestat;
|
|
|
|
char msg[128];
|
|
|
|
|
|
|
|
// initialize selection
|
|
|
|
selection = offset = 0;
|
|
|
|
|
2008-08-16 00:02:08 +00:00
|
|
|
// Clear any existing values
|
|
|
|
memset (&filelist, 0, sizeof (FILEENTRIES) * MAXFILES);
|
|
|
|
|
2008-08-10 03:14:39 +00:00
|
|
|
// open the directory
|
|
|
|
fatdir = diropen(currentdir);
|
|
|
|
if (fatdir == NULL)
|
2008-08-06 01:55:59 +00:00
|
|
|
{
|
2008-08-10 03:14:39 +00:00
|
|
|
sprintf(msg, "Couldn't open %s", currentdir);
|
|
|
|
WaitPrompt(msg);
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-08-10 03:14:39 +00:00
|
|
|
// if we can't open the dir, open root dir
|
|
|
|
sprintf(currentdir,"%s",ROOTFATDIR);
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-08-10 03:14:39 +00:00
|
|
|
fatdir = diropen(currentdir);
|
2008-08-06 01:55:59 +00:00
|
|
|
|
2008-08-10 03:14:39 +00:00
|
|
|
if (fatdir == NULL)
|
2008-08-07 03:25:02 +00:00
|
|
|
{
|
2008-08-10 03:14:39 +00:00
|
|
|
sprintf(msg, "Error opening %s", currentdir);
|
|
|
|
WaitPrompt(msg);
|
|
|
|
return 0;
|
2008-08-07 03:25:02 +00:00
|
|
|
}
|
|
|
|
}
|
2008-08-06 01:09:59 +00:00
|
|
|
|
2008-08-10 03:14:39 +00:00
|
|
|
// index files/folders
|
|
|
|
while(dirnext(fatdir,filename,&filestat) == 0)
|
2008-08-07 03:25:02 +00:00
|
|
|
{
|
2008-08-10 03:14:39 +00:00
|
|
|
if(strcmp(filename,".") != 0)
|
2008-08-07 03:25:02 +00:00
|
|
|
{
|
2008-08-10 03:14:39 +00:00
|
|
|
memset(&filelist[nbfiles], 0, sizeof(FILEENTRIES));
|
|
|
|
strncpy(filelist[nbfiles].filename, filename, MAXPATHLEN);
|
2008-08-06 01:39:43 +00:00
|
|
|
strncpy(filelist[nbfiles].displayname, filename, MAXDISPLAY+1); // crop name for display
|
2008-08-10 03:14:39 +00:00
|
|
|
filelist[nbfiles].length = filestat.st_size;
|
|
|
|
filelist[nbfiles].flags = (filestat.st_mode & _IFDIR) == 0 ? 0 : 1; // flag this as a dir
|
|
|
|
nbfiles++;
|
|
|
|
}
|
2008-08-06 01:39:43 +00:00
|
|
|
}
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-08-10 03:14:39 +00:00
|
|
|
// close directory
|
|
|
|
dirclose(fatdir);
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-08-10 03:14:39 +00:00
|
|
|
// Sort the file list
|
2008-08-06 01:55:59 +00:00
|
|
|
qsort(filelist, nbfiles, sizeof(FILEENTRIES), FileSortCallback);
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-08-10 03:14:39 +00:00
|
|
|
return nbfiles;
|
2008-08-06 01:09:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
2008-08-07 03:25:02 +00:00
|
|
|
* LoadFATFile
|
2008-09-12 05:28:40 +00:00
|
|
|
***************************************************************************/
|
2008-08-06 01:09:59 +00:00
|
|
|
int
|
2008-10-05 21:56:18 +00:00
|
|
|
LoadFATFile (char * rbuffer, int length)
|
2008-08-06 01:09:59 +00:00
|
|
|
{
|
2008-08-10 03:14:39 +00:00
|
|
|
char zipbuffer[2048];
|
|
|
|
char filepath[MAXPATHLEN];
|
|
|
|
u32 size;
|
|
|
|
|
|
|
|
/* Check filename length */
|
2008-10-14 09:21:34 +00:00
|
|
|
if (!MakeROMPath(filepath, METHOD_SD))
|
2008-08-10 03:14:39 +00:00
|
|
|
{
|
2008-08-12 03:25:16 +00:00
|
|
|
WaitPrompt((char*) "Maximum filepath length reached!");
|
2008-08-10 03:14:39 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2008-10-14 09:21:34 +00:00
|
|
|
fatfile = fopen (filepath, "rb");
|
|
|
|
if (fatfile > 0)
|
2008-08-10 03:14:39 +00:00
|
|
|
{
|
2008-10-15 06:35:14 +00:00
|
|
|
if(length > 0 && length <= 2048) // do a partial read (eg: to check file header)
|
2008-08-06 01:39:43 +00:00
|
|
|
{
|
2008-10-14 09:21:34 +00:00
|
|
|
fread (rbuffer, 1, length, fatfile);
|
2008-10-05 21:56:18 +00:00
|
|
|
size = length;
|
2008-08-06 01:39:43 +00:00
|
|
|
}
|
2008-10-05 21:56:18 +00:00
|
|
|
else // load whole file
|
2008-08-06 01:39:43 +00:00
|
|
|
{
|
2008-10-14 09:21:34 +00:00
|
|
|
fread (zipbuffer, 1, 2048, fatfile);
|
2008-10-05 21:56:18 +00:00
|
|
|
|
|
|
|
if (IsZipFile (zipbuffer))
|
|
|
|
{
|
2008-10-14 09:21:34 +00:00
|
|
|
size = UnZipBuffer ((unsigned char *)rbuffer, METHOD_SD); // unzip from FAT
|
2008-10-05 21:56:18 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Just load the file up
|
2008-10-14 09:21:34 +00:00
|
|
|
fseek(fatfile, 0, SEEK_END);
|
|
|
|
size = ftell(fatfile); // get filesize
|
|
|
|
fseek(fatfile, 2048, SEEK_SET); // seek back to point where we left off
|
2008-10-05 21:56:18 +00:00
|
|
|
memcpy (rbuffer, zipbuffer, 2048); // copy what we already read
|
2008-10-15 06:35:14 +00:00
|
|
|
|
2008-10-15 07:04:46 +00:00
|
|
|
ShowProgress ((char *)"Loading...", 2048, size);
|
|
|
|
|
2008-10-15 06:35:14 +00:00
|
|
|
u32 offset = 2048;
|
|
|
|
while(offset < size)
|
|
|
|
{
|
|
|
|
offset += fread (rbuffer + offset, 1, (1024*512), fatfile); // read in 512K chunks
|
|
|
|
ShowProgress ((char *)"Loading...", offset, size);
|
|
|
|
}
|
2008-10-05 21:56:18 +00:00
|
|
|
}
|
2008-08-06 01:39:43 +00:00
|
|
|
}
|
2008-10-14 09:21:34 +00:00
|
|
|
fclose (fatfile);
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
WaitPrompt((char*) "Error opening file");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* LoadFATSzFile
|
|
|
|
* Loads the selected file # from the specified 7z into rbuffer
|
|
|
|
* Returns file size
|
|
|
|
***************************************************************************/
|
|
|
|
int
|
|
|
|
LoadFATSzFile(char * filepath, unsigned char * rbuffer)
|
|
|
|
{
|
|
|
|
u32 size;
|
|
|
|
fatfile = fopen (filepath, "rb");
|
|
|
|
if (fatfile > 0)
|
|
|
|
{
|
|
|
|
size = SzExtractFile(filelist[selection].offset, rbuffer);
|
|
|
|
fclose (fatfile);
|
2008-08-22 02:47:08 +00:00
|
|
|
return size;
|
2008-08-10 03:14:39 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
WaitPrompt((char*) "Error opening file");
|
|
|
|
return 0;
|
|
|
|
}
|
2008-08-06 01:09:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
2008-08-07 03:25:02 +00:00
|
|
|
* Load savebuffer from FAT file
|
2008-09-12 05:28:40 +00:00
|
|
|
***************************************************************************/
|
2008-08-06 01:09:59 +00:00
|
|
|
int
|
2008-08-07 03:25:02 +00:00
|
|
|
LoadBufferFromFAT (char *filepath, bool silent)
|
2008-08-06 01:09:59 +00:00
|
|
|
{
|
2008-10-03 06:57:56 +00:00
|
|
|
int size = 0;
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-10-14 09:21:34 +00:00
|
|
|
fatfile = fopen (filepath, "rb");
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-10-14 09:21:34 +00:00
|
|
|
if (fatfile <= 0)
|
2008-08-06 01:09:59 +00:00
|
|
|
{
|
|
|
|
if ( !silent )
|
|
|
|
{
|
|
|
|
char msg[100];
|
|
|
|
sprintf(msg, "Couldn't open %s", filepath);
|
|
|
|
WaitPrompt (msg);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-10-03 05:26:01 +00:00
|
|
|
// Just load the file up
|
2008-10-14 09:21:34 +00:00
|
|
|
fseek(fatfile, 0, SEEK_END); // go to end of file
|
|
|
|
size = ftell(fatfile); // get filesize
|
|
|
|
fseek(fatfile, 0, SEEK_SET); // go to start of file
|
|
|
|
fread (savebuffer, 1, size, fatfile);
|
|
|
|
fclose (fatfile);
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-10-03 06:57:56 +00:00
|
|
|
return size;
|
2008-08-06 01:09:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
2008-08-07 03:25:02 +00:00
|
|
|
* Write savebuffer to FAT card file
|
2008-09-12 05:28:40 +00:00
|
|
|
***************************************************************************/
|
2008-08-06 01:09:59 +00:00
|
|
|
int
|
2008-08-07 03:25:02 +00:00
|
|
|
SaveBufferToFAT (char *filepath, int datasize, bool silent)
|
2008-08-06 01:09:59 +00:00
|
|
|
{
|
2008-10-14 09:21:34 +00:00
|
|
|
if (datasize)
|
2008-08-06 01:09:59 +00:00
|
|
|
{
|
2008-10-14 09:21:34 +00:00
|
|
|
fatfile = fopen (filepath, "wb");
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-10-14 09:21:34 +00:00
|
|
|
if (fatfile <= 0)
|
2008-08-06 01:09:59 +00:00
|
|
|
{
|
|
|
|
char msg[100];
|
|
|
|
sprintf(msg, "Couldn't save %s", filepath);
|
|
|
|
WaitPrompt (msg);
|
|
|
|
return 0;
|
|
|
|
}
|
2008-08-07 03:25:02 +00:00
|
|
|
|
2008-10-14 09:21:34 +00:00
|
|
|
fwrite (savebuffer, 1, datasize, fatfile);
|
|
|
|
fclose (fatfile);
|
2008-08-06 01:09:59 +00:00
|
|
|
}
|
2008-08-07 03:25:02 +00:00
|
|
|
return datasize;
|
2008-08-06 01:09:59 +00:00
|
|
|
}
|