⚡️ Handle shared enable pins (#22824)
This commit is contained in:
		
				
					committed by
					
						 Scott Lahteine
						Scott Lahteine
					
				
			
			
				
	
			
			
			
						parent
						
							25a131b942
						
					
				
				
					commit
					021ceeba0b
				
			| @@ -81,10 +81,6 @@ | |||||||
|   #endif |   #endif | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| #if ENABLED(EXTENSIBLE_UI) |  | ||||||
|   #include "lcd/extui/ui_api.h" |  | ||||||
| #endif |  | ||||||
|  |  | ||||||
| #if HAS_ETHERNET | #if HAS_ETHERNET | ||||||
|   #include "feature/ethernet.h" |   #include "feature/ethernet.h" | ||||||
| #endif | #endif | ||||||
| @@ -312,48 +308,6 @@ bool pin_is_protected(const pin_t pin) { | |||||||
|  |  | ||||||
| #pragma GCC diagnostic pop | #pragma GCC diagnostic pop | ||||||
|  |  | ||||||
| void enable_e_steppers() { |  | ||||||
|   #define _ENA_E(N) ENABLE_AXIS_E##N(); |  | ||||||
|   REPEAT(E_STEPPERS, _ENA_E) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void enable_all_steppers() { |  | ||||||
|   TERN_(AUTO_POWER_CONTROL, powerManager.power_on()); |  | ||||||
|   ENABLE_AXIS_X(); |  | ||||||
|   ENABLE_AXIS_Y(); |  | ||||||
|   ENABLE_AXIS_Z(); |  | ||||||
|   ENABLE_AXIS_I(); // Marlin 6-axis support by DerAndere (https://github.com/DerAndere1/Marlin/wiki) |  | ||||||
|   ENABLE_AXIS_J(); |  | ||||||
|   ENABLE_AXIS_K(); |  | ||||||
|   enable_e_steppers(); |  | ||||||
|  |  | ||||||
|   TERN_(EXTENSIBLE_UI, ExtUI::onSteppersEnabled()); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void disable_e_steppers() { |  | ||||||
|   #define _DIS_E(N) DISABLE_AXIS_E##N(); |  | ||||||
|   REPEAT(E_STEPPERS, _DIS_E) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void disable_e_stepper(const uint8_t e) { |  | ||||||
|   #define _CASE_DIS_E(N) case N: DISABLE_AXIS_E##N(); break; |  | ||||||
|   switch (e) { |  | ||||||
|     REPEAT(E_STEPPERS, _CASE_DIS_E) |  | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void disable_all_steppers() { |  | ||||||
|   DISABLE_AXIS_X(); |  | ||||||
|   DISABLE_AXIS_Y(); |  | ||||||
|   DISABLE_AXIS_Z(); |  | ||||||
|   DISABLE_AXIS_I(); |  | ||||||
|   DISABLE_AXIS_J(); |  | ||||||
|   DISABLE_AXIS_K(); |  | ||||||
|   disable_e_steppers(); |  | ||||||
|  |  | ||||||
|   TERN_(EXTENSIBLE_UI, ExtUI::onSteppersDisabled()); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * A Print Job exists when the timer is running or SD is printing |  * A Print Job exists when the timer is running or SD is printing | ||||||
|  */ |  */ | ||||||
| @@ -464,13 +418,13 @@ inline void manage_inactivity(const bool no_stepper_sleep=false) { | |||||||
|         already_shutdown_steppers = true;  // L6470 SPI will consume 99% of free time without this |         already_shutdown_steppers = true;  // L6470 SPI will consume 99% of free time without this | ||||||
|  |  | ||||||
|         // Individual axes will be disabled if configured |         // Individual axes will be disabled if configured | ||||||
|         if (ENABLED(DISABLE_INACTIVE_X)) DISABLE_AXIS_X(); |         TERN_(DISABLE_INACTIVE_X, stepper.disable_axis(X_AXIS)); | ||||||
|         if (ENABLED(DISABLE_INACTIVE_Y)) DISABLE_AXIS_Y(); |         TERN_(DISABLE_INACTIVE_Y, stepper.disable_axis(Y_AXIS)); | ||||||
|         if (ENABLED(DISABLE_INACTIVE_Z)) DISABLE_AXIS_Z(); |         TERN_(DISABLE_INACTIVE_Z, stepper.disable_axis(Z_AXIS)); | ||||||
|         if (ENABLED(DISABLE_INACTIVE_I)) DISABLE_AXIS_I(); |         TERN_(DISABLE_INACTIVE_I, stepper.disable_axis(I_AXIS)); | ||||||
|         if (ENABLED(DISABLE_INACTIVE_J)) DISABLE_AXIS_J(); |         TERN_(DISABLE_INACTIVE_J, stepper.disable_axis(J_AXIS)); | ||||||
|         if (ENABLED(DISABLE_INACTIVE_K)) DISABLE_AXIS_K(); |         TERN_(DISABLE_INACTIVE_K, stepper.disable_axis(K_AXIS)); | ||||||
|         if (ENABLED(DISABLE_INACTIVE_E)) disable_e_steppers(); |         TERN_(DISABLE_INACTIVE_E, stepper.disable_e_steppers()); | ||||||
|  |  | ||||||
|         TERN_(AUTO_BED_LEVELING_UBL, ubl.steppers_were_disabled()); |         TERN_(AUTO_BED_LEVELING_UBL, ubl.steppers_were_disabled()); | ||||||
|       } |       } | ||||||
| @@ -689,13 +643,13 @@ inline void manage_inactivity(const bool no_stepper_sleep=false) { | |||||||
|       #if ENABLED(SWITCHING_EXTRUDER) |       #if ENABLED(SWITCHING_EXTRUDER) | ||||||
|         bool oldstatus; |         bool oldstatus; | ||||||
|         switch (active_extruder) { |         switch (active_extruder) { | ||||||
|           default: oldstatus = E0_ENABLE_READ(); ENABLE_AXIS_E0(); break; |           default: oldstatus = stepper.AXIS_IS_ENABLED(E_AXIS, 0); stepper.ENABLE_EXTRUDER(0); break; | ||||||
|           #if E_STEPPERS > 1 |           #if E_STEPPERS > 1 | ||||||
|             case 2: case 3: oldstatus = E1_ENABLE_READ(); ENABLE_AXIS_E1(); break; |             case 2: case 3: oldstatus = stepper.AXIS_IS_ENABLED(E_AXIS, 1); stepper.ENABLE_EXTRUDER(1); break; | ||||||
|             #if E_STEPPERS > 2 |             #if E_STEPPERS > 2 | ||||||
|               case 4: case 5: oldstatus = E2_ENABLE_READ(); ENABLE_AXIS_E2(); break; |               case 4: case 5: oldstatus = stepper.AXIS_IS_ENABLED(E_AXIS, 2); stepper.ENABLE_EXTRUDER(2); break; | ||||||
|               #if E_STEPPERS > 3 |               #if E_STEPPERS > 3 | ||||||
|                 case 6: case 7: oldstatus = E3_ENABLE_READ(); ENABLE_AXIS_E3(); break; |                 case 6: case 7: oldstatus = stepper.AXIS_IS_ENABLED(E_AXIS, 3); stepper.ENABLE_EXTRUDER(3); break; | ||||||
|               #endif // E_STEPPERS > 3 |               #endif // E_STEPPERS > 3 | ||||||
|             #endif // E_STEPPERS > 2 |             #endif // E_STEPPERS > 2 | ||||||
|           #endif // E_STEPPERS > 1 |           #endif // E_STEPPERS > 1 | ||||||
| @@ -704,7 +658,7 @@ inline void manage_inactivity(const bool no_stepper_sleep=false) { | |||||||
|         bool oldstatus; |         bool oldstatus; | ||||||
|         switch (active_extruder) { |         switch (active_extruder) { | ||||||
|           default: |           default: | ||||||
|           #define _CASE_EN(N) case N: oldstatus = E##N##_ENABLE_READ(); ENABLE_AXIS_E##N(); break; |           #define _CASE_EN(N) case N: oldstatus = stepper.AXIS_IS_ENABLED(E_AXIS, N); stepper.ENABLE_EXTRUDER(N); break; | ||||||
|           REPEAT(E_STEPPERS, _CASE_EN); |           REPEAT(E_STEPPERS, _CASE_EN); | ||||||
|         } |         } | ||||||
|       #endif |       #endif | ||||||
| @@ -718,17 +672,17 @@ inline void manage_inactivity(const bool no_stepper_sleep=false) { | |||||||
|  |  | ||||||
|       #if ENABLED(SWITCHING_EXTRUDER) |       #if ENABLED(SWITCHING_EXTRUDER) | ||||||
|         switch (active_extruder) { |         switch (active_extruder) { | ||||||
|           default: oldstatus = E0_ENABLE_WRITE(oldstatus); break; |           default: if (oldstatus) stepper.ENABLE_EXTRUDER(0); else stepper.DISABLE_EXTRUDER(0); break; | ||||||
|           #if E_STEPPERS > 1 |           #if E_STEPPERS > 1 | ||||||
|             case 2: case 3: oldstatus = E1_ENABLE_WRITE(oldstatus); break; |             case 2: case 3: if (oldstatus) stepper.ENABLE_EXTRUDER(1); else stepper.DISABLE_EXTRUDER(1); break; | ||||||
|             #if E_STEPPERS > 2 |             #if E_STEPPERS > 2 | ||||||
|               case 4: case 5: oldstatus = E2_ENABLE_WRITE(oldstatus); break; |               case 4: case 5: if (oldstatus) stepper.ENABLE_EXTRUDER(2); else stepper.DISABLE_EXTRUDER(2); break; | ||||||
|             #endif // E_STEPPERS > 2 |             #endif // E_STEPPERS > 2 | ||||||
|           #endif // E_STEPPERS > 1 |           #endif // E_STEPPERS > 1 | ||||||
|         } |         } | ||||||
|       #else // !SWITCHING_EXTRUDER |       #else // !SWITCHING_EXTRUDER | ||||||
|         switch (active_extruder) { |         switch (active_extruder) { | ||||||
|           #define _CASE_RESTORE(N) case N: E##N##_ENABLE_WRITE(oldstatus); break; |           #define _CASE_RESTORE(N) case N: if (oldstatus) stepper.ENABLE_EXTRUDER(N); else stepper.DISABLE_EXTRUDER(N); break; | ||||||
|           REPEAT(E_STEPPERS, _CASE_RESTORE); |           REPEAT(E_STEPPERS, _CASE_RESTORE); | ||||||
|         } |         } | ||||||
|       #endif // !SWITCHING_EXTRUDER |       #endif // !SWITCHING_EXTRUDER | ||||||
| @@ -940,7 +894,7 @@ void minkill(const bool steppers_off/*=false*/) { | |||||||
|   TERN_(HAS_CUTTER, cutter.kill());  // Reiterate cutter shutdown |   TERN_(HAS_CUTTER, cutter.kill());  // Reiterate cutter shutdown | ||||||
|  |  | ||||||
|   // Power off all steppers (for M112) or just the E steppers |   // Power off all steppers (for M112) or just the E steppers | ||||||
|   steppers_off ? disable_all_steppers() : disable_e_steppers(); |   steppers_off ? stepper.disable_all_steppers() : stepper.disable_e_steppers(); | ||||||
|  |  | ||||||
|   TERN_(PSU_CONTROL, powerManager.power_off()); |   TERN_(PSU_CONTROL, powerManager.power_off()); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -38,15 +38,6 @@ inline void idle_no_sleep() { idle(true); } | |||||||
|   extern bool G38_did_trigger;      // Flag from the ISR to indicate the endstop changed |   extern bool G38_did_trigger;      // Flag from the ISR to indicate the endstop changed | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * The axis order in all axis related arrays is X, Y, Z, E |  | ||||||
|  */ |  | ||||||
| void enable_e_steppers(); |  | ||||||
| void enable_all_steppers(); |  | ||||||
| void disable_e_stepper(const uint8_t e); |  | ||||||
| void disable_e_steppers(); |  | ||||||
| void disable_all_steppers(); |  | ||||||
|  |  | ||||||
| void kill(PGM_P const lcd_error=nullptr, PGM_P const lcd_component=nullptr, const bool steppers_off=false); | void kill(PGM_P const lcd_error=nullptr, PGM_P const lcd_component=nullptr, const bool steppers_off=false); | ||||||
| void minkill(const bool steppers_off=false); | void minkill(const bool steppers_off=false); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -25,7 +25,7 @@ | |||||||
| #if ENABLED(USE_CONTROLLER_FAN) | #if ENABLED(USE_CONTROLLER_FAN) | ||||||
|  |  | ||||||
| #include "controllerfan.h" | #include "controllerfan.h" | ||||||
| #include "../module/stepper/indirection.h" | #include "../module/stepper.h" | ||||||
| #include "../module/temperature.h" | #include "../module/temperature.h" | ||||||
|  |  | ||||||
| ControllerFan controllerFan; | ControllerFan controllerFan; | ||||||
| @@ -54,33 +54,12 @@ void ControllerFan::update() { | |||||||
|   if (ELAPSED(ms, nextMotorCheck)) { |   if (ELAPSED(ms, nextMotorCheck)) { | ||||||
|     nextMotorCheck = ms + 2500UL; // Not a time critical function, so only check every 2.5s |     nextMotorCheck = ms + 2500UL; // Not a time critical function, so only check every 2.5s | ||||||
|  |  | ||||||
|     #define MOTOR_IS_ON(A,B) (A##_ENABLE_READ() == bool(B##_ENABLE_ON)) |  | ||||||
|     #define _OR_ENABLED_E(N) || MOTOR_IS_ON(E##N,E) |  | ||||||
|  |  | ||||||
|     const bool motor_on = ( |  | ||||||
|       ( DISABLED(CONTROLLER_FAN_IGNORE_Z) && |  | ||||||
|         (    MOTOR_IS_ON(Z,Z) |  | ||||||
|           || TERN0(HAS_Z2_ENABLE, MOTOR_IS_ON(Z2,Z)) |  | ||||||
|           || TERN0(HAS_Z3_ENABLE, MOTOR_IS_ON(Z3,Z)) |  | ||||||
|           || TERN0(HAS_Z4_ENABLE, MOTOR_IS_ON(Z4,Z)) |  | ||||||
|         ) |  | ||||||
|       ) || ( |  | ||||||
|         DISABLED(CONTROLLER_FAN_USE_Z_ONLY) && |  | ||||||
|         (    MOTOR_IS_ON(X,X) || MOTOR_IS_ON(Y,Y) |  | ||||||
|           || TERN0(HAS_X2_ENABLE, MOTOR_IS_ON(X2,X)) |  | ||||||
|           || TERN0(HAS_Y2_ENABLE, MOTOR_IS_ON(Y2,Y)) |  | ||||||
|           #if E_STEPPERS |  | ||||||
|             REPEAT(E_STEPPERS, _OR_ENABLED_E) |  | ||||||
|           #endif |  | ||||||
|         ) |  | ||||||
|       ) |  | ||||||
|     ); |  | ||||||
|  |  | ||||||
|     // If any triggers for the controller fan are true... |     // If any triggers for the controller fan are true... | ||||||
|     //   - At least one stepper driver is enabled |     //   - At least one stepper driver is enabled | ||||||
|     //   - The heated bed is enabled |     //   - The heated bed is enabled | ||||||
|     //   - TEMP_SENSOR_BOARD is reporting >= CONTROLLER_FAN_MIN_BOARD_TEMP |     //   - TEMP_SENSOR_BOARD is reporting >= CONTROLLER_FAN_MIN_BOARD_TEMP | ||||||
|     if ( motor_on |     const ena_mask_t axis_mask = TERN(CONTROLLER_FAN_USE_Z_ONLY, _BV(Z_AXIS), ~TERN0(CONTROLLER_FAN_IGNORE_Z, _BV(Z_AXIS))); | ||||||
|  |     if ( (stepper.axis_enabled.bits & axis_mask) | ||||||
|       || TERN0(HAS_HEATED_BED, thermalManager.temp_bed.soft_pwm_amount > 0) |       || TERN0(HAS_HEATED_BED, thermalManager.temp_bed.soft_pwm_amount > 0) | ||||||
|       || TERN0(HAS_CONTROLLER_FAN_MIN_BOARD_TEMP, thermalManager.wholeDegBoard() >= CONTROLLER_FAN_MIN_BOARD_TEMP) |       || TERN0(HAS_CONTROLLER_FAN_MIN_BOARD_TEMP, thermalManager.wholeDegBoard() >= CONTROLLER_FAN_MIN_BOARD_TEMP) | ||||||
|     ) lastMotorOn = ms; //... set time to NOW so the fan will turn on |     ) lastMotorOn = ms; //... set time to NOW so the fan will turn on | ||||||
|   | |||||||
| @@ -75,7 +75,7 @@ void FWRetract::reset() { | |||||||
|  |  | ||||||
|   LOOP_L_N(i, EXTRUDERS) { |   LOOP_L_N(i, EXTRUDERS) { | ||||||
|     retracted[i] = false; |     retracted[i] = false; | ||||||
|     TERN_(HAS_MULTI_EXTRUDER, retracted_swap[i] = false); |     E_TERN_(retracted_swap[i] = false); | ||||||
|     current_retract[i] = 0.0; |     current_retract[i] = 0.0; | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @@ -91,7 +91,7 @@ void FWRetract::reset() { | |||||||
|  * Note: Auto-retract will apply the set Z hop in addition to any Z hop |  * Note: Auto-retract will apply the set Z hop in addition to any Z hop | ||||||
|  *       included in the G-code. Use M207 Z0 to to prevent double hop. |  *       included in the G-code. Use M207 Z0 to to prevent double hop. | ||||||
|  */ |  */ | ||||||
| void FWRetract::retract(const bool retracting OPTARG(HAS_MULTI_EXTRUDER, bool swapping/*=false*/)) { | void FWRetract::retract(const bool retracting E_OPTARG(bool swapping/*=false*/)) { | ||||||
|   // Prevent two retracts or recovers in a row |   // Prevent two retracts or recovers in a row | ||||||
|   if (retracted[active_extruder] == retracting) return; |   if (retracted[active_extruder] == retracting) return; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -74,7 +74,7 @@ public: | |||||||
|     #endif |     #endif | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   static void retract(const bool retracting OPTARG(HAS_MULTI_EXTRUDER, bool swapping = false)); |   static void retract(const bool retracting E_OPTARG(bool swapping=false)); | ||||||
|  |  | ||||||
|   static void M207_report(); |   static void M207_report(); | ||||||
|   static void M207(); |   static void M207(); | ||||||
|   | |||||||
| @@ -26,6 +26,7 @@ | |||||||
|  |  | ||||||
| #include "../MarlinCore.h" | #include "../MarlinCore.h" | ||||||
| #include "../module/planner.h" | #include "../module/planner.h" | ||||||
|  | #include "../module/stepper.h" | ||||||
|  |  | ||||||
| void mmu_init() { | void mmu_init() { | ||||||
|   SET_OUTPUT(E_MUX0_PIN); |   SET_OUTPUT(E_MUX0_PIN); | ||||||
| @@ -35,7 +36,7 @@ void mmu_init() { | |||||||
|  |  | ||||||
| void select_multiplexed_stepper(const uint8_t e) { | void select_multiplexed_stepper(const uint8_t e) { | ||||||
|   planner.synchronize(); |   planner.synchronize(); | ||||||
|   disable_e_steppers(); |   stepper.disable_e_steppers(); | ||||||
|   WRITE(E_MUX0_PIN, TEST(e, 0) ? HIGH : LOW); |   WRITE(E_MUX0_PIN, TEST(e, 0) ? HIGH : LOW); | ||||||
|   WRITE(E_MUX1_PIN, TEST(e, 1) ? HIGH : LOW); |   WRITE(E_MUX1_PIN, TEST(e, 1) ? HIGH : LOW); | ||||||
|   WRITE(E_MUX2_PIN, TEST(e, 2) ? HIGH : LOW); |   WRITE(E_MUX2_PIN, TEST(e, 2) ? HIGH : LOW); | ||||||
|   | |||||||
| @@ -35,7 +35,7 @@ MMU2 mmu2; | |||||||
| #include "../../libs/nozzle.h" | #include "../../libs/nozzle.h" | ||||||
| #include "../../module/temperature.h" | #include "../../module/temperature.h" | ||||||
| #include "../../module/planner.h" | #include "../../module/planner.h" | ||||||
| #include "../../module/stepper/indirection.h" | #include "../../module/stepper.h" | ||||||
| #include "../../MarlinCore.h" | #include "../../MarlinCore.h" | ||||||
|  |  | ||||||
| #if ENABLED(HOST_PROMPT_SUPPORT) | #if ENABLED(HOST_PROMPT_SUPPORT) | ||||||
| @@ -486,7 +486,7 @@ static void mmu2_not_responding() { | |||||||
|  |  | ||||||
|     if (index != extruder) { |     if (index != extruder) { | ||||||
|  |  | ||||||
|       DISABLE_AXIS_E0(); |       stepper.disable_extruder(); | ||||||
|       ui.status_printf_P(0, GET_TEXT(MSG_MMU2_LOADING_FILAMENT), int(index + 1)); |       ui.status_printf_P(0, GET_TEXT(MSG_MMU2_LOADING_FILAMENT), int(index + 1)); | ||||||
|  |  | ||||||
|       command(MMU_CMD_T0 + index); |       command(MMU_CMD_T0 + index); | ||||||
| @@ -495,7 +495,7 @@ static void mmu2_not_responding() { | |||||||
|       if (load_to_gears()) { |       if (load_to_gears()) { | ||||||
|         extruder = index; // filament change is finished |         extruder = index; // filament change is finished | ||||||
|         active_extruder = 0; |         active_extruder = 0; | ||||||
|         ENABLE_AXIS_E0(); |         stepper.enable_extruder(); | ||||||
|         SERIAL_ECHO_MSG(STR_ACTIVE_EXTRUDER, extruder); |         SERIAL_ECHO_MSG(STR_ACTIVE_EXTRUDER, extruder); | ||||||
|       } |       } | ||||||
|       ui.reset_status(); |       ui.reset_status(); | ||||||
| @@ -531,13 +531,13 @@ static void mmu2_not_responding() { | |||||||
|           #if ENABLED(MMU2_MENUS) |           #if ENABLED(MMU2_MENUS) | ||||||
|             planner.synchronize(); |             planner.synchronize(); | ||||||
|             const uint8_t index = mmu2_choose_filament(); |             const uint8_t index = mmu2_choose_filament(); | ||||||
|             DISABLE_AXIS_E0(); |             stepper.disable_extruder(); | ||||||
|             command(MMU_CMD_T0 + index); |             command(MMU_CMD_T0 + index); | ||||||
|             manage_response(true, true); |             manage_response(true, true); | ||||||
|  |  | ||||||
|             if (load_to_gears()) { |             if (load_to_gears()) { | ||||||
|               mmu_loop(); |               mmu_loop(); | ||||||
|               ENABLE_AXIS_E0(); |               stepper.enable_extruder(); | ||||||
|               extruder = index; |               extruder = index; | ||||||
|               active_extruder = 0; |               active_extruder = 0; | ||||||
|             } |             } | ||||||
| @@ -566,7 +566,7 @@ static void mmu2_not_responding() { | |||||||
|     set_runout_valid(false); |     set_runout_valid(false); | ||||||
|  |  | ||||||
|     if (index != extruder) { |     if (index != extruder) { | ||||||
|       DISABLE_AXIS_E0(); |       stepper.disable_extruder(); | ||||||
|       if (FILAMENT_PRESENT()) { |       if (FILAMENT_PRESENT()) { | ||||||
|         DEBUG_ECHOLNPGM("Unloading\n"); |         DEBUG_ECHOLNPGM("Unloading\n"); | ||||||
|         mmu_loading_flag = false; |         mmu_loading_flag = false; | ||||||
| @@ -582,7 +582,7 @@ static void mmu2_not_responding() { | |||||||
|       extruder = index; |       extruder = index; | ||||||
|       active_extruder = 0; |       active_extruder = 0; | ||||||
|  |  | ||||||
|       ENABLE_AXIS_E0(); |       stepper.enable_extruder(); | ||||||
|       SERIAL_ECHO_MSG(STR_ACTIVE_EXTRUDER, extruder); |       SERIAL_ECHO_MSG(STR_ACTIVE_EXTRUDER, extruder); | ||||||
|  |  | ||||||
|       ui.reset_status(); |       ui.reset_status(); | ||||||
| @@ -620,14 +620,14 @@ static void mmu2_not_responding() { | |||||||
|         #if ENABLED(MMU2_MENUS) |         #if ENABLED(MMU2_MENUS) | ||||||
|           planner.synchronize(); |           planner.synchronize(); | ||||||
|           uint8_t index = mmu2_choose_filament(); |           uint8_t index = mmu2_choose_filament(); | ||||||
|           DISABLE_AXIS_E0(); |           stepper.disable_extruder(); | ||||||
|           command(MMU_CMD_T0 + index); |           command(MMU_CMD_T0 + index); | ||||||
|           manage_response(true, true); |           manage_response(true, true); | ||||||
|           mmu_continue_loading(); |           mmu_continue_loading(); | ||||||
|           command(MMU_CMD_C0); |           command(MMU_CMD_C0); | ||||||
|           mmu_loop(); |           mmu_loop(); | ||||||
|  |  | ||||||
|           ENABLE_AXIS_E0(); |           stepper.enable_extruder(); | ||||||
|           extruder = index; |           extruder = index; | ||||||
|           active_extruder = 0; |           active_extruder = 0; | ||||||
|         #else |         #else | ||||||
| @@ -670,14 +670,14 @@ static void mmu2_not_responding() { | |||||||
|     set_runout_valid(false); |     set_runout_valid(false); | ||||||
|  |  | ||||||
|     if (index != extruder) { |     if (index != extruder) { | ||||||
|       DISABLE_AXIS_E0(); |       stepper.disable_extruder(); | ||||||
|       ui.status_printf_P(0, GET_TEXT(MSG_MMU2_LOADING_FILAMENT), int(index + 1)); |       ui.status_printf_P(0, GET_TEXT(MSG_MMU2_LOADING_FILAMENT), int(index + 1)); | ||||||
|       command(MMU_CMD_T0 + index); |       command(MMU_CMD_T0 + index); | ||||||
|       manage_response(true, true); |       manage_response(true, true); | ||||||
|       command(MMU_CMD_C0); |       command(MMU_CMD_C0); | ||||||
|       extruder = index; //filament change is finished |       extruder = index; //filament change is finished | ||||||
|       active_extruder = 0; |       active_extruder = 0; | ||||||
|       ENABLE_AXIS_E0(); |       stepper.enable_extruder(); | ||||||
|       SERIAL_ECHO_MSG(STR_ACTIVE_EXTRUDER, extruder); |       SERIAL_ECHO_MSG(STR_ACTIVE_EXTRUDER, extruder); | ||||||
|       ui.reset_status(); |       ui.reset_status(); | ||||||
|     } |     } | ||||||
| @@ -714,13 +714,13 @@ static void mmu2_not_responding() { | |||||||
|         #if ENABLED(MMU2_MENUS) |         #if ENABLED(MMU2_MENUS) | ||||||
|           planner.synchronize(); |           planner.synchronize(); | ||||||
|           uint8_t index = mmu2_choose_filament(); |           uint8_t index = mmu2_choose_filament(); | ||||||
|           DISABLE_AXIS_E0(); |           stepper.disable_extruder(); | ||||||
|           command(MMU_CMD_T0 + index); |           command(MMU_CMD_T0 + index); | ||||||
|           manage_response(true, true); |           manage_response(true, true); | ||||||
|           command(MMU_CMD_C0); |           command(MMU_CMD_C0); | ||||||
|           mmu_loop(); |           mmu_loop(); | ||||||
|  |  | ||||||
|           ENABLE_AXIS_E0(); |           stepper.enable_extruder(); | ||||||
|           extruder = index; |           extruder = index; | ||||||
|           active_extruder = 0; |           active_extruder = 0; | ||||||
|         #else |         #else | ||||||
| @@ -912,7 +912,7 @@ bool MMU2::load_filament_to_nozzle(const uint8_t index) { | |||||||
|     return false; |     return false; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   DISABLE_AXIS_E0(); |   stepper.disable_extruder(); | ||||||
|   command(MMU_CMD_T0 + index); |   command(MMU_CMD_T0 + index); | ||||||
|   manage_response(true, true); |   manage_response(true, true); | ||||||
|  |  | ||||||
| @@ -950,7 +950,7 @@ bool MMU2::eject_filament(const uint8_t index, const bool recover) { | |||||||
|  |  | ||||||
|   LCD_MESSAGEPGM(MSG_MMU2_EJECTING_FILAMENT); |   LCD_MESSAGEPGM(MSG_MMU2_EJECTING_FILAMENT); | ||||||
|  |  | ||||||
|   ENABLE_AXIS_E0(); |   stepper.enable_extruder(); | ||||||
|   current_position.e -= MMU2_FILAMENTCHANGE_EJECT_FEED; |   current_position.e -= MMU2_FILAMENTCHANGE_EJECT_FEED; | ||||||
|   line_to_current_position(MMM_TO_MMS(2500)); |   line_to_current_position(MMM_TO_MMS(2500)); | ||||||
|   planner.synchronize(); |   planner.synchronize(); | ||||||
| @@ -979,7 +979,7 @@ bool MMU2::eject_filament(const uint8_t index, const bool recover) { | |||||||
|  |  | ||||||
|   BUZZ(200, 404); |   BUZZ(200, 404); | ||||||
|  |  | ||||||
|   DISABLE_AXIS_E0(); |   stepper.disable_extruder(); | ||||||
|  |  | ||||||
|   return true; |   return true; | ||||||
| } | } | ||||||
| @@ -1016,7 +1016,7 @@ bool MMU2::unload() { | |||||||
| void MMU2::execute_extruder_sequence(const E_Step * sequence, int steps) { | void MMU2::execute_extruder_sequence(const E_Step * sequence, int steps) { | ||||||
|  |  | ||||||
|   planner.synchronize(); |   planner.synchronize(); | ||||||
|   ENABLE_AXIS_E0(); |   stepper.enable_extruder(); | ||||||
|  |  | ||||||
|   const E_Step* step = sequence; |   const E_Step* step = sequence; | ||||||
|  |  | ||||||
| @@ -1034,7 +1034,7 @@ void MMU2::execute_extruder_sequence(const E_Step * sequence, int steps) { | |||||||
|     step++; |     step++; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   DISABLE_AXIS_E0(); |   stepper.disable_extruder(); | ||||||
| } | } | ||||||
|  |  | ||||||
| #endif // HAS_PRUSA_MMU2 | #endif // HAS_PRUSA_MMU2 | ||||||
|   | |||||||
| @@ -302,8 +302,8 @@ bool load_filament(const_float_t slow_load_length/*=0*/, const_float_t fast_load | |||||||
|  * send current back to their board, potentially frying it. |  * send current back to their board, potentially frying it. | ||||||
|  */ |  */ | ||||||
| inline void disable_active_extruder() { | inline void disable_active_extruder() { | ||||||
|   #if HAS_E_STEPPER_ENABLE |   #if HAS_EXTRUDERS | ||||||
|     disable_e_stepper(active_extruder); |     stepper.DISABLE_EXTRUDER(active_extruder); | ||||||
|     safe_delay(100); |     safe_delay(100); | ||||||
|   #endif |   #endif | ||||||
| } | } | ||||||
|   | |||||||
| @@ -27,7 +27,7 @@ | |||||||
| #include "../inc/MarlinConfig.h" | #include "../inc/MarlinConfig.h" | ||||||
|  |  | ||||||
| #include "power.h" | #include "power.h" | ||||||
| #include "../module/stepper/indirection.h" | #include "../module/stepper.h" | ||||||
| #include "../MarlinCore.h" | #include "../MarlinCore.h" | ||||||
|  |  | ||||||
| #if ENABLED(PS_OFF_SOUND) | #if ENABLED(PS_OFF_SOUND) | ||||||
| @@ -120,6 +120,9 @@ void Power::power_off() { | |||||||
|    */ |    */ | ||||||
|   bool Power::is_power_needed() { |   bool Power::is_power_needed() { | ||||||
|  |  | ||||||
|  |     // If any of the stepper drivers are enabled... | ||||||
|  |     if (stepper.axis_enabled.bits) return true; | ||||||
|  |  | ||||||
|     if (printJobOngoing() || printingIsPaused()) return true; |     if (printJobOngoing() || printingIsPaused()) return true; | ||||||
|  |  | ||||||
|     #if ENABLED(AUTO_POWER_FANS) |     #if ENABLED(AUTO_POWER_FANS) | ||||||
| @@ -140,23 +143,6 @@ void Power::power_off() { | |||||||
|     if (TERN0(AUTO_POWER_COOLER_FAN, thermalManager.coolerfan_speed)) |     if (TERN0(AUTO_POWER_COOLER_FAN, thermalManager.coolerfan_speed)) | ||||||
|       return true; |       return true; | ||||||
|  |  | ||||||
|     // If any of the drivers or the bed are enabled... |  | ||||||
|     if (X_ENABLE_READ() == X_ENABLE_ON || Y_ENABLE_READ() == Y_ENABLE_ON || Z_ENABLE_READ() == Z_ENABLE_ON |  | ||||||
|       #if HAS_X2_ENABLE |  | ||||||
|         || X2_ENABLE_READ() == X_ENABLE_ON |  | ||||||
|       #endif |  | ||||||
|       #if HAS_Y2_ENABLE |  | ||||||
|         || Y2_ENABLE_READ() == Y_ENABLE_ON |  | ||||||
|       #endif |  | ||||||
|       #if HAS_Z2_ENABLE |  | ||||||
|         || Z2_ENABLE_READ() == Z_ENABLE_ON |  | ||||||
|       #endif |  | ||||||
|       #if E_STEPPERS |  | ||||||
|         #define _OR_ENABLED_E(N) || E##N##_ENABLE_READ() == E_ENABLE_ON |  | ||||||
|         REPEAT(E_STEPPERS, _OR_ENABLED_E) |  | ||||||
|       #endif |  | ||||||
|     ) return true; |  | ||||||
|  |  | ||||||
|     #if HAS_HOTEND |     #if HAS_HOTEND | ||||||
|       HOTEND_LOOP() if (thermalManager.degTargetHotend(e) > 0 || thermalManager.temp_hotend[e].soft_pwm_amount > 0) return true; |       HOTEND_LOOP() if (thermalManager.degTargetHotend(e) > 0 || thermalManager.temp_hotend[e].soft_pwm_amount > 0) return true; | ||||||
|     #endif |     #endif | ||||||
|   | |||||||
| @@ -186,7 +186,7 @@ void PrintJobRecovery::save(const bool force/*=false*/, const float zraise/*=POW | |||||||
|     TERN_(GCODE_REPEAT_MARKERS, info.stored_repeat = repeat); |     TERN_(GCODE_REPEAT_MARKERS, info.stored_repeat = repeat); | ||||||
|     TERN_(HAS_HOME_OFFSET, info.home_offset = home_offset); |     TERN_(HAS_HOME_OFFSET, info.home_offset = home_offset); | ||||||
|     TERN_(HAS_POSITION_SHIFT, info.position_shift = position_shift); |     TERN_(HAS_POSITION_SHIFT, info.position_shift = position_shift); | ||||||
|     TERN_(HAS_MULTI_EXTRUDER, info.active_extruder = active_extruder); |     E_TERN_(info.active_extruder = active_extruder); | ||||||
|  |  | ||||||
|     #if DISABLED(NO_VOLUMETRICS) |     #if DISABLED(NO_VOLUMETRICS) | ||||||
|       info.flag.volumetric_enabled = parser.volumetric_enabled; |       info.flag.volumetric_enabled = parser.volumetric_enabled; | ||||||
|   | |||||||
| @@ -48,10 +48,10 @@ | |||||||
| void GcodeSuite::M301() { | void GcodeSuite::M301() { | ||||||
|   // multi-extruder PID patch: M301 updates or prints a single extruder's PID values |   // multi-extruder PID patch: M301 updates or prints a single extruder's PID values | ||||||
|   // default behavior (omitting E parameter) is to update for extruder 0 only |   // default behavior (omitting E parameter) is to update for extruder 0 only | ||||||
|   int8_t e = parser.byteval('E', -1); // extruder being updated |   int8_t e = E_TERN0(parser.byteval('E', -1)); // extruder being updated | ||||||
|  |  | ||||||
|   if (!parser.seen("PID" TERN_(PID_EXTRUSION_SCALING, "CL") TERN_(PID_FAN_SCALING, "F"))) |   if (!parser.seen("PID" TERN_(PID_EXTRUSION_SCALING, "CL") TERN_(PID_FAN_SCALING, "F"))) | ||||||
|     return M301_report(true, e); |     return M301_report(true E_OPTARG(e)); | ||||||
|  |  | ||||||
|   if (e == -1) e = 0; |   if (e == -1) e = 0; | ||||||
|  |  | ||||||
| @@ -78,8 +78,9 @@ void GcodeSuite::M301() { | |||||||
|     SERIAL_ERROR_MSG(STR_INVALID_EXTRUDER); |     SERIAL_ERROR_MSG(STR_INVALID_EXTRUDER); | ||||||
| } | } | ||||||
|  |  | ||||||
| void GcodeSuite::M301_report(const bool forReplay/*=true*/, const int8_t eindex/*=-1*/) { | void GcodeSuite::M301_report(const bool forReplay/*=true*/ E_OPTARG(const int8_t eindex/*=-1*/)) { | ||||||
|   report_heading(forReplay, PSTR(STR_HOTEND_PID)); |   report_heading(forReplay, PSTR(STR_HOTEND_PID)); | ||||||
|  |   IF_DISABLED(HAS_MULTI_EXTRUDER, constexpr int8_t eindex = -1); | ||||||
|   HOTEND_LOOP() { |   HOTEND_LOOP() { | ||||||
|     if (e == eindex || eindex == -1) { |     if (e == eindex || eindex == -1) { | ||||||
|       report_echo_start(forReplay); |       report_echo_start(forReplay); | ||||||
|   | |||||||
| @@ -29,29 +29,186 @@ | |||||||
|   #include "../../feature/bedlevel/bedlevel.h" |   #include "../../feature/bedlevel/bedlevel.h" | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| /** | #define DEBUG_OUT ENABLED(MARLIN_DEV_MODE) | ||||||
|  * M17: Enable stepper motors | #include "../../core/debug_out.h" | ||||||
|  */ | #include "../../libs/hex_print.h" | ||||||
| void GcodeSuite::M17() { |  | ||||||
|   if (parser.seen_axis()) { | inline axis_flags_t selected_axis_bits() { | ||||||
|     LOGICAL_AXIS_CODE( |   axis_flags_t selected{0}; | ||||||
|       if (TERN0(HAS_E_STEPPER_ENABLE, parser.seen_test('E'))) enable_e_steppers(), |   #if HAS_EXTRUDERS | ||||||
|       if (parser.seen_test('X'))        ENABLE_AXIS_X(), |     if (parser.seen('E')) { | ||||||
|       if (parser.seen_test('Y'))        ENABLE_AXIS_Y(), |       if (E_TERN0(parser.has_value())) { | ||||||
|       if (parser.seen_test('Z'))        ENABLE_AXIS_Z(), |         const uint8_t e = parser.value_int(); | ||||||
|       if (parser.seen_test(AXIS4_NAME)) ENABLE_AXIS_I(), |         if (e < EXTRUDERS) | ||||||
|       if (parser.seen_test(AXIS5_NAME)) ENABLE_AXIS_J(), |           selected.bits = _BV(INDEX_OF_AXIS(E_AXIS, e)); | ||||||
|       if (parser.seen_test(AXIS6_NAME)) ENABLE_AXIS_K() |       } | ||||||
|     ); |       else | ||||||
|  |         selected.bits = selected.e_bits(); | ||||||
|  |     } | ||||||
|  |   #endif | ||||||
|  |   selected.bits |= LINEAR_AXIS_GANG( | ||||||
|  |       (parser.seen_test('X')        << X_AXIS), | ||||||
|  |     | (parser.seen_test('Y')        << Y_AXIS), | ||||||
|  |     | (parser.seen_test('Z')        << Z_AXIS), | ||||||
|  |     | (parser.seen_test(AXIS4_NAME) << I_AXIS), | ||||||
|  |     | (parser.seen_test(AXIS5_NAME) << J_AXIS), | ||||||
|  |     | (parser.seen_test(AXIS6_NAME) << K_AXIS) | ||||||
|  |   ); | ||||||
|  |   return selected; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Enable specified axes and warn about other affected axes | ||||||
|  | void do_enable(const axis_flags_t to_enable) { | ||||||
|  |   const ena_mask_t was_enabled = stepper.axis_enabled.bits, | ||||||
|  |                   shall_enable = to_enable.bits & ~was_enabled; | ||||||
|  |  | ||||||
|  |   DEBUG_ECHOLNPGM("Now Enabled: ", hex_word(stepper.axis_enabled.bits), "  Enabling: ", hex_word(to_enable.bits), " | ", shall_enable); | ||||||
|  |  | ||||||
|  |   if (!shall_enable) return;    // All specified axes already enabled? | ||||||
|  |  | ||||||
|  |   ena_mask_t also_enabled = 0;    // Track steppers enabled due to overlap | ||||||
|  |  | ||||||
|  |   // Enable all flagged axes | ||||||
|  |   LOOP_LINEAR_AXES(a) { | ||||||
|  |     if (TEST(shall_enable, a)) { | ||||||
|  |       stepper.enable_axis(AxisEnum(a));         // Mark and enable the requested axis | ||||||
|  |       DEBUG_ECHOLNPGM("Enabled ", axis_codes[a], " (", a, ") with overlap ", hex_word(enable_overlap[a]), " ... Enabled: ", hex_word(stepper.axis_enabled.bits)); | ||||||
|  |       also_enabled |= enable_overlap[a]; | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|   else { |   #if HAS_EXTRUDERS | ||||||
|     LCD_MESSAGEPGM(MSG_NO_MOVE); |     LOOP_L_N(e, EXTRUDERS) { | ||||||
|     enable_all_steppers(); |       const uint8_t a = INDEX_OF_AXIS(E_AXIS, e); | ||||||
|  |       if (TEST(shall_enable, a)) { | ||||||
|  |         stepper.ENABLE_EXTRUDER(e); | ||||||
|  |         DEBUG_ECHOLNPGM("Enabled E", AS_DIGIT(e), " (", a, ") with overlap ", hex_word(enable_overlap[a]), " ... ", hex_word(stepper.axis_enabled.bits)); | ||||||
|  |         also_enabled |= enable_overlap[a]; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   #endif | ||||||
|  |  | ||||||
|  |   if ((also_enabled &= ~(shall_enable | was_enabled))) { | ||||||
|  |     SERIAL_CHAR('('); | ||||||
|  |     LOOP_LINEAR_AXES(a) if (TEST(also_enabled, a)) SERIAL_CHAR(axis_codes[a], ' '); | ||||||
|  |     #if HAS_EXTRUDERS | ||||||
|  |       #define _EN_ALSO(N) if (TEST(also_enabled, INDEX_OF_AXIS(E_AXIS, N))) SERIAL_CHAR('E', '0' + N, ' '); | ||||||
|  |       REPEAT(EXTRUDERS, _EN_ALSO) | ||||||
|  |     #endif | ||||||
|  |     SERIAL_ECHOLNPGM("also enabled)"); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   DEBUG_ECHOLNPGM("Enabled Now: ", hex_word(stepper.axis_enabled.bits)); | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * M18, M84: Disable stepper motors |  * M17: Enable stepper motor power for one or more axes. | ||||||
|  |  *      Print warnings for axes that share an ENABLE_PIN. | ||||||
|  |  * | ||||||
|  |  * Examples: | ||||||
|  |  * | ||||||
|  |  *  M17 XZ ; Enable X and Z axes | ||||||
|  |  *  M17 E  ; Enable all E steppers | ||||||
|  |  *  M17 E1 ; Enable just the E1 stepper | ||||||
|  |  */ | ||||||
|  | void GcodeSuite::M17() { | ||||||
|  |   if (parser.seen_axis()) { | ||||||
|  |     if (any_enable_overlap()) | ||||||
|  |       do_enable(selected_axis_bits()); | ||||||
|  |     else { | ||||||
|  |       #if HAS_EXTRUDERS | ||||||
|  |         if (parser.seen('E')) { | ||||||
|  |           if (parser.has_value()) { | ||||||
|  |             const uint8_t e = parser.value_int(); | ||||||
|  |             if (e < EXTRUDERS) stepper.ENABLE_EXTRUDER(e); | ||||||
|  |           } | ||||||
|  |           else | ||||||
|  |             stepper.enable_e_steppers(); | ||||||
|  |         } | ||||||
|  |       #endif | ||||||
|  |       LINEAR_AXIS_CODE( | ||||||
|  |         if (parser.seen_test('X'))        stepper.enable_axis(X_AXIS), | ||||||
|  |         if (parser.seen_test('Y'))        stepper.enable_axis(Y_AXIS), | ||||||
|  |         if (parser.seen_test('Z'))        stepper.enable_axis(Z_AXIS), | ||||||
|  |         if (parser.seen_test(AXIS4_NAME)) stepper.enable_axis(I_AXIS), | ||||||
|  |         if (parser.seen_test(AXIS5_NAME)) stepper.enable_axis(J_AXIS), | ||||||
|  |         if (parser.seen_test(AXIS6_NAME)) stepper.enable_axis(K_AXIS) | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   else { | ||||||
|  |     LCD_MESSAGEPGM(MSG_NO_MOVE); | ||||||
|  |     stepper.enable_all_steppers(); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void try_to_disable(const axis_flags_t to_disable) { | ||||||
|  |   ena_mask_t still_enabled = to_disable.bits & stepper.axis_enabled.bits; | ||||||
|  |  | ||||||
|  |   DEBUG_ECHOLNPGM("Enabled: ", hex_word(stepper.axis_enabled.bits), " To Disable: ", hex_word(to_disable.bits), " | ", hex_word(still_enabled)); | ||||||
|  |  | ||||||
|  |   if (!still_enabled) return; | ||||||
|  |  | ||||||
|  |   // Attempt to disable all flagged axes | ||||||
|  |   LOOP_LINEAR_AXES(a) | ||||||
|  |     if (TEST(to_disable.bits, a)) { | ||||||
|  |       DEBUG_ECHOPGM("Try to disable ", axis_codes[a], " (", a, ") with overlap ", hex_word(enable_overlap[a]), " ... "); | ||||||
|  |       if (stepper.disable_axis(AxisEnum(a))) {            // Mark the requested axis and request to disable | ||||||
|  |         DEBUG_ECHOPGM("OK"); | ||||||
|  |         still_enabled &= ~(_BV(a) | enable_overlap[a]);   // If actually disabled, clear one or more tracked bits | ||||||
|  |       } | ||||||
|  |       else | ||||||
|  |         DEBUG_ECHOPGM("OVERLAP"); | ||||||
|  |       DEBUG_ECHOLNPGM(" ... still_enabled=", hex_word(still_enabled)); | ||||||
|  |     } | ||||||
|  |   #if HAS_EXTRUDERS | ||||||
|  |     LOOP_L_N(e, EXTRUDERS) { | ||||||
|  |       const uint8_t a = INDEX_OF_AXIS(E_AXIS, e); | ||||||
|  |       if (TEST(to_disable.bits, a)) { | ||||||
|  |         DEBUG_ECHOPGM("Try to disable E", AS_DIGIT(e), " (", a, ") with overlap ", hex_word(enable_overlap[a]), " ... "); | ||||||
|  |         if (stepper.DISABLE_EXTRUDER(e)) { | ||||||
|  |           DEBUG_ECHOPGM("OK"); | ||||||
|  |           still_enabled &= ~(_BV(a) | enable_overlap[a]); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |           DEBUG_ECHOPGM("OVERLAP"); | ||||||
|  |         DEBUG_ECHOLNPGM(" ... still_enabled=", hex_word(still_enabled)); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   #endif | ||||||
|  |  | ||||||
|  |   auto overlap_warning = [](const ena_mask_t axis_bits) { | ||||||
|  |     SERIAL_ECHOPGM(" not disabled. Shared with"); | ||||||
|  |     LOOP_LINEAR_AXES(a) if (TEST(axis_bits, a)) SERIAL_CHAR(' ', axis_codes[a]); | ||||||
|  |     #if HAS_EXTRUDERS | ||||||
|  |       #define _EN_STILLON(N) if (TEST(axis_bits, INDEX_OF_AXIS(E_AXIS, N))) SERIAL_CHAR(' ', 'E', '0' + N); | ||||||
|  |       REPEAT(EXTRUDERS, _EN_STILLON) | ||||||
|  |     #endif | ||||||
|  |     SERIAL_ECHOLNPGM("."); | ||||||
|  |   }; | ||||||
|  |  | ||||||
|  |   // If any of the requested axes are still enabled, give a warning | ||||||
|  |   LOOP_LINEAR_AXES(a) { | ||||||
|  |     if (TEST(still_enabled, a)) { | ||||||
|  |       SERIAL_CHAR(axis_codes[a]); | ||||||
|  |       overlap_warning(stepper.axis_enabled.bits & enable_overlap[a]); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   #if HAS_EXTRUDERS | ||||||
|  |     LOOP_L_N(e, EXTRUDERS) { | ||||||
|  |       const uint8_t a = INDEX_OF_AXIS(E_AXIS, e); | ||||||
|  |       if (TEST(still_enabled, a)) { | ||||||
|  |         SERIAL_CHAR('E', '0' + e); | ||||||
|  |         overlap_warning(stepper.axis_enabled.bits & enable_overlap[a]); | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   #endif | ||||||
|  |  | ||||||
|  |   DEBUG_ECHOLNPGM("Enabled Now: ", hex_word(stepper.axis_enabled.bits)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * M18, M84: Disable stepper motor power for one or more axes. | ||||||
|  |  *           Print warnings for axes that share an ENABLE_PIN. | ||||||
|  */ |  */ | ||||||
| void GcodeSuite::M18_M84() { | void GcodeSuite::M18_M84() { | ||||||
|   if (parser.seenval('S')) { |   if (parser.seenval('S')) { | ||||||
| @@ -61,15 +218,26 @@ void GcodeSuite::M18_M84() { | |||||||
|   else { |   else { | ||||||
|     if (parser.seen_axis()) { |     if (parser.seen_axis()) { | ||||||
|       planner.synchronize(); |       planner.synchronize(); | ||||||
|       LOGICAL_AXIS_CODE( |       if (any_enable_overlap()) | ||||||
|         if (TERN0(HAS_E_STEPPER_ENABLE, parser.seen_test('E'))) disable_e_steppers(), |         try_to_disable(selected_axis_bits()); | ||||||
|         if (parser.seen_test('X'))        DISABLE_AXIS_X(), |       else { | ||||||
|         if (parser.seen_test('Y'))        DISABLE_AXIS_Y(), |         #if HAS_EXTRUDERS | ||||||
|         if (parser.seen_test('Z'))        DISABLE_AXIS_Z(), |           if (parser.seen('E')) { | ||||||
|         if (parser.seen_test(AXIS4_NAME)) DISABLE_AXIS_I(), |             if (E_TERN0(parser.has_value())) | ||||||
|         if (parser.seen_test(AXIS5_NAME)) DISABLE_AXIS_J(), |               stepper.DISABLE_EXTRUDER(parser.value_int()); | ||||||
|         if (parser.seen_test(AXIS6_NAME)) DISABLE_AXIS_K() |             else | ||||||
|       ); |               stepper.disable_e_steppers(); | ||||||
|  |           } | ||||||
|  |         #endif | ||||||
|  |         LINEAR_AXIS_CODE( | ||||||
|  |           if (parser.seen_test('X'))        stepper.disable_axis(X_AXIS), | ||||||
|  |           if (parser.seen_test('Y'))        stepper.disable_axis(Y_AXIS), | ||||||
|  |           if (parser.seen_test('Z'))        stepper.disable_axis(Z_AXIS), | ||||||
|  |           if (parser.seen_test(AXIS4_NAME)) stepper.disable_axis(I_AXIS), | ||||||
|  |           if (parser.seen_test(AXIS5_NAME)) stepper.disable_axis(J_AXIS), | ||||||
|  |           if (parser.seen_test(AXIS6_NAME)) stepper.disable_axis(K_AXIS) | ||||||
|  |         ); | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|     else |     else | ||||||
|       planner.finish_and_disable(); |       planner.finish_and_disable(); | ||||||
|   | |||||||
| @@ -32,7 +32,7 @@ | |||||||
|  * G10 - Retract filament according to settings of M207 |  * G10 - Retract filament according to settings of M207 | ||||||
|  *       TODO: Handle 'G10 P' for tool settings and 'G10 L' for workspace settings |  *       TODO: Handle 'G10 P' for tool settings and 'G10 L' for workspace settings | ||||||
|  */ |  */ | ||||||
| void GcodeSuite::G10() { fwretract.retract(true OPTARG(HAS_MULTI_EXTRUDER, parser.boolval('S'))); } | void GcodeSuite::G10() { fwretract.retract(true E_OPTARG(parser.boolval('S'))); } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * G11 - Recover filament according to settings of M208 |  * G11 - Recover filament according to settings of M208 | ||||||
|   | |||||||
| @@ -879,7 +879,7 @@ private: | |||||||
|  |  | ||||||
|   #if ENABLED(PIDTEMP) |   #if ENABLED(PIDTEMP) | ||||||
|     static void M301(); |     static void M301(); | ||||||
|     static void M301_report(const bool forReplay=true, const int8_t eindex=-1); |     static void M301_report(const bool forReplay=true E_OPTARG(const int8_t eindex=-1)); | ||||||
|   #endif |   #endif | ||||||
|  |  | ||||||
|   #if ENABLED(PREVENT_COLD_EXTRUSION) |   #if ENABLED(PREVENT_COLD_EXTRUSION) | ||||||
|   | |||||||
| @@ -587,6 +587,8 @@ | |||||||
|  *  HOTENDS      - Number of hotends, whether connected or separate |  *  HOTENDS      - Number of hotends, whether connected or separate | ||||||
|  *  E_STEPPERS   - Number of actual E stepper motors |  *  E_STEPPERS   - Number of actual E stepper motors | ||||||
|  *  E_MANUAL     - Number of E steppers for LCD move options |  *  E_MANUAL     - Number of E steppers for LCD move options | ||||||
|  |  * | ||||||
|  |  * These defines must be simple constants for use in REPEAT, etc. | ||||||
|  */ |  */ | ||||||
| #if EXTRUDERS | #if EXTRUDERS | ||||||
|   #define HAS_EXTRUDERS 1 |   #define HAS_EXTRUDERS 1 | ||||||
| @@ -605,9 +607,14 @@ | |||||||
|   #undef DISABLE_E |   #undef DISABLE_E | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | #define E_OPTARG(N) OPTARG(HAS_MULTI_EXTRUDER, N) | ||||||
|  | #define E_TERN_(N)  TERN_(HAS_MULTI_EXTRUDER, N) | ||||||
|  | #define E_TERN0(N)  TERN0(HAS_MULTI_EXTRUDER, N) | ||||||
|  |  | ||||||
| #if ENABLED(E_DUAL_STEPPER_DRIVERS) // E0/E1 steppers act in tandem as E0 | #if ENABLED(E_DUAL_STEPPER_DRIVERS) // E0/E1 steppers act in tandem as E0 | ||||||
|  |  | ||||||
|   #define E_STEPPERS      2 |   #define E_STEPPERS      2 | ||||||
|  |   #define E_MANUAL        1 | ||||||
|  |  | ||||||
| #elif ENABLED(SWITCHING_EXTRUDER)   // One stepper for every two EXTRUDERS | #elif ENABLED(SWITCHING_EXTRUDER)   // One stepper for every two EXTRUDERS | ||||||
|  |  | ||||||
| @@ -637,7 +644,8 @@ | |||||||
|  |  | ||||||
| #elif HAS_PRUSA_MMU2                // Průša Multi-Material Unit v2 | #elif HAS_PRUSA_MMU2                // Průša Multi-Material Unit v2 | ||||||
|  |  | ||||||
|   #define E_STEPPERS 1 |   #define E_STEPPERS      1 | ||||||
|  |   #define E_MANUAL        1 | ||||||
|  |  | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|   | |||||||
| @@ -37,6 +37,7 @@ | |||||||
| #include "FileNavigator.h" | #include "FileNavigator.h" | ||||||
|  |  | ||||||
| #include "../../../gcode/queue.h" | #include "../../../gcode/queue.h" | ||||||
|  | #include "../../../module/stepper.h" | ||||||
| #include "../../../sd/cardreader.h" | #include "../../../sd/cardreader.h" | ||||||
| #include "../../../libs/numtostr.h" | #include "../../../libs/numtostr.h" | ||||||
| #include "../../../MarlinCore.h" | #include "../../../MarlinCore.h" | ||||||
| @@ -665,7 +666,7 @@ void ChironTFT::PanelAction(uint8_t req) { | |||||||
|  |  | ||||||
|     case 19:   // A19 Motors off |     case 19:   // A19 Motors off | ||||||
|       if (!isPrinting()) { |       if (!isPrinting()) { | ||||||
|         disable_all_steppers(); // from marlincore.h |         stepper.disable_all_steppers(); | ||||||
|         SendtoTFTLN(AC_msg_ready); |         SendtoTFTLN(AC_msg_ready); | ||||||
|       } |       } | ||||||
|       break; |       break; | ||||||
|   | |||||||
| @@ -27,6 +27,7 @@ | |||||||
| #include "../ui_api.h" | #include "../ui_api.h" | ||||||
|  |  | ||||||
| #include "../../../libs/numtostr.h" | #include "../../../libs/numtostr.h" | ||||||
|  | #include "../../../module/stepper.h" // for disable_all_steppers | ||||||
| #include "../../../module/motion.h"  // for quickstop_stepper, A20 read printing speed, feedrate_percentage | #include "../../../module/motion.h"  // for quickstop_stepper, A20 read printing speed, feedrate_percentage | ||||||
| #include "../../../MarlinCore.h"     // for disable_steppers | #include "../../../MarlinCore.h"     // for disable_steppers | ||||||
| #include "../../../inc/MarlinConfig.h" | #include "../../../inc/MarlinConfig.h" | ||||||
| @@ -738,7 +739,7 @@ void AnycubicTFTClass::GetCommandFromTFT() { | |||||||
|           case 19: // A19 stop stepper drivers - sent on stop extrude command and on turn motors off command |           case 19: // A19 stop stepper drivers - sent on stop extrude command and on turn motors off command | ||||||
|             if (!isPrinting()) { |             if (!isPrinting()) { | ||||||
|               quickstop_stepper(); |               quickstop_stepper(); | ||||||
|               disable_all_steppers(); |               stepper.disable_all_steppers(); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             SENDLINE_PGM(""); |             SENDLINE_PGM(""); | ||||||
|   | |||||||
| @@ -33,6 +33,7 @@ | |||||||
| #include "../../../core/language.h" | #include "../../../core/language.h" | ||||||
| #include "../../../module/temperature.h" | #include "../../../module/temperature.h" | ||||||
| #include "../../../module/printcounter.h" | #include "../../../module/printcounter.h" | ||||||
|  | #include "../../../module/stepper.h" | ||||||
| #include "../../../gcode/queue.h" | #include "../../../gcode/queue.h" | ||||||
| #if ENABLED(ADVANCED_PAUSE_FEATURE) | #if ENABLED(ADVANCED_PAUSE_FEATURE) | ||||||
|   #include "../../../feature/pause.h" |   #include "../../../feature/pause.h" | ||||||
| @@ -375,10 +376,10 @@ void DGUSRxHandler::Steppers(DGUS_VP &vp, void *data_ptr) { | |||||||
|  |  | ||||||
|   switch (control) { |   switch (control) { | ||||||
|     case DGUS_Data::Control::ENABLE: |     case DGUS_Data::Control::ENABLE: | ||||||
|       enable_all_steppers(); |       stepper.enable_all_steppers(); | ||||||
|       break; |       break; | ||||||
|     case DGUS_Data::Control::DISABLE: |     case DGUS_Data::Control::DISABLE: | ||||||
|       disable_all_steppers(); |       stepper.disable_all_steppers(); | ||||||
|       break; |       break; | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -553,7 +554,7 @@ void DGUSRxHandler::FilamentSelect(DGUS_VP &vp, void *data_ptr) { | |||||||
|     default: return; |     default: return; | ||||||
|     case DGUS_Data::Extruder::CURRENT: |     case DGUS_Data::Extruder::CURRENT: | ||||||
|     case DGUS_Data::Extruder::E0: |     case DGUS_Data::Extruder::E0: | ||||||
|     TERN_(HAS_MULTI_EXTRUDER, case DGUS_Data::Extruder::E1:) |     E_TERN_(case DGUS_Data::Extruder::E1:) | ||||||
|       dgus_screen_handler.filament_extruder = extruder; |       dgus_screen_handler.filament_extruder = extruder; | ||||||
|       break; |       break; | ||||||
|   } |   } | ||||||
|   | |||||||
| @@ -286,14 +286,8 @@ void DGUSTxHandler::TempMax(DGUS_VP &vp) { | |||||||
| } | } | ||||||
|  |  | ||||||
| void DGUSTxHandler::StepperStatus(DGUS_VP &vp) { | void DGUSTxHandler::StepperStatus(DGUS_VP &vp) { | ||||||
|   if (X_ENABLE_READ() == X_ENABLE_ON |   const bool motor_on = stepper.axis_enabled.bits & (_BV(LINEAR_AXES) - 1); | ||||||
|       && Y_ENABLE_READ() == Y_ENABLE_ON |   dgus_display.Write((uint16_t)vp.addr, Swap16(uint16_t(motor_on ? DGUS_Data::Status::ENABLED : DGUS_Data::Status::DISABLED))); | ||||||
|       && Z_ENABLE_READ() == Z_ENABLE_ON) { |  | ||||||
|     dgus_display.Write((uint16_t)vp.addr, Swap16((uint16_t)DGUS_Data::Status::ENABLED)); |  | ||||||
|   } |  | ||||||
|   else { |  | ||||||
|     dgus_display.Write((uint16_t)vp.addr, Swap16((uint16_t)DGUS_Data::Status::DISABLED)); |  | ||||||
|   } |  | ||||||
| } | } | ||||||
|  |  | ||||||
| void DGUSTxHandler::StepIcons(DGUS_VP &vp) { | void DGUSTxHandler::StepIcons(DGUS_VP &vp) { | ||||||
|   | |||||||
| @@ -125,7 +125,7 @@ void StressTestScreen::onIdle() { | |||||||
|       injectCommands_P(PSTR( |       injectCommands_P(PSTR( | ||||||
|         "G0 X100 Y100 Z100 F6000\n" |         "G0 X100 Y100 Z100 F6000\n" | ||||||
|         "T0\nG4 S1" |         "T0\nG4 S1" | ||||||
|         TERN_(HAS_MULTI_EXTRUDER, "\nT1\nG4 S1") |         E_TERN_("\nT1\nG4 S1") | ||||||
|         "\nG0 X150 Y150 Z150" |         "\nG0 X150 Y150 Z150" | ||||||
|       )); |       )); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -33,6 +33,7 @@ | |||||||
|  |  | ||||||
| #include "../../../MarlinCore.h" | #include "../../../MarlinCore.h" | ||||||
| #include "../../../feature/pause.h" | #include "../../../feature/pause.h" | ||||||
|  | #include "../../../module/stepper.h" | ||||||
| #include "../../../gcode/queue.h" | #include "../../../gcode/queue.h" | ||||||
| #include "../../../libs/numtostr.h" | #include "../../../libs/numtostr.h" | ||||||
| #include "../../../sd/cardreader.h" | #include "../../../sd/cardreader.h" | ||||||
| @@ -536,7 +537,7 @@ void NextionTFT::PanelAction(uint8_t req) { | |||||||
|  |  | ||||||
|     case 57: // Disable Motors |     case 57: // Disable Motors | ||||||
|       if (!isPrinting()) { |       if (!isPrinting()) { | ||||||
|         disable_all_steppers(); // from marlincore.h |         stepper.disable_all_steppers(); | ||||||
|         SEND_TXT("tmppage.M117", "Motors disabled"); |         SEND_TXT("tmppage.M117", "Motors disabled"); | ||||||
|       } |       } | ||||||
|       break; |       break; | ||||||
|   | |||||||
| @@ -1375,13 +1375,13 @@ void Planner::check_axes_activity() { | |||||||
|   // Disable inactive axes |   // Disable inactive axes | ||||||
|   // |   // | ||||||
|   LOGICAL_AXIS_CODE( |   LOGICAL_AXIS_CODE( | ||||||
|     if (TERN0(DISABLE_E, !axis_active.e)) disable_e_steppers(), |     if (TERN0(DISABLE_E, !axis_active.e)) stepper.disable_e_steppers(), | ||||||
|     if (TERN0(DISABLE_X, !axis_active.x)) DISABLE_AXIS_X(), |     if (TERN0(DISABLE_X, !axis_active.x)) stepper.disable_axis(X_AXIS), | ||||||
|     if (TERN0(DISABLE_Y, !axis_active.y)) DISABLE_AXIS_Y(), |     if (TERN0(DISABLE_Y, !axis_active.y)) stepper.disable_axis(Y_AXIS), | ||||||
|     if (TERN0(DISABLE_Z, !axis_active.z)) DISABLE_AXIS_Z(), |     if (TERN0(DISABLE_Z, !axis_active.z)) stepper.disable_axis(Z_AXIS), | ||||||
|     if (TERN0(DISABLE_I, !axis_active.i)) DISABLE_AXIS_I(), |     if (TERN0(DISABLE_I, !axis_active.i)) stepper.disable_axis(I_AXIS), | ||||||
|     if (TERN0(DISABLE_J, !axis_active.j)) DISABLE_AXIS_J(), |     if (TERN0(DISABLE_J, !axis_active.j)) stepper.disable_axis(J_AXIS), | ||||||
|     if (TERN0(DISABLE_K, !axis_active.k)) DISABLE_AXIS_K() |     if (TERN0(DISABLE_K, !axis_active.k)) stepper.disable_axis(K_AXIS) | ||||||
|   ); |   ); | ||||||
|  |  | ||||||
|   // |   // | ||||||
| @@ -1707,7 +1707,7 @@ float Planner::triggered_position_mm(const AxisEnum axis) { | |||||||
|  |  | ||||||
| void Planner::finish_and_disable() { | void Planner::finish_and_disable() { | ||||||
|   while (has_blocks_queued() || cleaning_buffer_counter) idle(); |   while (has_blocks_queued() || cleaning_buffer_counter) idle(); | ||||||
|   disable_all_steppers(); |   stepper.disable_all_steppers(); | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -2144,7 +2144,7 @@ bool Planner::_populate_block(block_t * const block, bool split_move, | |||||||
|     block->e_to_p_pressure = baricuda_e_to_p_pressure; |     block->e_to_p_pressure = baricuda_e_to_p_pressure; | ||||||
|   #endif |   #endif | ||||||
|  |  | ||||||
|   TERN_(HAS_MULTI_EXTRUDER, block->extruder = extruder); |   E_TERN_(block->extruder = extruder); | ||||||
|  |  | ||||||
|   #if ENABLED(AUTO_POWER_CONTROL) |   #if ENABLED(AUTO_POWER_CONTROL) | ||||||
|     if (LINEAR_AXIS_GANG( |     if (LINEAR_AXIS_GANG( | ||||||
| @@ -2160,43 +2160,43 @@ bool Planner::_populate_block(block_t * const block, bool split_move, | |||||||
|   // Enable active axes |   // Enable active axes | ||||||
|   #if EITHER(CORE_IS_XY, MARKFORGED_XY) |   #if EITHER(CORE_IS_XY, MARKFORGED_XY) | ||||||
|     if (block->steps.a || block->steps.b) { |     if (block->steps.a || block->steps.b) { | ||||||
|       ENABLE_AXIS_X(); |       stepper.enable_axis(X_AXIS); | ||||||
|       ENABLE_AXIS_Y(); |       stepper.enable_axis(Y_AXIS); | ||||||
|     } |     } | ||||||
|     #if DISABLED(Z_LATE_ENABLE) |     #if DISABLED(Z_LATE_ENABLE) | ||||||
|       if (block->steps.z) ENABLE_AXIS_Z(); |       if (block->steps.z) stepper.enable_axis(Z_AXIS); | ||||||
|     #endif |     #endif | ||||||
|   #elif CORE_IS_XZ |   #elif CORE_IS_XZ | ||||||
|     if (block->steps.a || block->steps.c) { |     if (block->steps.a || block->steps.c) { | ||||||
|       ENABLE_AXIS_X(); |       stepper.enable_axis(X_AXIS); | ||||||
|       ENABLE_AXIS_Z(); |       stepper.enable_axis(Z_AXIS); | ||||||
|     } |     } | ||||||
|     if (block->steps.y) ENABLE_AXIS_Y(); |     if (block->steps.y) stepper.enable_axis(Y_AXIS); | ||||||
|   #elif CORE_IS_YZ |   #elif CORE_IS_YZ | ||||||
|     if (block->steps.b || block->steps.c) { |     if (block->steps.b || block->steps.c) { | ||||||
|       ENABLE_AXIS_Y(); |       stepper.enable_axis(Y_AXIS); | ||||||
|       ENABLE_AXIS_Z(); |       stepper.enable_axis(Z_AXIS); | ||||||
|     } |     } | ||||||
|     if (block->steps.x) ENABLE_AXIS_X(); |     if (block->steps.x) stepper.enable_axis(X_AXIS); | ||||||
|   #else |   #else | ||||||
|     LINEAR_AXIS_CODE( |     LINEAR_AXIS_CODE( | ||||||
|       if (block->steps.x) ENABLE_AXIS_X(), |       if (block->steps.x) stepper.enable_axis(X_AXIS), | ||||||
|       if (block->steps.y) ENABLE_AXIS_Y(), |       if (block->steps.y) stepper.enable_axis(Y_AXIS), | ||||||
|       if (TERN(Z_LATE_ENABLE, 0, block->steps.z)) ENABLE_AXIS_Z(), |       if (TERN(Z_LATE_ENABLE, 0, block->steps.z)) stepper.enable_axis(Z_AXIS), | ||||||
|       if (block->steps.i) ENABLE_AXIS_I(), |       if (block->steps.i) stepper.enable_axis(I_AXIS), | ||||||
|       if (block->steps.j) ENABLE_AXIS_J(), |       if (block->steps.j) stepper.enable_axis(J_AXIS), | ||||||
|       if (block->steps.k) ENABLE_AXIS_K() |       if (block->steps.k) stepper.enable_axis(K_AXIS) | ||||||
|     ); |     ); | ||||||
|   #endif |   #endif | ||||||
|   #if EITHER(IS_CORE, MARKFORGED_XY) |   #if EITHER(IS_CORE, MARKFORGED_XY) | ||||||
|     #if LINEAR_AXES >= 4 |     #if LINEAR_AXES >= 4 | ||||||
|       if (block->steps.i) ENABLE_AXIS_I(); |       if (block->steps.i) stepper.enable_axis(I_AXIS); | ||||||
|     #endif |     #endif | ||||||
|     #if LINEAR_AXES >= 5 |     #if LINEAR_AXES >= 5 | ||||||
|       if (block->steps.j) ENABLE_AXIS_J(); |       if (block->steps.j) stepper.enable_axis(J_AXIS); | ||||||
|     #endif |     #endif | ||||||
|     #if LINEAR_AXES >= 6 |     #if LINEAR_AXES >= 6 | ||||||
|       if (block->steps.k) ENABLE_AXIS_K(); |       if (block->steps.k) stepper.enable_axis(K_AXIS); | ||||||
|     #endif |     #endif | ||||||
|   #endif |   #endif | ||||||
|  |  | ||||||
| @@ -2214,27 +2214,27 @@ bool Planner::_populate_block(block_t * const block, bool split_move, | |||||||
|  |  | ||||||
|         #define ENABLE_ONE_E(N) do{ \ |         #define ENABLE_ONE_E(N) do{ \ | ||||||
|           if (E_STEPPER_INDEX(extruder) == N) { \ |           if (E_STEPPER_INDEX(extruder) == N) { \ | ||||||
|             ENABLE_AXIS_E##N(); \ |             stepper.ENABLE_EXTRUDER(N); \ | ||||||
|             g_uc_extruder_last_move[N] = (BLOCK_BUFFER_SIZE) * 2; \ |             g_uc_extruder_last_move[N] = (BLOCK_BUFFER_SIZE) * 2; \ | ||||||
|             if ((N) == 0 && TERN0(HAS_DUPLICATION_MODE, extruder_duplication_enabled)) \ |             if ((N) == 0 && TERN0(HAS_DUPLICATION_MODE, extruder_duplication_enabled)) \ | ||||||
|               ENABLE_AXIS_E1(); \ |               stepper.ENABLE_EXTRUDER(1); \ | ||||||
|           } \ |           } \ | ||||||
|           else if (!g_uc_extruder_last_move[N]) { \ |           else if (!g_uc_extruder_last_move[N]) { \ | ||||||
|             DISABLE_AXIS_E##N(); \ |             stepper.DISABLE_EXTRUDER(N); \ | ||||||
|             if ((N) == 0 && TERN0(HAS_DUPLICATION_MODE, extruder_duplication_enabled)) \ |             if ((N) == 0 && TERN0(HAS_DUPLICATION_MODE, extruder_duplication_enabled)) \ | ||||||
|               DISABLE_AXIS_E1(); \ |               stepper.DISABLE_EXTRUDER(1); \ | ||||||
|           } \ |           } \ | ||||||
|         }while(0); |         }while(0); | ||||||
|  |  | ||||||
|       #else |       #else | ||||||
|  |  | ||||||
|         #define ENABLE_ONE_E(N) ENABLE_AXIS_E##N(); |         #define ENABLE_ONE_E(N) stepper.ENABLE_EXTRUDER(N); | ||||||
|  |  | ||||||
|       #endif |       #endif | ||||||
|  |  | ||||||
|       REPEAT(E_STEPPERS, ENABLE_ONE_E); // (ENABLE_ONE_E must end with semicolon) |       REPEAT(E_STEPPERS, ENABLE_ONE_E); // (ENABLE_ONE_E must end with semicolon) | ||||||
|     } |     } | ||||||
|   #endif // EXTRUDERS |   #endif // HAS_EXTRUDERS | ||||||
|  |  | ||||||
|   if (esteps) |   if (esteps) | ||||||
|     NOLESS(fr_mm_s, settings.min_feedrate_mm_s); |     NOLESS(fr_mm_s, settings.min_feedrate_mm_s); | ||||||
| @@ -3049,7 +3049,7 @@ bool Planner::buffer_line(const xyze_pos_t &cart, const_feedRate_t fr_mm_s, cons | |||||||
|       FANS_LOOP(i) block->fan_speed[i] = thermalManager.fan_speed[i]; |       FANS_LOOP(i) block->fan_speed[i] = thermalManager.fan_speed[i]; | ||||||
|     #endif |     #endif | ||||||
|  |  | ||||||
|     TERN_(HAS_MULTI_EXTRUDER, block->extruder = extruder); |     E_TERN_(block->extruder = extruder); | ||||||
|  |  | ||||||
|     block->page_idx = page_idx; |     block->page_idx = page_idx; | ||||||
|  |  | ||||||
| @@ -3085,7 +3085,7 @@ bool Planner::buffer_line(const xyze_pos_t &cart, const_feedRate_t fr_mm_s, cons | |||||||
|     // Move buffer head |     // Move buffer head | ||||||
|     block_buffer_head = next_buffer_head; |     block_buffer_head = next_buffer_head; | ||||||
|  |  | ||||||
|     enable_all_steppers(); |     stepper.enable_all_steppers(); | ||||||
|     stepper.wake_up(); |     stepper.wake_up(); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -251,17 +251,17 @@ xyz_pos_t Probe::offset; // Initialized by settings.load() | |||||||
|   void Probe::set_probing_paused(const bool dopause) { |   void Probe::set_probing_paused(const bool dopause) { | ||||||
|     TERN_(PROBING_HEATERS_OFF, thermalManager.pause_heaters(dopause)); |     TERN_(PROBING_HEATERS_OFF, thermalManager.pause_heaters(dopause)); | ||||||
|     TERN_(PROBING_FANS_OFF, thermalManager.set_fans_paused(dopause)); |     TERN_(PROBING_FANS_OFF, thermalManager.set_fans_paused(dopause)); | ||||||
|     TERN_(PROBING_ESTEPPERS_OFF, if (dopause) disable_e_steppers()); |     TERN_(PROBING_ESTEPPERS_OFF, if (dopause) stepper.disable_e_steppers()); | ||||||
|     #if ENABLED(PROBING_STEPPERS_OFF) && DISABLED(DELTA) |     #if ENABLED(PROBING_STEPPERS_OFF) && DISABLED(DELTA) | ||||||
|       static uint8_t old_trusted; |       static uint8_t old_trusted; | ||||||
|       if (dopause) { |       if (dopause) { | ||||||
|         old_trusted = axis_trusted; |         old_trusted = axis_trusted; | ||||||
|         DISABLE_AXIS_X(); |         stepper.disable_axis(X_AXIS); | ||||||
|         DISABLE_AXIS_Y(); |         stepper.disable_axis(Y_AXIS); | ||||||
|       } |       } | ||||||
|       else { |       else { | ||||||
|         if (TEST(old_trusted, X_AXIS)) ENABLE_AXIS_X(); |         if (TEST(old_trusted, X_AXIS)) stepper.enable_axis(X_AXIS); | ||||||
|         if (TEST(old_trusted, Y_AXIS)) ENABLE_AXIS_Y(); |         if (TEST(old_trusted, Y_AXIS)) stepper.enable_axis(Y_AXIS); | ||||||
|         axis_trusted = old_trusted; |         axis_trusted = old_trusted; | ||||||
|       } |       } | ||||||
|     #endif |     #endif | ||||||
|   | |||||||
| @@ -3278,7 +3278,7 @@ void MarlinSettings::reset() { | |||||||
|     // |     // | ||||||
|     // Tool-changing Parameters |     // Tool-changing Parameters | ||||||
|     // |     // | ||||||
|     TERN_(HAS_MULTI_EXTRUDER, gcode.M217_report(forReplay)); |     E_TERN_(gcode.M217_report(forReplay)); | ||||||
|  |  | ||||||
|     // |     // | ||||||
|     // Backlash Compensation |     // Backlash Compensation | ||||||
|   | |||||||
| @@ -123,6 +123,10 @@ Stepper stepper; // Singleton | |||||||
|   bool L64XX_OK_to_power_up = false;  // flag to keep L64xx steppers powered down after a reset or power up |   bool L64XX_OK_to_power_up = false;  // flag to keep L64xx steppers powered down after a reset or power up | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | #if ENABLED(AUTO_POWER_CONTROL) | ||||||
|  |   #include "../feature/power.h" | ||||||
|  | #endif | ||||||
|  |  | ||||||
| #if ENABLED(POWER_LOSS_RECOVERY) | #if ENABLED(POWER_LOSS_RECOVERY) | ||||||
|   #include "../feature/powerloss.h" |   #include "../feature/powerloss.h" | ||||||
| #endif | #endif | ||||||
| @@ -131,6 +135,10 @@ Stepper stepper; // Singleton | |||||||
|   #include "../feature/spindle_laser.h" |   #include "../feature/spindle_laser.h" | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | #if ENABLED(EXTENSIBLE_UI) | ||||||
|  |   #include "../lcd/extui/ui_api.h" | ||||||
|  | #endif | ||||||
|  |  | ||||||
| // public: | // public: | ||||||
|  |  | ||||||
| #if EITHER(HAS_EXTRA_ENDSTOPS, Z_STEPPER_AUTO_ALIGN) | #if EITHER(HAS_EXTRA_ENDSTOPS, Z_STEPPER_AUTO_ALIGN) | ||||||
| @@ -145,6 +153,8 @@ Stepper stepper; // Singleton | |||||||
|   #endif |   #endif | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | axis_flags_t Stepper::axis_enabled; // {0} | ||||||
|  |  | ||||||
| // private: | // private: | ||||||
|  |  | ||||||
| block_t* Stepper::current_block; // (= nullptr) A pointer to the block currently being traced | block_t* Stepper::current_block; // (= nullptr) A pointer to the block currently being traced | ||||||
| @@ -473,6 +483,89 @@ xyze_int8_t Stepper::count_direction{0}; | |||||||
|   #define DIR_WAIT_AFTER() |   #define DIR_WAIT_AFTER() | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | void Stepper::enable_axis(const AxisEnum axis) { | ||||||
|  |   #define _CASE_ENABLE(N) case N##_AXIS: ENABLE_AXIS_##N(); break; | ||||||
|  |   switch (axis) { | ||||||
|  |     LINEAR_AXIS_CODE( | ||||||
|  |       _CASE_ENABLE(X), _CASE_ENABLE(Y), _CASE_ENABLE(Z), | ||||||
|  |       _CASE_ENABLE(I), _CASE_ENABLE(J), _CASE_ENABLE(K) | ||||||
|  |     ); | ||||||
|  |     default: break; | ||||||
|  |   } | ||||||
|  |   mark_axis_enabled(axis); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool Stepper::disable_axis(const AxisEnum axis) { | ||||||
|  |   mark_axis_disabled(axis); | ||||||
|  |   // If all the axes that share the enabled bit are disabled | ||||||
|  |   const bool can_disable = can_axis_disable(axis); | ||||||
|  |   if (can_disable) { | ||||||
|  |     #define _CASE_DISABLE(N) case N##_AXIS: DISABLE_AXIS_##N(); break; | ||||||
|  |     switch (axis) { | ||||||
|  |       LINEAR_AXIS_CODE( | ||||||
|  |         _CASE_DISABLE(X), _CASE_DISABLE(Y), _CASE_DISABLE(Z), | ||||||
|  |         _CASE_DISABLE(I), _CASE_DISABLE(J), _CASE_DISABLE(K) | ||||||
|  |       ); | ||||||
|  |       default: break; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   return can_disable; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #if HAS_EXTRUDERS | ||||||
|  |  | ||||||
|  |   void Stepper::enable_extruder(E_TERN_(const uint8_t eindex)) { | ||||||
|  |     IF_DISABLED(HAS_MULTI_EXTRUDER, constexpr uint8_t eindex = 0); | ||||||
|  |     #define _CASE_ENA_E(N) case N: ENABLE_AXIS_E##N(); mark_axis_enabled(E_AXIS E_OPTARG(eindex)); break; | ||||||
|  |     switch (eindex) { | ||||||
|  |       REPEAT(E_STEPPERS, _CASE_ENA_E) | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   bool Stepper::disable_extruder(E_TERN_(const uint8_t eindex)) { | ||||||
|  |     IF_DISABLED(HAS_MULTI_EXTRUDER, constexpr uint8_t eindex = 0); | ||||||
|  |     mark_axis_disabled(E_AXIS E_OPTARG(eindex)); | ||||||
|  |     const bool can_disable = can_axis_disable(E_AXIS E_OPTARG(eindex)); | ||||||
|  |     if (can_disable) { | ||||||
|  |       #define _CASE_DIS_E(N) case N: DISABLE_AXIS_E##N(); break; | ||||||
|  |       switch (eindex) { REPEAT(E_STEPPERS, _CASE_DIS_E) } | ||||||
|  |     } | ||||||
|  |     return can_disable; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void Stepper::enable_e_steppers() { | ||||||
|  |     #define _ENA_E(N) ENABLE_EXTRUDER(N); | ||||||
|  |     REPEAT(EXTRUDERS, _ENA_E) | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   void Stepper::disable_e_steppers() { | ||||||
|  |     #define _DIS_E(N) DISABLE_EXTRUDER(N); | ||||||
|  |     REPEAT(EXTRUDERS, _DIS_E) | ||||||
|  |   } | ||||||
|  |  | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | void Stepper::enable_all_steppers() { | ||||||
|  |   TERN_(AUTO_POWER_CONTROL, powerManager.power_on()); | ||||||
|  |   LINEAR_AXIS_CODE( | ||||||
|  |     enable_axis(X_AXIS), enable_axis(Y_AXIS), enable_axis(Z_AXIS), | ||||||
|  |     enable_axis(I_AXIS), enable_axis(J_AXIS), enable_axis(K_AXIS) | ||||||
|  |   ); | ||||||
|  |   enable_e_steppers(); | ||||||
|  |  | ||||||
|  |   TERN_(EXTENSIBLE_UI, ExtUI::onSteppersEnabled()); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void Stepper::disable_all_steppers() { | ||||||
|  |   LINEAR_AXIS_CODE( | ||||||
|  |     disable_axis(X_AXIS), disable_axis(Y_AXIS), disable_axis(Z_AXIS), | ||||||
|  |     disable_axis(I_AXIS), disable_axis(J_AXIS), disable_axis(K_AXIS) | ||||||
|  |   ); | ||||||
|  |   disable_e_steppers(); | ||||||
|  |  | ||||||
|  |   TERN_(EXTENSIBLE_UI, ExtUI::onSteppersDisabled()); | ||||||
|  | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Set the stepper direction of each axis |  * Set the stepper direction of each axis | ||||||
|  * |  * | ||||||
| @@ -494,24 +587,12 @@ void Stepper::set_directions() { | |||||||
|       count_direction[_AXIS(A)] = 1;            \ |       count_direction[_AXIS(A)] = 1;            \ | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   #if HAS_X_DIR |   TERN_(HAS_X_DIR, SET_STEP_DIR(X)); // A | ||||||
|     SET_STEP_DIR(X); // A |   TERN_(HAS_Y_DIR, SET_STEP_DIR(Y)); // B | ||||||
|   #endif |   TERN_(HAS_Z_DIR, SET_STEP_DIR(Z)); // C | ||||||
|   #if HAS_Y_DIR |   TERN_(HAS_I_DIR, SET_STEP_DIR(I)); | ||||||
|     SET_STEP_DIR(Y); // B |   TERN_(HAS_J_DIR, SET_STEP_DIR(J)); | ||||||
|   #endif |   TERN_(HAS_K_DIR, SET_STEP_DIR(K)); | ||||||
|   #if HAS_Z_DIR |  | ||||||
|     SET_STEP_DIR(Z); // C |  | ||||||
|   #endif |  | ||||||
|   #if HAS_I_DIR |  | ||||||
|     SET_STEP_DIR(I); |  | ||||||
|   #endif |  | ||||||
|   #if HAS_J_DIR |  | ||||||
|     SET_STEP_DIR(J); |  | ||||||
|   #endif |  | ||||||
|   #if HAS_K_DIR |  | ||||||
|     SET_STEP_DIR(K); |  | ||||||
|   #endif |  | ||||||
|  |  | ||||||
|   #if DISABLED(LIN_ADVANCE) |   #if DISABLED(LIN_ADVANCE) | ||||||
|     #if ENABLED(MIXING_EXTRUDER) |     #if ENABLED(MIXING_EXTRUDER) | ||||||
| @@ -2204,7 +2285,7 @@ uint32_t Stepper::block_phase_isr() { | |||||||
|  |  | ||||||
|       TERN_(MIXING_EXTRUDER, mixer.stepper_setup(current_block->b_color)); |       TERN_(MIXING_EXTRUDER, mixer.stepper_setup(current_block->b_color)); | ||||||
|  |  | ||||||
|       TERN_(HAS_MULTI_EXTRUDER, stepper_extruder = current_block->extruder); |       E_TERN_(stepper_extruder = current_block->extruder); | ||||||
|  |  | ||||||
|       // Initialize the trapezoid generator from the current block. |       // Initialize the trapezoid generator from the current block. | ||||||
|       #if ENABLED(LIN_ADVANCE) |       #if ENABLED(LIN_ADVANCE) | ||||||
| @@ -2227,7 +2308,7 @@ uint32_t Stepper::block_phase_isr() { | |||||||
|         || current_block->direction_bits != last_direction_bits |         || current_block->direction_bits != last_direction_bits | ||||||
|         || TERN(MIXING_EXTRUDER, false, stepper_extruder != last_moved_extruder) |         || TERN(MIXING_EXTRUDER, false, stepper_extruder != last_moved_extruder) | ||||||
|       ) { |       ) { | ||||||
|         TERN_(HAS_MULTI_EXTRUDER, last_moved_extruder = stepper_extruder); |         E_TERN_(last_moved_extruder = stepper_extruder); | ||||||
|         TERN_(HAS_L64XX, L64XX_OK_to_power_up = true); |         TERN_(HAS_L64XX, L64XX_OK_to_power_up = true); | ||||||
|         set_directions(current_block->direction_bits); |         set_directions(current_block->direction_bits); | ||||||
|       } |       } | ||||||
| @@ -2276,7 +2357,7 @@ uint32_t Stepper::block_phase_isr() { | |||||||
|         // If delayed Z enable, enable it now. This option will severely interfere with |         // If delayed Z enable, enable it now. This option will severely interfere with | ||||||
|         // timing between pulses when chaining motion between blocks, and it could lead |         // timing between pulses when chaining motion between blocks, and it could lead | ||||||
|         // to lost steps in both X and Y axis, so avoid using it unless strictly necessary!! |         // to lost steps in both X and Y axis, so avoid using it unless strictly necessary!! | ||||||
|         if (current_block->steps.z) ENABLE_AXIS_Z(); |         if (current_block->steps.z) enable_axis(Z_AXIS); | ||||||
|       #endif |       #endif | ||||||
|  |  | ||||||
|       // Mark the time_nominal as not calculated yet |       // Mark the time_nominal as not calculated yet | ||||||
| @@ -2872,7 +2953,7 @@ void Stepper::report_positions() { | |||||||
|  |  | ||||||
| #if ENABLED(BABYSTEPPING) | #if ENABLED(BABYSTEPPING) | ||||||
|  |  | ||||||
|   #define _ENABLE_AXIS(AXIS) ENABLE_AXIS_## AXIS() |   #define _ENABLE_AXIS(A) enable_axis(_AXIS(A)) | ||||||
|   #define _READ_DIR(AXIS) AXIS ##_DIR_READ() |   #define _READ_DIR(AXIS) AXIS ##_DIR_READ() | ||||||
|   #define _INVERT_DIR(AXIS) INVERT_## AXIS ##_DIR |   #define _INVERT_DIR(AXIS) INVERT_## AXIS ##_DIR | ||||||
|   #define _APPLY_DIR(AXIS, INVERT) AXIS ##_APPLY_DIR(INVERT, true) |   #define _APPLY_DIR(AXIS, INVERT) AXIS ##_APPLY_DIR(INVERT, true) | ||||||
| @@ -3000,8 +3081,10 @@ void Stepper::report_positions() { | |||||||
|  |  | ||||||
|           const bool z_direction = direction ^ BABYSTEP_INVERT_Z; |           const bool z_direction = direction ^ BABYSTEP_INVERT_Z; | ||||||
|  |  | ||||||
|           ENABLE_AXIS_X(); ENABLE_AXIS_Y(); ENABLE_AXIS_Z(); |           LINEAR_AXIS_CODE( | ||||||
|           ENABLE_AXIS_I(); ENABLE_AXIS_J(); ENABLE_AXIS_K(); |             enable_axis(X_AXIS), enable_axis(Y_AXIS), enable_axis(Z_AXIS), | ||||||
|  |             enable_axis(I_AXIS), enable_axis(J_AXIS), enable_axis(K_AXIS) | ||||||
|  |           ); | ||||||
|  |  | ||||||
|           DIR_WAIT_BEFORE(); |           DIR_WAIT_BEFORE(); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -236,6 +236,71 @@ | |||||||
| // Perhaps DISABLE_MULTI_STEPPING should be required with ADAPTIVE_STEP_SMOOTHING. | // Perhaps DISABLE_MULTI_STEPPING should be required with ADAPTIVE_STEP_SMOOTHING. | ||||||
| #define MIN_STEP_ISR_FREQUENCY (MAX_STEP_ISR_FREQUENCY_1X / 2) | #define MIN_STEP_ISR_FREQUENCY (MAX_STEP_ISR_FREQUENCY_1X / 2) | ||||||
|  |  | ||||||
|  | #define ENABLE_COUNT (LINEAR_AXES + E_STEPPERS) | ||||||
|  | typedef IF<(ENABLE_COUNT > 8), uint16_t, uint8_t>::type ena_mask_t; | ||||||
|  |  | ||||||
|  | // Axis flags type, for enabled state or other simple state | ||||||
|  | typedef struct { | ||||||
|  |   union { | ||||||
|  |     ena_mask_t bits; | ||||||
|  |     struct { | ||||||
|  |       bool LINEAR_AXIS_LIST(X:1, Y:1, Z:1, I:1, J:1, K:1); | ||||||
|  |       #if HAS_EXTRUDERS | ||||||
|  |         bool LIST_N(EXTRUDERS, E0:1, E1:1, E2:1, E3:1, E4:1, E5:1, E6:1, E7:1); | ||||||
|  |       #endif | ||||||
|  |     }; | ||||||
|  |   }; | ||||||
|  |   constexpr ena_mask_t linear_bits() { return _BV(LINEAR_AXES) - 1; } | ||||||
|  |   constexpr ena_mask_t e_bits() { return (_BV(EXTRUDERS) - 1) << LINEAR_AXES; } | ||||||
|  | } axis_flags_t; | ||||||
|  |  | ||||||
|  | // All the stepper enable pins | ||||||
|  | constexpr pin_t ena_pins[] = { | ||||||
|  |   LINEAR_AXIS_LIST(X_ENABLE_PIN, Y_ENABLE_PIN, Z_ENABLE_PIN, I_ENABLE_PIN, J_ENABLE_PIN, K_ENABLE_PIN), | ||||||
|  |   LIST_N(E_STEPPERS, E0_ENABLE_PIN, E1_ENABLE_PIN, E2_ENABLE_PIN, E3_ENABLE_PIN, E4_ENABLE_PIN, E5_ENABLE_PIN, E6_ENABLE_PIN, E7_ENABLE_PIN) | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | // Index of the axis or extruder element in a combined array | ||||||
|  | constexpr uint8_t index_of_axis(const AxisEnum axis E_OPTARG(const uint8_t eindex=0)) { | ||||||
|  |   return uint8_t(axis) + (E_TERN0(axis < LINEAR_AXES ? 0 : eindex)); | ||||||
|  | } | ||||||
|  | //#define __IAX_N(N,V...)           _IAX_##N(V) | ||||||
|  | //#define _IAX_N(N,V...)            __IAX_N(N,V) | ||||||
|  | //#define _IAX_1(A)                 index_of_axis(A) | ||||||
|  | //#define _IAX_2(A,B)               index_of_axis(A E_OPTARG(B)) | ||||||
|  | //#define INDEX_OF_AXIS(V...)       _IAX_N(TWO_ARGS(V),V) | ||||||
|  |  | ||||||
|  | #define INDEX_OF_AXIS(A,V...)     index_of_axis(A E_OPTARG(V+0)) | ||||||
|  |  | ||||||
|  | // Bit mask for a matching enable pin, or 0 | ||||||
|  | constexpr ena_mask_t ena_same(const uint8_t a, const uint8_t b) { | ||||||
|  |   return ena_pins[a] == ena_pins[b] ? _BV(b) : 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Recursively get the enable overlaps mask for a given linear axis or extruder | ||||||
|  | constexpr ena_mask_t ena_overlap(const uint8_t a=0, const uint8_t b=0) { | ||||||
|  |   return b >= ENABLE_COUNT ? 0 : (a == b ? 0 : ena_same(a, b)) | ena_overlap(a, b + 1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Recursively get whether there's any overlap at all | ||||||
|  | constexpr bool any_enable_overlap(const uint8_t a=0) { | ||||||
|  |   return a >= ENABLE_COUNT ? false : ena_overlap(a) || any_enable_overlap(a + 1); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Array of axes that overlap with each | ||||||
|  | // TODO: Consider cases where >=2 steppers are used by a linear axis or extruder | ||||||
|  | //       (e.g., CoreXY, Dual XYZ, or E with multiple steppers, etc.). | ||||||
|  | constexpr ena_mask_t enable_overlap[] = { | ||||||
|  |   #define _OVERLAP(N) ena_overlap(INDEX_OF_AXIS(AxisEnum(N))), | ||||||
|  |   REPEAT(LINEAR_AXES, _OVERLAP) | ||||||
|  |   #if HAS_EXTRUDERS | ||||||
|  |     #define _E_OVERLAP(N) ena_overlap(INDEX_OF_AXIS(E_AXIS, N)), | ||||||
|  |     REPEAT(E_STEPPERS, _E_OVERLAP) | ||||||
|  |   #endif | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | //static_assert(!any_enable_overlap(), "There is some overlap."); | ||||||
|  |  | ||||||
| // | // | ||||||
| // Stepper class definition | // Stepper class definition | ||||||
| // | // | ||||||
| @@ -519,6 +584,43 @@ class Stepper { | |||||||
|       static void refresh_motor_power(); |       static void refresh_motor_power(); | ||||||
|     #endif |     #endif | ||||||
|  |  | ||||||
|  |     static axis_flags_t axis_enabled;   // Axis stepper(s) ENABLED states | ||||||
|  |  | ||||||
|  |     static inline bool axis_is_enabled(const AxisEnum axis E_OPTARG(const uint8_t eindex=0)) { | ||||||
|  |       return TEST(axis_enabled.bits, INDEX_OF_AXIS(axis, eindex)); | ||||||
|  |     } | ||||||
|  |     static inline void mark_axis_enabled(const AxisEnum axis E_OPTARG(const uint8_t eindex=0)) { | ||||||
|  |       SBI(axis_enabled.bits, INDEX_OF_AXIS(axis, eindex)); | ||||||
|  |     } | ||||||
|  |     static inline void mark_axis_disabled(const AxisEnum axis E_OPTARG(const uint8_t eindex=0)) { | ||||||
|  |       CBI(axis_enabled.bits, INDEX_OF_AXIS(axis, eindex)); | ||||||
|  |     } | ||||||
|  |     static inline bool can_axis_disable(const AxisEnum axis E_OPTARG(const uint8_t eindex=0)) { | ||||||
|  |       return !any_enable_overlap() || !(axis_enabled.bits & enable_overlap[INDEX_OF_AXIS(axis, eindex)]); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     static void enable_axis(const AxisEnum axis); | ||||||
|  |     static bool disable_axis(const AxisEnum axis); | ||||||
|  |  | ||||||
|  |     #if HAS_EXTRUDERS | ||||||
|  |       static void enable_extruder(E_TERN_(const uint8_t eindex=0)); | ||||||
|  |       static bool disable_extruder(E_TERN_(const uint8_t eindex=0)); | ||||||
|  |       static void enable_e_steppers(); | ||||||
|  |       static void disable_e_steppers(); | ||||||
|  |     #else | ||||||
|  |       static inline void enable_extruder() {} | ||||||
|  |       static inline bool disable_extruder() {} | ||||||
|  |       static inline void enable_e_steppers() {} | ||||||
|  |       static inline void disable_e_steppers() {} | ||||||
|  |     #endif | ||||||
|  |  | ||||||
|  |     #define  ENABLE_EXTRUDER(N)  enable_extruder(E_TERN_(N)) | ||||||
|  |     #define DISABLE_EXTRUDER(N) disable_extruder(E_TERN_(N)) | ||||||
|  |     #define AXIS_IS_ENABLED(N,V...) axis_is_enabled(N E_OPTARG(#V)) | ||||||
|  |  | ||||||
|  |     static void enable_all_steppers(); | ||||||
|  |     static void disable_all_steppers(); | ||||||
|  |  | ||||||
|     // Update direction states for all steppers |     // Update direction states for all steppers | ||||||
|     static void set_directions(); |     static void set_directions(); | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user