Mercurial > public > mk2
changeset 192:c8816e4bc724
GF Model
+ Apply the Eric's Baker gradient formula.
+ Apply GF varying corrections for each compartiment.
+ B?hlmann 100/100 == Gradient factor 100/100.
+ Compute stop with gradient at current depth (not GF_low)
+ Add assert() on visual platform.
+ Overflow stop into the next bin at same depth.
+ debug last deco depth in range 3m..6m.
author | JeanDo |
---|---|
date | Fri, 11 Feb 2011 23:39:35 +0100 (2011-02-11) |
parents | a08fc66fee28 |
children | 652e17b6267a |
files | code_part1/OSTC_code_asm_part1/docu/README.txt code_part1/OSTC_code_c_part2/p2_deco.c code_part1/OSTC_code_c_part2/p2_deco.o code_part1/OSTC_code_c_part2/p2_definitions.h |
diffstat | 4 files changed, 247 insertions(+), 271 deletions(-) [+] |
line wrap: on
line diff
--- a/code_part1/OSTC_code_asm_part1/docu/README.txt Fri Feb 11 21:28:58 2011 +0100 +++ b/code_part1/OSTC_code_asm_part1/docu/README.txt Fri Feb 11 23:39:35 2011 +0100 @@ -62,9 +62,9 @@ - Click "Write Flash" button. Now you have 10sec to confirm on OSTC too. - - The bootloader should say it found a PIC18F6485 device, and start uploading. + - The bootloader should say it found a 18F6485 device, and start uploading. The OSTC should have the blue and red led blinking rapidly. - The upload time is ~ 1 minute. + The upload time is ~ 20 secondes. - Once done, the OSTC finishes its reboot.
--- a/code_part1/OSTC_code_c_part2/p2_deco.c Fri Feb 11 21:28:58 2011 +0100 +++ b/code_part1/OSTC_code_c_part2/p2_deco.c Fri Feb 11 23:39:35 2011 +0100 @@ -88,25 +88,34 @@ // ********************* #include <math.h> +// *********************************************** +// ** V A R I A B L E S D E F I N I T I O N S ** +// *********************************************** + +#include "p2_definitions.h" +#include "shared_definitions.h" + // ************************* // ** P R O T O T Y P E S ** // ************************* static void calc_hauptroutine(void); -static void calc_tissue_2_secs(void); -static void calc_tissue_1_min(void); static void calc_nullzeit(void); -static void backup_sim_pres_tissue(void); -static void restore_sim_pres_tissue(void); -static void calc_without_deco(void); +static void calc_tissue(PARAMETER unsigned char period); +static void calc_limit(PARAMETER float GF_current); + +static void calc_wo_deco_step_2s(void); static void clear_tissue(void); static void calc_ascenttime(void); static void update_startvalues(void); static void clear_deco_table(void); static void update_deco_table(void); -static void sim_tissue_1min(void); -static void sim_tissue_10min(void); + +static void backup_sim_pres_tissue(void); +static void restore_sim_pres_tissue(void); +static void sim_tissue(PARAMETER unsigned char period); +static void sim_limit(PARAMETER float GF_current); static void calc_gradient_factor(void); static void calc_wo_deco_step_1_min(void); @@ -117,33 +126,25 @@ static void calc_nextdecodepth(void); -// *********************************************** -// ** V A R I A B L E S D E F I N I T I O N S ** -// *********************************************** - -#include "p2_definitions.h" -#include "shared_definitions.h" - //---- Bank 3 parameters ----------------------------------------------------- #pragma udata bank4=0x400 -static unsigned char lock_GF_depth_list; +static unsigned char low_depth; static float temp_limit; static float GF_low; static float GF_high; static float GF_delta; -static float locked_GF_step; +static unsigned char low_depth; // Depth of deepest stop +static float locked_GF_step; // GF_delta / low_depth + +static unsigned char temp_depth_limit; -static float sim_pres_gtissue; -static float sim_pres_gtissue_diff; -static float sim_pres_gtissue_limit_GF_low; -static float sim_pres_gtissue_limit_GF_low_below_surface; -static unsigned char temp_depth_limit; -static float sim_pres_gtissue_limit; -static unsigned char sim_gtissue_no; +// Simulation context: used to predict ascent. +static unsigned char sim_lead_tissue_no; // Leading compatiment number. +static float sim_lead_tissue_limit; // Buhlmann tolerated pressure. -static unsigned char temp_depth_GF_low_meter; -static unsigned char internal_deco_pointer; +// Real context: what we are doing now. +static float calc_lead_tissue_limit; // static unsigned char internal_deco_time[32]; static unsigned char internal_deco_depth[32]; @@ -165,7 +166,6 @@ static float temp_atem; static float temp2_atem; static float temp_tissue; -static float temp_surface; static float N2_ratio; static float He_ratio; static float var_N2_a; @@ -176,7 +176,6 @@ static float var_He_e; static float var_N2_halftime; static float var_He_halftime; -static float pres_gtissue_limit; static float pres_diluent; // new in v.101 static float deco_diluent; // new in v.101 @@ -593,11 +592,7 @@ // new in v.102 // // INPUT, changing during dive: -// sim_pres_gtissue_limit_GF_low -// sim_pres_gtissue_limit_GF_low_below_surface -// sim_pres_gtissue -// sim_pres_gtissue_diff -// lock_GF_depth_list +// low_depth // // INPUT, fixed during dive: // pres_surface @@ -611,88 +606,99 @@ // locked_GF_step // temp_deco // temp_depth_limt -// lock_GF_depth_list +// low_depth // static void calc_nextdecodepth(void) { //---- ZH-L16 + GRADIENT FACTOR model ------------------------------------ if (char_I_deco_model == 1) { - overlay float next_stop; // Next stop to test, in Bar. + // Recompute leading gas limit, at current depth: + overlay float depth = (temp_deco - pres_surface) / 0.09995; + assert( depth >= 0.0 ); - if (lock_GF_depth_list == 0) - { - overlay unsigned char temp_depth_GF_low_number; + if( depth > low_depth ) + sim_limit( GF_low ); + else + sim_limit( GF_high - depth * locked_GF_step ); + + // Stops are needed ? + if( sim_lead_tissue_limit > pres_surface ) + { + // Deepest stop, in meter. + overlay unsigned char first_stop = 3 * (int)(0.99 + (sim_lead_tissue_limit - pres_surface) / 0.29985); + assert( first_stop < 128 ); - next_stop = sim_pres_gtissue_limit_GF_low_below_surface / 0.29985; - temp_depth_GF_low_number = (int) (next_stop + 0.99); // Deepest stop index. - if (temp_depth_GF_low_number > 31) - temp_depth_GF_low_number = 31; // deepest deco at 93 meter (31 deco stops) - if (temp_depth_GF_low_number < 0) - temp_depth_GF_low_number = 0; - temp_depth_GF_low_meter = 3 * temp_depth_GF_low_number; - next_stop = (float)temp_depth_GF_low_meter * 0.09995; - if (temp_depth_GF_low_number == 0) - locked_GF_step = 0; - else - locked_GF_step = GF_delta / (float)temp_depth_GF_low_number; - lock_GF_depth_list = 1; + // Apply correction for the first stop. + if( first_stop == 3 ) // new in v104 + first_stop = char_I_depth_last_deco; // Use last 3m..6m instead. - // new run - internal_deco_pointer = temp_depth_GF_low_number; - } + // Is it a new deepest first stop ? If yes this is the reference for + // the varying gradient factor. So reset that: + if( first_stop > low_depth ) + { + // Store the deepest stop depth, as reference for GF_low. + low_depth = first_stop; + locked_GF_step = GF_delta / low_depth; + } - // Check all stops until one is higher that tolerated presure - while(internal_deco_pointer > 0) - { - overlay unsigned char index; // Next index (0..30) - overlay unsigned char next_depth_limit; // Next depth (0m..90m) - overlay float press_tol; // Upper limit (lower pressure) tolerated. - overlay float GF_current; // Changing Gradient-Factor value at next depth. + // Check all stops until one is higher than tolerated presure + while(first_stop > 0) + { + overlay unsigned char next_stop; // Next index (0..30) + overlay float pres_stop; // Next depth (0m..90m) - index = internal_deco_pointer - 1; // Index of next (upper) stop. - if (index == 1) // new in v104 - next_depth_limit = char_I_depth_last_deco; // Use last 3m..6m instead. - else - next_depth_limit = 3 * index; // Convert index to meter. + if( first_stop <= char_I_depth_last_deco ) // new in v104 + next_stop = 0; + else + next_stop = first_stop - 3; // Index of next (upper) stop. + if( first_stop == 3 ) + first_stop = char_I_depth_last_deco; - next_stop = next_depth_limit * 0.09995 // Meters to bar - + pres_surface; - // current GF is GF_high - alpha (GF_high - GF_low) - // With alpha = currentStop / (totalStops-1), hence in [0..1] - GF_current = GF_high - (next_depth_limit/3) * locked_GF_step; - - // upper limit (lowest pressure tolerated): - press_tol = GF_current * sim_pres_gtissue_diff + sim_pres_gtissue; + pres_stop = next_stop * 0.09995 // Meters to bar + + pres_surface; + + // current GF is GF_high - alpha (GF_high - GF_low) + // With alpha = currentDepth / maxDepth, hence in [0..1] + sim_limit( GF_high - next_stop * locked_GF_step ); - if( press_tol > next_stop ) // check if ascent to next deco stop is ok - break; - - // Else, validate that stop and loop... - internal_deco_pointer = index; - } + // upper limit (lowest pressure tolerated): + if( sim_lead_tissue_limit >= pres_stop ) // check if ascent to next deco stop is ok + break; + + // Else, validate that stop and loop... + first_stop = next_stop; + } - // Redo the results once we do have the validated index - if( internal_deco_pointer==1 ) - temp_depth_limit = char_I_depth_last_deco; // Use last 3m..6m instead. + // first_stop is the last validated depth found: + temp_depth_limit = first_stop; // Stop depth, in meter. + temp_deco = first_stop * 0.09995 // Convert to bars, too. + + pres_surface; + if (first_stop > 0) + temp_deco += float_deco_distance; // Add security distance (bars too) + } else - temp_depth_limit = 3 * internal_deco_pointer; // Normal depth, from 0m to 96m - temp_deco = temp_depth_limit * 0.09995 // Convert to bars. - + pres_surface; - if (internal_deco_pointer != 0) - temp_deco += float_deco_distance; // Add security distance (bars too) + { + temp_deco = pres_surface; // surface ambient presure + temp_depth_limit = 0; // stop depth, in meter. + } } else //---- ZH-L16 model ------------------------------------------------- { - // calc_nextdecodepth - original model + overlay float pres_gradient; + + // Original model // optimized in v.101 // char_I_depth_last_deco included in v.101 - overlay float pres_gradient = sim_pres_gtissue_limit - pres_surface; + // Compute sim_lead_tissue_limit too, but just once. + sim_limit(1.0); + + pres_gradient = sim_lead_tissue_limit - pres_surface; if (pres_gradient >= 0) { - pres_gradient = pres_gradient / 0.29985; // == pres_gradient / 99.95 / 3m; - temp_depth_limit = 3 * (int) (pres_gradient + 0.99); // depth for deco [m] + pres_gradient /= 0.29985; // Bar --> stop number; + temp_depth_limit = 3 * (int) (pres_gradient + 0.99); // --> meter : depth for deco if (temp_depth_limit == 0) // At surface ? temp_deco = pres_surface; else @@ -707,7 +713,7 @@ else { temp_deco = pres_surface; // surface ambient presure - temp_depth_limit = 0; // stop number = 0 + temp_depth_limit = 0; // stop depth, in meter. } } } @@ -842,7 +848,7 @@ void deco_calc_without_deco(void) { RESET_C_STACK - calc_without_deco(); + calc_wo_deco_step_2s(); deco_calc_desaturation_time(); } @@ -922,24 +928,23 @@ } ////////////////////////////////////////////////////////////////////////////// -// calc_without_deco +// calc_wo_deco_step_2s // // optimized in v.101 (float_..saturation_multiplier) // // Note: fixed N2_ratio for standard air. -static void calc_without_deco(void) +static void calc_wo_deco_step_2s(void) { N2_ratio = 0.7902; // Sum as stated in b"uhlmann pres_respiration = (float)int_I_pres_respiration / 1000.0; // assembler code uses different digit system - pres_surface = (float)int_I_pres_surface / 1000.0; + pres_surface = (float)int_I_pres_surface / 1000.0; // the b"uhlmann formula using pres_surface does apply to the pressure without any inert ratio temp_atem = N2_ratio * (pres_respiration - 0.0627); // 0.0627 is the extra pressure in the body temp2_atem = 0.0; - temp_surface = pres_surface; // the b"uhlmann formula using temp_surface does apply to the pressure without any inert ratio float_desaturation_multiplier = char_I_desaturation_multiplier / 100.0; float_saturation_multiplier = char_I_saturation_multiplier / 100.0; - calc_tissue_2_secs(); // update the pressure in the 32 tissues in accordance with the new ambient pressure + calc_tissue(0); // update the pressure in the 32 tissues in accordance with the new ambient pressure for 2 seconds. clear_deco_table(); char_O_deco_status = 0; @@ -967,8 +972,7 @@ static void calc_hauptroutine(void) { static float backup_GF_step; - static unsigned char backup_lock; - static unsigned char backup_pointer; + static unsigned char backup_low_depth; calc_hauptroutine_data_input(); @@ -991,12 +995,10 @@ case 3: //---- At surface: start a new dive ------------------------------ clear_deco_table(); copy_deco_table(); - internal_deco_pointer = 0; update_startvalues(); - calc_nextdecodepth(); - lock_GF_depth_list = 0; - char_O_deco_status = 0; // Calc bottom-time/nullzeit next iteration. - backup_lock = 255; // backup is empty... + low_depth = 0; + char_O_deco_status = 0; // Calc bottom-time/nullzeit next iteration. + backup_low_depth = 255; // backup is empty... break; case 0: //---- bottom time ----------------------------------------------- @@ -1010,8 +1012,7 @@ // Backup ascention state, ie if we already fixed the low GF reference, // and its value. backup_GF_step = locked_GF_step; - backup_lock = lock_GF_depth_list; - backup_pointer = internal_deco_pointer; + backup_low_depth = low_depth; sim_ascent_to_first_stop(); char_O_deco_status = 1; // Cacl stops next time. @@ -1022,12 +1023,11 @@ // If simulation is finished, restore the GF low reference, so that // next ascent simulation is done from the current depth: - if( char_O_deco_status == 0 && backup_lock != 255) + if( char_O_deco_status == 0 && backup_low_depth != 255) { locked_GF_step = backup_GF_step; - lock_GF_depth_list = backup_lock; - internal_deco_pointer = backup_pointer; - backup_lock = 255; + low_depth = backup_low_depth; + backup_low_depth = 255; } break; @@ -1137,10 +1137,6 @@ // void calc_hauptroutine_update_tissues(void) { - overlay float gtissue_press; - overlay float gtissue_diff; - overlay float gtissue_limit; - if (char_I_const_ppO2 == 0) // new in v.101 pres_diluent = pres_respiration; // new in v.101 else @@ -1161,28 +1157,20 @@ temp2_atem = 0.0; // new in v.101 char_O_diluent = 0; } - temp_surface = pres_surface; if(!char_I_step_is_1min) - calc_tissue_2_secs(); + calc_tissue(0); else - calc_tissue_1_min(); - - gtissue_limit = pres_tissue_limit[char_O_gtissue_no]; - gtissue_press = pres_tissue[char_O_gtissue_no] + (pres_tissue+16)[char_O_gtissue_no]; - gtissue_diff = gtissue_limit - gtissue_press; - int_O_gtissue_limit = (int)(gtissue_limit * 1000); - int_O_gtissue_press = (int)(gtissue_press * 1000); + calc_tissue(1); - if (char_I_deco_model == 1) - // press_tol = GF_current * pres_gtissue_diff + pres_gtissue; - // With : pres_gtissue = pres_tissue[char_O_gtissue_no] + (pres_tissue+16)[char_O_gtissue_no] - // and gtissue_diff = sim_pres_gtissue_limit - sim_pres_gtissue - temp1 = GF_high * gtissue_diff + gtissue_press; - else - temp1 = pres_gtissue_limit; + // Calc limit for surface, ie. GF_high. + calc_limit(GF_high); - if( temp1 > temp_surface && char_O_deco_status == 0) // if guiding tissue can not be exposed to surface pressure immediately + int_O_gtissue_limit = (int)(calc_lead_tissue_limit * 1000); + int_O_gtissue_press = (int)((pres_tissue[char_O_gtissue_no] + (pres_tissue+16)[char_O_gtissue_no]) * 1000); + + // if guiding tissue can not be exposed to surface pressure immediately + if( calc_lead_tissue_limit > pres_surface && char_O_deco_status == 0) { char_O_nullzeit = 0; // deco necessary char_O_deco_status = 2; // calculate ascent on next iteration. @@ -1287,7 +1275,7 @@ //---- Else, continue simulating the stops --------------------------- check_gas_switch(); // Also updates ppN2 and ppHe. - sim_tissue_1min(); // Simulate compartiments for 1 minute. + sim_tissue(1); // Simulate compartiments for 1 minute. update_deco_table(); // Add one minute stops. } @@ -1312,30 +1300,28 @@ // Loop until first top or surface is reached. for(;;) { - temp_deco = temp_deco - 1.0; // Ascent 1 min, at 10m/min. == 1bar/min. + temp_deco -= 1.0; // Ascent 1 min, at 10m/min. == 1bar/min. - // TODO HERE TOO: WE MIGH HAVE STARTED ASCENT ! - if ( char_I_deco_model == 1) - temp_limit = sim_pres_gtissue_limit_GF_low; - else - temp_limit = sim_pres_gtissue_limit; + // Compute sim_lead_tissue_limit at GF_low (deepest stop). + sim_limit(GF_low); - // Can we ascent directly to surface ? - if( temp_deco <= temp_limit ) + // Did we reach first stop ? + if( temp_deco <= sim_lead_tissue_limit ) break; //Yes: finished ! // Next stop is surface ? if( temp_deco <= pres_surface ) break; // Yes: finished too. - //---- Simulat gas switches, at half the ascent - temp_deco += 0.5; // Check gas change 5 meter below new depth. + //---- Simulat gas switches, and compute ppN2/ppHe at half the ascent: + temp_deco += 0.5; // Check gas change 5 meter below new depth. check_gas_switch(); - temp_deco -= 0.5; // Back to new depth. + temp_deco -= 0.5; // Back to new depth. - // Then simulate with the new gas pressures...s - sim_tissue_1min(); + // Then simulate with the new gas pressures... (1min) + sim_tissue(1); } + temp_deco += 1.0; // Restore last correct depth. } ////////////////////////////////////////////////////////////////////////////// @@ -1345,9 +1331,6 @@ // static void calc_tissue(PARAMETER unsigned char period) { - char_O_gtissue_no = 255; - pres_gtissue_limit = 0.0; - for (ci=0;ci<16;ci++) { read_buhlmann_coefficients(period); // 2 sec or 1 min period. @@ -1361,40 +1344,50 @@ temp_tissue = (temp2_atem - (pres_tissue+16)[ci]) * var_He_e; temp_tissue_safety(); (pres_tissue+16)[ci] += temp_tissue; - - temp_tissue = pres_tissue[ci] + (pres_tissue+16)[ci]; - - var_N2_a = (var_N2_a * pres_tissue[ci] + var_He_a * (pres_tissue+16)[ci]) / temp_tissue; - var_N2_b = (var_N2_b * pres_tissue[ci] + var_He_b * (pres_tissue+16)[ci]) / temp_tissue; - pres_tissue_limit[ci] = (temp_tissue - var_N2_a) * var_N2_b; - if (pres_tissue_limit[ci] < 0) - pres_tissue_limit[ci] = 0; - if (pres_tissue_limit[ci] > pres_gtissue_limit) - { - pres_gtissue_limit = pres_tissue_limit[ci]; - char_O_gtissue_no = ci; - }//if - } // for + } } ////////////////////////////////////////////////////////////////////////////// -// calc_tissue_2_secs +// calc_limit // -// optimized in v.101 +// New in v.111 : separated from calc_tissue(), and depends on GF value. // -static void calc_tissue_2_secs(void) +static void calc_limit(PARAMETER float GF_current) { - calc_tissue(0); -} + char_O_gtissue_no = 255; + calc_lead_tissue_limit = 0.0; + + for (ci=0;ci<16;ci++) + { + overlay float p = pres_tissue[ci] + (pres_tissue+16)[ci]; + + read_buhlmann_coefficients(-1); // 2 sec or 1 min period. + var_N2_a = (var_N2_a * pres_tissue[ci] + var_He_a * (pres_tissue+16)[ci]) / p; + var_N2_b = (var_N2_b * pres_tissue[ci] + var_He_b * (pres_tissue+16)[ci]) / p; -////////////////////////////////////////////////////////////////////////////// -// calc_tissue_1_min -// -// optimized in v.101 -// -static void calc_tissue_1_min(void) -{ - calc_tissue(1); + // Apply the Eric Baker's varying gradient factor correction. + // Note: the correction factor depends both on GF and b, + // Actual values are in the 1.5 .. 1.0 range (for a GF=30%), + // so that can change who is the leading gas... + // Note: Also depends of the GF. So the calcul is different for + // GF_low, current GF, or GF_high... + // *BUT* calc_tissue() is used to compute bottom time, + // hence what would happend at surface, + // hence at GF_high. + if( char_I_deco_model == 1 ) + p = ( p - var_N2_a * GF_current) * var_N2_b + / (GF_current + var_N2_b * (1.0 - GF_current)); + else + p = (p - var_N2_a) * var_N2_b; + if( p < 0 ) p = 0; + + pres_tissue_limit[ci] = p; + if( p > calc_lead_tissue_limit ) + { + char_O_gtissue_no = ci; + calc_lead_tissue_limit = p; + } + } } ////////////////////////////////////////////////////////////////////////////// @@ -1412,13 +1405,10 @@ for(loop = 1; loop <= 17; loop++) { backup_sim_pres_tissue(); - sim_tissue_10min(); + sim_tissue(2); + sim_limit(GF_high); - if (char_I_deco_model == 1) - temp1 = GF_high * sim_pres_gtissue_diff + sim_pres_gtissue; - else - temp1 = sim_pres_gtissue_limit; - if (temp1 > temp_surface) // changed in v.102 , if guiding tissue can not be exposed to surface pressure immediately + if( sim_lead_tissue_limit > pres_surface ) // changed in v.102 , if guiding tissue can not be exposed to surface pressure immediately { restore_sim_pres_tissue(); break; @@ -1431,12 +1421,10 @@ { for(loop=1; loop <= 10; loop++) { - sim_tissue_1min(); - if (char_I_deco_model == 1) - temp1 = GF_high * sim_pres_gtissue_diff + sim_pres_gtissue; - else - temp1 = sim_pres_gtissue_limit; - if (temp1 > temp_surface) // changed in v.102 , if guiding tissue can not be exposed to surface pressure immediately + sim_tissue(1); + sim_limit(GF_high); + + if( sim_lead_tissue_limit > pres_surface) // changed in v.102 , if guiding tissue can not be exposed to surface pressure immediately break; char_O_nullzeit++; } @@ -1474,11 +1462,11 @@ { overlay unsigned char x; - // + 0.6 to count 1 minute ascent time from 4 meter to surface - overlay float ascent = pres_respiration - pres_surface + 0.6; + // + 0.7 to count 1 minute ascent time from 3 meter to surface + overlay float ascent = pres_respiration - pres_surface + 0.7; if (ascent < 0.0) ascent = 0.0; - int_O_ascenttime = (unsigned int)ascent; + int_O_ascenttime = (unsigned int)(ascent + 0.99); for(x=0; x<32 && internal_deco_depth[x]; x++) int_O_ascenttime += (unsigned int)internal_deco_time[x]; @@ -1496,16 +1484,6 @@ { overlay unsigned char x; - // Initialize data used to compute GF_low depth from current dive/simu - sim_gtissue_no = char_O_gtissue_no; - sim_pres_gtissue_limit = pres_gtissue_limit; - sim_pres_gtissue = pres_tissue[char_O_gtissue_no] + (pres_tissue+16)[char_O_gtissue_no]; - sim_pres_gtissue_diff = sim_pres_gtissue_limit - sim_pres_gtissue; // negative number - sim_pres_gtissue_limit_GF_low = GF_low * sim_pres_gtissue_diff + sim_pres_gtissue; - sim_pres_gtissue_limit_GF_low_below_surface = sim_pres_gtissue_limit_GF_low - pres_surface; - if (sim_pres_gtissue_limit_GF_low_below_surface < 0) - sim_pres_gtissue_limit_GF_low_below_surface = 0; - for (x = 0;x<16;x++) { sim_pres_tissue[x] = pres_tissue[x]; @@ -1515,21 +1493,16 @@ } ////////////////////////////////////////////////////////////////////////////// -// sim_tissue_1min +// sim_tissue // // optimized in v.101 // // Function very simular to calc_tissue, but: // + Use a 1min or 10min period. // + Do it on sim_pres_tissue, instead of pres_tissue. -// + Update GF_low state for GF decompression model. -// static void sim_tissue(PARAMETER unsigned char period) { - sim_pres_gtissue_limit = 0.0; - sim_gtissue_no = 255; - - for (ci=0;ci<16;ci++) + for(ci=0; ci<16; ci++) { read_buhlmann_coefficients(period); // 1 or 10 minute(s) interval @@ -1542,48 +1515,55 @@ temp_tissue = (temp2_atem - (sim_pres_tissue+16)[ci]) * var_He_e; temp_tissue_safety(); (sim_pres_tissue+16)[ci] += temp_tissue; - - // pressure limit - temp_tissue = sim_pres_tissue[ci] + (sim_pres_tissue+16)[ci]; - var_N2_a = (var_N2_a * sim_pres_tissue[ci] + var_He_a * (sim_pres_tissue+16)[ci]) / temp_tissue; - var_N2_b = (var_N2_b * sim_pres_tissue[ci] + var_He_b * (sim_pres_tissue+16)[ci]) / temp_tissue; - sim_pres_tissue_limit[ci] = (temp_tissue - var_N2_a) * var_N2_b; - - if (sim_pres_tissue_limit[ci] < 0) - sim_pres_tissue_limit[ci] = 0; - if (sim_pres_tissue_limit[ci] > sim_pres_gtissue_limit) - { - sim_pres_gtissue = temp_tissue; - sim_pres_gtissue_limit = sim_pres_tissue_limit[ci]; - sim_gtissue_no = ci; - } - } // for - - // Update data used to compute GF_low depth: - sim_pres_gtissue_diff = sim_pres_gtissue_limit - sim_pres_gtissue; - sim_pres_gtissue_limit_GF_low = GF_low * sim_pres_gtissue_diff + sim_pres_gtissue; - sim_pres_gtissue_limit_GF_low_below_surface = sim_pres_gtissue_limit_GF_low - pres_surface; - if (sim_pres_gtissue_limit_GF_low_below_surface < 0) - sim_pres_gtissue_limit_GF_low_below_surface = 0; + } } ////////////////////////////////////////////////////////////////////////////// -// sim_tissue_1min +// sim_limit() +// +// New in v.111 +// +// Function separated from sim_tissue() to allow recomputing limit on +// different depth, because it depends on current gradient factor. // -static void sim_tissue_1min(void) +static void sim_limit(PARAMETER float GF_current) { - sim_tissue(1); // 1 minute period + sim_lead_tissue_limit = 0.0; + sim_lead_tissue_no = 255; + + for(ci=0; ci<16; ci++) + { + overlay float p = sim_pres_tissue[ci] + (sim_pres_tissue+16)[ci]; + + read_buhlmann_coefficients(-1); + var_N2_a = (var_N2_a * sim_pres_tissue[ci] + var_He_a * (sim_pres_tissue+16)[ci]) / p; + var_N2_b = (var_N2_b * sim_pres_tissue[ci] + var_He_b * (sim_pres_tissue+16)[ci]) / p; + + // Apply the Eric Baker's varying gradient factor correction. + // Note: the correction factor depends both on GF and b, + // Actual values are in the 1.5 .. 1.0 range (for a GF=30%), + // so that can change who is the leading gas... + // Note: Also depends of the GF_current... + // *BUT* calc_tissue() is used to compute bottom time, + // hence what would happend at surface, + // hence at GF_high. + if( char_I_deco_model == 1 ) + p = ( p - var_N2_a * GF_current) * var_N2_b + / (GF_current + var_N2_b * (1.0 - GF_current)); + else + p = (p - var_N2_a) * var_N2_b; + if( p < 0 ) p = 0; + + sim_pres_tissue_limit[ci] = p; + if( p > sim_lead_tissue_limit ) + { + sim_lead_tissue_no = ci; + sim_lead_tissue_limit = p; + } + } // for ci } ////////////////////////////////////////////////////////////////////////////// -// sim_tissue_10min -// -static void sim_tissue_10min(void) -{ - sim_tissue(2); // 10 minutes period -} - -////////////////////////////////////////////////////////////////////////////// // clear_deco_table // // unchanged in v.101 @@ -1612,37 +1592,31 @@ // static void update_deco_table() { - overlay unsigned char x; - - if( temp_depth_limit > 255 ) // Can't store stops at more than 255m. - temp_depth_limit = 255; + overlay unsigned char x; - for(x=0; x<32; ++x) - { - // Did we found the right stop ? - if( internal_deco_depth[x] == temp_depth_limit ) - { - // Increment stop time, but do test overflow: - overlay int stop_time = 1 + (int)internal_deco_time[x]; - if( stop_time > 255 ) - stop_time = 255; - internal_deco_time[x] = stop_time; - - // Done ! - return; + for(x=0; x<32; ++x) + { + if( internal_deco_depth[x] == temp_depth_limit) + { + // Do not overflow (max 255') + if( internal_deco_time[x] < 255 ) + { + internal_deco_time[x]++; + return; + } + // But store extra in the next stop... } - + if( internal_deco_depth[x] == 0 ) { - // Found a free position: initialise it. - internal_deco_depth[x] = (unsigned char)temp_depth_limit; + internal_deco_depth[x] = temp_depth_limit; internal_deco_time[x] = 1; return; } } - // Here, there is no space left for stops. - // Ie. the first one starts at 3m*32 positions = 96m... + // Can't store stops at more than 96m. + // Or stops at less that 3m too. // Just do nothing with that... } @@ -1676,7 +1650,7 @@ if (char_I_deco_model == 1) // calculate relative gradient factor { - temp1 = (float)temp_depth_GF_low_meter * 0.09995; + temp1 = (float)low_depth * 0.09995; temp2 = pres_respiration - pres_surface; if (temp2 <= 0) temp1 = GF_high; @@ -1685,7 +1659,7 @@ else temp1 = GF_low + (temp1 - temp2)/temp1*GF_delta; - if (temp_depth_GF_low_meter == 0) + if (low_depth == 0) temp1 = GF_high; temp2 = temp3 / temp1; // temp3 is already in percent @@ -1820,14 +1794,13 @@ N2_ratio = 0.7902; // FIXED, sum lt. buehlmann pres_respiration = (float)int_I_pres_respiration / 1000.0; // assembler code uses different digit system - pres_surface = (float)int_I_pres_surface / 1000.0; + pres_surface = (float)int_I_pres_surface / 1000.0; // the b"uhlmann formula using pres_surface does not use the N2_ratio temp_atem = N2_ratio * (pres_respiration - 0.0627); // 0.0627 is the extra pressure in the body temp2_atem = 0.0; - temp_surface = pres_surface; // the b"uhlmann formula using temp_surface does not use the N2_ratio float_desaturation_multiplier = char_I_desaturation_multiplier / 142.0; // new in v.101 (70,42%/100.=142) float_saturation_multiplier = char_I_saturation_multiplier / 100.0; - calc_tissue_1_min(); // update the pressure in the 32 tissues in accordance with the new ambient pressure + calc_tissue(1); // update the pressure in the 32 tissues in accordance with the new ambient pressure clear_deco_table(); char_O_deco_status = 0;
--- a/code_part1/OSTC_code_c_part2/p2_definitions.h Fri Feb 11 21:28:58 2011 +0100 +++ b/code_part1/OSTC_code_c_part2/p2_definitions.h Fri Feb 11 23:39:35 2011 +0100 @@ -106,10 +106,13 @@ # define overlay # define PARAMETER +#include <assert.h> + // Avoid warnings about float/double mismatches: # pragma warning(disable: 4244 4068 4305) #else # define PARAMETER static +# define assert(predicate) #endif //////////////////////////////////////////////////////////////////////////////