249 lines
6.9 KiB
C++
249 lines
6.9 KiB
C++
/**
|
|
* Marlin 3D Printer Firmware
|
|
* Copyright (c) 2020 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 <https://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
#pragma once
|
|
|
|
/**
|
|
* endstops.h - manages endstops
|
|
*/
|
|
|
|
#include "../inc/MarlinConfig.h"
|
|
#include <stdint.h>
|
|
|
|
#define __ES_ITEM(N) N,
|
|
#define _ES_ITEM(K,N) TERN_(K,DEFER4(__ES_ITEM)(N))
|
|
|
|
enum EndstopEnum : char {
|
|
// Common XYZ (ABC) endstops. Defined according to USE_[XYZ](MIN|MAX)_PLUG settings.
|
|
_ES_ITEM(HAS_X_MIN, X_MIN)
|
|
_ES_ITEM(HAS_X_MAX, X_MAX)
|
|
_ES_ITEM(HAS_Y_MIN, Y_MIN)
|
|
_ES_ITEM(HAS_Y_MAX, Y_MAX)
|
|
_ES_ITEM(HAS_Z_MIN, Z_MIN)
|
|
_ES_ITEM(HAS_Z_MAX, Z_MAX)
|
|
_ES_ITEM(HAS_I_MIN, I_MIN)
|
|
_ES_ITEM(HAS_I_MAX, I_MAX)
|
|
_ES_ITEM(HAS_J_MIN, J_MIN)
|
|
_ES_ITEM(HAS_J_MAX, J_MAX)
|
|
_ES_ITEM(HAS_K_MIN, K_MIN)
|
|
_ES_ITEM(HAS_K_MAX, K_MAX)
|
|
|
|
// Extra Endstops for XYZ
|
|
_ES_ITEM(HAS_X2_MIN, X2_MIN)
|
|
_ES_ITEM(HAS_X2_MAX, X2_MAX)
|
|
_ES_ITEM(HAS_Y2_MIN, Y2_MIN)
|
|
_ES_ITEM(HAS_Y2_MAX, Y2_MAX)
|
|
_ES_ITEM(HAS_Z2_MIN, Z2_MIN)
|
|
_ES_ITEM(HAS_Z2_MAX, Z2_MAX)
|
|
_ES_ITEM(HAS_Z3_MIN, Z3_MIN)
|
|
_ES_ITEM(HAS_Z3_MAX, Z3_MAX)
|
|
_ES_ITEM(HAS_Z4_MIN, Z4_MIN)
|
|
_ES_ITEM(HAS_Z4_MAX, Z4_MAX)
|
|
|
|
// Bed Probe state is distinct or shared with Z_MIN (i.e., when the probe is the only Z endstop)
|
|
_ES_ITEM(HAS_BED_PROBE, Z_MIN_PROBE IF_DISABLED(USES_Z_MIN_PROBE_PIN, = Z_MIN))
|
|
|
|
// The total number of states
|
|
NUM_ENDSTOP_STATES
|
|
|
|
// Endstops can be either MIN or MAX but not both
|
|
#if HAS_X_MIN || HAS_X_MAX
|
|
, X_ENDSTOP = TERN(X_HOME_TO_MAX, X_MAX, X_MIN)
|
|
#endif
|
|
#if HAS_Y_MIN || HAS_Y_MAX
|
|
, Y_ENDSTOP = TERN(Y_HOME_TO_MAX, Y_MAX, Y_MIN)
|
|
#endif
|
|
#if HAS_Z_MIN || HAS_Z_MAX || HOMING_Z_WITH_PROBE
|
|
, Z_ENDSTOP = TERN(Z_HOME_TO_MAX, Z_MAX, TERN(HOMING_Z_WITH_PROBE, Z_MIN_PROBE, Z_MIN))
|
|
#endif
|
|
#if HAS_I_MIN || HAS_I_MAX
|
|
, I_ENDSTOP = TERN(I_HOME_TO_MAX, I_MAX, I_MIN)
|
|
#endif
|
|
#if HAS_J_MIN || HAS_J_MAX
|
|
, J_ENDSTOP = TERN(J_HOME_TO_MAX, J_MAX, J_MIN)
|
|
#endif
|
|
#if HAS_K_MIN || HAS_K_MAX
|
|
, K_ENDSTOP = TERN(K_HOME_TO_MAX, K_MAX, K_MIN)
|
|
#endif
|
|
};
|
|
|
|
#undef __ES_ITEM
|
|
#undef _ES_ITEM
|
|
|
|
class Endstops {
|
|
public:
|
|
|
|
typedef IF<(NUM_ENDSTOP_STATES > 8), uint16_t, uint8_t>::type endstop_mask_t;
|
|
|
|
#if ENABLED(X_DUAL_ENDSTOPS)
|
|
static float x2_endstop_adj;
|
|
#endif
|
|
#if ENABLED(Y_DUAL_ENDSTOPS)
|
|
static float y2_endstop_adj;
|
|
#endif
|
|
#if ENABLED(Z_MULTI_ENDSTOPS)
|
|
static float z2_endstop_adj;
|
|
#endif
|
|
#if ENABLED(Z_MULTI_ENDSTOPS) && NUM_Z_STEPPER_DRIVERS >= 3
|
|
static float z3_endstop_adj;
|
|
#endif
|
|
#if ENABLED(Z_MULTI_ENDSTOPS) && NUM_Z_STEPPER_DRIVERS >= 4
|
|
static float z4_endstop_adj;
|
|
#endif
|
|
|
|
private:
|
|
static bool enabled, enabled_globally;
|
|
static endstop_mask_t live_state;
|
|
static volatile endstop_mask_t hit_state; // Use X_MIN, Y_MIN, Z_MIN and Z_MIN_PROBE as BIT index
|
|
|
|
#if ENDSTOP_NOISE_THRESHOLD
|
|
static endstop_mask_t validated_live_state;
|
|
static uint8_t endstop_poll_count; // Countdown from threshold for polling
|
|
#endif
|
|
|
|
public:
|
|
Endstops() {};
|
|
|
|
/**
|
|
* Initialize the endstop pins
|
|
*/
|
|
static void init();
|
|
|
|
/**
|
|
* Are endstops or the probe set to abort the move?
|
|
*/
|
|
FORCE_INLINE static bool abort_enabled() {
|
|
return enabled || TERN0(HAS_BED_PROBE, z_probe_enabled);
|
|
}
|
|
|
|
static inline bool global_enabled() { return enabled_globally; }
|
|
|
|
/**
|
|
* Periodic call to poll endstops if required. Called from temperature ISR
|
|
*/
|
|
static void poll();
|
|
|
|
/**
|
|
* Update endstops bits from the pins. Apply filtering to get a verified state.
|
|
* If abort_enabled() and moving towards a triggered switch, abort the current move.
|
|
* Called from ISR contexts.
|
|
*/
|
|
static void update();
|
|
|
|
/**
|
|
* Get Endstop hit state.
|
|
*/
|
|
FORCE_INLINE static endstop_mask_t trigger_state() { return hit_state; }
|
|
|
|
/**
|
|
* Get current endstops state
|
|
*/
|
|
FORCE_INLINE static endstop_mask_t state() {
|
|
return
|
|
#if ENDSTOP_NOISE_THRESHOLD
|
|
validated_live_state
|
|
#else
|
|
live_state
|
|
#endif
|
|
;
|
|
}
|
|
|
|
static inline bool probe_switch_activated() {
|
|
return (true
|
|
#if ENABLED(PROBE_ACTIVATION_SWITCH)
|
|
&& READ(PROBE_ACTIVATION_SWITCH_PIN) == PROBE_ACTIVATION_SWITCH_STATE
|
|
#endif
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Report endstop hits to serial. Called from loop().
|
|
*/
|
|
static void event_handler();
|
|
|
|
/**
|
|
* Report endstop states in response to M119
|
|
*/
|
|
static void report_states();
|
|
|
|
// Enable / disable endstop checking globally
|
|
static void enable_globally(const bool onoff=true);
|
|
|
|
// Enable / disable endstop checking
|
|
static void enable(const bool onoff=true);
|
|
|
|
// Disable / Enable endstops based on ENSTOPS_ONLY_FOR_HOMING and global enable
|
|
static void not_homing();
|
|
|
|
#if ENABLED(VALIDATE_HOMING_ENDSTOPS)
|
|
// If the last move failed to trigger an endstop, call kill
|
|
static void validate_homing_move();
|
|
#else
|
|
FORCE_INLINE static void validate_homing_move() { hit_on_purpose(); }
|
|
#endif
|
|
|
|
// Clear endstops (i.e., they were hit intentionally) to suppress the report
|
|
FORCE_INLINE static void hit_on_purpose() { hit_state = 0; }
|
|
|
|
// Enable / disable endstop z-probe checking
|
|
#if HAS_BED_PROBE
|
|
static volatile bool z_probe_enabled;
|
|
static void enable_z_probe(const bool onoff=true);
|
|
#endif
|
|
|
|
static void resync();
|
|
|
|
// Debugging of endstops
|
|
#if ENABLED(PINS_DEBUGGING)
|
|
static bool monitor_flag;
|
|
static void monitor();
|
|
static void run_monitor();
|
|
#endif
|
|
|
|
#if ENABLED(SPI_ENDSTOPS)
|
|
typedef struct {
|
|
union {
|
|
bool any;
|
|
struct { bool LINEAR_AXIS_LIST(x:1, y:1, z:1, i:1, j:1, k:1); };
|
|
};
|
|
} tmc_spi_homing_t;
|
|
static tmc_spi_homing_t tmc_spi_homing;
|
|
static void clear_endstop_state();
|
|
static bool tmc_spi_homing_check();
|
|
#endif
|
|
};
|
|
|
|
extern Endstops endstops;
|
|
|
|
/**
|
|
* A class to save and change the endstop state,
|
|
* then restore it when it goes out of scope.
|
|
*/
|
|
class TemporaryGlobalEndstopsState {
|
|
bool saved;
|
|
|
|
public:
|
|
TemporaryGlobalEndstopsState(const bool enable) : saved(endstops.global_enabled()) {
|
|
endstops.enable_globally(enable);
|
|
}
|
|
~TemporaryGlobalEndstopsState() { endstops.enable_globally(saved); }
|
|
};
|