2012-01-26 21:00:45 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <stdlib.h>
|
2012-01-27 01:43:33 +00:00
|
|
|
#include <string.h>
|
2012-01-26 21:00:45 +00:00
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include "dive.h"
|
2012-01-27 20:43:40 +00:00
|
|
|
#include "file.h"
|
2012-01-26 21:00:45 +00:00
|
|
|
|
|
|
|
static int readfile(const char *filename, struct memblock *mem)
|
|
|
|
{
|
|
|
|
int ret, fd = open(filename, O_RDONLY);
|
|
|
|
struct stat st;
|
2012-01-27 18:56:36 +00:00
|
|
|
char *buf;
|
2012-01-26 21:00:45 +00:00
|
|
|
|
|
|
|
mem->buffer = NULL;
|
|
|
|
mem->size = 0;
|
|
|
|
|
|
|
|
fd = open(filename, O_RDONLY);
|
|
|
|
if (fd < 0)
|
|
|
|
return fd;
|
|
|
|
ret = fstat(fd, &st);
|
|
|
|
if (ret < 0)
|
|
|
|
goto out;
|
|
|
|
ret = -EINVAL;
|
|
|
|
if (!S_ISREG(st.st_mode))
|
|
|
|
goto out;
|
|
|
|
ret = 0;
|
|
|
|
if (!st.st_size)
|
|
|
|
goto out;
|
2012-01-27 18:56:36 +00:00
|
|
|
buf = malloc(st.st_size+1);
|
2012-01-26 21:00:45 +00:00
|
|
|
ret = -1;
|
|
|
|
errno = ENOMEM;
|
2012-01-27 18:56:36 +00:00
|
|
|
if (!buf)
|
2012-01-26 21:00:45 +00:00
|
|
|
goto out;
|
2012-01-27 18:56:36 +00:00
|
|
|
mem->buffer = buf;
|
2012-01-26 21:00:45 +00:00
|
|
|
mem->size = st.st_size;
|
2012-01-27 18:56:36 +00:00
|
|
|
ret = read(fd, buf, mem->size);
|
2012-01-26 21:00:45 +00:00
|
|
|
if (ret < 0)
|
|
|
|
goto free;
|
2012-01-27 18:56:36 +00:00
|
|
|
buf[ret] = 0;
|
2012-01-26 21:00:45 +00:00
|
|
|
if (ret == mem->size)
|
|
|
|
goto out;
|
|
|
|
errno = EIO;
|
|
|
|
ret = -1;
|
|
|
|
free:
|
|
|
|
free(mem->buffer);
|
|
|
|
mem->buffer = NULL;
|
|
|
|
mem->size = 0;
|
|
|
|
out:
|
|
|
|
close(fd);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2012-01-27 01:43:33 +00:00
|
|
|
#ifdef LIBZIP
|
|
|
|
#include <zip.h>
|
|
|
|
|
|
|
|
static void suunto_read(struct zip_file *file, GError **error)
|
|
|
|
{
|
|
|
|
int size = 1024, n, read = 0;
|
|
|
|
char *mem = malloc(size);
|
|
|
|
|
|
|
|
while ((n = zip_fread(file, mem+read, size-read)) > 0) {
|
|
|
|
read += n;
|
|
|
|
size = read * 3 / 2;
|
|
|
|
mem = realloc(mem, size);
|
|
|
|
}
|
|
|
|
parse_xml_buffer("SDE file", mem, read, error);
|
|
|
|
free(mem);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2012-01-27 18:56:36 +00:00
|
|
|
static int try_to_open_suunto(const char *filename, struct memblock *mem, GError **error)
|
2012-01-27 01:43:33 +00:00
|
|
|
{
|
|
|
|
int success = 0;
|
|
|
|
#ifdef LIBZIP
|
2012-01-27 18:56:36 +00:00
|
|
|
/* Grr. libzip needs to re-open the file, it can't take a buffer */
|
2012-01-27 01:43:33 +00:00
|
|
|
struct zip *zip = zip_open(filename, ZIP_CHECKCONS, NULL);
|
|
|
|
|
|
|
|
if (zip) {
|
|
|
|
int index;
|
|
|
|
for (index = 0; ;index++) {
|
|
|
|
struct zip_file *file = zip_fopen_index(zip, index, 0);
|
|
|
|
if (!file)
|
|
|
|
break;
|
|
|
|
suunto_read(file, error);
|
|
|
|
zip_fclose(file);
|
|
|
|
success++;
|
|
|
|
}
|
|
|
|
zip_close(zip);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
2012-01-27 18:56:36 +00:00
|
|
|
static int open_by_filename(const char *filename, const char *fmt, struct memblock *mem, GError **error)
|
2012-01-27 01:43:33 +00:00
|
|
|
{
|
|
|
|
/* Suunto Dive Manager files: SDE */
|
|
|
|
if (!strcasecmp(fmt, "SDE"))
|
2012-01-27 18:56:36 +00:00
|
|
|
return try_to_open_suunto(filename, mem, error);
|
2012-01-27 01:43:33 +00:00
|
|
|
|
2012-01-27 20:43:40 +00:00
|
|
|
/* Truly nasty intentionally obfuscated Cochran Anal software */
|
|
|
|
if (!strcasecmp(fmt, "CAN"))
|
|
|
|
return try_to_open_cochran(filename, mem, error);
|
|
|
|
|
2012-01-27 01:43:33 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-01-27 18:56:36 +00:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2012-01-26 21:00:45 +00:00
|
|
|
void parse_file(const char *filename, GError **error)
|
|
|
|
{
|
|
|
|
struct memblock mem;
|
|
|
|
|
|
|
|
if (readfile(filename, &mem) < 0) {
|
|
|
|
fprintf(stderr, "Failed to read '%s'.\n", filename);
|
|
|
|
if (error) {
|
|
|
|
*error = g_error_new(g_quark_from_string("subsurface"),
|
|
|
|
DIVE_ERROR_PARSE,
|
|
|
|
"Failed to read '%s'",
|
|
|
|
filename);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-01-27 18:56:36 +00:00
|
|
|
parse_file_buffer(filename, &mem, error);
|
2012-01-26 21:00:45 +00:00
|
|
|
free(mem.buffer);
|
|
|
|
}
|