/**************************************************************************** * Visual Boy Advance GX * * Tantric September 2008 * * filesel.cpp * * Generic file routines - reading, writing, browsing ***************************************************************************/ #include #include #include #include #include #include #include #ifdef WII_DVD extern "C" { #include } #endif #include "vba.h" #include "vbasupport.h" #include "vmmem.h" #include "menudraw.h" #include "video.h" #include "fileop.h" #include "memcardop.h" #include "input.h" #include "dvd.h" #include "smbop.h" #include "gcunzip.h" int offset; int selection; char currentdir[MAXPATHLEN]; char szpath[MAXPATHLEN]; int maxfiles; extern int screenheight; int havedir = -1; extern u64 dvddir; extern int dvddirlength; int hasloaded = 0; // Global file entry table FILEENTRIES filelist[MAXFILES]; bool inSz = false; char ROMFilename[512]; bool ROMLoaded = false; unsigned char *savebuffer = NULL; /**************************************************************************** * AllocSaveBuffer () * Clear and allocate the savebuffer ***************************************************************************/ void AllocSaveBuffer () { if (savebuffer != NULL) free(savebuffer); savebuffer = (unsigned char *) memalign(32, SAVEBUFFERSIZE); memset (savebuffer, 0, SAVEBUFFERSIZE); } /**************************************************************************** * FreeSaveBuffer () * Free the savebuffer memory ***************************************************************************/ void FreeSaveBuffer () { if (savebuffer != NULL) free(savebuffer); savebuffer = NULL; } /**************************************************************************** * autoLoadMethod() * Auto-determines and sets the load method * Returns method set ****************************************************************************/ int autoLoadMethod() { ShowAction ((char*) "Attempting to determine load method..."); if(ChangeFATInterface(METHOD_SD, SILENT)) return METHOD_SD; else if(ChangeFATInterface(METHOD_USB, SILENT)) return METHOD_USB; else if(TestDVD()) return METHOD_DVD; else if(ConnectShare (SILENT)) return METHOD_SMB; else { WaitPrompt((char*) "Unable to auto-determine load method!"); return 0; // no method found } } /**************************************************************************** * autoSaveMethod() * Auto-determines and sets the save method * Returns method set ****************************************************************************/ int autoSaveMethod() { ShowAction ((char*) "Attempting to determine save method..."); if(ChangeFATInterface(METHOD_SD, SILENT)) return METHOD_SD; else if(ChangeFATInterface(METHOD_USB, SILENT)) return METHOD_USB; else if(TestCard(CARD_SLOTA, SILENT)) return METHOD_MC_SLOTA; else if(TestCard(CARD_SLOTB, SILENT)) return METHOD_MC_SLOTB; else if(ConnectShare (SILENT)) return METHOD_SMB; else { WaitPrompt((char*) "Unable to auto-determine save method!"); return 0; // no method found } } /**************************************************************************** * UpdateDirName() * Update curent directory name for file browser ***************************************************************************/ int UpdateDirName(int method) { int size=0; char * test; char temp[1024]; // update DVD directory if(method == METHOD_DVD) { dvddir = filelist[selection].offset; dvddirlength = filelist[selection].length; } /* current directory doesn't change */ if (strcmp(filelist[selection].filename,".") == 0) { return 0; } /* go up to parent directory */ else if (strcmp(filelist[selection].filename,"..") == 0) { /* determine last subdirectory namelength */ sprintf(temp,"%s",currentdir); test = strtok(temp,"/"); while (test != NULL) { size = strlen(test); test = strtok(NULL,"/"); } /* remove last subdirectory name */ size = strlen(currentdir) - size - 1; currentdir[size] = 0; return 1; } /* Open a directory */ else { /* test new directory namelength */ if ((strlen(currentdir)+1+strlen(filelist[selection].filename)) < MAXPATHLEN) { /* update current directory name */ sprintf(currentdir, "%s/%s",currentdir, filelist[selection].filename); return 1; } else { WaitPrompt((char*)"Directory name is too long!"); return -1; } } } bool MakeFilePath(char filepath[], int type, int method) { char file[512]; char folder[1024]; char temppath[MAXPATHLEN]; if(type == FILE_ROM) { // Check path length if ((strlen(currentdir)+1+strlen(filelist[selection].filename)) >= MAXPATHLEN) { WaitPrompt((char*)"Maximum filepath length reached!"); filepath[0] = 0; return false; } else { sprintf(temppath, "%s/%s",currentdir,filelist[selection].filename); } } else { switch(type) { case FILE_SRAM: sprintf(folder, GCSettings.SaveFolder); sprintf(file, "%s.sav", ROMFilename); break; case FILE_SNAPSHOT: sprintf(folder, GCSettings.SaveFolder); sprintf(file, "%s.sgm", ROMFilename); break; case FILE_CHEAT: sprintf(folder, GCSettings.CheatFolder); sprintf(file, "%s.cht", ROMFilename); break; case FILE_PREF: sprintf(folder, appPath); sprintf(file, "%s", PREF_FILE_NAME); break; } switch(method) { case METHOD_SD: case METHOD_USB: sprintf (temppath, "%s/%s/%s", ROOTFATDIR, folder, file); break; case METHOD_DVD: case METHOD_SMB: sprintf (temppath, "%s/%s", folder, file); break; case METHOD_MC_SLOTA: case METHOD_MC_SLOTB: sprintf (temppath, "%s", file); temppath[31] = 0; // truncate filename break; } } strcpy(filepath, temppath); return true; } int LoadFile(char * buffer, char filepath[], int length, int method, bool silent) { int offset = 0; switch(method) { case METHOD_SD: case METHOD_USB: if(ChangeFATInterface(method, NOTSILENT)) offset = LoadFATFile (buffer, filepath, length, silent); break; case METHOD_SMB: offset = LoadSMBFile (buffer, filepath, length, silent); break; case METHOD_DVD: offset = LoadDVDFile (buffer, filepath, length, silent); break; case METHOD_MC_SLOTA: offset = LoadMCFile (buffer, CARD_SLOTA, filepath, silent); break; case METHOD_MC_SLOTB: offset = LoadMCFile (buffer, CARD_SLOTB, filepath, silent); break; } return offset; } int LoadFile(char filepath[], int method, bool silent) { return LoadFile((char *)savebuffer, filepath, 0, method, silent); } int SaveFile(char * buffer, char filepath[], int datasize, int method, bool silent) { int offset = 0; switch(method) { case METHOD_SD: case METHOD_USB: if(ChangeFATInterface(method, NOTSILENT)) offset = SaveFATFile (buffer, filepath, datasize, silent); break; case METHOD_SMB: offset = SaveSMBFile (buffer, filepath, datasize, silent); break; case METHOD_MC_SLOTA: offset = SaveMCFile (buffer, CARD_SLOTA, filepath, datasize, silent); break; case METHOD_MC_SLOTB: offset = SaveMCFile (buffer, CARD_SLOTB, filepath, datasize, silent); break; } return offset; } int SaveFile(char filepath[], int datasize, int method, bool silent) { return SaveFile((char *)savebuffer, filepath, datasize, method, silent); } /**************************************************************************** * FileSortCallback * * Quick sort callback to sort file entries with the following order: * . * .. * * ***************************************************************************/ int FileSortCallback(const void *f1, const void *f2) { /* Special case for implicit directories */ if(((FILEENTRIES *)f1)->filename[0] == '.' || ((FILEENTRIES *)f2)->filename[0] == '.') { if(strcmp(((FILEENTRIES *)f1)->filename, ".") == 0) { return -1; } if(strcmp(((FILEENTRIES *)f2)->filename, ".") == 0) { return 1; } if(strcmp(((FILEENTRIES *)f1)->filename, "..") == 0) { return -1; } if(strcmp(((FILEENTRIES *)f2)->filename, "..") == 0) { return 1; } } /* If one is a file and one is a directory the directory is first. */ if(((FILEENTRIES *)f1)->flags && !(((FILEENTRIES *)f2)->flags)) return -1; if(!(((FILEENTRIES *)f1)->flags) && ((FILEENTRIES *)f2)->flags) return 1; return stricmp(((FILEENTRIES *)f1)->filename, ((FILEENTRIES *)f2)->filename); } /**************************************************************************** * IsSz * * Checks if the specified file is a 7z ***************************************************************************/ bool IsSz() { if (strlen(filelist[selection].filename) > 4) { char * p = strrchr(filelist[selection].filename, '.'); if (p != NULL) if(stricmp(p, ".7z") == 0) return true; } return false; } /**************************************************************************** * StripExt * * Strips an extension from a filename ***************************************************************************/ void StripExt(char* returnstring, char * inputstring) { char* loc_dot; strcpy (returnstring, inputstring); loc_dot = strrchr(returnstring,'.'); if (loc_dot != NULL) *loc_dot = 0; // strip file extension } /**************************************************************************** * FileSelector * * Let user select a file from the listing ***************************************************************************/ int FileSelector (int method) { u32 p = 0; u32 wp = 0; u32 ph = 0; u32 wh = 0; signed char gc_ay = 0; signed char gc_sx = 0; signed char wm_ay = 0; signed char wm_sx = 0; int haverom = 0; int redraw = 1; int selectit = 0; int scroll_delay = 0; bool move_selection = 0; #define SCROLL_INITIAL_DELAY 15 #define SCROLL_LOOP_DELAY 2 while (haverom == 0) { if (redraw) ShowFiles (filelist, maxfiles, offset, selection); redraw = 0; VIDEO_WaitVSync(); // slow things down a bit so we don't overread the pads gc_ay = PAD_StickY (0); gc_sx = PAD_SubStickX (0); p = PAD_ButtonsDown (0); ph = PAD_ButtonsHeld (0); #ifdef HW_RVL wm_ay = WPAD_Stick (0, 0, 0); wm_sx = WPAD_Stick (0, 1, 1); wp = WPAD_ButtonsDown (0); wh = WPAD_ButtonsHeld (0); #endif /*** Check for exit combo ***/ if ( (gc_sx < -70) || (wm_sx < -70) || (wp & WPAD_BUTTON_HOME) || (wp & WPAD_CLASSIC_BUTTON_HOME) ) return 0; /*** Check buttons, perform actions ***/ if ( (p & PAD_BUTTON_A) || selectit || (wp & (WPAD_BUTTON_A | WPAD_CLASSIC_BUTTON_A)) ) { if ( selectit ) selectit = 0; if (filelist[selection].flags) // This is directory { /* update current directory and set new entry list if directory has changed */ int status; if(inSz && selection == 0) // inside a 7z, requesting to leave { if(method == METHOD_DVD) { // go to directory the 7z was in dvddir = filelist[0].offset; dvddirlength = filelist[0].length; } inSz = false; status = 1; SzClose(); } else { status = UpdateDirName(method); } if (status == 1) // ok, open directory { switch (method) { case METHOD_SD: case METHOD_USB: maxfiles = ParseFATdirectory(method); break; case METHOD_DVD: maxfiles = ParseDVDdirectory(); break; case METHOD_SMB: maxfiles = ParseSMBdirectory(); break; } if (!maxfiles) { WaitPrompt ((char*) "Error reading directory!"); haverom = 1; // quit menu } } else if (status == -1) // directory name too long { haverom = 1; // quit menu } } else // this is a file { // better do another unmount/remount, just in case if(method == METHOD_SD || method == METHOD_USB) if(!ChangeFATInterface(method, NOTSILENT)) return 0; // 7z file - let's open it up to select a file inside if(IsSz()) { // we'll store the 7z filepath for extraction later if(!MakeFilePath(szpath, FILE_ROM, method)) return 0; int szfiles = SzParse(szpath, method); if(szfiles) { maxfiles = szfiles; inSz = true; } else WaitPrompt((char*) "Error opening archive!"); } else { // store the filename (w/o ext) - used for sram/freeze naming StripExt(ROMFilename, filelist[selection].filename); ShowAction ((char *)"Loading..."); ROMLoaded = LoadVBAROM(method); inSz = false; if (ROMLoaded) { return 1; } else { WaitPrompt((char*) "Error loading ROM!"); } } } redraw = 1; } // End of A if ( (p & PAD_BUTTON_B) || (wp & (WPAD_BUTTON_B | WPAD_CLASSIC_BUTTON_B)) ) { while ( (PAD_ButtonsDown(0) & PAD_BUTTON_B) #ifdef HW_RVL || (WPAD_ButtonsDown(0) & (WPAD_BUTTON_B | WPAD_CLASSIC_BUTTON_B)) #endif ) VIDEO_WaitVSync(); if ( strcmp(filelist[0].filename,"..") == 0 ) { selection = 0; selectit = 1; } else if ( strcmp(filelist[1].filename,"..") == 0 ) { selection = selectit = 1; } else { return 0; } } // End of B if ( ((p | ph) & PAD_BUTTON_DOWN) || ((wp | wh) & (WPAD_BUTTON_DOWN | WPAD_CLASSIC_BUTTON_DOWN)) || (gc_ay < -PADCAL) || (wm_ay < -PADCAL) ) { if ( (p & PAD_BUTTON_DOWN) || (wp & (WPAD_BUTTON_DOWN | WPAD_CLASSIC_BUTTON_DOWN)) ) { /*** Button just pressed ***/ scroll_delay = SCROLL_INITIAL_DELAY; // reset scroll delay. move_selection = 1; //continue (move selection) } else if (scroll_delay == 0) { /*** Button is held ***/ scroll_delay = SCROLL_LOOP_DELAY; move_selection = 1; //continue (move selection) } else { scroll_delay--; // wait } if (move_selection) { selection++; if (selection == maxfiles) selection = offset = 0; if ((selection - offset) >= PAGESIZE) offset += PAGESIZE; redraw = 1; move_selection = 0; } } // End of down if ( ((p | ph) & PAD_BUTTON_UP) || ((wp | wh) & (WPAD_BUTTON_UP | WPAD_CLASSIC_BUTTON_UP)) || (gc_ay > PADCAL) || (wm_ay > PADCAL) ) { if ( (p & PAD_BUTTON_UP) || (wp & (WPAD_BUTTON_UP | WPAD_CLASSIC_BUTTON_UP)) ) { /*** Button just pressed***/ scroll_delay = SCROLL_INITIAL_DELAY; // reset scroll delay. move_selection = 1; //continue (move selection) } else if (scroll_delay == 0) { /*** Button is held ***/ scroll_delay = SCROLL_LOOP_DELAY; move_selection = 1; //continue (move selection) } else { scroll_delay--; // wait } if (move_selection) { selection--; if (selection < 0) { selection = maxfiles - 1; offset = selection - PAGESIZE + 1; } if (selection < offset) offset -= PAGESIZE; if (offset < 0) offset = 0; redraw = 1; move_selection = 0; } } // End of Up if ( (p & PAD_BUTTON_LEFT) || (wp & (WPAD_BUTTON_LEFT | WPAD_CLASSIC_BUTTON_LEFT)) ) { /*** Go back a page ***/ selection -= PAGESIZE; if (selection < 0) { selection = maxfiles - 1; offset = selection - PAGESIZE + 1; } if (selection < offset) offset -= PAGESIZE; if (offset < 0) offset = 0; redraw = 1; } if ( (p & PAD_BUTTON_RIGHT) || (wp & (WPAD_BUTTON_RIGHT | WPAD_CLASSIC_BUTTON_RIGHT)) ) { /*** Go forward a page ***/ selection += PAGESIZE; if (selection > maxfiles - 1) selection = offset = 0; if ((selection - offset) >= PAGESIZE) offset += PAGESIZE; redraw = 1; } } return 0; } /**************************************************************************** * OpenDVD * * Function to load a DVD directory and display to user. ***************************************************************************/ int OpenDVD (int method) { if (!getpvd()) { ShowAction((char*) "Loading DVD..."); #ifdef HW_DOL DVD_Mount(); // mount the DVD unit again #elif WII_DVD u32 val; DI_GetCoverRegister(&val); if(val & 0x1) // True if no disc inside, use (val & 0x2) for true if disc inside. { WaitPrompt((char *)"No disc inserted!"); return 0; } DI_Mount(); while(DI_GetStatus() & DVD_INIT); #endif if (!getpvd()) { WaitPrompt ((char *)"Invalid DVD."); return 0; // not a ISO9660 DVD } } currentdir[0] = 0; maxfiles = ParseDVDdirectory(); // load root folder // switch to rom folder SwitchDVDFolder(GCSettings.LoadFolder); if (maxfiles > 0) { return FileSelector (method); } else { // no entries found WaitPrompt ((char *)"No Files Found!"); return 0; } } /**************************************************************************** * OpenSMB * * Function to load from an SMB share ***************************************************************************/ int OpenSMB (int method) { // Connect to network share if(ConnectShare (NOTSILENT)) { // change current dir to root dir sprintf(currentdir, "/%s", GCSettings.LoadFolder); maxfiles = ParseSMBdirectory (); if (maxfiles > 0) { return FileSelector (method); } else { // no entries found WaitPrompt ((char *)"No Files Found!"); return 0; } } return 0; } /**************************************************************************** * OpenFAT * * Function to load from FAT ***************************************************************************/ int OpenFAT (int method) { if(ChangeFATInterface(method, NOTSILENT)) { // change current dir to vba roms directory sprintf ( currentdir, "%s/%s", ROOTFATDIR, GCSettings.LoadFolder ); // Parse initial root directory and get entries list maxfiles = ParseFATdirectory (method); if (maxfiles > 0) { // Select an entry return FileSelector (method); } else { // no entries found WaitPrompt ((char *)"No Files Found!"); return 0; } } return 0; } /**************************************************************************** * OpenROM * Opens device specified by method, displays a list of ROMS ***************************************************************************/ int OpenROM (int method) { int loadROM = 0; if(method == METHOD_AUTO) method = autoLoadMethod(); switch (method) { case METHOD_SD: case METHOD_USB: loadROM = OpenFAT (method); break; case METHOD_DVD: // Load from DVD loadROM = OpenDVD (method); break; case METHOD_SMB: // Load from Network (SMB) loadROM = OpenSMB (method); break; } return loadROM; }