f53df7da64
Code submissions have continually suffered from formatting inconsistencies that constantly have to be addressed. Using clang-format simplifies this by making code formatting more consistent, and allows automation of the code formatting so that maintainers can focus more on the code itself instead of code formatting.
187 lines
4.0 KiB
C
187 lines
4.0 KiB
C
/*
|
|
* Copyright (c) 2015 Hugh Bailey <obs.jim@gmail.com>
|
|
*
|
|
* Permission to use, copy, modify, and distribute this software for any
|
|
* purpose with or without fee is hereby granted, provided that the above
|
|
* copyright notice and this permission notice appear in all copies.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
*/
|
|
|
|
#include "dstr.h"
|
|
#include "file-serializer.h"
|
|
#include "platform.h"
|
|
|
|
static size_t file_input_read(void *file, void *data, size_t size)
|
|
{
|
|
return fread(data, 1, size, file);
|
|
}
|
|
|
|
static int64_t file_input_seek(void *file, int64_t offset,
|
|
enum serialize_seek_type seek_type)
|
|
{
|
|
int origin = SEEK_SET;
|
|
|
|
switch (seek_type) {
|
|
case SERIALIZE_SEEK_START:
|
|
origin = SEEK_SET;
|
|
break;
|
|
case SERIALIZE_SEEK_CURRENT:
|
|
origin = SEEK_CUR;
|
|
break;
|
|
case SERIALIZE_SEEK_END:
|
|
origin = SEEK_END;
|
|
break;
|
|
}
|
|
|
|
if (os_fseeki64(file, offset, origin) == -1)
|
|
return -1;
|
|
|
|
return os_ftelli64(file);
|
|
}
|
|
|
|
static int64_t file_input_get_pos(void *file)
|
|
{
|
|
return os_ftelli64(file);
|
|
}
|
|
|
|
bool file_input_serializer_init(struct serializer *s, const char *path)
|
|
{
|
|
s->data = os_fopen(path, "rb");
|
|
if (!s->data)
|
|
return false;
|
|
|
|
s->read = file_input_read;
|
|
s->write = NULL;
|
|
s->seek = file_input_seek;
|
|
s->get_pos = file_input_get_pos;
|
|
return true;
|
|
}
|
|
|
|
void file_input_serializer_free(struct serializer *s)
|
|
{
|
|
if (s->data)
|
|
fclose(s->data);
|
|
}
|
|
|
|
/* ------------------------------------------------------------------------- */
|
|
|
|
struct file_output_data {
|
|
FILE *file;
|
|
char *temp_name;
|
|
char *file_name;
|
|
};
|
|
|
|
static size_t file_output_write(void *sdata, const void *data, size_t size)
|
|
{
|
|
struct file_output_data *out = sdata;
|
|
return fwrite(data, 1, size, out->file);
|
|
}
|
|
|
|
static int64_t file_output_seek(void *sdata, int64_t offset,
|
|
enum serialize_seek_type seek_type)
|
|
{
|
|
struct file_output_data *out = sdata;
|
|
int origin = SEEK_SET;
|
|
|
|
switch (seek_type) {
|
|
case SERIALIZE_SEEK_START:
|
|
origin = SEEK_SET;
|
|
break;
|
|
case SERIALIZE_SEEK_CURRENT:
|
|
origin = SEEK_CUR;
|
|
break;
|
|
case SERIALIZE_SEEK_END:
|
|
origin = SEEK_END;
|
|
break;
|
|
}
|
|
|
|
if (os_fseeki64(out->file, offset, origin) == -1)
|
|
return -1;
|
|
|
|
return os_ftelli64(out->file);
|
|
}
|
|
|
|
static int64_t file_output_get_pos(void *sdata)
|
|
{
|
|
struct file_output_data *out = sdata;
|
|
return os_ftelli64(out->file);
|
|
}
|
|
|
|
bool file_output_serializer_init(struct serializer *s, const char *path)
|
|
{
|
|
FILE *file = os_fopen(path, "wb");
|
|
struct file_output_data *out;
|
|
|
|
if (!file)
|
|
return false;
|
|
|
|
out = bzalloc(sizeof(*out));
|
|
out->file = file;
|
|
|
|
s->data = out;
|
|
s->read = NULL;
|
|
s->write = file_output_write;
|
|
s->seek = file_output_seek;
|
|
s->get_pos = file_output_get_pos;
|
|
return true;
|
|
}
|
|
|
|
bool file_output_serializer_init_safe(struct serializer *s, const char *path,
|
|
const char *temp_ext)
|
|
{
|
|
struct dstr temp_name = {0};
|
|
struct file_output_data *out;
|
|
FILE *file;
|
|
|
|
if (!temp_ext || !*temp_ext)
|
|
return false;
|
|
|
|
dstr_copy(&temp_name, path);
|
|
if (*temp_ext != '.')
|
|
dstr_cat_ch(&temp_name, '.');
|
|
dstr_cat(&temp_name, temp_ext);
|
|
|
|
file = os_fopen(temp_name.array, "wb");
|
|
if (!file) {
|
|
dstr_free(&temp_name);
|
|
return false;
|
|
}
|
|
|
|
out = bzalloc(sizeof(*out));
|
|
out->file_name = bstrdup(path);
|
|
out->temp_name = temp_name.array;
|
|
out->file = file;
|
|
|
|
s->data = out;
|
|
s->read = NULL;
|
|
s->write = file_output_write;
|
|
s->seek = file_output_seek;
|
|
s->get_pos = file_output_get_pos;
|
|
return true;
|
|
}
|
|
|
|
void file_output_serializer_free(struct serializer *s)
|
|
{
|
|
struct file_output_data *out = s->data;
|
|
|
|
if (out) {
|
|
fclose(out->file);
|
|
|
|
if (out->temp_name) {
|
|
os_unlink(out->file_name);
|
|
os_rename(out->temp_name, out->file_name);
|
|
}
|
|
|
|
bfree(out->file_name);
|
|
bfree(out->temp_name);
|
|
bfree(out);
|
|
}
|
|
}
|