godot_voxel/streams/file_utils.cpp

120 lines
3.1 KiB
C++

#include "file_utils.h"
#include "../engine/voxel_engine.h"
#include <core/io/dir_access.h>
namespace zylann {
const char *to_string(FileResult res) {
switch (res) {
case FILE_OK:
return "OK";
case FILE_CANT_OPEN:
return "Can't open";
case FILE_DOES_NOT_EXIST:
return "Doesn't exist";
case FILE_UNEXPECTED_EOF:
return "Unexpected end of file";
case FILE_INVALID_MAGIC:
return "Invalid magic";
case FILE_INVALID_VERSION:
return "Invalid version";
case FILE_INVALID_DATA:
return "Invalid data";
default:
CRASH_NOW();
return nullptr;
}
}
FileResult 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 FILE_UNEXPECTED_EOF;
}
for (int i = 0; i < 4; ++i) {
if (magic[i] != expected_magic[i]) {
return FILE_INVALID_MAGIC;
}
}
out_version = f.get_8();
if (out_version != expected_version) {
return FILE_INVALID_VERSION;
}
return FILE_OK;
}
Error check_directory_created(const std::string &p_directory_path) {
const String directory_path(p_directory_path.c_str());
Ref<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");
return err;
}
}
return OK;
}
namespace voxel {
Error check_directory_created_using_file_locker(const std::string &directory_path) {
VoxelFileLockerWrite file_wlock(directory_path);
return check_directory_created(directory_path);
}
} // namespace voxel
// 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(temp_chunk_size == 0);
const size_t prev_file_len = f.get_length();
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_length();
// 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 zylann