// SPDX-License-Identifier: GPL-2.0 /* gaspressures.cpp * ---------------- * This file contains the routines to calculate the gas pressures in the cylinders. * The functions below support the code in profile.cpp. * The high-level function is populate_pressure_information(), called by function * create_plot_info_new() in profile.cpp. The other functions below are, in turn, * called by populate_pressure_information(). The calling sequence is as follows: * * populate_pressure_information() -> calc_pressure_time() * -> fill_missing_tank_pressures() -> fill_missing_segment_pressures() * -> get_pr_interpolate_data() */ #include "dive.h" #include "event.h" #include "profile.h" #include "gaspressures.h" #include "pref.h" #include #include /* * simple structure to track the beginning and end tank pressure as * well as the integral of depth over time spent while we have no * pressure reading from the tank */ struct pr_track_t { int start; int end; int t_start; int t_end; int pressure_time; pr_track_t(int start, int t_start) : start(start), end(0), t_start(t_start), t_end(t_start), pressure_time(0) { } }; struct pr_interpolate_t { int start; int end; int pressure_time; int acc_pressure_time; }; enum interpolation_strategy {SAC, TIME, CONSTANT}; #ifdef DEBUG_PR_TRACK static void dump_pr_track(int cyl, std::vector &track_pr) { printf("cyl%d:\n", cyl); for (const auto &item: track_pr) { printf(" start %f end %f t_start %d:%02d t_end %d:%02d pt %d\n", mbar_to_PSI(item.start), mbar_to_PSI(item.end), FRACTION_TUPLE(item.t_start, 60), FRACTION_TUPLE(item.t_end, 60), item.pressure_time); } } #endif /* * This looks at the pressures for one cylinder, and * calculates any missing beginning/end pressures for * each segment by taking the over-all SAC-rate into * account for that cylinder. * * NOTE! Many segments have full pressure information * (both beginning and ending pressure). But if we have * switched away from a cylinder, we will have the * beginning pressure for the first segment with a * missing end pressure. We may then have one or more * segments without beginning or end pressures, until * we finally have a segment with an end pressure. * * We want to spread out the pressure over these missing * segments according to how big of a time_pressure area * they have. */ static void fill_missing_segment_pressures(std::vector &list, enum interpolation_strategy strategy) { for (auto it = list.begin(); it != list.end(); ++it) { int start = it->start, end; int pt_sum = 0, pt = 0; auto tmp = it; for (;;) { pt_sum += tmp->pressure_time; end = tmp->end; if (end) break; end = start; if (std::next(tmp) == list.end()) break; ++tmp; } if (!start) start = end; /* * Now 'start' and 'end' contain the pressure values * for the set of segments described by 'list'..'tmp'. * pt_sum is the sum of all the pressure-times of the * segments. * * Now dole out the pressures relative to pressure-time. */ it->start = start; tmp->end = end; switch (strategy) { case SAC: for (;;) { int pressure; pt += it->pressure_time; pressure = start; if (pt_sum) pressure -= lrint((start - end) * (double)pt / pt_sum); it->end = pressure; if (it == tmp) break; ++it; it->start = pressure; } break; case TIME: if (it->t_end && (tmp->t_start - tmp->t_end)) { double magic = (it->t_start - tmp->t_end) / (tmp->t_start - tmp->t_end); it->end = lrint(start - (start - end) * magic); } else { it->end = start; } break; case CONSTANT: it->end = start; } } } #ifdef DEBUG_PR_INTERPOLATE void dump_pr_interpolate(int i, pr_interpolate_t interpolate_pr) { printf("Interpolate for entry %d: start %d - end %d - pt %d - acc_pt %d\n", i, interpolate_pr.start, interpolate_pr.end, interpolate_pr.pressure_time, interpolate_pr.acc_pressure_time); } #endif static pr_interpolate_t get_pr_interpolate_data(const pr_track_t &segment, struct plot_info &pi, int cur) { // cur = index to pi.entry corresponding to t_end of segment; pr_interpolate_t interpolate; int i; interpolate.start = segment.start; interpolate.end = segment.end; interpolate.acc_pressure_time = 0; interpolate.pressure_time = 0; for (i = 0; i < pi.nr; i++) { const plot_data &entry = pi.entry[i]; if (entry.sec < segment.t_start) continue; interpolate.pressure_time += entry.pressure_time; if (entry.sec >= segment.t_end) break; if (i <= cur) interpolate.acc_pressure_time += entry.pressure_time; } return interpolate; } static void fill_missing_tank_pressures(const struct dive *dive, struct plot_info &pi, std::vector &track_pr, int cyl) { int i; pr_interpolate_t interpolate = { 0, 0, 0, 0 }; int cur_pr; enum interpolation_strategy strategy; /* no segment where this cylinder is used */ if (track_pr.empty()) return; if (dive->get_cylinder(cyl)->cylinder_use == OC_GAS) strategy = SAC; else strategy = TIME; fill_missing_segment_pressures(track_pr, strategy); // Interpolate the missing tank pressure values .. cur_pr = track_pr[0].start; // in the pr_track_t lists of structures // and keep the starting pressure for each cylinder. #ifdef DEBUG_PR_TRACK dump_pr_track(cyl, track_pr); #endif /* Transfer interpolated cylinder pressures from pr_track strucktures to plotdata * Go down the list of tank pressures in plot_info. Align them with the start & * end times of each profile segment represented by a pr_track_t structure. Get * the accumulated pressure_depths from the pr_track_t structures and then * interpolate the pressure where these do not exist in the plot_info pressure * variables. Pressure values are transferred from the pr_track_t structures * to the plot_info structure, allowing us to plot the tank pressure. * * The first two pi structures are "fillers", but in case we don't have a sample * at time 0 we need to process the second of them here, therefore i=1 */ auto last_segment = track_pr.end(); for (i = 1; i < pi.nr; i++) { // For each point on the profile: const struct plot_data &entry = pi.entry[i]; int pressure = get_plot_pressure(pi, i, cyl); if (pressure) { // If there is a valid pressure value, last_segment = track_pr.end(); // get rid of interpolation data, cur_pr = pressure; // set current pressure continue; // and skip to next point. } // If there is NO valid pressure value.. // Find the pressure segment corresponding to this entry.. auto it = track_pr.begin(); while (it != track_pr.end() && it->t_end < entry.sec) // Find the track_pr with end time.. ++it; // ..that matches the plot_info time (entry.sec) // After last segment? All done. if (it == track_pr.end()) break; // Before first segment, or between segments.. Go on, no interpolation. if (it->t_start > entry.sec) continue; if (!it->pressure_time) { // Empty segment? set_plot_pressure_data(pi, i, SENSOR_PR, cyl, cur_pr); // Just use our current pressure continue; // and skip to next point. } // If there is a valid segment but no tank pressure .. if (it == last_segment) { interpolate.acc_pressure_time += entry.pressure_time; } else { // Set up an interpolation structure interpolate = get_pr_interpolate_data(*it, pi, i); last_segment = it; } if(dive->get_cylinder(cyl)->cylinder_use == OC_GAS) { /* if this segment has pressure_time, then calculate a new interpolated pressure */ if (interpolate.pressure_time) { /* Overall pressure change over total pressure-time for this segment*/ double magic = (interpolate.end - interpolate.start) / (double)interpolate.pressure_time; /* Use that overall pressure change to update the current pressure */ cur_pr = lrint(interpolate.start + magic * interpolate.acc_pressure_time); } } else { double magic = (interpolate.end - interpolate.start) / static_cast(it->t_end - it->t_start); cur_pr = lrint(it->start + magic * (entry.sec - it->t_start)); } set_plot_pressure_data(pi, i, INTERPOLATED_PR, cyl, cur_pr); // and store the interpolated data in plot_info } } /* * What's the pressure-time between two plot data entries? * We're calculating the integral of pressure over time by * adding these up. * * The units won't matter as long as everybody agrees about * them, since they'll cancel out - we use this to calculate * a constant SAC-rate-equivalent, but we only use it to * scale pressures, so it ends up being a unitless scaling * factor. */ static inline int calc_pressure_time(const struct dive *dive, const struct plot_data &a, const struct plot_data &b) { int time = b.sec - a.sec; int depth = (a.depth + b.depth) / 2; if (depth <= SURFACE_THRESHOLD) return 0; return dive->depth_to_mbar(depth) * time; } #ifdef PRINT_PRESSURES_DEBUG // A CCR debugging tool that prints the gas pressures in cylinder 0 and in the diluent cylinder, used in populate_pressure_information(): static void debug_print_pressures(struct plot_info &pi) { int i; for (i = 0; i < pi.nr; i++) printf("%5d |%9d | %9d |\n", i, get_plot_sensor_pressure(pi, i), get_plot_interpolated_pressure(pi, i)); } #endif /* This function goes through the list of tank pressures, of structure plot_info for the dive profile where each * item in the list corresponds to one point (node) of the profile. It finds values for which there are no tank * pressures (pressure==0). For each missing item (node) of tank pressure it creates a pr_track_t structure * that represents a segment on the dive profile and that contains tank pressures. There is a linked list of * pr_track_t structures for each cylinder. These pr_track_t structures ultimately allow for filling * the missing tank pressure values on the dive profile using the depth_pressure of the dive. To do this, it * calculates the summed pressure-time value for the duration of the dive and stores these in the pr_track_t * structures. This function is called by create_plot_info_new() in profile.cpp */ void populate_pressure_information(const struct dive *dive, const struct divecomputer *dc, struct plot_info &pi, int sensor) { int first, last, cyl; const cylinder_t *cylinder = dive->get_cylinder(sensor); std::vector track; size_t current = std::string::npos; int missing_pr = 0, dense = 1; const double gasfactor[5] = {1.0, 0.0, prefs.pscr_ratio/1000.0, 1.0, 1.0 }; if (sensor < 0 || static_cast(sensor) >= dive->cylinders.size()) return; /* if we have no pressure data whatsoever, this is pointless, so let's just return */ if (!cylinder->start.mbar && !cylinder->end.mbar && !cylinder->sample_start.mbar && !cylinder->sample_end.mbar) return; /* Get a rough range of where we have any pressures at all */ first = last = -1; for (int i = 0; i < pi.nr; i++) { int pressure = get_plot_sensor_pressure(pi, i, sensor); if (!pressure) continue; if (first < 0) first = i; last = i; } /* No sensor data at all? */ if (first == last) return; /* * Split the range: * - missing pressure data * - gas change events to other cylinders * * Note that we only look at gas switches if this cylinder * itself has a gas change event. */ cyl = sensor; event_loop loop_gas("gaschange"); const struct event *ev = dive->has_gaschange_event(dc, sensor) ? loop_gas.next(*dc) : nullptr; divemode_loop loop_mode(*dc); for (int i = first; i <= last; i++) { struct plot_data &entry = pi.entry[i]; int pressure = get_plot_sensor_pressure(pi, i, sensor); int time = entry.sec; while (ev && ev->time.seconds <= time) { // Find 1st gaschange event after cyl = dive->get_cylinder_index(*ev); // the current gas change. if (cyl < 0) cyl = sensor; ev = loop_gas.next(*dc); } divemode_t dmode = loop_mode.next(time); if (current != std::string::npos) { // calculate pressure-time, taking into account the dive mode for this specific segment. entry.pressure_time = (int)(calc_pressure_time(dive, pi.entry[i - 1], entry) * gasfactor[dmode] + 0.5); track[current].pressure_time += entry.pressure_time; track[current].t_end = entry.sec; if (pressure) track[current].end = pressure; } // We have a final pressure for 'current' // If a gas switch has occurred, finish the // current pressure track entry and continue // until we get back to this cylinder. if (cyl != sensor) { current = std::string::npos; set_plot_pressure_data(pi, i, SENSOR_PR, sensor, 0); continue; } // If we have no pressure information, we will need to // continue with or without a tracking entry. Mark any // existing tracking entry as non-dense, and remember // to fill in interpolated data. if (current != std::string::npos && !pressure) { missing_pr = 1; dense = 0; continue; } // If we already have a pressure tracking entry, and // it has not had any missing samples, just continue // using it - there's nothing to interpolate yet. if (current != std::string::npos && dense) continue; // We need to start a new tracking entry, either // because the previous was interrupted by a gas // switch event, or because the previous one has // missing entries that need to be interpolated. // Or maybe we didn't have a previous one at all, // and this is the first pressure entry. track.emplace_back(pressure, entry.sec); current = track.size() - 1; dense = 1; } if (missing_pr) fill_missing_tank_pressures(dive, pi, track, sensor); #ifdef PRINT_PRESSURES_DEBUG debug_print_pressures(pi); #endif }