Rewrite/modularize the model checker

- Allow all parameters to be changed at runtime through M310
- Move the model prototypes into a separate temp_model.h header
- Allow the checked to be enabled/disabled at runtime
- Introduce a warning threshold
This commit is contained in:
Yuri D'Elia 2022-06-24 16:04:00 +02:00
parent fabf511b97
commit b0b2ff5f9e
5 changed files with 462 additions and 147 deletions

View File

@ -7761,6 +7761,54 @@ Sigma_Exit:
}
break;
/*!
### M310 - Temperature model
#### Usage
M310 [ C ] [ P ] [ I R ] [ S ] [ E ] [ W ] [ A ] [ T ]
#### Parameters
- `P` - power
- `C` - capacitance
- `I` - resistance index position
- `R` - resistance value (requires `I`)
- `S` - set 0=disable 1=enable (default)
- `E` - error threshold (define min/max values in variants)
- `W` - warning threshold (define min/max values in variants)
- `T` - ambient temperature correction
- `A` - autotune C+R values
*/
case 310:
{
// parse all parameters
float P = NAN, C = NAN, R = NAN, E = NAN, W = NAN, T = NAN, A = NAN;
int8_t I = -1, S = -1;
if(code_seen('C')) C = code_value();
if(code_seen('P')) P = code_value();
if(code_seen('I')) I = code_value_short();
if(code_seen('R')) R = code_value();
if(code_seen('S')) S = code_value_short();
if(code_seen('E')) E = code_value();
if(code_seen('W')) W = code_value();
if(code_seen('T')) T = code_value();
if(code_seen('A')) A = code_value();
// report values if nothing has been requested
if(isnan(C) && isnan(P) && isnan(R) && isnan(E) && isnan(W) && isnan(T) && isnan(A) && I < 0 && S < 0) {
temp_model_report_settings();
break;
}
// update all set parameters
if(S >= 0) temp_model_set_enabled(S);
if(!isnan(C) || !isnan(P) || !isnan(T) || !isnan(W) || !isnan(E)) temp_model_set_params(C, P, T, W, E);
if(I >= 0 && !isnan(R)) temp_model_set_resistance(I, R);
// run autotune
if(!isnan(A)) temp_model_autotune(A != 0? A: NAN);
}
break;
/*!
### M400 - Wait for all moves to finish <a href="https://reprap.org/wiki/G-code#M400:_Wait_for_current_moves_to_finish">M400: Wait for current moves to finish</a>
Finishes all current moves and and thus clears the buffer.
@ -9144,7 +9192,16 @@ Sigma_Exit:
};
#endif
#ifdef TEMP_MODEL_LOGGING
#ifdef TEMP_MODEL_DEBUG
/*!
## D70 - Enable low-level temperature model logging for offline simulation
#### Usage
D70 [ I ]
#### Parameters
- `I` - Enable 0-1 (default 0)
*/
case 70: {
if(code_seen('I'))
temp_model_log_enable(code_value_short());

110
Firmware/temp_model.h Normal file
View File

@ -0,0 +1,110 @@
// model-based temperature safety checker declarations
#ifndef TEMP_MGR_INTV
#error "this file is not a public interface, it should be used *only* within temperature.cpp!"
#endif
#include "planner.h"
constexpr uint8_t TEMP_MODEL_CAL_S = 60; // Maximum recording lenght during calibration (s)
constexpr float TEMP_MODEL_fS = 0.065; // simulation filter (1st-order IIR factor)
constexpr float TEMP_MODEL_fE = 0.05; // error filter (1st-order IIR factor)
// transport delay buffer size (samples)
constexpr uint8_t TEMP_MODEL_LAG_SIZE = (TEMP_MODEL_LAG / TEMP_MGR_INTV + 0.5);
// resistance values for all fan levels
constexpr uint8_t TEMP_MODEL_R_SIZE = (1 << FAN_SOFT_PWM_BITS);
namespace temp_model {
// recording scratch buffer
struct rec_entry
{
float temp; // heater temperature
uint8_t pwm; // heater PWM
};
constexpr uint16_t rec_buffer_size = TEMP_MODEL_CAL_S / TEMP_MGR_INTV;
static rec_entry* const rec_buffer = (rec_entry*)block_buffer; // oh-hey, free memory!
static_assert(sizeof(rec_entry[rec_buffer_size]) <= sizeof(block_buffer),
"recording length too long to fit within available buffer");
struct model_data
{
// temporary buffers
float dT_lag_buf[TEMP_MODEL_LAG_SIZE]; // transport delay buffer
uint8_t dT_lag_idx = 0; // transport delay buffer index
float dT_err_prev = 0; // previous temperature delta error
float T_prev = 0; // last temperature extruder
// configurable parameters
float P; // heater power (W)
float C; // heatblock capacitance (J/K)
float R[TEMP_MODEL_R_SIZE]; // heatblock resistance for all fan levels (K/W)
float Ta_corr; // ambient temperature correction (K)
// thresholds
float warn; // pre-warning threshold (K/s)
float err; // error threshold (K/s)
// status flags
union
{
bool flags;
struct
{
bool uninitialized: 1; // model is not initialized
bool error: 1; // error threshold set
bool warning: 1; // warning threshold set
} flag_bits;
};
// pre-computed values (initialized via reset)
float C_i; // heatblock capacitance (precomputed dT/C)
float warn_s; // pre-warning threshold (per sample)
float err_s; // error threshold (per sample)
// simulation functions
void reset(uint8_t heater_pwm, uint8_t fan_pwm, float heater_temp, float ambient_temp);
void step(uint8_t heater_pwm, uint8_t fan_pwm, float heater_temp, float ambient_temp);
};
static bool enabled; // model check enabled
static model_data data; // default heater data
static void init(); // initialize and setup the model subsystem
static bool calibrated(); // return calibration/model validity status
static void check(); // check and trigger errors or warnings based on current state
// warning state (updated from from isr context)
volatile static struct
{
float dT_err; // temperature delta error (per sample)
bool warning: 1; // warning condition
bool assert: 1; // warning is still asserted
} warning_state;
static void handle_warning(); // handle warnings from user context
#ifdef TEMP_MODEL_DEBUG
static struct
{
volatile struct
{
uint32_t stamp;
int8_t delta_ms;
uint8_t counter;
uint8_t cur_pwm;
float cur_temp;
float cur_amb;
} entry;
uint8_t serial;
bool enabled;
} log_buf;
static void log_usr(); // user log handler
static void log_isr(); // isr log handler
#endif
} // namespace temp_model

View File

@ -48,6 +48,19 @@
#error "ADC_OVRSAMPL oversampling must match OVERSAMPLENR"
#endif
// temperature manager timer configuration
#define TEMP_MGR_INTV 0.27 // seconds, ~3.7Hz
#define TIMER5_PRESCALE 256
#define TIMER5_OCRA_OVF (uint16_t)(TEMP_MGR_INTV / ((long double)TIMER5_PRESCALE / F_CPU))
#define TEMP_MGR_INTERRUPT_STATE() (TIMSK5 & (1<<OCIE5A))
#define ENABLE_TEMP_MGR_INTERRUPT() TIMSK5 |= (1<<OCIE5A)
#define DISABLE_TEMP_MGR_INTERRUPT() TIMSK5 &= ~(1<<OCIE5A)
#ifdef TEMP_MODEL
// temperature model interface
#include "temp_model.h"
#endif
//===========================================================================
//=============================public variables============================
//===========================================================================
@ -454,7 +467,9 @@ enum class TempErrorType : uint8_t
min,
preheat,
runaway,
#ifdef TEMP_MODEL
model,
#endif
};
// error state (updated via set_temp_error from isr context)
@ -477,14 +492,8 @@ volatile static union
void set_temp_error(TempErrorSource source, uint8_t index, TempErrorType type)
{
// keep disabling heaters and keep fans on as long as the condition is asserted
#ifdef TEMP_MODEL_CHECK_WARN_ONLY
if(type != TempErrorType::model) {
#endif
disable_heater();
hotendFanSetFullSpeed();
#ifdef TEMP_MODEL_CHECK_WARN_ONLY
}
#endif
// set the initial error source to the highest priority error
if(!temp_error_state.error || (uint8_t)type < temp_error_state.type) {
@ -500,11 +509,6 @@ void set_temp_error(TempErrorSource source, uint8_t index, TempErrorType type)
void handle_temp_error();
#ifdef TEMP_MODEL_LOGGING
static void temp_model_log_usr();
static void temp_model_log_isr();
#endif
void manage_heater()
{
#ifdef WATCHDOG
@ -519,6 +523,12 @@ void manage_heater()
// syncronize temperatures with isr
updateTemperatures();
#ifdef TEMP_MODEL
// handle model warnings first, so not to override the error handler
if(temp_model::warning_state.warning)
temp_model::handle_warning();
#endif
// handle temperature errors
if(temp_error_state.v)
handle_temp_error();
@ -526,8 +536,8 @@ void manage_heater()
// periodically check fans
checkFans();
#ifdef TEMP_MODEL_LOGGING
temp_model_log_usr();
#ifdef TEMP_MODEL_DEBUG
temp_model::log_usr();
#endif
}
@ -1765,26 +1775,19 @@ void handle_temp_error()
break;
}
break;
#ifdef TEMP_MODEL
case TempErrorType::model:
#ifdef TEMP_MODEL_CHECK
static bool is_new = true;
static bool beep_on = false;
printf_P(PSTR("TM: err:%u ass:%u\n"), (unsigned)temp_error_state.error,
(unsigned)temp_error_state.assert);
if(is_new) {
beep_on = true;
is_new = false;
}
WRITE(BEEPER, beep_on);
beep_on = !beep_on;
if(!temp_error_state.assert) {
printf_P(PSTR("TM: assertion cleared - resetting\n"));
if(temp_error_state.assert) {
// TODO: do something meaningful
SERIAL_ECHOLNPGM("TM: error triggered!");
WRITE(BEEPER, HIGH);
} else {
temp_error_state.v = 0;
WRITE(BEEPER, LOW);
is_new = true;
SERIAL_ECHOLNPGM("TM: error cleared");
}
#endif
break;
#endif
}
}
@ -1841,13 +1844,6 @@ bool has_temperature_compensation()
#endif //PINDA_THERMISTOR
#define TEMP_MGR_INTV 0.27 // seconds, ~3.7Hz
#define TIMER5_PRESCALE 256
#define TIMER5_OCRA_OVF (uint16_t)(TEMP_MGR_INTV / ((long double)TIMER5_PRESCALE / F_CPU))
#define TEMP_MGR_INTERRUPT_STATE() (TIMSK5 & (1<<OCIE5A))
#define ENABLE_TEMP_MGR_INTERRUPT() TIMSK5 |= (1<<OCIE5A)
#define DISABLE_TEMP_MGR_INTERRUPT() TIMSK5 &= ~(1<<OCIE5A)
// RAII helper class to run a code block with temp_mgr_isr disabled
class TempMgrGuard
{
@ -1895,6 +1891,10 @@ void temp_mgr_init()
TCNT5 = 0;
OCR5A = TIMER5_OCRA_OVF;
#ifdef TEMP_MODEL
temp_model::init();
#endif
// clear pending interrupts, enable COMPA
TIFR5 |= (1<<OCF5A);
ENABLE_TEMP_MGR_INTERRUPT();
@ -2192,9 +2192,6 @@ static void check_temp_runaway()
}
static void check_temp_raw();
#ifdef TEMP_MODEL_CHECK
static void check_temp_model();
#endif
static void temp_mgr_isr()
{
@ -2205,12 +2202,11 @@ static void temp_mgr_isr()
temp_error_state.assert = false;
check_temp_raw(); // check min/max temp using raw values
check_temp_runaway(); // classic temperature hysteresis check
#ifdef TEMP_MODEL_CHECK
check_temp_model(); // model-based heater check
#ifdef TEMP_MODEL
temp_model::check(); // model-based heater check
#ifdef TEMP_MODEL_DEBUG
temp_model::log_isr();
#endif
#ifdef TEMP_MODEL_LOGGING
temp_model_log_isr();
#endif
// PID regulation
@ -2332,147 +2328,266 @@ static void check_temp_raw()
check_min_temp_raw();
}
#ifdef TEMP_MODEL_CHECK
#ifndef TEMP_MODEL_VARS
static const float TM_P = 38.; // heater power (W)
static const float TM_C = 11.9; // heatblock capacitance (J/K)
static const float TM_R = 27.; // heatblock resistance
static const float TM_Rf = -20.; // full-power fan resistance change
static const float TM_aC = -5; // ambient temperature correction (K)
#endif
#ifdef TEMP_MODEL
namespace temp_model {
static const float TM_dTl = 2.1; // temperature transport delay (s)
static const uint8_t TM_dTs = (TM_dTl / TEMP_MGR_INTV + 0.5); // temperature transport delay (samples)
static const float TM_fS = 0.065; // simulation (1st-order IIR factor)
static const float TM_fE = 0.05; // error (1st-order IIR factor)
static const float TM_err = 1.; // error threshold (K/s)
static const float TM_err_s = (TM_err * TEMP_MGR_INTV); // error threshold (per sample)
static float TM_dT_buf[TM_dTs]; // transport delay buffer
static uint8_t TM_dT_idx = 0; // transport delay buffer index
static float TM_dT_err = 0; // last error
static float TM_T = 0; // last temperature
#ifndef MAX
#define MAX(A, B) (A >= B? A: B)
#endif
// samples required for settling the model (crude approximation)
static uint8_t TM_dT_smp = MAX(TM_dTs, MAX(3/TM_fS, 3/TM_fE));
static void check_temp_model()
void model_data::reset(uint8_t heater_pwm, uint8_t fan_pwm, float heater_temp, float ambient_temp)
{
const float soft_pwm_inv = 1. / ((1 << 7) - 1);
const float soft_pwm_fan_inv = 1. / ((1 << FAN_SOFT_PWM_BITS) - 1);
// pre-compute invariant values
C_i = (TEMP_MGR_INTV / C);
warn_s = warn * TEMP_MGR_INTV;
err_s = err * TEMP_MGR_INTV;
// initial values
memset(dT_lag_buf, 0, sizeof(dT_lag_buf));
dT_lag_idx = 0;
dT_err_prev = 0;
T_prev = heater_temp;
// perform one step to initialize the first delta
step(heater_pwm, fan_pwm, heater_temp, ambient_temp);
// clear the initialization flag
flag_bits.uninitialized = false;
}
void model_data::step(uint8_t heater_pwm, uint8_t fan_pwm, float heater_temp, float ambient_temp)
{
constexpr float soft_pwm_inv = 1. / ((1 << 7) - 1);
// input values
float heater_scale = soft_pwm_inv * soft_pwm[0];
float fan_scale = soft_pwm_fan_inv * soft_pwm_fan;
float cur_temp_heater = current_temperature_isr[0];
float cur_temp_ambient = current_temperature_ambient_isr + TM_aC;
const float heater_scale = soft_pwm_inv * heater_pwm;
const float cur_heater_temp = heater_temp;
const float cur_ambient_temp = ambient_temp + Ta_corr;
const float cur_R = R[fan_pwm]; // resistance at current fan power (K/W)
// model invariants
float C_i = (TEMP_MGR_INTV / TM_C);
float dP = TM_P * heater_scale; // current power [W]
float R = TM_R + TM_Rf * fan_scale; // resistance (constant + fan modulation)
float dPl = (cur_temp_heater - cur_temp_ambient) / R; // [W] leakage power
float dP = P * heater_scale; // current power [W]
float dPl = (cur_heater_temp - cur_ambient_temp) / cur_R; // [W] leakage power
float dT = (dP - dPl) * C_i; // expected temperature difference (K)
// filter and lag dT
uint8_t next_dT_idx = (TM_dT_idx == (TM_dTs - 1) ? 0: TM_dT_idx + 1);
float lag_dT = TM_dT_buf[next_dT_idx];
float prev_dT = TM_dT_buf[TM_dT_idx];
float dTf = (prev_dT * (1. - TM_fS)) + (dT * TM_fS);
TM_dT_buf[next_dT_idx] = dTf;
TM_dT_idx = next_dT_idx;
uint8_t dT_next_idx = (dT_lag_idx == (TEMP_MODEL_LAG_SIZE - 1) ? 0: dT_lag_idx + 1);
float dT_lag = dT_lag_buf[dT_next_idx];
float dT_lag_prev = dT_lag_buf[dT_lag_idx];
float dT_f = (dT_lag_prev * (1.f - TEMP_MODEL_fS)) + (dT * TEMP_MODEL_fS);
dT_lag_buf[dT_next_idx] = dT_f;
dT_lag_idx = dT_next_idx;
// calculate and filter dT_err
float dT_err = (cur_temp_heater - TM_T) - lag_dT;
float dT_err_f = (TM_dT_err * (1. - TM_fE)) + (dT_err * TM_fE);
TM_T = cur_temp_heater;
TM_dT_err = dT_err_f;
float dT_err = (cur_heater_temp - T_prev) - dT_lag;
float dT_err_f = (dT_err_prev * (1.f - TEMP_MODEL_fE)) + (dT_err * TEMP_MODEL_fE);
T_prev = cur_heater_temp;
dT_err_prev = dT_err_f;
// check and trigger errors
if(TM_dT_smp) {
// model not ready
--TM_dT_smp;
} else {
if(dT_err_f > TM_err_s || dT_err_f < -TM_err_s)
flag_bits.error = (fabsf(dT_err_f) > err_s);
flag_bits.warning = (fabsf(dT_err_f) > warn_s);
}
// verify calibration status and trigger a model reset if valid
void setup()
{
if(!calibrated()) enabled = false;
data.flag_bits.uninitialized = true;
}
void init()
{
// TODO: initialize the model with eeprom values
data.P = TEMP_MODEL_P;
data.C = TEMP_MODEL_C;
for(uint8_t i = 0; i != TEMP_MODEL_R_SIZE; ++i)
data.R[i] = TEMP_MODEL_R;
data.Ta_corr = TEMP_MODEL_Ta_corr;
data.warn = TEMP_MODEL_W;
data.err = TEMP_MODEL_E;
enabled = true;
setup();
}
bool calibrated()
{
if(!(data.P >= 0)) return false;
if(!(data.C >= 0)) return false;
if(!(data.Ta_corr != NAN)) return false;
for(uint8_t i = 0; i != TEMP_MODEL_R_SIZE; ++i) {
if(!(temp_model::data.R[i] >= 0))
return false;
}
if(!(data.warn != NAN)) return false;
if(!(data.err != NAN)) return false;
return true;
}
void check()
{
if(!enabled) return;
uint8_t heater_pwm = soft_pwm[0];
uint8_t fan_pwm = soft_pwm_fan;
float heater_temp = current_temperature_isr[0];
float ambient_temp = current_temperature_ambient_isr;
// check if a reset is required to seed the model: this needs to be done with valid
// ADC values, so we can't do that directly in init()
if(data.flag_bits.uninitialized)
data.reset(heater_pwm, fan_pwm, heater_temp, ambient_temp);
// step the model
data.step(heater_pwm, fan_pwm, heater_temp, ambient_temp);
// handle errors
if(data.flag_bits.error)
set_temp_error(TempErrorSource::hotend, 0, TempErrorType::model);
// handle warning conditions as lower-priority but with greater feedback
warning_state.assert = data.flag_bits.warning;
if(warning_state.assert) {
warning_state.warning = true;
warning_state.dT_err = temp_model::data.dT_err_prev;
}
}
#ifdef TEMP_MODEL_LOGGING
static struct
void handle_warning()
{
volatile struct
{
uint32_t stamp;
int8_t delta_ms;
uint8_t counter;
uint8_t cur_pwm;
float cur_temp;
float cur_amb;
} entry;
uint8_t serial;
bool enabled;
} temp_model_log_buf;
static void temp_model_log_usr()
{
if(!temp_model_log_buf.enabled)
return;
uint8_t counter = temp_model_log_buf.entry.counter;
if (counter == temp_model_log_buf.serial)
return;
int8_t delta_ms;
uint8_t cur_pwm;
float cur_temp;
float cur_amb;
// update values
float warn = data.warn;
float dT_err;
{
TempMgrGuard temp_mgr_guard;
delta_ms = temp_model_log_buf.entry.delta_ms;
counter = temp_model_log_buf.entry.counter;
cur_pwm = temp_model_log_buf.entry.cur_pwm;
cur_temp = temp_model_log_buf.entry.cur_temp;
cur_amb = temp_model_log_buf.entry.cur_amb;
dT_err = warning_state.dT_err;
}
dT_err /= TEMP_MGR_INTV; // per-sample => K/s
// TODO: alert the user on the lcd
printf_P(PSTR("TM: error |%f|>%f\n"), (double)dT_err, (double)warn);
static bool beeper = false;
if(warning_state.assert) {
// beep periodically
beeper = !beeper;
WRITE(BEEPER, beeper);
} else {
// warning cleared, reset state
warning_state.warning = false;
beeper = false;
WRITE(BEEPER, LOW);
}
}
#ifdef TEMP_MODEL_DEBUG
void log_usr()
{
if(!log_buf.enabled) return;
uint8_t counter = log_buf.entry.counter;
if (counter == log_buf.serial) return;
int8_t delta_ms;
uint8_t cur_pwm;
// avoid strict-aliasing warnings
union { float cur_temp; uint32_t cur_temp_b; };
union { float cur_amb; uint32_t cur_amb_b; };
{
TempMgrGuard temp_mgr_guard;
delta_ms = log_buf.entry.delta_ms;
counter = log_buf.entry.counter;
cur_pwm = log_buf.entry.cur_pwm;
cur_temp = log_buf.entry.cur_temp;
cur_amb = log_buf.entry.cur_amb;
}
uint8_t d = counter - temp_model_log_buf.serial;
temp_model_log_buf.serial = counter;
uint8_t d = counter - log_buf.serial;
log_buf.serial = counter;
printf_P(PSTR("TML %d %d %x %lx %lx\n"), (unsigned)d - 1, (int)delta_ms + 1, (int)cur_pwm,
*(unsigned long*)&cur_temp, *(unsigned long*)&cur_amb);
printf_P(PSTR("TML %d %d %x %lx %lx\n"), (unsigned)d - 1, (int)delta_ms + 1,
(int)cur_pwm, (unsigned long)cur_temp_b, (unsigned long)cur_amb_b);
}
static void temp_model_log_isr()
void log_isr()
{
if(!temp_model_log_buf.enabled)
return;
if(!log_buf.enabled) return;
uint32_t stamp = _millis();
uint8_t delta_ms = stamp - temp_model_log_buf.entry.stamp - (TEMP_MGR_INTV * 1000);
temp_model_log_buf.entry.stamp = stamp;
uint8_t delta_ms = stamp - log_buf.entry.stamp - (TEMP_MGR_INTV * 1000);
log_buf.entry.stamp = stamp;
++temp_model_log_buf.entry.counter;
temp_model_log_buf.entry.delta_ms = delta_ms;
temp_model_log_buf.entry.cur_pwm = soft_pwm[0];
temp_model_log_buf.entry.cur_temp = current_temperature_isr[0];
temp_model_log_buf.entry.cur_amb = current_temperature_ambient_isr;
++log_buf.entry.counter;
log_buf.entry.delta_ms = delta_ms;
log_buf.entry.cur_pwm = soft_pwm[0];
log_buf.entry.cur_temp = current_temperature_isr[0];
log_buf.entry.cur_amb = current_temperature_ambient_isr;
}
#endif
} // namespace temp_model
void temp_model_set_enabled(bool enabled)
{
// set the enabled flag
{
TempMgrGuard temp_mgr_guard;
temp_model::enabled = enabled;
temp_model::setup();
}
// verify that the model has been enabled
if(enabled && !temp_model::enabled)
SERIAL_ECHOLNPGM("TM: invalid parameters, cannot enable");
}
void temp_model_set_params(float C, float P, float Ta_corr, float warn, float err)
{
TempMgrGuard temp_mgr_guard;
if(!isnan(C) && C > 0) temp_model::data.C = C;
if(!isnan(P) && P > 0) temp_model::data.P = P;
if(!isnan(Ta_corr)) temp_model::data.Ta_corr = Ta_corr;
if(!isnan(err) && err > 0) temp_model::data.err = err;
if(!isnan(warn) && warn > 0) temp_model::data.warn = warn;
// ensure warn <= err
if (temp_model::data.warn > temp_model::data.err)
temp_model::data.warn = temp_model::data.err;
temp_model::setup();
}
void temp_model_set_resistance(uint8_t index, float R)
{
if(index >= TEMP_MODEL_R_SIZE || R <= 0)
return;
TempMgrGuard temp_mgr_guard;
temp_model::data.R[index] = R;
temp_model::setup();
}
void temp_model_report_settings()
{
printf_P(PSTR("%STemperature Model settings:\n"
"%S M310 P%.2f C%.2f S%u E%.2f W%.2f T%.2f\n"),
echomagic, echomagic, (double)temp_model::data.P, (double)temp_model::data.C, (unsigned)temp_model::enabled,
(double)temp_model::data.err, (double)temp_model::data.warn, (double)temp_model::data.Ta_corr);
for(uint8_t i = 0; i != TEMP_MODEL_R_SIZE; ++i)
printf_P(PSTR("%S M310 I%u R%.2f\n"), echomagic, (unsigned)i, (double)temp_model::data.R[i]);
}
void temp_model_autotune(float temp)
{
// TODO
}
#ifdef TEMP_MODEL_DEBUG
void temp_model_log_enable(bool enable)
{
if(enable) {
TempMgrGuard temp_mgr_guard;
temp_model_log_buf.entry.stamp = _millis();
temp_model::log_buf.entry.stamp = _millis();
}
temp_model_log_buf.enabled = enable;
temp_model::log_buf.enabled = enable;
}
#endif
#endif

View File

@ -228,9 +228,18 @@ FORCE_INLINE void autotempShutdown(){
}
void PID_autotune(float temp, int extruder, int ncycles);
#ifdef TEMP_MODEL_LOGGING
#ifdef TEMP_MODEL
void temp_model_set_enabled(bool enabled);
void temp_model_set_params(float C = NAN, float P = NAN, float Ta_corr = NAN, float warn = NAN, float err = NAN);
void temp_model_set_resistance(uint8_t index, float R);
void temp_model_report_settings();
void temp_model_autotune(float temp = NAN);
#ifdef TEMP_MODEL_DEBUG
void temp_model_log_enable(bool enable);
#endif
#endif
#ifdef FAN_SOFT_PWM
extern unsigned char fanSpeedSoftPwm;

View File

@ -418,9 +418,33 @@
#define TEMP_RUNAWAY_EXTRUDER_TIMEOUT 45
// model-based temperature check
#define TEMP_MODEL_CHECK 1
#define TEMP_MODEL_CHECK_WARN_ONLY 1
#define TEMP_MODEL_LOGGING 1
#define TEMP_MODEL 1 // enable model-based temperature checks
#define TEMP_MODEL_DEBUG 1 // extended runtime logging
#define TEMP_MODEL_P 38. // heater power (W)
#define TEMP_MODEL_C 11. // initial guess for heatblock capacitance (J/K)
#define TEMP_MODEL_Cl 5 // C estimation lower limit
#define TEMP_MODEL_Ch 20 // C estimation upper limit
#define TEMP_MODEL_C_thr 0.01 // C estimation iteration threshold
#define TEMP_MODEL_C_itr 30 // C estimation iteration limit
#define TEMP_MODEL_R 25 // initial guess for heatblock resistance (K/W)
#define TEMP_MODEL_Rl 5 // R estimation lower limit
#define TEMP_MODEL_Rh 50 // R estimation upper limit
#define TEMP_MODEL_R_thr 0.01 // R estimation iteration threshold
#define TEMP_MODEL_R_itr 30 // R estimation iteration limit
#define TEMP_MODEL_Rf_D -15 // initial guess for resistance change with full-power fan
#define TEMP_MODEL_Ta_corr -7 // Default ambient temperature correction
#define TEMP_MODEL_LAG 2.1 // Temperature transport delay (s)
#define TEMP_MODEL_W 1.2 // Default pre-warning threshold (K/s)
#define TEMP_MODEL_E 1.74 // Default error threshold (K/s)
#define TEMP_MODEL_CAL_Th 230 // Default calibration working temperature (C)
#define TEMP_MODEL_CAL_Tl 50 // Default calibration cooling temperature (C)
/*------------------------------------
MOTOR CURRENT SETTINGS