1
0
Fork 0
mirror of https://github.com/AquariaOSE/Aquaria.git synced 2025-01-25 01:36:38 +00:00
Aquaria/ExternalLibs/ttvfs_cfileapi/ttvfs_stdio.cpp
fgenesis 70b8dcdc3a Rework texture loading, part 1
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
2023-05-31 00:55:16 +02:00

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;
}