mirror of
https://github.com/bulletphysics/bullet3
synced 2024-12-15 06:00:12 +00:00
f9f8d0fbb0
also show how to disable file caching (used in the fileIOPlugin)
262 lines
5.7 KiB
C++
262 lines
5.7 KiB
C++
|
|
#include "minizip/unzip.h"
|
|
|
|
#define B3_ZIP_FILEIO_MAX_FILES 1024
|
|
|
|
struct ZipFileIO : public CommonFileIOInterface
|
|
{
|
|
std::string m_zipfileName;
|
|
|
|
unzFile m_fileHandles[B3_ZIP_FILEIO_MAX_FILES ];
|
|
int m_numFileHandles;
|
|
|
|
ZipFileIO(int fileIOType, const char* zipfileName, CommonFileIOInterface* wrapperFileIO)
|
|
:CommonFileIOInterface(fileIOType,0),
|
|
m_zipfileName(zipfileName),
|
|
m_numFileHandles(0)
|
|
{
|
|
m_pathPrefix = m_zipfileName.c_str();
|
|
for (int i=0;i<B3_ZIP_FILEIO_MAX_FILES ;i++)
|
|
{
|
|
m_fileHandles[i]=0;
|
|
}
|
|
}
|
|
|
|
|
|
static bool FileIOPluginFindFile(void* userPtr, const char* orgFileName, char* relativeFileName, int maxRelativeFileNameMaxLen)
|
|
{
|
|
ZipFileIO* fileIo = (ZipFileIO*) userPtr;
|
|
return fileIo->findFile(orgFileName, relativeFileName, maxRelativeFileNameMaxLen);
|
|
}
|
|
|
|
virtual ~ZipFileIO()
|
|
{
|
|
for (int i=0;i<B3_ZIP_FILEIO_MAX_FILES;i++)
|
|
{
|
|
fileClose(i);
|
|
}
|
|
}
|
|
|
|
virtual int fileOpen(const char* fileName, const char* mode)
|
|
{
|
|
|
|
//search a free slot
|
|
int slot = -1;
|
|
for (int i=0;i<B3_ZIP_FILEIO_MAX_FILES ;i++)
|
|
{
|
|
if (m_fileHandles[i]==0)
|
|
{
|
|
slot=i;
|
|
break;
|
|
}
|
|
}
|
|
if (slot>=0)
|
|
{
|
|
unzFile zipfile;
|
|
unz_global_info m_global_info;
|
|
zipfile = unzOpen(m_zipfileName.c_str());
|
|
if (zipfile == NULL)
|
|
{
|
|
printf("%s: not found\n", m_zipfileName.c_str());
|
|
slot = -1;
|
|
} else
|
|
{
|
|
|
|
int result = 0;
|
|
result = unzGetGlobalInfo(zipfile, &m_global_info );
|
|
if (result != UNZ_OK)
|
|
{
|
|
printf("could not read file global info from %s\n", m_zipfileName.c_str());
|
|
unzClose(zipfile);
|
|
zipfile = 0;
|
|
slot = -1;
|
|
}
|
|
}
|
|
if (slot >=0)
|
|
{
|
|
int result = unzLocateFile(zipfile, fileName, 0);
|
|
if (result == UNZ_OK)
|
|
{
|
|
unz_file_info info;
|
|
result = unzGetCurrentFileInfo(zipfile, &info, NULL, 0, NULL, 0, NULL, 0);
|
|
if (result != UNZ_OK)
|
|
{
|
|
printf("unzGetCurrentFileInfo() != UNZ_OK (%d)\n", result);
|
|
slot=-1;
|
|
unzClose(zipfile);
|
|
zipfile = 0;
|
|
}
|
|
else
|
|
{
|
|
result = unzOpenCurrentFile(zipfile);
|
|
if (result == UNZ_OK)
|
|
{
|
|
printf("zipFile::fileOpen %s in mode %s in fileHandle %d\n", fileName, mode, slot);
|
|
m_fileHandles[slot] = zipfile;
|
|
} else
|
|
{
|
|
slot=-1;
|
|
unzClose(zipfile);
|
|
zipfile = 0;
|
|
}
|
|
}
|
|
} else
|
|
{
|
|
slot=-1;
|
|
unzClose(zipfile);
|
|
zipfile = 0;
|
|
}
|
|
}
|
|
}
|
|
return slot;
|
|
}
|
|
virtual int fileRead(int fileHandle, char* destBuffer, int numBytes)
|
|
{
|
|
int result = -1;
|
|
if (fileHandle>=0 && fileHandle < B3_ZIP_FILEIO_MAX_FILES )
|
|
{
|
|
unzFile f = m_fileHandles[fileHandle];
|
|
if (f)
|
|
{
|
|
result = unzReadCurrentFile(f, destBuffer,numBytes);
|
|
//::fread(destBuffer, 1, numBytes, f);
|
|
}
|
|
}
|
|
return result;
|
|
|
|
}
|
|
virtual int fileWrite(int fileHandle,const char* sourceBuffer, int numBytes)
|
|
{
|
|
#if 0
|
|
if (fileHandle>=0 && fileHandle < B3_ZIP_FILEIO_MAX_FILES )
|
|
{
|
|
FILE* f = m_fileHandles[fileHandle];
|
|
if (f)
|
|
{
|
|
return ::fwrite(sourceBuffer, 1, numBytes,m_fileHandles[fileHandle]);
|
|
}
|
|
}
|
|
#endif
|
|
return -1;
|
|
}
|
|
virtual void fileClose(int fileHandle)
|
|
{
|
|
if (fileHandle>=0 && fileHandle < B3_ZIP_FILEIO_MAX_FILES )
|
|
{
|
|
unzFile f = m_fileHandles[fileHandle];
|
|
if (f)
|
|
{
|
|
printf("zipFile::fileClose slot %d\n", fileHandle);
|
|
unzClose(f);
|
|
m_fileHandles[fileHandle]=0;
|
|
}
|
|
}
|
|
}
|
|
|
|
virtual bool findResourcePath(const char* fileName, char* relativeFileName, int relativeFileNameSizeInBytes)
|
|
{
|
|
return b3ResourcePath::findResourcePath(fileName, relativeFileName, relativeFileNameSizeInBytes, ZipFileIO::FileIOPluginFindFile, this);
|
|
}
|
|
|
|
|
|
virtual bool findFile(const char* orgFileName, char* relativeFileName, int maxRelativeFileNameMaxLen)
|
|
{
|
|
int fileHandle = -1;
|
|
fileHandle = fileOpen(orgFileName, "rb");
|
|
if (fileHandle>=0)
|
|
{
|
|
//printf("original file found: [%s]\n", orgFileName);
|
|
sprintf(relativeFileName, "%s", orgFileName);
|
|
fileClose(fileHandle);
|
|
return true;
|
|
}
|
|
|
|
//printf("Trying various directories, relative to current working directory\n");
|
|
const char* prefix[] = {"./", "./data/", "../data/", "../../data/", "../../../data/", "../../../../data/"};
|
|
int numPrefixes = sizeof(prefix) / sizeof(const char*);
|
|
|
|
int f = 0;
|
|
bool fileFound = false;
|
|
|
|
for (int i = 0; !f && i < numPrefixes; i++)
|
|
{
|
|
#ifdef _MSC_VER
|
|
sprintf_s(relativeFileName, maxRelativeFileNameMaxLen, "%s%s", prefix[i], orgFileName);
|
|
#else
|
|
sprintf(relativeFileName, "%s%s", prefix[i], orgFileName);
|
|
#endif
|
|
f = fileOpen(relativeFileName, "rb");
|
|
if (f>=0)
|
|
{
|
|
fileFound = true;
|
|
break;
|
|
}
|
|
}
|
|
if (f>=0)
|
|
{
|
|
fileClose(f);
|
|
}
|
|
|
|
return fileFound;
|
|
}
|
|
virtual char* readLine(int fileHandle, char* destBuffer, int numBytes)
|
|
{
|
|
int numRead = 0;
|
|
|
|
if (fileHandle>=0 && fileHandle < B3_ZIP_FILEIO_MAX_FILES )
|
|
{
|
|
unzFile f = m_fileHandles[fileHandle];
|
|
if (f)
|
|
{
|
|
//return ::fgets(destBuffer, numBytes, m_fileHandles[fileHandle]);
|
|
char c = 0;
|
|
do
|
|
{
|
|
fileRead(fileHandle,&c,1);
|
|
if (c && c!='\n')
|
|
{
|
|
if (c!=13)
|
|
{
|
|
destBuffer[numRead++]=c;
|
|
} else
|
|
{
|
|
destBuffer[numRead++]=0;
|
|
}
|
|
}
|
|
} while (c != 0 && c != '\n' && numRead<(numBytes-1));
|
|
}
|
|
}
|
|
if (numRead<numBytes && numRead>0)
|
|
{
|
|
destBuffer[numRead]=0;
|
|
return &destBuffer[0];
|
|
}
|
|
return 0;
|
|
}
|
|
virtual int getFileSize(int fileHandle)
|
|
{
|
|
int size=0;
|
|
|
|
if (fileHandle>=0 && fileHandle < B3_ZIP_FILEIO_MAX_FILES )
|
|
{
|
|
unzFile f = m_fileHandles[fileHandle];
|
|
if (f)
|
|
{
|
|
unz_file_info info;
|
|
int result = unzGetCurrentFileInfo(f, &info, NULL, 0, NULL, 0, NULL, 0);
|
|
if (result == UNZ_OK)
|
|
{
|
|
size = info.uncompressed_size;
|
|
}
|
|
}
|
|
}
|
|
return size;
|
|
}
|
|
|
|
virtual void enableFileCaching(bool enable)
|
|
{
|
|
(void)enable;
|
|
}
|
|
|
|
};
|