mirror of
https://github.com/AquariaOSE/Aquaria.git
synced 2025-01-25 01:36:38 +00:00
70b8dcdc3a
Major code/logic cleanups + it has a multithreaded batch mode now. The VFS code doesn't like multiple threads at all, so for now there's a big lock in front of where it matters. Changed precacher, map/tileset, and worldmap loading to batched mode. Still TODO: - fix broken mod preview images - reloading resources on debug key - make mod recache entirely unnecessary - actually drop resources when no longer needed
153 lines
2.7 KiB
C++
153 lines
2.7 KiB
C++
#define VFS_ENABLE_C_API 1
|
|
|
|
#include <ttvfs.h>
|
|
#include "ttvfs_stdio.h"
|
|
#include <assert.h>
|
|
#include <sstream>
|
|
#include <stdio.h>
|
|
|
|
// HACK: add a big lock to make this thing not crash when multiple threads are active
|
|
#include "../../BBGE/MT.h"
|
|
static Lockable lock;
|
|
|
|
|
|
static ttvfs::Root *vfs = NULL;
|
|
|
|
void ttvfs_setroot(ttvfs::Root *root)
|
|
{
|
|
vfs = root;
|
|
}
|
|
|
|
VFILE* vfgetfile(const char* fn)
|
|
{
|
|
MTGuard g(lock);
|
|
return vfs->GetFile(fn);
|
|
}
|
|
|
|
VFILE *vfopen(const char *fn, const char *mode)
|
|
{
|
|
MTGuard g(lock);
|
|
VFILE *vf = vfs->GetFile(fn);
|
|
if (!vf || !vf->open(mode))
|
|
return NULL;
|
|
vf->incref(); // keep the file alive until closed.
|
|
return vf;
|
|
}
|
|
|
|
size_t vfread(void *ptr, size_t size, size_t count, VFILE *vf)
|
|
{
|
|
return vf->read(ptr, size * count) / size;
|
|
}
|
|
|
|
int vfclose(VFILE *vf)
|
|
{
|
|
MTGuard g(lock);
|
|
vf->close();
|
|
vf->decref();
|
|
return 0;
|
|
}
|
|
|
|
size_t vfwrite(const void *ptr, size_t size, size_t count, VFILE *vf)
|
|
{
|
|
return vf->write(ptr, size * count) / size;
|
|
}
|
|
|
|
// return 0 on success, -1 on error
|
|
int vfseek(VFILE *vf, long int offset, int origin)
|
|
{
|
|
return vf->seek(offset, origin) ? 0 : -1;
|
|
}
|
|
|
|
// warning: slow
|
|
char *vfgets(char *str, int num, VFILE *vf)
|
|
{
|
|
char *s = str;
|
|
if (vf->iseof())
|
|
return NULL;
|
|
unsigned int remain = int(vf->size() - vf->getpos());
|
|
if (remain < (unsigned int)num)
|
|
num = remain;
|
|
else
|
|
--num; // be sure to keep space for the final null char
|
|
|
|
for(int i = 0; i < num; ++i)
|
|
{
|
|
char c;
|
|
if(!vf->read(&c, 1))
|
|
break;
|
|
*s++ = c;
|
|
if(c == '\n' || c == '\r')
|
|
break;
|
|
}
|
|
|
|
*s++ = 0;
|
|
return str;
|
|
}
|
|
|
|
int vfsize(VFILE *f, size_t *sizep)
|
|
{
|
|
ttvfs::vfspos sz = f->size();
|
|
if(sz == ttvfs::npos)
|
|
return -1;
|
|
*sizep = (size_t)sz;
|
|
return 0;
|
|
}
|
|
|
|
long int vftell(VFILE *vf)
|
|
{
|
|
return (long int)vf->getpos();
|
|
}
|
|
|
|
int vfeof(VFILE *vf)
|
|
{
|
|
return vf->iseof();
|
|
}
|
|
|
|
InStream::InStream(const std::string& fn)
|
|
: std::istringstream()
|
|
{
|
|
open(fn.c_str());
|
|
}
|
|
|
|
InStream::InStream(const char *fn)
|
|
: std::istringstream()
|
|
{
|
|
open(fn);
|
|
}
|
|
|
|
bool InStream::open(const char *fn)
|
|
{
|
|
ttvfs::File *vf = NULL;
|
|
{
|
|
MTGuard g(lock);
|
|
vf = vfs->GetFile(fn);
|
|
if(!vf || !vf->open("r"))
|
|
{
|
|
setstate(std::ios::failbit);
|
|
return false;
|
|
}
|
|
}
|
|
size_t sz = (size_t)vf->size();
|
|
std::string s;
|
|
s.resize(sz);
|
|
size_t bytes = vf->read(&s[0], sz);
|
|
s.resize(bytes);
|
|
str(s);
|
|
vf->close();
|
|
return true;
|
|
}
|
|
|
|
int ttvfs_stdio_fsize(VFILE *f, size_t *sizep)
|
|
{
|
|
long int sz = 0;
|
|
if ( vfseek(f, 0, SEEK_END) != 0
|
|
|| (sz = vftell(f)) < 0
|
|
|| vfseek(f, 0, SEEK_SET) != 0)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
*sizep = sz;
|
|
return 0;
|
|
}
|
|
|