summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorGravatar Linus Torvalds <torvalds@linux-foundation.org>2016-02-21 15:34:04 -0800
committerGravatar Dirk Hohndel <dirk@hohndel.org>2016-02-21 16:36:10 -0800
commit10ae00db3cf994dad1d784dfd7855c55c9153924 (patch)
tree33f4d4bf859a13bf114ed99cdb45939f8dbbce48
parentb71fc7ffb2d1a43cee859a9862d3f85303a5310a (diff)
downloadsubsurface-10ae00db3cf994dad1d784dfd7855c55c9153924.tar.gz
pressure interpolation: incrementally update interpolation data
Instead of re-calculating all the interpolation data for each plot entry (which means that we have a quadratic algorithm that walks over all the plot-info points for each plot-info point), we can just update it incrementally within any particular interpolation segment. The previous cleanups made the code sane enough to understand, and makes it trivial to see how you don't have to recalculate the full thing. This gets rid of the O(n**2) algorithm, and it instead becomes O(n*m) where 'n' is the number of plot entries, and 'm' is the number of gas segments (which is usually a much smaller numer, typically "1"). Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org> Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
-rw-r--r--subsurface-core/gaspressures.c13
1 files changed, 11 insertions, 2 deletions
diff --git a/subsurface-core/gaspressures.c b/subsurface-core/gaspressures.c
index 5e7cd72ea..a8bb80812 100644
--- a/subsurface-core/gaspressures.c
+++ b/subsurface-core/gaspressures.c
@@ -200,6 +200,8 @@ static void fill_missing_tank_pressures(struct dive *dive, struct plot_info *pi,
{
int cyl, i;
struct plot_data *entry;
+ pr_interpolate_t interpolate = { 0 };
+ pr_track_t *last_segment = NULL;
int cur_pr[MAX_CYLINDERS]; // cur_pr[MAX_CYLINDERS] is the CCR diluent cylinder
for (cyl = 0; cyl < MAX_CYLINDERS; cyl++) {
@@ -235,7 +237,6 @@ static void fill_missing_tank_pressures(struct dive *dive, struct plot_info *pi,
for (i = 1; i < pi->nr; i++) { // For each point on the profile:
double magic;
pr_track_t *segment;
- pr_interpolate_t interpolate;
int pressure;
int *save_pressure, *save_interpolated;
@@ -257,6 +258,7 @@ static void fill_missing_tank_pressures(struct dive *dive, struct plot_info *pi,
}
if (pressure) { // If there is a valid pressure value,
+ last_segment = NULL; // get rid of interpolation data,
cur_pr[cyl] = pressure; // set current pressure
continue; // and skip to next point.
}
@@ -272,7 +274,14 @@ static void fill_missing_tank_pressures(struct dive *dive, struct plot_info *pi,
}
// If there is a valid segment but no tank pressure ..
- interpolate = get_pr_interpolate_data(segment, pi, i); // Set up an interpolation structure
+ if (segment == last_segment) {
+ interpolate.acc_pressure_time += entry->pressure_time;
+ } else {
+ // Set up an interpolation structure
+ interpolate = get_pr_interpolate_data(segment, pi, i);
+ last_segment = segment;
+ }
+
if(dive->cylinder[cyl].cylinder_use == OC_GAS) {
/* if this segment has pressure_time, then calculate a new interpolated pressure */