Filament Runout handling for Mixing Extruder (#20327)
Co-authored-by: Scott Lahteine <thinkyhead@users.noreply.github.com>
This commit is contained in:
		| @@ -47,12 +47,12 @@ bool FilamentMonitorBase::enabled = true, | ||||
|  | ||||
| #if HAS_FILAMENT_RUNOUT_DISTANCE | ||||
|   float RunoutResponseDelayed::runout_distance_mm = FILAMENT_RUNOUT_DISTANCE_MM; | ||||
|   volatile float RunoutResponseDelayed::runout_mm_countdown[EXTRUDERS]; | ||||
|   volatile float RunoutResponseDelayed::runout_mm_countdown[NUM_RUNOUT_SENSORS]; | ||||
|   #if ENABLED(FILAMENT_MOTION_SENSOR) | ||||
|     uint8_t FilamentSensorEncoder::motion_detected; | ||||
|   #endif | ||||
| #else | ||||
|   int8_t RunoutResponseDebounced::runout_count; // = 0 | ||||
|   int8_t RunoutResponseDebounced::runout_count[NUM_RUNOUT_SENSORS]; // = 0 | ||||
| #endif | ||||
|  | ||||
| // | ||||
| @@ -70,7 +70,7 @@ bool FilamentMonitorBase::enabled = true, | ||||
|   #include "../lcd/extui/ui_api.h" | ||||
| #endif | ||||
|  | ||||
| void event_filament_runout() { | ||||
| void event_filament_runout(const uint8_t extruder) { | ||||
|  | ||||
|   if (did_pause_print) return;  // Action already in progress. Purge triggered repeated runout. | ||||
|  | ||||
| @@ -85,10 +85,10 @@ void event_filament_runout() { | ||||
|     } | ||||
|   #endif | ||||
|  | ||||
|   TERN_(EXTENSIBLE_UI, ExtUI::onFilamentRunout(ExtUI::getActiveTool())); | ||||
|   TERN_(EXTENSIBLE_UI, ExtUI::onFilamentRunout(ExtUI::getTool(extruder))); | ||||
|  | ||||
|   #if EITHER(HOST_PROMPT_SUPPORT, HOST_ACTION_COMMANDS) | ||||
|     const char tool = '0' + TERN0(MULTI_FILAMENT_SENSOR, active_extruder); | ||||
|   #if ANY(HOST_PROMPT_SUPPORT, HOST_ACTION_COMMANDS, MULTI_FILAMENT_SENSOR) | ||||
|     const char tool = '0' + TERN0(MULTI_FILAMENT_SENSOR, extruder); | ||||
|   #endif | ||||
|  | ||||
|   //action:out_of_filament | ||||
| @@ -124,8 +124,22 @@ void event_filament_runout() { | ||||
|     SERIAL_EOL(); | ||||
|   #endif // HOST_ACTION_COMMANDS | ||||
|  | ||||
|   if (run_runout_script) | ||||
|     queue.inject_P(PSTR(FILAMENT_RUNOUT_SCRIPT)); | ||||
|   if (run_runout_script) { | ||||
|     #if MULTI_FILAMENT_SENSOR | ||||
|       char script[strlen(FILAMENT_RUNOUT_SCRIPT) + 1]; | ||||
|       sprintf_P(script, PSTR(FILAMENT_RUNOUT_SCRIPT), tool); | ||||
|       #if ENABLED(FILAMENT_RUNOUT_SENSOR_DEBUG) | ||||
|         SERIAL_ECHOLNPAIR("Runout Command: ", script); | ||||
|       #endif | ||||
|       queue.inject(script); | ||||
|     #else | ||||
|       #if ENABLED(FILAMENT_RUNOUT_SENSOR_DEBUG) | ||||
|         SERIAL_ECHOPGM("Runout Command: "); | ||||
|         SERIAL_ECHOLNPGM(FILAMENT_RUNOUT_SCRIPT); | ||||
|       #endif | ||||
|       queue.inject_P(PSTR(FILAMENT_RUNOUT_SCRIPT)); | ||||
|     #endif | ||||
|   } | ||||
| } | ||||
|  | ||||
| #endif // HAS_FILAMENT_SENSOR | ||||
|   | ||||
| @@ -43,7 +43,7 @@ | ||||
|   #define FILAMENT_RUNOUT_THRESHOLD 5 | ||||
| #endif | ||||
|  | ||||
| void event_filament_runout(); | ||||
| void event_filament_runout(const uint8_t extruder); | ||||
|  | ||||
| template<class RESPONSE_T, class SENSOR_T> | ||||
| class TFilamentMonitor; | ||||
| @@ -119,11 +119,41 @@ class TFilamentMonitor : public FilamentMonitorBase { | ||||
|         TERN_(HAS_FILAMENT_RUNOUT_DISTANCE, cli()); // Prevent RunoutResponseDelayed::block_completed from accumulating here | ||||
|         response.run(); | ||||
|         sensor.run(); | ||||
|         const bool ran_out = response.has_run_out(); | ||||
|         const uint8_t runout_flags = response.has_run_out(); | ||||
|         TERN_(HAS_FILAMENT_RUNOUT_DISTANCE, sei()); | ||||
|         #if MULTI_FILAMENT_SENSOR | ||||
|           #if ENABLED(WATCH_ALL_RUNOUT_SENSORS) | ||||
|             const bool ran_out = !!runout_flags;  // any sensor triggers | ||||
|             uint8_t extruder = 0; | ||||
|             if (ran_out) { | ||||
|               uint8_t bitmask = runout_flags; | ||||
|               while (!(bitmask & 1)) { | ||||
|                 bitmask >>= 1; | ||||
|                 extruder++; | ||||
|               } | ||||
|             } | ||||
|           #else | ||||
|             const bool ran_out = TEST(runout_flags, active_extruder);  // suppress non active extruders | ||||
|             uint8_t extruder = active_extruder; | ||||
|           #endif | ||||
|         #else | ||||
|           const bool ran_out = !!runout_flags; | ||||
|           uint8_t extruder = active_extruder; | ||||
|         #endif | ||||
|  | ||||
|         #if ENABLED(FILAMENT_RUNOUT_SENSOR_DEBUG) | ||||
|           if (runout_flags) { | ||||
|             SERIAL_ECHOPGM("Runout Sensors: "); | ||||
|             LOOP_L_N(i, 8) SERIAL_ECHO('0' + TEST(runout_flags, i)); | ||||
|             SERIAL_ECHOPAIR(" -> ", extruder); | ||||
|             if (ran_out) SERIAL_ECHOPGM(" RUN OUT"); | ||||
|             SERIAL_EOL(); | ||||
|           } | ||||
|         #endif | ||||
|  | ||||
|         if (ran_out) { | ||||
|           filament_ran_out = true; | ||||
|           event_filament_runout(); | ||||
|           event_filament_runout(extruder); | ||||
|           planner.synchronize(); | ||||
|         } | ||||
|       } | ||||
| @@ -280,16 +310,17 @@ class FilamentSensorBase { | ||||
|       static inline void block_completed(const block_t* const) {} | ||||
|  | ||||
|       static inline void run() { | ||||
|         const bool out = poll_runout_state(active_extruder); | ||||
|         if (!out) filament_present(active_extruder); | ||||
|         #if ENABLED(FILAMENT_RUNOUT_SENSOR_DEBUG) | ||||
|           static bool was_out = false; | ||||
|           if (out != was_out) { | ||||
|             was_out = out; | ||||
|             SERIAL_ECHOPGM("Filament "); | ||||
|             SERIAL_ECHOPGM_P(out ? PSTR("OUT\n") : PSTR("IN\n")); | ||||
|           } | ||||
|         #endif | ||||
|         LOOP_L_N(s, NUM_RUNOUT_SENSORS) { | ||||
|           const bool out = poll_runout_state(s); | ||||
|           if (!out) filament_present(s); | ||||
|           #if ENABLED(FILAMENT_RUNOUT_SENSOR_DEBUG) | ||||
|             static uint8_t was_out; // = 0 | ||||
|             if (out != TEST(was_out, s)) { | ||||
|               TBI(was_out, s); | ||||
|               SERIAL_ECHOLNPAIR_P(PSTR("Filament Sensor "), '0' + s, out ? PSTR(" OUT") : PSTR(" IN")); | ||||
|             } | ||||
|           #endif | ||||
|         } | ||||
|       } | ||||
|   }; | ||||
|  | ||||
| @@ -305,13 +336,13 @@ class FilamentSensorBase { | ||||
|   // during a runout condition. | ||||
|   class RunoutResponseDelayed { | ||||
|     private: | ||||
|       static volatile float runout_mm_countdown[EXTRUDERS]; | ||||
|       static volatile float runout_mm_countdown[NUM_RUNOUT_SENSORS]; | ||||
|  | ||||
|     public: | ||||
|       static float runout_distance_mm; | ||||
|  | ||||
|       static inline void reset() { | ||||
|         LOOP_L_N(i, EXTRUDERS) filament_present(i); | ||||
|         LOOP_L_N(i, NUM_RUNOUT_SENSORS) filament_present(i); | ||||
|       } | ||||
|  | ||||
|       static inline void run() { | ||||
| @@ -320,15 +351,17 @@ class FilamentSensorBase { | ||||
|           const millis_t ms = millis(); | ||||
|           if (ELAPSED(ms, t)) { | ||||
|             t = millis() + 1000UL; | ||||
|             LOOP_L_N(i, EXTRUDERS) | ||||
|             LOOP_L_N(i, NUM_RUNOUT_SENSORS) | ||||
|               SERIAL_ECHOPAIR_P(i ? PSTR(", ") : PSTR("Remaining mm: "), runout_mm_countdown[i]); | ||||
|             SERIAL_EOL(); | ||||
|           } | ||||
|         #endif | ||||
|       } | ||||
|  | ||||
|       static inline bool has_run_out() { | ||||
|         return runout_mm_countdown[active_extruder] < 0; | ||||
|       static inline uint8_t has_run_out() { | ||||
|         uint8_t runout_flags = 0; | ||||
|         LOOP_L_N(i, NUM_RUNOUT_SENSORS) if (runout_mm_countdown[i] < 0) SBI(runout_flags, i); | ||||
|         return runout_flags; | ||||
|       } | ||||
|  | ||||
|       static inline void filament_present(const uint8_t extruder) { | ||||
| @@ -353,13 +386,28 @@ class FilamentSensorBase { | ||||
|   class RunoutResponseDebounced { | ||||
|     private: | ||||
|       static constexpr int8_t runout_threshold = FILAMENT_RUNOUT_THRESHOLD; | ||||
|       static int8_t runout_count; | ||||
|       static int8_t runout_count[NUM_RUNOUT_SENSORS]; | ||||
|  | ||||
|     public: | ||||
|       static inline void reset()                                  { runout_count = runout_threshold; } | ||||
|       static inline void run()                                    { if (runout_count >= 0) runout_count--; } | ||||
|       static inline bool has_run_out()                            { return runout_count < 0; } | ||||
|       static inline void block_completed(const block_t* const)    { } | ||||
|       static inline void filament_present(const uint8_t)          { runout_count = runout_threshold; } | ||||
|       static inline void reset() { | ||||
|         LOOP_L_N(i, NUM_RUNOUT_SENSORS) filament_present(i); | ||||
|       } | ||||
|  | ||||
|       static inline void run() { | ||||
|         LOOP_L_N(i, NUM_RUNOUT_SENSORS) if (runout_count[i] >= 0) runout_count[i]--; | ||||
|       } | ||||
|  | ||||
|       static inline uint8_t has_run_out() { | ||||
|         uint8_t runout_flags = 0; | ||||
|         LOOP_L_N(i, NUM_RUNOUT_SENSORS) if (runout_count[i] < 0) SBI(runout_flags, i); | ||||
|         return runout_flags; | ||||
|       } | ||||
|  | ||||
|       static inline void block_completed(const block_t* const) { } | ||||
|  | ||||
|       static inline void filament_present(const uint8_t extruder) { | ||||
|         runout_count[extruder] = runout_threshold; | ||||
|       } | ||||
|   }; | ||||
|  | ||||
| #endif // !HAS_FILAMENT_RUNOUT_DISTANCE | ||||
|   | ||||
		Reference in New Issue
	
	Block a user