Stepper and Endstops as singleton objects
This commit is contained in:
		@@ -216,7 +216,7 @@ void manage_inactivity(bool ignore_stepper_queue = false);
 | 
				
			|||||||
 */
 | 
					 */
 | 
				
			||||||
enum AxisEnum {X_AXIS = 0, A_AXIS = 0, Y_AXIS = 1, B_AXIS = 1, Z_AXIS = 2, C_AXIS = 2, E_AXIS = 3, X_HEAD = 4, Y_HEAD = 5, Z_HEAD = 5};
 | 
					enum AxisEnum {X_AXIS = 0, A_AXIS = 0, Y_AXIS = 1, B_AXIS = 1, Z_AXIS = 2, C_AXIS = 2, E_AXIS = 3, X_HEAD = 4, Y_HEAD = 5, Z_HEAD = 5};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
enum EndstopEnum {X_MIN = 0, Y_MIN = 1, Z_MIN = 2, Z_MIN_PROBE = 3, X_MAX = 4, Y_MAX = 5, Z_MAX = 6, Z2_MIN = 7, Z2_MAX = 8};
 | 
					#define _AXIS(AXIS) AXIS ##_AXIS
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void enable_all_steppers();
 | 
					void enable_all_steppers();
 | 
				
			||||||
void disable_all_steppers();
 | 
					void disable_all_steppers();
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -48,6 +48,7 @@
 | 
				
			|||||||
#include "ultralcd.h"
 | 
					#include "ultralcd.h"
 | 
				
			||||||
#include "planner.h"
 | 
					#include "planner.h"
 | 
				
			||||||
#include "stepper.h"
 | 
					#include "stepper.h"
 | 
				
			||||||
 | 
					#include "endstops.h"
 | 
				
			||||||
#include "temperature.h"
 | 
					#include "temperature.h"
 | 
				
			||||||
#include "cardreader.h"
 | 
					#include "cardreader.h"
 | 
				
			||||||
#include "configuration_store.h"
 | 
					#include "configuration_store.h"
 | 
				
			||||||
@@ -547,10 +548,6 @@ static void report_current_position();
 | 
				
			|||||||
  float extrude_min_temp = EXTRUDE_MINTEMP;
 | 
					  float extrude_min_temp = EXTRUDE_MINTEMP;
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if ENABLED(HAS_Z_MIN_PROBE)
 | 
					 | 
				
			||||||
  extern volatile bool z_probe_is_active;
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#if ENABLED(SDSUPPORT)
 | 
					#if ENABLED(SDSUPPORT)
 | 
				
			||||||
  #include "SdFatUtil.h"
 | 
					  #include "SdFatUtil.h"
 | 
				
			||||||
  int freeMemory() { return SdFatUtil::FreeRam(); }
 | 
					  int freeMemory() { return SdFatUtil::FreeRam(); }
 | 
				
			||||||
@@ -711,7 +708,7 @@ void servo_init() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
   #if HAS_SERVO_ENDSTOPS
 | 
					   #if HAS_SERVO_ENDSTOPS
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    z_probe_is_active = false;
 | 
					    endstops.enable_z_probe(false);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    /**
 | 
					    /**
 | 
				
			||||||
     * Set position of all defined Servo Endstops
 | 
					     * Set position of all defined Servo Endstops
 | 
				
			||||||
@@ -831,7 +828,7 @@ void setup() {
 | 
				
			|||||||
    watchdog_init();
 | 
					    watchdog_init();
 | 
				
			||||||
  #endif
 | 
					  #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  st_init();    // Initialize stepper, this enables interrupts!
 | 
					  stepper.init();    // Initialize stepper, this enables interrupts!
 | 
				
			||||||
  setup_photpin();
 | 
					  setup_photpin();
 | 
				
			||||||
  servo_init();
 | 
					  servo_init();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -915,7 +912,7 @@ void loop() {
 | 
				
			|||||||
    commands_in_queue--;
 | 
					    commands_in_queue--;
 | 
				
			||||||
    cmd_queue_index_r = (cmd_queue_index_r + 1) % BUFSIZE;
 | 
					    cmd_queue_index_r = (cmd_queue_index_r + 1) % BUFSIZE;
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
  checkHitEndstops();
 | 
					  endstops.report_state();
 | 
				
			||||||
  idle();
 | 
					  idle();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1445,9 +1442,9 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
  feedrate_multiplier = 100;
 | 
					  feedrate_multiplier = 100;
 | 
				
			||||||
  refresh_cmd_timeout();
 | 
					  refresh_cmd_timeout();
 | 
				
			||||||
  #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
					  #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
				
			||||||
    if (DEBUGGING(LEVELING)) SERIAL_ECHOLNPGM("setup_for_endstop_move > enable_endstops(true)");
 | 
					    if (DEBUGGING(LEVELING)) SERIAL_ECHOLNPGM("setup_for_endstop_move > endstops.enable()");
 | 
				
			||||||
  #endif
 | 
					  #endif
 | 
				
			||||||
  enable_endstops(true);
 | 
					  endstops.enable();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if ENABLED(AUTO_BED_LEVELING_FEATURE)
 | 
					#if ENABLED(AUTO_BED_LEVELING_FEATURE)
 | 
				
			||||||
@@ -1553,7 +1550,7 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
    #if ENABLED(DELTA)
 | 
					    #if ENABLED(DELTA)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      float start_z = current_position[Z_AXIS];
 | 
					      float start_z = current_position[Z_AXIS];
 | 
				
			||||||
      long start_steps = st_get_position(Z_AXIS);
 | 
					      long start_steps = stepper.position(Z_AXIS);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
					      #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
				
			||||||
        if (DEBUGGING(LEVELING)) SERIAL_ECHOLNPGM("run_z_probe (DELTA) 1");
 | 
					        if (DEBUGGING(LEVELING)) SERIAL_ECHOLNPGM("run_z_probe (DELTA) 1");
 | 
				
			||||||
@@ -1563,14 +1560,14 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
      feedrate = homing_feedrate[Z_AXIS] / 4;
 | 
					      feedrate = homing_feedrate[Z_AXIS] / 4;
 | 
				
			||||||
      destination[Z_AXIS] = -10;
 | 
					      destination[Z_AXIS] = -10;
 | 
				
			||||||
      prepare_move_raw(); // this will also set_current_to_destination
 | 
					      prepare_move_raw(); // this will also set_current_to_destination
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
      endstops_hit_on_purpose(); // clear endstop hit flags
 | 
					      endstops.hit_on_purpose(); // clear endstop hit flags
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      /**
 | 
					      /**
 | 
				
			||||||
       * We have to let the planner know where we are right now as it
 | 
					       * We have to let the planner know where we are right now as it
 | 
				
			||||||
       * is not where we said to go.
 | 
					       * is not where we said to go.
 | 
				
			||||||
       */
 | 
					       */
 | 
				
			||||||
      long stop_steps = st_get_position(Z_AXIS);
 | 
					      long stop_steps = stepper.position(Z_AXIS);
 | 
				
			||||||
      float mm = start_z - float(start_steps - stop_steps) / axis_steps_per_unit[Z_AXIS];
 | 
					      float mm = start_z - float(start_steps - stop_steps) / axis_steps_per_unit[Z_AXIS];
 | 
				
			||||||
      current_position[Z_AXIS] = mm;
 | 
					      current_position[Z_AXIS] = mm;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1588,10 +1585,10 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
      // Move down until the Z probe (or endstop?) is triggered
 | 
					      // Move down until the Z probe (or endstop?) is triggered
 | 
				
			||||||
      float zPosition = -(Z_MAX_LENGTH + 10);
 | 
					      float zPosition = -(Z_MAX_LENGTH + 10);
 | 
				
			||||||
      line_to_z(zPosition);
 | 
					      line_to_z(zPosition);
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      // Tell the planner where we ended up - Get this from the stepper handler
 | 
					      // Tell the planner where we ended up - Get this from the stepper handler
 | 
				
			||||||
      zPosition = st_get_axis_position_mm(Z_AXIS);
 | 
					      zPosition = stepper.get_axis_position_mm(Z_AXIS);
 | 
				
			||||||
      plan_set_position(
 | 
					      plan_set_position(
 | 
				
			||||||
        current_position[X_AXIS], current_position[Y_AXIS], zPosition,
 | 
					        current_position[X_AXIS], current_position[Y_AXIS], zPosition,
 | 
				
			||||||
        current_position[E_AXIS]
 | 
					        current_position[E_AXIS]
 | 
				
			||||||
@@ -1600,19 +1597,19 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
      // move up the retract distance
 | 
					      // move up the retract distance
 | 
				
			||||||
      zPosition += home_bump_mm(Z_AXIS);
 | 
					      zPosition += home_bump_mm(Z_AXIS);
 | 
				
			||||||
      line_to_z(zPosition);
 | 
					      line_to_z(zPosition);
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
      endstops_hit_on_purpose(); // clear endstop hit flags
 | 
					      endstops.hit_on_purpose(); // clear endstop hit flags
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      // move back down slowly to find bed
 | 
					      // move back down slowly to find bed
 | 
				
			||||||
      set_homing_bump_feedrate(Z_AXIS);
 | 
					      set_homing_bump_feedrate(Z_AXIS);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      zPosition -= home_bump_mm(Z_AXIS) * 2;
 | 
					      zPosition -= home_bump_mm(Z_AXIS) * 2;
 | 
				
			||||||
      line_to_z(zPosition);
 | 
					      line_to_z(zPosition);
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
      endstops_hit_on_purpose(); // clear endstop hit flags
 | 
					      endstops.hit_on_purpose(); // clear endstop hit flags
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      // Get the current stepper position after bumping an endstop
 | 
					      // Get the current stepper position after bumping an endstop
 | 
				
			||||||
      current_position[Z_AXIS] = st_get_axis_position_mm(Z_AXIS);
 | 
					      current_position[Z_AXIS] = stepper.get_axis_position_mm(Z_AXIS);
 | 
				
			||||||
      sync_plan_position();
 | 
					      sync_plan_position();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
					      #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
				
			||||||
@@ -1641,7 +1638,7 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
      destination[Y_AXIS] = y;
 | 
					      destination[Y_AXIS] = y;
 | 
				
			||||||
      destination[Z_AXIS] = z;
 | 
					      destination[Z_AXIS] = z;
 | 
				
			||||||
      prepare_move_raw(); // this will also set_current_to_destination
 | 
					      prepare_move_raw(); // this will also set_current_to_destination
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    #else
 | 
					    #else
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1649,14 +1646,14 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
      current_position[Z_AXIS] = z;
 | 
					      current_position[Z_AXIS] = z;
 | 
				
			||||||
      line_to_current_position();
 | 
					      line_to_current_position();
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      feedrate = xy_travel_speed;
 | 
					      feedrate = xy_travel_speed;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      current_position[X_AXIS] = x;
 | 
					      current_position[X_AXIS] = x;
 | 
				
			||||||
      current_position[Y_AXIS] = y;
 | 
					      current_position[Y_AXIS] = y;
 | 
				
			||||||
      line_to_current_position();
 | 
					      line_to_current_position();
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1681,9 +1678,9 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  static void clean_up_after_endstop_move() {
 | 
					  static void clean_up_after_endstop_move() {
 | 
				
			||||||
    #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
					    #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
				
			||||||
      if (DEBUGGING(LEVELING)) SERIAL_ECHOLNPGM("clean_up_after_endstop_move > ENDSTOPS_ONLY_FOR_HOMING > endstops_not_homing()");
 | 
					      if (DEBUGGING(LEVELING)) SERIAL_ECHOLNPGM("clean_up_after_endstop_move > ENDSTOPS_ONLY_FOR_HOMING > endstops.not_homing()");
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
    endstops_not_homing();
 | 
					    endstops.not_homing();
 | 
				
			||||||
    feedrate = saved_feedrate;
 | 
					    feedrate = saved_feedrate;
 | 
				
			||||||
    feedrate_multiplier = saved_feedrate_multiplier;
 | 
					    feedrate_multiplier = saved_feedrate_multiplier;
 | 
				
			||||||
    refresh_cmd_timeout();
 | 
					    refresh_cmd_timeout();
 | 
				
			||||||
@@ -1697,7 +1694,7 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
      if (DEBUGGING(LEVELING)) DEBUG_POS("deploy_z_probe", current_position);
 | 
					      if (DEBUGGING(LEVELING)) DEBUG_POS("deploy_z_probe", current_position);
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (z_probe_is_active) return;
 | 
					    if (endstops.z_probe_enabled) return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    #if HAS_SERVO_ENDSTOPS
 | 
					    #if HAS_SERVO_ENDSTOPS
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1757,7 +1754,7 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
      destination[Y_AXIS] = destination[Y_AXIS] * 0.75;
 | 
					      destination[Y_AXIS] = destination[Y_AXIS] * 0.75;
 | 
				
			||||||
      prepare_move_raw(); // this will also set_current_to_destination
 | 
					      prepare_move_raw(); // this will also set_current_to_destination
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      #if ENABLED(Z_MIN_PROBE_ENDSTOP)
 | 
					      #if ENABLED(Z_MIN_PROBE_ENDSTOP)
 | 
				
			||||||
        z_probe_endstop = (READ(Z_MIN_PROBE_PIN) != Z_MIN_PROBE_ENDSTOP_INVERTING);
 | 
					        z_probe_endstop = (READ(Z_MIN_PROBE_PIN) != Z_MIN_PROBE_ENDSTOP_INVERTING);
 | 
				
			||||||
@@ -1778,10 +1775,10 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
    #endif // Z_PROBE_ALLEN_KEY
 | 
					    #endif // Z_PROBE_ALLEN_KEY
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    #if ENABLED(FIX_MOUNTED_PROBE)
 | 
					    #if ENABLED(FIX_MOUNTED_PROBE)
 | 
				
			||||||
      // Noting to be done. Just set z_probe_is_active
 | 
					      // Noting to be done. Just set endstops.z_probe_enabled
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    z_probe_is_active = true;
 | 
					    endstops.enable_z_probe();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1793,7 +1790,7 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
      if (DEBUGGING(LEVELING)) DEBUG_POS("stow_z_probe", current_position);
 | 
					      if (DEBUGGING(LEVELING)) DEBUG_POS("stow_z_probe", current_position);
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (!z_probe_is_active) return;
 | 
					    if (!endstops.z_probe_enabled) return;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    #if HAS_SERVO_ENDSTOPS
 | 
					    #if HAS_SERVO_ENDSTOPS
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1811,7 +1808,7 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
              }
 | 
					              }
 | 
				
			||||||
            #endif
 | 
					            #endif
 | 
				
			||||||
            raise_z_after_probing(); // this also updates current_position
 | 
					            raise_z_after_probing(); // this also updates current_position
 | 
				
			||||||
            st_synchronize();
 | 
					            stepper.synchronize();
 | 
				
			||||||
          }
 | 
					          }
 | 
				
			||||||
        #endif
 | 
					        #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1861,7 +1858,7 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
      destination[Y_AXIS] = 0;
 | 
					      destination[Y_AXIS] = 0;
 | 
				
			||||||
      prepare_move_raw(); // this will also set_current_to_destination
 | 
					      prepare_move_raw(); // this will also set_current_to_destination
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      #if ENABLED(Z_MIN_PROBE_ENDSTOP)
 | 
					      #if ENABLED(Z_MIN_PROBE_ENDSTOP)
 | 
				
			||||||
        bool z_probe_endstop = (READ(Z_MIN_PROBE_PIN) != Z_MIN_PROBE_ENDSTOP_INVERTING);
 | 
					        bool z_probe_endstop = (READ(Z_MIN_PROBE_PIN) != Z_MIN_PROBE_ENDSTOP_INVERTING);
 | 
				
			||||||
