mirror of
https://github.com/subsurface/subsurface.git
synced 2025-02-19 22:16:15 +00:00
Libdivecomputer: start actually importing the dive data
So this actually reports the dive data that libdivecomputer generates. It doesn't import special events etc, but neither do we for the xml importer. It is also slow as heck, since it doesn't try to do the "hey, I already have this dive" logic and always imports everything, but the basics are definitely there. Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
parent
aa416e3c96
commit
42f627b8b1
5 changed files with 50 additions and 29 deletions
3
dive.h
3
dive.h
|
@ -193,12 +193,15 @@ static inline unsigned int dive_size(int samples)
|
||||||
return sizeof(struct dive) + samples*sizeof(struct sample);
|
return sizeof(struct dive) + samples*sizeof(struct sample);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
extern time_t utc_mktime(struct tm *tm);
|
||||||
|
|
||||||
extern struct dive *alloc_dive(void);
|
extern struct dive *alloc_dive(void);
|
||||||
extern void record_dive(struct dive *dive);
|
extern void record_dive(struct dive *dive);
|
||||||
|
|
||||||
extern struct sample *prepare_sample(struct dive **divep);
|
extern struct sample *prepare_sample(struct dive **divep);
|
||||||
extern void finish_sample(struct dive *dive, struct sample *sample);
|
extern void finish_sample(struct dive *dive, struct sample *sample);
|
||||||
|
|
||||||
|
extern void report_dives(void);
|
||||||
extern struct dive *fixup_dive(struct dive *dive);
|
extern struct dive *fixup_dive(struct dive *dive);
|
||||||
extern struct dive *try_to_merge(struct dive *a, struct dive *b);
|
extern struct dive *try_to_merge(struct dive *a, struct dive *b);
|
||||||
|
|
||||||
|
|
|
@ -10,6 +10,7 @@ struct DiveList {
|
||||||
GtkTreeViewColumn *date, *depth, *duration;
|
GtkTreeViewColumn *date, *depth, *duration;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
extern struct DiveList dive_list;
|
||||||
extern struct DiveList dive_list_create(void);
|
extern struct DiveList dive_list_create(void);
|
||||||
extern void dive_list_update_dives(struct DiveList);
|
extern void dive_list_update_dives(struct DiveList);
|
||||||
extern void update_dive_list_units(struct DiveList *);
|
extern void update_dive_list_units(struct DiveList *);
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
#include <gtk/gtk.h>
|
#include <gtk/gtk.h>
|
||||||
|
|
||||||
#include "dive.h"
|
#include "dive.h"
|
||||||
|
#include "divelist.h"
|
||||||
#include "display.h"
|
#include "display.h"
|
||||||
|
|
||||||
/* libdivecomputer */
|
/* libdivecomputer */
|
||||||
|
@ -116,7 +117,7 @@ static parser_status_t create_parser(device_data_t *devdata, parser_t **parser)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int parse_gasmixes(parser_t *parser, int ngases)
|
static int parse_gasmixes(struct dive *dive, parser_t *parser, int ngases)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
|
@ -128,20 +129,17 @@ static int parse_gasmixes(parser_t *parser, int ngases)
|
||||||
if (rc != PARSER_STATUS_SUCCESS && rc != PARSER_STATUS_UNSUPPORTED)
|
if (rc != PARSER_STATUS_SUCCESS && rc != PARSER_STATUS_UNSUPPORTED)
|
||||||
return rc;
|
return rc;
|
||||||
|
|
||||||
printf("<gasmix>\n"
|
if (i >= MAX_CYLINDERS)
|
||||||
" <he>%.1f</he>\n"
|
continue;
|
||||||
" <o2>%.1f</o2>\n"
|
|
||||||
" <n2>%.1f</n2>\n"
|
dive->cylinder[i].gasmix.o2.permille = gasmix.oxygen * 1000 + 0.5;
|
||||||
"</gasmix>\n",
|
dive->cylinder[i].gasmix.he.permille = gasmix.helium * 1000 + 0.5;
|
||||||
gasmix.helium * 100.0,
|
|
||||||
gasmix.oxygen * 100.0,
|
|
||||||
gasmix.nitrogen * 100.0);
|
|
||||||
}
|
}
|
||||||
return PARSER_STATUS_SUCCESS;
|
return PARSER_STATUS_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
sample_cb (parser_sample_type_t type, parser_sample_value_t value, void *userdata)
|
sample_cb(parser_sample_type_t type, parser_sample_value_t value, void *userdata)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
static const char *events[] = {
|
static const char *events[] = {
|
||||||
|
@ -150,20 +148,31 @@ sample_cb (parser_sample_type_t type, parser_sample_value_t value, void *userdat
|
||||||
"safety stop (voluntary)", "safety stop (mandatory)", "deepstop",
|
"safety stop (voluntary)", "safety stop (mandatory)", "deepstop",
|
||||||
"ceiling (safety stop)", "unknown", "divetime", "maxdepth",
|
"ceiling (safety stop)", "unknown", "divetime", "maxdepth",
|
||||||
"OLF", "PO2", "airtime", "rgbm", "heading", "tissue level warning"};
|
"OLF", "PO2", "airtime", "rgbm", "heading", "tissue level warning"};
|
||||||
|
struct dive **divep = userdata;
|
||||||
|
struct dive *dive = *divep;
|
||||||
|
struct sample *sample;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* We fill in the "previous" sample - except for SAMPLE_TYPE_TIME,
|
||||||
|
* which creates a new one.
|
||||||
|
*/
|
||||||
|
sample = dive->samples ? dive->sample+dive->samples-1 : NULL;
|
||||||
|
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case SAMPLE_TYPE_TIME:
|
case SAMPLE_TYPE_TIME:
|
||||||
printf("<sample>\n");
|
sample = prepare_sample(divep);
|
||||||
printf(" <time>%02u:%02u</time>\n", value.time / 60, value.time % 60);
|
sample->time.seconds = value.time;
|
||||||
|
finish_sample(*divep, sample);
|
||||||
break;
|
break;
|
||||||
case SAMPLE_TYPE_DEPTH:
|
case SAMPLE_TYPE_DEPTH:
|
||||||
printf(" <depth>%.2f</depth>\n", value.depth);
|
sample->depth.mm = value.depth * 1000 + 0.5;
|
||||||
break;
|
break;
|
||||||
case SAMPLE_TYPE_PRESSURE:
|
case SAMPLE_TYPE_PRESSURE:
|
||||||
printf(" <pressure tank=\"%u\">%.2f</pressure>\n", value.pressure.tank, value.pressure.value);
|
sample->cylinderindex = value.pressure.tank;
|
||||||
|
sample->cylinderpressure.mbar = value.pressure.value * 1000 + 0.5;
|
||||||
break;
|
break;
|
||||||
case SAMPLE_TYPE_TEMPERATURE:
|
case SAMPLE_TYPE_TEMPERATURE:
|
||||||
printf(" <temperature>%.2f</temperature>\n", value.temperature);
|
sample->temperature.mkelvin = (value.temperature + 273.15) * 1000 + 0.5;
|
||||||
break;
|
break;
|
||||||
case SAMPLE_TYPE_EVENT:
|
case SAMPLE_TYPE_EVENT:
|
||||||
printf(" <event type=\"%u\" time=\"%u\" flags=\"%u\" value=\"%u\">%s</event>\n",
|
printf(" <event type=\"%u\" time=\"%u\" flags=\"%u\" value=\"%u\">%s</event>\n",
|
||||||
|
@ -190,11 +199,11 @@ sample_cb (parser_sample_type_t type, parser_sample_value_t value, void *userdat
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static int parse_samples(parser_t *parser)
|
static int parse_samples(struct dive **divep, parser_t *parser)
|
||||||
{
|
{
|
||||||
// Parse the sample data.
|
// Parse the sample data.
|
||||||
printf("Parsing the sample data.\n");
|
printf("Parsing the sample data.\n");
|
||||||
return parser_samples_foreach(parser, sample_cb, NULL);
|
return parser_samples_foreach(parser, sample_cb, divep);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int dive_cb(const unsigned char *data, unsigned int size,
|
static int dive_cb(const unsigned char *data, unsigned int size,
|
||||||
|
@ -205,6 +214,8 @@ static int dive_cb(const unsigned char *data, unsigned int size,
|
||||||
parser_t *parser = NULL;
|
parser_t *parser = NULL;
|
||||||
device_data_t *devdata = userdata;
|
device_data_t *devdata = userdata;
|
||||||
dc_datetime_t dt = {0};
|
dc_datetime_t dt = {0};
|
||||||
|
struct tm tm;
|
||||||
|
struct dive *dive;
|
||||||
|
|
||||||
/* Christ, this is hacky */
|
/* Christ, this is hacky */
|
||||||
run_gtk_mainloop();
|
run_gtk_mainloop();
|
||||||
|
@ -222,6 +233,7 @@ static int dive_cb(const unsigned char *data, unsigned int size,
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
dive = alloc_dive();
|
||||||
rc = parser_get_datetime(parser, &dt);
|
rc = parser_get_datetime(parser, &dt);
|
||||||
if (rc != PARSER_STATUS_SUCCESS && rc != PARSER_STATUS_UNSUPPORTED) {
|
if (rc != PARSER_STATUS_SUCCESS && rc != PARSER_STATUS_UNSUPPORTED) {
|
||||||
error("Error parsing the datetime.");
|
error("Error parsing the datetime.");
|
||||||
|
@ -229,9 +241,13 @@ static int dive_cb(const unsigned char *data, unsigned int size,
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
printf("<datetime>%04i-%02i-%02i %02i:%02i:%02i</datetime>\n",
|
tm.tm_year = dt.year;
|
||||||
dt.year, dt.month, dt.day,
|
tm.tm_mon = dt.month-1;
|
||||||
dt.hour, dt.minute, dt.second);
|
tm.tm_mday = dt.day;
|
||||||
|
tm.tm_hour = dt.hour;
|
||||||
|
tm.tm_min = dt.minute;
|
||||||
|
tm.tm_sec = dt.second;
|
||||||
|
dive->when = utc_mktime(&tm);
|
||||||
|
|
||||||
// Parse the divetime.
|
// Parse the divetime.
|
||||||
printf("Parsing the divetime.\n");
|
printf("Parsing the divetime.\n");
|
||||||
|
@ -242,9 +258,7 @@ static int dive_cb(const unsigned char *data, unsigned int size,
|
||||||
parser_destroy(parser);
|
parser_destroy(parser);
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
dive->duration.seconds = divetime;
|
||||||
printf("<divetime>%02u:%02u</divetime>\n",
|
|
||||||
divetime / 60, divetime % 60);
|
|
||||||
|
|
||||||
// Parse the maxdepth.
|
// Parse the maxdepth.
|
||||||
printf("Parsing the maxdepth.\n");
|
printf("Parsing the maxdepth.\n");
|
||||||
|
@ -255,8 +269,7 @@ static int dive_cb(const unsigned char *data, unsigned int size,
|
||||||
parser_destroy(parser);
|
parser_destroy(parser);
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
dive->maxdepth.mm = maxdepth * 1000 + 0.5;
|
||||||
printf("<maxdepth>%.2f</maxdepth>\n", maxdepth);
|
|
||||||
|
|
||||||
// Parse the gas mixes.
|
// Parse the gas mixes.
|
||||||
printf("Parsing the gas mixes.\n");
|
printf("Parsing the gas mixes.\n");
|
||||||
|
@ -268,7 +281,7 @@ static int dive_cb(const unsigned char *data, unsigned int size,
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
|
||||||
rc = parse_gasmixes(parser, ngases);
|
rc = parse_gasmixes(dive, parser, ngases);
|
||||||
if (rc != PARSER_STATUS_SUCCESS) {
|
if (rc != PARSER_STATUS_SUCCESS) {
|
||||||
error("Error parsing the gas mix.");
|
error("Error parsing the gas mix.");
|
||||||
parser_destroy(parser);
|
parser_destroy(parser);
|
||||||
|
@ -276,12 +289,13 @@ static int dive_cb(const unsigned char *data, unsigned int size,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Initialize the sample data.
|
// Initialize the sample data.
|
||||||
rc = parse_samples(parser);
|
rc = parse_samples(&dive, parser);
|
||||||
if (rc != PARSER_STATUS_SUCCESS) {
|
if (rc != PARSER_STATUS_SUCCESS) {
|
||||||
error("Error parsing the samples.");
|
error("Error parsing the samples.");
|
||||||
parser_destroy(parser);
|
parser_destroy(parser);
|
||||||
return rc;
|
return rc;
|
||||||
}
|
}
|
||||||
|
record_dive(dive);
|
||||||
|
|
||||||
parser_destroy(parser);
|
parser_destroy(parser);
|
||||||
return 1;
|
return 1;
|
||||||
|
@ -565,4 +579,7 @@ void import_dialog(GtkWidget *w, gpointer data)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
gtk_widget_destroy(dialog);
|
gtk_widget_destroy(dialog);
|
||||||
|
|
||||||
|
report_dives();
|
||||||
|
dive_list_update_dives(dive_list);
|
||||||
}
|
}
|
||||||
|
|
2
main.c
2
main.c
|
@ -37,7 +37,7 @@ static int sortfn(const void *_a, const void *_b)
|
||||||
* This doesn't really report anything at all. We just sort the
|
* This doesn't really report anything at all. We just sort the
|
||||||
* dives, the GUI does the reporting
|
* dives, the GUI does the reporting
|
||||||
*/
|
*/
|
||||||
static void report_dives(void)
|
void report_dives(void)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
|
|
|
@ -104,7 +104,7 @@ static enum import_source {
|
||||||
UDDF,
|
UDDF,
|
||||||
} import_source;
|
} import_source;
|
||||||
|
|
||||||
static time_t utc_mktime(struct tm *tm)
|
time_t utc_mktime(struct tm *tm)
|
||||||
{
|
{
|
||||||
static const int mdays[] = {
|
static const int mdays[] = {
|
||||||
0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334
|
0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334
|
||||||
|
|
Loading…
Add table
Reference in a new issue