89750a20f8
- Added a global file locker so threads can synchronize when accessing files - Added mutexes to serialize access to file streams (i.e to call in series) - VoxelStream API now expects implementations to be thread-safe - Did some logic/data separation in RegionFile in case we need an in-memory implementation, however I might just use SQLite in the future instead
113 lines
3.0 KiB
C++
113 lines
3.0 KiB
C++
#include "file_utils.h"
|
|
#include "../server/voxel_server.h"
|
|
|
|
const char *to_string(VoxelFileResult res) {
|
|
switch (res) {
|
|
case VOXEL_FILE_OK:
|
|
return "OK";
|
|
case VOXEL_FILE_CANT_OPEN:
|
|
return "Can't open";
|
|
case VOXEL_FILE_DOES_NOT_EXIST:
|
|
return "Doesn't exist";
|
|
case VOXEL_FILE_UNEXPECTED_EOF:
|
|
return "Unexpected end of file";
|
|
case VOXEL_FILE_INVALID_MAGIC:
|
|
return "Invalid magic";
|
|
case VOXEL_FILE_INVALID_VERSION:
|
|
return "Invalid version";
|
|
case VOXEL_FILE_INVALID_DATA:
|
|
return "Invalid data";
|
|
default:
|
|
CRASH_NOW();
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
VoxelFileResult check_magic_and_version(FileAccess *f, uint8_t expected_version, const char *expected_magic, uint8_t &out_version) {
|
|
uint8_t magic[5] = { '\0' };
|
|
int count = f->get_buffer(magic, 4);
|
|
if (count != 4) {
|
|
return VOXEL_FILE_UNEXPECTED_EOF;
|
|
}
|
|
for (int i = 0; i < 4; ++i) {
|
|
if (magic[i] != expected_magic[i]) {
|
|
return VOXEL_FILE_INVALID_MAGIC;
|
|
}
|
|
}
|
|
|
|
out_version = f->get_8();
|
|
if (out_version != expected_version) {
|
|
return VOXEL_FILE_INVALID_VERSION;
|
|
}
|
|
|
|
return VOXEL_FILE_OK;
|
|
}
|
|
|
|
Error check_directory_created(const String &directory_path) {
|
|
VoxelFileLockerWrite file_wlock(directory_path);
|
|
DirAccess *d = DirAccess::create_for_path(directory_path);
|
|
|
|
if (d == nullptr) {
|
|
ERR_PRINT("Could not access to filesystem");
|
|
return ERR_FILE_CANT_OPEN;
|
|
}
|
|
|
|
if (!d->exists(directory_path)) {
|
|
// Create if not exist
|
|
Error err = d->make_dir_recursive(directory_path);
|
|
if (err != OK) {
|
|
ERR_PRINT("Could not create directory");
|
|
memdelete(d);
|
|
return err;
|
|
}
|
|
}
|
|
|
|
memdelete(d);
|
|
return OK;
|
|
}
|
|
|
|
namespace VoxelFileUtils {
|
|
|
|
// TODO Write tests
|
|
|
|
// Makes the file bigger to move the half from the current position further,
|
|
// so that it makes room for the specified amount of bytes.
|
|
// The new allocated "free" bytes have undefined values, which may be later overwritten by the caller anyways.
|
|
void insert_bytes(FileAccess *f, size_t count, size_t temp_chunk_size) {
|
|
CRASH_COND(f == nullptr);
|
|
CRASH_COND(temp_chunk_size == 0);
|
|
|
|
const size_t prev_file_len = f->get_len();
|
|
const size_t insert_pos = f->get_position();
|
|
|
|
// Make the file larger
|
|
f->seek(prev_file_len);
|
|
for (size_t i = 0; i < count; ++i) {
|
|
f->store_8(0);
|
|
}
|
|
|
|
const size_t initial_bytes_to_move = prev_file_len - insert_pos;
|
|
size_t bytes_to_move = initial_bytes_to_move;
|
|
std::vector<uint8_t> temp;
|
|
size_t src_pos = prev_file_len;
|
|
size_t dst_pos = f->get_len();
|
|
|
|
// Copy chunks of the file at a later position, from last to first.
|
|
// The last copied chunk can be smaller.
|
|
while (bytes_to_move > 0) {
|
|
size_t chunk_size = bytes_to_move >= temp_chunk_size ? temp_chunk_size : bytes_to_move;
|
|
src_pos -= chunk_size;
|
|
dst_pos -= chunk_size;
|
|
temp.resize(chunk_size);
|
|
f->seek(src_pos);
|
|
const size_t read_size = f->get_buffer(temp.data(), temp.size());
|
|
CRASH_COND(read_size != temp.size());
|
|
f->seek(dst_pos);
|
|
f->store_buffer(temp.data(), temp.size());
|
|
bytes_to_move -= temp.size();
|
|
CRASH_COND(bytes_to_move >= initial_bytes_to_move);
|
|
}
|
|
}
|
|
|
|
} // namespace VoxelFileUtils
|