@@ -1881,10 +1878,10 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
    #endif // Z_PROBE_ALLEN_KEY
 | 
					    #endif // Z_PROBE_ALLEN_KEY
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    #if ENABLED(FIX_MOUNTED_PROBE)
 | 
					    #if ENABLED(FIX_MOUNTED_PROBE)
 | 
				
			||||||
      // Nothing to do here. Just clear z_probe_is_active
 | 
					      // Nothing to do here. Just clear endstops.z_probe_enabled
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    z_probe_is_active = false;
 | 
					    endstops.enable_z_probe(false);
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
  #endif // HAS_Z_MIN_PROBE
 | 
					  #endif // HAS_Z_MIN_PROBE
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -2081,13 +2078,13 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
      }
 | 
					      }
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    if (z_probe_is_active == dock) return;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    if (!axis_homed[X_AXIS] || !axis_homed[Y_AXIS]) {
 | 
					    if (!axis_homed[X_AXIS] || !axis_homed[Y_AXIS]) {
 | 
				
			||||||
      axis_unhomed_error();
 | 
					      axis_unhomed_error();
 | 
				
			||||||
      return;
 | 
					      return;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    if (endstops.z_probe_enabled == !dock) return; // already docked/undocked?
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    float oldXpos = current_position[X_AXIS]; // save x position
 | 
					    float oldXpos = current_position[X_AXIS]; // save x position
 | 
				
			||||||
    if (dock) {
 | 
					    if (dock) {
 | 
				
			||||||
      #if Z_RAISE_AFTER_PROBING > 0
 | 
					      #if Z_RAISE_AFTER_PROBING > 0
 | 
				
			||||||
@@ -2105,7 +2102,7 @@ static void setup_for_endstop_move() {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
    do_blocking_move_to_x(oldXpos); // return to position before docking
 | 
					    do_blocking_move_to_x(oldXpos); // return to position before docking
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    z_probe_is_active = dock;
 | 
					    endstops.enable_z_probe(!dock); // logically disable docked probe
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#endif // Z_PROBE_SLED
 | 
					#endif // Z_PROBE_SLED
 | 
				
			||||||
@@ -2167,39 +2164,39 @@ static void homeaxis(AxisEnum axis) {
 | 
				
			|||||||
      // Engage an X or Y Servo endstop if enabled
 | 
					      // Engage an X or Y Servo endstop if enabled
 | 
				
			||||||
      if (_Z_SERVO_TEST && servo_endstop_id[axis] >= 0) {
 | 
					      if (_Z_SERVO_TEST && servo_endstop_id[axis] >= 0) {
 | 
				
			||||||
        servo[servo_endstop_id[axis]].move(servo_endstop_angle[axis][0]);
 | 
					        servo[servo_endstop_id[axis]].move(servo_endstop_angle[axis][0]);
 | 
				
			||||||
        if (_Z_PROBE_SUBTEST) z_probe_is_active = true;
 | 
					        if (_Z_PROBE_SUBTEST) endstops.z_probe_enabled = true;
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // Set a flag for Z motor locking
 | 
					    // Set a flag for Z motor locking
 | 
				
			||||||
    #if ENABLED(Z_DUAL_ENDSTOPS)
 | 
					    #if ENABLED(Z_DUAL_ENDSTOPS)
 | 
				
			||||||
      if (axis == Z_AXIS) In_Homing_Process(true);
 | 
					      if (axis == Z_AXIS) stepper.set_homing_flag(true);
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // Move towards the endstop until an endstop is triggered
 | 
					    // Move towards the endstop until an endstop is triggered
 | 
				
			||||||
    destination[axis] = 1.5 * max_length(axis) * axis_home_dir;
 | 
					    destination[axis] = 1.5 * max_length(axis) * axis_home_dir;
 | 
				
			||||||
    feedrate = homing_feedrate[axis];
 | 
					    feedrate = homing_feedrate[axis];
 | 
				
			||||||
    line_to_destination();
 | 
					    line_to_destination();
 | 
				
			||||||
    st_synchronize();
 | 
					    stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // Set the axis position as setup for the move
 | 
					    // Set the axis position as setup for the move
 | 
				
			||||||
    current_position[axis] = 0;
 | 
					    current_position[axis] = 0;
 | 
				
			||||||
    sync_plan_position();
 | 
					    sync_plan_position();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
					    #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
				
			||||||
      if (DEBUGGING(LEVELING)) SERIAL_ECHOLNPGM("> enable_endstops(false)");
 | 
					      if (DEBUGGING(LEVELING)) SERIAL_ECHOLNPGM("> endstops.enable(false)");
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
    enable_endstops(false); // Disable endstops while moving away
 | 
					    endstops.enable(false); // Disable endstops while moving away
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // Move away from the endstop by the axis HOME_BUMP_MM
 | 
					    // Move away from the endstop by the axis HOME_BUMP_MM
 | 
				
			||||||
    destination[axis] = -home_bump_mm(axis) * axis_home_dir;
 | 
					    destination[axis] = -home_bump_mm(axis) * axis_home_dir;
 | 
				
			||||||
    line_to_destination();
 | 
					    line_to_destination();
 | 
				
			||||||
    st_synchronize();
 | 
					    stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
					    #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
				
			||||||
      if (DEBUGGING(LEVELING)) SERIAL_ECHOLNPGM("> enable_endstops(true)");
 | 
					      if (DEBUGGING(LEVELING)) SERIAL_ECHOLNPGM("> endstops.enable(true)");
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
    enable_endstops(true); // Enable endstops for next homing move
 | 
					    endstops.enable(true); // Enable endstops for next homing move
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // Slow down the feedrate for the next move
 | 
					    // Slow down the feedrate for the next move
 | 
				
			||||||
    set_homing_bump_feedrate(axis);
 | 
					    set_homing_bump_feedrate(axis);
 | 
				
			||||||
@@ -2207,7 +2204,7 @@ static void homeaxis(AxisEnum axis) {
 | 
				
			|||||||
    // Move slowly towards the endstop until triggered
 | 
					    // Move slowly towards the endstop until triggered
 | 
				
			||||||
    destination[axis] = 2 * home_bump_mm(axis) * axis_home_dir;
 | 
					    destination[axis] = 2 * home_bump_mm(axis) * axis_home_dir;
 | 
				
			||||||
    line_to_destination();
 | 
					    line_to_destination();
 | 
				
			||||||
    st_synchronize();
 | 
					    stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
					    #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
				
			||||||
      if (DEBUGGING(LEVELING)) DEBUG_POS("> TRIGGER ENDSTOP", current_position);
 | 
					      if (DEBUGGING(LEVELING)) DEBUG_POS("> TRIGGER ENDSTOP", current_position);
 | 
				
			||||||
@@ -2224,17 +2221,17 @@ static void homeaxis(AxisEnum axis) {
 | 
				
			|||||||
        else
 | 
					        else
 | 
				
			||||||
          lockZ1 = (z_endstop_adj < 0);
 | 
					          lockZ1 = (z_endstop_adj < 0);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (lockZ1) Lock_z_motor(true); else Lock_z2_motor(true);
 | 
					        if (lockZ1) stepper.set_z_lock(true); else stepper.set_z2_lock(true);
 | 
				
			||||||
        sync_plan_position();
 | 
					        sync_plan_position();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        // Move to the adjusted endstop height
 | 
					        // Move to the adjusted endstop height
 | 
				
			||||||
        feedrate = homing_feedrate[axis];
 | 
					        feedrate = homing_feedrate[axis];
 | 
				
			||||||
        destination[Z_AXIS] = adj;
 | 
					        destination[Z_AXIS] = adj;
 | 
				
			||||||
        line_to_destination();
 | 
					        line_to_destination();
 | 
				
			||||||
        st_synchronize();
 | 
					        stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (lockZ1) Lock_z_motor(false); else Lock_z2_motor(false);
 | 
					        if (lockZ1) stepper.set_z_lock(false); else stepper.set_z2_lock(false);
 | 
				
			||||||
        In_Homing_Process(false);
 | 
					        stepper.set_homing_flag(false);
 | 
				
			||||||
      } // Z_AXIS
 | 
					      } // Z_AXIS
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -2242,9 +2239,9 @@ static void homeaxis(AxisEnum axis) {
 | 
				
			|||||||
      // retrace by the amount specified in endstop_adj
 | 
					      // retrace by the amount specified in endstop_adj
 | 
				
			||||||
      if (endstop_adj[axis] * axis_home_dir < 0) {
 | 
					      if (endstop_adj[axis] * axis_home_dir < 0) {
 | 
				
			||||||
        #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
					        #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
				
			||||||
          if (DEBUGGING(LEVELING)) SERIAL_ECHOLNPGM("> enable_endstops(false)");
 | 
					          if (DEBUGGING(LEVELING)) SERIAL_ECHOLNPGM("> endstops.enable(false)");
 | 
				
			||||||
        #endif
 | 
					        #endif
 | 
				
			||||||
        enable_endstops(false); // Disable endstops while moving away
 | 
					        endstops.enable(false); // Disable endstops while moving away
 | 
				
			||||||
        sync_plan_position();
 | 
					        sync_plan_position();
 | 
				
			||||||
        destination[axis] = endstop_adj[axis];
 | 
					        destination[axis] = endstop_adj[axis];
 | 
				
			||||||
        #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
					        #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
				
			||||||
@@ -2254,11 +2251,11 @@ static void homeaxis(AxisEnum axis) {
 | 
				
			|||||||
          }
 | 
					          }
 | 
				
			||||||
        #endif
 | 
					        #endif
 | 
				
			||||||
        line_to_destination();
 | 
					        line_to_destination();
 | 
				
			||||||
        st_synchronize();
 | 
					        stepper.synchronize();
 | 
				
			||||||
        #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
					        #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
				
			||||||
          if (DEBUGGING(LEVELING)) SERIAL_ECHOLNPGM("> enable_endstops(true)");
 | 
					          if (DEBUGGING(LEVELING)) SERIAL_ECHOLNPGM("> endstops.enable(true)");
 | 
				
			||||||
        #endif
 | 
					        #endif
 | 
				
			||||||
        enable_endstops(true); // Enable endstops for next homing move
 | 
					        endstops.enable(true); // Enable endstops for next homing move
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
      #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
					      #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
				
			||||||
        else {
 | 
					        else {
 | 
				
			||||||
@@ -2280,7 +2277,7 @@ static void homeaxis(AxisEnum axis) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    destination[axis] = current_position[axis];
 | 
					    destination[axis] = current_position[axis];
 | 
				
			||||||
    feedrate = 0.0;
 | 
					    feedrate = 0.0;
 | 
				
			||||||
    endstops_hit_on_purpose(); // clear endstop hit flags
 | 
					    endstops.hit_on_purpose(); // clear endstop hit flags
 | 
				
			||||||
    axis_known_position[axis] = true;
 | 
					    axis_known_position[axis] = true;
 | 
				
			||||||
    axis_homed[axis] = true;
 | 
					    axis_homed[axis] = true;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -2301,7 +2298,7 @@ static void homeaxis(AxisEnum axis) {
 | 
				
			|||||||
          if (DEBUGGING(LEVELING)) SERIAL_ECHOLNPGM("> SERVO_ENDSTOPS > Stow with servo.move()");
 | 
					          if (DEBUGGING(LEVELING)) SERIAL_ECHOLNPGM("> SERVO_ENDSTOPS > Stow with servo.move()");
 | 
				
			||||||
        #endif
 | 
					        #endif
 | 
				
			||||||
        servo[servo_endstop_id[axis]].move(servo_endstop_angle[axis][1]);
 | 
					        servo[servo_endstop_id[axis]].move(servo_endstop_angle[axis][1]);
 | 
				
			||||||
        if (_Z_PROBE_SUBTEST) z_probe_is_active = false;
 | 
					        if (_Z_PROBE_SUBTEST) endstops.enable_z_probe(false);
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -2499,7 +2496,7 @@ inline void gcode_G4() {
 | 
				
			|||||||
  if (code_seen('P')) codenum = code_value_long(); // milliseconds to wait
 | 
					  if (code_seen('P')) codenum = code_value_long(); // milliseconds to wait
 | 
				
			||||||
  if (code_seen('S')) codenum = code_value() * 1000UL; // seconds to wait
 | 
					  if (code_seen('S')) codenum = code_value() * 1000UL; // seconds to wait
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  st_synchronize();
 | 
					  stepper.synchronize();
 | 
				
			||||||
  refresh_cmd_timeout();
 | 
					  refresh_cmd_timeout();
 | 
				
			||||||
  codenum += previous_cmd_ms;  // keep track of when we started waiting
 | 
					  codenum += previous_cmd_ms;  // keep track of when we started waiting
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -2551,7 +2548,7 @@ inline void gcode_G28() {
 | 
				
			|||||||
  #endif
 | 
					  #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  // Wait for planner moves to finish!
 | 
					  // Wait for planner moves to finish!
 | 
				
			||||||
  st_synchronize();
 | 
					  stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  // For auto bed leveling, clear the level matrix
 | 
					  // For auto bed leveling, clear the level matrix
 | 
				
			||||||
  #if ENABLED(AUTO_BED_LEVELING_FEATURE)
 | 
					  #if ENABLED(AUTO_BED_LEVELING_FEATURE)
 | 
				
			||||||
@@ -2594,8 +2591,8 @@ inline void gcode_G28() {
 | 
				
			|||||||
    for (int i = X_AXIS; i <= Z_AXIS; i++) destination[i] = 3 * (Z_MAX_LENGTH);
 | 
					    for (int i = X_AXIS; i <= Z_AXIS; i++) destination[i] = 3 * (Z_MAX_LENGTH);
 | 
				
			||||||
    feedrate = 1.732 * homing_feedrate[X_AXIS];
 | 
					    feedrate = 1.732 * homing_feedrate[X_AXIS];
 | 
				
			||||||
    line_to_destination();
 | 
					    line_to_destination();
 | 
				
			||||||
    st_synchronize();
 | 
					    stepper.synchronize();
 | 
				
			||||||
    endstops_hit_on_purpose(); // clear endstop hit flags
 | 
					    endstops.hit_on_purpose(); // clear endstop hit flags
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // Destination reached
 | 
					    // Destination reached
 | 
				
			||||||
    for (int i = X_AXIS; i <= Z_AXIS; i++) current_position[i] = destination[i];
 | 
					    for (int i = X_AXIS; i <= Z_AXIS; i++) current_position[i] = destination[i];
 | 
				
			||||||
@@ -2643,7 +2640,7 @@ inline void gcode_G28() {
 | 
				
			|||||||
          }
 | 
					          }
 | 
				
			||||||
        #endif
 | 
					        #endif
 | 
				
			||||||
        line_to_destination();
 | 
					        line_to_destination();
 | 
				
			||||||
        st_synchronize();
 | 
					        stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        /**
 | 
					        /**
 | 
				
			||||||
         * Update the current Z position even if it currently not real from
 | 
					         * Update the current Z position even if it currently not real from
 | 
				
			||||||
@@ -2676,7 +2673,7 @@ inline void gcode_G28() {
 | 
				
			|||||||
        destination[Y_AXIS] = 1.5 * mly * home_dir(Y_AXIS);
 | 
					        destination[Y_AXIS] = 1.5 * mly * home_dir(Y_AXIS);
 | 
				
			||||||
        feedrate = min(homing_feedrate[X_AXIS], homing_feedrate[Y_AXIS]) * sqrt(mlratio * mlratio + 1);
 | 
					        feedrate = min(homing_feedrate[X_AXIS], homing_feedrate[Y_AXIS]) * sqrt(mlratio * mlratio + 1);
 | 
				
			||||||
        line_to_destination();
 | 
					        line_to_destination();
 | 
				
			||||||
        st_synchronize();
 | 
					        stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        set_axis_is_at_home(X_AXIS);
 | 
					        set_axis_is_at_home(X_AXIS);
 | 
				
			||||||
        set_axis_is_at_home(Y_AXIS);
 | 
					        set_axis_is_at_home(Y_AXIS);
 | 
				
			||||||
@@ -2690,8 +2687,8 @@ inline void gcode_G28() {
 | 
				
			|||||||
        destination[Y_AXIS] = current_position[Y_AXIS];
 | 
					        destination[Y_AXIS] = current_position[Y_AXIS];
 | 
				
			||||||
        line_to_destination();
 | 
					        line_to_destination();
 | 
				
			||||||
        feedrate = 0.0;
 | 
					        feedrate = 0.0;
 | 
				
			||||||
        st_synchronize();
 | 
					        stepper.synchronize();
 | 
				
			||||||
        endstops_hit_on_purpose(); // clear endstop hit flags
 | 
					        endstops.hit_on_purpose(); // clear endstop hit flags
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        current_position[X_AXIS] = destination[X_AXIS];
 | 
					        current_position[X_AXIS] = destination[X_AXIS];
 | 
				
			||||||
        current_position[Y_AXIS] = destination[Y_AXIS];
 | 
					        current_position[Y_AXIS] = destination[Y_AXIS];
 | 
				
			||||||
@@ -2784,7 +2781,7 @@ inline void gcode_G28() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
            // Move in the XY plane
 | 
					            // Move in the XY plane
 | 
				
			||||||
            line_to_destination();
 | 
					            line_to_destination();
 | 
				
			||||||
            st_synchronize();
 | 
					            stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            /**
 | 
					            /**
 | 
				
			||||||
             * Update the current positions for XY, Z is still at least at
 | 
					             * Update the current positions for XY, Z is still at least at
 | 
				
			||||||
@@ -2857,10 +2854,10 @@ inline void gcode_G28() {
 | 
				
			|||||||
  #endif
 | 
					  #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  #if ENABLED(ENDSTOPS_ONLY_FOR_HOMING)
 | 
					  #if ENABLED(ENDSTOPS_ONLY_FOR_HOMING)
 | 
				
			||||||
    enable_endstops(false);
 | 
					    endstops.enable(false);
 | 
				
			||||||
    #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
					    #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
				
			||||||
      if (DEBUGGING(LEVELING)) {
 | 
					      if (DEBUGGING(LEVELING)) {
 | 
				
			||||||
        SERIAL_ECHOLNPGM("ENDSTOPS_ONLY_FOR_HOMING enable_endstops(false)");
 | 
					        SERIAL_ECHOLNPGM("ENDSTOPS_ONLY_FOR_HOMING endstops.enable(false)");
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
  #endif
 | 
					  #endif
 | 
				
			||||||
@@ -2875,7 +2872,7 @@ inline void gcode_G28() {
 | 
				
			|||||||
      set_destination_to_current();
 | 
					      set_destination_to_current();
 | 
				
			||||||
      feedrate = homing_feedrate[Z_AXIS];
 | 
					      feedrate = homing_feedrate[Z_AXIS];
 | 
				
			||||||
      line_to_destination();
 | 
					      line_to_destination();
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
      #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
					      #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
				
			||||||
        if (DEBUGGING(LEVELING)) DEBUG_POS("mbl_was_active", current_position);
 | 
					        if (DEBUGGING(LEVELING)) DEBUG_POS("mbl_was_active", current_position);
 | 
				
			||||||
      #endif
 | 
					      #endif
 | 
				
			||||||
@@ -2885,7 +2882,7 @@ inline void gcode_G28() {
 | 
				
			|||||||
  feedrate = saved_feedrate;
 | 
					  feedrate = saved_feedrate;
 | 
				
			||||||
  feedrate_multiplier = saved_feedrate_multiplier;
 | 
					  feedrate_multiplier = saved_feedrate_multiplier;
 | 
				
			||||||
  refresh_cmd_timeout();
 | 
					  refresh_cmd_timeout();
 | 
				
			||||||
  endstops_hit_on_purpose(); // clear endstop hit flags
 | 
					  endstops.hit_on_purpose(); // clear endstop hit flags
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
					  #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
				
			||||||
    if (DEBUGGING(LEVELING)) {
 | 
					    if (DEBUGGING(LEVELING)) {
 | 
				
			||||||
@@ -2921,7 +2918,7 @@ inline void gcode_G28() {
 | 
				
			|||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    feedrate = saved_feedrate;
 | 
					    feedrate = saved_feedrate;
 | 
				
			||||||
    st_synchronize();
 | 
					    stepper.synchronize();
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /**
 | 
					  /**
 | 
				
			||||||
@@ -3015,7 +3012,7 @@ inline void gcode_G28() {
 | 
				
			|||||||
            #endif
 | 
					            #endif
 | 
				
			||||||
          ;
 | 
					          ;
 | 
				
			||||||
          line_to_current_position();
 | 
					          line_to_current_position();
 | 
				
			||||||
          st_synchronize();
 | 
					          stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
          // After recording the last point, activate the mbl and home
 | 
					          // After recording the last point, activate the mbl and home
 | 
				
			||||||
          SERIAL_PROTOCOLLNPGM("Mesh probing done.");
 | 
					          SERIAL_PROTOCOLLNPGM("Mesh probing done.");
 | 
				
			||||||
@@ -3240,7 +3237,7 @@ inline void gcode_G28() {
 | 
				
			|||||||
      deploy_z_probe();
 | 
					      deploy_z_probe();
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    st_synchronize();
 | 
					    stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    setup_for_endstop_move();
 | 
					    setup_for_endstop_move();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -3511,7 +3508,7 @@ inline void gcode_G28() {
 | 
				
			|||||||
        float x_tmp = current_position[X_AXIS] + X_PROBE_OFFSET_FROM_EXTRUDER,
 | 
					        float x_tmp = current_position[X_AXIS] + X_PROBE_OFFSET_FROM_EXTRUDER,
 | 
				
			||||||
              y_tmp = current_position[Y_AXIS] + Y_PROBE_OFFSET_FROM_EXTRUDER,
 | 
					              y_tmp = current_position[Y_AXIS] + Y_PROBE_OFFSET_FROM_EXTRUDER,
 | 
				
			||||||
              z_tmp = current_position[Z_AXIS],
 | 
					              z_tmp = current_position[Z_AXIS],
 | 
				
			||||||
              real_z = st_get_axis_position_mm(Z_AXIS);  //get the real Z (since plan_get_position is now correcting the plane)
 | 
					              real_z = stepper.get_axis_position_mm(Z_AXIS);  //get the real Z (since plan_get_position is now correcting the plane)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
					        #if ENABLED(DEBUG_LEVELING_FEATURE)
 | 
				
			||||||
          if (DEBUGGING(LEVELING)) {
 | 
					          if (DEBUGGING(LEVELING)) {
 | 
				
			||||||
@@ -3588,9 +3585,9 @@ inline void gcode_G28() {
 | 
				
			|||||||
      #endif
 | 
					      #endif
 | 
				
			||||||
      enqueue_and_echo_commands_P(PSTR(Z_PROBE_END_SCRIPT));
 | 
					      enqueue_and_echo_commands_P(PSTR(Z_PROBE_END_SCRIPT));
 | 
				
			||||||
      #if ENABLED(HAS_Z_MIN_PROBE)
 | 
					      #if ENABLED(HAS_Z_MIN_PROBE)
 | 
				
			||||||
        z_probe_is_active = false;
 | 
					        endstops.enable_z_probe(false);
 | 
				
			||||||
      #endif
 | 
					      #endif
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    KEEPALIVE_STATE(IN_HANDLER);
 | 
					    KEEPALIVE_STATE(IN_HANDLER);
 | 
				
			||||||
@@ -3615,7 +3612,7 @@ inline void gcode_G28() {
 | 
				
			|||||||
      #endif
 | 
					      #endif
 | 
				
			||||||
      deploy_z_probe(); // Engage Z Servo endstop if available. Z_PROBE_SLED is missed here.
 | 
					      deploy_z_probe(); // Engage Z Servo endstop if available. Z_PROBE_SLED is missed here.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
      // TODO: clear the leveling matrix or the planner will be set incorrectly
 | 
					      // TODO: clear the leveling matrix or the planner will be set incorrectly
 | 
				
			||||||
      setup_for_endstop_move(); // Too late. Must be done before deploying.
 | 
					      setup_for_endstop_move(); // Too late. Must be done before deploying.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -3650,7 +3647,7 @@ inline void gcode_G28() {
 | 
				
			|||||||
inline void gcode_G92() {
 | 
					inline void gcode_G92() {
 | 
				
			||||||
  bool didE = code_seen(axis_codes[E_AXIS]);
 | 
					  bool didE = code_seen(axis_codes[E_AXIS]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  if (!didE) st_synchronize();
 | 
					  if (!didE) stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  bool didXYZ = false;
 | 
					  bool didXYZ = false;
 | 
				
			||||||
  for (int i = 0; i < NUM_AXIS; i++) {
 | 
					  for (int i = 0; i < NUM_AXIS; i++) {
 | 
				
			||||||
@@ -3712,7 +3709,7 @@ inline void gcode_G92() {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    lcd_ignore_click();
 | 
					    lcd_ignore_click();
 | 
				
			||||||
    st_synchronize();
 | 
					    stepper.synchronize();
 | 
				
			||||||
    refresh_cmd_timeout();
 | 
					    refresh_cmd_timeout();
 | 
				
			||||||
    if (codenum > 0) {
 | 
					    if (codenum > 0) {
 | 
				
			||||||
      codenum += previous_cmd_ms;  // wait until this time for a click
 | 
					      codenum += previous_cmd_ms;  // wait until this time for a click
 | 
				
			||||||
@@ -3853,7 +3850,7 @@ inline void gcode_M31() {
 | 
				
			|||||||
   */
 | 
					   */
 | 
				
			||||||
  inline void gcode_M32() {
 | 
					  inline void gcode_M32() {
 | 
				
			||||||
    if (card.sdprinting)
 | 
					    if (card.sdprinting)
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    char* namestartpos = strchr(current_command_args, '!');  // Find ! to indicate filename string start.
 | 
					    char* namestartpos = strchr(current_command_args, '!');  // Find ! to indicate filename string start.
 | 
				
			||||||
    if (!namestartpos)
 | 
					    if (!namestartpos)
 | 
				
			||||||
@@ -4819,7 +4816,7 @@ inline void gcode_M140() {
 | 
				
			|||||||
 */
 | 
					 */
 | 
				
			||||||
inline void gcode_M81() {
 | 
					inline void gcode_M81() {
 | 
				
			||||||
  disable_all_heaters();
 | 
					  disable_all_heaters();
 | 
				
			||||||
  finishAndDisableSteppers();
 | 
					  stepper.finish_and_disable();
 | 
				
			||||||
  #if FAN_COUNT > 0
 | 
					  #if FAN_COUNT > 0
 | 
				
			||||||
    #if FAN_COUNT > 1
 | 
					    #if FAN_COUNT > 1
 | 
				
			||||||
      for (uint8_t i = 0; i < FAN_COUNT; i++) fanSpeeds[i] = 0;
 | 
					      for (uint8_t i = 0; i < FAN_COUNT; i++) fanSpeeds[i] = 0;
 | 
				
			||||||
@@ -4829,7 +4826,7 @@ inline void gcode_M81() {
 | 
				
			|||||||
  #endif
 | 
					  #endif
 | 
				
			||||||
  delay(1000); // Wait 1 second before switching off
 | 
					  delay(1000); // Wait 1 second before switching off
 | 
				
			||||||
  #if HAS_SUICIDE
 | 
					  #if HAS_SUICIDE
 | 
				
			||||||
    st_synchronize();
 | 
					    stepper.synchronize();
 | 
				
			||||||
    suicide();
 | 
					    suicide();
 | 
				
			||||||
  #elif HAS_POWER_SWITCH
 | 
					  #elif HAS_POWER_SWITCH
 | 
				
			||||||
    OUT_WRITE(PS_ON_PIN, PS_ON_ASLEEP);
 | 
					    OUT_WRITE(PS_ON_PIN, PS_ON_ASLEEP);
 | 
				
			||||||
@@ -4864,10 +4861,10 @@ inline void gcode_M18_M84() {
 | 
				
			|||||||
  else {
 | 
					  else {
 | 
				
			||||||
    bool all_axis = !((code_seen(axis_codes[X_AXIS])) || (code_seen(axis_codes[Y_AXIS])) || (code_seen(axis_codes[Z_AXIS])) || (code_seen(axis_codes[E_AXIS])));
 | 
					    bool all_axis = !((code_seen(axis_codes[X_AXIS])) || (code_seen(axis_codes[Y_AXIS])) || (code_seen(axis_codes[Z_AXIS])) || (code_seen(axis_codes[E_AXIS])));
 | 
				
			||||||
    if (all_axis) {
 | 
					    if (all_axis) {
 | 
				
			||||||
      finishAndDisableSteppers();
 | 
					      stepper.finish_and_disable();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    else {
 | 
					    else {
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
      if (code_seen('X')) disable_x();
 | 
					      if (code_seen('X')) disable_x();
 | 
				
			||||||
      if (code_seen('Y')) disable_y();
 | 
					      if (code_seen('Y')) disable_y();
 | 
				
			||||||
      if (code_seen('Z')) disable_z();
 | 
					      if (code_seen('Z')) disable_z();
 | 
				
			||||||
@@ -4927,35 +4924,7 @@ static void report_current_position() {
 | 
				
			|||||||
  SERIAL_PROTOCOLPGM(" E:");
 | 
					  SERIAL_PROTOCOLPGM(" E:");
 | 
				
			||||||
  SERIAL_PROTOCOL(current_position[E_AXIS]);
 | 
					  SERIAL_PROTOCOL(current_position[E_AXIS]);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  CRITICAL_SECTION_START;
 | 
					  stepper.report_positions();
 | 
				
			||||||
  extern volatile long count_position[NUM_AXIS];
 | 
					 | 
				
			||||||
  long xpos = count_position[X_AXIS],
 | 
					 | 
				
			||||||
       ypos = count_position[Y_AXIS],
 | 
					 | 
				
			||||||
       zpos = count_position[Z_AXIS];
 | 
					 | 
				
			||||||
  CRITICAL_SECTION_END;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if ENABLED(COREXY) || ENABLED(COREXZ)
 | 
					 | 
				
			||||||
    SERIAL_PROTOCOLPGM(MSG_COUNT_A);
 | 
					 | 
				
			||||||
  #else
 | 
					 | 
				
			||||||
    SERIAL_PROTOCOLPGM(MSG_COUNT_X);
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
  SERIAL_PROTOCOL(xpos);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if ENABLED(COREXY)
 | 
					 | 
				
			||||||
    SERIAL_PROTOCOLPGM(" B:");
 | 
					 | 
				
			||||||
  #else
 | 
					 | 
				
			||||||
    SERIAL_PROTOCOLPGM(" Y:");
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
  SERIAL_PROTOCOL(ypos);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if ENABLED(COREXZ)
 | 
					 | 
				
			||||||
    SERIAL_PROTOCOLPGM(" C:");
 | 
					 | 
				
			||||||
  #else
 | 
					 | 
				
			||||||
    SERIAL_PROTOCOLPGM(" Z:");
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
  SERIAL_PROTOCOL(zpos);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  SERIAL_EOL;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
  #if ENABLED(SCARA)
 | 
					  #if ENABLED(SCARA)
 | 
				
			||||||
    SERIAL_PROTOCOLPGM("SCARA Theta:");
 | 
					    SERIAL_PROTOCOLPGM("SCARA Theta:");
 | 
				
			||||||
@@ -5039,12 +5008,12 @@ inline void gcode_M119() {
 | 
				
			|||||||
/**
 | 
					/**
 | 
				
			||||||
 * M120: Enable endstops and set non-homing endstop state to "enabled"
 | 
					 * M120: Enable endstops and set non-homing endstop state to "enabled"
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
inline void gcode_M120() { enable_endstops_globally(true); }
 | 
					inline void gcode_M120() { endstops.enable_globally(true); }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
 * M121: Disable endstops and set non-homing endstop state to "disabled"
 | 
					 * M121: Disable endstops and set non-homing endstop state to "disabled"
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
inline void gcode_M121() { enable_endstops_globally(false); }
 | 
					inline void gcode_M121() { endstops.enable_globally(false); }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if ENABLED(BLINKM)
 | 
					#if ENABLED(BLINKM)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -5439,7 +5408,7 @@ inline void gcode_M226() {
 | 
				
			|||||||
      if (pin_number > -1) {
 | 
					      if (pin_number > -1) {
 | 
				
			||||||
        int target = LOW;
 | 
					        int target = LOW;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        st_synchronize();
 | 
					        stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        pinMode(pin_number, INPUT);
 | 
					        pinMode(pin_number, INPUT);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -5801,7 +5770,7 @@ inline void gcode_M303() {
 | 
				
			|||||||
/**
 | 
					/**
 | 
				
			||||||
 * M400: Finish all moves
 | 
					 * M400: Finish all moves
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
inline void gcode_M400() { st_synchronize(); }
 | 
					inline void gcode_M400() { stepper.synchronize(); }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if ENABLED(AUTO_BED_LEVELING_FEATURE) && DISABLED(Z_PROBE_SLED) && (HAS_SERVO_ENDSTOPS || ENABLED(Z_PROBE_ALLEN_KEY))
 | 
					#if ENABLED(AUTO_BED_LEVELING_FEATURE) && DISABLED(Z_PROBE_SLED) && (HAS_SERVO_ENDSTOPS || ENABLED(Z_PROBE_ALLEN_KEY))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -5887,7 +5856,7 @@ inline void gcode_M400() { st_synchronize(); }
 | 
				
			|||||||
 * This will stop the carriages mid-move, so most likely they
 | 
					 * This will stop the carriages mid-move, so most likely they
 | 
				
			||||||
 * will be out of sync with the stepper position after this.
 | 
					 * will be out of sync with the stepper position after this.
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
inline void gcode_M410() { quickStop(); }
 | 
					inline void gcode_M410() { stepper.quick_stop(); }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if ENABLED(MESH_BED_LEVELING)
 | 
					#if ENABLED(MESH_BED_LEVELING)
 | 
				
			||||||
@@ -6111,7 +6080,7 @@ inline void gcode_M503() {
 | 
				
			|||||||
    RUNPLAN;
 | 
					    RUNPLAN;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    //finish moves
 | 
					    //finish moves
 | 
				
			||||||
    st_synchronize();
 | 
					    stepper.synchronize();
 | 
				
			||||||
    //disable extruder steppers so filament can be removed
 | 
					    //disable extruder steppers so filament can be removed
 | 
				
			||||||
    disable_e0();
 | 
					    disable_e0();
 | 
				
			||||||
    disable_e1();
 | 
					    disable_e1();
 | 
				
			||||||
@@ -6135,7 +6104,7 @@ inline void gcode_M503() {
 | 
				
			|||||||
        current_position[E_AXIS] += AUTO_FILAMENT_CHANGE_LENGTH;
 | 
					        current_position[E_AXIS] += AUTO_FILAMENT_CHANGE_LENGTH;
 | 
				
			||||||
        destination[E_AXIS] = current_position[E_AXIS];
 | 
					        destination[E_AXIS] = current_position[E_AXIS];
 | 
				
			||||||
        line_to_destination(AUTO_FILAMENT_CHANGE_FEEDRATE);
 | 
					        line_to_destination(AUTO_FILAMENT_CHANGE_FEEDRATE);
 | 
				
			||||||
        st_synchronize();
 | 
					        stepper.synchronize();
 | 
				
			||||||
      #endif
 | 
					      #endif
 | 
				
			||||||
    } // while(!lcd_clicked)
 | 
					    } // while(!lcd_clicked)
 | 
				
			||||||
    KEEPALIVE_STATE(IN_HANDLER);
 | 
					    KEEPALIVE_STATE(IN_HANDLER);
 | 
				
			||||||
@@ -6143,7 +6112,7 @@ inline void gcode_M503() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    #if ENABLED(AUTO_FILAMENT_CHANGE)
 | 
					    #if ENABLED(AUTO_FILAMENT_CHANGE)
 | 
				
			||||||
      current_position[E_AXIS] = 0;
 | 
					      current_position[E_AXIS] = 0;
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    //return to normal
 | 
					    //return to normal
 | 
				
			||||||
@@ -6198,7 +6167,7 @@ inline void gcode_M503() {
 | 
				
			|||||||
   *    Note: the X axis should be homed after changing dual x-carriage mode.
 | 
					   *    Note: the X axis should be homed after changing dual x-carriage mode.
 | 
				
			||||||
   */
 | 
					   */
 | 
				
			||||||
  inline void gcode_M605() {
 | 
					  inline void gcode_M605() {
 | 
				
			||||||
    st_synchronize();
 | 
					    stepper.synchronize();
 | 
				
			||||||
    if (code_seen('S')) dual_x_carriage_mode = code_value();
 | 
					    if (code_seen('S')) dual_x_carriage_mode = code_value();
 | 
				
			||||||
    switch (dual_x_carriage_mode) {
 | 
					    switch (dual_x_carriage_mode) {
 | 
				
			||||||
      case DXC_DUPLICATION_MODE:
 | 
					      case DXC_DUPLICATION_MODE:
 | 
				
			||||||
@@ -6375,7 +6344,7 @@ inline void gcode_T(uint8_t tmp_extruder) {
 | 
				
			|||||||
                           current_position[E_AXIS], max_feedrate[X_AXIS], active_extruder);
 | 
					                           current_position[E_AXIS], max_feedrate[X_AXIS], active_extruder);
 | 
				
			||||||
          plan_buffer_line(x_home_pos(active_extruder), current_position[Y_AXIS], current_position[Z_AXIS],
 | 
					          plan_buffer_line(x_home_pos(active_extruder), current_position[Y_AXIS], current_position[Z_AXIS],
 | 
				
			||||||
                           current_position[E_AXIS], max_feedrate[Z_AXIS], active_extruder);
 | 
					                           current_position[E_AXIS], max_feedrate[Z_AXIS], active_extruder);
 | 
				
			||||||
          st_synchronize();
 | 
					          stepper.synchronize();
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        // apply Y & Z extruder offset (x offset is already used in determining home pos)
 | 
					        // apply Y & Z extruder offset (x offset is already used in determining home pos)
 | 
				
			||||||
@@ -6460,7 +6429,7 @@ inline void gcode_T(uint8_t tmp_extruder) {
 | 
				
			|||||||
    } // (tmp_extruder != active_extruder)
 | 
					    } // (tmp_extruder != active_extruder)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    #if ENABLED(EXT_SOLENOID)
 | 
					    #if ENABLED(EXT_SOLENOID)
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
      disable_all_solenoids();
 | 
					      disable_all_solenoids();
 | 
				
			||||||
      enable_solenoid_on_active_extruder();
 | 
					      enable_solenoid_on_active_extruder();
 | 
				
			||||||
    #endif // EXT_SOLENOID
 | 
					    #endif // EXT_SOLENOID
 | 
				
			||||||
@@ -7400,7 +7369,7 @@ void mesh_plan_buffer_line(float x, float y, float z, const float e, float feed_
 | 
				
			|||||||
        plan_buffer_line(current_position[X_AXIS] + duplicate_extruder_x_offset,
 | 
					        plan_buffer_line(current_position[X_AXIS] + duplicate_extruder_x_offset,
 | 
				
			||||||
                         current_position[Y_AXIS], current_position[Z_AXIS], current_position[E_AXIS], max_feedrate[X_AXIS], 1);
 | 
					                         current_position[Y_AXIS], current_position[Z_AXIS], current_position[E_AXIS], max_feedrate[X_AXIS], 1);
 | 
				
			||||||
        sync_plan_position();
 | 
					        sync_plan_position();
 | 
				
			||||||
        st_synchronize();
 | 
					        stepper.synchronize();
 | 
				
			||||||
        extruder_duplication_enabled = true;
 | 
					        extruder_duplication_enabled = true;
 | 
				
			||||||
        active_extruder_parked = false;
 | 
					        active_extruder_parked = false;
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
@@ -7927,7 +7896,7 @@ void manage_inactivity(bool ignore_stepper_queue/*=false*/) {
 | 
				
			|||||||
      destination[E_AXIS] = oldedes;
 | 
					      destination[E_AXIS] = oldedes;
 | 
				
			||||||
      plan_set_e_position(oldepos);
 | 
					      plan_set_e_position(oldepos);
 | 
				
			||||||
      previous_cmd_ms = ms; // refresh_cmd_timeout()
 | 
					      previous_cmd_ms = ms; // refresh_cmd_timeout()
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
      switch (active_extruder) {
 | 
					      switch (active_extruder) {
 | 
				
			||||||
        case 0:
 | 
					        case 0:
 | 
				
			||||||
          E0_ENABLE_WRITE(oldstatus);
 | 
					          E0_ENABLE_WRITE(oldstatus);
 | 
				
			||||||
@@ -8004,7 +7973,7 @@ void kill(const char* lcd_msg) {
 | 
				
			|||||||
    if (!filament_ran_out) {
 | 
					    if (!filament_ran_out) {
 | 
				
			||||||
      filament_ran_out = true;
 | 
					      filament_ran_out = true;
 | 
				
			||||||
      enqueue_and_echo_commands_P(PSTR(FILAMENT_RUNOUT_SCRIPT));
 | 
					      enqueue_and_echo_commands_P(PSTR(FILAMENT_RUNOUT_SCRIPT));
 | 
				
			||||||
      st_synchronize();
 | 
					      stepper.synchronize();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -596,7 +596,7 @@ void CardReader::updir() {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void CardReader::printingHasFinished() {
 | 
					void CardReader::printingHasFinished() {
 | 
				
			||||||
  st_synchronize();
 | 
					  stepper.synchronize();
 | 
				
			||||||
  if (file_subcall_ctr > 0) { // Heading up to a parent file that called current as a procedure.
 | 
					  if (file_subcall_ctr > 0) { // Heading up to a parent file that called current as a procedure.
 | 
				
			||||||
    file.close();
 | 
					    file.close();
 | 
				
			||||||
    file_subcall_ctr--;
 | 
					    file_subcall_ctr--;
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										317
									
								
								Marlin/endstops.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										317
									
								
								Marlin/endstops.cpp
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,317 @@
 | 
				
			|||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * Marlin 3D Printer Firmware
 | 
				
			||||||
 | 
					 * Copyright (C) 2016 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Based on Sprinter and grbl.
 | 
				
			||||||
 | 
					 * Copyright (C) 2011 Camiel Gubbels / Erik van der Zalm
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * This program is free software: you can redistribute it and/or modify
 | 
				
			||||||
 | 
					 * it under the terms of the GNU General Public License as published by
 | 
				
			||||||
 | 
					 * the Free Software Foundation, either version 3 of the License, or
 | 
				
			||||||
 | 
					 * (at your option) any later version.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * This program is distributed in the hope that it will be useful,
 | 
				
			||||||
 | 
					 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
				
			||||||
 | 
					 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
				
			||||||
 | 
					 * GNU General Public License for more details.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * You should have received a copy of the GNU General Public License
 | 
				
			||||||
 | 
					 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * endstops.cpp - A singleton object to manage endstops
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#include "Marlin.h"
 | 
				
			||||||
 | 
					#include "endstops.h"
 | 
				
			||||||
 | 
					#include "stepper.h"
 | 
				
			||||||
 | 
					#include "ultralcd.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// TEST_ENDSTOP: test the old and the current status of an endstop
 | 
				
			||||||
 | 
					#define TEST_ENDSTOP(ENDSTOP) (TEST(current_endstop_bits & old_endstop_bits, ENDSTOP))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Endstops endstops;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Endstops::Endstops() {
 | 
				
			||||||
 | 
					  enable_globally(ENABLED(ENDSTOPS_ONLY_FOR_HOMING));
 | 
				
			||||||
 | 
					  enable(true);
 | 
				
			||||||
 | 
					  #if ENABLED(HAS_Z_MIN_PROBE)
 | 
				
			||||||
 | 
					    enable_z_probe(false);
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					} // Endstops::Endstops
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void Endstops::init() {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #if HAS_X_MIN
 | 
				
			||||||
 | 
					    SET_INPUT(X_MIN_PIN);
 | 
				
			||||||
 | 
					    #if ENABLED(ENDSTOPPULLUP_XMIN)
 | 
				
			||||||
 | 
					      WRITE(X_MIN_PIN,HIGH);
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #if HAS_Y_MIN
 | 
				
			||||||
 | 
					    SET_INPUT(Y_MIN_PIN);
 | 
				
			||||||
 | 
					    #if ENABLED(ENDSTOPPULLUP_YMIN)
 | 
				
			||||||
 | 
					      WRITE(Y_MIN_PIN,HIGH);
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #if HAS_Z_MIN
 | 
				
			||||||
 | 
					    SET_INPUT(Z_MIN_PIN);
 | 
				
			||||||
 | 
					    #if ENABLED(ENDSTOPPULLUP_ZMIN)
 | 
				
			||||||
 | 
					      WRITE(Z_MIN_PIN,HIGH);
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #if HAS_Z2_MIN
 | 
				
			||||||
 | 
					    SET_INPUT(Z2_MIN_PIN);
 | 
				
			||||||
 | 
					    #if ENABLED(ENDSTOPPULLUP_ZMIN)
 | 
				
			||||||
 | 
					      WRITE(Z2_MIN_PIN,HIGH);
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #if HAS_X_MAX
 | 
				
			||||||
 | 
					    SET_INPUT(X_MAX_PIN);
 | 
				
			||||||
 | 
					    #if ENABLED(ENDSTOPPULLUP_XMAX)
 | 
				
			||||||
 | 
					      WRITE(X_MAX_PIN,HIGH);
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #if HAS_Y_MAX
 | 
				
			||||||
 | 
					    SET_INPUT(Y_MAX_PIN);
 | 
				
			||||||
 | 
					    #if ENABLED(ENDSTOPPULLUP_YMAX)
 | 
				
			||||||
 | 
					      WRITE(Y_MAX_PIN,HIGH);
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #if HAS_Z_MAX
 | 
				
			||||||
 | 
					    SET_INPUT(Z_MAX_PIN);
 | 
				
			||||||
 | 
					    #if ENABLED(ENDSTOPPULLUP_ZMAX)
 | 
				
			||||||
 | 
					      WRITE(Z_MAX_PIN,HIGH);
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #if HAS_Z2_MAX
 | 
				
			||||||
 | 
					    SET_INPUT(Z2_MAX_PIN);
 | 
				
			||||||
 | 
					    #if ENABLED(ENDSTOPPULLUP_ZMAX)
 | 
				
			||||||
 | 
					      WRITE(Z2_MAX_PIN,HIGH);
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #if HAS_Z_PROBE && ENABLED(Z_MIN_PROBE_ENDSTOP) // Check for Z_MIN_PROBE_ENDSTOP so we don't pull a pin high unless it's to be used.
 | 
				
			||||||
 | 
					    SET_INPUT(Z_MIN_PROBE_PIN);
 | 
				
			||||||
 | 
					    #if ENABLED(ENDSTOPPULLUP_ZMIN_PROBE)
 | 
				
			||||||
 | 
					      WRITE(Z_MIN_PROBE_PIN,HIGH);
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					} // Endstops::init
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void Endstops::report_state() {
 | 
				
			||||||
 | 
					  if (endstop_hit_bits) {
 | 
				
			||||||
 | 
					    #if ENABLED(ULTRA_LCD)
 | 
				
			||||||
 | 
					      char chrX = ' ', chrY = ' ', chrZ = ' ', chrP = ' ';
 | 
				
			||||||
 | 
					      #define _SET_STOP_CHAR(A,C) (chr## A = C)
 | 
				
			||||||
 | 
					    #else
 | 
				
			||||||
 | 
					      #define _SET_STOP_CHAR(A,C) ;
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #define _ENDSTOP_HIT_ECHO(A,C) do{ \
 | 
				
			||||||
 | 
					      SERIAL_ECHOPAIR(" " STRINGIFY(A) ":", stepper.triggered_position_mm(A ##_AXIS)); \
 | 
				
			||||||
 | 
					      _SET_STOP_CHAR(A,C); }while(0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #define _ENDSTOP_HIT_TEST(A,C) \
 | 
				
			||||||
 | 
					      if (TEST(endstop_hit_bits, A ##_MIN) || TEST(endstop_hit_bits, A ##_MAX)) \
 | 
				
			||||||
 | 
					        _ENDSTOP_HIT_ECHO(A,C)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    SERIAL_ECHO_START;
 | 
				
			||||||
 | 
					    SERIAL_ECHOPGM(MSG_ENDSTOPS_HIT);
 | 
				
			||||||
 | 
					    _ENDSTOP_HIT_TEST(X, 'X');
 | 
				
			||||||
 | 
					    _ENDSTOP_HIT_TEST(Y, 'Y');
 | 
				
			||||||
 | 
					    _ENDSTOP_HIT_TEST(Z, 'Z');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #if ENABLED(Z_MIN_PROBE_ENDSTOP)
 | 
				
			||||||
 | 
					      #define P_AXIS Z_AXIS
 | 
				
			||||||
 | 
					      if (TEST(endstop_hit_bits, Z_MIN_PROBE)) _ENDSTOP_HIT_ECHO(P, 'P');
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					    SERIAL_EOL;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #if ENABLED(ULTRA_LCD)
 | 
				
			||||||
 | 
					      char msg[3 * strlen(MSG_LCD_ENDSTOPS) + 8 + 1]; // Room for a UTF 8 string
 | 
				
			||||||
 | 
					      sprintf_P(msg, PSTR(MSG_LCD_ENDSTOPS " %c %c %c %c"), chrX, chrY, chrZ, chrP);
 | 
				
			||||||
 | 
					      lcd_setstatus(msg);
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    hit_on_purpose();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #if ENABLED(ABORT_ON_ENDSTOP_HIT_FEATURE_ENABLED) && ENABLED(SDSUPPORT)
 | 
				
			||||||
 | 
					      if (abort_on_endstop_hit) {
 | 
				
			||||||
 | 
					        card.sdprinting = false;
 | 
				
			||||||
 | 
					        card.closefile();
 | 
				
			||||||
 | 
					        stepper.quick_stop();
 | 
				
			||||||
 | 
					        disable_all_heaters(); // switch off all heaters.
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					} // Endstops::report_state
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Check endstops - Called from ISR!
 | 
				
			||||||
 | 
					void Endstops::update() {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #define _ENDSTOP_PIN(AXIS, MINMAX) AXIS ##_## MINMAX ##_PIN
 | 
				
			||||||
 | 
					  #define _ENDSTOP_INVERTING(AXIS, MINMAX) AXIS ##_## MINMAX ##_ENDSTOP_INVERTING
 | 
				
			||||||
 | 
					  #define _ENDSTOP_HIT(AXIS) SBI(endstop_hit_bits, _ENDSTOP(AXIS, MIN))
 | 
				
			||||||
 | 
					  #define _ENDSTOP(AXIS, MINMAX) AXIS ##_## MINMAX
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  // UPDATE_ENDSTOP_BIT: set the current endstop bits for an endstop to its status
 | 
				
			||||||
 | 
					  #define UPDATE_ENDSTOP_BIT(AXIS, MINMAX) SET_BIT(current_endstop_bits, _ENDSTOP(AXIS, MINMAX), (READ(_ENDSTOP_PIN(AXIS, MINMAX)) != _ENDSTOP_INVERTING(AXIS, MINMAX)))
 | 
				
			||||||
 | 
					  // COPY_BIT: copy the value of COPY_BIT to BIT in bits
 | 
				
			||||||
 | 
					  #define COPY_BIT(bits, COPY_BIT, BIT) SET_BIT(bits, BIT, TEST(bits, COPY_BIT))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #define UPDATE_ENDSTOP(AXIS,MINMAX) do { \
 | 
				
			||||||
 | 
					      UPDATE_ENDSTOP_BIT(AXIS, MINMAX); \
 | 
				
			||||||
 | 
					      if (TEST_ENDSTOP(_ENDSTOP(AXIS, MINMAX)) && stepper.current_block->steps[_AXIS(AXIS)] > 0) { \
 | 
				
			||||||
 | 
					        _ENDSTOP_HIT(AXIS); \
 | 
				
			||||||
 | 
					        stepper.endstop_triggered(_AXIS(AXIS)); \
 | 
				
			||||||
 | 
					      } \
 | 
				
			||||||
 | 
					    } while(0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #if ENABLED(COREXY) || ENABLED(COREXZ)
 | 
				
			||||||
 | 
					    // Head direction in -X axis for CoreXY and CoreXZ bots.
 | 
				
			||||||
 | 
					    // If Delta1 == -Delta2, the movement is only in Y or Z axis
 | 
				
			||||||
 | 
					    if ((stepper.current_block->steps[A_AXIS] != stepper.current_block->steps[CORE_AXIS_2]) || (stepper.motor_direction(A_AXIS) == stepper.motor_direction(CORE_AXIS_2))) {
 | 
				
			||||||
 | 
					      if (stepper.motor_direction(X_HEAD))
 | 
				
			||||||
 | 
					  #else
 | 
				
			||||||
 | 
					    if (stepper.motor_direction(X_AXIS))   // stepping along -X axis (regular Cartesian bot)
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					      { // -direction
 | 
				
			||||||
 | 
					        #if ENABLED(DUAL_X_CARRIAGE)
 | 
				
			||||||
 | 
					          // with 2 x-carriages, endstops are only checked in the homing direction for the active extruder
 | 
				
			||||||
 | 
					          if ((stepper.current_block->active_extruder == 0 && X_HOME_DIR == -1) || (stepper.current_block->active_extruder != 0 && X2_HOME_DIR == -1))
 | 
				
			||||||
 | 
					        #endif
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            #if HAS_X_MIN
 | 
				
			||||||
 | 
					              UPDATE_ENDSTOP(X, MIN);
 | 
				
			||||||
 | 
					            #endif
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					      else { // +direction
 | 
				
			||||||
 | 
					        #if ENABLED(DUAL_X_CARRIAGE)
 | 
				
			||||||
 | 
					          // with 2 x-carriages, endstops are only checked in the homing direction for the active extruder
 | 
				
			||||||
 | 
					          if ((stepper.current_block->active_extruder == 0 && X_HOME_DIR == 1) || (stepper.current_block->active_extruder != 0 && X2_HOME_DIR == 1))
 | 
				
			||||||
 | 
					        #endif
 | 
				
			||||||
 | 
					          {
 | 
				
			||||||
 | 
					            #if HAS_X_MAX
 | 
				
			||||||
 | 
					              UPDATE_ENDSTOP(X, MAX);
 | 
				
			||||||
 | 
					            #endif
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					  #if ENABLED(COREXY) || ENABLED(COREXZ)
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #if ENABLED(COREXY)
 | 
				
			||||||
 | 
					    // Head direction in -Y axis for CoreXY bots.
 | 
				
			||||||
 | 
					    // If DeltaX == DeltaY, the movement is only in X axis
 | 
				
			||||||
 | 
					    if ((stepper.current_block->steps[A_AXIS] != stepper.current_block->steps[B_AXIS]) || (stepper.motor_direction(A_AXIS) != stepper.motor_direction(B_AXIS))) {
 | 
				
			||||||
 | 
					      if (stepper.motor_direction(Y_HEAD))
 | 
				
			||||||
 | 
					  #else
 | 
				
			||||||
 | 
					      if (stepper.motor_direction(Y_AXIS))   // -direction
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					      { // -direction
 | 
				
			||||||
 | 
					        #if HAS_Y_MIN
 | 
				
			||||||
 | 
					          UPDATE_ENDSTOP(Y, MIN);
 | 
				
			||||||
 | 
					        #endif
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					      else { // +direction
 | 
				
			||||||
 | 
					        #if HAS_Y_MAX
 | 
				
			||||||
 | 
					          UPDATE_ENDSTOP(Y, MAX);
 | 
				
			||||||
 | 
					        #endif
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					  #if ENABLED(COREXY)
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #if ENABLED(COREXZ)
 | 
				
			||||||
 | 
					    // Head direction in -Z axis for CoreXZ bots.
 | 
				
			||||||
 | 
					    // If DeltaX == DeltaZ, the movement is only in X axis
 | 
				
			||||||
 | 
					    if ((stepper.current_block->steps[A_AXIS] != stepper.current_block->steps[C_AXIS]) || (stepper.motor_direction(A_AXIS) != stepper.motor_direction(C_AXIS))) {
 | 
				
			||||||
 | 
					      if (stepper.motor_direction(Z_HEAD))
 | 
				
			||||||
 | 
					  #else
 | 
				
			||||||
 | 
					      if (stepper.motor_direction(Z_AXIS))
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					      { // z -direction
 | 
				
			||||||
 | 
					        #if HAS_Z_MIN
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          #if ENABLED(Z_DUAL_ENDSTOPS)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            UPDATE_ENDSTOP_BIT(Z, MIN);
 | 
				
			||||||
 | 
					            #if HAS_Z2_MIN
 | 
				
			||||||
 | 
					              UPDATE_ENDSTOP_BIT(Z2, MIN);
 | 
				
			||||||
 | 
					            #else
 | 
				
			||||||
 | 
					              COPY_BIT(current_endstop_bits, Z_MIN, Z2_MIN);
 | 
				
			||||||
 | 
					            #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            byte z_test = TEST_ENDSTOP(Z_MIN) | (TEST_ENDSTOP(Z2_MIN) << 1); // bit 0 for Z, bit 1 for Z2
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if (z_test && stepper.current_block->steps[Z_AXIS] > 0) { // z_test = Z_MIN || Z2_MIN
 | 
				
			||||||
 | 
					              stepper.endstop_triggered(Z_AXIS);
 | 
				
			||||||
 | 
					              SBI(endstop_hit_bits, Z_MIN);
 | 
				
			||||||
 | 
					              if (!performing_homing || (z_test == 0x3))  //if not performing home or if both endstops were trigged during homing...
 | 
				
			||||||
 | 
					                stepper.kill_current_block();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          #else // !Z_DUAL_ENDSTOPS
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            #if ENABLED(Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN) && ENABLED(HAS_Z_MIN_PROBE)
 | 
				
			||||||
 | 
					              if (z_probe_enabled) UPDATE_ENDSTOP(Z, MIN);
 | 
				
			||||||
 | 
					            #else
 | 
				
			||||||
 | 
					              UPDATE_ENDSTOP(Z, MIN);
 | 
				
			||||||
 | 
					            #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          #endif // !Z_DUAL_ENDSTOPS
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #endif // HAS_Z_MIN
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        #if ENABLED(Z_MIN_PROBE_ENDSTOP) && DISABLED(Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN) && ENABLED(HAS_Z_MIN_PROBE)
 | 
				
			||||||
 | 
					          if (z_probe_enabled) {
 | 
				
			||||||
 | 
					            UPDATE_ENDSTOP(Z, MIN_PROBE);
 | 
				
			||||||
 | 
					            if (TEST_ENDSTOP(Z_MIN_PROBE)) SBI(endstop_hit_bits, Z_MIN_PROBE);
 | 
				
			||||||
 | 
					          }
 | 
				
			||||||
 | 
					        #endif
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					      else { // z +direction
 | 
				
			||||||
 | 
					        #if HAS_Z_MAX
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          #if ENABLED(Z_DUAL_ENDSTOPS)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            UPDATE_ENDSTOP_BIT(Z, MAX);
 | 
				
			||||||
 | 
					            #if HAS_Z2_MAX
 | 
				
			||||||
 | 
					              UPDATE_ENDSTOP_BIT(Z2, MAX);
 | 
				
			||||||
 | 
					            #else
 | 
				
			||||||
 | 
					              COPY_BIT(current_endstop_bits, Z_MAX, Z2_MAX);
 | 
				
			||||||
 | 
					            #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            byte z_test = TEST_ENDSTOP(Z_MAX) | (TEST_ENDSTOP(Z2_MAX) << 1); // bit 0 for Z, bit 1 for Z2
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            if (z_test && stepper.current_block->steps[Z_AXIS] > 0) {  // t_test = Z_MAX || Z2_MAX
 | 
				
			||||||
 | 
					              stepper.endstop_triggered(Z_AXIS);
 | 
				
			||||||
 | 
					              SBI(endstop_hit_bits, Z_MIN);
 | 
				
			||||||
 | 
					              if (!performing_homing || (z_test == 0x3))  //if not performing home or if both endstops were trigged during homing...
 | 
				
			||||||
 | 
					                stepper.kill_current_block();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          #else // !Z_DUAL_ENDSTOPS
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					            UPDATE_ENDSTOP(Z, MAX);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					          #endif // !Z_DUAL_ENDSTOPS
 | 
				
			||||||
 | 
					        #endif // Z_MAX_PIN
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					  #if ENABLED(COREXZ)
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					  old_endstop_bits = current_endstop_bits;
 | 
				
			||||||
 | 
					} // Endstops::update()
 | 
				
			||||||
							
								
								
									
										94
									
								
								Marlin/endstops.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										94
									
								
								Marlin/endstops.h
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,94 @@
 | 
				
			|||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * Marlin 3D Printer Firmware
 | 
				
			||||||
 | 
					 * Copyright (C) 2016 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Based on Sprinter and grbl.
 | 
				
			||||||
 | 
					 * Copyright (C) 2011 Camiel Gubbels / Erik van der Zalm
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * This program is free software: you can redistribute it and/or modify
 | 
				
			||||||
 | 
					 * it under the terms of the GNU General Public License as published by
 | 
				
			||||||
 | 
					 * the Free Software Foundation, either version 3 of the License, or
 | 
				
			||||||
 | 
					 * (at your option) any later version.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * This program is distributed in the hope that it will be useful,
 | 
				
			||||||
 | 
					 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
				
			||||||
 | 
					 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
				
			||||||
 | 
					 * GNU General Public License for more details.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * You should have received a copy of the GNU General Public License
 | 
				
			||||||
 | 
					 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 *  endstops.h - manages endstops
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#ifndef ENDSTOPS_H
 | 
				
			||||||
 | 
					#define ENDSTOPS_H
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					enum EndstopEnum {X_MIN = 0, Y_MIN = 1, Z_MIN = 2, Z_MIN_PROBE = 3, X_MAX = 4, Y_MAX = 5, Z_MAX = 6, Z2_MIN = 7, Z2_MAX = 8};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class Endstops {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  public:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    volatile char endstop_hit_bits; // use X_MIN, Y_MIN, Z_MIN and Z_MIN_PROBE as BIT value
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #if ENABLED(Z_DUAL_ENDSTOPS)
 | 
				
			||||||
 | 
					      uint16_t current_endstop_bits = 0,
 | 
				
			||||||
 | 
					                   old_endstop_bits = 0;
 | 
				
			||||||
 | 
					    #else
 | 
				
			||||||
 | 
					      byte current_endstop_bits = 0,
 | 
				
			||||||
 | 
					               old_endstop_bits = 0;
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					        
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    bool enabled = true;
 | 
				
			||||||
 | 
					    bool enabled_globally =
 | 
				
			||||||
 | 
					      #if ENABLED(ENDSTOPS_ONLY_FOR_HOMING)
 | 
				
			||||||
 | 
					        false
 | 
				
			||||||
 | 
					      #else
 | 
				
			||||||
 | 
					        true
 | 
				
			||||||
 | 
					      #endif
 | 
				
			||||||
 | 
					    ;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Endstops();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * Initialize the endstop pins
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    void init();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * Update the endstops bits from the pins
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    void update();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    /**
 | 
				
			||||||
 | 
					     * Print an error message reporting the position when the endstops were last hit.
 | 
				
			||||||
 | 
					     */
 | 
				
			||||||
 | 
					    void report_state(); //call from somewhere to create an serial error message with the locations the endstops where hit, in case they were triggered
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Enable / disable endstop checking globally
 | 
				
			||||||
 | 
					    FORCE_INLINE void enable_globally(bool onoff=true) { enabled_globally = enabled = onoff; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Enable / disable endstop checking
 | 
				
			||||||
 | 
					    FORCE_INLINE void enable(bool onoff=true) { enabled = onoff; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Disable / Enable endstops based on ENSTOPS_ONLY_FOR_HOMING and global enable
 | 
				
			||||||
 | 
					    FORCE_INLINE void not_homing() { enabled = enabled_globally; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Clear endstops (i.e., they were hit intentionally) to suppress the report
 | 
				
			||||||
 | 
					    FORCE_INLINE void hit_on_purpose() { endstop_hit_bits = 0; }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Enable / disable endstop z-probe checking
 | 
				
			||||||
 | 
					    #if ENABLED(HAS_Z_MIN_PROBE)
 | 
				
			||||||
 | 
					      volatile bool z_probe_enabled = false;
 | 
				
			||||||
 | 
					      FORCE_INLINE void enable_z_probe(bool onoff=true) { z_probe_enabled = onoff; }
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					extern Endstops endstops;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#endif // ENDSTOPS_H
 | 
				
			||||||
@@ -1085,7 +1085,7 @@ float junction_deviation = 0.1;
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  planner_recalculate();
 | 
					  planner_recalculate();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  st_wake_up();
 | 
					  stepper.wake_up();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
} // plan_buffer_line()
 | 
					} // plan_buffer_line()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1097,7 +1097,7 @@ float junction_deviation = 0.1;
 | 
				
			|||||||
   * On CORE machines XYZ is derived from ABC.
 | 
					   * On CORE machines XYZ is derived from ABC.
 | 
				
			||||||
   */
 | 
					   */
 | 
				
			||||||
  vector_3 plan_get_position() {
 | 
					  vector_3 plan_get_position() {
 | 
				
			||||||
    vector_3 position = vector_3(st_get_axis_position_mm(X_AXIS), st_get_axis_position_mm(Y_AXIS), st_get_axis_position_mm(Z_AXIS));
 | 
					    vector_3 position = vector_3(stepper.get_axis_position_mm(X_AXIS), stepper.get_axis_position_mm(Y_AXIS), stepper.get_axis_position_mm(Z_AXIS));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    //position.debug("in plan_get position");
 | 
					    //position.debug("in plan_get position");
 | 
				
			||||||
    //plan_bed_level_matrix.debug("in plan_get_position");
 | 
					    //plan_bed_level_matrix.debug("in plan_get_position");
 | 
				
			||||||
@@ -1132,7 +1132,7 @@ float junction_deviation = 0.1;
 | 
				
			|||||||
         ny = position[Y_AXIS] = lround(y * axis_steps_per_unit[Y_AXIS]),
 | 
					         ny = position[Y_AXIS] = lround(y * axis_steps_per_unit[Y_AXIS]),
 | 
				
			||||||
         nz = position[Z_AXIS] = lround(z * axis_steps_per_unit[Z_AXIS]),
 | 
					         nz = position[Z_AXIS] = lround(z * axis_steps_per_unit[Z_AXIS]),
 | 
				
			||||||
         ne = position[E_AXIS] = lround(e * axis_steps_per_unit[E_AXIS]);
 | 
					         ne = position[E_AXIS] = lround(e * axis_steps_per_unit[E_AXIS]);
 | 
				
			||||||
    st_set_position(nx, ny, nz, ne);
 | 
					    stepper.set_position(nx, ny, nz, ne);
 | 
				
			||||||
    previous_nominal_speed = 0.0; // Resets planner junction speeds. Assumes start from rest.
 | 
					    previous_nominal_speed = 0.0; // Resets planner junction speeds. Assumes start from rest.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    for (int i = 0; i < NUM_AXIS; i++) previous_speed[i] = 0.0;
 | 
					    for (int i = 0; i < NUM_AXIS; i++) previous_speed[i] = 0.0;
 | 
				
			||||||
@@ -1140,7 +1140,7 @@ float junction_deviation = 0.1;
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
void plan_set_e_position(const float& e) {
 | 
					void plan_set_e_position(const float& e) {
 | 
				
			||||||
  position[E_AXIS] = lround(e * axis_steps_per_unit[E_AXIS]);
 | 
					  position[E_AXIS] = lround(e * axis_steps_per_unit[E_AXIS]);
 | 
				
			||||||
  st_set_e_position(position[E_AXIS]);
 | 
					  stepper.set_e_position(position[E_AXIS]);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Calculate the steps/s^2 acceleration rates, based on the mm/s^s
 | 
					// Calculate the steps/s^2 acceleration rates, based on the mm/s^s
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -21,7 +21,7 @@
 | 
				
			|||||||
 */
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
 * stepper.cpp - stepper motor driver: executes motion plans using stepper motors
 | 
					 * stepper.cpp - A singleton object to execute motion plans using stepper motors
 | 
				
			||||||
 * Marlin Firmware
 | 
					 * Marlin Firmware
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * Derived from Grbl
 | 
					 * Derived from Grbl
 | 
				
			||||||
@@ -46,6 +46,7 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
#include "Marlin.h"
 | 
					#include "Marlin.h"
 | 
				
			||||||
#include "stepper.h"
 | 
					#include "stepper.h"
 | 
				
			||||||
 | 
					#include "endstops.h"
 | 
				
			||||||
#include "planner.h"
 | 
					#include "planner.h"
 | 
				
			||||||
#include "temperature.h"
 | 
					#include "temperature.h"
 | 
				
			||||||
#include "ultralcd.h"
 | 
					#include "ultralcd.h"
 | 
				
			||||||
@@ -57,85 +58,7 @@
 | 
				
			|||||||
  #include <SPI.h>
 | 
					  #include <SPI.h>
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
//===========================================================================
 | 
					Stepper stepper; // Singleton
 | 
				
			||||||
//============================= public variables ============================
 | 
					 | 
				
			||||||
//===========================================================================
 | 
					 | 
				
			||||||
block_t* current_block;  // A pointer to the block currently being traced
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#if ENABLED(HAS_Z_MIN_PROBE)
 | 
					 | 
				
			||||||
  volatile bool z_probe_is_active = false;
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
//===========================================================================
 | 
					 | 
				
			||||||
//============================= private variables ===========================
 | 
					 | 
				
			||||||
//===========================================================================
 | 
					 | 
				
			||||||
//static makes it impossible to be called from outside of this file by extern.!
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// Variables used by The Stepper Driver Interrupt
 | 
					 | 
				
			||||||
static unsigned char out_bits = 0;        // The next stepping-bits to be output
 | 
					 | 
				
			||||||
static unsigned int cleaning_buffer_counter;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#if ENABLED(Z_DUAL_ENDSTOPS)
 | 
					 | 
				
			||||||
  static bool performing_homing = false,
 | 
					 | 
				
			||||||
              locked_z_motor = false,
 | 
					 | 
				
			||||||
              locked_z2_motor = false;
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// Counter variables for the Bresenham line tracer
 | 
					 | 
				
			||||||
static long counter_x, counter_y, counter_z, counter_e;
 | 
					 | 
				
			||||||
volatile static unsigned long step_events_completed; // The number of step events executed in the current block
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#if ENABLED(ADVANCE)
 | 
					 | 
				
			||||||
  static long advance_rate, advance, final_advance = 0;
 | 
					 | 
				
			||||||
  static long old_advance = 0;
 | 
					 | 
				
			||||||
  static long e_steps[4];
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static long acceleration_time, deceleration_time;
 | 
					 | 
				
			||||||
//static unsigned long accelerate_until, decelerate_after, acceleration_rate, initial_rate, final_rate, nominal_rate;
 | 
					 | 
				
			||||||
static unsigned short acc_step_rate; // needed for deceleration start point
 | 
					 | 
				
			||||||
static uint8_t step_loops;
 | 
					 | 
				
			||||||
static uint8_t step_loops_nominal;
 | 
					 | 
				
			||||||
static unsigned short OCR1A_nominal;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
volatile long endstops_trigsteps[3] = { 0 };
 | 
					 | 
				
			||||||
volatile long endstops_stepsTotal, endstops_stepsDone;
 | 
					 | 
				
			||||||
static volatile char endstop_hit_bits = 0; // use X_MIN, Y_MIN, Z_MIN and Z_MIN_PROBE as BIT value
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#if DISABLED(Z_DUAL_ENDSTOPS)
 | 
					 | 
				
			||||||
  static byte
 | 
					 | 
				
			||||||
#else
 | 
					 | 
				
			||||||
  static uint16_t
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
    old_endstop_bits = 0; // use X_MIN, X_MAX... Z_MAX, Z_MIN_PROBE, Z2_MIN, Z2_MAX
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#if ENABLED(ABORT_ON_ENDSTOP_HIT_FEATURE_ENABLED)
 | 
					 | 
				
			||||||
  bool abort_on_endstop_hit = false;
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#if HAS_MOTOR_CURRENT_PWM
 | 
					 | 
				
			||||||
  #ifndef PWM_MOTOR_CURRENT
 | 
					 | 
				
			||||||
    #define PWM_MOTOR_CURRENT DEFAULT_PWM_MOTOR_CURRENT
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
  const int motor_current_setting[3] = PWM_MOTOR_CURRENT;
 | 
					 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
static bool check_endstops = true;
 | 
					 | 
				
			||||||
static bool check_endstops_global =
 | 
					 | 
				
			||||||
  #if ENABLED(ENDSTOPS_ONLY_FOR_HOMING)
 | 
					 | 
				
			||||||
    false
 | 
					 | 
				
			||||||
  #else
 | 
					 | 
				
			||||||
    true
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
volatile long count_position[NUM_AXIS] = { 0 }; // Positions of stepper motors, in step units
 | 
					 | 
				
			||||||
volatile signed char count_direction[NUM_AXIS] = { 1 };
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
//===========================================================================
 | 
					 | 
				
			||||||
//================================ functions ================================
 | 
					 | 
				
			||||||
//===========================================================================
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if ENABLED(DUAL_X_CARRIAGE)
 | 
					#if ENABLED(DUAL_X_CARRIAGE)
 | 
				
			||||||
  #define X_APPLY_DIR(v,ALWAYS) \
 | 
					  #define X_APPLY_DIR(v,ALWAYS) \
 | 
				
			||||||
@@ -173,12 +96,12 @@ volatile signed char count_direction[NUM_AXIS] = { 1 };
 | 
				
			|||||||
    #define Z_APPLY_STEP(v,Q) \
 | 
					    #define Z_APPLY_STEP(v,Q) \
 | 
				
			||||||
    if (performing_homing) { \
 | 
					    if (performing_homing) { \
 | 
				
			||||||
      if (Z_HOME_DIR > 0) {\
 | 
					      if (Z_HOME_DIR > 0) {\
 | 
				
			||||||
        if (!(TEST(old_endstop_bits, Z_MAX) && (count_direction[Z_AXIS] > 0)) && !locked_z_motor) Z_STEP_WRITE(v); \
 | 
					        if (!(TEST(endstops.old_endstop_bits, Z_MAX) && (count_direction[Z_AXIS] > 0)) && !locked_z_motor) Z_STEP_WRITE(v); \
 | 
				
			||||||
        if (!(TEST(old_endstop_bits, Z2_MAX) && (count_direction[Z_AXIS] > 0)) && !locked_z2_motor) Z2_STEP_WRITE(v); \
 | 
					        if (!(TEST(endstops.old_endstop_bits, Z2_MAX) && (count_direction[Z_AXIS] > 0)) && !locked_z2_motor) Z2_STEP_WRITE(v); \
 | 
				
			||||||
      } \
 | 
					      } \
 | 
				
			||||||
      else { \
 | 
					      else { \
 | 
				
			||||||
        if (!(TEST(old_endstop_bits, Z_MIN) && (count_direction[Z_AXIS] < 0)) && !locked_z_motor) Z_STEP_WRITE(v); \
 | 
					        if (!(TEST(endstops.old_endstop_bits, Z_MIN) && (count_direction[Z_AXIS] < 0)) && !locked_z_motor) Z_STEP_WRITE(v); \
 | 
				
			||||||
        if (!(TEST(old_endstop_bits, Z2_MIN) && (count_direction[Z_AXIS] < 0)) && !locked_z2_motor) Z2_STEP_WRITE(v); \
 | 
					        if (!(TEST(endstops.old_endstop_bits, Z2_MIN) && (count_direction[Z_AXIS] < 0)) && !locked_z2_motor) Z2_STEP_WRITE(v); \
 | 
				
			||||||
      } \
 | 
					      } \
 | 
				
			||||||
    } \
 | 
					    } \
 | 
				
			||||||
    else { \
 | 
					    else { \
 | 
				
			||||||
@@ -195,31 +118,6 @@ volatile signed char count_direction[NUM_AXIS] = { 1 };
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
#define E_APPLY_STEP(v,Q) E_STEP_WRITE(v)
 | 
					#define E_APPLY_STEP(v,Q) E_STEP_WRITE(v)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// intRes = intIn1 * intIn2 >> 16
 | 
					 | 
				
			||||||
// uses:
 | 
					 | 
				
			||||||
// r26 to store 0
 | 
					 | 
				
			||||||
// r27 to store the byte 1 of the 24 bit result
 | 
					 | 
				
			||||||
#define MultiU16X8toH16(intRes, charIn1, intIn2) \
 | 
					 | 
				
			||||||
  asm volatile ( \
 | 
					 | 
				
			||||||
                 "clr r26 \n\t" \
 | 
					 | 
				
			||||||
                 "mul %A1, %B2 \n\t" \
 | 
					 | 
				
			||||||
                 "movw %A0, r0 \n\t" \
 | 
					 | 
				
			||||||
                 "mul %A1, %A2 \n\t" \
 | 
					 | 
				
			||||||
                 "add %A0, r1 \n\t" \
 | 
					 | 
				
			||||||
                 "adc %B0, r26 \n\t" \
 | 
					 | 
				
			||||||
                 "lsr r0 \n\t" \
 | 
					 | 
				
			||||||
                 "adc %A0, r26 \n\t" \
 | 
					 | 
				
			||||||
                 "adc %B0, r26 \n\t" \
 | 
					 | 
				
			||||||
                 "clr r1 \n\t" \
 | 
					 | 
				
			||||||
                 : \
 | 
					 | 
				
			||||||
                 "=&r" (intRes) \
 | 
					 | 
				
			||||||
                 : \
 | 
					 | 
				
			||||||
                 "d" (charIn1), \
 | 
					 | 
				
			||||||
                 "d" (intIn2) \
 | 
					 | 
				
			||||||
                 : \
 | 
					 | 
				
			||||||
                 "r26" \
 | 
					 | 
				
			||||||
               )
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// intRes = longIn1 * longIn2 >> 24
 | 
					// intRes = longIn1 * longIn2 >> 24
 | 
				
			||||||
// uses:
 | 
					// uses:
 | 
				
			||||||
// r26 to store 0
 | 
					// r26 to store 0
 | 
				
			||||||
@@ -281,312 +179,38 @@ volatile signed char count_direction[NUM_AXIS] = { 1 };
 | 
				
			|||||||
#define ENABLE_STEPPER_DRIVER_INTERRUPT()  SBI(TIMSK1, OCIE1A)
 | 
					#define ENABLE_STEPPER_DRIVER_INTERRUPT()  SBI(TIMSK1, OCIE1A)
 | 
				
			||||||
#define DISABLE_STEPPER_DRIVER_INTERRUPT() CBI(TIMSK1, OCIE1A)
 | 
					#define DISABLE_STEPPER_DRIVER_INTERRUPT() CBI(TIMSK1, OCIE1A)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void enable_endstops(bool check) { check_endstops = check; }
 | 
					/**
 | 
				
			||||||
 | 
					 *         __________________________
 | 
				
			||||||
void enable_endstops_globally(bool check) { check_endstops_global = check_endstops = check; }
 | 
					 *        /|                        |\     _________________         ^
 | 
				
			||||||
 | 
					 *       / |                        | \   /|               |\        |
 | 
				
			||||||
void endstops_not_homing() { check_endstops = check_endstops_global; }
 | 
					 *      /  |                        |  \ / |               | \       s
 | 
				
			||||||
 | 
					 *     /   |                        |   |  |               |  \      p
 | 
				
			||||||
void endstops_hit_on_purpose() { endstop_hit_bits = 0; }
 | 
					 *    /    |                        |   |  |               |   \     e
 | 
				
			||||||
 | 
					 *   +-----+------------------------+---+--+---------------+----+    e
 | 
				
			||||||
void checkHitEndstops() {
 | 
					 *   |               BLOCK 1            |      BLOCK 2          |    d
 | 
				
			||||||
  if (endstop_hit_bits) {
 | 
					 *
 | 
				
			||||||
    #if ENABLED(ULTRA_LCD)
 | 
					 *                           time ----->
 | 
				
			||||||
      char chrX = ' ', chrY = ' ', chrZ = ' ', chrP = ' ';
 | 
					 *
 | 
				
			||||||
      #define _SET_STOP_CHAR(A,C) (chr## A = C)
 | 
					 *  The trapezoid is the shape the speed curve over time. It starts at block->initial_rate, accelerates
 | 
				
			||||||
    #else
 | 
					 *  first block->accelerate_until step_events_completed, then keeps going at constant speed until
 | 
				
			||||||
      #define _SET_STOP_CHAR(A,C) ;
 | 
					 *  step_events_completed reaches block->decelerate_after after which it decelerates until the trapezoid generator is reset.
 | 
				
			||||||
    #endif
 | 
					 *  The slope of acceleration is calculated using v = u + at where t is the accumulated timer values of the steps so far.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
    #define _ENDSTOP_HIT_ECHO(A,C) do{ \
 | 
					void Stepper::wake_up() {
 | 
				
			||||||
      SERIAL_ECHOPAIR(" " STRINGIFY(A) ":", endstops_trigsteps[A ##_AXIS] / axis_steps_per_unit[A ##_AXIS]); \
 | 
					 | 
				
			||||||
      _SET_STOP_CHAR(A,C); }while(0)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    #define _ENDSTOP_HIT_TEST(A,C) \
 | 
					 | 
				
			||||||
      if (TEST(endstop_hit_bits, A ##_MIN) || TEST(endstop_hit_bits, A ##_MAX)) \
 | 
					 | 
				
			||||||
        _ENDSTOP_HIT_ECHO(A,C)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    SERIAL_ECHO_START;
 | 
					 | 
				
			||||||
    SERIAL_ECHOPGM(MSG_ENDSTOPS_HIT);
 | 
					 | 
				
			||||||
    _ENDSTOP_HIT_TEST(X, 'X');
 | 
					 | 
				
			||||||
    _ENDSTOP_HIT_TEST(Y, 'Y');
 | 
					 | 
				
			||||||
    _ENDSTOP_HIT_TEST(Z, 'Z');
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    #if ENABLED(Z_MIN_PROBE_ENDSTOP)
 | 
					 | 
				
			||||||
      #define P_AXIS Z_AXIS
 | 
					 | 
				
			||||||
      if (TEST(endstop_hit_bits, Z_MIN_PROBE)) _ENDSTOP_HIT_ECHO(P, 'P');
 | 
					 | 
				
			||||||
    #endif
 | 
					 | 
				
			||||||
    SERIAL_EOL;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    #if ENABLED(ULTRA_LCD)
 | 
					 | 
				
			||||||
      char msg[3 * strlen(MSG_LCD_ENDSTOPS) + 8 + 1]; // Room for a UTF 8 string
 | 
					 | 
				
			||||||
      sprintf_P(msg, PSTR(MSG_LCD_ENDSTOPS " %c %c %c %c"), chrX, chrY, chrZ, chrP);
 | 
					 | 
				
			||||||
      lcd_setstatus(msg);
 | 
					 | 
				
			||||||
    #endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    endstops_hit_on_purpose();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    #if ENABLED(ABORT_ON_ENDSTOP_HIT_FEATURE_ENABLED) && ENABLED(SDSUPPORT)
 | 
					 | 
				
			||||||
      if (abort_on_endstop_hit) {
 | 
					 | 
				
			||||||
        card.sdprinting = false;
 | 
					 | 
				
			||||||
        card.closefile();
 | 
					 | 
				
			||||||
        quickStop();
 | 
					 | 
				
			||||||
        disable_all_heaters(); // switch off all heaters.
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
    #endif
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// Check endstops - Called from ISR!
 | 
					 | 
				
			||||||
inline void update_endstops() {
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if ENABLED(Z_DUAL_ENDSTOPS)
 | 
					 | 
				
			||||||
    uint16_t
 | 
					 | 
				
			||||||
  #else
 | 
					 | 
				
			||||||
    byte
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
      current_endstop_bits = 0;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #define _ENDSTOP_PIN(AXIS, MINMAX) AXIS ##_## MINMAX ##_PIN
 | 
					 | 
				
			||||||
  #define _ENDSTOP_INVERTING(AXIS, MINMAX) AXIS ##_## MINMAX ##_ENDSTOP_INVERTING
 | 
					 | 
				
			||||||
  #define _AXIS(AXIS) AXIS ##_AXIS
 | 
					 | 
				
			||||||
  #define _ENDSTOP_HIT(AXIS) SBI(endstop_hit_bits, _ENDSTOP(AXIS, MIN))
 | 
					 | 
				
			||||||
  #define _ENDSTOP(AXIS, MINMAX) AXIS ##_## MINMAX
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  // SET_ENDSTOP_BIT: set the current endstop bits for an endstop to its status
 | 
					 | 
				
			||||||
  #define SET_ENDSTOP_BIT(AXIS, MINMAX) SET_BIT(current_endstop_bits, _ENDSTOP(AXIS, MINMAX), (READ(_ENDSTOP_PIN(AXIS, MINMAX)) != _ENDSTOP_INVERTING(AXIS, MINMAX)))
 | 
					 | 
				
			||||||
  // COPY_BIT: copy the value of COPY_BIT to BIT in bits
 | 
					 | 
				
			||||||
  #define COPY_BIT(bits, COPY_BIT, BIT) SET_BIT(bits, BIT, TEST(bits, COPY_BIT))
 | 
					 | 
				
			||||||
  // TEST_ENDSTOP: test the old and the current status of an endstop
 | 
					 | 
				
			||||||
  #define TEST_ENDSTOP(ENDSTOP) (TEST(current_endstop_bits, ENDSTOP) && TEST(old_endstop_bits, ENDSTOP))
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if ENABLED(COREXY) || ENABLED(COREXZ)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    #define _SET_TRIGSTEPS(AXIS) do { \
 | 
					 | 
				
			||||||
        float axis_pos = count_position[_AXIS(AXIS)]; \
 | 
					 | 
				
			||||||
        if (_AXIS(AXIS) == A_AXIS) \
 | 
					 | 
				
			||||||
          axis_pos = (axis_pos + count_position[CORE_AXIS_2]) / 2; \
 | 
					 | 
				
			||||||
        else if (_AXIS(AXIS) == CORE_AXIS_2) \
 | 
					 | 
				
			||||||
          axis_pos = (count_position[A_AXIS] - axis_pos) / 2; \
 | 
					 | 
				
			||||||
        endstops_trigsteps[_AXIS(AXIS)] = axis_pos; \
 | 
					 | 
				
			||||||
      } while(0)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #else
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    #define _SET_TRIGSTEPS(AXIS) endstops_trigsteps[_AXIS(AXIS)] = count_position[_AXIS(AXIS)]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #endif // COREXY || COREXZ
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #define UPDATE_ENDSTOP(AXIS,MINMAX) do { \
 | 
					 | 
				
			||||||
      SET_ENDSTOP_BIT(AXIS, MINMAX); \
 | 
					 | 
				
			||||||
      if (TEST_ENDSTOP(_ENDSTOP(AXIS, MINMAX)) && current_block->steps[_AXIS(AXIS)] > 0) { \
 | 
					 | 
				
			||||||
        _SET_TRIGSTEPS(AXIS); \
 | 
					 | 
				
			||||||
        _ENDSTOP_HIT(AXIS); \
 | 
					 | 
				
			||||||
        step_events_completed = current_block->step_event_count; \
 | 
					 | 
				
			||||||
      } \
 | 
					 | 
				
			||||||
    } while(0)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if ENABLED(COREXY) || ENABLED(COREXZ)
 | 
					 | 
				
			||||||
    // Head direction in -X axis for CoreXY and CoreXZ bots.
 | 
					 | 
				
			||||||
    // If Delta1 == -Delta2, the movement is only in Y or Z axis
 | 
					 | 
				
			||||||
    if ((current_block->steps[A_AXIS] != current_block->steps[CORE_AXIS_2]) || (TEST(out_bits, A_AXIS) == TEST(out_bits, CORE_AXIS_2))) {
 | 
					 | 
				
			||||||
      if (TEST(out_bits, X_HEAD))
 | 
					 | 
				
			||||||
  #else
 | 
					 | 
				
			||||||
    if (TEST(out_bits, X_AXIS))   // stepping along -X axis (regular Cartesian bot)
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
      { // -direction
 | 
					 | 
				
			||||||
        #if ENABLED(DUAL_X_CARRIAGE)
 | 
					 | 
				
			||||||
          // with 2 x-carriages, endstops are only checked in the homing direction for the active extruder
 | 
					 | 
				
			||||||
          if ((current_block->active_extruder == 0 && X_HOME_DIR == -1) || (current_block->active_extruder != 0 && X2_HOME_DIR == -1))
 | 
					 | 
				
			||||||
        #endif
 | 
					 | 
				
			||||||
          {
 | 
					 | 
				
			||||||
            #if HAS_X_MIN
 | 
					 | 
				
			||||||
              UPDATE_ENDSTOP(X, MIN);
 | 
					 | 
				
			||||||
            #endif
 | 
					 | 
				
			||||||
          }
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
      else { // +direction
 | 
					 | 
				
			||||||
        #if ENABLED(DUAL_X_CARRIAGE)
 | 
					 | 
				
			||||||
          // with 2 x-carriages, endstops are only checked in the homing direction for the active extruder
 | 
					 | 
				
			||||||
          if ((current_block->active_extruder == 0 && X_HOME_DIR == 1) || (current_block->active_extruder != 0 && X2_HOME_DIR == 1))
 | 
					 | 
				
			||||||
        #endif
 | 
					 | 
				
			||||||
          {
 | 
					 | 
				
			||||||
            #if HAS_X_MAX
 | 
					 | 
				
			||||||
              UPDATE_ENDSTOP(X, MAX);
 | 
					 | 
				
			||||||
            #endif
 | 
					 | 
				
			||||||
          }
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
  #if ENABLED(COREXY) || ENABLED(COREXZ)
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if ENABLED(COREXY)
 | 
					 | 
				
			||||||
    // Head direction in -Y axis for CoreXY bots.
 | 
					 | 
				
			||||||
    // If DeltaX == DeltaY, the movement is only in X axis
 | 
					 | 
				
			||||||
    if ((current_block->steps[A_AXIS] != current_block->steps[B_AXIS]) || (TEST(out_bits, A_AXIS) != TEST(out_bits, B_AXIS))) {
 | 
					 | 
				
			||||||
      if (TEST(out_bits, Y_HEAD))
 | 
					 | 
				
			||||||
  #else
 | 
					 | 
				
			||||||
      if (TEST(out_bits, Y_AXIS))   // -direction
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
      { // -direction
 | 
					 | 
				
			||||||
        #if HAS_Y_MIN
 | 
					 | 
				
			||||||
          UPDATE_ENDSTOP(Y, MIN);
 | 
					 | 
				
			||||||
        #endif
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
      else { // +direction
 | 
					 | 
				
			||||||
        #if HAS_Y_MAX
 | 
					 | 
				
			||||||
          UPDATE_ENDSTOP(Y, MAX);
 | 
					 | 
				
			||||||
        #endif
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
  #if ENABLED(COREXY)
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if ENABLED(COREXZ)
 | 
					 | 
				
			||||||
    // Head direction in -Z axis for CoreXZ bots.
 | 
					 | 
				
			||||||
    // If DeltaX == DeltaZ, the movement is only in X axis
 | 
					 | 
				
			||||||
    if ((current_block->steps[A_AXIS] != current_block->steps[C_AXIS]) || (TEST(out_bits, A_AXIS) != TEST(out_bits, C_AXIS))) {
 | 
					 | 
				
			||||||
      if (TEST(out_bits, Z_HEAD))
 | 
					 | 
				
			||||||
  #else
 | 
					 | 
				
			||||||
      if (TEST(out_bits, Z_AXIS))
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
      { // z -direction
 | 
					 | 
				
			||||||
        #if HAS_Z_MIN
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
          #if ENABLED(Z_DUAL_ENDSTOPS)
 | 
					 | 
				
			||||||
            SET_ENDSTOP_BIT(Z, MIN);
 | 
					 | 
				
			||||||
            #if HAS_Z2_MIN
 | 
					 | 
				
			||||||
              SET_ENDSTOP_BIT(Z2, MIN);
 | 
					 | 
				
			||||||
            #else
 | 
					 | 
				
			||||||
              COPY_BIT(current_endstop_bits, Z_MIN, Z2_MIN);
 | 
					 | 
				
			||||||
            #endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            byte z_test = TEST_ENDSTOP(Z_MIN) | (TEST_ENDSTOP(Z2_MIN) << 1); // bit 0 for Z, bit 1 for Z2
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            if (z_test && current_block->steps[Z_AXIS] > 0) { // z_test = Z_MIN || Z2_MIN
 | 
					 | 
				
			||||||
              endstops_trigsteps[Z_AXIS] = count_position[Z_AXIS];
 | 
					 | 
				
			||||||
              SBI(endstop_hit_bits, Z_MIN);
 | 
					 | 
				
			||||||
              if (!performing_homing || (z_test == 0x3))  //if not performing home or if both endstops were trigged during homing...
 | 
					 | 
				
			||||||
                step_events_completed = current_block->step_event_count;
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
          #else // !Z_DUAL_ENDSTOPS
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            #if ENABLED(Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN) && ENABLED(HAS_Z_MIN_PROBE)
 | 
					 | 
				
			||||||
              if (z_probe_is_active) UPDATE_ENDSTOP(Z, MIN);
 | 
					 | 
				
			||||||
            #else
 | 
					 | 
				
			||||||
              UPDATE_ENDSTOP(Z, MIN);
 | 
					 | 
				
			||||||
            #endif
 | 
					 | 
				
			||||||
          #endif // !Z_DUAL_ENDSTOPS
 | 
					 | 
				
			||||||
        #endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        #if ENABLED(Z_MIN_PROBE_ENDSTOP) && DISABLED(Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN) && ENABLED(HAS_Z_MIN_PROBE)
 | 
					 | 
				
			||||||
          if (z_probe_is_active) {
 | 
					 | 
				
			||||||
            UPDATE_ENDSTOP(Z, MIN_PROBE);
 | 
					 | 
				
			||||||
            if (TEST_ENDSTOP(Z_MIN_PROBE)) SBI(endstop_hit_bits, Z_MIN_PROBE);
 | 
					 | 
				
			||||||
          }
 | 
					 | 
				
			||||||
        #endif
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
      else { // z +direction
 | 
					 | 
				
			||||||
        #if HAS_Z_MAX
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
          #if ENABLED(Z_DUAL_ENDSTOPS)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            SET_ENDSTOP_BIT(Z, MAX);
 | 
					 | 
				
			||||||
            #if HAS_Z2_MAX
 | 
					 | 
				
			||||||
              SET_ENDSTOP_BIT(Z2, MAX);
 | 
					 | 
				
			||||||
            #else
 | 
					 | 
				
			||||||
              COPY_BIT(current_endstop_bits, Z_MAX, Z2_MAX);
 | 
					 | 
				
			||||||
            #endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            byte z_test = TEST_ENDSTOP(Z_MAX) | (TEST_ENDSTOP(Z2_MAX) << 1); // bit 0 for Z, bit 1 for Z2
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            if (z_test && current_block->steps[Z_AXIS] > 0) {  // t_test = Z_MAX || Z2_MAX
 | 
					 | 
				
			||||||
              endstops_trigsteps[Z_AXIS] = count_position[Z_AXIS];
 | 
					 | 
				
			||||||
              SBI(endstop_hit_bits, Z_MIN);
 | 
					 | 
				
			||||||
              if (!performing_homing || (z_test == 0x3))  //if not performing home or if both endstops were trigged during homing...
 | 
					 | 
				
			||||||
                step_events_completed = current_block->step_event_count;
 | 
					 | 
				
			||||||
            }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
          #else // !Z_DUAL_ENDSTOPS
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
            UPDATE_ENDSTOP(Z, MAX);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
          #endif // !Z_DUAL_ENDSTOPS
 | 
					 | 
				
			||||||
        #endif // Z_MAX_PIN
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
  #if ENABLED(COREXZ)
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
  old_endstop_bits = current_endstop_bits;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
//         __________________________
 | 
					 | 
				
			||||||
//        /|                        |\     _________________         ^
 | 
					 | 
				
			||||||
//       / |                        | \   /|               |\        |
 | 
					 | 
				
			||||||
//      /  |                        |  \ / |               | \       s
 | 
					 | 
				
			||||||
//     /   |                        |   |  |               |  \      p
 | 
					 | 
				
			||||||
//    /    |                        |   |  |               |   \     e
 | 
					 | 
				
			||||||
//   +-----+------------------------+---+--+---------------+----+    e
 | 
					 | 
				
			||||||
//   |               BLOCK 1            |      BLOCK 2          |    d
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
//                           time ----->
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
//  The trapezoid is the shape the speed curve over time. It starts at block->initial_rate, accelerates
 | 
					 | 
				
			||||||
//  first block->accelerate_until step_events_completed, then keeps going at constant speed until
 | 
					 | 
				
			||||||
//  step_events_completed reaches block->decelerate_after after which it decelerates until the trapezoid generator is reset.
 | 
					 | 
				
			||||||
//  The slope of acceleration is calculated using v = u + at where t is the accumulated timer values of the steps so far.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
void st_wake_up() {
 | 
					 | 
				
			||||||
  //  TCNT1 = 0;
 | 
					  //  TCNT1 = 0;
 | 
				
			||||||
  ENABLE_STEPPER_DRIVER_INTERRUPT();
 | 
					  ENABLE_STEPPER_DRIVER_INTERRUPT();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
FORCE_INLINE unsigned short calc_timer(unsigned short step_rate) {
 | 
					 | 
				
			||||||
  unsigned short timer;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  NOMORE(step_rate, MAX_STEP_FREQUENCY);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  if (step_rate > 20000) { // If steprate > 20kHz >> step 4 times
 | 
					 | 
				
			||||||
    step_rate = (step_rate >> 2) & 0x3fff;
 | 
					 | 
				
			||||||
    step_loops = 4;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  else if (step_rate > 10000) { // If steprate > 10kHz >> step 2 times
 | 
					 | 
				
			||||||
    step_rate = (step_rate >> 1) & 0x7fff;
 | 
					 | 
				
			||||||
    step_loops = 2;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  else {
 | 
					 | 
				
			||||||
    step_loops = 1;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  NOLESS(step_rate, F_CPU / 500000);
 | 
					 | 
				
			||||||
  step_rate -= F_CPU / 500000; // Correct for minimal speed
 | 
					 | 
				
			||||||
  if (step_rate >= (8 * 256)) { // higher step rate
 | 
					 | 
				
			||||||
    unsigned short table_address = (unsigned short)&speed_lookuptable_fast[(unsigned char)(step_rate >> 8)][0];
 | 
					 | 
				
			||||||
    unsigned char tmp_step_rate = (step_rate & 0x00ff);
 | 
					 | 
				
			||||||
    unsigned short gain = (unsigned short)pgm_read_word_near(table_address + 2);
 | 
					 | 
				
			||||||
    MultiU16X8toH16(timer, tmp_step_rate, gain);
 | 
					 | 
				
			||||||
    timer = (unsigned short)pgm_read_word_near(table_address) - timer;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  else { // lower step rates
 | 
					 | 
				
			||||||
    unsigned short table_address = (unsigned short)&speed_lookuptable_slow[0][0];
 | 
					 | 
				
			||||||
    table_address += ((step_rate) >> 1) & 0xfffc;
 | 
					 | 
				
			||||||
    timer = (unsigned short)pgm_read_word_near(table_address);
 | 
					 | 
				
			||||||
    timer -= (((unsigned short)pgm_read_word_near(table_address + 2) * (unsigned char)(step_rate & 0x0007)) >> 3);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
  if (timer < 100) { timer = 100; MYSERIAL.print(MSG_STEPPER_TOO_HIGH); MYSERIAL.println(step_rate); }//(20kHz this should never happen)
 | 
					 | 
				
			||||||
  return timer;
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
 * Set the stepper direction of each axis
 | 
					 * Set the stepper direction of each axis
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 *   X_AXIS=A_AXIS and Y_AXIS=B_AXIS for COREXY
 | 
					 *   X_AXIS=A_AXIS and Y_AXIS=B_AXIS for COREXY
 | 
				
			||||||
 *   X_AXIS=A_AXIS and Z_AXIS=C_AXIS for COREXZ
 | 
					 *   X_AXIS=A_AXIS and Z_AXIS=C_AXIS for COREXZ
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
void set_stepper_direction() {
 | 
					void Stepper::set_directions() {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  #define SET_STEP_DIR(AXIS) \
 | 
					  #define SET_STEP_DIR(AXIS) \
 | 
				
			||||||
    if (TEST(out_bits, AXIS ##_AXIS)) { \
 | 
					    if (motor_direction(AXIS ##_AXIS)) { \
 | 
				
			||||||
      AXIS ##_APPLY_DIR(INVERT_## AXIS ##_DIR, false); \
 | 
					      AXIS ##_APPLY_DIR(INVERT_## AXIS ##_DIR, false); \
 | 
				
			||||||
      count_direction[AXIS ##_AXIS] = -1; \
 | 
					      count_direction[AXIS ##_AXIS] = -1; \
 | 
				
			||||||
    } \
 | 
					    } \
 | 
				
			||||||
@@ -600,7 +224,7 @@ void set_stepper_direction() {
 | 
				
			|||||||
  SET_STEP_DIR(Z); // C
 | 
					  SET_STEP_DIR(Z); // C
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  #if DISABLED(ADVANCE)
 | 
					  #if DISABLED(ADVANCE)
 | 
				
			||||||
    if (TEST(out_bits, E_AXIS)) {
 | 
					    if (motor_direction(E_AXIS)) {
 | 
				
			||||||
      REV_E_DIR();
 | 
					      REV_E_DIR();
 | 
				
			||||||
      count_direction[E_AXIS] = -1;
 | 
					      count_direction[E_AXIS] = -1;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
@@ -611,49 +235,11 @@ void set_stepper_direction() {
 | 
				
			|||||||
  #endif //!ADVANCE
 | 
					  #endif //!ADVANCE
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Initializes the trapezoid generator from the current block. Called whenever a new
 | 
					 | 
				
			||||||
// block begins.
 | 
					 | 
				
			||||||
FORCE_INLINE void trapezoid_generator_reset() {
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  static int8_t last_extruder = -1;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  if (current_block->direction_bits != out_bits || current_block->active_extruder != last_extruder) {
 | 
					 | 
				
			||||||
    out_bits = current_block->direction_bits;
 | 
					 | 
				
			||||||
    last_extruder = current_block->active_extruder;
 | 
					 | 
				
			||||||
    set_stepper_direction();
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if ENABLED(ADVANCE)
 | 
					 | 
				
			||||||
    advance = current_block->initial_advance;
 | 
					 | 
				
			||||||
    final_advance = current_block->final_advance;
 | 
					 | 
				
			||||||
    // Do E steps + advance steps
 | 
					 | 
				
			||||||
    e_steps[current_block->active_extruder] += ((advance >>8) - old_advance);
 | 
					 | 
				
			||||||
    old_advance = advance >>8;
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
  deceleration_time = 0;
 | 
					 | 
				
			||||||
  // step_rate to timer interval
 | 
					 | 
				
			||||||
  OCR1A_nominal = calc_timer(current_block->nominal_rate);
 | 
					 | 
				
			||||||
  // make a note of the number of step loops required at nominal speed
 | 
					 | 
				
			||||||
  step_loops_nominal = step_loops;
 | 
					 | 
				
			||||||
  acc_step_rate = current_block->initial_rate;
 | 
					 | 
				
			||||||
  acceleration_time = calc_timer(acc_step_rate);
 | 
					 | 
				
			||||||
  OCR1A = acceleration_time;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  // SERIAL_ECHO_START;
 | 
					 | 
				
			||||||
  // SERIAL_ECHOPGM("advance :");
 | 
					 | 
				
			||||||
  // SERIAL_ECHO(current_block->advance/256.0);
 | 
					 | 
				
			||||||
  // SERIAL_ECHOPGM("advance rate :");
 | 
					 | 
				
			||||||
  // SERIAL_ECHO(current_block->advance_rate/256.0);
 | 
					 | 
				
			||||||
  // SERIAL_ECHOPGM("initial advance :");
 | 
					 | 
				
			||||||
  // SERIAL_ECHO(current_block->initial_advance/256.0);
 | 
					 | 
				
			||||||
  // SERIAL_ECHOPGM("final advance :");
 | 
					 | 
				
			||||||
  // SERIAL_ECHOLN(current_block->final_advance/256.0);
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
// "The Stepper Driver Interrupt" - This timer interrupt is the workhorse.
 | 
					// "The Stepper Driver Interrupt" - This timer interrupt is the workhorse.
 | 
				
			||||||
// It pops blocks from the block_buffer and executes them by pulsing the stepper pins appropriately.
 | 
					// It pops blocks from the block_buffer and executes them by pulsing the stepper pins appropriately.
 | 
				
			||||||
ISR(TIMER1_COMPA_vect) {
 | 
					ISR(TIMER1_COMPA_vect) { stepper.isr(); }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void Stepper::isr() {
 | 
				
			||||||
  if (cleaning_buffer_counter) {
 | 
					  if (cleaning_buffer_counter) {
 | 
				
			||||||
    current_block = NULL;
 | 
					    current_block = NULL;
 | 
				
			||||||
    plan_discard_current_block();
 | 
					    plan_discard_current_block();
 | 
				
			||||||
@@ -672,8 +258,8 @@ ISR(TIMER1_COMPA_vect) {
 | 
				
			|||||||
    if (current_block) {
 | 
					    if (current_block) {
 | 
				
			||||||
      current_block->busy = true;
 | 
					      current_block->busy = true;
 | 
				
			||||||
      trapezoid_generator_reset();
 | 
					      trapezoid_generator_reset();
 | 
				
			||||||
      counter_x = -(current_block->step_event_count >> 1);
 | 
					      counter_X = -(current_block->step_event_count >> 1);
 | 
				
			||||||
      counter_y = counter_z = counter_e = counter_x;
 | 
					      counter_Y = counter_Z = counter_E = counter_X;
 | 
				
			||||||
      step_events_completed = 0;
 | 
					      step_events_completed = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      #if ENABLED(Z_LATE_ENABLE)
 | 
					      #if ENABLED(Z_LATE_ENABLE)
 | 
				
			||||||
@@ -697,9 +283,9 @@ ISR(TIMER1_COMPA_vect) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    // Update endstops state, if enabled
 | 
					    // Update endstops state, if enabled
 | 
				
			||||||
    #if ENABLED(HAS_Z_MIN_PROBE)
 | 
					    #if ENABLED(HAS_Z_MIN_PROBE)
 | 
				
			||||||
      if (check_endstops || z_probe_is_active) update_endstops();
 | 
					      if (endstops.enabled || endstops.z_probe_enabled) endstops.update();
 | 
				
			||||||
    #else
 | 
					    #else
 | 
				
			||||||
      if (check_endstops) update_endstops();
 | 
					      if (endstops.enabled) endstops.update();
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // Take multiple steps per interrupt (For high speed moves)
 | 
					    // Take multiple steps per interrupt (For high speed moves)
 | 
				
			||||||
@@ -709,48 +295,47 @@ ISR(TIMER1_COMPA_vect) {
 | 
				
			|||||||
      #endif
 | 
					      #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      #if ENABLED(ADVANCE)
 | 
					      #if ENABLED(ADVANCE)
 | 
				
			||||||
        counter_e += current_block->steps[E_AXIS];
 | 
					        counter_E += current_block->steps[E_AXIS];
 | 
				
			||||||
        if (counter_e > 0) {
 | 
					        if (counter_E > 0) {
 | 
				
			||||||
          counter_e -= current_block->step_event_count;
 | 
					          counter_E -= current_block->step_event_count;
 | 
				
			||||||
          e_steps[current_block->active_extruder] += TEST(out_bits, E_AXIS) ? -1 : 1;
 | 
					          e_steps[current_block->active_extruder] += motor_direction(E_AXIS) ? -1 : 1;
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
      #endif //ADVANCE
 | 
					      #endif //ADVANCE
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      #define _COUNTER(axis) counter_## axis
 | 
					      #define _COUNTER(AXIS) counter_## AXIS
 | 
				
			||||||
      #define _APPLY_STEP(AXIS) AXIS ##_APPLY_STEP
 | 
					      #define _APPLY_STEP(AXIS) AXIS ##_APPLY_STEP
 | 
				
			||||||
      #define _INVERT_STEP_PIN(AXIS) INVERT_## AXIS ##_STEP_PIN
 | 
					      #define _INVERT_STEP_PIN(AXIS) INVERT_## AXIS ##_STEP_PIN
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      #define STEP_ADD(axis, AXIS) \
 | 
					      #define STEP_ADD(AXIS) \
 | 
				
			||||||
        _COUNTER(axis) += current_block->steps[_AXIS(AXIS)]; \
 | 
					        _COUNTER(AXIS) += current_block->steps[_AXIS(AXIS)]; \
 | 
				
			||||||
        if (_COUNTER(axis) > 0) { _APPLY_STEP(AXIS)(!_INVERT_STEP_PIN(AXIS),0); }
 | 
					        if (_COUNTER(AXIS) > 0) { _APPLY_STEP(AXIS)(!_INVERT_STEP_PIN(AXIS),0); }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      STEP_ADD(x,X);
 | 
					      STEP_ADD(X);
 | 
				
			||||||
      STEP_ADD(y,Y);
 | 
					      STEP_ADD(Y);
 | 
				
			||||||
      STEP_ADD(z,Z);
 | 
					      STEP_ADD(Z);
 | 
				
			||||||
      #if DISABLED(ADVANCE)
 | 
					      #if DISABLED(ADVANCE)
 | 
				
			||||||
        STEP_ADD(e,E);
 | 
					        STEP_ADD(E);
 | 
				
			||||||
      #endif
 | 
					      #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      #define STEP_IF_COUNTER(axis, AXIS) \
 | 
					      #define STEP_IF_COUNTER(AXIS) \
 | 
				
			||||||
        if (_COUNTER(axis) > 0) { \
 | 
					        if (_COUNTER(AXIS) > 0) { \
 | 
				
			||||||
          _COUNTER(axis) -= current_block->step_event_count; \
 | 
					          _COUNTER(AXIS) -= current_block->step_event_count; \
 | 
				
			||||||
          count_position[_AXIS(AXIS)] += count_direction[_AXIS(AXIS)]; \
 | 
					          count_position[_AXIS(AXIS)] += count_direction[_AXIS(AXIS)]; \
 | 
				
			||||||
          _APPLY_STEP(AXIS)(_INVERT_STEP_PIN(AXIS),0); \
 | 
					          _APPLY_STEP(AXIS)(_INVERT_STEP_PIN(AXIS),0); \
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      STEP_IF_COUNTER(x, X);
 | 
					      STEP_IF_COUNTER(X);
 | 
				
			||||||
      STEP_IF_COUNTER(y, Y);
 | 
					      STEP_IF_COUNTER(Y);
 | 
				
			||||||
      STEP_IF_COUNTER(z, Z);
 | 
					      STEP_IF_COUNTER(Z);
 | 
				
			||||||
      #if DISABLED(ADVANCE)
 | 
					      #if DISABLED(ADVANCE)
 | 
				
			||||||
        STEP_IF_COUNTER(e, E);
 | 
					        STEP_IF_COUNTER(E);
 | 
				
			||||||
      #endif
 | 
					      #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      step_events_completed++;
 | 
					      step_events_completed++;
 | 
				
			||||||
      if (step_events_completed >= current_block->step_event_count) break;
 | 
					      if (step_events_completed >= current_block->step_event_count) break;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    // Calculate new timer value
 | 
					    // Calculate new timer value
 | 
				
			||||||
    unsigned short timer;
 | 
					    unsigned short timer, step_rate;
 | 
				
			||||||
    unsigned short step_rate;
 | 
					 | 
				
			||||||
    if (step_events_completed <= (unsigned long)current_block->accelerate_until) {
 | 
					    if (step_events_completed <= (unsigned long)current_block->accelerate_until) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      MultiU24X32toH16(acc_step_rate, acceleration_time, current_block->acceleration_rate);
 | 
					      MultiU24X32toH16(acc_step_rate, acceleration_time, current_block->acceleration_rate);
 | 
				
			||||||
@@ -817,10 +402,11 @@ ISR(TIMER1_COMPA_vect) {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if ENABLED(ADVANCE)
 | 
					#if ENABLED(ADVANCE)
 | 
				
			||||||
  unsigned char old_OCR0A;
 | 
					 | 
				
			||||||
  // Timer interrupt for E. e_steps is set in the main routine;
 | 
					  // Timer interrupt for E. e_steps is set in the main routine;
 | 
				
			||||||
  // Timer 0 is shared with millies
 | 
					  // Timer 0 is shared with millies
 | 
				
			||||||
  ISR(TIMER0_COMPA_vect) {
 | 
					  ISR(TIMER0_COMPA_vect) { stepper.advance_isr(); }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  void Stepper::advance_isr() {
 | 
				
			||||||
    old_OCR0A += 52; // ~10kHz interrupt (250000 / 26 = 9615kHz)
 | 
					    old_OCR0A += 52; // ~10kHz interrupt (250000 / 26 = 9615kHz)
 | 
				
			||||||
    OCR0A = old_OCR0A;
 | 
					    OCR0A = old_OCR0A;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -852,9 +438,10 @@ ISR(TIMER1_COMPA_vect) {
 | 
				
			|||||||
      #endif
 | 
					      #endif
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#endif // ADVANCE
 | 
					#endif // ADVANCE
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void st_init() {
 | 
					void Stepper::init() {
 | 
				
			||||||
  digipot_init(); //Initialize Digipot Motor Current
 | 
					  digipot_init(); //Initialize Digipot Motor Current
 | 
				
			||||||
  microstep_init(); //Initialize Microstepping Pins
 | 
					  microstep_init(); //Initialize Microstepping Pins
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -944,70 +531,10 @@ void st_init() {
 | 
				
			|||||||
    if (!E_ENABLE_ON) E3_ENABLE_WRITE(HIGH);
 | 
					    if (!E_ENABLE_ON) E3_ENABLE_WRITE(HIGH);
 | 
				
			||||||
  #endif
 | 
					  #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  //endstops and pullups
 | 
					  //
 | 
				
			||||||
 | 
					  // Init endstops and pullups here
 | 
				
			||||||
  #if HAS_X_MIN
 | 
					  //
 | 
				
			||||||
    SET_INPUT(X_MIN_PIN);
 | 
					  endstops.init();
 | 
				
			||||||
    #if ENABLED(ENDSTOPPULLUP_XMIN)
 | 
					 | 
				
			||||||
      WRITE(X_MIN_PIN,HIGH);
 | 
					 | 
				
			||||||
    #endif
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if HAS_Y_MIN
 | 
					 | 
				
			||||||
    SET_INPUT(Y_MIN_PIN);
 | 
					 | 
				
			||||||
    #if ENABLED(ENDSTOPPULLUP_YMIN)
 | 
					 | 
				
			||||||
      WRITE(Y_MIN_PIN,HIGH);
 | 
					 | 
				
			||||||
    #endif
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if HAS_Z_MIN
 | 
					 | 
				
			||||||
    SET_INPUT(Z_MIN_PIN);
 | 
					 | 
				
			||||||
    #if ENABLED(ENDSTOPPULLUP_ZMIN)
 | 
					 | 
				
			||||||
      WRITE(Z_MIN_PIN,HIGH);
 | 
					 | 
				
			||||||
    #endif
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if HAS_Z2_MIN
 | 
					 | 
				
			||||||
    SET_INPUT(Z2_MIN_PIN);
 | 
					 | 
				
			||||||
    #if ENABLED(ENDSTOPPULLUP_ZMIN)
 | 
					 | 
				
			||||||
      WRITE(Z2_MIN_PIN,HIGH);
 | 
					 | 
				
			||||||
    #endif
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if HAS_X_MAX
 | 
					 | 
				
			||||||
    SET_INPUT(X_MAX_PIN);
 | 
					 | 
				
			||||||
    #if ENABLED(ENDSTOPPULLUP_XMAX)
 | 
					 | 
				
			||||||
      WRITE(X_MAX_PIN,HIGH);
 | 
					 | 
				
			||||||
    #endif
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if HAS_Y_MAX
 | 
					 | 
				
			||||||
    SET_INPUT(Y_MAX_PIN);
 | 
					 | 
				
			||||||
    #if ENABLED(ENDSTOPPULLUP_YMAX)
 | 
					 | 
				
			||||||
      WRITE(Y_MAX_PIN,HIGH);
 | 
					 | 
				
			||||||
    #endif
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if HAS_Z_MAX
 | 
					 | 
				
			||||||
    SET_INPUT(Z_MAX_PIN);
 | 
					 | 
				
			||||||
    #if ENABLED(ENDSTOPPULLUP_ZMAX)
 | 
					 | 
				
			||||||
      WRITE(Z_MAX_PIN,HIGH);
 | 
					 | 
				
			||||||
    #endif
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if HAS_Z2_MAX
 | 
					 | 
				
			||||||
    SET_INPUT(Z2_MAX_PIN);
 | 
					 | 
				
			||||||
    #if ENABLED(ENDSTOPPULLUP_ZMAX)
 | 
					 | 
				
			||||||
      WRITE(Z2_MAX_PIN,HIGH);
 | 
					 | 
				
			||||||
    #endif
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  #if HAS_Z_PROBE && ENABLED(Z_MIN_PROBE_ENDSTOP) // Check for Z_MIN_PROBE_ENDSTOP so we don't pull a pin high unless it's to be used.
 | 
					 | 
				
			||||||
    SET_INPUT(Z_MIN_PROBE_PIN);
 | 
					 | 
				
			||||||
    #if ENABLED(ENDSTOPPULLUP_ZMIN_PROBE)
 | 
					 | 
				
			||||||
      WRITE(Z_MIN_PROBE_PIN,HIGH);
 | 
					 | 
				
			||||||
    #endif
 | 
					 | 
				
			||||||
  #endif
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
  #define _STEP_INIT(AXIS) AXIS ##_STEP_INIT
 | 
					  #define _STEP_INIT(AXIS) AXIS ##_STEP_INIT
 | 
				
			||||||
  #define _WRITE_STEP(AXIS, HIGHLOW) AXIS ##_STEP_WRITE(HIGHLOW)
 | 
					  #define _WRITE_STEP(AXIS, HIGHLOW) AXIS ##_STEP_WRITE(HIGHLOW)
 | 
				
			||||||
@@ -1083,17 +610,17 @@ void st_init() {
 | 
				
			|||||||
    SBI(TIMSK0, OCIE0A);
 | 
					    SBI(TIMSK0, OCIE0A);
 | 
				
			||||||
  #endif //ADVANCE
 | 
					  #endif //ADVANCE
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  enable_endstops(true); // Start with endstops active. After homing they can be disabled
 | 
					  endstops.enable(true); // Start with endstops active. After homing they can be disabled
 | 
				
			||||||
  sei();
 | 
					  sei();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  set_stepper_direction(); // Init directions to out_bits = 0
 | 
					  set_directions(); // Init directions to last_direction_bits = 0
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
 * Block until all buffered steps are executed
 | 
					 * Block until all buffered steps are executed
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
void st_synchronize() { while (blocks_queued()) idle(); }
 | 
					void Stepper::synchronize() { while (blocks_queued()) idle(); }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
 * Set the stepper positions directly in steps
 | 
					 * Set the stepper positions directly in steps
 | 
				
			||||||
@@ -1101,10 +628,10 @@ void st_synchronize() { while (blocks_queued()) idle(); }
 | 
				
			|||||||
 * The input is based on the typical per-axis XYZ steps.
 | 
					 * The input is based on the typical per-axis XYZ steps.
 | 
				
			||||||
 * For CORE machines XYZ needs to be translated to ABC.
 | 
					 * For CORE machines XYZ needs to be translated to ABC.
 | 
				
			||||||
 *
 | 
					 *
 | 
				
			||||||
 * This allows st_get_axis_position_mm to correctly
 | 
					 * This allows get_axis_position_mm to correctly
 | 
				
			||||||
 * derive the current XYZ position later on.
 | 
					 * derive the current XYZ position later on.
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
void st_set_position(const long& x, const long& y, const long& z, const long& e) {
 | 
					void Stepper::set_position(const long& x, const long& y, const long& z, const long& e) {
 | 
				
			||||||
  CRITICAL_SECTION_START;
 | 
					  CRITICAL_SECTION_START;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  #if ENABLED(COREXY)
 | 
					  #if ENABLED(COREXY)
 | 
				
			||||||
@@ -1129,7 +656,7 @@ void st_set_position(const long& x, const long& y, const long& z, const long& e)
 | 
				
			|||||||
  CRITICAL_SECTION_END;
 | 
					  CRITICAL_SECTION_END;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void st_set_e_position(const long& e) {
 | 
					void Stepper::set_e_position(const long& e) {
 | 
				
			||||||
  CRITICAL_SECTION_START;
 | 
					  CRITICAL_SECTION_START;
 | 
				
			||||||
  count_position[E_AXIS] = e;
 | 
					  count_position[E_AXIS] = e;
 | 
				
			||||||
  CRITICAL_SECTION_END;
 | 
					  CRITICAL_SECTION_END;
 | 
				
			||||||
@@ -1138,7 +665,7 @@ void st_set_e_position(const long& e) {
 | 
				
			|||||||
/**
 | 
					/**
 | 
				
			||||||
 * Get a stepper's position in steps.
 | 
					 * Get a stepper's position in steps.
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
long st_get_position(AxisEnum axis) {
 | 
					long Stepper::position(AxisEnum axis) {
 | 
				
			||||||
  CRITICAL_SECTION_START;
 | 
					  CRITICAL_SECTION_START;
 | 
				
			||||||
  long count_pos = count_position[axis];
 | 
					  long count_pos = count_position[axis];
 | 
				
			||||||
  CRITICAL_SECTION_END;
 | 
					  CRITICAL_SECTION_END;
 | 
				
			||||||
@@ -1149,7 +676,7 @@ long st_get_position(AxisEnum axis) {
 | 
				
			|||||||
 * Get an axis position according to stepper position(s)
 | 
					 * Get an axis position according to stepper position(s)
 | 
				
			||||||
 * For CORE machines apply translation from ABC to XYZ.
 | 
					 * For CORE machines apply translation from ABC to XYZ.
 | 
				
			||||||
 */
 | 
					 */
 | 
				
			||||||
float st_get_axis_position_mm(AxisEnum axis) {
 | 
					float Stepper::get_axis_position_mm(AxisEnum axis) {
 | 
				
			||||||
  float axis_steps;
 | 
					  float axis_steps;
 | 
				
			||||||
  #if ENABLED(COREXY) | ENABLED(COREXZ)
 | 
					  #if ENABLED(COREXY) | ENABLED(COREXZ)
 | 
				
			||||||
    if (axis == X_AXIS || axis == CORE_AXIS_2) {
 | 
					    if (axis == X_AXIS || axis == CORE_AXIS_2) {
 | 
				
			||||||
@@ -1162,19 +689,19 @@ float st_get_axis_position_mm(AxisEnum axis) {
 | 
				
			|||||||
      axis_steps = (pos1 + ((axis == X_AXIS) ? pos2 : -pos2)) / 2.0f;
 | 
					      axis_steps = (pos1 + ((axis == X_AXIS) ? pos2 : -pos2)) / 2.0f;
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    else
 | 
					    else
 | 
				
			||||||
      axis_steps = st_get_position(axis);
 | 
					      axis_steps = position(axis);
 | 
				
			||||||
  #else
 | 
					  #else
 | 
				
			||||||
    axis_steps = st_get_position(axis);
 | 
					    axis_steps = position(axis);
 | 
				
			||||||
  #endif
 | 
					  #endif
 | 
				
			||||||
  return axis_steps / axis_steps_per_unit[axis];
 | 
					  return axis_steps / axis_steps_per_unit[axis];
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void finishAndDisableSteppers() {
 | 
					void Stepper::finish_and_disable() {
 | 
				
			||||||
  st_synchronize();
 | 
					  synchronize();
 | 
				
			||||||
  disable_all_steppers();
 | 
					  disable_all_steppers();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void quickStop() {
 | 
					void Stepper::quick_stop() {
 | 
				
			||||||
  cleaning_buffer_counter = 5000;
 | 
					  cleaning_buffer_counter = 5000;
 | 
				
			||||||
  DISABLE_STEPPER_DRIVER_INTERRUPT();
 | 
					  DISABLE_STEPPER_DRIVER_INTERRUPT();
 | 
				
			||||||
  while (blocks_queued()) plan_discard_current_block();
 | 
					  while (blocks_queued()) plan_discard_current_block();
 | 
				
			||||||
@@ -1182,11 +709,62 @@ void quickStop() {
 | 
				
			|||||||
  ENABLE_STEPPER_DRIVER_INTERRUPT();
 | 
					  ENABLE_STEPPER_DRIVER_INTERRUPT();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void Stepper::endstop_triggered(AxisEnum axis) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #if ENABLED(COREXY) || ENABLED(COREXZ)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    float axis_pos = count_position[axis];
 | 
				
			||||||
 | 
					    if (axis == A_AXIS)
 | 
				
			||||||
 | 
					      axis_pos = (axis_pos + count_position[CORE_AXIS_2]) / 2;
 | 
				
			||||||
 | 
					    else if (axis == CORE_AXIS_2)
 | 
				
			||||||
 | 
					      axis_pos = (count_position[A_AXIS] - axis_pos) / 2;
 | 
				
			||||||
 | 
					    endstops_trigsteps[axis] = axis_pos;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #else // !COREXY && !COREXZ
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    endstops_trigsteps[axis] = count_position[axis];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #endif // !COREXY && !COREXZ
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  kill_current_block();
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void Stepper::report_positions() {
 | 
				
			||||||
 | 
					  CRITICAL_SECTION_START;
 | 
				
			||||||
 | 
					  long xpos = count_position[X_AXIS],
 | 
				
			||||||
 | 
					       ypos = count_position[Y_AXIS],
 | 
				
			||||||
 | 
					       zpos = count_position[Z_AXIS];
 | 
				
			||||||
 | 
					  CRITICAL_SECTION_END;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #if ENABLED(COREXY) || ENABLED(COREXZ)
 | 
				
			||||||
 | 
					    SERIAL_PROTOCOLPGM(MSG_COUNT_A);
 | 
				
			||||||
 | 
					  #else
 | 
				
			||||||
 | 
					    SERIAL_PROTOCOLPGM(MSG_COUNT_X);
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					  SERIAL_PROTOCOL(xpos);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #if ENABLED(COREXY) || ENABLED(COREXZ)
 | 
				
			||||||
 | 
					    SERIAL_PROTOCOLPGM(" B:");
 | 
				
			||||||
 | 
					  #else
 | 
				
			||||||
 | 
					    SERIAL_PROTOCOLPGM(" Y:");
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					  SERIAL_PROTOCOL(ypos);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  #if ENABLED(COREXZ) || ENABLED(COREXZ)
 | 
				
			||||||
 | 
					    SERIAL_PROTOCOLPGM(" C:");
 | 
				
			||||||
 | 
					  #else
 | 
				
			||||||
 | 
					    SERIAL_PROTOCOLPGM(" Z:");
 | 
				
			||||||
 | 
					  #endif
 | 
				
			||||||
 | 
					  SERIAL_PROTOCOL(zpos);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  SERIAL_EOL;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if ENABLED(BABYSTEPPING)
 | 
					#if ENABLED(BABYSTEPPING)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  // MUST ONLY BE CALLED BY AN ISR,
 | 
					  // MUST ONLY BE CALLED BY AN ISR,
 | 
				
			||||||
  // No other ISR should ever interrupt this!
 | 
					  // No other ISR should ever interrupt this!
 | 
				
			||||||
  void babystep(const uint8_t axis, const bool direction) {
 | 
					  void Stepper::babystep(const uint8_t axis, const bool direction) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    #define _ENABLE(axis) enable_## axis()
 | 
					    #define _ENABLE(axis) enable_## axis()
 | 
				
			||||||
    #define _READ_DIR(AXIS) AXIS ##_DIR_READ
 | 
					    #define _READ_DIR(AXIS) AXIS ##_DIR_READ
 | 
				
			||||||
@@ -1256,10 +834,14 @@ void quickStop() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
#endif //BABYSTEPPING
 | 
					#endif //BABYSTEPPING
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/**
 | 
				
			||||||
 | 
					 * Software-controlled Stepper Motor Current
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if HAS_DIGIPOTSS
 | 
					#if HAS_DIGIPOTSS
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  // From Arduino DigitalPotControl example
 | 
					  // From Arduino DigitalPotControl example
 | 
				
			||||||
  void digitalPotWrite(int address, int value) {
 | 
					  void Stepper::digitalPotWrite(int address, int value) {
 | 
				
			||||||
    digitalWrite(DIGIPOTSS_PIN, LOW); // take the SS pin low to select the chip
 | 
					    digitalWrite(DIGIPOTSS_PIN, LOW); // take the SS pin low to select the chip
 | 
				
			||||||
    SPI.transfer(address); //  send in the address and value via SPI:
 | 
					    SPI.transfer(address); //  send in the address and value via SPI:
 | 
				
			||||||
    SPI.transfer(value);
 | 
					    SPI.transfer(value);
 | 
				
			||||||
@@ -1269,8 +851,7 @@ void quickStop() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
#endif //HAS_DIGIPOTSS
 | 
					#endif //HAS_DIGIPOTSS
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Initialize Digipot Motor Current
 | 
					void Stepper::digipot_init() {
 | 
				
			||||||
void digipot_init() {
 | 
					 | 
				
			||||||
  #if HAS_DIGIPOTSS
 | 
					  #if HAS_DIGIPOTSS
 | 
				
			||||||
    const uint8_t digipot_motor_current[] = DIGIPOT_MOTOR_CURRENT;
 | 
					    const uint8_t digipot_motor_current[] = DIGIPOT_MOTOR_CURRENT;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -1299,7 +880,7 @@ void digipot_init() {
 | 
				
			|||||||
  #endif
 | 
					  #endif
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void digipot_current(uint8_t driver, int current) {
 | 
					void Stepper::digipot_current(uint8_t driver, int current) {
 | 
				
			||||||
  #if HAS_DIGIPOTSS
 | 
					  #if HAS_DIGIPOTSS
 | 
				
			||||||
    const uint8_t digipot_ch[] = DIGIPOT_CHANNELS;
 | 
					    const uint8_t digipot_ch[] = DIGIPOT_CHANNELS;
 | 
				
			||||||
    digitalPotWrite(digipot_ch[driver], current);
 | 
					    digitalPotWrite(digipot_ch[driver], current);
 | 
				
			||||||
@@ -1322,7 +903,7 @@ void digipot_current(uint8_t driver, int current) {
 | 
				
			|||||||
  #endif
 | 
					  #endif
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void microstep_init() {
 | 
					void Stepper::microstep_init() {
 | 
				
			||||||
  #if HAS_MICROSTEPS_E1
 | 
					  #if HAS_MICROSTEPS_E1
 | 
				
			||||||
    pinMode(E1_MS1_PIN, OUTPUT);
 | 
					    pinMode(E1_MS1_PIN, OUTPUT);
 | 
				
			||||||
    pinMode(E1_MS2_PIN, OUTPUT);
 | 
					    pinMode(E1_MS2_PIN, OUTPUT);
 | 
				
			||||||
@@ -1343,7 +924,11 @@ void microstep_init() {
 | 
				
			|||||||
  #endif
 | 
					  #endif
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void microstep_ms(uint8_t driver, int8_t ms1, int8_t ms2) {
 | 
					/**
 | 
				
			||||||
 | 
					 * Software-controlled Microstepping
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void Stepper::microstep_ms(uint8_t driver, int8_t ms1, int8_t ms2) {
 | 
				
			||||||
  if (ms1 >= 0) switch (driver) {
 | 
					  if (ms1 >= 0) switch (driver) {
 | 
				
			||||||
    case 0: digitalWrite(X_MS1_PIN, ms1); break;
 | 
					    case 0: digitalWrite(X_MS1_PIN, ms1); break;
 | 
				
			||||||
    case 1: digitalWrite(Y_MS1_PIN, ms1); break;
 | 
					    case 1: digitalWrite(Y_MS1_PIN, ms1); break;
 | 
				
			||||||
@@ -1364,7 +949,7 @@ void microstep_ms(uint8_t driver, int8_t ms1, int8_t ms2) {
 | 
				
			|||||||
  }
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void microstep_mode(uint8_t driver, uint8_t stepping_mode) {
 | 
					void Stepper::microstep_mode(uint8_t driver, uint8_t stepping_mode) {
 | 
				
			||||||
  switch (stepping_mode) {
 | 
					  switch (stepping_mode) {
 | 
				
			||||||
    case 1: microstep_ms(driver, MICROSTEP1); break;
 | 
					    case 1: microstep_ms(driver, MICROSTEP1); break;
 | 
				
			||||||
    case 2: microstep_ms(driver, MICROSTEP2); break;
 | 
					    case 2: microstep_ms(driver, MICROSTEP2); break;
 | 
				
			||||||
@@ -1374,7 +959,7 @@ void microstep_mode(uint8_t driver, uint8_t stepping_mode) {
 | 
				
			|||||||
  }
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void microstep_readings() {
 | 
					void Stepper::microstep_readings() {
 | 
				
			||||||
  SERIAL_PROTOCOLPGM("MS1,MS2 Pins\n");
 | 
					  SERIAL_PROTOCOLPGM("MS1,MS2 Pins\n");
 | 
				
			||||||
  SERIAL_PROTOCOLPGM("X: ");
 | 
					  SERIAL_PROTOCOLPGM("X: ");
 | 
				
			||||||
  SERIAL_PROTOCOL(digitalRead(X_MS1_PIN));
 | 
					  SERIAL_PROTOCOL(digitalRead(X_MS1_PIN));
 | 
				
			||||||
@@ -1396,7 +981,7 @@ void microstep_readings() {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if ENABLED(Z_DUAL_ENDSTOPS)
 | 
					#if ENABLED(Z_DUAL_ENDSTOPS)
 | 
				
			||||||
  void In_Homing_Process(bool state) { performing_homing = state; }
 | 
					  void Stepper::set_homing_flag(bool state) { performing_homing = state; }
 | 
				
			||||||
  void Lock_z_motor(bool state) { locked_z_motor = state; }
 | 
					  void Stepper::set_z_lock(bool state) { locked_z_motor = state; }
 | 
				
			||||||
  void Lock_z2_motor(bool state) { locked_z2_motor = state; }
 | 
					  void Stepper::set_z2_lock(bool state) { locked_z2_motor = state; }
 | 
				
			||||||
#endif
 | 
					#endif
 | 
				
			||||||
 
 | 
				
			|||||||
							
								
								
									
										347
									
								
								Marlin/stepper.h
									
									
									
									
									
								
							
							
						
						
									
										347
									
								
								Marlin/stepper.h
									
									
									
									
									
								
							@@ -21,90 +21,313 @@
 | 
				
			|||||||
 */
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/**
 | 
					/**
 | 
				
			||||||
  stepper.h - stepper motor driver: executes motion plans of planner.c using the stepper motors
 | 
					 * stepper.h - stepper motor driver: executes motion plans of planner.c using the stepper motors
 | 
				
			||||||
  Part of Grbl
 | 
					 * Part of Grbl
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Copyright (c) 2009-2011 Simen Svale Skogsrud
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Grbl is free software: you can redistribute it and/or modify
 | 
				
			||||||
 | 
					 * it under the terms of the GNU General Public License as published by
 | 
				
			||||||
 | 
					 * the Free Software Foundation, either version 3 of the License, or
 | 
				
			||||||
 | 
					 * (at your option) any later version.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * Grbl is distributed in the hope that it will be useful,
 | 
				
			||||||
 | 
					 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
				
			||||||
 | 
					 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
				
			||||||
 | 
					 * GNU General Public License for more details.
 | 
				
			||||||
 | 
					 *
 | 
				
			||||||
 | 
					 * You should have received a copy of the GNU General Public License
 | 
				
			||||||
 | 
					 * along with Grbl.  If not, see <http://www.gnu.org/licenses/>.
 | 
				
			||||||
 | 
					 */
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  Copyright (c) 2009-2011 Simen Svale Skogsrud
 | 
					#ifndef STEPPER_H
 | 
				
			||||||
 | 
					#define STEPPER_H
 | 
				
			||||||
  Grbl is free software: you can redistribute it and/or modify
 | 
					 | 
				
			||||||
  it under the terms of the GNU General Public License as published by
 | 
					 | 
				
			||||||
  the Free Software Foundation, either version 3 of the License, or
 | 
					 | 
				
			||||||
  (at your option) any later version.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  Grbl is distributed in the hope that it will be useful,
 | 
					 | 
				
			||||||
  but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					 | 
				
			||||||
  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | 
					 | 
				
			||||||
  GNU General Public License for more details.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  You should have received a copy of the GNU General Public License
 | 
					 | 
				
			||||||
  along with Grbl.  If not, see <http://www.gnu.org/licenses/>.
 | 
					 | 
				
			||||||
*/
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#ifndef stepper_h
 | 
					 | 
				
			||||||
#define stepper_h
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
#include "planner.h"
 | 
					#include "planner.h"
 | 
				
			||||||
 | 
					#include "speed_lookuptable.h"
 | 
				
			||||||
#include "stepper_indirection.h"
 | 
					#include "stepper_indirection.h"
 | 
				
			||||||
 | 
					#include "language.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if ENABLED(ABORT_ON_ENDSTOP_HIT_FEATURE_ENABLED)
 | 
					class Stepper;
 | 
				
			||||||
  extern bool abort_on_endstop_hit;
 | 
					extern Stepper stepper;
 | 
				
			||||||
#endif
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Initialize and start the stepper motor subsystem
 | 
					// intRes = intIn1 * intIn2 >> 16
 | 
				
			||||||
void st_init();
 | 
					// uses:
 | 
				
			||||||
 | 
					// r26 to store 0
 | 
				
			||||||
 | 
					// r27 to store the byte 1 of the 24 bit result
 | 
				
			||||||
 | 
					#define MultiU16X8toH16(intRes, charIn1, intIn2) \
 | 
				
			||||||
 | 
					  asm volatile ( \
 | 
				
			||||||
 | 
					                 "clr r26 \n\t" \
 | 
				
			||||||
 | 
					                 "mul %A1, %B2 \n\t" \
 | 
				
			||||||
 | 
					                 "movw %A0, r0 \n\t" \
 | 
				
			||||||
 | 
					                 "mul %A1, %A2 \n\t" \
 | 
				
			||||||
 | 
					                 "add %A0, r1 \n\t" \
 | 
				
			||||||
 | 
					                 "adc %B0, r26 \n\t" \
 | 
				
			||||||
 | 
					                 "lsr r0 \n\t" \
 | 
				
			||||||
 | 
					                 "adc %A0, r26 \n\t" \
 | 
				
			||||||
 | 
					                 "adc %B0, r26 \n\t" \
 | 
				
			||||||
 | 
					                 "clr r1 \n\t" \
 | 
				
			||||||
 | 
					                 : \
 | 
				
			||||||
 | 
					                 "=&r" (intRes) \
 | 
				
			||||||
 | 
					                 : \
 | 
				
			||||||
 | 
					                 "d" (charIn1), \
 | 
				
			||||||
 | 
					                 "d" (intIn2) \
 | 
				
			||||||
 | 
					                 : \
 | 
				
			||||||
 | 
					                 "r26" \
 | 
				
			||||||
 | 
					               )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Block until all buffered steps are executed
 | 
					class Stepper {
 | 
				
			||||||
void st_synchronize();
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Set current position in steps
 | 
					  public:
 | 
				
			||||||
void st_set_position(const long& x, const long& y, const long& z, const long& e);
 | 
					 | 
				
			||||||
void st_set_e_position(const long& e);
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Get current position in steps
 | 
					    block_t* current_block = NULL;  // A pointer to the block currently being traced
 | 
				
			||||||
long st_get_position(AxisEnum axis);
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Get current axis position in mm
 | 
					    #if ENABLED(ABORT_ON_ENDSTOP_HIT_FEATURE_ENABLED)
 | 
				
			||||||
float st_get_axis_position_mm(AxisEnum axis);
 | 
					      bool abort_on_endstop_hit = false;
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// The stepper subsystem goes to sleep when it runs out of things to execute. Call this
 | 
					    #if ENABLED(Z_DUAL_ENDSTOPS)
 | 
				
			||||||
// to notify the subsystem that it is time to go to work.
 | 
					      bool performing_homing = false;
 | 
				
			||||||
void st_wake_up();
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #if ENABLED(ADVANCE)
 | 
				
			||||||
 | 
					      long e_steps[4];
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void checkHitEndstops(); //call from somewhere to create an serial error message with the locations the endstops where hit, in case they were triggered
 | 
					  private:
 | 
				
			||||||
void endstops_hit_on_purpose(); //avoid creation of the message, i.e. after homing and before a routine call of checkHitEndstops();
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
void enable_endstops(bool check); // Enable/disable endstop checking
 | 
					    unsigned char last_direction_bits = 0;        // The next stepping-bits to be output
 | 
				
			||||||
 | 
					    unsigned int cleaning_buffer_counter = 0;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void enable_endstops_globally(bool check);
 | 
					    #if ENABLED(Z_DUAL_ENDSTOPS)
 | 
				
			||||||
void endstops_not_homing();
 | 
					      bool locked_z_motor = false,
 | 
				
			||||||
 | 
					           locked_z2_motor = false;
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void checkStepperErrors(); //Print errors detected by the stepper
 | 
					    // Counter variables for the Bresenham line tracer
 | 
				
			||||||
 | 
					    long counter_X = 0, counter_Y = 0, counter_Z = 0, counter_E = 0;
 | 
				
			||||||
 | 
					    volatile unsigned long step_events_completed = 0; // The number of step events executed in the current block
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void finishAndDisableSteppers();
 | 
					    #if ENABLED(ADVANCE)
 | 
				
			||||||
 | 
					      unsigned char old_OCR0A;
 | 
				
			||||||
 | 
					      long advance_rate, advance, final_advance = 0;
 | 
				
			||||||
 | 
					      long old_advance = 0;
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
extern block_t* current_block;  // A pointer to the block currently being traced
 | 
					    long acceleration_time, deceleration_time;
 | 
				
			||||||
 | 
					    //unsigned long accelerate_until, decelerate_after, acceleration_rate, initial_rate, final_rate, nominal_rate;
 | 
				
			||||||
 | 
					    unsigned short acc_step_rate; // needed for deceleration start point
 | 
				
			||||||
 | 
					    uint8_t step_loops;
 | 
				
			||||||
 | 
					    uint8_t step_loops_nominal;
 | 
				
			||||||
 | 
					    unsigned short OCR1A_nominal;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void quickStop();
 | 
					    volatile long endstops_trigsteps[3];
 | 
				
			||||||
 | 
					    volatile long endstops_stepsTotal, endstops_stepsDone;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if HAS_DIGIPOTSS
 | 
					    #if HAS_MOTOR_CURRENT_PWM
 | 
				
			||||||
 | 
					      #ifndef PWM_MOTOR_CURRENT
 | 
				
			||||||
 | 
					        #define PWM_MOTOR_CURRENT DEFAULT_PWM_MOTOR_CURRENT
 | 
				
			||||||
 | 
					      #endif
 | 
				
			||||||
 | 
					      const int motor_current_setting[3] = PWM_MOTOR_CURRENT;
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // Positions of stepper motors, in step units
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    volatile long count_position[NUM_AXIS] = { 0 };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // Current direction of stepper motors (+1 or -1)
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    volatile signed char count_direction[NUM_AXIS] = { 1 };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  public:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // Constructor / initializer
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    Stepper() {};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // Initialize stepper hardware
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    void init();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // Interrupt Service Routines
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    void isr();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #if ENABLED(ADVANCE)
 | 
				
			||||||
 | 
					      void advance_isr();
 | 
				
			||||||
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // Block until all buffered steps are executed
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    void synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // Set the current position in steps
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    void set_position(const long& x, const long& y, const long& z, const long& e);
 | 
				
			||||||
 | 
					    void set_e_position(const long& e);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // Set direction bits for all steppers
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    void set_directions();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // Get the position of a stepper, in steps
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    long position(AxisEnum axis);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // Report the positions of the steppers, in steps
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    void report_positions();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // Get the position (mm) of an axis based on stepper position(s)
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    float get_axis_position_mm(AxisEnum axis);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // The stepper subsystem goes to sleep when it runs out of things to execute. Call this
 | 
				
			||||||
 | 
					    // to notify the subsystem that it is time to go to work.
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    void wake_up();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // Wait for moves to finish and disable all steppers
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    void finish_and_disable();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // Quickly stop all steppers and clear the blocks queue
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    void quick_stop();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // The direction of a single motor
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    FORCE_INLINE bool motor_direction(AxisEnum axis) { return TEST(last_direction_bits, axis); }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    #if HAS_DIGIPOTSS
 | 
				
			||||||
      void digitalPotWrite(int address, int value);
 | 
					      void digitalPotWrite(int address, int value);
 | 
				
			||||||
#endif
 | 
					    #endif
 | 
				
			||||||
void microstep_ms(uint8_t driver, int8_t ms1, int8_t ms2);
 | 
					    void microstep_ms(uint8_t driver, int8_t ms1, int8_t ms2);
 | 
				
			||||||
void microstep_mode(uint8_t driver, uint8_t stepping);
 | 
					    void digipot_current(uint8_t driver, int current);
 | 
				
			||||||
void digipot_init();
 | 
					    void microstep_readings();
 | 
				
			||||||
void digipot_current(uint8_t driver, int current);
 | 
					 | 
				
			||||||
void microstep_init();
 | 
					 | 
				
			||||||
void microstep_readings();
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if ENABLED(Z_DUAL_ENDSTOPS)
 | 
					    #if ENABLED(Z_DUAL_ENDSTOPS)
 | 
				
			||||||
  void In_Homing_Process(bool state);
 | 
					      void set_homing_flag(bool state);
 | 
				
			||||||
  void Lock_z_motor(bool state);
 | 
					      void set_z_lock(bool state);
 | 
				
			||||||
  void Lock_z2_motor(bool state);
 | 
					      void set_z2_lock(bool state);
 | 
				
			||||||
#endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#if ENABLED(BABYSTEPPING)
 | 
					    #if ENABLED(BABYSTEPPING)
 | 
				
			||||||
      void babystep(const uint8_t axis, const bool direction); // perform a short step with a single stepper motor, outside of any convention
 | 
					      void babystep(const uint8_t axis, const bool direction); // perform a short step with a single stepper motor, outside of any convention
 | 
				
			||||||
#endif
 | 
					    #endif
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#endif
 | 
					    inline void kill_current_block() {
 | 
				
			||||||
 | 
					      step_events_completed = current_block->step_event_count;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // Handle a triggered endstop
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    void endstop_triggered(AxisEnum axis);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    // Triggered position of an axis in mm (not core-savvy)
 | 
				
			||||||
 | 
					    //
 | 
				
			||||||
 | 
					    FORCE_INLINE float triggered_position_mm(AxisEnum axis) {
 | 
				
			||||||
 | 
					      return endstops_trigsteps[axis] / axis_steps_per_unit[axis];
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    FORCE_INLINE unsigned short calc_timer(unsigned short step_rate) {
 | 
				
			||||||
 | 
					      unsigned short timer;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      NOMORE(step_rate, MAX_STEP_FREQUENCY);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      if (step_rate > 20000) { // If steprate > 20kHz >> step 4 times
 | 
				
			||||||
 | 
					        step_rate = (step_rate >> 2) & 0x3fff;
 | 
				
			||||||
 | 
					        step_loops = 4;
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					      else if (step_rate > 10000) { // If steprate > 10kHz >> step 2 times
 | 
				
			||||||
 | 
					        step_rate = (step_rate >> 1) & 0x7fff;
 | 
				
			||||||
 | 
					        step_loops = 2;
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					      else {
 | 
				
			||||||
 | 
					        step_loops = 1;
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      NOLESS(step_rate, F_CPU / 500000);
 | 
				
			||||||
 | 
					      step_rate -= F_CPU / 500000; // Correct for minimal speed
 | 
				
			||||||
 | 
					      if (step_rate >= (8 * 256)) { // higher step rate
 | 
				
			||||||
 | 
					        unsigned short table_address = (unsigned short)&speed_lookuptable_fast[(unsigned char)(step_rate >> 8)][0];
 | 
				
			||||||
 | 
					        unsigned char tmp_step_rate = (step_rate & 0x00ff);
 | 
				
			||||||
 | 
					        unsigned short gain = (unsigned short)pgm_read_word_near(table_address + 2);
 | 
				
			||||||
 | 
					        MultiU16X8toH16(timer, tmp_step_rate, gain);
 | 
				
			||||||
 | 
					        timer = (unsigned short)pgm_read_word_near(table_address) - timer;
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					      else { // lower step rates
 | 
				
			||||||
 | 
					        unsigned short table_address = (unsigned short)&speed_lookuptable_slow[0][0];
 | 
				
			||||||
 | 
					        table_address += ((step_rate) >> 1) & 0xfffc;
 | 
				
			||||||
 | 
					        timer = (unsigned short)pgm_read_word_near(table_address);
 | 
				
			||||||
 | 
					        timer -= (((unsigned short)pgm_read_word_near(table_address + 2) * (unsigned char)(step_rate & 0x0007)) >> 3);
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					      if (timer < 100) { timer = 100; MYSERIAL.print(MSG_STEPPER_TOO_HIGH); MYSERIAL.println(step_rate); }//(20kHz this should never happen)
 | 
				
			||||||
 | 
					      return timer;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    // Initializes the trapezoid generator from the current block. Called whenever a new
 | 
				
			||||||
 | 
					    // block begins.
 | 
				
			||||||
 | 
					    FORCE_INLINE void trapezoid_generator_reset() {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      static int8_t last_extruder = -1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      if (current_block->direction_bits != last_direction_bits || current_block->active_extruder != last_extruder) {
 | 
				
			||||||
 | 
					        last_direction_bits = current_block->direction_bits;
 | 
				
			||||||
 | 
					        last_extruder = current_block->active_extruder;
 | 
				
			||||||
 | 
					        set_directions();
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      #if ENABLED(ADVANCE)
 | 
				
			||||||
 | 
					        advance = current_block->initial_advance;
 | 
				
			||||||
 | 
					        final_advance = current_block->final_advance;
 | 
				
			||||||
 | 
					        // Do E steps + advance steps
 | 
				
			||||||
 | 
					        e_steps[current_block->active_extruder] += ((advance >>8) - old_advance);
 | 
				
			||||||
 | 
					        old_advance = advance >>8;
 | 
				
			||||||
 | 
					      #endif
 | 
				
			||||||
 | 
					      deceleration_time = 0;
 | 
				
			||||||
 | 
					      // step_rate to timer interval
 | 
				
			||||||
 | 
					      OCR1A_nominal = calc_timer(current_block->nominal_rate);
 | 
				
			||||||
 | 
					      // make a note of the number of step loops required at nominal speed
 | 
				
			||||||
 | 
					      step_loops_nominal = step_loops;
 | 
				
			||||||
 | 
					      acc_step_rate = current_block->initial_rate;
 | 
				
			||||||
 | 
					      acceleration_time = calc_timer(acc_step_rate);
 | 
				
			||||||
 | 
					      OCR1A = acceleration_time;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      // SERIAL_ECHO_START;
 | 
				
			||||||
 | 
					      // SERIAL_ECHOPGM("advance :");
 | 
				
			||||||
 | 
					      // SERIAL_ECHO(current_block->advance/256.0);
 | 
				
			||||||
 | 
					      // SERIAL_ECHOPGM("advance rate :");
 | 
				
			||||||
 | 
					      // SERIAL_ECHO(current_block->advance_rate/256.0);
 | 
				
			||||||
 | 
					      // SERIAL_ECHOPGM("initial advance :");
 | 
				
			||||||
 | 
					      // SERIAL_ECHO(current_block->initial_advance/256.0);
 | 
				
			||||||
 | 
					      // SERIAL_ECHOPGM("final advance :");
 | 
				
			||||||
 | 
					      // SERIAL_ECHOLN(current_block->final_advance/256.0);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  private:
 | 
				
			||||||
 | 
					    void microstep_mode(uint8_t driver, uint8_t stepping);
 | 
				
			||||||
 | 
					    void digipot_init();
 | 
				
			||||||
 | 
					    void microstep_init();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#endif // STEPPER_H
 | 
				
			||||||
@@ -604,7 +604,7 @@ float get_pid_output(int e) {
 | 
				
			|||||||
        #if ENABLED(PID_ADD_EXTRUSION_RATE)
 | 
					        #if ENABLED(PID_ADD_EXTRUSION_RATE)
 | 
				
			||||||
          cTerm[e] = 0;
 | 
					          cTerm[e] = 0;
 | 
				
			||||||
          if (e == active_extruder) {
 | 
					          if (e == active_extruder) {
 | 
				
			||||||
            long e_position = st_get_position(E_AXIS);
 | 
					            long e_position = stepper.position(E_AXIS);
 | 
				
			||||||
            if (e_position > last_position[e]) {
 | 
					            if (e_position > last_position[e]) {
 | 
				
			||||||
              lpq[lpq_ptr++] = e_position - last_position[e];
 | 
					              lpq[lpq_ptr++] = e_position - last_position[e];
 | 
				
			||||||
              last_position[e] = e_position;
 | 
					              last_position[e] = e_position;
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -476,7 +476,7 @@ inline void line_to_current(AxisEnum axis) {
 | 
				
			|||||||
  static void lcd_sdcard_resume() { card.startFileprint(); }
 | 
					  static void lcd_sdcard_resume() { card.startFileprint(); }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static void lcd_sdcard_stop() {
 | 
					  static void lcd_sdcard_stop() {
 | 
				
			||||||
    quickStop();
 | 
					    stepper.quick_stop();
 | 
				
			||||||
    card.sdprinting = false;
 | 
					    card.sdprinting = false;
 | 
				
			||||||
    card.closefile();
 | 
					    card.closefile();
 | 
				
			||||||
    autotempShutdown();
 | 
					    autotempShutdown();
 | 
				
			||||||
@@ -911,7 +911,7 @@ void lcd_cooldown() {
 | 
				
			|||||||
      current_position[Z_AXIS] = MESH_HOME_SEARCH_Z;
 | 
					      current_position[Z_AXIS] = MESH_HOME_SEARCH_Z;
 | 
				
			||||||
      line_to_current(Z_AXIS);
 | 
					      line_to_current(Z_AXIS);
 | 
				
			||||||
    #endif
 | 
					    #endif
 | 
				
			||||||
    st_synchronize();
 | 
					    stepper.synchronize();
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  static void _lcd_level_goto_next_point();
 | 
					  static void _lcd_level_goto_next_point();
 | 
				
			||||||
@@ -964,7 +964,7 @@ void lcd_cooldown() {
 | 
				
			|||||||
            #endif
 | 
					            #endif
 | 
				
			||||||
          ;
 | 
					          ;
 | 
				
			||||||
          line_to_current(Z_AXIS);
 | 
					          line_to_current(Z_AXIS);
 | 
				
			||||||
          st_synchronize();
 | 
					          stepper.synchronize();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
          mbl.active = true;
 | 
					          mbl.active = true;
 | 
				
			||||||
          enqueue_and_echo_commands_P(PSTR("G28"));
 | 
					          enqueue_and_echo_commands_P(PSTR("G28"));
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user