Split up file reading from 'parse-xml.c' into 'file.c'

We're going to eventually import non-xml files too, so let's begin
splitting the logic up.

Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
Linus Torvalds 2012-01-26 13:00:45 -08:00
parent 6e0ea1ce61
commit 4d10bc017a
6 changed files with 88 additions and 71 deletions

View file

@ -115,7 +115,7 @@ LIBS = $(LIBXML2) $(LIBXSLT) $(LIBGTK) $(LIBGCONF2) $(LIBDIVECOMPUTER) $(EXTRALI
OBJS = main.o dive.o profile.o info.o equipment.o divelist.o \ OBJS = main.o dive.o profile.o info.o equipment.o divelist.o \
parse-xml.o save-xml.o libdivecomputer.o print.o uemis.o \ parse-xml.o save-xml.o libdivecomputer.o print.o uemis.o \
gtk-gui.o statistics.o $(OSSUPPORT).o $(RESFILE) gtk-gui.o statistics.o file.o $(OSSUPPORT).o $(RESFILE)
$(NAME): $(OBJS) $(NAME): $(OBJS)
$(CC) $(LDFLAGS) -o $(NAME) $(OBJS) $(LIBS) $(CC) $(LDFLAGS) -o $(NAME) $(OBJS) $(LIBS)
@ -147,6 +147,9 @@ install-macosx: $(NAME)
$(INSTALL) $(ICONFILE) $(MACOSXINSTALL)/Contents/Resources/ $(INSTALL) $(ICONFILE) $(MACOSXINSTALL)/Contents/Resources/
$(INSTALL) $(MACOSXFILES)/Subsurface.icns $(MACOSXINSTALL)/Contents/Resources/ $(INSTALL) $(MACOSXFILES)/Subsurface.icns $(MACOSXINSTALL)/Contents/Resources/
file.o: file.c dive.h
$(CC) $(CFLAGS) $(GLIB2CFLAGS) $(XML2CFLAGS) $(XSLT) -c file.c
parse-xml.o: parse-xml.c dive.h parse-xml.o: parse-xml.c dive.h
$(CC) $(CFLAGS) $(GLIB2CFLAGS) $(XML2CFLAGS) $(XSLT) -c parse-xml.c $(CC) $(CFLAGS) $(GLIB2CFLAGS) $(XML2CFLAGS) $(XSLT) -c parse-xml.c

4
dive.h
View file

@ -254,9 +254,11 @@ static inline struct dive *get_dive(unsigned int nr)
} }
extern void parse_xml_init(void); extern void parse_xml_init(void);
extern void parse_xml_file(const char *filename, GError **error); extern void parse_xml_buffer(const char *url, const char *buf, int size, GError **error);
extern void set_filename(const char *filename); extern void set_filename(const char *filename);
extern void parse_file(const char *filename, GError **error);
#ifdef XSLT #ifdef XSLT
extern xmlDoc *test_xslt_transforms(xmlDoc *doc); extern xmlDoc *test_xslt_transforms(xmlDoc *doc);
#endif #endif

73
file.c Normal file
View file

@ -0,0 +1,73 @@
#include <unistd.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <stdlib.h>
#include <errno.h>
#include "dive.h"
struct memblock {
void *buffer;
size_t size;
};
static int readfile(const char *filename, struct memblock *mem)
{
int ret, fd = open(filename, O_RDONLY);
struct stat st;
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;
mem->buffer = malloc(st.st_size);
ret = -1;
errno = ENOMEM;
if (!mem->buffer)
goto out;
mem->size = st.st_size;
ret = read(fd, mem->buffer, mem->size);
if (ret < 0)
goto free;
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;
}
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;
}
parse_xml_buffer(filename, mem.buffer, mem.size, error);
free(mem.buffer);
}

View file

@ -114,7 +114,7 @@ static void file_open(GtkWidget *w, gpointer data)
GError *error = NULL; GError *error = NULL;
while(filenames != NULL) { while(filenames != NULL) {
filename = filenames->data; filename = filenames->data;
parse_xml_file(filename, &error); parse_file(filename, &error);
if (error != NULL) if (error != NULL)
{ {
report_error(error); report_error(error);
@ -961,7 +961,7 @@ static GtkWidget *xml_file_selector(GtkWidget *vbox, GtkWidget *main_dialog)
static void do_import_file(gpointer data, gpointer user_data) static void do_import_file(gpointer data, gpointer user_data)
{ {
GError *error = NULL; GError *error = NULL;
parse_xml_file(data, &error); parse_file(data, &error);
if (error != NULL) if (error != NULL)
{ {

2
main.c
View file

@ -226,7 +226,7 @@ int main(int argc, char **argv)
continue; continue;
} }
GError *error = NULL; GError *error = NULL;
parse_xml_file(a, &error); parse_file(a, &error);
if (error != NULL) if (error != NULL)
{ {

View file

@ -4,8 +4,6 @@
#include <stdlib.h> #include <stdlib.h>
#include <errno.h> #include <errno.h>
#include <unistd.h> #include <unistd.h>
#include <fcntl.h>
#include <sys/stat.h>
#define __USE_XOPEN #define __USE_XOPEN
#include <time.h> #include <time.h>
#include <libxml/parser.h> #include <libxml/parser.h>
@ -1304,84 +1302,25 @@ static void reset_all(void)
import_source = UNKNOWN; import_source = UNKNOWN;
} }
struct memblock { void parse_xml_buffer(const char *url, const char *buffer, int size, GError **error)
void *buffer;
size_t size;
};
static int readfile(const char *filename, struct memblock *mem)
{
int ret, fd = open(filename, O_RDONLY);
struct stat st;
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;
mem->buffer = malloc(st.st_size);
ret = -1;
errno = ENOMEM;
if (!mem->buffer)
goto out;
mem->size = st.st_size;
ret = read(fd, mem->buffer, mem->size);
if (ret < 0)
goto free;
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;
}
void parse_xml_file(const char *filename, GError **error)
{ {
xmlDoc *doc; xmlDoc *doc;
struct memblock mem;
if (readfile(filename, &mem) < 0) { doc = xmlReadMemory(buffer, size, url, NULL, 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;
}
doc = xmlReadMemory(mem.buffer, mem.size, filename, NULL, 0);
if (!doc) { if (!doc) {
fprintf(stderr, "Failed to parse '%s'.\n", filename); fprintf(stderr, "Failed to parse '%s'.\n", url);
if (error != NULL) if (error != NULL)
{ {
*error = g_error_new(g_quark_from_string("subsurface"), *error = g_error_new(g_quark_from_string("subsurface"),
DIVE_ERROR_PARSE, DIVE_ERROR_PARSE,
"Failed to parse '%s'", "Failed to parse '%s'",
filename); url);
} }
return; return;
} }
/* we assume that the last (or only) filename passed as argument is a /* we assume that the last (or only) filename passed as argument is a
* great filename to use as default when saving the dives */ * great filename to use as default when saving the dives */
set_filename(filename); set_filename(url);
reset_all(); reset_all();
dive_start(); dive_start();
#ifdef XSLT #ifdef XSLT