mirror of
https://github.com/Oibaf66/frodo-wii.git
synced 2024-12-01 16:04:21 +01:00
542 lines
13 KiB
C++
542 lines
13 KiB
C++
|
/*
|
||
|
* file.cpp
|
||
|
*
|
||
|
* This file is taken from the ARAnyM project which builds a new and powerful
|
||
|
* TOS/FreeMiNT compatible virtual machine running on almost any hardware.
|
||
|
*
|
||
|
* It 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 2 of the License, or
|
||
|
* (at your option) any later version.
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License
|
||
|
* along with Frodo; if not, write to the Free Software
|
||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||
|
*/
|
||
|
|
||
|
#include "sysdeps.h"
|
||
|
#include "file.h"
|
||
|
|
||
|
#define DEBUG 0
|
||
|
#include "debug.h"
|
||
|
|
||
|
# include <cstdlib>
|
||
|
# include <cstdio>
|
||
|
# include <cstring>
|
||
|
|
||
|
#ifdef OS_solaris
|
||
|
#define DIRHANDLE dirp->d_fd
|
||
|
#elif defined(OS_mingw)
|
||
|
#define DIRHANDLE dirp->dd_handle
|
||
|
#else
|
||
|
#define DIRHANDLE dirp->fd
|
||
|
#endif
|
||
|
|
||
|
#if defined(__BEOS__) || defined(OS_solaris) || defined(OS_mingw)
|
||
|
/* The scandir() and alphasort() functions aren't available on BeOS, */
|
||
|
/* so let's declare them here... */
|
||
|
#ifndef OS_mingw
|
||
|
#include <strings.h>
|
||
|
#endif
|
||
|
|
||
|
#undef DIRSIZ
|
||
|
|
||
|
#define DIRSIZ(dp) \
|
||
|
((sizeof(struct dirent) - sizeof(dp)->d_name) + \
|
||
|
(((dp)->d_reclen + 1 + 3) &~ 3))
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Alphabetic order comparison routine for those who want it.
|
||
|
*/
|
||
|
int alphasort(const void *d1, const void *d2)
|
||
|
{
|
||
|
return(strcmp((*(struct dirent **)d1)->d_name, (*(struct dirent **)d2)->d_name));
|
||
|
}
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Scan a directory for all its entries
|
||
|
*/
|
||
|
int scandir(const char *dirname, struct dirent ***namelist, int(*select) (const struct dirent *), int (*dcomp) (const void *, const void *))
|
||
|
{
|
||
|
register struct dirent *d, *p, **names;
|
||
|
register size_t nitems;
|
||
|
struct stat stb;
|
||
|
unsigned long arraysz;
|
||
|
DIR *dirp;
|
||
|
|
||
|
if ((dirp = opendir(dirname)) == NULL)
|
||
|
return(-1);
|
||
|
|
||
|
if (fstat(DIRHANDLE, &stb) < 0)
|
||
|
return(-1);
|
||
|
|
||
|
/*
|
||
|
* estimate the array size by taking the size of the directory file
|
||
|
* and dividing it by a multiple of the minimum size entry.
|
||
|
*/
|
||
|
arraysz = (stb.st_size / 24);
|
||
|
|
||
|
names = (struct dirent **)malloc(arraysz * sizeof(struct dirent *));
|
||
|
if (names == NULL)
|
||
|
return(-1);
|
||
|
|
||
|
nitems = 0;
|
||
|
|
||
|
while ((d = readdir(dirp)) != NULL) {
|
||
|
|
||
|
if (select != NULL && !(*select)(d))
|
||
|
continue; /* just selected names */
|
||
|
|
||
|
/*
|
||
|
* Make a minimum size copy of the data
|
||
|
*/
|
||
|
|
||
|
p = (struct dirent *)malloc(DIRSIZ(d));
|
||
|
if (p == NULL)
|
||
|
return(-1);
|
||
|
|
||
|
p->d_ino = d->d_ino;
|
||
|
p->d_reclen = d->d_reclen;
|
||
|
/*p->d_namlen = d->d_namlen;*/
|
||
|
bcopy(d->d_name, p->d_name, p->d_reclen + 1);
|
||
|
|
||
|
/*
|
||
|
* Check to make sure the array has space left and
|
||
|
* realloc the maximum size.
|
||
|
*/
|
||
|
|
||
|
if (++nitems >= arraysz) {
|
||
|
|
||
|
if (fstat(DIRHANDLE, &stb) < 0)
|
||
|
return(-1); /* just might have grown */
|
||
|
|
||
|
arraysz = stb.st_size / 12;
|
||
|
|
||
|
names = (struct dirent **)realloc((char *)names, arraysz * sizeof(struct dirent *));
|
||
|
if (names == NULL)
|
||
|
return(-1);
|
||
|
}
|
||
|
|
||
|
names[nitems-1] = p;
|
||
|
}
|
||
|
|
||
|
closedir(dirp);
|
||
|
|
||
|
if (nitems && dcomp != NULL)
|
||
|
qsort(names, nitems, sizeof(struct dirent *), dcomp);
|
||
|
|
||
|
*namelist = names;
|
||
|
|
||
|
return(nitems);
|
||
|
}
|
||
|
|
||
|
|
||
|
#endif /* __BEOS__ */
|
||
|
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Remove any '/'s from end of filenames, but keeps / intact
|
||
|
*/
|
||
|
void File_CleanFileName(char *pszFileName)
|
||
|
{
|
||
|
int len;
|
||
|
|
||
|
len = strlen(pszFileName);
|
||
|
|
||
|
/* Security length check: */
|
||
|
if( len>MAX_FILENAME_LENGTH )
|
||
|
{
|
||
|
pszFileName[MAX_FILENAME_LENGTH-1] = 0;
|
||
|
len = MAX_FILENAME_LENGTH;
|
||
|
}
|
||
|
|
||
|
/* Remove end slash from filename! But / remains! Doh! */
|
||
|
if( len>2 && pszFileName[len-1]=='/' )
|
||
|
pszFileName[len-1] = 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Add '/' to end of filename
|
||
|
*/
|
||
|
void File_AddSlashToEndFileName(char *pszFileName)
|
||
|
{
|
||
|
/* Check dir/filenames */
|
||
|
if (strlen(pszFileName)!=0) {
|
||
|
if (pszFileName[strlen(pszFileName)-1]!='/')
|
||
|
strcat(pszFileName,"/"); /* Must use end slash */
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Does filename extension match? If so, return true
|
||
|
*/
|
||
|
bool File_DoesFileExtensionMatch(char *pszFileName, char *pszExtension)
|
||
|
{
|
||
|
if ( strlen(pszFileName) < strlen(pszExtension) )
|
||
|
return(false);
|
||
|
/* Is matching extension? */
|
||
|
if ( !strcasecmp(&pszFileName[strlen(pszFileName)-strlen(pszExtension)], pszExtension) )
|
||
|
return(true);
|
||
|
|
||
|
/* No */
|
||
|
return(false);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Check if filename is from root
|
||
|
|
||
|
Return true if filename is '/', else give false
|
||
|
*/
|
||
|
bool File_IsRootFileName(char *pszFileName)
|
||
|
{
|
||
|
if (pszFileName[0]=='\0') /* If NULL string return! */
|
||
|
return(false);
|
||
|
|
||
|
if (pszFileName[0]=='/')
|
||
|
return(true);
|
||
|
|
||
|
return(false);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Return string, to remove 'C:' part of filename
|
||
|
*/
|
||
|
char *File_RemoveFileNameDrive(char *pszFileName)
|
||
|
{
|
||
|
if ( (pszFileName[0]!='\0') && (pszFileName[1]==':') )
|
||
|
return(&pszFileName[2]);
|
||
|
else
|
||
|
return(pszFileName);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Check if filename end with a '/'
|
||
|
|
||
|
Return true if filename ends with '/'
|
||
|
*/
|
||
|
bool File_DoesFileNameEndWithSlash(char *pszFileName)
|
||
|
{
|
||
|
if (pszFileName[0]=='\0') /* If NULL string return! */
|
||
|
return(false);
|
||
|
|
||
|
/* Does string end in a '/'? */
|
||
|
if (pszFileName[strlen(pszFileName)-1]=='/')
|
||
|
return(true);
|
||
|
|
||
|
return(false);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Remove any double '/'s from end of filenames. So just the one
|
||
|
*/
|
||
|
void File_RemoveFileNameTrailingSlashes(char *pszFileName)
|
||
|
{
|
||
|
int Length;
|
||
|
|
||
|
/* Do have slash at end of filename? */
|
||
|
Length = strlen(pszFileName);
|
||
|
if (Length>=3) {
|
||
|
if (pszFileName[Length-1]=='/') { /* Yes, have one previous? */
|
||
|
if (pszFileName[Length-2]=='/')
|
||
|
pszFileName[Length-1] = '\0'; /* then remove it! */
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Does filename end with a .MSA extension? If so, return true
|
||
|
*/
|
||
|
bool File_FileNameIsMSA(char *pszFileName)
|
||
|
{
|
||
|
return(File_DoesFileExtensionMatch(pszFileName,".msa"));
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Does filename end with a .ST extension? If so, return true
|
||
|
*/
|
||
|
bool File_FileNameIsST(char *pszFileName)
|
||
|
{
|
||
|
return(File_DoesFileExtensionMatch(pszFileName,".st"));
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Read file from PC into memory, allocate memory for it if need to (pass Address as NULL)
|
||
|
Also may pass 'unsigned long' if want to find size of file read (may pass as NULL)
|
||
|
*/
|
||
|
void *File_Read(const char *pszFileName, void *pAddress, long *pFileSize, char *ppszExts[])
|
||
|
{
|
||
|
// TODO DUNUSED(ppszExts);
|
||
|
FILE *DiscFile;
|
||
|
void *pFile=NULL;
|
||
|
long FileSize=0;
|
||
|
|
||
|
/* Open our file */
|
||
|
DiscFile = fopen(pszFileName, "rb");
|
||
|
if (DiscFile!=NULL) {
|
||
|
/* Find size of TOS image - 192k or 256k */
|
||
|
fseek(DiscFile, 0, SEEK_END);
|
||
|
FileSize = ftell(DiscFile);
|
||
|
fseek(DiscFile, 0, SEEK_SET);
|
||
|
/* Find pointer to where to load, allocate memory if pass NULL */
|
||
|
if (pAddress)
|
||
|
pFile = pAddress;
|
||
|
else
|
||
|
pFile = malloc(FileSize);
|
||
|
/* Read in... */
|
||
|
if (pFile)
|
||
|
fread((char *)pFile, 1, FileSize, DiscFile);
|
||
|
|
||
|
fclose(DiscFile);
|
||
|
}
|
||
|
/* Store size of file we read in (or 0 if failed) */
|
||
|
if (pFileSize)
|
||
|
*pFileSize = FileSize;
|
||
|
|
||
|
return(pFile); /* Return to where read in/allocated */
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Save file to PC, return false if errors
|
||
|
*/
|
||
|
bool File_Save(char *pszFileName, void *pAddress,unsigned long Size,bool bQueryOverwrite)
|
||
|
{
|
||
|
FILE *DiscFile;
|
||
|
bool bRet=false;
|
||
|
|
||
|
/* Check if need to ask user if to overwrite */
|
||
|
if (bQueryOverwrite) {
|
||
|
/* If file exists, ask if OK to overwrite */
|
||
|
if (!File_QueryOverwrite(pszFileName))
|
||
|
return(false);
|
||
|
}
|
||
|
|
||
|
/* Create our file */
|
||
|
DiscFile = fopen(pszFileName, "wb");
|
||
|
if (DiscFile!=NULL) {
|
||
|
/* Write data, set success flag */
|
||
|
if ( fwrite(pAddress, 1, Size, DiscFile)==Size )
|
||
|
bRet = true;
|
||
|
|
||
|
fclose(DiscFile);
|
||
|
}
|
||
|
|
||
|
return(bRet);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Return size of file, -1 if error
|
||
|
*/
|
||
|
int File_Length(char *pszFileName)
|
||
|
{
|
||
|
FILE *DiscFile;
|
||
|
int FileSize;
|
||
|
DiscFile = fopen(pszFileName, "rb");
|
||
|
if (DiscFile!=NULL) {
|
||
|
fseek(DiscFile, 0, SEEK_END);
|
||
|
FileSize = ftell(DiscFile);
|
||
|
fseek(DiscFile, 0, SEEK_SET);
|
||
|
fclose(DiscFile);
|
||
|
return(FileSize);
|
||
|
}
|
||
|
|
||
|
return(-1);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Return true if file exists
|
||
|
*/
|
||
|
bool File_Exists(const char *pszFileName)
|
||
|
{
|
||
|
FILE *DiscFile;
|
||
|
|
||
|
/* Attempt to open file */
|
||
|
DiscFile = fopen(pszFileName, "rb");
|
||
|
if (DiscFile!=NULL) {
|
||
|
fclose(DiscFile);
|
||
|
return(true);
|
||
|
}
|
||
|
return(false);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Delete file, return true if OK
|
||
|
*/
|
||
|
bool File_Delete(char *pszFileName)
|
||
|
{
|
||
|
/* Delete the file (must be closed first) */
|
||
|
return( remove(pszFileName) );
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Find if file exists, and if so ask user if OK to overwrite
|
||
|
*/
|
||
|
bool File_QueryOverwrite(char *pszFileName)
|
||
|
{
|
||
|
|
||
|
char szString[MAX_FILENAME_LENGTH];
|
||
|
|
||
|
/* Try and find if file exists */
|
||
|
if (File_Exists(pszFileName)) {
|
||
|
/* File does exist, are we OK to overwrite? */
|
||
|
sprintf(szString,"File '%s' exists, overwrite?",pszFileName);
|
||
|
/* FIXME: */
|
||
|
// if (MessageBox(hWnd,szString,PROG_NAME,MB_YESNO | MB_DEFBUTTON2 | MB_ICONSTOP)==IDNO)
|
||
|
// return(false);
|
||
|
}
|
||
|
|
||
|
return(true);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Try filename with various extensions and check if file exists - if so return correct name
|
||
|
*/
|
||
|
bool File_FindPossibleExtFileName(char *pszFileName, char *ppszExts[])
|
||
|
{
|
||
|
char szSrcDir[256], szSrcName[128], szSrcExt[32];
|
||
|
char szTempFileName[MAX_FILENAME_LENGTH];
|
||
|
int i=0;
|
||
|
|
||
|
/* Split filename into parts */
|
||
|
File_splitpath(pszFileName, szSrcDir, szSrcName, szSrcExt);
|
||
|
|
||
|
/* Scan possible extensions */
|
||
|
while(ppszExts[i]) {
|
||
|
/* Re-build with new file extension */
|
||
|
File_makepath(szTempFileName, szSrcDir, szSrcName, ppszExts[i]);
|
||
|
/* Does this file exist? */
|
||
|
if (File_Exists(szTempFileName)) {
|
||
|
/* Copy name for return */
|
||
|
strcpy(pszFileName,szTempFileName);
|
||
|
return(true);
|
||
|
}
|
||
|
|
||
|
/* Next one */
|
||
|
i++;
|
||
|
}
|
||
|
|
||
|
/* No, none of the files exist */
|
||
|
return(false);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Split a complete filename into path, filename and extension.
|
||
|
If pExt is NULL, don't split the extension from the file name!
|
||
|
*/
|
||
|
void File_splitpath(char *pSrcFileName, char *pDir, char *pName, char *pExt)
|
||
|
{
|
||
|
char *ptr1, *ptr2;
|
||
|
|
||
|
/* Build pathname: */
|
||
|
ptr1 = strrchr(pSrcFileName, '/');
|
||
|
if( ptr1 )
|
||
|
{
|
||
|
strcpy(pDir, pSrcFileName);
|
||
|
strcpy(pName, ptr1+1);
|
||
|
pDir[ptr1-pSrcFileName+1] = 0;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
strcpy(pDir, "");
|
||
|
strcpy(pName, pSrcFileName);
|
||
|
}
|
||
|
|
||
|
/* Build the raw filename: */
|
||
|
if( pExt!=NULL )
|
||
|
{
|
||
|
ptr2 = strrchr(pName+1, '.');
|
||
|
if( ptr2 )
|
||
|
{
|
||
|
pName[ptr2-pName] = 0;
|
||
|
/* Copy the file extension: */
|
||
|
strcpy(pExt, ptr2+1);
|
||
|
}
|
||
|
else
|
||
|
pExt[0] = 0;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Build a complete filename from path, filename and extension.
|
||
|
pExt can also be NULL.
|
||
|
*/
|
||
|
void File_makepath(char *pDestFileName, char *pDir, char *pName, char *pExt)
|
||
|
{
|
||
|
strcpy(pDestFileName, pDir);
|
||
|
if( strlen(pDestFileName)==0 )
|
||
|
strcpy(pDestFileName, "");
|
||
|
if( pDestFileName[strlen(pDestFileName)-1]!='/' )
|
||
|
strcat(pDestFileName, "/");
|
||
|
|
||
|
strcat(pDestFileName, pName);
|
||
|
|
||
|
if( pExt!=NULL )
|
||
|
{
|
||
|
if( strlen(pExt)>0 && pExt[0]!='.' )
|
||
|
strcat(pDestFileName, ".");
|
||
|
strcat(pDestFileName, pExt);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/*-----------------------------------------------------------------------*/
|
||
|
/*
|
||
|
Shrink a file name to a certain length and insert some dots if we cut
|
||
|
something away (usefull for showing file names in a dialog).
|
||
|
*/
|
||
|
void File_ShrinkName(char *pDestFileName, char *pSrcFileName, int maxlen)
|
||
|
{
|
||
|
int srclen = strlen(pSrcFileName);
|
||
|
if( srclen<maxlen )
|
||
|
strcpy(pDestFileName, pSrcFileName); /* It fits! */
|
||
|
else
|
||
|
{
|
||
|
strncpy(pDestFileName, pSrcFileName, maxlen/2);
|
||
|
if(maxlen&1) /* even or uneven? */
|
||
|
pDestFileName[maxlen/2-1] = 0;
|
||
|
else
|
||
|
pDestFileName[maxlen/2-2] = 0;
|
||
|
strcat(pDestFileName, "...");
|
||
|
strcat(pDestFileName, &pSrcFileName[strlen(pSrcFileName)-maxlen/2+1]);
|
||
|
}
|
||
|
}
|
||
|
|