105 lines
2.3 KiB
C
105 lines
2.3 KiB
C
|
#include "skip.h"
|
||
|
|
||
|
#include "error.h"
|
||
|
#include "file.h"
|
||
|
#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;
|
||
|
|
||
|
skip = sw_skip_create();
|
||
|
fil = fopen(filename, "rb");
|
||
|
|
||
|
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);
|
||
|
}
|
||
|
|
||
|
fclose(fil);
|
||
|
return skip;
|
||
|
}
|
||
|
|
||
|
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);
|
||
|
}
|
||
|
|
||
|
void sw_skip_save(struct sw_skip *skip, char *filename) {
|
||
|
FILE *fil;
|
||
|
u32 size;
|
||
|
u16 i;
|
||
|
|
||
|
size = 0;
|
||
|
for(i = 0; i < skip->file_count; ++i) {
|
||
|
size += skip->files[i].size;
|
||
|
}
|
||
|
|
||
|
fil = fopen(filename, "wb");
|
||
|
sw_fwrite_str_nozero(fil, "SWZP");
|
||
|
sw_fwrite_u8(fil, 0); /* version */
|
||
|
sw_fwrite_u8(fil, 0); /* compression */
|
||
|
sw_fwrite_u32(fil, size); /* uncompressed size */
|
||
|
sw_fwrite_u32(fil, size); /* compressed size */
|
||
|
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 */
|
||
|
}
|
||
|
|
||
|
sw_fwrite_u32(fil, ftell(fil) + 4); /* file length */
|
||
|
|
||
|
fclose(fil);
|
||
|
}
|