mirror of
https://github.com/subsurface/subsurface.git
synced 2025-01-19 14:25:27 +00:00
Plot tank pressures for multiple tanks
The code keeps track of the segments of time when a specific tank was used and interpolates the pressure values for that tank based on a simulated average SAC rate for the times in which no pressure readings are available. This changes the way we used to plot the pressure when only beginning and end pressure of a tank are known; it used to be a straight line, now it is a sloped line where the steepness of the slope is proportional to the depth at that point - which is much more realistic. We also plot the pressures in two colors now. The old green for pressure data that came from the input file (that is not the same thing as saying it came from the computer - divelog for example appear to create pressure readings in the samples even if it only has beginning and end pressure). Interpolated values are plotted in yellow. If you have a sub-standard dive computer which has a frequently failing pressure sensor, you can now tell the parts of the plot where data was missing and we are filling in. The function that prints the pressure text labels had to be completely redone as it previously assumed one tank for the whole dive and simplisticly printed that tank's start and end pressure at the beginning and end of the profile plot with the y-values being the maximum and minimum pressure... This commit introduces a custom simplistic single linked list data structure to keep track of the pressure information per segment - Linus hated the idea of using GList for this purpose, and I have to admit that in the end this was very straight forward to implement and made the code easier to read and debug. Signed-off-by: Dirk Hohndel <dirk@hohndel.org>
This commit is contained in:
parent
bf1dc48dfe
commit
473fb14b56
1 changed files with 264 additions and 56 deletions
304
profile.c
304
profile.c
|
@ -26,8 +26,12 @@ struct plot_info {
|
||||||
int mintemp, maxtemp;
|
int mintemp, maxtemp;
|
||||||
struct plot_data {
|
struct plot_data {
|
||||||
unsigned int same_cylinder:1;
|
unsigned int same_cylinder:1;
|
||||||
|
unsigned int cylinderindex;
|
||||||
int sec;
|
int sec;
|
||||||
int pressure, temperature;
|
/* pressure[0] is sensor pressure
|
||||||
|
* pressure[1] is interpolated pressure */
|
||||||
|
int pressure[2];
|
||||||
|
int temperature;
|
||||||
/* Depth info */
|
/* Depth info */
|
||||||
int depth;
|
int depth;
|
||||||
int smoothed;
|
int smoothed;
|
||||||
|
@ -37,6 +41,10 @@ struct plot_info {
|
||||||
int avg[3];
|
int avg[3];
|
||||||
} entry[];
|
} entry[];
|
||||||
};
|
};
|
||||||
|
#define SENSOR_PR 0
|
||||||
|
#define INTERPOLATED_PR 1
|
||||||
|
#define SENSOR_PRESSURE(_entry) (_entry)->pressure[SENSOR_PR]
|
||||||
|
#define INTERPOLATED_PRESSURE(_entry) (_entry)->pressure[INTERPOLATED_PR]
|
||||||
|
|
||||||
/* convert velocity to colors */
|
/* convert velocity to colors */
|
||||||
typedef struct { double r, g, b; } rgb_t;
|
typedef struct { double r, g, b; } rgb_t;
|
||||||
|
@ -487,36 +495,55 @@ static int get_cylinder_pressure_range(struct graphics_context *gc, struct plot_
|
||||||
return pi->maxpressure != 0;
|
return pi->maxpressure != 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void plot_cylinder_pressure(struct graphics_context *gc, struct plot_info *pi)
|
static void plot_pressure_helper(struct graphics_context *gc, struct plot_info *pi, int type)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
int have_pressure = FALSE;
|
int lift_pen = FALSE;
|
||||||
|
|
||||||
if (!get_cylinder_pressure_range(gc, pi))
|
|
||||||
return;
|
|
||||||
|
|
||||||
set_source_rgba(gc, 0.2, 1.0, 0.2, 0.80);
|
|
||||||
|
|
||||||
for (i = 0; i < pi->nr; i++) {
|
for (i = 0; i < pi->nr; i++) {
|
||||||
int mbar;
|
int mbar;
|
||||||
struct plot_data *entry = pi->entry + i;
|
struct plot_data *entry = pi->entry + i;
|
||||||
|
|
||||||
mbar = entry->pressure;
|
mbar = entry->pressure[type];
|
||||||
if (!mbar)
|
if (!entry->same_cylinder)
|
||||||
|
lift_pen = TRUE;
|
||||||
|
if (!mbar) {
|
||||||
|
lift_pen = TRUE;
|
||||||
continue;
|
continue;
|
||||||
have_pressure = TRUE;
|
|
||||||
if (entry->same_cylinder)
|
|
||||||
line_to(gc, entry->sec, mbar);
|
|
||||||
else
|
|
||||||
move_to(gc, entry->sec, mbar);
|
|
||||||
}
|
}
|
||||||
/* if we have valid samples, we don't want to draw a line to the minpressure
|
if (lift_pen) {
|
||||||
* but just end wherever the dive ended (think valve shutdowns during dive)
|
if (i > 0 && entry->same_cylinder) {
|
||||||
* but that doesn't work so well if we have only max and min
|
/* if we have a previous event from the same tank,
|
||||||
*/
|
* draw at least a short line .
|
||||||
if (! have_pressure)
|
* This uses the implementation detail that the
|
||||||
line_to(gc, pi->maxtime, pi->minpressure);
|
* type is either 0 or 1 */
|
||||||
|
int prev_pr;
|
||||||
|
prev_pr = (entry-1)->pressure[type] ? : (entry-1)->pressure[1 - type];
|
||||||
|
move_to(gc, (entry-1)->sec, prev_pr);
|
||||||
|
line_to(gc, entry->sec, mbar);
|
||||||
|
} else
|
||||||
|
move_to(gc, entry->sec, mbar);
|
||||||
|
lift_pen = FALSE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
line_to(gc, entry->sec, mbar);
|
||||||
|
}
|
||||||
cairo_stroke(gc->cr);
|
cairo_stroke(gc->cr);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
static void plot_cylinder_pressure(struct graphics_context *gc, struct plot_info *pi)
|
||||||
|
{
|
||||||
|
if (!get_cylinder_pressure_range(gc, pi))
|
||||||
|
return;
|
||||||
|
|
||||||
|
/* first plot the pressure readings we have from the dive computer */
|
||||||
|
set_source_rgba(gc, 0.2, 1.0, 0.2, 0.80);
|
||||||
|
plot_pressure_helper(gc, pi, SENSOR_PR);
|
||||||
|
|
||||||
|
/* then, in a different color, the interpolated values */
|
||||||
|
set_source_rgba(gc, 1.0, 1.0, 0.2, 0.80);
|
||||||
|
plot_pressure_helper(gc, pi, INTERPOLATED_PR);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int mbar_to_PSI(int mbar)
|
static int mbar_to_PSI(int mbar)
|
||||||
|
@ -525,34 +552,71 @@ static int mbar_to_PSI(int mbar)
|
||||||
return to_PSI(p);
|
return to_PSI(p);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void plot_cylinder_pressure_text(struct graphics_context *gc, struct plot_info *pi)
|
static void plot_pressure_value(struct graphics_context *gc, int mbar, int sec,
|
||||||
|
int xalign, int yalign)
|
||||||
{
|
{
|
||||||
if (get_cylinder_pressure_range(gc, pi)) {
|
int pressure;
|
||||||
int start, end;
|
const char *unit;
|
||||||
const char *unit = "bar";
|
|
||||||
|
|
||||||
switch (output_units.pressure) {
|
switch (output_units.pressure) {
|
||||||
case PASCAL:
|
case PASCAL:
|
||||||
start = pi->maxpressure * 100;
|
pressure = mbar * 100;
|
||||||
end = pi->endpressure * 100;
|
|
||||||
unit = "pascal";
|
unit = "pascal";
|
||||||
break;
|
break;
|
||||||
case BAR:
|
case BAR:
|
||||||
start = (pi->maxpressure + 500) / 1000;
|
pressure = (mbar + 500) / 1000;
|
||||||
end = (pi->endpressure + 500) / 1000;
|
|
||||||
unit = "bar";
|
unit = "bar";
|
||||||
break;
|
break;
|
||||||
case PSI:
|
case PSI:
|
||||||
start = mbar_to_PSI(pi->maxpressure);
|
pressure = mbar_to_PSI(mbar);
|
||||||
end = mbar_to_PSI(pi->endpressure);
|
|
||||||
unit = "psi";
|
unit = "psi";
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
text_render_options_t tro = {10, 0.2, 1.0, 0.2, xalign, yalign};
|
||||||
|
plot_text(gc, &tro, sec, mbar, "%d %s", pressure, unit);
|
||||||
|
}
|
||||||
|
|
||||||
text_render_options_t tro = {10, 0.2, 1.0, 0.2, LEFT, TOP};
|
static void plot_cylinder_pressure_text(struct graphics_context *gc, struct plot_info *pi)
|
||||||
plot_text(gc, &tro, 0, pi->maxpressure, "%d %s", start, unit);
|
{
|
||||||
plot_text(gc, &tro, pi->maxtime, pi->endpressure,
|
int i;
|
||||||
"%d %s", end, unit);
|
int mbar, cyl;
|
||||||
|
int seen_cyl[MAX_CYLINDERS] = { FALSE, };
|
||||||
|
int last_pressure[MAX_CYLINDERS] = { 0, };
|
||||||
|
int last_time[MAX_CYLINDERS] = { 0, };
|
||||||
|
struct plot_data *entry;
|
||||||
|
|
||||||
|
if (!get_cylinder_pressure_range(gc, pi))
|
||||||
|
return;
|
||||||
|
|
||||||
|
/* only loop over the actual events from the dive computer */
|
||||||
|
for (i = 2; i < pi->nr - 2; i++) {
|
||||||
|
entry = pi->entry + i;
|
||||||
|
|
||||||
|
if (!entry->same_cylinder) {
|
||||||
|
cyl = entry->cylinderindex;
|
||||||
|
if (!seen_cyl[cyl]) {
|
||||||
|
mbar = SENSOR_PRESSURE(entry) ? : INTERPOLATED_PRESSURE(entry);
|
||||||
|
plot_pressure_value(gc, mbar, entry->sec, LEFT, BOTTOM);
|
||||||
|
seen_cyl[cyl] = TRUE;
|
||||||
|
}
|
||||||
|
if (i > 2) {
|
||||||
|
/* remember the last pressure and time of
|
||||||
|
* the previous cylinder */
|
||||||
|
cyl = (entry - 1)->cylinderindex;
|
||||||
|
last_pressure[cyl] =
|
||||||
|
SENSOR_PRESSURE(entry - 1) ? : INTERPOLATED_PRESSURE(entry - 1);
|
||||||
|
last_time[cyl] = (entry - 1)->sec;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cyl = entry->cylinderindex;
|
||||||
|
last_pressure[cyl] = SENSOR_PRESSURE(entry) ? : INTERPOLATED_PRESSURE(entry);
|
||||||
|
last_time[cyl] = entry->sec;
|
||||||
|
|
||||||
|
for (cyl = 0; cyl < MAX_CYLINDERS; cyl++) {
|
||||||
|
if (last_time[cyl]) {
|
||||||
|
plot_pressure_value(gc, last_pressure[cyl], last_time[cyl], CENTER, TOP);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -633,7 +697,7 @@ static struct plot_info *analyze_plot_info(struct plot_info *pi)
|
||||||
/* Do pressure min/max based on the non-surface data */
|
/* Do pressure min/max based on the non-surface data */
|
||||||
for (i = 0; i < nr; i++) {
|
for (i = 0; i < nr; i++) {
|
||||||
struct plot_data *entry = pi->entry+i;
|
struct plot_data *entry = pi->entry+i;
|
||||||
int pressure = entry->pressure;
|
int pressure = SENSOR_PRESSURE(entry) ? : INTERPOLATED_PRESSURE(entry);
|
||||||
int temperature = entry->temperature;
|
int temperature = entry->temperature;
|
||||||
|
|
||||||
if (pressure) {
|
if (pressure) {
|
||||||
|
@ -685,6 +749,117 @@ static struct plot_info *analyze_plot_info(struct plot_info *pi)
|
||||||
return pi;
|
return pi;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 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 */
|
||||||
|
typedef struct pr_track_struct pr_track_t;
|
||||||
|
struct pr_track_struct {
|
||||||
|
int start;
|
||||||
|
int end;
|
||||||
|
int t_start;
|
||||||
|
int t_end;
|
||||||
|
double pressure_time;
|
||||||
|
pr_track_t *next;
|
||||||
|
};
|
||||||
|
|
||||||
|
static pr_track_t *pr_track_alloc(int start, int t_start) {
|
||||||
|
pr_track_t *pt = malloc(sizeof(pr_track_t));
|
||||||
|
pt->start = start;
|
||||||
|
pt->t_start = t_start;
|
||||||
|
pt->end = 0;
|
||||||
|
pt->t_end = 0;
|
||||||
|
pt->pressure_time = 0.0;
|
||||||
|
pt->next = NULL;
|
||||||
|
return pt;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* poor man's linked list */
|
||||||
|
static pr_track_t *list_last(pr_track_t *list)
|
||||||
|
{
|
||||||
|
pr_track_t *tail = list;
|
||||||
|
if (!tail)
|
||||||
|
return NULL;
|
||||||
|
while (tail->next) {
|
||||||
|
tail = tail->next;
|
||||||
|
}
|
||||||
|
return tail;
|
||||||
|
}
|
||||||
|
|
||||||
|
static pr_track_t *list_add(pr_track_t *list, pr_track_t *element)
|
||||||
|
{
|
||||||
|
pr_track_t *tail = list_last(list);
|
||||||
|
if (!tail)
|
||||||
|
return element;
|
||||||
|
tail->next = element;
|
||||||
|
return list;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void list_free(pr_track_t *list)
|
||||||
|
{
|
||||||
|
if (!list)
|
||||||
|
return;
|
||||||
|
list_free(list->next);
|
||||||
|
free(list);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void fill_missing_tank_pressures(struct dive *dive, struct plot_info *pi,
|
||||||
|
pr_track_t **track_pr)
|
||||||
|
{
|
||||||
|
pr_track_t *list = NULL;
|
||||||
|
pr_track_t *nlist = NULL;
|
||||||
|
double pt, magic;
|
||||||
|
int cyl, i;
|
||||||
|
struct plot_data *entry;
|
||||||
|
int cur_pr[MAX_CYLINDERS];
|
||||||
|
|
||||||
|
for (cyl = 0; cyl < MAX_CYLINDERS; cyl++) {
|
||||||
|
cur_pr[cyl] = track_pr[cyl]->start;
|
||||||
|
}
|
||||||
|
for (i = 0; i < dive->samples; i++) {
|
||||||
|
entry = pi->entry + i + 2;
|
||||||
|
if (SENSOR_PRESSURE(entry)) {
|
||||||
|
cur_pr[entry->cylinderindex] = SENSOR_PRESSURE(entry);
|
||||||
|
} else {
|
||||||
|
if(!list || list->t_end < entry->sec) {
|
||||||
|
nlist = track_pr[entry->cylinderindex];
|
||||||
|
list = NULL;
|
||||||
|
while (nlist && nlist->t_start <= entry->sec) {
|
||||||
|
list = nlist;
|
||||||
|
nlist = list->next;
|
||||||
|
}
|
||||||
|
/* there may be multiple segments - so
|
||||||
|
* let's assemble the length */
|
||||||
|
nlist = list;
|
||||||
|
pt = list->pressure_time;
|
||||||
|
while (!nlist->end) {
|
||||||
|
nlist = nlist->next;
|
||||||
|
if (!nlist) {
|
||||||
|
/* oops - we have no end pressure,
|
||||||
|
* so this means this is a tank without
|
||||||
|
* gas consumption information */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
pt += nlist->pressure_time;
|
||||||
|
}
|
||||||
|
if (!nlist) {
|
||||||
|
/* just continue without calculating
|
||||||
|
* interpolated values */
|
||||||
|
list = NULL;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
magic = (nlist->end - cur_pr[entry->cylinderindex]) / pt; }
|
||||||
|
if (pt != 0.0) {
|
||||||
|
double cur_pt = (entry->sec - (entry-1)->sec) *
|
||||||
|
(1 + entry->depth / 10000.0);
|
||||||
|
INTERPOLATED_PRESSURE(entry) =
|
||||||
|
cur_pr[entry->cylinderindex] + cur_pt * magic;
|
||||||
|
cur_pr[entry->cylinderindex] = INTERPOLATED_PRESSURE(entry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Create a plot-info with smoothing and ranged min/max
|
* Create a plot-info with smoothing and ranged min/max
|
||||||
*
|
*
|
||||||
|
@ -696,9 +871,13 @@ static struct plot_info *create_plot_info(struct dive *dive)
|
||||||
{
|
{
|
||||||
int cylinderindex = -1;
|
int cylinderindex = -1;
|
||||||
int lastdepth, lastindex;
|
int lastdepth, lastindex;
|
||||||
int i, nr = dive->samples + 4, sec;
|
int i, nr = dive->samples + 4, sec, cyl;
|
||||||
size_t alloc_size = plot_info_size(nr);
|
size_t alloc_size = plot_info_size(nr);
|
||||||
struct plot_info *pi;
|
struct plot_info *pi;
|
||||||
|
pr_track_t *track_pr[MAX_CYLINDERS] = {NULL, };
|
||||||
|
pr_track_t *pr_track, *current;
|
||||||
|
gboolean missing_pr = FALSE;
|
||||||
|
struct plot_data *entry;
|
||||||
|
|
||||||
pi = malloc(alloc_size);
|
pi = malloc(alloc_size);
|
||||||
if (!pi)
|
if (!pi)
|
||||||
|
@ -708,19 +887,40 @@ static struct plot_info *create_plot_info(struct dive *dive)
|
||||||
sec = 0;
|
sec = 0;
|
||||||
lastindex = 0;
|
lastindex = 0;
|
||||||
lastdepth = -1;
|
lastdepth = -1;
|
||||||
|
for (cyl = 0; cyl < MAX_CYLINDERS; cyl++) /* initialize the start pressures */
|
||||||
|
track_pr[cyl] = pr_track_alloc(dive->cylinder[cyl].start.mbar, -1);
|
||||||
|
current = track_pr[dive->sample[0].cylinderindex];
|
||||||
for (i = 0; i < dive->samples; i++) {
|
for (i = 0; i < dive->samples; i++) {
|
||||||
int depth;
|
int depth;
|
||||||
struct sample *sample = dive->sample+i;
|
struct sample *sample = dive->sample+i;
|
||||||
struct plot_data *entry = pi->entry + i + 2;
|
|
||||||
|
|
||||||
|
entry = pi->entry + i + 2;
|
||||||
sec = entry->sec = sample->time.seconds;
|
sec = entry->sec = sample->time.seconds;
|
||||||
depth = entry->depth = sample->depth.mm;
|
depth = entry->depth = sample->depth.mm;
|
||||||
|
|
||||||
entry->same_cylinder = sample->cylinderindex == cylinderindex;
|
entry->same_cylinder = sample->cylinderindex == cylinderindex;
|
||||||
cylinderindex = sample->cylinderindex;
|
entry->cylinderindex = cylinderindex = sample->cylinderindex;
|
||||||
entry->pressure = sample->cylinderpressure.mbar;
|
SENSOR_PRESSURE(entry) = sample->cylinderpressure.mbar;
|
||||||
if (!entry->same_cylinder && !entry->pressure)
|
/* track the segments per cylinder and their pressure/time integral */
|
||||||
entry->pressure = dive->cylinder[cylinderindex].start.mbar;
|
if (!entry->same_cylinder) {
|
||||||
|
current->end = SENSOR_PRESSURE(entry-1);
|
||||||
|
current->t_end = (entry-1)->sec;
|
||||||
|
current = pr_track_alloc(SENSOR_PRESSURE(entry), entry->sec);
|
||||||
|
track_pr[cylinderindex] = list_add(track_pr[cylinderindex], current);
|
||||||
|
} else { /* same cylinder */
|
||||||
|
if ((!SENSOR_PRESSURE(entry) && SENSOR_PRESSURE(entry-1)) ||
|
||||||
|
(SENSOR_PRESSURE(entry) && !SENSOR_PRESSURE(entry-1))) {
|
||||||
|
/* transmitter changed its working status */
|
||||||
|
current->end = SENSOR_PRESSURE(entry-1);
|
||||||
|
current->t_end = (entry-1)->sec;
|
||||||
|
current = pr_track_alloc(SENSOR_PRESSURE(entry), entry->sec);
|
||||||
|
track_pr[cylinderindex] =
|
||||||
|
list_add(track_pr[cylinderindex], current);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/* finally, do the discrete integration to get the SAC rate equivalent */
|
||||||
|
current->pressure_time += (entry->sec - (entry-1)->sec) *
|
||||||
|
(1 + entry->depth / 10000.0);
|
||||||
|
missing_pr |= !SENSOR_PRESSURE(entry);
|
||||||
entry->temperature = sample->temperature.mkelvin;
|
entry->temperature = sample->temperature.mkelvin;
|
||||||
|
|
||||||
if (depth || lastdepth)
|
if (depth || lastdepth)
|
||||||
|
@ -730,17 +930,20 @@ static struct plot_info *create_plot_info(struct dive *dive)
|
||||||
if (depth > pi->maxdepth)
|
if (depth > pi->maxdepth)
|
||||||
pi->maxdepth = depth;
|
pi->maxdepth = depth;
|
||||||
}
|
}
|
||||||
|
current->t_end = entry->sec;
|
||||||
|
for (cyl = 0; cyl < MAX_CYLINDERS; cyl++) { /* initialize the end pressures */
|
||||||
|
int pr = dive->cylinder[cyl].end.mbar;
|
||||||
|
if (pr && track_pr[cyl]) {
|
||||||
|
pr_track = list_last(track_pr[cyl]);
|
||||||
|
pr_track->end = pr;
|
||||||
|
}
|
||||||
|
}
|
||||||
if (lastdepth)
|
if (lastdepth)
|
||||||
lastindex = i + 2;
|
lastindex = i + 2;
|
||||||
/* Fill in the last two entries with empty values but valid times */
|
/* Fill in the last two entries with empty values but valid times */
|
||||||
i = dive->samples + 2;
|
i = dive->samples + 2;
|
||||||
pi->entry[i].sec = sec + 20;
|
pi->entry[i].sec = sec + 20;
|
||||||
pi->entry[i+1].sec = sec + 40;
|
pi->entry[i+1].sec = sec + 40;
|
||||||
if (cylinderindex >= 0) {
|
|
||||||
pi->entry[i].pressure = dive->cylinder[cylinderindex].end.mbar;
|
|
||||||
pi->entry[i].same_cylinder = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
pi->nr = lastindex+1;
|
pi->nr = lastindex+1;
|
||||||
pi->maxtime = pi->entry[lastindex].sec;
|
pi->maxtime = pi->entry[lastindex].sec;
|
||||||
|
|
||||||
|
@ -749,6 +952,11 @@ static struct plot_info *create_plot_info(struct dive *dive)
|
||||||
|
|
||||||
pi->meandepth = dive->meandepth.mm;
|
pi->meandepth = dive->meandepth.mm;
|
||||||
|
|
||||||
|
if (missing_pr) {
|
||||||
|
fill_missing_tank_pressures(dive, pi, track_pr);
|
||||||
|
}
|
||||||
|
for (cyl = 0; cyl < MAX_CYLINDERS; cyl++)
|
||||||
|
list_free(track_pr[cyl]);
|
||||||
return analyze_plot_info(pi);
|
return analyze_plot_info(pi);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue