Implementation compiles and links with no actual archive support. No test
case available at this moment. :)
This commit is contained in:
parent
f053b90111
commit
2622be385d
2
Makefile
2
Makefile
|
@ -182,7 +182,7 @@ $(MAINLIB) : $(BINDIR) $(MAINOBJS)
|
||||||
|
|
||||||
$(BINDIR):
|
$(BINDIR):
|
||||||
mkdir -p $(BINDIR)
|
mkdir -p $(BINDIR)
|
||||||
mkdir -p $(BINDIR)/archive_drivers
|
mkdir -p $(BINDIR)/archivers
|
||||||
mkdir -p $(BINDIR)/platform
|
mkdir -p $(BINDIR)/platform
|
||||||
|
|
||||||
distclean: clean
|
distclean: clean
|
||||||
|
|
311
archivers/dir.c
311
archivers/dir.c
|
@ -8,46 +8,309 @@
|
||||||
|
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
#define __PHYSICSFS_INTERNAL__
|
#define __PHYSICSFS_INTERNAL__
|
||||||
#include "physfs_internal.h"
|
#include "physfs_internal.h"
|
||||||
|
|
||||||
|
extern const DirFunctions __PHYSFS_DirFunctions_DIR;
|
||||||
|
static const FileFunctions __PHYSFS_FileHandle_DIR;
|
||||||
|
static const FileFunctions __PHYSFS_FileHandle_DIRW;
|
||||||
|
|
||||||
|
static int DIR_read(FileHandle *handle, void *buffer,
|
||||||
|
unsigned int objSize, unsigned int objCount)
|
||||||
|
{
|
||||||
|
FILE *h = (FILE *) (handle->opaque);
|
||||||
|
int retval;
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
retval = fread(buffer, objSize, objCount, h);
|
||||||
|
if ( (retval < objCount) && (ferror(h)) )
|
||||||
|
__PHYSFS_setError(strerror(errno));
|
||||||
|
|
||||||
|
return(retval);
|
||||||
|
} /* DIR_read */
|
||||||
|
|
||||||
|
|
||||||
|
static int DIR_write(FileHandle *handle, void *buffer,
|
||||||
|
unsigned int objSize, unsigned int objCount)
|
||||||
|
{
|
||||||
|
FILE *h = (FILE *) (handle->opaque);
|
||||||
|
int retval;
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
retval = fwrite(buffer, objSize, objCount, h);
|
||||||
|
if ( (retval < objCount) && (ferror(h)) )
|
||||||
|
__PHYSFS_setError(strerror(errno));
|
||||||
|
|
||||||
|
return(retval);
|
||||||
|
} /* DIR_write */
|
||||||
|
|
||||||
|
|
||||||
|
static int DIR_eof(FileHandle *handle)
|
||||||
|
{
|
||||||
|
return(feof((FILE *) (handle->opaque)));
|
||||||
|
} /* DIR_eof */
|
||||||
|
|
||||||
|
|
||||||
|
static int DIR_tell(FileHandle *handle)
|
||||||
|
{
|
||||||
|
return(ftell((FILE *) (handle->opaque)));
|
||||||
|
} /* DIR_tell */
|
||||||
|
|
||||||
|
|
||||||
|
static int DIR_seek(FileHandle *handle, int offset)
|
||||||
|
{
|
||||||
|
return(fseek((FILE *) (handle->opaque), offset, SEEK_SET) == 0);
|
||||||
|
} /* DIR_seek */
|
||||||
|
|
||||||
|
|
||||||
|
static int DIR_fileClose(FileHandle *handle)
|
||||||
|
{
|
||||||
|
FILE *h = (FILE *) (handle->opaque);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* we manually fflush() the buffer, since that's the place fclose() will
|
||||||
|
* most likely fail, but that will leave the file handle in an undefined
|
||||||
|
* state if it fails. fflush() failures we can recover from.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* keep trying until there's success or an unrecoverable error... */
|
||||||
|
do {
|
||||||
|
__PHYSFS_platformTimeslice();
|
||||||
|
errno = 0;
|
||||||
|
} while ( (fflush(h) == EOF) && ((errno == EAGAIN) || (errno == EINTR)) );
|
||||||
|
|
||||||
|
/* EBADF == "Not open for writing". That's fine. */
|
||||||
|
BAIL_IF_MACRO((errno != 0) && (errno != EBADF), strerror(errno), 0);
|
||||||
|
|
||||||
|
/* if fclose fails anyhow, we just have to pray that it's still usable. */
|
||||||
|
errno = 0;
|
||||||
|
BAIL_IF_MACRO(fclose(h) == EOF, strerror(errno), 0); /* (*shrug*) */
|
||||||
|
|
||||||
|
free(handle);
|
||||||
|
return(1);
|
||||||
|
} /* DIR_fileClose */
|
||||||
|
|
||||||
|
|
||||||
|
static int DIR_isArchive(const char *filename, int forWriting)
|
||||||
|
{
|
||||||
|
/* directories ARE archives in this driver... */
|
||||||
|
return(__PHYSFS_platformIsDirectory(filename));
|
||||||
|
} /* DIR_isArchive */
|
||||||
|
|
||||||
|
|
||||||
|
static DirHandle *DIR_openArchive(const char *name, int forWriting)
|
||||||
|
{
|
||||||
|
const char *dirsep = __PHYSFS_platformDirSeparator;
|
||||||
|
DirHandle *retval = NULL;
|
||||||
|
int namelen = strlen(name);
|
||||||
|
int seplen = strlen(dirsep);
|
||||||
|
|
||||||
|
BAIL_IF_MACRO(!DIR_isArchive(name, 0), ERR_UNSUPPORTED_ARCHIVE, NULL);
|
||||||
|
|
||||||
|
retval = malloc(sizeof (DirHandle));
|
||||||
|
BAIL_IF_MACRO(retval == NULL, ERR_OUT_OF_MEMORY, NULL);
|
||||||
|
retval->opaque = malloc(namelen + seplen + 1);
|
||||||
|
if (retval->opaque == NULL)
|
||||||
|
{
|
||||||
|
free(retval);
|
||||||
|
BAIL_IF_MACRO(0, ERR_OUT_OF_MEMORY, NULL);
|
||||||
|
} /* if */
|
||||||
|
|
||||||
|
/* make sure there's a dir separator at the end of the string */
|
||||||
|
strcpy((char *) (retval->opaque), name);
|
||||||
|
if (strcmp((name + namelen) - seplen, dirsep) != 0)
|
||||||
|
strcat((char *) (retval->opaque), dirsep);
|
||||||
|
|
||||||
|
retval->funcs = &__PHYSFS_DirFunctions_DIR;
|
||||||
|
return(retval);
|
||||||
|
} /* DIR_openArchive */
|
||||||
|
|
||||||
|
|
||||||
|
static LinkedStringList *DIR_enumerateFiles(DirHandle *h, const char *dname)
|
||||||
|
{
|
||||||
|
char *d = __PHYSFS_platformCvtToDependent((char *)(h->opaque),dname,NULL);
|
||||||
|
LinkedStringList *retval;
|
||||||
|
|
||||||
|
BAIL_IF_MACRO(d == NULL, NULL, NULL);
|
||||||
|
retval = __PHYSFS_platformEnumerateFiles(d);
|
||||||
|
free(d);
|
||||||
|
return(retval);
|
||||||
|
} /* DIR_enumerateFiles */
|
||||||
|
|
||||||
|
|
||||||
|
static int DIR_exists(DirHandle *h, const char *name)
|
||||||
|
{
|
||||||
|
char *f = __PHYSFS_platformCvtToDependent((char *)(h->opaque), name, NULL);
|
||||||
|
int retval;
|
||||||
|
|
||||||
|
BAIL_IF_MACRO(f == NULL, NULL, 0);
|
||||||
|
retval = __PHYSFS_platformExists(f);
|
||||||
|
free(f);
|
||||||
|
return(retval);
|
||||||
|
} /* DIR_exists */
|
||||||
|
|
||||||
|
|
||||||
|
static int DIR_isDirectory(DirHandle *h, const char *name)
|
||||||
|
{
|
||||||
|
char *d = __PHYSFS_platformCvtToDependent((char *)(h->opaque), name, NULL);
|
||||||
|
int retval;
|
||||||
|
|
||||||
|
BAIL_IF_MACRO(d == NULL, NULL, 0);
|
||||||
|
retval = __PHYSFS_platformIsDirectory(d);
|
||||||
|
free(d);
|
||||||
|
return(retval);
|
||||||
|
} /* DIR_isDirectory */
|
||||||
|
|
||||||
|
|
||||||
|
static int DIR_isSymLink(DirHandle *h, const char *name)
|
||||||
|
{
|
||||||
|
char *f = __PHYSFS_platformCvtToDependent((char *)(h->opaque), name, NULL);
|
||||||
|
int retval;
|
||||||
|
|
||||||
|
BAIL_IF_MACRO(f == NULL, NULL, 0); /* !!! might be a problem. */
|
||||||
|
retval = __PHYSFS_platformIsSymLink(f);
|
||||||
|
free(f);
|
||||||
|
return(retval);
|
||||||
|
} /* DIR_isSymLink */
|
||||||
|
|
||||||
|
|
||||||
|
static FileHandle *doOpen(DirHandle *h, const char *name, const char *mode)
|
||||||
|
{
|
||||||
|
char *f = __PHYSFS_platformCvtToDependent((char *)(h->opaque), name, NULL);
|
||||||
|
FILE *rc;
|
||||||
|
FileHandle *retval;
|
||||||
|
char *str;
|
||||||
|
|
||||||
|
BAIL_IF_MACRO(f == NULL, NULL, NULL);
|
||||||
|
|
||||||
|
retval = (FileHandle *) malloc(sizeof (FileHandle));
|
||||||
|
if (!retval)
|
||||||
|
{
|
||||||
|
free(f);
|
||||||
|
BAIL_IF_MACRO(0, ERR_OUT_OF_MEMORY, NULL);
|
||||||
|
} /* if */
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
rc = fopen(f, mode);
|
||||||
|
str = strerror(errno);
|
||||||
|
free(f);
|
||||||
|
|
||||||
|
if (!rc)
|
||||||
|
{
|
||||||
|
free(retval);
|
||||||
|
BAIL_IF_MACRO(0, str, NULL);
|
||||||
|
} /* if */
|
||||||
|
|
||||||
|
retval->opaque = (void *) rc;
|
||||||
|
retval->dirHandle = h;
|
||||||
|
retval->funcs = &__PHYSFS_FileHandle_DIR;
|
||||||
|
return(retval);
|
||||||
|
} /* doOpen */
|
||||||
|
|
||||||
|
|
||||||
|
static FileHandle *DIR_openRead(DirHandle *h, const char *filename)
|
||||||
|
{
|
||||||
|
return(doOpen(h, filename, "rb"));
|
||||||
|
} /* DIR_openRead */
|
||||||
|
|
||||||
|
|
||||||
|
static FileHandle *DIR_openWrite(DirHandle *h, const char *filename)
|
||||||
|
{
|
||||||
|
return(doOpen(h, filename, "wb"));
|
||||||
|
} /* DIR_openWrite */
|
||||||
|
|
||||||
|
|
||||||
|
static FileHandle *DIR_openAppend(DirHandle *h, const char *filename)
|
||||||
|
{
|
||||||
|
return(doOpen(h, filename, "ab"));
|
||||||
|
} /* DIR_openAppend */
|
||||||
|
|
||||||
|
|
||||||
|
static int DIR_remove(DirHandle *h, const char *name)
|
||||||
|
{
|
||||||
|
char *f = __PHYSFS_platformCvtToDependent((char *)(h->opaque), name, NULL);
|
||||||
|
int retval;
|
||||||
|
|
||||||
|
BAIL_IF_MACRO(f == NULL, NULL, 0);
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
retval = (remove(f) == 0);
|
||||||
|
if (!retval)
|
||||||
|
__PHYSFS_setError(strerror(errno));
|
||||||
|
|
||||||
|
free(f);
|
||||||
|
return(retval);
|
||||||
|
} /* DIR_remove */
|
||||||
|
|
||||||
|
|
||||||
|
static int DIR_mkdir(DirHandle *h, const char *name)
|
||||||
|
{
|
||||||
|
char *f = __PHYSFS_platformCvtToDependent((char *)(h->opaque), name, NULL);
|
||||||
|
int retval;
|
||||||
|
|
||||||
|
BAIL_IF_MACRO(f == NULL, NULL, 0);
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
retval = (mkdir(f, S_IRWXU) == 0);
|
||||||
|
if (!retval)
|
||||||
|
__PHYSFS_setError(strerror(errno));
|
||||||
|
|
||||||
|
free(f);
|
||||||
|
return(retval);
|
||||||
|
} /* DIR_mkdir */
|
||||||
|
|
||||||
|
|
||||||
|
static void DIR_dirClose(DirHandle *h)
|
||||||
|
{
|
||||||
|
free(h->opaque);
|
||||||
|
free(h);
|
||||||
|
} /* DIR_dirClose */
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
static const FileFunctions __PHYSFS_FileHandle_DIR =
|
static const FileFunctions __PHYSFS_FileHandle_DIR =
|
||||||
{
|
{
|
||||||
DIR_read, /* read() method */
|
DIR_read, /* read() method */
|
||||||
NULL, /* write() method */
|
NULL, /* write() method */
|
||||||
DIR_eof, /* eof() method */
|
DIR_eof, /* eof() method */
|
||||||
DIR_tell, /* tell() method */
|
DIR_tell, /* tell() method */
|
||||||
DIR_seek, /* seek() method */
|
DIR_seek, /* seek() method */
|
||||||
DIR_close, /* close() method */
|
DIR_fileClose, /* fileClose() method */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
static const FileFunctions __PHYSFS_FileHandle_DIRW =
|
static const FileFunctions __PHYSFS_FileHandle_DIRW =
|
||||||
{
|
{
|
||||||
NULL, /* read() method */
|
NULL, /* read() method */
|
||||||
DIR_write, /* write() method */
|
DIR_write, /* write() method */
|
||||||
DIR_eof, /* eof() method */
|
DIR_eof, /* eof() method */
|
||||||
DIR_tell, /* tell() method */
|
DIR_tell, /* tell() method */
|
||||||
DIR_seek, /* seek() method */
|
DIR_seek, /* seek() method */
|
||||||
DIR_close, /* close() method */
|
DIR_fileClose, /* fileClose() method */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
const DirFunctions __PHYSFS_DirFunctions_DIR =
|
const DirFunctions __PHYSFS_DirFunctions_DIR =
|
||||||
{
|
{
|
||||||
DIR_isArchive, /* isArchive() method */
|
DIR_isArchive, /* isArchive() method */
|
||||||
DIR_openArchive, /* openArchive() method */
|
DIR_openArchive, /* openArchive() method */
|
||||||
DIR_enumerate, /* enumerateFiles() method */
|
DIR_enumerateFiles, /* enumerateFiles() method */
|
||||||
DIR_exists, /* exists() method */
|
DIR_exists, /* exists() method */
|
||||||
DIR_isDirectory, /* isDirectory() method */
|
DIR_isDirectory, /* isDirectory() method */
|
||||||
DIR_isSymLink, /* isSymLink() method */
|
DIR_isSymLink, /* isSymLink() method */
|
||||||
DIR_openRead, /* openRead() method */
|
DIR_openRead, /* openRead() method */
|
||||||
DIR_openWrite, /* openWrite() method */
|
DIR_openWrite, /* openWrite() method */
|
||||||
DIR_openAppend, /* openAppend() method */
|
DIR_openAppend, /* openAppend() method */
|
||||||
DIR_remove, /* remove() method */
|
DIR_remove, /* remove() method */
|
||||||
DIR_mkdir, /* mkdir() method */
|
DIR_mkdir, /* mkdir() method */
|
||||||
DIR_close, /* close() method */
|
DIR_dirClose, /* dirClose() method */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
105
archivers/zip.c
105
archivers/zip.c
|
@ -16,32 +16,101 @@
|
||||||
#error PHYSFS_SUPPORTS_ZIP must be defined.
|
#error PHYSFS_SUPPORTS_ZIP must be defined.
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
extern const DirFunctions __PHYSFS_DirFunctions_ZIP;
|
||||||
|
extern const FileFunctions __PHYSFS_FileHandle_ZIP;
|
||||||
|
|
||||||
|
|
||||||
|
static int ZIP_read(FileHandle *handle, void *buffer,
|
||||||
|
unsigned int objSize, unsigned int objCount)
|
||||||
|
{
|
||||||
|
} /* ZIP_read */
|
||||||
|
|
||||||
|
|
||||||
|
static int ZIP_eof(FileHandle *handle)
|
||||||
|
{
|
||||||
|
} /* ZIP_eof */
|
||||||
|
|
||||||
|
|
||||||
|
static int ZIP_tell(FileHandle *handle)
|
||||||
|
{
|
||||||
|
} /* ZIP_tell */
|
||||||
|
|
||||||
|
|
||||||
|
static int ZIP_seek(FileHandle *handle, int offset)
|
||||||
|
{
|
||||||
|
} /* ZIP_seek */
|
||||||
|
|
||||||
|
|
||||||
|
static int ZIP_fileClose(FileHandle *handle)
|
||||||
|
{
|
||||||
|
} /* ZIP_fileClose */
|
||||||
|
|
||||||
|
|
||||||
|
static int ZIP_isArchive(const char *filename, int forWriting)
|
||||||
|
{
|
||||||
|
} /* ZIP_isArchive */
|
||||||
|
|
||||||
|
|
||||||
|
static DirHandle *ZIP_openArchive(const char *name, int forWriting)
|
||||||
|
{
|
||||||
|
} /* ZIP_openArchive */
|
||||||
|
|
||||||
|
|
||||||
|
static LinkedStringList *ZIP_enumerateFiles(DirHandle *r, const char *dirname)
|
||||||
|
{
|
||||||
|
} /* ZIP_enumerateFiles */
|
||||||
|
|
||||||
|
|
||||||
|
static int ZIP_exists(DirHandle *r, const char *name)
|
||||||
|
{
|
||||||
|
} /* ZIP_exists */
|
||||||
|
|
||||||
|
|
||||||
|
static int ZIP_isDirectory(DirHandle *r, const char *name)
|
||||||
|
{
|
||||||
|
} /* ZIP_isDirectory */
|
||||||
|
|
||||||
|
|
||||||
|
static int ZIP_isSymLink(DirHandle *r, const char *name)
|
||||||
|
{
|
||||||
|
} /* ZIP_isSymLink */
|
||||||
|
|
||||||
|
|
||||||
|
static FileHandle *ZIP_openRead(DirHandle *r, const char *filename)
|
||||||
|
{
|
||||||
|
} /* ZIP_openRead */
|
||||||
|
|
||||||
|
|
||||||
|
static void ZIP_dirClose(DirHandle *r)
|
||||||
|
{
|
||||||
|
} /* ZIP_dirClose */
|
||||||
|
|
||||||
|
|
||||||
static const FileFunctions __PHYSFS_FileHandle_ZIP =
|
static const FileFunctions __PHYSFS_FileHandle_ZIP =
|
||||||
{
|
{
|
||||||
ZIP_read, /* read() method */
|
ZIP_read, /* read() method */
|
||||||
NULL, /* write() method */
|
NULL, /* write() method */
|
||||||
ZIP_eof, /* eof() method */
|
ZIP_eof, /* eof() method */
|
||||||
ZIP_tell, /* tell() method */
|
ZIP_tell, /* tell() method */
|
||||||
ZIP_seek, /* seek() method */
|
ZIP_seek, /* seek() method */
|
||||||
ZIP_close, /* close() method */
|
ZIP_fileClose, /* fileClose() method */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
const DirFunctions __PHYSFS_DirFunctions_ZIP =
|
const DirFunctions __PHYSFS_DirFunctions_ZIP =
|
||||||
{
|
{
|
||||||
ZIP_isArchive, /* isArchive() method */
|
ZIP_isArchive, /* isArchive() method */
|
||||||
ZIP_openArchive, /* openArchive() method */
|
ZIP_openArchive, /* openArchive() method */
|
||||||
ZIP_enumerate, /* enumerateFiles() method */
|
ZIP_enumerateFiles, /* enumerateFiles() method */
|
||||||
ZIP_exists, /* exists() method */
|
ZIP_exists, /* exists() method */
|
||||||
ZIP_isDirectory, /* isDirectory() method */
|
ZIP_isDirectory, /* isDirectory() method */
|
||||||
ZIP_isSymLink, /* isSymLink() method */
|
ZIP_isSymLink, /* isSymLink() method */
|
||||||
ZIP_openRead, /* openRead() method */
|
ZIP_openRead, /* openRead() method */
|
||||||
NULL, /* openWrite() method */
|
NULL, /* openWrite() method */
|
||||||
NULL, /* openAppend() method */
|
NULL, /* openAppend() method */
|
||||||
NULL, /* remove() method */
|
NULL, /* remove() method */
|
||||||
NULL, /* mkdir() method */
|
NULL, /* mkdir() method */
|
||||||
ZIP_close, /* close() method */
|
ZIP_dirClose, /* dirClose() method */
|
||||||
};
|
};
|
||||||
|
|
||||||
const __PHYSFS_ArchiveInfo __PHYSFS_ArchiveInfo_ZIP =
|
const __PHYSFS_ArchiveInfo __PHYSFS_ArchiveInfo_ZIP =
|
||||||
|
|
24
physfs.c
24
physfs.c
|
@ -194,14 +194,14 @@ static DirInfo *buildDirInfo(const char *newDir, int forWriting)
|
||||||
|
|
||||||
di = (DirInfo *) malloc(sizeof (DirInfo));
|
di = (DirInfo *) malloc(sizeof (DirInfo));
|
||||||
if (di == NULL)
|
if (di == NULL)
|
||||||
dirHandle->funcs->close(dirHandle);
|
dirHandle->funcs->dirClose(dirHandle);
|
||||||
BAIL_IF_MACRO(di == NULL, ERR_OUT_OF_MEMORY, 0);
|
BAIL_IF_MACRO(di == NULL, ERR_OUT_OF_MEMORY, 0);
|
||||||
|
|
||||||
di->dirName = (char *) malloc(strlen(newDir) + 1);
|
di->dirName = (char *) malloc(strlen(newDir) + 1);
|
||||||
if (di->dirName == NULL)
|
if (di->dirName == NULL)
|
||||||
{
|
{
|
||||||
free(di);
|
free(di);
|
||||||
dirHandle->funcs->close(dirHandle);
|
dirHandle->funcs->dirClose(dirHandle);
|
||||||
__PHYSFS_setError(ERR_OUT_OF_MEMORY);
|
__PHYSFS_setError(ERR_OUT_OF_MEMORY);
|
||||||
return(0);
|
return(0);
|
||||||
} /* if */
|
} /* if */
|
||||||
|
@ -226,7 +226,7 @@ static int freeDirInfo(DirInfo *di, FileHandleList *openList)
|
||||||
BAIL_IF_MACRO(h == di->dirHandle, ERR_FILES_STILL_OPEN, 0);
|
BAIL_IF_MACRO(h == di->dirHandle, ERR_FILES_STILL_OPEN, 0);
|
||||||
} /* for */
|
} /* for */
|
||||||
|
|
||||||
di->dirHandle->funcs->close(di->dirHandle);
|
di->dirHandle->funcs->dirClose(di->dirHandle);
|
||||||
free(di->dirName);
|
free(di->dirName);
|
||||||
free(di);
|
free(di);
|
||||||
return(1);
|
return(1);
|
||||||
|
@ -300,7 +300,7 @@ static int closeFileHandleList(FileHandleList **list)
|
||||||
{
|
{
|
||||||
next = i->next;
|
next = i->next;
|
||||||
h = (FileHandle *) (i->handle->opaque);
|
h = (FileHandle *) (i->handle->opaque);
|
||||||
if (!h->funcs->close(i->handle->opaque))
|
if (!h->funcs->fileClose(i->handle->opaque))
|
||||||
{
|
{
|
||||||
*list = i;
|
*list = i;
|
||||||
return(0);
|
return(0);
|
||||||
|
@ -624,9 +624,9 @@ void PHYSFS_permitSymbolicLinks(int allow)
|
||||||
|
|
||||||
|
|
||||||
/* string manipulation in C makes my ass itch. */
|
/* string manipulation in C makes my ass itch. */
|
||||||
char *__PHYSFS_convertToDependentNotation(const char *prepend,
|
char *__PHYSFS_convertToDependent(const char *prepend,
|
||||||
const char *dirName,
|
const char *dirName,
|
||||||
const char *append)
|
const char *append)
|
||||||
{
|
{
|
||||||
const char *dirsep = PHYSFS_getDirSeparator();
|
const char *dirsep = PHYSFS_getDirSeparator();
|
||||||
int sepsize = strlen(dirsep);
|
int sepsize = strlen(dirsep);
|
||||||
|
@ -658,13 +658,14 @@ char *__PHYSFS_convertToDependentNotation(const char *prepend,
|
||||||
|
|
||||||
str = (char *) malloc(allocSize);
|
str = (char *) malloc(allocSize);
|
||||||
BAIL_IF_MACRO(str == NULL, ERR_OUT_OF_MEMORY, NULL);
|
BAIL_IF_MACRO(str == NULL, ERR_OUT_OF_MEMORY, NULL);
|
||||||
*str = '\0';
|
|
||||||
|
|
||||||
if (prepend)
|
if (prepend == NULL)
|
||||||
|
*str = '\0';
|
||||||
|
else
|
||||||
{
|
{
|
||||||
strcpy(str, prepend);
|
strcpy(str, prepend);
|
||||||
strcat(str, dirsep);
|
strcat(str, dirsep);
|
||||||
} /* if */
|
} /* else */
|
||||||
|
|
||||||
for (i1 = (char *) dirName, i2 = str + strlen(str); *i1; i1++, i2++)
|
for (i1 = (char *) dirName, i2 = str + strlen(str); *i1; i1++, i2++)
|
||||||
{
|
{
|
||||||
|
@ -709,6 +710,7 @@ int __PHYSFS_verifySecurity(DirHandle *h, const char *fname)
|
||||||
|
|
||||||
if ( (strcmp(start, ".") == 0) ||
|
if ( (strcmp(start, ".") == 0) ||
|
||||||
(strcmp(start, "..") == 0) ||
|
(strcmp(start, "..") == 0) ||
|
||||||
|
(strchr(start, '\\') != NULL) ||
|
||||||
(strchr(start, ':') != NULL) )
|
(strchr(start, ':') != NULL) )
|
||||||
{
|
{
|
||||||
__PHYSFS_setError(ERR_INSECURE_FNAME);
|
__PHYSFS_setError(ERR_INSECURE_FNAME);
|
||||||
|
@ -1058,7 +1060,7 @@ int PHYSFS_close(PHYSFS_file *handle)
|
||||||
{
|
{
|
||||||
if (i->handle == handle)
|
if (i->handle == handle)
|
||||||
{
|
{
|
||||||
rc = h->funcs->close(h);
|
rc = h->funcs->fileClose(h);
|
||||||
if (!rc)
|
if (!rc)
|
||||||
return(0);
|
return(0);
|
||||||
|
|
||||||
|
|
|
@ -17,6 +17,14 @@
|
||||||
struct __PHYSFS_DIRHANDLE__;
|
struct __PHYSFS_DIRHANDLE__;
|
||||||
struct __PHYSFS_FILEFUNCTIONS__;
|
struct __PHYSFS_FILEFUNCTIONS__;
|
||||||
|
|
||||||
|
|
||||||
|
typedef struct __PHYSFS_LINKEDSTRINGLIST__
|
||||||
|
{
|
||||||
|
char *str;
|
||||||
|
struct __PHYSFS_LINKEDSTRINGLIST__ *next;
|
||||||
|
} LinkedStringList;
|
||||||
|
|
||||||
|
|
||||||
typedef struct __PHYSFS_FILEHANDLE__
|
typedef struct __PHYSFS_FILEHANDLE__
|
||||||
{
|
{
|
||||||
/*
|
/*
|
||||||
|
@ -79,7 +87,7 @@ typedef struct __PHYSFS_FILEFUNCTIONS__
|
||||||
* returns non-zero on success, zero if can't close file.
|
* returns non-zero on success, zero if can't close file.
|
||||||
* On failure, call __PHYSFS_setError().
|
* On failure, call __PHYSFS_setError().
|
||||||
*/
|
*/
|
||||||
int (*close)(FileHandle *handle);
|
int (*fileClose)(FileHandle *handle);
|
||||||
} FileFunctions;
|
} FileFunctions;
|
||||||
|
|
||||||
|
|
||||||
|
@ -97,12 +105,6 @@ typedef struct __PHYSFS_DIRHANDLE__
|
||||||
} DirHandle;
|
} DirHandle;
|
||||||
|
|
||||||
|
|
||||||
typedef struct __PHYSFS_LINKEDSTRINGLIST__
|
|
||||||
{
|
|
||||||
char *str;
|
|
||||||
struct __PHYSFS_LINKEDSTRINGLIST__ *next;
|
|
||||||
} LinkedStringList;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Symlinks should always be followed; PhysicsFS will use
|
* Symlinks should always be followed; PhysicsFS will use
|
||||||
* DirFunctions->isSymLink() and make a judgement on whether to
|
* DirFunctions->isSymLink() and make a judgement on whether to
|
||||||
|
@ -162,13 +164,17 @@ typedef struct __PHYSFS_DIRFUNCTIONS__
|
||||||
* This filename is in platform-independent notation.
|
* This filename is in platform-independent notation.
|
||||||
* If you can't handle multiple opens of the same file,
|
* If you can't handle multiple opens of the same file,
|
||||||
* you can opt to fail for the second call.
|
* you can opt to fail for the second call.
|
||||||
|
* Fail if the file does not exist.
|
||||||
* Returns NULL on failure, and calls __PHYSFS_setError().
|
* Returns NULL on failure, and calls __PHYSFS_setError().
|
||||||
*/
|
*/
|
||||||
FileHandle *(*openRead)(DirHandle *r, const char *filename);
|
FileHandle *(*openRead)(DirHandle *r, const char *filename);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Open file for writing, and return a FileHandle.
|
* Open file for writing, and return a FileHandle.
|
||||||
* This filename is in platform-independent notation.
|
* If the file does not exist, it should be created. If it exists,
|
||||||
|
* it should be truncated to zero bytes. The writing
|
||||||
|
* offset should be the start of the file.
|
||||||
|
* This filename is in platform-independent notation.
|
||||||
* This method may be NULL.
|
* This method may be NULL.
|
||||||
* If you can't handle multiple opens of the same file,
|
* If you can't handle multiple opens of the same file,
|
||||||
* you can opt to fail for the second call.
|
* you can opt to fail for the second call.
|
||||||
|
@ -178,7 +184,9 @@ typedef struct __PHYSFS_DIRFUNCTIONS__
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Open file for appending, and return a FileHandle.
|
* Open file for appending, and return a FileHandle.
|
||||||
* This filename is in platform-independent notation.
|
* If the file does not exist, it should be created. The writing
|
||||||
|
* offset should be the end of the file.
|
||||||
|
* This filename is in platform-independent notation.
|
||||||
* This method may be NULL.
|
* This method may be NULL.
|
||||||
* If you can't handle multiple opens of the same file,
|
* If you can't handle multiple opens of the same file,
|
||||||
* you can opt to fail for the second call.
|
* you can opt to fail for the second call.
|
||||||
|
@ -212,7 +220,7 @@ typedef struct __PHYSFS_DIRFUNCTIONS__
|
||||||
* the "opaque" entry. This should assume that it won't be called if
|
* the "opaque" entry. This should assume that it won't be called if
|
||||||
* there are still files open from this DirHandle.
|
* there are still files open from this DirHandle.
|
||||||
*/
|
*/
|
||||||
void (*close)(DirHandle *r);
|
void (*dirClose)(DirHandle *r);
|
||||||
} DirFunctions;
|
} DirFunctions;
|
||||||
|
|
||||||
|
|
||||||
|
@ -248,7 +256,7 @@ void __PHYSFS_setError(const char *err);
|
||||||
* and append (append) to the converted string.
|
* and append (append) to the converted string.
|
||||||
*
|
*
|
||||||
* So, on Win32, calling:
|
* So, on Win32, calling:
|
||||||
* __PHYSFS_convertToDependentNotation("C:\", "my/files", NULL);
|
* __PHYSFS_convertToDependent("C:\", "my/files", NULL);
|
||||||
* ...will return the string "C:\my\files".
|
* ...will return the string "C:\my\files".
|
||||||
*
|
*
|
||||||
* This is a convenience function; you might want to hack something out that
|
* This is a convenience function; you might want to hack something out that
|
||||||
|
@ -256,9 +264,9 @@ void __PHYSFS_setError(const char *err);
|
||||||
*
|
*
|
||||||
* Be sure to free() the return value when done with it.
|
* Be sure to free() the return value when done with it.
|
||||||
*/
|
*/
|
||||||
char *__PHYSFS_convertToDependentNotation(const char *prepend,
|
char *__PHYSFS_convertToDependent(const char *prepend,
|
||||||
const char *dirName,
|
const char *dirName,
|
||||||
const char *append);
|
const char *append);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Verify that (fname) (in platform-independent notation), in relation
|
* Verify that (fname) (in platform-independent notation), in relation
|
||||||
|
@ -337,15 +345,61 @@ int __PHYSFS_platformGetThreadID(void);
|
||||||
int __PHYSFS_platformStricmp(const char *str1, const char *str2);
|
int __PHYSFS_platformStricmp(const char *str1, const char *str2);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Return non-zero if filename (in platform-dependent notation) is a symlink.
|
* Return non-zero if filename (in platform-dependent notation) exists.
|
||||||
|
* Symlinks should be followed; if what the symlink points to is missing,
|
||||||
|
* then the retval is false.
|
||||||
*/
|
*/
|
||||||
int __PHYSFS_platformIsSymlink(const char *fname);
|
int __PHYSFS_platformExists(const char *fname);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Return non-zero if filename (in platform-dependent notation) is a symlink.
|
* Return non-zero if filename (in platform-dependent notation) is a symlink.
|
||||||
*/
|
*/
|
||||||
|
int __PHYSFS_platformIsSymLink(const char *fname);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Return non-zero if filename (in platform-dependent notation) is a symlink.
|
||||||
|
* Symlinks should be followed; if what the symlink points to is missing,
|
||||||
|
* or isn't a directory, then the retval is false.
|
||||||
|
*/
|
||||||
int __PHYSFS_platformIsDirectory(const char *fname);
|
int __PHYSFS_platformIsDirectory(const char *fname);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Convert (dirName) to platform-dependent notation, then prepend (prepend)
|
||||||
|
* and append (append) to the converted string.
|
||||||
|
*
|
||||||
|
* So, on Win32, calling:
|
||||||
|
* __PHYSFS_platformCvtToDependent("C:\", "my/files", NULL);
|
||||||
|
* ...will return the string "C:\my\files".
|
||||||
|
*
|
||||||
|
* This can be implemented in a platform-specific manner, so you can get
|
||||||
|
* get a speed boost that the default implementation can't, since
|
||||||
|
* you can make assumptions about the size of strings, etc..
|
||||||
|
*
|
||||||
|
* Platforms that choose not to implement this may just call
|
||||||
|
* __PHYSFS_convertToDependent() as a passthrough.
|
||||||
|
*
|
||||||
|
* Be sure to free() the return value when done with it.
|
||||||
|
*/
|
||||||
|
char *__PHYSFS_platformCvtToDependent(const char *prepend,
|
||||||
|
const char *dirName,
|
||||||
|
const char *append);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Make the current thread give up a timeslice. This is called in a loop
|
||||||
|
* while waiting for various external forces to get back to us.
|
||||||
|
*/
|
||||||
|
void __PHYSFS_platformTimeslice(void);
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Enumerate a directory of files. This follows the rules for the
|
||||||
|
* DirFunctions->enumerateFiles() method (see above), except that the
|
||||||
|
* (dirName) that is passed to this function is converted to
|
||||||
|
* platform-DEPENDENT notation by the caller. The DirFunctions version
|
||||||
|
* uses platform-independent notation.
|
||||||
|
*/
|
||||||
|
LinkedStringList *__PHYSFS_platformEnumerateFiles(const char *dirname);
|
||||||
|
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
extern "C" {
|
extern "C" {
|
||||||
|
|
232
platform/unix.c
232
platform/unix.c
|
@ -6,9 +6,39 @@
|
||||||
* This file written by Ryan C. Gordon.
|
* This file written by Ryan C. Gordon.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#if (defined __STRICT_ANSI__)
|
||||||
|
#define __PHYSFS_DOING_STRICT_ANSI__
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* We cheat a little: I want the symlink version of stat() (lstat), and
|
||||||
|
* GCC/Linux will not declare it if compiled with the -ansi flag.
|
||||||
|
* If you are really lacking symlink support on your platform,
|
||||||
|
* you should #define __PHYSFS_NO_SYMLINKS__ before compiling this
|
||||||
|
* file. That will open a security hole, though, if you really DO have
|
||||||
|
* symlinks on your platform; it renders PHYSFS_permitSymbolicLinks(0)
|
||||||
|
* useless, since every symlink will be reported as a regular file/dir.
|
||||||
|
*/
|
||||||
|
#if (defined __PHYSFS_DOING_STRICT_ANSI__)
|
||||||
|
#undef __STRICT_ANSI__
|
||||||
|
#endif
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
#if (defined __PHYSFS_DOING_STRICT_ANSI__)
|
||||||
|
#define __STRICT_ANSI__
|
||||||
|
#endif
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <ctype.h>
|
||||||
#include <pthread.h>
|
#include <pthread.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <pwd.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <dirent.h>
|
||||||
|
#include <time.h>
|
||||||
|
#include <errno.h>
|
||||||
|
|
||||||
|
|
||||||
#define __PHYSICSFS_INTERNAL__
|
#define __PHYSICSFS_INTERNAL__
|
||||||
#include "physfs_internal.h"
|
#include "physfs_internal.h"
|
||||||
|
@ -18,22 +48,88 @@ const char *__PHYSFS_platformDirSeparator = "/";
|
||||||
|
|
||||||
char **__PHYSFS_platformDetectAvailableCDs(void)
|
char **__PHYSFS_platformDetectAvailableCDs(void)
|
||||||
{
|
{
|
||||||
|
/* !!! write me. */
|
||||||
|
char **retval = malloc(sizeof (char *));
|
||||||
|
if (retval != NULL)
|
||||||
|
*retval = NULL;
|
||||||
|
|
||||||
|
return(retval);
|
||||||
} /* __PHYSFS_detectAvailableCDs */
|
} /* __PHYSFS_detectAvailableCDs */
|
||||||
|
|
||||||
|
|
||||||
char *__PHYSFS_platformCalcBaseDir(char *argv0)
|
char *__PHYSFS_platformCalcBaseDir(char *argv0)
|
||||||
{
|
{
|
||||||
return(NULL);
|
return(NULL); /* default PhysicsFS behaviour is acceptable. */
|
||||||
} /* __PHYSFS_platformCalcBaseDir */
|
} /* __PHYSFS_platformCalcBaseDir */
|
||||||
|
|
||||||
|
|
||||||
|
static char *copyEnvironmentVariable(const char *varname)
|
||||||
|
{
|
||||||
|
const char *envr = getenv(varname);
|
||||||
|
char *retval = NULL;
|
||||||
|
|
||||||
|
if (envr != NULL)
|
||||||
|
{
|
||||||
|
retval = malloc(strlen(envr) + 1);
|
||||||
|
if (retval != NULL)
|
||||||
|
strcpy(retval, envr);
|
||||||
|
} /* if */
|
||||||
|
|
||||||
|
return(retval);
|
||||||
|
} /* copyEnvironmentVariable */
|
||||||
|
|
||||||
|
|
||||||
|
static char *getUserNameByUID(void)
|
||||||
|
{
|
||||||
|
uid_t uid = getuid();
|
||||||
|
struct passwd *pw;
|
||||||
|
char *retval = NULL;
|
||||||
|
|
||||||
|
pw = getpwuid(uid);
|
||||||
|
if ((pw != NULL) && (pw->pw_name != NULL))
|
||||||
|
{
|
||||||
|
retval = malloc(strlen(pw->pw_name) + 1);
|
||||||
|
if (retval != NULL)
|
||||||
|
strcpy(retval, pw->pw_name);
|
||||||
|
} /* if */
|
||||||
|
|
||||||
|
return(retval);
|
||||||
|
} /* getUserNameByUID */
|
||||||
|
|
||||||
|
|
||||||
|
static char *getUserDirByUID(void)
|
||||||
|
{
|
||||||
|
uid_t uid = getuid();
|
||||||
|
struct passwd *pw;
|
||||||
|
char *retval = NULL;
|
||||||
|
|
||||||
|
pw = getpwuid(uid);
|
||||||
|
if ((pw != NULL) && (pw->pw_dir != NULL))
|
||||||
|
{
|
||||||
|
retval = malloc(strlen(pw->pw_dir) + 1);
|
||||||
|
if (retval != NULL)
|
||||||
|
strcpy(retval, pw->pw_dir);
|
||||||
|
} /* if */
|
||||||
|
|
||||||
|
return(retval);
|
||||||
|
} /* getUserDirByUID */
|
||||||
|
|
||||||
|
|
||||||
char *__PHYSFS_platformGetUserName(void)
|
char *__PHYSFS_platformGetUserName(void)
|
||||||
{
|
{
|
||||||
|
char *retval = getUserNameByUID();
|
||||||
|
if (retval == NULL)
|
||||||
|
retval = copyEnvironmentVariable("USER");
|
||||||
|
return(retval);
|
||||||
} /* __PHYSFS_platformGetUserName */
|
} /* __PHYSFS_platformGetUserName */
|
||||||
|
|
||||||
|
|
||||||
char *__PHYSFS_platformGetUserDir(void)
|
char *__PHYSFS_platformGetUserDir(void)
|
||||||
{
|
{
|
||||||
|
char *retval = copyEnvironmentVariable("HOME");
|
||||||
|
if (retval == NULL)
|
||||||
|
retval = getUserDirByUID();
|
||||||
|
return(retval);
|
||||||
} /* __PHYSFS_platformGetUserDir */
|
} /* __PHYSFS_platformGetUserDir */
|
||||||
|
|
||||||
|
|
||||||
|
@ -43,21 +139,149 @@ int __PHYSFS_platformGetThreadID(void)
|
||||||
} /* __PHYSFS_platformGetThreadID */
|
} /* __PHYSFS_platformGetThreadID */
|
||||||
|
|
||||||
|
|
||||||
int __PHYSFS_platformStricmp(const char *str1, const char *str2)
|
/* -ansi and -pedantic flags prevent use of strcasecmp() on Linux. */
|
||||||
|
int __PHYSFS_platformStricmp(const char *x, const char *y)
|
||||||
{
|
{
|
||||||
return(strcasecmp(str1, str2));
|
int ux, uy;
|
||||||
|
|
||||||
|
do
|
||||||
|
{
|
||||||
|
ux = toupper((int) *x);
|
||||||
|
uy = toupper((int) *y);
|
||||||
|
if (ux > uy)
|
||||||
|
return(1);
|
||||||
|
else if (ux < uy)
|
||||||
|
return(-1);
|
||||||
|
x++;
|
||||||
|
y++;
|
||||||
|
} while ((ux) && (uy));
|
||||||
|
|
||||||
|
return(0);
|
||||||
} /* __PHYSFS_platformStricmp */
|
} /* __PHYSFS_platformStricmp */
|
||||||
|
|
||||||
|
|
||||||
int __PHYSFS_platformIsSymlink(const char *fname)
|
int __PHYSFS_platformExists(const char *fname)
|
||||||
{
|
{
|
||||||
|
struct stat statbuf;
|
||||||
|
return(stat(fname, &statbuf) == 0);
|
||||||
|
} /* __PHYSFS_platformExists */
|
||||||
|
|
||||||
|
|
||||||
|
int __PHYSFS_platformIsSymLink(const char *fname)
|
||||||
|
{
|
||||||
|
#if (defined __PHYSFS_NO_SYMLINKS__)
|
||||||
|
return(0);
|
||||||
|
#else
|
||||||
|
|
||||||
|
struct stat statbuf;
|
||||||
|
int retval = 0;
|
||||||
|
|
||||||
|
if (lstat(fname, &statbuf) == 0)
|
||||||
|
{
|
||||||
|
if (S_ISLNK(statbuf.st_mode))
|
||||||
|
retval = 1;
|
||||||
|
} /* if */
|
||||||
|
|
||||||
|
return(retval);
|
||||||
|
|
||||||
|
#endif
|
||||||
} /* __PHYSFS_platformIsSymlink */
|
} /* __PHYSFS_platformIsSymlink */
|
||||||
|
|
||||||
|
|
||||||
int __PHYSFS_platformIsDirectory(const char *fname)
|
int __PHYSFS_platformIsDirectory(const char *fname)
|
||||||
{
|
{
|
||||||
|
struct stat statbuf;
|
||||||
|
int retval = 0;
|
||||||
|
|
||||||
|
if (stat(fname, &statbuf) == 0)
|
||||||
|
{
|
||||||
|
if (S_ISDIR(statbuf.st_mode))
|
||||||
|
retval = 1;
|
||||||
|
} /* if */
|
||||||
|
|
||||||
|
return(retval);
|
||||||
} /* __PHYSFS_platformIsDirectory */
|
} /* __PHYSFS_platformIsDirectory */
|
||||||
|
|
||||||
|
|
||||||
|
char *__PHYSFS_platformCvtToDependent(const char *prepend,
|
||||||
|
const char *dirName,
|
||||||
|
const char *append)
|
||||||
|
{
|
||||||
|
int len = ((prepend) ? strlen(prepend) : 0) +
|
||||||
|
((append) ? strlen(append) : 0) +
|
||||||
|
strlen(dirName) + 1;
|
||||||
|
char *retval = malloc(len);
|
||||||
|
|
||||||
|
BAIL_IF_MACRO(retval == NULL, ERR_OUT_OF_MEMORY, NULL);
|
||||||
|
|
||||||
|
/* platform-independent notation is Unix-style already. :) */
|
||||||
|
|
||||||
|
if (prepend)
|
||||||
|
strcpy(retval, prepend);
|
||||||
|
else
|
||||||
|
retval[0] = '\0';
|
||||||
|
|
||||||
|
strcat(retval, dirName);
|
||||||
|
|
||||||
|
if (append)
|
||||||
|
strcat(retval, append);
|
||||||
|
|
||||||
|
return(retval);
|
||||||
|
} /* __PHYSFS_platformCvtToDependent */
|
||||||
|
|
||||||
|
|
||||||
|
/* Much like my college days, try to sleep for 10 milliseconds at a time... */
|
||||||
|
void __PHYSFS_platformTimeslice(void)
|
||||||
|
{
|
||||||
|
struct timespec napTime;
|
||||||
|
napTime.tv_sec = 0;
|
||||||
|
napTime.tv_nsec = 10 * 1000 * 1000; /* specified in nanoseconds. */
|
||||||
|
nanosleep(&napTime, NULL); /* don't care if it fails. */
|
||||||
|
} /* __PHYSFS_platformTimeslice */
|
||||||
|
|
||||||
|
|
||||||
|
LinkedStringList *__PHYSFS_platformEnumerateFiles(const char *dirname)
|
||||||
|
{
|
||||||
|
LinkedStringList *retval = NULL;
|
||||||
|
LinkedStringList *l = NULL;
|
||||||
|
LinkedStringList *prev = NULL;
|
||||||
|
DIR *dir;
|
||||||
|
struct dirent *ent;
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
dir = opendir(dirname);
|
||||||
|
BAIL_IF_MACRO(dir == NULL, strerror(errno), NULL);
|
||||||
|
|
||||||
|
while (1)
|
||||||
|
{
|
||||||
|
ent = readdir(dir);
|
||||||
|
if (ent == NULL) /* we're done. */
|
||||||
|
break;
|
||||||
|
|
||||||
|
l = (LinkedStringList *) malloc(sizeof (LinkedStringList));
|
||||||
|
if (l != NULL)
|
||||||
|
break;
|
||||||
|
|
||||||
|
l->str = (char *) malloc(strlen(ent->d_name) + 1);
|
||||||
|
if (l->str == NULL)
|
||||||
|
{
|
||||||
|
free(l);
|
||||||
|
break;
|
||||||
|
} /* if */
|
||||||
|
|
||||||
|
if (retval == NULL)
|
||||||
|
retval = l;
|
||||||
|
else
|
||||||
|
prev->next = l;
|
||||||
|
|
||||||
|
prev = l;
|
||||||
|
l->next = NULL;
|
||||||
|
} /* while */
|
||||||
|
|
||||||
|
closedir(dir);
|
||||||
|
return(retval);
|
||||||
|
} /* __PHYSFS_platformEnumerateFiles */
|
||||||
|
|
||||||
|
|
||||||
/* end of unix.c ... */
|
/* end of unix.c ... */
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue