Filament sensor refactoring initial
This commit is contained in:
parent
ce0d8f2f12
commit
45e43137a5
|
|
@ -0,0 +1,3 @@
|
||||||
|
#include "Filament_sensor.h"
|
||||||
|
|
||||||
|
IR_sensor_analog fsensor;
|
||||||
|
|
@ -0,0 +1,163 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <inttypes.h>
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <avr/pgmspace.h>
|
||||||
|
|
||||||
|
#include "Marlin.h"
|
||||||
|
#include "ultralcd.h"
|
||||||
|
#include "menu.h"
|
||||||
|
#include "cardreader.h"
|
||||||
|
#include "temperature.h"
|
||||||
|
#include "cmdqueue.h"
|
||||||
|
#include "eeprom.h"
|
||||||
|
#include "pins.h"
|
||||||
|
#include "fastio.h"
|
||||||
|
|
||||||
|
class Filament_sensor {
|
||||||
|
public:
|
||||||
|
virtual void init() = 0;
|
||||||
|
virtual void update() = 0;
|
||||||
|
virtual bool getFilamentPresent() = 0;
|
||||||
|
|
||||||
|
enum class SensorActionOnError : uint8_t {
|
||||||
|
_Continue = 0,
|
||||||
|
_Pause = 1,
|
||||||
|
_Undef = EEPROM_EMPTY_VALUE
|
||||||
|
};
|
||||||
|
|
||||||
|
void setAutoLoadEnabled(bool state, bool updateEEPROM = false) {
|
||||||
|
autoLoadEnabled = state;
|
||||||
|
if (updateEEPROM) {
|
||||||
|
eeprom_update_byte((uint8_t *)EEPROM_FSENS_AUTOLOAD_ENABLED, state);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void setRunoutEnabled(bool state, bool updateEEPROM = false) {
|
||||||
|
runoutEnabled = state;
|
||||||
|
if (updateEEPROM) {
|
||||||
|
eeprom_update_byte((uint8_t *)EEPROM_FSENSOR, state);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void settings_init() {
|
||||||
|
autoLoadEnabled = eeprom_read_byte((uint8_t*)EEPROM_FSENS_AUTOLOAD_ENABLED);
|
||||||
|
runoutEnabled = eeprom_read_byte((uint8_t*)EEPROM_FSENSOR);
|
||||||
|
sensorActionOnError = (SensorActionOnError)eeprom_read_byte((uint8_t*)EEPROM_FSENSOR_ACTION_NA);
|
||||||
|
if (sensorActionOnError == SensorActionOnError::_Undef) {
|
||||||
|
sensorActionOnError = SensorActionOnError::_Pause;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void checkFilamentEvents() {
|
||||||
|
if (!ready)
|
||||||
|
return;
|
||||||
|
|
||||||
|
bool newFilamentPresent = getFilamentPresent();
|
||||||
|
if (oldFilamentPresent != newFilamentPresent) {
|
||||||
|
oldFilamentPresent = newFilamentPresent;
|
||||||
|
if (newFilamentPresent) { //filament insertion
|
||||||
|
puts_P(PSTR("filament inserted"));
|
||||||
|
triggerFilamentInserted();
|
||||||
|
}
|
||||||
|
else { //filament removal
|
||||||
|
puts_P(PSTR("filament removed"));
|
||||||
|
triggerFilamentRemoved();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
void triggerFilamentInserted() {
|
||||||
|
if (autoLoadEnabled/* && (eFilamentAction == FilamentAction::None) */ && !(moves_planned() || IS_SD_PRINTING || usb_timer.running() || (lcd_commands_type == LcdCommands::Layer1Cal) || eeprom_read_byte((uint8_t*)EEPROM_WIZARD_ACTIVE))) {
|
||||||
|
eFilamentAction = FilamentAction::AutoLoad;
|
||||||
|
if(target_temperature[0] >= EXTRUDE_MINTEMP){
|
||||||
|
bFilamentPreheatState = true;
|
||||||
|
menu_submenu(mFilamentItemForce);
|
||||||
|
} else {
|
||||||
|
menu_submenu(lcd_generic_preheat_menu);
|
||||||
|
lcd_timeoutToStatus.start();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void triggerFilamentRemoved() {
|
||||||
|
if (runoutEnabled/* && (eFilamentAction == FilamentAction::None) */ && !saved_printing && (moves_planned() || IS_SD_PRINTING || usb_timer.running() || (lcd_commands_type == LcdCommands::Layer1Cal) || eeprom_read_byte((uint8_t*)EEPROM_WIZARD_ACTIVE))) {
|
||||||
|
runoutEnabled = false;
|
||||||
|
autoLoadEnabled = false;
|
||||||
|
stop_and_save_print_to_ram(0, 0);
|
||||||
|
restore_print_from_ram_and_continue(0);
|
||||||
|
eeprom_update_byte((uint8_t*)EEPROM_FERROR_COUNT, eeprom_read_byte((uint8_t*)EEPROM_FERROR_COUNT) + 1);
|
||||||
|
eeprom_update_word((uint16_t*)EEPROM_FERROR_COUNT_TOT, eeprom_read_word((uint16_t*)EEPROM_FERROR_COUNT_TOT) + 1);
|
||||||
|
enquecommand_front_P((PSTR("M600")));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool autoLoadEnabled;
|
||||||
|
bool runoutEnabled;
|
||||||
|
bool oldFilamentPresent; //for creating filament presence switching events.
|
||||||
|
bool ready;
|
||||||
|
SensorActionOnError sensorActionOnError;
|
||||||
|
};
|
||||||
|
|
||||||
|
class IR_sensor: public Filament_sensor {
|
||||||
|
public:
|
||||||
|
void init() {
|
||||||
|
SET_INPUT(IR_SENSOR_PIN); //input mode
|
||||||
|
WRITE(IR_SENSOR_PIN, 1); //pullup
|
||||||
|
settings_init();
|
||||||
|
}
|
||||||
|
|
||||||
|
void update() {
|
||||||
|
if (!ready) {
|
||||||
|
ready = true; //the IR sensor gets ready instantly as it's just a gpio read operation.
|
||||||
|
oldFilamentPresent = getFilamentPresent(); //initialize the current filament state so that we don't create a switching event right after the sensor is ready.
|
||||||
|
}
|
||||||
|
checkFilamentEvents();
|
||||||
|
;//
|
||||||
|
}
|
||||||
|
|
||||||
|
bool getFilamentPresent() {
|
||||||
|
return !READ(IR_SENSOR_PIN);
|
||||||
|
}
|
||||||
|
|
||||||
|
void settings_init() {
|
||||||
|
Filament_sensor::settings_init();
|
||||||
|
}
|
||||||
|
protected:
|
||||||
|
};
|
||||||
|
|
||||||
|
class IR_sensor_analog: public IR_sensor {
|
||||||
|
public:
|
||||||
|
void init() {
|
||||||
|
IR_sensor::init();
|
||||||
|
;//
|
||||||
|
}
|
||||||
|
|
||||||
|
void update() {
|
||||||
|
IR_sensor::update();
|
||||||
|
;//
|
||||||
|
}
|
||||||
|
|
||||||
|
void voltUpdate(uint16_t raw) { //to be called from the ADC ISR when a cycle is finished
|
||||||
|
voltRaw = raw;
|
||||||
|
voltReady = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void settings_init() {
|
||||||
|
IR_sensor::settings_init();
|
||||||
|
sensorRevision = (SensorRevision)eeprom_read_byte((uint8_t*)EEPROM_FSENSOR_PCB);
|
||||||
|
}
|
||||||
|
|
||||||
|
enum class SensorRevision : uint8_t {
|
||||||
|
_Old = 0,
|
||||||
|
_Rev04 = 1,
|
||||||
|
_Undef = EEPROM_EMPTY_VALUE
|
||||||
|
};
|
||||||
|
private:
|
||||||
|
SensorRevision sensorRevision;
|
||||||
|
bool voltReady; //this gets set by the adc ISR
|
||||||
|
uint16_t voltRaw;
|
||||||
|
};
|
||||||
|
|
||||||
|
extern IR_sensor_analog fsensor;
|
||||||
|
|
@ -96,6 +96,7 @@
|
||||||
#include "spi.h"
|
#include "spi.h"
|
||||||
|
|
||||||
#ifdef FILAMENT_SENSOR
|
#ifdef FILAMENT_SENSOR
|
||||||
|
#include "Filament_sensor.h"
|
||||||
#include "fsensor.h"
|
#include "fsensor.h"
|
||||||
#ifdef IR_SENSOR
|
#ifdef IR_SENSOR
|
||||||
#include "pat9125.h" // for pat9125_probe
|
#include "pat9125.h" // for pat9125_probe
|
||||||
|
|
@ -1365,6 +1366,7 @@ void setup()
|
||||||
xflash_err_msg();
|
xflash_err_msg();
|
||||||
|
|
||||||
#ifdef FILAMENT_SENSOR
|
#ifdef FILAMENT_SENSOR
|
||||||
|
fsensor.init();
|
||||||
fsensor_init();
|
fsensor_init();
|
||||||
#endif //FILAMENT_SENSOR
|
#endif //FILAMENT_SENSOR
|
||||||
|
|
||||||
|
|
@ -3608,6 +3610,10 @@ static void gcode_M600(bool automatic, float x_position, float y_position, float
|
||||||
fsensor_check_autoload();
|
fsensor_check_autoload();
|
||||||
#endif //IR_SENSOR
|
#endif //IR_SENSOR
|
||||||
|
|
||||||
|
#ifdef FILAMENT_SENSOR
|
||||||
|
fsensor.settings_init();
|
||||||
|
#endif
|
||||||
|
|
||||||
lcd_setstatuspgm(MSG_WELCOME);
|
lcd_setstatuspgm(MSG_WELCOME);
|
||||||
custom_message_type = CustomMsg::Status;
|
custom_message_type = CustomMsg::Status;
|
||||||
}
|
}
|
||||||
|
|
@ -3616,7 +3622,12 @@ void gcode_M701()
|
||||||
{
|
{
|
||||||
printf_P(PSTR("gcode_M701 begin\n"));
|
printf_P(PSTR("gcode_M701 begin\n"));
|
||||||
|
|
||||||
prusa_statistics(22);
|
#ifdef FILAMENT_SENSOR
|
||||||
|
fsensor.setRunoutEnabled(false); //suppress filament runouts while loading filament.
|
||||||
|
fsensor.setAutoLoadEnabled(false); //suppress filament autoloads while loading filament.
|
||||||
|
#endif
|
||||||
|
|
||||||
|
prusa_statistics(22);
|
||||||
|
|
||||||
if (mmu_enabled)
|
if (mmu_enabled)
|
||||||
{
|
{
|
||||||
|
|
@ -3672,6 +3683,10 @@ void gcode_M701()
|
||||||
}
|
}
|
||||||
#endif //FSENSOR_QUALITY
|
#endif //FSENSOR_QUALITY
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef FILAMENT_SENSOR
|
||||||
|
fsensor.settings_init(); //restore filament runout state.
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
/**
|
/**
|
||||||
* @brief Get serial number from 32U2 processor
|
* @brief Get serial number from 32U2 processor
|
||||||
|
|
@ -9426,105 +9441,9 @@ void manage_inactivity_IR_ANALOG_Check(uint16_t &nFSCheckCount, ClFsensorPCB isV
|
||||||
void manage_inactivity(bool ignore_stepper_queue/*=false*/) //default argument set in Marlin.h
|
void manage_inactivity(bool ignore_stepper_queue/*=false*/) //default argument set in Marlin.h
|
||||||
{
|
{
|
||||||
#ifdef FILAMENT_SENSOR
|
#ifdef FILAMENT_SENSOR
|
||||||
bool bInhibitFlag = false;
|
fsensor.update();
|
||||||
#ifdef IR_SENSOR_ANALOG
|
|
||||||
static uint16_t nFSCheckCount=0;
|
|
||||||
#endif // IR_SENSOR_ANALOG
|
#endif // IR_SENSOR_ANALOG
|
||||||
|
|
||||||
if (mmu_enabled == false)
|
|
||||||
{
|
|
||||||
//-// if (mcode_in_progress != 600) //M600 not in progress
|
|
||||||
if (!printer_active()) bInhibitFlag=(menu_menu==lcd_menu_show_sensors_state); //Block Filament sensor actions if PRINTER is not active and Support::SensorInfo menu active
|
|
||||||
#ifdef IR_SENSOR_ANALOG
|
|
||||||
bInhibitFlag=bInhibitFlag||bMenuFSDetect; // Block Filament sensor actions if Settings::HWsetup::FSdetect menu active
|
|
||||||
#endif // IR_SENSOR_ANALOG
|
|
||||||
if ((mcode_in_progress != 600) && (eFilamentAction != FilamentAction::AutoLoad) && (!bInhibitFlag) && (menu_menu != lcd_move_e)) //M600 not in progress, preHeat @ autoLoad menu not active
|
|
||||||
{
|
|
||||||
if (!moves_planned() && !IS_SD_PRINTING && !usb_timer.running() && (lcd_commands_type != LcdCommands::Layer1Cal) && ! eeprom_read_byte((uint8_t*)EEPROM_WIZARD_ACTIVE))
|
|
||||||
{
|
|
||||||
#ifdef IR_SENSOR_ANALOG
|
|
||||||
static uint16_t minVolt = Voltage2Raw(6.F), maxVolt = 0;
|
|
||||||
// detect min-max, some long term sliding window for filtration may be added
|
|
||||||
// avoiding floating point operations, thus computing in raw
|
|
||||||
if( current_voltage_raw_IR > maxVolt )maxVolt = current_voltage_raw_IR;
|
|
||||||
if( current_voltage_raw_IR < minVolt )minVolt = current_voltage_raw_IR;
|
|
||||||
|
|
||||||
#if 0 // Start: IR Sensor debug info
|
|
||||||
{ // debug print
|
|
||||||
static uint16_t lastVolt = ~0U;
|
|
||||||
if( current_voltage_raw_IR != lastVolt ){
|
|
||||||
printf_P(PSTR("fs volt=%4.2fV (min=%4.2f max=%4.2f)\n"), Raw2Voltage(current_voltage_raw_IR), Raw2Voltage(minVolt), Raw2Voltage(maxVolt) );
|
|
||||||
lastVolt = current_voltage_raw_IR;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif // End: IR Sensor debug info
|
|
||||||
//! The trouble is, I can hold the filament in the hole in such a way, that it creates the exact voltage
|
|
||||||
//! to be detected as the new fsensor
|
|
||||||
//! We can either fake it by extending the detection window to a looooong time
|
|
||||||
//! or do some other countermeasures
|
|
||||||
|
|
||||||
//! what we want to detect:
|
|
||||||
//! if minvolt gets below ~0.3V, it means there is an old fsensor
|
|
||||||
//! if maxvolt gets above 4.6V, it means we either have an old fsensor or broken cables/fsensor
|
|
||||||
//! So I'm waiting for a situation, when minVolt gets to range <0, 1.5> and maxVolt gets into range <3.0, 5>
|
|
||||||
//! If and only if minVolt is in range <0.3, 1.5> and maxVolt is in range <3.0, 4.6>, I'm considering a situation with the new fsensor
|
|
||||||
if( minVolt >= IRsensor_Ldiode_TRESHOLD && minVolt <= IRsensor_Lmax_TRESHOLD
|
|
||||||
&& maxVolt >= IRsensor_Hmin_TRESHOLD && maxVolt <= IRsensor_Hopen_TRESHOLD
|
|
||||||
){
|
|
||||||
manage_inactivity_IR_ANALOG_Check(nFSCheckCount, ClFsensorPCB::_Old, ClFsensorPCB::_Rev04, _i("FS v0.4 or newer") ); ////MSG_FS_V_04_OR_NEWER c=18
|
|
||||||
}
|
|
||||||
//! If and only if minVolt is in range <0.0, 0.3> and maxVolt is in range <4.6, 5.0V>, I'm considering a situation with the old fsensor
|
|
||||||
//! Note, we are not relying on one voltage here - getting just +5V can mean an old fsensor or a broken new sensor - that's why
|
|
||||||
//! we need to have both voltages detected correctly to allow switching back to the old fsensor.
|
|
||||||
else if( minVolt < IRsensor_Ldiode_TRESHOLD
|
|
||||||
&& maxVolt > IRsensor_Hopen_TRESHOLD && maxVolt <= IRsensor_VMax_TRESHOLD
|
|
||||||
){
|
|
||||||
manage_inactivity_IR_ANALOG_Check(nFSCheckCount, ClFsensorPCB::_Rev04, oFsensorPCB=ClFsensorPCB::_Old, _i("FS v0.3 or older")); ////MSG_FS_V_03_OR_OLDER c=18
|
|
||||||
}
|
|
||||||
#endif // IR_SENSOR_ANALOG
|
|
||||||
if (fsensor_check_autoload())
|
|
||||||
{
|
|
||||||
#ifdef PAT9125
|
|
||||||
fsensor_autoload_check_stop();
|
|
||||||
#endif //PAT9125
|
|
||||||
//-// if ((int)degHotend0() > extrude_min_temp)
|
|
||||||
if(0)
|
|
||||||
{
|
|
||||||
Sound_MakeCustom(50,1000,false);
|
|
||||||
loading_flag = true;
|
|
||||||
enquecommand_front_P((PSTR("M701")));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
/*
|
|
||||||
lcd_update_enable(false);
|
|
||||||
show_preheat_nozzle_warning();
|
|
||||||
lcd_update_enable(true);
|
|
||||||
*/
|
|
||||||
eFilamentAction=FilamentAction::AutoLoad;
|
|
||||||
if(target_temperature[0] >= extrude_min_temp){
|
|
||||||
bFilamentPreheatState=true;
|
|
||||||
// mFilamentItem(target_temperature[0],target_temperature_bed);
|
|
||||||
menu_submenu(mFilamentItemForce);
|
|
||||||
} else {
|
|
||||||
menu_submenu(lcd_generic_preheat_menu);
|
|
||||||
lcd_timeoutToStatus.start();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
#ifdef PAT9125
|
|
||||||
fsensor_autoload_check_stop();
|
|
||||||
#endif //PAT9125
|
|
||||||
if (fsensor_enabled && !saved_printing)
|
|
||||||
fsensor_update();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif //FILAMENT_SENSOR
|
|
||||||
|
|
||||||
#ifdef SAFETYTIMER
|
#ifdef SAFETYTIMER
|
||||||
handleSafetyTimer();
|
handleSafetyTimer();
|
||||||
#endif //SAFETYTIMER
|
#endif //SAFETYTIMER
|
||||||
|
|
|
||||||
|
|
@ -130,6 +130,8 @@ void menu_back_if_clicked_fb(void)
|
||||||
|
|
||||||
void menu_submenu(menu_func_t submenu)
|
void menu_submenu(menu_func_t submenu)
|
||||||
{
|
{
|
||||||
|
if (menu_menu == submenu)
|
||||||
|
return; //do not enter into the current menu.
|
||||||
if (menu_depth < MENU_DEPTH_MAX)
|
if (menu_depth < MENU_DEPTH_MAX)
|
||||||
{
|
{
|
||||||
menu_stack[menu_depth].menu = menu_menu;
|
menu_stack[menu_depth].menu = menu_menu;
|
||||||
|
|
|
||||||
|
|
@ -97,6 +97,8 @@
|
||||||
#include "temp_model.h"
|
#include "temp_model.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include "Filament_sensor.h"
|
||||||
|
|
||||||
//===========================================================================
|
//===========================================================================
|
||||||
//=============================public variables============================
|
//=============================public variables============================
|
||||||
//===========================================================================
|
//===========================================================================
|
||||||
|
|
|
||||||
|
|
@ -5186,6 +5186,7 @@ static void mmu_cut_filament_menu()
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
eFilamentAction=FilamentAction::MmuCut;
|
eFilamentAction=FilamentAction::MmuCut;
|
||||||
|
bFilamentFirstRun=false;
|
||||||
if(target_temperature[0] >= extrude_min_temp)
|
if(target_temperature[0] >= extrude_min_temp)
|
||||||
{
|
{
|
||||||
bFilamentPreheatState=true;
|
bFilamentPreheatState=true;
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue