skunkworks-c/src/skip.c

201 lines
4.4 KiB
C
Raw Normal View History

2023-02-09 19:36:39 +01:00
#include "skip.h"
#include "error.h"
#include "file.h"
2023-02-25 00:14:05 +01:00
#include "str.h"
2023-02-09 19:36:39 +01:00
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
struct sw_skip *sw_skip_create() {
struct sw_skip *skip;
skip = calloc(1, sizeof(struct sw_skip));
return skip;
}
struct sw_skip *sw_skip_load(char *filename) {
FILE *fil;
struct sw_skip *skip;
char magic[5] = {0};
u8 version, compression;
u16 i;
char str[257];
u8 str_len;
2023-02-25 00:14:05 +01:00
u32 off;
2023-02-09 19:36:39 +01:00
skip = sw_skip_create();
fil = fopen(filename, "rb");
2023-02-25 00:14:05 +01:00
fseek(fil, -4, SEEK_END);
fread(&off, 4, 1, fil);
fseek(fil, -off, SEEK_END);
2023-02-09 19:36:39 +01:00
fread(magic, 1, 4, fil);
if(strcmp("SWZP", magic) != 0) {
sw_error("not a .skip file");
}
fread(&version, 1, 1, fil);
if(version > 0) {
sw_error("version %i of .skip not supported", version);
}
fread(&compression, 1, 1, fil);
if(compression > 0) {
sw_error("unsupported .skip compression method");
}
fread(&skip->data_size, 1, 4, fil);
fseek(fil, 4, SEEK_CUR); /* ignore compressed size */
fread(&skip->file_count, 1, 2, fil);
skip->files = malloc(sizeof(skip->files[0]) * skip->file_count);
for(i = 0; i < skip->file_count; ++i) {
memset(str, 0, 257);
fread(&skip->files[i].offset, 1, 4, fil);
fread(&skip->files[i].size, 1, 4, fil);
fread(&str_len, 1, 1, fil);
fread(str, str_len, 1, fil);
str[str_len] = 0;
skip->files[i].name = malloc(str_len + 1);
strcpy(skip->files[i].name, str);
}
2023-02-25 00:14:05 +01:00
skip->data = malloc(skip->data_size);
fread(skip->data, 1, skip->data_size, fil);
2023-02-09 19:36:39 +01:00
fclose(fil);
return skip;
}
2023-02-25 00:14:05 +01:00
static void sw_skip_save_to(struct sw_skip *skip, FILE *fil);
2023-02-09 19:36:39 +01:00
void sw_skip_destroy(struct sw_skip *skip) {
i32 i;
for(i = 0; i < skip->file_count; ++i) {
free(skip->files[i].name);
}
free(skip->files);
free(skip->data);
free(skip);
}
2023-02-25 00:14:05 +01:00
void sw_skip_add(struct sw_skip *skip, u8 *data, u32 data_size, char *name) {
u32 off;
off = skip->data_size + 1;
skip->data_size += data_size;
skip->data = realloc(skip->data, skip->data_size);
memcpy(skip->data + off, data, data_size);
skip->file_count += 1;
skip->files =
realloc(skip->files, sizeof(skip->files[0]) * skip->file_count);
2023-02-26 23:29:17 +01:00
skip->files[skip->file_count - 1].name = sw_str_dup(name);
2023-02-25 00:14:05 +01:00
skip->files[skip->file_count - 1].offset = off;
skip->files[skip->file_count - 1].size = data_size;
}
void sw_skip_add_file(struct sw_skip *skip, char *name, char *filename) {
2023-02-09 19:36:39 +01:00
FILE *fil;
2023-02-25 00:14:05 +01:00
u32 len;
u8 *data;
fil = fopen(filename, "rb");
fseek(fil, 0, SEEK_END);
len = ftell(fil);
fseek(fil, 0, SEEK_SET);
data = malloc(len);
fread(data, 1, len, fil);
fclose(fil);
sw_skip_add(skip, data, len, name);
free(data);
}
void sw_skip_attach(struct sw_skip *skip, char *filename) {
FILE *fil;
fil = fopen(filename, "rb+");
fseek(fil, 0, SEEK_END);
sw_skip_save_to(skip, fil);
}
void sw_skip_dettach(char *filename) {
FILE *fil;
char magic[5] = {0};
u32 off;
u32 len;
u8 *data;
fil = fopen(filename, "rb");
fseek(fil, -4, SEEK_END);
fread(&off, 4, 1, fil);
fseek(fil, -off, SEEK_END);
fread(magic, 1, 4, fil);
sw_log("magic: [%s]", magic);
sw_log("off: %i", off);
if(strcmp("SWZP", magic) == 0) {
sw_log("detaching .skip from %s", filename);
len = ftell(fil) - 4;
fseek(fil, 0, SEEK_SET);
data = malloc(len);
fread(data, 1, len, fil);
fclose(fil);
fil = fopen(filename, "wb");
fwrite(data, 1, len, fil);
}
fclose(fil);
}
u8 *sw_skip_get(struct sw_skip *skip, char *name, u32 *out_size) {
u32 i;
2023-02-09 19:36:39 +01:00
for(i = 0; i < skip->file_count; ++i) {
2023-02-25 00:14:05 +01:00
if(strcmp(name, skip->files[i].name) == 0) {
if(out_size) {
*out_size = skip->files[i].size;
}
return skip->data + skip->files[i].offset;
}
2023-02-09 19:36:39 +01:00
}
2023-02-25 00:14:05 +01:00
return 0;
}
2023-02-09 19:36:39 +01:00
2023-02-25 00:14:05 +01:00
void sw_skip_save(struct sw_skip *skip, char *filename) {
sw_skip_save_to(skip, fopen(filename, "wb"));
}
static void sw_skip_save_to(struct sw_skip *skip, FILE *fil) {
u16 i;
u32 start;
start = ftell(fil);
2023-02-09 19:36:39 +01:00
sw_fwrite_str_nozero(fil, "SWZP");
sw_fwrite_u8(fil, 0); /* version */
sw_fwrite_u8(fil, 0); /* compression */
2023-02-25 00:14:05 +01:00
sw_fwrite_u32(fil, skip->data_size); /* uncompressed size */
sw_fwrite_u32(fil, skip->data_size); /* compressed size */
2023-02-09 19:36:39 +01:00
sw_fwrite_u16(fil, skip->file_count); /* file count */
for(i = 0; i < skip->file_count; ++i) {
sw_fwrite_u32(fil, skip->files[i].offset); /* offset */
sw_fwrite_u32(fil, skip->files[i].size); /* file size */
sw_fwrite_str_prefix8(fil, skip->files[i].name); /* file name */
}
2023-02-25 00:14:05 +01:00
fwrite(skip->data, 1, skip->data_size, fil);
sw_fwrite_u32(fil, ftell(fil) + 4 - start); /* file length */
2023-02-09 19:36:39 +01:00
fclose(fil);
}