X-Git-Url: http://git.tdb.fi/?a=blobdiff_plain;f=file.c;h=aff1d51f2d2c87b470f2b61b23d6c8b945870f73;hb=8add7917ce9da2faa7751798c3fb15a24715ae1e;hp=192dda675df9a1f5cfc39f04deb5670596765e33;hpb=34d682416faee9c8beef8b0747d5f1587dea847c;p=ext%2Fsubsurface.git diff --git a/file.c b/file.c index 192dda6..aff1d51 100644 --- a/file.c +++ b/file.c @@ -6,16 +6,13 @@ #include #include "dive.h" - -struct memblock { - void *buffer; - size_t size; -}; +#include "file.h" static int readfile(const char *filename, struct memblock *mem) { int ret, fd = open(filename, O_RDONLY); struct stat st; + char *buf; mem->buffer = NULL; mem->size = 0; @@ -32,15 +29,17 @@ static int readfile(const char *filename, struct memblock *mem) ret = 0; if (!st.st_size) goto out; - mem->buffer = malloc(st.st_size); + buf = malloc(st.st_size+1); ret = -1; errno = ENOMEM; - if (!mem->buffer) + if (!buf) goto out; + mem->buffer = buf; mem->size = st.st_size; - ret = read(fd, mem->buffer, mem->size); + ret = read(fd, buf, mem->size); if (ret < 0) goto free; + buf[ret] = 0; if (ret == mem->size) goto out; errno = EIO; @@ -72,10 +71,11 @@ static void suunto_read(struct zip_file *file, GError **error) } #endif -static int try_to_open_suunto(const char *filename, GError **error) +static int try_to_open_suunto(const char *filename, struct memblock *mem, GError **error) { int success = 0; #ifdef LIBZIP + /* Grr. libzip needs to re-open the file, it can't take a buffer */ struct zip *zip = zip_open(filename, ZIP_CHECKCONS, NULL); if (zip) { @@ -94,24 +94,32 @@ static int try_to_open_suunto(const char *filename, GError **error) return success; } -static int open_by_filename(const char *filename, const char *fmt, GError **error) +static int open_by_filename(const char *filename, const char *fmt, struct memblock *mem, GError **error) { /* Suunto Dive Manager files: SDE */ if (!strcasecmp(fmt, "SDE")) - return try_to_open_suunto(filename, error); + return try_to_open_suunto(filename, mem, error); + + /* Truly nasty intentionally obfuscated Cochran Anal software */ + if (!strcasecmp(fmt, "CAN")) + return try_to_open_cochran(filename, mem, error); return 0; } +static void parse_file_buffer(const char *filename, struct memblock *mem, GError **error) +{ + char *fmt = strrchr(filename, '.'); + if (fmt && open_by_filename(filename, fmt+1, mem, error)) + return; + + parse_xml_buffer(filename, mem->buffer, mem->size, error); +} + void parse_file(const char *filename, GError **error) { - char *fmt; struct memblock mem; - fmt = strrchr(filename, '.'); - if (fmt && open_by_filename(filename, fmt+1, error)) - return; - if (readfile(filename, &mem) < 0) { fprintf(stderr, "Failed to read '%s'.\n", filename); if (error) { @@ -123,6 +131,6 @@ void parse_file(const char *filename, GError **error) return; } - parse_xml_buffer(filename, mem.buffer, mem.size, error); + parse_file_buffer(filename, &mem, error); free(mem.buffer); }