comparison Small_CPU/Src/pressure.c @ 382:14fd5f35cb50 MotionDetection

merge default
author Ideenmodellierer
date Thu, 10 Oct 2019 22:26:03 +0200
parents 1f24022345d1
children cb3870f79e9d
comparison
equal deleted inserted replaced
381:695434a6dcf6 382:14fd5f35cb50
44 #define CMD_ADC_1024 0x04 // ADC OSR=1024 44 #define CMD_ADC_1024 0x04 // ADC OSR=1024
45 #define CMD_ADC_2048 0x06 // ADC OSR=2056 45 #define CMD_ADC_2048 0x06 // ADC OSR=2056
46 #define CMD_ADC_4096 0x08 // ADC OSR=4096 46 #define CMD_ADC_4096 0x08 // ADC OSR=4096
47 #define CMD_PROM_RD 0xA0 // Prom read command 47 #define CMD_PROM_RD 0xA0 // Prom read command
48 48
49 #define PRESSURE_SURFACE_MAX_MBAR (1070.0f) /* It is very unlikely that pressure at surface is greater than this value => clip to it */ 49 /* remove comment to use a predefined profile for pressure changes instead of real world data */
50 #define PRESSURE_HISTORY_SIZE (8u) 50 /* #define SIMULATE_PRESSURE */
51 #define PRESSURE_JUMP_VALID_MBAR (500.0f) /* values are measure several times a second => jumps > 5m very unlikely */ 51
52 52 #define PRESSURE_SURFACE_MAX_MBAR (1030.0f) /* It is unlikely that pressure at surface is greater than this value => clip to it */
53 #define PRESSURE_SURFACE_QUE (30u) /* history buffer [minutes] for past pressure measurements */ 53 #define PRESSURE_HISTORY_SIZE (8u)
54 #define PRESSURE_JUMP_VALID_MBAR (500.0f) /* values are measure several times a second => jumps > 5m very unlikely */
55
56 #define PRESSURE_SURFACE_QUE (30u) /* history buffer [minutes] for past pressure measurements */
57 #define PRESSURE_SURFACE_EVA_WINDOW (15u) /* Number of entries evaluated during instability test. Used to avoid detection while dive enters water */
58 #define PRESSURE_SURFACE_STABLE_LIMIT (10u) /* Define pressure as stable if delta (mBar) is below this value */
59 #define PRESSURE_SURFACE_DETECT_STABLE_CNT (5u) /* Event count to detect stable condition */
60 #define PRESSURE_SURFACE_UNSTABLE_LIMIT (50u) /* Define pressure as not stable if delta (mBar) is larger than this value */
61 #define PRESSURE_SURFACE_DETECT_UNSTABLE_CNT (3u) /* Event count to detect unstable condition */
62
54 63
55 static uint16_t get_ci_by_coef_num(uint8_t coef_num); 64 static uint16_t get_ci_by_coef_num(uint8_t coef_num);
56 //void pressure_calculation_new(void); 65 //void pressure_calculation_new(void);
57 //void pressure_calculation_old(void); 66 //void pressure_calculation_old(void);
58 static void pressure_calculation_AN520_004_mod_MS5803_30BA__09_2015(void); 67 static void pressure_calculation_AN520_004_mod_MS5803_30BA__09_2015(void);
84 static float ambient_temperature = 0; 93 static float ambient_temperature = 0;
85 static float ambient_pressure_mbar = 1000.0; 94 static float ambient_pressure_mbar = 1000.0;
86 static float surface_pressure_mbar = 1000.0; 95 static float surface_pressure_mbar = 1000.0;
87 static float surface_ring_mbar[PRESSURE_SURFACE_QUE] = { 0 }; 96 static float surface_ring_mbar[PRESSURE_SURFACE_QUE] = { 0 };
88 97
98 static uint8_t surface_pressure_writeIndex = 0;
99 static float surface_pressure_stable_value = 0;
100 static uint8_t surface_pressure_stable = 0;
101
89 static float pressure_history_mbar[PRESSURE_HISTORY_SIZE]; 102 static float pressure_history_mbar[PRESSURE_HISTORY_SIZE];
90 103
91 static uint8_t secondCounterSurfaceRing = 0; 104 static uint8_t secondCounterSurfaceRing = 0;
92 static uint8_t avgCount = 0; 105 static uint8_t avgCount = 0;
93 static float runningAvg = 0; 106 static float runningAvg = 0;
117 runningAvg = 0; 130 runningAvg = 0;
118 131
119 for(int i=0; i<PRESSURE_SURFACE_QUE; i++) 132 for(int i=0; i<PRESSURE_SURFACE_QUE; i++)
120 surface_ring_mbar[i] = ambient_pressure_mbar; 133 surface_ring_mbar[i] = ambient_pressure_mbar;
121 surface_pressure_mbar = ambient_pressure_mbar; 134 surface_pressure_mbar = ambient_pressure_mbar;
135 surface_pressure_writeIndex = 0; /* index of the oldest value in the ring buffer */
122 } 136 }
123 } 137 }
124 138
125 void init_pressure_history(void) 139 void init_pressure_history(void)
126 { 140 {
128 { 142 {
129 pressure_history_mbar[i] = 1000.0; 143 pressure_history_mbar[i] = 1000.0;
130 } 144 }
131 } 145 }
132 146
133 147 uint8_t is_surface_pressure_stable(void)
148 {
149 return surface_pressure_stable;
150 }
151
152 float set_last_surface_pressure_stable(void)
153 {
154 surface_pressure_mbar = surface_pressure_stable_value;
155 return surface_pressure_stable_value;
156 }
157
158 /* iterate backward through the history memory and evaluate the changes pressure changes during the last 30 minutes */
159 void evaluate_surface_pressure()
160 {
161 uint8_t index;
162 float lastvalue;
163 uint8_t stablecnt = 0;
164 uint8_t unstablecnt = 0;
165 uint8_t EvaluationWindow = PRESSURE_SURFACE_QUE - PRESSURE_SURFACE_EVA_WINDOW; /* do not use the latest 15 values to avoid unstable condition due to something like fin handling */
166 uint8_t EvaluatedValues = 0;
167
168 lastvalue = surface_ring_mbar[surface_pressure_writeIndex];
169 surface_pressure_stable_value = surface_ring_mbar[surface_pressure_writeIndex]; /* default: if no stable value is found return the oldest value */
170 index = surface_pressure_writeIndex;
171 surface_pressure_stable = 1;
172
173 if(index == 0)
174 {
175 index = PRESSURE_SURFACE_QUE - 1;
176 }
177 else
178 {
179 index = index - 1;
180 }
181 do
182 {
183 if((EvaluatedValues < EvaluationWindow) &&
184 (fabs(surface_pressure_stable_value - surface_ring_mbar[index]) > PRESSURE_SURFACE_UNSTABLE_LIMIT)) /* unusual change during last 30 minutes */
185 {
186 unstablecnt++;
187 if(unstablecnt > PRESSURE_SURFACE_DETECT_UNSTABLE_CNT)
188 {
189 surface_pressure_stable = 0;
190 }
191 }
192 /* search for a value which does not change for several iterations */
193 if (fabs(lastvalue - surface_ring_mbar[index]) < PRESSURE_SURFACE_STABLE_LIMIT)
194 {
195 stablecnt++;
196 }
197 else
198 {
199 stablecnt = 0;
200 }
201 if ((stablecnt >= PRESSURE_SURFACE_DETECT_STABLE_CNT) && (surface_pressure_stable == 0)&&(surface_pressure_stable_value == surface_ring_mbar[surface_pressure_writeIndex])) /* pressure is unstable => search for new stable value */
202 {
203 surface_pressure_stable_value = surface_ring_mbar[index];
204 unstablecnt = 0;
205 }
206
207 lastvalue = surface_ring_mbar[index];
208
209 if(index == 0)
210 {
211 index = PRESSURE_SURFACE_QUE - 1;
212 }
213 else
214 {
215 index = index - 1;
216 }
217 EvaluatedValues++;
218 } while (index != surface_pressure_writeIndex);
219 }
134 void update_surface_pressure(uint8_t call_rhythm_seconds) 220 void update_surface_pressure(uint8_t call_rhythm_seconds)
135 { 221 {
136 static uint8_t writeIndex = 0; /* Reinitialization will reset all entries to the same value => no need to reinit write index */
137 222
138 223
139 if(is_init_pressure_done()) 224 if(is_init_pressure_done())
140 { 225 {
141 runningAvg = (runningAvg * avgCount + ambient_pressure_mbar) / (avgCount +1); 226 runningAvg = (runningAvg * avgCount + ambient_pressure_mbar) / (avgCount +1);
144 229
145 if(secondCounterSurfaceRing >= 60) 230 if(secondCounterSurfaceRing >= 60)
146 { 231 {
147 if(runningAvg < PRESSURE_SURFACE_MAX_MBAR) 232 if(runningAvg < PRESSURE_SURFACE_MAX_MBAR)
148 { 233 {
149 surface_ring_mbar[writeIndex] = runningAvg; 234 surface_ring_mbar[surface_pressure_writeIndex] = runningAvg;
150 } 235 }
151 else 236 else
152 { 237 {
153 surface_ring_mbar[writeIndex] = PRESSURE_SURFACE_MAX_MBAR; 238 surface_ring_mbar[surface_pressure_writeIndex] = PRESSURE_SURFACE_MAX_MBAR;
154 } 239 }
155 writeIndex++; /* the write index is now pointing to the oldest value in the buffer which will be overwritten next time */ 240 surface_pressure_writeIndex++; /* the write index is now pointing to the oldest value in the buffer which will be overwritten next time */
156 241
157 if(writeIndex == PRESSURE_SURFACE_QUE) 242 if(surface_pressure_writeIndex == PRESSURE_SURFACE_QUE)
158 { 243 {
159 writeIndex = 0; 244 surface_pressure_writeIndex = 0;
160 } 245 }
161 246
162 surface_pressure_mbar = surface_ring_mbar[writeIndex]; /* 30 minutes old measurement */ 247 surface_pressure_mbar = surface_ring_mbar[surface_pressure_writeIndex]; /* 30 minutes old measurement */
163 248
164 secondCounterSurfaceRing = 0; 249 secondCounterSurfaceRing = 0;
165 avgCount = 1; /* use the current value as starting point but restart the weight decrement of the measurements */ 250 avgCount = 1; /* use the current value as starting point but restart the weight decrement of the measurements */
166 } 251 }
252 evaluate_surface_pressure();
167 } 253 }
168 } 254 }
169 255
170 #ifdef DEMOMODE 256 #ifdef DEMOMODE
171 float demo_modify_temperature_helper(float bottom_mbar_diff_to_surface) 257 float demo_modify_temperature_helper(float bottom_mbar_diff_to_surface)
423 } 509 }
424 return statusReturn; 510 return statusReturn;
425 } 511 }
426 512
427 513
514 #ifdef SIMULATE_PRESSURE
515 void pressure_simulation()
516 {
517 static uint32_t tickstart = 0;
518 static float pressure_sim_mbar = 0;
519 static uint32_t passedSecond = 0;
520 static uint32_t secondtick = 0;
521
522 uint32_t lasttick = 0;
523
524
525
526 if( tickstart == 0)
527 {
528 tickstart = HAL_GetTick(); /* init time stamp */
529 secondtick = tickstart;
530 pressure_sim_mbar = 1000;
531 }
532
533 lasttick = HAL_GetTick();
534 if(time_elapsed_ms(secondtick,lasttick) > 1000) /* one second passed since last tick */
535 {
536 secondtick = lasttick;
537 passedSecond++;
538
539 #ifdef DIVE_AFTER_LANDING
540 if(passedSecond < 10) pressure_sim_mbar = 1000.0; /* stay stable for 10 seconds */
541 else if(passedSecond < 300) pressure_sim_mbar -= 1.0; /* decrease pressure in 5 minutes target 770mbar => delta 330 */
542 else if(passedSecond < 900) pressure_sim_mbar += 0.0; /*stay stable 10 minutes*/
543 else if(passedSecond < 1500) pressure_sim_mbar += 0.5; /* return to 1 bar in 10 Minutes*/
544 else if(passedSecond < 1800) pressure_sim_mbar += 0.0; /* 5 minutes break */
545 else if(passedSecond < 2000) pressure_sim_mbar += 10.0; /* start dive */
546 else if(passedSecond < 2300) pressure_sim_mbar += 0.0; /* stay on depth */
547 else if(passedSecond < 2500) pressure_sim_mbar -= 10.0; /* return to surface */
548 else pressure_sim_mbar = 1000.0; /* final state */
549 #else /* short dive */
550 if(passedSecond < 10) pressure_sim_mbar = 1000.0; /* stay stable for 10 seconds */
551 else if(passedSecond < 180) pressure_sim_mbar += 10.0; /* Start dive */
552 else if(passedSecond < 300) pressure_sim_mbar += 0.0; /*stay on depth*/
553 else if(passedSecond < 460) pressure_sim_mbar -= 10.0; /* return to surface */
554 else if(passedSecond < 600) pressure_sim_mbar += 0.0; /* stay */
555 else if(passedSecond < 610) pressure_sim_mbar = 1000.0; /* get ready for second dive */
556 else if(passedSecond < 780) pressure_sim_mbar += 10.0; /* Start dive */
557 else if(passedSecond < 900) pressure_sim_mbar += 0.0; /*stay on depth*/
558 else if(passedSecond < 1060) pressure_sim_mbar -= 10.0; /* return to surface */
559 else if(passedSecond < 1200) pressure_sim_mbar += 0.0; /* stay */
560 else pressure_sim_mbar = 1000.0; /* final state */
561 #endif
562 }
563
564
565 ambient_pressure_mbar = pressure_sim_mbar;
566 ambient_temperature = 25.0;
567 return;
568 }
569
570 #endif
571
428 void pressure_calculation(void) 572 void pressure_calculation(void)
429 { 573 {
430 if(global.I2C_SystemStatus != HAL_OK) 574 if(global.I2C_SystemStatus != HAL_OK)
431 return; 575 return;
432 576
577 #ifdef SIMULATE_PRESSURE
578 pressure_simulation();
579 #else
433 pressure_calculation_AN520_004_mod_MS5803_30BA__09_2015(); 580 pressure_calculation_AN520_004_mod_MS5803_30BA__09_2015();
581 #endif
434 } 582 }
435 583
436 static uint8_t pressure_plausible(float pressurevalue) 584 static uint8_t pressure_plausible(float pressurevalue)
437 { 585 {
438 static uint8_t pressurewriteindex = 0; 586 static uint8_t pressurewriteindex = 0;
550 avgCnt++; /* by the measurement range of the sensor which is focused on under water pressure measurement */ 698 avgCnt++; /* by the measurement range of the sensor which is focused on under water pressure measurement */
551 } 699 }
552 ambient_pressure_mbar = runningAvg; 700 ambient_pressure_mbar = runningAvg;
553 } 701 }
554 } 702 }
555
556
557 /*
558 void pressure_calculation_new(void)
559 {
560 #define POW2_8 (256)
561 #define POW2_17 (131072)
562 #define POW2_6 (64)
563 #define POW2_16 (65536)
564 #define POW2_7 (128)
565 #define POW2_23 (8388608)
566 #define POW2_21 (2097152)
567 #define POW2_15 (32768)
568 #define POW2_13 (8192)
569 #define POW2_37 (137438953472)
570 #define POW2_4 (16)
571 #define POW2_33 (8589934592)
572 #define POW2_3 (8)
573
574 int32_t P; // compensated pressure value
575 int32_t T; // compensated temperature value
576 int32_t dT; // difference between actual and measured temperature
577 int64_t OFF; // offset at actual temperature
578 int64_t SENS;
579
580 int32_t T2;
581 int64_t OFF2;
582 int64_t SENS2;
583
584 dT = ((int32_t)D2) - ((int32_t)C[5]) * POW2_8;
585 OFF = ((int64_t)C[2]) * POW2_16 + ((int64_t)dT) * ((int64_t)C[4]) / POW2_7;
586 SENS = ((int64_t)C[1]) * POW2_15 + ((int64_t)dT) * ((int64_t)C[3]) / POW2_8;
587
588 T = 2000 + (dT * ((int32_t)C[6])) / POW2_23;
589
590
591 if(T < 2000) // low temperature
592 {
593 T2 = 3 * dT * dT;
594 T2 /= POW2_33;
595 OFF2 = ((int64_t)T) - 2000;
596 OFF2 *= OFF2;
597 OFF2 *= 3;
598 OFF2 /= 2;
599 SENS2 = ((int64_t)T) - 2000;
600 SENS2 *= SENS2;
601 SENS2 *= 5;
602 SENS2 /= POW2_3;
603 }
604 else // high temperature
605 {
606 T2 = 7 * dT * dT;
607 T2 /= POW2_37;
608 OFF2 = ((int64_t)T) - 2000;
609 OFF2 *= OFF2;
610 OFF2 /= POW2_4;
611 SENS2 = 0;
612 }
613
614 T = T - T2;
615 OFF = OFF - OFF2;
616 SENS = SENS - SENS2;
617
618 P = (int32_t)(((((int64_t)D1) * SENS) / POW2_21 - OFF) / POW2_13);
619
620 ambient_temperature = ((float)T) / 100;
621 ambient_pressure_mbar = ((float)P) / 10;
622 }
623 */
624
625 /*
626 void pressure_calculation_old(void) {
627 //
628 double ambient_temperature_centigrad = 0;
629 double ambient_pressure_decimbar = 0;
630
631 // static for debug
632 static int64_t dt = 0;
633 static int64_t temp = 0;
634 static int64_t ms_off = 0;
635 static int64_t sens = 0;
636 //
637 static int64_t ms_off2 = 0;
638 static int64_t sens2 = 0;
639 static int64_t t2 = 0;
640
641 if((D2 == 0) || (D1 == 0))
642 return;
643 //
644
645 // dT = D2 - C[5] * POW2_8;
646 // T = 2000 + (dT * C[6]) / POW2_23;
647 dt = (int64_t)D2 - C5_x_2p8;
648 //temp ; // in 10 milliGrad Celcius
649 ambient_temperature_centigrad = 2000 + dt * C[6] / 8388608;
650
651
652 if(ambient_temperature_centigrad < 2000) // low temperature
653 {
654 t2 = 3 * dt;
655 t2 *= dt;
656 t2 /= 8589934592;
657 ms_off2 = ambient_temperature_centigrad - 2000;
658 ms_off2 *= ms_off2;
659 sens2 = ms_off2;
660 ms_off2 *= 3;
661 ms_off2 /= 2;
662 sens2 *= 5;
663 sens2 /= 8;
664 }
665 else // high temperature
666 {
667 t2 = 7 * dt;
668 t2 *= dt;
669 t2 /= 137438953472;
670 ms_off2 = ambient_temperature_centigrad - 2000;
671 ms_off2 *= ms_off2;
672 ms_off2 /= 16;
673 sens2 = 0;
674 }
675
676
677 //
678
679 // pressure
680 // OFF = C[2] * POW2_16 + dT * C[4] / POW2_7;
681 // SENS = C[1] * POW2_15 + dT * C[3] / POW2_8;
682 ms_off = C[4] * dt;
683 ms_off /= 128;
684 ms_off += C2_x_2p16;
685 //
686 sens = C[3] * dt;
687 sens /= 256;
688 sens += C1_x_2p15;
689
690 // 2nd order correction
691 ambient_temperature_centigrad -= t2;
692 ms_off -= ms_off2;
693 sens -= sens2;
694
695 ambient_temperature = ambient_temperature_centigrad / 100;
696 // P = (D1 * SENS / POW2_21 - OFF) / POW2_13;
697 temp = D1 * sens;
698 temp /= 2097152;
699 temp -= ms_off;
700 temp /= 8192;
701 ambient_pressure_decimbar = temp; // to float/double
702 ambient_pressure_mbar = ambient_pressure_decimbar / 10;
703 }
704 */
705 703
706 704
707 /* taken from AN520 by meas-spec.com dated 9. Aug. 2011 705 /* taken from AN520 by meas-spec.com dated 9. Aug. 2011
708 * short and int are both 16bit according to AVR/GCC google results 706 * short and int are both 16bit according to AVR/GCC google results
709 */ 707 */