|  |  |  | @ -60,6 +60,7 @@ cCommandStream::CommandFn cmdGetScaleA; | 
		
	
		
			
				|  |  |  |  | cCommandStream::CommandFn cmdGetScaleB; | 
		
	
		
			
				|  |  |  |  | cCommandStream::CommandFn cmdCalibrateZeroScaleA; | 
		
	
		
			
				|  |  |  |  | cCommandStream::CommandFn cmdCalibrateZeroScaleB; | 
		
	
		
			
				|  |  |  |  | cCommandStream::CommandFn cmdCalibrateScales; | 
		
	
		
			
				|  |  |  |  | cCommandStream::CommandFn cmdCalibrateScaleA; | 
		
	
		
			
				|  |  |  |  | cCommandStream::CommandFn cmdCalibrateScaleB; | 
		
	
		
			
				|  |  |  |  | cCommandStream::CommandFn cmdSetDebugLevel; | 
		
	
	
		
			
				
					|  |  |  | @ -72,6 +73,7 @@ static const cCommandStream::cEntry sMyExtraCommmands[] = | 
		
	
		
			
				|  |  |  |  |   { "hello", cmdHello }, | 
		
	
		
			
				|  |  |  |  |   { "get_calibration_settings", cmdGetCalibrationSettings }, | 
		
	
		
			
				|  |  |  |  |   { "get_sensor_readings", cmdGetSensorReadings }, | 
		
	
		
			
				|  |  |  |  |   { "calibrate_scales", cmdCalibrateScales }, | 
		
	
		
			
				|  |  |  |  |   { "calibrate_zero_scale_a", cmdCalibrateZeroScaleA }, | 
		
	
		
			
				|  |  |  |  |   { "calibrate_zero_scale_b", cmdCalibrateZeroScaleB }, | 
		
	
		
			
				|  |  |  |  |   { "calibrate_scale_a", cmdCalibrateScaleA }, | 
		
	
	
		
			
				
					|  |  |  | @ -99,7 +101,7 @@ sMyExtraCommands_top( | 
		
	
		
			
				|  |  |  |  |   \****************************************************************************/ | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | byte my_position = 0;    // what is our actual measurement, starts with 0
 | 
		
	
		
			
				|  |  |  |  | long timer_pos0; | 
		
	
		
			
				|  |  |  |  | unsigned long timer_pos0; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | // Global Variables
 | 
		
	
		
			
				|  |  |  |  | LORA_data lora_data; | 
		
	
	
		
			
				
					|  |  |  | @ -110,6 +112,7 @@ long iteration = 0;       // what iteration number do we have, starts with 0 | 
		
	
		
			
				|  |  |  |  | long package_counter = 0; // sent package counter
 | 
		
	
		
			
				|  |  |  |  | bool send_in_progress = false; | 
		
	
		
			
				|  |  |  |  | bool stop_iterations = false; | 
		
	
		
			
				|  |  |  |  | bool start_new_iteration = false; | 
		
	
		
			
				|  |  |  |  | bool next_package_is_init_package = true; | 
		
	
		
			
				|  |  |  |  | uint32_t gRebootMs; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
	
		
			
				
					|  |  |  | @ -143,24 +146,17 @@ bool fFlash; | 
		
	
		
			
				|  |  |  |  | bool fUsbPower; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | // have we printed the sleep info?
 | 
		
	
		
			
				|  |  |  |  | bool g_fPrintedSleeping = false; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | //  the job that's used to synchronize us with the LMIC code
 | 
		
	
		
			
				|  |  |  |  | static osjob_t iterationJob; | 
		
	
		
			
				|  |  |  |  | static osjob_t sendJob; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | // the cycle time to use
 | 
		
	
		
			
				|  |  |  |  | unsigned gTxCycle; | 
		
	
		
			
				|  |  |  |  | // remaining before we reset to default
 | 
		
	
		
			
				|  |  |  |  | unsigned gTxCycleCount; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | void setup(void) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   gCatena.begin(); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   setup_platform(); | 
		
	
		
			
				|  |  |  |  |   SetupScales(config_data.debug_level); | 
		
	
		
			
				|  |  |  |  |   ClearLoraData(); | 
		
	
		
			
				|  |  |  |  |   ClearLoraData(true); | 
		
	
		
			
				|  |  |  |  |   setup_bme280(); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   setup_flash(); | 
		
	
	
		
			
				
					|  |  |  | @ -253,7 +249,6 @@ void setup_platform(void) | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   gLoRaWAN.SetReceiveBufferBufferCb(receiveMessage); | 
		
	
		
			
				|  |  |  |  |   setTxCycleTime(CATCFG_T_CYCLE_INITIAL, CATCFG_INTERVAL_COUNT_INITIAL); | 
		
	
		
			
				|  |  |  |  |   gCatena.registerObject(&gLoRaWAN); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   /* find the platform */ | 
		
	
	
		
			
				
					|  |  |  | @ -348,6 +343,9 @@ void setup_uplink(void) | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   LMIC_setClockError(1 * 65536 / 100); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   // explicitly enable LinkCheckMode
 | 
		
	
		
			
				|  |  |  |  |   gLoRaWAN.SetLinkCheckMode(true); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   /* figure out when to reboot */ | 
		
	
		
			
				|  |  |  |  |   gRebootMs = (CATCFG_T_REBOOT + os_getRndU2() - 32768) * 1000; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
	
		
			
				
					|  |  |  | @ -379,9 +377,12 @@ void setup_uplink(void) | 
		
	
		
			
				|  |  |  |  | void loop() | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   gCatena.poll(); | 
		
	
		
			
				|  |  |  |  |   if (start_new_iteration) { | 
		
	
		
			
				|  |  |  |  |     StartNewIteration(); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | void ClearLoraData(void) | 
		
	
		
			
				|  |  |  |  | void ClearLoraData(bool clearLastValues) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   lora_data.version = LORA_DATA_VERSION; | 
		
	
		
			
				|  |  |  |  |   lora_data.vbat = 0; | 
		
	
	
		
			
				
					|  |  |  | @ -411,6 +412,7 @@ void ClearLoraData(void) | 
		
	
		
			
				|  |  |  |  |   my_position = 0; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   // We initialize last_sensor_reading
 | 
		
	
		
			
				|  |  |  |  |   if (clearLastValues) { | 
		
	
		
			
				|  |  |  |  |     last_sensor_reading.vbat = 0; | 
		
	
		
			
				|  |  |  |  |     last_sensor_reading.weight1 = 0; | 
		
	
		
			
				|  |  |  |  |     last_sensor_reading.weight2 = 0; | 
		
	
	
		
			
				
					|  |  |  | @ -419,6 +421,7 @@ void ClearLoraData(void) | 
		
	
		
			
				|  |  |  |  |     last_sensor_reading.humidity = 0; | 
		
	
		
			
				|  |  |  |  |     last_sensor_reading.pressure = 0; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | void ShowLORAData(bool firstTime) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
	
		
			
				
					|  |  |  | @ -503,7 +506,7 @@ uint8_t GetVBatValue(int millivolts) | 
		
	
		
			
				|  |  |  |  | void DoDeepSleep(uint32_t sleep_time) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("DoDeepSleep, now going to deep sleep\n"); | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("DoDeepSleep, now going to deep sleep, millis: %d\n", millis()); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   // Prepare Deep Sleep
 | 
		
	
	
		
			
				
					|  |  |  | @ -520,13 +523,19 @@ void DoDeepSleep(uint32_t sleep_time) | 
		
	
		
			
				|  |  |  |  |   deepSleepRecovery(); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("done with deep sleep\n"); | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("done with deep sleep, millis: %d\n", millis()); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | void ReadSensors(SENSOR_data &sensor_data) { | 
		
	
		
			
				|  |  |  |  | // Returns true if measurements are plausible, otherwise false
 | 
		
	
		
			
				|  |  |  |  | bool ReadSensors(SENSOR_data &sensor_data) { | 
		
	
		
			
				|  |  |  |  |   bool plausible; | 
		
	
		
			
				|  |  |  |  |   bool plausible_a; | 
		
	
		
			
				|  |  |  |  |   bool plausible_b; | 
		
	
		
			
				|  |  |  |  |   SENSOR_data res; | 
		
	
		
			
				|  |  |  |  |   int32_t weight_current32; | 
		
	
		
			
				|  |  |  |  |   int32_t weight_current32_a; | 
		
	
		
			
				|  |  |  |  |   int32_t weight_current32_b; | 
		
	
		
			
				|  |  |  |  |   long w1_0_real; | 
		
	
		
			
				|  |  |  |  |   long w2_0_real; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
	
		
			
				
					|  |  |  | @ -544,30 +553,14 @@ void ReadSensors(SENSOR_data &sensor_data) { | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("LoadCell is ready.\n"); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     if (config_data.cal_w1_0 != NOT_ATTACHED) { | 
		
	
		
			
				|  |  |  |  |     res.weight1 = (int32_t)ReadScale('A'); | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("Load_cell 1 weight1_current: %ld\n", res.weight1); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     } else { | 
		
	
		
			
				|  |  |  |  |       res.weight1 = 0; | 
		
	
		
			
				|  |  |  |  |       w1_0_real = 0; | 
		
	
		
			
				|  |  |  |  |       if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |         gCatena.SafePrintf("Load_cell 1 is disabled\n"); | 
		
	
		
			
				|  |  |  |  |       } | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     if (config_data.cal_w2_0 != NOT_ATTACHED) { | 
		
	
		
			
				|  |  |  |  |     res.weight2 = (int32_t)ReadScale('B'); | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("Load_cell 2 weight2_current: %ld\n", res.weight2); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     } else { | 
		
	
		
			
				|  |  |  |  |       res.weight2 = 0; | 
		
	
		
			
				|  |  |  |  |       w2_0_real = 0; | 
		
	
		
			
				|  |  |  |  |       if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |         gCatena.SafePrintf("Load_cell 2 is disabled\n"); | 
		
	
		
			
				|  |  |  |  |       } | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  |   else { | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 0) { | 
		
	
	
		
			
				
					|  |  |  | @ -579,18 +572,32 @@ void ReadSensors(SENSOR_data &sensor_data) { | 
		
	
		
			
				|  |  |  |  |   PowerdownScale(); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   // Gewicht berechnen
 | 
		
	
		
			
				|  |  |  |  |   weight_current32 = (int32_t)((((res.weight1 - w1_0_real) / config_data.cal_w1_factor) + ((res.weight2 - w2_0_real) / config_data.cal_w2_factor)) / 5.0); | 
		
	
		
			
				|  |  |  |  |   weight_current32_a = (int32_t)((res.weight1 - w1_0_real) / config_data.cal_w1_factor); | 
		
	
		
			
				|  |  |  |  |   weight_current32_b = (int32_t)((res.weight2 - w2_0_real) / config_data.cal_w2_factor); | 
		
	
		
			
				|  |  |  |  |   weight_current32 = (int32_t)((weight_current32_a + weight_current32_b) / 5.0); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   // we check if weights are plausible
 | 
		
	
		
			
				|  |  |  |  |   plausible_a = (weight_current32_a > -10000) && (weight_current32_a < 150000); | 
		
	
		
			
				|  |  |  |  |   plausible_b = (weight_current32_b > -10000) && (weight_current32_b < 150000); | 
		
	
		
			
				|  |  |  |  |   plausible = (plausible_a && plausible_b); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (weight_current32 < 0) { | 
		
	
		
			
				|  |  |  |  |     if (plausible) {  | 
		
	
		
			
				|  |  |  |  |       weight_current32 = 0; | 
		
	
		
			
				|  |  |  |  |     }  | 
		
	
		
			
				|  |  |  |  |   } else if (weight_current32 > UINT16_MAX) { | 
		
	
		
			
				|  |  |  |  |     //weight_current32 = UINT16_MAX;
 | 
		
	
		
			
				|  |  |  |  |     // we set the weight to 0, as such high values are not realistic and probably a sign for bad calibration...
 | 
		
	
		
			
				|  |  |  |  |     weight_current32 = 0; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  |   if (config_data.cal_w1_0 == NOT_ATTACHED || config_data.cal_w2_0 == NOT_ATTACHED) { | 
		
	
		
			
				|  |  |  |  |     // when at least one load cell is disabled, we multiply the measured weight by 2
 | 
		
	
		
			
				|  |  |  |  |     weight_current32 = weight_current32 * 2; | 
		
	
		
			
				|  |  |  |  |   if (!plausible) { | 
		
	
		
			
				|  |  |  |  |     weight_current32 = NOT_PLAUSIBLE_16; | 
		
	
		
			
				|  |  |  |  |     if (!plausible_a) { | 
		
	
		
			
				|  |  |  |  |       res.weight1 = NOT_PLAUSIBLE_32; | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     if (!plausible_b) { | 
		
	
		
			
				|  |  |  |  |       res.weight2 = NOT_PLAUSIBLE_32; | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  |   res.weight = (uint16_t)weight_current32; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
	
		
			
				
					|  |  |  | @ -618,9 +625,11 @@ void ReadSensors(SENSOR_data &sensor_data) { | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   sensor_data = res; | 
		
	
		
			
				|  |  |  |  |   return plausible; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | void StartNewIteration() { | 
		
	
		
			
				|  |  |  |  |   start_new_iteration = false; | 
		
	
		
			
				|  |  |  |  |   uint32_t wait_time; | 
		
	
		
			
				|  |  |  |  |   wait_time = 0; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
	
		
			
				
					|  |  |  | @ -628,7 +637,18 @@ void StartNewIteration() { | 
		
	
		
			
				|  |  |  |  |   iteration++; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   SENSOR_data current_sensor_reading; | 
		
	
		
			
				|  |  |  |  |   ReadSensors(current_sensor_reading); | 
		
	
		
			
				|  |  |  |  |   if (!ReadSensors(current_sensor_reading)) { | 
		
	
		
			
				|  |  |  |  |     // we try a second time if Readings do not seem plausible
 | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("Readings do not seem plausible, try a second time\n"); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     delay(500); | 
		
	
		
			
				|  |  |  |  |     if (!ReadSensors(current_sensor_reading)) { | 
		
	
		
			
				|  |  |  |  |       if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |         gCatena.SafePrintf("Readings do not seem plausible for a second time, we give up!\n"); | 
		
	
		
			
				|  |  |  |  |       } | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  |   int16_t temp_change; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   // vBus
 | 
		
	
	
		
			
				
					|  |  |  | @ -677,35 +697,45 @@ void StartNewIteration() { | 
		
	
		
			
				|  |  |  |  |   // we send data the first time the system is started, when the array is full
 | 
		
	
		
			
				|  |  |  |  |   // or when the weight has fallen more than threshold or the first measurement is
 | 
		
	
		
			
				|  |  |  |  |   // more than one hour old (which should not happen :-) )
 | 
		
	
		
			
				|  |  |  |  |   if ( (next_package_is_init_package) || (my_position >= MAX_VALUES_TO_SEND) || (abs(last_sensor_reading.weight - current_sensor_reading.weight) > SEND_DIFF_THRESHOLD_5GRAMS) || ((millis() - timer_pos0) > 3600000)) { | 
		
	
		
			
				|  |  |  |  |   bool big_difference = (abs(last_sensor_reading.weight - current_sensor_reading.weight) > SEND_DIFF_THRESHOLD_5GRAMS); | 
		
	
		
			
				|  |  |  |  |   if ( (next_package_is_init_package) || (my_position >= MAX_VALUES_TO_SEND) || (big_difference) || ((millis() - timer_pos0) > 3600000)) { | 
		
	
		
			
				|  |  |  |  |     lora_data.offset_last_reading = (uint8_t)((millis() - timer_pos0) / 1000 / 60); | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("startSendingUplink(), my_position: %d, iteration: %d, package_counter: %d\n", my_position, iteration, package_counter); | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("startSendingUplink(), my_position: %d, iteration: %d, package_counter: %d, big_difference: %d\n", my_position, iteration, package_counter, big_difference); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     // the first <INIT_PACKETS> packets are "Init-Packets" or each INIT_PACKAGE_INTERVAL ...
 | 
		
	
		
			
				|  |  |  |  |     startSendingUplink(next_package_is_init_package); | 
		
	
		
			
				|  |  |  |  |     // send confirmed if big_difference in weight
 | 
		
	
		
			
				|  |  |  |  |     startSendingUplink(next_package_is_init_package, big_difference); | 
		
	
		
			
				|  |  |  |  |     next_package_is_init_package = ((iteration < INIT_PACKETS) || ((package_counter % INIT_PACKAGE_INTERVAL) == 0)); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 1) { | 
		
	
		
			
				|  |  |  |  |       gLed.Set(LedPattern::TwoShort); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |     // Loop while sending is in progress, timeout just in case after 300 seconds
 | 
		
	
		
			
				|  |  |  |  |     long start_time = millis(); | 
		
	
		
			
				|  |  |  |  |     // Loop sending is in progress, timeout just in case after 600 seconds
 | 
		
	
		
			
				|  |  |  |  |     unsigned long start_time = millis(); | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("waiting while send is in progress\n"); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     while (send_in_progress && ((millis() - start_time) < 300000)) | 
		
	
		
			
				|  |  |  |  |     while (send_in_progress && ((millis() - start_time) < 600000)) | 
		
	
		
			
				|  |  |  |  |     { | 
		
	
		
			
				|  |  |  |  |       os_runloop_once(); | 
		
	
		
			
				|  |  |  |  | //      gCatena.poll();
 | 
		
	
		
			
				|  |  |  |  | //      yield();
 | 
		
	
		
			
				|  |  |  |  |       gCatena.poll(); | 
		
	
		
			
				|  |  |  |  |       yield(); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     // handle timeout...
 | 
		
	
		
			
				|  |  |  |  |     if (send_in_progress) { | 
		
	
		
			
				|  |  |  |  |       if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |         gCatena.SafePrintf("looks like we timed out waiting for sending to finish...\n", wait_time); | 
		
	
		
			
				|  |  |  |  |       } | 
		
	
		
			
				|  |  |  |  |       LMIC_clrTxData(); | 
		
	
		
			
				|  |  |  |  |       // we sleep 10 seconds...
 | 
		
	
		
			
				|  |  |  |  |       start_time = millis(); | 
		
	
		
			
				|  |  |  |  |       while ((millis() - start_time) < 10000) | 
		
	
		
			
				|  |  |  |  |       { | 
		
	
		
			
				|  |  |  |  |         gCatena.poll(); | 
		
	
		
			
				|  |  |  |  |         yield(); | 
		
	
		
			
				|  |  |  |  |       } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |       send_in_progress = false; | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     wait_time = (uint32_t)((millis() - start_time) / 1000); | 
		
	
	
		
			
				
					|  |  |  | @ -717,6 +747,9 @@ void StartNewIteration() { | 
		
	
		
			
				|  |  |  |  |   if (not(next_package_is_init_package)) { | 
		
	
		
			
				|  |  |  |  |     // we make the current sensor reading to the last one...
 | 
		
	
		
			
				|  |  |  |  |     last_sensor_reading = current_sensor_reading; | 
		
	
		
			
				|  |  |  |  |   } else { | 
		
	
		
			
				|  |  |  |  |     // we only copy the last weight
 | 
		
	
		
			
				|  |  |  |  |     last_sensor_reading.weight = current_sensor_reading.weight; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   uint32_t sleep_time_sec; | 
		
	
	
		
			
				
					|  |  |  | @ -727,9 +760,9 @@ void StartNewIteration() { | 
		
	
		
			
				|  |  |  |  |     sleep_time_sec = 5; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   // for the first <INIT_PACKETS> iterations, we set the sleep time to 10 seconds only...
 | 
		
	
		
			
				|  |  |  |  |   // for the first <INIT_PACKETS> iterations, we set the sleep time to 120 seconds only...
 | 
		
	
		
			
				|  |  |  |  |   if (iteration <= INIT_PACKETS) { | 
		
	
		
			
				|  |  |  |  |     sleep_time_sec = 10; | 
		
	
		
			
				|  |  |  |  |     sleep_time_sec = 120; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (config_data.debug_level > 0) { | 
		
	
	
		
			
				
					|  |  |  | @ -745,18 +778,31 @@ void StartNewIteration() { | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (!fUsbPower) { | 
		
	
		
			
				|  |  |  |  |     DoDeepSleep(sleep_time_sec); | 
		
	
		
			
				|  |  |  |  |     if (! stop_iterations) { | 
		
	
		
			
				|  |  |  |  |       StartNewIteration(); | 
		
	
		
			
				|  |  |  |  |   // if we need to periodically reboot, we can do it now...
 | 
		
	
		
			
				|  |  |  |  |   if (uint32_t(millis()) > gRebootMs) { | 
		
	
		
			
				|  |  |  |  |     // time to reboot
 | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("Reached threshold to reboot...\n"); | 
		
	
		
			
				|  |  |  |  |       Serial.flush(); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     NVIC_SystemReset(); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("LMIC.opmode just before Sleeping: %#x\n", LMIC.opmode); | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("LMIC.globalDutyRate: %d, LMIC.globalDutyAvail: %d, os_getTime: %d\n", LMIC.globalDutyRate, LMIC.globalDutyAvail, os_getTime()); | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("LMIC.seqnoUp: %d, LMIC.seqnoDn: %d\n", LMIC.seqnoUp, LMIC.seqnoDn); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (!fUsbPower && !stop_iterations)  { | 
		
	
		
			
				|  |  |  |  |     DoDeepSleep(sleep_time_sec); | 
		
	
		
			
				|  |  |  |  |     if (! stop_iterations) { | 
		
	
		
			
				|  |  |  |  |       start_new_iteration = true; | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |     //os_setTimedCallback(
 | 
		
	
		
			
				|  |  |  |  |     //  &iterationJob,
 | 
		
	
		
			
				|  |  |  |  |     //  os_getTime() + sec2osticks(2),
 | 
		
	
		
			
				|  |  |  |  |     //  startNewIterationCb);
 | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  |   else { | 
		
	
		
			
				|  |  |  |  |     if (! stop_iterations) { | 
		
	
		
			
				|  |  |  |  |       if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |         gCatena.SafePrintf("light sleep; os_setTimedCallback for startNewIterationCb in %d...seconds\n", sleep_time_sec); | 
		
	
		
			
				|  |  |  |  |       } | 
		
	
	
		
			
				
					|  |  |  | @ -766,8 +812,9 @@ void StartNewIteration() { | 
		
	
		
			
				|  |  |  |  |         startNewIterationCb); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | void startSendingUplink(bool firstTime) | 
		
	
		
			
				|  |  |  |  | void startSendingUplink(bool firstTime, bool confirmed) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   send_in_progress = true; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
	
		
			
				
					|  |  |  | @ -791,24 +838,50 @@ void startSendingUplink(bool firstTime) | 
		
	
		
			
				|  |  |  |  |     fConfirmed = true; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   os_runloop_once(); | 
		
	
		
			
				|  |  |  |  |   //gCatena.poll();
 | 
		
	
		
			
				|  |  |  |  |   // we can overwrite fConfirmed
 | 
		
	
		
			
				|  |  |  |  |   if (confirmed) { | 
		
	
		
			
				|  |  |  |  |     fConfirmed = true; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (firstTime) { | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("SendBuffer firstTime\n"); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     gLoRaWAN.SendBuffer((uint8_t*)&lora_data_first, sizeof(LORA_data_first), sendBufferDoneCb, NULL, fConfirmed, kUplinkPort); | 
		
	
		
			
				|  |  |  |  |     if (gLoRaWAN.SendBuffer((uint8_t*)&lora_data_first, sizeof(LORA_data_first), sendBufferDoneCb, NULL, fConfirmed, kUplinkPort)) { | 
		
	
		
			
				|  |  |  |  |       package_counter++; | 
		
	
		
			
				|  |  |  |  |       if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |         gCatena.SafePrintf("LMIC.opmode just after SendBuffer (successful): %#x\n", LMIC.opmode); | 
		
	
		
			
				|  |  |  |  |         gCatena.SafePrintf("LMIC.globalDutyRate: %d, LMIC.globalDutyAvail: %d, os_getTime: %d\n", LMIC.globalDutyRate, LMIC.globalDutyAvail, os_getTime()); | 
		
	
		
			
				|  |  |  |  |         gCatena.SafePrintf("LMIC.seqnoUp: %d, LMIC.seqnoDn: %d\n", LMIC.seqnoUp, LMIC.seqnoDn); | 
		
	
		
			
				|  |  |  |  |       } | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     else { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("LMIC.opmode just before SendBuffer (failed): %#x\n", LMIC.opmode); | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("LMIC.globalDutyRate: %d, LMIC.globalDutyAvail: %d, os_getTime: %d\n", LMIC.globalDutyRate, LMIC.globalDutyAvail, os_getTime()); | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("LMIC.seqnoUp: %d, LMIC.seqnoDn: %d\n", LMIC.seqnoUp, LMIC.seqnoDn); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |   } else { | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("LMIC.opmode just before SendBuffer: %#x\n", LMIC.opmode); | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("LMIC.globalDutyRate: %d, LMIC.globalDutyAvail: %d, os_getTime: %d\n", LMIC.globalDutyRate, LMIC.globalDutyAvail, os_getTime()); | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("LMIC.seqnoUp: %d, LMIC.seqnoDn: %d\n", LMIC.seqnoUp, LMIC.seqnoDn); | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("SendBuffer not firstTime\n"); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     gLoRaWAN.SendBuffer((uint8_t*)&lora_data, sizeof(LORA_data), sendBufferDoneCb, NULL, fConfirmed, kUplinkPort); | 
		
	
		
			
				|  |  |  |  |     if (gLoRaWAN.SendBuffer((uint8_t*)&lora_data, sizeof(LORA_data), sendBufferDoneCb, NULL, fConfirmed, kUplinkPort)) { | 
		
	
		
			
				|  |  |  |  |       package_counter++; | 
		
	
		
			
				|  |  |  |  |       if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |         gCatena.SafePrintf("LMIC.opmode just after SendBuffer (successful): %#x\n", LMIC.opmode); | 
		
	
		
			
				|  |  |  |  |         gCatena.SafePrintf("LMIC.globalDutyRate: %d, LMIC.globalDutyAvail: %d, os_getTime: %d\n", LMIC.globalDutyRate, LMIC.globalDutyAvail, os_getTime()); | 
		
	
		
			
				|  |  |  |  |         gCatena.SafePrintf("LMIC.seqnoUp: %d, LMIC.seqnoDn: %d\n", LMIC.seqnoUp, LMIC.seqnoDn); | 
		
	
		
			
				|  |  |  |  |       } | 
		
	
		
			
				|  |  |  |  |     } else { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("LMIC.opmode just before SendBuffer (failed): %#x\n", LMIC.opmode); | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("LMIC.globalDutyRate: %d, LMIC.globalDutyAvail: %d, os_getTime: %d\n", LMIC.globalDutyRate, LMIC.globalDutyAvail, os_getTime()); | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("LMIC.seqnoUp: %d, LMIC.seqnoDn: %d\n", LMIC.seqnoUp, LMIC.seqnoDn); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   ClearLoraData(); | 
		
	
		
			
				|  |  |  |  |   ClearLoraData(false); | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | static void sendBufferDoneCb( | 
		
	
	
		
			
				
					|  |  |  | @ -819,12 +892,12 @@ static void sendBufferDoneCb( | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (config_data.debug_level > 1) { | 
		
	
		
			
				|  |  |  |  |     gLed.Set(LedPattern::Settling); | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("LMIC.opmode in sendBufferDoneCb: %#x\n", LMIC.opmode); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   pFn = settleDoneCb; | 
		
	
		
			
				|  |  |  |  |   if (! fStatus) | 
		
	
		
			
				|  |  |  |  |   { | 
		
	
		
			
				|  |  |  |  |     send_in_progress = false; | 
		
	
		
			
				|  |  |  |  |     if (!gLoRaWAN.IsProvisioned()) | 
		
	
		
			
				|  |  |  |  |     { | 
		
	
		
			
				|  |  |  |  |       // we'll talk about it at the callback.
 | 
		
	
	
		
			
				
					|  |  |  | @ -834,7 +907,7 @@ static void sendBufferDoneCb( | 
		
	
		
			
				|  |  |  |  |       gLoRaWAN.Shutdown(); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     else if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("send buffer failed\n"); | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("send buffer failed, LMIC.opmode: %#x\n", LMIC.opmode); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
	
		
			
				
					|  |  |  | @ -858,180 +931,56 @@ static void txNotProvisionedCb( | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | static void settleDoneCb( | 
		
	
		
			
				|  |  |  |  |   osjob_t* pSendJob) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   const bool fDeepSleep = checkDeepSleep(); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("settleDoneCb\n"); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (config_data.debug_level > 2) { | 
		
	
		
			
				|  |  |  |  |     // Terry vv
 | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("LMIC.rxDelay: %i\n", LMIC.rxDelay); | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("LMIC.dn2Dr: %i\n", LMIC.dn2Dr); | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("LMIC.dn2Freq: %i\n", LMIC.dn2Freq); | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("LMIC.rx1DrOffset: %i\n", LMIC.rx1DrOffset); | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("LMIC.adrAckReq: %i\n", LMIC.adrAckReq); | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("LMIC.adrEnabled: %i\n", LMIC.adrEnabled); | 
		
	
		
			
				|  |  |  |  |     // Terry ^^
 | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (uint32_t(millis()) > gRebootMs) { | 
		
	
		
			
				|  |  |  |  |     // time to reboot
 | 
		
	
		
			
				|  |  |  |  |     NVIC_SystemReset(); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (! g_fPrintedSleeping) | 
		
	
		
			
				|  |  |  |  |     doSleepAlert(fDeepSleep); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   /* count what we're up to */ | 
		
	
		
			
				|  |  |  |  |   updateSleepCounters(); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (fDeepSleep) | 
		
	
		
			
				|  |  |  |  |     doDeepSleep(pSendJob); | 
		
	
		
			
				|  |  |  |  |   else | 
		
	
		
			
				|  |  |  |  |     doLightSleep(pSendJob); | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | bool checkDeepSleep(void) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   bool const fDeepSleepTest = gCatena.GetOperatingFlags() & | 
		
	
		
			
				|  |  |  |  |                               static_cast<uint32_t>(gCatena.OPERATING_FLAGS::fDeepSleepTest); | 
		
	
		
			
				|  |  |  |  |   bool fDeepSleep; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (fDeepSleepTest) | 
		
	
		
			
				|  |  |  |  |   { | 
		
	
		
			
				|  |  |  |  |     fDeepSleep = true; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | #ifdef USBCON | 
		
	
		
			
				|  |  |  |  |   else if (Serial.dtr()) | 
		
	
		
			
				|  |  |  |  |   { | 
		
	
		
			
				|  |  |  |  |     fDeepSleep = false; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | #endif | 
		
	
		
			
				|  |  |  |  |   else if (gCatena.GetOperatingFlags() & | 
		
	
		
			
				|  |  |  |  |            static_cast<uint32_t>(gCatena.OPERATING_FLAGS::fDisableDeepSleep)) | 
		
	
		
			
				|  |  |  |  |   { | 
		
	
		
			
				|  |  |  |  |     fDeepSleep = false; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  |   else if ((gCatena.GetOperatingFlags() & | 
		
	
		
			
				|  |  |  |  |             static_cast<uint32_t>(gCatena.OPERATING_FLAGS::fUnattended)) != 0) | 
		
	
		
			
				|  |  |  |  |   { | 
		
	
		
			
				|  |  |  |  |     fDeepSleep = true; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  |   else | 
		
	
		
			
				|  |  |  |  |   { | 
		
	
		
			
				|  |  |  |  |     fDeepSleep = false; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   return fDeepSleep; | 
		
	
		
			
				|  |  |  |  |   return !fUsbPower; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | void doSleepAlert(const bool fDeepSleep) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   g_fPrintedSleeping = true; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (fDeepSleep) | 
		
	
		
			
				|  |  |  |  |   { | 
		
	
		
			
				|  |  |  |  |     bool const fDeepSleepTest = gCatena.GetOperatingFlags() & | 
		
	
		
			
				|  |  |  |  |                                 static_cast<uint32_t>(gCatena.OPERATING_FLAGS::fDeepSleepTest); | 
		
	
		
			
				|  |  |  |  |     const uint32_t deepSleepDelay = fDeepSleepTest ? 10 : 30; | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 2) { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("using deep sleep in %u secs" | 
		
	
		
			
				|  |  |  |  | #ifdef USBCON | 
		
	
		
			
				|  |  |  |  |                          " (USB will disconnect while asleep)" | 
		
	
		
			
				|  |  |  |  | #endif | 
		
	
		
			
				|  |  |  |  |                          ": ", | 
		
	
		
			
				|  |  |  |  |                          deepSleepDelay | 
		
	
		
			
				|  |  |  |  |                         ); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     // sleep and print
 | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 1) { | 
		
	
		
			
				|  |  |  |  |       gLed.Set(LedPattern::TwoShort); | 
		
	
		
			
				|  |  |  |  |   if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("We wait until is is safe to go to sleep...\n"); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |     for (auto n = deepSleepDelay; n > 0; --n) | 
		
	
		
			
				|  |  |  |  |   for (int i = 0; i <= 15; i++) { | 
		
	
		
			
				|  |  |  |  |     unsigned long prevPrint = millis(); | 
		
	
		
			
				|  |  |  |  |     while (os_queryTimeCriticalJobs(ms2osticks(2000)) != 0) | 
		
	
		
			
				|  |  |  |  |     { | 
		
	
		
			
				|  |  |  |  |       uint32_t tNow = millis(); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |       while (uint32_t(millis() - tNow) < 1000) | 
		
	
		
			
				|  |  |  |  |       { | 
		
	
		
			
				|  |  |  |  |         os_runloop_once(); | 
		
	
		
			
				|  |  |  |  |         //gCatena.poll();
 | 
		
	
		
			
				|  |  |  |  |         //yield();
 | 
		
	
		
			
				|  |  |  |  |       } | 
		
	
		
			
				|  |  |  |  |       if (config_data.debug_level > 2) { | 
		
	
		
			
				|  |  |  |  |         gCatena.SafePrintf("."); | 
		
	
		
			
				|  |  |  |  |       gCatena.poll(); | 
		
	
		
			
				|  |  |  |  |       yield(); | 
		
	
		
			
				|  |  |  |  |       if (millis() - prevPrint > 1000) { | 
		
	
		
			
				|  |  |  |  |         prevPrint = millis(); | 
		
	
		
			
				|  |  |  |  |         if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |           gCatena.SafePrintf("LMIC.opmode: %#x in loop %d\n", LMIC.opmode, i); | 
		
	
		
			
				|  |  |  |  |         } | 
		
	
		
			
				|  |  |  |  |       } | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 2) { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("\nStarting deep sleep.\n"); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     uint32_t tNow = millis(); | 
		
	
		
			
				|  |  |  |  |     while (uint32_t(millis() - tNow) < 100) | 
		
	
		
			
				|  |  |  |  |     { | 
		
	
		
			
				|  |  |  |  |       os_runloop_once(); | 
		
	
		
			
				|  |  |  |  |       //gCatena.poll();
 | 
		
	
		
			
				|  |  |  |  |       //yield();
 | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  |   else if (config_data.debug_level > 2) { | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("using light sleep\n"); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | void updateSleepCounters(void) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   // update the sleep parameters
 | 
		
	
		
			
				|  |  |  |  |   if (gTxCycleCount > 1) | 
		
	
		
			
				|  |  |  |  |   { | 
		
	
		
			
				|  |  |  |  |     // values greater than one are decremented and ultimately reset to default.
 | 
		
	
		
			
				|  |  |  |  |     --gTxCycleCount; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  |   else if (gTxCycleCount == 1) | 
		
	
		
			
				|  |  |  |  |   { | 
		
	
		
			
				|  |  |  |  |     // it's now one (otherwise we couldn't be here.)
 | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 2) { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("resetting tx cycle to default: %u\n", CATCFG_T_CYCLE); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |     gTxCycleCount = 0; | 
		
	
		
			
				|  |  |  |  |     gTxCycle = CATCFG_T_CYCLE; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  |   else | 
		
	
		
			
				|  |  |  |  |   { | 
		
	
		
			
				|  |  |  |  |     // it's zero. Leave it alone.
 | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | void doDeepSleep(osjob_t *pJob) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   bool const fDeepSleepTest = gCatena.GetOperatingFlags() & | 
		
	
		
			
				|  |  |  |  |                               static_cast<uint32_t>(gCatena.OPERATING_FLAGS::fDeepSleepTest); | 
		
	
		
			
				|  |  |  |  |   uint32_t const sleepInterval = CATCFG_GetInterval( | 
		
	
		
			
				|  |  |  |  |                                    fDeepSleepTest ? CATCFG_T_CYCLE_TEST : gTxCycle | 
		
	
		
			
				|  |  |  |  |                                  ); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("doDeepSleep, sleepInterval: %d...\n", sleepInterval); | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("Now it is safe to go to sleep\n"); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   /* ok... now it's time for a deep sleep */ | 
		
	
		
			
				|  |  |  |  |   gLed.Set(LedPattern::Off); | 
		
	
		
			
				|  |  |  |  |   deepSleepPrepare(); | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   /* sleep */ | 
		
	
		
			
				|  |  |  |  |   gCatena.Sleep(sleepInterval); | 
		
	
		
			
				|  |  |  |  | static void settleDoneCb( | 
		
	
		
			
				|  |  |  |  |   osjob_t* pSendJob) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("settleDoneCb - we are at the end of the callback chain!\n"); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   /* recover from sleep */ | 
		
	
		
			
				|  |  |  |  |   deepSleepRecovery(); | 
		
	
		
			
				|  |  |  |  |   const bool fDeepSleep = checkDeepSleep(); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   /* and now... we're awake again. trigger another measurement */ | 
		
	
		
			
				|  |  |  |  |   sleepDoneCb(pJob); | 
		
	
		
			
				|  |  |  |  |   if (uint32_t(millis()) > gRebootMs) | 
		
	
		
			
				|  |  |  |  |   { | 
		
	
		
			
				|  |  |  |  |     // time to reboot
 | 
		
	
		
			
				|  |  |  |  |     NVIC_SystemReset(); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   doSleepAlert(fDeepSleep); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   send_in_progress = false; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | void deepSleepPrepare(void) | 
		
	
	
		
			
				
					|  |  |  | @ -1053,53 +1002,6 @@ void deepSleepRecovery(void) | 
		
	
		
			
				|  |  |  |  |     gSPI2.begin(); | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | void doLightSleep(osjob_t *pJob) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   uint32_t interval = sec2osticks(CATCFG_GetInterval(gTxCycle)); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (config_data.debug_level > 1) { | 
		
	
		
			
				|  |  |  |  |     gLed.Set(LedPattern::Sleeping); | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("doLightSleep\n"); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (gCatena.GetOperatingFlags() & | 
		
	
		
			
				|  |  |  |  |       static_cast<uint32_t>(gCatena.OPERATING_FLAGS::fQuickLightSleep)) | 
		
	
		
			
				|  |  |  |  |   { | 
		
	
		
			
				|  |  |  |  |     interval = 1; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   os_setTimedCallback( | 
		
	
		
			
				|  |  |  |  |     &iterationJob, | 
		
	
		
			
				|  |  |  |  |     os_getTime() + interval, | 
		
	
		
			
				|  |  |  |  |     sleepDoneCb | 
		
	
		
			
				|  |  |  |  |   ); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | static void sleepDoneCb(osjob_t* pJob) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   if (config_data.debug_level > 1) { | 
		
	
		
			
				|  |  |  |  |     gLed.Set(LedPattern::WarmingUp); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("sleepDoneCb\n"); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   os_setTimedCallback( | 
		
	
		
			
				|  |  |  |  |     pJob, | 
		
	
		
			
				|  |  |  |  |     os_getTime() + sec2osticks(CATCFG_T_WARMUP), | 
		
	
		
			
				|  |  |  |  |     warmupDoneCb); | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | static void warmupDoneCb(osjob_t* pJob) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("warmupDoneCb\n"); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  |   send_in_progress = false; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | static void startNewIterationCb(osjob_t* pJob) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
	
		
			
				
					|  |  |  | @ -1108,14 +1010,12 @@ static void startNewIterationCb(osjob_t* pJob) | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (! stop_iterations) { | 
		
	
		
			
				|  |  |  |  |     StartNewIteration(); | 
		
	
		
			
				|  |  |  |  |     start_new_iteration = true; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | static void receiveMessage(void *pContext, uint8_t port, const uint8_t *pMessage, size_t nMessage) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   unsigned txCycle; | 
		
	
		
			
				|  |  |  |  |   unsigned txCount; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   long cal_w1_0; | 
		
	
		
			
				|  |  |  |  |   long cal_w2_0; | 
		
	
	
		
			
				
					|  |  |  | @ -1231,59 +1131,8 @@ static void receiveMessage(void *pContext, uint8_t port, const uint8_t *pMessage | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (port == 0) | 
		
	
		
			
				|  |  |  |  |   { | 
		
	
		
			
				|  |  |  |  |     return; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   else if (! (port == 1 && 2 <= nMessage && nMessage <= 3)) | 
		
	
		
			
				|  |  |  |  |   { | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("invalid message port(%02x)/length(%x)\n", | 
		
	
		
			
				|  |  |  |  |                          port, nMessage | 
		
	
		
			
				|  |  |  |  |                         ); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     return; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   txCycle = (pMessage[0] << 8) | pMessage[1]; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (txCycle < CATCFG_T_MIN || txCycle > CATCFG_T_MAX) | 
		
	
		
			
				|  |  |  |  |   { | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("tx cycle time out of range: %u\n", txCycle); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |     return; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   // byte [2], if present, is the repeat count.
 | 
		
	
		
			
				|  |  |  |  |   // explicitly sending zero causes it to stick.
 | 
		
	
		
			
				|  |  |  |  |   txCount = CATCFG_INTERVAL_COUNT; | 
		
	
		
			
				|  |  |  |  |   if (nMessage >= 3) | 
		
	
		
			
				|  |  |  |  |   { | 
		
	
		
			
				|  |  |  |  |     txCount = pMessage[2]; | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   setTxCycleTime(txCycle, txCount); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | void setTxCycleTime(unsigned txCycle, unsigned txCount) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   if (txCount > 0) { | 
		
	
		
			
				|  |  |  |  |     if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |       gCatena.SafePrintf("message cycle time %u seconds for %u messages\n", txCycle, txCount); | 
		
	
		
			
				|  |  |  |  |     } | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  |   else if (config_data.debug_level > 0) { | 
		
	
		
			
				|  |  |  |  |     gCatena.SafePrintf("message cycle time %u seconds indefinitely\n", txCycle); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   gTxCycle = txCycle; | 
		
	
		
			
				|  |  |  |  |   gTxCycleCount = txCount; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | /* process "application hello" -- args are ignored */ | 
		
	
		
			
				|  |  |  |  | // argv[0] is "hello"
 | 
		
	
		
			
				|  |  |  |  | // argv[1..argc-1] are the (ignored) arguments
 | 
		
	
	
		
			
				
					|  |  |  | @ -1294,7 +1143,6 @@ cCommandStream::CommandStatus cmdHello(cCommandStream * pThis, void *pContext, i | 
		
	
		
			
				|  |  |  |  |   return cCommandStream::CommandStatus::kSuccess; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | cCommandStream::CommandStatus cmdGetCalibrationSettings(cCommandStream * pThis, void *pContext, int argc, char **argv) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   pThis->printf("{\n"); | 
		
	
	
		
			
				
					|  |  |  | @ -1359,20 +1207,33 @@ cCommandStream::CommandStatus cmdCalibrateZeroScaleB(cCommandStream * pThis, voi | 
		
	
		
			
				|  |  |  |  |   return cCommandStream::CommandStatus::kSuccess; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | cCommandStream::CommandStatus cmdCalibrateScales(cCommandStream * pThis, void *pContext, int argc, char **argv) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   String s_cal_w1_0(argv[1]); | 
		
	
		
			
				|  |  |  |  |   String s_cal_w1_factor(argv[2]); | 
		
	
		
			
				|  |  |  |  |   String s_cal_w2_0(argv[3]); | 
		
	
		
			
				|  |  |  |  |   String s_cal_w2_factor(argv[4]); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   config_data.cal_w1_0 = s_cal_w1_0.toInt(); | 
		
	
		
			
				|  |  |  |  |   config_data.cal_w1_factor = s_cal_w1_factor.toFloat(); | 
		
	
		
			
				|  |  |  |  |   config_data.cal_w2_0 = s_cal_w2_0.toInt(); | 
		
	
		
			
				|  |  |  |  |   config_data.cal_w2_factor = s_cal_w2_factor.toFloat(); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   gCatena.getFram()->saveField(cFramStorage::kAppConf, (const uint8_t *)&config_data, sizeof(config_data)); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   pThis->printf("{ \"msg\": \"calibrate_scales was successful\" }\n"); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   return cCommandStream::CommandStatus::kSuccess; | 
		
	
		
			
				|  |  |  |  | } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  | cCommandStream::CommandStatus cmdCalibrateScaleA(cCommandStream * pThis, void *pContext, int argc, char **argv) | 
		
	
		
			
				|  |  |  |  | { | 
		
	
		
			
				|  |  |  |  |   String w1_gramm(argv[1]); | 
		
	
		
			
				|  |  |  |  |   long weight1; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (w1_gramm == "NA") { | 
		
	
		
			
				|  |  |  |  |     // scale a is not connected
 | 
		
	
		
			
				|  |  |  |  |     config_data.cal_w1_factor = 1.0; | 
		
	
		
			
				|  |  |  |  |     config_data.cal_w1_0 = NOT_ATTACHED; | 
		
	
		
			
				|  |  |  |  |   } else { | 
		
	
		
			
				|  |  |  |  |   setup_scales(); | 
		
	
		
			
				|  |  |  |  |   weight1 = ReadScale('A'); | 
		
	
		
			
				|  |  |  |  |   config_data.cal_w1_factor = (float)((weight1 - config_data.cal_w1_0) / w1_gramm.toFloat()); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   gCatena.getFram()->saveField(cFramStorage::kAppConf, (const uint8_t *)&config_data, sizeof(config_data)); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
	
		
			
				
					|  |  |  | @ -1386,15 +1247,9 @@ cCommandStream::CommandStatus cmdCalibrateScaleB(cCommandStream * pThis, void *p | 
		
	
		
			
				|  |  |  |  |   String w2_gramm(argv[1]); | 
		
	
		
			
				|  |  |  |  |   long weight2; | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   if (w2_gramm == "NA") { | 
		
	
		
			
				|  |  |  |  |     // scale b is not connected
 | 
		
	
		
			
				|  |  |  |  |     config_data.cal_w2_factor = 1.0; | 
		
	
		
			
				|  |  |  |  |     config_data.cal_w2_0 = NOT_ATTACHED; | 
		
	
		
			
				|  |  |  |  |   } else { | 
		
	
		
			
				|  |  |  |  |   setup_scales(); | 
		
	
		
			
				|  |  |  |  |   weight2 = ReadScale('B'); | 
		
	
		
			
				|  |  |  |  |   config_data.cal_w2_factor = (float)((weight2 - config_data.cal_w2_0) / w2_gramm.toFloat()); | 
		
	
		
			
				|  |  |  |  |   } | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
		
			
				|  |  |  |  |   gCatena.getFram()->saveField(cFramStorage::kAppConf, (const uint8_t *)&config_data, sizeof(config_data)); | 
		
	
		
			
				|  |  |  |  | 
 | 
		
	
	
		
			
				
					|  |  |  | 
 |