diff options
Diffstat (limited to 'smtk-import/smartrak.c')
-rw-r--r-- | smtk-import/smartrak.c | 807 |
1 files changed, 807 insertions, 0 deletions
diff --git a/smtk-import/smartrak.c b/smtk-import/smartrak.c new file mode 100644 index 000000000..9d3ac501e --- /dev/null +++ b/smtk-import/smartrak.c @@ -0,0 +1,807 @@ +/* + * An .slg file is composed of a main table (Dives), a bunch of tables directly + * linked to Dives by their indexes (Site, Suit, Tank, etc) and another group of + * independent tables (Activity, Type, Gear, Fish ...) which connect with the dives + * via a related group of tables (ActivityRelation, TypeRelation ...) that contain + * just the dive index number and the related table index number. + * The data stored in the main group of tables are very extensive, going far beyond + * the actual scope of Subsurface in most of cases; e.g. Dives table keeps + * information which can be directly uploaded to DAN's database of dives, or Buddy + * table can include telephones, photo or, even, buddy mother's maiden name. + * + * Although these funcs are suposed to work in a standalone tool, will also work + * on main Subsurface import menu, by simply tweaking file.c and main_window.cpp + * to call smartrak_import() + */ + +#ifndef __USE_XOPEN +#define __USE_XOPEN +#endif + +#ifndef _GNU_SOURCE +#define _GNU_SOURCE +#endif + +#include <stdlib.h> +#include <stdio.h> +#include <string.h> +#include <mdbtools.h> +#include <stdarg.h> + +#include "dive.h" +#include "gettext.h" +#include "divelist.h" +#include "libdivecomputer.h" +#include "divesite.h" +#include "membuffer.h" + +/* SmartTrak version, constant for every single file */ +int smtk_version; +int tanks; + +/* + * There are AFAIK two versions of Smarttrak. The newer one supports trimix and up + * to 10 tanks. The other one just 3 tanks and no trimix but only nitrox. This is a + * problem for an automated parser which has to support both formats. + * In this solution I made an enum of fields with the same order they would have in + * a smarttrak db, and a tiny function which returns the number of the column where + * a field is expected to be, taking into account the different db formats . + */ +enum field_pos {IDX = 0, DIVENUM, DATE, INTIME, INTVAL, DURATION, OUTTIME, DESATBEFORE, DESATAFTER, NOFLYBEFORE, + NOFLYAFTER, NOSTOPDECO, MAXDEPTH, VISIBILITY, WEIGHT, O2FRAC, HEFRAC, PSTART, PEND, AIRTEMP, + MINWATERTEMP, MAXWATERTEMP, SECFACT, ALARMS, MODE, REMARKS, DCNUMBER, DCMODEL, DIVETIMECOUNT, LOG, + PROFILE, SITEIDX, ALTIDX, SUITIDX, WEATHERIDX, SURFACEIDX, UNDERWATERIDX, TANKIDX}; + +/* + * Returns calculated column number depending on smartrak version, as there are more + * tanks (10) in later versions than in older (3). + * Older versions also lacks of 3 columns storing he fraction, one for each tank. + */ +static int coln(enum field_pos pos) +{ + int amnd = (smtk_version < 10213) ? 3 : 0; + + if (pos <= O2FRAC) + return pos; + if (pos >= AIRTEMP) { + pos += 4 * (tanks - 1) - amnd; + return pos; + } + switch (pos) { + case HEFRAC: + pos = O2FRAC + tanks; + return pos; + case PSTART: + pos = O2FRAC + 2 * tanks - amnd; + return pos; + case PEND: + pos = O2FRAC + 2 * tanks + 1 - amnd; + return pos; + } +} + +/* + * Fills the date part of a tm structure with the string data obtained + * from smartrak in format "DD/MM/YY HH:MM:SS" where time is irrelevant. + * TODO: Verify localization. + */ +static void smtk_date_to_tm(char *d_buffer, struct tm *tm_date) +{ + char *temp = NULL; + + temp = copy_string(d_buffer); + strtok(temp, " "); + if (temp) + strptime(temp, "%x", tm_date); +} + +/* + * Fills the time part of a tm structure with the string data obtained + * from smartrak in format "DD/MM/YY HH:MM:SS" where date is irrelevant. + * TODO: Verify localization. + */ +static void smtk_time_to_tm(char *t_buffer, struct tm *tm_date) +{ + char *temp = NULL; + + temp = rindex(copy_string(t_buffer), ' '); + if (temp) + strptime(temp, "%X", tm_date); +} + +/* + * Converts to seconds a time lapse returned from smartrak in string format + * "DD/MM/YY HH:MM:SS" where date is irrelevant. + * TODO: modify to support times > 24h where date means difference in days + * from 29/12/99. + */ +static unsigned int smtk_time_to_secs(char *t_buffer) +{ + char *temp; + unsigned int hr, min, sec; + + if (!same_string(t_buffer, "")) { + temp = rindex(copy_string(t_buffer), ' '); + hr = atoi(strtok(temp, ":")); + min = atoi(strtok(NULL, ":")); + sec = atoi(strtok(NULL, "\0")); + return((((hr*60)+min)*60)+sec); + } else { + return 0; + } +} + +/* + * Returns an opened table given its name and mdb. outcol and outbounder have to be allocated + * by the caller and are filled here. + */ +static MdbTableDef *smtk_open_table(MdbHandle *mdb, char *tablename, MdbColumn **outcol, char **outbounder) +{ + MdbCatalogEntry *entry; + MdbTableDef *table; + int i; + + entry = mdb_get_catalogentry_by_name(mdb, tablename); + if (!entry) + return NULL; + table = mdb_read_table(entry); + if (!table) + return NULL; + mdb_read_columns(table); + for (i = 0; i < table->num_cols; i++) { + outcol[i] = g_ptr_array_index(table->columns, i); + outbounder[i] = (char *) g_malloc(MDB_BIND_SIZE); + mdb_bind_column(table, i+1, outbounder[i], NULL); + } + mdb_rewind_table(table); + + return table; +} + +/* + * Utility function which returns the value from a given column in a given table, + * whose row equals the given idx string. + * Idx should be a numeric value, but all values obtained from mdbtools are strings, + * so let's compare strings instead of numbers to avoid unnecessary transforms. + */ +static char *smtk_value_by_idx(MdbHandle *mdb, char *tablename, int colnum, char *idx) +{ + MdbCatalogEntry *entry; + MdbTableDef *table; + MdbColumn *idxcol, *valuecol; + char *bounder[MDB_MAX_COLS], *str = NULL; + int i = 0; + + entry = mdb_get_catalogentry_by_name(mdb, tablename); + table = mdb_read_table(entry); + if (!table) { + report_error("[Error][smartrak_import]\t%s table doesn't exist\n", tablename); + return str; + } + mdb_read_columns(table); + idxcol = g_ptr_array_index(table->columns, 0); + valuecol = g_ptr_array_index(table->columns, colnum); + for (i = 0; i < table->num_cols; i++) { + bounder[i] = (char *) g_malloc(MDB_BIND_SIZE); + mdb_bind_column(table, i+1, bounder[i], NULL); + } + mdb_rewind_table(table); + for (i = 0; i < table->num_rows; i++) { + mdb_fetch_row(table); + if (!strcmp(idxcol->bind_ptr, idx)) { + str = copy_string(valuecol->bind_ptr); + break; + } + } + for (i = 0; i < table->num_cols; i++) + free(bounder[i]); + mdb_free_tabledef(table); + return str; +} + +/* + * Utility function which joins three strings, being the second a separator string, + * usually a "\n". The third is a format string with an argument list. + * If the original string is NULL, then just returns the third. + * This is based in add_to_string() and add_to_string_va(), and, as its parents + * frees the original string. + */ +static char *smtk_concat_str(const char *orig, const char *sep, const char *fmt, ...) +{ + char *str; + va_list args; + struct membuffer out = { 0 }, in = { 0 }; + + va_start(args, fmt); + put_vformat(&in, fmt, args); + if (orig != NULL) { + put_format(&out, "%s%s%s", orig, sep, mb_cstring(&in)); + str = copy_string(mb_cstring(&out)); + } else { + str = copy_string(mb_cstring(&in)); + } + va_end(args); + + free_buffer(&out); + free_buffer(&in); + free((void *)orig); + + return str; +} + +/* + * A site may be a wreck, which has its own table. + * Parse this table referred by the site idx. If found, put the different info items in + * Subsurface's dive_site notes. + * Wreck format: + * | Idx | SiteIdx | Text | Built | Sank | SankTime | Reason | ... | Notes | TrakId | + */ +static void smtk_wreck_site(MdbHandle *mdb, char *site_idx, struct dive_site *ds) +{ + MdbTableDef *table; + MdbColumn *col[MDB_MAX_COLS]; + char *bound_values[MDB_MAX_COLS]; + char *tmp = NULL, *notes = NULL; + int rc, i; + uint32_t d; + const char *wreck_fields[] = {QT_TRANSLATE_NOOP("gettextFromC", "Built"), QT_TRANSLATE_NOOP("gettextFromC", "Sank"), QT_TRANSLATE_NOOP("gettextFromC", "SankTime"), + QT_TRANSLATE_NOOP("gettextFromC", "Reason"), QT_TRANSLATE_NOOP("gettextFromC", "Nationality"), QT_TRANSLATE_NOOP("gettextFromC", "Shipyard"), + QT_TRANSLATE_NOOP("gettextFromC", "ShipType"), QT_TRANSLATE_NOOP("gettextFromC", "Length"), QT_TRANSLATE_NOOP("gettextFromC", "Beam"), + QT_TRANSLATE_NOOP("gettextFromC", "Draught"), QT_TRANSLATE_NOOP("gettextFromC", "Displacement"), QT_TRANSLATE_NOOP("gettextFromC", "Cargo"), + QT_TRANSLATE_NOOP("gettextFromC", "Notes")}; + + table = smtk_open_table(mdb, "Wreck", col, bound_values); + + /* Sanity check for table, unlikely but ... */ + if (!table) + return; + + /* Begin parsing. Write strings to notes only if available.*/ + while (mdb_fetch_row(table)) { + if (!strcmp(col[1]->bind_ptr, site_idx)) { + tmp = copy_string(col[1]->bind_ptr); + notes = smtk_concat_str(notes, "\n", translate("gettextFromC", "Wreck Data")); + for (i = 3; i < 16; i++) { + switch (i) { + case 3: + case 4: + if (memcmp(col[i]->bind_ptr, "\0", 1)) + notes = smtk_concat_str(notes, "\n", "%s: %s", wreck_fields[i - 3], strtok(copy_string(col[i]->bind_ptr), " ")); + break; + case 5: + if (strcmp(rindex(copy_string(col[i]->bind_ptr), ' '), "\0")) + notes = smtk_concat_str(notes, "\n", "%s: %s", wreck_fields[i - 3], rindex(col[i]->bind_ptr, ' ')); + break; + case 6 ... 9: + case 14: + case 15: + if (memcmp(col[i]->bind_ptr, "\0", 1)) + notes = smtk_concat_str(notes, "\n", "%s: %s", wreck_fields[i - 3], col[i]->bind_ptr); + break; + default: + d = strtold(col[i]->bind_ptr, NULL); + if (d) + notes = smtk_concat_str(notes, "\n", "%s: %d", wreck_fields[i - 3], d); + break; + } + } + ds->notes = smtk_concat_str(ds->notes, "\n", "%s", notes); + break; + } + } + /* Clean up and exit */ + for (i = 0; i < table->num_cols; i++) + free(bound_values[i]); + mdb_free_tabledef(table); + free(notes); + free(tmp); +} + +/* + * Smartrak locations db is quite extensive. This builds a string joining some of + * the data in the style: "Country, State, Locality, Site" if this data are + * available. Uses two tables, Site and Location. Puts Altitude, Depth and Notes + * in Subsurface's dive site structure's notes if they are available. + * Site format: + * | Idx | Text | LocationIdx | WaterIdx | PlatformIdx | BottomIdx | Latitude | Longitude | Altitude | Depth | Notes | TrakId | + * Location format: + * | Idx | Text | Province | Country | Depth | + */ +static void smtk_build_location(MdbHandle *mdb, char *idx, timestamp_t when, uint32_t *location) +{ + MdbTableDef *table; + MdbColumn *col[MDB_MAX_COLS]; + char *bound_values[MDB_MAX_COLS]; + int i; + uint32_t d; + struct dive_site *ds; + degrees_t lat, lon; + char *str = NULL, *loc_idx = NULL, *site = NULL, *notes = NULL; + const char *site_fields[] = {QT_TRANSLATE_NOOP("gettextFromC", "Altitude"), QT_TRANSLATE_NOOP("gettextFromC", "Depth"), + QT_TRANSLATE_NOOP("gettextFromC", "Notes")}; + + /* Read data from Site table. Format notes for the dive site if any.*/ + table = smtk_open_table(mdb, "Site", col, bound_values); + if (!table) + return; + + for (i = 1; i <= atoi(idx); i++) + mdb_fetch_row(table); + loc_idx = copy_string(col[2]->bind_ptr); + site = copy_string(col[1]->bind_ptr); + lat.udeg = lrint(strtod(copy_string(col[6]->bind_ptr), NULL) * 1000000); + lon.udeg = lrint(strtod(copy_string(col[7]->bind_ptr), NULL) * 1000000); + + for (i = 8; i < 11; i++) { + switch (i) { + case 8: + case 9: + d = strtold(col[i]->bind_ptr, NULL); + if (d) + notes = smtk_concat_str(notes, "\n", "%s: %d m", site_fields[i - 8], d); + break; + case 10: + if (memcmp(col[i]->bind_ptr, "\0", 1)) + notes = smtk_concat_str(notes, "\n", "%s: %s", site_fields[i - 8], col[i]->bind_ptr); + break; + } + } + for (i = 0; i < table->num_cols; i++) { + bound_values[i] = NULL; + col[i] = NULL; + } + + /* Read data from Location table, linked to Site by loc_idx */ + table = smtk_open_table(mdb, "Location", col, bound_values); + mdb_rewind_table(table); + for (i = 1; i <= atoi(loc_idx); i++) + mdb_fetch_row(table); + /* + * Create a string for Subsurface's dive site structure with coordinates + * if available, if the site's name doesn't previously exists. + */ + if (memcmp(col[3]->bind_ptr, "\0", 1)) + str = smtk_concat_str(str, ", ", "%s", col[3]->bind_ptr); // Country + if (memcmp(col[2]->bind_ptr, "\0", 1)) + str = smtk_concat_str(str, ", ", "%s", col[2]->bind_ptr); // State - Province + if (memcmp(col[1]->bind_ptr, "\0", 1)) + str = smtk_concat_str(str, ", ", "%s", col[1]->bind_ptr); // Locality + str = smtk_concat_str(str, ", ", "%s", site); + + *location = get_dive_site_uuid_by_name(str, NULL); + if (*location == 0) { + if (lat.udeg == 0 && lon.udeg == 0) + *location = create_dive_site(str, when); + else + *location = create_dive_site_with_gps(str, lat, lon, when); + } + for (i = 0; i < table->num_cols; i++) { + bound_values[i] = NULL; + col[i] = NULL; + } + + /* Insert site notes */ + ds = get_dive_site_by_uuid(*location); + ds->notes = copy_string(notes); + free(notes); + + /* Check if we have a wreck */ + smtk_wreck_site(mdb, idx, ds); + + /* Clean up and exit */ + for (i = 0; i < table->num_cols; i++) + free(bound_values[i]); + mdb_free_tabledef(table); + free(loc_idx); + free(site); + free(str); +} + +static void smtk_build_tank_info(MdbHandle *mdb, struct dive *dive, int tanknum, char *idx) +{ + MdbTableDef *table; + MdbColumn *col[MDB_MAX_COLS]; + char *bound_values[MDB_MAX_COLS]; + int i; + + table = smtk_open_table(mdb, "Tank", col, bound_values); + if (!table) + return; + + for (i = 1; i <= atoi(idx); i++) + mdb_fetch_row(table); + dive->cylinder[tanknum].type.description = copy_string(col[1]->bind_ptr); + dive->cylinder[tanknum].type.size.mliter = strtod(col[2]->bind_ptr, NULL) * 1000; + dive->cylinder[tanknum].type.workingpressure.mbar = strtod(col[4]->bind_ptr, NULL) * 1000; + + for (i = 0; i < table->num_cols; i++) + free(bound_values[i]); + mdb_free_tabledef(table); +} + +/* + * Parses a relation table and fills a list with the relations for a dive idx. + * Returns the number of relations found for a given dive idx. + * Table relation format: + * | Diveidx | Idx | + */ +static int smtk_index_list(MdbHandle *mdb, char *table_name, char *dive_idx, int idx_list[]) +{ + int n = 0, i = 0; + MdbTableDef *table; + MdbColumn *cols[MDB_MAX_COLS]; + char *bounders[MDB_MAX_COLS]; + + table = smtk_open_table(mdb, table_name, cols, bounders); + + /* Sanity check */ + if (!table) + return 0; + + /* Parse the table searching for dive_idx */ + while (mdb_fetch_row(table)) { + if (!strcmp(dive_idx, cols[0]->bind_ptr)) { + idx_list[n] = atoi(cols[1]->bind_ptr); + n++; + } + } + + /* Clean up and exit */ + for (i = 0; i < table->num_cols; i++) + free(bounders[i]); + mdb_free_tabledef(table); + return n; +} + +/* + * Returns string with buddies names as registered in smartrak (may be a nickname). + * "Buddy" table is a buddies relation with lots and lots and lots of data (even buddy mother's + * maiden name ;-) ) most of them useless for a dive log. Let's just consider the nickname as main + * field and the full name if this exists and its construction is different from the nickname. + * Buddy table format: + * | Idx | Text (nickname) | Name | Firstname | Middlename | Title | Picture | Phone | ... + */ +static char *smtk_locate_buddy(MdbHandle *mdb, char *dive_idx) +{ + char *str = NULL, *fullname = NULL, *bounder[MDB_MAX_COLS] = { NULL }, *buddies[256] = { NULL }; + MdbTableDef *table; + MdbColumn *col[MDB_MAX_COLS]; + int i, n, rel[256] = { 0 }; + + n = smtk_index_list(mdb, "BuddyRelation", dive_idx, rel); + if (!n) + return str; + table = smtk_open_table(mdb, "Buddy", col, bounder); + if (!table) + return str; + /* + * Buddies in a single dive aren't (usually) a big number, so probably + * it's not a good idea to use a complex data structure and algorithm. + */ + while (mdb_fetch_row(table)) { + if (!same_string(col[3]->bind_ptr, "")) + fullname = smtk_concat_str(fullname, " ", "%s", col[3]->bind_ptr); + if (!same_string(col[4]->bind_ptr, "")) + fullname = smtk_concat_str(fullname, " ", "%s", col[4]->bind_ptr); + if (!same_string(col[2]->bind_ptr, "")) + fullname = smtk_concat_str(fullname, " ", "%s", col[2]->bind_ptr); + if (fullname && !same_string(col[1]->bind_ptr, fullname)) + buddies[atoi(col[0]->bind_ptr)] = smtk_concat_str(buddies[atoi(col[0]->bind_ptr)], "", "%s (%s)", col[1]->bind_ptr, fullname); + else + buddies[atoi(col[0]->bind_ptr)] = smtk_concat_str(buddies[atoi(col[0]->bind_ptr)], "", "%s", col[1]->bind_ptr); + fullname = NULL; + } + free(fullname); + for (i = 0; i < n; i++) + str = smtk_concat_str(str, ", ", "%s", buddies[rel[i]]); + + /* Clean up and exit */ + for (i = 0; i < table->num_rows; i++) + free(buddies[i]); + for (i = 0; i < table->num_cols; i++) + free(bounder[i]); + mdb_free_tabledef(table); + return str; +} + +/* Parses the dive_type mdb tables and import the data into dive's + * taglist structure or notes. If there are tags that affects dive's dive_mode + * (SCR, CCR or so), set the dive mode too. + * The "tag" parameter is used to mark if we want this table to be imported + * into tags or into notes. + * Managed tables formats: Just consider Idx and Text + * Type: + * | Idx | Text | Default (bool) + * Activity: + * | Idx | Text | Default (bool) + * Gear: + * | Idx | Text | Vendor | Type | Typenum | Notes | Default (bool) | TrakId + * Fish: + * | Idx | Text | Name | Latin name | Typelength | Maxlength | Picture | Default (bool)| TrakId + */ +static void smtk_parse_relations(MdbHandle *mdb, struct dive *dive, char *dive_idx, char *table_name, char *rel_table_name, bool tag) +{ + MdbTableDef *table; + MdbColumn *col[MDB_MAX_COLS]; + char *bound_values[MDB_MAX_COLS], *tmp = NULL, *types[64] = { NULL }; + int i = 0, n = 0, rels[256] = { 0 }; + + n = smtk_index_list(mdb, rel_table_name, dive_idx, rels); + if (!n) + return; + table = smtk_open_table(mdb, table_name, col, bound_values); + if (!table) + return; + while (mdb_fetch_row(table)) + types[atoi(col[0]->bind_ptr)] = copy_string(col[1]->bind_ptr); + + for (i = 0; i < n; i++) { + if (tag) + taglist_add_tag(&dive->tag_list, copy_string(types[rels[i]])); + else + tmp = smtk_concat_str(tmp, ", ", "%s", types[rels[i]]); + if (strstr(types[rels[i]], "SCR")) + dive->dc.divemode = PSCR; + else if (strstr(types[rels[i]], "CCR")) + dive->dc.divemode = CCR; + } + if (tmp) + dive->notes = smtk_concat_str(dive->notes, "\n", "Smartrak %s: %s", table_name, tmp); + free(tmp); + + /* clean up and exit */ + for (i = 1; i < 64; i++) + free(types[i]); + for (i = 0; i < table->num_cols; i++) + free(bound_values[i]); + mdb_free_tabledef(table); +} + +/* + * Returns a dc_descriptor_t structure based on dc model's number. + * This ensures the model pased to libdc_buffer_parser() is a supported model and avoids + * problems with shared model num devices by taking the family into account. The family + * is estimated elsewhere based in dive header length. + */ +static dc_descriptor_t *get_data_descriptor(int data_model, dc_family_t data_fam) +{ + dc_descriptor_t *descriptor = NULL, *current = NULL; + dc_iterator_t *iterator = NULL; + dc_status_t rc; + + rc = dc_descriptor_iterator(&iterator); + if (rc != DC_STATUS_SUCCESS) { + report_error("[Error][libdc]\t\t\tCreating the device descriptor iterator.\n"); + return current; + } + while ((dc_iterator_next(iterator, &descriptor)) == DC_STATUS_SUCCESS) { + int desc_model = dc_descriptor_get_model(descriptor); + dc_family_t desc_fam = dc_descriptor_get_type(descriptor); + + if (data_fam == DC_FAMILY_UWATEC_ALADIN) { + if (data_model == desc_model && data_fam == desc_fam) { + current = descriptor; + break; + } + } else if (data_model == desc_model && (desc_fam == DC_FAMILY_UWATEC_SMART || + desc_fam == DC_FAMILY_UWATEC_MERIDIAN)) { + current = descriptor; + break; + } + dc_descriptor_free(descriptor); + } + dc_iterator_free(iterator); + return current; +} + +/* + * Fills a device_data_t structure with known dc data. + * Completes a dc_descriptor_t structure with libdc's g_descriptors[] table so + * we ensure that data passed for parsing to libdc have a supported and known + * DC. dc_family_t is certainly known *only* if it is Aladin/Memomouse family + * otherwise it will be known after get_data_descriptor call. + */ +static int prepare_data(int data_model, dc_family_t dc_fam, device_data_t *dev_data) +{ + dev_data->device = NULL; + dev_data->context = NULL; + dev_data->descriptor = get_data_descriptor(data_model, dc_fam); + if (dev_data->descriptor) { + dev_data->vendor = copy_string(dev_data->descriptor->vendor); + dev_data->product = copy_string(dev_data->descriptor->product); + dev_data->model = smtk_concat_str(dev_data->model, "", "%s %s", dev_data->vendor, dev_data->product); + return DC_STATUS_SUCCESS; + } else { + return DC_STATUS_UNSUPPORTED; + } +} + +/* + * Returns a buffer prepared for libdc parsing. + * Aladin and memomouse dives were imported from datatrak, so they lack of a + * header. Creates a fake header for them and inserts the dc model where libdc + * expects it to be. + */ +static dc_status_t libdc_buffer_complete(device_data_t *dev_data, unsigned char *hdr_buffer, int hdr_length, + unsigned char *prf_buffer, int prf_length, unsigned char *compl_buf) +{ + switch (dev_data->descriptor->type) { + case DC_FAMILY_UWATEC_ALADIN: + case DC_FAMILY_UWATEC_MEMOMOUSE: + compl_buf[3] = (unsigned char) dev_data->descriptor->model; + memcpy(compl_buf+hdr_length, prf_buffer, prf_length); + break; + case DC_FAMILY_UWATEC_SMART: + case DC_FAMILY_UWATEC_MERIDIAN: + memcpy(compl_buf, hdr_buffer, hdr_length); + memcpy(compl_buf+hdr_length, prf_buffer, prf_length); + break; + default: + return -2; + } + return 0; +} + +/* + * Main function. + * Two looping levels using a database for the first level ("Dives" table) + * and a clone of the first DB for the second level (each dive items). Using + * a DB clone is necessary as calling mdb_fetch_row() over different tables in + * a single DB breaks binded row data, and so would break the top loop. + */ + +void smartrak_import(const char *file, struct dive_table *divetable) +{ + MdbHandle *mdb, *mdb_clon; + MdbTableDef *mdb_table; + MdbColumn *col[MDB_MAX_COLS]; + + char *bound_values[MDB_MAX_COLS]; + int i, dc_model; + + mdb = mdb_open(file, MDB_NOFLAGS); + if (!mdb) { + report_error("[Error][smartrak_import]\tFile %s does not seem to be an Access database.", file); + return; + } + if (!mdb_read_catalog(mdb, MDB_TABLE)) { + report_error("[Error][smartrak_import]\tFile %s does not seem to be an Access database.", file); + return; + } + mdb_clon = mdb_clone_handle(mdb); + mdb_read_catalog(mdb_clon, MDB_TABLE); + + /* Check Smarttrak version (different number of supported tanks, mixes and so) */ + smtk_version = atoi(smtk_value_by_idx(mdb_clon, "SmartTrak", 1, "1")); + tanks = (smtk_version < 10213) ? 3 : 10; + + mdb_table = smtk_open_table(mdb, "Dives", col, bound_values); + if (!mdb_table) { + report_error("[Error][smartrak_import]\tFile %s does not seem to be an SmartTrak file.", file); + return; + } + while (mdb_fetch_row(mdb_table)) { + device_data_t *devdata = calloc(1, sizeof(device_data_t)); + dc_family_t dc_fam = DC_FAMILY_NULL; + unsigned char *prf_buffer, *hdr_buffer, *compl_buffer; + struct dive *smtkdive = alloc_dive(); + struct tm *tm_date = malloc(sizeof(struct tm)); + size_t hdr_length, prf_length; + dc_status_t rc = 0; + + smtkdive->number = strtod(col[1]->bind_ptr, NULL); + /* + * If there is a DC model (no zero) try to create a buffer for the + * dive and parse it with libdivecomputer + */ + dc_model = (int) strtod(col[coln(DCMODEL)]->bind_ptr, NULL) & 0xFF; + if (dc_model) { + hdr_buffer = mdb_ole_read_full(mdb, col[coln(LOG)], &hdr_length); + if (hdr_length > 0 && hdr_length < 20) // We have a profile but it's imported from datatrak + dc_fam = DC_FAMILY_UWATEC_ALADIN; + rc = prepare_data(dc_model, dc_fam, devdata); + } else { + rc = DC_STATUS_NODEVICE; + } + smtkdive->dc.model = devdata->model; + smtkdive->dc.serial = copy_string(col[coln(DCNUMBER)]->bind_ptr); + if (rc == DC_STATUS_SUCCESS) { + prf_buffer = mdb_ole_read_full(mdb, col[coln(PROFILE)], &prf_length); + if (prf_length > 0) { + if (devdata->descriptor->type == DC_FAMILY_UWATEC_ALADIN || devdata->descriptor->type == DC_FAMILY_UWATEC_MEMOMOUSE) + hdr_length = 18; + compl_buffer = calloc(hdr_length+prf_length, sizeof(char)); + rc = libdc_buffer_complete(devdata, hdr_buffer, hdr_length, prf_buffer, prf_length, compl_buffer); + if (rc != DC_STATUS_SUCCESS) { + report_error("[Error][smartrak_import]\t- %s - for dive %d", errmsg(rc), smtkdive->number); + } else { + rc = libdc_buffer_parser(smtkdive, devdata, compl_buffer, hdr_length + prf_length); + if (rc != DC_STATUS_SUCCESS) + report_error("[Error][libdc]\t\t- %s - for dive %d", errmsg(rc), smtkdive->number); + } + free(compl_buffer); + } else { + /* Dives without profile samples (usual in older aladin series) */ + report_error("[Warning][smartrak_import]\t No profile for dive %d", smtkdive->number); + smtkdive->dc.duration.seconds = smtkdive->duration.seconds = smtk_time_to_secs(col[coln(DURATION)]->bind_ptr); + smtkdive->dc.maxdepth.mm = smtkdive->maxdepth.mm = strtod(col[coln(MAXDEPTH)]->bind_ptr, NULL) * 1000; + } + free(prf_buffer); + } else { + /* Manual dives or unknown DCs */ + report_error("[Warning][smartrak_import]\t Manual or unknown dive computer for dive %d", smtkdive->number); + smtkdive->dc.duration.seconds = smtkdive->duration.seconds = smtk_time_to_secs(col[coln(DURATION)]->bind_ptr); + smtkdive->dc.maxdepth.mm = smtkdive->maxdepth.mm = strtod(col[coln(MAXDEPTH)]->bind_ptr, NULL) * 1000; + } + free(hdr_buffer); + /* + * Cylinder and gasmixes completion. + * Revisit data under some circunstances, e.g. a start pressure = 0 may mean + * that dc don't support gas control, in this situation let's look into mdb data + */ + int numtanks = (tanks == 10) ? 8 : 3; // Subsurface supports up to 8 tanks + int pstartcol = coln(PSTART); + int o2fraccol = coln(O2FRAC); + int hefraccol = coln(HEFRAC); + int tankidxcol = coln(TANKIDX); + + for (i = 0; i < numtanks; i++) { + if (smtkdive->cylinder[i].start.mbar == 0) { + smtkdive->cylinder[i].start.mbar = strtod(col[(i*2)+pstartcol]->bind_ptr, NULL) * 1000; + smtkdive->cylinder[i].gasmix.o2.permille = strtod(col[i+o2fraccol]->bind_ptr, NULL) * 10; + if (smtk_version == 10213) + smtkdive->cylinder[i].gasmix.he.permille = strtod(col[i+hefraccol]->bind_ptr, NULL) * 10; + else + smtkdive->cylinder[i].gasmix.he.permille = 0; + } + /* + * If there is a start pressure ensure that end pressure is not zero as + * will be registered in DCs which only keep track of differential pressures, + * and collect the data registered by the user in mdb + */ + if (smtkdive->cylinder[i].start.mbar != 0) { + if (smtkdive->cylinder[i].end.mbar == 0) + smtkdive->cylinder[i].end.mbar = strtod(col[(i * 2) + 1 + pstartcol]->bind_ptr, NULL) * 1000 ? : 1000; + smtk_build_tank_info(mdb_clon, smtkdive, i, col[i + tankidxcol]->bind_ptr); + } + } + + /* Date issues with libdc parser - Take date time from mdb */ + smtk_date_to_tm(col[coln(DATE)]->bind_ptr, tm_date); + smtk_time_to_tm(col[coln(INTIME)]->bind_ptr, tm_date); + smtkdive->dc.when = smtkdive->when = mktime(tm_date); + free(tm_date); + smtkdive->dc.surfacetime.seconds = smtk_time_to_secs(col[coln(INTVAL)]->bind_ptr); + + /* Data that user may have registered manually if not supported by DC, or not parsed */ + if (!smtkdive->airtemp.mkelvin) + smtkdive->airtemp.mkelvin = C_to_mkelvin(strtod(col[coln(AIRTEMP)]->bind_ptr, NULL)); + if (!smtkdive->watertemp.mkelvin) + smtkdive->watertemp.mkelvin = smtkdive->mintemp.mkelvin = C_to_mkelvin(strtod(col[coln(MINWATERTEMP)]->bind_ptr, NULL)); + if (!smtkdive->maxtemp.mkelvin) + smtkdive->maxtemp.mkelvin = C_to_mkelvin(strtod(col[coln(MAXWATERTEMP)]->bind_ptr, NULL)); + + /* No DC related data */ + smtkdive->visibility = strtod(col[coln(VISIBILITY)]->bind_ptr, NULL) > 25 ? 5 : strtod(col[13]->bind_ptr, NULL) / 5; + smtkdive->weightsystem[0].weight.grams = strtod(col[coln(WEIGHT)]->bind_ptr, NULL) * 1000; + smtkdive->suit = smtk_value_by_idx(mdb_clon, "Suit", 1, col[coln(SUITIDX)]->bind_ptr); + smtk_build_location(mdb_clon, col[coln(SITEIDX)]->bind_ptr, smtkdive->when, &smtkdive->dive_site_uuid); + smtkdive->buddy = smtk_locate_buddy(mdb_clon, col[0]->bind_ptr); + smtk_parse_relations(mdb_clon, smtkdive, col[0]->bind_ptr, "Type", "TypeRelation", true); + smtk_parse_relations(mdb_clon, smtkdive, col[0]->bind_ptr, "Activity", "ActivityRelation", false); + smtk_parse_relations(mdb_clon, smtkdive, col[0]->bind_ptr, "Gear", "GearRelation", false); + smtk_parse_relations(mdb_clon, smtkdive, col[0]->bind_ptr, "Fish", "FishRelation", false); + smtkdive->notes = smtk_concat_str(smtkdive->notes, "\n", "%s", col[coln(REMARKS)]->bind_ptr); + + record_dive_to_table(smtkdive, divetable); + free(devdata); + } + for (i = 0; i < mdb_table->num_cols; i++) + free(bound_values[i]); + mdb_free_tabledef(mdb_table); + mdb_free_catalog(mdb_clon); + mdb->catalog = NULL; + mdb_close(mdb_clon); + mdb_close(mdb); + sort_table(divetable); +} |