summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--core/liquivision.c77
1 files changed, 57 insertions, 20 deletions
diff --git a/core/liquivision.c b/core/liquivision.c
index fb2b44446..e4e5c5f80 100644
--- a/core/liquivision.c
+++ b/core/liquivision.c
@@ -21,7 +21,18 @@ struct lv_event {
} pressure;
};
-uint16_t primary_sensor;
+// Liquivision supports the following sensor configurations:
+// Primary sensor only
+// Primary + Buddy sensor
+// Primary + Up to 4 additional sensors
+// Primary + Up to 9 addiitonal sensors
+struct lv_sensor_ids {
+ uint16_t primary;
+ uint16_t buddy;
+ uint16_t group[9];
+};
+
+struct lv_sensor_ids sensor_ids;
static int handle_event_ver2(int code, const unsigned char *ps, unsigned int ps_ptr, struct lv_event *event)
{
@@ -61,28 +72,52 @@ static int handle_event_ver3(int code, const unsigned char *ps, unsigned int ps_
case 0x000f:
// Tank pressure
event->time = array_uint32_le(ps + ps_ptr);
+ current_sensor = array_uint16_le(ps + ps_ptr + 4);
- /* As far as I know, Liquivision supports 2 sensors, own and buddie's. This is my
- * best guess how it is represented. */
+ event->pressure.sensor = -1;
+ event->pressure.mbar = array_uint16_le(ps + ps_ptr + 6) * 10; // cb->mb
- current_sensor = array_uint16_le(ps + ps_ptr + 4);
- if (primary_sensor == 0) {
- primary_sensor = current_sensor;
- }
- if (current_sensor == primary_sensor) {
+ if (current_sensor == sensor_ids.primary) {
event->pressure.sensor = 0;
- event->pressure.mbar = array_uint16_le(ps + ps_ptr + 6) * 10; // cb->mb
- } else {
- /* Ignoring the buddy sensor for no as we cannot draw it on the profile.
+ } else if (current_sensor == sensor_ids.buddy) {
event->pressure.sensor = 1;
- event->pressure.mbar = array_uint16_le(ps + ps_ptr + 6) * 10; // cb->mb
- */
+ } else {
+ int i;
+ for (i = 0; i < 9; ++i) {
+ if (current_sensor == sensor_ids.group[i]) {
+ event->pressure.sensor = i + 2;
+ break;
+ }
+ }
}
+
// 1 byte PSR
// 1 byte ST
skip = 10;
break;
case 0x0010:
+ // 4 byte time
+ // 2 byte primary transmitter S/N
+ // 2 byte buddy transmitter S/N
+ // 2 byte group transmitter S/N (9x)
+
+ // I don't think it's possible to change sensor IDs once a dive has started but disallow it here just in case
+ if (sensor_ids.primary == 0) {
+ sensor_ids.primary = array_uint16_le(ps + ps_ptr + 4);
+ }
+
+ if (sensor_ids.buddy == 0) {
+ sensor_ids.buddy = array_uint16_le(ps + ps_ptr + 6);
+ }
+
+ int i;
+ const unsigned char *group_ptr = ps + ps_ptr + 8;
+ for (i = 0; i < 9; ++i, group_ptr += 2) {
+ if (sensor_ids.group[i] == 0) {
+ sensor_ids.group[i] = array_uint16_le(group_ptr);
+ }
+ }
+
skip = 26;
break;
case 0x0015: // Unknown
@@ -109,7 +144,7 @@ static void parse_dives (int log_version, const unsigned char *buf, unsigned int
int i;
bool found_divesite = false;
dive = alloc_dive();
- primary_sensor = 0;
+ memset(&sensor_ids, 0, sizeof(sensor_ids));
dc = &dive->dc;
/* Just the main cylinder until we can handle the buddy cylinder porperly */
@@ -239,7 +274,7 @@ static void parse_dives (int log_version, const unsigned char *buf, unsigned int
ptr += 4;
dive->otu = lrintf(*(float *) (buf + ptr));
ptr += 4;
- dive_mode = *(buf + ptr++); // 0=Deco, 1=Gauge, 2=None
+ dive_mode = *(buf + ptr++); // 0=Deco, 1=Gauge, 2=None, 35=Rec
algorithm = *(buf + ptr++); // 0=ZH-L16C+GF
sample_count = array_uint32_le(buf + ptr);
}
@@ -275,8 +310,9 @@ static void parse_dives (int log_version, const unsigned char *buf, unsigned int
if (log_version == 3) {
ps_ptr += handle_event_ver3(event_code, ps, ps_ptr, &event);
- if (event_code != 0xf)
- continue; // ignore all by pressure sensor event
+ // Ignoring the buddy sensor for now as we cannot draw it on the profile.
+ if ((event_code != 0xf) || (event.pressure.sensor != 0))
+ continue; // ignore all but pressure sensor event
} else { // version 2
ps_ptr += handle_event_ver2(event_code, ps, ps_ptr, &event);
continue; // ignore all events
@@ -319,6 +355,7 @@ static void parse_dives (int log_version, const unsigned char *buf, unsigned int
sample->sensor = event.pressure.sensor;
sample->cylinderpressure.mbar = event.pressure.mbar;
finish_sample(dc);
+ d++;
break;
} else { // Event is prior to sample
@@ -333,9 +370,9 @@ static void parse_dives (int log_version, const unsigned char *buf, unsigned int
last_depth = array_uint16_le(ds + (d - 1) * 2) * 10; // cm->mm
last_temp = C_to_mkelvin((float) array_uint16_le(ts + (d - 1) * 2) / 10); // dC->mK
sample->depth.mm = last_depth + (depth_mm - last_depth)
- * (event.time - last_time) / sample_interval;
+ * ((int)event.time - last_time) / sample_interval;
sample->temperature.mkelvin = last_temp + (temp_mk - last_temp)
- * (event.time - last_time) / sample_interval;
+ * ((int)event.time - last_time) / sample_interval;
}
finish_sample(dc);
@@ -375,7 +412,7 @@ static void parse_dives (int log_version, const unsigned char *buf, unsigned int
break;
}
- while (*(ps + ps_ptr) != 0x04)
+ while (((ptr + ps_ptr + 4) < buf_size) && (*(ps + ps_ptr) != 0x04))
ps_ptr++;
}