2016-03-24 23:19:46 -07:00
|
|
|
/**
|
2016-03-24 18:01:20 +00:00
|
|
|
* Marlin 3D Printer Firmware
|
2020-02-03 08:00:57 -06:00
|
|
|
* Copyright (c) 2020 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
|
2016-03-24 18:01:20 +00:00
|
|
|
*
|
|
|
|
* Based on Sprinter and grbl.
|
2019-06-27 23:57:50 -05:00
|
|
|
* Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
|
2016-03-24 18:01:20 +00:00
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
2020-07-23 05:20:14 +02:00
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
2016-03-24 18:01:20 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2016-03-24 23:19:46 -07:00
|
|
|
/**
|
2016-04-28 18:18:13 -07:00
|
|
|
* temperature.cpp - temperature control
|
|
|
|
*/
|
2011-12-22 12:38:50 +01:00
|
|
|
|
2020-07-11 01:23:48 -04:00
|
|
|
// Useful when debugging thermocouples
|
|
|
|
//#define IGNORE_THERMOCOUPLE_ERRORS
|
|
|
|
|
2021-01-21 03:40:07 -06:00
|
|
|
#include "../MarlinCore.h"
|
|
|
|
#include "../HAL/shared/Delay.h"
|
|
|
|
#include "../lcd/marlinui.h"
|
|
|
|
|
2011-11-15 22:50:43 +01:00
|
|
|
#include "temperature.h"
|
2018-05-09 02:17:53 -03:00
|
|
|
#include "endstops.h"
|
2017-05-03 20:43:00 -05:00
|
|
|
#include "planner.h"
|
2021-07-06 17:36:41 -07:00
|
|
|
#include "printcounter.h"
|
2020-04-27 04:35:20 -05:00
|
|
|
|
2021-04-01 19:29:49 -05:00
|
|
|
#if EITHER(HAS_COOLER, LASER_COOLANT_FLOW_METER)
|
2021-03-06 14:13:28 -06:00
|
|
|
#include "../feature/cooler.h"
|
|
|
|
#include "../feature/spindle_laser.h"
|
|
|
|
#endif
|
|
|
|
|
2021-11-14 05:55:31 -06:00
|
|
|
#if ENABLED(USE_CONTROLLER_FAN)
|
|
|
|
#include "../feature/controllerfan.h"
|
|
|
|
#endif
|
|
|
|
|
2021-01-21 03:40:07 -06:00
|
|
|
#if ENABLED(EMERGENCY_PARSER)
|
|
|
|
#include "motion.h"
|
|
|
|
#endif
|
2020-06-16 01:45:27 -05:00
|
|
|
|
|
|
|
#if ENABLED(DWIN_CREALITY_LCD)
|
2021-07-31 05:32:13 -05:00
|
|
|
#include "../lcd/e3v2/creality/dwin.h"
|
2021-09-07 02:15:24 -05:00
|
|
|
#elif ENABLED(DWIN_CREALITY_LCD_ENHANCED)
|
|
|
|
#include "../lcd/e3v2/enhanced/dwin.h"
|
2020-06-16 01:45:27 -05:00
|
|
|
#endif
|
|
|
|
|
2020-01-04 11:00:44 +08:00
|
|
|
#if ENABLED(EXTENSIBLE_UI)
|
2020-03-13 16:29:29 -05:00
|
|
|
#include "../lcd/extui/ui_api.h"
|
2020-01-04 11:00:44 +08:00
|
|
|
#endif
|
2017-04-15 16:22:42 +01:00
|
|
|
|
2021-03-29 01:41:56 -05:00
|
|
|
#if ENABLED(HOST_PROMPT_SUPPORT)
|
|
|
|
#include "../feature/host_actions.h"
|
|
|
|
#endif
|
|
|
|
|
2021-09-07 18:06:10 -05:00
|
|
|
#if HAS_TEMP_SENSOR
|
|
|
|
#include "../gcode/gcode.h"
|
|
|
|
#endif
|
|
|
|
|
2021-12-05 05:14:19 +07:00
|
|
|
#if ENABLED(NOZZLE_PARK_FEATURE)
|
|
|
|
#include "../libs/nozzle.h"
|
|
|
|
#endif
|
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
// MAX TC related macros
|
2021-08-18 20:12:41 -05:00
|
|
|
#define TEMP_SENSOR_IS_MAX(n, M) (ENABLED(TEMP_SENSOR_##n##_IS_MAX##M) || (ENABLED(TEMP_SENSOR_REDUNDANT_IS_MAX##M) && REDUNDANT_TEMP_MATCH(SOURCE, E##n)))
|
|
|
|
#define TEMP_SENSOR_IS_ANY_MAX_TC(n) (ENABLED(TEMP_SENSOR_##n##_IS_MAX_TC) || (ENABLED(TEMP_SENSOR_REDUNDANT_IS_MAX_TC) && REDUNDANT_TEMP_MATCH(SOURCE, E##n)))
|
2021-06-11 13:51:29 -07:00
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
// LIB_MAX6675 can be added to the build_flags in platformio.ini to use a user-defined library
|
|
|
|
// If LIB_MAX6675 is not on the build_flags then raw SPI reads will be used.
|
2021-08-31 15:40:49 +12:00
|
|
|
#if HAS_MAX6675 && USE_LIB_MAX6675
|
2021-07-06 17:36:41 -07:00
|
|
|
#include <max6675.h>
|
|
|
|
#define HAS_MAX6675_LIBRARY 1
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// LIB_MAX31855 can be added to the build_flags in platformio.ini to use a user-defined library.
|
|
|
|
// If LIB_MAX31855 is not on the build_flags then raw SPI reads will be used.
|
2021-08-31 15:40:49 +12:00
|
|
|
#if HAS_MAX31855 && USE_ADAFRUIT_MAX31855
|
2021-02-08 20:51:57 -05:00
|
|
|
#include <Adafruit_MAX31855.h>
|
2021-07-06 17:36:41 -07:00
|
|
|
#define HAS_MAX31855_LIBRARY 1
|
|
|
|
typedef Adafruit_MAX31855 MAX31855;
|
2021-02-08 20:51:57 -05:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if HAS_MAX31865
|
2021-08-31 15:40:49 +12:00
|
|
|
#if USE_ADAFRUIT_MAX31865
|
2021-07-06 17:36:41 -07:00
|
|
|
#include <Adafruit_MAX31865.h>
|
|
|
|
typedef Adafruit_MAX31865 MAX31865;
|
|
|
|
#else
|
|
|
|
#include "../libs/MAX31865.h"
|
2021-02-08 20:51:57 -05:00
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
#if HAS_MAX6675_LIBRARY || HAS_MAX31855_LIBRARY || HAS_MAX31865
|
|
|
|
#define HAS_MAXTC_LIBRARIES 1
|
2019-11-21 11:15:10 +02:00
|
|
|
#endif
|
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
// If we have a MAX TC with SCK and MISO pins defined, it's either on a separate/dedicated Hardware
|
|
|
|
// SPI bus, or some pins for Software SPI. Alternate Hardware SPI buses are not supported yet, so
|
|
|
|
// your SPI options are:
|
|
|
|
//
|
|
|
|
// 1. Only CS pin(s) defined: Hardware SPI on the default bus (usually the SD card SPI).
|
|
|
|
// 2. CS, MISO, and SCK pins defined: Software SPI on a separate bus, as defined by MISO, SCK.
|
|
|
|
// 3. CS, MISO, and SCK pins w/ FORCE_HW_SPI: Hardware SPI on the default bus, ignoring MISO, SCK.
|
|
|
|
//
|
|
|
|
#if TEMP_SENSOR_IS_ANY_MAX_TC(0) && TEMP_SENSOR_0_HAS_SPI_PINS && DISABLED(TEMP_SENSOR_FORCE_HW_SPI)
|
|
|
|
#define TEMP_SENSOR_0_USES_SW_SPI 1
|
2021-02-08 20:51:57 -05:00
|
|
|
#endif
|
2021-07-06 17:36:41 -07:00
|
|
|
#if TEMP_SENSOR_IS_ANY_MAX_TC(1) && TEMP_SENSOR_1_HAS_SPI_PINS && DISABLED(TEMP_SENSOR_FORCE_HW_SPI)
|
|
|
|
#define TEMP_SENSOR_1_USES_SW_SPI 1
|
2020-11-11 16:52:35 -05:00
|
|
|
#endif
|
2019-02-05 22:56:13 -06:00
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
#if (TEMP_SENSOR_0_USES_SW_SPI || TEMP_SENSOR_1_USES_SW_SPI) && !HAS_MAXTC_LIBRARIES
|
2017-09-06 06:28:32 -05:00
|
|
|
#include "../libs/private_spi.h"
|
2021-07-06 17:36:41 -07:00
|
|
|
#define HAS_MAXTC_SW_SPI 1
|
2021-07-18 18:24:27 -07:00
|
|
|
|
|
|
|
// Define pins for SPI-based sensors
|
|
|
|
#if TEMP_SENSOR_0_USES_SW_SPI
|
|
|
|
#define SW_SPI_SCK_PIN TEMP_0_SCK_PIN
|
|
|
|
#define SW_SPI_MISO_PIN TEMP_0_MISO_PIN
|
|
|
|
#if PIN_EXISTS(TEMP_0_MOSI)
|
|
|
|
#define SW_SPI_MOSI_PIN TEMP_0_MOSI_PIN
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
#define SW_SPI_SCK_PIN TEMP_1_SCK_PIN
|
|
|
|
#define SW_SPI_MISO_PIN TEMP_1_MISO_PIN
|
|
|
|
#if PIN_EXISTS(TEMP_1_MOSI)
|
|
|
|
#define SW_SPI_MOSI_PIN TEMP_1_MOSI_PIN
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
#ifndef SW_SPI_MOSI_PIN
|
|
|
|
#define SW_SPI_MOSI_PIN SD_MOSI_PIN
|
|
|
|
#endif
|
2017-04-15 16:22:42 +01:00
|
|
|
#endif
|
|
|
|
|
2020-02-14 05:14:37 -06:00
|
|
|
#if ENABLED(PID_EXTRUSION_SCALING)
|
2016-08-02 21:05:31 +02:00
|
|
|
#include "stepper.h"
|
|
|
|
#endif
|
2015-01-15 18:52:10 -05:00
|
|
|
|
2020-02-15 21:42:28 -06:00
|
|
|
#if ENABLED(BABYSTEPPING) && DISABLED(INTEGRATED_BABYSTEPPING)
|
2019-04-06 18:04:34 -05:00
|
|
|
#include "../feature/babystep.h"
|
2019-02-27 05:38:56 -05:00
|
|
|
#endif
|
|
|
|
|
2017-09-12 15:02:17 -05:00
|
|
|
#if ENABLED(FILAMENT_WIDTH_SENSOR)
|
|
|
|
#include "../feature/filwidth.h"
|
|
|
|
#endif
|
|
|
|
|
2020-06-18 15:23:03 -05:00
|
|
|
#if HAS_POWER_MONITOR
|
|
|
|
#include "../feature/power_monitor.h"
|
|
|
|
#endif
|
|
|
|
|
2018-04-25 12:44:26 +01:00
|
|
|
#if ENABLED(EMERGENCY_PARSER)
|
2020-03-13 16:29:29 -05:00
|
|
|
#include "../feature/e_parser.h"
|
2018-04-25 12:44:26 +01:00
|
|
|
#endif
|
|
|
|
|
2018-09-29 01:44:47 -05:00
|
|
|
#if ENABLED(PRINTER_EVENT_LEDS)
|
2018-10-11 04:25:43 +02:00
|
|
|
#include "../feature/leds/printer_event_leds.h"
|
2018-09-29 01:44:47 -05:00
|
|
|
#endif
|
|
|
|
|
2019-09-08 00:55:34 -05:00
|
|
|
#if ENABLED(JOYSTICK)
|
|
|
|
#include "../feature/joystick.h"
|
|
|
|
#endif
|
|
|
|
|
2019-01-12 01:41:48 -05:00
|
|
|
#if ENABLED(SINGLENOZZLE)
|
|
|
|
#include "tool_change.h"
|
|
|
|
#endif
|
|
|
|
|
2019-08-20 09:01:37 +02:00
|
|
|
#if USE_BEEPER
|
2019-05-22 03:31:05 +01:00
|
|
|
#include "../libs/buzzer.h"
|
|
|
|
#endif
|
|
|
|
|
2020-10-07 01:36:01 +02:00
|
|
|
#if HAS_SERVOS
|
2021-03-24 18:21:11 +01:00
|
|
|
#include "servo.h"
|
2020-10-07 01:36:01 +02:00
|
|
|
#endif
|
2020-11-09 01:17:37 -06:00
|
|
|
|
2021-02-07 16:58:06 -06:00
|
|
|
#if ANY(TEMP_SENSOR_0_IS_THERMISTOR, TEMP_SENSOR_1_IS_THERMISTOR, TEMP_SENSOR_2_IS_THERMISTOR, TEMP_SENSOR_3_IS_THERMISTOR, \
|
|
|
|
TEMP_SENSOR_4_IS_THERMISTOR, TEMP_SENSOR_5_IS_THERMISTOR, TEMP_SENSOR_6_IS_THERMISTOR, TEMP_SENSOR_7_IS_THERMISTOR )
|
2020-11-09 01:17:37 -06:00
|
|
|
#define HAS_HOTEND_THERMISTOR 1
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if HAS_HOTEND_THERMISTOR
|
2021-06-11 13:51:29 -07:00
|
|
|
#define NEXT_TEMPTABLE(N) ,TEMPTABLE_##N
|
|
|
|
#define NEXT_TEMPTABLE_LEN(N) ,TEMPTABLE_##N##_LEN
|
|
|
|
static const temp_entry_t* heater_ttbl_map[HOTENDS] = ARRAY_BY_HOTENDS(TEMPTABLE_0 REPEAT_S(1, HOTENDS, NEXT_TEMPTABLE));
|
|
|
|
static constexpr uint8_t heater_ttbllen_map[HOTENDS] = ARRAY_BY_HOTENDS(TEMPTABLE_0_LEN REPEAT_S(1, HOTENDS, NEXT_TEMPTABLE_LEN));
|
2013-06-06 15:49:25 -07:00
|
|
|
#endif
|
2012-12-09 23:00:06 +01:00
|
|
|
|
2016-04-28 18:18:13 -07:00
|
|
|
Temperature thermalManager;
|
2012-03-08 21:43:21 +01:00
|
|
|
|
2021-09-28 20:15:52 -05:00
|
|
|
PGMSTR(str_t_thermal_runaway, STR_T_THERMAL_RUNAWAY);
|
|
|
|
PGMSTR(str_t_heating_failed, STR_T_HEATING_FAILED);
|
2020-02-26 03:02:03 -06:00
|
|
|
|
2018-04-04 17:29:27 -06:00
|
|
|
/**
|
|
|
|
* Macros to include the heater id in temp errors. The compiler's dead-code
|
|
|
|
* elimination should (hopefully) optimize out the unused strings.
|
|
|
|
*/
|
2019-03-07 00:09:39 -08:00
|
|
|
|
2018-04-23 17:13:01 -05:00
|
|
|
#if HAS_HEATED_BED
|
2021-09-28 20:15:52 -05:00
|
|
|
#define _BED_FSTR(h) (h) == H_BED ? GET_TEXT_F(MSG_BED) :
|
2019-03-07 00:09:39 -08:00
|
|
|
#else
|
2021-09-28 20:15:52 -05:00
|
|
|
#define _BED_FSTR(h)
|
2019-03-07 00:09:39 -08:00
|
|
|
#endif
|
|
|
|
#if HAS_HEATED_CHAMBER
|
2021-09-28 20:15:52 -05:00
|
|
|
#define _CHAMBER_FSTR(h) (h) == H_CHAMBER ? GET_TEXT_F(MSG_CHAMBER) :
|
2018-04-04 17:29:27 -06:00
|
|
|
#else
|
2021-09-28 20:15:52 -05:00
|
|
|
#define _CHAMBER_FSTR(h)
|
2018-04-04 17:29:27 -06:00
|
|
|
#endif
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_COOLER
|
2021-09-28 20:15:52 -05:00
|
|
|
#define _COOLER_FSTR(h) (h) == H_COOLER ? GET_TEXT_F(MSG_COOLER) :
|
2021-03-06 14:13:28 -06:00
|
|
|
#else
|
2021-09-28 20:15:52 -05:00
|
|
|
#define _COOLER_FSTR(h)
|
2021-03-06 14:13:28 -06:00
|
|
|
#endif
|
2021-09-28 20:15:52 -05:00
|
|
|
#define _E_FSTR(h,N) ((HOTENDS) > N && (h) == N) ? F(LCD_STR_E##N) :
|
|
|
|
#define HEATER_FSTR(h) _BED_FSTR(h) _CHAMBER_FSTR(h) _COOLER_FSTR(h) _E_FSTR(h,1) _E_FSTR(h,2) _E_FSTR(h,3) _E_FSTR(h,4) _E_FSTR(h,5) F(LCD_STR_E0)
|
2018-04-04 17:29:27 -06:00
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
//
|
|
|
|
// Initialize MAX TC objects/SPI
|
|
|
|
//
|
|
|
|
#if HAS_MAX_TC
|
|
|
|
|
|
|
|
#if HAS_MAXTC_SW_SPI
|
|
|
|
// Initialize SoftSPI for non-lib Software SPI; Libraries take care of it themselves.
|
|
|
|
template<uint8_t MisoPin, uint8_t MosiPin, uint8_t SckPin>
|
|
|
|
SoftSPI<MisoPin, MosiPin, SckPin> SPIclass<MisoPin, MosiPin, SckPin>::softSPI;
|
2021-07-18 18:24:27 -07:00
|
|
|
SPIclass<SW_SPI_MISO_PIN, SW_SPI_MOSI_PIN, SW_SPI_SCK_PIN> max_tc_spi;
|
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#define MAXTC_INIT(n, M) \
|
|
|
|
MAX##M max##M##_##n = MAX##M( \
|
|
|
|
TEMP_##n##_CS_PIN \
|
|
|
|
OPTARG(_MAX31865_##n##_SW, TEMP_##n##_MOSI_PIN) \
|
|
|
|
OPTARG(TEMP_SENSOR_##n##_USES_SW_SPI, TEMP_##n##_MISO_PIN, TEMP_##n##_SCK_PIN) \
|
|
|
|
OPTARG(LARGE_PINMAP, HIGH) \
|
|
|
|
)
|
|
|
|
|
|
|
|
#if HAS_MAX6675_LIBRARY
|
|
|
|
#if TEMP_SENSOR_IS_MAX(0, 6675)
|
|
|
|
MAXTC_INIT(0, 6675);
|
|
|
|
#endif
|
|
|
|
#if TEMP_SENSOR_IS_MAX(1, 6675)
|
|
|
|
MAXTC_INIT(1, 6675);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if HAS_MAX31855_LIBRARY
|
|
|
|
#if TEMP_SENSOR_IS_MAX(0, 31855)
|
|
|
|
MAXTC_INIT(0, 31855);
|
|
|
|
#endif
|
|
|
|
#if TEMP_SENSOR_IS_MAX(1, 31855)
|
|
|
|
MAXTC_INIT(1, 31855);
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// MAX31865 always uses a library, unlike '55 & 6675
|
|
|
|
#if HAS_MAX31865
|
|
|
|
#define _MAX31865_0_SW TEMP_SENSOR_0_USES_SW_SPI
|
|
|
|
#define _MAX31865_1_SW TEMP_SENSOR_1_USES_SW_SPI
|
|
|
|
|
|
|
|
#if TEMP_SENSOR_IS_MAX(0, 31865)
|
|
|
|
MAXTC_INIT(0, 31865);
|
|
|
|
#endif
|
|
|
|
#if TEMP_SENSOR_IS_MAX(1, 31865)
|
|
|
|
MAXTC_INIT(1, 31865);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#undef _MAX31865_0_SW
|
|
|
|
#undef _MAX31865_1_SW
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#undef MAXTC_INIT
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* public:
|
|
|
|
*/
|
2016-05-26 11:58:38 -07:00
|
|
|
|
2019-02-04 07:12:41 +01:00
|
|
|
#if ENABLED(NO_FAN_SLOWING_IN_PID_TUNING)
|
|
|
|
bool Temperature::adaptive_fan_slowing = true;
|
|
|
|
#endif
|
|
|
|
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_HOTEND
|
2021-04-27 16:36:39 +12:00
|
|
|
hotend_info_t Temperature::temp_hotend[HOTENDS];
|
|
|
|
#define _HMT(N) HEATER_##N##_MAXTEMP,
|
2021-03-24 04:11:43 -05:00
|
|
|
const celsius_t Temperature::hotend_maxtemp[HOTENDS] = ARRAY_BY_HOTENDS(HEATER_0_MAXTEMP, HEATER_1_MAXTEMP, HEATER_2_MAXTEMP, HEATER_3_MAXTEMP, HEATER_4_MAXTEMP, HEATER_5_MAXTEMP, HEATER_6_MAXTEMP, HEATER_7_MAXTEMP);
|
2019-09-10 02:20:49 -05:00
|
|
|
#endif
|
2017-05-28 11:11:17 -05:00
|
|
|
|
2021-06-11 13:51:29 -07:00
|
|
|
#if HAS_TEMP_REDUNDANT
|
2021-08-18 20:12:41 -05:00
|
|
|
redundant_info_t Temperature::temp_redundant;
|
2021-06-11 13:51:29 -07:00
|
|
|
#endif
|
|
|
|
|
2018-02-06 00:22:30 -06:00
|
|
|
#if ENABLED(AUTO_POWER_E_FANS)
|
2019-01-12 01:41:48 -05:00
|
|
|
uint8_t Temperature::autofan_speed[HOTENDS]; // = { 0 }
|
2018-02-06 00:22:30 -06:00
|
|
|
#endif
|
|
|
|
|
2019-05-05 05:51:47 +02:00
|
|
|
#if ENABLED(AUTO_POWER_CHAMBER_FAN)
|
|
|
|
uint8_t Temperature::chamberfan_speed; // = 0
|
|
|
|
#endif
|
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
#if ENABLED(AUTO_POWER_COOLER_FAN)
|
|
|
|
uint8_t Temperature::coolerfan_speed; // = 0
|
|
|
|
#endif
|
2021-03-19 16:34:10 -05:00
|
|
|
|
2021-11-14 05:55:31 -06:00
|
|
|
#if BOTH(FAN_SOFT_PWM, USE_CONTROLLER_FAN)
|
|
|
|
uint8_t Temperature::soft_pwm_controller_speed;
|
|
|
|
#endif
|
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
// Init fans according to whether they're native PWM or Software PWM
|
|
|
|
#ifdef BOARD_OPENDRAIN_MOSFETS
|
|
|
|
#define _INIT_SOFT_FAN(P) OUT_WRITE_OD(P, FAN_INVERTING ? LOW : HIGH)
|
|
|
|
#else
|
|
|
|
#define _INIT_SOFT_FAN(P) OUT_WRITE(P, FAN_INVERTING ? LOW : HIGH)
|
|
|
|
#endif
|
|
|
|
#if ENABLED(FAN_SOFT_PWM)
|
|
|
|
#define _INIT_FAN_PIN(P) _INIT_SOFT_FAN(P)
|
|
|
|
#else
|
|
|
|
#define _INIT_FAN_PIN(P) do{ if (PWM_PIN(P)) SET_PWM(P); else _INIT_SOFT_FAN(P); }while(0)
|
|
|
|
#endif
|
|
|
|
#if ENABLED(FAST_PWM_FAN)
|
|
|
|
#define SET_FAST_PWM_FREQ(P) set_pwm_frequency(P, FAST_PWM_FAN_FREQUENCY)
|
|
|
|
#else
|
|
|
|
#define SET_FAST_PWM_FREQ(P) NOOP
|
|
|
|
#endif
|
|
|
|
#define INIT_FAN_PIN(P) do{ _INIT_FAN_PIN(P); SET_FAST_PWM_FREQ(P); }while(0)
|
|
|
|
|
|
|
|
// HAS_FAN does not include CONTROLLER_FAN
|
2020-04-27 04:41:18 -05:00
|
|
|
#if HAS_FAN
|
2019-01-12 01:41:48 -05:00
|
|
|
|
|
|
|
uint8_t Temperature::fan_speed[FAN_COUNT]; // = { 0 }
|
|
|
|
|
|
|
|
#if ENABLED(EXTRA_FAN_SPEED)
|
|
|
|
|
2021-03-19 16:39:05 -05:00
|
|
|
Temperature::extra_fan_t Temperature::extra_fan_speed[FAN_COUNT];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle the M106 P<fan> T<speed> command:
|
|
|
|
* T1 = Restore fan speed saved on the last T2
|
|
|
|
* T2 = Save the fan speed, then set to the last T<3-255> value
|
|
|
|
* T<3-255> = Set the "extra fan speed"
|
|
|
|
*/
|
|
|
|
void Temperature::set_temp_fan_speed(const uint8_t fan, const uint16_t command_or_speed) {
|
|
|
|
switch (command_or_speed) {
|
2019-01-12 01:41:48 -05:00
|
|
|
case 1:
|
2021-03-19 16:39:05 -05:00
|
|
|
set_fan_speed(fan, extra_fan_speed[fan].saved);
|
2019-01-12 01:41:48 -05:00
|
|
|
break;
|
|
|
|
case 2:
|
2021-03-19 16:39:05 -05:00
|
|
|
extra_fan_speed[fan].saved = fan_speed[fan];
|
|
|
|
set_fan_speed(fan, extra_fan_speed[fan].speed);
|
2019-01-12 01:41:48 -05:00
|
|
|
break;
|
|
|
|
default:
|
2021-03-19 16:39:05 -05:00
|
|
|
extra_fan_speed[fan].speed = _MIN(command_or_speed, 255U);
|
2019-01-12 01:41:48 -05:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2019-05-07 21:14:12 +02:00
|
|
|
#if EITHER(PROBING_FANS_OFF, ADVANCED_PAUSE_FANS_PAUSE)
|
2019-01-12 01:41:48 -05:00
|
|
|
bool Temperature::fans_paused; // = false;
|
2019-05-07 21:14:12 +02:00
|
|
|
uint8_t Temperature::saved_fan_speed[FAN_COUNT]; // = { 0 }
|
2019-01-12 01:41:48 -05:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if ENABLED(ADAPTIVE_FAN_SLOWING)
|
2021-05-11 04:01:21 -05:00
|
|
|
uint8_t Temperature::fan_speed_scaler[FAN_COUNT] = ARRAY_N_1(FAN_COUNT, 128);
|
2019-01-12 01:41:48 -05:00
|
|
|
#endif
|
|
|
|
|
2019-06-28 13:58:40 -05:00
|
|
|
/**
|
|
|
|
* Set the print fan speed for a target extruder
|
|
|
|
*/
|
2021-04-13 04:00:39 -05:00
|
|
|
void Temperature::set_fan_speed(uint8_t fan, uint16_t speed) {
|
2019-01-12 01:41:48 -05:00
|
|
|
|
|
|
|
NOMORE(speed, 255U);
|
|
|
|
|
2020-04-28 07:21:23 +02:00
|
|
|
#if ENABLED(SINGLENOZZLE_STANDBY_FAN)
|
2021-04-13 04:00:39 -05:00
|
|
|
if (fan != active_extruder) {
|
|
|
|
if (fan < EXTRUDERS) singlenozzle_fan_speed[fan] = speed;
|
2019-01-12 01:41:48 -05:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2021-08-05 23:24:20 -05:00
|
|
|
TERN_(SINGLENOZZLE, if (fan < EXTRUDERS) fan = 0); // Always fan 0 for SINGLENOZZLE E fan
|
2020-08-26 06:23:56 -04:00
|
|
|
|
2021-04-13 04:00:39 -05:00
|
|
|
if (fan >= FAN_COUNT) return;
|
2019-01-17 18:03:58 -06:00
|
|
|
|
2021-04-13 04:00:39 -05:00
|
|
|
fan_speed[fan] = speed;
|
2021-06-15 11:45:54 +12:00
|
|
|
#if REDUNDANT_PART_COOLING_FAN
|
|
|
|
if (fan == 0) fan_speed[REDUNDANT_PART_COOLING_FAN] = speed;
|
|
|
|
#endif
|
2020-05-26 04:16:35 +05:30
|
|
|
|
2021-04-13 04:00:39 -05:00
|
|
|
TERN_(REPORT_FAN_CHANGE, report_fan_speed(fan));
|
2019-01-12 01:41:48 -05:00
|
|
|
}
|
|
|
|
|
2020-07-14 19:30:48 -05:00
|
|
|
#if ENABLED(REPORT_FAN_CHANGE)
|
|
|
|
/**
|
|
|
|
* Report print fan speed for a target extruder
|
|
|
|
*/
|
2021-04-13 04:00:39 -05:00
|
|
|
void Temperature::report_fan_speed(const uint8_t fan) {
|
|
|
|
if (fan >= FAN_COUNT) return;
|
2021-03-09 10:20:37 +01:00
|
|
|
PORT_REDIRECT(SerialMask::All);
|
2021-09-09 04:57:05 -05:00
|
|
|
SERIAL_ECHOLNPGM("M106 P", fan, " S", fan_speed[fan]);
|
2020-07-14 19:30:48 -05:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-05-04 16:50:44 -04:00
|
|
|
#if EITHER(PROBING_FANS_OFF, ADVANCED_PAUSE_FANS_PAUSE)
|
2019-01-12 01:41:48 -05:00
|
|
|
|
|
|
|
void Temperature::set_fans_paused(const bool p) {
|
|
|
|
if (p != fans_paused) {
|
|
|
|
fans_paused = p;
|
|
|
|
if (p)
|
2019-05-04 14:36:59 -05:00
|
|
|
FANS_LOOP(i) { saved_fan_speed[i] = fan_speed[i]; fan_speed[i] = 0; }
|
2019-01-12 01:41:48 -05:00
|
|
|
else
|
2019-05-04 14:36:59 -05:00
|
|
|
FANS_LOOP(i) fan_speed[i] = saved_fan_speed[i];
|
2019-01-12 01:41:48 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-04 16:50:44 -04:00
|
|
|
#endif
|
2019-01-12 01:41:48 -05:00
|
|
|
|
2020-04-27 04:41:18 -05:00
|
|
|
#endif // HAS_FAN
|
2019-01-12 01:41:48 -05:00
|
|
|
|
2019-03-07 00:09:39 -08:00
|
|
|
#if WATCH_HOTENDS
|
2020-01-30 03:24:43 -06:00
|
|
|
hotend_watch_t Temperature::watch_hotend[HOTENDS]; // = { { 0 } }
|
2019-03-07 00:09:39 -08:00
|
|
|
#endif
|
|
|
|
#if HEATER_IDLE_HANDLER
|
2020-09-14 16:58:39 +12:00
|
|
|
Temperature::heater_idle_t Temperature::heater_idle[NR_HEATER_IDLE]; // = { { 0 } }
|
2019-03-07 00:09:39 -08:00
|
|
|
#endif
|
|
|
|
|
2018-04-23 17:13:01 -05:00
|
|
|
#if HAS_HEATED_BED
|
2019-03-07 00:09:39 -08:00
|
|
|
bed_info_t Temperature::temp_bed; // = { 0 }
|
|
|
|
// Init min and max temp with extreme values to prevent false errors during startup
|
2021-03-24 04:11:43 -05:00
|
|
|
int16_t Temperature::mintemp_raw_BED = TEMP_SENSOR_BED_RAW_LO_TEMP,
|
|
|
|
Temperature::maxtemp_raw_BED = TEMP_SENSOR_BED_RAW_HI_TEMP;
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(WATCH_BED, bed_watch_t Temperature::watch_bed); // = { 0 }
|
2020-11-07 18:28:29 -06:00
|
|
|
IF_DISABLED(PIDTEMPBED, millis_t Temperature::next_bed_check_ms);
|
2021-03-06 14:13:28 -06:00
|
|
|
#endif
|
2018-04-23 17:13:01 -05:00
|
|
|
|
|
|
|
#if HAS_TEMP_CHAMBER
|
2019-03-10 15:51:13 +01:00
|
|
|
chamber_info_t Temperature::temp_chamber; // = { 0 }
|
2019-03-07 00:09:39 -08:00
|
|
|
#if HAS_HEATED_CHAMBER
|
2020-10-07 01:36:01 +02:00
|
|
|
millis_t next_cool_check_ms_2 = 0;
|
2021-04-23 19:14:49 -05:00
|
|
|
celsius_float_t old_temp = 9999;
|
2021-03-24 04:11:43 -05:00
|
|
|
int16_t Temperature::mintemp_raw_CHAMBER = TEMP_SENSOR_CHAMBER_RAW_LO_TEMP,
|
|
|
|
Temperature::maxtemp_raw_CHAMBER = TEMP_SENSOR_CHAMBER_RAW_HI_TEMP;
|
2021-02-24 16:26:51 -08:00
|
|
|
TERN_(WATCH_CHAMBER, chamber_watch_t Temperature::watch_chamber{0});
|
|
|
|
IF_DISABLED(PIDTEMPCHAMBER, millis_t Temperature::next_chamber_check_ms);
|
2021-03-06 14:13:28 -06:00
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if HAS_TEMP_COOLER
|
|
|
|
cooler_info_t Temperature::temp_cooler; // = { 0 }
|
|
|
|
#if HAS_COOLER
|
|
|
|
bool flag_cooler_state;
|
|
|
|
//bool flag_cooler_excess = false;
|
2021-04-23 19:14:49 -05:00
|
|
|
celsius_float_t previous_temp = 9999;
|
2021-03-24 04:11:43 -05:00
|
|
|
int16_t Temperature::mintemp_raw_COOLER = TEMP_SENSOR_COOLER_RAW_LO_TEMP,
|
|
|
|
Temperature::maxtemp_raw_COOLER = TEMP_SENSOR_COOLER_RAW_HI_TEMP;
|
2021-03-06 14:13:28 -06:00
|
|
|
#if WATCH_COOLER
|
|
|
|
cooler_watch_t Temperature::watch_cooler{0};
|
|
|
|
#endif
|
|
|
|
millis_t Temperature::next_cooler_check_ms, Temperature::cooler_fan_flush_ms;
|
|
|
|
#endif
|
|
|
|
#endif
|
2016-05-26 11:58:38 -07:00
|
|
|
|
2020-01-18 00:16:45 +01:00
|
|
|
#if HAS_TEMP_PROBE
|
|
|
|
probe_info_t Temperature::temp_probe; // = { 0 }
|
|
|
|
#endif
|
|
|
|
|
2021-08-18 20:12:41 -05:00
|
|
|
#if HAS_TEMP_BOARD
|
|
|
|
board_info_t Temperature::temp_board; // = { 0 }
|
|
|
|
#if ENABLED(THERMAL_PROTECTION_BOARD)
|
|
|
|
int16_t Temperature::mintemp_raw_BOARD = TEMP_SENSOR_BOARD_RAW_LO_TEMP,
|
|
|
|
Temperature::maxtemp_raw_BOARD = TEMP_SENSOR_BOARD_RAW_HI_TEMP;
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2016-08-20 23:34:24 -05:00
|
|
|
#if ENABLED(PREVENT_COLD_EXTRUSION)
|
2016-07-06 08:28:09 -07:00
|
|
|
bool Temperature::allow_cold_extrude = false;
|
2021-03-24 04:11:43 -05:00
|
|
|
celsius_t Temperature::extrude_min_temp = EXTRUDE_MINTEMP;
|
2016-05-26 11:58:38 -07:00
|
|
|
#endif
|
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
#if HAS_ADC_BUTTONS
|
|
|
|
uint32_t Temperature::current_ADCKey_raw = HAL_ADC_RANGE;
|
|
|
|
uint16_t Temperature::ADCKey_count = 0;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if ENABLED(PID_EXTRUSION_SCALING)
|
|
|
|
int16_t Temperature::lpq_len; // Initialized in settings.cpp
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* private:
|
|
|
|
*/
|
2016-05-26 11:58:38 -07:00
|
|
|
|
2020-01-26 17:39:45 -06:00
|
|
|
volatile bool Temperature::raw_temps_ready = false;
|
2016-05-26 11:58:38 -07:00
|
|
|
|
2019-02-27 05:38:56 -05:00
|
|
|
#if ENABLED(PID_EXTRUSION_SCALING)
|
2019-03-07 00:09:39 -08:00
|
|
|
int32_t Temperature::last_e_position, Temperature::lpq[LPQ_MAX_LEN];
|
|
|
|
lpq_ptr_t Temperature::lpq_ptr = 0;
|
2016-05-26 11:58:38 -07:00
|
|
|
#endif
|
|
|
|
|
2021-02-07 16:58:06 -06:00
|
|
|
#define TEMPDIR(N) ((TEMP_SENSOR_##N##_RAW_LO_TEMP) < (TEMP_SENSOR_##N##_RAW_HI_TEMP) ? 1 : -1)
|
2019-03-07 00:09:39 -08:00
|
|
|
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_HOTEND
|
2019-09-10 02:20:49 -05:00
|
|
|
// Init mintemp and maxtemp with extreme values to prevent false errors during startup
|
2021-02-07 16:58:06 -06:00
|
|
|
constexpr temp_range_t sensor_heater_0 { TEMP_SENSOR_0_RAW_LO_TEMP, TEMP_SENSOR_0_RAW_HI_TEMP, 0, 16383 },
|
|
|
|
sensor_heater_1 { TEMP_SENSOR_1_RAW_LO_TEMP, TEMP_SENSOR_1_RAW_HI_TEMP, 0, 16383 },
|
|
|
|
sensor_heater_2 { TEMP_SENSOR_2_RAW_LO_TEMP, TEMP_SENSOR_2_RAW_HI_TEMP, 0, 16383 },
|
|
|
|
sensor_heater_3 { TEMP_SENSOR_3_RAW_LO_TEMP, TEMP_SENSOR_3_RAW_HI_TEMP, 0, 16383 },
|
|
|
|
sensor_heater_4 { TEMP_SENSOR_4_RAW_LO_TEMP, TEMP_SENSOR_4_RAW_HI_TEMP, 0, 16383 },
|
|
|
|
sensor_heater_5 { TEMP_SENSOR_5_RAW_LO_TEMP, TEMP_SENSOR_5_RAW_HI_TEMP, 0, 16383 },
|
|
|
|
sensor_heater_6 { TEMP_SENSOR_6_RAW_LO_TEMP, TEMP_SENSOR_6_RAW_HI_TEMP, 0, 16383 },
|
|
|
|
sensor_heater_7 { TEMP_SENSOR_7_RAW_LO_TEMP, TEMP_SENSOR_7_RAW_HI_TEMP, 0, 16383 };
|
2016-05-26 11:58:38 -07:00
|
|
|
|
2020-01-25 16:13:39 +08:00
|
|
|
temp_range_t Temperature::temp_range[HOTENDS] = ARRAY_BY_HOTENDS(sensor_heater_0, sensor_heater_1, sensor_heater_2, sensor_heater_3, sensor_heater_4, sensor_heater_5, sensor_heater_6, sensor_heater_7);
|
2019-09-10 02:20:49 -05:00
|
|
|
#endif
|
2016-05-26 11:58:38 -07:00
|
|
|
|
2021-05-02 21:32:21 -05:00
|
|
|
#if MAX_CONSECUTIVE_LOW_TEMPERATURE_ERROR_ALLOWED > 1
|
2017-05-03 17:12:14 -05:00
|
|
|
uint8_t Temperature::consecutive_low_temperature_error[HOTENDS] = { 0 };
|
2016-07-08 19:28:37 -07:00
|
|
|
#endif
|
|
|
|
|
2021-05-02 21:32:21 -05:00
|
|
|
#if MILLISECONDS_PREHEAT_TIME > 0
|
2017-05-03 17:12:14 -05:00
|
|
|
millis_t Temperature::preheat_end_time[HOTENDS] = { 0 };
|
2016-07-08 19:28:37 -07:00
|
|
|
#endif
|
|
|
|
|
2016-05-26 11:58:38 -07:00
|
|
|
#if HAS_AUTO_FAN
|
2016-08-13 19:52:24 -07:00
|
|
|
millis_t Temperature::next_auto_fan_check_ms = 0;
|
2016-05-26 11:58:38 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if ENABLED(FAN_SOFT_PWM)
|
2017-04-20 14:04:26 -05:00
|
|
|
uint8_t Temperature::soft_pwm_amount_fan[FAN_COUNT],
|
|
|
|
Temperature::soft_pwm_count_fan[FAN_COUNT];
|
2016-05-26 11:58:38 -07:00
|
|
|
#endif
|
|
|
|
|
2021-01-19 20:58:50 -06:00
|
|
|
#if ENABLED(SINGLENOZZLE_STANDBY_TEMP)
|
2021-03-24 04:11:43 -05:00
|
|
|
celsius_t Temperature::singlenozzle_temp[EXTRUDERS];
|
2021-05-01 20:21:18 +12:00
|
|
|
#endif
|
|
|
|
#if ENABLED(SINGLENOZZLE_STANDBY_FAN)
|
|
|
|
uint8_t Temperature::singlenozzle_fan_speed[EXTRUDERS];
|
2021-01-19 20:58:50 -06:00
|
|
|
#endif
|
|
|
|
|
2017-05-07 07:06:06 -04:00
|
|
|
#if ENABLED(PROBING_HEATERS_OFF)
|
2021-05-02 21:32:21 -05:00
|
|
|
bool Temperature::paused_for_probing;
|
2017-05-26 13:01:02 -05:00
|
|
|
#endif
|
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
/**
|
|
|
|
* public:
|
|
|
|
* Class and Instance Methods
|
|
|
|
*/
|
2018-05-12 02:13:40 -05:00
|
|
|
|
2016-04-19 18:09:29 -07:00
|
|
|
#if HAS_PID_HEATING
|
2016-10-09 16:13:58 -05:00
|
|
|
|
2018-11-29 16:58:58 -06:00
|
|
|
inline void say_default_() { SERIAL_ECHOPGM("#define DEFAULT_"); }
|
2018-10-10 09:45:20 -05:00
|
|
|
|
2017-11-10 00:48:00 -06:00
|
|
|
/**
|
|
|
|
* PID Autotuning (M303)
|
|
|
|
*
|
|
|
|
* Alternately heat and cool the nozzle, observing its behavior to
|
|
|
|
* determine the best PID values to achieve a stable temperature.
|
2019-02-10 10:54:45 +01:00
|
|
|
* Needs sufficient heater power to make some overshoot at target
|
|
|
|
* temperature to succeed.
|
2017-11-10 00:48:00 -06:00
|
|
|
*/
|
2021-04-23 20:19:23 -05:00
|
|
|
void Temperature::PID_autotune(const celsius_t target, const heater_id_t heater_id, const int8_t ncycles, const bool set_result/*=false*/) {
|
2021-04-23 22:29:15 -03:00
|
|
|
celsius_float_t current_temp = 0.0;
|
2016-04-18 19:02:11 -07:00
|
|
|
int cycles = 0;
|
|
|
|
bool heating = true;
|
2012-03-08 21:43:21 +01:00
|
|
|
|
2017-10-29 04:30:50 -05:00
|
|
|
millis_t next_temp_ms = millis(), t1 = next_temp_ms, t2 = next_temp_ms;
|
2016-04-18 19:02:11 -07:00
|
|
|
long t_high = 0, t_low = 0;
|
2012-03-08 21:43:21 +01:00
|
|
|
|
2018-10-10 09:45:20 -05:00
|
|
|
PID_t tune_pid = { 0, 0, 0 };
|
2021-04-23 19:14:49 -05:00
|
|
|
celsius_float_t maxT = 0, minT = 10000;
|
2017-11-03 03:24:19 -05:00
|
|
|
|
2020-09-13 18:06:14 -05:00
|
|
|
const bool isbed = (heater_id == H_BED);
|
2021-02-24 16:26:51 -08:00
|
|
|
const bool ischamber = (heater_id == H_CHAMBER);
|
2019-06-26 07:40:29 +02:00
|
|
|
|
2021-02-24 16:26:51 -08:00
|
|
|
#if ENABLED(PIDTEMPCHAMBER)
|
|
|
|
#define C_TERN(T,A,B) ((T) ? (A) : (B))
|
2018-01-03 21:30:45 -06:00
|
|
|
#else
|
2021-02-24 16:26:51 -08:00
|
|
|
#define C_TERN(T,A,B) (B)
|
2018-01-03 21:30:45 -06:00
|
|
|
#endif
|
2021-02-24 16:26:51 -08:00
|
|
|
#if ENABLED(PIDTEMPBED)
|
|
|
|
#define B_TERN(T,A,B) ((T) ? (A) : (B))
|
|
|
|
#else
|
|
|
|
#define B_TERN(T,A,B) (B)
|
|
|
|
#endif
|
|
|
|
#define GHV(C,B,H) C_TERN(ischamber, C, B_TERN(isbed, B, H))
|
|
|
|
#define SHV(V) C_TERN(ischamber, temp_chamber.soft_pwm_amount = V, B_TERN(isbed, temp_bed.soft_pwm_amount = V, temp_hotend[heater_id].soft_pwm_amount = V))
|
|
|
|
#define ONHEATINGSTART() C_TERN(ischamber, printerEventLEDs.onChamberHeatingStart(), B_TERN(isbed, printerEventLEDs.onBedHeatingStart(), printerEventLEDs.onHotendHeatingStart()))
|
|
|
|
#define ONHEATING(S,C,T) C_TERN(ischamber, printerEventLEDs.onChamberHeating(S,C,T), B_TERN(isbed, printerEventLEDs.onBedHeating(S,C,T), printerEventLEDs.onHotendHeating(S,C,T)))
|
|
|
|
|
2021-10-22 13:21:26 -05:00
|
|
|
#define WATCH_PID DISABLED(NO_WATCH_PID_TUNING) && (BOTH(WATCH_CHAMBER, PIDTEMPCHAMBER) || BOTH(WATCH_BED, PIDTEMPBED) || BOTH(WATCH_HOTENDS, PIDTEMP))
|
2018-01-03 21:30:45 -06:00
|
|
|
|
2020-05-18 11:51:32 -07:00
|
|
|
#if WATCH_PID
|
2021-02-24 16:26:51 -08:00
|
|
|
#if BOTH(THERMAL_PROTECTION_CHAMBER, PIDTEMPCHAMBER)
|
|
|
|
#define C_GTV(T,A,B) ((T) ? (A) : (B))
|
2020-05-18 11:51:32 -07:00
|
|
|
#else
|
2021-02-24 16:26:51 -08:00
|
|
|
#define C_GTV(T,A,B) (B)
|
2018-07-15 16:07:31 -05:00
|
|
|
#endif
|
2021-02-24 16:26:51 -08:00
|
|
|
#if BOTH(THERMAL_PROTECTION_BED, PIDTEMPBED)
|
|
|
|
#define B_GTV(T,A,B) ((T) ? (A) : (B))
|
|
|
|
#else
|
|
|
|
#define B_GTV(T,A,B) (B)
|
|
|
|
#endif
|
|
|
|
#define GTV(C,B,H) C_GTV(ischamber, C, B_GTV(isbed, B, H))
|
|
|
|
const uint16_t watch_temp_period = GTV(WATCH_CHAMBER_TEMP_PERIOD, WATCH_BED_TEMP_PERIOD, WATCH_TEMP_PERIOD);
|
|
|
|
const uint8_t watch_temp_increase = GTV(WATCH_CHAMBER_TEMP_INCREASE, WATCH_BED_TEMP_INCREASE, WATCH_TEMP_INCREASE);
|
2021-04-30 08:51:26 +02:00
|
|
|
const celsius_float_t watch_temp_target = celsius_float_t(target - (watch_temp_increase + GTV(TEMP_CHAMBER_HYSTERESIS, TEMP_BED_HYSTERESIS, TEMP_HYSTERESIS) + 1));
|
2020-04-03 19:49:45 -05:00
|
|
|
millis_t temp_change_ms = next_temp_ms + SEC_TO_MS(watch_temp_period);
|
2021-04-23 19:14:49 -05:00
|
|
|
celsius_float_t next_watch_temp = 0.0;
|
2017-11-03 03:24:19 -05:00
|
|
|
bool heated = false;
|
|
|
|
#endif
|
2012-09-12 21:01:31 -05:00
|
|
|
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(HAS_AUTO_FAN, next_auto_fan_check_ms = next_temp_ms + 2500UL);
|
2015-10-02 23:08:58 -07:00
|
|
|
|
2021-05-02 04:28:49 +03:00
|
|
|
TERN_(EXTENSIBLE_UI, ExtUI::onPidTuning(ExtUI::result_t::PID_STARTED));
|
2021-09-07 02:15:24 -05:00
|
|
|
TERN_(DWIN_CREALITY_LCD_ENHANCED, DWIN_PidTuning(isbed ? PID_BED_START : PID_EXTR_START));
|
2021-05-02 04:28:49 +03:00
|
|
|
|
2021-03-19 16:34:10 -05:00
|
|
|
if (target > GHV(CHAMBER_MAX_TARGET, BED_MAX_TARGET, temp_range[heater_id].maxtemp - (HOTEND_OVERSHOOT))) {
|
2020-02-26 03:02:03 -06:00
|
|
|
SERIAL_ECHOLNPGM(STR_PID_TEMP_TOO_HIGH);
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(EXTENSIBLE_UI, ExtUI::onPidTuning(ExtUI::result_t::PID_TEMP_TOO_HIGH));
|
2021-09-07 02:15:24 -05:00
|
|
|
TERN_(DWIN_CREALITY_LCD_ENHANCED, DWIN_PidTuning(PID_TEMP_TOO_HIGH));
|
2018-12-20 22:55:30 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-02-26 03:02:03 -06:00
|
|
|
SERIAL_ECHOLNPGM(STR_PID_AUTOTUNE_START);
|
2012-09-12 21:01:31 -05:00
|
|
|
|
2018-10-18 20:20:56 -06:00
|
|
|
disable_all_heaters();
|
2020-08-24 16:54:25 +02:00
|
|
|
TERN_(AUTO_POWER_CONTROL, powerManager.power_on());
|
2012-09-12 21:01:31 -05:00
|
|
|
|
2021-02-24 16:26:51 -08:00
|
|
|
long bias = GHV(MAX_CHAMBER_POWER, MAX_BED_POWER, PID_MAX) >> 1, d = bias;
|
|
|
|
SHV(bias);
|
2012-09-12 21:01:31 -05:00
|
|
|
|
2018-10-11 04:25:43 +02:00
|
|
|
#if ENABLED(PRINTER_EVENT_LEDS)
|
2021-04-23 19:06:55 -05:00
|
|
|
const celsius_float_t start_temp = GHV(degChamber(), degBed(), degHotend(heater_id));
|
2018-11-01 23:44:41 +01:00
|
|
|
LEDColor color = ONHEATINGSTART();
|
2018-10-11 04:25:43 +02:00
|
|
|
#endif
|
Add an emergency-command parser to MarlinSerial (supporting M108)
Add an emergency-command parser to MarlinSerial's RX interrupt.
The parser tries to find and execute M108,M112,M410 before the commands disappear in the RX-buffer.
To avoid false positives for M117, comments and commands followed by filenames (M23, M28, M30, M32, M33) are filtered.
This enables Marlin to receive and react on the Emergency command at all times - regardless of whether the buffers are full or not. It remains to convince hosts to send the commands. To inform the hosts about the new feature a new entry in the M115-report was made. "`EMERGENCY_CODES:M112,M108,M410;`".
The parser is fast. It only ever needs two switch decisions and one assignment of the new state for every character.
One problem remains. If the host has sent an incomplete line before sending an emergency command the emergency command could be omitted when the parser is in `state_IGNORE`.
In that case the host should send "\ncommand\n"
Also introduces M108 to break the waiting for the heaters in M109, M190 and M303.
Rename `cancel_heatup` to `wait_for_heatup` to better see the purpose.
2016-07-04 23:23:22 +02:00
|
|
|
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(NO_FAN_SLOWING_IN_PID_TUNING, adaptive_fan_slowing = false);
|
2019-02-04 07:12:41 +01:00
|
|
|
|
2016-04-18 19:02:11 -07:00
|
|
|
// PID Tuning loop
|
2020-09-12 05:51:19 +02:00
|
|
|
wait_for_heatup = true; // Can be interrupted with M108
|
2021-09-25 17:05:11 -05:00
|
|
|
TERN_(HAS_STATUS_MESSAGE, ui.set_status(F("Wait for heat up...")));
|
Add an emergency-command parser to MarlinSerial (supporting M108)
Add an emergency-command parser to MarlinSerial's RX interrupt.
The parser tries to find and execute M108,M112,M410 before the commands disappear in the RX-buffer.
To avoid false positives for M117, comments and commands followed by filenames (M23, M28, M30, M32, M33) are filtered.
This enables Marlin to receive and react on the Emergency command at all times - regardless of whether the buffers are full or not. It remains to convince hosts to send the commands. To inform the hosts about the new feature a new entry in the M115-report was made. "`EMERGENCY_CODES:M112,M108,M410;`".
The parser is fast. It only ever needs two switch decisions and one assignment of the new state for every character.
One problem remains. If the host has sent an incomplete line before sending an emergency command the emergency command could be omitted when the parser is in `state_IGNORE`.
In that case the host should send "\ncommand\n"
Also introduces M108 to break the waiting for the heaters in M109, M190 and M303.
Rename `cancel_heatup` to `wait_for_heatup` to better see the purpose.
2016-07-04 23:23:22 +02:00
|
|
|
while (wait_for_heatup) {
|
2012-03-08 21:43:21 +01:00
|
|
|
|
2017-10-29 04:30:50 -05:00
|
|
|
const millis_t ms = millis();
|
2012-12-09 23:00:06 +01:00
|
|
|
|
2021-05-08 17:34:21 -05:00
|
|
|
if (updateTemperaturesIfReady()) { // temp sample ready
|
2012-09-12 21:01:31 -05:00
|
|
|
|
2018-01-03 21:30:45 -06:00
|
|
|
// Get the current temperature and constrain it
|
2021-04-23 19:06:55 -05:00
|
|
|
current_temp = GHV(degChamber(), degBed(), degHotend(heater_id));
|
2019-09-09 02:56:23 -05:00
|
|
|
NOLESS(maxT, current_temp);
|
|
|
|
NOMORE(minT, current_temp);
|
2014-12-23 23:35:54 -05:00
|
|
|
|
2018-10-11 04:25:43 +02:00
|
|
|
#if ENABLED(PRINTER_EVENT_LEDS)
|
2019-09-09 02:56:23 -05:00
|
|
|
ONHEATING(start_temp, current_temp, target);
|
2018-10-11 04:25:43 +02:00
|
|
|
#endif
|
|
|
|
|
2016-04-18 19:02:11 -07:00
|
|
|
#if HAS_AUTO_FAN
|
|
|
|
if (ELAPSED(ms, next_auto_fan_check_ms)) {
|
|
|
|
checkExtruderAutoFans();
|
|
|
|
next_auto_fan_check_ms = ms + 2500UL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2021-02-24 16:26:51 -08:00
|
|
|
if (heating && current_temp > target && ELAPSED(ms, t2 + 5000UL)) {
|
|
|
|
heating = false;
|
|
|
|
SHV((bias - d) >> 1);
|
|
|
|
t1 = ms;
|
|
|
|
t_high = t1 - t2;
|
|
|
|
maxT = target;
|
2012-03-08 21:43:21 +01:00
|
|
|
}
|
2015-05-13 02:02:19 -07:00
|
|
|
|
2021-02-24 16:26:51 -08:00
|
|
|
if (!heating && current_temp < target && ELAPSED(ms, t1 + 5000UL)) {
|
|
|
|
heating = true;
|
|
|
|
t2 = ms;
|
|
|
|
t_low = t2 - t1;
|
|
|
|
if (cycles > 0) {
|
|
|
|
const long max_pow = GHV(MAX_CHAMBER_POWER, MAX_BED_POWER, PID_MAX);
|
|
|
|
bias += (d * (t_high - t_low)) / (t_low + t_high);
|
|
|
|
LIMIT(bias, 20, max_pow - 20);
|
|
|
|
d = (bias > max_pow >> 1) ? max_pow - 1 - bias : bias;
|
|
|
|
|
2021-09-09 04:57:05 -05:00
|
|
|
SERIAL_ECHOPGM(STR_BIAS, bias, STR_D_COLON, d, STR_T_MIN, minT, STR_T_MAX, maxT);
|
2021-02-24 16:26:51 -08:00
|
|
|
if (cycles > 2) {
|
|
|
|
const float Ku = (4.0f * d) / (float(M_PI) * (maxT - minT) * 0.5f),
|
|
|
|
Tu = float(t_low + t_high) * 0.001f,
|
2021-10-22 13:21:26 -05:00
|
|
|
pf = (ischamber || isbed) ? 0.2f : 0.6f,
|
|
|
|
df = (ischamber || isbed) ? 1.0f / 3.0f : 1.0f / 8.0f;
|
2021-02-24 16:26:51 -08:00
|
|
|
|
|
|
|
tune_pid.Kp = Ku * pf;
|
|
|
|
tune_pid.Ki = tune_pid.Kp * 2.0f / Tu;
|
|
|
|
tune_pid.Kd = tune_pid.Kp * Tu * df;
|
|
|
|
|
2021-09-09 04:57:05 -05:00
|
|
|
SERIAL_ECHOLNPGM(STR_KU, Ku, STR_TU, Tu);
|
2021-02-24 16:26:51 -08:00
|
|
|
if (ischamber || isbed)
|
|
|
|
SERIAL_ECHOLNPGM(" No overshoot");
|
|
|
|
else
|
|
|
|
SERIAL_ECHOLNPGM(STR_CLASSIC_PID);
|
2021-09-09 04:57:05 -05:00
|
|
|
SERIAL_ECHOLNPGM(STR_KP, tune_pid.Kp, STR_KI, tune_pid.Ki, STR_KD, tune_pid.Kd);
|
2012-03-08 21:43:21 +01:00
|
|
|
}
|
|
|
|
}
|
2021-02-24 16:26:51 -08:00
|
|
|
SHV((bias + d) >> 1);
|
2021-09-25 17:05:11 -05:00
|
|
|
TERN_(HAS_STATUS_MESSAGE, ui.status_printf(0, F(S_FMT " %i/%i"), GET_TEXT(MSG_PID_CYCLE), cycles, ncycles));
|
2021-02-24 16:26:51 -08:00
|
|
|
cycles++;
|
|
|
|
minT = target;
|
2012-03-08 21:43:21 +01:00
|
|
|
}
|
2015-10-02 23:08:58 -07:00
|
|
|
}
|
2018-01-03 21:30:45 -06:00
|
|
|
|
|
|
|
// Did the temperature overshoot very far?
|
2018-03-20 22:48:36 +01:00
|
|
|
#ifndef MAX_OVERSHOOT_PID_AUTOTUNE
|
2019-09-25 21:43:00 +07:00
|
|
|
#define MAX_OVERSHOOT_PID_AUTOTUNE 30
|
2018-03-20 22:48:36 +01:00
|
|
|
#endif
|
2019-09-09 02:56:23 -05:00
|
|
|
if (current_temp > target + MAX_OVERSHOOT_PID_AUTOTUNE) {
|
2020-02-26 03:02:03 -06:00
|
|
|
SERIAL_ECHOLNPGM(STR_PID_TEMP_TOO_HIGH);
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(EXTENSIBLE_UI, ExtUI::onPidTuning(ExtUI::result_t::PID_TEMP_TOO_HIGH));
|
2021-09-07 02:15:24 -05:00
|
|
|
TERN_(DWIN_CREALITY_LCD_ENHANCED, DWIN_PidTuning(PID_TEMP_TOO_HIGH));
|
2017-10-29 04:30:50 -05:00
|
|
|
break;
|
2016-04-18 19:02:11 -07:00
|
|
|
}
|
2018-01-03 21:30:45 -06:00
|
|
|
|
|
|
|
// Report heater states every 2 seconds
|
2017-10-29 04:30:50 -05:00
|
|
|
if (ELAPSED(ms, next_temp_ms)) {
|
2018-03-07 01:53:50 -06:00
|
|
|
#if HAS_TEMP_SENSOR
|
2021-02-24 16:26:51 -08:00
|
|
|
print_heater_states(ischamber ? active_extruder : (isbed ? active_extruder : heater_id));
|
2017-06-09 10:51:23 -05:00
|
|
|
SERIAL_EOL();
|
2016-04-18 19:02:11 -07:00
|
|
|
#endif
|
2017-10-29 04:30:50 -05:00
|
|
|
next_temp_ms = ms + 2000UL;
|
2017-11-03 03:24:19 -05:00
|
|
|
|
2018-01-03 21:30:45 -06:00
|
|
|
// Make sure heating is actually working
|
2020-05-18 11:51:32 -07:00
|
|
|
#if WATCH_PID
|
2021-02-24 16:26:51 -08:00
|
|
|
if (BOTH(WATCH_BED, WATCH_HOTENDS) || isbed == DISABLED(WATCH_HOTENDS) || ischamber == DISABLED(WATCH_HOTENDS)) {
|
2020-04-01 18:53:58 -05:00
|
|
|
if (!heated) { // If not yet reached target...
|
|
|
|
if (current_temp > next_watch_temp) { // Over the watch temp?
|
|
|
|
next_watch_temp = current_temp + watch_temp_increase; // - set the next temp to watch for
|
2021-10-22 13:21:26 -05:00
|
|
|
temp_change_ms = ms + SEC_TO_MS(watch_temp_period); // - move the expiration timer up
|
2020-04-01 18:53:58 -05:00
|
|
|
if (current_temp > watch_temp_target) heated = true; // - Flag if target temperature reached
|
2018-03-20 22:48:36 +01:00
|
|
|
}
|
2020-04-01 18:53:58 -05:00
|
|
|
else if (ELAPSED(ms, temp_change_ms)) // Watch timer expired
|
2021-09-28 20:15:52 -05:00
|
|
|
_temp_error(heater_id, FPSTR(str_t_heating_failed), GET_TEXT_F(MSG_HEATING_FAILED_LCD));
|
2018-01-03 21:30:45 -06:00
|
|
|
}
|
2019-09-09 02:56:23 -05:00
|
|
|
else if (current_temp < target - (MAX_OVERSHOOT_PID_AUTOTUNE)) // Heated, then temperature fell too far?
|
2021-09-28 20:15:52 -05:00
|
|
|
_temp_error(heater_id, FPSTR(str_t_thermal_runaway), GET_TEXT_F(MSG_THERMAL_RUNAWAY));
|
2017-11-03 03:24:19 -05:00
|
|
|
}
|
|
|
|
#endif
|
2016-04-18 19:02:11 -07:00
|
|
|
} // every 2 seconds
|
2018-01-03 21:30:45 -06:00
|
|
|
|
2018-03-20 22:48:36 +01:00
|
|
|
// Timeout after MAX_CYCLE_TIME_PID_AUTOTUNE minutes since the last undershoot/overshoot cycle
|
|
|
|
#ifndef MAX_CYCLE_TIME_PID_AUTOTUNE
|
|
|
|
#define MAX_CYCLE_TIME_PID_AUTOTUNE 20L
|
|
|
|
#endif
|
2020-06-18 23:05:12 +02:00
|
|
|
if ((ms - _MIN(t1, t2)) > (MAX_CYCLE_TIME_PID_AUTOTUNE * 60L * 1000L)) {
|
2020-09-17 06:35:04 -04:00
|
|
|
TERN_(DWIN_CREALITY_LCD, DWIN_Popup_Temperature(0));
|
2021-09-07 02:15:24 -05:00
|
|
|
TERN_(DWIN_CREALITY_LCD_ENHANCED, DWIN_PidTuning(PID_TUNING_TIMEOUT));
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(EXTENSIBLE_UI, ExtUI::onPidTuning(ExtUI::result_t::PID_TUNING_TIMEOUT));
|
2020-02-26 03:02:03 -06:00
|
|
|
SERIAL_ECHOLNPGM(STR_PID_TIMEOUT);
|
2017-10-29 04:30:50 -05:00
|
|
|
break;
|
2016-04-18 19:02:11 -07:00
|
|
|
}
|
2018-01-03 21:30:45 -06:00
|
|
|
|
2019-02-10 10:54:45 +01:00
|
|
|
if (cycles > ncycles && cycles > 2) {
|
2020-02-26 03:02:03 -06:00
|
|
|
SERIAL_ECHOLNPGM(STR_PID_AUTOTUNE_FINISHED);
|
2016-04-19 18:09:29 -07:00
|
|
|
|
2021-02-24 16:26:51 -08:00
|
|
|
#if EITHER(PIDTEMPBED, PIDTEMPCHAMBER)
|
2021-09-27 11:03:07 -05:00
|
|
|
FSTR_P const estring = GHV(F("chamber"), F("bed"), FPSTR(NUL_STR));
|
|
|
|
say_default_(); SERIAL_ECHOF(estring); SERIAL_ECHOLNPGM("Kp ", tune_pid.Kp);
|
|
|
|
say_default_(); SERIAL_ECHOF(estring); SERIAL_ECHOLNPGM("Ki ", tune_pid.Ki);
|
|
|
|
say_default_(); SERIAL_ECHOF(estring); SERIAL_ECHOLNPGM("Kd ", tune_pid.Kd);
|
2021-02-24 16:26:51 -08:00
|
|
|
#else
|
2021-09-09 04:57:05 -05:00
|
|
|
say_default_(); SERIAL_ECHOLNPGM("Kp ", tune_pid.Kp);
|
|
|
|
say_default_(); SERIAL_ECHOLNPGM("Ki ", tune_pid.Ki);
|
|
|
|
say_default_(); SERIAL_ECHOLNPGM("Kd ", tune_pid.Kd);
|
2016-04-19 18:09:29 -07:00
|
|
|
#endif
|
|
|
|
|
2021-02-24 16:26:51 -08:00
|
|
|
auto _set_hotend_pid = [](const uint8_t e, const PID_t &in_pid) {
|
|
|
|
#if ENABLED(PIDTEMP)
|
|
|
|
PID_PARAM(Kp, e) = in_pid.Kp;
|
|
|
|
PID_PARAM(Ki, e) = scalePID_i(in_pid.Ki);
|
|
|
|
PID_PARAM(Kd, e) = scalePID_d(in_pid.Kd);
|
|
|
|
updatePID();
|
|
|
|
#else
|
|
|
|
UNUSED(e); UNUSED(in_pid);
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
|
|
|
#if ENABLED(PIDTEMPBED)
|
|
|
|
auto _set_bed_pid = [](const PID_t &in_pid) {
|
|
|
|
temp_bed.pid.Kp = in_pid.Kp;
|
|
|
|
temp_bed.pid.Ki = scalePID_i(in_pid.Ki);
|
|
|
|
temp_bed.pid.Kd = scalePID_d(in_pid.Kd);
|
|
|
|
};
|
|
|
|
#endif
|
2016-04-19 18:09:29 -07:00
|
|
|
|
2021-02-24 16:26:51 -08:00
|
|
|
#if ENABLED(PIDTEMPCHAMBER)
|
|
|
|
auto _set_chamber_pid = [](const PID_t &in_pid) {
|
|
|
|
temp_chamber.pid.Kp = in_pid.Kp;
|
|
|
|
temp_chamber.pid.Ki = scalePID_i(in_pid.Ki);
|
|
|
|
temp_chamber.pid.Kd = scalePID_d(in_pid.Kd);
|
|
|
|
};
|
|
|
|
#endif
|
2016-04-18 19:02:11 -07:00
|
|
|
|
|
|
|
// Use the result? (As with "M303 U1")
|
2021-02-24 16:26:51 -08:00
|
|
|
if (set_result)
|
|
|
|
GHV(_set_chamber_pid(tune_pid), _set_bed_pid(tune_pid), _set_hotend_pid(heater_id, tune_pid));
|
2019-02-04 07:12:41 +01:00
|
|
|
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(PRINTER_EVENT_LEDS, printerEventLEDs.onPidTuningDone(color));
|
|
|
|
|
|
|
|
TERN_(EXTENSIBLE_UI, ExtUI::onPidTuning(ExtUI::result_t::PID_DONE));
|
2021-09-07 02:15:24 -05:00
|
|
|
TERN_(DWIN_CREALITY_LCD_ENHANCED, DWIN_PidTuning(PID_DONE));
|
2018-11-01 23:44:41 +01:00
|
|
|
|
2019-02-04 07:12:41 +01:00
|
|
|
goto EXIT_M303;
|
2016-03-20 14:56:11 +01:00
|
|
|
}
|
2020-10-10 01:35:41 -03:00
|
|
|
|
|
|
|
// Run HAL idle tasks
|
|
|
|
TERN_(HAL_IDLETASK, HAL_idletask());
|
|
|
|
|
|
|
|
// Run UI update
|
2021-09-07 02:15:24 -05:00
|
|
|
TERN(HAS_DWIN_E3V2_BASIC, DWIN_Update(), ui.update());
|
2012-04-15 19:17:33 +02:00
|
|
|
}
|
2020-09-12 05:51:19 +02:00
|
|
|
wait_for_heatup = false;
|
2019-02-04 07:12:41 +01:00
|
|
|
|
2017-10-29 04:30:50 -05:00
|
|
|
disable_all_heaters();
|
2019-02-04 07:12:41 +01:00
|
|
|
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(PRINTER_EVENT_LEDS, printerEventLEDs.onPidTuningDone(color));
|
|
|
|
|
|
|
|
TERN_(EXTENSIBLE_UI, ExtUI::onPidTuning(ExtUI::result_t::PID_DONE));
|
2021-09-07 02:15:24 -05:00
|
|
|
TERN_(DWIN_CREALITY_LCD_ENHANCED, DWIN_PidTuning(PID_DONE));
|
2019-02-04 07:12:41 +01:00
|
|
|
|
|
|
|
EXIT_M303:
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(NO_FAN_SLOWING_IN_PID_TUNING, adaptive_fan_slowing = true);
|
2019-02-04 07:12:41 +01:00
|
|
|
return;
|
2012-03-08 21:43:21 +01:00
|
|
|
}
|
|
|
|
|
2016-04-28 18:18:13 -07:00
|
|
|
#endif // HAS_PID_HEATING
|
|
|
|
|
2020-09-13 18:06:14 -05:00
|
|
|
int16_t Temperature::getHeaterPower(const heater_id_t heater_id) {
|
2019-07-02 08:39:55 -05:00
|
|
|
switch (heater_id) {
|
2018-04-23 17:13:01 -05:00
|
|
|
#if HAS_HEATED_BED
|
2019-07-02 08:39:55 -05:00
|
|
|
case H_BED: return temp_bed.soft_pwm_amount;
|
|
|
|
#endif
|
|
|
|
#if HAS_HEATED_CHAMBER
|
|
|
|
case H_CHAMBER: return temp_chamber.soft_pwm_amount;
|
2018-04-23 17:13:01 -05:00
|
|
|
#endif
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_COOLER
|
|
|
|
case H_COOLER: return temp_cooler.soft_pwm_amount;
|
|
|
|
#endif
|
2019-09-10 02:20:49 -05:00
|
|
|
default:
|
2020-04-22 16:35:03 -05:00
|
|
|
return TERN0(HAS_HOTEND, temp_hotend[heater_id].soft_pwm_amount);
|
2019-07-02 08:39:55 -05:00
|
|
|
}
|
2011-12-12 19:34:37 +01:00
|
|
|
}
|
|
|
|
|
2019-10-09 18:46:10 -06:00
|
|
|
#define _EFANOVERLAP(A,B) _FANOVERLAP(E##A,B)
|
|
|
|
|
2015-02-26 01:14:59 -08:00
|
|
|
#if HAS_AUTO_FAN
|
2021-07-06 17:36:41 -07:00
|
|
|
#if EXTRUDER_AUTO_FAN_SPEED != 255
|
|
|
|
#define INIT_E_AUTO_FAN_PIN(P) do{ if (P == FAN1_PIN || P == FAN2_PIN) { SET_PWM(P); SET_FAST_PWM_FREQ(P); } else SET_OUTPUT(P); }while(0)
|
|
|
|
#else
|
|
|
|
#define INIT_E_AUTO_FAN_PIN(P) SET_OUTPUT(P)
|
|
|
|
#endif
|
|
|
|
#if CHAMBER_AUTO_FAN_SPEED != 255
|
|
|
|
#define INIT_CHAMBER_AUTO_FAN_PIN(P) do{ if (P == FAN1_PIN || P == FAN2_PIN) { SET_PWM(P); SET_FAST_PWM_FREQ(P); } else SET_OUTPUT(P); }while(0)
|
|
|
|
#else
|
|
|
|
#define INIT_CHAMBER_AUTO_FAN_PIN(P) SET_OUTPUT(P)
|
|
|
|
#endif
|
2016-10-09 16:13:58 -05:00
|
|
|
|
2019-05-05 05:51:47 +02:00
|
|
|
#define CHAMBER_FAN_INDEX HOTENDS
|
2019-03-13 06:51:15 -05:00
|
|
|
|
2016-04-28 18:18:13 -07:00
|
|
|
void Temperature::checkExtruderAutoFans() {
|
2019-11-09 17:59:04 -06:00
|
|
|
#define _EFAN(B,A) _EFANOVERLAP(A,B) ? B :
|
2017-06-06 06:33:06 -05:00
|
|
|
static const uint8_t fanBit[] PROGMEM = {
|
2019-05-05 05:51:47 +02:00
|
|
|
0
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_MULTI_HOTEND
|
2020-02-01 21:00:53 -06:00
|
|
|
#define _NEXT_FAN(N) , REPEAT2(N,_EFAN,N) N
|
|
|
|
RREPEAT_S(1, HOTENDS, _NEXT_FAN)
|
2019-05-05 05:51:47 +02:00
|
|
|
#endif
|
|
|
|
#if HAS_AUTO_CHAMBER_FAN
|
2019-10-09 18:46:10 -06:00
|
|
|
#define _CFAN(B) _FANOVERLAP(CHAMBER,B) ? B :
|
2019-11-09 17:59:04 -06:00
|
|
|
, REPEAT(HOTENDS,_CFAN) (HOTENDS)
|
2019-03-05 00:41:31 -06:00
|
|
|
#endif
|
2016-04-28 18:18:13 -07:00
|
|
|
};
|
2016-10-29 13:42:43 -05:00
|
|
|
|
2019-11-09 17:59:04 -06:00
|
|
|
uint8_t fanState = 0;
|
2021-11-27 17:58:05 -05:00
|
|
|
HOTEND_LOOP() {
|
|
|
|
if (temp_hotend[e].celsius >= EXTRUDER_AUTO_FAN_TEMPERATURE) {
|
2017-06-06 06:33:06 -05:00
|
|
|
SBI(fanState, pgm_read_byte(&fanBit[e]));
|
2021-11-27 17:58:05 -05:00
|
|
|
#if MOTHERBOARD == BOARD_ULTIMAIN_2
|
|
|
|
// For the UM2 the head fan is connected to PJ6, which does not have an Arduino PIN definition. So use direct register access.
|
|
|
|
// https://github.com/Ultimaker/Ultimaker2Marlin/blob/master/Marlin/temperature.cpp#L553
|
|
|
|
SBI(DDRJ, 6); SBI(PORTJ, 6);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
#if MOTHERBOARD == BOARD_ULTIMAIN_2
|
|
|
|
SBI(DDRJ, 6); CBI(PORTJ, 6);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
2018-04-24 07:55:58 -05:00
|
|
|
|
2019-05-05 05:51:47 +02:00
|
|
|
#if HAS_AUTO_CHAMBER_FAN
|
2019-09-09 02:56:23 -05:00
|
|
|
if (temp_chamber.celsius >= CHAMBER_AUTO_FAN_TEMPERATURE)
|
2019-05-05 05:51:47 +02:00
|
|
|
SBI(fanState, pgm_read_byte(&fanBit[CHAMBER_FAN_INDEX]));
|
2018-04-24 07:55:58 -05:00
|
|
|
#endif
|
2016-10-29 13:42:43 -05:00
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_AUTO_COOLER_FAN
|
|
|
|
if (temp_cooler.celsius >= COOLER_AUTO_FAN_TEMPERATURE)
|
|
|
|
SBI(fanState, pgm_read_byte(&fanBit[COOLER_FAN_INDEX]));
|
|
|
|
#endif
|
|
|
|
|
2021-11-02 01:47:16 -04:00
|
|
|
#define _UPDATE_AUTO_FAN(P,D,A) do{ \
|
|
|
|
if (PWM_PIN(P##_AUTO_FAN_PIN) && A < 255) \
|
|
|
|
set_pwm_duty(pin_t(P##_AUTO_FAN_PIN), D ? A : 0); \
|
|
|
|
else \
|
|
|
|
WRITE(P##_AUTO_FAN_PIN, D); \
|
2019-03-05 00:41:31 -06:00
|
|
|
}while(0)
|
|
|
|
|
2016-05-15 18:54:37 -07:00
|
|
|
uint8_t fanDone = 0;
|
2020-03-13 23:18:16 -05:00
|
|
|
LOOP_L_N(f, COUNT(fanBit)) {
|
2019-05-05 05:51:47 +02:00
|
|
|
const uint8_t realFan = pgm_read_byte(&fanBit[f]);
|
|
|
|
if (TEST(fanDone, realFan)) continue;
|
|
|
|
const bool fan_on = TEST(fanState, realFan);
|
|
|
|
switch (f) {
|
2019-08-16 16:42:24 -07:00
|
|
|
#if ENABLED(AUTO_POWER_CHAMBER_FAN)
|
2019-05-05 05:51:47 +02:00
|
|
|
case CHAMBER_FAN_INDEX:
|
|
|
|
chamberfan_speed = fan_on ? CHAMBER_AUTO_FAN_SPEED : 0;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
default:
|
|
|
|
#if ENABLED(AUTO_POWER_E_FANS)
|
|
|
|
autofan_speed[realFan] = fan_on ? EXTRUDER_AUTO_FAN_SPEED : 0;
|
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-03-05 00:41:31 -06:00
|
|
|
switch (f) {
|
|
|
|
#if HAS_AUTO_FAN_0
|
2019-05-05 05:51:47 +02:00
|
|
|
case 0: _UPDATE_AUTO_FAN(E0, fan_on, EXTRUDER_AUTO_FAN_SPEED); break;
|
2019-03-05 00:41:31 -06:00
|
|
|
#endif
|
|
|
|
#if HAS_AUTO_FAN_1
|
2019-05-05 05:51:47 +02:00
|
|
|
case 1: _UPDATE_AUTO_FAN(E1, fan_on, EXTRUDER_AUTO_FAN_SPEED); break;
|
2019-03-05 00:41:31 -06:00
|
|
|
#endif
|
|
|
|
#if HAS_AUTO_FAN_2
|
2019-05-05 05:51:47 +02:00
|
|
|
case 2: _UPDATE_AUTO_FAN(E2, fan_on, EXTRUDER_AUTO_FAN_SPEED); break;
|
2019-03-05 00:41:31 -06:00
|
|
|
#endif
|
|
|
|
#if HAS_AUTO_FAN_3
|
2019-05-05 05:51:47 +02:00
|
|
|
case 3: _UPDATE_AUTO_FAN(E3, fan_on, EXTRUDER_AUTO_FAN_SPEED); break;
|
2019-03-05 00:41:31 -06:00
|
|
|
#endif
|
|
|
|
#if HAS_AUTO_FAN_4
|
2019-05-05 05:51:47 +02:00
|
|
|
case 4: _UPDATE_AUTO_FAN(E4, fan_on, EXTRUDER_AUTO_FAN_SPEED); break;
|
2019-03-05 00:41:31 -06:00
|
|
|
#endif
|
|
|
|
#if HAS_AUTO_FAN_5
|
2019-05-05 05:51:47 +02:00
|
|
|
case 5: _UPDATE_AUTO_FAN(E5, fan_on, EXTRUDER_AUTO_FAN_SPEED); break;
|
2019-03-05 00:41:31 -06:00
|
|
|
#endif
|
2020-01-25 16:13:39 +08:00
|
|
|
#if HAS_AUTO_FAN_6
|
|
|
|
case 6: _UPDATE_AUTO_FAN(E6, fan_on, EXTRUDER_AUTO_FAN_SPEED); break;
|
|
|
|
#endif
|
|
|
|
#if HAS_AUTO_FAN_7
|
|
|
|
case 7: _UPDATE_AUTO_FAN(E7, fan_on, EXTRUDER_AUTO_FAN_SPEED); break;
|
|
|
|
#endif
|
2019-05-05 05:51:47 +02:00
|
|
|
#if HAS_AUTO_CHAMBER_FAN && !AUTO_CHAMBER_IS_E
|
|
|
|
case CHAMBER_FAN_INDEX: _UPDATE_AUTO_FAN(CHAMBER, fan_on, CHAMBER_AUTO_FAN_SPEED); break;
|
2018-02-06 00:22:30 -06:00
|
|
|
#endif
|
2016-05-14 14:46:14 -07:00
|
|
|
}
|
2019-05-05 05:51:47 +02:00
|
|
|
SBI(fanDone, realFan);
|
2015-01-23 23:13:06 +01:00
|
|
|
}
|
2016-04-28 18:18:13 -07:00
|
|
|
}
|
2013-06-06 15:49:25 -07:00
|
|
|
|
2015-06-09 17:01:47 -07:00
|
|
|
#endif // HAS_AUTO_FAN
|
2013-06-06 15:49:25 -07:00
|
|
|
|
2015-02-24 04:46:11 -08:00
|
|
|
//
|
2015-03-19 22:22:23 -07:00
|
|
|
// Temperature Error Handlers
|
2015-02-24 04:46:11 -08:00
|
|
|
//
|
2019-07-09 21:35:07 -05:00
|
|
|
|
2021-09-28 20:15:52 -05:00
|
|
|
inline void loud_kill(FSTR_P const lcd_msg, const heater_id_t heater_id) {
|
2020-03-14 18:47:44 -05:00
|
|
|
marlin_state = MF_KILLED;
|
2021-12-19 05:33:21 +07:00
|
|
|
thermalManager.disable_all_heaters();
|
2019-08-20 09:01:37 +02:00
|
|
|
#if USE_BEEPER
|
2019-07-09 21:35:07 -05:00
|
|
|
for (uint8_t i = 20; i--;) {
|
2021-01-03 20:39:15 -07:00
|
|
|
WRITE(BEEPER_PIN, HIGH);
|
|
|
|
delay(25);
|
|
|
|
watchdog_refresh();
|
|
|
|
WRITE(BEEPER_PIN, LOW);
|
|
|
|
delay(40);
|
|
|
|
watchdog_refresh();
|
|
|
|
delay(40);
|
|
|
|
watchdog_refresh();
|
2019-07-09 21:35:07 -05:00
|
|
|
}
|
|
|
|
WRITE(BEEPER_PIN, HIGH);
|
|
|
|
#endif
|
2021-12-05 05:14:19 +07:00
|
|
|
#if ENABLED(NOZZLE_PARK_FEATURE)
|
|
|
|
if (!homing_needed_error()) {
|
|
|
|
nozzle.park(0);
|
|
|
|
planner.synchronize();
|
|
|
|
}
|
|
|
|
#endif
|
2021-09-28 20:15:52 -05:00
|
|
|
kill(lcd_msg, HEATER_FSTR(heater_id));
|
2019-07-09 21:35:07 -05:00
|
|
|
}
|
|
|
|
|
2021-09-28 20:15:52 -05:00
|
|
|
void Temperature::_temp_error(const heater_id_t heater_id, FSTR_P const serial_msg, FSTR_P const lcd_msg) {
|
2019-07-09 21:35:07 -05:00
|
|
|
|
|
|
|
static uint8_t killed = 0;
|
|
|
|
|
2020-04-22 16:35:03 -05:00
|
|
|
if (IsRunning() && TERN1(BOGUS_TEMPERATURE_GRACE_PERIOD, killed == 2)) {
|
2017-06-09 10:51:23 -05:00
|
|
|
SERIAL_ERROR_START();
|
2021-09-28 20:15:52 -05:00
|
|
|
SERIAL_ECHOF(serial_msg);
|
2020-02-26 03:02:03 -06:00
|
|
|
SERIAL_ECHOPGM(STR_STOPPED_HEATER);
|
2021-08-18 20:12:41 -05:00
|
|
|
|
|
|
|
heater_id_t real_heater_id = heater_id;
|
|
|
|
|
|
|
|
#if HAS_TEMP_REDUNDANT
|
|
|
|
if (heater_id == H_REDUNDANT) {
|
|
|
|
SERIAL_ECHOPGM(STR_REDUNDANT); // print redundant and cascade to print target, too.
|
|
|
|
real_heater_id = (heater_id_t)HEATER_ID(TEMP_SENSOR_REDUNDANT_TARGET);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
switch (real_heater_id) {
|
|
|
|
OPTCODE(HAS_TEMP_COOLER, case H_COOLER: SERIAL_ECHOPGM(STR_COOLER); break)
|
|
|
|
OPTCODE(HAS_TEMP_PROBE, case H_PROBE: SERIAL_ECHOPGM(STR_PROBE); break)
|
|
|
|
OPTCODE(HAS_TEMP_BOARD, case H_BOARD: SERIAL_ECHOPGM(STR_MOTHERBOARD); break)
|
|
|
|
OPTCODE(HAS_TEMP_CHAMBER, case H_CHAMBER: SERIAL_ECHOPGM(STR_HEATER_CHAMBER); break)
|
|
|
|
OPTCODE(HAS_TEMP_BED, case H_BED: SERIAL_ECHOPGM(STR_HEATER_BED); break)
|
|
|
|
default:
|
|
|
|
if (real_heater_id >= 0)
|
2021-09-09 04:57:05 -05:00
|
|
|
SERIAL_ECHOLNPGM("E", real_heater_id);
|
2021-08-18 20:12:41 -05:00
|
|
|
}
|
2019-03-07 00:09:39 -08:00
|
|
|
SERIAL_EOL();
|
2015-03-11 16:38:42 -07:00
|
|
|
}
|
2019-05-22 03:31:05 +01:00
|
|
|
|
2019-07-09 21:35:07 -05:00
|
|
|
disable_all_heaters(); // always disable (even for bogus temp)
|
2021-01-03 20:39:15 -07:00
|
|
|
watchdog_refresh();
|
2019-05-22 03:31:05 +01:00
|
|
|
|
2019-07-09 21:35:07 -05:00
|
|
|
#if BOGUS_TEMPERATURE_GRACE_PERIOD
|
|
|
|
const millis_t ms = millis();
|
|
|
|
static millis_t expire_ms;
|
|
|
|
switch (killed) {
|
|
|
|
case 0:
|
|
|
|
expire_ms = ms + BOGUS_TEMPERATURE_GRACE_PERIOD;
|
|
|
|
++killed;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
if (ELAPSED(ms, expire_ms)) ++killed;
|
|
|
|
break;
|
|
|
|
case 2:
|
2020-09-13 18:06:14 -05:00
|
|
|
loud_kill(lcd_msg, heater_id);
|
2019-07-09 21:35:07 -05:00
|
|
|
++killed;
|
|
|
|
break;
|
2015-05-25 12:44:03 +02:00
|
|
|
}
|
2019-07-09 21:35:07 -05:00
|
|
|
#elif defined(BOGUS_TEMPERATURE_GRACE_PERIOD)
|
2019-06-18 07:15:41 +02:00
|
|
|
UNUSED(killed);
|
2019-07-09 21:35:07 -05:00
|
|
|
#else
|
2020-09-13 18:06:14 -05:00
|
|
|
if (!killed) { killed = 1; loud_kill(lcd_msg, heater_id); }
|
2015-03-11 16:38:42 -07:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2020-09-13 18:06:14 -05:00
|
|
|
void Temperature::max_temp_error(const heater_id_t heater_id) {
|
2021-09-07 02:15:24 -05:00
|
|
|
#if HAS_DWIN_E3V2_BASIC && (HAS_HOTEND || HAS_HEATED_BED)
|
2020-09-17 06:35:04 -04:00
|
|
|
DWIN_Popup_Temperature(1);
|
|
|
|
#endif
|
2021-09-28 20:15:52 -05:00
|
|
|
_temp_error(heater_id, F(STR_T_MAXTEMP), GET_TEXT_F(MSG_ERR_MAXTEMP));
|
2015-03-11 16:38:42 -07:00
|
|
|
}
|
2018-04-23 17:13:01 -05:00
|
|
|
|
2020-09-13 18:06:14 -05:00
|
|
|
void Temperature::min_temp_error(const heater_id_t heater_id) {
|
2021-09-07 02:15:24 -05:00
|
|
|
#if HAS_DWIN_E3V2_BASIC && (HAS_HOTEND || HAS_HEATED_BED)
|
2020-09-17 06:35:04 -04:00
|
|
|
DWIN_Popup_Temperature(0);
|
|
|
|
#endif
|
2021-09-28 20:15:52 -05:00
|
|
|
_temp_error(heater_id, F(STR_T_MINTEMP), GET_TEXT_F(MSG_ERR_MINTEMP));
|
2015-03-11 16:38:42 -07:00
|
|
|
}
|
|
|
|
|
2021-02-24 16:26:51 -08:00
|
|
|
#if ANY(PID_DEBUG, PID_BED_DEBUG, PID_CHAMBER_DEBUG)
|
|
|
|
bool Temperature::pid_debug_flag; // = 0
|
|
|
|
#endif
|
|
|
|
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_HOTEND
|
2019-09-10 02:20:49 -05:00
|
|
|
|
2019-09-30 21:44:07 -05:00
|
|
|
float Temperature::get_pid_output_hotend(const uint8_t E_NAME) {
|
2019-09-10 02:20:49 -05:00
|
|
|
const uint8_t ee = HOTEND_INDEX;
|
|
|
|
#if ENABLED(PIDTEMP)
|
|
|
|
#if DISABLED(PID_OPENLOOP)
|
|
|
|
static hotend_pid_t work_pid[HOTENDS];
|
|
|
|
static float temp_iState[HOTENDS] = { 0 },
|
|
|
|
temp_dState[HOTENDS] = { 0 };
|
|
|
|
static bool pid_reset[HOTENDS] = { false };
|
|
|
|
const float pid_error = temp_hotend[ee].target - temp_hotend[ee].celsius;
|
|
|
|
|
2019-09-24 22:30:13 -05:00
|
|
|
float pid_output;
|
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
if (temp_hotend[ee].target == 0
|
|
|
|
|| pid_error < -(PID_FUNCTIONAL_RANGE)
|
2020-09-14 16:58:39 +12:00
|
|
|
|| TERN0(HEATER_IDLE_HANDLER, heater_idle[ee].timed_out)
|
2019-09-10 02:20:49 -05:00
|
|
|
) {
|
|
|
|
pid_output = 0;
|
|
|
|
pid_reset[ee] = true;
|
2017-05-26 13:01:02 -05:00
|
|
|
}
|
2019-09-10 02:20:49 -05:00
|
|
|
else if (pid_error > PID_FUNCTIONAL_RANGE) {
|
|
|
|
pid_output = BANG_MAX;
|
|
|
|
pid_reset[ee] = true;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (pid_reset[ee]) {
|
|
|
|
temp_iState[ee] = 0.0;
|
|
|
|
work_pid[ee].Kd = 0.0;
|
|
|
|
pid_reset[ee] = false;
|
|
|
|
}
|
2019-06-22 16:52:56 -05:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
work_pid[ee].Kd = work_pid[ee].Kd + PID_K2 * (PID_PARAM(Kd, ee) * (temp_dState[ee] - temp_hotend[ee].celsius) - work_pid[ee].Kd);
|
|
|
|
const float max_power_over_i_gain = float(PID_MAX) / PID_PARAM(Ki, ee) - float(MIN_POWER);
|
|
|
|
temp_iState[ee] = constrain(temp_iState[ee] + pid_error, 0, max_power_over_i_gain);
|
|
|
|
work_pid[ee].Kp = PID_PARAM(Kp, ee) * pid_error;
|
|
|
|
work_pid[ee].Ki = PID_PARAM(Ki, ee) * temp_iState[ee];
|
|
|
|
|
|
|
|
pid_output = work_pid[ee].Kp + work_pid[ee].Ki + work_pid[ee].Kd + float(MIN_POWER);
|
|
|
|
|
|
|
|
#if ENABLED(PID_EXTRUSION_SCALING)
|
2019-09-30 21:44:07 -05:00
|
|
|
#if HOTENDS == 1
|
|
|
|
constexpr bool this_hotend = true;
|
|
|
|
#else
|
2019-10-06 01:27:03 +02:00
|
|
|
const bool this_hotend = (ee == active_extruder);
|
2019-09-30 21:44:07 -05:00
|
|
|
#endif
|
2019-09-10 02:20:49 -05:00
|
|
|
work_pid[ee].Kc = 0;
|
2019-09-30 21:44:07 -05:00
|
|
|
if (this_hotend) {
|
2019-09-10 02:20:49 -05:00
|
|
|
const long e_position = stepper.position(E_AXIS);
|
|
|
|
if (e_position > last_e_position) {
|
|
|
|
lpq[lpq_ptr] = e_position - last_e_position;
|
|
|
|
last_e_position = e_position;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
lpq[lpq_ptr] = 0;
|
2019-03-03 05:29:55 -06:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
if (++lpq_ptr >= lpq_len) lpq_ptr = 0;
|
2021-09-27 21:05:52 +02:00
|
|
|
work_pid[ee].Kc = (lpq[lpq_ptr] * planner.mm_per_step[E_AXIS]) * PID_PARAM(Kc, ee);
|
2019-09-10 02:20:49 -05:00
|
|
|
pid_output += work_pid[ee].Kc;
|
|
|
|
}
|
|
|
|
#endif // PID_EXTRUSION_SCALING
|
2019-11-26 10:34:43 +01:00
|
|
|
#if ENABLED(PID_FAN_SCALING)
|
2021-01-03 20:39:15 -07:00
|
|
|
if (fan_speed[active_extruder] > PID_FAN_SCALING_MIN_SPEED) {
|
|
|
|
work_pid[ee].Kf = PID_PARAM(Kf, ee) + (PID_FAN_SCALING_LIN_FACTOR) * fan_speed[active_extruder];
|
2019-11-26 10:34:43 +01:00
|
|
|
pid_output += work_pid[ee].Kf;
|
|
|
|
}
|
|
|
|
//pid_output -= work_pid[ee].Ki;
|
|
|
|
//pid_output += work_pid[ee].Ki * work_pid[ee].Kf
|
|
|
|
#endif // PID_FAN_SCALING
|
2019-09-10 02:20:49 -05:00
|
|
|
LIMIT(pid_output, 0, PID_MAX);
|
|
|
|
}
|
|
|
|
temp_dState[ee] = temp_hotend[ee].celsius;
|
2015-08-30 19:04:30 -07:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
#else // PID_OPENLOOP
|
2018-10-10 09:45:20 -05:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
const float pid_output = constrain(temp_hotend[ee].target, 0, PID_MAX);
|
2018-10-10 09:45:20 -05:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
#endif // PID_OPENLOOP
|
2015-03-13 19:49:41 -07:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
#if ENABLED(PID_DEBUG)
|
2020-04-11 11:03:04 -05:00
|
|
|
if (ee == active_extruder && pid_debug_flag) {
|
2021-02-24 16:26:51 -08:00
|
|
|
SERIAL_ECHO_MSG(STR_PID_DEBUG, ee, STR_PID_DEBUG_INPUT, temp_hotend[ee].celsius, STR_PID_DEBUG_OUTPUT, pid_output
|
|
|
|
#if DISABLED(PID_OPENLOOP)
|
|
|
|
, STR_PID_DEBUG_PTERM, work_pid[ee].Kp
|
|
|
|
, STR_PID_DEBUG_ITERM, work_pid[ee].Ki
|
|
|
|
, STR_PID_DEBUG_DTERM, work_pid[ee].Kd
|
2019-09-10 02:20:49 -05:00
|
|
|
#if ENABLED(PID_EXTRUSION_SCALING)
|
2020-02-26 03:02:03 -06:00
|
|
|
, STR_PID_DEBUG_CTERM, work_pid[ee].Kc
|
2019-09-10 02:20:49 -05:00
|
|
|
#endif
|
2021-02-24 16:26:51 -08:00
|
|
|
#endif
|
|
|
|
);
|
2019-09-10 02:20:49 -05:00
|
|
|
}
|
2021-02-24 16:26:51 -08:00
|
|
|
#endif
|
2015-03-13 19:49:41 -07:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
#else // No PID enabled
|
|
|
|
|
2020-09-14 16:58:39 +12:00
|
|
|
const bool is_idling = TERN0(HEATER_IDLE_HANDLER, heater_idle[ee].timed_out);
|
2019-09-30 21:44:07 -05:00
|
|
|
const float pid_output = (!is_idling && temp_hotend[ee].celsius < temp_hotend[ee].target) ? BANG_MAX : 0;
|
2019-07-02 08:39:55 -05:00
|
|
|
|
2017-05-26 13:01:02 -05:00
|
|
|
#endif
|
2019-07-02 08:39:55 -05:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
return pid_output;
|
|
|
|
}
|
2015-03-13 19:49:41 -07:00
|
|
|
|
2020-07-01 16:27:28 -05:00
|
|
|
#endif // HAS_HOTEND
|
2015-03-13 19:49:41 -07:00
|
|
|
|
2015-07-30 22:29:30 -07:00
|
|
|
#if ENABLED(PIDTEMPBED)
|
2018-10-10 09:45:20 -05:00
|
|
|
|
2016-04-28 18:18:13 -07:00
|
|
|
float Temperature::get_pid_output_bed() {
|
2018-10-10 09:45:20 -05:00
|
|
|
|
2015-07-30 22:29:30 -07:00
|
|
|
#if DISABLED(PID_OPENLOOP)
|
2015-03-13 19:49:41 -07:00
|
|
|
|
2019-09-17 18:16:28 -05:00
|
|
|
static PID_t work_pid{0};
|
2018-10-10 09:45:20 -05:00
|
|
|
static float temp_iState = 0, temp_dState = 0;
|
2019-07-28 15:50:25 -05:00
|
|
|
static bool pid_reset = true;
|
|
|
|
float pid_output = 0;
|
|
|
|
const float max_power_over_i_gain = float(MAX_BED_POWER) / temp_bed.pid.Ki - float(MIN_BED_POWER),
|
2019-09-09 02:56:23 -05:00
|
|
|
pid_error = temp_bed.target - temp_bed.celsius;
|
2019-06-22 16:52:56 -05:00
|
|
|
|
2019-07-28 15:50:25 -05:00
|
|
|
if (!temp_bed.target || pid_error < -(PID_FUNCTIONAL_RANGE)) {
|
|
|
|
pid_output = 0;
|
|
|
|
pid_reset = true;
|
|
|
|
}
|
|
|
|
else if (pid_error > PID_FUNCTIONAL_RANGE) {
|
2019-10-01 06:19:12 +03:00
|
|
|
pid_output = MAX_BED_POWER;
|
2019-07-28 15:50:25 -05:00
|
|
|
pid_reset = true;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (pid_reset) {
|
|
|
|
temp_iState = 0.0;
|
|
|
|
work_pid.Kd = 0.0;
|
|
|
|
pid_reset = false;
|
|
|
|
}
|
2019-06-22 16:52:56 -05:00
|
|
|
|
2019-07-28 15:50:25 -05:00
|
|
|
temp_iState = constrain(temp_iState + pid_error, 0, max_power_over_i_gain);
|
2018-10-10 09:45:20 -05:00
|
|
|
|
2019-07-28 15:50:25 -05:00
|
|
|
work_pid.Kp = temp_bed.pid.Kp * pid_error;
|
|
|
|
work_pid.Ki = temp_bed.pid.Ki * temp_iState;
|
2019-09-09 02:56:23 -05:00
|
|
|
work_pid.Kd = work_pid.Kd + PID_K2 * (temp_bed.pid.Kd * (temp_dState - temp_bed.celsius) - work_pid.Kd);
|
2015-03-13 19:49:41 -07:00
|
|
|
|
2019-09-09 02:56:23 -05:00
|
|
|
temp_dState = temp_bed.celsius;
|
2019-07-28 15:50:25 -05:00
|
|
|
|
|
|
|
pid_output = constrain(work_pid.Kp + work_pid.Ki + work_pid.Kd + float(MIN_BED_POWER), 0, MAX_BED_POWER);
|
|
|
|
}
|
2018-10-10 09:45:20 -05:00
|
|
|
|
|
|
|
#else // PID_OPENLOOP
|
|
|
|
|
2019-03-07 00:09:39 -08:00
|
|
|
const float pid_output = constrain(temp_bed.target, 0, MAX_BED_POWER);
|
2018-10-10 09:45:20 -05:00
|
|
|
|
2015-03-13 19:49:41 -07:00
|
|
|
#endif // PID_OPENLOOP
|
|
|
|
|
2015-07-30 22:29:30 -07:00
|
|
|
#if ENABLED(PID_BED_DEBUG)
|
2021-02-24 16:26:51 -08:00
|
|
|
if (pid_debug_flag) {
|
|
|
|
SERIAL_ECHO_MSG(
|
|
|
|
" PID_BED_DEBUG : Input ", temp_bed.celsius, " Output ", pid_output
|
|
|
|
#if DISABLED(PID_OPENLOOP)
|
|
|
|
, STR_PID_DEBUG_PTERM, work_pid.Kp
|
|
|
|
, STR_PID_DEBUG_ITERM, work_pid.Ki
|
|
|
|
, STR_PID_DEBUG_DTERM, work_pid.Kd
|
|
|
|
#endif
|
|
|
|
);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return pid_output;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // PIDTEMPBED
|
|
|
|
|
|
|
|
#if ENABLED(PIDTEMPCHAMBER)
|
|
|
|
|
|
|
|
float Temperature::get_pid_output_chamber() {
|
|
|
|
|
|
|
|
#if DISABLED(PID_OPENLOOP)
|
|
|
|
|
|
|
|
static PID_t work_pid{0};
|
|
|
|
static float temp_iState = 0, temp_dState = 0;
|
|
|
|
static bool pid_reset = true;
|
|
|
|
float pid_output = 0;
|
|
|
|
const float max_power_over_i_gain = float(MAX_CHAMBER_POWER) / temp_chamber.pid.Ki - float(MIN_CHAMBER_POWER),
|
|
|
|
pid_error = temp_chamber.target - temp_chamber.celsius;
|
|
|
|
|
|
|
|
if (!temp_chamber.target || pid_error < -(PID_FUNCTIONAL_RANGE)) {
|
|
|
|
pid_output = 0;
|
|
|
|
pid_reset = true;
|
|
|
|
}
|
|
|
|
else if (pid_error > PID_FUNCTIONAL_RANGE) {
|
|
|
|
pid_output = MAX_CHAMBER_POWER;
|
|
|
|
pid_reset = true;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (pid_reset) {
|
|
|
|
temp_iState = 0.0;
|
|
|
|
work_pid.Kd = 0.0;
|
|
|
|
pid_reset = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
temp_iState = constrain(temp_iState + pid_error, 0, max_power_over_i_gain);
|
|
|
|
|
|
|
|
work_pid.Kp = temp_chamber.pid.Kp * pid_error;
|
|
|
|
work_pid.Ki = temp_chamber.pid.Ki * temp_iState;
|
|
|
|
work_pid.Kd = work_pid.Kd + PID_K2 * (temp_chamber.pid.Kd * (temp_dState - temp_chamber.celsius) - work_pid.Kd);
|
|
|
|
|
|
|
|
temp_dState = temp_chamber.celsius;
|
|
|
|
|
|
|
|
pid_output = constrain(work_pid.Kp + work_pid.Ki + work_pid.Kd + float(MIN_CHAMBER_POWER), 0, MAX_CHAMBER_POWER);
|
|
|
|
}
|
|
|
|
|
|
|
|
#else // PID_OPENLOOP
|
|
|
|
|
|
|
|
const float pid_output = constrain(temp_chamber.target, 0, MAX_CHAMBER_POWER);
|
|
|
|
|
|
|
|
#endif // PID_OPENLOOP
|
|
|
|
|
|
|
|
#if ENABLED(PID_CHAMBER_DEBUG)
|
2019-09-30 21:44:07 -05:00
|
|
|
{
|
2021-02-04 23:22:42 -06:00
|
|
|
SERIAL_ECHO_MSG(
|
2021-02-24 16:26:51 -08:00
|
|
|
" PID_CHAMBER_DEBUG : Input ", temp_chamber.celsius, " Output ", pid_output
|
2019-03-05 06:46:19 -06:00
|
|
|
#if DISABLED(PID_OPENLOOP)
|
2021-02-24 16:26:51 -08:00
|
|
|
, STR_PID_DEBUG_PTERM, work_pid.Kp
|
|
|
|
, STR_PID_DEBUG_ITERM, work_pid.Ki
|
|
|
|
, STR_PID_DEBUG_DTERM, work_pid.Kd
|
2019-03-05 06:46:19 -06:00
|
|
|
#endif
|
|
|
|
);
|
2019-09-30 21:44:07 -05:00
|
|
|
}
|
2018-10-10 09:45:20 -05:00
|
|
|
#endif
|
2015-03-20 00:07:36 +01:00
|
|
|
|
2015-03-13 19:49:41 -07:00
|
|
|
return pid_output;
|
|
|
|
}
|
2018-10-10 09:45:20 -05:00
|
|
|
|
2021-02-24 16:26:51 -08:00
|
|
|
#endif // PIDTEMPCHAMBER
|
2015-03-13 19:49:41 -07:00
|
|
|
|
2015-04-13 17:17:36 -07:00
|
|
|
/**
|
|
|
|
* Manage heating activities for extruder hot-ends and a heated bed
|
|
|
|
* - Acquire updated temperature readings
|
2016-05-14 17:37:42 -07:00
|
|
|
* - Also resets the watchdog timer
|
2015-04-13 17:17:36 -07:00
|
|
|
* - Invoke thermal runaway protection
|
|
|
|
* - Manage extruder auto-fan
|
|
|
|
* - Apply filament width to the extrusion rate (may move)
|
|
|
|
* - Update the heated bed PID output value
|
|
|
|
*/
|
2016-04-28 18:18:13 -07:00
|
|
|
void Temperature::manage_heater() {
|
2021-05-02 23:37:54 -03:00
|
|
|
if (marlin_state == MF_INITIALIZING) return watchdog_refresh(); // If Marlin isn't started, at least reset the watchdog!
|
2018-03-21 21:04:45 -03:00
|
|
|
|
2021-07-08 21:48:11 -07:00
|
|
|
static bool no_reentry = false; // Prevent recursion
|
|
|
|
if (no_reentry) return;
|
|
|
|
REMEMBER(mh, no_reentry, true);
|
|
|
|
|
2020-10-16 18:11:00 -03:00
|
|
|
#if ENABLED(EMERGENCY_PARSER)
|
2021-09-28 20:15:52 -05:00
|
|
|
if (emergency_parser.killed_by_M112) kill(FPSTR(M112_KILL_STR), nullptr, true);
|
2020-10-16 18:11:00 -03:00
|
|
|
|
|
|
|
if (emergency_parser.quickstop_by_M410) {
|
|
|
|
emergency_parser.quickstop_by_M410 = false; // quickstop_stepper may call idle so clear this now!
|
|
|
|
quickstop_stepper();
|
|
|
|
}
|
|
|
|
#endif
|
2018-03-03 23:14:01 -06:00
|
|
|
|
2021-05-08 17:34:21 -05:00
|
|
|
if (!updateTemperaturesIfReady()) return; // Will also reset the watchdog if temperatures are ready
|
2011-11-15 22:50:43 +01:00
|
|
|
|
2020-07-11 01:23:48 -04:00
|
|
|
#if DISABLED(IGNORE_THERMOCOUPLE_ERRORS)
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_0_IS_MAX_TC
|
2021-04-27 16:36:39 +12:00
|
|
|
if (degHotend(0) > _MIN(HEATER_0_MAXTEMP, TEMP_SENSOR_0_MAX_TC_TMAX - 1.0)) max_temp_error(H_E0);
|
|
|
|
if (degHotend(0) < _MAX(HEATER_0_MINTEMP, TEMP_SENSOR_0_MAX_TC_TMIN + .01)) min_temp_error(H_E0);
|
2020-07-11 01:23:48 -04:00
|
|
|
#endif
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_1_IS_MAX_TC
|
2021-06-11 13:51:29 -07:00
|
|
|
if (degHotend(1) > _MIN(HEATER_1_MAXTEMP, TEMP_SENSOR_1_MAX_TC_TMAX - 1.0)) max_temp_error(H_E1);
|
|
|
|
if (degHotend(1) < _MAX(HEATER_1_MINTEMP, TEMP_SENSOR_1_MAX_TC_TMIN + .01)) min_temp_error(H_E1);
|
|
|
|
#endif
|
|
|
|
#if TEMP_SENSOR_REDUNDANT_IS_MAX_TC
|
|
|
|
if (degRedundant() > TEMP_SENSOR_REDUNDANT_MAX_TC_TMAX - 1.0) max_temp_error(H_REDUNDANT);
|
|
|
|
if (degRedundant() < TEMP_SENSOR_REDUNDANT_MAX_TC_TMIN + .01) min_temp_error(H_REDUNDANT);
|
2020-07-11 01:23:48 -04:00
|
|
|
#endif
|
2015-05-08 22:25:51 -07:00
|
|
|
#endif
|
2015-03-25 11:08:24 +01:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
millis_t ms = millis();
|
2011-12-05 23:33:33 -05:00
|
|
|
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_HOTEND
|
2014-06-30 15:22:37 -03:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
HOTEND_LOOP() {
|
|
|
|
#if ENABLED(THERMAL_PROTECTION_HOTENDS)
|
2020-09-14 16:58:39 +12:00
|
|
|
if (degHotend(e) > temp_range[e].maxtemp) max_temp_error((heater_id_t)e);
|
2019-09-10 02:20:49 -05:00
|
|
|
#endif
|
2017-05-26 13:01:02 -05:00
|
|
|
|
2020-09-14 16:58:39 +12:00
|
|
|
TERN_(HEATER_IDLE_HANDLER, heater_idle[e].update(ms));
|
2015-02-24 04:46:11 -08:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
#if ENABLED(THERMAL_PROTECTION_HOTENDS)
|
|
|
|
// Check for thermal runaway
|
2020-09-14 16:58:39 +12:00
|
|
|
tr_state_machine[e].run(temp_hotend[e].celsius, temp_hotend[e].target, (heater_id_t)e, THERMAL_PROTECTION_PERIOD, THERMAL_PROTECTION_HYSTERESIS);
|
2019-09-10 02:20:49 -05:00
|
|
|
#endif
|
2011-12-05 23:33:33 -05:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
temp_hotend[e].soft_pwm_amount = (temp_hotend[e].celsius > temp_range[e].mintemp || is_preheating(e)) && temp_hotend[e].celsius < temp_range[e].maxtemp ? (int)get_pid_output_hotend(e) >> 1 : 0;
|
2015-05-12 18:22:47 -07:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
#if WATCH_HOTENDS
|
|
|
|
// Make sure temperature is increasing
|
2021-04-30 08:51:26 +02:00
|
|
|
if (watch_hotend[e].elapsed(ms)) { // Enabled and time to check?
|
|
|
|
if (watch_hotend[e].check(degHotend(e))) // Increased enough?
|
|
|
|
start_watching_hotend(e); // If temp reached, turn off elapsed check
|
|
|
|
else {
|
2021-09-07 02:15:24 -05:00
|
|
|
TERN_(HAS_DWIN_E3V2_BASIC, DWIN_Popup_Temperature(0));
|
2021-09-28 20:15:52 -05:00
|
|
|
_temp_error((heater_id_t)e, FPSTR(str_t_heating_failed), GET_TEXT_F(MSG_HEATING_FAILED_LCD));
|
2020-06-16 01:45:27 -05:00
|
|
|
}
|
2019-09-10 02:20:49 -05:00
|
|
|
}
|
|
|
|
#endif
|
2015-02-24 04:46:11 -08:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
} // HOTEND_LOOP
|
|
|
|
|
2020-07-01 16:27:28 -05:00
|
|
|
#endif // HAS_HOTEND
|
2012-08-11 09:17:47 +03:00
|
|
|
|
2021-06-11 13:51:29 -07:00
|
|
|
#if HAS_TEMP_REDUNDANT
|
|
|
|
// Make sure measured temperatures are close together
|
|
|
|
if (ABS(degRedundantTarget() - degRedundant()) > TEMP_SENSOR_REDUNDANT_MAX_DIFF)
|
2021-09-28 20:15:52 -05:00
|
|
|
_temp_error((heater_id_t)HEATER_ID(TEMP_SENSOR_REDUNDANT_TARGET), F(STR_REDUNDANCY), GET_TEXT_F(MSG_ERR_REDUNDANT_TEMP));
|
2021-06-11 13:51:29 -07:00
|
|
|
#endif
|
|
|
|
|
2015-02-26 01:14:59 -08:00
|
|
|
#if HAS_AUTO_FAN
|
2016-04-11 18:48:27 -07:00
|
|
|
if (ELAPSED(ms, next_auto_fan_check_ms)) { // only need to check fan state very infrequently
|
2015-02-24 04:46:11 -08:00
|
|
|
checkExtruderAutoFans();
|
2016-04-10 15:55:12 -07:00
|
|
|
next_auto_fan_check_ms = ms + 2500UL;
|
2015-02-24 04:46:11 -08:00
|
|
|
}
|
2015-10-02 23:08:58 -07:00
|
|
|
#endif
|
2015-04-13 17:17:36 -07:00
|
|
|
|
2016-03-29 20:28:23 -07:00
|
|
|
#if ENABLED(FILAMENT_WIDTH_SENSOR)
|
2017-12-13 02:32:34 -06:00
|
|
|
/**
|
2019-09-10 18:48:58 -05:00
|
|
|
* Dynamically set the volumetric multiplier based
|
|
|
|
* on the delayed Filament Width measurement.
|
2017-12-13 02:32:34 -06:00
|
|
|
*/
|
2019-09-10 18:48:58 -05:00
|
|
|
filwidth.update_volumetric();
|
|
|
|
#endif
|
2015-04-13 17:17:36 -07:00
|
|
|
|
2018-04-23 17:13:01 -05:00
|
|
|
#if HAS_HEATED_BED
|
|
|
|
|
2019-04-22 03:50:32 +02:00
|
|
|
#if ENABLED(THERMAL_PROTECTION_BED)
|
2020-09-14 16:58:39 +12:00
|
|
|
if (degBed() > BED_MAXTEMP) max_temp_error(H_BED);
|
2019-04-22 03:50:32 +02:00
|
|
|
#endif
|
2019-04-19 20:37:12 -06:00
|
|
|
|
2019-03-07 00:09:39 -08:00
|
|
|
#if WATCH_BED
|
2018-04-23 17:13:01 -05:00
|
|
|
// Make sure temperature is increasing
|
2021-04-30 08:51:26 +02:00
|
|
|
if (watch_bed.elapsed(ms)) { // Time to check the bed?
|
|
|
|
if (watch_bed.check(degBed())) // Increased enough?
|
|
|
|
start_watching_bed(); // If temp reached, turn off elapsed check
|
|
|
|
else {
|
2021-09-07 02:15:24 -05:00
|
|
|
TERN_(HAS_DWIN_E3V2_BASIC, DWIN_Popup_Temperature(0));
|
2021-09-28 20:15:52 -05:00
|
|
|
_temp_error(H_BED, FPSTR(str_t_heating_failed), GET_TEXT_F(MSG_HEATING_FAILED_LCD));
|
2020-06-16 01:45:27 -05:00
|
|
|
}
|
2018-04-23 17:13:01 -05:00
|
|
|
}
|
2019-03-07 00:09:39 -08:00
|
|
|
#endif // WATCH_BED
|
2017-05-26 13:01:02 -05:00
|
|
|
|
2020-04-22 16:35:03 -05:00
|
|
|
#if BOTH(PROBING_HEATERS_OFF, BED_LIMIT_SWITCHING)
|
|
|
|
#define PAUSE_CHANGE_REQD 1
|
|
|
|
#endif
|
|
|
|
|
2020-03-07 22:34:33 -06:00
|
|
|
#if PAUSE_CHANGE_REQD
|
|
|
|
static bool last_pause_state;
|
|
|
|
#endif
|
|
|
|
|
2019-05-07 02:55:01 +02:00
|
|
|
do {
|
|
|
|
|
|
|
|
#if DISABLED(PIDTEMPBED)
|
|
|
|
if (PENDING(ms, next_bed_check_ms)
|
2021-05-02 21:32:21 -05:00
|
|
|
&& TERN1(PAUSE_CHANGE_REQD, paused_for_probing == last_pause_state)
|
2019-05-07 02:55:01 +02:00
|
|
|
) break;
|
|
|
|
next_bed_check_ms = ms + BED_CHECK_INTERVAL;
|
2021-05-02 21:32:21 -05:00
|
|
|
TERN_(PAUSE_CHANGE_REQD, last_pause_state = paused_for_probing);
|
2018-02-08 20:36:05 -06:00
|
|
|
#endif
|
2017-05-26 13:01:02 -05:00
|
|
|
|
2020-09-14 16:58:39 +12:00
|
|
|
TERN_(HEATER_IDLE_HANDLER, heater_idle[IDLE_INDEX_BED].update(ms));
|
2015-10-02 23:08:58 -07:00
|
|
|
|
2020-09-14 16:58:39 +12:00
|
|
|
#if HAS_THERMALLY_PROTECTED_BED
|
|
|
|
tr_state_machine[RUNAWAY_IND_BED].run(temp_bed.celsius, temp_bed.target, H_BED, THERMAL_PROTECTION_BED_PERIOD, THERMAL_PROTECTION_BED_HYSTERESIS);
|
|
|
|
#endif
|
2014-06-30 15:22:37 -03:00
|
|
|
|
2019-05-07 02:55:01 +02:00
|
|
|
#if HEATER_IDLE_HANDLER
|
2020-09-14 16:58:39 +12:00
|
|
|
if (heater_idle[IDLE_INDEX_BED].timed_out) {
|
2019-03-07 00:09:39 -08:00
|
|
|
temp_bed.soft_pwm_amount = 0;
|
2019-05-07 02:55:01 +02:00
|
|
|
#if DISABLED(PIDTEMPBED)
|
|
|
|
WRITE_HEATER_BED(LOW);
|
|
|
|
#endif
|
2017-05-26 13:01:02 -05:00
|
|
|
}
|
2019-05-07 02:55:01 +02:00
|
|
|
else
|
2017-05-26 13:01:02 -05:00
|
|
|
#endif
|
2019-05-07 02:55:01 +02:00
|
|
|
{
|
|
|
|
#if ENABLED(PIDTEMPBED)
|
2019-09-09 02:56:23 -05:00
|
|
|
temp_bed.soft_pwm_amount = WITHIN(temp_bed.celsius, BED_MINTEMP, BED_MAXTEMP) ? (int)get_pid_output_bed() >> 1 : 0;
|
2019-05-07 02:55:01 +02:00
|
|
|
#else
|
|
|
|
// Check if temperature is within the correct band
|
2019-09-09 02:56:23 -05:00
|
|
|
if (WITHIN(temp_bed.celsius, BED_MINTEMP, BED_MAXTEMP)) {
|
2019-05-07 02:55:01 +02:00
|
|
|
#if ENABLED(BED_LIMIT_SWITCHING)
|
2019-09-09 02:56:23 -05:00
|
|
|
if (temp_bed.celsius >= temp_bed.target + BED_HYSTERESIS)
|
2019-05-07 02:55:01 +02:00
|
|
|
temp_bed.soft_pwm_amount = 0;
|
2019-09-09 02:56:23 -05:00
|
|
|
else if (temp_bed.celsius <= temp_bed.target - (BED_HYSTERESIS))
|
2019-05-07 02:55:01 +02:00
|
|
|
temp_bed.soft_pwm_amount = MAX_BED_POWER >> 1;
|
|
|
|
#else // !PIDTEMPBED && !BED_LIMIT_SWITCHING
|
2019-09-09 02:56:23 -05:00
|
|
|
temp_bed.soft_pwm_amount = temp_bed.celsius < temp_bed.target ? MAX_BED_POWER >> 1 : 0;
|
2019-05-07 02:55:01 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
temp_bed.soft_pwm_amount = 0;
|
|
|
|
WRITE_HEATER_BED(LOW);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
} while (false);
|
|
|
|
|
2018-04-23 17:13:01 -05:00
|
|
|
#endif // HAS_HEATED_BED
|
2019-03-07 00:09:39 -08:00
|
|
|
|
2019-05-05 05:51:47 +02:00
|
|
|
#if HAS_HEATED_CHAMBER
|
2019-03-07 00:09:39 -08:00
|
|
|
|
|
|
|
#ifndef CHAMBER_CHECK_INTERVAL
|
|
|
|
#define CHAMBER_CHECK_INTERVAL 1000UL
|
|
|
|
#endif
|
|
|
|
|
2019-05-05 05:51:47 +02:00
|
|
|
#if ENABLED(THERMAL_PROTECTION_CHAMBER)
|
2020-09-14 16:58:39 +12:00
|
|
|
if (degChamber() > CHAMBER_MAXTEMP) max_temp_error(H_CHAMBER);
|
2019-05-05 05:51:47 +02:00
|
|
|
#endif
|
2019-03-07 00:09:39 -08:00
|
|
|
|
2019-05-05 05:51:47 +02:00
|
|
|
#if WATCH_CHAMBER
|
|
|
|
// Make sure temperature is increasing
|
2021-04-30 08:51:26 +02:00
|
|
|
if (watch_chamber.elapsed(ms)) { // Time to check the chamber?
|
|
|
|
if (watch_chamber.check(degChamber())) // Increased enough? Error below.
|
|
|
|
start_watching_chamber(); // If temp reached, turn off elapsed check.
|
2019-05-05 05:51:47 +02:00
|
|
|
else
|
2021-09-28 20:15:52 -05:00
|
|
|
_temp_error(H_CHAMBER, FPSTR(str_t_heating_failed), GET_TEXT_F(MSG_HEATING_FAILED_LCD));
|
2019-03-07 00:09:39 -08:00
|
|
|
}
|
2019-05-07 02:55:01 +02:00
|
|
|
#endif
|
2019-05-05 05:51:47 +02:00
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
#if EITHER(CHAMBER_FAN, CHAMBER_VENT) || DISABLED(PIDTEMPCHAMBER)
|
|
|
|
static bool flag_chamber_excess_heat; // = false;
|
|
|
|
#endif
|
|
|
|
|
2020-10-07 01:36:01 +02:00
|
|
|
#if EITHER(CHAMBER_FAN, CHAMBER_VENT)
|
2021-03-06 14:13:28 -06:00
|
|
|
static bool flag_chamber_off; // = false
|
|
|
|
|
2020-10-07 01:36:01 +02:00
|
|
|
if (temp_chamber.target > CHAMBER_MINTEMP) {
|
|
|
|
flag_chamber_off = false;
|
|
|
|
|
|
|
|
#if ENABLED(CHAMBER_FAN)
|
2021-03-06 14:13:28 -06:00
|
|
|
int16_t fan_chamber_pwm;
|
2020-10-07 01:36:01 +02:00
|
|
|
#if CHAMBER_FAN_MODE == 0
|
2021-01-01 12:56:59 -08:00
|
|
|
fan_chamber_pwm = CHAMBER_FAN_BASE;
|
2020-10-07 01:36:01 +02:00
|
|
|
#elif CHAMBER_FAN_MODE == 1
|
2020-10-06 19:42:03 -05:00
|
|
|
fan_chamber_pwm = (temp_chamber.celsius > temp_chamber.target) ? (CHAMBER_FAN_BASE) + (CHAMBER_FAN_FACTOR) * (temp_chamber.celsius - temp_chamber.target) : 0;
|
2020-10-07 01:36:01 +02:00
|
|
|
#elif CHAMBER_FAN_MODE == 2
|
|
|
|
fan_chamber_pwm = (CHAMBER_FAN_BASE) + (CHAMBER_FAN_FACTOR) * ABS(temp_chamber.celsius - temp_chamber.target);
|
|
|
|
if (temp_chamber.soft_pwm_amount)
|
|
|
|
fan_chamber_pwm += (CHAMBER_FAN_FACTOR) * 2;
|
2021-04-13 03:28:13 +02:00
|
|
|
#elif CHAMBER_FAN_MODE == 3
|
|
|
|
fan_chamber_pwm = CHAMBER_FAN_BASE + _MAX((CHAMBER_FAN_FACTOR) * (temp_chamber.celsius - temp_chamber.target), 0);
|
2020-10-07 01:36:01 +02:00
|
|
|
#endif
|
2021-10-18 01:11:16 -07:00
|
|
|
NOMORE(fan_chamber_pwm, 255);
|
|
|
|
set_fan_speed(CHAMBER_FAN_INDEX, fan_chamber_pwm);
|
2020-10-07 01:36:01 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if ENABLED(CHAMBER_VENT)
|
|
|
|
#ifndef MIN_COOLING_SLOPE_TIME_CHAMBER_VENT
|
|
|
|
#define MIN_COOLING_SLOPE_TIME_CHAMBER_VENT 20
|
|
|
|
#endif
|
|
|
|
#ifndef MIN_COOLING_SLOPE_DEG_CHAMBER_VENT
|
|
|
|
#define MIN_COOLING_SLOPE_DEG_CHAMBER_VENT 1.5
|
|
|
|
#endif
|
2020-10-29 01:14:02 -05:00
|
|
|
if (!flag_chamber_excess_heat && temp_chamber.celsius - temp_chamber.target >= HIGH_EXCESS_HEAT_LIMIT) {
|
|
|
|
// Open vent after MIN_COOLING_SLOPE_TIME_CHAMBER_VENT seconds if the
|
|
|
|
// temperature didn't drop at least MIN_COOLING_SLOPE_DEG_CHAMBER_VENT
|
2020-10-07 01:36:01 +02:00
|
|
|
if (next_cool_check_ms_2 == 0 || ELAPSED(ms, next_cool_check_ms_2)) {
|
2021-03-06 14:13:28 -06:00
|
|
|
if (temp_chamber.celsius - old_temp > MIN_COOLING_SLOPE_DEG_CHAMBER_VENT)
|
|
|
|
flag_chamber_excess_heat = true; // the bed is heating the chamber too much
|
2021-01-28 20:40:20 -06:00
|
|
|
next_cool_check_ms_2 = ms + SEC_TO_MS(MIN_COOLING_SLOPE_TIME_CHAMBER_VENT);
|
2020-10-07 01:36:01 +02:00
|
|
|
old_temp = temp_chamber.celsius;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
next_cool_check_ms_2 = 0;
|
|
|
|
old_temp = 9999;
|
|
|
|
}
|
2021-03-06 14:13:28 -06:00
|
|
|
if (flag_chamber_excess_heat && (temp_chamber.target - temp_chamber.celsius >= LOW_EXCESS_HEAT_LIMIT))
|
2020-10-07 01:36:01 +02:00
|
|
|
flag_chamber_excess_heat = false;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else if (!flag_chamber_off) {
|
|
|
|
#if ENABLED(CHAMBER_FAN)
|
|
|
|
flag_chamber_off = true;
|
2021-08-05 23:24:20 -05:00
|
|
|
set_fan_speed(CHAMBER_FAN_INDEX, 0);
|
2020-10-07 01:36:01 +02:00
|
|
|
#endif
|
|
|
|
#if ENABLED(CHAMBER_VENT)
|
|
|
|
flag_chamber_excess_heat = false;
|
|
|
|
MOVE_SERVO(CHAMBER_VENT_SERVO_NR, 90);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2021-02-24 16:26:51 -08:00
|
|
|
#if ENABLED(PIDTEMPCHAMBER)
|
2021-08-03 20:02:34 -04:00
|
|
|
// PIDTEMPCHAMBER doesn't support a CHAMBER_VENT yet.
|
2021-02-24 16:26:51 -08:00
|
|
|
temp_chamber.soft_pwm_amount = WITHIN(temp_chamber.celsius, CHAMBER_MINTEMP, CHAMBER_MAXTEMP) ? (int)get_pid_output_chamber() >> 1 : 0;
|
|
|
|
#else
|
2021-03-06 14:13:28 -06:00
|
|
|
if (ELAPSED(ms, next_chamber_check_ms)) {
|
|
|
|
next_chamber_check_ms = ms + CHAMBER_CHECK_INTERVAL;
|
2019-03-07 00:09:39 -08:00
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
if (WITHIN(temp_chamber.celsius, CHAMBER_MINTEMP, CHAMBER_MAXTEMP)) {
|
2021-02-24 16:26:51 -08:00
|
|
|
if (flag_chamber_excess_heat) {
|
|
|
|
temp_chamber.soft_pwm_amount = 0;
|
|
|
|
#if ENABLED(CHAMBER_VENT)
|
|
|
|
if (!flag_chamber_off) MOVE_SERVO(CHAMBER_VENT_SERVO_NR, temp_chamber.celsius <= temp_chamber.target ? 0 : 90);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
#if ENABLED(CHAMBER_LIMIT_SWITCHING)
|
|
|
|
if (temp_chamber.celsius >= temp_chamber.target + TEMP_CHAMBER_HYSTERESIS)
|
|
|
|
temp_chamber.soft_pwm_amount = 0;
|
|
|
|
else if (temp_chamber.celsius <= temp_chamber.target - (TEMP_CHAMBER_HYSTERESIS))
|
|
|
|
temp_chamber.soft_pwm_amount = (MAX_CHAMBER_POWER) >> 1;
|
|
|
|
#else
|
|
|
|
temp_chamber.soft_pwm_amount = temp_chamber.celsius < temp_chamber.target ? (MAX_CHAMBER_POWER) >> 1 : 0;
|
|
|
|
#endif
|
|
|
|
#if ENABLED(CHAMBER_VENT)
|
|
|
|
if (!flag_chamber_off) MOVE_SERVO(CHAMBER_VENT_SERVO_NR, 0);
|
|
|
|
#endif
|
|
|
|
}
|
2020-10-06 19:42:03 -05:00
|
|
|
}
|
|
|
|
else {
|
2021-02-24 16:26:51 -08:00
|
|
|
temp_chamber.soft_pwm_amount = 0;
|
|
|
|
WRITE_HEATER_CHAMBER(LOW);
|
2020-10-07 01:36:01 +02:00
|
|
|
}
|
2021-02-24 16:26:51 -08:00
|
|
|
}
|
|
|
|
#if ENABLED(THERMAL_PROTECTION_CHAMBER)
|
|
|
|
tr_state_machine[RUNAWAY_IND_CHAMBER].run(temp_chamber.celsius, temp_chamber.target, H_CHAMBER, THERMAL_PROTECTION_CHAMBER_PERIOD, THERMAL_PROTECTION_CHAMBER_HYSTERESIS);
|
|
|
|
#endif
|
|
|
|
#endif
|
2019-03-07 00:09:39 -08:00
|
|
|
|
2019-05-05 05:51:47 +02:00
|
|
|
#endif // HAS_HEATED_CHAMBER
|
2019-09-10 02:20:49 -05:00
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_COOLER
|
|
|
|
|
|
|
|
#ifndef COOLER_CHECK_INTERVAL
|
|
|
|
#define COOLER_CHECK_INTERVAL 2000UL
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if ENABLED(THERMAL_PROTECTION_COOLER)
|
|
|
|
if (degCooler() > COOLER_MAXTEMP) max_temp_error(H_COOLER);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if WATCH_COOLER
|
|
|
|
// Make sure temperature is decreasing
|
|
|
|
if (watch_cooler.elapsed(ms)) { // Time to check the cooler?
|
|
|
|
if (degCooler() > watch_cooler.target) // Failed to decrease enough?
|
2021-09-28 20:15:52 -05:00
|
|
|
_temp_error(H_COOLER, GET_TEXT_F(MSG_COOLING_FAILED), GET_TEXT_F(MSG_COOLING_FAILED));
|
2021-03-06 14:13:28 -06:00
|
|
|
else
|
|
|
|
start_watching_cooler(); // Start again if the target is still far off
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static bool flag_cooler_state; // = false
|
|
|
|
|
2021-03-29 01:41:56 -05:00
|
|
|
if (cooler.enabled) {
|
2021-03-06 14:13:28 -06:00
|
|
|
flag_cooler_state = true; // used to allow M106 fan control when cooler is disabled
|
2021-03-19 16:34:10 -05:00
|
|
|
if (temp_cooler.target == 0) temp_cooler.target = COOLER_MIN_TARGET;
|
2021-03-06 14:13:28 -06:00
|
|
|
if (ELAPSED(ms, next_cooler_check_ms)) {
|
|
|
|
next_cooler_check_ms = ms + COOLER_CHECK_INTERVAL;
|
|
|
|
if (temp_cooler.celsius > temp_cooler.target) {
|
|
|
|
temp_cooler.soft_pwm_amount = temp_cooler.celsius > temp_cooler.target ? MAX_COOLER_POWER : 0;
|
|
|
|
flag_cooler_state = temp_cooler.soft_pwm_amount > 0 ? true : false; // used to allow M106 fan control when cooler is disabled
|
|
|
|
#if ENABLED(COOLER_FAN)
|
|
|
|
int16_t fan_cooler_pwm = (COOLER_FAN_BASE) + (COOLER_FAN_FACTOR) * ABS(temp_cooler.celsius - temp_cooler.target);
|
|
|
|
NOMORE(fan_cooler_pwm, 255);
|
|
|
|
set_fan_speed(COOLER_FAN_INDEX, fan_cooler_pwm); // Set cooler fan pwm
|
|
|
|
cooler_fan_flush_ms = ms + 5000;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
temp_cooler.soft_pwm_amount = 0;
|
|
|
|
#if ENABLED(COOLER_FAN)
|
|
|
|
set_fan_speed(COOLER_FAN_INDEX, temp_cooler.celsius > temp_cooler.target - 2 ? COOLER_FAN_BASE : 0);
|
|
|
|
#endif
|
|
|
|
WRITE_HEATER_COOLER(LOW);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
temp_cooler.soft_pwm_amount = 0;
|
|
|
|
if (flag_cooler_state) {
|
|
|
|
flag_cooler_state = false;
|
|
|
|
thermalManager.set_fan_speed(COOLER_FAN_INDEX, 0);
|
|
|
|
}
|
|
|
|
WRITE_HEATER_COOLER(LOW);
|
|
|
|
}
|
|
|
|
|
|
|
|
#if ENABLED(THERMAL_PROTECTION_COOLER)
|
|
|
|
tr_state_machine[RUNAWAY_IND_COOLER].run(temp_cooler.celsius, temp_cooler.target, H_COOLER, THERMAL_PROTECTION_COOLER_PERIOD, THERMAL_PROTECTION_COOLER_HYSTERESIS);
|
|
|
|
#endif
|
2021-03-29 01:41:56 -05:00
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
#endif // HAS_COOLER
|
|
|
|
|
2021-04-01 19:29:49 -05:00
|
|
|
#if ENABLED(LASER_COOLANT_FLOW_METER)
|
2021-03-29 01:41:56 -05:00
|
|
|
cooler.flowmeter_task(ms);
|
|
|
|
#if ENABLED(FLOWMETER_SAFETY)
|
|
|
|
if (cutter.enabled() && cooler.check_flow_too_low()) {
|
|
|
|
cutter.disable();
|
2021-07-12 00:22:08 -05:00
|
|
|
TERN_(HAS_DISPLAY, ui.flow_fault());
|
2021-03-29 01:41:56 -05:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
UNUSED(ms);
|
2011-11-15 22:50:43 +01:00
|
|
|
}
|
|
|
|
|
2020-02-28 21:28:17 -08:00
|
|
|
#define TEMP_AD595(RAW) ((RAW) * 5.0 * 100.0 / float(HAL_ADC_RANGE) / (OVERSAMPLENR) * (TEMP_SENSOR_AD595_GAIN) + TEMP_SENSOR_AD595_OFFSET)
|
|
|
|
#define TEMP_AD8495(RAW) ((RAW) * 6.6 * 100.0 / float(HAL_ADC_RANGE) / (OVERSAMPLENR) * (TEMP_SENSOR_AD8495_GAIN) + TEMP_SENSOR_AD8495_OFFSET)
|
2018-05-11 01:01:51 -05:00
|
|
|
|
2018-05-27 23:27:51 -05:00
|
|
|
/**
|
|
|
|
* Bisect search for the range of the 'raw' value, then interpolate
|
|
|
|
* proportionally between the under and over values.
|
|
|
|
*/
|
2021-03-24 04:11:43 -05:00
|
|
|
#define SCAN_THERMISTOR_TABLE(TBL,LEN) do{ \
|
|
|
|
uint8_t l = 0, r = LEN, m; \
|
|
|
|
for (;;) { \
|
|
|
|
m = (l + r) >> 1; \
|
|
|
|
if (!m) return celsius_t(pgm_read_word(&TBL[0].celsius)); \
|
|
|
|
if (m == l || m == r) return celsius_t(pgm_read_word(&TBL[LEN-1].celsius)); \
|
|
|
|
int16_t v00 = pgm_read_word(&TBL[m-1].value), \
|
|
|
|
v10 = pgm_read_word(&TBL[m-0].value); \
|
|
|
|
if (raw < v00) r = m; \
|
|
|
|
else if (raw > v10) l = m; \
|
|
|
|
else { \
|
|
|
|
const celsius_t v01 = celsius_t(pgm_read_word(&TBL[m-1].celsius)), \
|
|
|
|
v11 = celsius_t(pgm_read_word(&TBL[m-0].celsius)); \
|
|
|
|
return v01 + (raw - v00) * float(v11 - v01) / float(v10 - v00); \
|
|
|
|
} \
|
|
|
|
} \
|
2018-05-11 01:01:51 -05:00
|
|
|
}while(0)
|
2016-04-28 18:18:13 -07:00
|
|
|
|
2019-05-07 00:51:06 +01:00
|
|
|
#if HAS_USER_THERMISTORS
|
|
|
|
|
|
|
|
user_thermistor_t Temperature::user_thermistor[USER_THERMISTORS]; // Initialized by settings.load()
|
|
|
|
|
|
|
|
void Temperature::reset_user_thermistors() {
|
2021-01-03 20:39:15 -07:00
|
|
|
user_thermistor_t default_user_thermistor[USER_THERMISTORS] = {
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_0_IS_CUSTOM
|
2019-05-07 00:51:06 +01:00
|
|
|
{ true, 0, 0, HOTEND0_PULLUP_RESISTOR_OHMS, HOTEND0_RESISTANCE_25C_OHMS, 0, 0, HOTEND0_BETA, 0 },
|
|
|
|
#endif
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_1_IS_CUSTOM
|
2019-05-07 00:51:06 +01:00
|
|
|
{ true, 0, 0, HOTEND1_PULLUP_RESISTOR_OHMS, HOTEND1_RESISTANCE_25C_OHMS, 0, 0, HOTEND1_BETA, 0 },
|
|
|
|
#endif
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_2_IS_CUSTOM
|
2019-05-07 00:51:06 +01:00
|
|
|
{ true, 0, 0, HOTEND2_PULLUP_RESISTOR_OHMS, HOTEND2_RESISTANCE_25C_OHMS, 0, 0, HOTEND2_BETA, 0 },
|
|
|
|
#endif
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_3_IS_CUSTOM
|
2019-05-07 00:51:06 +01:00
|
|
|
{ true, 0, 0, HOTEND3_PULLUP_RESISTOR_OHMS, HOTEND3_RESISTANCE_25C_OHMS, 0, 0, HOTEND3_BETA, 0 },
|
|
|
|
#endif
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_4_IS_CUSTOM
|
2019-05-07 00:51:06 +01:00
|
|
|
{ true, 0, 0, HOTEND4_PULLUP_RESISTOR_OHMS, HOTEND4_RESISTANCE_25C_OHMS, 0, 0, HOTEND4_BETA, 0 },
|
|
|
|
#endif
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_5_IS_CUSTOM
|
2019-05-07 00:51:06 +01:00
|
|
|
{ true, 0, 0, HOTEND5_PULLUP_RESISTOR_OHMS, HOTEND5_RESISTANCE_25C_OHMS, 0, 0, HOTEND5_BETA, 0 },
|
|
|
|
#endif
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_6_IS_CUSTOM
|
2020-01-25 16:13:39 +08:00
|
|
|
{ true, 0, 0, HOTEND6_PULLUP_RESISTOR_OHMS, HOTEND6_RESISTANCE_25C_OHMS, 0, 0, HOTEND6_BETA, 0 },
|
|
|
|
#endif
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_7_IS_CUSTOM
|
2020-01-25 16:13:39 +08:00
|
|
|
{ true, 0, 0, HOTEND7_PULLUP_RESISTOR_OHMS, HOTEND7_RESISTANCE_25C_OHMS, 0, 0, HOTEND7_BETA, 0 },
|
|
|
|
#endif
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_BED_IS_CUSTOM
|
2019-05-07 00:51:06 +01:00
|
|
|
{ true, 0, 0, BED_PULLUP_RESISTOR_OHMS, BED_RESISTANCE_25C_OHMS, 0, 0, BED_BETA, 0 },
|
|
|
|
#endif
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_CHAMBER_IS_CUSTOM
|
2021-06-11 13:51:29 -07:00
|
|
|
{ true, 0, 0, CHAMBER_PULLUP_RESISTOR_OHMS, CHAMBER_RESISTANCE_25C_OHMS, 0, 0, CHAMBER_BETA, 0 },
|
2019-05-07 00:51:06 +01:00
|
|
|
#endif
|
2021-03-06 14:13:28 -06:00
|
|
|
#if TEMP_SENSOR_COOLER_IS_CUSTOM
|
2021-06-11 13:51:29 -07:00
|
|
|
{ true, 0, 0, COOLER_PULLUP_RESISTOR_OHMS, COOLER_RESISTANCE_25C_OHMS, 0, 0, COOLER_BETA, 0 },
|
2021-03-06 14:13:28 -06:00
|
|
|
#endif
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_PROBE_IS_CUSTOM
|
2021-06-11 13:51:29 -07:00
|
|
|
{ true, 0, 0, PROBE_PULLUP_RESISTOR_OHMS, PROBE_RESISTANCE_25C_OHMS, 0, 0, PROBE_BETA, 0 },
|
|
|
|
#endif
|
2021-08-18 20:12:41 -05:00
|
|
|
#if TEMP_SENSOR_BOARD_IS_CUSTOM
|
|
|
|
{ true, 0, 0, BOARD_PULLUP_RESISTOR_OHMS, BOARD_RESISTANCE_25C_OHMS, 0, 0, BOARD_BETA, 0 },
|
|
|
|
#endif
|
2021-06-11 13:51:29 -07:00
|
|
|
#if TEMP_SENSOR_REDUNDANT_IS_CUSTOM
|
|
|
|
{ true, 0, 0, REDUNDANT_PULLUP_RESISTOR_OHMS, REDUNDANT_RESISTANCE_25C_OHMS, 0, 0, REDUNDANT_BETA, 0 },
|
2021-02-07 16:58:06 -06:00
|
|
|
#endif
|
2019-05-07 00:51:06 +01:00
|
|
|
};
|
2021-01-03 20:39:15 -07:00
|
|
|
COPY(user_thermistor, default_user_thermistor);
|
2019-05-07 00:51:06 +01:00
|
|
|
}
|
|
|
|
|
2021-09-07 18:06:10 -05:00
|
|
|
void Temperature::M305_report(const uint8_t t_index, const bool forReplay/*=true*/) {
|
2021-09-27 13:40:01 -05:00
|
|
|
gcode.report_heading_etc(forReplay, F(STR_USER_THERMISTORS));
|
2021-09-09 04:57:05 -05:00
|
|
|
SERIAL_ECHOPGM(" M305 P", AS_DIGIT(t_index));
|
2019-05-07 00:51:06 +01:00
|
|
|
|
|
|
|
const user_thermistor_t &t = user_thermistor[t_index];
|
|
|
|
|
|
|
|
SERIAL_ECHOPAIR_F(" R", t.series_res, 1);
|
2020-02-01 17:05:42 -06:00
|
|
|
SERIAL_ECHOPAIR_F_P(SP_T_STR, t.res_25, 1);
|
2020-06-27 23:27:28 -05:00
|
|
|
SERIAL_ECHOPAIR_F_P(SP_B_STR, t.beta, 1);
|
|
|
|
SERIAL_ECHOPAIR_F_P(SP_C_STR, t.sh_c_coeff, 9);
|
2019-05-07 00:51:06 +01:00
|
|
|
SERIAL_ECHOPGM(" ; ");
|
2021-09-27 11:03:07 -05:00
|
|
|
SERIAL_ECHOF(
|
|
|
|
TERN_(TEMP_SENSOR_0_IS_CUSTOM, t_index == CTI_HOTEND_0 ? F("HOTEND 0") :)
|
|
|
|
TERN_(TEMP_SENSOR_1_IS_CUSTOM, t_index == CTI_HOTEND_1 ? F("HOTEND 1") :)
|
|
|
|
TERN_(TEMP_SENSOR_2_IS_CUSTOM, t_index == CTI_HOTEND_2 ? F("HOTEND 2") :)
|
|
|
|
TERN_(TEMP_SENSOR_3_IS_CUSTOM, t_index == CTI_HOTEND_3 ? F("HOTEND 3") :)
|
|
|
|
TERN_(TEMP_SENSOR_4_IS_CUSTOM, t_index == CTI_HOTEND_4 ? F("HOTEND 4") :)
|
|
|
|
TERN_(TEMP_SENSOR_5_IS_CUSTOM, t_index == CTI_HOTEND_5 ? F("HOTEND 5") :)
|
|
|
|
TERN_(TEMP_SENSOR_6_IS_CUSTOM, t_index == CTI_HOTEND_6 ? F("HOTEND 6") :)
|
|
|
|
TERN_(TEMP_SENSOR_7_IS_CUSTOM, t_index == CTI_HOTEND_7 ? F("HOTEND 7") :)
|
|
|
|
TERN_(TEMP_SENSOR_BED_IS_CUSTOM, t_index == CTI_BED ? F("BED") :)
|
|
|
|
TERN_(TEMP_SENSOR_CHAMBER_IS_CUSTOM, t_index == CTI_CHAMBER ? F("CHAMBER") :)
|
|
|
|
TERN_(TEMP_SENSOR_COOLER_IS_CUSTOM, t_index == CTI_COOLER ? F("COOLER") :)
|
|
|
|
TERN_(TEMP_SENSOR_PROBE_IS_CUSTOM, t_index == CTI_PROBE ? F("PROBE") :)
|
|
|
|
TERN_(TEMP_SENSOR_BOARD_IS_CUSTOM, t_index == CTI_BOARD ? F("BOARD") :)
|
|
|
|
TERN_(TEMP_SENSOR_REDUNDANT_IS_CUSTOM, t_index == CTI_REDUNDANT ? F("REDUNDANT") :)
|
2019-05-09 11:45:55 -05:00
|
|
|
nullptr
|
2019-05-07 00:51:06 +01:00
|
|
|
);
|
|
|
|
SERIAL_EOL();
|
|
|
|
}
|
|
|
|
|
2021-04-27 16:36:39 +12:00
|
|
|
celsius_float_t Temperature::user_thermistor_to_deg_c(const uint8_t t_index, const int16_t raw) {
|
2019-05-07 00:51:06 +01:00
|
|
|
|
|
|
|
if (!WITHIN(t_index, 0, COUNT(user_thermistor) - 1)) return 25;
|
|
|
|
|
2019-09-14 03:05:10 -05:00
|
|
|
user_thermistor_t &t = user_thermistor[t_index];
|
|
|
|
if (t.pre_calc) { // pre-calculate some variables
|
|
|
|
t.pre_calc = false;
|
|
|
|
t.res_25_recip = 1.0f / t.res_25;
|
|
|
|
t.res_25_log = logf(t.res_25);
|
|
|
|
t.beta_recip = 1.0f / t.beta;
|
|
|
|
t.sh_alpha = RECIPROCAL(THERMISTOR_RESISTANCE_NOMINAL_C - (THERMISTOR_ABS_ZERO_C))
|
|
|
|
- (t.beta_recip * t.res_25_log) - (t.sh_c_coeff * cu(t.res_25_log));
|
2019-05-07 00:51:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// maximum adc value .. take into account the over sampling
|
2019-11-07 02:49:17 +03:00
|
|
|
const int adc_max = MAX_RAW_THERMISTOR_VALUE,
|
2019-05-07 00:51:06 +01:00
|
|
|
adc_raw = constrain(raw, 1, adc_max - 1); // constrain to prevent divide-by-zero
|
|
|
|
|
|
|
|
const float adc_inverse = (adc_max - adc_raw) - 0.5f,
|
2019-09-14 03:05:10 -05:00
|
|
|
resistance = t.series_res * (adc_raw + 0.5f) / adc_inverse,
|
2019-05-07 00:51:06 +01:00
|
|
|
log_resistance = logf(resistance);
|
|
|
|
|
2019-09-14 03:05:10 -05:00
|
|
|
float value = t.sh_alpha;
|
|
|
|
value += log_resistance * t.beta_recip;
|
|
|
|
if (t.sh_c_coeff != 0)
|
|
|
|
value += t.sh_c_coeff * cu(log_resistance);
|
2019-05-07 00:51:06 +01:00
|
|
|
value = 1.0f / value;
|
|
|
|
|
2019-05-22 03:31:05 +01:00
|
|
|
// Return degrees C (up to 999, as the LCD only displays 3 digits)
|
2019-07-05 18:01:21 -05:00
|
|
|
return _MIN(value + THERMISTOR_ABS_ZERO_C, 999);
|
2019-05-07 00:51:06 +01:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_HOTEND
|
2019-09-10 02:20:49 -05:00
|
|
|
// Derived from RepRap FiveD extruder::getTemperature()
|
|
|
|
// For hot end temperature measurement.
|
2021-04-27 16:36:39 +12:00
|
|
|
celsius_float_t Temperature::analog_to_celsius_hotend(const int16_t raw, const uint8_t e) {
|
2021-09-07 18:06:10 -05:00
|
|
|
if (e >= HOTENDS) {
|
|
|
|
SERIAL_ERROR_START();
|
|
|
|
SERIAL_ECHO(e);
|
|
|
|
SERIAL_ECHOLNPGM(STR_INVALID_EXTRUDER_NUM);
|
|
|
|
kill();
|
|
|
|
return 0;
|
|
|
|
}
|
2015-04-13 17:17:36 -07:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
switch (e) {
|
|
|
|
case 0:
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_0_IS_CUSTOM
|
2019-09-10 02:20:49 -05:00
|
|
|
return user_thermistor_to_deg_c(CTI_HOTEND_0, raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_0_IS_MAX_TC
|
2021-07-06 17:36:41 -07:00
|
|
|
#if TEMP_SENSOR_0_IS_MAX31865
|
|
|
|
return TERN(LIB_INTERNAL_MAX31865,
|
|
|
|
max31865_0.temperature((uint16_t)raw),
|
|
|
|
max31865_0.temperature(MAX31865_SENSOR_OHMS_0, MAX31865_CALIBRATION_OHMS_0)
|
|
|
|
);
|
|
|
|
#else
|
|
|
|
return raw * 0.25;
|
|
|
|
#endif
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_0_IS_AD595
|
2019-09-10 02:20:49 -05:00
|
|
|
return TEMP_AD595(raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_0_IS_AD8495
|
2019-09-10 02:20:49 -05:00
|
|
|
return TEMP_AD8495(raw);
|
|
|
|
#else
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case 1:
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_1_IS_CUSTOM
|
2019-09-10 02:20:49 -05:00
|
|
|
return user_thermistor_to_deg_c(CTI_HOTEND_1, raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_1_IS_MAX_TC
|
2021-07-06 17:36:41 -07:00
|
|
|
#if TEMP_SENSOR_0_IS_MAX31865
|
|
|
|
return TERN(LIB_INTERNAL_MAX31865,
|
|
|
|
max31865_1.temperature((uint16_t)raw),
|
|
|
|
max31865_1.temperature(MAX31865_SENSOR_OHMS_1, MAX31865_CALIBRATION_OHMS_1)
|
|
|
|
);
|
|
|
|
#else
|
|
|
|
return raw * 0.25;
|
|
|
|
#endif
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_1_IS_AD595
|
2019-09-10 02:20:49 -05:00
|
|
|
return TEMP_AD595(raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_1_IS_AD8495
|
2019-09-10 02:20:49 -05:00
|
|
|
return TEMP_AD8495(raw);
|
|
|
|
#else
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case 2:
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_2_IS_CUSTOM
|
2019-09-10 02:20:49 -05:00
|
|
|
return user_thermistor_to_deg_c(CTI_HOTEND_2, raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_2_IS_AD595
|
2019-09-10 02:20:49 -05:00
|
|
|
return TEMP_AD595(raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_2_IS_AD8495
|
2019-09-10 02:20:49 -05:00
|
|
|
return TEMP_AD8495(raw);
|
|
|
|
#else
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case 3:
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_3_IS_CUSTOM
|
2019-09-10 02:20:49 -05:00
|
|
|
return user_thermistor_to_deg_c(CTI_HOTEND_3, raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_3_IS_AD595
|
2019-09-10 02:20:49 -05:00
|
|
|
return TEMP_AD595(raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_3_IS_AD8495
|
2019-09-10 02:20:49 -05:00
|
|
|
return TEMP_AD8495(raw);
|
|
|
|
#else
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case 4:
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_4_IS_CUSTOM
|
2019-09-10 02:20:49 -05:00
|
|
|
return user_thermistor_to_deg_c(CTI_HOTEND_4, raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_4_IS_AD595
|
2019-09-10 02:20:49 -05:00
|
|
|
return TEMP_AD595(raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_4_IS_AD8495
|
2019-09-10 02:20:49 -05:00
|
|
|
return TEMP_AD8495(raw);
|
|
|
|
#else
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case 5:
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_5_IS_CUSTOM
|
2019-09-10 02:20:49 -05:00
|
|
|
return user_thermistor_to_deg_c(CTI_HOTEND_5, raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_5_IS_AD595
|
2019-09-10 02:20:49 -05:00
|
|
|
return TEMP_AD595(raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_5_IS_AD8495
|
2019-09-10 02:20:49 -05:00
|
|
|
return TEMP_AD8495(raw);
|
|
|
|
#else
|
|
|
|
break;
|
|
|
|
#endif
|
2020-01-25 16:13:39 +08:00
|
|
|
case 6:
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_6_IS_CUSTOM
|
2020-01-25 16:13:39 +08:00
|
|
|
return user_thermistor_to_deg_c(CTI_HOTEND_6, raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_6_IS_AD595
|
2020-01-25 16:13:39 +08:00
|
|
|
return TEMP_AD595(raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_6_IS_AD8495
|
2020-01-25 16:13:39 +08:00
|
|
|
return TEMP_AD8495(raw);
|
|
|
|
#else
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case 7:
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_7_IS_CUSTOM
|
2020-01-25 16:13:39 +08:00
|
|
|
return user_thermistor_to_deg_c(CTI_HOTEND_7, raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_7_IS_AD595
|
2020-01-25 16:13:39 +08:00
|
|
|
return TEMP_AD595(raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_7_IS_AD8495
|
2020-01-25 16:13:39 +08:00
|
|
|
return TEMP_AD8495(raw);
|
|
|
|
#else
|
|
|
|
break;
|
|
|
|
#endif
|
2019-09-10 02:20:49 -05:00
|
|
|
default: break;
|
|
|
|
}
|
2018-05-01 19:33:41 -05:00
|
|
|
|
2020-11-09 01:17:37 -06:00
|
|
|
#if HAS_HOTEND_THERMISTOR
|
2019-09-10 02:20:49 -05:00
|
|
|
// Thermistor with conversion table?
|
2020-07-01 16:27:28 -05:00
|
|
|
const temp_entry_t(*tt)[] = (temp_entry_t(*)[])(heater_ttbl_map[e]);
|
2019-09-10 02:20:49 -05:00
|
|
|
SCAN_THERMISTOR_TABLE((*tt), heater_ttbllen_map[e]);
|
|
|
|
#endif
|
2018-05-12 05:27:51 -05:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
return 0;
|
|
|
|
}
|
2020-07-01 16:27:28 -05:00
|
|
|
#endif // HAS_HOTEND
|
2011-11-15 22:50:43 +01:00
|
|
|
|
2018-04-23 17:13:01 -05:00
|
|
|
#if HAS_HEATED_BED
|
2018-01-01 18:24:54 -06:00
|
|
|
// For bed temperature measurement.
|
2021-04-27 16:36:39 +12:00
|
|
|
celsius_float_t Temperature::analog_to_celsius_bed(const int16_t raw) {
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_BED_IS_CUSTOM
|
2019-05-07 00:51:06 +01:00
|
|
|
return user_thermistor_to_deg_c(CTI_BED, raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_BED_IS_THERMISTOR
|
|
|
|
SCAN_THERMISTOR_TABLE(TEMPTABLE_BED, TEMPTABLE_BED_LEN);
|
|
|
|
#elif TEMP_SENSOR_BED_IS_AD595
|
2018-05-11 01:01:51 -05:00
|
|
|
return TEMP_AD595(raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_BED_IS_AD8495
|
2018-05-11 01:01:51 -05:00
|
|
|
return TEMP_AD8495(raw);
|
2018-01-01 18:24:54 -06:00
|
|
|
#else
|
2020-01-18 00:16:45 +01:00
|
|
|
UNUSED(raw);
|
2018-05-11 01:01:51 -05:00
|
|
|
return 0;
|
2018-01-01 18:24:54 -06:00
|
|
|
#endif
|
|
|
|
}
|
2018-04-23 17:13:01 -05:00
|
|
|
#endif // HAS_HEATED_BED
|
2011-11-15 22:50:43 +01:00
|
|
|
|
2018-02-24 19:38:58 +13:00
|
|
|
#if HAS_TEMP_CHAMBER
|
|
|
|
// For chamber temperature measurement.
|
2021-04-27 16:36:39 +12:00
|
|
|
celsius_float_t Temperature::analog_to_celsius_chamber(const int16_t raw) {
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_CHAMBER_IS_CUSTOM
|
2019-05-07 00:51:06 +01:00
|
|
|
return user_thermistor_to_deg_c(CTI_CHAMBER, raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_CHAMBER_IS_THERMISTOR
|
|
|
|
SCAN_THERMISTOR_TABLE(TEMPTABLE_CHAMBER, TEMPTABLE_CHAMBER_LEN);
|
|
|
|
#elif TEMP_SENSOR_CHAMBER_IS_AD595
|
2018-05-11 01:01:51 -05:00
|
|
|
return TEMP_AD595(raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_CHAMBER_IS_AD8495
|
2018-05-11 01:01:51 -05:00
|
|
|
return TEMP_AD8495(raw);
|
2018-02-24 19:38:58 +13:00
|
|
|
#else
|
2020-01-18 00:16:45 +01:00
|
|
|
UNUSED(raw);
|
2018-05-11 01:01:51 -05:00
|
|
|
return 0;
|
2018-02-24 19:38:58 +13:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif // HAS_TEMP_CHAMBER
|
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_TEMP_COOLER
|
|
|
|
// For cooler temperature measurement.
|
2021-04-27 16:36:39 +12:00
|
|
|
celsius_float_t Temperature::analog_to_celsius_cooler(const int16_t raw) {
|
2021-03-06 14:13:28 -06:00
|
|
|
#if TEMP_SENSOR_COOLER_IS_CUSTOM
|
|
|
|
return user_thermistor_to_deg_c(CTI_COOLER, raw);
|
|
|
|
#elif TEMP_SENSOR_COOLER_IS_THERMISTOR
|
|
|
|
SCAN_THERMISTOR_TABLE(TEMPTABLE_COOLER, TEMPTABLE_COOLER_LEN);
|
|
|
|
#elif TEMP_SENSOR_COOLER_IS_AD595
|
|
|
|
return TEMP_AD595(raw);
|
|
|
|
#elif TEMP_SENSOR_COOLER_IS_AD8495
|
|
|
|
return TEMP_AD8495(raw);
|
|
|
|
#else
|
|
|
|
UNUSED(raw);
|
|
|
|
return 0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif // HAS_TEMP_COOLER
|
|
|
|
|
2020-01-18 00:16:45 +01:00
|
|
|
#if HAS_TEMP_PROBE
|
|
|
|
// For probe temperature measurement.
|
2021-04-27 16:36:39 +12:00
|
|
|
celsius_float_t Temperature::analog_to_celsius_probe(const int16_t raw) {
|
2021-02-07 16:58:06 -06:00
|
|
|
#if TEMP_SENSOR_PROBE_IS_CUSTOM
|
2020-01-18 00:16:45 +01:00
|
|
|
return user_thermistor_to_deg_c(CTI_PROBE, raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_PROBE_IS_THERMISTOR
|
|
|
|
SCAN_THERMISTOR_TABLE(TEMPTABLE_PROBE, TEMPTABLE_PROBE_LEN);
|
|
|
|
#elif TEMP_SENSOR_PROBE_IS_AD595
|
2020-01-18 00:16:45 +01:00
|
|
|
return TEMP_AD595(raw);
|
2021-02-07 16:58:06 -06:00
|
|
|
#elif TEMP_SENSOR_PROBE_IS_AD8495
|
2020-01-18 00:16:45 +01:00
|
|
|
return TEMP_AD8495(raw);
|
|
|
|
#else
|
|
|
|
UNUSED(raw);
|
|
|
|
return 0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif // HAS_TEMP_PROBE
|
|
|
|
|
2021-08-18 20:12:41 -05:00
|
|
|
#if HAS_TEMP_BOARD
|
|
|
|
// For motherboard temperature measurement.
|
|
|
|
celsius_float_t Temperature::analog_to_celsius_board(const int16_t raw) {
|
|
|
|
#if TEMP_SENSOR_BOARD_IS_CUSTOM
|
|
|
|
return user_thermistor_to_deg_c(CTI_BOARD, raw);
|
|
|
|
#elif TEMP_SENSOR_BOARD_IS_THERMISTOR
|
|
|
|
SCAN_THERMISTOR_TABLE(TEMPTABLE_BOARD, TEMPTABLE_BOARD_LEN);
|
|
|
|
#elif TEMP_SENSOR_BOARD_IS_AD595
|
|
|
|
return TEMP_AD595(raw);
|
|
|
|
#elif TEMP_SENSOR_BOARD_IS_AD8495
|
|
|
|
return TEMP_AD8495(raw);
|
|
|
|
#else
|
|
|
|
UNUSED(raw);
|
|
|
|
return 0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif // HAS_TEMP_BOARD
|
|
|
|
|
2021-06-11 13:51:29 -07:00
|
|
|
#if HAS_TEMP_REDUNDANT
|
|
|
|
// For redundant temperature measurement.
|
|
|
|
celsius_float_t Temperature::analog_to_celsius_redundant(const int16_t raw) {
|
|
|
|
#if TEMP_SENSOR_REDUNDANT_IS_CUSTOM
|
|
|
|
return user_thermistor_to_deg_c(CTI_REDUNDANT, raw);
|
2021-08-18 20:12:41 -05:00
|
|
|
#elif TEMP_SENSOR_REDUNDANT_IS_MAX_TC && REDUNDANT_TEMP_MATCH(SOURCE, E0)
|
2021-07-06 17:36:41 -07:00
|
|
|
return TERN(TEMP_SENSOR_REDUNDANT_IS_MAX31865, max31865_0.temperature((uint16_t)raw), raw * 0.25);
|
2021-08-18 20:12:41 -05:00
|
|
|
#elif TEMP_SENSOR_REDUNDANT_IS_MAX_TC && REDUNDANT_TEMP_MATCH(SOURCE, E1)
|
2021-07-06 17:36:41 -07:00
|
|
|
return TERN(TEMP_SENSOR_REDUNDANT_IS_MAX31865, max31865_1.temperature((uint16_t)raw), raw * 0.25);
|
2021-06-11 13:51:29 -07:00
|
|
|
#elif TEMP_SENSOR_REDUNDANT_IS_THERMISTOR
|
|
|
|
SCAN_THERMISTOR_TABLE(TEMPTABLE_REDUNDANT, TEMPTABLE_REDUNDANT_LEN);
|
|
|
|
#elif TEMP_SENSOR_REDUNDANT_IS_AD595
|
|
|
|
return TEMP_AD595(raw);
|
|
|
|
#elif TEMP_SENSOR_REDUNDANT_IS_AD8495
|
|
|
|
return TEMP_AD8495(raw);
|
|
|
|
#else
|
|
|
|
UNUSED(raw);
|
|
|
|
return 0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#endif // HAS_TEMP_REDUNDANT
|
|
|
|
|
2016-04-28 18:18:13 -07:00
|
|
|
/**
|
2021-05-08 17:34:21 -05:00
|
|
|
* Convert the raw sensor readings into actual Celsius temperatures and
|
|
|
|
* validate raw temperatures. Bad readings generate min/maxtemp errors.
|
|
|
|
*
|
|
|
|
* The raw values are generated entirely in interrupt context, and this
|
|
|
|
* method is called from normal context once 'raw_temps_ready' has been
|
|
|
|
* set by update_raw_temperatures().
|
|
|
|
*
|
|
|
|
* The watchdog is dependent on this method. If 'raw_temps_ready' stops
|
|
|
|
* being set by the interrupt so that this method is not called for over
|
|
|
|
* 4 seconds then something has gone afoul and the machine will be reset.
|
2016-04-28 18:18:13 -07:00
|
|
|
*/
|
|
|
|
void Temperature::updateTemperaturesFromRawValues() {
|
2021-05-08 17:34:21 -05:00
|
|
|
|
|
|
|
watchdog_refresh(); // Reset because raw_temps_ready was set by the interrupt
|
|
|
|
|
2021-02-07 16:58:06 -06:00
|
|
|
TERN_(TEMP_SENSOR_0_IS_MAX_TC, temp_hotend[0].raw = READ_MAX_TC(0));
|
2021-06-11 13:51:29 -07:00
|
|
|
TERN_(TEMP_SENSOR_1_IS_MAX_TC, temp_hotend[1].raw = READ_MAX_TC(1));
|
2021-08-18 20:12:41 -05:00
|
|
|
TERN_(TEMP_SENSOR_REDUNDANT_IS_MAX_TC, temp_redundant.raw = READ_MAX_TC(HEATER_ID(TEMP_SENSOR_REDUNDANT_SOURCE)));
|
2021-07-06 17:36:41 -07:00
|
|
|
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_HOTEND
|
2019-09-10 02:20:49 -05:00
|
|
|
HOTEND_LOOP() temp_hotend[e].celsius = analog_to_celsius_hotend(temp_hotend[e].raw, e);
|
|
|
|
#endif
|
2021-05-08 17:34:21 -05:00
|
|
|
|
2021-06-11 13:51:29 -07:00
|
|
|
TERN_(HAS_HEATED_BED, temp_bed.celsius = analog_to_celsius_bed(temp_bed.raw));
|
|
|
|
TERN_(HAS_TEMP_CHAMBER, temp_chamber.celsius = analog_to_celsius_chamber(temp_chamber.raw));
|
|
|
|
TERN_(HAS_TEMP_COOLER, temp_cooler.celsius = analog_to_celsius_cooler(temp_cooler.raw));
|
|
|
|
TERN_(HAS_TEMP_PROBE, temp_probe.celsius = analog_to_celsius_probe(temp_probe.raw));
|
2021-08-18 20:12:41 -05:00
|
|
|
TERN_(HAS_TEMP_BOARD, temp_board.celsius = analog_to_celsius_board(temp_board.raw));
|
2021-06-11 13:51:29 -07:00
|
|
|
TERN_(HAS_TEMP_REDUNDANT, temp_redundant.celsius = analog_to_celsius_redundant(temp_redundant.raw));
|
2021-05-08 17:34:21 -05:00
|
|
|
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(FILAMENT_WIDTH_SENSOR, filwidth.update_measured_mm());
|
2021-05-08 17:34:21 -05:00
|
|
|
TERN_(HAS_POWER_MONITOR, power_monitor.capture_values());
|
2015-10-13 03:57:36 -07:00
|
|
|
|
2021-05-08 17:34:21 -05:00
|
|
|
#if HAS_HOTEND
|
|
|
|
static constexpr int8_t temp_dir[] = {
|
2021-06-11 13:51:29 -07:00
|
|
|
#if TEMP_SENSOR_IS_ANY_MAX_TC(0)
|
|
|
|
0
|
|
|
|
#else
|
|
|
|
TEMPDIR(0)
|
|
|
|
#endif
|
2021-05-08 17:34:21 -05:00
|
|
|
#if HAS_MULTI_HOTEND
|
2021-06-11 13:51:29 -07:00
|
|
|
#if TEMP_SENSOR_IS_ANY_MAX_TC(1)
|
|
|
|
, 0
|
|
|
|
#else
|
|
|
|
, TEMPDIR(1)
|
|
|
|
#endif
|
2021-05-08 17:34:21 -05:00
|
|
|
#if HOTENDS > 2
|
|
|
|
#define _TEMPDIR(N) , TEMPDIR(N)
|
|
|
|
REPEAT_S(2, HOTENDS, _TEMPDIR)
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
|
|
|
LOOP_L_N(e, COUNT(temp_dir)) {
|
|
|
|
const int8_t tdir = temp_dir[e];
|
|
|
|
if (tdir) {
|
|
|
|
const int16_t rawtemp = temp_hotend[e].raw * tdir; // normal direction, +rawtemp, else -rawtemp
|
|
|
|
if (rawtemp > temp_range[e].raw_max * tdir) max_temp_error((heater_id_t)e);
|
2015-02-26 00:33:30 -08:00
|
|
|
|
2021-05-08 17:34:21 -05:00
|
|
|
const bool heater_on = temp_hotend[e].target > 0;
|
|
|
|
if (heater_on && rawtemp < temp_range[e].raw_min * tdir && !is_preheating(e)) {
|
|
|
|
#if MAX_CONSECUTIVE_LOW_TEMPERATURE_ERROR_ALLOWED > 1
|
|
|
|
if (++consecutive_low_temperature_error[e] >= MAX_CONSECUTIVE_LOW_TEMPERATURE_ERROR_ALLOWED)
|
|
|
|
#endif
|
|
|
|
min_temp_error((heater_id_t)e);
|
|
|
|
}
|
|
|
|
#if MAX_CONSECUTIVE_LOW_TEMPERATURE_ERROR_ALLOWED > 1
|
|
|
|
else
|
|
|
|
consecutive_low_temperature_error[e] = 0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // HAS_HOTEND
|
|
|
|
|
2021-08-18 20:12:41 -05:00
|
|
|
#define TP_CMP(S,A,B) (TEMPDIR(S) < 0 ? ((A)<(B)) : ((A)>(B)))
|
2021-05-08 17:34:21 -05:00
|
|
|
#if ENABLED(THERMAL_PROTECTION_BED)
|
2021-08-18 20:12:41 -05:00
|
|
|
if (TP_CMP(BED, temp_bed.raw, maxtemp_raw_BED)) max_temp_error(H_BED);
|
|
|
|
if (temp_bed.target > 0 && TP_CMP(BED, mintemp_raw_BED, temp_bed.raw)) min_temp_error(H_BED);
|
2021-05-08 17:34:21 -05:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if BOTH(HAS_HEATED_CHAMBER, THERMAL_PROTECTION_CHAMBER)
|
2021-08-18 20:12:41 -05:00
|
|
|
if (TP_CMP(CHAMBER, temp_chamber.raw, maxtemp_raw_CHAMBER)) max_temp_error(H_CHAMBER);
|
|
|
|
if (temp_chamber.target > 0 && TP_CMP(CHAMBER, mintemp_raw_CHAMBER, temp_chamber.raw)) min_temp_error(H_CHAMBER);
|
2021-05-08 17:34:21 -05:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if BOTH(HAS_COOLER, THERMAL_PROTECTION_COOLER)
|
2021-08-18 20:12:41 -05:00
|
|
|
if (cutter.unitPower > 0 && TP_CMP(COOLER, temp_cooler.raw, maxtemp_raw_COOLER)) max_temp_error(H_COOLER);
|
|
|
|
if (TP_CMP(COOLER, mintemp_raw_COOLER, temp_cooler.raw)) min_temp_error(H_COOLER);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if BOTH(HAS_TEMP_BOARD, THERMAL_PROTECTION_BOARD)
|
|
|
|
if (TP_CMP(BOARD, temp_board.raw, maxtemp_raw_BOARD)) max_temp_error(H_BOARD);
|
|
|
|
if (TP_CMP(BOARD, mintemp_raw_BOARD, temp_board.raw)) min_temp_error(H_BOARD);
|
2021-05-08 17:34:21 -05:00
|
|
|
#endif
|
2021-08-18 20:12:41 -05:00
|
|
|
#undef TP_CMP
|
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
} // Temperature::updateTemperaturesFromRawValues
|
2019-03-13 06:51:15 -05:00
|
|
|
|
2015-04-13 17:17:36 -07:00
|
|
|
/**
|
|
|
|
* Initialize the temperature manager
|
2021-05-02 21:32:21 -05:00
|
|
|
*
|
2015-04-13 17:17:36 -07:00
|
|
|
* The manager is implemented by periodic calls to manage_heater()
|
2021-05-02 21:32:21 -05:00
|
|
|
*
|
|
|
|
* - Init (and disable) SPI thermocouples like MAX6675 and MAX31865
|
|
|
|
* - Disable RUMBA JTAG to accommodate a thermocouple extension
|
|
|
|
* - Read-enable thermistors with a read-enable pin
|
|
|
|
* - Init HEATER and COOLER pins for OUTPUT in OFF state
|
|
|
|
* - Init the FAN pins as PWM or OUTPUT
|
|
|
|
* - Init the SPI interface for SPI thermocouples
|
|
|
|
* - Init ADC according to the HAL
|
|
|
|
* - Set thermistor pins to analog inputs according to the HAL
|
|
|
|
* - Start the Temperature ISR timer
|
|
|
|
* - Init the AUTO FAN pins as PWM or OUTPUT
|
|
|
|
* - Wait 250ms for temperatures to settle
|
|
|
|
* - Init temp_range[], used for catching min/maxtemp
|
2015-04-13 17:17:36 -07:00
|
|
|
*/
|
2016-04-28 18:18:13 -07:00
|
|
|
void Temperature::init() {
|
2021-05-02 21:32:21 -05:00
|
|
|
|
|
|
|
TERN_(PROBING_HEATERS_OFF, paused_for_probing = false);
|
|
|
|
|
|
|
|
#if BOTH(PIDTEMP, PID_EXTRUSION_SCALING)
|
|
|
|
last_e_position = 0;
|
|
|
|
#endif
|
|
|
|
|
2021-02-08 20:51:57 -05:00
|
|
|
// Init (and disable) SPI thermocouples
|
2021-07-06 17:36:41 -07:00
|
|
|
#if TEMP_SENSOR_IS_ANY_MAX_TC(0) && PIN_EXISTS(TEMP_0_CS)
|
|
|
|
OUT_WRITE(TEMP_0_CS_PIN, HIGH);
|
|
|
|
#endif
|
|
|
|
#if TEMP_SENSOR_IS_ANY_MAX_TC(1) && PIN_EXISTS(TEMP_1_CS)
|
|
|
|
OUT_WRITE(TEMP_1_CS_PIN, HIGH);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Setup objects for library-based polling of MAX TCs
|
|
|
|
#if HAS_MAXTC_LIBRARIES
|
|
|
|
#define _MAX31865_WIRES(n) MAX31865_##n##WIRE
|
|
|
|
#define MAX31865_WIRES(n) _MAX31865_WIRES(n)
|
|
|
|
|
|
|
|
#if TEMP_SENSOR_IS_MAX(0, 6675) && HAS_MAX6675_LIBRARY
|
|
|
|
max6675_0.begin();
|
|
|
|
#elif TEMP_SENSOR_IS_MAX(0, 31855) && HAS_MAX31855_LIBRARY
|
|
|
|
max31855_0.begin();
|
|
|
|
#elif TEMP_SENSOR_IS_MAX(0, 31865)
|
|
|
|
max31865_0.begin(
|
|
|
|
MAX31865_WIRES(MAX31865_SENSOR_WIRES_0) // MAX31865_2WIRE, MAX31865_3WIRE, MAX31865_4WIRE
|
|
|
|
OPTARG(LIB_INTERNAL_MAX31865, MAX31865_SENSOR_OHMS_0, MAX31865_CALIBRATION_OHMS_0)
|
|
|
|
);
|
|
|
|
#if defined(LIB_INTERNAL_MAX31865) && ENABLED(MAX31865_50HZ_FILTER)
|
|
|
|
max31865_0.enable50HzFilter(1);
|
|
|
|
#endif
|
2021-06-19 11:44:28 -07:00
|
|
|
#endif
|
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
#if TEMP_SENSOR_IS_MAX(1, 6675) && HAS_MAX6675_LIBRARY
|
|
|
|
max6675_1.begin();
|
|
|
|
#elif TEMP_SENSOR_IS_MAX(1, 31855) && HAS_MAX31855_LIBRARY
|
|
|
|
max31855_1.begin();
|
|
|
|
#elif TEMP_SENSOR_IS_MAX(1, 31865)
|
|
|
|
max31865_1.begin(
|
|
|
|
MAX31865_WIRES(MAX31865_SENSOR_WIRES_1) // MAX31865_2WIRE, MAX31865_3WIRE, MAX31865_4WIRE
|
|
|
|
OPTARG(LIB_INTERNAL_MAX31865, MAX31865_SENSOR_OHMS_1, MAX31865_CALIBRATION_OHMS_1)
|
|
|
|
);
|
|
|
|
#if defined(LIB_INTERNAL_MAX31865) && ENABLED(MAX31865_50HZ_FILTER)
|
|
|
|
max31865_1.enable50HzFilter(1);
|
|
|
|
#endif
|
2021-06-19 11:44:28 -07:00
|
|
|
#endif
|
2021-07-06 17:36:41 -07:00
|
|
|
#undef MAX31865_WIRES
|
|
|
|
#undef _MAX31865_WIRES
|
2021-02-08 20:51:57 -05:00
|
|
|
#endif
|
2019-11-21 11:15:10 +02:00
|
|
|
|
2019-03-07 00:09:39 -08:00
|
|
|
#if MB(RUMBA)
|
2020-02-27 07:47:40 -06:00
|
|
|
// Disable RUMBA JTAG in case the thermocouple extension is plugged on top of JTAG connector
|
2021-02-07 16:58:06 -06:00
|
|
|
#define _AD(N) (TEMP_SENSOR_##N##_IS_AD595 || TEMP_SENSOR_##N##_IS_AD8495)
|
2021-06-11 13:51:29 -07:00
|
|
|
#if _AD(0) || _AD(1) || _AD(2) || _AD(BED) || _AD(CHAMBER) || _AD(REDUNDANT)
|
2019-03-07 00:09:39 -08:00
|
|
|
MCUCR = _BV(JTD);
|
|
|
|
MCUCR = _BV(JTD);
|
|
|
|
#endif
|
2015-02-24 04:46:11 -08:00
|
|
|
#endif
|
2015-10-02 23:08:58 -07:00
|
|
|
|
2020-07-06 15:08:52 -07:00
|
|
|
// Thermistor activation by MCU pin
|
2021-02-07 16:58:06 -06:00
|
|
|
#if PIN_EXISTS(TEMP_0_TR_ENABLE)
|
2021-10-18 01:49:35 -05:00
|
|
|
OUT_WRITE(TEMP_0_TR_ENABLE_PIN, (
|
2021-06-11 13:51:29 -07:00
|
|
|
#if TEMP_SENSOR_IS_ANY_MAX_TC(0)
|
2021-10-18 01:49:35 -05:00
|
|
|
HIGH
|
2021-06-11 13:51:29 -07:00
|
|
|
#else
|
2021-10-18 01:49:35 -05:00
|
|
|
LOW
|
2021-06-11 13:51:29 -07:00
|
|
|
#endif
|
2021-10-18 01:49:35 -05:00
|
|
|
));
|
2020-07-06 15:08:52 -07:00
|
|
|
#endif
|
2021-02-07 16:58:06 -06:00
|
|
|
#if PIN_EXISTS(TEMP_1_TR_ENABLE)
|
2021-10-18 01:49:35 -05:00
|
|
|
OUT_WRITE(TEMP_1_TR_ENABLE_PIN, (
|
2021-06-11 13:51:29 -07:00
|
|
|
#if TEMP_SENSOR_IS_ANY_MAX_TC(1)
|
2021-10-18 01:49:35 -05:00
|
|
|
HIGH
|
2021-06-11 13:51:29 -07:00
|
|
|
#else
|
2021-10-18 01:49:35 -05:00
|
|
|
LOW
|
2021-06-11 13:51:29 -07:00
|
|
|
#endif
|
2021-10-18 01:49:35 -05:00
|
|
|
));
|
2020-07-06 15:08:52 -07:00
|
|
|
#endif
|
|
|
|
|
2015-02-26 00:33:30 -08:00
|
|
|
#if HAS_HEATER_0
|
2021-02-25 11:35:18 +01:00
|
|
|
#ifdef BOARD_OPENDRAIN_MOSFETS
|
2019-07-15 01:05:24 +02:00
|
|
|
OUT_WRITE_OD(HEATER_0_PIN, HEATER_0_INVERTING);
|
|
|
|
#else
|
|
|
|
OUT_WRITE(HEATER_0_PIN, HEATER_0_INVERTING);
|
|
|
|
#endif
|
2015-01-23 23:13:06 +01:00
|
|
|
#endif
|
2015-02-26 00:33:30 -08:00
|
|
|
#if HAS_HEATER_1
|
2018-04-03 02:37:41 +02:00
|
|
|
OUT_WRITE(HEATER_1_PIN, HEATER_1_INVERTING);
|
2015-01-23 10:37:39 -08:00
|
|
|
#endif
|
2015-02-26 00:33:30 -08:00
|
|
|
#if HAS_HEATER_2
|
2018-04-03 02:37:41 +02:00
|
|
|
OUT_WRITE(HEATER_2_PIN, HEATER_2_INVERTING);
|
2015-01-23 23:13:06 +01:00
|
|
|
#endif
|
2015-02-26 00:33:30 -08:00
|
|
|
#if HAS_HEATER_3
|
2018-04-03 02:37:41 +02:00
|
|
|
OUT_WRITE(HEATER_3_PIN, HEATER_3_INVERTING);
|
2015-01-23 23:13:06 +01:00
|
|
|
#endif
|
2017-04-06 16:46:52 -05:00
|
|
|
#if HAS_HEATER_4
|
2019-01-11 22:54:15 -06:00
|
|
|
OUT_WRITE(HEATER_4_PIN, HEATER_4_INVERTING);
|
2017-04-06 16:46:52 -05:00
|
|
|
#endif
|
2019-03-07 00:09:39 -08:00
|
|
|
#if HAS_HEATER_5
|
|
|
|
OUT_WRITE(HEATER_5_PIN, HEATER_5_INVERTING);
|
|
|
|
#endif
|
2020-01-25 16:13:39 +08:00
|
|
|
#if HAS_HEATER_6
|
|
|
|
OUT_WRITE(HEATER_6_PIN, HEATER_6_INVERTING);
|
|
|
|
#endif
|
|
|
|
#if HAS_HEATER_7
|
|
|
|
OUT_WRITE(HEATER_7_PIN, HEATER_7_INVERTING);
|
|
|
|
#endif
|
2019-07-15 01:05:24 +02:00
|
|
|
|
2018-04-23 17:13:01 -05:00
|
|
|
#if HAS_HEATED_BED
|
2021-02-25 11:35:18 +01:00
|
|
|
#ifdef BOARD_OPENDRAIN_MOSFETS
|
2019-07-15 01:05:24 +02:00
|
|
|
OUT_WRITE_OD(HEATER_BED_PIN, HEATER_BED_INVERTING);
|
|
|
|
#else
|
|
|
|
OUT_WRITE(HEATER_BED_PIN, HEATER_BED_INVERTING);
|
|
|
|
#endif
|
2015-10-02 23:08:58 -07:00
|
|
|
#endif
|
2019-07-15 01:05:24 +02:00
|
|
|
|
2019-03-07 00:09:39 -08:00
|
|
|
#if HAS_HEATED_CHAMBER
|
|
|
|
OUT_WRITE(HEATER_CHAMBER_PIN, HEATER_CHAMBER_INVERTING);
|
|
|
|
#endif
|
2019-03-13 06:51:15 -05:00
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_COOLER
|
|
|
|
OUT_WRITE(COOLER_PIN, COOLER_INVERTING);
|
|
|
|
#endif
|
|
|
|
|
2016-08-13 20:16:13 -07:00
|
|
|
#if HAS_FAN0
|
2019-03-13 06:51:15 -05:00
|
|
|
INIT_FAN_PIN(FAN_PIN);
|
2016-08-13 20:16:13 -07:00
|
|
|
#endif
|
|
|
|
#if HAS_FAN1
|
2019-03-13 06:51:15 -05:00
|
|
|
INIT_FAN_PIN(FAN1_PIN);
|
2016-08-13 20:16:13 -07:00
|
|
|
#endif
|
|
|
|
#if HAS_FAN2
|
2019-03-13 06:51:15 -05:00
|
|
|
INIT_FAN_PIN(FAN2_PIN);
|
2016-08-13 20:16:13 -07:00
|
|
|
#endif
|
2020-01-25 16:13:39 +08:00
|
|
|
#if HAS_FAN3
|
|
|
|
INIT_FAN_PIN(FAN3_PIN);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN4
|
|
|
|
INIT_FAN_PIN(FAN4_PIN);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN5
|
|
|
|
INIT_FAN_PIN(FAN5_PIN);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN6
|
|
|
|
INIT_FAN_PIN(FAN6_PIN);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN7
|
|
|
|
INIT_FAN_PIN(FAN7_PIN);
|
|
|
|
#endif
|
2018-11-23 23:03:39 +01:00
|
|
|
#if ENABLED(USE_CONTROLLER_FAN)
|
2019-03-13 06:51:15 -05:00
|
|
|
INIT_FAN_PIN(CONTROLLER_FAN_PIN);
|
2018-11-23 23:03:39 +01:00
|
|
|
#endif
|
|
|
|
|
2021-08-21 18:00:55 -05:00
|
|
|
TERN_(HAS_MAXTC_SW_SPI, max_tc_spi.init());
|
2018-11-17 01:20:33 +01:00
|
|
|
|
2017-06-18 00:36:10 +01:00
|
|
|
HAL_adc_init();
|
2015-02-24 04:46:11 -08:00
|
|
|
|
2018-05-01 19:33:41 -05:00
|
|
|
#if HAS_TEMP_ADC_0
|
2017-06-18 00:36:10 +01:00
|
|
|
HAL_ANALOG_SELECT(TEMP_0_PIN);
|
2011-11-20 13:14:58 +01:00
|
|
|
#endif
|
2018-05-01 19:33:41 -05:00
|
|
|
#if HAS_TEMP_ADC_1
|
2017-06-18 00:36:10 +01:00
|
|
|
HAL_ANALOG_SELECT(TEMP_1_PIN);
|
2011-11-20 13:14:58 +01:00
|
|
|
#endif
|
2018-05-01 19:33:41 -05:00
|
|
|
#if HAS_TEMP_ADC_2
|
2017-06-18 00:36:10 +01:00
|
|
|
HAL_ANALOG_SELECT(TEMP_2_PIN);
|
2015-01-23 23:13:06 +01:00
|
|
|
#endif
|
2018-05-01 19:33:41 -05:00
|
|
|
#if HAS_TEMP_ADC_3
|
2017-06-18 00:36:10 +01:00
|
|
|
HAL_ANALOG_SELECT(TEMP_3_PIN);
|
2011-11-20 13:14:58 +01:00
|
|
|
#endif
|
2018-05-01 19:33:41 -05:00
|
|
|
#if HAS_TEMP_ADC_4
|
2017-06-18 00:36:10 +01:00
|
|
|
HAL_ANALOG_SELECT(TEMP_4_PIN);
|
2017-04-06 16:46:52 -05:00
|
|
|
#endif
|
2018-09-13 01:35:55 -05:00
|
|
|
#if HAS_TEMP_ADC_5
|
|
|
|
HAL_ANALOG_SELECT(TEMP_5_PIN);
|
|
|
|
#endif
|
2020-01-25 16:13:39 +08:00
|
|
|
#if HAS_TEMP_ADC_6
|
|
|
|
HAL_ANALOG_SELECT(TEMP_6_PIN);
|
|
|
|
#endif
|
|
|
|
#if HAS_TEMP_ADC_7
|
|
|
|
HAL_ANALOG_SELECT(TEMP_7_PIN);
|
|
|
|
#endif
|
2019-09-08 00:55:34 -05:00
|
|
|
#if HAS_JOY_ADC_X
|
|
|
|
HAL_ANALOG_SELECT(JOY_X_PIN);
|
|
|
|
#endif
|
|
|
|
#if HAS_JOY_ADC_Y
|
|
|
|
HAL_ANALOG_SELECT(JOY_Y_PIN);
|
|
|
|
#endif
|
|
|
|
#if HAS_JOY_ADC_Z
|
|
|
|
HAL_ANALOG_SELECT(JOY_Z_PIN);
|
|
|
|
#endif
|
|
|
|
#if HAS_JOY_ADC_EN
|
|
|
|
SET_INPUT_PULLUP(JOY_EN_PIN);
|
|
|
|
#endif
|
2020-11-22 16:47:52 -08:00
|
|
|
#if HAS_TEMP_ADC_BED
|
2017-06-18 00:36:10 +01:00
|
|
|
HAL_ANALOG_SELECT(TEMP_BED_PIN);
|
2011-12-05 23:33:33 -05:00
|
|
|
#endif
|
2020-11-22 16:47:52 -08:00
|
|
|
#if HAS_TEMP_ADC_CHAMBER
|
2018-02-24 19:38:58 +13:00
|
|
|
HAL_ANALOG_SELECT(TEMP_CHAMBER_PIN);
|
|
|
|
#endif
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_TEMP_ADC_COOLER
|
|
|
|
HAL_ANALOG_SELECT(TEMP_COOLER_PIN);
|
|
|
|
#endif
|
2020-11-22 16:47:52 -08:00
|
|
|
#if HAS_TEMP_ADC_PROBE
|
2020-01-18 00:16:45 +01:00
|
|
|
HAL_ANALOG_SELECT(TEMP_PROBE_PIN);
|
|
|
|
#endif
|
2021-08-18 20:12:41 -05:00
|
|
|
#if HAS_TEMP_ADC_BOARD
|
|
|
|
HAL_ANALOG_SELECT(TEMP_BOARD_PIN);
|
|
|
|
#endif
|
2021-06-11 13:51:29 -07:00
|
|
|
#if HAS_TEMP_ADC_REDUNDANT
|
|
|
|
HAL_ANALOG_SELECT(TEMP_REDUNDANT_PIN);
|
|
|
|
#endif
|
2016-03-29 20:28:23 -07:00
|
|
|
#if ENABLED(FILAMENT_WIDTH_SENSOR)
|
2017-06-18 00:36:10 +01:00
|
|
|
HAL_ANALOG_SELECT(FILWIDTH_PIN);
|
|
|
|
#endif
|
2019-07-11 23:47:41 +05:00
|
|
|
#if HAS_ADC_BUTTONS
|
|
|
|
HAL_ANALOG_SELECT(ADC_KEYPAD_PIN);
|
|
|
|
#endif
|
2020-06-18 15:23:03 -05:00
|
|
|
#if ENABLED(POWER_MONITOR_CURRENT)
|
|
|
|
HAL_ANALOG_SELECT(POWER_MONITOR_CURRENT_PIN);
|
|
|
|
#endif
|
|
|
|
#if ENABLED(POWER_MONITOR_VOLTAGE)
|
|
|
|
HAL_ANALOG_SELECT(POWER_MONITOR_VOLTAGE_PIN);
|
|
|
|
#endif
|
2017-06-18 00:36:10 +01:00
|
|
|
|
2018-06-03 01:43:00 -05:00
|
|
|
HAL_timer_start(TEMP_TIMER_NUM, TEMP_TIMER_FREQUENCY);
|
2018-04-24 00:05:07 -03:00
|
|
|
ENABLE_TEMPERATURE_INTERRUPT();
|
2015-06-09 17:01:47 -07:00
|
|
|
|
2015-06-09 16:54:28 -07:00
|
|
|
#if HAS_AUTO_FAN_0
|
2019-05-05 05:51:47 +02:00
|
|
|
INIT_E_AUTO_FAN_PIN(E0_AUTO_FAN_PIN);
|
2015-06-09 16:54:28 -07:00
|
|
|
#endif
|
2019-10-09 18:46:10 -06:00
|
|
|
#if HAS_AUTO_FAN_1 && !_EFANOVERLAP(1,0)
|
2019-05-05 05:51:47 +02:00
|
|
|
INIT_E_AUTO_FAN_PIN(E1_AUTO_FAN_PIN);
|
2015-06-09 16:54:28 -07:00
|
|
|
#endif
|
2019-10-09 18:46:10 -06:00
|
|
|
#if HAS_AUTO_FAN_2 && !(_EFANOVERLAP(2,0) || _EFANOVERLAP(2,1))
|
2019-05-05 05:51:47 +02:00
|
|
|
INIT_E_AUTO_FAN_PIN(E2_AUTO_FAN_PIN);
|
2015-06-09 16:54:28 -07:00
|
|
|
#endif
|
2019-10-09 18:46:10 -06:00
|
|
|
#if HAS_AUTO_FAN_3 && !(_EFANOVERLAP(3,0) || _EFANOVERLAP(3,1) || _EFANOVERLAP(3,2))
|
2019-05-05 05:51:47 +02:00
|
|
|
INIT_E_AUTO_FAN_PIN(E3_AUTO_FAN_PIN);
|
2015-06-09 16:54:28 -07:00
|
|
|
#endif
|
2019-10-09 18:46:10 -06:00
|
|
|
#if HAS_AUTO_FAN_4 && !(_EFANOVERLAP(4,0) || _EFANOVERLAP(4,1) || _EFANOVERLAP(4,2) || _EFANOVERLAP(4,3))
|
2019-05-05 05:51:47 +02:00
|
|
|
INIT_E_AUTO_FAN_PIN(E4_AUTO_FAN_PIN);
|
2017-04-06 16:46:52 -05:00
|
|
|
#endif
|
2019-10-09 18:46:10 -06:00
|
|
|
#if HAS_AUTO_FAN_5 && !(_EFANOVERLAP(5,0) || _EFANOVERLAP(5,1) || _EFANOVERLAP(5,2) || _EFANOVERLAP(5,3) || _EFANOVERLAP(5,4))
|
2019-05-05 05:51:47 +02:00
|
|
|
INIT_E_AUTO_FAN_PIN(E5_AUTO_FAN_PIN);
|
2018-09-13 01:35:55 -05:00
|
|
|
#endif
|
2020-01-25 16:13:39 +08:00
|
|
|
#if HAS_AUTO_FAN_6 && !(_EFANOVERLAP(6,0) || _EFANOVERLAP(6,1) || _EFANOVERLAP(6,2) || _EFANOVERLAP(6,3) || _EFANOVERLAP(6,4) || _EFANOVERLAP(6,5))
|
|
|
|
INIT_E_AUTO_FAN_PIN(E6_AUTO_FAN_PIN);
|
|
|
|
#endif
|
|
|
|
#if HAS_AUTO_FAN_7 && !(_EFANOVERLAP(7,0) || _EFANOVERLAP(7,1) || _EFANOVERLAP(7,2) || _EFANOVERLAP(7,3) || _EFANOVERLAP(7,4) || _EFANOVERLAP(7,5) || _EFANOVERLAP(7,6))
|
|
|
|
INIT_E_AUTO_FAN_PIN(E7_AUTO_FAN_PIN);
|
|
|
|
#endif
|
2019-05-05 05:51:47 +02:00
|
|
|
#if HAS_AUTO_CHAMBER_FAN && !AUTO_CHAMBER_IS_E
|
|
|
|
INIT_CHAMBER_AUTO_FAN_PIN(CHAMBER_AUTO_FAN_PIN);
|
2018-02-24 19:38:58 +13:00
|
|
|
#endif
|
2015-06-09 16:54:28 -07:00
|
|
|
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_HOTEND
|
2019-03-07 00:09:39 -08:00
|
|
|
#define _TEMP_MIN_E(NR) do{ \
|
2021-04-13 14:02:21 +12:00
|
|
|
const celsius_t tmin = _MAX(HEATER_##NR##_MINTEMP, TERN(TEMP_SENSOR_##NR##_IS_CUSTOM, 0, (int)pgm_read_word(&TEMPTABLE_##NR [TEMP_SENSOR_##NR##_MINTEMP_IND].celsius))); \
|
2020-07-01 16:27:28 -05:00
|
|
|
temp_range[NR].mintemp = tmin; \
|
|
|
|
while (analog_to_celsius_hotend(temp_range[NR].raw_min, NR) < tmin) \
|
2019-03-07 00:09:39 -08:00
|
|
|
temp_range[NR].raw_min += TEMPDIR(NR) * (OVERSAMPLENR); \
|
|
|
|
}while(0)
|
|
|
|
#define _TEMP_MAX_E(NR) do{ \
|
2021-04-13 14:02:21 +12:00
|
|
|
const celsius_t tmax = _MIN(HEATER_##NR##_MAXTEMP, TERN(TEMP_SENSOR_##NR##_IS_CUSTOM, 2000, (int)pgm_read_word(&TEMPTABLE_##NR [TEMP_SENSOR_##NR##_MAXTEMP_IND].celsius) - 1)); \
|
2020-07-01 16:27:28 -05:00
|
|
|
temp_range[NR].maxtemp = tmax; \
|
|
|
|
while (analog_to_celsius_hotend(temp_range[NR].raw_max, NR) > tmax) \
|
2019-03-07 00:09:39 -08:00
|
|
|
temp_range[NR].raw_max -= TEMPDIR(NR) * (OVERSAMPLENR); \
|
|
|
|
}while(0)
|
|
|
|
|
2021-06-11 13:51:29 -07:00
|
|
|
#define _MINMAX_TEST(N,M) (HOTENDS > N && TEMP_SENSOR_##N > 0 && TEMP_SENSOR_##N != 998 && TEMP_SENSOR_##N != 999 && defined(HEATER_##N##_##M##TEMP))
|
2020-07-07 01:50:57 -05:00
|
|
|
|
2020-07-05 13:04:31 +12:00
|
|
|
#if _MINMAX_TEST(0, MIN)
|
2019-03-07 00:09:39 -08:00
|
|
|
_TEMP_MIN_E(0);
|
2015-02-24 04:46:11 -08:00
|
|
|
#endif
|
2020-07-05 13:04:31 +12:00
|
|
|
#if _MINMAX_TEST(0, MAX)
|
2020-07-07 01:50:57 -05:00
|
|
|
_TEMP_MAX_E(0);
|
2020-07-01 16:27:28 -05:00
|
|
|
#endif
|
2020-07-05 13:04:31 +12:00
|
|
|
#if _MINMAX_TEST(1, MIN)
|
|
|
|
_TEMP_MIN_E(1);
|
2020-07-01 16:27:28 -05:00
|
|
|
#endif
|
2020-07-05 13:04:31 +12:00
|
|
|
#if _MINMAX_TEST(1, MAX)
|
|
|
|
_TEMP_MAX_E(1);
|
2020-07-01 16:27:28 -05:00
|
|
|
#endif
|
2020-07-05 13:04:31 +12:00
|
|
|
#if _MINMAX_TEST(2, MIN)
|
|
|
|
_TEMP_MIN_E(2);
|
2020-07-01 16:27:28 -05:00
|
|
|
#endif
|
2020-07-05 13:04:31 +12:00
|
|
|
#if _MINMAX_TEST(2, MAX)
|
|
|
|
_TEMP_MAX_E(2);
|
2020-07-01 16:27:28 -05:00
|
|
|
#endif
|
2020-07-05 13:04:31 +12:00
|
|
|
#if _MINMAX_TEST(3, MIN)
|
|
|
|
_TEMP_MIN_E(3);
|
2020-07-01 16:27:28 -05:00
|
|
|
#endif
|
2020-07-05 13:04:31 +12:00
|
|
|
#if _MINMAX_TEST(3, MAX)
|
|
|
|
_TEMP_MAX_E(3);
|
|
|
|
#endif
|
|
|
|
#if _MINMAX_TEST(4, MIN)
|
|
|
|
_TEMP_MIN_E(4);
|
|
|
|
#endif
|
|
|
|
#if _MINMAX_TEST(4, MAX)
|
|
|
|
_TEMP_MAX_E(4);
|
|
|
|
#endif
|
|
|
|
#if _MINMAX_TEST(5, MIN)
|
|
|
|
_TEMP_MIN_E(5);
|
|
|
|
#endif
|
|
|
|
#if _MINMAX_TEST(5, MAX)
|
|
|
|
_TEMP_MAX_E(5);
|
|
|
|
#endif
|
|
|
|
#if _MINMAX_TEST(6, MIN)
|
|
|
|
_TEMP_MIN_E(6);
|
|
|
|
#endif
|
|
|
|
#if _MINMAX_TEST(6, MAX)
|
|
|
|
_TEMP_MAX_E(6);
|
|
|
|
#endif
|
|
|
|
#if _MINMAX_TEST(7, MIN)
|
|
|
|
_TEMP_MIN_E(7);
|
|
|
|
#endif
|
|
|
|
#if _MINMAX_TEST(7, MAX)
|
|
|
|
_TEMP_MAX_E(7);
|
2020-07-01 16:27:28 -05:00
|
|
|
#endif
|
|
|
|
#endif // HAS_HOTEND
|
2015-02-24 04:46:11 -08:00
|
|
|
|
2021-08-18 20:12:41 -05:00
|
|
|
// TODO: combine these into the macros above
|
2018-04-23 17:13:01 -05:00
|
|
|
#if HAS_HEATED_BED
|
2021-03-24 04:11:43 -05:00
|
|
|
while (analog_to_celsius_bed(mintemp_raw_BED) < BED_MINTEMP) mintemp_raw_BED += TEMPDIR(BED) * (OVERSAMPLENR);
|
|
|
|
while (analog_to_celsius_bed(maxtemp_raw_BED) > BED_MAXTEMP) maxtemp_raw_BED -= TEMPDIR(BED) * (OVERSAMPLENR);
|
|
|
|
#endif
|
2017-05-07 07:06:06 -04:00
|
|
|
|
2019-03-07 00:09:39 -08:00
|
|
|
#if HAS_HEATED_CHAMBER
|
2021-03-24 04:11:43 -05:00
|
|
|
while (analog_to_celsius_chamber(mintemp_raw_CHAMBER) < CHAMBER_MINTEMP) mintemp_raw_CHAMBER += TEMPDIR(CHAMBER) * (OVERSAMPLENR);
|
|
|
|
while (analog_to_celsius_chamber(maxtemp_raw_CHAMBER) > CHAMBER_MAXTEMP) maxtemp_raw_CHAMBER -= TEMPDIR(CHAMBER) * (OVERSAMPLENR);
|
2019-03-07 00:09:39 -08:00
|
|
|
#endif
|
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_COOLER
|
2021-03-24 04:11:43 -05:00
|
|
|
while (analog_to_celsius_cooler(mintemp_raw_COOLER) > COOLER_MINTEMP) mintemp_raw_COOLER += TEMPDIR(COOLER) * (OVERSAMPLENR);
|
|
|
|
while (analog_to_celsius_cooler(maxtemp_raw_COOLER) < COOLER_MAXTEMP) maxtemp_raw_COOLER -= TEMPDIR(COOLER) * (OVERSAMPLENR);
|
2021-03-06 14:13:28 -06:00
|
|
|
#endif
|
2021-06-11 13:51:29 -07:00
|
|
|
|
2021-08-18 20:12:41 -05:00
|
|
|
#if BOTH(HAS_TEMP_BOARD, THERMAL_PROTECTION_BOARD)
|
|
|
|
while (analog_to_celsius_board(mintemp_raw_BOARD) < BOARD_MINTEMP) mintemp_raw_BOARD += TEMPDIR(BOARD) * (OVERSAMPLENR);
|
|
|
|
while (analog_to_celsius_board(maxtemp_raw_BOARD) > BOARD_MAXTEMP) maxtemp_raw_BOARD -= TEMPDIR(BOARD) * (OVERSAMPLENR);
|
|
|
|
#endif
|
|
|
|
|
2021-06-11 13:51:29 -07:00
|
|
|
#if HAS_TEMP_REDUNDANT
|
|
|
|
temp_redundant.target = &(
|
2021-08-18 20:12:41 -05:00
|
|
|
#if REDUNDANT_TEMP_MATCH(TARGET, COOLER) && HAS_TEMP_COOLER
|
2021-06-11 13:51:29 -07:00
|
|
|
temp_cooler
|
2021-08-18 20:12:41 -05:00
|
|
|
#elif REDUNDANT_TEMP_MATCH(TARGET, PROBE) && HAS_TEMP_PROBE
|
2021-06-11 13:51:29 -07:00
|
|
|
temp_probe
|
2021-08-18 20:12:41 -05:00
|
|
|
#elif REDUNDANT_TEMP_MATCH(TARGET, BOARD) && HAS_TEMP_BOARD
|
|
|
|
temp_board
|
|
|
|
#elif REDUNDANT_TEMP_MATCH(TARGET, CHAMBER) && HAS_TEMP_CHAMBER
|
2021-06-11 13:51:29 -07:00
|
|
|
temp_chamber
|
2021-08-18 20:12:41 -05:00
|
|
|
#elif REDUNDANT_TEMP_MATCH(TARGET, BED) && HAS_TEMP_BED
|
2021-06-11 13:51:29 -07:00
|
|
|
temp_bed
|
|
|
|
#else
|
2021-08-18 20:12:41 -05:00
|
|
|
temp_hotend[HEATER_ID(TEMP_SENSOR_REDUNDANT_TARGET)]
|
2021-06-11 13:51:29 -07:00
|
|
|
#endif
|
|
|
|
);
|
|
|
|
#endif
|
2011-11-15 22:50:43 +01:00
|
|
|
}
|
|
|
|
|
2019-04-22 03:50:32 +02:00
|
|
|
#if HAS_THERMAL_PROTECTION
|
2015-04-02 05:10:14 -07:00
|
|
|
|
2020-09-14 16:58:39 +12:00
|
|
|
Temperature::tr_state_machine_t Temperature::tr_state_machine[NR_HEATER_RUNAWAY]; // = { { TRInactive, 0 } };
|
2016-05-26 11:58:38 -07:00
|
|
|
|
2020-09-14 16:58:39 +12:00
|
|
|
/**
|
|
|
|
* @brief Thermal Runaway state machine for a single heater
|
|
|
|
* @param current current measured temperature
|
|
|
|
* @param target current target temperature
|
|
|
|
* @param heater_id extruder index
|
|
|
|
* @param period_seconds missed temperature allowed time
|
|
|
|
* @param hysteresis_degc allowed distance from target
|
|
|
|
*
|
|
|
|
* TODO: Embed the last 3 parameters during init, if not less optimal
|
|
|
|
*/
|
2021-04-23 19:14:49 -05:00
|
|
|
void Temperature::tr_state_machine_t::run(const_celsius_float_t current, const_celsius_float_t target, const heater_id_t heater_id, const uint16_t period_seconds, const celsius_t hysteresis_degc) {
|
2015-04-03 19:34:55 -07:00
|
|
|
|
2020-09-14 16:58:39 +12:00
|
|
|
#if HEATER_IDLE_HANDLER
|
|
|
|
// Convert the given heater_id_t to an idle array index
|
|
|
|
const IdleIndex idle_index = idle_index_for_id(heater_id);
|
|
|
|
#endif
|
2015-04-03 19:34:55 -07:00
|
|
|
|
2016-03-24 23:19:46 -07:00
|
|
|
/**
|
2019-04-22 03:50:32 +02:00
|
|
|
SERIAL_ECHO_START();
|
2020-03-14 18:47:44 -05:00
|
|
|
SERIAL_ECHOPGM("Thermal Runaway Running. Heater ID: ");
|
2020-09-14 16:58:39 +12:00
|
|
|
switch (heater_id) {
|
|
|
|
case H_BED: SERIAL_ECHOPGM("bed"); break;
|
|
|
|
case H_CHAMBER: SERIAL_ECHOPGM("chamber"); break;
|
2021-02-08 07:37:24 +01:00
|
|
|
default: SERIAL_ECHO(heater_id);
|
2020-09-14 16:58:39 +12:00
|
|
|
}
|
2021-09-09 04:57:05 -05:00
|
|
|
SERIAL_ECHOLNPGM(
|
2020-09-14 16:58:39 +12:00
|
|
|
" ; sizeof(running_temp):", sizeof(running_temp),
|
|
|
|
" ; State:", state, " ; Timer:", timer, " ; Temperature:", current, " ; Target Temp:", target
|
|
|
|
#if HEATER_IDLE_HANDLER
|
|
|
|
, " ; Idle Timeout:", heater_idle[idle_index].timed_out
|
|
|
|
#endif
|
|
|
|
);
|
2021-06-11 13:51:29 -07:00
|
|
|
*/
|
2015-04-04 05:45:36 -07:00
|
|
|
|
2017-06-12 00:22:31 -05:00
|
|
|
#if HEATER_IDLE_HANDLER
|
2017-05-26 13:01:02 -05:00
|
|
|
// If the heater idle timeout expires, restart
|
2020-09-14 16:58:39 +12:00
|
|
|
if (heater_idle[idle_index].timed_out) {
|
|
|
|
state = TRInactive;
|
|
|
|
running_temp = 0;
|
2017-05-26 13:01:02 -05:00
|
|
|
}
|
|
|
|
else
|
|
|
|
#endif
|
2018-01-01 18:24:54 -06:00
|
|
|
{
|
|
|
|
// If the target temperature changes, restart
|
2020-09-14 16:58:39 +12:00
|
|
|
if (running_temp != target) {
|
|
|
|
running_temp = target;
|
|
|
|
state = target > 0 ? TRFirstHeating : TRInactive;
|
2018-01-01 18:24:54 -06:00
|
|
|
}
|
2016-05-08 17:07:45 -07:00
|
|
|
}
|
2015-04-02 05:10:14 -07:00
|
|
|
|
2020-09-14 16:58:39 +12:00
|
|
|
switch (state) {
|
2015-04-02 06:11:03 -07:00
|
|
|
// Inactive state waits for a target temperature to be set
|
2016-05-08 17:07:45 -07:00
|
|
|
case TRInactive: break;
|
2019-01-12 01:41:48 -05:00
|
|
|
|
2015-04-02 05:10:14 -07:00
|
|
|
// When first heating, wait for the temperature to be reached then go to Stable state
|
|
|
|
case TRFirstHeating:
|
2020-09-14 16:58:39 +12:00
|
|
|
if (current < running_temp) break;
|
|
|
|
state = TRStable;
|
2019-01-12 01:41:48 -05:00
|
|
|
|
2015-04-02 05:10:14 -07:00
|
|
|
// While the temperature is stable watch for a bad temperature
|
|
|
|
case TRStable:
|
2019-01-12 01:41:48 -05:00
|
|
|
|
2019-02-04 07:12:41 +01:00
|
|
|
#if ENABLED(ADAPTIVE_FAN_SLOWING)
|
|
|
|
if (adaptive_fan_slowing && heater_id >= 0) {
|
2019-07-05 18:01:21 -05:00
|
|
|
const int fan_index = _MIN(heater_id, FAN_COUNT - 1);
|
2020-09-14 16:58:39 +12:00
|
|
|
if (fan_speed[fan_index] == 0 || current >= running_temp - (hysteresis_degc * 0.25f))
|
2019-01-12 01:41:48 -05:00
|
|
|
fan_speed_scaler[fan_index] = 128;
|
2020-09-14 16:58:39 +12:00
|
|
|
else if (current >= running_temp - (hysteresis_degc * 0.3335f))
|
2019-01-12 01:41:48 -05:00
|
|
|
fan_speed_scaler[fan_index] = 96;
|
2020-09-14 16:58:39 +12:00
|
|
|
else if (current >= running_temp - (hysteresis_degc * 0.5f))
|
2019-01-12 01:41:48 -05:00
|
|
|
fan_speed_scaler[fan_index] = 64;
|
2020-09-14 16:58:39 +12:00
|
|
|
else if (current >= running_temp - (hysteresis_degc * 0.8f))
|
2019-01-12 01:41:48 -05:00
|
|
|
fan_speed_scaler[fan_index] = 32;
|
|
|
|
else
|
|
|
|
fan_speed_scaler[fan_index] = 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-09-14 16:58:39 +12:00
|
|
|
if (current >= running_temp - hysteresis_degc) {
|
|
|
|
timer = millis() + SEC_TO_MS(period_seconds);
|
2016-05-08 16:51:33 -07:00
|
|
|
break;
|
|
|
|
}
|
2020-09-14 16:58:39 +12:00
|
|
|
else if (PENDING(millis(), timer)) break;
|
|
|
|
state = TRRunaway;
|
2019-01-12 01:41:48 -05:00
|
|
|
|
2015-04-04 05:45:36 -07:00
|
|
|
case TRRunaway:
|
2021-09-07 02:15:24 -05:00
|
|
|
TERN_(HAS_DWIN_E3V2_BASIC, DWIN_Popup_Temperature(0));
|
2021-09-28 20:15:52 -05:00
|
|
|
_temp_error(heater_id, FPSTR(str_t_thermal_runaway), GET_TEXT_F(MSG_THERMAL_RUNAWAY));
|
2015-04-02 05:10:14 -07:00
|
|
|
}
|
2014-06-30 15:22:37 -03:00
|
|
|
}
|
2011-11-15 22:50:43 +01:00
|
|
|
|
2019-07-14 08:56:29 -05:00
|
|
|
#endif // HAS_THERMAL_PROTECTION
|
2015-02-24 04:46:11 -08:00
|
|
|
|
2016-04-28 18:18:13 -07:00
|
|
|
void Temperature::disable_all_heaters() {
|
2017-05-03 20:43:00 -05:00
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
// Disable autotemp, unpause and reset everything
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(AUTOTEMP, planner.autotemp_enabled = false);
|
2021-05-22 21:08:57 -05:00
|
|
|
TERN_(PROBING_HEATERS_OFF, pause_heaters(false));
|
2017-05-07 07:06:06 -04:00
|
|
|
|
2020-07-14 23:14:03 -07:00
|
|
|
#if HAS_HOTEND
|
|
|
|
HOTEND_LOOP() {
|
|
|
|
setTargetHotend(0, e);
|
|
|
|
temp_hotend[e].soft_pwm_amount = 0;
|
|
|
|
}
|
|
|
|
#endif
|
2015-03-23 15:18:22 -07:00
|
|
|
|
2016-04-03 16:18:49 -07:00
|
|
|
#if HAS_TEMP_HOTEND
|
2020-07-15 01:20:52 -05:00
|
|
|
#define DISABLE_HEATER(N) WRITE_HEATER_##N(LOW);
|
2019-11-09 17:59:04 -06:00
|
|
|
REPEAT(HOTENDS, DISABLE_HEATER);
|
2015-02-24 04:46:11 -08:00
|
|
|
#endif
|
2015-01-23 23:13:06 +01:00
|
|
|
|
2018-04-23 17:13:01 -05:00
|
|
|
#if HAS_HEATED_BED
|
2020-07-14 23:14:03 -07:00
|
|
|
setTargetBed(0);
|
2019-03-07 00:09:39 -08:00
|
|
|
temp_bed.soft_pwm_amount = 0;
|
2019-06-29 01:54:22 +02:00
|
|
|
WRITE_HEATER_BED(LOW);
|
2015-02-24 04:46:11 -08:00
|
|
|
#endif
|
2019-03-07 00:09:39 -08:00
|
|
|
|
2019-03-10 15:51:13 +01:00
|
|
|
#if HAS_HEATED_CHAMBER
|
2020-07-14 23:14:03 -07:00
|
|
|
setTargetChamber(0);
|
2019-03-07 00:09:39 -08:00
|
|
|
temp_chamber.soft_pwm_amount = 0;
|
2019-06-29 01:54:22 +02:00
|
|
|
WRITE_HEATER_CHAMBER(LOW);
|
2019-03-07 00:09:39 -08:00
|
|
|
#endif
|
2021-03-06 14:13:28 -06:00
|
|
|
|
|
|
|
#if HAS_COOLER
|
|
|
|
setTargetCooler(0);
|
|
|
|
temp_cooler.soft_pwm_amount = 0;
|
|
|
|
WRITE_HEATER_COOLER(LOW);
|
|
|
|
#endif
|
2011-11-15 22:50:43 +01:00
|
|
|
}
|
|
|
|
|
2020-01-30 03:24:43 -06:00
|
|
|
#if ENABLED(PRINTJOB_TIMER_AUTOSTART)
|
|
|
|
|
2020-11-07 17:55:31 -06:00
|
|
|
bool Temperature::auto_job_over_threshold() {
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_HOTEND
|
2020-02-01 02:51:57 -08:00
|
|
|
HOTEND_LOOP() if (degTargetHotend(e) > (EXTRUDE_MINTEMP) / 2) return true;
|
2020-01-30 03:24:43 -06:00
|
|
|
#endif
|
2020-04-22 16:35:03 -05:00
|
|
|
return TERN0(HAS_HEATED_BED, degTargetBed() > BED_MINTEMP)
|
|
|
|
|| TERN0(HAS_HEATED_CHAMBER, degTargetChamber() > CHAMBER_MINTEMP);
|
2020-01-30 03:24:43 -06:00
|
|
|
}
|
|
|
|
|
2020-11-07 17:55:31 -06:00
|
|
|
void Temperature::auto_job_check_timer(const bool can_start, const bool can_stop) {
|
|
|
|
if (auto_job_over_threshold()) {
|
2020-01-30 03:24:43 -06:00
|
|
|
if (can_start) startOrResumeJob();
|
|
|
|
}
|
|
|
|
else if (can_stop) {
|
|
|
|
print_job_timer.stop();
|
|
|
|
ui.reset_status();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
#endif // PRINTJOB_TIMER_AUTOSTART
|
2020-01-30 03:24:43 -06:00
|
|
|
|
2017-05-07 07:06:06 -04:00
|
|
|
#if ENABLED(PROBING_HEATERS_OFF)
|
|
|
|
|
2021-05-22 21:08:57 -05:00
|
|
|
void Temperature::pause_heaters(const bool p) {
|
2021-05-02 21:32:21 -05:00
|
|
|
if (p != paused_for_probing) {
|
|
|
|
paused_for_probing = p;
|
2017-05-08 14:09:37 -05:00
|
|
|
if (p) {
|
2020-09-14 16:58:39 +12:00
|
|
|
HOTEND_LOOP() heater_idle[e].expire(); // Timeout immediately
|
|
|
|
TERN_(HAS_HEATED_BED, heater_idle[IDLE_INDEX_BED].expire()); // Timeout immediately
|
2017-05-08 14:09:37 -05:00
|
|
|
}
|
|
|
|
else {
|
2020-01-30 03:24:43 -06:00
|
|
|
HOTEND_LOOP() reset_hotend_idle_timer(e);
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(HAS_HEATED_BED, reset_bed_idle_timer());
|
2017-05-07 07:06:06 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-08 07:40:50 -04:00
|
|
|
#endif // PROBING_HEATERS_OFF
|
2017-05-07 07:06:06 -04:00
|
|
|
|
2021-05-01 20:21:18 +12:00
|
|
|
#if EITHER(SINGLENOZZLE_STANDBY_TEMP, SINGLENOZZLE_STANDBY_FAN)
|
2021-01-19 20:58:50 -06:00
|
|
|
|
|
|
|
void Temperature::singlenozzle_change(const uint8_t old_tool, const uint8_t new_tool) {
|
2021-05-01 20:21:18 +12:00
|
|
|
#if ENABLED(SINGLENOZZLE_STANDBY_FAN)
|
2021-01-19 20:58:50 -06:00
|
|
|
singlenozzle_fan_speed[old_tool] = fan_speed[0];
|
|
|
|
fan_speed[0] = singlenozzle_fan_speed[new_tool];
|
|
|
|
#endif
|
2021-05-01 20:21:18 +12:00
|
|
|
#if ENABLED(SINGLENOZZLE_STANDBY_TEMP)
|
|
|
|
singlenozzle_temp[old_tool] = temp_hotend[0].target;
|
|
|
|
if (singlenozzle_temp[new_tool] && singlenozzle_temp[new_tool] != singlenozzle_temp[old_tool]) {
|
|
|
|
setTargetHotend(singlenozzle_temp[new_tool], 0);
|
|
|
|
TERN_(AUTOTEMP, planner.autotemp_update());
|
2021-05-22 17:56:31 -05:00
|
|
|
set_heating_message(0);
|
2021-05-01 20:21:18 +12:00
|
|
|
(void)wait_for_hotend(0, false); // Wait for heating or cooling
|
|
|
|
}
|
|
|
|
#endif
|
2021-01-19 20:58:50 -06:00
|
|
|
}
|
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
#endif // SINGLENOZZLE_STANDBY_TEMP || SINGLENOZZLE_STANDBY_FAN
|
2021-01-19 20:58:50 -06:00
|
|
|
|
2021-02-07 16:58:06 -06:00
|
|
|
#if HAS_MAX_TC
|
2016-03-08 06:58:11 -08:00
|
|
|
|
2020-05-20 15:38:29 -05:00
|
|
|
#ifndef THERMOCOUPLE_MAX_ERRORS
|
|
|
|
#define THERMOCOUPLE_MAX_ERRORS 15
|
|
|
|
#endif
|
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
/**
|
|
|
|
* @brief Read MAX Thermocouple temperature.
|
|
|
|
*
|
|
|
|
* Reads the thermocouple board via HW or SW SPI, using a library (LIB_USR_x) or raw SPI reads.
|
|
|
|
* Doesn't strictly return a temperature; returns an "ADC Value" (i.e. raw register content).
|
|
|
|
*
|
|
|
|
* @param hindex the hotend we're referencing (if MULTI_MAX_TC)
|
|
|
|
* @return integer representing the board's buffer, to be converted later if needed
|
|
|
|
*/
|
|
|
|
int16_t Temperature::read_max_tc(TERN_(HAS_MULTI_MAX_TC, const uint8_t hindex/*=0*/)) {
|
|
|
|
#define MAXTC_HEAT_INTERVAL 250UL
|
2012-02-05 13:05:07 +01:00
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
#if HAS_MAX31855
|
|
|
|
#define MAX_TC_ERROR_MASK 7 // D2-0: SCV, SCG, OC
|
|
|
|
#define MAX_TC_DISCARD_BITS 18 // Data D31-18; sign bit D31
|
|
|
|
#define MAX_TC_SPEED_BITS 3 // ~1MHz
|
|
|
|
#elif HAS_MAX31865
|
|
|
|
#define MAX_TC_ERROR_MASK 1 // D0 Bit on fault only
|
2021-02-07 16:58:06 -06:00
|
|
|
#define MAX_TC_DISCARD_BITS 1 // Data is in D15-D1
|
2021-07-06 17:36:41 -07:00
|
|
|
#define MAX_TC_SPEED_BITS 3 // ~1MHz
|
|
|
|
#else // MAX6675
|
|
|
|
#define MAX_TC_ERROR_MASK 3 // D2 only; 1 = open circuit
|
|
|
|
#define MAX_TC_DISCARD_BITS 3 // Data D15-D1
|
|
|
|
#define MAX_TC_SPEED_BITS 2 // ~2MHz
|
2018-11-17 01:20:33 +01:00
|
|
|
#endif
|
2015-02-24 04:46:11 -08:00
|
|
|
|
2021-02-07 16:58:06 -06:00
|
|
|
#if HAS_MULTI_MAX_TC
|
2020-11-11 16:52:35 -05:00
|
|
|
// Needed to return the correct temp when this is called between readings
|
2021-07-06 17:36:41 -07:00
|
|
|
static int16_t max_tc_temp_previous[MAX_TC_COUNT] = { 0 };
|
2021-02-07 16:58:06 -06:00
|
|
|
#define THERMO_TEMP(I) max_tc_temp_previous[I]
|
|
|
|
#define THERMO_SEL(A,B) (hindex ? (B) : (A))
|
2021-07-06 17:36:41 -07:00
|
|
|
#define MAXTC_CS_WRITE(V) do{ switch (hindex) { case 1: WRITE(TEMP_1_CS_PIN, V); break; default: WRITE(TEMP_0_CS_PIN, V); } }while(0)
|
2020-11-11 16:52:35 -05:00
|
|
|
#else
|
2021-07-06 17:36:41 -07:00
|
|
|
// When we have only 1 max tc, THERMO_SEL will pick the appropriate sensor
|
|
|
|
// variable, and MAXTC_*() macros will be hardcoded to the correct CS pin.
|
2020-11-11 16:52:35 -05:00
|
|
|
constexpr uint8_t hindex = 0;
|
2021-02-07 16:58:06 -06:00
|
|
|
#define THERMO_TEMP(I) max_tc_temp
|
2021-07-06 17:36:41 -07:00
|
|
|
#if TEMP_SENSOR_IS_ANY_MAX_TC(0)
|
2021-02-07 16:58:06 -06:00
|
|
|
#define THERMO_SEL(A,B) A
|
2021-07-06 17:36:41 -07:00
|
|
|
#define MAXTC_CS_WRITE(V) WRITE(TEMP_0_CS_PIN, V)
|
2020-11-11 16:52:35 -05:00
|
|
|
#else
|
2021-07-06 17:36:41 -07:00
|
|
|
#define THERMO_SEL(A,B) B
|
|
|
|
#define MAXTC_CS_WRITE(V) WRITE(TEMP_1_CS_PIN, V)
|
2020-11-11 16:52:35 -05:00
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
static TERN(HAS_MAX31855, uint32_t, uint16_t) max_tc_temp = THERMO_SEL(
|
|
|
|
TEMP_SENSOR_0_MAX_TC_TMAX,
|
|
|
|
TEMP_SENSOR_1_MAX_TC_TMAX
|
|
|
|
);
|
|
|
|
|
2021-02-07 16:58:06 -06:00
|
|
|
static uint8_t max_tc_errors[MAX_TC_COUNT] = { 0 };
|
2021-07-06 17:36:41 -07:00
|
|
|
static millis_t next_max_tc_ms[MAX_TC_COUNT] = { 0 };
|
2020-11-11 16:52:35 -05:00
|
|
|
|
2018-11-17 01:20:33 +01:00
|
|
|
// Return last-read value between readings
|
2015-04-12 18:07:08 -07:00
|
|
|
millis_t ms = millis();
|
2021-07-06 17:36:41 -07:00
|
|
|
if (PENDING(ms, next_max_tc_ms[hindex]))
|
|
|
|
return (int16_t)THERMO_TEMP(hindex);
|
2018-11-17 01:20:33 +01:00
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
next_max_tc_ms[hindex] = ms + MAXTC_HEAT_INTERVAL;
|
2015-04-12 23:06:50 -07:00
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
#if !HAS_MAXTC_LIBRARIES
|
|
|
|
max_tc_temp = 0;
|
|
|
|
|
|
|
|
#if !HAS_MAXTC_SW_SPI
|
|
|
|
// Initialize SPI using the default Hardware SPI bus.
|
|
|
|
// FIXME: spiBegin, spiRec and spiInit doesn't work when soft spi is used.
|
|
|
|
spiBegin();
|
|
|
|
spiInit(MAX_TC_SPEED_BITS);
|
|
|
|
#endif
|
2012-02-05 13:05:07 +01:00
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
MAXTC_CS_WRITE(LOW); // enable MAXTC
|
|
|
|
DELAY_NS(100); // Ensure 100ns delay
|
2015-02-24 04:46:11 -08:00
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
// Read a big-endian temperature value without using a library
|
2021-02-08 20:51:57 -05:00
|
|
|
for (uint8_t i = sizeof(max_tc_temp); i--;) {
|
2021-07-06 17:36:41 -07:00
|
|
|
max_tc_temp |= TERN(HAS_MAXTC_SW_SPI, max_tc_spi.receive(), spiRec());
|
2021-02-08 20:51:57 -05:00
|
|
|
if (i > 0) max_tc_temp <<= 8; // shift left if not the last byte
|
|
|
|
}
|
2015-02-24 04:46:11 -08:00
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
MAXTC_CS_WRITE(HIGH); // disable MAXTC
|
|
|
|
#else
|
|
|
|
#if HAS_MAX6675_LIBRARY
|
|
|
|
MAX6675 &max6675ref = THERMO_SEL(max6675_0, max6675_1);
|
|
|
|
max_tc_temp = max6675ref.readRaw16();
|
2021-02-08 20:51:57 -05:00
|
|
|
#endif
|
2020-11-11 16:52:35 -05:00
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
#if HAS_MAX31855_LIBRARY
|
|
|
|
MAX31855 &max855ref = THERMO_SEL(max31855_0, max31855_1);
|
|
|
|
max_tc_temp = max855ref.readRaw32();
|
|
|
|
#endif
|
2021-02-08 20:51:57 -05:00
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
#if HAS_MAX31865
|
|
|
|
MAX31865 &max865ref = THERMO_SEL(max31865_0, max31865_1);
|
|
|
|
max_tc_temp = TERN(LIB_INTERNAL_MAX31865, max865ref.readRaw(), max865ref.readRTD_with_Fault());
|
|
|
|
#endif
|
2021-02-08 20:51:57 -05:00
|
|
|
#endif
|
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
// Handle an error. If there have been more than THERMOCOUPLE_MAX_ERRORS, send an error over serial.
|
|
|
|
// Either way, return the TMAX for the thermocouple to trigger a max_temp_error()
|
|
|
|
if (max_tc_temp & MAX_TC_ERROR_MASK) {
|
2021-02-07 16:58:06 -06:00
|
|
|
max_tc_errors[hindex]++;
|
2021-07-06 17:36:41 -07:00
|
|
|
|
2021-02-07 16:58:06 -06:00
|
|
|
if (max_tc_errors[hindex] > THERMOCOUPLE_MAX_ERRORS) {
|
2020-05-20 15:38:29 -05:00
|
|
|
SERIAL_ERROR_START();
|
|
|
|
SERIAL_ECHOPGM("Temp measurement error! ");
|
2021-07-06 17:36:41 -07:00
|
|
|
#if HAS_MAX31855
|
2021-09-09 04:57:05 -05:00
|
|
|
SERIAL_ECHOPGM("MAX31855 Fault: (", max_tc_temp & 0x7, ") >> ");
|
2021-02-08 20:51:57 -05:00
|
|
|
if (max_tc_temp & 0x1)
|
2020-05-20 15:38:29 -05:00
|
|
|
SERIAL_ECHOLNPGM("Open Circuit");
|
2021-02-08 20:51:57 -05:00
|
|
|
else if (max_tc_temp & 0x2)
|
2020-05-20 15:38:29 -05:00
|
|
|
SERIAL_ECHOLNPGM("Short to GND");
|
2021-02-08 20:51:57 -05:00
|
|
|
else if (max_tc_temp & 0x4)
|
2020-05-20 15:38:29 -05:00
|
|
|
SERIAL_ECHOLNPGM("Short to VCC");
|
2021-02-08 20:51:57 -05:00
|
|
|
#elif HAS_MAX31865
|
2021-07-06 17:36:41 -07:00
|
|
|
const uint8_t fault_31865 = max865ref.readFault();
|
2021-02-08 20:51:57 -05:00
|
|
|
max865ref.clearFault();
|
2020-11-11 16:52:35 -05:00
|
|
|
if (fault_31865) {
|
2021-02-08 20:51:57 -05:00
|
|
|
SERIAL_EOL();
|
2021-09-09 04:57:05 -05:00
|
|
|
SERIAL_ECHOLNPGM("\nMAX31865 Fault: (", fault_31865, ") >>");
|
2020-11-11 16:52:35 -05:00
|
|
|
if (fault_31865 & MAX31865_FAULT_HIGHTHRESH)
|
|
|
|
SERIAL_ECHOLNPGM("RTD High Threshold");
|
2021-02-08 20:51:57 -05:00
|
|
|
if (fault_31865 & MAX31865_FAULT_LOWTHRESH)
|
2020-11-11 16:52:35 -05:00
|
|
|
SERIAL_ECHOLNPGM("RTD Low Threshold");
|
2021-02-08 20:51:57 -05:00
|
|
|
if (fault_31865 & MAX31865_FAULT_REFINLOW)
|
2021-07-06 17:36:41 -07:00
|
|
|
SERIAL_ECHOLNPGM("REFIN- > 0.85 x V bias");
|
2021-02-08 20:51:57 -05:00
|
|
|
if (fault_31865 & MAX31865_FAULT_REFINHIGH)
|
2021-07-06 17:36:41 -07:00
|
|
|
SERIAL_ECHOLNPGM("REFIN- < 0.85 x V bias (FORCE- open)");
|
2021-02-08 20:51:57 -05:00
|
|
|
if (fault_31865 & MAX31865_FAULT_RTDINLOW)
|
2021-07-06 17:36:41 -07:00
|
|
|
SERIAL_ECHOLNPGM("REFIN- < 0.85 x V bias (FORCE- open)");
|
2021-02-08 20:51:57 -05:00
|
|
|
if (fault_31865 & MAX31865_FAULT_OVUV)
|
2020-11-11 16:52:35 -05:00
|
|
|
SERIAL_ECHOLNPGM("Under/Over voltage");
|
|
|
|
}
|
2021-07-06 17:36:41 -07:00
|
|
|
#else // MAX6675
|
|
|
|
SERIAL_ECHOLNPGM("MAX6675 Fault: Open Circuit");
|
2018-11-17 01:20:33 +01:00
|
|
|
#endif
|
2020-05-20 15:38:29 -05:00
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
// Set thermocouple above max temperature (TMAX)
|
|
|
|
max_tc_temp = THERMO_SEL(TEMP_SENSOR_0_MAX_TC_TMAX, TEMP_SENSOR_1_MAX_TC_TMAX) << (MAX_TC_DISCARD_BITS + 1);
|
2020-05-20 15:38:29 -05:00
|
|
|
}
|
2016-10-09 17:12:50 -05:00
|
|
|
}
|
2020-05-20 15:38:29 -05:00
|
|
|
else {
|
2021-07-06 17:36:41 -07:00
|
|
|
max_tc_errors[hindex] = 0; // No error bit, reset error count
|
2020-05-20 15:38:29 -05:00
|
|
|
}
|
2019-01-14 14:29:55 -06:00
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
max_tc_temp >>= MAX_TC_DISCARD_BITS;
|
2019-01-14 14:29:55 -06:00
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
#if HAS_MAX31855
|
|
|
|
// Support negative temperature for MAX38155
|
|
|
|
if (max_tc_temp & 0x00002000) max_tc_temp |= 0xFFFFC000;
|
2021-02-08 20:51:57 -05:00
|
|
|
#endif
|
2020-11-18 21:12:03 -08:00
|
|
|
|
2021-02-07 16:58:06 -06:00
|
|
|
THERMO_TEMP(hindex) = max_tc_temp;
|
2015-02-24 04:46:11 -08:00
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
return (int16_t)max_tc_temp;
|
2015-02-24 04:46:11 -08:00
|
|
|
}
|
|
|
|
|
2021-02-07 16:58:06 -06:00
|
|
|
#endif // HAS_MAX_TC
|
2015-02-24 04:46:11 -08:00
|
|
|
|
2016-04-28 18:18:13 -07:00
|
|
|
/**
|
2020-01-26 17:39:45 -06:00
|
|
|
* Update raw temperatures
|
2021-05-08 17:34:21 -05:00
|
|
|
*
|
|
|
|
* Called by ISR => readings_ready when new temperatures have been set by updateTemperaturesFromRawValues.
|
|
|
|
* Applies all the accumulators to the current raw temperatures.
|
2016-04-28 18:18:13 -07:00
|
|
|
*/
|
2020-01-26 17:39:45 -06:00
|
|
|
void Temperature::update_raw_temperatures() {
|
2018-11-17 01:20:33 +01:00
|
|
|
|
2021-08-18 20:12:41 -05:00
|
|
|
// TODO: can this be collapsed into a HOTEND_LOOP()?
|
2021-02-07 16:58:06 -06:00
|
|
|
#if HAS_TEMP_ADC_0 && !TEMP_SENSOR_0_IS_MAX_TC
|
2019-09-05 19:44:55 -05:00
|
|
|
temp_hotend[0].update();
|
2015-03-25 11:08:24 +01:00
|
|
|
#endif
|
2018-11-17 01:20:33 +01:00
|
|
|
|
2021-06-11 13:51:29 -07:00
|
|
|
#if HAS_TEMP_ADC_1 && !TEMP_SENSOR_1_IS_MAX_TC
|
|
|
|
temp_hotend[1].update();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if HAS_TEMP_ADC_REDUNDANT && !TEMP_SENSOR_REDUNDANT_IS_MAX_TC
|
|
|
|
temp_redundant.update();
|
2020-02-01 15:18:01 -06:00
|
|
|
#endif
|
|
|
|
|
2021-07-06 17:36:41 -07:00
|
|
|
TERN_(HAS_TEMP_ADC_2, temp_hotend[2].update());
|
|
|
|
TERN_(HAS_TEMP_ADC_3, temp_hotend[3].update());
|
|
|
|
TERN_(HAS_TEMP_ADC_4, temp_hotend[4].update());
|
|
|
|
TERN_(HAS_TEMP_ADC_5, temp_hotend[5].update());
|
|
|
|
TERN_(HAS_TEMP_ADC_6, temp_hotend[6].update());
|
|
|
|
TERN_(HAS_TEMP_ADC_7, temp_hotend[7].update());
|
|
|
|
TERN_(HAS_TEMP_ADC_BED, temp_bed.update());
|
2020-11-22 16:47:52 -08:00
|
|
|
TERN_(HAS_TEMP_ADC_CHAMBER, temp_chamber.update());
|
2021-07-06 17:36:41 -07:00
|
|
|
TERN_(HAS_TEMP_ADC_PROBE, temp_probe.update());
|
2021-08-18 20:12:41 -05:00
|
|
|
TERN_(HAS_TEMP_ADC_BOARD, temp_board.update());
|
2021-07-06 17:36:41 -07:00
|
|
|
TERN_(HAS_TEMP_ADC_COOLER, temp_cooler.update());
|
2019-03-07 00:09:39 -08:00
|
|
|
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(HAS_JOY_ADC_X, joystick.x.update());
|
|
|
|
TERN_(HAS_JOY_ADC_Y, joystick.y.update());
|
|
|
|
TERN_(HAS_JOY_ADC_Z, joystick.z.update());
|
2015-03-25 11:08:24 +01:00
|
|
|
}
|
|
|
|
|
2021-05-08 17:34:21 -05:00
|
|
|
/**
|
|
|
|
* Called by the Temperature ISR when all the ADCs have been processed.
|
|
|
|
* Reset all the ADC accumulators for another round of updates.
|
|
|
|
*/
|
2018-07-26 09:59:19 +01:00
|
|
|
void Temperature::readings_ready() {
|
2019-05-28 21:36:03 +01:00
|
|
|
|
2021-05-08 17:34:21 -05:00
|
|
|
// Update raw values only if they're not already set.
|
|
|
|
if (!raw_temps_ready) {
|
|
|
|
update_raw_temperatures();
|
|
|
|
raw_temps_ready = true;
|
|
|
|
}
|
2018-07-26 09:59:19 +01:00
|
|
|
|
|
|
|
// Filament Sensor - can be read any time since IIR filtering is used
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(FILAMENT_WIDTH_SENSOR, filwidth.reading_ready());
|
2018-07-26 09:59:19 +01:00
|
|
|
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_HOTEND
|
2019-09-10 02:20:49 -05:00
|
|
|
HOTEND_LOOP() temp_hotend[e].reset();
|
2019-08-06 21:25:47 -07:00
|
|
|
#endif
|
2018-07-26 09:59:19 +01:00
|
|
|
|
2021-08-18 20:12:41 -05:00
|
|
|
TERN_(HAS_HEATED_BED, temp_bed.reset());
|
|
|
|
TERN_(HAS_TEMP_CHAMBER, temp_chamber.reset());
|
|
|
|
TERN_(HAS_TEMP_PROBE, temp_probe.reset());
|
|
|
|
TERN_(HAS_TEMP_COOLER, temp_cooler.reset());
|
|
|
|
TERN_(HAS_TEMP_BOARD, temp_board.reset());
|
2021-06-11 13:51:29 -07:00
|
|
|
TERN_(HAS_TEMP_REDUNDANT, temp_redundant.reset());
|
2018-07-26 09:59:19 +01:00
|
|
|
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(HAS_JOY_ADC_X, joystick.x.reset());
|
|
|
|
TERN_(HAS_JOY_ADC_Y, joystick.y.reset());
|
|
|
|
TERN_(HAS_JOY_ADC_Z, joystick.z.reset());
|
2018-07-26 09:59:19 +01:00
|
|
|
}
|
|
|
|
|
2015-04-13 17:17:36 -07:00
|
|
|
/**
|
2016-08-27 20:06:26 -05:00
|
|
|
* Timer 0 is shared with millies so don't change the prescaler.
|
|
|
|
*
|
2017-12-08 22:13:03 -06:00
|
|
|
* On AVR this ISR uses the compare method so it runs at the base
|
2016-10-09 10:06:31 -05:00
|
|
|
* frequency (16 MHz / 64 / 256 = 976.5625 Hz), but at the TCNT0 set
|
2016-08-27 20:06:26 -05:00
|
|
|
* in OCR0B above (128 or halfway between OVFs).
|
|
|
|
*
|
2015-04-13 17:17:36 -07:00
|
|
|
* - Manage PWM to all the heaters and fan
|
2017-03-06 01:43:08 -06:00
|
|
|
* - Prepare or Measure one of the raw ADC sensor values
|
|
|
|
* - Check new temperature values for MIN/MAX errors (kill on error)
|
2015-04-13 17:17:36 -07:00
|
|
|
* - Step the babysteps value for each axis towards 0
|
2017-03-06 01:43:08 -06:00
|
|
|
* - For PINS_DEBUGGING, monitor and report endstop pins
|
|
|
|
* - For ENDSTOP_INTERRUPTS_FEATURE check endstops if flagged
|
2021-04-08 15:43:16 -05:00
|
|
|
* - Call planner.isr to count down its "ignore" time
|
2015-04-13 17:17:36 -07:00
|
|
|
*/
|
2019-03-10 20:43:59 -05:00
|
|
|
HAL_TEMP_TIMER_ISR() {
|
2017-08-24 12:18:54 -05:00
|
|
|
HAL_timer_isr_prologue(TEMP_TIMER_NUM);
|
2018-04-24 00:05:07 -03:00
|
|
|
|
2021-04-08 15:43:16 -05:00
|
|
|
Temperature::isr();
|
2016-04-28 18:18:13 -07:00
|
|
|
|
2018-04-24 00:05:07 -03:00
|
|
|
HAL_timer_isr_epilogue(TEMP_TIMER_NUM);
|
|
|
|
}
|
2017-02-12 18:00:14 +01:00
|
|
|
|
2019-03-07 00:09:39 -08:00
|
|
|
#if ENABLED(SLOW_PWM_HEATERS) && !defined(MIN_STATE_TIME)
|
|
|
|
#define MIN_STATE_TIME 16 // MIN_STATE_TIME * 65.5 = time in milliseconds
|
|
|
|
#endif
|
|
|
|
|
|
|
|
class SoftPWM {
|
|
|
|
public:
|
|
|
|
uint8_t count;
|
|
|
|
inline bool add(const uint8_t mask, const uint8_t amount) {
|
|
|
|
count = (count & mask) + amount; return (count > mask);
|
|
|
|
}
|
|
|
|
#if ENABLED(SLOW_PWM_HEATERS)
|
|
|
|
bool state_heater;
|
|
|
|
uint8_t state_timer_heater;
|
|
|
|
inline void dec() { if (state_timer_heater > 0) state_timer_heater--; }
|
|
|
|
inline bool ready(const bool v) {
|
|
|
|
const bool rdy = !state_timer_heater;
|
|
|
|
if (rdy && state_heater != v) {
|
|
|
|
state_heater = v;
|
|
|
|
state_timer_heater = MIN_STATE_TIME;
|
|
|
|
}
|
|
|
|
return rdy;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2019-11-13 02:23:02 +01:00
|
|
|
/**
|
|
|
|
* Handle various ~1KHz tasks associated with temperature
|
|
|
|
* - Heater PWM (~1KHz with scaler)
|
|
|
|
* - LCD Button polling (~500Hz)
|
|
|
|
* - Start / Read one ADC sensor
|
|
|
|
* - Advance Babysteps
|
|
|
|
* - Endstop polling
|
|
|
|
* - Planner clean buffer
|
|
|
|
*/
|
2021-04-08 15:43:16 -05:00
|
|
|
void Temperature::isr() {
|
2015-04-13 17:17:36 -07:00
|
|
|
|
2017-04-06 18:42:57 -05:00
|
|
|
static int8_t temp_count = -1;
|
|
|
|
static ADCSensorState adc_sensor_state = StartupDelay;
|
2016-09-24 00:51:21 -05:00
|
|
|
static uint8_t pwm_count = _BV(SOFT_PWM_SCALE);
|
2019-11-13 02:23:02 +01:00
|
|
|
|
2017-03-14 20:24:17 +01:00
|
|
|
// avoid multiple loads of pwm_count
|
|
|
|
uint8_t pwm_count_tmp = pwm_count;
|
2018-11-03 04:29:32 +01:00
|
|
|
|
2018-11-19 03:39:49 +01:00
|
|
|
#if HAS_ADC_BUTTONS
|
2017-06-10 00:12:18 -05:00
|
|
|
static unsigned int raw_ADCKey_value = 0;
|
2019-05-19 06:52:55 +05:00
|
|
|
static bool ADCKey_pressed = false;
|
2017-06-10 00:12:18 -05:00
|
|
|
#endif
|
2015-01-23 23:13:06 +01:00
|
|
|
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_HOTEND
|
2019-09-10 02:20:49 -05:00
|
|
|
static SoftPWM soft_pwm_hotend[HOTENDS];
|
2019-03-07 00:09:39 -08:00
|
|
|
#endif
|
|
|
|
|
2018-04-23 17:13:01 -05:00
|
|
|
#if HAS_HEATED_BED
|
2019-03-07 00:09:39 -08:00
|
|
|
static SoftPWM soft_pwm_bed;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if HAS_HEATED_CHAMBER
|
|
|
|
static SoftPWM soft_pwm_chamber;
|
2015-02-24 04:46:11 -08:00
|
|
|
#endif
|
2015-01-23 23:13:06 +01:00
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_COOLER
|
|
|
|
static SoftPWM soft_pwm_cooler;
|
|
|
|
#endif
|
|
|
|
|
2021-11-14 05:55:31 -06:00
|
|
|
#if BOTH(FAN_SOFT_PWM, USE_CONTROLLER_FAN)
|
|
|
|
static SoftPWM soft_pwm_controller;
|
|
|
|
#endif
|
|
|
|
|
2021-01-06 21:32:30 -06:00
|
|
|
#define WRITE_FAN(n, v) WRITE(FAN##n##_PIN, (v) ^ FAN_INVERTING)
|
|
|
|
|
2015-07-30 22:29:30 -07:00
|
|
|
#if DISABLED(SLOW_PWM_HEATERS)
|
2019-09-10 02:20:49 -05:00
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
#if ANY(HAS_HOTEND, HAS_HEATED_BED, HAS_HEATED_CHAMBER, HAS_COOLER, FAN_SOFT_PWM)
|
2020-09-17 06:35:04 -04:00
|
|
|
constexpr uint8_t pwm_mask = TERN0(SOFT_PWM_DITHER, _BV(SOFT_PWM_SCALE) - 1);
|
2019-09-10 02:20:49 -05:00
|
|
|
#define _PWM_MOD(N,S,T) do{ \
|
|
|
|
const bool on = S.add(pwm_mask, T.soft_pwm_amount); \
|
|
|
|
WRITE_HEATER_##N(on); \
|
|
|
|
}while(0)
|
|
|
|
#endif
|
2017-03-12 06:00:26 +01:00
|
|
|
|
2015-02-24 04:46:11 -08:00
|
|
|
/**
|
2018-11-28 19:28:31 -06:00
|
|
|
* Standard heater PWM modulation
|
2015-02-24 04:46:11 -08:00
|
|
|
*/
|
2017-03-14 20:24:17 +01:00
|
|
|
if (pwm_count_tmp >= 127) {
|
|
|
|
pwm_count_tmp -= 127;
|
2019-09-10 02:20:49 -05:00
|
|
|
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_HOTEND
|
2019-11-09 17:59:04 -06:00
|
|
|
#define _PWM_MOD_E(N) _PWM_MOD(N,soft_pwm_hotend[N],temp_hotend[N]);
|
|
|
|
REPEAT(HOTENDS, _PWM_MOD_E);
|
|
|
|
#endif
|
2015-01-23 23:13:06 +01:00
|
|
|
|
2018-04-23 17:13:01 -05:00
|
|
|
#if HAS_HEATED_BED
|
2021-08-21 18:00:55 -05:00
|
|
|
_PWM_MOD(BED, soft_pwm_bed, temp_bed);
|
2019-03-07 00:09:39 -08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if HAS_HEATED_CHAMBER
|
2021-08-21 18:00:55 -05:00
|
|
|
_PWM_MOD(CHAMBER, soft_pwm_chamber, temp_chamber);
|
2015-02-24 04:46:11 -08:00
|
|
|
#endif
|
2016-03-05 18:27:45 -08:00
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_COOLER
|
2021-08-21 18:00:55 -05:00
|
|
|
_PWM_MOD(COOLER, soft_pwm_cooler, temp_cooler);
|
2021-03-06 14:13:28 -06:00
|
|
|
#endif
|
|
|
|
|
2021-11-14 05:55:31 -06:00
|
|
|
#if BOTH(USE_CONTROLLER_FAN, FAN_SOFT_PWM)
|
|
|
|
WRITE(CONTROLLER_FAN_PIN, soft_pwm_controller.add(pwm_mask, soft_pwm_controller_speed));
|
|
|
|
#endif
|
|
|
|
|
2015-07-30 22:29:30 -07:00
|
|
|
#if ENABLED(FAN_SOFT_PWM)
|
2020-02-01 23:07:12 -06:00
|
|
|
#define _FAN_PWM(N) do{ \
|
|
|
|
uint8_t &spcf = soft_pwm_count_fan[N]; \
|
|
|
|
spcf = (spcf & pwm_mask) + (soft_pwm_amount_fan[N] >> 1); \
|
|
|
|
WRITE_FAN(N, spcf > pwm_mask ? HIGH : LOW); \
|
2019-03-07 00:09:39 -08:00
|
|
|
}while(0)
|
2016-03-05 18:27:45 -08:00
|
|
|
#if HAS_FAN0
|
2019-03-07 00:09:39 -08:00
|
|
|
_FAN_PWM(0);
|
2016-03-05 18:27:45 -08:00
|
|
|
#endif
|
|
|
|
#if HAS_FAN1
|
2019-03-07 00:09:39 -08:00
|
|
|
_FAN_PWM(1);
|
2016-03-05 18:27:45 -08:00
|
|
|
#endif
|
|
|
|
#if HAS_FAN2
|
2019-03-07 00:09:39 -08:00
|
|
|
_FAN_PWM(2);
|
2016-03-05 18:27:45 -08:00
|
|
|
#endif
|
2020-01-25 16:13:39 +08:00
|
|
|
#if HAS_FAN3
|
|
|
|
_FAN_PWM(3);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN4
|
|
|
|
_FAN_PWM(4);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN5
|
|
|
|
_FAN_PWM(5);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN6
|
|
|
|
_FAN_PWM(6);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN7
|
|
|
|
_FAN_PWM(7);
|
|
|
|
#endif
|
2015-02-24 04:46:11 -08:00
|
|
|
#endif
|
2015-01-23 23:13:06 +01:00
|
|
|
}
|
2017-03-18 20:28:57 +01:00
|
|
|
else {
|
2019-03-07 00:09:39 -08:00
|
|
|
#define _PWM_LOW(N,S) do{ if (S.count <= pwm_count_tmp) WRITE_HEATER_##N(LOW); }while(0)
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_HOTEND
|
2019-11-09 17:59:04 -06:00
|
|
|
#define _PWM_LOW_E(N) _PWM_LOW(N, soft_pwm_hotend[N]);
|
|
|
|
REPEAT(HOTENDS, _PWM_LOW_E);
|
|
|
|
#endif
|
2017-03-18 20:28:57 +01:00
|
|
|
|
2018-04-23 17:13:01 -05:00
|
|
|
#if HAS_HEATED_BED
|
2019-03-07 00:09:39 -08:00
|
|
|
_PWM_LOW(BED, soft_pwm_bed);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if HAS_HEATED_CHAMBER
|
|
|
|
_PWM_LOW(CHAMBER, soft_pwm_chamber);
|
2016-03-05 18:27:45 -08:00
|
|
|
#endif
|
2017-03-18 20:28:57 +01:00
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_COOLER
|
|
|
|
_PWM_LOW(COOLER, soft_pwm_cooler);
|
|
|
|
#endif
|
|
|
|
|
2017-03-18 20:28:57 +01:00
|
|
|
#if ENABLED(FAN_SOFT_PWM)
|
|
|
|
#if HAS_FAN0
|
2019-06-28 17:03:43 -05:00
|
|
|
if (soft_pwm_count_fan[0] <= pwm_count_tmp) WRITE_FAN(0, LOW);
|
2017-03-18 20:28:57 +01:00
|
|
|
#endif
|
|
|
|
#if HAS_FAN1
|
2019-06-28 17:03:43 -05:00
|
|
|
if (soft_pwm_count_fan[1] <= pwm_count_tmp) WRITE_FAN(1, LOW);
|
2017-03-18 20:28:57 +01:00
|
|
|
#endif
|
|
|
|
#if HAS_FAN2
|
2019-06-28 17:03:43 -05:00
|
|
|
if (soft_pwm_count_fan[2] <= pwm_count_tmp) WRITE_FAN(2, LOW);
|
2017-03-18 20:28:57 +01:00
|
|
|
#endif
|
2020-01-25 16:13:39 +08:00
|
|
|
#if HAS_FAN3
|
|
|
|
if (soft_pwm_count_fan[3] <= pwm_count_tmp) WRITE_FAN(3, LOW);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN4
|
|
|
|
if (soft_pwm_count_fan[4] <= pwm_count_tmp) WRITE_FAN(4, LOW);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN5
|
|
|
|
if (soft_pwm_count_fan[5] <= pwm_count_tmp) WRITE_FAN(5, LOW);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN6
|
|
|
|
if (soft_pwm_count_fan[6] <= pwm_count_tmp) WRITE_FAN(6, LOW);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN7
|
|
|
|
if (soft_pwm_count_fan[7] <= pwm_count_tmp) WRITE_FAN(7, LOW);
|
|
|
|
#endif
|
2021-11-14 05:55:31 -06:00
|
|
|
#if ENABLED(USE_CONTROLLER_FAN)
|
|
|
|
if (soft_pwm_controller.count <= pwm_count_tmp) WRITE(CONTROLLER_FAN_PIN, LOW);
|
|
|
|
#endif
|
2016-03-05 18:27:45 -08:00
|
|
|
#endif
|
2017-03-18 20:28:57 +01:00
|
|
|
}
|
2015-10-02 23:08:58 -07:00
|
|
|
|
2016-10-09 10:06:31 -05:00
|
|
|
// SOFT_PWM_SCALE to frequency:
|
|
|
|
//
|
|
|
|
// 0: 16000000/64/256/128 = 7.6294 Hz
|
|
|
|
// 1: / 64 = 15.2588 Hz
|
|
|
|
// 2: / 32 = 30.5176 Hz
|
|
|
|
// 3: / 16 = 61.0352 Hz
|
|
|
|
// 4: / 8 = 122.0703 Hz
|
|
|
|
// 5: / 4 = 244.1406 Hz
|
2017-03-14 20:24:17 +01:00
|
|
|
pwm_count = pwm_count_tmp + _BV(SOFT_PWM_SCALE);
|
2015-10-02 23:08:58 -07:00
|
|
|
|
2015-02-24 04:46:11 -08:00
|
|
|
#else // SLOW_PWM_HEATERS
|
2015-10-02 23:08:58 -07:00
|
|
|
|
2016-03-24 23:19:46 -07:00
|
|
|
/**
|
2015-02-24 04:46:11 -08:00
|
|
|
* SLOW PWM HEATERS
|
|
|
|
*
|
2016-08-27 20:06:26 -05:00
|
|
|
* For relay-driven heaters
|
2015-02-24 04:46:11 -08:00
|
|
|
*/
|
2019-03-07 00:09:39 -08:00
|
|
|
#define _SLOW_SET(NR,PWM,V) do{ if (PWM.ready(V)) WRITE_HEATER_##NR(V); }while(0)
|
|
|
|
#define _SLOW_PWM(NR,PWM,SRC) do{ PWM.count = SRC.soft_pwm_amount; _SLOW_SET(NR,PWM,(PWM.count > 0)); }while(0)
|
|
|
|
#define _PWM_OFF(NR,PWM) do{ if (PWM.count < slow_pwm_count) _SLOW_SET(NR,PWM,0); }while(0)
|
2015-01-23 23:13:06 +01:00
|
|
|
|
2019-09-10 02:20:49 -05:00
|
|
|
static uint8_t slow_pwm_count = 0;
|
|
|
|
|
2015-02-24 04:46:11 -08:00
|
|
|
if (slow_pwm_count == 0) {
|
|
|
|
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_HOTEND
|
2019-11-09 17:59:04 -06:00
|
|
|
#define _SLOW_PWM_E(N) _SLOW_PWM(N, soft_pwm_hotend[N], temp_hotend[N]);
|
|
|
|
REPEAT(HOTENDS, _SLOW_PWM_E);
|
|
|
|
#endif
|
2019-03-07 00:09:39 -08:00
|
|
|
|
|
|
|
#if HAS_HEATED_BED
|
|
|
|
_SLOW_PWM(BED, soft_pwm_bed, temp_bed);
|
|
|
|
#endif
|
|
|
|
|
2020-07-01 16:27:28 -05:00
|
|
|
#if HAS_HEATED_CHAMBER
|
|
|
|
_SLOW_PWM(CHAMBER, soft_pwm_chamber, temp_chamber);
|
|
|
|
#endif
|
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_COOLER
|
|
|
|
_SLOW_PWM(COOLER, soft_pwm_cooler, temp_cooler);
|
|
|
|
#endif
|
|
|
|
|
2019-03-07 00:09:39 -08:00
|
|
|
} // slow_pwm_count == 0
|
|
|
|
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_HOTEND
|
2019-03-07 00:09:39 -08:00
|
|
|
#define _PWM_OFF_E(N) _PWM_OFF(N, soft_pwm_hotend[N]);
|
2019-11-09 17:59:04 -06:00
|
|
|
REPEAT(HOTENDS, _PWM_OFF_E);
|
|
|
|
#endif
|
2015-02-24 04:46:11 -08:00
|
|
|
|
2018-04-23 17:13:01 -05:00
|
|
|
#if HAS_HEATED_BED
|
2019-03-07 00:09:39 -08:00
|
|
|
_PWM_OFF(BED, soft_pwm_bed);
|
2015-02-24 04:46:11 -08:00
|
|
|
#endif
|
|
|
|
|
2020-07-01 16:27:28 -05:00
|
|
|
#if HAS_HEATED_CHAMBER
|
|
|
|
_PWM_OFF(CHAMBER, soft_pwm_chamber);
|
|
|
|
#endif
|
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_COOLER
|
|
|
|
_PWM_OFF(COOLER, soft_pwm_cooler, temp_cooler);
|
|
|
|
#endif
|
|
|
|
|
2015-07-30 22:29:30 -07:00
|
|
|
#if ENABLED(FAN_SOFT_PWM)
|
2017-03-14 20:24:17 +01:00
|
|
|
if (pwm_count_tmp >= 127) {
|
|
|
|
pwm_count_tmp = 0;
|
2019-06-28 17:03:43 -05:00
|
|
|
#define _PWM_FAN(N) do{ \
|
|
|
|
soft_pwm_count_fan[N] = soft_pwm_amount_fan[N] >> 1; \
|
|
|
|
WRITE_FAN(N, soft_pwm_count_fan[N] > 0 ? HIGH : LOW); \
|
2019-03-07 00:09:39 -08:00
|
|
|
}while(0)
|
2016-03-05 18:27:45 -08:00
|
|
|
#if HAS_FAN0
|
2019-06-28 17:03:43 -05:00
|
|
|
_PWM_FAN(0);
|
2016-03-05 18:27:45 -08:00
|
|
|
#endif
|
|
|
|
#if HAS_FAN1
|
2019-06-28 17:03:43 -05:00
|
|
|
_PWM_FAN(1);
|
2016-03-05 18:27:45 -08:00
|
|
|
#endif
|
|
|
|
#if HAS_FAN2
|
2019-06-28 17:03:43 -05:00
|
|
|
_PWM_FAN(2);
|
2016-03-05 18:27:45 -08:00
|
|
|
#endif
|
2020-01-25 16:13:39 +08:00
|
|
|
#if HAS_FAN3
|
|
|
|
_FAN_PWM(3);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN4
|
|
|
|
_FAN_PWM(4);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN5
|
|
|
|
_FAN_PWM(5);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN6
|
|
|
|
_FAN_PWM(6);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN7
|
|
|
|
_FAN_PWM(7);
|
|
|
|
#endif
|
2015-02-24 04:46:11 -08:00
|
|
|
}
|
2016-03-05 18:27:45 -08:00
|
|
|
#if HAS_FAN0
|
2019-06-28 17:03:43 -05:00
|
|
|
if (soft_pwm_count_fan[0] <= pwm_count_tmp) WRITE_FAN(0, LOW);
|
2016-03-05 18:27:45 -08:00
|
|
|
#endif
|
|
|
|
#if HAS_FAN1
|
2019-06-28 17:03:43 -05:00
|
|
|
if (soft_pwm_count_fan[1] <= pwm_count_tmp) WRITE_FAN(1, LOW);
|
2016-03-05 18:27:45 -08:00
|
|
|
#endif
|
|
|
|
#if HAS_FAN2
|
2019-06-28 17:03:43 -05:00
|
|
|
if (soft_pwm_count_fan[2] <= pwm_count_tmp) WRITE_FAN(2, LOW);
|
2016-03-05 18:27:45 -08:00
|
|
|
#endif
|
2020-01-25 16:13:39 +08:00
|
|
|
#if HAS_FAN3
|
|
|
|
if (soft_pwm_count_fan[3] <= pwm_count_tmp) WRITE_FAN(3, LOW);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN4
|
|
|
|
if (soft_pwm_count_fan[4] <= pwm_count_tmp) WRITE_FAN(4, LOW);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN5
|
|
|
|
if (soft_pwm_count_fan[5] <= pwm_count_tmp) WRITE_FAN(5, LOW);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN6
|
|
|
|
if (soft_pwm_count_fan[6] <= pwm_count_tmp) WRITE_FAN(6, LOW);
|
|
|
|
#endif
|
|
|
|
#if HAS_FAN7
|
|
|
|
if (soft_pwm_count_fan[7] <= pwm_count_tmp) WRITE_FAN(7, LOW);
|
|
|
|
#endif
|
2017-03-23 04:20:25 -05:00
|
|
|
#endif // FAN_SOFT_PWM
|
2015-02-24 04:46:11 -08:00
|
|
|
|
2016-10-09 10:06:31 -05:00
|
|
|
// SOFT_PWM_SCALE to frequency:
|
|
|
|
//
|
|
|
|
// 0: 16000000/64/256/128 = 7.6294 Hz
|
|
|
|
// 1: / 64 = 15.2588 Hz
|
|
|
|
// 2: / 32 = 30.5176 Hz
|
|
|
|
// 3: / 16 = 61.0352 Hz
|
|
|
|
// 4: / 8 = 122.0703 Hz
|
|
|
|
// 5: / 4 = 244.1406 Hz
|
2017-03-14 20:24:17 +01:00
|
|
|
pwm_count = pwm_count_tmp + _BV(SOFT_PWM_SCALE);
|
2015-02-24 04:46:11 -08:00
|
|
|
|
2017-03-12 22:10:21 +01:00
|
|
|
// increment slow_pwm_count only every 64th pwm_count,
|
|
|
|
// i.e. yielding a PWM frequency of 16/128 Hz (8s).
|
2017-03-23 04:20:25 -05:00
|
|
|
if (((pwm_count >> SOFT_PWM_SCALE) & 0x3F) == 0) {
|
2015-02-24 04:46:11 -08:00
|
|
|
slow_pwm_count++;
|
2017-03-23 04:20:25 -05:00
|
|
|
slow_pwm_count &= 0x7F;
|
2015-10-02 23:08:58 -07:00
|
|
|
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_HOTEND
|
2019-11-09 17:59:04 -06:00
|
|
|
HOTEND_LOOP() soft_pwm_hotend[e].dec();
|
|
|
|
#endif
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(HAS_HEATED_BED, soft_pwm_bed.dec());
|
2020-07-01 16:27:28 -05:00
|
|
|
TERN_(HAS_HEATED_CHAMBER, soft_pwm_chamber.dec());
|
2021-03-06 14:13:28 -06:00
|
|
|
TERN_(HAS_COOLER, soft_pwm_cooler.dec());
|
2020-04-22 16:35:03 -05:00
|
|
|
}
|
2015-10-02 23:08:58 -07:00
|
|
|
|
2015-02-24 04:46:11 -08:00
|
|
|
#endif // SLOW_PWM_HEATERS
|
|
|
|
|
2017-04-06 18:42:57 -05:00
|
|
|
//
|
|
|
|
// Update lcd buttons 488 times per second
|
|
|
|
//
|
|
|
|
static bool do_buttons;
|
2018-11-11 12:16:24 -06:00
|
|
|
if ((do_buttons ^= true)) ui.update_buttons();
|
2017-04-06 18:42:57 -05:00
|
|
|
|
|
|
|
/**
|
2020-02-15 22:37:16 -06:00
|
|
|
* One sensor is sampled on every other call of the ISR.
|
|
|
|
* Each sensor is read 16 (OVERSAMPLENR) times, taking the average.
|
2017-04-06 18:42:57 -05:00
|
|
|
*
|
2020-02-15 22:37:16 -06:00
|
|
|
* On each Prepare pass, ADC is started for a sensor pin.
|
|
|
|
* On the next pass, the ADC value is read and accumulated.
|
2017-04-06 18:42:57 -05:00
|
|
|
*
|
2020-02-15 22:37:16 -06:00
|
|
|
* This gives each ADC 0.9765ms to charge up.
|
2017-04-06 18:42:57 -05:00
|
|
|
*/
|
2020-02-15 22:37:16 -06:00
|
|
|
#define ACCUMULATE_ADC(obj) do{ \
|
|
|
|
if (!HAL_ADC_READY()) next_sensor_state = adc_sensor_state; \
|
|
|
|
else obj.sample(HAL_READ_ADC()); \
|
2018-07-26 09:59:19 +01:00
|
|
|
}while(0)
|
|
|
|
|
2020-02-15 22:37:16 -06:00
|
|
|
ADCSensorState next_sensor_state = adc_sensor_state < SensorsReady ? (ADCSensorState)(int(adc_sensor_state) + 1) : StartSampling;
|
2020-02-15 19:10:46 -06:00
|
|
|
|
|
|
|
switch (adc_sensor_state) {
|
|
|
|
|
2017-04-06 18:42:57 -05:00
|
|
|
case SensorsReady: {
|
|
|
|
// All sensors have been read. Stay in this state for a few
|
|
|
|
// ISRs to save on calls to temp update/checking code below.
|
2017-06-15 15:14:08 -05:00
|
|
|
constexpr int8_t extra_loops = MIN_ADC_ISR_LOOPS - (int8_t)SensorsReady;
|
2017-04-06 18:42:57 -05:00
|
|
|
static uint8_t delay_count = 0;
|
|
|
|
if (extra_loops > 0) {
|
2018-07-26 09:59:19 +01:00
|
|
|
if (delay_count == 0) delay_count = extra_loops; // Init this delay
|
|
|
|
if (--delay_count) // While delaying...
|
|
|
|
next_sensor_state = SensorsReady; // retain this state (else, next state will be 0)
|
2020-02-15 22:37:16 -06:00
|
|
|
break;
|
2017-04-06 18:42:57 -05:00
|
|
|
}
|
2018-07-26 09:59:19 +01:00
|
|
|
else {
|
2020-02-15 22:37:16 -06:00
|
|
|
adc_sensor_state = StartSampling; // Fall-through to start sampling
|
|
|
|
next_sensor_state = (ADCSensorState)(int(StartSampling) + 1);
|
2018-07-26 09:59:19 +01:00
|
|
|
}
|
2017-04-06 18:42:57 -05:00
|
|
|
}
|
|
|
|
|
2018-07-26 09:59:19 +01:00
|
|
|
case StartSampling: // Start of sampling loops. Do updates/checks.
|
2020-02-15 22:37:16 -06:00
|
|
|
if (++temp_count >= OVERSAMPLENR) { // 10 * 16 * 1/(16000000/64/256) = 164ms.
|
2018-07-26 09:59:19 +01:00
|
|
|
temp_count = 0;
|
|
|
|
readings_ready();
|
|
|
|
}
|
2020-02-15 22:37:16 -06:00
|
|
|
break;
|
2018-07-26 09:59:19 +01:00
|
|
|
|
2018-05-01 19:33:41 -05:00
|
|
|
#if HAS_TEMP_ADC_0
|
2020-02-15 22:37:16 -06:00
|
|
|
case PrepareTemp_0: HAL_START_ADC(TEMP_0_PIN); break;
|
|
|
|
case MeasureTemp_0: ACCUMULATE_ADC(temp_hotend[0]); break;
|
2017-04-06 18:42:57 -05:00
|
|
|
#endif
|
2020-02-15 22:37:16 -06:00
|
|
|
|
2020-11-22 16:47:52 -08:00
|
|
|
#if HAS_TEMP_ADC_BED
|
2020-02-15 22:37:16 -06:00
|
|
|
case PrepareTemp_BED: HAL_START_ADC(TEMP_BED_PIN); break;
|
|
|
|
case MeasureTemp_BED: ACCUMULATE_ADC(temp_bed); break;
|
2017-04-06 18:42:57 -05:00
|
|
|
#endif
|
2020-02-15 22:37:16 -06:00
|
|
|
|
2020-11-22 16:47:52 -08:00
|
|
|
#if HAS_TEMP_ADC_CHAMBER
|
2019-09-05 20:01:38 -05:00
|
|
|
case PrepareTemp_CHAMBER: HAL_START_ADC(TEMP_CHAMBER_PIN); break;
|
2020-02-15 22:37:16 -06:00
|
|
|
case MeasureTemp_CHAMBER: ACCUMULATE_ADC(temp_chamber); break;
|
2018-02-24 19:38:58 +13:00
|
|
|
#endif
|
2020-02-15 22:37:16 -06:00
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_TEMP_ADC_COOLER
|
|
|
|
case PrepareTemp_COOLER: HAL_START_ADC(TEMP_COOLER_PIN); break;
|
|
|
|
case MeasureTemp_COOLER: ACCUMULATE_ADC(temp_cooler); break;
|
|
|
|
#endif
|
|
|
|
|
2020-11-22 16:47:52 -08:00
|
|
|
#if HAS_TEMP_ADC_PROBE
|
2020-02-15 22:37:16 -06:00
|
|
|
case PrepareTemp_PROBE: HAL_START_ADC(TEMP_PROBE_PIN); break;
|
|
|
|
case MeasureTemp_PROBE: ACCUMULATE_ADC(temp_probe); break;
|
2020-01-18 00:16:45 +01:00
|
|
|
#endif
|
2020-02-15 22:37:16 -06:00
|
|
|
|
2021-08-18 20:12:41 -05:00
|
|
|
#if HAS_TEMP_ADC_BOARD
|
|
|
|
case PrepareTemp_BOARD: HAL_START_ADC(TEMP_BOARD_PIN); break;
|
|
|
|
case MeasureTemp_BOARD: ACCUMULATE_ADC(temp_board); break;
|
|
|
|
#endif
|
|
|
|
|
2021-06-11 13:51:29 -07:00
|
|
|
#if HAS_TEMP_ADC_REDUNDANT
|
|
|
|
case PrepareTemp_REDUNDANT: HAL_START_ADC(TEMP_REDUNDANT_PIN); break;
|
|
|
|
case MeasureTemp_REDUNDANT: ACCUMULATE_ADC(temp_redundant); break;
|
|
|
|
#endif
|
|
|
|
|
2018-05-01 19:33:41 -05:00
|
|
|
#if HAS_TEMP_ADC_1
|
2020-02-15 22:37:16 -06:00
|
|
|
case PrepareTemp_1: HAL_START_ADC(TEMP_1_PIN); break;
|
2021-06-11 13:51:29 -07:00
|
|
|
case MeasureTemp_1: ACCUMULATE_ADC(temp_hotend[1]); break;
|
2017-04-06 18:42:57 -05:00
|
|
|
#endif
|
2020-02-15 22:37:16 -06:00
|
|
|
|
2018-05-01 19:33:41 -05:00
|
|
|
#if HAS_TEMP_ADC_2
|
2020-02-15 22:37:16 -06:00
|
|
|
case PrepareTemp_2: HAL_START_ADC(TEMP_2_PIN); break;
|
|
|
|
case MeasureTemp_2: ACCUMULATE_ADC(temp_hotend[2]); break;
|
2017-04-06 18:42:57 -05:00
|
|
|
#endif
|
2020-02-15 22:37:16 -06:00
|
|
|
|
2018-05-01 19:33:41 -05:00
|
|
|
#if HAS_TEMP_ADC_3
|
2020-02-15 22:37:16 -06:00
|
|
|
case PrepareTemp_3: HAL_START_ADC(TEMP_3_PIN); break;
|
|
|
|
case MeasureTemp_3: ACCUMULATE_ADC(temp_hotend[3]); break;
|
2017-04-06 18:42:57 -05:00
|
|
|
#endif
|
2020-02-15 22:37:16 -06:00
|
|
|
|
2018-05-01 19:33:41 -05:00
|
|
|
#if HAS_TEMP_ADC_4
|
2020-02-15 22:37:16 -06:00
|
|
|
case PrepareTemp_4: HAL_START_ADC(TEMP_4_PIN); break;
|
|
|
|
case MeasureTemp_4: ACCUMULATE_ADC(temp_hotend[4]); break;
|
2017-04-06 18:42:57 -05:00
|
|
|
#endif
|
2020-02-15 22:37:16 -06:00
|
|
|
|
2018-09-13 01:35:55 -05:00
|
|
|
#if HAS_TEMP_ADC_5
|
2020-02-15 22:37:16 -06:00
|
|
|
case PrepareTemp_5: HAL_START_ADC(TEMP_5_PIN); break;
|
|
|
|
case MeasureTemp_5: ACCUMULATE_ADC(temp_hotend[5]); break;
|
2018-09-13 01:35:55 -05:00
|
|
|
#endif
|
2020-02-15 22:37:16 -06:00
|
|
|
|
2020-01-25 16:13:39 +08:00
|
|
|
#if HAS_TEMP_ADC_6
|
2020-02-15 22:37:16 -06:00
|
|
|
case PrepareTemp_6: HAL_START_ADC(TEMP_6_PIN); break;
|
|
|
|
case MeasureTemp_6: ACCUMULATE_ADC(temp_hotend[6]); break;
|
2020-01-25 16:13:39 +08:00
|
|
|
#endif
|
2020-02-15 22:37:16 -06:00
|
|
|
|
2020-01-25 16:13:39 +08:00
|
|
|
#if HAS_TEMP_ADC_7
|
2020-02-15 22:37:16 -06:00
|
|
|
case PrepareTemp_7: HAL_START_ADC(TEMP_7_PIN); break;
|
|
|
|
case MeasureTemp_7: ACCUMULATE_ADC(temp_hotend[7]); break;
|
2020-01-25 16:13:39 +08:00
|
|
|
#endif
|
2020-02-15 22:37:16 -06:00
|
|
|
|
2017-04-06 18:42:57 -05:00
|
|
|
#if ENABLED(FILAMENT_WIDTH_SENSOR)
|
2020-02-15 22:37:16 -06:00
|
|
|
case Prepare_FILWIDTH: HAL_START_ADC(FILWIDTH_PIN); break;
|
|
|
|
case Measure_FILWIDTH:
|
2020-06-18 15:23:03 -05:00
|
|
|
if (!HAL_ADC_READY()) next_sensor_state = adc_sensor_state; // Redo this state
|
|
|
|
else filwidth.accumulate(HAL_READ_ADC());
|
2020-02-15 22:37:16 -06:00
|
|
|
break;
|
2017-04-06 18:42:57 -05:00
|
|
|
#endif
|
2020-02-15 22:37:16 -06:00
|
|
|
|
2020-06-18 15:23:03 -05:00
|
|
|
#if ENABLED(POWER_MONITOR_CURRENT)
|
|
|
|
case Prepare_POWER_MONITOR_CURRENT:
|
|
|
|
HAL_START_ADC(POWER_MONITOR_CURRENT_PIN);
|
|
|
|
break;
|
|
|
|
case Measure_POWER_MONITOR_CURRENT:
|
|
|
|
if (!HAL_ADC_READY()) next_sensor_state = adc_sensor_state; // Redo this state
|
|
|
|
else power_monitor.add_current_sample(HAL_READ_ADC());
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if ENABLED(POWER_MONITOR_VOLTAGE)
|
|
|
|
case Prepare_POWER_MONITOR_VOLTAGE:
|
|
|
|
HAL_START_ADC(POWER_MONITOR_VOLTAGE_PIN);
|
|
|
|
break;
|
|
|
|
case Measure_POWER_MONITOR_VOLTAGE:
|
|
|
|
if (!HAL_ADC_READY()) next_sensor_state = adc_sensor_state; // Redo this state
|
|
|
|
else power_monitor.add_voltage_sample(HAL_READ_ADC());
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
|
2019-09-08 00:55:34 -05:00
|
|
|
#if HAS_JOY_ADC_X
|
2020-02-15 22:37:16 -06:00
|
|
|
case PrepareJoy_X: HAL_START_ADC(JOY_X_PIN); break;
|
|
|
|
case MeasureJoy_X: ACCUMULATE_ADC(joystick.x); break;
|
2019-09-08 00:55:34 -05:00
|
|
|
#endif
|
2020-02-15 22:37:16 -06:00
|
|
|
|
2019-09-08 00:55:34 -05:00
|
|
|
#if HAS_JOY_ADC_Y
|
2020-02-15 22:37:16 -06:00
|
|
|
case PrepareJoy_Y: HAL_START_ADC(JOY_Y_PIN); break;
|
|
|
|
case MeasureJoy_Y: ACCUMULATE_ADC(joystick.y); break;
|
2019-09-08 00:55:34 -05:00
|
|
|
#endif
|
2020-02-15 22:37:16 -06:00
|
|
|
|
2019-09-08 00:55:34 -05:00
|
|
|
#if HAS_JOY_ADC_Z
|
2020-02-15 22:37:16 -06:00
|
|
|
case PrepareJoy_Z: HAL_START_ADC(JOY_Z_PIN); break;
|
|
|
|
case MeasureJoy_Z: ACCUMULATE_ADC(joystick.z); break;
|
2019-09-08 00:55:34 -05:00
|
|
|
#endif
|
2020-02-15 22:37:16 -06:00
|
|
|
|
2018-11-19 03:39:49 +01:00
|
|
|
#if HAS_ADC_BUTTONS
|
2020-02-15 22:37:16 -06:00
|
|
|
#ifndef ADC_BUTTON_DEBOUNCE_DELAY
|
|
|
|
#define ADC_BUTTON_DEBOUNCE_DELAY 16
|
|
|
|
#endif
|
|
|
|
case Prepare_ADC_KEY: HAL_START_ADC(ADC_KEYPAD_PIN); break;
|
|
|
|
case Measure_ADC_KEY:
|
|
|
|
if (!HAL_ADC_READY())
|
|
|
|
next_sensor_state = adc_sensor_state; // redo this state
|
|
|
|
else if (ADCKey_count < ADC_BUTTON_DEBOUNCE_DELAY) {
|
|
|
|
raw_ADCKey_value = HAL_READ_ADC();
|
|
|
|
if (raw_ADCKey_value <= 900UL * HAL_ADC_RANGE / 1024UL) {
|
|
|
|
NOMORE(current_ADCKey_raw, raw_ADCKey_value);
|
|
|
|
ADCKey_count++;
|
|
|
|
}
|
|
|
|
else { //ADC Key release
|
|
|
|
if (ADCKey_count > 0) ADCKey_count++; else ADCKey_pressed = false;
|
|
|
|
if (ADCKey_pressed) {
|
|
|
|
ADCKey_count = 0;
|
|
|
|
current_ADCKey_raw = HAL_ADC_RANGE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (ADCKey_count == ADC_BUTTON_DEBOUNCE_DELAY) ADCKey_pressed = true;
|
|
|
|
break;
|
|
|
|
#endif // HAS_ADC_BUTTONS
|
|
|
|
|
|
|
|
case StartupDelay: break;
|
2015-02-24 04:46:11 -08:00
|
|
|
|
2017-04-06 18:42:57 -05:00
|
|
|
} // switch(adc_sensor_state)
|
2015-03-23 00:27:29 -07:00
|
|
|
|
2018-07-26 09:59:19 +01:00
|
|
|
// Go to the next state
|
|
|
|
adc_sensor_state = next_sensor_state;
|
2015-01-23 23:13:06 +01:00
|
|
|
|
2018-07-26 09:59:19 +01:00
|
|
|
//
|
|
|
|
// Additional ~1KHz Tasks
|
|
|
|
//
|
2017-04-06 18:42:57 -05:00
|
|
|
|
2020-02-15 21:42:28 -06:00
|
|
|
#if ENABLED(BABYSTEPPING) && DISABLED(INTEGRATED_BABYSTEPPING)
|
2019-04-06 18:04:34 -05:00
|
|
|
babystep.task();
|
2019-02-27 05:38:56 -05:00
|
|
|
#endif
|
2016-10-31 07:28:51 -05:00
|
|
|
|
2018-05-16 04:08:43 -03:00
|
|
|
// Poll endstops state, if required
|
|
|
|
endstops.poll();
|
2018-05-09 02:17:53 -03:00
|
|
|
|
2021-04-08 15:43:16 -05:00
|
|
|
// Periodically call the planner timer service routine
|
|
|
|
planner.isr();
|
2011-11-15 22:50:43 +01:00
|
|
|
}
|
2017-09-16 04:44:37 -05:00
|
|
|
|
2018-03-07 01:53:50 -06:00
|
|
|
#if HAS_TEMP_SENSOR
|
2021-04-23 19:14:49 -05:00
|
|
|
/**
|
|
|
|
* Print a single heater state in the form:
|
|
|
|
* Bed: " B:nnn.nn /nnn.nn"
|
|
|
|
* Chamber: " C:nnn.nn /nnn.nn"
|
|
|
|
* Probe: " P:nnn.nn /nnn.nn"
|
|
|
|
* Cooler: " L:nnn.nn /nnn.nn"
|
2021-06-11 13:51:29 -07:00
|
|
|
* Redundant: " R:nnn.nn /nnn.nn"
|
2021-04-23 19:14:49 -05:00
|
|
|
* Extruder: " T0:nnn.nn /nnn.nn"
|
|
|
|
* With ADC: " T0:nnn.nn /nnn.nn (nnn.nn)"
|
|
|
|
*/
|
2021-05-23 21:33:22 -05:00
|
|
|
static void print_heater_state(const heater_id_t e, const_celsius_float_t c, const_celsius_float_t t
|
|
|
|
OPTARG(SHOW_TEMP_ADC_VALUES, const float r)
|
2017-09-16 04:44:37 -05:00
|
|
|
) {
|
2019-06-27 19:29:53 -07:00
|
|
|
char k;
|
|
|
|
switch (e) {
|
2021-03-06 14:13:28 -06:00
|
|
|
default:
|
|
|
|
#if HAS_TEMP_HOTEND
|
|
|
|
k = 'T'; break;
|
|
|
|
#endif
|
|
|
|
#if HAS_TEMP_BED
|
|
|
|
case H_BED: k = 'B'; break;
|
|
|
|
#endif
|
2019-06-27 19:29:53 -07:00
|
|
|
#if HAS_TEMP_CHAMBER
|
2019-07-02 08:39:55 -05:00
|
|
|
case H_CHAMBER: k = 'C'; break;
|
2017-09-16 04:44:37 -05:00
|
|
|
#endif
|
2020-01-18 00:16:45 +01:00
|
|
|
#if HAS_TEMP_PROBE
|
|
|
|
case H_PROBE: k = 'P'; break;
|
|
|
|
#endif
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_TEMP_COOLER
|
|
|
|
case H_COOLER: k = 'L'; break;
|
|
|
|
#endif
|
2021-08-18 20:12:41 -05:00
|
|
|
#if HAS_TEMP_BOARD
|
|
|
|
case H_BOARD: k = 'M'; break;
|
|
|
|
#endif
|
2021-06-11 13:51:29 -07:00
|
|
|
#if HAS_TEMP_REDUNDANT
|
2021-03-06 14:13:28 -06:00
|
|
|
case H_REDUNDANT: k = 'R'; break;
|
2019-06-27 19:29:53 -07:00
|
|
|
#endif
|
|
|
|
}
|
2021-01-05 13:18:09 +01:00
|
|
|
SERIAL_CHAR(' ', k);
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_MULTI_HOTEND
|
2019-02-23 22:53:01 -06:00
|
|
|
if (e >= 0) SERIAL_CHAR('0' + e);
|
2017-09-16 04:44:37 -05:00
|
|
|
#endif
|
2021-01-05 13:18:09 +01:00
|
|
|
#ifdef SERIAL_FLOAT_PRECISION
|
|
|
|
#define SFP _MIN(SERIAL_FLOAT_PRECISION, 2)
|
|
|
|
#else
|
|
|
|
#define SFP 2
|
|
|
|
#endif
|
2019-02-23 22:53:01 -06:00
|
|
|
SERIAL_CHAR(':');
|
2021-01-05 13:18:09 +01:00
|
|
|
SERIAL_PRINT(c, SFP);
|
2021-01-05 06:57:58 +01:00
|
|
|
SERIAL_ECHOPGM(" /");
|
2021-01-05 13:18:09 +01:00
|
|
|
SERIAL_PRINT(t, SFP);
|
2017-09-16 04:44:37 -05:00
|
|
|
#if ENABLED(SHOW_TEMP_ADC_VALUES)
|
2021-02-08 20:51:57 -05:00
|
|
|
// Temperature MAX SPI boards do not have an OVERSAMPLENR defined
|
2021-09-09 04:57:05 -05:00
|
|
|
SERIAL_ECHOPGM(" (", TERN(HAS_MAXTC_LIBRARIES, k == 'T', false) ? r : r * RECIPROCAL(OVERSAMPLENR));
|
2019-02-23 22:53:01 -06:00
|
|
|
SERIAL_CHAR(')');
|
2017-09-16 04:44:37 -05:00
|
|
|
#endif
|
2018-01-20 14:13:17 -06:00
|
|
|
delay(2);
|
2017-09-16 04:44:37 -05:00
|
|
|
}
|
|
|
|
|
2019-06-27 19:29:53 -07:00
|
|
|
void Temperature::print_heater_states(const uint8_t target_extruder
|
2021-06-11 13:51:29 -07:00
|
|
|
OPTARG(HAS_TEMP_REDUNDANT, const bool include_r/*=false*/)
|
2019-06-27 19:29:53 -07:00
|
|
|
) {
|
2017-09-16 04:44:37 -05:00
|
|
|
#if HAS_TEMP_HOTEND
|
2021-08-18 20:12:41 -05:00
|
|
|
print_heater_state(H_E0, degHotend(target_extruder), degTargetHotend(target_extruder) OPTARG(SHOW_TEMP_ADC_VALUES, rawHotendTemp(target_extruder)));
|
2017-09-16 04:44:37 -05:00
|
|
|
#endif
|
2018-04-23 17:13:01 -05:00
|
|
|
#if HAS_HEATED_BED
|
2021-05-23 21:33:22 -05:00
|
|
|
print_heater_state(H_BED, degBed(), degTargetBed() OPTARG(SHOW_TEMP_ADC_VALUES, rawBedTemp()));
|
2017-09-16 04:44:37 -05:00
|
|
|
#endif
|
2018-02-24 19:38:58 +13:00
|
|
|
#if HAS_TEMP_CHAMBER
|
2021-05-23 21:33:22 -05:00
|
|
|
print_heater_state(H_CHAMBER, degChamber(), TERN0(HAS_HEATED_CHAMBER, degTargetChamber()) OPTARG(SHOW_TEMP_ADC_VALUES, rawChamberTemp()));
|
|
|
|
#endif
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_TEMP_COOLER
|
2021-05-23 21:33:22 -05:00
|
|
|
print_heater_state(H_COOLER, degCooler(), TERN0(HAS_COOLER, degTargetCooler()) OPTARG(SHOW_TEMP_ADC_VALUES, rawCoolerTemp()));
|
|
|
|
#endif
|
2020-01-18 00:16:45 +01:00
|
|
|
#if HAS_TEMP_PROBE
|
2021-08-18 20:12:41 -05:00
|
|
|
print_heater_state(H_PROBE, degProbe(), 0 OPTARG(SHOW_TEMP_ADC_VALUES, rawProbeTemp()));
|
|
|
|
#endif
|
|
|
|
#if HAS_TEMP_BOARD
|
|
|
|
print_heater_state(H_BOARD, degBoard(), 0 OPTARG(SHOW_TEMP_ADC_VALUES, rawBoardTemp()));
|
2021-02-08 10:36:57 +08:00
|
|
|
#endif
|
2021-06-11 13:51:29 -07:00
|
|
|
#if HAS_TEMP_REDUNDANT
|
|
|
|
if (include_r) print_heater_state(H_REDUNDANT, degRedundant(), degRedundantTarget() OPTARG(SHOW_TEMP_ADC_VALUES, rawRedundantTemp()));
|
|
|
|
#endif
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_MULTI_HOTEND
|
2021-05-23 21:33:22 -05:00
|
|
|
HOTEND_LOOP() print_heater_state((heater_id_t)e, degHotend(e), degTargetHotend(e) OPTARG(SHOW_TEMP_ADC_VALUES, rawHotendTemp(e)));
|
2017-09-16 04:44:37 -05:00
|
|
|
#endif
|
2021-09-09 04:57:05 -05:00
|
|
|
SERIAL_ECHOPGM(" @:", getHeaterPower((heater_id_t)target_extruder));
|
2018-04-23 17:13:01 -05:00
|
|
|
#if HAS_HEATED_BED
|
2021-09-09 04:57:05 -05:00
|
|
|
SERIAL_ECHOPGM(" B@:", getHeaterPower(H_BED));
|
2019-04-12 22:38:10 +02:00
|
|
|
#endif
|
|
|
|
#if HAS_HEATED_CHAMBER
|
2021-09-09 04:57:05 -05:00
|
|
|
SERIAL_ECHOPGM(" C@:", getHeaterPower(H_CHAMBER));
|
2017-09-16 04:44:37 -05:00
|
|
|
#endif
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_COOLER
|
2021-09-09 04:57:05 -05:00
|
|
|
SERIAL_ECHOPGM(" C@:", getHeaterPower(H_COOLER));
|
2021-03-06 14:13:28 -06:00
|
|
|
#endif
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_MULTI_HOTEND
|
2017-09-16 04:44:37 -05:00
|
|
|
HOTEND_LOOP() {
|
2021-09-09 04:57:05 -05:00
|
|
|
SERIAL_ECHOPGM(" @", e);
|
2019-02-23 22:53:01 -06:00
|
|
|
SERIAL_CHAR(':');
|
2020-09-13 18:06:14 -05:00
|
|
|
SERIAL_ECHO(getHeaterPower((heater_id_t)e));
|
2017-09-16 04:44:37 -05:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-09-16 13:05:01 -05:00
|
|
|
#if ENABLED(AUTO_REPORT_TEMPERATURES)
|
2021-02-01 01:11:50 +01:00
|
|
|
AutoReporter<Temperature::AutoReportTemp> Temperature::auto_reporter;
|
2021-05-23 21:33:22 -05:00
|
|
|
void Temperature::AutoReportTemp::report() { print_heater_states(active_extruder); SERIAL_EOL(); }
|
2021-01-28 20:40:20 -06:00
|
|
|
#endif
|
2017-09-16 13:05:01 -05:00
|
|
|
|
2021-03-24 10:12:57 -05:00
|
|
|
#if HAS_HOTEND && HAS_STATUS_MESSAGE
|
2018-10-07 18:06:14 -04:00
|
|
|
void Temperature::set_heating_message(const uint8_t e) {
|
|
|
|
const bool heating = isHeatingHotend(e);
|
2021-09-25 17:05:11 -05:00
|
|
|
ui.status_printf(0,
|
2020-04-19 23:56:55 -05:00
|
|
|
#if HAS_MULTI_HOTEND
|
2021-09-25 17:05:11 -05:00
|
|
|
F("E%c " S_FMT), '1' + e
|
2019-09-27 03:38:43 -06:00
|
|
|
#else
|
2021-09-25 17:05:11 -05:00
|
|
|
F("E1 " S_FMT)
|
2019-09-27 03:38:43 -06:00
|
|
|
#endif
|
2019-10-09 18:46:10 -06:00
|
|
|
, heating ? GET_TEXT(MSG_HEATING) : GET_TEXT(MSG_COOLING)
|
2019-09-27 03:38:43 -06:00
|
|
|
);
|
2018-10-07 18:06:14 -04:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-09-29 01:44:47 -05:00
|
|
|
#if HAS_TEMP_HOTEND
|
|
|
|
|
|
|
|
#ifndef MIN_COOLING_SLOPE_DEG
|
|
|
|
#define MIN_COOLING_SLOPE_DEG 1.50
|
|
|
|
#endif
|
|
|
|
#ifndef MIN_COOLING_SLOPE_TIME
|
|
|
|
#define MIN_COOLING_SLOPE_TIME 60
|
|
|
|
#endif
|
|
|
|
|
2018-11-10 18:07:38 -06:00
|
|
|
bool Temperature::wait_for_hotend(const uint8_t target_extruder, const bool no_wait_for_cooling/*=true*/
|
2021-05-23 21:33:22 -05:00
|
|
|
OPTARG(G26_CLICK_CAN_CANCEL, const bool click_to_cancel/*=false*/)
|
2018-11-10 18:07:38 -06:00
|
|
|
) {
|
2020-05-03 07:32:25 +02:00
|
|
|
#if ENABLED(AUTOTEMP)
|
|
|
|
REMEMBER(1, planner.autotemp_enabled, false);
|
|
|
|
#endif
|
|
|
|
|
2018-09-29 01:44:47 -05:00
|
|
|
#if TEMP_RESIDENCY_TIME > 0
|
|
|
|
millis_t residency_start_ms = 0;
|
2020-01-14 23:55:26 -08:00
|
|
|
bool first_loop = true;
|
2018-09-29 01:44:47 -05:00
|
|
|
// Loop until the temperature has stabilized
|
2020-04-03 19:49:45 -05:00
|
|
|
#define TEMP_CONDITIONS (!residency_start_ms || PENDING(now, residency_start_ms + SEC_TO_MS(TEMP_RESIDENCY_TIME)))
|
2018-09-29 01:44:47 -05:00
|
|
|
#else
|
|
|
|
// Loop until the temperature is very close target
|
|
|
|
#define TEMP_CONDITIONS (wants_to_cool ? isCoolingHotend(target_extruder) : isHeatingHotend(target_extruder))
|
|
|
|
#endif
|
|
|
|
|
2018-09-29 17:03:23 -05:00
|
|
|
#if DISABLED(BUSY_WHILE_HEATING) && ENABLED(HOST_KEEPALIVE_FEATURE)
|
2018-09-29 01:44:47 -05:00
|
|
|
KEEPALIVE_STATE(NOT_BUSY);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if ENABLED(PRINTER_EVENT_LEDS)
|
2021-04-23 19:14:49 -05:00
|
|
|
const celsius_float_t start_temp = degHotend(target_extruder);
|
2018-10-11 04:25:43 +02:00
|
|
|
printerEventLEDs.onHotendHeatingStart();
|
2018-09-29 01:44:47 -05:00
|
|
|
#endif
|
|
|
|
|
2020-01-14 23:55:26 -08:00
|
|
|
bool wants_to_cool = false;
|
2021-04-23 19:14:49 -05:00
|
|
|
celsius_float_t target_temp = -1.0, old_temp = 9999.0;
|
2018-09-29 01:44:47 -05:00
|
|
|
millis_t now, next_temp_ms = 0, next_cool_check_ms = 0;
|
2020-09-12 05:51:19 +02:00
|
|
|
wait_for_heatup = true;
|
2018-09-29 01:44:47 -05:00
|
|
|
do {
|
|
|
|
// Target temperature might be changed during the loop
|
|
|
|
if (target_temp != degTargetHotend(target_extruder)) {
|
|
|
|
wants_to_cool = isCoolingHotend(target_extruder);
|
|
|
|
target_temp = degTargetHotend(target_extruder);
|
|
|
|
|
|
|
|
// Exit if S<lower>, continue if S<higher>, R<lower>, or R<higher>
|
|
|
|
if (no_wait_for_cooling && wants_to_cool) break;
|
|
|
|
}
|
|
|
|
|
|
|
|
now = millis();
|
2018-11-14 17:33:04 -06:00
|
|
|
if (ELAPSED(now, next_temp_ms)) { // Print temp & remaining time every 1s while waiting
|
2018-09-29 01:44:47 -05:00
|
|
|
next_temp_ms = now + 1000UL;
|
2018-11-14 17:33:04 -06:00
|
|
|
print_heater_states(target_extruder);
|
2018-09-29 01:44:47 -05:00
|
|
|
#if TEMP_RESIDENCY_TIME > 0
|
2018-11-29 16:58:58 -06:00
|
|
|
SERIAL_ECHOPGM(" W:");
|
2018-09-29 01:44:47 -05:00
|
|
|
if (residency_start_ms)
|
2020-04-03 19:49:45 -05:00
|
|
|
SERIAL_ECHO(long((SEC_TO_MS(TEMP_RESIDENCY_TIME) - (now - residency_start_ms)) / 1000UL));
|
2018-09-29 01:44:47 -05:00
|
|
|
else
|
2018-11-29 16:58:58 -06:00
|
|
|
SERIAL_CHAR('?');
|
2018-09-29 01:44:47 -05:00
|
|
|
#endif
|
|
|
|
SERIAL_EOL();
|
|
|
|
}
|
|
|
|
|
2019-09-02 19:21:47 -05:00
|
|
|
idle();
|
2018-09-29 01:44:47 -05:00
|
|
|
gcode.reset_stepper_timeout(); // Keep steppers powered
|
|
|
|
|
2021-04-23 19:14:49 -05:00
|
|
|
const celsius_float_t temp = degHotend(target_extruder);
|
2018-09-29 01:44:47 -05:00
|
|
|
|
|
|
|
#if ENABLED(PRINTER_EVENT_LEDS)
|
|
|
|
// Gradually change LED strip from violet to red as nozzle heats up
|
2018-10-11 04:25:43 +02:00
|
|
|
if (!wants_to_cool) printerEventLEDs.onHotendHeating(start_temp, temp, target_temp);
|
2018-09-29 01:44:47 -05:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if TEMP_RESIDENCY_TIME > 0
|
|
|
|
|
2021-04-23 19:14:49 -05:00
|
|
|
const celsius_float_t temp_diff = ABS(target_temp - temp);
|
2018-09-29 01:44:47 -05:00
|
|
|
|
|
|
|
if (!residency_start_ms) {
|
|
|
|
// Start the TEMP_RESIDENCY_TIME timer when we reach target temp for the first time.
|
2020-07-07 01:50:57 -05:00
|
|
|
if (temp_diff < TEMP_WINDOW)
|
2020-07-07 01:52:21 +02:00
|
|
|
residency_start_ms = now + (first_loop ? SEC_TO_MS(TEMP_RESIDENCY_TIME) / 3 : 0);
|
2018-09-29 01:44:47 -05:00
|
|
|
}
|
|
|
|
else if (temp_diff > TEMP_HYSTERESIS) {
|
|
|
|
// Restart the timer whenever the temperature falls outside the hysteresis.
|
|
|
|
residency_start_ms = now;
|
|
|
|
}
|
|
|
|
|
2020-01-14 23:55:26 -08:00
|
|
|
first_loop = false;
|
|
|
|
|
2018-09-29 01:44:47 -05:00
|
|
|
#endif
|
|
|
|
|
|
|
|
// Prevent a wait-forever situation if R is misused i.e. M109 R0
|
|
|
|
if (wants_to_cool) {
|
|
|
|
// break after MIN_COOLING_SLOPE_TIME seconds
|
|
|
|
// if the temperature did not drop at least MIN_COOLING_SLOPE_DEG
|
|
|
|
if (!next_cool_check_ms || ELAPSED(now, next_cool_check_ms)) {
|
|
|
|
if (old_temp - temp < float(MIN_COOLING_SLOPE_DEG)) break;
|
2021-01-28 20:40:20 -06:00
|
|
|
next_cool_check_ms = now + SEC_TO_MS(MIN_COOLING_SLOPE_TIME);
|
2018-09-29 01:44:47 -05:00
|
|
|
old_temp = temp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-10 18:07:38 -06:00
|
|
|
#if G26_CLICK_CAN_CANCEL
|
2018-11-11 12:16:24 -06:00
|
|
|
if (click_to_cancel && ui.use_click()) {
|
2018-11-10 18:07:38 -06:00
|
|
|
wait_for_heatup = false;
|
2021-06-13 23:08:46 -04:00
|
|
|
TERN_(HAS_LCD_MENU, ui.quick_feedback());
|
2018-11-10 18:07:38 -06:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-09-29 01:44:47 -05:00
|
|
|
} while (wait_for_heatup && TEMP_CONDITIONS);
|
|
|
|
|
|
|
|
if (wait_for_heatup) {
|
2020-09-12 05:51:19 +02:00
|
|
|
wait_for_heatup = false;
|
2021-09-07 02:15:24 -05:00
|
|
|
#if HAS_DWIN_E3V2_BASIC
|
2020-06-16 01:45:27 -05:00
|
|
|
HMI_flag.heat_flag = 0;
|
|
|
|
duration_t elapsed = print_job_timer.duration(); // print timer
|
2020-06-24 19:44:50 -05:00
|
|
|
dwin_heat_time = elapsed.value;
|
2021-11-10 11:55:20 -05:00
|
|
|
#else
|
|
|
|
ui.reset_status();
|
2020-06-16 01:45:27 -05:00
|
|
|
#endif
|
2020-04-22 16:35:03 -05:00
|
|
|
TERN_(PRINTER_EVENT_LEDS, printerEventLEDs.onHeatingDone());
|
2020-09-12 05:51:19 +02:00
|
|
|
return true;
|
2018-09-29 01:44:47 -05:00
|
|
|
}
|
|
|
|
|
2020-09-12 05:51:19 +02:00
|
|
|
return false;
|
2018-09-29 01:44:47 -05:00
|
|
|
}
|
|
|
|
|
2021-02-25 06:28:27 -08:00
|
|
|
#if ENABLED(WAIT_FOR_HOTEND)
|
|
|
|
void Temperature::wait_for_hotend_heating(const uint8_t target_extruder) {
|
|
|
|
if (isHeatingHotend(target_extruder)) {
|
|
|
|
SERIAL_ECHOLNPGM("Wait for hotend heating...");
|
2021-09-25 17:05:11 -05:00
|
|
|
LCD_MESSAGE(MSG_HEATING);
|
2021-02-25 06:28:27 -08:00
|
|
|
wait_for_hotend(target_extruder);
|
|
|
|
ui.reset_status();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-09-29 01:44:47 -05:00
|
|
|
#endif // HAS_TEMP_HOTEND
|
|
|
|
|
|
|
|
#if HAS_HEATED_BED
|
|
|
|
|
|
|
|
#ifndef MIN_COOLING_SLOPE_DEG_BED
|
2020-05-05 00:23:35 +02:00
|
|
|
#define MIN_COOLING_SLOPE_DEG_BED 1.00
|
2018-09-29 01:44:47 -05:00
|
|
|
#endif
|
|
|
|
#ifndef MIN_COOLING_SLOPE_TIME_BED
|
|
|
|
#define MIN_COOLING_SLOPE_TIME_BED 60
|
|
|
|
#endif
|
|
|
|
|
2018-11-25 18:05:18 -06:00
|
|
|
bool Temperature::wait_for_bed(const bool no_wait_for_cooling/*=true*/
|
2021-05-23 21:33:22 -05:00
|
|
|
OPTARG(G26_CLICK_CAN_CANCEL, const bool click_to_cancel/*=false*/)
|
2018-11-10 18:07:38 -06:00
|
|
|
) {
|
2018-09-29 01:44:47 -05:00
|
|
|
#if TEMP_BED_RESIDENCY_TIME > 0
|
|
|
|
millis_t residency_start_ms = 0;
|
2019-05-01 22:55:58 -04:00
|
|
|
bool first_loop = true;
|
2018-09-29 01:44:47 -05:00
|
|
|
// Loop until the temperature has stabilized
|
2020-04-03 19:49:45 -05:00
|
|
|
#define TEMP_BED_CONDITIONS (!residency_start_ms || PENDING(now, residency_start_ms + SEC_TO_MS(TEMP_BED_RESIDENCY_TIME)))
|
2018-09-29 01:44:47 -05:00
|
|
|
#else
|
|
|
|
// Loop until the temperature is very close target
|
|
|
|
#define TEMP_BED_CONDITIONS (wants_to_cool ? isCoolingBed() : isHeatingBed())
|
|
|
|
#endif
|
|
|
|
|
2018-09-29 17:03:23 -05:00
|
|
|
#if DISABLED(BUSY_WHILE_HEATING) && ENABLED(HOST_KEEPALIVE_FEATURE)
|
2018-09-29 01:44:47 -05:00
|
|
|
KEEPALIVE_STATE(NOT_BUSY);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if ENABLED(PRINTER_EVENT_LEDS)
|
2021-04-23 19:14:49 -05:00
|
|
|
const celsius_float_t start_temp = degBed();
|
2018-10-11 04:25:43 +02:00
|
|
|
printerEventLEDs.onBedHeatingStart();
|
2018-09-29 01:44:47 -05:00
|
|
|
#endif
|
|
|
|
|
2020-09-12 05:51:19 +02:00
|
|
|
bool wants_to_cool = false;
|
2021-04-23 19:14:49 -05:00
|
|
|
celsius_float_t target_temp = -1, old_temp = 9999;
|
2020-09-12 05:51:19 +02:00
|
|
|
millis_t now, next_temp_ms = 0, next_cool_check_ms = 0;
|
|
|
|
wait_for_heatup = true;
|
2018-09-29 01:44:47 -05:00
|
|
|
do {
|
|
|
|
// Target temperature might be changed during the loop
|
|
|
|
if (target_temp != degTargetBed()) {
|
|
|
|
wants_to_cool = isCoolingBed();
|
|
|
|
target_temp = degTargetBed();
|
|
|
|
|
|
|
|
// Exit if S<lower>, continue if S<higher>, R<lower>, or R<higher>
|
|
|
|
if (no_wait_for_cooling && wants_to_cool) break;
|
|
|
|
}
|
|
|
|
|
|
|
|
now = millis();
|
|
|
|
if (ELAPSED(now, next_temp_ms)) { //Print Temp Reading every 1 second while heating up.
|
|
|
|
next_temp_ms = now + 1000UL;
|
2018-11-14 17:33:04 -06:00
|
|
|
print_heater_states(active_extruder);
|
2018-09-29 01:44:47 -05:00
|
|
|
#if TEMP_BED_RESIDENCY_TIME > 0
|
2018-11-29 16:58:58 -06:00
|
|
|
SERIAL_ECHOPGM(" W:");
|
2018-09-29 01:44:47 -05:00
|
|
|
if (residency_start_ms)
|
2020-04-03 19:49:45 -05:00
|
|
|
SERIAL_ECHO(long((SEC_TO_MS(TEMP_BED_RESIDENCY_TIME) - (now - residency_start_ms)) / 1000UL));
|
2018-09-29 01:44:47 -05:00
|
|
|
else
|
2018-11-29 16:58:58 -06:00
|
|
|
SERIAL_CHAR('?');
|
2018-09-29 01:44:47 -05:00
|
|
|
#endif
|
|
|
|
SERIAL_EOL();
|
|
|
|
}
|
|
|
|
|
|
|
|
idle();
|
|
|
|
gcode.reset_stepper_timeout(); // Keep steppers powered
|
|
|
|
|
2021-04-23 19:14:49 -05:00
|
|
|
const celsius_float_t temp = degBed();
|
2018-09-29 01:44:47 -05:00
|
|
|
|
|
|
|
#if ENABLED(PRINTER_EVENT_LEDS)
|
|
|
|
// Gradually change LED strip from blue to violet as bed heats up
|
2018-10-11 04:25:43 +02:00
|
|
|
if (!wants_to_cool) printerEventLEDs.onBedHeating(start_temp, temp, target_temp);
|
2018-09-29 01:44:47 -05:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if TEMP_BED_RESIDENCY_TIME > 0
|
|
|
|
|
2021-04-23 19:14:49 -05:00
|
|
|
const celsius_float_t temp_diff = ABS(target_temp - temp);
|
2018-09-29 01:44:47 -05:00
|
|
|
|
|
|
|
if (!residency_start_ms) {
|
|
|
|
// Start the TEMP_BED_RESIDENCY_TIME timer when we reach target temp for the first time.
|
2020-07-07 01:52:21 +02:00
|
|
|
if (temp_diff < TEMP_BED_WINDOW)
|
|
|
|
residency_start_ms = now + (first_loop ? SEC_TO_MS(TEMP_BED_RESIDENCY_TIME) / 3 : 0);
|
2018-09-29 01:44:47 -05:00
|
|
|
}
|
|
|
|
else if (temp_diff > TEMP_BED_HYSTERESIS) {
|
|
|
|
// Restart the timer whenever the temperature falls outside the hysteresis.
|
|
|
|
residency_start_ms = now;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // TEMP_BED_RESIDENCY_TIME > 0
|
|
|
|
|
|
|
|
// Prevent a wait-forever situation if R is misused i.e. M190 R0
|
|
|
|
if (wants_to_cool) {
|
|
|
|
// Break after MIN_COOLING_SLOPE_TIME_BED seconds
|
|
|
|
// if the temperature did not drop at least MIN_COOLING_SLOPE_DEG_BED
|
|
|
|
if (!next_cool_check_ms || ELAPSED(now, next_cool_check_ms)) {
|
|
|
|
if (old_temp - temp < float(MIN_COOLING_SLOPE_DEG_BED)) break;
|
2021-01-28 20:40:20 -06:00
|
|
|
next_cool_check_ms = now + SEC_TO_MS(MIN_COOLING_SLOPE_TIME_BED);
|
2018-09-29 01:44:47 -05:00
|
|
|
old_temp = temp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-10 18:07:38 -06:00
|
|
|
#if G26_CLICK_CAN_CANCEL
|
2018-11-11 12:16:24 -06:00
|
|
|
if (click_to_cancel && ui.use_click()) {
|
2018-11-10 18:07:38 -06:00
|
|
|
wait_for_heatup = false;
|
2021-06-13 23:08:46 -04:00
|
|
|
TERN_(HAS_LCD_MENU, ui.quick_feedback());
|
2018-11-10 18:07:38 -06:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-05-01 22:55:58 -04:00
|
|
|
#if TEMP_BED_RESIDENCY_TIME > 0
|
|
|
|
first_loop = false;
|
|
|
|
#endif
|
2019-02-28 14:35:05 -06:00
|
|
|
|
2018-09-29 01:44:47 -05:00
|
|
|
} while (wait_for_heatup && TEMP_BED_CONDITIONS);
|
|
|
|
|
2020-09-12 05:51:19 +02:00
|
|
|
if (wait_for_heatup) {
|
|
|
|
wait_for_heatup = false;
|
|
|
|
ui.reset_status();
|
|
|
|
return true;
|
|
|
|
}
|
2018-09-29 01:44:47 -05:00
|
|
|
|
2020-09-12 05:51:19 +02:00
|
|
|
return false;
|
2018-09-29 01:44:47 -05:00
|
|
|
}
|
|
|
|
|
2020-03-04 12:50:43 -06:00
|
|
|
void Temperature::wait_for_bed_heating() {
|
|
|
|
if (isHeatingBed()) {
|
|
|
|
SERIAL_ECHOLNPGM("Wait for bed heating...");
|
2021-09-25 17:05:11 -05:00
|
|
|
LCD_MESSAGE(MSG_BED_HEATING);
|
2020-03-04 12:50:43 -06:00
|
|
|
wait_for_bed();
|
|
|
|
ui.reset_status();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-29 01:44:47 -05:00
|
|
|
#endif // HAS_HEATED_BED
|
|
|
|
|
2020-09-12 05:51:19 +02:00
|
|
|
#if HAS_TEMP_PROBE
|
|
|
|
|
|
|
|
#ifndef MIN_DELTA_SLOPE_DEG_PROBE
|
|
|
|
#define MIN_DELTA_SLOPE_DEG_PROBE 1.0
|
|
|
|
#endif
|
|
|
|
#ifndef MIN_DELTA_SLOPE_TIME_PROBE
|
|
|
|
#define MIN_DELTA_SLOPE_TIME_PROBE 600
|
|
|
|
#endif
|
|
|
|
|
2021-04-23 20:19:23 -05:00
|
|
|
bool Temperature::wait_for_probe(const celsius_t target_temp, bool no_wait_for_cooling/*=true*/) {
|
2020-09-12 05:51:19 +02:00
|
|
|
|
2021-04-23 20:19:23 -05:00
|
|
|
const bool wants_to_cool = isProbeAboveTemp(target_temp),
|
|
|
|
will_wait = !(wants_to_cool && no_wait_for_cooling);
|
2020-09-12 05:51:19 +02:00
|
|
|
if (will_wait)
|
2021-09-28 20:15:52 -05:00
|
|
|
SERIAL_ECHOLNPGM("Waiting for probe to ", wants_to_cool ? F("cool down") : F("heat up"), " to ", target_temp, " degrees.");
|
2020-09-12 05:51:19 +02:00
|
|
|
|
|
|
|
#if DISABLED(BUSY_WHILE_HEATING) && ENABLED(HOST_KEEPALIVE_FEATURE)
|
|
|
|
KEEPALIVE_STATE(NOT_BUSY);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
float old_temp = 9999;
|
|
|
|
millis_t next_temp_ms = 0, next_delta_check_ms = 0;
|
|
|
|
wait_for_heatup = true;
|
|
|
|
while (will_wait && wait_for_heatup) {
|
|
|
|
|
|
|
|
// Print Temp Reading every 10 seconds while heating up.
|
|
|
|
millis_t now = millis();
|
|
|
|
if (!next_temp_ms || ELAPSED(now, next_temp_ms)) {
|
|
|
|
next_temp_ms = now + 10000UL;
|
|
|
|
print_heater_states(active_extruder);
|
|
|
|
SERIAL_EOL();
|
|
|
|
}
|
|
|
|
|
|
|
|
idle();
|
|
|
|
gcode.reset_stepper_timeout(); // Keep steppers powered
|
|
|
|
|
|
|
|
// Break after MIN_DELTA_SLOPE_TIME_PROBE seconds if the temperature
|
|
|
|
// did not drop at least MIN_DELTA_SLOPE_DEG_PROBE. This avoids waiting
|
|
|
|
// forever as the probe is not actively heated.
|
|
|
|
if (!next_delta_check_ms || ELAPSED(now, next_delta_check_ms)) {
|
|
|
|
const float temp = degProbe(),
|
|
|
|
delta_temp = old_temp > temp ? old_temp - temp : temp - old_temp;
|
|
|
|
if (delta_temp < float(MIN_DELTA_SLOPE_DEG_PROBE)) {
|
|
|
|
SERIAL_ECHOLNPGM("Timed out waiting for probe temperature.");
|
|
|
|
break;
|
|
|
|
}
|
2021-01-28 20:40:20 -06:00
|
|
|
next_delta_check_ms = now + SEC_TO_MS(MIN_DELTA_SLOPE_TIME_PROBE);
|
2020-09-12 05:51:19 +02:00
|
|
|
old_temp = temp;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Loop until the temperature is very close target
|
|
|
|
if (!(wants_to_cool ? isProbeAboveTemp(target_temp) : isProbeBelowTemp(target_temp))) {
|
|
|
|
SERIAL_ECHOLN(wants_to_cool ? PSTR("Cooldown") : PSTR("Heatup"));
|
|
|
|
SERIAL_ECHOLNPGM(" complete, target probe temperature reached.");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wait_for_heatup) {
|
|
|
|
wait_for_heatup = false;
|
|
|
|
ui.reset_status();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else if (will_wait)
|
|
|
|
SERIAL_ECHOLNPGM("Canceled wait for probe temperature.");
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // HAS_TEMP_PROBE
|
|
|
|
|
2019-10-16 22:02:37 +02:00
|
|
|
#if HAS_HEATED_CHAMBER
|
2019-05-05 05:51:47 +02:00
|
|
|
|
|
|
|
#ifndef MIN_COOLING_SLOPE_DEG_CHAMBER
|
|
|
|
#define MIN_COOLING_SLOPE_DEG_CHAMBER 1.50
|
|
|
|
#endif
|
|
|
|
#ifndef MIN_COOLING_SLOPE_TIME_CHAMBER
|
2020-10-07 01:36:01 +02:00
|
|
|
#define MIN_COOLING_SLOPE_TIME_CHAMBER 120
|
2019-05-05 05:51:47 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
bool Temperature::wait_for_chamber(const bool no_wait_for_cooling/*=true*/) {
|
|
|
|
#if TEMP_CHAMBER_RESIDENCY_TIME > 0
|
|
|
|
millis_t residency_start_ms = 0;
|
2020-01-14 23:55:26 -08:00
|
|
|
bool first_loop = true;
|
2019-05-05 05:51:47 +02:00
|
|
|
// Loop until the temperature has stabilized
|
2020-04-03 19:49:45 -05:00
|
|
|
#define TEMP_CHAMBER_CONDITIONS (!residency_start_ms || PENDING(now, residency_start_ms + SEC_TO_MS(TEMP_CHAMBER_RESIDENCY_TIME)))
|
2019-05-05 05:51:47 +02:00
|
|
|
#else
|
|
|
|
// Loop until the temperature is very close target
|
|
|
|
#define TEMP_CHAMBER_CONDITIONS (wants_to_cool ? isCoolingChamber() : isHeatingChamber())
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if DISABLED(BUSY_WHILE_HEATING) && ENABLED(HOST_KEEPALIVE_FEATURE)
|
|
|
|
KEEPALIVE_STATE(NOT_BUSY);
|
|
|
|
#endif
|
|
|
|
|
2020-09-12 05:51:19 +02:00
|
|
|
bool wants_to_cool = false;
|
|
|
|
float target_temp = -1, old_temp = 9999;
|
|
|
|
millis_t now, next_temp_ms = 0, next_cool_check_ms = 0;
|
|
|
|
wait_for_heatup = true;
|
2019-05-05 05:51:47 +02:00
|
|
|
do {
|
|
|
|
// Target temperature might be changed during the loop
|
|
|
|
if (target_temp != degTargetChamber()) {
|
|
|
|
wants_to_cool = isCoolingChamber();
|
|
|
|
target_temp = degTargetChamber();
|
|
|
|
|
|
|
|
// Exit if S<lower>, continue if S<higher>, R<lower>, or R<higher>
|
|
|
|
if (no_wait_for_cooling && wants_to_cool) break;
|
|
|
|
}
|
|
|
|
|
|
|
|
now = millis();
|
2020-07-07 01:52:21 +02:00
|
|
|
if (ELAPSED(now, next_temp_ms)) { // Print Temp Reading every 1 second while heating up.
|
2019-05-05 05:51:47 +02:00
|
|
|
next_temp_ms = now + 1000UL;
|
|
|
|
print_heater_states(active_extruder);
|
|
|
|
#if TEMP_CHAMBER_RESIDENCY_TIME > 0
|
|
|
|
SERIAL_ECHOPGM(" W:");
|
|
|
|
if (residency_start_ms)
|
2020-04-03 19:49:45 -05:00
|
|
|
SERIAL_ECHO(long((SEC_TO_MS(TEMP_CHAMBER_RESIDENCY_TIME) - (now - residency_start_ms)) / 1000UL));
|
2019-05-05 05:51:47 +02:00
|
|
|
else
|
|
|
|
SERIAL_CHAR('?');
|
|
|
|
#endif
|
|
|
|
SERIAL_EOL();
|
|
|
|
}
|
|
|
|
|
|
|
|
idle();
|
|
|
|
gcode.reset_stepper_timeout(); // Keep steppers powered
|
|
|
|
|
|
|
|
const float temp = degChamber();
|
|
|
|
|
|
|
|
#if TEMP_CHAMBER_RESIDENCY_TIME > 0
|
|
|
|
|
|
|
|
const float temp_diff = ABS(target_temp - temp);
|
|
|
|
|
|
|
|
if (!residency_start_ms) {
|
|
|
|
// Start the TEMP_CHAMBER_RESIDENCY_TIME timer when we reach target temp for the first time.
|
2020-07-07 01:52:21 +02:00
|
|
|
if (temp_diff < TEMP_CHAMBER_WINDOW)
|
|
|
|
residency_start_ms = now + (first_loop ? SEC_TO_MS(TEMP_CHAMBER_RESIDENCY_TIME) / 3 : 0);
|
2019-05-05 05:51:47 +02:00
|
|
|
}
|
|
|
|
else if (temp_diff > TEMP_CHAMBER_HYSTERESIS) {
|
|
|
|
// Restart the timer whenever the temperature falls outside the hysteresis.
|
|
|
|
residency_start_ms = now;
|
|
|
|
}
|
|
|
|
|
2020-01-14 23:55:26 -08:00
|
|
|
first_loop = false;
|
2019-05-05 05:51:47 +02:00
|
|
|
#endif // TEMP_CHAMBER_RESIDENCY_TIME > 0
|
|
|
|
|
|
|
|
// Prevent a wait-forever situation if R is misused i.e. M191 R0
|
|
|
|
if (wants_to_cool) {
|
|
|
|
// Break after MIN_COOLING_SLOPE_TIME_CHAMBER seconds
|
|
|
|
// if the temperature did not drop at least MIN_COOLING_SLOPE_DEG_CHAMBER
|
|
|
|
if (!next_cool_check_ms || ELAPSED(now, next_cool_check_ms)) {
|
|
|
|
if (old_temp - temp < float(MIN_COOLING_SLOPE_DEG_CHAMBER)) break;
|
2021-01-28 20:40:20 -06:00
|
|
|
next_cool_check_ms = now + SEC_TO_MS(MIN_COOLING_SLOPE_TIME_CHAMBER);
|
2019-05-05 05:51:47 +02:00
|
|
|
old_temp = temp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} while (wait_for_heatup && TEMP_CHAMBER_CONDITIONS);
|
|
|
|
|
2020-09-12 05:51:19 +02:00
|
|
|
if (wait_for_heatup) {
|
|
|
|
wait_for_heatup = false;
|
|
|
|
ui.reset_status();
|
|
|
|
return true;
|
|
|
|
}
|
2019-05-05 05:51:47 +02:00
|
|
|
|
2020-09-12 05:51:19 +02:00
|
|
|
return false;
|
2019-05-05 05:51:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif // HAS_HEATED_CHAMBER
|
|
|
|
|
2021-03-06 14:13:28 -06:00
|
|
|
#if HAS_COOLER
|
|
|
|
|
|
|
|
#ifndef MIN_COOLING_SLOPE_DEG_COOLER
|
|
|
|
#define MIN_COOLING_SLOPE_DEG_COOLER 1.50
|
|
|
|
#endif
|
|
|
|
#ifndef MIN_COOLING_SLOPE_TIME_COOLER
|
|
|
|
#define MIN_COOLING_SLOPE_TIME_COOLER 120
|
|
|
|
#endif
|
|
|
|
|
|
|
|
bool Temperature::wait_for_cooler(const bool no_wait_for_cooling/*=true*/) {
|
|
|
|
|
|
|
|
#if TEMP_COOLER_RESIDENCY_TIME > 0
|
|
|
|
millis_t residency_start_ms = 0;
|
|
|
|
bool first_loop = true;
|
|
|
|
// Loop until the temperature has stabilized
|
|
|
|
#define TEMP_COOLER_CONDITIONS (!residency_start_ms || PENDING(now, residency_start_ms + SEC_TO_MS(TEMP_COOLER_RESIDENCY_TIME)))
|
|
|
|
#else
|
|
|
|
// Loop until the temperature is very close target
|
|
|
|
#define TEMP_COOLER_CONDITIONS (wants_to_cool ? isLaserHeating() : isLaserCooling())
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if DISABLED(BUSY_WHILE_HEATING) && ENABLED(HOST_KEEPALIVE_FEATURE)
|
|
|
|
KEEPALIVE_STATE(NOT_BUSY);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
bool wants_to_cool = false;
|
|
|
|
float target_temp = -1, previous_temp = 9999;
|
|
|
|
millis_t now, next_temp_ms = 0, next_cooling_check_ms = 0;
|
|
|
|
wait_for_heatup = true;
|
|
|
|
do {
|
|
|
|
// Target temperature might be changed during the loop
|
|
|
|
if (target_temp != degTargetCooler()) {
|
|
|
|
wants_to_cool = isLaserHeating();
|
|
|
|
target_temp = degTargetCooler();
|
|
|
|
|
|
|
|
// Exit if S<lower>, continue if S<higher>, R<lower>, or R<higher>
|
|
|
|
if (no_wait_for_cooling && wants_to_cool) break;
|
|
|
|
}
|
|
|
|
|
|
|
|
now = millis();
|
|
|
|
if (ELAPSED(now, next_temp_ms)) { // Print Temp Reading every 1 second while heating up.
|
|
|
|
next_temp_ms = now + 1000UL;
|
|
|
|
print_heater_states(active_extruder);
|
|
|
|
#if TEMP_COOLER_RESIDENCY_TIME > 0
|
|
|
|
SERIAL_ECHOPGM(" W:");
|
|
|
|
if (residency_start_ms)
|
|
|
|
SERIAL_ECHO(long((SEC_TO_MS(TEMP_COOLER_RESIDENCY_TIME) - (now - residency_start_ms)) / 1000UL));
|
|
|
|
else
|
|
|
|
SERIAL_CHAR('?');
|
|
|
|
#endif
|
|
|
|
SERIAL_EOL();
|
|
|
|
}
|
|
|
|
|
|
|
|
idle();
|
|
|
|
gcode.reset_stepper_timeout(); // Keep steppers powered
|
|
|
|
|
2021-04-23 19:14:49 -05:00
|
|
|
const celsius_float_t current_temp = degCooler();
|
2021-03-06 14:13:28 -06:00
|
|
|
|
|
|
|
#if TEMP_COOLER_RESIDENCY_TIME > 0
|
|
|
|
|
2021-04-23 19:14:49 -05:00
|
|
|
const celsius_float_t temp_diff = ABS(target_temp - temp);
|
2021-03-06 14:13:28 -06:00
|
|
|
|
|
|
|
if (!residency_start_ms) {
|
|
|
|
// Start the TEMP_COOLER_RESIDENCY_TIME timer when we reach target temp for the first time.
|
|
|
|
if (temp_diff < TEMP_COOLER_WINDOW)
|
|
|
|
residency_start_ms = now + (first_loop ? SEC_TO_MS(TEMP_COOLER_RESIDENCY_TIME) / 3 : 0);
|
|
|
|
}
|
|
|
|
else if (temp_diff > TEMP_COOLER_HYSTERESIS) {
|
|
|
|
// Restart the timer whenever the temperature falls outside the hysteresis.
|
|
|
|
residency_start_ms = now;
|
|
|
|
}
|
|
|
|
|
|
|
|
first_loop = false;
|
|
|
|
#endif // TEMP_COOLER_RESIDENCY_TIME > 0
|
|
|
|
|
|
|
|
if (wants_to_cool) {
|
|
|
|
// Break after MIN_COOLING_SLOPE_TIME_CHAMBER seconds
|
|
|
|
// if the temperature did not drop at least MIN_COOLING_SLOPE_DEG_CHAMBER
|
|
|
|
if (!next_cooling_check_ms || ELAPSED(now, next_cooling_check_ms)) {
|
|
|
|
if (previous_temp - current_temp < float(MIN_COOLING_SLOPE_DEG_COOLER)) break;
|
|
|
|
next_cooling_check_ms = now + SEC_TO_MS(MIN_COOLING_SLOPE_TIME_COOLER);
|
|
|
|
previous_temp = current_temp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} while (wait_for_heatup && TEMP_COOLER_CONDITIONS);
|
|
|
|
|
|
|
|
// Prevent a wait-forever situation if R is misused i.e. M191 R0
|
|
|
|
if (wait_for_heatup) {
|
|
|
|
wait_for_heatup = false;
|
|
|
|
ui.reset_status();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // HAS_COOLER
|
|
|
|
|
2018-03-07 01:53:50 -06:00
|
|
|
#endif // HAS_TEMP_SENSOR
|