2016-03-25 07:19:46 +01:00
/**
2016-03-24 19:01:20 +01:00
* Marlin 3 D Printer Firmware
2017-03-18 16:15:54 +01:00
* Copyright ( C ) 2016 , 2017 MarlinFirmware [ https : //github.com/MarlinFirmware/Marlin]
2015-05-17 10:44:53 +02:00
*
* 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/>.
2016-03-24 19:01:20 +01:00
*
*/
2016-03-25 07:19:46 +01:00
/**
2015-05-17 10:44:53 +02:00
* About Marlin
*
* This firmware is a mashup between Sprinter and grbl .
* - https : //github.com/kliment/Sprinter
* - https : //github.com/simen/grbl/tree
2012-11-06 12:06:41 +01:00
*/
2012-11-21 20:53:56 +01:00
2015-04-14 02:17:36 +02:00
/**
2016-11-19 04:54:38 +01:00
* - - - - - - - - - - - - - - - - -
* G - Codes in Marlin
* - - - - - - - - - - - - - - - - -
*
* Helpful G - code references :
2015-04-14 02:17:36 +02:00
* - http : //linuxcnc.org/handbook/gcode/g-code.html
* - http : //objects.reprap.org/wiki/Mendel_User_Manual:_RepRapGCodes
*
2016-11-19 04:54:38 +01:00
* Help to document Marlin ' s G - codes online :
2015-04-14 02:17:36 +02:00
* - http : //reprap.org/wiki/G-code
2016-11-19 04:54:38 +01:00
* - https : //github.com/MarlinFirmware/MarlinDocumentation
2015-05-17 10:44:53 +02:00
*
* - - - - - - - - - - - - - - - - -
2015-04-14 02:17:36 +02:00
*
* " G " Codes
*
2017-05-21 22:47:09 +02:00
* G0 - > G1
* G1 - Coordinated Movement X Y Z E
* G2 - CW ARC
* G3 - CCW ARC
* G4 - Dwell S < seconds > or P < milliseconds >
* G5 - Cubic B - spline with XYZE destination and IJPQ offsets
2017-07-18 05:02:12 +02:00
* G10 - Retract filament according to settings of M207 ( Requires FWRETRACT )
* G11 - Retract recover filament according to settings of M208 ( Requires FWRETRACT )
* G12 - Clean tool ( Requires NOZZLE_CLEAN_FEATURE )
2017-06-23 21:28:28 +02:00
* G17 - Select Plane XY ( Requires CNC_WORKSPACE_PLANES )
* G18 - Select Plane ZX ( Requires CNC_WORKSPACE_PLANES )
* G19 - Select Plane YZ ( Requires CNC_WORKSPACE_PLANES )
2017-07-18 05:02:12 +02:00
* G20 - Set input units to inches ( Requires INCH_MODE_SUPPORT )
* G21 - Set input units to millimeters ( Requires INCH_MODE_SUPPORT )
2017-05-21 22:47:09 +02:00
* G26 - Mesh Validation Pattern ( Requires UBL_G26_MESH_VALIDATION )
* G27 - Park Nozzle ( Requires NOZZLE_PARK_FEATURE )
* G28 - Home one or more axes
2017-07-18 05:02:12 +02:00
* G29 - Start or continue the bed leveling probe procedure ( Requires bed leveling )
2017-05-21 22:47:09 +02:00
* G30 - Single Z probe , probes bed at X Y location ( defaults to current XY location )
* G31 - Dock sled ( Z_PROBE_SLED only )
* G32 - Undock sled ( Z_PROBE_SLED only )
* G33 - Delta Auto - Calibration ( Requires DELTA_AUTO_CALIBRATION )
2017-07-18 05:02:12 +02:00
* G38 - Probe in any direction using the Z_MIN_PROBE ( Requires G38_PROBE_TARGET )
2017-05-21 22:47:09 +02:00
* G42 - Coordinated move to a mesh point ( Requires AUTO_BED_LEVELING_UBL )
* G90 - Use Absolute Coordinates
* G91 - Use Relative Coordinates
* G92 - Set current position to coordinates given
2015-04-14 02:17:36 +02:00
*
* " M " Codes
*
* M0 - Unconditional stop - Wait for user to press a button on the LCD ( Only if ULTRA_LCD is enabled )
2017-06-22 16:44:39 +02:00
* M1 - > M0
2017-04-07 20:52:45 +02:00
* M3 - Turn laser / spindle on , set spindle / laser speed / power , set rotation to clockwise
* M4 - Turn laser / spindle on , set spindle / laser speed / power , set rotation to counter - clockwise
* M5 - Turn laser / spindle off
2015-04-14 02:17:36 +02:00
* M17 - Enable / Power all stepper motors
* M18 - Disable all stepper motors ; same as M84
2016-09-25 14:50:11 +02:00
* M20 - List SD card . ( Requires SDSUPPORT )
* M21 - Init SD card . ( Requires SDSUPPORT )
* M22 - Release SD card . ( Requires SDSUPPORT )
* M23 - Select SD file : " M23 /path/file.gco " . ( Requires SDSUPPORT )
* M24 - Start / resume SD print . ( Requires SDSUPPORT )
* M25 - Pause SD print . ( Requires SDSUPPORT )
* M26 - Set SD position in bytes : " M26 S12345 " . ( Requires SDSUPPORT )
* M27 - Report SD print status . ( Requires SDSUPPORT )
* M28 - Start SD write : " M28 /path/file.gco " . ( Requires SDSUPPORT )
* M29 - Stop SD write . ( Requires SDSUPPORT )
* M30 - Delete file from SD : " M30 /path/file.gco "
* M31 - Report time since last M109 or SD card start to serial .
* M32 - Select file and start SD print : " M32 [S<bytepos>] !/path/file.gco# " . ( Requires SDSUPPORT )
* Use P to run other files as sub - programs : " M32 P !filename# "
2015-04-14 02:17:36 +02:00
* The ' # ' is necessary when calling from within sd files , as it stops buffer prereading
2016-09-25 14:50:11 +02:00
* M33 - Get the longname version of a path . ( Requires LONG_FILENAME_HOST_SUPPORT )
2017-02-09 14:02:25 +01:00
* M34 - Set SD Card sorting options . ( Requires SDCARD_SORT_ALPHA )
2016-09-25 14:50:11 +02:00
* M42 - Change pin status via gcode : M42 P < pin > S < value > . LED pin assumed if P is omitted .
2017-04-02 21:18:59 +02:00
* M43 - Display pin status , watch pins for changes , watch endstops & toggle LED , Z servo probe test , toggle pins
2016-09-25 14:50:11 +02:00
* M48 - Measure Z Probe repeatability : M48 P < points > X < pos > Y < pos > V < level > E < engage > L < legs > . ( Requires Z_MIN_PROBE_REPEATABILITY_TEST )
* M75 - Start the print job timer .
* M76 - Pause the print job timer .
* M77 - Stop the print job timer .
* M78 - Show statistical information about the print jobs . ( Requires PRINTCOUNTER )
2017-05-11 09:32:08 +02:00
* M80 - Turn on Power Supply . ( Requires POWER_SUPPLY > 0 )
* M81 - Turn off Power Supply . ( Requires POWER_SUPPLY > 0 )
2016-09-25 14:50:11 +02:00
* M82 - Set E codes absolute ( default ) .
* M83 - Set E codes relative while in Absolute ( G90 ) mode .
* M84 - Disable steppers until next move , or use S < seconds > to specify an idle
* duration after which steppers should turn off . S0 disables the timeout .
2015-04-14 02:17:36 +02:00
* M85 - Set inactivity shutdown timer with parameter S < seconds > . To disable set zero ( default )
2016-09-25 14:50:11 +02:00
* M92 - Set planner . axis_steps_per_mm for one or more axes .
2017-05-12 05:25:06 +02:00
* M100 - Watch Free Memory ( for debugging ) ( Requires M100_FREE_MEMORY_WATCHER )
2016-09-25 14:50:11 +02:00
* M104 - Set extruder target temp .
* M105 - Report current temperatures .
* M106 - Fan on .
* M107 - Fan off .
* M108 - Break out of heating loops ( M109 , M190 , M303 ) . With no controller , breaks out of M0 / M1 . ( Requires EMERGENCY_PARSER )
2015-04-14 02:17:36 +02:00
* M109 - Sxxx Wait for extruder current temp to reach target temp . Waits only when heating
* Rxxx Wait for extruder current temp to reach target temp . Waits when heating and cooling
2016-11-30 02:51:13 +01:00
* If AUTOTEMP is enabled , S < mintemp > B < maxtemp > F < factor > . Exit autotemp by any M109 without F
2016-09-25 14:50:11 +02:00
* M110 - Set the current line number . ( Used by host printing )
* M111 - Set debug flags : " M111 S<flagbits> " . See flag bits defined in enum . h .
* M112 - Emergency stop .
* M113 - Get or set the timeout interval for Host Keepalive " busy " messages . ( Requires HOST_KEEPALIVE_FEATURE )
* M114 - Report current position .
2016-11-09 01:05:59 +01:00
* M115 - Report capabilities . ( Extended capabilities requires EXTENDED_CAPABILITIES_REPORT )
2016-09-25 14:50:11 +02:00
* M117 - Display a message on the controller screen . ( Requires an LCD )
2017-06-27 21:49:21 +02:00
* M118 - Display a message in the host console .
2016-09-25 14:50:11 +02:00
* M119 - Report endstops status .
* M120 - Enable endstops detection .
* M121 - Disable endstops detection .
2017-03-19 10:26:22 +01:00
* M125 - Save current position and move to filament change position . ( Requires PARK_HEAD_ON_PAUSE )
2016-09-25 14:50:11 +02:00
* M126 - Solenoid Air Valve Open . ( Requires BARICUDA )
* M127 - Solenoid Air Valve Closed . ( Requires BARICUDA )
* M128 - EtoP Open . ( Requires BARICUDA )
* M129 - EtoP Closed . ( Requires BARICUDA )
* M140 - Set bed target temp . S < temp >
* M145 - Set heatup values for materials on the LCD . H < hotend > B < bed > F < fan speed > for S < material > ( 0 = PLA , 1 = ABS )
* M149 - Set temperature units . ( Requires TEMPERATURE_UNITS_SUPPORT )
2017-06-10 08:02:15 +02:00
* M150 - Set Status LED Color as R < red > U < green > B < blue > . Values 0 - 255. ( Requires BLINKM , RGB_LED , RGBW_LED , or PCA9632 )
2016-11-09 01:03:40 +01:00
* M155 - Auto - report temperatures with interval of S < seconds > . ( Requires AUTO_REPORT_TEMPERATURES )
2016-09-25 14:50:11 +02:00
* M163 - Set a single proportion for a mixing extruder . ( Requires MIXING_EXTRUDER )
* M164 - Save the mix as a virtual extruder . ( Requires MIXING_EXTRUDER and MIXING_VIRTUAL_TOOLS )
* M165 - Set the proportions for a mixing extruder . Use parameters ABCDHI to set the mixing factors . ( Requires MIXING_EXTRUDER )
* M190 - Sxxx Wait for bed current temp to reach target temp . * * Waits only when heating ! * *
* Rxxx Wait for bed current temp to reach target temp . * * Waits for heating or cooling . * *
2016-07-04 05:08:22 +02:00
* M200 - Set filament diameter , D < diameter > , setting E axis units to cubic . ( Use S0 to revert to linear units . )
2016-09-25 14:50:11 +02:00
* M201 - Set max acceleration in units / s ^ 2 for print moves : " M201 X<accel> Y<accel> Z<accel> E<accel> "
* M202 - Set max acceleration in units / s ^ 2 for travel moves : " M202 X<accel> Y<accel> Z<accel> E<accel> " * * UNUSED IN MARLIN ! * *
* M203 - Set maximum feedrate : " M203 X<fr> Y<fr> Z<fr> E<fr> " in units / sec .
* M204 - Set default acceleration in units / sec ^ 2 : P < printing > R < extruder_only > T < travel >
2016-07-04 05:08:22 +02:00
* M205 - Set advanced settings . Current units apply :
S < print > T < travel > minimum speeds
B < minimum segment time >
2016-10-03 08:54:15 +02:00
X < max X jerk > , Y < max Y jerk > , Z < max Z jerk > , E < max E jerk >
2017-04-15 04:41:21 +02:00
* M206 - Set additional homing offset . ( Disabled by NO_WORKSPACE_OFFSETS or DELTA )
2016-09-25 14:50:11 +02:00
* M207 - Set Retract Length : S < length > , Feedrate : F < units / min > , and Z lift : Z < distance > . ( Requires FWRETRACT )
* M208 - Set Recover ( unretract ) Additional ( ! ) Length : S < length > and Feedrate : F < units / min > . ( Requires FWRETRACT )
* M209 - Turn Automatic Retract Detection on / off : S < 0 | 1 > ( For slicers that don ' t support G10 / 11 ) . ( Requires FWRETRACT )
2016-07-04 05:08:22 +02:00
Every normal extrude - only move will be classified as retract depending on the direction .
2017-03-19 10:26:22 +01:00
* M211 - Enable , Disable , and / or Report software endstops : S < 0 | 1 > ( Requires MIN_SOFTWARE_ENDSTOPS or MAX_SOFTWARE_ENDSTOPS )
2016-09-25 14:50:11 +02:00
* M218 - Set a tool offset : " M218 T<index> X<offset> Y<offset> " . ( Requires 2 or more extruders )
* M220 - Set Feedrate Percentage : " M220 S<percent> " ( i . e . , " FR " on the LCD )
* M221 - Set Flow Percentage : " M221 S<percent> "
* M226 - Wait until a pin is in a given state : " M226 P<pin> S<state> "
* M240 - Trigger a camera to take a photograph . ( Requires CHDK or PHOTOGRAPH_PIN )
* M250 - Set LCD contrast : " M250 C<contrast> " ( 0 - 63 ) . ( Requires LCD support )
2016-11-08 23:28:42 +01:00
* M260 - i2c Send Data ( Requires EXPERIMENTAL_I2CBUS )
* M261 - i2c Request Data ( Requires EXPERIMENTAL_I2CBUS )
2016-09-25 14:50:11 +02:00
* M280 - Set servo position absolute : " M280 P<index> S<angle|µs> " . ( Requires servos )
2015-04-14 02:17:36 +02:00
* M300 - Play beep sound S < frequency Hz > P < duration ms >
2016-09-25 14:50:11 +02:00
* M301 - Set PID parameters P I and D . ( Requires PIDTEMP )
* M302 - Allow cold extrudes , or set the minimum extrude S < temperature > . ( Requires PREVENT_COLD_EXTRUSION )
* M303 - PID relay autotune S < temperature > sets the target temperature . Default 150 C . ( Requires PIDTEMP )
* M304 - Set bed PID parameters P I and D . ( Requires PIDTEMPBED )
2017-05-13 07:43:12 +02:00
* M350 - Set microstepping mode . ( Requires digital microstepping pins . )
* M351 - Toggle MS1 MS2 pins directly . ( Requires digital microstepping pins . )
2017-05-19 22:21:43 +02:00
* M355 - Set Case Light on / off and set brightness . ( Requires CASE_LIGHT_PIN )
2016-09-25 14:50:11 +02:00
* M380 - Activate solenoid on active extruder . ( Requires EXT_SOLENOID )
* M381 - Disable all solenoids . ( Requires EXT_SOLENOID )
* M400 - Finish all moves .
* M401 - Lower Z probe . ( Requires a probe )
* M402 - Raise Z probe . ( Requires a probe )
* M404 - Display or set the Nominal Filament Width : " W<diameter> " . ( Requires FILAMENT_WIDTH_SENSOR )
* M405 - Enable Filament Sensor flow control . " M405 D<delay_cm> " . ( Requires FILAMENT_WIDTH_SENSOR )
* M406 - Disable Filament Sensor flow control . ( Requires FILAMENT_WIDTH_SENSOR )
* M407 - Display measured filament diameter in millimeters . ( Requires FILAMENT_WIDTH_SENSOR )
* M410 - Quickstop . Abort all planned moves .
2016-09-29 08:03:28 +02:00
* M420 - Enable / Disable Leveling ( with current values ) S1 = enable S0 = disable ( Requires MESH_BED_LEVELING or ABL )
2017-04-02 23:13:15 +02:00
* M421 - Set a single Z coordinate in the Mesh Leveling grid . X < units > Y < units > Z < units > ( Requires MESH_BED_LEVELING or AUTO_BED_LEVELING_UBL )
2017-04-15 04:41:21 +02:00
* M428 - Set the home_offset based on the current_position . Nearest edge applies . ( Disabled by NO_WORKSPACE_OFFSETS or DELTA )
2016-09-25 14:50:11 +02:00
* M500 - Store parameters in EEPROM . ( Requires EEPROM_SETTINGS )
* M501 - Restore parameters from EEPROM . ( Requires EEPROM_SETTINGS )
* M502 - Revert to the default " factory settings " . * * Does not write them to EEPROM ! * *
* M503 - Print the current settings ( in memory ) : " M503 S<verbose> " . S0 specifies compact output .
* M540 - Enable / disable SD card abort on endstop hit : " M540 S<state> " . ( Requires ABORT_ON_ENDSTOP_HIT_FEATURE_ENABLED )
2017-05-26 20:01:02 +02:00
* M600 - Pause for filament change : " M600 X<pos> Y<pos> Z<raise> E<first_retract> L<later_retract> " . ( Requires ADVANCED_PAUSE_FEATURE )
2017-03-07 23:35:27 +01:00
* M665 - Set delta configurations : " M665 L<diagonal rod> R<delta radius> S<segments/s> A<rod A trim mm> B<rod B trim mm> C<rod C trim mm> I<tower A trim angle> J<tower B trim angle> K<tower C trim angle> " ( Requires DELTA )
2016-09-25 14:50:11 +02:00
* M666 - Set delta endstop adjustment . ( Requires DELTA )
* M605 - Set dual x - carriage movement mode : " M605 S<mode> [X<x_offset>] [R<temp_offset>] " . ( Requires DUAL_X_CARRIAGE )
* M851 - Set Z probe ' s Z offset in current units . ( Negative = below the nozzle . )
2017-06-09 14:06:23 +02:00
* M860 - Report the position of position encoder modules .
* M861 - Report the status of position encoder modules .
* M862 - Perform an axis continuity test for position encoder modules .
* M863 - Perform steps - per - mm calibration for position encoder modules .
* M864 - Change position encoder module I2C address .
* M865 - Check position encoder module firmware version .
* M866 - Report or reset position encoder module error count .
* M867 - Enable / disable or toggle error correction for position encoder modules .
* M868 - Report or set position encoder module error correction threshold .
* M869 - Report position encoder module error .
2017-05-13 07:43:12 +02:00
* M900 - Get and / or Set advance K factor and WH / D ratio . ( Requires LIN_ADVANCE )
2017-03-07 06:00:43 +01:00
* M906 - Set or get motor current in milliamps using axis codes X , Y , Z , E . Report values if no axis codes given . ( Requires HAVE_TMC2130 )
2016-09-25 14:50:11 +02:00
* M907 - Set digital trimpot motor current using axis codes . ( Requires a board with digital trimpots )
* M908 - Control digital trimpot directly . ( Requires DAC_STEPPER_CURRENT or DIGIPOTSS_PIN )
* M909 - Print digipot / DAC current value . ( Requires DAC_STEPPER_CURRENT )
* M910 - Commit digipot / DAC value to external EEPROM via I2C . ( Requires DAC_STEPPER_CURRENT )
2017-03-07 06:00:43 +01:00
* M911 - Report stepper driver overtemperature pre - warn condition . ( Requires HAVE_TMC2130 )
* M912 - Clear stepper driver overtemperature pre - warn condition flag . ( Requires HAVE_TMC2130 )
2017-04-15 05:44:08 +02:00
* M913 - Set HYBRID_THRESHOLD speed . ( Requires HYBRID_THRESHOLD )
* M914 - Set SENSORLESS_HOMING sensitivity . ( Requires SENSORLESS_HOMING )
2015-04-14 02:17:36 +02:00
*
* M360 - SCARA calibration : Move to cal - position ThetaA ( 0 deg calibration )
* M361 - SCARA calibration : Move to cal - position ThetaB ( 90 deg calibration - steps per degree )
* M362 - SCARA calibration : Move to cal - position PsiA ( 0 deg calibration )
* M363 - SCARA calibration : Move to cal - position PsiB ( 90 deg calibration - steps per degree )
* M364 - SCARA calibration : Move to cal - position PSIC ( 90 deg to Theta calibration position )
*
2015-05-19 13:25:15 +02:00
* * * * * * * * * * * * * Custom codes - This can change to suit future G - code regulations
2016-09-25 14:50:11 +02:00
* M928 - Start SD logging : " M928 filename.gco " . Stop with M29 . ( Requires SDSUPPORT )
2015-04-14 02:17:36 +02:00
* M999 - Restart after being stopped by error
2015-05-17 10:44:53 +02:00
*
* " T " Codes
*
2016-09-25 14:50:11 +02:00
* T0 - T3 - Select an extruder ( tool ) by index : " T<n> F<units/min> "
2015-05-17 10:44:53 +02:00
*
2015-04-14 02:17:36 +02:00
*/
2012-11-06 12:06:41 +01:00
2016-11-19 04:54:38 +01:00
# include "Marlin.h"
2017-09-06 13:28:33 +02:00
# include "lcd/ultralcd.h"
# include "module/planner.h"
# include "module/stepper.h"
# include "module/endstops.h"
# include "module/temperature.h"
# include "sd/cardreader.h"
# include "module/configuration_store.h"
2017-06-18 01:36:10 +02:00
# ifdef ARDUINO
2017-09-06 13:28:33 +02:00
# include <pins_arduino.h>
2017-06-18 01:36:10 +02:00
# endif
2017-09-06 13:28:33 +02:00
# include <math.h>
# include "libs/nozzle.h"
# include "libs/duration_t.h"
# include "gcode/parser.h"
2016-11-19 04:54:38 +01:00
# if HAS_ABL
2017-09-06 13:28:33 +02:00
# include "libs/vector_3.h"
2016-11-19 04:54:38 +01:00
# if ENABLED(AUTO_BED_LEVELING_LINEAR)
2017-09-06 13:28:33 +02:00
# include "libs/least_squares_fit.h"
2016-11-19 04:54:38 +01:00
# endif
# elif ENABLED(MESH_BED_LEVELING)
2017-09-06 13:28:33 +02:00
# include "feature/mbl/mesh_bed_leveling.h"
2016-11-19 04:54:38 +01:00
# endif
# if ENABLED(BEZIER_CURVE_SUPPORT)
2017-09-06 13:28:33 +02:00
# include "module/planner_bezier.h"
2016-11-19 04:54:38 +01:00
# endif
# if HAS_BUZZER && DISABLED(LCD_USE_I2C_BUZZER)
2017-09-06 13:28:33 +02:00
# include "libs/buzzer.h"
2016-11-19 04:54:38 +01:00
# endif
2017-08-26 00:03:07 +02:00
# if ENABLED(MAX7219_DEBUG)
2017-09-06 13:28:33 +02:00
# include "feature/leds/Max7219_Debug_LEDs.h"
2017-08-26 00:03:07 +02:00
# endif
2017-07-13 17:01:21 +02:00
# if ENABLED(NEOPIXEL_RGBW_LED)
# include <Adafruit_NeoPixel.h>
# endif
2016-11-19 04:54:38 +01:00
# if ENABLED(BLINKM)
2017-09-06 13:28:33 +02:00
# include "feature/leds/blinkm.h"
2016-11-19 04:54:38 +01:00
# endif
2017-06-10 08:02:15 +02:00
# if ENABLED(PCA9632)
2017-09-06 13:28:33 +02:00
# include "feature/leds/pca9632.h"
2017-06-10 08:02:15 +02:00
# endif
2016-11-19 04:54:38 +01:00
# if HAS_SERVOS
2017-09-06 13:28:33 +02:00
# include "HAL/servo.h"
2016-11-19 04:54:38 +01:00
# endif
# if HAS_DIGIPOTSS
# include <SPI.h>
# endif
# if ENABLED(DAC_STEPPER_CURRENT)
2017-09-06 13:28:33 +02:00
# include "feature/dac/stepper_dac.h"
2016-11-19 04:54:38 +01:00
# endif
# if ENABLED(EXPERIMENTAL_I2CBUS)
2017-09-06 13:28:33 +02:00
# include "feature/twibus.h"
2016-11-19 04:54:38 +01:00
# endif
2017-06-09 14:06:23 +02:00
# if ENABLED(I2C_POSITION_ENCODERS)
2017-09-06 13:28:33 +02:00
# include "feature/I2CPositionEncoder.h"
2017-06-09 14:06:23 +02:00
# endif
2016-11-19 04:54:38 +01:00
# if ENABLED(ENDSTOP_INTERRUPTS_FEATURE)
2017-09-06 13:28:33 +02:00
# include "HAL/HAL_endstop_interrupts.h"
2016-11-19 04:54:38 +01:00
# endif
2015-07-31 07:24:43 +02:00
# if ENABLED(M100_FREE_MEMORY_WATCHER)
2015-07-19 01:59:12 +02:00
void gcode_M100 ( ) ;
2017-04-17 00:33:22 +02:00
void M100_dump_routine ( const char * const title , const char * start , const char * end ) ;
2015-07-06 02:42:13 +02:00
# endif
2015-07-31 07:24:43 +02:00
# if ENABLED(SDSUPPORT)
2015-01-30 03:52:21 +01:00
CardReader card ;
2012-11-06 12:06:41 +01:00
# endif
2015-01-30 03:52:21 +01:00
2016-03-27 00:25:28 +01:00
# if ENABLED(EXPERIMENTAL_I2CBUS)
TWIBus i2c ;
# endif
2016-09-29 22:06:01 +02:00
# if ENABLED(G38_PROBE_TARGET)
2016-09-26 08:30:34 +02:00
bool G38_move = false ,
G38_endstop_hit = false ;
2016-09-19 21:17:10 +02:00
# endif
2017-03-18 16:15:54 +01:00
# if ENABLED(AUTO_BED_LEVELING_UBL)
2017-09-06 13:28:33 +02:00
# include "feature/ubl/ubl.h"
2017-06-13 01:26:49 +02:00
extern bool defer_return_to_status ;
2017-03-20 07:42:41 +01:00
unified_bed_leveling ubl ;
2017-03-31 04:32:50 +02:00
# define UBL_MESH_VALID !( ( ubl.z_values[0][0] == ubl.z_values[0][1] && ubl.z_values[0][1] == ubl.z_values[0][2] \
& & ubl . z_values [ 1 ] [ 0 ] = = ubl . z_values [ 1 ] [ 1 ] & & ubl . z_values [ 1 ] [ 1 ] = = ubl . z_values [ 1 ] [ 2 ] \
& & ubl . z_values [ 2 ] [ 0 ] = = ubl . z_values [ 2 ] [ 1 ] & & ubl . z_values [ 2 ] [ 1 ] = = ubl . z_values [ 2 ] [ 2 ] \
& & ubl . z_values [ 0 ] [ 0 ] = = 0 & & ubl . z_values [ 1 ] [ 0 ] = = 0 & & ubl . z_values [ 2 ] [ 0 ] = = 0 ) \
| | isnan ( ubl . z_values [ 0 ] [ 0 ] ) )
2017-03-18 16:15:54 +01:00
# endif
2015-04-08 09:56:19 +02:00
bool Running = true ;
2016-10-05 04:32:37 +02:00
/**
* Cartesian Current Position
* Used to track the logical position as moves are queued .
* Used by ' line_to_current_position ' to do a move after changing it .
* Used by ' SYNC_PLAN_POSITION_KINEMATIC ' to update ' planner . position ' .
*/
float current_position [ XYZE ] = { 0.0 } ;
/**
* Cartesian Destination
* A temporary position , usually applied to ' current_position ' .
* Set with ' gcode_get_destination ' or ' set_destination_to_current ' .
* ' line_to_destination ' sets ' current_position ' to ' destination ' .
*/
2017-03-18 16:15:54 +01:00
float destination [ XYZE ] = { 0.0 } ;
2016-10-05 04:32:37 +02:00
/**
* axis_homed
* Flags that each linear axis was homed .
* XYZ on cartesian , ABC on delta , ABZ on SCARA .
*
* axis_known_position
* Flags that the position is known in each linear axis . Set when homed .
* Cleared whenever a stepper powers off , potentially losing its position .
*/
bool axis_homed [ XYZ ] = { false } , axis_known_position [ XYZ ] = { false } ;
2015-04-08 09:56:19 +02:00
2016-10-05 04:32:37 +02:00
/**
* GCode line number handling . Hosts may opt to include line numbers when
* sending commands to Marlin , and lines will be checked for sequentiality .
2017-01-13 12:19:16 +01:00
* M110 N < int > sets the current line number .
2016-10-05 04:32:37 +02:00
*/
2015-04-08 09:56:19 +02:00
static long gcode_N , gcode_LastN , Stopped_gcode_LastN = 0 ;
2015-04-14 02:17:36 +02:00
2016-10-05 04:32:37 +02:00
/**
* GCode Command Queue
* A simple ring buffer of BUFSIZE command strings .
*
* Commands are copied into this buffer by the command injectors
* ( immediate , serial , sd card ) and they are processed sequentially by
* the main loop . The process_next_command function parses the next
* command and hands off execution to individual handler functions .
*/
2017-03-15 09:32:00 +01:00
uint8_t commands_in_queue = 0 ; // Count of commands in the queue
2016-10-05 04:32:37 +02:00
static uint8_t cmd_queue_index_r = 0 , // Ring buffer read position
2017-03-15 09:32:00 +01:00
cmd_queue_index_w = 0 ; // Ring buffer write position
2017-04-16 02:14:40 +02:00
# if ENABLED(M100_FREE_MEMORY_WATCHER)
char command_queue [ BUFSIZE ] [ MAX_CMD_SIZE ] ; // Necessary so M100 Free Memory Dumper can show us the commands and any corruption
# else // This can be collapsed back to the way it was soon.
2017-03-15 09:32:00 +01:00
static char command_queue [ BUFSIZE ] [ MAX_CMD_SIZE ] ;
2017-04-16 02:14:40 +02:00
# endif
2016-10-05 04:32:37 +02:00
/**
* Next Injected Command pointer . NULL if no commands are being injected .
* Used by Marlin internally to ensure that commands initiated from within
* are enqueued ahead of any pending serial or sd card commands .
*/
static const char * injected_commands_P = NULL ;
2015-04-08 09:56:19 +02:00
2016-05-31 17:26:08 +02:00
# if ENABLED(TEMPERATURE_UNITS_SUPPORT)
TempUnit input_temp_units = TEMPUNIT_C ;
# endif
2016-07-16 03:49:34 +02:00
/**
* Feed rates are often configured with mm / m
* but the planner and stepper like mm / s units .
*/
2017-06-06 01:49:16 +02:00
static const float homing_feedrate_mm_s [ ] PROGMEM = {
2016-07-20 02:11:57 +02:00
# if ENABLED(DELTA)
2016-08-07 09:20:25 +02:00
MMM_TO_MMS ( HOMING_FEEDRATE_Z ) , MMM_TO_MMS ( HOMING_FEEDRATE_Z ) ,
2016-07-20 02:11:57 +02:00
# else
2016-08-07 09:20:25 +02:00
MMM_TO_MMS ( HOMING_FEEDRATE_XY ) , MMM_TO_MMS ( HOMING_FEEDRATE_XY ) ,
2016-07-20 02:11:57 +02:00
# endif
2016-08-07 09:20:25 +02:00
MMM_TO_MMS ( HOMING_FEEDRATE_Z ) , 0
2016-07-20 02:11:57 +02:00
} ;
2017-06-06 01:49:16 +02:00
FORCE_INLINE float homing_feedrate ( const AxisEnum a ) { return pgm_read_float ( & homing_feedrate_mm_s [ a ] ) ; }
2017-05-20 10:03:08 +02:00
float feedrate_mm_s = MMM_TO_MMS ( 1500.0 ) ;
static float saved_feedrate_mm_s ;
2017-06-20 05:39:23 +02:00
int16_t feedrate_percentage = 100 , saved_feedrate_percentage ,
2016-10-05 04:32:37 +02:00
flow_percentage [ EXTRUDERS ] = ARRAY_BY_EXTRUDERS1 ( 100 ) ;
2016-04-28 18:57:21 +02:00
2017-07-18 10:17:50 +02:00
// Initialized by settings.load()
2016-10-05 04:32:37 +02:00
bool axis_relative_modes [ ] = AXIS_RELATIVE_MODES ,
2017-07-18 10:17:50 +02:00
volumetric_enabled ;
float filament_size [ EXTRUDERS ] , volumetric_multiplier [ EXTRUDERS ] ;
2016-03-24 11:55:18 +01:00
2017-04-15 04:41:21 +02:00
# if HAS_WORKSPACE_OFFSET
# if HAS_POSITION_SHIFT
// The distance that XYZ has been offset by G92. Reset by G28.
float position_shift [ XYZ ] = { 0 } ;
# endif
# if HAS_HOME_OFFSET
// This offset is added to the configured home position.
// Set by M206, M428, or menu item. Saved to EEPROM.
float home_offset [ XYZ ] = { 0 } ;
# endif
# if HAS_HOME_OFFSET && HAS_POSITION_SHIFT
// The above two are combined to save on computes
float workspace_offset [ XYZ ] = { 0 } ;
# endif
2017-03-05 01:01:33 +01:00
# endif
2016-04-18 08:11:35 +02:00
2016-08-21 03:07:42 +02:00
// Software Endstops are based on the configured limits.
2017-03-16 23:20:24 +01:00
# if HAS_SOFTWARE_ENDSTOPS
2016-08-21 03:07:42 +02:00
bool soft_endstops_enabled = true ;
# endif
2017-07-22 05:54:39 +02:00
float soft_endstop_min [ XYZ ] = { X_MIN_BED , Y_MIN_BED , Z_MIN_POS } ,
soft_endstop_max [ XYZ ] = { X_MAX_BED , Y_MAX_BED , Z_MAX_POS } ;
2015-04-08 09:56:19 +02:00
2016-03-06 03:27:45 +01:00
# if FAN_COUNT > 0
2017-05-04 00:12:14 +02:00
int16_t fanSpeeds [ FAN_COUNT ] = { 0 } ;
2017-05-07 13:06:06 +02:00
# if ENABLED(PROBING_FANS_OFF)
bool fans_paused = false ;
int16_t paused_fanSpeeds [ FAN_COUNT ] = { 0 } ;
# endif
2016-03-06 03:27:45 +01:00
# endif
2016-04-18 08:13:07 +02:00
// The active extruder (tool). Set with T<extruder> command.
2015-04-04 00:45:41 +02:00
uint8_t active_extruder = 0 ;
2016-04-18 08:13:07 +02:00
// Relative Mode. Enable with G91, disable with G90.
static bool relative_mode = false ;
2016-10-07 05:06:33 +02:00
// For M109 and M190, this flag may be cleared (by M108) to exit the wait loop
2016-07-07 04:59:19 +02:00
volatile bool wait_for_heatup = true ;
2015-04-08 09:56:19 +02:00
2016-10-07 05:06:33 +02:00
// For M0/M1, this flag may be cleared (by M108) to exit the wait-for-user loop
2017-04-09 18:12:08 +02:00
# if HAS_RESUME_CONTINUE
2016-09-15 09:53:46 +02:00
volatile bool wait_for_user = false ;
2016-09-12 03:51:53 +02:00
# endif
2017-06-06 00:41:38 +02:00
const char axis_codes [ XYZE ] = { ' X ' , ' Y ' , ' Z ' , ' E ' } ;
2015-04-08 09:56:19 +02:00
2016-10-07 05:06:33 +02:00
// Number of characters read in the current line of serial input
2015-04-04 00:45:41 +02:00
static int serial_count = 0 ;
2016-04-18 08:13:07 +02:00
2015-04-04 00:45:41 +02:00
// Inactivity shutdown
2015-04-13 03:07:08 +02:00
millis_t previous_cmd_ms = 0 ;
static millis_t max_inactive_time = 0 ;
2016-04-11 00:55:12 +02:00
static millis_t stepper_inactive_time = ( DEFAULT_STEPPER_DEACTIVE_TIME ) * 1000UL ;
2016-04-18 08:13:07 +02:00
// Print Job Timer
2016-04-27 03:13:27 +02:00
# if ENABLED(PRINTCOUNTER)
PrintCounter print_job_timer = PrintCounter ( ) ;
# else
Stopwatch print_job_timer = Stopwatch ( ) ;
# endif
2016-04-18 08:13:07 +02:00
2016-08-02 06:10:53 +02:00
// Buzzer - I2C on the LCD or a BEEPER_PIN
# if ENABLED(LCD_USE_I2C_BUZZER)
# define BUZZ(d,f) lcd_buzz(d, f)
2016-10-05 04:31:50 +02:00
# elif PIN_EXISTS(BEEPER)
2016-08-02 06:10:53 +02:00
Buzzer buzzer ;
# define BUZZ(d,f) buzzer.tone(d, f)
# else
# define BUZZ(d,f) NOOP
2016-06-04 21:29:01 +02:00
# endif
2015-04-04 06:43:30 +02:00
static uint8_t target_extruder ;
2015-04-04 00:45:41 +02:00
2016-06-15 03:05:20 +02:00
# if HAS_BED_PROBE
2017-07-18 10:17:50 +02:00
float zprobe_zoffset ; // Initialized by settings.load()
2016-06-15 03:05:20 +02:00
# endif
2016-09-26 06:17:39 +02:00
# if HAS_ABL
2016-08-07 09:20:25 +02:00
float xy_probe_feedrate_mm_s = MMM_TO_MMS ( XY_PROBE_SPEED ) ;
# define XY_PROBE_FEEDRATE_MM_S xy_probe_feedrate_mm_s
2016-06-22 11:44:55 +02:00
# elif defined(XY_PROBE_SPEED)
2016-08-07 09:20:25 +02:00
# define XY_PROBE_FEEDRATE_MM_S MMM_TO_MMS(XY_PROBE_SPEED)
2016-06-22 11:22:43 +02:00
# else
2016-08-07 09:20:25 +02:00
# define XY_PROBE_FEEDRATE_MM_S PLANNER_XY_FEEDRATE()
2015-04-04 00:45:41 +02:00
# endif
2016-10-09 22:19:39 +02:00
# if ENABLED(AUTO_BED_LEVELING_BILINEAR)
2016-10-11 00:05:11 +02:00
# if ENABLED(DELTA)
# define ADJUST_DELTA(V) \
if ( planner . abl_enabled ) { \
const float zadj = bilinear_z_offset ( V ) ; \
delta [ A_AXIS ] + = zadj ; \
delta [ B_AXIS ] + = zadj ; \
delta [ C_AXIS ] + = zadj ; \
}
# else
# define ADJUST_DELTA(V) if (planner.abl_enabled) { delta[Z_AXIS] += bilinear_z_offset(V); }
# endif
2016-10-09 22:19:39 +02:00
# elif IS_KINEMATIC
# define ADJUST_DELTA(V) NOOP
# endif
2016-08-08 01:54:56 +02:00
# if ENABLED(Z_DUAL_ENDSTOPS)
2017-07-18 10:17:50 +02:00
float z_endstop_adj ;
2015-04-04 00:45:41 +02:00
# endif
2013-08-07 16:10:26 +02:00
2015-03-31 11:49:47 +02:00
// Extruder offsets
2016-05-27 02:43:20 +02:00
# if HOTENDS > 1
2017-07-18 10:17:50 +02:00
float hotend_offset [ XYZ ] [ HOTENDS ] ; // Initialized by settings.load()
2013-02-27 12:32:07 +01:00
# endif
2015-01-30 03:52:21 +01:00
2016-06-17 03:15:48 +02:00
# if HAS_Z_SERVO_ENDSTOP
2016-06-14 05:18:24 +02:00
const int z_servo_angle [ 2 ] = Z_SERVO_ANGLES ;
2013-06-06 18:18:03 +02:00
# endif
2015-01-30 03:52:21 +01:00
2015-07-31 07:24:43 +02:00
# if ENABLED(BARICUDA)
2017-07-18 05:01:20 +02:00
uint8_t baricuda_valve_pressure = 0 ,
baricuda_e_to_p_pressure = 0 ;
2013-06-07 00:49:25 +02:00
# endif
2012-11-06 12:06:41 +01:00
2017-07-18 05:01:41 +02:00
# if ENABLED(FWRETRACT) // Initialized by settings.load()...
2017-07-24 01:56:56 +02:00
bool autoretract_enabled , // M209 S - Autoretract switch
retracted [ EXTRUDERS ] = { false } ; // Which extruders are currently retracted
2017-07-18 05:01:41 +02:00
float retract_length , // M207 S - G10 Retract length
retract_feedrate_mm_s , // M207 F - G10 Retract feedrate
retract_zlift , // M207 Z - G10 Retract hop size
retract_recover_length , // M208 S - G11 Recover length
retract_recover_feedrate_mm_s , // M208 F - G11 Recover feedrate
2017-07-27 05:11:22 +02:00
swap_retract_length , // M207 W - G10 Swap Retract length
swap_retract_recover_length , // M208 W - G11 Swap Recover length
2017-07-18 05:01:41 +02:00
swap_retract_recover_feedrate_mm_s ; // M208 R - G11 Swap Recover feedrate
# if EXTRUDERS > 1
bool retracted_swap [ EXTRUDERS ] = { false } ; // Which extruders are swap-retracted
# else
constexpr bool retracted_swap [ 1 ] = { false } ;
# endif
2015-01-23 23:13:06 +01:00
# endif // FWRETRACT
2012-11-06 12:06:41 +01:00
2017-05-10 18:07:58 +02:00
# if HAS_POWER_SWITCH
bool powersupply_on =
2015-07-31 07:24:43 +02:00
# if ENABLED(PS_DEFAULT_OFF)
2015-01-30 03:52:21 +01:00
false
# else
2015-03-05 13:27:24 +01:00
true
2015-01-30 03:52:21 +01:00
# endif
;
2013-03-05 14:52:51 +01:00
# endif
2015-07-31 07:24:43 +02:00
# if ENABLED(DELTA)
2015-09-03 07:38:47 +02:00
2016-08-21 10:10:55 +02:00
float delta [ ABC ] ,
endstop_adj [ ABC ] = { 0 } ;
2017-07-18 10:17:50 +02:00
// Initialized by settings.load()
2017-03-08 00:42:04 +01:00
float delta_radius ,
2017-04-18 14:43:25 +02:00
delta_tower_angle_trim [ 2 ] ,
2017-03-08 00:42:04 +01:00
delta_tower [ ABC ] [ 2 ] ,
delta_diagonal_rod ,
2017-04-18 14:43:25 +02:00
delta_calibration_radius ,
2017-03-08 00:42:04 +01:00
delta_diagonal_rod_2_tower [ ABC ] ,
delta_segments_per_second ,
2016-08-21 10:10:55 +02:00
delta_clip_start_height = Z_MAX_POS ;
2016-07-21 23:37:10 +02:00
float delta_safe_distance_from_top ( ) ;
2016-08-21 10:10:55 +02:00
2015-04-04 00:45:41 +02:00
# endif
2014-07-01 16:45:03 +02:00
2016-09-26 06:17:39 +02:00
# if ENABLED(AUTO_BED_LEVELING_BILINEAR)
2016-12-10 08:54:09 +01:00
int bilinear_grid_spacing [ 2 ] , bilinear_start [ 2 ] ;
2017-04-21 08:34:03 +02:00
float bilinear_grid_factor [ 2 ] ,
z_values [ GRID_MAX_POINTS_X ] [ GRID_MAX_POINTS_Y ] ;
2016-09-12 03:56:16 +02:00
# endif
2016-09-12 03:11:30 +02:00
# if IS_SCARA
// Float constants for SCARA calculations
const float L1 = SCARA_LINKAGE_1 , L2 = SCARA_LINKAGE_2 ,
L1_2 = sq ( float ( L1 ) ) , L1_2_2 = 2.0 * L1_2 ,
L2_2 = sq ( float ( L2 ) ) ;
2016-08-21 10:10:55 +02:00
float delta_segments_per_second = SCARA_SEGMENTS_PER_SECOND ,
2016-09-16 21:31:53 +02:00
delta [ ABC ] ;
2015-04-04 01:46:56 +02:00
# endif
2016-09-12 10:48:29 +02:00
float cartes [ XYZ ] = { 0 } ;
2016-03-30 05:28:23 +02:00
# if ENABLED(FILAMENT_WIDTH_SENSOR)
2017-04-26 09:43:11 +02:00
bool filament_sensor = false ; // M405 turns on filament sensor control. M406 turns it off.
float filament_width_nominal = DEFAULT_NOMINAL_FILAMENT_DIA , // Nominal filament width. Change with M404.
2016-09-06 06:57:12 +02:00
filament_width_meas = DEFAULT_MEASURED_FILAMENT_DIA ; // Measured filament diameter
2017-06-23 21:48:02 +02:00
uint8_t meas_delay_cm = MEASUREMENT_DELAY_CM , // Distance delay setting
measurement_delay [ MAX_MEASUREMENT_DELAY + 1 ] ; // Ring buffer to delayed measurement. Store extruder factor after subtracting 100
int8_t filwidth_delay_index [ 2 ] = { 0 , - 1 } ; // Indexes into ring buffer
2014-08-07 02:30:57 +02:00
# endif
2015-07-31 07:24:43 +02:00
# if ENABLED(FILAMENT_RUNOUT_SENSOR)
2016-04-18 08:16:49 +02:00
static bool filament_ran_out = false ;
2015-03-07 21:43:15 +01:00
# endif
2017-05-26 20:01:02 +02:00
# if ENABLED(ADVANCED_PAUSE_FEATURE)
AdvancedPauseMenuResponse advanced_pause_menu_response ;
2016-04-28 18:57:21 +02:00
# endif
2016-06-29 00:06:56 +02:00
# if ENABLED(MIXING_EXTRUDER)
2016-11-12 21:33:07 +01:00
float mixing_factor [ MIXING_STEPPERS ] ; // Reciprocal of mix proportion. 0.0 = off, otherwise >= 1.0.
2016-06-29 00:06:56 +02:00
# if MIXING_VIRTUAL_TOOLS > 1
float mixing_virtual_tool_mix [ MIXING_VIRTUAL_TOOLS ] [ MIXING_STEPPERS ] ;
# endif
# endif
2016-02-21 02:35:35 +01:00
static bool send_ok [ BUFSIZE ] ;
2012-11-06 12:06:41 +01:00
2015-07-31 07:24:43 +02:00
# if HAS_SERVOS
2017-06-18 01:36:10 +02:00
HAL_SERVO_LIB servo [ NUM_SERVOS ] ;
2016-06-13 00:35:01 +02:00
# define MOVE_SERVO(I, P) servo[I].move(P)
2016-06-22 23:04:23 +02:00
# if HAS_Z_SERVO_ENDSTOP
# define DEPLOY_Z_SERVO() MOVE_SERVO(Z_ENDSTOP_SERVO_NR, z_servo_angle[0])
# define STOW_Z_SERVO() MOVE_SERVO(Z_ENDSTOP_SERVO_NR, z_servo_angle[1])
# endif
2013-06-07 00:49:25 +02:00
# endif
2014-03-10 21:57:08 +01:00
# ifdef CHDK
2016-04-11 00:55:12 +02:00
millis_t chdkHigh = 0 ;
2017-01-13 13:03:52 +01:00
bool chdkActive = false ;
2014-03-10 21:57:08 +01:00
# endif
2016-08-01 02:49:34 +02:00
# if ENABLED(PID_EXTRUSION_SCALING)
2015-08-31 04:04:30 +02:00
int lpq_len = 20 ;
# endif
2016-03-07 12:48:14 +01:00
# if ENABLED(HOST_KEEPALIVE_FEATURE)
2017-03-31 08:28:07 +02:00
MarlinBusyState busy_state = NOT_BUSY ;
2016-04-11 00:55:12 +02:00
static millis_t next_busy_signal_ms = 0 ;
2016-04-07 02:31:18 +02:00
uint8_t host_keepalive_interval = DEFAULT_KEEPALIVE_INTERVAL ;
2016-03-07 12:48:14 +01:00
# else
2017-03-29 02:45:54 +02:00
# define host_keepalive() NOOP
# endif
2016-03-07 12:48:14 +01:00
2017-06-09 14:06:23 +02:00
# if ENABLED(I2C_POSITION_ENCODERS)
I2CPositionEncodersMgr I2CPEM ;
uint8_t blockBufferIndexRef = 0 ;
millis_t lastUpdateMillis ;
# endif
2017-06-23 21:28:28 +02:00
# if ENABLED(CNC_WORKSPACE_PLANES)
static WorkspacePlane workspace_plane = PLANE_XY ;
# endif
2017-06-06 01:32:14 +02:00
FORCE_INLINE float pgm_read_any ( const float * p ) { return pgm_read_float_near ( p ) ; }
FORCE_INLINE signed char pgm_read_any ( const signed char * p ) { return pgm_read_byte_near ( p ) ; }
2016-09-06 02:56:03 +02:00
# define XYZ_CONSTS_FROM_CONFIG(type, array, CONFIG) \
2016-10-24 19:03:41 +02:00
static const PROGMEM type array # # _P [ XYZ ] = { X_ # # CONFIG , Y_ # # CONFIG , Z_ # # CONFIG } ; \
2017-04-24 20:16:54 +02:00
static inline type array ( AxisEnum axis ) { return pgm_read_any ( & array # # _P [ axis ] ) ; } \
typedef void __void_ # # CONFIG # # __
2016-09-06 02:56:03 +02:00
2017-04-24 20:16:54 +02:00
XYZ_CONSTS_FROM_CONFIG ( float , base_min_pos , MIN_POS ) ;
XYZ_CONSTS_FROM_CONFIG ( float , base_max_pos , MAX_POS ) ;
XYZ_CONSTS_FROM_CONFIG ( float , base_home_pos , HOME_POS ) ;
XYZ_CONSTS_FROM_CONFIG ( float , max_length , MAX_LENGTH ) ;
XYZ_CONSTS_FROM_CONFIG ( float , home_bump_mm , HOME_BUMP_MM ) ;
XYZ_CONSTS_FROM_CONFIG ( signed char , home_dir , HOME_DIR ) ;
2016-09-06 02:56:03 +02:00
2016-03-27 05:36:36 +02:00
/**
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * FUNCTIONS * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
*/
2012-11-06 12:06:41 +01:00
2016-04-18 09:05:22 +02:00
void stop ( ) ;
2016-03-28 14:25:27 +02:00
void get_available_commands ( ) ;
2015-05-17 05:47:40 +02:00
void process_next_command ( ) ;
2016-06-11 04:23:41 +02:00
void prepare_move_to_destination ( ) ;
2016-09-12 10:48:29 +02:00
void get_cartesian_from_steppers ( ) ;
2016-09-15 06:50:17 +02:00
void set_current_from_steppers_for_axis ( const AxisEnum axis ) ;
2015-05-17 05:47:40 +02:00
2016-05-12 22:33:47 +02:00
# if ENABLED(BEZIER_CURVE_SUPPORT)
void plan_cubic_move ( const float offset [ 4 ] ) ;
# endif
2017-06-04 00:11:43 +02:00
void report_current_position ( ) ;
2016-03-23 13:30:50 +01:00
2016-07-24 22:27:09 +02:00
/**
* sync_plan_position
2016-09-15 07:38:02 +02:00
*
* Set the planner / stepper positions directly from current_position with
* no kinematic translation . Used for homing axes and cartesian / core syncing .
2016-07-24 22:27:09 +02:00
*/
2017-06-13 01:26:49 +02:00
void sync_plan_position ( ) {
2016-07-24 22:27:09 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) DEBUG_POS ( " sync_plan_position " , current_position ) ;
# endif
planner . set_position_mm ( current_position [ X_AXIS ] , current_position [ Y_AXIS ] , current_position [ Z_AXIS ] , current_position [ E_AXIS ] ) ;
}
inline void sync_plan_position_e ( ) { planner . set_e_position_mm ( current_position [ E_AXIS ] ) ; }
2016-09-12 03:11:30 +02:00
# if IS_KINEMATIC
2016-09-15 08:21:31 +02:00
2016-09-12 04:30:59 +02:00
inline void sync_plan_position_kinematic ( ) {
2016-04-12 03:25:39 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
2016-09-12 04:30:59 +02:00
if ( DEBUGGING ( LEVELING ) ) DEBUG_POS ( " sync_plan_position_kinematic " , current_position ) ;
2016-04-12 03:25:39 +02:00
# endif
2016-10-06 10:56:05 +02:00
planner . set_position_mm_kinematic ( current_position ) ;
2016-04-12 03:25:39 +02:00
}
2016-09-12 04:30:59 +02:00
# define SYNC_PLAN_POSITION_KINEMATIC() sync_plan_position_kinematic()
2016-09-15 08:21:31 +02:00
2016-06-22 03:56:25 +02:00
# else
2016-09-15 08:21:31 +02:00
2016-06-22 03:56:25 +02:00
# define SYNC_PLAN_POSITION_KINEMATIC() sync_plan_position()
2016-09-15 08:21:31 +02:00
2016-04-12 03:25:39 +02:00
# endif
2016-05-18 03:22:41 +02:00
# if ENABLED(DIGIPOT_I2C)
2017-06-06 13:10:10 +02:00
extern void digipot_i2c_set_current ( uint8_t channel , float current ) ;
2016-05-18 03:22:41 +02:00
extern void digipot_i2c_init ( ) ;
# endif
2015-04-14 02:17:36 +02:00
/**
2017-03-20 03:31:06 +01:00
* Inject the next " immediate " command , when possible , onto the front of the queue .
2016-03-19 01:37:31 +01:00
* Return true if any immediate commands remain to inject .
2015-04-14 02:17:36 +02:00
*/
2016-10-05 04:32:37 +02:00
static bool drain_injected_commands_P ( ) {
if ( injected_commands_P ! = NULL ) {
2016-03-19 01:37:31 +01:00
size_t i = 0 ;
2016-03-20 04:05:23 +01:00
char c , cmd [ 30 ] ;
2016-10-05 04:32:37 +02:00
strncpy_P ( cmd , injected_commands_P , sizeof ( cmd ) - 1 ) ;
2016-03-20 04:05:23 +01:00
cmd [ sizeof ( cmd ) - 1 ] = ' \0 ' ;
while ( ( c = cmd [ i ] ) & & c ! = ' \n ' ) i + + ; // find the end of this gcode command
cmd [ i ] = ' \0 ' ;
2017-04-02 04:32:36 +02:00
if ( enqueue_and_echo_command ( cmd ) ) // success?
2016-10-24 19:03:41 +02:00
injected_commands_P = c ? injected_commands_P + i + 1 : NULL ; // next command or done
2012-11-06 12:06:41 +01:00
}
2017-03-20 03:31:06 +01:00
return ( injected_commands_P ! = NULL ) ; // return whether any more remain
2012-11-06 12:06:41 +01:00
}
2015-04-14 02:17:36 +02:00
/**
* Record one or many commands to run from program memory .
* Aborts the current queue , if any .
2016-10-05 04:32:37 +02:00
* Note : drain_injected_commands_P ( ) must be called repeatedly to drain the commands afterwards
2015-04-14 02:17:36 +02:00
*/
2017-05-28 02:29:29 +02:00
void enqueue_and_echo_commands_P ( const char * const pgcode ) {
2016-10-05 04:32:37 +02:00
injected_commands_P = pgcode ;
drain_injected_commands_P ( ) ; // first command executed asap (when possible)
2012-11-28 10:30:34 +01:00
}
2017-03-20 03:31:06 +01:00
/**
* Clear the Marlin command queue
*/
2016-06-01 02:41:32 +02:00
void clear_command_queue ( ) {
cmd_queue_index_r = cmd_queue_index_w ;
commands_in_queue = 0 ;
}
2015-04-14 02:17:36 +02:00
/**
2016-02-21 02:35:35 +01:00
* Once a new command is in the ring buffer , call this to commit it
2015-04-14 02:17:36 +02:00
*/
2016-02-21 02:35:35 +01:00
inline void _commit_command ( bool say_ok ) {
send_ok [ cmd_queue_index_w ] = say_ok ;
2017-05-05 09:57:22 +02:00
if ( + + cmd_queue_index_w > = BUFSIZE ) cmd_queue_index_w = 0 ;
2015-04-14 02:17:36 +02:00
commands_in_queue + + ;
2016-02-21 02:35:35 +01:00
}
/**
2017-03-20 03:31:06 +01:00
* Copy a command from RAM into the main command buffer .
* Return true if the command was successfully added .
* Return false for a full buffer , or if the ' command ' is a comment .
2016-02-21 02:35:35 +01:00
*/
inline bool _enqueuecommand ( const char * cmd , bool say_ok = false ) {
if ( * cmd = = ' ; ' | | commands_in_queue > = BUFSIZE ) return false ;
strcpy ( command_queue [ cmd_queue_index_w ] , cmd ) ;
_commit_command ( say_ok ) ;
2015-02-07 23:24:20 +01:00
return true ;
}
2016-02-21 02:35:35 +01:00
/**
* Enqueue with Serial Echo
*/
bool enqueue_and_echo_command ( const char * cmd , bool say_ok /*=false*/ ) {
if ( _enqueuecommand ( cmd , say_ok ) ) {
2017-06-09 17:51:23 +02:00
SERIAL_ECHO_START ( ) ;
2017-04-02 15:36:58 +02:00
SERIAL_ECHOPAIR ( MSG_ENQUEUEING , cmd ) ;
SERIAL_CHAR ( ' " ' ) ;
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2016-02-21 02:35:35 +01:00
return true ;
}
return false ;
}
2015-04-14 02:17:36 +02:00
void setup_killpin ( ) {
2015-04-04 00:31:35 +02:00
# if HAS_KILL
2017-03-08 06:43:33 +01:00
SET_INPUT_PULLUP ( KILL_PIN ) ;
2012-11-06 12:06:41 +01:00
# endif
}
2013-06-07 00:49:25 +02:00
2016-06-20 03:05:57 +02:00
# if ENABLED(FILAMENT_RUNOUT_SENSOR)
void setup_filrunoutpin ( ) {
2015-07-31 07:24:43 +02:00
# if ENABLED(ENDSTOPPULLUP_FIL_RUNOUT)
2017-03-08 06:43:33 +01:00
SET_INPUT_PULLUP ( FIL_RUNOUT_PIN ) ;
# else
SET_INPUT ( FIL_RUNOUT_PIN ) ;
2015-04-04 00:31:35 +02:00
# endif
2016-06-20 03:05:57 +02:00
}
# endif
2015-03-07 21:43:15 +01:00
2015-04-14 02:17:36 +02:00
void setup_powerhold ( ) {
2015-04-04 00:31:35 +02:00
# if HAS_SUICIDE
2015-03-03 09:48:20 +01:00
OUT_WRITE ( SUICIDE_PIN , HIGH ) ;
2013-06-07 00:49:25 +02:00
# endif
2015-03-31 01:39:47 +02:00
# if HAS_POWER_SWITCH
2015-07-31 07:24:43 +02:00
# if ENABLED(PS_DEFAULT_OFF)
2015-03-03 09:48:20 +01:00
OUT_WRITE ( PS_ON_PIN , PS_ON_ASLEEP ) ;
2015-03-05 13:27:24 +01:00
# else
2015-03-03 09:48:20 +01:00
OUT_WRITE ( PS_ON_PIN , PS_ON_AWAKE ) ;
# endif
2013-06-07 00:49:25 +02:00
# endif
2012-11-06 12:06:41 +01:00
}
2015-04-14 02:17:36 +02:00
void suicide ( ) {
2015-04-04 00:31:35 +02:00
# if HAS_SUICIDE
2015-03-03 09:48:20 +01:00
OUT_WRITE ( SUICIDE_PIN , LOW ) ;
2013-06-07 00:49:25 +02:00
# endif
}
2015-04-14 02:17:36 +02:00
void servo_init ( ) {
2015-04-04 00:31:35 +02:00
# if NUM_SERVOS >= 1 && HAS_SERVO_0
2015-04-14 12:13:25 +02:00
servo [ 0 ] . attach ( SERVO0_PIN ) ;
2015-07-24 11:38:15 +02:00
servo [ 0 ] . detach ( ) ; // Just set up the pin. We don't have a position yet. Don't move to a random position.
2013-06-07 00:49:25 +02:00
# endif
2015-04-04 00:31:35 +02:00
# if NUM_SERVOS >= 2 && HAS_SERVO_1
2015-04-14 12:13:25 +02:00
servo [ 1 ] . attach ( SERVO1_PIN ) ;
2015-07-24 11:38:15 +02:00
servo [ 1 ] . detach ( ) ;
2013-06-07 00:49:25 +02:00
# endif
2015-04-04 00:31:35 +02:00
# if NUM_SERVOS >= 3 && HAS_SERVO_2
2015-04-14 12:13:25 +02:00
servo [ 2 ] . attach ( SERVO2_PIN ) ;
2015-07-24 11:38:15 +02:00
servo [ 2 ] . detach ( ) ;
2013-06-07 00:49:25 +02:00
# endif
2015-04-04 00:31:35 +02:00
# if NUM_SERVOS >= 4 && HAS_SERVO_3
2015-04-14 12:13:25 +02:00
servo [ 3 ] . attach ( SERVO3_PIN ) ;
2015-07-24 11:38:15 +02:00
servo [ 3 ] . detach ( ) ;
2013-06-07 00:49:25 +02:00
# endif
2013-06-06 16:36:52 +02:00
2016-06-17 03:15:48 +02:00
# if HAS_Z_SERVO_ENDSTOP
2015-12-03 14:19:29 +01:00
/**
2016-06-14 05:18:24 +02:00
* Set position of Z Servo Endstop
2015-12-03 14:19:29 +01:00
*
* The servo might be deployed and positioned too low to stow
* when starting up the machine or rebooting the board .
* There ' s no way to know where the nozzle is positioned until
* homing has been done - no homing with z - probe without init !
*
*/
2016-06-14 05:18:24 +02:00
STOW_Z_SERVO ( ) ;
2016-06-13 00:15:37 +02:00
# endif
2012-11-06 12:06:41 +01:00
}
2015-07-11 03:15:24 +02:00
/**
* Stepper Reset ( RigidBoard , et . al . )
*/
# if HAS_STEPPER_RESET
void disableStepperDrivers ( ) {
2016-09-25 12:52:10 +02:00
OUT_WRITE ( STEPPER_RESET_PIN , LOW ) ; // drive it down to hold in reset motor driver chips
2015-07-11 03:15:24 +02:00
}
2016-09-25 12:52:10 +02:00
void enableStepperDrivers ( ) { SET_INPUT ( STEPPER_RESET_PIN ) ; } // set to input, which allows it to be pulled high by pullups
2015-07-11 03:15:24 +02:00
# endif
2016-08-11 08:04:18 +02:00
# if ENABLED(EXPERIMENTAL_I2CBUS) && I2C_SLAVE_ADDRESS > 0
2016-08-12 02:57:22 +02:00
void i2c_on_receive ( int bytes ) { // just echo all bytes received to serial
i2c . receive ( bytes ) ;
}
void i2c_on_request ( ) { // just send dummy data for now
2016-08-14 03:06:10 +02:00
i2c . reply ( " Hello World! \n " ) ;
2016-08-11 08:04:18 +02:00
}
# endif
2017-04-04 23:26:30 +02:00
# if HAS_COLOR_LEDS
2017-07-13 17:01:21 +02:00
# if ENABLED(NEOPIXEL_RGBW_LED)
Adafruit_NeoPixel pixels ( NEOPIXEL_PIXELS , NEOPIXEL_PIN , NEO_GRBW + NEO_KHZ800 ) ;
void set_neopixel_color ( const uint32_t color ) {
for ( uint16_t i = 0 ; i < pixels . numPixels ( ) ; + + i )
pixels . setPixelColor ( i , color ) ;
pixels . show ( ) ;
}
void setup_neopixel ( ) {
pixels . setBrightness ( 255 ) ; // 0 - 255 range
pixels . begin ( ) ;
pixels . show ( ) ; // initialize to all off
# if ENABLED(NEOPIXEL_STARTUP_TEST)
delay ( 2000 ) ;
set_neopixel_color ( pixels . Color ( 255 , 0 , 0 , 0 ) ) ; // red
delay ( 2000 ) ;
set_neopixel_color ( pixels . Color ( 0 , 255 , 0 , 0 ) ) ; // green
delay ( 2000 ) ;
set_neopixel_color ( pixels . Color ( 0 , 0 , 255 , 0 ) ) ; // blue
delay ( 2000 ) ;
# endif
set_neopixel_color ( pixels . Color ( 0 , 0 , 0 , 255 ) ) ; // white
}
# endif // NEOPIXEL_RGBW_LED
2017-04-11 19:58:55 +02:00
void set_led_color (
const uint8_t r , const uint8_t g , const uint8_t b
2017-07-13 17:01:21 +02:00
# if ENABLED(RGBW_LED) || ENABLED(NEOPIXEL_RGBW_LED)
, const uint8_t w = 0
# if ENABLED(NEOPIXEL_RGBW_LED)
, bool isSequence = false
# endif
2017-04-11 19:58:55 +02:00
# endif
) {
2017-04-04 23:26:30 +02:00
2017-07-13 17:01:21 +02:00
# if ENABLED(NEOPIXEL_RGBW_LED)
const uint32_t color = pixels . Color ( r , g , b , w ) ;
2017-07-28 09:19:50 +02:00
static uint16_t nextLed = 0 ;
2017-07-13 17:01:21 +02:00
if ( ! isSequence )
set_neopixel_color ( color ) ;
else {
pixels . setPixelColor ( nextLed , color ) ;
pixels . show ( ) ;
if ( + + nextLed > = pixels . numPixels ( ) ) nextLed = 0 ;
return ;
}
# endif
2017-04-04 23:26:30 +02:00
# if ENABLED(BLINKM)
// This variant uses i2c to send the RGB components to the device.
SendColors ( r , g , b ) ;
2017-06-10 08:02:15 +02:00
# endif
# if ENABLED(RGB_LED) || ENABLED(RGBW_LED)
2017-04-04 23:26:30 +02:00
// This variant uses 3 separate pins for the RGB components.
// If the pins can do PWM then their intensity will be set.
2017-04-14 23:36:12 +02:00
WRITE ( RGB_LED_R_PIN , r ? HIGH : LOW ) ;
WRITE ( RGB_LED_G_PIN , g ? HIGH : LOW ) ;
WRITE ( RGB_LED_B_PIN , b ? HIGH : LOW ) ;
2017-04-04 23:26:30 +02:00
analogWrite ( RGB_LED_R_PIN , r ) ;
analogWrite ( RGB_LED_G_PIN , g ) ;
analogWrite ( RGB_LED_B_PIN , b ) ;
2017-04-11 19:58:55 +02:00
# if ENABLED(RGBW_LED)
2017-04-14 23:36:12 +02:00
WRITE ( RGB_LED_W_PIN , w ? HIGH : LOW ) ;
2017-04-11 19:58:55 +02:00
analogWrite ( RGB_LED_W_PIN , w ) ;
# endif
2017-04-04 23:26:30 +02:00
# endif
2017-06-10 08:02:15 +02:00
# if ENABLED(PCA9632)
// Update I2C LED driver
PCA9632_SetColor ( r , g , b ) ;
# endif
2017-04-04 23:26:30 +02:00
}
# endif // HAS_COLOR_LEDS
2015-10-03 08:08:58 +02:00
void gcode_line_error ( const char * err , bool doFlush = true ) {
2017-06-09 17:51:23 +02:00
SERIAL_ERROR_START ( ) ;
2015-05-17 05:54:58 +02:00
serialprintPGM ( err ) ;
SERIAL_ERRORLN ( gcode_LastN ) ;
//Serial.println(gcode_N);
if ( doFlush ) FlushSerialRequestResend ( ) ;
serial_count = 0 ;
}
2017-03-20 03:31:06 +01:00
/**
* Get all commands waiting on the serial port and queue them .
* Exit when the buffer is full or when no more characters are
* left on the serial port .
*/
2016-03-28 14:25:27 +02:00
inline void get_serial_commands ( ) {
2016-02-21 02:35:35 +01:00
static char serial_line_buffer [ MAX_CMD_SIZE ] ;
2017-01-13 13:03:52 +01:00
static bool serial_comment_mode = false ;
2016-02-21 02:35:35 +01:00
2016-03-28 14:25:27 +02:00
// If the command buffer is empty for too long,
// send "wait" to indicate Marlin is still waiting.
2016-02-21 02:35:35 +01:00
# if defined(NO_TIMEOUTS) && NO_TIMEOUTS > 0
2015-04-16 21:42:54 +02:00
static millis_t last_command_time = 0 ;
2017-03-29 02:45:54 +02:00
const millis_t ms = millis ( ) ;
2016-04-11 00:55:12 +02:00
if ( commands_in_queue = = 0 & & ! MYSERIAL . available ( ) & & ELAPSED ( ms , last_command_time + NO_TIMEOUTS ) ) {
2015-04-16 21:42:54 +02:00
SERIAL_ECHOLNPGM ( MSG_WAIT ) ;
last_command_time = ms ;
}
# endif
2015-05-17 05:59:04 +02:00
2016-03-27 05:36:36 +02:00
/**
* Loop while serial characters are incoming and the queue is not full
*/
2015-05-17 05:59:04 +02:00
while ( commands_in_queue < BUFSIZE & & MYSERIAL . available ( ) > 0 ) {
2016-02-21 02:35:35 +01:00
char serial_char = MYSERIAL . read ( ) ;
2015-04-14 02:17:36 +02:00
2016-03-27 05:36:36 +02:00
/**
* If the character ends the line
*/
2016-02-21 02:35:35 +01:00
if ( serial_char = = ' \n ' | | serial_char = = ' \r ' ) {
2015-05-17 05:59:04 +02:00
2016-02-21 02:35:35 +01:00
serial_comment_mode = false ; // end of line == end of comment
2015-03-05 23:30:34 +01:00
2016-03-28 13:42:51 +02:00
if ( ! serial_count ) continue ; // skip empty lines
2015-04-13 03:07:08 +02:00
2016-02-21 02:35:35 +01:00
serial_line_buffer [ serial_count ] = 0 ; // terminate string
serial_count = 0 ; //reset buffer
2015-04-13 03:07:08 +02:00
2016-02-21 02:35:35 +01:00
char * command = serial_line_buffer ;
2015-04-13 03:07:08 +02:00
2015-10-13 12:58:11 +02:00
while ( * command = = ' ' ) command + + ; // skip any leading spaces
2017-05-20 10:03:08 +02:00
char * npos = ( * command = = ' N ' ) ? command : NULL , // Require the N parameter to start the line
* apos = strchr ( command , ' * ' ) ;
2015-10-13 12:58:11 +02:00
2015-05-17 14:00:09 +02:00
if ( npos ) {
2015-06-24 02:50:09 +02:00
2017-01-13 13:03:52 +01:00
bool M110 = strstr_P ( command , PSTR ( " M110 " ) ) ! = NULL ;
2015-06-24 02:50:09 +02:00
if ( M110 ) {
2015-10-03 08:08:58 +02:00
char * n2pos = strchr ( command + 4 , ' N ' ) ;
2015-06-24 02:50:09 +02:00
if ( n2pos ) npos = n2pos ;
}
2015-05-17 14:00:09 +02:00
gcode_N = strtol ( npos + 1 , NULL , 10 ) ;
2015-06-24 02:50:09 +02:00
2015-06-24 02:57:38 +02:00
if ( gcode_N ! = gcode_LastN + 1 & & ! M110 ) {
2015-05-17 05:54:58 +02:00
gcode_line_error ( PSTR ( MSG_ERR_LINE_NO ) ) ;
2015-03-05 23:30:34 +01:00
return ;
}
2015-05-17 14:00:09 +02:00
if ( apos ) {
byte checksum = 0 , count = 0 ;
2015-04-14 02:17:36 +02:00
while ( command [ count ] ! = ' * ' ) checksum ^ = command [ count + + ] ;
2012-11-06 12:06:41 +01:00
2015-05-17 14:00:09 +02:00
if ( strtol ( apos + 1 , NULL , 10 ) ! = checksum ) {
2015-05-17 05:54:58 +02:00
gcode_line_error ( PSTR ( MSG_ERR_CHECKSUM_MISMATCH ) ) ;
2012-11-06 12:06:41 +01:00
return ;
}
2015-05-17 05:54:58 +02:00
// if no errors, continue parsing
2012-11-06 12:06:41 +01:00
}
2016-03-08 11:25:11 +01:00
else {
2015-05-17 05:54:58 +02:00
gcode_line_error ( PSTR ( MSG_ERR_NO_CHECKSUM ) ) ;
2015-03-05 23:30:34 +01:00
return ;
2012-11-06 12:06:41 +01:00
}
2015-03-05 23:30:34 +01:00
gcode_LastN = gcode_N ;
2015-05-17 05:54:58 +02:00
// if no errors, continue parsing
2015-03-05 23:30:34 +01:00
}
2015-05-17 14:00:09 +02:00
else if ( apos ) { // No '*' without 'N'
gcode_line_error ( PSTR ( MSG_ERR_NO_LINENUMBER_WITH_CHECKSUM ) , false ) ;
return ;
2015-03-05 23:30:34 +01:00
}
2015-04-13 03:07:08 +02:00
2015-05-17 14:00:09 +02:00
// Movement commands alert when stopped
if ( IsStopped ( ) ) {
2015-10-03 08:08:58 +02:00
char * gpos = strchr ( command , ' G ' ) ;
2015-05-17 14:00:09 +02:00
if ( gpos ) {
2017-04-18 21:39:45 +02:00
const int codenum = strtol ( gpos + 1 , NULL , 10 ) ;
2015-05-17 14:00:09 +02:00
switch ( codenum ) {
case 0 :
case 1 :
case 2 :
case 3 :
2015-04-13 03:07:08 +02:00
SERIAL_ERRORLNPGM ( MSG_ERR_STOPPED ) ;
LCD_MESSAGEPGM ( MSG_STOPPED ) ;
2015-05-17 14:00:09 +02:00
break ;
}
2015-08-05 13:40:36 +02:00
}
2012-11-06 12:06:41 +01:00
}
2015-03-05 23:30:34 +01:00
Add an emergency-command parser to MarlinSerial (supporting M108)
Add an emergency-command parser to MarlinSerial's RX interrupt.
The parser tries to find and execute M108,M112,M410 before the commands disappear in the RX-buffer.
To avoid false positives for M117, comments and commands followed by filenames (M23, M28, M30, M32, M33) are filtered.
This enables Marlin to receive and react on the Emergency command at all times - regardless of whether the buffers are full or not. It remains to convince hosts to send the commands. To inform the hosts about the new feature a new entry in the M115-report was made. "`EMERGENCY_CODES:M112,M108,M410;`".
The parser is fast. It only ever needs two switch decisions and one assignment of the new state for every character.
One problem remains. If the host has sent an incomplete line before sending an emergency command the emergency command could be omitted when the parser is in `state_IGNORE`.
In that case the host should send "\ncommand\n"
Also introduces M108 to break the waiting for the heaters in M109, M190 and M303.
Rename `cancel_heatup` to `wait_for_heatup` to better see the purpose.
2016-07-04 23:23:22 +02:00
# if DISABLED(EMERGENCY_PARSER)
// If command was e-stop process now
2016-10-26 08:42:52 +02:00
if ( strcmp ( command , " M108 " ) = = 0 ) {
wait_for_heatup = false ;
# if ENABLED(ULTIPANEL)
wait_for_user = false ;
# endif
}
Add an emergency-command parser to MarlinSerial (supporting M108)
Add an emergency-command parser to MarlinSerial's RX interrupt.
The parser tries to find and execute M108,M112,M410 before the commands disappear in the RX-buffer.
To avoid false positives for M117, comments and commands followed by filenames (M23, M28, M30, M32, M33) are filtered.
This enables Marlin to receive and react on the Emergency command at all times - regardless of whether the buffers are full or not. It remains to convince hosts to send the commands. To inform the hosts about the new feature a new entry in the M115-report was made. "`EMERGENCY_CODES:M112,M108,M410;`".
The parser is fast. It only ever needs two switch decisions and one assignment of the new state for every character.
One problem remains. If the host has sent an incomplete line before sending an emergency command the emergency command could be omitted when the parser is in `state_IGNORE`.
In that case the host should send "\ncommand\n"
Also introduces M108 to break the waiting for the heaters in M109, M190 and M303.
Rename `cancel_heatup` to `wait_for_heatup` to better see the purpose.
2016-07-04 23:23:22 +02:00
if ( strcmp ( command , " M112 " ) = = 0 ) kill ( PSTR ( MSG_KILLED ) ) ;
2016-07-06 21:00:28 +02:00
if ( strcmp ( command , " M410 " ) = = 0 ) { quickstop_stepper ( ) ; }
Add an emergency-command parser to MarlinSerial (supporting M108)
Add an emergency-command parser to MarlinSerial's RX interrupt.
The parser tries to find and execute M108,M112,M410 before the commands disappear in the RX-buffer.
To avoid false positives for M117, comments and commands followed by filenames (M23, M28, M30, M32, M33) are filtered.
This enables Marlin to receive and react on the Emergency command at all times - regardless of whether the buffers are full or not. It remains to convince hosts to send the commands. To inform the hosts about the new feature a new entry in the M115-report was made. "`EMERGENCY_CODES:M112,M108,M410;`".
The parser is fast. It only ever needs two switch decisions and one assignment of the new state for every character.
One problem remains. If the host has sent an incomplete line before sending an emergency command the emergency command could be omitted when the parser is in `state_IGNORE`.
In that case the host should send "\ncommand\n"
Also introduces M108 to break the waiting for the heaters in M109, M190 and M303.
Rename `cancel_heatup` to `wait_for_heatup` to better see the purpose.
2016-07-04 23:23:22 +02:00
# endif
2015-03-05 23:30:34 +01:00
2016-02-21 02:35:35 +01:00
# if defined(NO_TIMEOUTS) && NO_TIMEOUTS > 0
last_command_time = ms ;
# endif
2015-03-05 23:30:34 +01:00
2016-02-21 02:35:35 +01:00
// Add the command to the queue
_enqueuecommand ( serial_line_buffer , true ) ;
}
else if ( serial_count > = MAX_CMD_SIZE - 1 ) {
// Keep fetching, but ignore normal characters beyond the max length
// The command will be injected when EOL is reached
2012-11-06 12:06:41 +01:00
}
2015-04-13 03:07:08 +02:00
else if ( serial_char = = ' \\ ' ) { // Handle escapes
2016-02-21 02:35:35 +01:00
if ( MYSERIAL . available ( ) > 0 ) {
2015-04-13 03:07:08 +02:00
// if we have one more character, copy it over
serial_char = MYSERIAL . read ( ) ;
2016-03-28 13:36:14 +02:00
if ( ! serial_comment_mode ) serial_line_buffer [ serial_count + + ] = serial_char ;
2015-04-13 03:07:08 +02:00
}
// otherwise do nothing
2015-03-05 21:22:37 +01:00
}
2016-02-21 02:35:35 +01:00
else { // it's not a newline, carriage return or escape char
if ( serial_char = = ' ; ' ) serial_comment_mode = true ;
if ( ! serial_comment_mode ) serial_line_buffer [ serial_count + + ] = serial_char ;
2012-11-06 12:06:41 +01:00
}
2016-02-21 02:35:35 +01:00
} // queue has space, serial has data
2016-03-28 14:25:27 +02:00
}
2013-12-08 21:35:57 +01:00
2016-03-28 14:25:27 +02:00
# if ENABLED(SDSUPPORT)
2013-06-07 00:49:25 +02:00
2017-03-20 03:31:06 +01:00
/**
* Get commands from the SD Card until the command buffer is full
* or until the end of the file is reached . The special character ' # '
* can also interrupt buffering .
*/
2016-03-28 14:25:27 +02:00
inline void get_sdcard_commands ( ) {
2016-02-21 02:35:35 +01:00
static bool stop_buffering = false ,
sd_comment_mode = false ;
2017-09-06 13:28:33 +02:00
if ( ! IS_SD_PRINTING ) return ;
2015-04-13 03:07:08 +02:00
2016-03-27 05:36:36 +02:00
/**
* ' # ' stops reading from SD to the buffer prematurely , so procedural
* macro calls are possible . If it occurs , stop_buffering is triggered
* and the buffer is run dry ; this character _can_ occur in serial com
* due to checksums , however , no checksums are used in SD printing .
*/
2015-04-13 03:07:08 +02:00
2015-04-14 02:17:36 +02:00
if ( commands_in_queue = = 0 ) stop_buffering = false ;
2015-04-13 03:07:08 +02:00
2016-02-21 02:35:35 +01:00
uint16_t sd_count = 0 ;
bool card_eof = card . eof ( ) ;
while ( commands_in_queue < BUFSIZE & & ! card_eof & & ! stop_buffering ) {
2017-03-20 03:31:06 +01:00
const int16_t n = card . get ( ) ;
2016-02-21 02:35:35 +01:00
char sd_char = ( char ) n ;
card_eof = card . eof ( ) ;
if ( card_eof | | n = = - 1
| | sd_char = = ' \n ' | | sd_char = = ' \r '
| | ( ( sd_char = = ' # ' | | sd_char = = ' : ' ) & & ! sd_comment_mode )
2015-04-13 03:07:08 +02:00
) {
2016-02-21 02:35:35 +01:00
if ( card_eof ) {
2015-04-13 03:07:08 +02:00
SERIAL_PROTOCOLLNPGM ( MSG_FILE_PRINTED ) ;
card . printingHasFinished ( ) ;
2017-04-04 23:26:30 +02:00
# if ENABLED(PRINTER_EVENT_LEDS)
LCD_MESSAGEPGM ( MSG_INFO_COMPLETED_PRINTS ) ;
2017-04-11 19:58:55 +02:00
set_led_color ( 0 , 255 , 0 ) ; // Green
2017-04-04 23:26:30 +02:00
# if HAS_RESUME_CONTINUE
2017-05-04 05:14:40 +02:00
enqueue_and_echo_commands_P ( PSTR ( " M0 " ) ) ; // end of the queue!
2017-04-04 23:26:30 +02:00
# else
safe_delay ( 1000 ) ;
# endif
2017-04-11 19:58:55 +02:00
set_led_color ( 0 , 0 , 0 ) ; // OFF
2017-04-04 23:26:30 +02:00
# endif
2015-04-13 03:07:08 +02:00
card . checkautostart ( true ) ;
}
2016-06-20 02:14:56 +02:00
else if ( n = = - 1 ) {
2017-06-09 17:51:23 +02:00
SERIAL_ERROR_START ( ) ;
2016-06-20 02:14:56 +02:00
SERIAL_ECHOLNPGM ( MSG_SD_ERR_READ ) ;
}
2016-02-21 02:35:35 +01:00
if ( sd_char = = ' # ' ) stop_buffering = true ;
2013-12-08 21:35:57 +01:00
2017-03-20 03:31:06 +01:00
sd_comment_mode = false ; // for new command
2016-02-21 02:35:35 +01:00
2017-03-20 03:31:06 +01:00
if ( ! sd_count ) continue ; // skip empty lines (and comment lines)
2016-02-21 02:35:35 +01:00
2017-03-20 03:31:06 +01:00
command_queue [ cmd_queue_index_w ] [ sd_count ] = ' \0 ' ; // terminate string
sd_count = 0 ; // clear sd line buffer
2016-02-21 02:35:35 +01:00
_commit_command ( false ) ;
}
else if ( sd_count > = MAX_CMD_SIZE - 1 ) {
2016-03-27 05:36:36 +02:00
/**
* Keep fetching , but ignore normal characters beyond the max length
* The command will be injected when EOL is reached
*/
2015-04-13 03:07:08 +02:00
}
else {
2016-02-21 02:35:35 +01:00
if ( sd_char = = ' ; ' ) sd_comment_mode = true ;
if ( ! sd_comment_mode ) command_queue [ cmd_queue_index_w ] [ sd_count + + ] = sd_char ;
2015-04-13 03:07:08 +02:00
}
2012-11-06 12:06:41 +01:00
}
2016-03-28 14:25:27 +02:00
}
2012-11-06 12:06:41 +01:00
2016-03-28 14:25:27 +02:00
# endif // SDSUPPORT
/**
* Add to the circular command queue the next command from :
2016-10-05 04:32:37 +02:00
* - The command - injection queue ( injected_commands_P )
2016-03-28 14:25:27 +02:00
* - The active serial input ( usually USB )
* - The SD card file being actively printed
*/
void get_available_commands ( ) {
// if any immediate commands remain, don't get other commands yet
2016-10-05 04:32:37 +02:00
if ( drain_injected_commands_P ( ) ) return ;
2016-03-28 14:25:27 +02:00
get_serial_commands ( ) ;
# if ENABLED(SDSUPPORT)
get_sdcard_commands ( ) ;
# endif
2012-11-06 12:06:41 +01:00
}
2016-04-18 09:01:28 +02:00
/**
* Set target_extruder from the T parameter or the active_extruder
*
* Returns TRUE if the target is invalid
*/
2017-06-27 09:36:19 +02:00
bool get_target_extruder_from_command ( const uint16_t code ) {
2017-06-27 06:31:45 +02:00
if ( parser . seenval ( ' T ' ) ) {
2017-06-27 09:36:19 +02:00
const int8_t e = parser . value_byte ( ) ;
if ( e > = EXTRUDERS ) {
2017-06-09 17:51:23 +02:00
SERIAL_ECHO_START ( ) ;
2016-04-18 09:01:28 +02:00
SERIAL_CHAR ( ' M ' ) ;
SERIAL_ECHO ( code ) ;
2017-06-27 09:36:19 +02:00
SERIAL_ECHOLNPAIR ( " " MSG_INVALID_EXTRUDER " " , e ) ;
2016-04-18 09:01:28 +02:00
return true ;
}
2017-06-27 09:36:19 +02:00
target_extruder = e ;
2016-04-18 09:01:28 +02:00
}
else
target_extruder = active_extruder ;
return false ;
}
2016-07-20 19:30:10 +02:00
# if ENABLED(DUAL_X_CARRIAGE) || ENABLED(DUAL_NOZZLE_DUPLICATION_MODE)
bool extruder_duplication_enabled = false ; // Used in Dual X mode 2
# endif
2015-07-31 07:24:43 +02:00
# if ENABLED(DUAL_X_CARRIAGE)
2013-08-01 15:06:39 +02:00
2016-11-03 03:37:46 +01:00
static DualXMode dual_x_carriage_mode = DEFAULT_DUAL_X_CARRIAGE_MODE ;
2015-03-20 06:22:23 +01:00
2016-11-07 07:20:59 +01:00
static float x_home_pos ( const int extruder ) {
2015-03-20 06:22:23 +01:00
if ( extruder = = 0 )
2016-07-25 01:34:01 +02:00
return LOGICAL_X_POSITION ( base_home_pos ( X_AXIS ) ) ;
2015-03-20 06:22:23 +01:00
else
2016-03-27 05:36:36 +02:00
/**
* In dual carriage mode the extruder offset provides an override of the
2016-11-07 07:20:19 +01:00
* second X - carriage position when homed - otherwise X2_HOME_POS is used .
* This allows soft recalibration of the second extruder home position
2016-03-27 05:36:36 +02:00
* without firmware reflash ( through the M218 command ) .
*/
2016-11-07 07:20:19 +01:00
return LOGICAL_X_POSITION ( hotend_offset [ X_AXIS ] [ 1 ] > 0 ? hotend_offset [ X_AXIS ] [ 1 ] : X2_HOME_POS ) ;
2015-03-20 06:22:23 +01:00
}
2016-11-07 07:20:59 +01:00
static int x_home_dir ( const int extruder ) { return extruder ? X2_HOME_DIR : X_HOME_DIR ; }
2015-03-20 06:22:23 +01:00
static float inactive_extruder_x_pos = X2_MAX_POS ; // used in mode 0 & 1
2016-07-25 01:34:31 +02:00
static bool active_extruder_parked = false ; // used in mode 1 & 2
2017-03-18 03:14:05 +01:00
static float raised_parked_position [ XYZE ] ; // used in mode 1
2016-07-25 01:34:31 +02:00
static millis_t delayed_move_time = 0 ; // used in mode 1
2015-03-20 06:22:23 +01:00
static float duplicate_extruder_x_offset = DEFAULT_DUPLICATION_X_OFFSET ; // used in mode 2
2017-05-04 00:12:14 +02:00
static int16_t duplicate_extruder_temp_offset = 0 ; // used in mode 2
2013-07-17 14:44:45 +02:00
2016-10-05 12:50:22 +02:00
# endif // DUAL_X_CARRIAGE
2013-07-17 14:44:45 +02:00
2017-04-15 04:41:21 +02:00
# if HAS_WORKSPACE_OFFSET || ENABLED(DUAL_X_CARRIAGE)
2016-05-23 23:46:29 +02:00
2017-03-05 01:01:33 +01:00
/**
* Software endstops can be used to monitor the open end of
* an axis that has a hardware endstop on the other end . Or
* they can prevent axes from moving past endstops and grinding .
*
* To keep doing their job as the coordinate system changes ,
* the software endstop positions must be refreshed to remain
* at the same positions relative to the machine .
*/
void update_software_endstops ( const AxisEnum axis ) {
2017-04-15 04:41:21 +02:00
const float offs = 0.0
# if HAS_HOME_OFFSET
+ home_offset [ axis ]
# endif
# if HAS_POSITION_SHIFT
+ position_shift [ axis ]
# endif
;
# if HAS_HOME_OFFSET && HAS_POSITION_SHIFT
workspace_offset [ axis ] = offs ;
# endif
2016-12-03 07:27:10 +01:00
2017-03-05 01:01:33 +01:00
# if ENABLED(DUAL_X_CARRIAGE)
if ( axis = = X_AXIS ) {
2016-12-03 07:27:10 +01:00
2017-03-05 01:01:33 +01:00
// In Dual X mode hotend_offset[X] is T1's home position
float dual_max_x = max ( hotend_offset [ X_AXIS ] [ 1 ] , X2_MAX_POS ) ;
if ( active_extruder ! = 0 ) {
// T1 can move from X2_MIN_POS to X2_MAX_POS or X2 home position (whichever is larger)
soft_endstop_min [ X_AXIS ] = X2_MIN_POS + offs ;
soft_endstop_max [ X_AXIS ] = dual_max_x + offs ;
}
else if ( dual_x_carriage_mode = = DXC_DUPLICATION_MODE ) {
// In Duplication Mode, T0 can move as far left as X_MIN_POS
// but not so far to the right that T1 would move past the end
soft_endstop_min [ X_AXIS ] = base_min_pos ( X_AXIS ) + offs ;
soft_endstop_max [ X_AXIS ] = min ( base_max_pos ( X_AXIS ) , dual_max_x - duplicate_extruder_x_offset ) + offs ;
}
else {
// In other modes, T0 can move from X_MIN_POS to X_MAX_POS
soft_endstop_min [ axis ] = base_min_pos ( axis ) + offs ;
soft_endstop_max [ axis ] = base_max_pos ( axis ) + offs ;
}
2016-03-24 11:55:18 +01:00
}
2017-08-03 05:52:40 +02:00
# elif ENABLED(DELTA)
soft_endstop_min [ axis ] = base_min_pos ( axis ) + ( axis = = Z_AXIS ? 0 : offs ) ;
soft_endstop_max [ axis ] = base_max_pos ( axis ) + offs ;
2017-03-05 01:01:33 +01:00
# else
soft_endstop_min [ axis ] = base_min_pos ( axis ) + offs ;
soft_endstop_max [ axis ] = base_max_pos ( axis ) + offs ;
# endif
2016-07-15 15:20:54 +02:00
2017-03-05 01:01:33 +01:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) {
SERIAL_ECHOPAIR ( " For " , axis_codes [ axis ] ) ;
2017-04-15 04:41:21 +02:00
# if HAS_HOME_OFFSET
2017-03-05 01:01:33 +01:00
SERIAL_ECHOPAIR ( " axis: \n home_offset = " , home_offset [ axis ] ) ;
2017-04-15 04:41:21 +02:00
# endif
# if HAS_POSITION_SHIFT
2017-03-05 01:01:33 +01:00
SERIAL_ECHOPAIR ( " \n position_shift = " , position_shift [ axis ] ) ;
# endif
SERIAL_ECHOPAIR ( " \n soft_endstop_min = " , soft_endstop_min [ axis ] ) ;
SERIAL_ECHOLNPAIR ( " \n soft_endstop_max = " , soft_endstop_max [ axis ] ) ;
}
# endif
2016-07-15 15:20:54 +02:00
2017-03-05 01:01:33 +01:00
# if ENABLED(DELTA)
if ( axis = = Z_AXIS )
delta_clip_start_height = soft_endstop_max [ axis ] - delta_safe_distance_from_top ( ) ;
# endif
}
2016-07-15 15:20:54 +02:00
2017-04-15 04:41:21 +02:00
# endif // HAS_WORKSPACE_OFFSET || DUAL_X_CARRIAGE
2016-03-24 11:55:18 +01:00
2017-04-15 04:41:21 +02:00
# if HAS_M206_COMMAND
2017-03-05 01:01:33 +01:00
/**
* Change the home offset for an axis , update the current
* position and the software endstops to retain the same
* relative distance to the new home .
*
* Since this changes the current_position , code should
* call sync_plan_position soon after this .
*/
2017-03-05 02:18:11 +01:00
static void set_home_offset ( const AxisEnum axis , const float v ) {
2017-03-05 01:01:33 +01:00
current_position [ axis ] + = v - home_offset [ axis ] ;
home_offset [ axis ] = v ;
update_software_endstops ( axis ) ;
}
2017-04-15 04:41:21 +02:00
# endif // HAS_M206_COMMAND
2016-03-24 11:55:18 +01:00
2016-09-15 09:47:32 +02:00
/**
* Set an axis ' current position to its home position ( after homing ) .
*
* For Core and Cartesian robots this applies one - to - one when an
* individual axis has been homed .
*
* DELTA should wait until all homing is done before setting the XYZ
* current_position to home , because homing is a single operation .
* In the case where the axis positions are already known and previously
* homed , DELTA could home to X or Y individually by moving either one
* to the center . However , homing Z always homes XY and Z .
*
* SCARA should wait until all XY homing is done before setting the XY
* current_position to home , because neither X nor Y is at home until
* both are at home . Z can however be homed individually .
2016-09-26 08:30:34 +02:00
*
2016-11-07 07:21:45 +01:00
* Callers must sync the planner position after calling this !
2016-09-15 09:47:32 +02:00
*/
2017-06-06 00:41:38 +02:00
static void set_axis_is_at_home ( const AxisEnum axis ) {
2016-04-06 03:30:53 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) {
2016-08-19 08:09:03 +02:00
SERIAL_ECHOPAIR ( " >>> set_axis_is_at_home( " , axis_codes [ axis ] ) ;
2016-10-02 08:48:17 +02:00
SERIAL_CHAR ( ' ) ' ) ;
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2016-04-06 03:30:53 +02:00
}
# endif
2015-03-28 04:29:05 +01:00
2016-09-15 10:10:33 +02:00
axis_known_position [ axis ] = axis_homed [ axis ] = true ;
2017-04-15 04:41:21 +02:00
# if HAS_POSITION_SHIFT
2017-03-05 01:01:33 +01:00
position_shift [ axis ] = 0 ;
update_software_endstops ( axis ) ;
# endif
2016-03-24 11:55:18 +01:00
2015-07-31 07:24:43 +02:00
# if ENABLED(DUAL_X_CARRIAGE)
2016-11-07 07:23:39 +01:00
if ( axis = = X_AXIS & & ( active_extruder = = 1 | | dual_x_carriage_mode = = DXC_DUPLICATION_MODE ) ) {
current_position [ X_AXIS ] = x_home_pos ( active_extruder ) ;
2016-03-24 11:55:18 +01:00
return ;
2013-08-07 16:10:26 +02:00
}
2015-03-28 04:29:05 +01:00
# endif
2016-09-12 10:48:29 +02:00
# if ENABLED(MORGAN_SCARA)
2015-08-05 13:40:36 +02:00
2016-09-21 23:47:12 +02:00
/**
* Morgan SCARA homes XY at the same time
*/
2015-04-24 03:16:44 +02:00
if ( axis = = X_AXIS | | axis = = Y_AXIS ) {
2015-03-28 04:29:05 +01:00
2016-08-21 05:38:32 +02:00
float homeposition [ XYZ ] ;
2016-11-02 23:42:24 +01:00
LOOP_XYZ ( i ) homeposition [ i ] = LOGICAL_POSITION ( base_home_pos ( ( AxisEnum ) i ) , i ) ;
2015-03-28 04:29:05 +01:00
2016-09-13 00:49:35 +02:00
// SERIAL_ECHOPAIR("homeposition X:", homeposition[X_AXIS]);
// SERIAL_ECHOLNPAIR(" Y:", homeposition[Y_AXIS]);
2016-03-27 05:36:36 +02:00
/**
2016-09-15 20:34:24 +02:00
* Get Home position SCARA arm angles using inverse kinematics ,
* and calculate homing offset using forward kinematics
2016-03-27 05:36:36 +02:00
*/
2016-07-22 00:46:22 +02:00
inverse_kinematics ( homeposition ) ;
2016-09-12 10:48:29 +02:00
forward_kinematics_SCARA ( delta [ A_AXIS ] , delta [ B_AXIS ] ) ;
2015-08-05 13:40:36 +02:00
2016-09-13 00:49:35 +02:00
// SERIAL_ECHOPAIR("Cartesian X:", cartes[X_AXIS]);
// SERIAL_ECHOLNPAIR(" Y:", cartes[Y_AXIS]);
2015-08-05 13:40:36 +02:00
2016-09-12 10:48:29 +02:00
current_position [ axis ] = LOGICAL_POSITION ( cartes [ axis ] , axis ) ;
2015-08-05 13:40:36 +02:00
2016-03-27 05:36:36 +02:00
/**
* SCARA home positions are based on configuration since the actual
* limits are determined by the inverse kinematic transform .
*/
2016-09-12 10:48:29 +02:00
soft_endstop_min [ axis ] = base_min_pos ( axis ) ; // + (cartes[axis] - base_home_pos(axis));
soft_endstop_max [ axis ] = base_max_pos ( axis ) ; // + (cartes[axis] - base_home_pos(axis));
2015-03-28 04:29:05 +01:00
}
2015-04-24 03:16:44 +02:00
else
# endif
{
2016-07-23 02:46:05 +02:00
current_position [ axis ] = LOGICAL_POSITION ( base_home_pos ( axis ) , axis ) ;
2016-09-21 23:47:12 +02:00
}
2015-04-16 16:41:19 +02:00
2016-09-21 23:47:12 +02:00
/**
* Z Probe Z Homing ? Account for the probe ' s Z offset .
*/
# if HAS_BED_PROBE && Z_HOME_DIR < 0
2016-08-26 22:42:16 +02:00
if ( axis = = Z_AXIS ) {
2016-09-21 23:47:12 +02:00
# if HOMING_Z_WITH_PROBE
current_position [ Z_AXIS ] - = zprobe_zoffset ;
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) {
SERIAL_ECHOLNPGM ( " *** Z HOMED WITH PROBE (Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN) *** " ) ;
SERIAL_ECHOLNPAIR ( " > zprobe_zoffset = " , zprobe_zoffset ) ;
}
2016-08-26 22:42:16 +02:00
# endif
2016-09-21 23:47:12 +02:00
# elif ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) SERIAL_ECHOLNPGM ( " *** Z HOMED TO ENDSTOP (Z_MIN_PROBE_ENDSTOP) *** " ) ;
2016-08-26 22:42:16 +02:00
# endif
}
2016-09-21 23:47:12 +02:00
# endif
2016-08-26 22:42:16 +02:00
2016-04-06 03:30:53 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) {
2017-04-15 04:41:21 +02:00
# if HAS_HOME_OFFSET
2017-03-05 01:01:33 +01:00
SERIAL_ECHOPAIR ( " > home_offset[ " , axis_codes [ axis ] ) ;
SERIAL_ECHOLNPAIR ( " ] = " , home_offset [ axis ] ) ;
# endif
2016-09-21 23:47:12 +02:00
DEBUG_POS ( " " , current_position ) ;
2016-08-19 08:09:03 +02:00
SERIAL_ECHOPAIR ( " <<< set_axis_is_at_home( " , axis_codes [ axis ] ) ;
2016-10-02 08:48:17 +02:00
SERIAL_CHAR ( ' ) ' ) ;
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2016-04-06 03:30:53 +02:00
}
# endif
2017-06-09 14:06:23 +02:00
# if ENABLED(I2C_POSITION_ENCODERS)
I2CPEM . homed ( axis ) ;
# endif
2012-11-06 12:06:41 +01:00
}
2015-03-29 05:33:21 +02:00
/**
2015-03-30 08:16:12 +02:00
* Some planner shorthand inline functions
2015-03-29 05:33:21 +02:00
*/
2017-06-06 00:41:38 +02:00
inline float get_homing_bump_feedrate ( const AxisEnum axis ) {
2017-06-08 21:59:21 +02:00
static const uint8_t homing_bump_divisor [ ] PROGMEM = HOMING_BUMP_DIVISOR ;
2017-06-06 09:58:31 +02:00
uint8_t hbd = pgm_read_byte ( & homing_bump_divisor [ axis ] ) ;
2015-07-24 22:43:03 +02:00
if ( hbd < 1 ) {
hbd = 10 ;
2017-06-09 17:51:23 +02:00
SERIAL_ECHO_START ( ) ;
2015-07-24 22:43:03 +02:00
SERIAL_ECHOLNPGM ( " Warning: Homing Bump Divisor < 1 " ) ;
2015-04-07 08:21:33 +02:00
}
2017-06-06 01:49:16 +02:00
return homing_feedrate ( axis ) / hbd ;
2015-04-07 08:21:33 +02:00
}
2016-07-19 17:24:44 +02:00
2017-06-06 00:41:38 +02:00
/**
* Move the planner to the current position from wherever it last moved
* ( or from wherever it has been told it is located ) .
*/
2016-09-22 14:49:49 +02:00
inline void line_to_current_position ( ) {
planner . buffer_line ( current_position [ X_AXIS ] , current_position [ Y_AXIS ] , current_position [ Z_AXIS ] , current_position [ E_AXIS ] , feedrate_mm_s , active_extruder ) ;
}
2016-09-15 10:02:42 +02:00
2017-06-06 00:41:38 +02:00
/**
* Move the planner to the position stored in the destination array , which is
* used by G0 / G1 / G2 / G3 / G5 and many other functions to set a destination .
*/
inline void line_to_destination ( const float fr_mm_s ) {
2016-09-22 14:49:49 +02:00
planner . buffer_line ( destination [ X_AXIS ] , destination [ Y_AXIS ] , destination [ Z_AXIS ] , destination [ E_AXIS ] , fr_mm_s , active_extruder ) ;
}
inline void line_to_destination ( ) { line_to_destination ( feedrate_mm_s ) ; }
2016-04-28 18:57:21 +02:00
2017-03-05 02:19:06 +01:00
inline void set_current_to_destination ( ) { COPY ( current_position , destination ) ; }
inline void set_destination_to_current ( ) { COPY ( destination , current_position ) ; }
2015-03-29 05:33:21 +02:00
2016-09-15 20:21:59 +02:00
# if IS_KINEMATIC
2016-06-15 03:03:17 +02:00
/**
2016-07-15 01:26:37 +02:00
* Calculate delta , start a line , and set current_position to destination
2016-06-15 03:03:17 +02:00
*/
2016-09-21 23:47:12 +02:00
void prepare_uninterpolated_move_to_destination ( const float fr_mm_s = 0.0 ) {
2016-06-15 03:03:17 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
2016-09-15 09:55:23 +02:00
if ( DEBUGGING ( LEVELING ) ) DEBUG_POS ( " prepare_uninterpolated_move_to_destination " , destination ) ;
2016-06-15 03:03:17 +02:00
# endif
2016-09-22 01:47:11 +02:00
2016-07-15 01:26:37 +02:00
refresh_cmd_timeout ( ) ;
2017-06-04 00:11:43 +02:00
# if UBL_DELTA
// ubl segmented line will do z-only moves in single segment
ubl . prepare_segmented_line_to ( destination , MMS_SCALED ( fr_mm_s ? fr_mm_s : feedrate_mm_s ) ) ;
# else
if ( current_position [ X_AXIS ] = = destination [ X_AXIS ]
& & current_position [ Y_AXIS ] = = destination [ Y_AXIS ]
& & current_position [ Z_AXIS ] = = destination [ Z_AXIS ]
& & current_position [ E_AXIS ] = = destination [ E_AXIS ]
) return ;
planner . buffer_line_kinematic ( destination , MMS_SCALED ( fr_mm_s ? fr_mm_s : feedrate_mm_s ) , active_extruder ) ;
# endif
2017-06-04 18:30:03 +02:00
2016-07-15 01:26:37 +02:00
set_current_to_destination ( ) ;
}
2016-09-22 00:27:37 +02:00
# endif // IS_KINEMATIC
2016-06-15 03:03:17 +02:00
2016-07-15 01:26:37 +02:00
/**
* Plan a move to ( X , Y , Z ) and set the current_position
* The final current_position may not be the one that was requested
*/
2017-06-07 14:12:19 +02:00
void do_blocking_move_to ( const float & lx , const float & ly , const float & lz , const float & fr_mm_s /*=0.0*/ ) {
2017-03-04 01:23:31 +01:00
const float old_feedrate_mm_s = feedrate_mm_s ;
2016-06-15 03:03:17 +02:00
2016-07-15 01:26:37 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
2017-06-07 14:12:19 +02:00
if ( DEBUGGING ( LEVELING ) ) print_xyz ( PSTR ( " >>> do_blocking_move_to " ) , NULL , lx , ly , lz ) ;
2016-07-15 01:26:37 +02:00
# endif
2016-06-15 03:03:17 +02:00
2016-07-15 01:26:37 +02:00
# if ENABLED(DELTA)
2016-06-15 03:03:17 +02:00
2017-06-07 14:12:19 +02:00
if ( ! position_is_reachable_xy ( lx , ly ) ) return ;
2017-05-12 05:33:47 +02:00
2016-09-21 23:47:12 +02:00
feedrate_mm_s = fr_mm_s ? fr_mm_s : XY_PROBE_FEEDRATE_MM_S ;
2016-06-15 03:03:17 +02:00
2016-07-22 23:18:08 +02:00
set_destination_to_current ( ) ; // sync destination at the start
2016-07-23 08:52:42 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
2016-07-23 09:37:46 +02:00
if ( DEBUGGING ( LEVELING ) ) DEBUG_POS ( " set_destination_to_current " , destination ) ;
2016-07-23 08:52:42 +02:00
# endif
2016-07-22 03:12:46 +02:00
// when in the danger zone
if ( current_position [ Z_AXIS ] > delta_clip_start_height ) {
2017-06-07 14:12:19 +02:00
if ( lz > delta_clip_start_height ) { // staying in the danger zone
destination [ X_AXIS ] = lx ; // move directly (uninterpolated)
destination [ Y_AXIS ] = ly ;
destination [ Z_AXIS ] = lz ;
2016-09-15 09:55:23 +02:00
prepare_uninterpolated_move_to_destination ( ) ; // set_current_to_destination
2016-07-23 08:52:42 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
2016-07-23 09:37:46 +02:00
if ( DEBUGGING ( LEVELING ) ) DEBUG_POS ( " danger zone move " , current_position ) ;
2016-07-23 08:52:42 +02:00
# endif
2016-07-22 03:12:46 +02:00
return ;
2016-07-22 23:18:08 +02:00
}
else {
2016-07-22 03:12:46 +02:00
destination [ Z_AXIS ] = delta_clip_start_height ;
2016-09-15 09:55:23 +02:00
prepare_uninterpolated_move_to_destination ( ) ; // set_current_to_destination
2016-07-23 08:52:42 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
2016-07-23 09:37:46 +02:00
if ( DEBUGGING ( LEVELING ) ) DEBUG_POS ( " zone border move " , current_position ) ;
2016-07-23 08:52:42 +02:00
# endif
2016-07-22 03:12:46 +02:00
}
}
2016-07-22 23:18:08 +02:00
2017-06-07 14:12:19 +02:00
if ( lz > current_position [ Z_AXIS ] ) { // raising?
destination [ Z_AXIS ] = lz ;
2016-09-15 09:55:23 +02:00
prepare_uninterpolated_move_to_destination ( ) ; // set_current_to_destination
2016-07-23 08:52:42 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
2016-07-23 09:37:46 +02:00
if ( DEBUGGING ( LEVELING ) ) DEBUG_POS ( " z raise move " , current_position ) ;
2016-07-23 08:52:42 +02:00
# endif
2016-07-22 03:12:46 +02:00
}
2016-07-22 23:18:08 +02:00
2017-06-07 14:12:19 +02:00
destination [ X_AXIS ] = lx ;
destination [ Y_AXIS ] = ly ;
2016-07-22 23:18:08 +02:00
prepare_move_to_destination ( ) ; // set_current_to_destination
2016-07-23 08:52:42 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
2016-07-23 09:37:46 +02:00
if ( DEBUGGING ( LEVELING ) ) DEBUG_POS ( " xy move " , current_position ) ;
2016-07-23 08:52:42 +02:00
# endif
2016-06-15 03:03:17 +02:00
2017-06-07 14:12:19 +02:00
if ( lz < current_position [ Z_AXIS ] ) { // lowering?
destination [ Z_AXIS ] = lz ;
2016-09-15 09:55:23 +02:00
prepare_uninterpolated_move_to_destination ( ) ; // set_current_to_destination
2016-07-23 08:52:42 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
2016-07-23 09:37:46 +02:00
if ( DEBUGGING ( LEVELING ) ) DEBUG_POS ( " z lower move " , current_position ) ;
2016-07-23 08:52:42 +02:00
# endif
2016-07-22 03:12:46 +02:00
}
2016-07-15 01:26:37 +02:00
2016-09-15 20:21:59 +02:00
# elif IS_SCARA
2017-06-07 14:12:19 +02:00
if ( ! position_is_reachable_xy ( lx , ly ) ) return ;
2017-05-12 05:33:47 +02:00
2016-09-15 20:21:59 +02:00
set_destination_to_current ( ) ;
// If Z needs to raise, do it before moving XY
2017-06-07 14:12:19 +02:00
if ( destination [ Z_AXIS ] < lz ) {
destination [ Z_AXIS ] = lz ;
2017-06-06 01:49:16 +02:00
prepare_uninterpolated_move_to_destination ( fr_mm_s ? fr_mm_s : homing_feedrate ( Z_AXIS ) ) ;
2016-09-15 20:21:59 +02:00
}
2017-06-07 14:12:19 +02:00
destination [ X_AXIS ] = lx ;
destination [ Y_AXIS ] = ly ;
2016-09-21 23:47:12 +02:00
prepare_uninterpolated_move_to_destination ( fr_mm_s ? fr_mm_s : XY_PROBE_FEEDRATE_MM_S ) ;
2016-09-15 20:21:59 +02:00
// If Z needs to lower, do it after moving XY
2017-06-07 14:12:19 +02:00
if ( destination [ Z_AXIS ] > lz ) {
destination [ Z_AXIS ] = lz ;
2017-06-06 01:49:16 +02:00
prepare_uninterpolated_move_to_destination ( fr_mm_s ? fr_mm_s : homing_feedrate ( Z_AXIS ) ) ;
2016-09-15 20:21:59 +02:00
}
2016-07-15 01:26:37 +02:00
# else
2016-06-15 03:03:17 +02:00
2016-07-15 01:26:37 +02:00
// If Z needs to raise, do it before moving XY
2017-06-07 14:12:19 +02:00
if ( current_position [ Z_AXIS ] < lz ) {
2017-06-06 01:49:16 +02:00
feedrate_mm_s = fr_mm_s ? fr_mm_s : homing_feedrate ( Z_AXIS ) ;
2017-06-07 14:12:19 +02:00
current_position [ Z_AXIS ] = lz ;
2016-06-15 03:03:17 +02:00
line_to_current_position ( ) ;
2016-07-15 01:26:37 +02:00
}
2016-06-15 03:03:17 +02:00
2016-09-21 23:47:12 +02:00
feedrate_mm_s = fr_mm_s ? fr_mm_s : XY_PROBE_FEEDRATE_MM_S ;
2017-06-07 14:12:19 +02:00
current_position [ X_AXIS ] = lx ;
current_position [ Y_AXIS ] = ly ;
2016-07-15 01:26:37 +02:00
line_to_current_position ( ) ;
2016-06-24 00:50:13 +02:00
2016-07-15 01:26:37 +02:00
// If Z needs to lower, do it after moving XY
2017-06-07 14:12:19 +02:00
if ( current_position [ Z_AXIS ] > lz ) {
2017-06-06 01:49:16 +02:00
feedrate_mm_s = fr_mm_s ? fr_mm_s : homing_feedrate ( Z_AXIS ) ;
2017-06-07 14:12:19 +02:00
current_position [ Z_AXIS ] = lz ;
2016-07-15 01:26:37 +02:00
line_to_current_position ( ) ;
}
2016-06-15 03:03:17 +02:00
2016-07-15 01:26:37 +02:00
# endif
2016-06-15 03:03:17 +02:00
2016-07-15 01:26:37 +02:00
stepper . synchronize ( ) ;
2016-06-15 03:03:17 +02:00
2016-08-07 09:20:25 +02:00
feedrate_mm_s = old_feedrate_mm_s ;
2016-09-22 00:27:37 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) SERIAL_ECHOLNPGM ( " <<< do_blocking_move_to " ) ;
# endif
2016-07-15 01:26:37 +02:00
}
2017-06-07 14:12:19 +02:00
void do_blocking_move_to_x ( const float & lx , const float & fr_mm_s /*=0.0*/ ) {
do_blocking_move_to ( lx , current_position [ Y_AXIS ] , current_position [ Z_AXIS ] , fr_mm_s ) ;
2016-07-23 09:58:53 +02:00
}
2017-06-07 14:12:19 +02:00
void do_blocking_move_to_z ( const float & lz , const float & fr_mm_s /*=0.0*/ ) {
do_blocking_move_to ( current_position [ X_AXIS ] , current_position [ Y_AXIS ] , lz , fr_mm_s ) ;
2016-07-23 09:58:53 +02:00
}
2017-06-07 14:12:19 +02:00
void do_blocking_move_to_xy ( const float & lx , const float & ly , const float & fr_mm_s /*=0.0*/ ) {
do_blocking_move_to ( lx , ly , current_position [ Z_AXIS ] , fr_mm_s ) ;
2016-07-15 01:26:37 +02:00
}
2016-06-15 03:03:17 +02:00
2016-07-16 04:13:02 +02:00
//
// Prepare to do endstop or probe moves
// with custom feedrates.
//
// - Save current feedrates
// - Reset the rate multiplier
// - Reset the command timeout
// - Enable the endstops (for endstop moves)
//
static void setup_for_endstop_or_probe_move ( ) {
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) DEBUG_POS ( " setup_for_endstop_or_probe_move " , current_position ) ;
# endif
2016-08-07 09:20:25 +02:00
saved_feedrate_mm_s = feedrate_mm_s ;
2016-07-16 03:49:34 +02:00
saved_feedrate_percentage = feedrate_percentage ;
feedrate_percentage = 100 ;
2016-07-16 04:13:02 +02:00
refresh_cmd_timeout ( ) ;
}
static void clean_up_after_endstop_or_probe_move ( ) {
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) DEBUG_POS ( " clean_up_after_endstop_or_probe_move " , current_position ) ;
# endif
2016-08-07 09:20:25 +02:00
feedrate_mm_s = saved_feedrate_mm_s ;
2016-07-16 03:49:34 +02:00
feedrate_percentage = saved_feedrate_percentage ;
2016-07-16 04:13:02 +02:00
refresh_cmd_timeout ( ) ;
}
2016-07-15 01:26:37 +02:00
# if HAS_BED_PROBE
2016-06-13 00:15:37 +02:00
/**
2016-06-22 23:40:15 +02:00
* Raise Z to a minimum height to make room for a probe to move
2016-06-13 00:15:37 +02:00
*/
2017-06-07 14:12:19 +02:00
inline void do_probe_raise ( const float z_raise ) {
2016-06-22 23:40:15 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) {
SERIAL_ECHOPAIR ( " do_probe_raise( " , z_raise ) ;
2016-10-02 08:48:17 +02:00
SERIAL_CHAR ( ' ) ' ) ;
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2016-06-22 23:40:15 +02:00
}
# endif
2016-09-14 03:25:12 +02:00
2017-08-03 05:52:40 +02:00
float z_dest = z_raise ;
2016-09-14 03:25:12 +02:00
if ( zprobe_zoffset < 0 ) z_dest - = zprobe_zoffset ;
2017-06-26 12:25:57 +02:00
2016-07-06 03:32:24 +02:00
if ( z_dest > current_position [ Z_AXIS ] )
2016-06-22 23:40:15 +02:00
do_blocking_move_to_z ( z_dest ) ;
2016-06-13 00:15:37 +02:00
}
2017-05-09 19:35:43 +02:00
# endif // HAS_BED_PROBE
2016-06-13 00:15:37 +02:00
2017-04-29 00:17:01 +02:00
# if HAS_PROBING_PROCEDURE || HOTENDS > 1 || ENABLED(Z_PROBE_ALLEN_KEY) || ENABLED(Z_PROBE_SLED) || ENABLED(NOZZLE_CLEAN_FEATURE) || ENABLED(NOZZLE_PARK_FEATURE) || ENABLED(DELTA_AUTO_CALIBRATION)
2017-05-14 22:57:37 +02:00
bool axis_unhomed_error ( const bool x /*=true*/ , const bool y /*=true*/ , const bool z /*=true*/ ) {
2017-05-29 19:52:56 +02:00
# if ENABLED(HOME_AFTER_DEACTIVATE)
const bool xx = x & & ! axis_known_position [ X_AXIS ] ,
yy = y & & ! axis_known_position [ Y_AXIS ] ,
zz = z & & ! axis_known_position [ Z_AXIS ] ;
# else
const bool xx = x & & ! axis_homed [ X_AXIS ] ,
yy = y & & ! axis_homed [ Y_AXIS ] ,
zz = z & & ! axis_homed [ Z_AXIS ] ;
# endif
2016-07-16 12:57:35 +02:00
if ( xx | | yy | | zz ) {
2017-06-09 17:51:23 +02:00
SERIAL_ECHO_START ( ) ;
2016-07-16 12:57:35 +02:00
SERIAL_ECHOPGM ( MSG_HOME " " ) ;
if ( xx ) SERIAL_ECHOPGM ( MSG_X ) ;
if ( yy ) SERIAL_ECHOPGM ( MSG_Y ) ;
if ( zz ) SERIAL_ECHOPGM ( MSG_Z ) ;
SERIAL_ECHOLNPGM ( " " MSG_FIRST ) ;
# if ENABLED(ULTRA_LCD)
2017-04-02 07:46:37 +02:00
lcd_status_printf_P ( 0 , PSTR ( MSG_HOME " %s%s%s " MSG_FIRST ) , xx ? MSG_X : " " , yy ? MSG_Y : " " , zz ? MSG_Z : " " ) ;
2016-07-16 12:57:35 +02:00
# endif
return true ;
}
return false ;
2016-06-22 23:03:22 +02:00
}
2017-04-29 00:17:01 +02:00
2016-07-16 12:57:35 +02:00
# endif
2016-06-22 23:03:22 +02:00
2016-06-21 00:50:51 +02:00
# if ENABLED(Z_PROBE_SLED)
# ifndef SLED_DOCKING_OFFSET
# define SLED_DOCKING_OFFSET 0
# endif
/**
* Method to dock / undock a sled designed by Charles Bell .
*
2016-06-27 18:09:22 +02:00
* stow [ in ] If false , move to MAX_X and engage the solenoid
* If true , move to MAX_X and release the solenoid
2016-06-21 00:50:51 +02:00
*/
2016-06-27 18:09:22 +02:00
static void dock_sled ( bool stow ) {
2016-06-21 00:50:51 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) {
2016-06-27 18:09:22 +02:00
SERIAL_ECHOPAIR ( " dock_sled( " , stow ) ;
2016-10-02 08:48:17 +02:00
SERIAL_CHAR ( ' ) ' ) ;
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2016-06-21 00:50:51 +02:00
}
# endif
2016-06-27 18:09:22 +02:00
// Dock sled a bit closer to ensure proper capturing
do_blocking_move_to_x ( X_MAX_POS + SLED_DOCKING_OFFSET - ( ( stow ) ? 1 : 0 ) ) ;
2017-04-14 23:36:02 +02:00
# if HAS_SOLENOID_1 && DISABLED(EXT_SOLENOID)
WRITE ( SOL1_PIN , ! stow ) ; // switch solenoid
2016-07-30 11:01:46 +02:00
# endif
2016-06-21 00:50:51 +02:00
}
2016-10-03 01:18:05 +02:00
# elif ENABLED(Z_PROBE_ALLEN_KEY)
2017-06-07 14:12:19 +02:00
FORCE_INLINE void do_blocking_move_to ( const float logical [ XYZ ] , const float & fr_mm_s ) {
do_blocking_move_to ( logical [ X_AXIS ] , logical [ Y_AXIS ] , logical [ Z_AXIS ] , fr_mm_s ) ;
}
2016-07-04 09:46:22 +02:00
void run_deploy_moves_script ( ) {
# if defined(Z_PROBE_ALLEN_KEY_DEPLOY_1_X) || defined(Z_PROBE_ALLEN_KEY_DEPLOY_1_Y) || defined(Z_PROBE_ALLEN_KEY_DEPLOY_1_Z)
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_1_X
# define Z_PROBE_ALLEN_KEY_DEPLOY_1_X current_position[X_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_1_Y
# define Z_PROBE_ALLEN_KEY_DEPLOY_1_Y current_position[Y_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_1_Z
# define Z_PROBE_ALLEN_KEY_DEPLOY_1_Z current_position[Z_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_1_FEEDRATE
# define Z_PROBE_ALLEN_KEY_DEPLOY_1_FEEDRATE 0.0
# endif
2017-06-07 14:12:19 +02:00
const float deploy_1 [ ] = { Z_PROBE_ALLEN_KEY_DEPLOY_1_X , Z_PROBE_ALLEN_KEY_DEPLOY_1_Y , Z_PROBE_ALLEN_KEY_DEPLOY_1_Z } ;
do_blocking_move_to ( deploy_1 , MMM_TO_MMS ( Z_PROBE_ALLEN_KEY_DEPLOY_1_FEEDRATE ) ) ;
2016-07-04 09:46:22 +02:00
# endif
# if defined(Z_PROBE_ALLEN_KEY_DEPLOY_2_X) || defined(Z_PROBE_ALLEN_KEY_DEPLOY_2_Y) || defined(Z_PROBE_ALLEN_KEY_DEPLOY_2_Z)
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_2_X
# define Z_PROBE_ALLEN_KEY_DEPLOY_2_X current_position[X_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_2_Y
# define Z_PROBE_ALLEN_KEY_DEPLOY_2_Y current_position[Y_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_2_Z
# define Z_PROBE_ALLEN_KEY_DEPLOY_2_Z current_position[Z_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_2_FEEDRATE
# define Z_PROBE_ALLEN_KEY_DEPLOY_2_FEEDRATE 0.0
# endif
2017-06-07 14:12:19 +02:00
const float deploy_2 [ ] = { Z_PROBE_ALLEN_KEY_DEPLOY_2_X , Z_PROBE_ALLEN_KEY_DEPLOY_2_Y , Z_PROBE_ALLEN_KEY_DEPLOY_2_Z } ;
do_blocking_move_to ( deploy_2 , MMM_TO_MMS ( Z_PROBE_ALLEN_KEY_DEPLOY_2_FEEDRATE ) ) ;
2016-07-04 09:46:22 +02:00
# endif
# if defined(Z_PROBE_ALLEN_KEY_DEPLOY_3_X) || defined(Z_PROBE_ALLEN_KEY_DEPLOY_3_Y) || defined(Z_PROBE_ALLEN_KEY_DEPLOY_3_Z)
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_3_X
# define Z_PROBE_ALLEN_KEY_DEPLOY_3_X current_position[X_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_3_Y
# define Z_PROBE_ALLEN_KEY_DEPLOY_3_Y current_position[Y_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_3_Z
# define Z_PROBE_ALLEN_KEY_DEPLOY_3_Z current_position[Z_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_3_FEEDRATE
# define Z_PROBE_ALLEN_KEY_DEPLOY_3_FEEDRATE 0.0
# endif
2017-06-07 14:12:19 +02:00
const float deploy_3 [ ] = { Z_PROBE_ALLEN_KEY_DEPLOY_3_X , Z_PROBE_ALLEN_KEY_DEPLOY_3_Y , Z_PROBE_ALLEN_KEY_DEPLOY_3_Z } ;
do_blocking_move_to ( deploy_3 , MMM_TO_MMS ( Z_PROBE_ALLEN_KEY_DEPLOY_3_FEEDRATE ) ) ;
2016-07-04 09:46:22 +02:00
# endif
# if defined(Z_PROBE_ALLEN_KEY_DEPLOY_4_X) || defined(Z_PROBE_ALLEN_KEY_DEPLOY_4_Y) || defined(Z_PROBE_ALLEN_KEY_DEPLOY_4_Z)
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_4_X
# define Z_PROBE_ALLEN_KEY_DEPLOY_4_X current_position[X_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_4_Y
# define Z_PROBE_ALLEN_KEY_DEPLOY_4_Y current_position[Y_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_4_Z
# define Z_PROBE_ALLEN_KEY_DEPLOY_4_Z current_position[Z_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_4_FEEDRATE
# define Z_PROBE_ALLEN_KEY_DEPLOY_4_FEEDRATE 0.0
# endif
2017-06-07 14:12:19 +02:00
const float deploy_4 [ ] = { Z_PROBE_ALLEN_KEY_DEPLOY_4_X , Z_PROBE_ALLEN_KEY_DEPLOY_4_Y , Z_PROBE_ALLEN_KEY_DEPLOY_4_Z } ;
do_blocking_move_to ( deploy_4 , MMM_TO_MMS ( Z_PROBE_ALLEN_KEY_DEPLOY_4_FEEDRATE ) ) ;
2016-07-04 09:46:22 +02:00
# endif
# if defined(Z_PROBE_ALLEN_KEY_DEPLOY_5_X) || defined(Z_PROBE_ALLEN_KEY_DEPLOY_5_Y) || defined(Z_PROBE_ALLEN_KEY_DEPLOY_5_Z)
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_5_X
# define Z_PROBE_ALLEN_KEY_DEPLOY_5_X current_position[X_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_5_Y
# define Z_PROBE_ALLEN_KEY_DEPLOY_5_Y current_position[Y_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_5_Z
# define Z_PROBE_ALLEN_KEY_DEPLOY_5_Z current_position[Z_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_DEPLOY_5_FEEDRATE
# define Z_PROBE_ALLEN_KEY_DEPLOY_5_FEEDRATE 0.0
# endif
2017-06-07 14:12:19 +02:00
const float deploy_5 [ ] = { Z_PROBE_ALLEN_KEY_DEPLOY_5_X , Z_PROBE_ALLEN_KEY_DEPLOY_5_Y , Z_PROBE_ALLEN_KEY_DEPLOY_5_Z } ;
do_blocking_move_to ( deploy_5 , MMM_TO_MMS ( Z_PROBE_ALLEN_KEY_DEPLOY_5_FEEDRATE ) ) ;
2016-07-04 09:46:22 +02:00
# endif
}
2016-10-03 01:18:05 +02:00
2016-07-04 09:46:22 +02:00
void run_stow_moves_script ( ) {
# if defined(Z_PROBE_ALLEN_KEY_STOW_1_X) || defined(Z_PROBE_ALLEN_KEY_STOW_1_Y) || defined(Z_PROBE_ALLEN_KEY_STOW_1_Z)
# ifndef Z_PROBE_ALLEN_KEY_STOW_1_X
# define Z_PROBE_ALLEN_KEY_STOW_1_X current_position[X_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_STOW_1_Y
# define Z_PROBE_ALLEN_KEY_STOW_1_Y current_position[Y_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_STOW_1_Z
# define Z_PROBE_ALLEN_KEY_STOW_1_Z current_position[Z_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_STOW_1_FEEDRATE
# define Z_PROBE_ALLEN_KEY_STOW_1_FEEDRATE 0.0
# endif
2017-06-07 14:12:19 +02:00
const float stow_1 [ ] = { Z_PROBE_ALLEN_KEY_STOW_1_X , Z_PROBE_ALLEN_KEY_STOW_1_Y , Z_PROBE_ALLEN_KEY_STOW_1_Z } ;
do_blocking_move_to ( stow_1 , MMM_TO_MMS ( Z_PROBE_ALLEN_KEY_STOW_1_FEEDRATE ) ) ;
2016-07-04 09:46:22 +02:00
# endif
# if defined(Z_PROBE_ALLEN_KEY_STOW_2_X) || defined(Z_PROBE_ALLEN_KEY_STOW_2_Y) || defined(Z_PROBE_ALLEN_KEY_STOW_2_Z)
# ifndef Z_PROBE_ALLEN_KEY_STOW_2_X
# define Z_PROBE_ALLEN_KEY_STOW_2_X current_position[X_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_STOW_2_Y
# define Z_PROBE_ALLEN_KEY_STOW_2_Y current_position[Y_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_STOW_2_Z
# define Z_PROBE_ALLEN_KEY_STOW_2_Z current_position[Z_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_STOW_2_FEEDRATE
# define Z_PROBE_ALLEN_KEY_STOW_2_FEEDRATE 0.0
# endif
2017-06-07 14:12:19 +02:00
const float stow_2 [ ] = { Z_PROBE_ALLEN_KEY_STOW_2_X , Z_PROBE_ALLEN_KEY_STOW_2_Y , Z_PROBE_ALLEN_KEY_STOW_2_Z } ;
do_blocking_move_to ( stow_2 , MMM_TO_MMS ( Z_PROBE_ALLEN_KEY_STOW_2_FEEDRATE ) ) ;
2016-07-04 09:46:22 +02:00
# endif
# if defined(Z_PROBE_ALLEN_KEY_STOW_3_X) || defined(Z_PROBE_ALLEN_KEY_STOW_3_Y) || defined(Z_PROBE_ALLEN_KEY_STOW_3_Z)
# ifndef Z_PROBE_ALLEN_KEY_STOW_3_X
# define Z_PROBE_ALLEN_KEY_STOW_3_X current_position[X_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_STOW_3_Y
# define Z_PROBE_ALLEN_KEY_STOW_3_Y current_position[Y_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_STOW_3_Z
# define Z_PROBE_ALLEN_KEY_STOW_3_Z current_position[Z_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_STOW_3_FEEDRATE
# define Z_PROBE_ALLEN_KEY_STOW_3_FEEDRATE 0.0
# endif
2017-06-07 14:12:19 +02:00
const float stow_3 [ ] = { Z_PROBE_ALLEN_KEY_STOW_3_X , Z_PROBE_ALLEN_KEY_STOW_3_Y , Z_PROBE_ALLEN_KEY_STOW_3_Z } ;
do_blocking_move_to ( stow_3 , MMM_TO_MMS ( Z_PROBE_ALLEN_KEY_STOW_3_FEEDRATE ) ) ;
2016-07-04 09:46:22 +02:00
# endif
# if defined(Z_PROBE_ALLEN_KEY_STOW_4_X) || defined(Z_PROBE_ALLEN_KEY_STOW_4_Y) || defined(Z_PROBE_ALLEN_KEY_STOW_4_Z)
# ifndef Z_PROBE_ALLEN_KEY_STOW_4_X
# define Z_PROBE_ALLEN_KEY_STOW_4_X current_position[X_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_STOW_4_Y
# define Z_PROBE_ALLEN_KEY_STOW_4_Y current_position[Y_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_STOW_4_Z
# define Z_PROBE_ALLEN_KEY_STOW_4_Z current_position[Z_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_STOW_4_FEEDRATE
# define Z_PROBE_ALLEN_KEY_STOW_4_FEEDRATE 0.0
# endif
2017-06-07 14:12:19 +02:00
const float stow_4 [ ] = { Z_PROBE_ALLEN_KEY_STOW_4_X , Z_PROBE_ALLEN_KEY_STOW_4_Y , Z_PROBE_ALLEN_KEY_STOW_4_Z } ;
do_blocking_move_to ( stow_4 , MMM_TO_MMS ( Z_PROBE_ALLEN_KEY_STOW_4_FEEDRATE ) ) ;
2016-07-04 09:46:22 +02:00
# endif
# if defined(Z_PROBE_ALLEN_KEY_STOW_5_X) || defined(Z_PROBE_ALLEN_KEY_STOW_5_Y) || defined(Z_PROBE_ALLEN_KEY_STOW_5_Z)
# ifndef Z_PROBE_ALLEN_KEY_STOW_5_X
# define Z_PROBE_ALLEN_KEY_STOW_5_X current_position[X_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_STOW_5_Y
# define Z_PROBE_ALLEN_KEY_STOW_5_Y current_position[Y_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_STOW_5_Z
# define Z_PROBE_ALLEN_KEY_STOW_5_Z current_position[Z_AXIS]
# endif
# ifndef Z_PROBE_ALLEN_KEY_STOW_5_FEEDRATE
# define Z_PROBE_ALLEN_KEY_STOW_5_FEEDRATE 0.0
# endif
2017-06-07 14:12:19 +02:00
const float stow_5 [ ] = { Z_PROBE_ALLEN_KEY_STOW_5_X , Z_PROBE_ALLEN_KEY_STOW_5_Y , Z_PROBE_ALLEN_KEY_STOW_5_Z } ;
do_blocking_move_to ( stow_5 , MMM_TO_MMS ( Z_PROBE_ALLEN_KEY_STOW_5_FEEDRATE ) ) ;
2016-07-04 09:46:22 +02:00
# endif
}
2016-10-03 01:18:05 +02:00
2016-07-04 09:46:22 +02:00
# endif
2016-06-21 00:50:51 +02:00
2017-05-07 13:06:06 +02:00
# if ENABLED(PROBING_FANS_OFF)
2017-05-08 21:09:37 +02:00
void fans_pause ( const bool p ) {
if ( p ! = fans_paused ) {
fans_paused = p ;
if ( p )
for ( uint8_t x = 0 ; x < FAN_COUNT ; x + + ) {
paused_fanSpeeds [ x ] = fanSpeeds [ x ] ;
fanSpeeds [ x ] = 0 ;
}
else
for ( uint8_t x = 0 ; x < FAN_COUNT ; x + + )
fanSpeeds [ x ] = paused_fanSpeeds [ x ] ;
}
2017-05-07 13:06:06 +02:00
}
2017-05-08 21:09:37 +02:00
2017-05-08 13:40:50 +02:00
# endif // PROBING_FANS_OFF
2017-05-07 13:06:06 +02:00
2016-06-20 23:31:01 +02:00
# if HAS_BED_PROBE
2017-04-08 08:53:22 +02:00
// TRIGGERED_WHEN_STOWED_TEST can easily be extended to servo probes, ... if needed.
2016-07-06 03:56:17 +02:00
# if ENABLED(PROBE_IS_TRIGGERED_WHEN_STOWED_TEST)
# if ENABLED(Z_MIN_PROBE_ENDSTOP)
# define _TRIGGERED_WHEN_STOWED_TEST (READ(Z_MIN_PROBE_PIN) != Z_MIN_PROBE_ENDSTOP_INVERTING)
# else
# define _TRIGGERED_WHEN_STOWED_TEST (READ(Z_MIN_PIN) != Z_MIN_ENDSTOP_INVERTING)
# endif
# endif
2017-05-07 13:06:06 +02:00
# if QUIET_PROBING
2017-05-08 21:09:37 +02:00
void probing_pause ( const bool p ) {
2017-05-07 13:06:06 +02:00
# if ENABLED(PROBING_HEATERS_OFF)
2017-05-08 21:09:37 +02:00
thermalManager . pause ( p ) ;
2017-05-07 13:06:06 +02:00
# endif
# if ENABLED(PROBING_FANS_OFF)
2017-05-08 21:09:37 +02:00
fans_pause ( p ) ;
2017-05-07 13:06:06 +02:00
# endif
2017-07-13 18:15:59 +02:00
if ( p ) safe_delay (
# if DELAY_BEFORE_PROBING > 25
DELAY_BEFORE_PROBING
# else
25
# endif
) ;
2017-05-07 13:06:06 +02:00
}
2017-05-08 21:09:37 +02:00
# endif // QUIET_PROBING
2017-05-07 13:06:06 +02:00
2016-09-20 21:37:18 +02:00
# if ENABLED(BLTOUCH)
2017-04-29 01:22:31 +02:00
2017-01-22 01:10:02 +01:00
void bltouch_command ( int angle ) {
2017-08-15 06:52:23 +02:00
MOVE_SERVO ( Z_ENDSTOP_SERVO_NR , angle ) ; // Give the BL-Touch the command and wait
2017-04-02 17:55:00 +02:00
safe_delay ( BLTOUCH_DELAY ) ;
2017-01-22 01:10:02 +01:00
}
2017-08-03 05:52:40 +02:00
bool set_bltouch_deployed ( const bool deploy ) {
2017-03-24 17:23:00 +01:00
if ( deploy & & TEST_BLTOUCH ( ) ) { // If BL-Touch says it's triggered
2017-04-29 00:36:31 +02:00
bltouch_command ( BLTOUCH_RESET ) ; // try to reset it.
2017-03-24 17:23:00 +01:00
bltouch_command ( BLTOUCH_DEPLOY ) ; // Also needs to deploy and stow to
2017-04-29 00:36:31 +02:00
bltouch_command ( BLTOUCH_STOW ) ; // clear the triggered condition.
safe_delay ( 1500 ) ; // Wait for internal self-test to complete.
// (Measured completion time was 0.65 seconds
// after reset, deploy, and stow sequence)
2017-03-24 17:23:00 +01:00
if ( TEST_BLTOUCH ( ) ) { // If it still claims to be triggered...
2017-06-09 17:51:23 +02:00
SERIAL_ERROR_START ( ) ;
2017-03-24 17:23:00 +01:00
SERIAL_ERRORLNPGM ( MSG_STOP_BLTOUCH ) ;
stop ( ) ; // punt!
2017-08-03 05:52:40 +02:00
return true ;
2017-03-24 17:23:00 +01:00
}
}
2017-05-01 23:10:39 +02:00
2017-01-22 01:10:02 +01:00
bltouch_command ( deploy ? BLTOUCH_DEPLOY : BLTOUCH_STOW ) ;
2017-05-01 23:10:39 +02:00
2016-10-02 08:48:17 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) {
SERIAL_ECHOPAIR ( " set_bltouch_deployed( " , deploy ) ;
SERIAL_CHAR ( ' ) ' ) ;
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2016-10-02 08:48:17 +02:00
}
# endif
2017-08-03 05:52:40 +02:00
return false ;
2016-09-20 21:37:18 +02:00
}
2017-04-29 01:22:31 +02:00
# endif // BLTOUCH
2016-09-20 21:37:18 +02:00
2016-07-06 03:40:21 +02:00
// returns false for ok and true for failure
2017-03-18 16:15:54 +01:00
bool set_probe_deployed ( bool deploy ) {
2016-06-20 23:31:01 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
2016-07-06 03:40:21 +02:00
if ( DEBUGGING ( LEVELING ) ) {
DEBUG_POS ( " set_probe_deployed " , current_position ) ;
2016-08-26 22:42:16 +02:00
SERIAL_ECHOLNPAIR ( " deploy: " , deploy ) ;
2016-07-06 03:40:21 +02:00
}
2016-06-20 23:31:01 +02:00
# endif
2017-05-01 09:27:52 +02:00
if ( endstops . z_probe_enabled = = deploy ) return false ;
2016-06-24 03:00:20 +02:00
// Make room for probe
2016-09-20 10:49:42 +02:00
do_probe_raise ( _Z_CLEARANCE_DEPLOY_PROBE ) ;
2016-06-24 03:00:20 +02:00
2017-08-03 05:52:40 +02:00
# if ENABLED(Z_PROBE_SLED) || ENABLED(Z_PROBE_ALLEN_KEY)
2017-05-14 22:57:37 +02:00
# if ENABLED(Z_PROBE_SLED)
# define _AUE_ARGS true, false, false
# else
# define _AUE_ARGS
# endif
if ( axis_unhomed_error ( _AUE_ARGS ) ) {
2017-06-09 17:51:23 +02:00
SERIAL_ERROR_START ( ) ;
2017-03-14 18:24:46 +01:00
SERIAL_ERRORLNPGM ( MSG_STOP_UNHOMED ) ;
stop ( ) ;
return true ;
}
2016-06-20 23:31:01 +02:00
# endif
2017-03-04 01:23:31 +01:00
const float oldXpos = current_position [ X_AXIS ] ,
oldYpos = current_position [ Y_AXIS ] ;
2016-06-24 03:00:20 +02:00
2016-07-06 03:56:17 +02:00
# ifdef _TRIGGERED_WHEN_STOWED_TEST
2016-08-06 23:24:05 +02:00
2016-07-06 03:56:17 +02:00
// If endstop is already false, the Z probe is deployed
2016-08-06 23:24:05 +02:00
if ( _TRIGGERED_WHEN_STOWED_TEST = = deploy ) { // closed after the probe specific actions.
// Would a goto be less ugly?
//while (!_TRIGGERED_WHEN_STOWED_TEST) idle(); // would offer the opportunity
// for a triggered when stowed manual probe.
2016-07-06 03:56:17 +02:00
2016-08-06 23:24:05 +02:00
if ( ! deploy ) endstops . enable_z_probe ( false ) ; // Switch off triggered when stowed probes early
// otherwise an Allen-Key probe can't be stowed.
2016-06-20 23:31:01 +02:00
# endif
2017-04-14 23:36:02 +02:00
# if ENABLED(SOLENOID_PROBE)
# if HAS_SOLENOID_1
WRITE ( SOL1_PIN , deploy ) ;
# endif
# elif ENABLED(Z_PROBE_SLED)
2016-08-06 23:24:05 +02:00
dock_sled ( ! deploy ) ;
2016-09-20 21:37:18 +02:00
# elif HAS_Z_SERVO_ENDSTOP && DISABLED(BLTOUCH)
2016-08-06 23:24:05 +02:00
2017-08-15 06:52:23 +02:00
MOVE_SERVO ( Z_ENDSTOP_SERVO_NR , z_servo_angle [ deploy ? 0 : 1 ] ) ;
2016-08-06 23:24:05 +02:00
# elif ENABLED(Z_PROBE_ALLEN_KEY)
deploy ? run_deploy_moves_script ( ) : run_stow_moves_script ( ) ;
# endif
2016-07-06 03:56:17 +02:00
# ifdef _TRIGGERED_WHEN_STOWED_TEST
2016-08-06 23:24:05 +02:00
} // _TRIGGERED_WHEN_STOWED_TEST == deploy
if ( _TRIGGERED_WHEN_STOWED_TEST = = deploy ) { // State hasn't changed?
2016-07-06 03:40:21 +02:00
2016-07-06 03:56:17 +02:00
if ( IsRunning ( ) ) {
2017-06-09 17:51:23 +02:00
SERIAL_ERROR_START ( ) ;
2016-07-06 03:40:21 +02:00
SERIAL_ERRORLNPGM ( " Z-Probe failed " ) ;
2016-07-06 03:56:17 +02:00
LCD_ALERTMESSAGEPGM ( " Err: ZPROBE " ) ;
}
stop ( ) ;
return true ;
2016-08-06 23:24:05 +02:00
} // _TRIGGERED_WHEN_STOWED_TEST == deploy
2016-07-06 03:56:17 +02:00
# endif
2016-07-06 03:40:21 +02:00
do_blocking_move_to ( oldXpos , oldYpos , current_position [ Z_AXIS ] ) ; // return to position before deploy
2016-08-28 02:53:02 +02:00
endstops . enable_z_probe ( deploy ) ;
2016-07-06 03:40:21 +02:00
return false ;
2016-06-20 23:31:01 +02:00
}
2017-08-11 23:39:55 +02:00
/**
* @ brief Used by run_z_probe to do a single Z probe move .
*
* @ param z Z destination
* @ param fr_mm_s Feedrate in mm / s
* @ return true to indicate an error
*/
static bool do_probe_move ( const float z , const float fr_mm_m ) {
2016-08-19 08:10:41 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) DEBUG_POS ( " >>> do_probe_move " , current_position ) ;
# endif
2016-09-20 21:37:18 +02:00
// Deploy BLTouch at the start of any probe
# if ENABLED(BLTOUCH)
2017-08-03 05:52:40 +02:00
if ( set_bltouch_deployed ( true ) ) return true ;
2016-09-20 21:37:18 +02:00
# endif
2017-05-07 13:06:06 +02:00
# if QUIET_PROBING
probing_pause ( true ) ;
# endif
2016-08-19 08:10:41 +02:00
// Move down until probe triggered
2017-08-03 05:52:40 +02:00
do_blocking_move_to_z ( z , MMM_TO_MMS ( fr_mm_m ) ) ;
// Check to see if the probe was triggered
const bool probe_triggered = TEST ( Endstops : : endstop_hit_bits ,
2017-08-11 23:39:55 +02:00
# if ENABLED(Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN)
2017-08-03 05:52:40 +02:00
Z_MIN
# else
Z_MIN_PROBE
# endif
) ;
2016-08-19 08:10:41 +02:00
2017-05-07 13:06:06 +02:00
# if QUIET_PROBING
probing_pause ( false ) ;
# endif
2017-08-03 05:52:40 +02:00
// Retract BLTouch immediately after a probe if it was triggered
2016-09-20 21:37:18 +02:00
# if ENABLED(BLTOUCH)
2017-08-03 05:52:40 +02:00
if ( probe_triggered & & set_bltouch_deployed ( false ) ) return true ;
2016-09-20 21:37:18 +02:00
# endif
2016-08-19 08:10:41 +02:00
// Clear endstop flags
endstops . hit_on_purpose ( ) ;
// Get Z where the steppers were interrupted
set_current_from_steppers_for_axis ( Z_AXIS ) ;
2016-10-23 02:45:13 +02:00
// Tell the planner where we actually are
SYNC_PLAN_POSITION_KINEMATIC ( ) ;
2016-08-19 08:10:41 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) DEBUG_POS ( " <<< do_probe_move " , current_position ) ;
# endif
2017-08-03 05:52:40 +02:00
return ! probe_triggered ;
2016-08-19 08:10:41 +02:00
}
2017-08-11 23:39:55 +02:00
/**
* @ details Used by probe_pt to do a single Z probe .
* Leaves current_position [ Z_AXIS ] at the height where the probe triggered .
*
* @ param short_move Flag for a shorter probe move towards the bed
* @ return The raw Z position where the probe was triggered
*/
static float run_z_probe ( const bool short_move = true ) {
2015-03-30 08:16:12 +02:00
2016-08-19 08:09:03 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) DEBUG_POS ( " >>> run_z_probe " , current_position ) ;
# endif
2016-06-22 00:14:04 +02:00
// Prevent stepper_inactive_time from running out and EXTRUDER_RUNOUT_PREVENT from extruding
2016-03-27 05:36:36 +02:00
refresh_cmd_timeout ( ) ;
2015-11-03 20:14:15 +01:00
double bump probing as a feature
Why double touch probing is not a good thing.
It's widely believed we can get better __probing__ results when using a double touch when probing.
Let's compare to double touch __homing__.
Or better let's begin with single touch __homing__.
We home to find out out position, so our position is unknown.
To find the endstop we have to move into the direction of the endstop.
The maximum way we have to move is a bit longer than the axis length.
When we arrive at the endstop - when it triggers, the stepper pulses are stopped immediately.
It's a sudden stop. No smooth deacceleration is possible.
Depending on the speed and the moving mass we lose steps here.
Only if we approached slow enough (below jerk speed?) we will not lose steps.
Moving a complete axis length, that slow, takes for ever.
To speed up homing, we now make the first approach faster, get a guess about our position,
back up a bit and make a second slower approach to get a exact result without losing steps.
What we do in double touch probing is the same. But the difference here is:
a. we already know where we are
b. if the first approach is to fast we will lose steps here to.
But this time there is no second approach to set the position to 0. We are measuring only.
The lost steps are permanent until we home the next time.
So if you experienced permanently rising values in M48 you now know why. (Too fast, suddenly stopped, first approach)
What can we do to improve probing?
We can use the information about our current position.
We can make a really fast, but deaccelerated, move to a place we know it is a bit before the trigger point.
And then move the rest of the way really slow.
2016-07-30 03:00:49 +02:00
# if ENABLED(PROBE_DOUBLE_TOUCH)
2016-08-19 08:10:41 +02:00
// Do a first probe at the fast speed
2017-08-03 05:52:40 +02:00
if ( do_probe_move ( - 10 , Z_PROBE_SPEED_FAST ) ) return NAN ;
2016-08-19 08:10:41 +02:00
2016-09-28 21:01:59 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
float first_probe_z = current_position [ Z_AXIS ] ;
2016-10-03 00:35:40 +02:00
if ( DEBUGGING ( LEVELING ) ) SERIAL_ECHOLNPAIR ( " 1st Probe Z: " , first_probe_z ) ;
2016-09-28 21:01:59 +02:00
# endif
2017-08-03 05:52:40 +02:00
// move up to make clearance for the probe
do_blocking_move_to_z ( current_position [ Z_AXIS ] + Z_CLEARANCE_BETWEEN_PROBES , MMM_TO_MMS ( Z_PROBE_SPEED_FAST ) ) ;
2016-08-19 08:17:56 +02:00
double bump probing as a feature
Why double touch probing is not a good thing.
It's widely believed we can get better __probing__ results when using a double touch when probing.
Let's compare to double touch __homing__.
Or better let's begin with single touch __homing__.
We home to find out out position, so our position is unknown.
To find the endstop we have to move into the direction of the endstop.
The maximum way we have to move is a bit longer than the axis length.
When we arrive at the endstop - when it triggers, the stepper pulses are stopped immediately.
It's a sudden stop. No smooth deacceleration is possible.
Depending on the speed and the moving mass we lose steps here.
Only if we approached slow enough (below jerk speed?) we will not lose steps.
Moving a complete axis length, that slow, takes for ever.
To speed up homing, we now make the first approach faster, get a guess about our position,
back up a bit and make a second slower approach to get a exact result without losing steps.
What we do in double touch probing is the same. But the difference here is:
a. we already know where we are
b. if the first approach is to fast we will lose steps here to.
But this time there is no second approach to set the position to 0. We are measuring only.
The lost steps are permanent until we home the next time.
So if you experienced permanently rising values in M48 you now know why. (Too fast, suddenly stopped, first approach)
What can we do to improve probing?
We can use the information about our current position.
We can make a really fast, but deaccelerated, move to a place we know it is a bit before the trigger point.
And then move the rest of the way really slow.
2016-07-30 03:00:49 +02:00
# else
2016-08-19 08:17:56 +02:00
2016-08-29 02:13:08 +02:00
// If the nozzle is above the travel height then
// move down quickly before doing the slow probe
2017-08-03 05:52:40 +02:00
float z = Z_CLEARANCE_DEPLOY_PROBE ;
2016-10-03 23:18:04 +02:00
if ( zprobe_zoffset < 0 ) z - = zprobe_zoffset ;
2017-06-26 12:25:57 +02:00
2017-08-03 05:52:40 +02:00
if ( z < current_position [ Z_AXIS ] ) {
2016-08-19 08:17:56 +02:00
2017-08-03 05:52:40 +02:00
// If we don't make it to the z position (i.e. the probe triggered), move up to make clearance for the probe
if ( ! do_probe_move ( z , Z_PROBE_SPEED_FAST ) )
do_blocking_move_to_z ( current_position [ Z_AXIS ] + Z_CLEARANCE_BETWEEN_PROBES , MMM_TO_MMS ( Z_PROBE_SPEED_FAST ) ) ;
}
double bump probing as a feature
Why double touch probing is not a good thing.
It's widely believed we can get better __probing__ results when using a double touch when probing.
Let's compare to double touch __homing__.
Or better let's begin with single touch __homing__.
We home to find out out position, so our position is unknown.
To find the endstop we have to move into the direction of the endstop.
The maximum way we have to move is a bit longer than the axis length.
When we arrive at the endstop - when it triggers, the stepper pulses are stopped immediately.
It's a sudden stop. No smooth deacceleration is possible.
Depending on the speed and the moving mass we lose steps here.
Only if we approached slow enough (below jerk speed?) we will not lose steps.
Moving a complete axis length, that slow, takes for ever.
To speed up homing, we now make the first approach faster, get a guess about our position,
back up a bit and make a second slower approach to get a exact result without losing steps.
What we do in double touch probing is the same. But the difference here is:
a. we already know where we are
b. if the first approach is to fast we will lose steps here to.
But this time there is no second approach to set the position to 0. We are measuring only.
The lost steps are permanent until we home the next time.
So if you experienced permanently rising values in M48 you now know why. (Too fast, suddenly stopped, first approach)
What can we do to improve probing?
We can use the information about our current position.
We can make a really fast, but deaccelerated, move to a place we know it is a bit before the trigger point.
And then move the rest of the way really slow.
2016-07-30 03:00:49 +02:00
# endif
2016-07-18 15:52:41 +02:00
double bump probing as a feature
Why double touch probing is not a good thing.
It's widely believed we can get better __probing__ results when using a double touch when probing.
Let's compare to double touch __homing__.
Or better let's begin with single touch __homing__.
We home to find out out position, so our position is unknown.
To find the endstop we have to move into the direction of the endstop.
The maximum way we have to move is a bit longer than the axis length.
When we arrive at the endstop - when it triggers, the stepper pulses are stopped immediately.
It's a sudden stop. No smooth deacceleration is possible.
Depending on the speed and the moving mass we lose steps here.
Only if we approached slow enough (below jerk speed?) we will not lose steps.
Moving a complete axis length, that slow, takes for ever.
To speed up homing, we now make the first approach faster, get a guess about our position,
back up a bit and make a second slower approach to get a exact result without losing steps.
What we do in double touch probing is the same. But the difference here is:
a. we already know where we are
b. if the first approach is to fast we will lose steps here to.
But this time there is no second approach to set the position to 0. We are measuring only.
The lost steps are permanent until we home the next time.
So if you experienced permanently rising values in M48 you now know why. (Too fast, suddenly stopped, first approach)
What can we do to improve probing?
We can use the information about our current position.
We can make a really fast, but deaccelerated, move to a place we know it is a bit before the trigger point.
And then move the rest of the way really slow.
2016-07-30 03:00:49 +02:00
// move down slowly to find bed
2017-08-11 23:39:55 +02:00
if ( do_probe_move ( - 10 + ( short_move ? 0 : - ( Z_MAX_LENGTH ) ) , Z_PROBE_SPEED_SLOW ) ) return NAN ;
2016-06-28 04:33:37 +02:00
2016-07-18 15:52:41 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
2016-08-19 08:09:03 +02:00
if ( DEBUGGING ( LEVELING ) ) DEBUG_POS ( " <<< run_z_probe " , current_position ) ;
2016-07-18 15:52:41 +02:00
# endif
2016-06-23 01:52:32 +02:00
2016-09-28 21:01:59 +02:00
// Debug: compare probe heights
# if ENABLED(PROBE_DOUBLE_TOUCH) && ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) {
SERIAL_ECHOPAIR ( " 2nd Probe Z: " , current_position [ Z_AXIS ] ) ;
SERIAL_ECHOLNPAIR ( " Discrepancy: " , first_probe_z - current_position [ Z_AXIS ] ) ;
}
# endif
2017-08-03 05:52:40 +02:00
2017-06-26 12:25:57 +02:00
return RAW_CURRENT_POSITION ( Z ) + zprobe_zoffset
# if ENABLED(DELTA)
+ home_offset [ Z_AXIS ] // Account for delta height adjustment
# endif
;
2015-03-30 08:16:12 +02:00
}
2013-09-29 18:20:06 +02:00
2017-04-29 00:36:31 +02:00
/**
* - Move to the given XY
* - Deploy the probe , if not already deployed
* - Probe the bed , get the Z position
* - Depending on the ' stow ' flag
* - Stow the probe , or
* - Raise to the BETWEEN height
* - Return the probed Z position
*/
2017-06-29 16:42:42 +02:00
float probe_pt ( const float & lx , const float & ly , const bool stow , const uint8_t verbose_level , const bool printable = true ) {
2015-08-05 13:40:36 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
2016-03-30 04:50:01 +02:00
if ( DEBUGGING ( LEVELING ) ) {
2017-06-29 16:42:42 +02:00
SERIAL_ECHOPAIR ( " >>> probe_pt( " , lx ) ;
SERIAL_ECHOPAIR ( " , " , ly ) ;
2016-10-03 01:18:05 +02:00
SERIAL_ECHOPAIR ( " , " , stow ? " " : " no " ) ;
SERIAL_ECHOLNPGM ( " stow) " ) ;
2016-04-06 04:04:42 +02:00
DEBUG_POS ( " " , current_position ) ;
2015-08-05 13:40:36 +02:00
}
2015-07-10 01:57:44 +02:00
# endif
2017-06-29 16:42:42 +02:00
const float nx = lx - ( X_PROBE_OFFSET_FROM_EXTRUDER ) , ny = ly - ( Y_PROBE_OFFSET_FROM_EXTRUDER ) ;
2017-08-09 19:31:00 +02:00
if ( printable
? ! position_is_reachable_xy ( nx , ny )
: ! position_is_reachable_by_probe_xy ( lx , ly )
) return NAN ;
2017-05-12 05:33:47 +02:00
2017-03-04 01:23:31 +01:00
const float old_feedrate_mm_s = feedrate_mm_s ;
2016-06-23 01:39:51 +02:00
2016-12-10 13:49:46 +01:00
# if ENABLED(DELTA)
if ( current_position [ Z_AXIS ] > delta_clip_start_height )
do_blocking_move_to_z ( delta_clip_start_height ) ;
# endif
2017-08-03 05:52:40 +02:00
# if HAS_SOFTWARE_ENDSTOPS
// Store the status of the soft endstops and disable if we're probing a non-printable location
static bool enable_soft_endstops = soft_endstops_enabled ;
if ( ! printable ) soft_endstops_enabled = false ;
# endif
2016-06-26 04:19:31 +02:00
2016-08-07 09:20:25 +02:00
feedrate_mm_s = XY_PROBE_FEEDRATE_MM_S ;
2016-09-15 20:34:24 +02:00
// Move the probe to the given XY
2017-06-29 16:42:42 +02:00
do_blocking_move_to_xy ( nx , ny ) ;
2015-03-30 08:16:12 +02:00
2017-08-03 05:52:40 +02:00
float measured_z = NAN ;
if ( ! DEPLOY_PROBE ( ) ) {
measured_z = run_z_probe ( printable ) ;
2016-06-22 00:15:59 +02:00
2017-08-03 05:52:40 +02:00
if ( ! stow )
do_blocking_move_to_z ( current_position [ Z_AXIS ] + Z_CLEARANCE_BETWEEN_PROBES , MMM_TO_MMS ( Z_PROBE_SPEED_FAST ) ) ;
else
if ( STOW_PROBE ( ) ) measured_z = NAN ;
}
2014-02-16 19:38:29 +01:00
2017-08-03 05:52:40 +02:00
# if HAS_SOFTWARE_ENDSTOPS
// Restore the soft endstop status
soft_endstops_enabled = enable_soft_endstops ;
# endif
2014-02-16 19:38:29 +01:00
2015-03-30 08:16:12 +02:00
if ( verbose_level > 2 ) {
2015-05-17 10:49:52 +02:00
SERIAL_PROTOCOLPGM ( " Bed X: " ) ;
2017-06-29 16:42:42 +02:00
SERIAL_PROTOCOL_F ( lx , 3 ) ;
2015-03-30 08:16:12 +02:00
SERIAL_PROTOCOLPGM ( " Y: " ) ;
2017-06-29 16:42:42 +02:00
SERIAL_PROTOCOL_F ( ly , 3 ) ;
2015-03-30 08:16:12 +02:00
SERIAL_PROTOCOLPGM ( " Z: " ) ;
2017-03-31 21:58:40 +02:00
SERIAL_PROTOCOL_F ( measured_z , 3 ) ;
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2015-03-07 19:36:21 +01:00
}
2015-07-10 01:57:44 +02:00
2015-08-05 13:40:36 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
2016-04-06 04:04:42 +02:00
if ( DEBUGGING ( LEVELING ) ) SERIAL_ECHOLNPGM ( " <<< probe_pt " ) ;
2015-07-10 01:57:44 +02:00
# endif
2016-08-07 09:20:25 +02:00
feedrate_mm_s = old_feedrate_mm_s ;
2016-06-23 00:49:05 +02:00
2017-08-11 23:39:55 +02:00
if ( isnan ( measured_z ) ) {
LCD_MESSAGEPGM ( MSG_ERR_PROBING_FAILED ) ;
SERIAL_ERROR_START ( ) ;
SERIAL_ERRORLNPGM ( MSG_ERR_PROBING_FAILED ) ;
}
2015-03-30 08:16:12 +02:00
return measured_z ;
2015-03-07 19:36:21 +01:00
}
2016-06-24 04:00:29 +02:00
# endif // HAS_BED_PROBE
2016-06-22 23:03:22 +02:00
2017-05-01 23:13:09 +02:00
# if HAS_LEVELING
2017-05-28 02:29:29 +02:00
bool leveling_is_valid ( ) {
return
# if ENABLED(MESH_BED_LEVELING)
mbl . has_mesh ( )
# elif ENABLED(AUTO_BED_LEVELING_BILINEAR)
! ! bilinear_grid_spacing [ X_AXIS ]
# elif ENABLED(AUTO_BED_LEVELING_UBL)
true
# else // 3POINT, LINEAR
true
# endif
;
}
bool leveling_is_active ( ) {
return
# if ENABLED(MESH_BED_LEVELING)
mbl . active ( )
# elif ENABLED(AUTO_BED_LEVELING_UBL)
ubl . state . active
# else
planner . abl_enabled
# endif
;
}
2016-09-29 07:59:06 +02:00
/**
* Turn bed leveling on or off , fixing the current
* position as - needed .
*
* Disable : Current position = physical position
* Enable : Current position = " unleveled " physical position
*/
2017-05-28 02:29:29 +02:00
void set_bed_leveling_enabled ( const bool enable /*=true*/ ) {
# if ENABLED(AUTO_BED_LEVELING_BILINEAR)
const bool can_change = ( ! enable | | leveling_is_valid ( ) ) ;
# else
constexpr bool can_change = true ;
# endif
2016-09-29 07:59:06 +02:00
2017-05-28 02:29:29 +02:00
if ( can_change & & enable ! = leveling_is_active ( ) ) {
# if ENABLED(MESH_BED_LEVELING)
2016-09-29 07:59:06 +02:00
2016-12-05 08:53:36 +01:00
if ( ! enable )
planner . apply_leveling ( current_position [ X_AXIS ] , current_position [ Y_AXIS ] , current_position [ Z_AXIS ] ) ;
2017-05-28 02:29:29 +02:00
const bool enabling = enable & & leveling_is_valid ( ) ;
mbl . set_active ( enabling ) ;
if ( enabling ) planner . unapply_leveling ( current_position ) ;
2016-12-05 08:53:36 +01:00
2017-05-28 02:29:29 +02:00
# elif ENABLED(AUTO_BED_LEVELING_UBL)
# if PLANNER_LEVELING
2017-06-04 00:11:43 +02:00
if ( ubl . state . active ) { // leveling from on to off
// change unleveled current_position to physical current_position without moving steppers.
2017-05-12 08:05:11 +02:00
planner . apply_leveling ( current_position [ X_AXIS ] , current_position [ Y_AXIS ] , current_position [ Z_AXIS ] ) ;
2017-06-04 00:11:43 +02:00
ubl . state . active = false ; // disable only AFTER calling apply_leveling
}
else { // leveling from off to on
ubl . state . active = true ; // enable BEFORE calling unapply_leveling, otherwise ignored
// change physical current_position to unleveled current_position without moving steppers.
2017-06-04 18:30:03 +02:00
planner . unapply_leveling ( current_position ) ;
2017-06-04 00:11:43 +02:00
}
# else
ubl . state . active = enable ; // just flip the bit, current_position will be wrong until next move.
2017-05-28 02:29:29 +02:00
# endif
2016-09-29 07:59:06 +02:00
2017-05-28 02:29:29 +02:00
# else // ABL
2017-04-21 08:34:03 +02:00
# if ENABLED(AUTO_BED_LEVELING_BILINEAR)
// Force bilinear_z_offset to re-calculate next time
const float reset [ XYZ ] = { - 9999.999 , - 9999.999 , 0 } ;
( void ) bilinear_z_offset ( reset ) ;
# endif
2017-06-04 18:30:03 +02:00
// Enable or disable leveling compensation in the planner
2016-10-11 00:05:40 +02:00
planner . abl_enabled = enable ;
2017-06-04 18:30:03 +02:00
2016-10-11 00:05:40 +02:00
if ( ! enable )
2017-06-04 18:30:03 +02:00
// When disabling just get the current position from the steppers.
// This will yield the smallest error when first converted back to steps.
2016-09-29 07:59:06 +02:00
set_current_from_steppers_for_axis (
# if ABL_PLANAR
ALL_AXES
# else
Z_AXIS
# endif
) ;
else
2017-06-04 18:30:03 +02:00
// When enabling, remove compensation from the current position,
// so compensation will give the right stepper counts.
2016-09-29 07:59:06 +02:00
planner . unapply_leveling ( current_position ) ;
2017-05-28 02:29:29 +02:00
2017-06-04 18:30:03 +02:00
# endif // ABL
2017-05-28 02:29:29 +02:00
}
2016-09-29 07:59:06 +02:00
}
2016-11-26 06:32:47 +01:00
# if ENABLED(ENABLE_LEVELING_FADE_HEIGHT)
void set_z_fade_height ( const float zfh ) {
2017-06-04 18:30:03 +02:00
const bool level_active = leveling_is_active ( ) ;
2017-06-04 00:11:43 +02:00
# if ENABLED(AUTO_BED_LEVELING_UBL)
2017-06-04 18:30:03 +02:00
if ( level_active )
2017-06-04 00:11:43 +02:00
set_bed_leveling_enabled ( false ) ; // turn off before changing fade height for proper apply/unapply leveling to maintain current_position
planner . z_fade_height = zfh ;
planner . inverse_z_fade_height = RECIPROCAL ( zfh ) ;
2017-06-04 18:30:03 +02:00
if ( level_active )
2017-06-04 00:11:43 +02:00
set_bed_leveling_enabled ( true ) ; // turn back on after changing fade height
# else
planner . z_fade_height = zfh ;
planner . inverse_z_fade_height = RECIPROCAL ( zfh ) ;
2017-06-04 18:30:03 +02:00
if ( level_active ) {
2017-06-04 00:11:43 +02:00
set_current_from_steppers_for_axis (
# if ABL_PLANAR
ALL_AXES
# else
Z_AXIS
# endif
) ;
}
# endif
2016-11-26 06:32:47 +01:00
}
# endif // LEVELING_FADE_HEIGHT
2016-06-22 23:03:22 +02:00
2016-09-13 09:11:45 +02:00
/**
* Reset calibration results to zero .
*/
void reset_bed_level ( ) {
2016-12-10 08:54:09 +01:00
set_bed_leveling_enabled ( false ) ;
2016-09-29 07:59:06 +02:00
# if ENABLED(MESH_BED_LEVELING)
2017-05-28 02:29:29 +02:00
if ( leveling_is_valid ( ) ) {
2016-09-29 07:59:06 +02:00
mbl . reset ( ) ;
mbl . set_has_mesh ( false ) ;
}
# else
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) SERIAL_ECHOLNPGM ( " reset_bed_level " ) ;
# endif
# if ABL_PLANAR
planner . bed_level_matrix . set_to_identity ( ) ;
# elif ENABLED(AUTO_BED_LEVELING_BILINEAR)
2016-12-10 08:54:09 +01:00
bilinear_start [ X_AXIS ] = bilinear_start [ Y_AXIS ] =
bilinear_grid_spacing [ X_AXIS ] = bilinear_grid_spacing [ Y_AXIS ] = 0 ;
2017-04-06 05:29:44 +02:00
for ( uint8_t x = 0 ; x < GRID_MAX_POINTS_X ; x + + )
for ( uint8_t y = 0 ; y < GRID_MAX_POINTS_Y ; y + + )
2017-04-22 05:46:19 +02:00
z_values [ x ] [ y ] = NAN ;
2017-03-23 21:56:38 +01:00
# elif ENABLED(AUTO_BED_LEVELING_UBL)
ubl . reset ( ) ;
2016-09-29 07:59:06 +02:00
# endif
2016-09-13 09:11:45 +02:00
# endif
}
2016-06-22 23:03:22 +02:00
2017-05-01 23:13:09 +02:00
# endif // HAS_LEVELING
2016-09-12 09:52:52 +02:00
2017-04-05 09:55:25 +02:00
# if ENABLED(AUTO_BED_LEVELING_BILINEAR) || ENABLED(MESH_BED_LEVELING)
2017-04-29 00:36:31 +02:00
/**
* Enable to produce output in JSON format suitable
* for SCAD or JavaScript mesh visualizers .
*
* Visualize meshes in OpenSCAD using the included script .
*
* buildroot / shared / scripts / MarlinMesh . scad
*/
2017-04-05 09:55:25 +02:00
//#define SCAD_MESH_OUTPUT
/**
* Print calibration results for plotting or manual frame adjustment .
*/
static void print_2d_array ( const uint8_t sx , const uint8_t sy , const uint8_t precision , float ( * fn ) ( const uint8_t , const uint8_t ) ) {
# ifndef SCAD_MESH_OUTPUT
for ( uint8_t x = 0 ; x < sx ; x + + ) {
for ( uint8_t i = 0 ; i < precision + 2 + ( x < 10 ? 1 : 0 ) ; i + + )
SERIAL_PROTOCOLCHAR ( ' ' ) ;
SERIAL_PROTOCOL ( ( int ) x ) ;
}
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2017-04-05 09:55:25 +02:00
# endif
# ifdef SCAD_MESH_OUTPUT
SERIAL_PROTOCOLLNPGM ( " measured_z = [ " ) ; // open 2D array
# endif
for ( uint8_t y = 0 ; y < sy ; y + + ) {
# ifdef SCAD_MESH_OUTPUT
2017-05-29 22:44:20 +02:00
SERIAL_PROTOCOLPGM ( " [ " ) ; // open sub-array
2017-04-05 09:55:25 +02:00
# else
if ( y < 10 ) SERIAL_PROTOCOLCHAR ( ' ' ) ;
SERIAL_PROTOCOL ( ( int ) y ) ;
# endif
for ( uint8_t x = 0 ; x < sx ; x + + ) {
SERIAL_PROTOCOLCHAR ( ' ' ) ;
const float offset = fn ( x , y ) ;
2017-04-09 20:28:46 +02:00
if ( ! isnan ( offset ) ) {
2017-04-05 09:55:25 +02:00
if ( offset > = 0 ) SERIAL_PROTOCOLCHAR ( ' + ' ) ;
SERIAL_PROTOCOL_F ( offset , precision ) ;
}
else {
# ifdef SCAD_MESH_OUTPUT
for ( uint8_t i = 3 ; i < precision + 3 ; i + + )
SERIAL_PROTOCOLCHAR ( ' ' ) ;
SERIAL_PROTOCOLPGM ( " NAN " ) ;
# else
for ( uint8_t i = 0 ; i < precision + 3 ; i + + )
SERIAL_PROTOCOLCHAR ( i ? ' = ' : ' ' ) ;
# endif
}
# ifdef SCAD_MESH_OUTPUT
if ( x < sx - 1 ) SERIAL_PROTOCOLCHAR ( ' , ' ) ;
# endif
}
# ifdef SCAD_MESH_OUTPUT
SERIAL_PROTOCOLCHAR ( ' ' ) ;
SERIAL_PROTOCOLCHAR ( ' ] ' ) ; // close sub-array
if ( y < sy - 1 ) SERIAL_PROTOCOLCHAR ( ' , ' ) ;
# endif
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2017-04-05 09:55:25 +02:00
}
# ifdef SCAD_MESH_OUTPUT
2017-05-29 22:44:20 +02:00
SERIAL_PROTOCOLPGM ( " ]; " ) ; // close 2D array
2017-04-05 09:55:25 +02:00
# endif
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2017-04-05 09:55:25 +02:00
}
# endif
2016-09-26 06:17:39 +02:00
# if ENABLED(AUTO_BED_LEVELING_BILINEAR)
2015-03-07 19:36:21 +01:00
2016-09-13 09:11:45 +02:00
/**
2016-09-15 10:40:34 +02:00
* Extrapolate a single point from its neighbors
2016-09-13 09:11:45 +02:00
*/
2017-05-03 08:47:16 +02:00
static void extrapolate_one_point ( const uint8_t x , const uint8_t y , const int8_t xdir , const int8_t ydir ) {
2016-09-27 06:38:03 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) {
SERIAL_ECHOPGM ( " Extrapolate [ " ) ;
if ( x < 10 ) SERIAL_CHAR ( ' ' ) ;
SERIAL_ECHO ( ( int ) x ) ;
SERIAL_CHAR ( xdir ? ( xdir > 0 ? ' + ' : ' - ' ) : ' ' ) ;
SERIAL_CHAR ( ' ' ) ;
if ( y < 10 ) SERIAL_CHAR ( ' ' ) ;
SERIAL_ECHO ( ( int ) y ) ;
SERIAL_CHAR ( ydir ? ( ydir > 0 ? ' + ' : ' - ' ) : ' ' ) ;
2016-10-07 00:04:22 +02:00
SERIAL_CHAR ( ' ] ' ) ;
2016-09-27 06:38:03 +02:00
}
# endif
2017-04-22 05:46:19 +02:00
if ( ! isnan ( z_values [ x ] [ y ] ) ) {
2016-09-27 06:38:03 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) SERIAL_ECHOLNPGM ( " (done) " ) ;
# endif
return ; // Don't overwrite good values.
}
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2016-09-27 06:38:03 +02:00
// Get X neighbors, Y neighbors, and XY neighbors
2017-05-03 08:47:16 +02:00
const uint8_t x1 = x + xdir , y1 = y + ydir , x2 = x1 + xdir , y2 = y1 + ydir ;
float a1 = z_values [ x1 ] [ y ] , a2 = z_values [ x2 ] [ y ] ,
b1 = z_values [ x ] [ y1 ] , b2 = z_values [ x ] [ y2 ] ,
c1 = z_values [ x1 ] [ y1 ] , c2 = z_values [ x2 ] [ y2 ] ;
2016-09-27 06:38:03 +02:00
// Treat far unprobed points as zero, near as equal to far
2017-04-09 20:28:46 +02:00
if ( isnan ( a2 ) ) a2 = 0.0 ; if ( isnan ( a1 ) ) a1 = a2 ;
if ( isnan ( b2 ) ) b2 = 0.0 ; if ( isnan ( b1 ) ) b1 = b2 ;
if ( isnan ( c2 ) ) c2 = 0.0 ; if ( isnan ( c1 ) ) c1 = c2 ;
2016-09-27 06:38:03 +02:00
2017-03-04 01:23:31 +01:00
const float a = 2 * a1 - a2 , b = 2 * b1 - b2 , c = 2 * c1 - c2 ;
2016-09-27 06:38:03 +02:00
2016-12-14 08:41:26 +01:00
// Take the average instead of the median
2017-04-22 05:46:19 +02:00
z_values [ x ] [ y ] = ( a + b + c ) / 3.0 ;
2016-09-27 06:38:03 +02:00
2016-09-13 09:11:45 +02:00
// Median is robust (ignores outliers).
2017-04-22 05:46:19 +02:00
// z_values[x][y] = (a < b) ? ((b < c) ? b : (c < a) ? a : c)
2016-09-27 06:38:03 +02:00
// : ((c < b) ? b : (a < c) ? a : c);
2016-09-13 09:11:45 +02:00
}
2016-10-14 21:44:28 +02:00
//Enable this if your SCARA uses 180° of total area
//#define EXTRAPOLATE_FROM_EDGE
2016-09-27 06:38:03 +02:00
# if ENABLED(EXTRAPOLATE_FROM_EDGE)
2017-04-06 05:29:44 +02:00
# if GRID_MAX_POINTS_X < GRID_MAX_POINTS_Y
2016-09-27 06:38:03 +02:00
# define HALF_IN_X
2017-04-06 05:29:44 +02:00
# elif GRID_MAX_POINTS_Y < GRID_MAX_POINTS_X
2016-09-27 06:38:03 +02:00
# define HALF_IN_Y
# endif
# endif
2016-09-13 09:11:45 +02:00
/**
* Fill in the unprobed points ( corners of circular print surface )
* using linear extrapolation , away from the center .
*/
static void extrapolate_unprobed_bed_level ( ) {
2016-09-27 06:38:03 +02:00
# ifdef HALF_IN_X
2017-05-03 08:47:16 +02:00
constexpr uint8_t ctrx2 = 0 , xlen = GRID_MAX_POINTS_X - 1 ;
2016-09-27 06:38:03 +02:00
# else
2017-05-03 08:47:16 +02:00
constexpr uint8_t ctrx1 = ( GRID_MAX_POINTS_X - 1 ) / 2 , // left-of-center
ctrx2 = ( GRID_MAX_POINTS_X ) / 2 , // right-of-center
xlen = ctrx1 ;
2016-09-27 06:38:03 +02:00
# endif
# ifdef HALF_IN_Y
2017-05-03 08:47:16 +02:00
constexpr uint8_t ctry2 = 0 , ylen = GRID_MAX_POINTS_Y - 1 ;
2016-09-27 06:38:03 +02:00
# else
2017-05-03 08:47:16 +02:00
constexpr uint8_t ctry1 = ( GRID_MAX_POINTS_Y - 1 ) / 2 , // top-of-center
ctry2 = ( GRID_MAX_POINTS_Y ) / 2 , // bottom-of-center
ylen = ctry1 ;
2016-09-27 06:38:03 +02:00
# endif
for ( uint8_t xo = 0 ; xo < = xlen ; xo + + )
for ( uint8_t yo = 0 ; yo < = ylen ; yo + + ) {
uint8_t x2 = ctrx2 + xo , y2 = ctry2 + yo ;
# ifndef HALF_IN_X
2017-03-04 01:23:31 +01:00
const uint8_t x1 = ctrx1 - xo ;
2016-09-27 06:38:03 +02:00
# endif
# ifndef HALF_IN_Y
2017-03-04 01:23:31 +01:00
const uint8_t y1 = ctry1 - yo ;
2016-09-27 06:38:03 +02:00
# ifndef HALF_IN_X
extrapolate_one_point ( x1 , y1 , + 1 , + 1 ) ; // left-below + +
# endif
extrapolate_one_point ( x2 , y1 , - 1 , + 1 ) ; // right-below - +
# endif
# ifndef HALF_IN_X
extrapolate_one_point ( x1 , y2 , + 1 , - 1 ) ; // left-above + -
# endif
extrapolate_one_point ( x2 , y2 , - 1 , - 1 ) ; // right-above - -
2015-03-30 08:16:12 +02:00
}
2016-09-27 06:38:03 +02:00
2016-09-13 09:11:45 +02:00
}
2015-03-07 19:36:21 +01:00
2016-12-23 17:00:04 +01:00
static void print_bilinear_leveling_grid ( ) {
SERIAL_ECHOLNPGM ( " Bilinear Leveling Grid: " ) ;
2017-04-06 05:29:44 +02:00
print_2d_array ( GRID_MAX_POINTS_X , GRID_MAX_POINTS_Y , 3 ,
2017-04-22 05:46:19 +02:00
[ ] ( const uint8_t ix , const uint8_t iy ) { return z_values [ ix ] [ iy ] ; }
2016-12-23 17:00:04 +01:00
) ;
}
2016-11-05 17:01:26 +01:00
# if ENABLED(ABL_BILINEAR_SUBDIVISION)
2016-12-23 17:00:04 +01:00
2017-04-06 05:29:44 +02:00
# define ABL_GRID_POINTS_VIRT_X (GRID_MAX_POINTS_X - 1) * (BILINEAR_SUBDIVISIONS) + 1
# define ABL_GRID_POINTS_VIRT_Y (GRID_MAX_POINTS_Y - 1) * (BILINEAR_SUBDIVISIONS) + 1
# define ABL_TEMP_POINTS_X (GRID_MAX_POINTS_X + 2)
# define ABL_TEMP_POINTS_Y (GRID_MAX_POINTS_Y + 2)
2017-04-22 05:46:19 +02:00
float z_values_virt [ ABL_GRID_POINTS_VIRT_X ] [ ABL_GRID_POINTS_VIRT_Y ] ;
2016-11-05 17:01:26 +01:00
int bilinear_grid_spacing_virt [ 2 ] = { 0 } ;
2017-04-21 08:34:03 +02:00
float bilinear_grid_factor_virt [ 2 ] = { 0 } ;
2016-11-05 17:01:26 +01:00
2017-07-24 01:47:11 +02:00
static void print_bilinear_leveling_grid_virt ( ) {
2016-11-05 17:01:26 +01:00
SERIAL_ECHOLNPGM ( " Subdivided with CATMULL ROM Leveling Grid: " ) ;
2016-12-23 17:00:04 +01:00
print_2d_array ( ABL_GRID_POINTS_VIRT_X , ABL_GRID_POINTS_VIRT_Y , 5 ,
2017-04-22 05:46:19 +02:00
[ ] ( const uint8_t ix , const uint8_t iy ) { return z_values_virt [ ix ] [ iy ] ; }
2016-12-23 17:00:04 +01:00
) ;
2016-11-05 17:01:26 +01:00
}
2016-12-23 17:00:04 +01:00
2016-12-23 15:02:39 +01:00
# define LINEAR_EXTRAPOLATION(E, I) ((E) * 2 - (I))
float bed_level_virt_coord ( const uint8_t x , const uint8_t y ) {
uint8_t ep = 0 , ip = 1 ;
if ( ! x | | x = = ABL_TEMP_POINTS_X - 1 ) {
if ( x ) {
2017-04-06 05:29:44 +02:00
ep = GRID_MAX_POINTS_X - 1 ;
ip = GRID_MAX_POINTS_X - 2 ;
2016-12-23 15:02:39 +01:00
}
2017-03-31 16:00:49 +02:00
if ( WITHIN ( y , 1 , ABL_TEMP_POINTS_Y - 2 ) )
2016-12-23 15:02:39 +01:00
return LINEAR_EXTRAPOLATION (
2017-04-22 05:46:19 +02:00
z_values [ ep ] [ y - 1 ] ,
z_values [ ip ] [ y - 1 ]
2016-12-23 15:02:39 +01:00
) ;
else
return LINEAR_EXTRAPOLATION (
bed_level_virt_coord ( ep + 1 , y ) ,
bed_level_virt_coord ( ip + 1 , y )
2016-11-05 17:01:26 +01:00
) ;
}
2016-12-23 15:02:39 +01:00
if ( ! y | | y = = ABL_TEMP_POINTS_Y - 1 ) {
if ( y ) {
2017-04-06 05:29:44 +02:00
ep = GRID_MAX_POINTS_Y - 1 ;
ip = GRID_MAX_POINTS_Y - 2 ;
2016-12-23 15:02:39 +01:00
}
2017-03-31 16:00:49 +02:00
if ( WITHIN ( x , 1 , ABL_TEMP_POINTS_X - 2 ) )
2016-12-23 15:02:39 +01:00
return LINEAR_EXTRAPOLATION (
2017-04-22 05:46:19 +02:00
z_values [ x - 1 ] [ ep ] ,
z_values [ x - 1 ] [ ip ]
2016-12-23 15:02:39 +01:00
) ;
else
return LINEAR_EXTRAPOLATION (
bed_level_virt_coord ( x , ep + 1 ) ,
bed_level_virt_coord ( x , ip + 1 )
2016-11-05 17:01:26 +01:00
) ;
}
2017-04-22 05:46:19 +02:00
return z_values [ x - 1 ] [ y - 1 ] ;
2016-11-05 17:01:26 +01:00
}
2016-12-23 17:00:04 +01:00
2016-11-05 17:01:26 +01:00
static float bed_level_virt_cmr ( const float p [ 4 ] , const uint8_t i , const float t ) {
return (
p [ i - 1 ] * - t * sq ( 1 - t )
+ p [ i ] * ( 2 - 5 * sq ( t ) + 3 * t * sq ( t ) )
+ p [ i + 1 ] * t * ( 1 + 4 * t - 3 * sq ( t ) )
- p [ i + 2 ] * sq ( t ) * ( 1 - t )
) * 0.5 ;
}
2016-12-23 17:00:04 +01:00
2016-11-05 17:01:26 +01:00
static float bed_level_virt_2cmr ( const uint8_t x , const uint8_t y , const float & tx , const float & ty ) {
float row [ 4 ] , column [ 4 ] ;
for ( uint8_t i = 0 ; i < 4 ; i + + ) {
2016-12-23 15:02:39 +01:00
for ( uint8_t j = 0 ; j < 4 ; j + + ) {
column [ j ] = bed_level_virt_coord ( i + x - 1 , j + y - 1 ) ;
}
2016-11-05 17:01:26 +01:00
row [ i ] = bed_level_virt_cmr ( column , 1 , ty ) ;
}
return bed_level_virt_cmr ( row , 1 , tx ) ;
}
2016-12-23 17:00:04 +01:00
2016-12-10 07:17:49 +01:00
void bed_level_virt_interpolate ( ) {
2017-04-02 16:55:38 +02:00
bilinear_grid_spacing_virt [ X_AXIS ] = bilinear_grid_spacing [ X_AXIS ] / ( BILINEAR_SUBDIVISIONS ) ;
bilinear_grid_spacing_virt [ Y_AXIS ] = bilinear_grid_spacing [ Y_AXIS ] / ( BILINEAR_SUBDIVISIONS ) ;
2017-04-21 08:34:03 +02:00
bilinear_grid_factor_virt [ X_AXIS ] = RECIPROCAL ( bilinear_grid_spacing_virt [ X_AXIS ] ) ;
bilinear_grid_factor_virt [ Y_AXIS ] = RECIPROCAL ( bilinear_grid_spacing_virt [ Y_AXIS ] ) ;
2017-04-06 05:29:44 +02:00
for ( uint8_t y = 0 ; y < GRID_MAX_POINTS_Y ; y + + )
for ( uint8_t x = 0 ; x < GRID_MAX_POINTS_X ; x + + )
2016-11-05 17:01:26 +01:00
for ( uint8_t ty = 0 ; ty < BILINEAR_SUBDIVISIONS ; ty + + )
for ( uint8_t tx = 0 ; tx < BILINEAR_SUBDIVISIONS ; tx + + ) {
2017-04-06 05:29:44 +02:00
if ( ( ty & & y = = GRID_MAX_POINTS_Y - 1 ) | | ( tx & & x = = GRID_MAX_POINTS_X - 1 ) )
2016-11-05 17:01:26 +01:00
continue ;
2017-04-22 05:46:19 +02:00
z_values_virt [ x * ( BILINEAR_SUBDIVISIONS ) + tx ] [ y * ( BILINEAR_SUBDIVISIONS ) + ty ] =
2016-11-05 17:01:26 +01:00
bed_level_virt_2cmr (
x + 1 ,
y + 1 ,
( float ) tx / ( BILINEAR_SUBDIVISIONS ) ,
( float ) ty / ( BILINEAR_SUBDIVISIONS )
) ;
}
}
# endif // ABL_BILINEAR_SUBDIVISION
2017-04-22 05:45:02 +02:00
// Refresh after other values have been updated
void refresh_bed_level ( ) {
2017-04-21 08:34:03 +02:00
bilinear_grid_factor [ X_AXIS ] = RECIPROCAL ( bilinear_grid_spacing [ X_AXIS ] ) ;
bilinear_grid_factor [ Y_AXIS ] = RECIPROCAL ( bilinear_grid_spacing [ Y_AXIS ] ) ;
2017-04-22 05:45:02 +02:00
# if ENABLED(ABL_BILINEAR_SUBDIVISION)
bed_level_virt_interpolate ( ) ;
# endif
}
2016-09-26 06:17:39 +02:00
# endif // AUTO_BED_LEVELING_BILINEAR
2015-08-12 01:04:40 +02:00
2015-04-01 10:44:13 +02:00
/**
2016-09-15 20:34:24 +02:00
* Home an individual linear axis
2015-04-01 10:44:13 +02:00
*/
2017-06-06 00:41:38 +02:00
static void do_homing_move ( const AxisEnum axis , const float distance , const float fr_mm_s = 0.0 ) {
2016-09-20 21:37:18 +02:00
2016-10-02 08:48:17 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) {
SERIAL_ECHOPAIR ( " >>> do_homing_move( " , axis_codes [ axis ] ) ;
SERIAL_ECHOPAIR ( " , " , distance ) ;
SERIAL_ECHOPAIR ( " , " , fr_mm_s ) ;
SERIAL_CHAR ( ' ) ' ) ;
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2016-10-02 08:48:17 +02:00
}
# endif
2016-09-20 21:37:18 +02:00
# if HOMING_Z_WITH_PROBE && ENABLED(BLTOUCH)
2017-03-04 01:23:31 +01:00
const bool deploy_bltouch = ( axis = = Z_AXIS & & distance < 0 ) ;
2016-09-22 21:27:22 +02:00
if ( deploy_bltouch ) set_bltouch_deployed ( true ) ;
2016-09-20 21:37:18 +02:00
# endif
2017-05-07 13:06:06 +02:00
# if QUIET_PROBING
if ( axis = = Z_AXIS ) probing_pause ( true ) ;
# endif
2016-09-22 01:35:40 +02:00
// Tell the planner we're at Z=0
2016-08-28 02:53:02 +02:00
current_position [ axis ] = 0 ;
2016-09-22 01:35:40 +02:00
# if IS_SCARA
SYNC_PLAN_POSITION_KINEMATIC ( ) ;
current_position [ axis ] = distance ;
inverse_kinematics ( current_position ) ;
2017-06-06 01:49:16 +02:00
planner . buffer_line ( delta [ A_AXIS ] , delta [ B_AXIS ] , delta [ C_AXIS ] , current_position [ E_AXIS ] , fr_mm_s ? fr_mm_s : homing_feedrate ( axis ) , active_extruder ) ;
2016-09-22 01:35:40 +02:00
# else
sync_plan_position ( ) ;
current_position [ axis ] = distance ;
2017-06-06 01:49:16 +02:00
planner . buffer_line ( current_position [ X_AXIS ] , current_position [ Y_AXIS ] , current_position [ Z_AXIS ] , current_position [ E_AXIS ] , fr_mm_s ? fr_mm_s : homing_feedrate ( axis ) , active_extruder ) ;
2016-09-22 01:35:40 +02:00
# endif
2016-08-19 08:11:41 +02:00
stepper . synchronize ( ) ;
2016-09-20 21:37:18 +02:00
2017-05-07 13:06:06 +02:00
# if QUIET_PROBING
if ( axis = = Z_AXIS ) probing_pause ( false ) ;
# endif
2016-09-20 21:37:18 +02:00
# if HOMING_Z_WITH_PROBE && ENABLED(BLTOUCH)
2016-09-22 21:27:22 +02:00
if ( deploy_bltouch ) set_bltouch_deployed ( false ) ;
2016-09-20 21:37:18 +02:00
# endif
2016-08-28 02:53:02 +02:00
endstops . hit_on_purpose ( ) ;
2016-10-02 08:48:17 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) {
SERIAL_ECHOPAIR ( " <<< do_homing_move( " , axis_codes [ axis ] ) ;
SERIAL_CHAR ( ' ) ' ) ;
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2016-10-02 08:48:17 +02:00
}
# endif
2016-08-19 08:11:41 +02:00
}
2017-04-15 05:44:08 +02:00
/**
* TMC2130 specific sensorless homing using stallGuard2 .
* stallGuard2 only works when in spreadCycle mode .
* spreadCycle and stealthChop are mutually exclusive .
*/
# if ENABLED(SENSORLESS_HOMING)
void tmc2130_sensorless_homing ( TMC2130Stepper & st , bool enable = true ) {
# if ENABLED(STEALTHCHOP)
if ( enable ) {
st . coolstep_min_speed ( 1024UL * 1024UL - 1UL ) ;
st . stealthChop ( 0 ) ;
}
else {
st . coolstep_min_speed ( 0 ) ;
st . stealthChop ( 1 ) ;
}
# endif
st . diag1_stall ( enable ? 1 : 0 ) ;
}
# endif
2016-09-15 20:22:23 +02:00
/**
* Home an individual " raw axis " to its endstop .
* This applies to XYZ on Cartesian and Core robots , and
* to the individual ABC steppers on DELTA and SCARA .
*
* At the end of the procedure the axis is marked as
* homed and the current position of that axis is updated .
* Kinematic robots should wait till all axes are homed
* before updating the current position .
*/
2015-04-01 10:44:13 +02:00
# define HOMEAXIS(LETTER) homeaxis(LETTER##_AXIS)
2017-03-04 01:23:31 +01:00
static void homeaxis ( const AxisEnum axis ) {
2016-07-08 15:59:33 +02:00
2016-09-15 22:43:06 +02:00
# if IS_SCARA
// Only Z homing (with probe) is permitted
if ( axis ! = Z_AXIS ) { BUZZ ( 100 , 880 ) ; return ; }
# else
# define CAN_HOME(A) \
( axis = = A # # _AXIS & & ( ( A # # _MIN_PIN > - 1 & & A # # _HOME_DIR < 0 ) | | ( A # # _MAX_PIN > - 1 & & A # # _HOME_DIR > 0 ) ) )
if ( ! CAN_HOME ( X ) & & ! CAN_HOME ( Y ) & & ! CAN_HOME ( Z ) ) return ;
# endif
2016-07-08 15:59:33 +02:00
2015-08-05 13:40:36 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
2016-03-30 04:50:01 +02:00
if ( DEBUGGING ( LEVELING ) ) {
2016-08-19 08:09:03 +02:00
SERIAL_ECHOPAIR ( " >>> homeaxis( " , axis_codes [ axis ] ) ;
2016-10-02 08:48:17 +02:00
SERIAL_CHAR ( ' ) ' ) ;
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2015-08-05 13:40:36 +02:00
}
2015-07-10 01:57:44 +02:00
# endif
2013-06-06 16:36:52 +02:00
2017-03-04 01:23:31 +01:00
const int axis_home_dir =
2016-07-08 15:59:33 +02:00
# if ENABLED(DUAL_X_CARRIAGE)
( axis = = X_AXIS ) ? x_home_dir ( active_extruder ) :
2016-06-02 04:48:17 +02:00
# endif
2016-07-08 15:59:33 +02:00
home_dir ( axis ) ;
2016-03-31 02:19:54 +02:00
2016-07-08 15:59:33 +02:00
// Homing Z towards the bed? Deploy the Z probe or endstop.
2016-08-30 20:29:13 +02:00
# if HOMING_Z_WITH_PROBE
2016-09-13 00:49:35 +02:00
if ( axis = = Z_AXIS & & DEPLOY_PROBE ( ) ) return ;
2016-07-08 15:59:33 +02:00
# endif
2016-07-05 10:42:33 +02:00
2016-07-08 15:59:33 +02:00
// Set a flag for Z motor locking
# if ENABLED(Z_DUAL_ENDSTOPS)
if ( axis = = Z_AXIS ) stepper . set_homing_flag ( true ) ;
# endif
2013-06-07 00:49:25 +02:00
2017-04-15 05:44:08 +02:00
// Disable stealthChop if used. Enable diag1 pin on driver.
# if ENABLED(SENSORLESS_HOMING)
# if ENABLED(X_IS_TMC2130)
if ( axis = = X_AXIS ) tmc2130_sensorless_homing ( stepperX ) ;
# endif
# if ENABLED(Y_IS_TMC2130)
if ( axis = = Y_AXIS ) tmc2130_sensorless_homing ( stepperY ) ;
# endif
# endif
2016-09-22 13:23:46 +02:00
// Fast move towards endstop until triggered
2016-10-02 08:48:17 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) SERIAL_ECHOLNPGM ( " Home 1 Fast: " ) ;
# endif
2016-09-22 13:23:46 +02:00
do_homing_move ( axis , 1.5 * max_length ( axis ) * axis_home_dir ) ;
2016-09-22 22:21:53 +02:00
// When homing Z with probe respect probe clearance
const float bump = axis_home_dir * (
# if HOMING_Z_WITH_PROBE
( axis = = Z_AXIS ) ? max ( Z_CLEARANCE_BETWEEN_PROBES , home_bump_mm ( Z_AXIS ) ) :
# endif
home_bump_mm ( axis )
) ;
2016-09-22 13:23:46 +02:00
// If a second homing move is configured...
if ( bump ) {
// Move away from the endstop by the axis HOME_BUMP_MM
2016-10-02 08:48:17 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) SERIAL_ECHOLNPGM ( " Move Away: " ) ;
# endif
2016-09-22 13:23:46 +02:00
do_homing_move ( axis , - bump ) ;
2016-10-02 08:48:17 +02:00
2016-09-22 13:23:46 +02:00
// Slow move towards endstop until triggered
2016-10-02 08:48:17 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) SERIAL_ECHOLNPGM ( " Home 2 Slow: " ) ;
# endif
2016-09-22 13:23:46 +02:00
do_homing_move ( axis , 2 * bump , get_homing_bump_feedrate ( axis ) ) ;
}
2015-07-10 01:57:44 +02:00
2016-07-08 15:59:33 +02:00
# if ENABLED(Z_DUAL_ENDSTOPS)
if ( axis = = Z_AXIS ) {
2017-06-20 05:39:23 +02:00
float adj = FABS ( z_endstop_adj ) ;
2016-07-08 15:59:33 +02:00
bool lockZ1 ;
if ( axis_home_dir > 0 ) {
adj = - adj ;
lockZ1 = ( z_endstop_adj > 0 ) ;
}
else
lockZ1 = ( z_endstop_adj < 0 ) ;
2015-04-01 10:44:13 +02:00
2016-07-08 15:59:33 +02:00
if ( lockZ1 ) stepper . set_z_lock ( true ) ; else stepper . set_z2_lock ( true ) ;
2015-04-01 10:44:13 +02:00
2016-07-08 15:59:33 +02:00
// Move to the adjusted endstop height
2016-08-19 08:11:41 +02:00
do_homing_move ( axis , adj ) ;
2015-04-01 10:44:13 +02:00
2016-07-08 15:59:33 +02:00
if ( lockZ1 ) stepper . set_z_lock ( false ) ; else stepper . set_z2_lock ( false ) ;
stepper . set_homing_flag ( false ) ;
} // Z_AXIS
# endif
2015-03-24 18:06:44 +01:00
2016-09-15 22:43:06 +02:00
# if IS_SCARA
set_axis_is_at_home ( axis ) ;
SYNC_PLAN_POSITION_KINEMATIC ( ) ;
# elif ENABLED(DELTA)
// Delta has already moved all three towers up in G28
// so here it re-homes each tower in turn.
// Delta homing treats the axes as normal linear axes.
2016-08-28 02:53:02 +02:00
2017-05-21 02:23:39 +02:00
// retrace by the amount specified in endstop_adj + additional 0.1mm in order to have minimum steps
if ( endstop_adj [ axis ] * Z_HOME_DIR < = 0 ) {
2016-07-08 15:59:33 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
2016-10-02 08:48:17 +02:00
if ( DEBUGGING ( LEVELING ) ) SERIAL_ECHOLNPGM ( " endstop_adj: " ) ;
2016-07-08 15:59:33 +02:00
# endif
2017-05-21 02:23:39 +02:00
do_homing_move ( axis , endstop_adj [ axis ] - 0.1 ) ;
2016-07-08 15:59:33 +02:00
}
2016-06-22 01:17:48 +02:00
2016-08-28 02:53:02 +02:00
# else
2015-04-01 10:44:13 +02:00
2016-09-21 23:47:12 +02:00
// For cartesian/core machines,
// set the axis to its home position
2016-08-28 02:53:02 +02:00
set_axis_is_at_home ( axis ) ;
sync_plan_position ( ) ;
2015-07-10 01:57:44 +02:00
2016-08-28 02:53:02 +02:00
destination [ axis ] = current_position [ axis ] ;
2013-06-06 16:36:52 +02:00
2016-08-28 02:53:02 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) DEBUG_POS ( " > AFTER set_axis_is_at_home " , current_position ) ;
# endif
# endif
2015-04-29 04:10:07 +02:00
2017-04-15 05:44:08 +02:00
// Re-enable stealthChop if used. Disable diag1 pin on driver.
# if ENABLED(SENSORLESS_HOMING)
# if ENABLED(X_IS_TMC2130)
if ( axis = = X_AXIS ) tmc2130_sensorless_homing ( stepperX , false ) ;
# endif
# if ENABLED(Y_IS_TMC2130)
if ( axis = = Y_AXIS ) tmc2130_sensorless_homing ( stepperY , false ) ;
# endif
# endif
2016-07-08 15:59:33 +02:00
// Put away the Z probe
2016-08-30 08:40:29 +02:00
# if HOMING_Z_WITH_PROBE
2016-09-13 00:49:35 +02:00
if ( axis = = Z_AXIS & & STOW_PROBE ( ) ) return ;
2016-07-08 15:59:33 +02:00
# endif
2015-07-10 01:57:44 +02:00
2015-08-05 13:40:36 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
2016-03-30 04:50:01 +02:00
if ( DEBUGGING ( LEVELING ) ) {
2016-08-19 08:09:03 +02:00
SERIAL_ECHOPAIR ( " <<< homeaxis( " , axis_codes [ axis ] ) ;
2016-10-02 08:48:17 +02:00
SERIAL_CHAR ( ' ) ' ) ;
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2015-08-05 13:40:36 +02:00
}
2015-07-10 01:57:44 +02:00
# endif
2016-08-19 08:12:41 +02:00
} // homeaxis()
2014-06-23 17:09:57 +02:00
2015-07-31 07:24:43 +02:00
# if ENABLED(FWRETRACT)
2015-04-01 03:52:19 +02:00
2017-07-18 05:01:41 +02:00
/**
* Retract or recover according to firmware settings
*
* This function handles retract / recover moves for G10 and G11 ,
* plus auto - retract moves sent from G0 / G1 when E - only moves are done .
*
* To simplify the logic , doubled retract / recover moves are ignored .
*
* Note : Z lift is done transparently to the planner . Aborting
* a print between G10 and G11 may corrupt the Z position .
*
* Note : Auto - retract will apply the set Z hop in addition to any Z hop
* included in the G - code . Use M207 Z0 to to prevent double hop .
*/
void retract ( const bool retracting
# if EXTRUDERS > 1
, bool swapping = false
# endif
) {
static float hop_height , // Remember where the Z height started
hop_amount = 0.0 ; // Total amount lifted, for use in recover
2015-04-01 03:52:19 +02:00
2017-07-18 05:01:41 +02:00
// Simply never allow two retracts or recovers in a row
if ( retracted [ active_extruder ] = = retracting ) return ;
2016-12-13 17:23:43 +01:00
2017-07-18 05:01:41 +02:00
# if EXTRUDERS < 2
bool swapping = false ;
# endif
if ( ! retracting ) swapping = retracted_swap [ active_extruder ] ;
2015-04-01 03:52:19 +02:00
2017-07-18 05:01:41 +02:00
/* // debugging
SERIAL_ECHOLNPAIR ( " retracting " , retracting ) ;
SERIAL_ECHOLNPAIR ( " swapping " , swapping ) ;
SERIAL_ECHOLNPAIR ( " active extruder " , active_extruder ) ;
for ( uint8_t i = 0 ; i < EXTRUDERS ; + + i ) {
SERIAL_ECHOPAIR ( " retracted[ " , i ) ;
SERIAL_ECHOLNPAIR ( " ] " , retracted [ i ] ) ;
SERIAL_ECHOPAIR ( " retracted_swap[ " , i ) ;
SERIAL_ECHOLNPAIR ( " ] " , retracted_swap [ i ] ) ;
}
SERIAL_ECHOLNPAIR ( " current_position[z] " , current_position [ Z_AXIS ] ) ;
SERIAL_ECHOLNPAIR ( " hop_amount " , hop_amount ) ;
//*/
2016-12-13 17:23:43 +01:00
2017-07-18 05:01:41 +02:00
const bool has_zhop = retract_zlift > 0.01 ; // Is there a hop set?
2015-04-01 03:52:19 +02:00
2017-03-04 01:23:31 +01:00
const float old_feedrate_mm_s = feedrate_mm_s ;
2017-07-21 21:12:41 +02:00
const int16_t old_flow = flow_percentage [ active_extruder ] ;
// Don't apply flow multiplication to retract/recover
flow_percentage [ active_extruder ] = 100 ;
2015-04-01 03:52:19 +02:00
2017-07-18 05:01:41 +02:00
// The current position will be the destination for E and Z moves
2015-04-04 08:42:50 +02:00
set_destination_to_current ( ) ;
2015-04-01 03:52:19 +02:00
if ( retracting ) {
2017-07-18 05:01:41 +02:00
// Remember the Z height since G-code may include its own Z-hop
// For best results turn off Z hop if G-code already includes it
hop_height = destination [ Z_AXIS ] ;
2015-04-01 03:52:19 +02:00
2017-07-18 05:01:41 +02:00
// Retract by moving from a faux E position back to the current E position
2016-08-07 09:20:25 +02:00
feedrate_mm_s = retract_feedrate_mm_s ;
2017-07-27 05:11:22 +02:00
current_position [ E_AXIS ] + = ( swapping ? swap_retract_length : retract_length ) / volumetric_multiplier [ active_extruder ] ;
2016-04-20 22:02:19 +02:00
sync_plan_position_e ( ) ;
2016-06-11 04:23:41 +02:00
prepare_move_to_destination ( ) ;
2015-04-01 03:52:19 +02:00
2017-07-18 05:01:41 +02:00
// Is a Z hop set, and has the hop not yet been done?
if ( has_zhop ) {
hop_amount + = retract_zlift ; // Carriage is raised for retraction hop
current_position [ Z_AXIS ] - = retract_zlift ; // Pretend current pos is lower. Next move raises Z.
SYNC_PLAN_POSITION_KINEMATIC ( ) ; // Set the planner to the new position
prepare_move_to_destination ( ) ; // Raise up to the old current pos
2015-01-09 21:33:02 +01:00
}
2015-04-01 03:52:19 +02:00
}
else {
2017-07-18 05:01:41 +02:00
// If a hop was done and Z hasn't changed, undo the Z hop
if ( hop_amount & & NEAR ( hop_height , destination [ Z_AXIS ] ) ) {
current_position [ Z_AXIS ] + = hop_amount ; // Pretend current pos is higher. Next move lowers Z.
SYNC_PLAN_POSITION_KINEMATIC ( ) ; // Set the planner to the new position
prepare_move_to_destination ( ) ; // Lower to the old current pos
hop_amount = 0.0 ;
2014-06-02 17:02:10 +02:00
}
2015-04-01 03:52:19 +02:00
2017-07-18 05:01:41 +02:00
// A retract multiplier has been added here to get faster swap recovery
feedrate_mm_s = swapping ? swap_retract_recover_feedrate_mm_s : retract_recover_feedrate_mm_s ;
2017-07-27 05:11:22 +02:00
const float move_e = swapping ? swap_retract_length + swap_retract_recover_length : retract_length + retract_recover_length ;
2015-04-01 03:52:19 +02:00
current_position [ E_AXIS ] - = move_e / volumetric_multiplier [ active_extruder ] ;
2016-04-20 22:02:19 +02:00
sync_plan_position_e ( ) ;
2016-12-13 17:23:43 +01:00
2017-07-18 05:01:41 +02:00
prepare_move_to_destination ( ) ; // Recover E
2014-02-17 03:59:04 +01:00
}
2015-04-01 03:52:19 +02:00
2017-07-21 21:12:41 +02:00
// Restore flow and feedrate
flow_percentage [ active_extruder ] = old_flow ;
2016-08-07 09:20:25 +02:00
feedrate_mm_s = old_feedrate_mm_s ;
2017-07-18 05:01:41 +02:00
// The active extruder is now retracted or recovered
2015-04-07 02:04:22 +02:00
retracted [ active_extruder ] = retracting ;
2015-04-01 03:52:19 +02:00
2017-07-18 05:01:41 +02:00
// If swap retract/recover then update the retracted_swap flag too
# if EXTRUDERS > 1
if ( swapping ) retracted_swap [ active_extruder ] = retracting ;
# endif
/* // debugging
SERIAL_ECHOLNPAIR ( " retracting " , retracting ) ;
SERIAL_ECHOLNPAIR ( " swapping " , swapping ) ;
SERIAL_ECHOLNPAIR ( " active_extruder " , active_extruder ) ;
for ( uint8_t i = 0 ; i < EXTRUDERS ; + + i ) {
SERIAL_ECHOPAIR ( " retracted[ " , i ) ;
SERIAL_ECHOLNPAIR ( " ] " , retracted [ i ] ) ;
SERIAL_ECHOPAIR ( " retracted_swap[ " , i ) ;
SERIAL_ECHOLNPAIR ( " ] " , retracted_swap [ i ] ) ;
}
SERIAL_ECHOLNPAIR ( " current_position[z] " , current_position [ Z_AXIS ] ) ;
SERIAL_ECHOLNPAIR ( " hop_amount " , hop_amount ) ;
//*/
2015-04-01 03:52:19 +02:00
} // retract()
# endif // FWRETRACT
2014-02-17 04:00:28 +01:00
2016-06-29 00:06:56 +02:00
# if ENABLED(MIXING_EXTRUDER)
void normalize_mix ( ) {
float mix_total = 0.0 ;
2017-03-04 01:23:31 +01:00
for ( uint8_t i = 0 ; i < MIXING_STEPPERS ; i + + ) mix_total + = RECIPROCAL ( mixing_factor [ i ] ) ;
2016-06-29 00:06:56 +02:00
// Scale all values if they don't add up to ~1.0
2016-11-12 21:33:07 +01:00
if ( ! NEAR ( mix_total , 1.0 ) ) {
2016-06-29 00:06:56 +02:00
SERIAL_PROTOCOLLNPGM ( " Warning: Mix factors must add up to 1.0. Scaling. " ) ;
2017-03-04 01:23:31 +01:00
for ( uint8_t i = 0 ; i < MIXING_STEPPERS ; i + + ) mixing_factor [ i ] * = mix_total ;
2016-06-29 00:06:56 +02:00
}
}
# if ENABLED(DIRECT_MIXING_IN_G1)
// Get mixing parameters from the GCode
// The total "must" be 1.0 (but it will be normalized)
2016-11-15 22:50:54 +01:00
// If no mix factors are given, the old mix is preserved
2016-06-29 00:06:56 +02:00
void gcode_get_mix ( ) {
const char * mixing_codes = " ABCDHI " ;
2016-11-15 22:50:54 +01:00
byte mix_bits = 0 ;
for ( uint8_t i = 0 ; i < MIXING_STEPPERS ; i + + ) {
2017-06-27 06:31:45 +02:00
if ( parser . seenval ( mixing_codes [ i ] ) ) {
2016-11-15 22:50:54 +01:00
SBI ( mix_bits , i ) ;
2017-05-20 10:03:08 +02:00
float v = parser . value_float ( ) ;
2016-11-15 22:50:54 +01:00
NOLESS ( v , 0.0 ) ;
mixing_factor [ i ] = RECIPROCAL ( v ) ;
}
}
// If any mixing factors were included, clear the rest
// If none were included, preserve the last mix
if ( mix_bits ) {
for ( uint8_t i = 0 ; i < MIXING_STEPPERS ; i + + )
if ( ! TEST ( mix_bits , i ) ) mixing_factor [ i ] = 0.0 ;
normalize_mix ( ) ;
2016-11-12 21:33:07 +01:00
}
2016-06-29 00:06:56 +02:00
}
# endif
# endif
2015-03-05 13:27:24 +01:00
/**
2016-03-27 05:36:36 +02:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * G - CODE HANDLING * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
2015-03-05 13:27:24 +01:00
*/
2013-06-07 00:49:25 +02:00
2015-05-14 03:52:41 +02:00
/**
* Set XYZE destination and feedrate from the current GCode command
*
* - Set destination from included axis codes
* - Set to current for missing axis codes
* - Set the feedrate , if included
*/
void gcode_get_destination ( ) {
2016-07-23 22:07:23 +02:00
LOOP_XYZE ( i ) {
2017-05-20 10:03:08 +02:00
if ( parser . seen ( axis_codes [ i ] ) )
destination [ i ] = parser . value_axis_units ( ( AxisEnum ) i ) + ( axis_relative_modes [ i ] | | relative_mode ? current_position [ i ] : 0 ) ;
2015-05-14 03:52:41 +02:00
else
destination [ i ] = current_position [ i ] ;
}
2016-07-14 03:18:59 +02:00
2017-06-27 07:49:21 +02:00
if ( parser . linearval ( ' F ' ) > 0.0 )
2017-05-20 10:03:08 +02:00
feedrate_mm_s = MMM_TO_MMS ( parser . value_feedrate ( ) ) ;
2016-07-14 03:18:59 +02:00
# if ENABLED(PRINTCOUNTER)
2016-07-16 03:49:34 +02:00
if ( ! DEBUGGING ( DRYRUN ) )
2016-07-14 03:18:59 +02:00
print_job_timer . incFilamentUsed ( destination [ E_AXIS ] - current_position [ E_AXIS ] ) ;
# endif
2016-06-29 00:06:56 +02:00
// Get ABCDHI mixing factors
# if ENABLED(MIXING_EXTRUDER) && ENABLED(DIRECT_MIXING_IN_G1)
gcode_get_mix ( ) ;
# endif
2015-05-14 03:52:41 +02:00
}
2016-03-07 12:48:14 +01:00
# if ENABLED(HOST_KEEPALIVE_FEATURE)
2016-03-26 02:36:21 +01:00
/**
* Output a " busy " message at regular intervals
* while the machine is not accepting commands .
*/
2016-03-07 12:48:14 +01:00
void host_keepalive ( ) {
2017-03-04 01:23:31 +01:00
const millis_t ms = millis ( ) ;
2016-04-10 01:58:17 +02:00
if ( host_keepalive_interval & & busy_state ! = NOT_BUSY ) {
2016-04-11 00:55:12 +02:00
if ( PENDING ( ms , next_busy_signal_ms ) ) return ;
2016-03-07 12:48:14 +01:00
switch ( busy_state ) {
case IN_HANDLER :
case IN_PROCESS :
2017-06-09 17:51:23 +02:00
SERIAL_ECHO_START ( ) ;
2016-03-07 12:48:14 +01:00
SERIAL_ECHOLNPGM ( MSG_BUSY_PROCESSING ) ;
break ;
case PAUSED_FOR_USER :
2017-06-09 17:51:23 +02:00
SERIAL_ECHO_START ( ) ;
2016-03-07 12:48:14 +01:00
SERIAL_ECHOLNPGM ( MSG_BUSY_PAUSED_FOR_USER ) ;
break ;
case PAUSED_FOR_INPUT :
2017-06-09 17:51:23 +02:00
SERIAL_ECHO_START ( ) ;
2016-03-07 12:48:14 +01:00
SERIAL_ECHOLNPGM ( MSG_BUSY_PAUSED_FOR_INPUT ) ;
break ;
2016-04-02 23:28:17 +02:00
default :
break ;
2016-03-07 12:48:14 +01:00
}
}
2016-04-11 00:55:12 +02:00
next_busy_signal_ms = ms + host_keepalive_interval * 1000UL ;
2016-03-07 12:48:14 +01:00
}
2017-05-09 19:35:43 +02:00
# endif // HOST_KEEPALIVE_FEATURE
2016-03-07 12:48:14 +01:00
2016-09-12 05:21:32 +02:00
2016-09-13 09:11:45 +02:00
/**************************************************
* * * * * * * * * * * * * * * * * GCode Handlers * * * * * * * * * * * * * * * * *
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2017-09-06 13:28:33 +02:00
# include "gcode/motion/G0_G1.h"
2015-03-05 13:27:24 +01:00
2016-05-13 13:27:45 +02:00
# if ENABLED(ARC_SUPPORT)
2017-09-06 13:28:33 +02:00
# include "gcode/motion/G2_G3.h"
# endif
2015-03-05 13:27:24 +01:00
2017-08-13 23:35:59 +02:00
void dwell ( millis_t time ) {
refresh_cmd_timeout ( ) ;
time + = previous_cmd_ms ;
while ( PENDING ( millis ( ) , time ) ) idle ( ) ;
}
2017-09-06 13:28:33 +02:00
# include "gcode/motion/G4.h"
2013-09-29 18:20:06 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(BEZIER_CURVE_SUPPORT)
# include "gcode/motion/G5.h"
# endif
2015-03-05 13:27:24 +01:00
2017-09-06 13:28:33 +02:00
# if ENABLED(FWRETRACT)
# include "gcode/feature/fwretract/G10_G11.h"
# endif
2015-04-25 05:13:01 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(NOZZLE_CLEAN_FEATURE)
# include "gcode/feature/clean/G12.h"
# endif
2015-04-25 05:13:01 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(CNC_WORKSPACE_PLANES)
# include "gcode/feature/clean/G17-G19.h"
# endif
2013-06-07 00:49:25 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(INCH_MODE_SUPPORT)
# include "gcode/units/G20_G21.h"
# endif
2016-05-12 22:33:47 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(UBL_G26_MESH_VALIDATION)
# include "gcode/calibrate/G26.h"
# endif
2016-05-12 22:33:47 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(NOZZLE_PARK_FEATURE)
# include "gcode/feature/pause/G27.h"
# endif
2016-05-12 22:33:47 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(PROBE_MANUALLY)
bool g29_in_progress = false ;
# else
constexpr bool g29_in_progress = false ;
# endif
2017-06-22 15:57:33 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/calibrate/G28.h"
2016-05-12 22:33:47 +02:00
2017-09-06 13:28:33 +02:00
void home_all_axes ( ) { gcode_G28 ( true ) ; }
2016-05-12 22:33:47 +02:00
2017-09-06 13:28:33 +02:00
# if HAS_PROBING_PROCEDURE
void out_of_range_error ( const char * p_edge ) {
SERIAL_PROTOCOLPGM ( " ?Probe " ) ;
serialprintPGM ( p_edge ) ;
SERIAL_PROTOCOLLNPGM ( " position out of range. " ) ;
2016-05-12 22:33:47 +02:00
}
2017-09-06 13:28:33 +02:00
# endif
2016-05-12 22:33:47 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/calibrate/G29.h"
2013-06-07 00:49:25 +02:00
2017-09-06 13:28:33 +02:00
# if HAS_BED_PROBE
# include "gcode/probe/G30.h"
# if ENABLED(Z_PROBE_SLED)
# include "gcode/probe/G31_G32.h"
# endif
# endif
2013-06-10 06:10:00 +02:00
2017-09-06 13:28:33 +02:00
# if PROBE_SELECTED && ENABLED(DELTA_AUTO_CALIBRATION)
# include "gcode/calibrate/G33.h"
# endif
2017-07-18 08:37:54 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(G38_PROBE_TARGET)
# include "gcode/probe/G38.h"
# endif
2013-06-10 06:10:00 +02:00
2017-09-06 13:28:33 +02:00
# if HAS_MESH
# include "gcode/probe/G42.h"
# endif
2016-06-16 06:18:44 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/geometry/G92.h"
2016-06-16 06:18:44 +02:00
2017-09-06 13:28:33 +02:00
# if HAS_RESUME_CONTINUE
# include "gcode/lcd/M0_M1.h"
2016-06-16 06:18:44 +02:00
# endif
2017-09-06 13:28:33 +02:00
# if ENABLED(SPINDLE_LASER_ENABLE)
# include "gcode/control/M3-M5.h"
# endif
2017-06-23 21:28:28 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/control/M17.h"
2017-06-23 21:28:28 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(ADVANCED_PAUSE_FEATURE)
// For M125, M600, M24
# include "gcode/feature/pause/common.h"
# endif
2017-06-23 21:28:28 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(SDSUPPORT)
# include "gcode/sdcard/M20.h" // M20 - List SD card. (Requires SDSUPPORT)
# include "gcode/sdcard/M21.h" // M21 - Init SD card. (Requires SDSUPPORT)
# include "gcode/sdcard/M22.h" // M22 - Release SD card. (Requires SDSUPPORT)
# include "gcode/sdcard/M23.h" // M23 - Select SD file: "M23 /path/file.gco". (Requires SDSUPPORT)
# include "gcode/sdcard/M24.h" // M24 - Start/resume SD print. (Requires SDSUPPORT)
# include "gcode/sdcard/M25.h" // M25 - Pause SD print. (Requires SDSUPPORT)
# include "gcode/sdcard/M26.h" // M26 - Set SD position in bytes: "M26 S12345". (Requires SDSUPPORT)
# include "gcode/sdcard/M27.h" // M27 - Report SD print status. (Requires SDSUPPORT)
# include "gcode/sdcard/M28.h" // M28 - Start SD write: "M28 /path/file.gco". (Requires SDSUPPORT)
# include "gcode/sdcard/M29.h" // M29 - Stop SD write. (Requires SDSUPPORT)
# include "gcode/sdcard/M30.h" // M30 - Delete file from SD: "M30 /path/file.gco"
# endif
2017-06-23 21:28:28 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/stats/M31.h" // M31: Get the time since the start of SD Print (or last M109)
2016-05-31 17:26:08 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(SDSUPPORT)
# include "gcode/sdcard/M32.h"
# if ENABLED(LONG_FILENAME_HOST_SUPPORT)
# include "gcode/sdcard/M33.h"
# endif
# if ENABLED(SDCARD_SORT_ALPHA) && ENABLED(SDSORT_GCODE)
# include "gcode/sdcard/M34.h"
# endif
# include "gcode/sdcard/M928.h"
2016-05-31 17:26:08 +02:00
# endif
2017-09-06 13:28:33 +02:00
/**
* Sensitive pin test for M42 , M226
*/
static bool pin_is_protected ( const int8_t pin ) {
static const int8_t sensitive_pins [ ] PROGMEM = SENSITIVE_PINS ;
for ( uint8_t i = 0 ; i < COUNT ( sensitive_pins ) ; i + + )
if ( pin = = ( int8_t ) pgm_read_byte ( & sensitive_pins [ i ] ) ) return true ;
return false ;
}
2016-07-19 17:24:44 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/control/M42.h"
2016-07-06 05:10:15 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(PINS_DEBUGGING)
# include "gcode/config/M43.h"
# endif
2016-07-06 05:10:15 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(Z_MIN_PROBE_REPEATABILITY_TEST)
# include "gcode/calibrate/M48.h"
# endif
2016-07-19 17:24:44 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(AUTO_BED_LEVELING_UBL) && ENABLED(UBL_G26_MESH_VALIDATION)
# include "gcode/calibrate/M49.h"
# endif
2016-07-06 05:10:15 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/stats/M75.h"
# include "gcode/stats/M76.h"
# include "gcode/stats/M77.h"
2016-07-06 05:10:15 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(PRINTCOUNTER)
# include "gcode/stats/M78.h"
# endif
2016-07-06 05:10:15 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/temperature/M104.h"
2016-07-06 05:10:15 +02:00
2017-09-06 13:28:33 +02:00
# if HAS_TEMP_HOTEND || HAS_TEMP_BED
2016-08-29 03:45:18 +02:00
2017-09-06 13:28:33 +02:00
void print_heater_state ( const float & c , const float & t ,
# if ENABLED(SHOW_TEMP_ADC_VALUES)
const float r ,
2016-08-29 03:45:18 +02:00
# endif
2017-09-06 13:28:33 +02:00
const int8_t e = - 2
) {
# if !(HAS_TEMP_BED && HAS_TEMP_HOTEND) && HOTENDS <= 1
UNUSED ( e ) ;
2016-08-29 03:45:18 +02:00
# endif
2017-09-06 13:28:33 +02:00
SERIAL_PROTOCOLCHAR ( ' ' ) ;
SERIAL_PROTOCOLCHAR (
# if HAS_TEMP_BED && HAS_TEMP_HOTEND
e = = - 1 ? ' B ' : ' T '
# elif HAS_TEMP_HOTEND
' T '
2016-09-15 20:06:00 +02:00
# else
2017-09-06 13:28:33 +02:00
' B '
2016-08-29 03:45:18 +02:00
# endif
2017-09-06 13:28:33 +02:00
) ;
# if HOTENDS > 1
if ( e > = 0 ) SERIAL_PROTOCOLCHAR ( ' 0 ' + e ) ;
# endif
SERIAL_PROTOCOLCHAR ( ' : ' ) ;
SERIAL_PROTOCOL ( c ) ;
SERIAL_PROTOCOLPAIR ( " / " , t ) ;
# if ENABLED(SHOW_TEMP_ADC_VALUES)
SERIAL_PROTOCOLPAIR ( " ( " , r / OVERSAMPLENR ) ;
SERIAL_PROTOCOLCHAR ( ' ) ' ) ;
2016-08-29 03:45:18 +02:00
# endif
2017-09-06 13:28:33 +02:00
}
2016-09-27 06:38:52 +02:00
2017-09-06 13:28:33 +02:00
void print_heaterstates ( ) {
# if HAS_TEMP_HOTEND
print_heater_state ( thermalManager . degHotend ( target_extruder ) , thermalManager . degTargetHotend ( target_extruder )
# if ENABLED(SHOW_TEMP_ADC_VALUES)
, thermalManager . rawHotendTemp ( target_extruder )
# endif
) ;
# endif
# if HAS_TEMP_BED
print_heater_state ( thermalManager . degBed ( ) , thermalManager . degTargetBed ( ) ,
# if ENABLED(SHOW_TEMP_ADC_VALUES)
thermalManager . rawBedTemp ( ) ,
# endif
- 1 // BED
) ;
# endif
# if HOTENDS > 1
HOTEND_LOOP ( ) print_heater_state ( thermalManager . degHotend ( e ) , thermalManager . degTargetHotend ( e ) ,
# if ENABLED(SHOW_TEMP_ADC_VALUES)
thermalManager . rawHotendTemp ( e ) ,
2016-09-27 06:38:52 +02:00
# endif
2017-09-06 13:28:33 +02:00
e
) ;
# endif
SERIAL_PROTOCOLPGM ( " @: " ) ;
SERIAL_PROTOCOL ( thermalManager . getHeaterPower ( target_extruder ) ) ;
# if HAS_TEMP_BED
SERIAL_PROTOCOLPGM ( " B@: " ) ;
SERIAL_PROTOCOL ( thermalManager . getHeaterPower ( - 1 ) ) ;
# endif
# if HOTENDS > 1
HOTEND_LOOP ( ) {
SERIAL_PROTOCOLPAIR ( " @ " , e ) ;
SERIAL_PROTOCOLCHAR ( ' : ' ) ;
SERIAL_PROTOCOL ( thermalManager . getHeaterPower ( e ) ) ;
2016-09-27 06:38:52 +02:00
}
2017-09-06 13:28:33 +02:00
# endif
}
2017-03-20 07:42:41 +01:00
2017-09-06 13:28:33 +02:00
# endif // HAS_TEMP_HOTEND || HAS_TEMP_BED
2017-03-20 07:42:41 +01:00
2017-09-06 13:28:33 +02:00
# include "gcode/temperature/M105.h"
2017-03-20 07:42:41 +01:00
2017-09-06 13:28:33 +02:00
# if ENABLED(AUTO_REPORT_TEMPERATURES) && (HAS_TEMP_HOTEND || HAS_TEMP_BED)
2017-03-20 07:42:41 +01:00
2017-09-06 13:28:33 +02:00
static uint8_t auto_report_temp_interval ;
static millis_t next_temp_report_ms ;
2016-09-27 06:38:52 +02:00
2017-09-06 13:28:33 +02:00
inline void auto_report_temperatures ( ) {
if ( auto_report_temp_interval & & ELAPSED ( millis ( ) , next_temp_report_ms ) ) {
next_temp_report_ms = millis ( ) + 1000UL * auto_report_temp_interval ;
print_heaterstates ( ) ;
SERIAL_EOL ( ) ;
}
2016-08-29 03:45:18 +02:00
}
2017-09-06 13:28:33 +02:00
# include "gcode/temperature/M155.h"
2016-08-29 03:45:18 +02:00
2017-09-06 13:28:33 +02:00
# endif // AUTO_REPORT_TEMPERATURES && (HAS_TEMP_HOTEND || HAS_TEMP_BED)
2016-09-12 03:25:44 +02:00
2017-09-06 13:28:33 +02:00
# if FAN_COUNT > 0
# include "gcode/temperature/M106.h"
# include "gcode/temperature/M107.h"
2017-03-15 09:32:00 +01:00
# endif
2017-09-06 13:28:33 +02:00
# if DISABLED(EMERGENCY_PARSER)
# include "gcode/control/M108.h"
# include "gcode/control/M112.h"
# include "gcode/control/M410.h"
# endif
2013-09-29 18:20:06 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/temperature/M109.h"
2015-03-03 05:00:17 +01:00
2017-09-06 13:28:33 +02:00
# if HAS_TEMP_BED
# include "gcode/temperature/M190.h"
# endif
2015-03-03 05:00:17 +01:00
2017-09-06 13:28:33 +02:00
# include "gcode/host/M110.h"
2017-05-12 06:22:58 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/control/M111.h"
2016-10-02 08:48:17 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(HOST_KEEPALIVE_FEATURE)
# include "gcode/host/M113.h"
# endif
2016-06-22 23:20:34 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(BARICUDA)
# if HAS_HEATER_1
# include "gcode/feature/baricuda/M126.h"
# include "gcode/feature/baricuda/M127.h"
2016-07-20 04:37:31 +02:00
# endif
2017-09-06 13:28:33 +02:00
# if HAS_HEATER_2
# include "gcode/feature/baricuda/M128.h"
# include "gcode/feature/baricuda/M129.h"
2017-03-15 09:20:41 +01:00
# endif
2016-06-22 23:03:22 +02:00
# endif
2017-09-06 13:28:33 +02:00
# include "gcode/temperature/M140.h"
2016-04-13 11:37:41 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(ULTIPANEL)
# include "gcode/lcd/M145.h"
2017-03-15 09:32:00 +01:00
# endif
2017-09-06 13:28:33 +02:00
# if ENABLED(TEMPERATURE_UNITS_SUPPORT)
# include "gcode/units/M149.h"
# endif
2015-03-30 08:16:12 +02:00
2017-09-06 13:28:33 +02:00
# if HAS_POWER_SWITCH
# include "gcode/control/M80.h"
# endif
2017-03-15 09:20:41 +01:00
2017-09-06 13:28:33 +02:00
# include "gcode/control/M81.h"
2015-03-21 14:50:47 +01:00
2017-09-06 13:28:33 +02:00
# include "gcode/units/M82_M83.h"
2015-04-05 04:16:16 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/control/M18_M84.h"
2015-03-21 14:50:47 +01:00
2017-09-06 13:28:33 +02:00
# include "gcode/control/M85.h"
2015-03-21 14:50:47 +01:00
2017-09-06 13:28:33 +02:00
/**
* Multi - stepper support for M92 , M201 , M203
*/
# if ENABLED(DISTINCT_E_FACTORS)
# define GET_TARGET_EXTRUDER(CMD) if (get_target_extruder_from_command(CMD)) return
# define TARGET_EXTRUDER target_extruder
# else
# define GET_TARGET_EXTRUDER(CMD) NOOP
# define TARGET_EXTRUDER 0
# endif
2017-03-15 09:20:41 +01:00
2017-09-06 13:28:33 +02:00
# include "gcode/config/M92.h"
2015-04-05 04:06:02 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(M100_FREE_MEMORY_WATCHER)
# include "gcode/calibrate/M100.h"
# endif
2016-07-20 04:37:31 +02:00
2015-03-05 13:27:24 +01:00
/**
2017-09-06 13:28:33 +02:00
* Output the current position to serial
2015-03-05 13:27:24 +01:00
*/
2017-09-06 13:28:33 +02:00
void report_current_position ( ) {
SERIAL_PROTOCOLPGM ( " X: " ) ;
SERIAL_PROTOCOL ( current_position [ X_AXIS ] ) ;
SERIAL_PROTOCOLPGM ( " Y: " ) ;
SERIAL_PROTOCOL ( current_position [ Y_AXIS ] ) ;
SERIAL_PROTOCOLPGM ( " Z: " ) ;
SERIAL_PROTOCOL ( current_position [ Z_AXIS ] ) ;
SERIAL_PROTOCOLPGM ( " E: " ) ;
SERIAL_PROTOCOL ( current_position [ E_AXIS ] ) ;
2016-11-08 23:28:42 +01:00
2017-09-06 13:28:33 +02:00
stepper . report_positions ( ) ;
2015-02-18 21:33:38 +01:00
2017-09-06 13:28:33 +02:00
# if IS_SCARA
SERIAL_PROTOCOLPAIR ( " SCARA Theta: " , stepper . get_axis_position_degrees ( A_AXIS ) ) ;
SERIAL_PROTOCOLLNPAIR ( " Psi+Theta: " , stepper . get_axis_position_degrees ( B_AXIS ) ) ;
SERIAL_EOL ( ) ;
# endif
}
2014-02-05 10:47:12 +01:00
2017-09-06 13:28:33 +02:00
# include "gcode/host/M114.h"
# include "gcode/host/M115.h"
2013-08-07 16:10:26 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/lcd/M117.h"
2013-08-07 16:10:26 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/host/M118.h"
# include "gcode/host/M119.h"
2014-02-05 10:47:12 +01:00
2017-09-06 13:28:33 +02:00
# include "gcode/control/M120_M121.h"
2014-02-05 10:47:12 +01:00
2017-09-06 13:28:33 +02:00
# if ENABLED(PARK_HEAD_ON_PAUSE)
# include "gcode/feature/pause/M125.h"
# endif
2016-10-28 20:53:48 +02:00
2017-09-06 13:28:33 +02:00
# if HAS_COLOR_LEDS
# include "gcode/feature/leds/M150.h"
# endif
2015-04-27 03:44:01 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/config/M200.h"
# include "gcode/config/M201.h"
2015-04-30 03:26:16 +02:00
2017-09-06 13:28:33 +02:00
#if 0 // Not used for Sprinter/grbl gen6
# include "gcode/config/M202.h"
# endif
2013-08-07 16:10:26 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/config/M203.h"
# include "gcode/config/M204.h"
# include "gcode/config/M205.h"
2014-02-05 10:47:12 +01:00
2017-09-06 13:28:33 +02:00
# if HAS_M206_COMMAND
# include "gcode/geometry/M206.h"
# endif
2014-02-05 10:47:12 +01:00
2017-09-06 13:28:33 +02:00
# if IS_KINEMATIC
# include "gcode/calibrate/M665.h"
# endif
2013-07-17 14:44:45 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(DELTA) || ENABLED(Z_DUAL_ENDSTOPS)
# include "gcode/calibrate/M666.h"
# endif
2016-06-16 20:24:11 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(FWRETRACT)
# include "gcode/feature/fwretract/M207.h"
# include "gcode/feature/fwretract/M208.h"
# include "gcode/feature/fwretract/M209.h"
# endif
2017-06-02 20:57:31 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/control/M211.h"
2013-07-17 14:44:45 +02:00
2017-09-06 13:28:33 +02:00
# if HOTENDS > 1
# include "gcode/config/M218.h"
# endif
2017-03-07 06:00:43 +01:00
2017-09-06 13:28:33 +02:00
# include "gcode/config/M220.h"
# include "gcode/config/M221.h"
2014-04-01 03:26:19 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/control/M226.h"
2016-03-20 02:19:41 +01:00
2017-09-06 13:28:33 +02:00
# if ENABLED(EXPERIMENTAL_I2CBUS)
# include "gcode/feature/i2c/M260_M261.h"
# endif
2016-03-20 02:19:41 +01:00
2017-09-06 13:28:33 +02:00
# if HAS_SERVOS
# include "gcode/control/M280.h"
# endif
2016-03-20 02:19:41 +01:00
2017-09-06 13:28:33 +02:00
# if HAS_BUZZER
# include "gcode/lcd/M300.h"
# endif
2016-03-20 02:19:41 +01:00
2017-09-06 13:28:33 +02:00
# if ENABLED(PIDTEMP)
# include "gcode/config/M301.h"
# endif
2016-03-20 02:19:41 +01:00
2017-09-06 13:28:33 +02:00
# if ENABLED(PIDTEMPBED)
# include "gcode/config/M304.h"
# endif
2016-03-20 02:19:41 +01:00
2017-09-06 13:28:33 +02:00
# if defined(CHDK) || HAS_PHOTOGRAPH
# include "gcode/control/M240.h"
# endif
2014-04-01 03:26:19 +02:00
2017-09-06 13:28:33 +02:00
# if HAS_LCD_CONTRAST
# include "gcode/control/M250.h"
# endif
2017-03-07 06:00:43 +01:00
2017-09-06 13:28:33 +02:00
# if ENABLED(PREVENT_COLD_EXTRUSION)
# include "gcode/config/M302.h"
# endif
2017-04-15 05:44:08 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/temperature/M303.h"
2017-04-15 05:44:08 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(MORGAN_SCARA)
# include "gcode/scara/M360-M364.h"
# endif
2017-03-07 06:00:43 +01:00
2017-09-06 13:28:33 +02:00
# if ENABLED(EXT_SOLENOID)
# include "gcode/control/M380_M381.h"
# endif
2015-03-03 02:07:12 +01:00
2017-09-06 13:28:33 +02:00
# include "gcode/control/M400.h"
2015-04-04 00:31:35 +02:00
2017-09-06 13:28:33 +02:00
# if HAS_BED_PROBE
# include "gcode/probe/M401_M402.h"
# endif
2015-04-04 00:31:35 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(FILAMENT_WIDTH_SENSOR)
# include "gcode/sensor/M404.h"
# include "gcode/sensor/M405.h"
# include "gcode/sensor/M406.h"
# include "gcode/sensor/M407.h"
# endif
2015-03-03 02:07:12 +01:00
2017-09-06 13:28:33 +02:00
void quickstop_stepper ( ) {
stepper . quick_stop ( ) ;
stepper . synchronize ( ) ;
set_current_from_steppers_for_axis ( ALL_AXES ) ;
SYNC_PLAN_POSITION_KINEMATIC ( ) ;
}
2016-10-05 21:34:07 +02:00
2017-09-06 13:28:33 +02:00
# if HAS_LEVELING
# include "gcode/calibrate/M420.h"
# include "gcode/calibrate/M421.h"
# endif
2017-05-20 10:03:08 +02:00
2017-09-06 13:28:33 +02:00
# if HAS_M206_COMMAND
# include "gcode/geometry/M428.h"
# endif
2017-06-09 14:06:23 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/eeprom/M500.h"
# include "gcode/eeprom/M501.h"
# include "gcode/eeprom/M502.h"
# if DISABLED(DISABLE_M503)
# include "gcode/eeprom/M503.h"
# endif
2017-06-09 14:06:23 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(ABORT_ON_ENDSTOP_HIT_FEATURE_ENABLED)
# include "gcode/config/M540.h"
# endif
2017-06-09 14:06:23 +02:00
2017-09-06 13:28:33 +02:00
# if HAS_BED_PROBE
# include "gcode/probe/M851.h"
# endif
2017-06-09 14:06:23 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(ADVANCED_PAUSE_FEATURE)
# include "gcode/feature/pause/M600.h"
# endif
2017-06-09 14:06:23 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(MK2_MULTIPLEXER)
# include "gcode/feature/snmm/M702.h"
# endif
2017-06-09 14:06:23 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(DUAL_X_CARRIAGE) || ENABLED(DUAL_NOZZLE_DUPLICATION_MODE)
# include "gcode/control/M605.h"
# endif
2017-06-09 14:06:23 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(LIN_ADVANCE)
# include "gcode/feature/advance/M900.h"
# endif
2017-06-09 14:06:23 +02:00
2017-09-06 13:28:33 +02:00
# if ENABLED(HAVE_TMC2130)
# include "feature/tmc2130.h"
# include "gcode/feature/trinamic/M906.h"
# include "gcode/feature/trinamic/M911.h"
# include "gcode/feature/trinamic/M912.h"
# if ENABLED(HYBRID_THRESHOLD)
# include "gcode/feature/trinamic/M913.h"
# endif
# if ENABLED(SENSORLESS_HOMING)
# include "gcode/feature/trinamic/M914.h"
# endif
# endif
2017-06-09 14:06:23 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/feature/digipot/M907.h"
2017-06-09 14:06:23 +02:00
2017-09-06 13:28:33 +02:00
# if HAS_DIGIPOTSS || ENABLED(DAC_STEPPER_CURRENT)
# include "gcode/feature/digipot/M908.h"
# if ENABLED(DAC_STEPPER_CURRENT) // As with Printrbot RevF
# include "gcode/feature/digipot/M909.h"
# include "gcode/feature/digipot/M910.h"
# endif
# endif
2017-06-09 14:06:23 +02:00
2017-09-06 13:28:33 +02:00
# if HAS_MICROSTEPS
# include "gcode/control/M350.h"
# include "gcode/control/M351.h"
# endif
2017-06-09 14:06:23 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/feature/caselight/M355.h"
2012-11-06 12:06:41 +01:00
2017-09-06 13:28:33 +02:00
# if ENABLED(MIXING_EXTRUDER)
# include "gcode/feature/mixing/M163.h"
# if MIXING_VIRTUAL_TOOLS > 1
# include "gcode/feature/mixing/M164.h"
# endif
# if ENABLED(DIRECT_MIXING_IN_G1)
# include "gcode/feature/mixing/M165.h"
# endif
# endif
2015-05-29 02:44:09 +02:00
2017-09-06 13:28:33 +02:00
# include "gcode/control/M999.h"
2015-03-05 13:27:24 +01:00
2017-09-06 13:28:33 +02:00
# include "gcode/control/T.h"
2016-03-07 12:48:14 +01:00
2017-09-06 13:28:33 +02:00
# include "gcode/process_next_command.h"
2012-11-06 12:06:41 +01:00
2016-09-13 09:11:45 +02:00
/**
* Send a " Resend: nnn " message to the host to
* indicate that a command needs to be re - sent .
*/
2015-04-04 00:31:35 +02:00
void FlushSerialRequestResend ( ) {
2015-04-14 02:17:36 +02:00
//char command_queue[cmd_queue_index_r][100]="Resend:";
2012-11-06 12:06:41 +01:00
MYSERIAL . flush ( ) ;
SERIAL_PROTOCOLPGM ( MSG_RESEND ) ;
SERIAL_PROTOCOLLN ( gcode_LastN + 1 ) ;
2015-05-12 11:08:20 +02:00
ok_to_send ( ) ;
2012-11-06 12:06:41 +01:00
}
2016-09-13 09:11:45 +02:00
/**
* Send an " ok " message to the host , indicating
* that a command was successfully processed .
*
* If ADVANCED_OK is enabled also include :
* N < int > Line number of the command , if any
* P < int > Planner space remaining
* B < int > Block queue space remaining
*/
2015-05-12 11:08:20 +02:00
void ok_to_send ( ) {
2015-04-04 00:31:35 +02:00
refresh_cmd_timeout ( ) ;
2016-02-21 02:35:35 +01:00
if ( ! send_ok [ cmd_queue_index_r ] ) return ;
2015-04-20 00:22:40 +02:00
SERIAL_PROTOCOLPGM ( MSG_OK ) ;
2015-07-31 07:24:43 +02:00
# if ENABLED(ADVANCED_OK)
2016-02-09 19:23:26 +01:00
char * p = command_queue [ cmd_queue_index_r ] ;
if ( * p = = ' N ' ) {
SERIAL_PROTOCOL ( ' ' ) ;
SERIAL_ECHO ( * p + + ) ;
2016-03-28 12:52:49 +02:00
while ( NUMERIC_SIGNED ( * p ) )
2016-02-09 19:23:26 +01:00
SERIAL_ECHO ( * p + + ) ;
}
2016-05-02 15:04:26 +02:00
SERIAL_PROTOCOLPGM ( " P " ) ; SERIAL_PROTOCOL ( int ( BLOCK_BUFFER_SIZE - planner . movesplanned ( ) - 1 ) ) ;
2015-05-11 13:04:00 +02:00
SERIAL_PROTOCOLPGM ( " B " ) ; SERIAL_PROTOCOL ( BUFSIZE - commands_in_queue ) ;
2015-04-20 00:22:40 +02:00
# endif
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2012-11-06 12:06:41 +01:00
}
2017-03-16 23:20:24 +01:00
# if HAS_SOFTWARE_ENDSTOPS
2016-08-21 03:07:42 +02:00
2016-09-13 09:11:45 +02:00
/**
* Constrain the given coordinates to the software endstops .
*/
2017-06-04 00:11:43 +02:00
// NOTE: This makes no sense for delta beds other than Z-axis.
// For delta the X/Y would need to be clamped at
// DELTA_PRINTABLE_RADIUS from center of bed, but delta
// now enforces is_position_reachable for X/Y regardless
// of HAS_SOFTWARE_ENDSTOPS, so that enforcement would be
2017-08-03 05:52:40 +02:00
// redundant here.
2017-06-04 00:11:43 +02:00
2016-08-21 03:07:42 +02:00
void clamp_to_software_endstops ( float target [ XYZ ] ) {
2017-03-16 23:28:44 +01:00
if ( ! soft_endstops_enabled ) return ;
2017-03-17 02:03:57 +01:00
# if ENABLED(MIN_SOFTWARE_ENDSTOPS)
2017-08-03 05:52:40 +02:00
# if DISABLED(DELTA)
NOLESS ( target [ X_AXIS ] , soft_endstop_min [ X_AXIS ] ) ;
NOLESS ( target [ Y_AXIS ] , soft_endstop_min [ Y_AXIS ] ) ;
# endif
2016-08-21 03:07:42 +02:00
NOLESS ( target [ Z_AXIS ] , soft_endstop_min [ Z_AXIS ] ) ;
# endif
2017-03-17 02:03:57 +01:00
# if ENABLED(MAX_SOFTWARE_ENDSTOPS)
2017-08-03 05:52:40 +02:00
# if DISABLED(DELTA)
NOMORE ( target [ X_AXIS ] , soft_endstop_max [ X_AXIS ] ) ;
NOMORE ( target [ Y_AXIS ] , soft_endstop_max [ Y_AXIS ] ) ;
# endif
2016-08-21 03:07:42 +02:00
NOMORE ( target [ Z_AXIS ] , soft_endstop_max [ Z_AXIS ] ) ;
# endif
2012-11-06 12:06:41 +01:00
}
2016-08-21 03:07:42 +02:00
# endif
2012-11-06 12:06:41 +01:00
2016-09-26 06:17:39 +02:00
# if ENABLED(AUTO_BED_LEVELING_BILINEAR)
2016-09-15 07:20:30 +02:00
2016-11-05 17:01:26 +01:00
# if ENABLED(ABL_BILINEAR_SUBDIVISION)
# define ABL_BG_SPACING(A) bilinear_grid_spacing_virt[A]
2017-04-21 08:34:03 +02:00
# define ABL_BG_FACTOR(A) bilinear_grid_factor_virt[A]
2016-11-05 17:01:26 +01:00
# define ABL_BG_POINTS_X ABL_GRID_POINTS_VIRT_X
# define ABL_BG_POINTS_Y ABL_GRID_POINTS_VIRT_Y
2017-04-22 05:46:19 +02:00
# define ABL_BG_GRID(X,Y) z_values_virt[X][Y]
2016-11-05 17:01:26 +01:00
# else
# define ABL_BG_SPACING(A) bilinear_grid_spacing[A]
2017-04-21 08:34:03 +02:00
# define ABL_BG_FACTOR(A) bilinear_grid_factor[A]
2017-04-06 05:29:44 +02:00
# define ABL_BG_POINTS_X GRID_MAX_POINTS_X
# define ABL_BG_POINTS_Y GRID_MAX_POINTS_Y
2017-04-22 05:46:19 +02:00
# define ABL_BG_GRID(X,Y) z_values[X][Y]
2016-11-05 17:01:26 +01:00
# endif
2016-09-15 07:20:30 +02:00
// Get the Z adjustment for non-linear bed leveling
2017-04-21 08:34:03 +02:00
float bilinear_z_offset ( const float logical [ XYZ ] ) {
2016-09-27 09:38:38 +02:00
2017-04-21 08:34:03 +02:00
static float z1 , d2 , z3 , d4 , L , D , ratio_x , ratio_y ,
last_x = - 999.999 , last_y = - 999.999 ;
2016-09-27 09:38:38 +02:00
2016-10-27 01:11:26 +02:00
// Whole units for the grid line indices. Constrained within bounds.
2017-04-21 08:34:03 +02:00
static int8_t gridx , gridy , nextx , nexty ,
last_gridx = - 99 , last_gridy = - 99 ;
2016-09-28 19:56:35 +02:00
2017-04-21 08:34:03 +02:00
// XY relative to the probed area
const float x = RAW_X_POSITION ( logical [ X_AXIS ] ) - bilinear_start [ X_AXIS ] ,
y = RAW_Y_POSITION ( logical [ Y_AXIS ] ) - bilinear_start [ Y_AXIS ] ;
2017-05-04 07:19:07 +02:00
# if ENABLED(EXTRAPOLATE_BEYOND_GRID)
// Keep using the last grid box
# define FAR_EDGE_OR_BOX 2
# else
// Just use the grid far edge
# define FAR_EDGE_OR_BOX 1
# endif
2017-04-21 08:34:03 +02:00
if ( last_x ! = x ) {
last_x = x ;
ratio_x = x * ABL_BG_FACTOR ( X_AXIS ) ;
2017-06-20 05:39:23 +02:00
const float gx = constrain ( FLOOR ( ratio_x ) , 0 , ABL_BG_POINTS_X - FAR_EDGE_OR_BOX ) ;
2017-04-21 08:34:03 +02:00
ratio_x - = gx ; // Subtract whole to get the ratio within the grid box
2017-05-04 07:19:07 +02:00
# if DISABLED(EXTRAPOLATE_BEYOND_GRID)
// Beyond the grid maintain height at grid edges
NOLESS ( ratio_x , 0 ) ; // Never < 0.0. (> 1.0 is ok when nextx==gridx.)
# endif
2017-04-21 08:34:03 +02:00
gridx = gx ;
nextx = min ( gridx + 1 , ABL_BG_POINTS_X - 1 ) ;
}
if ( last_y ! = y | | last_gridx ! = gridx ) {
if ( last_y ! = y ) {
last_y = y ;
ratio_y = y * ABL_BG_FACTOR ( Y_AXIS ) ;
2017-06-20 05:39:23 +02:00
const float gy = constrain ( FLOOR ( ratio_y ) , 0 , ABL_BG_POINTS_Y - FAR_EDGE_OR_BOX ) ;
2017-04-21 08:34:03 +02:00
ratio_y - = gy ;
2017-05-04 07:19:07 +02:00
# if DISABLED(EXTRAPOLATE_BEYOND_GRID)
// Beyond the grid maintain height at grid edges
NOLESS ( ratio_y , 0 ) ; // Never < 0.0. (> 1.0 is ok when nexty==gridy.)
# endif
2017-04-21 08:34:03 +02:00
gridy = gy ;
nexty = min ( gridy + 1 , ABL_BG_POINTS_Y - 1 ) ;
}
2016-10-27 01:11:26 +02:00
2017-04-21 08:34:03 +02:00
if ( last_gridx ! = gridx | | last_gridy ! = gridy ) {
last_gridx = gridx ;
last_gridy = gridy ;
// Z at the box corners
z1 = ABL_BG_GRID ( gridx , gridy ) ; // left-front
d2 = ABL_BG_GRID ( gridx , nexty ) - z1 ; // left-back (delta)
z3 = ABL_BG_GRID ( nextx , gridy ) ; // right-front
d4 = ABL_BG_GRID ( nextx , nexty ) - z3 ; // right-back (delta)
}
2016-09-27 09:38:38 +02:00
2017-04-21 08:34:03 +02:00
// Bilinear interpolate. Needed since y or gridx has changed.
L = z1 + d2 * ratio_y ; // Linear interp. LF -> LB
const float R = z3 + d4 * ratio_y ; // Linear interp. RF -> RB
D = R - L ;
}
2016-09-27 09:38:38 +02:00
2017-04-21 08:34:03 +02:00
const float offset = L + ratio_x * D ; // the offset almost always changes
2016-09-15 07:20:30 +02:00
/*
2016-09-29 08:01:38 +02:00
static float last_offset = 0 ;
2017-06-20 05:39:23 +02:00
if ( FABS ( last_offset - offset ) > 0.2 ) {
2016-09-29 08:01:38 +02:00
SERIAL_ECHOPGM ( " Sudden Shift at " ) ;
SERIAL_ECHOPAIR ( " x= " , x ) ;
SERIAL_ECHOPAIR ( " / " , bilinear_grid_spacing [ X_AXIS ] ) ;
SERIAL_ECHOLNPAIR ( " -> gridx= " , gridx ) ;
SERIAL_ECHOPAIR ( " y= " , y ) ;
SERIAL_ECHOPAIR ( " / " , bilinear_grid_spacing [ Y_AXIS ] ) ;
SERIAL_ECHOLNPAIR ( " -> gridy= " , gridy ) ;
2016-09-15 07:20:30 +02:00
SERIAL_ECHOPAIR ( " ratio_x= " , ratio_x ) ;
2016-09-29 08:01:38 +02:00
SERIAL_ECHOLNPAIR ( " ratio_y= " , ratio_y ) ;
2016-09-15 07:20:30 +02:00
SERIAL_ECHOPAIR ( " z1= " , z1 ) ;
SERIAL_ECHOPAIR ( " z2= " , z2 ) ;
SERIAL_ECHOPAIR ( " z3= " , z3 ) ;
2016-09-29 08:01:38 +02:00
SERIAL_ECHOLNPAIR ( " z4= " , z4 ) ;
2016-09-27 09:38:38 +02:00
SERIAL_ECHOPAIR ( " L= " , L ) ;
SERIAL_ECHOPAIR ( " R= " , R ) ;
2016-09-29 08:01:38 +02:00
SERIAL_ECHOLNPAIR ( " offset= " , offset ) ;
}
last_offset = offset ;
2017-04-21 08:34:03 +02:00
//*/
2016-09-15 07:20:30 +02:00
2016-09-29 08:01:38 +02:00
return offset ;
2016-09-15 07:20:30 +02:00
}
2016-09-26 06:17:39 +02:00
# endif // AUTO_BED_LEVELING_BILINEAR
2016-09-15 07:20:30 +02:00
2015-07-31 07:24:43 +02:00
# if ENABLED(DELTA)
2014-02-18 05:50:59 +01:00
2016-09-13 09:11:45 +02:00
/**
* Recalculate factors used for delta kinematics whenever
* settings have been changed ( e . g . , by M665 ) .
*/
2015-04-01 04:58:03 +02:00
void recalc_delta_settings ( float radius , float diagonal_rod ) {
2017-04-18 14:43:25 +02:00
const float trt [ ABC ] = DELTA_RADIUS_TRIM_TOWER ,
drt [ ABC ] = DELTA_DIAGONAL_ROD_TRIM_TOWER ;
2017-04-27 23:24:08 +02:00
delta_tower [ A_AXIS ] [ X_AXIS ] = cos ( RADIANS ( 210 + delta_tower_angle_trim [ A_AXIS ] ) ) * ( radius + trt [ A_AXIS ] ) ; // front left tower
delta_tower [ A_AXIS ] [ Y_AXIS ] = sin ( RADIANS ( 210 + delta_tower_angle_trim [ A_AXIS ] ) ) * ( radius + trt [ A_AXIS ] ) ;
delta_tower [ B_AXIS ] [ X_AXIS ] = cos ( RADIANS ( 330 + delta_tower_angle_trim [ B_AXIS ] ) ) * ( radius + trt [ B_AXIS ] ) ; // front right tower
delta_tower [ B_AXIS ] [ Y_AXIS ] = sin ( RADIANS ( 330 + delta_tower_angle_trim [ B_AXIS ] ) ) * ( radius + trt [ B_AXIS ] ) ;
2017-04-18 14:43:25 +02:00
delta_tower [ C_AXIS ] [ X_AXIS ] = 0.0 ; // back middle tower
delta_tower [ C_AXIS ] [ Y_AXIS ] = ( radius + trt [ C_AXIS ] ) ;
delta_diagonal_rod_2_tower [ A_AXIS ] = sq ( diagonal_rod + drt [ A_AXIS ] ) ;
delta_diagonal_rod_2_tower [ B_AXIS ] = sq ( diagonal_rod + drt [ B_AXIS ] ) ;
delta_diagonal_rod_2_tower [ C_AXIS ] = sq ( diagonal_rod + drt [ C_AXIS ] ) ;
2015-04-01 04:58:03 +02:00
}
2015-03-07 19:36:21 +01:00
2017-06-18 01:36:10 +02:00
# if ENABLED(DELTA_FAST_SQRT) && defined(ARDUINO_ARCH_AVR)
2016-09-13 09:11:45 +02:00
/**
* Fast inverse sqrt from Quake III Arena
* See : https : //en.wikipedia.org/wiki/Fast_inverse_square_root
*/
float Q_rsqrt ( float number ) {
long i ;
float x2 , y ;
const float threehalfs = 1.5f ;
x2 = number * 0.5f ;
y = number ;
i = * ( long * ) & y ; // evil floating point bit level hacking
2017-04-22 04:42:27 +02:00
i = 0x5F3759DF - ( i > > 1 ) ; // what the f***?
2016-09-13 09:11:45 +02:00
y = * ( float * ) & i ;
y = y * ( threehalfs - ( x2 * y * y ) ) ; // 1st iteration
// y = y * ( threehalfs - ( x2 * y * y ) ); // 2nd iteration, this can be removed
return y ;
}
# define _SQRT(n) (1.0f / Q_rsqrt(n))
# else
2017-06-20 05:39:23 +02:00
# define _SQRT(n) SQRT(n)
2016-09-13 09:11:45 +02:00
# endif
/**
* Delta Inverse Kinematics
*
* Calculate the tower positions for a given logical
* position , storing the result in the delta [ ] array .
*
* This is an expensive calculation , requiring 3 square
* roots per segmented linear move , and strains the limits
* of a Mega2560 with a Graphical Display .
*
* Suggested optimizations include :
*
* - Disable the home_offset ( M206 ) and / or position_shift ( G92 )
* features to remove up to 12 float additions .
*
* - Use a fast - inverse - sqrt function and add the reciprocal .
* ( see above )
*/
2015-04-01 04:58:03 +02:00
2016-09-16 20:07:43 +02:00
// Macro to obtain the Z position of an individual tower
2017-03-08 00:42:04 +01:00
# define DELTA_Z(T) raw[Z_AXIS] + _SQRT( \
delta_diagonal_rod_2_tower [ T ] - HYPOT2 ( \
delta_tower [ T ] [ X_AXIS ] - raw [ X_AXIS ] , \
delta_tower [ T ] [ Y_AXIS ] - raw [ Y_AXIS ] \
) \
2016-09-16 20:07:43 +02:00
)
2017-03-17 12:30:22 +01:00
# define DELTA_RAW_IK() do { \
2017-03-08 00:42:04 +01:00
delta [ A_AXIS ] = DELTA_Z ( A_AXIS ) ; \
delta [ B_AXIS ] = DELTA_Z ( B_AXIS ) ; \
delta [ C_AXIS ] = DELTA_Z ( C_AXIS ) ; \
2017-06-15 22:17:41 +02:00
} while ( 0 )
2016-09-22 00:43:06 +02:00
2016-09-16 20:07:43 +02:00
# define DELTA_LOGICAL_IK() do { \
const float raw [ XYZ ] = { \
RAW_X_POSITION ( logical [ X_AXIS ] ) , \
RAW_Y_POSITION ( logical [ Y_AXIS ] ) , \
RAW_Z_POSITION ( logical [ Z_AXIS ] ) \
} ; \
2016-09-22 00:43:06 +02:00
DELTA_RAW_IK ( ) ; \
2017-06-15 22:17:41 +02:00
} while ( 0 )
2016-09-16 20:07:43 +02:00
# define DELTA_DEBUG() do { \
SERIAL_ECHOPAIR ( " cartesian X: " , raw [ X_AXIS ] ) ; \
SERIAL_ECHOPAIR ( " Y: " , raw [ Y_AXIS ] ) ; \
SERIAL_ECHOLNPAIR ( " Z: " , raw [ Z_AXIS ] ) ; \
SERIAL_ECHOPAIR ( " delta A: " , delta [ A_AXIS ] ) ; \
SERIAL_ECHOPAIR ( " B: " , delta [ B_AXIS ] ) ; \
SERIAL_ECHOLNPAIR ( " C: " , delta [ C_AXIS ] ) ; \
2017-06-15 22:17:41 +02:00
} while ( 0 )
2016-09-13 09:11:45 +02:00
2016-09-16 20:07:43 +02:00
void inverse_kinematics ( const float logical [ XYZ ] ) {
DELTA_LOGICAL_IK ( ) ;
// DELTA_DEBUG();
2015-04-01 04:58:03 +02:00
}
2015-03-07 19:36:21 +01:00
2016-09-13 09:11:45 +02:00
/**
* Calculate the highest Z position where the
* effector has the full range of XY motion .
*/
2016-07-15 15:20:54 +02:00
float delta_safe_distance_from_top ( ) {
2016-08-21 05:38:32 +02:00
float cartesian [ XYZ ] = {
2016-07-25 01:34:01 +02:00
LOGICAL_X_POSITION ( 0 ) ,
LOGICAL_Y_POSITION ( 0 ) ,
LOGICAL_Z_POSITION ( 0 )
2016-07-23 02:46:05 +02:00
} ;
2016-07-22 00:46:22 +02:00
inverse_kinematics ( cartesian ) ;
2016-08-21 10:10:55 +02:00
float distance = delta [ A_AXIS ] ;
2016-07-25 01:34:01 +02:00
cartesian [ Y_AXIS ] = LOGICAL_Y_POSITION ( DELTA_PRINTABLE_RADIUS ) ;
2016-07-22 00:46:22 +02:00
inverse_kinematics ( cartesian ) ;
2017-06-20 05:39:23 +02:00
return FABS ( distance - delta [ A_AXIS ] ) ;
2016-07-15 15:20:54 +02:00
}
2016-09-13 09:11:45 +02:00
/**
* Delta Forward Kinematics
*
* See the Wikipedia article " Trilateration "
* https : //en.wikipedia.org/wiki/Trilateration
*
* Establish a new coordinate system in the plane of the
* three carriage points . This system has its origin at
* tower1 , with tower2 on the X axis . Tower3 is in the X - Y
* plane with a Z component of zero .
* We will define unit vectors in this coordinate system
* in our original coordinate system . Then when we calculate
* the Xnew , Ynew and Znew values , we can translate back into
* the original system by moving along those unit vectors
* by the corresponding values .
*
* Variable names matched to Marlin , c - version , and avoid the
* use of any vector library .
*
* by Andreas Hardtung 2016 - 06 - 07
* based on a Java function from " Delta Robot Kinematics V3 "
* by Steve Graves
*
* The result is stored in the cartes [ ] array .
*/
2016-07-22 00:46:22 +02:00
void forward_kinematics_DELTA ( float z1 , float z2 , float z3 ) {
2016-09-13 09:11:45 +02:00
// Create a vector in old coordinates along x axis of new coordinate
2017-03-08 00:42:04 +01:00
float p12 [ 3 ] = { delta_tower [ B_AXIS ] [ X_AXIS ] - delta_tower [ A_AXIS ] [ X_AXIS ] , delta_tower [ B_AXIS ] [ Y_AXIS ] - delta_tower [ A_AXIS ] [ Y_AXIS ] , z2 - z1 } ;
2016-06-07 01:44:14 +02:00
2016-09-13 09:11:45 +02:00
// Get the Magnitude of vector.
2017-06-20 05:39:23 +02:00
float d = SQRT ( sq ( p12 [ 0 ] ) + sq ( p12 [ 1 ] ) + sq ( p12 [ 2 ] ) ) ;
2016-06-07 01:44:14 +02:00
2016-09-13 09:11:45 +02:00
// Create unit vector by dividing by magnitude.
float ex [ 3 ] = { p12 [ 0 ] / d , p12 [ 1 ] / d , p12 [ 2 ] / d } ;
2016-06-07 01:44:14 +02:00
2016-09-13 09:11:45 +02:00
// Get the vector from the origin of the new system to the third point.
2017-03-08 00:42:04 +01:00
float p13 [ 3 ] = { delta_tower [ C_AXIS ] [ X_AXIS ] - delta_tower [ A_AXIS ] [ X_AXIS ] , delta_tower [ C_AXIS ] [ Y_AXIS ] - delta_tower [ A_AXIS ] [ Y_AXIS ] , z3 - z1 } ;
2016-06-07 01:44:14 +02:00
2016-09-13 09:11:45 +02:00
// Use the dot product to find the component of this vector on the X axis.
float i = ex [ 0 ] * p13 [ 0 ] + ex [ 1 ] * p13 [ 1 ] + ex [ 2 ] * p13 [ 2 ] ;
2016-06-07 01:44:14 +02:00
2016-09-13 09:11:45 +02:00
// Create a vector along the x axis that represents the x component of p13.
float iex [ 3 ] = { ex [ 0 ] * i , ex [ 1 ] * i , ex [ 2 ] * i } ;
2016-06-07 01:44:14 +02:00
2016-09-13 09:11:45 +02:00
// Subtract the X component from the original vector leaving only Y. We use the
// variable that will be the unit vector after we scale it.
float ey [ 3 ] = { p13 [ 0 ] - iex [ 0 ] , p13 [ 1 ] - iex [ 1 ] , p13 [ 2 ] - iex [ 2 ] } ;
2016-06-07 01:44:14 +02:00
2016-09-13 09:11:45 +02:00
// The magnitude of Y component
2017-06-20 05:39:23 +02:00
float j = SQRT ( sq ( ey [ 0 ] ) + sq ( ey [ 1 ] ) + sq ( ey [ 2 ] ) ) ;
2016-06-07 01:44:14 +02:00
2016-09-13 09:11:45 +02:00
// Convert to a unit vector
2016-06-07 01:44:14 +02:00
ey [ 0 ] / = j ; ey [ 1 ] / = j ; ey [ 2 ] / = j ;
2016-09-13 09:11:45 +02:00
// The cross product of the unit x and y is the unit z
// float[] ez = vectorCrossProd(ex, ey);
float ez [ 3 ] = {
ex [ 1 ] * ey [ 2 ] - ex [ 2 ] * ey [ 1 ] ,
ex [ 2 ] * ey [ 0 ] - ex [ 0 ] * ey [ 2 ] ,
ex [ 0 ] * ey [ 1 ] - ex [ 1 ] * ey [ 0 ]
} ;
// We now have the d, i and j values defined in Wikipedia.
// Plug them into the equations defined in Wikipedia for Xnew, Ynew and Znew
2017-03-08 00:42:04 +01:00
float Xnew = ( delta_diagonal_rod_2_tower [ A_AXIS ] - delta_diagonal_rod_2_tower [ B_AXIS ] + sq ( d ) ) / ( d * 2 ) ,
Ynew = ( ( delta_diagonal_rod_2_tower [ A_AXIS ] - delta_diagonal_rod_2_tower [ C_AXIS ] + HYPOT2 ( i , j ) ) / 2 - i * Xnew ) / j ,
2017-06-20 05:39:23 +02:00
Znew = SQRT ( delta_diagonal_rod_2_tower [ A_AXIS ] - HYPOT2 ( Xnew , Ynew ) ) ;
2016-09-13 09:11:45 +02:00
// Start from the origin of the old coordinates and add vectors in the
// old coords that represent the Xnew, Ynew and Znew to find the point
// in the old system.
2017-03-08 00:42:04 +01:00
cartes [ X_AXIS ] = delta_tower [ A_AXIS ] [ X_AXIS ] + ex [ 0 ] * Xnew + ey [ 0 ] * Ynew - ez [ 0 ] * Znew ;
cartes [ Y_AXIS ] = delta_tower [ A_AXIS ] [ Y_AXIS ] + ex [ 1 ] * Xnew + ey [ 1 ] * Ynew - ez [ 1 ] * Znew ;
2016-09-13 09:11:45 +02:00
cartes [ Z_AXIS ] = z1 + ex [ 2 ] * Xnew + ey [ 2 ] * Ynew - ez [ 2 ] * Znew ;
2016-10-13 17:51:44 +02:00
}
2016-06-07 01:44:14 +02:00
2016-08-21 05:38:32 +02:00
void forward_kinematics_DELTA ( float point [ ABC ] ) {
forward_kinematics_DELTA ( point [ A_AXIS ] , point [ B_AXIS ] , point [ C_AXIS ] ) ;
2016-07-21 18:05:48 +02:00
}
2015-04-01 04:58:03 +02:00
# endif // DELTA
2012-12-10 10:04:12 +01:00
2016-09-13 09:11:45 +02:00
/**
* Get the stepper positions in the cartes [ ] array .
* Forward kinematics are applied for DELTA and SCARA .
*
* The result is in the current coordinate space with
* leveling applied . The coordinates need to be run through
* unapply_leveling to obtain the " ideal " coordinates
* suitable for current_position , etc .
*/
void get_cartesian_from_steppers ( ) {
# if ENABLED(DELTA)
forward_kinematics_DELTA (
stepper . get_axis_position_mm ( A_AXIS ) ,
stepper . get_axis_position_mm ( B_AXIS ) ,
stepper . get_axis_position_mm ( C_AXIS )
) ;
2016-09-15 22:07:14 +02:00
cartes [ X_AXIS ] + = LOGICAL_X_POSITION ( 0 ) ;
cartes [ Y_AXIS ] + = LOGICAL_Y_POSITION ( 0 ) ;
cartes [ Z_AXIS ] + = LOGICAL_Z_POSITION ( 0 ) ;
2016-09-13 09:11:45 +02:00
# elif IS_SCARA
forward_kinematics_SCARA (
stepper . get_axis_position_degrees ( A_AXIS ) ,
stepper . get_axis_position_degrees ( B_AXIS )
) ;
2016-09-15 22:07:14 +02:00
cartes [ X_AXIS ] + = LOGICAL_X_POSITION ( 0 ) ;
cartes [ Y_AXIS ] + = LOGICAL_Y_POSITION ( 0 ) ;
2016-09-13 09:11:45 +02:00
cartes [ Z_AXIS ] = stepper . get_axis_position_mm ( Z_AXIS ) ;
# else
cartes [ X_AXIS ] = stepper . get_axis_position_mm ( X_AXIS ) ;
cartes [ Y_AXIS ] = stepper . get_axis_position_mm ( Y_AXIS ) ;
cartes [ Z_AXIS ] = stepper . get_axis_position_mm ( Z_AXIS ) ;
# endif
}
/**
* Set the current_position for an axis based on
* the stepper positions , removing any leveling that
* may have been applied .
*/
2016-09-15 06:50:17 +02:00
void set_current_from_steppers_for_axis ( const AxisEnum axis ) {
2016-09-15 22:02:36 +02:00
get_cartesian_from_steppers ( ) ;
2017-05-01 23:13:09 +02:00
# if PLANNER_LEVELING
2016-09-23 10:01:27 +02:00
planner . unapply_leveling ( cartes ) ;
2016-07-21 23:35:48 +02:00
# endif
2016-09-15 22:02:36 +02:00
if ( axis = = ALL_AXES )
2017-03-05 02:19:06 +01:00
COPY ( current_position , cartes ) ;
2016-09-15 22:02:36 +02:00
else
current_position [ axis ] = cartes [ axis ] ;
2016-07-21 23:35:48 +02:00
}
2015-07-31 07:24:43 +02:00
# if ENABLED(MESH_BED_LEVELING)
2015-04-01 04:58:03 +02:00
2016-09-13 09:11:45 +02:00
/**
* Prepare a mesh - leveled linear move in a Cartesian setup ,
* splitting the move where it crosses mesh borders .
*/
2017-04-22 04:42:27 +02:00
void mesh_line_to_destination ( float fr_mm_s , uint8_t x_splits = 0xFF , uint8_t y_splits = 0xFF ) {
2017-04-15 04:41:21 +02:00
int cx1 = mbl . cell_index_x ( RAW_CURRENT_POSITION ( X ) ) ,
cy1 = mbl . cell_index_y ( RAW_CURRENT_POSITION ( Y ) ) ,
2016-09-13 09:11:45 +02:00
cx2 = mbl . cell_index_x ( RAW_X_POSITION ( destination [ X_AXIS ] ) ) ,
cy2 = mbl . cell_index_y ( RAW_Y_POSITION ( destination [ Y_AXIS ] ) ) ;
2017-04-06 05:29:44 +02:00
NOMORE ( cx1 , GRID_MAX_POINTS_X - 2 ) ;
NOMORE ( cy1 , GRID_MAX_POINTS_Y - 2 ) ;
NOMORE ( cx2 , GRID_MAX_POINTS_X - 2 ) ;
NOMORE ( cy2 , GRID_MAX_POINTS_Y - 2 ) ;
2016-09-13 09:11:45 +02:00
if ( cx1 = = cx2 & & cy1 = = cy2 ) {
// Start and end on same mesh square
line_to_destination ( fr_mm_s ) ;
set_current_to_destination ( ) ;
return ;
}
2016-07-17 04:08:54 +02:00
2016-09-13 09:11:45 +02:00
# define MBL_SEGMENT_END(A) (current_position[A ##_AXIS] + (destination[A ##_AXIS] - current_position[A ##_AXIS]) * normalized_dist)
2016-07-17 04:20:16 +02:00
2016-11-03 10:43:59 +01:00
float normalized_dist , end [ XYZE ] ;
2016-07-17 21:21:28 +02:00
2016-09-13 09:11:45 +02:00
// Split at the left/front border of the right/top square
2017-03-24 06:53:28 +01:00
const int8_t gcx = max ( cx1 , cx2 ) , gcy = max ( cy1 , cy2 ) ;
2016-09-13 09:11:45 +02:00
if ( cx2 ! = cx1 & & TEST ( x_splits , gcx ) ) {
2017-03-05 02:19:06 +01:00
COPY ( end , destination ) ;
2017-03-31 08:28:07 +02:00
destination [ X_AXIS ] = LOGICAL_X_POSITION ( mbl . index_to_xpos [ gcx ] ) ;
2016-09-13 09:11:45 +02:00
normalized_dist = ( destination [ X_AXIS ] - current_position [ X_AXIS ] ) / ( end [ X_AXIS ] - current_position [ X_AXIS ] ) ;
destination [ Y_AXIS ] = MBL_SEGMENT_END ( Y ) ;
CBI ( x_splits , gcx ) ;
}
else if ( cy2 ! = cy1 & & TEST ( y_splits , gcy ) ) {
2017-03-05 02:19:06 +01:00
COPY ( end , destination ) ;
2017-03-31 08:28:07 +02:00
destination [ Y_AXIS ] = LOGICAL_Y_POSITION ( mbl . index_to_ypos [ gcy ] ) ;
2016-09-13 09:11:45 +02:00
normalized_dist = ( destination [ Y_AXIS ] - current_position [ Y_AXIS ] ) / ( end [ Y_AXIS ] - current_position [ Y_AXIS ] ) ;
destination [ X_AXIS ] = MBL_SEGMENT_END ( X ) ;
CBI ( y_splits , gcy ) ;
}
else {
// Already split on a border
line_to_destination ( fr_mm_s ) ;
set_current_to_destination ( ) ;
return ;
}
2016-07-17 04:16:12 +02:00
2016-09-13 09:11:45 +02:00
destination [ Z_AXIS ] = MBL_SEGMENT_END ( Z ) ;
destination [ E_AXIS ] = MBL_SEGMENT_END ( E ) ;
2016-07-17 04:28:13 +02:00
2016-09-13 09:11:45 +02:00
// Do the split and look for more borders
mesh_line_to_destination ( fr_mm_s , x_splits , y_splits ) ;
// Restore destination from stack
2017-03-05 02:19:06 +01:00
COPY ( destination , end ) ;
2016-09-13 09:11:45 +02:00
mesh_line_to_destination ( fr_mm_s , x_splits , y_splits ) ;
}
2016-07-17 04:28:13 +02:00
2016-11-03 20:09:51 +01:00
# elif ENABLED(AUTO_BED_LEVELING_BILINEAR) && !IS_KINEMATIC
2016-11-01 13:19:00 +01:00
2017-04-21 08:34:03 +02:00
# define CELL_INDEX(A,V) ((RAW_##A##_POSITION(V) - bilinear_start[A##_AXIS]) * ABL_BG_FACTOR(A##_AXIS))
2016-11-03 10:43:59 +01:00
2016-11-01 13:19:00 +01:00
/**
2016-11-03 10:43:59 +01:00
* Prepare a bilinear - leveled linear move on Cartesian ,
* splitting the move where it crosses grid borders .
2016-11-01 13:19:00 +01:00
*/
2016-11-03 10:43:59 +01:00
void bilinear_line_to_destination ( float fr_mm_s , uint16_t x_splits = 0xFFFF , uint16_t y_splits = 0xFFFF ) {
int cx1 = CELL_INDEX ( X , current_position [ X_AXIS ] ) ,
cy1 = CELL_INDEX ( Y , current_position [ Y_AXIS ] ) ,
cx2 = CELL_INDEX ( X , destination [ X_AXIS ] ) ,
cy2 = CELL_INDEX ( Y , destination [ Y_AXIS ] ) ;
2016-11-05 17:01:26 +01:00
cx1 = constrain ( cx1 , 0 , ABL_BG_POINTS_X - 2 ) ;
cy1 = constrain ( cy1 , 0 , ABL_BG_POINTS_Y - 2 ) ;
cx2 = constrain ( cx2 , 0 , ABL_BG_POINTS_X - 2 ) ;
cy2 = constrain ( cy2 , 0 , ABL_BG_POINTS_Y - 2 ) ;
2016-11-01 13:19:00 +01:00
if ( cx1 = = cx2 & & cy1 = = cy2 ) {
// Start and end on same mesh square
line_to_destination ( fr_mm_s ) ;
set_current_to_destination ( ) ;
return ;
}
# define LINE_SEGMENT_END(A) (current_position[A ##_AXIS] + (destination[A ##_AXIS] - current_position[A ##_AXIS]) * normalized_dist)
2016-11-03 10:43:59 +01:00
float normalized_dist , end [ XYZE ] ;
2016-11-01 13:19:00 +01:00
// Split at the left/front border of the right/top square
2017-03-24 06:53:28 +01:00
const int8_t gcx = max ( cx1 , cx2 ) , gcy = max ( cy1 , cy2 ) ;
2016-11-01 13:19:00 +01:00
if ( cx2 ! = cx1 & & TEST ( x_splits , gcx ) ) {
2017-03-05 02:19:06 +01:00
COPY ( end , destination ) ;
2016-11-05 17:01:26 +01:00
destination [ X_AXIS ] = LOGICAL_X_POSITION ( bilinear_start [ X_AXIS ] + ABL_BG_SPACING ( X_AXIS ) * gcx ) ;
2016-11-01 13:19:00 +01:00
normalized_dist = ( destination [ X_AXIS ] - current_position [ X_AXIS ] ) / ( end [ X_AXIS ] - current_position [ X_AXIS ] ) ;
destination [ Y_AXIS ] = LINE_SEGMENT_END ( Y ) ;
CBI ( x_splits , gcx ) ;
}
else if ( cy2 ! = cy1 & & TEST ( y_splits , gcy ) ) {
2017-03-05 02:19:06 +01:00
COPY ( end , destination ) ;
2016-11-05 17:01:26 +01:00
destination [ Y_AXIS ] = LOGICAL_Y_POSITION ( bilinear_start [ Y_AXIS ] + ABL_BG_SPACING ( Y_AXIS ) * gcy ) ;
2016-11-01 13:19:00 +01:00
normalized_dist = ( destination [ Y_AXIS ] - current_position [ Y_AXIS ] ) / ( end [ Y_AXIS ] - current_position [ Y_AXIS ] ) ;
destination [ X_AXIS ] = LINE_SEGMENT_END ( X ) ;
CBI ( y_splits , gcy ) ;
}
else {
// Already split on a border
line_to_destination ( fr_mm_s ) ;
set_current_to_destination ( ) ;
return ;
}
destination [ Z_AXIS ] = LINE_SEGMENT_END ( Z ) ;
destination [ E_AXIS ] = LINE_SEGMENT_END ( E ) ;
// Do the split and look for more borders
bilinear_line_to_destination ( fr_mm_s , x_splits , y_splits ) ;
// Restore destination from stack
2017-03-05 02:19:06 +01:00
COPY ( destination , end ) ;
2016-11-01 13:19:00 +01:00
bilinear_line_to_destination ( fr_mm_s , x_splits , y_splits ) ;
}
# endif // AUTO_BED_LEVELING_BILINEAR
2015-03-15 10:43:26 +01:00
2017-05-12 08:05:11 +02:00
# if IS_KINEMATIC && !UBL_DELTA
2015-03-30 08:16:12 +02:00
2016-09-13 09:11:45 +02:00
/**
* Prepare a linear move in a DELTA or SCARA setup .
*
* This calls planner . buffer_line several times , adding
* small incremental moves for DELTA or SCARA .
*/
2017-03-18 03:14:05 +01:00
inline bool prepare_kinematic_move_to ( float ltarget [ XYZE ] ) {
2016-09-15 22:38:48 +02:00
// Get the top feedrate of the move in the XY plane
2017-05-12 06:22:58 +02:00
const float _feedrate_mm_s = MMS_SCALED ( feedrate_mm_s ) ;
2016-09-15 22:38:48 +02:00
2016-09-23 08:46:52 +02:00
// If the move is only in Z/E don't split up the move
if ( ltarget [ X_AXIS ] = = current_position [ X_AXIS ] & & ltarget [ Y_AXIS ] = = current_position [ Y_AXIS ] ) {
2016-10-06 10:56:05 +02:00
planner . buffer_line_kinematic ( ltarget , _feedrate_mm_s , active_extruder ) ;
2017-04-15 06:48:30 +02:00
return false ;
2016-09-15 22:38:48 +02:00
}
2017-05-12 05:33:47 +02:00
// Fail if attempting move outside printable radius
2017-05-12 08:05:11 +02:00
if ( ! position_is_reachable_xy ( ltarget [ X_AXIS ] , ltarget [ Y_AXIS ] ) ) return true ;
2017-05-12 05:33:47 +02:00
2016-09-23 08:46:52 +02:00
// Get the cartesian distances moved in XYZE
2017-04-27 17:36:15 +02:00
const float difference [ XYZE ] = {
ltarget [ X_AXIS ] - current_position [ X_AXIS ] ,
ltarget [ Y_AXIS ] - current_position [ Y_AXIS ] ,
ltarget [ Z_AXIS ] - current_position [ Z_AXIS ] ,
ltarget [ E_AXIS ] - current_position [ E_AXIS ]
} ;
2015-03-30 08:16:12 +02:00
2016-09-23 08:46:52 +02:00
// Get the linear distance in XYZ
2017-06-20 05:39:23 +02:00
float cartesian_mm = SQRT ( sq ( difference [ X_AXIS ] ) + sq ( difference [ Y_AXIS ] ) + sq ( difference [ Z_AXIS ] ) ) ;
2016-09-23 08:46:52 +02:00
// If the move is very short, check the E move distance
2017-06-20 05:39:23 +02:00
if ( UNEAR_ZERO ( cartesian_mm ) ) cartesian_mm = FABS ( difference [ E_AXIS ] ) ;
2016-09-23 08:46:52 +02:00
// No E move either? Game over.
2017-04-15 06:48:30 +02:00
if ( UNEAR_ZERO ( cartesian_mm ) ) return true ;
2016-09-15 22:38:48 +02:00
// Minimum number of seconds to move the given distance
2017-05-12 06:22:58 +02:00
const float seconds = cartesian_mm / _feedrate_mm_s ;
2015-03-30 08:16:12 +02:00
2016-09-15 22:38:48 +02:00
// The number of segments-per-second times the duration
2016-09-23 08:46:52 +02:00
// gives the number of segments
2016-09-15 22:38:48 +02:00
uint16_t segments = delta_segments_per_second * seconds ;
2015-03-30 08:16:12 +02:00
2017-03-28 23:37:11 +02:00
// For SCARA minimum segment size is 0.25mm
2016-09-15 22:38:48 +02:00
# if IS_SCARA
2017-03-28 23:37:11 +02:00
NOMORE ( segments , cartesian_mm * 4 ) ;
2016-09-15 22:38:48 +02:00
# endif
2015-05-12 11:08:20 +02:00
2016-09-23 08:46:52 +02:00
// At least one segment is required
2016-09-15 22:38:48 +02:00
NOLESS ( segments , 1 ) ;
2016-09-23 08:46:52 +02:00
// The approximate length of each segment
2017-03-28 23:37:11 +02:00
const float inv_segments = 1.0 / float ( segments ) ,
segment_distance [ XYZE ] = {
difference [ X_AXIS ] * inv_segments ,
difference [ Y_AXIS ] * inv_segments ,
difference [ Z_AXIS ] * inv_segments ,
difference [ E_AXIS ] * inv_segments
} ;
2015-03-30 08:16:12 +02:00
2016-09-13 00:49:35 +02:00
// SERIAL_ECHOPAIR("mm=", cartesian_mm);
// SERIAL_ECHOPAIR(" seconds=", seconds);
2016-09-15 22:38:48 +02:00
// SERIAL_ECHOLNPAIR(" segments=", segments);
2015-03-30 08:16:12 +02:00
2017-05-08 21:41:03 +02:00
# if IS_SCARA && ENABLED(SCARA_FEEDRATE_SCALING)
2017-03-28 23:37:11 +02:00
// SCARA needs to scale the feed rate from mm/s to degrees/s
const float inv_segment_length = min ( 10.0 , float ( segments ) / cartesian_mm ) , // 1/mm/segs
feed_factor = inv_segment_length * _feedrate_mm_s ;
float oldA = stepper . get_axis_position_degrees ( A_AXIS ) ,
oldB = stepper . get_axis_position_degrees ( B_AXIS ) ;
# endif
2016-09-23 08:46:52 +02:00
2017-03-17 12:30:22 +01:00
// Get the logical current position as starting point
float logical [ XYZE ] ;
COPY ( logical , current_position ) ;
2015-05-12 11:08:20 +02:00
2017-03-28 23:37:11 +02:00
// Drop one segment so the last move is to the exact target.
// If there's only 1 segment, loops will be skipped entirely.
- - segments ;
2017-03-17 12:32:11 +01:00
// Calculate and execute the segments
for ( uint16_t s = segments + 1 ; - - s ; ) {
LOOP_XYZE ( i ) logical [ i ] + = segment_distance [ i ] ;
2016-09-16 20:07:43 +02:00
# if ENABLED(DELTA)
2017-03-17 12:32:11 +01:00
DELTA_LOGICAL_IK ( ) ; // Delta can inline its kinematics
2016-09-16 20:07:43 +02:00
# else
2017-03-17 12:32:11 +01:00
inverse_kinematics ( logical ) ;
2016-09-16 20:07:43 +02:00
# endif
2017-03-28 23:37:11 +02:00
2017-03-17 12:32:11 +01:00
ADJUST_DELTA ( logical ) ; // Adjust Z if bed leveling is enabled
2017-03-28 23:37:11 +02:00
2017-05-08 21:41:03 +02:00
# if IS_SCARA && ENABLED(SCARA_FEEDRATE_SCALING)
2017-03-28 23:37:11 +02:00
// For SCARA scale the feed rate from mm/s to degrees/s
// Use ratio between the length of the move and the larger angle change
const float adiff = abs ( delta [ A_AXIS ] - oldA ) ,
bdiff = abs ( delta [ B_AXIS ] - oldB ) ;
planner . buffer_line ( delta [ A_AXIS ] , delta [ B_AXIS ] , delta [ C_AXIS ] , logical [ E_AXIS ] , max ( adiff , bdiff ) * feed_factor , active_extruder ) ;
oldA = delta [ A_AXIS ] ;
oldB = delta [ B_AXIS ] ;
# else
planner . buffer_line ( delta [ A_AXIS ] , delta [ B_AXIS ] , delta [ C_AXIS ] , logical [ E_AXIS ] , _feedrate_mm_s , active_extruder ) ;
# endif
2017-03-17 12:32:11 +01:00
}
2015-05-12 11:08:20 +02:00
2016-09-23 08:46:52 +02:00
// Since segment_distance is only approximate,
// the final move must be to the exact destination.
2017-03-28 23:37:11 +02:00
2017-05-08 21:41:03 +02:00
# if IS_SCARA && ENABLED(SCARA_FEEDRATE_SCALING)
2017-03-28 23:37:11 +02:00
// For SCARA scale the feed rate from mm/s to degrees/s
// With segments > 1 length is 1 segment, otherwise total length
inverse_kinematics ( ltarget ) ;
2017-05-12 05:33:47 +02:00
ADJUST_DELTA ( ltarget ) ;
2017-03-28 23:37:11 +02:00
const float adiff = abs ( delta [ A_AXIS ] - oldA ) ,
bdiff = abs ( delta [ B_AXIS ] - oldB ) ;
planner . buffer_line ( delta [ A_AXIS ] , delta [ B_AXIS ] , delta [ C_AXIS ] , logical [ E_AXIS ] , max ( adiff , bdiff ) * feed_factor , active_extruder ) ;
# else
planner . buffer_line_kinematic ( ltarget , _feedrate_mm_s , active_extruder ) ;
# endif
2017-04-15 06:48:30 +02:00
return false ;
2015-05-12 11:08:20 +02:00
}
2015-03-30 08:16:12 +02:00
2017-05-12 08:05:11 +02:00
# else // !IS_KINEMATIC || UBL_DELTA
2016-09-13 09:11:45 +02:00
/**
* Prepare a linear move in a Cartesian setup .
* If Mesh Bed Leveling is enabled , perform a mesh move .
2017-04-02 05:29:35 +02:00
*
* Returns true if the caller didn ' t update current_position .
2016-09-13 09:11:45 +02:00
*/
inline bool prepare_move_to_destination_cartesian ( ) {
2017-05-13 11:09:21 +02:00
# if ENABLED(AUTO_BED_LEVELING_UBL)
const float fr_scaled = MMS_SCALED ( feedrate_mm_s ) ;
2017-05-28 02:29:29 +02:00
if ( ubl . state . active ) { // direct use of ubl.state.active for speed
2017-05-22 21:41:09 +02:00
ubl . line_to_destination_cartesian ( fr_scaled , active_extruder ) ;
2017-05-13 11:09:21 +02:00
return true ;
}
else
line_to_destination ( fr_scaled ) ;
# else
// Do not use feedrate_percentage for E or Z only moves
if ( current_position [ X_AXIS ] = = destination [ X_AXIS ] & & current_position [ Y_AXIS ] = = destination [ Y_AXIS ] )
line_to_destination ( ) ;
else {
const float fr_scaled = MMS_SCALED ( feedrate_mm_s ) ;
# if ENABLED(MESH_BED_LEVELING)
2017-05-28 02:29:29 +02:00
if ( mbl . active ( ) ) { // direct used of mbl.active() for speed
2017-05-13 11:09:21 +02:00
mesh_line_to_destination ( fr_scaled ) ;
return true ;
}
else
# elif ENABLED(AUTO_BED_LEVELING_BILINEAR)
2017-05-28 02:29:29 +02:00
if ( planner . abl_enabled ) { // direct use of abl_enabled for speed
2017-05-13 11:09:21 +02:00
bilinear_line_to_destination ( fr_scaled ) ;
return true ;
}
else
# endif
line_to_destination ( fr_scaled ) ;
}
# endif
2017-04-15 06:48:30 +02:00
return false ;
2016-09-13 09:11:45 +02:00
}
2017-05-12 08:05:11 +02:00
# endif // !IS_KINEMATIC || UBL_DELTA
2013-08-07 16:10:26 +02:00
2015-07-31 07:24:43 +02:00
# if ENABLED(DUAL_X_CARRIAGE)
2015-05-12 11:08:20 +02:00
2016-09-13 09:11:45 +02:00
/**
* Prepare a linear move in a dual X axis setup
*/
2016-07-01 00:37:32 +02:00
inline bool prepare_move_to_destination_dualx ( ) {
2015-04-04 08:42:50 +02:00
if ( active_extruder_parked ) {
2016-11-03 03:38:26 +01:00
switch ( dual_x_carriage_mode ) {
case DXC_FULL_CONTROL_MODE :
break ;
case DXC_AUTO_PARK_MODE :
if ( current_position [ E_AXIS ] = = destination [ E_AXIS ] ) {
// This is a travel move (with no extrusion)
// Skip it, but keep track of the current position
// (so it can be used as the start of the next non-travel move)
if ( delayed_move_time ! = 0xFFFFFFFFUL ) {
set_current_to_destination ( ) ;
NOLESS ( raised_parked_position [ Z_AXIS ] , destination [ Z_AXIS ] ) ;
delayed_move_time = millis ( ) ;
2017-04-15 06:48:30 +02:00
return true ;
2016-11-03 03:38:26 +01:00
}
}
// unpark extruder: 1) raise, 2) move into starting XY position, 3) lower
2016-11-07 07:33:12 +01:00
for ( uint8_t i = 0 ; i < 3 ; i + + )
planner . buffer_line (
i = = 0 ? raised_parked_position [ X_AXIS ] : current_position [ X_AXIS ] ,
i = = 0 ? raised_parked_position [ Y_AXIS ] : current_position [ Y_AXIS ] ,
i = = 2 ? current_position [ Z_AXIS ] : raised_parked_position [ Z_AXIS ] ,
current_position [ E_AXIS ] ,
i = = 1 ? PLANNER_XY_FEEDRATE ( ) : planner . max_feedrate_mm_s [ Z_AXIS ] ,
active_extruder
) ;
delayed_move_time = 0 ;
2016-11-03 03:38:26 +01:00
active_extruder_parked = false ;
2017-04-15 06:47:53 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) SERIAL_ECHOLNPGM ( " Clear active_extruder_parked " ) ;
# endif
2016-11-03 03:38:26 +01:00
break ;
2016-11-07 07:18:44 +01:00
case DXC_DUPLICATION_MODE :
if ( active_extruder = = 0 ) {
2017-04-15 06:47:53 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) {
SERIAL_ECHOPAIR ( " Set planner X " , LOGICAL_X_POSITION ( inactive_extruder_x_pos ) ) ;
SERIAL_ECHOLNPAIR ( " ... Line to X " , current_position [ X_AXIS ] + duplicate_extruder_x_offset ) ;
}
# endif
2016-11-07 07:18:44 +01:00
// move duplicate extruder into correct duplication position.
planner . set_position_mm (
LOGICAL_X_POSITION ( inactive_extruder_x_pos ) ,
current_position [ Y_AXIS ] ,
current_position [ Z_AXIS ] ,
current_position [ E_AXIS ]
) ;
2016-11-07 07:21:45 +01:00
planner . buffer_line (
current_position [ X_AXIS ] + duplicate_extruder_x_offset ,
current_position [ Y_AXIS ] , current_position [ Z_AXIS ] , current_position [ E_AXIS ] ,
planner . max_feedrate_mm_s [ X_AXIS ] , 1
) ;
2016-11-07 07:18:44 +01:00
SYNC_PLAN_POSITION_KINEMATIC ( ) ;
stepper . synchronize ( ) ;
extruder_duplication_enabled = true ;
active_extruder_parked = false ;
2017-04-15 06:47:53 +02:00
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) SERIAL_ECHOLNPGM ( " Set extruder_duplication_enabled \n Clear active_extruder_parked " ) ;
# endif
}
else {
# if ENABLED(DEBUG_LEVELING_FEATURE)
if ( DEBUGGING ( LEVELING ) ) SERIAL_ECHOLNPGM ( " Active extruder not 0 " ) ;
# endif
2016-11-07 07:18:44 +01:00
}
break ;
2013-08-07 16:10:26 +02:00
}
}
2017-04-15 06:48:30 +02:00
return false ;
2015-05-12 11:08:20 +02:00
}
2013-08-07 16:10:26 +02:00
2015-05-12 11:08:20 +02:00
# endif // DUAL_X_CARRIAGE
/**
* Prepare a single move and get ready for the next one
2015-06-28 06:49:37 +02:00
*
2016-09-13 09:11:45 +02:00
* This may result in several calls to planner . buffer_line to
* do smaller moves for DELTA , SCARA , mesh moves , etc .
2015-05-12 11:08:20 +02:00
*/
2016-06-11 04:23:41 +02:00
void prepare_move_to_destination ( ) {
2015-05-12 11:08:20 +02:00
clamp_to_software_endstops ( destination ) ;
refresh_cmd_timeout ( ) ;
2016-08-21 06:34:24 +02:00
# if ENABLED(PREVENT_COLD_EXTRUSION)
2016-09-13 09:11:45 +02:00
if ( ! DEBUGGING ( DRYRUN ) ) {
if ( destination [ E_AXIS ] ! = current_position [ E_AXIS ] ) {
if ( thermalManager . tooColdToExtrude ( active_extruder ) ) {
current_position [ E_AXIS ] = destination [ E_AXIS ] ; // Behave as if the move really took place, but ignore E part
2017-06-09 17:51:23 +02:00
SERIAL_ECHO_START ( ) ;
2016-09-13 09:11:45 +02:00
SERIAL_ECHOLNPGM ( MSG_ERR_COLD_EXTRUDE_STOP ) ;
}
# if ENABLED(PREVENT_LENGTHY_EXTRUDE)
2016-09-29 07:32:33 +02:00
if ( destination [ E_AXIS ] - current_position [ E_AXIS ] > EXTRUDE_MAXLENGTH ) {
2016-09-13 09:11:45 +02:00
current_position [ E_AXIS ] = destination [ E_AXIS ] ; // Behave as if the move really took place, but ignore E part
2017-06-09 17:51:23 +02:00
SERIAL_ECHO_START ( ) ;
2016-09-13 09:11:45 +02:00
SERIAL_ECHOLNPGM ( MSG_ERR_LONG_EXTRUDE_STOP ) ;
}
# endif
}
}
2015-05-12 11:08:20 +02:00
# endif
2017-05-12 08:05:11 +02:00
if (
2017-05-29 16:36:41 +02:00
# if UBL_DELTA // Also works for CARTESIAN (smaller segments follow mesh more closely)
ubl . prepare_segmented_line_to ( destination , feedrate_mm_s )
# elif IS_KINEMATIC
prepare_kinematic_move_to ( destination )
2017-05-12 08:05:11 +02:00
# elif ENABLED(DUAL_X_CARRIAGE)
2017-07-31 18:39:33 +02:00
prepare_move_to_destination_dualx ( ) | | prepare_move_to_destination_cartesian ( )
2017-05-12 05:33:47 +02:00
# else
2017-05-12 08:05:11 +02:00
prepare_move_to_destination_cartesian ( )
2016-04-26 05:02:24 +02:00
# endif
2017-05-12 08:05:11 +02:00
) return ;
2014-06-23 17:09:57 +02:00
2015-04-04 08:42:50 +02:00
set_current_to_destination ( ) ;
2012-11-06 12:06:41 +01:00
}
2017-05-04 23:38:29 +02:00
# if ENABLED(USE_CONTROLLER_FAN)
2013-06-07 00:49:25 +02:00
2015-05-12 11:08:20 +02:00
void controllerFan ( ) {
2017-03-29 02:45:54 +02:00
static millis_t lastMotorOn = 0 , // Last time a motor was turned on
nextMotorCheck = 0 ; // Last time the state was checked
const millis_t ms = millis ( ) ;
2016-04-11 00:55:12 +02:00
if ( ELAPSED ( ms , nextMotorCheck ) ) {
nextMotorCheck = ms + 2500UL ; // Not a time critical function, so only check every 2.5s
2017-04-20 21:04:26 +02:00
if ( X_ENABLE_READ = = X_ENABLE_ON | | Y_ENABLE_READ = = Y_ENABLE_ON | | Z_ENABLE_READ = = Z_ENABLE_ON | | thermalManager . soft_pwm_amount_bed > 0
2015-10-03 08:08:58 +02:00
| | E0_ENABLE_READ = = E_ENABLE_ON // If any of the drivers are enabled...
2016-06-29 00:06:56 +02:00
# if E_STEPPERS > 1
2015-10-03 08:08:58 +02:00
| | E1_ENABLE_READ = = E_ENABLE_ON
# if HAS_X2_ENABLE
| | X2_ENABLE_READ = = X_ENABLE_ON
# endif
2016-06-29 00:06:56 +02:00
# if E_STEPPERS > 2
2015-10-03 08:08:58 +02:00
| | E2_ENABLE_READ = = E_ENABLE_ON
2016-06-29 00:06:56 +02:00
# if E_STEPPERS > 3
2015-10-03 08:08:58 +02:00
| | E3_ENABLE_READ = = E_ENABLE_ON
2017-04-06 23:46:52 +02:00
# if E_STEPPERS > 4
| | E4_ENABLE_READ = = E_ENABLE_ON
# endif // E_STEPPERS > 4
# endif // E_STEPPERS > 3
# endif // E_STEPPERS > 2
# endif // E_STEPPERS > 1
2015-05-12 11:08:20 +02:00
) {
2016-03-21 02:48:31 +01:00
lastMotorOn = ms ; //... set time to NOW so the fan will turn on
2015-05-12 11:08:20 +02:00
}
2016-03-21 02:48:31 +01:00
// Fan off if no steppers have been enabled for CONTROLLERFAN_SECS seconds
2016-04-11 00:55:12 +02:00
uint8_t speed = ( ! lastMotorOn | | ELAPSED ( ms , lastMotorOn + ( CONTROLLERFAN_SECS ) * 1000UL ) ) ? 0 : CONTROLLERFAN_SPEED ;
2016-03-21 02:48:31 +01:00
2015-05-12 11:08:20 +02:00
// allows digital or PWM fan output to be used (see M42 handling)
2017-05-03 03:18:31 +02:00
WRITE ( CONTROLLER_FAN_PIN , speed ) ;
analogWrite ( CONTROLLER_FAN_PIN , speed ) ;
2012-11-06 12:06:41 +01:00
}
}
2015-05-12 11:08:20 +02:00
2017-05-03 03:18:31 +02:00
# endif // USE_CONTROLLER_FAN
2012-11-06 12:06:41 +01:00
2016-09-15 07:28:26 +02:00
# if ENABLED(MORGAN_SCARA)
2015-05-17 10:47:02 +02:00
2016-09-15 07:28:26 +02:00
/**
* Morgan SCARA Forward Kinematics . Results in cartes [ ] .
* Maths and first version by QHARLEY .
* Integrated into Marlin and slightly restructured by Joachim Cerny .
*/
2016-09-12 10:48:29 +02:00
void forward_kinematics_SCARA ( const float & a , const float & b ) {
2015-05-17 10:47:02 +02:00
2016-09-15 07:28:26 +02:00
float a_sin = sin ( RADIANS ( a ) ) * L1 ,
a_cos = cos ( RADIANS ( a ) ) * L1 ,
b_sin = sin ( RADIANS ( b ) ) * L2 ,
b_cos = cos ( RADIANS ( b ) ) * L2 ;
2015-05-17 10:47:02 +02:00
2016-09-12 10:48:29 +02:00
cartes [ X_AXIS ] = a_cos + b_cos + SCARA_OFFSET_X ; //theta
cartes [ Y_AXIS ] = a_sin + b_sin + SCARA_OFFSET_Y ; //theta+phi
2015-05-13 11:02:19 +02:00
2016-09-13 00:49:35 +02:00
/*
2016-09-22 00:54:05 +02:00
SERIAL_ECHOPAIR ( " SCARA FK Angle a= " , a ) ;
2016-09-15 07:28:26 +02:00
SERIAL_ECHOPAIR ( " b= " , b ) ;
2016-09-13 00:49:35 +02:00
SERIAL_ECHOPAIR ( " a_sin= " , a_sin ) ;
SERIAL_ECHOPAIR ( " a_cos= " , a_cos ) ;
SERIAL_ECHOPAIR ( " b_sin= " , b_sin ) ;
SERIAL_ECHOLNPAIR ( " b_cos= " , b_cos ) ;
SERIAL_ECHOPAIR ( " cartes[X_AXIS]= " , cartes [ X_AXIS ] ) ;
SERIAL_ECHOLNPAIR ( " cartes[Y_AXIS]= " , cartes [ Y_AXIS ] ) ;
//*/
2015-08-05 13:40:36 +02:00
}
2014-06-23 17:09:57 +02:00
2016-09-15 07:28:26 +02:00
/**
* Morgan SCARA Inverse Kinematics . Results in delta [ ] .
*
* See http : //forums.reprap.org/read.php?185,283327
2016-09-26 08:30:34 +02:00
*
2016-09-15 07:28:26 +02:00
* Maths and first version by QHARLEY .
* Integrated into Marlin and slightly restructured by Joachim Cerny .
*/
2016-09-15 09:31:36 +02:00
void inverse_kinematics ( const float logical [ XYZ ] ) {
2015-08-05 13:40:36 +02:00
2016-09-12 03:11:39 +02:00
static float C2 , S2 , SK1 , SK2 , THETA , PSI ;
2015-08-05 13:40:36 +02:00
2016-09-15 09:31:36 +02:00
float sx = RAW_X_POSITION ( logical [ X_AXIS ] ) - SCARA_OFFSET_X , // Translate SCARA to standard X Y
sy = RAW_Y_POSITION ( logical [ Y_AXIS ] ) - SCARA_OFFSET_Y ; // With scaling factor.
2015-08-05 13:40:36 +02:00
2016-09-15 07:28:26 +02:00
if ( L1 = = L2 )
C2 = HYPOT2 ( sx , sy ) / L1_2_2 - 1 ;
else
C2 = ( HYPOT2 ( sx , sy ) - ( L1_2 + L2_2 ) ) / ( 2.0 * L1 * L2 ) ;
2015-08-05 13:40:36 +02:00
2017-06-20 05:39:23 +02:00
S2 = SQRT ( 1 - sq ( C2 ) ) ;
2015-08-05 13:40:36 +02:00
2016-09-15 07:28:26 +02:00
// Unrotated Arm1 plus rotated Arm2 gives the distance from Center to End
2016-09-12 03:11:39 +02:00
SK1 = L1 + L2 * C2 ;
2016-09-15 07:28:26 +02:00
// Rotated Arm2 gives the distance from Arm1 to Arm2
2016-09-12 03:11:39 +02:00
SK2 = L2 * S2 ;
2015-08-05 13:40:36 +02:00
2016-09-15 07:28:26 +02:00
// Angle of Arm1 is the difference between Center-to-End angle and the Center-to-Elbow
2017-06-20 05:39:23 +02:00
THETA = ATAN2 ( SK1 , SK2 ) - ATAN2 ( sx , sy ) ;
2016-09-15 07:28:26 +02:00
// Angle of Arm2
2017-06-20 05:39:23 +02:00
PSI = ATAN2 ( S2 , C2 ) ;
2015-08-05 13:40:36 +02:00
2016-09-12 03:11:39 +02:00
delta [ A_AXIS ] = DEGREES ( THETA ) ; // theta is support arm angle
delta [ B_AXIS ] = DEGREES ( THETA + PSI ) ; // equal to sub arm angle (inverted motor)
2016-09-15 08:21:31 +02:00
delta [ C_AXIS ] = logical [ Z_AXIS ] ;
2015-08-05 13:40:36 +02:00
2016-09-15 09:31:36 +02:00
/*
DEBUG_POS ( " SCARA IK " , logical ) ;
2016-09-12 03:11:39 +02:00
DEBUG_POS ( " SCARA IK " , delta ) ;
SERIAL_ECHOPAIR ( " SCARA (x,y) " , sx ) ;
SERIAL_ECHOPAIR ( " , " , sy ) ;
SERIAL_ECHOPAIR ( " C2= " , C2 ) ;
SERIAL_ECHOPAIR ( " S2= " , S2 ) ;
SERIAL_ECHOPAIR ( " Theta= " , THETA ) ;
SERIAL_ECHOLNPAIR ( " Phi= " , PHI ) ;
//*/
}
2016-09-15 07:28:26 +02:00
# endif // MORGAN_SCARA
2016-09-12 10:48:29 +02:00
2015-07-31 07:24:43 +02:00
# if ENABLED(TEMP_STAT_LEDS)
2015-04-13 03:07:08 +02:00
static bool red_led = false ;
static millis_t next_status_led_update_ms = 0 ;
void handle_status_leds ( void ) {
2016-04-11 00:55:12 +02:00
if ( ELAPSED ( millis ( ) , next_status_led_update_ms ) ) {
2015-04-13 03:07:08 +02:00
next_status_led_update_ms + = 500 ; // Update every 0.5s
2016-08-16 18:19:51 +02:00
float max_temp = 0.0 ;
# if HAS_TEMP_BED
max_temp = MAX3 ( max_temp , thermalManager . degTargetBed ( ) , thermalManager . degBed ( ) ) ;
# endif
2017-05-21 11:49:25 +02:00
HOTEND_LOOP ( )
2016-08-19 05:13:47 +02:00
max_temp = MAX3 ( max_temp , thermalManager . degHotend ( e ) , thermalManager . degTargetHotend ( e ) ) ;
2017-06-22 16:44:39 +02:00
const bool new_led = ( max_temp > 55.0 ) ? true : ( max_temp < 54.0 ) ? false : red_led ;
2015-04-13 03:07:08 +02:00
if ( new_led ! = red_led ) {
red_led = new_led ;
2016-10-23 03:07:48 +02:00
# if PIN_EXISTS(STAT_LED_RED)
WRITE ( STAT_LED_RED_PIN , new_led ? HIGH : LOW ) ;
# if PIN_EXISTS(STAT_LED_BLUE)
WRITE ( STAT_LED_BLUE_PIN , new_led ? LOW : HIGH ) ;
# endif
# else
WRITE ( STAT_LED_BLUE_PIN , new_led ? HIGH : LOW ) ;
# endif
2015-04-13 03:07:08 +02:00
}
2013-10-20 12:12:35 +02:00
}
}
2015-04-13 03:07:08 +02:00
2013-10-20 12:12:35 +02:00
# endif
2016-09-13 09:11:45 +02:00
# if ENABLED(FILAMENT_RUNOUT_SENSOR)
void handle_filament_runout ( ) {
if ( ! filament_ran_out ) {
filament_ran_out = true ;
enqueue_and_echo_commands_P ( PSTR ( FILAMENT_RUNOUT_SCRIPT ) ) ;
stepper . synchronize ( ) ;
}
}
# endif // FILAMENT_RUNOUT_SENSOR
2017-07-18 10:18:05 +02:00
float calculate_volumetric_multiplier ( const float diameter ) {
2016-09-13 09:11:45 +02:00
if ( ! volumetric_enabled | | diameter = = 0 ) return 1.0 ;
2017-03-24 06:53:28 +01:00
return 1.0 / ( M_PI * sq ( diameter * 0.5 ) ) ;
2016-09-13 09:11:45 +02:00
}
void calculate_volumetric_multipliers ( ) {
for ( uint8_t i = 0 ; i < COUNT ( filament_size ) ; i + + )
volumetric_multiplier [ i ] = calculate_volumetric_multiplier ( filament_size [ i ] ) ;
}
2015-04-04 04:25:22 +02:00
void enable_all_steppers ( ) {
2017-04-11 18:10:26 +02:00
enable_X ( ) ;
enable_Y ( ) ;
enable_Z ( ) ;
enable_E0 ( ) ;
enable_E1 ( ) ;
enable_E2 ( ) ;
enable_E3 ( ) ;
enable_E4 ( ) ;
2015-04-04 04:25:22 +02:00
}
2017-03-18 03:12:19 +01:00
void disable_e_steppers ( ) {
2017-04-11 18:10:26 +02:00
disable_E0 ( ) ;
disable_E1 ( ) ;
disable_E2 ( ) ;
disable_E3 ( ) ;
disable_E4 ( ) ;
2015-04-04 00:31:35 +02:00
}
2017-03-18 03:12:19 +01:00
void disable_all_steppers ( ) {
2017-04-11 18:10:26 +02:00
disable_X ( ) ;
disable_Y ( ) ;
disable_Z ( ) ;
2017-03-18 03:12:19 +01:00
disable_e_steppers ( ) ;
}
2015-04-04 00:31:35 +02:00
/**
2015-04-04 08:42:50 +02:00
* Manage several activities :
* - Check for Filament Runout
* - Keep the command buffer full
* - Check for maximum inactive time between commands
* - Check for maximum inactive time between stepper commands
* - Check if pin CHDK needs to go LOW
* - Check for KILL button held down
* - Check for HOME button held down
* - Check if cooling fan needs to be switched on
* - Check if an idle but hot extruder needs filament extruded ( EXTRUDER_RUNOUT_PREVENT )
2015-04-04 00:31:35 +02:00
*/
void manage_inactivity ( bool ignore_stepper_queue /*=false*/ ) {
2015-08-05 13:40:36 +02:00
2016-06-20 03:05:57 +02:00
# if ENABLED(FILAMENT_RUNOUT_SENSOR)
2016-11-09 17:21:21 +01:00
if ( ( IS_SD_PRINTING | | print_job_timer . isRunning ( ) ) & & ( READ ( FIL_RUNOUT_PIN ) = = FIL_RUNOUT_INVERTING ) )
2016-04-18 08:16:49 +02:00
handle_filament_runout ( ) ;
2015-04-04 00:31:35 +02:00
# endif
2016-03-28 14:25:27 +02:00
if ( commands_in_queue < BUFSIZE ) get_available_commands ( ) ;
2015-04-04 00:31:35 +02:00
2017-03-29 02:45:54 +02:00
const millis_t ms = millis ( ) ;
2015-04-04 00:31:35 +02:00
2017-03-14 18:24:46 +01:00
if ( max_inactive_time & & ELAPSED ( ms , previous_cmd_ms + max_inactive_time ) ) {
2017-06-09 17:51:23 +02:00
SERIAL_ERROR_START ( ) ;
2017-05-20 10:03:08 +02:00
SERIAL_ECHOLNPAIR ( MSG_KILL_INACTIVE_TIME , parser . command_ptr ) ;
2017-03-14 18:24:46 +01:00
kill ( PSTR ( MSG_KILLED ) ) ;
}
2017-02-10 07:13:58 +01:00
// Prevent steppers timing-out in the middle of M600
2017-05-26 20:01:02 +02:00
# if ENABLED(ADVANCED_PAUSE_FEATURE) && ENABLED(PAUSE_PARK_NO_STEPPER_TIMEOUT)
# define MOVE_AWAY_TEST !move_away_flag
2017-02-10 07:13:58 +01:00
# else
2017-05-26 20:01:02 +02:00
# define MOVE_AWAY_TEST true
2017-01-22 00:10:57 +01:00
# endif
2017-03-14 08:25:34 +01:00
2017-05-26 20:01:02 +02:00
if ( MOVE_AWAY_TEST & & stepper_inactive_time & & ELAPSED ( ms , previous_cmd_ms + stepper_inactive_time )
2016-04-28 03:06:32 +02:00
& & ! ignore_stepper_queue & & ! planner . blocks_queued ( ) ) {
2016-03-14 06:15:45 +01:00
# if ENABLED(DISABLE_INACTIVE_X)
2017-04-11 18:10:26 +02:00
disable_X ( ) ;
2015-06-01 10:42:28 +02:00
# endif
2016-03-14 06:15:45 +01:00
# if ENABLED(DISABLE_INACTIVE_Y)
2017-04-11 18:10:26 +02:00
disable_Y ( ) ;
2015-06-01 10:42:28 +02:00
# endif
2016-03-14 06:15:45 +01:00
# if ENABLED(DISABLE_INACTIVE_Z)
2017-04-11 18:10:26 +02:00
disable_Z ( ) ;
2015-06-01 10:42:28 +02:00
# endif
2016-03-14 06:15:45 +01:00
# if ENABLED(DISABLE_INACTIVE_E)
2017-03-18 03:12:19 +01:00
disable_e_steppers ( ) ;
2015-06-01 10:42:28 +02:00
# endif
2017-07-02 01:33:54 +02:00
# if ENABLED(AUTO_BED_LEVELING_UBL) && ENABLED(ULTRA_LCD) // Only needed with an LCD
ubl_lcd_map_control = defer_return_to_status = false ;
2017-06-13 01:26:49 +02:00
# endif
2015-06-01 10:42:28 +02:00
}
2015-03-07 21:43:15 +01:00
2015-04-04 08:42:50 +02:00
# ifdef CHDK // Check if pin should be set to LOW after M240 set it to HIGH
2017-03-17 03:49:39 +01:00
if ( chdkActive & & ELAPSED ( ms , chdkHigh + CHDK_DELAY ) ) {
2014-03-10 21:57:08 +01:00
chdkActive = false ;
WRITE ( CHDK , LOW ) ;
}
# endif
2015-04-04 00:31:35 +02:00
# if HAS_KILL
2015-08-05 13:40:36 +02:00
2014-12-28 19:54:06 +01:00
// Check if the kill button was pressed and wait just in case it was an accidental
// key kill key press
// -------------------------------------------------------------------------------
2015-04-04 00:45:41 +02:00
static int killCount = 0 ; // make the inactivity button a bit less responsive
const int KILL_DELAY = 750 ;
2015-04-04 00:31:35 +02:00
if ( ! READ ( KILL_PIN ) )
2015-10-03 08:08:58 +02:00
killCount + + ;
2014-12-28 19:54:06 +01:00
else if ( killCount > 0 )
2015-10-03 08:08:58 +02:00
killCount - - ;
2015-04-04 00:31:35 +02:00
2014-12-28 19:54:06 +01:00
// Exceeded threshold and we can confirm that it was not accidental
// KILL the machine
// ----------------------------------------------------------------
2017-03-14 18:24:46 +01:00
if ( killCount > = KILL_DELAY ) {
2017-06-09 17:51:23 +02:00
SERIAL_ERROR_START ( ) ;
2017-03-14 18:24:46 +01:00
SERIAL_ERRORLNPGM ( MSG_KILL_BUTTON ) ;
kill ( PSTR ( MSG_KILLED ) ) ;
}
2012-11-06 12:06:41 +01:00
# endif
2014-12-28 19:54:06 +01:00
2015-04-04 00:31:35 +02:00
# if HAS_HOME
2014-12-28 19:54:06 +01:00
// Check to see if we have to home, use poor man's debouncer
// ---------------------------------------------------------
2015-04-04 00:45:41 +02:00
static int homeDebounceCount = 0 ; // poor man's debouncing count
2015-09-13 20:20:43 +02:00
const int HOME_DEBOUNCE_DELAY = 2500 ;
2016-12-06 21:22:34 +01:00
if ( ! IS_SD_PRINTING & & ! READ ( HOME_PIN ) ) {
2015-04-04 00:31:35 +02:00
if ( ! homeDebounceCount ) {
2016-02-21 02:35:35 +01:00
enqueue_and_echo_commands_P ( PSTR ( " G28 " ) ) ;
2015-05-04 19:48:49 +02:00
LCD_MESSAGEPGM ( MSG_AUTO_HOME ) ;
2015-04-04 00:31:35 +02:00
}
if ( homeDebounceCount < HOME_DEBOUNCE_DELAY )
homeDebounceCount + + ;
else
homeDebounceCount = 0 ;
2014-12-28 19:54:06 +01:00
}
2015-04-04 00:31:35 +02:00
# endif
2015-08-05 13:40:36 +02:00
2017-05-04 23:38:29 +02:00
# if ENABLED(USE_CONTROLLER_FAN)
2015-04-04 08:42:50 +02:00
controllerFan ( ) ; // Check if fan should be turned on to cool stepper drivers down
2012-11-06 12:06:41 +01:00
# endif
2015-04-04 00:31:35 +02:00
2015-07-31 07:24:43 +02:00
# if ENABLED(EXTRUDER_RUNOUT_PREVENT)
2016-06-29 00:06:56 +02:00
if ( ELAPSED ( ms , previous_cmd_ms + ( EXTRUDER_RUNOUT_SECONDS ) * 1000UL )
& & thermalManager . degHotend ( active_extruder ) > EXTRUDER_RUNOUT_MINTEMP ) {
# if ENABLED(SWITCHING_EXTRUDER)
2017-07-06 19:10:06 +02:00
const bool oldstatus = E0_ENABLE_READ ;
2017-04-11 18:10:26 +02:00
enable_E0 ( ) ;
2016-06-29 00:06:56 +02:00
# else // !SWITCHING_EXTRUDER
2017-07-06 19:10:06 +02:00
bool oldstatus ;
2015-10-03 08:08:58 +02:00
switch ( active_extruder ) {
2017-07-06 19:10:06 +02:00
default : oldstatus = E0_ENABLE_READ ; enable_E0 ( ) ; break ;
2016-06-29 00:06:56 +02:00
# if E_STEPPERS > 1
2017-04-18 21:39:45 +02:00
case 1 : oldstatus = E1_ENABLE_READ ; enable_E1 ( ) ; break ;
2016-06-29 00:06:56 +02:00
# if E_STEPPERS > 2
2017-04-18 21:39:45 +02:00
case 2 : oldstatus = E2_ENABLE_READ ; enable_E2 ( ) ; break ;
2016-06-29 00:06:56 +02:00
# if E_STEPPERS > 3
2017-04-18 21:39:45 +02:00
case 3 : oldstatus = E3_ENABLE_READ ; enable_E3 ( ) ; break ;
2017-04-06 23:46:52 +02:00
# if E_STEPPERS > 4
2017-04-18 21:39:45 +02:00
case 4 : oldstatus = E4_ENABLE_READ ; enable_E4 ( ) ; break ;
2017-04-06 23:46:52 +02:00
# endif // E_STEPPERS > 4
# endif // E_STEPPERS > 3
# endif // E_STEPPERS > 2
# endif // E_STEPPERS > 1
2015-10-03 08:08:58 +02:00
}
2016-06-29 00:06:56 +02:00
# endif // !SWITCHING_EXTRUDER
2015-04-13 03:07:08 +02:00
previous_cmd_ms = ms ; // refresh_cmd_timeout()
2016-10-09 22:21:36 +02:00
2017-03-20 22:41:59 +01:00
const float olde = current_position [ E_AXIS ] ;
current_position [ E_AXIS ] + = EXTRUDER_RUNOUT_EXTRUDE ;
planner . buffer_line_kinematic ( current_position , MMM_TO_MMS ( EXTRUDER_RUNOUT_SPEED ) , active_extruder ) ;
current_position [ E_AXIS ] = olde ;
planner . set_e_position_mm ( olde ) ;
2016-04-27 16:15:20 +02:00
stepper . synchronize ( ) ;
2016-06-29 00:06:56 +02:00
# if ENABLED(SWITCHING_EXTRUDER)
E0_ENABLE_WRITE ( oldstatus ) ;
# else
switch ( active_extruder ) {
2017-04-18 21:39:45 +02:00
case 0 : E0_ENABLE_WRITE ( oldstatus ) ; break ;
2016-06-29 00:06:56 +02:00
# if E_STEPPERS > 1
2017-04-18 21:39:45 +02:00
case 1 : E1_ENABLE_WRITE ( oldstatus ) ; break ;
2016-06-29 00:06:56 +02:00
# if E_STEPPERS > 2
2017-04-18 21:39:45 +02:00
case 2 : E2_ENABLE_WRITE ( oldstatus ) ; break ;
2016-06-29 00:06:56 +02:00
# if E_STEPPERS > 3
2017-04-18 21:39:45 +02:00
case 3 : E3_ENABLE_WRITE ( oldstatus ) ; break ;
2017-04-06 23:46:52 +02:00
# if E_STEPPERS > 4
2017-04-18 21:39:45 +02:00
case 4 : E4_ENABLE_WRITE ( oldstatus ) ; break ;
2017-04-06 23:46:52 +02:00
# endif // E_STEPPERS > 4
# endif // E_STEPPERS > 3
# endif // E_STEPPERS > 2
# endif // E_STEPPERS > 1
2016-06-29 00:06:56 +02:00
}
# endif // !SWITCHING_EXTRUDER
2012-11-06 12:06:41 +01:00
}
2016-06-29 00:06:56 +02:00
# endif // EXTRUDER_RUNOUT_PREVENT
2015-04-04 00:31:35 +02:00
2015-07-31 07:24:43 +02:00
# if ENABLED(DUAL_X_CARRIAGE)
2013-08-07 16:10:26 +02:00
// handle delayed move timeout
2016-04-11 00:55:12 +02:00
if ( delayed_move_time & & ELAPSED ( ms , delayed_move_time + 1000UL ) & & IsRunning ( ) ) {
2013-08-07 16:10:26 +02:00
// travel moves have been received so enact them
delayed_move_time = 0xFFFFFFFFUL ; // force moves to be done
2015-04-04 08:42:50 +02:00
set_destination_to_current ( ) ;
2016-06-11 04:23:41 +02:00
prepare_move_to_destination ( ) ;
2013-08-07 16:10:26 +02:00
}
2013-10-20 12:12:35 +02:00
# endif
2015-04-04 00:31:35 +02:00
2015-07-31 07:24:43 +02:00
# if ENABLED(TEMP_STAT_LEDS)
2015-04-04 00:31:35 +02:00
handle_status_leds ( ) ;
2013-10-20 12:12:35 +02:00
# endif
2015-04-04 00:31:35 +02:00
2017-04-15 05:44:08 +02:00
# if ENABLED(HAVE_TMC2130)
2017-09-06 13:28:33 +02:00
tmc2130_checkOverTemp ( ) ;
2017-03-07 06:00:43 +01:00
# endif
2016-04-28 03:06:32 +02:00
planner . check_axes_activity ( ) ;
2012-11-06 12:06:41 +01:00
}
2016-09-13 09:11:45 +02:00
/**
* Standard idle routine keeps the machine alive
*/
void idle (
2017-05-26 20:01:02 +02:00
# if ENABLED(ADVANCED_PAUSE_FEATURE)
2016-09-13 09:11:45 +02:00
bool no_stepper_sleep /*=false*/
# endif
) {
2017-08-26 00:03:07 +02:00
# if ENABLED(MAX7219_DEBUG)
Max7219_idle_tasks ( ) ;
# endif // MAX7219_DEBUG
2016-09-13 09:11:45 +02:00
lcd_update ( ) ;
2016-11-09 01:03:40 +01:00
2016-09-13 09:11:45 +02:00
host_keepalive ( ) ;
2016-11-09 01:03:40 +01:00
# if ENABLED(AUTO_REPORT_TEMPERATURES) && (HAS_TEMP_HOTEND || HAS_TEMP_BED)
auto_report_temperatures ( ) ;
# endif
2016-09-13 09:11:45 +02:00
manage_inactivity (
2017-05-26 20:01:02 +02:00
# if ENABLED(ADVANCED_PAUSE_FEATURE)
2016-09-13 09:11:45 +02:00
no_stepper_sleep
# endif
) ;
thermalManager . manage_heater ( ) ;
# if ENABLED(PRINTCOUNTER)
print_job_timer . tick ( ) ;
# endif
2016-10-05 04:31:50 +02:00
# if HAS_BUZZER && DISABLED(LCD_USE_I2C_BUZZER)
2016-09-13 09:11:45 +02:00
buzzer . tick ( ) ;
# endif
2017-06-09 14:06:23 +02:00
# if ENABLED(I2C_POSITION_ENCODERS)
if ( planner . blocks_queued ( ) & &
( ( blockBufferIndexRef ! = planner . block_buffer_head ) | |
( ( lastUpdateMillis + I2CPE_MIN_UPD_TIME_MS ) < millis ( ) ) ) ) {
blockBufferIndexRef = planner . block_buffer_head ;
I2CPEM . update ( ) ;
lastUpdateMillis = millis ( ) ;
}
# endif
2016-09-13 09:11:45 +02:00
}
/**
* Kill all activity and lock the machine .
* After this the machine will need to be reset .
*/
2015-10-03 08:08:58 +02:00
void kill ( const char * lcd_msg ) {
2017-06-09 17:51:23 +02:00
SERIAL_ERROR_START ( ) ;
2016-07-10 04:50:45 +02:00
SERIAL_ERRORLNPGM ( MSG_ERR_KILLED ) ;
2017-03-21 15:06:01 +01:00
thermalManager . disable_all_heaters ( ) ;
disable_all_steppers ( ) ;
2017-03-29 02:45:54 +02:00
2015-07-31 07:24:43 +02:00
# if ENABLED(ULTRA_LCD)
2016-07-10 04:50:45 +02:00
kill_screen ( lcd_msg ) ;
2015-10-04 17:33:55 +02:00
# else
UNUSED ( lcd_msg ) ;
2015-05-20 22:03:16 +02:00
# endif
2015-05-20 20:53:48 +02:00
2017-04-01 07:23:14 +02:00
_delay_ms ( 600 ) ; // Wait a short time (allows messages to get out before shutting down.
2012-11-06 12:06:41 +01:00
cli ( ) ; // Stop interrupts
2017-03-29 02:45:54 +02:00
2017-03-21 15:06:01 +01:00
_delay_ms ( 250 ) ; //Wait to ensure all interrupts routines stopped
thermalManager . disable_all_heaters ( ) ; //turn off heaters again
2013-06-07 00:49:25 +02:00
2017-07-28 06:42:01 +02:00
# ifdef ACTION_ON_KILL
2017-07-02 10:44:24 +02:00
SERIAL_ECHOLNPGM ( " //action: " ACTION_ON_KILL ) ;
# endif
2017-07-07 04:24:30 +02:00
2015-03-31 01:50:05 +02:00
# if HAS_POWER_SWITCH
2016-12-03 05:40:18 +01:00
SET_INPUT ( PS_ON_PIN ) ;
2015-03-31 01:50:05 +02:00
# endif
2012-11-06 12:06:41 +01:00
suicide ( ) ;
2016-03-04 00:35:18 +01:00
while ( 1 ) {
2016-03-07 22:27:01 +01:00
# if ENABLED(USE_WATCHDOG)
watchdog_reset ( ) ;
# endif
2016-03-04 00:35:18 +01:00
} // Wait for reset
2012-11-06 12:06:41 +01:00
}
2016-09-13 09:11:45 +02:00
/**
* Turn off heaters and stop the print in progress
* After a stop the machine may be resumed with M999
*/
2016-04-18 09:05:22 +02:00
void stop ( ) {
2017-05-07 13:06:06 +02:00
thermalManager . disable_all_heaters ( ) ; // 'unpause' taken care of in here
# if ENABLED(PROBING_FANS_OFF)
if ( fans_paused ) fans_pause ( false ) ; // put things back the way they were
# endif
2015-05-17 10:47:02 +02:00
if ( IsRunning ( ) ) {
Stopped_gcode_LastN = gcode_LastN ; // Save last g_code for restart
2017-06-09 17:51:23 +02:00
SERIAL_ERROR_START ( ) ;
2015-05-17 10:47:02 +02:00
SERIAL_ERRORLNPGM ( MSG_ERR_STOPPED ) ;
LCD_MESSAGEPGM ( MSG_STOPPED ) ;
2017-04-01 07:23:14 +02:00
safe_delay ( 350 ) ; // allow enough time for messages to get out before stopping
Running = false ;
2012-11-06 12:06:41 +01:00
}
}
2016-09-12 03:21:54 +02:00
/**
* Marlin entry - point : Set up before the program loop
* - Set up the kill pin , filament runout , power hold
* - Start the serial port
* - Print startup messages and diagnostics
* - Get EEPROM or default settings
* - Initialize managers for :
* • temperature
* • planner
* • watchdog
* • stepper
* • photo pin
* • servos
* • LCD controller
* • Digipot I2C
* • Z probe sled
* • status LEDs
*/
void setup ( ) {
2017-08-26 00:03:07 +02:00
# if ENABLED(MAX7219_DEBUG)
Max7219_init ( ) ;
# endif
2016-09-23 00:10:20 +02:00
# ifdef DISABLE_JTAG
2016-09-12 03:21:54 +02:00
// Disable JTAG on AT90USB chips to free up pins for IO
MCUCR = 0x80 ;
MCUCR = 0x80 ;
# endif
# if ENABLED(FILAMENT_RUNOUT_SENSOR)
setup_filrunoutpin ( ) ;
# endif
setup_killpin ( ) ;
setup_powerhold ( ) ;
# if HAS_STEPPER_RESET
disableStepperDrivers ( ) ;
# endif
MYSERIAL . begin ( BAUDRATE ) ;
2017-06-18 01:36:10 +02:00
while ( ! MYSERIAL ) ;
2016-09-12 03:21:54 +02:00
SERIAL_PROTOCOLLNPGM ( " start " ) ;
2017-06-09 17:51:23 +02:00
SERIAL_ECHO_START ( ) ;
2016-09-12 03:21:54 +02:00
// Check startup - does nothing if bootloader sets MCUSR to 0
2017-06-18 01:36:10 +02:00
byte mcu = HAL_get_reset_source ( ) ;
2017-06-16 21:20:58 +02:00
if ( mcu & 1 ) SERIAL_ECHOLNPGM ( MSG_POWERUP ) ;
if ( mcu & 2 ) SERIAL_ECHOLNPGM ( MSG_EXTERNAL_RESET ) ;
if ( mcu & 4 ) SERIAL_ECHOLNPGM ( MSG_BROWNOUT_RESET ) ;
if ( mcu & 8 ) SERIAL_ECHOLNPGM ( MSG_WATCHDOG_RESET ) ;
2016-09-12 03:21:54 +02:00
if ( mcu & 32 ) SERIAL_ECHOLNPGM ( MSG_SOFTWARE_RESET ) ;
2017-06-18 01:36:10 +02:00
HAL_clear_reset_source ( ) ;
# if ENABLED(USE_WATCHDOG) //reinit watchdog after HAL_get_reset_source call
watchdog_init ( ) ;
# endif
2016-09-12 03:21:54 +02:00
SERIAL_ECHOPGM ( MSG_MARLIN ) ;
2016-09-13 00:49:35 +02:00
SERIAL_CHAR ( ' ' ) ;
SERIAL_ECHOLNPGM ( SHORT_BUILD_VERSION ) ;
2017-06-09 17:51:23 +02:00
SERIAL_EOL ( ) ;
2016-09-12 03:21:54 +02:00
2016-09-13 00:49:35 +02:00
# if defined(STRING_DISTRIBUTION_DATE) && defined(STRING_CONFIG_H_AUTHOR)
2017-06-09 17:51:23 +02:00
SERIAL_ECHO_START ( ) ;
2016-09-13 00:49:35 +02:00
SERIAL_ECHOPGM ( MSG_CONFIGURATION_VER ) ;
SERIAL_ECHOPGM ( STRING_DISTRIBUTION_DATE ) ;
SERIAL_ECHOLNPGM ( MSG_AUTHOR STRING_CONFIG_H_AUTHOR ) ;
2017-08-21 23:30:08 +02:00
SERIAL_ECHO_START ( ) ;
2016-09-13 00:49:35 +02:00
SERIAL_ECHOLNPGM ( " Compiled: " __DATE__ ) ;
# endif
2016-09-12 03:21:54 +02:00
2017-06-09 17:51:23 +02:00
SERIAL_ECHO_START ( ) ;
2016-09-13 00:49:35 +02:00
SERIAL_ECHOPAIR ( MSG_FREE_MEMORY , freeMemory ( ) ) ;
SERIAL_ECHOLNPAIR ( MSG_PLANNER_BUFFER_BYTES , ( int ) sizeof ( block_t ) * BLOCK_BUFFER_SIZE ) ;
2016-09-12 03:21:54 +02:00
// Send "ok" after commands by default
for ( int8_t i = 0 ; i < BUFSIZE ; i + + ) send_ok [ i ] = true ;
// Load data from EEPROM if available (or use defaults)
// This also updates variables in the planner, elsewhere
2017-04-10 04:47:49 +02:00
( void ) settings . load ( ) ;
2016-09-12 03:21:54 +02:00
2017-04-15 04:41:21 +02:00
# if HAS_M206_COMMAND
2017-03-05 01:01:33 +01:00
// Initialize current position based on home_offset
2017-03-05 02:19:06 +01:00
COPY ( current_position , home_offset ) ;
2017-03-05 01:01:33 +01:00
# else
ZERO ( current_position ) ;
# endif
2016-09-12 03:21:54 +02:00
// Vital to init stepper/planner equivalent for current_position
SYNC_PLAN_POSITION_KINEMATIC ( ) ;
thermalManager . init ( ) ; // Initialize temperature loop
stepper . init ( ) ; // Initialize stepper, this enables interrupts!
servo_init ( ) ;
2016-11-01 12:24:21 +01:00
# if HAS_PHOTOGRAPH
OUT_WRITE ( PHOTOGRAPH_PIN , LOW ) ;
# endif
2016-10-05 21:34:07 +02:00
# if HAS_CASE_LIGHT
2017-05-19 22:21:43 +02:00
case_light_on = CASE_LIGHT_DEFAULT_ON ;
case_light_brightness = CASE_LIGHT_DEFAULT_BRIGHTNESS ;
2016-11-28 06:51:51 +01:00
update_case_light ( ) ;
2016-10-05 21:34:07 +02:00
# endif
2017-04-07 20:52:45 +02:00
# if ENABLED(SPINDLE_LASER_ENABLE)
OUT_WRITE ( SPINDLE_LASER_ENABLE_PIN , ! SPINDLE_LASER_ENABLE_INVERT ) ; // init spindle to off
# if SPINDLE_DIR_CHANGE
OUT_WRITE ( SPINDLE_DIR_PIN , SPINDLE_INVERT_DIR ? 255 : 0 ) ; // init rotation to clockwise (M3)
# endif
2017-09-06 13:28:33 +02:00
# if ENABLED(SPINDLE_LASER_PWM) && defined(SPINDLE_LASER_PWM_PIN) && SPINDLE_LASER_PWM_PIN >= 0
2017-04-07 20:52:45 +02:00
SET_OUTPUT ( SPINDLE_LASER_PWM_PIN ) ;
analogWrite ( SPINDLE_LASER_PWM_PIN , SPINDLE_LASER_PWM_INVERT ? 255 : 0 ) ; // set to lowest speed
# endif
# endif
2016-09-12 03:21:54 +02:00
# if HAS_BED_PROBE
endstops . enable_z_probe ( false ) ;
# endif
2017-05-04 23:38:29 +02:00
# if ENABLED(USE_CONTROLLER_FAN)
2017-05-03 03:18:31 +02:00
SET_OUTPUT ( CONTROLLER_FAN_PIN ) ; //Set pin used for driver cooling fan
2016-09-12 03:21:54 +02:00
# endif
# if HAS_STEPPER_RESET
enableStepperDrivers ( ) ;
# endif
# if ENABLED(DIGIPOT_I2C)
digipot_i2c_init ( ) ;
# endif
# if ENABLED(DAC_STEPPER_CURRENT)
dac_init ( ) ;
# endif
2017-04-14 23:36:02 +02:00
# if (ENABLED(Z_PROBE_SLED) || ENABLED(SOLENOID_PROBE)) && HAS_SOLENOID_1
OUT_WRITE ( SOL1_PIN , LOW ) ; // turn it off
# endif
2016-09-12 03:21:54 +02:00
2017-07-02 09:26:49 +02:00
# if HAS_HOME
SET_INPUT_PULLUP ( HOME_PIN ) ;
# endif
2016-09-12 03:21:54 +02:00
2016-09-24 22:33:29 +02:00
# if PIN_EXISTS(STAT_LED_RED)
OUT_WRITE ( STAT_LED_RED_PIN , LOW ) ; // turn it off
2016-09-12 03:21:54 +02:00
# endif
2016-09-24 22:33:29 +02:00
# if PIN_EXISTS(STAT_LED_BLUE)
OUT_WRITE ( STAT_LED_BLUE_PIN , LOW ) ; // turn it off
2016-09-12 03:21:54 +02:00
# endif
2017-07-13 17:01:21 +02:00
# if ENABLED(NEOPIXEL_RGBW_LED)
SET_OUTPUT ( NEOPIXEL_PIN ) ;
setup_neopixel ( ) ;
# endif
2017-04-14 23:54:53 +02:00
# if ENABLED(RGB_LED) || ENABLED(RGBW_LED)
2017-03-18 16:17:00 +01:00
SET_OUTPUT ( RGB_LED_R_PIN ) ;
SET_OUTPUT ( RGB_LED_G_PIN ) ;
SET_OUTPUT ( RGB_LED_B_PIN ) ;
2017-04-14 23:54:53 +02:00
# if ENABLED(RGBW_LED)
SET_OUTPUT ( RGB_LED_W_PIN ) ;
# endif
2016-11-30 02:51:13 +01:00
# endif
2017-06-02 20:57:31 +02:00
# if ENABLED(MK2_MULTIPLEXER)
SET_OUTPUT ( E_MUX0_PIN ) ;
SET_OUTPUT ( E_MUX1_PIN ) ;
SET_OUTPUT ( E_MUX2_PIN ) ;
# endif
2017-08-26 00:03:07 +02:00
2017-08-15 14:48:10 +02:00
# if HAS_FANMUX
fanmux_init ( ) ;
# endif
2017-08-26 00:03:07 +02:00
2016-09-12 03:21:54 +02:00
lcd_init ( ) ;
2017-07-02 07:35:41 +02:00
2017-07-03 05:35:28 +02:00
# ifndef CUSTOM_BOOTSCREEN_TIMEOUT
# define CUSTOM_BOOTSCREEN_TIMEOUT 2500
# endif
2016-09-12 03:21:54 +02:00
# if ENABLED(SHOW_BOOTSCREEN)
2017-07-02 07:35:41 +02:00
# if ENABLED(DOGLCD) // On DOGM the first bootscreen is already drawn
# if ENABLED(SHOW_CUSTOM_BOOTSCREEN)
safe_delay ( CUSTOM_BOOTSCREEN_TIMEOUT ) ; // Custom boot screen pause
lcd_bootscreen ( ) ; // Show Marlin boot screen
# endif
safe_delay ( BOOTSCREEN_TIMEOUT ) ; // Pause
2016-09-12 03:21:54 +02:00
# elif ENABLED(ULTRA_LCD)
2017-07-02 07:35:41 +02:00
lcd_bootscreen ( ) ;
2016-10-13 09:03:20 +02:00
# if DISABLED(SDSUPPORT)
lcd_init ( ) ;
# endif
2016-09-12 03:21:54 +02:00
# endif
# endif
# if ENABLED(MIXING_EXTRUDER) && MIXING_VIRTUAL_TOOLS > 1
// Initialize mixing to 100% color 1
for ( uint8_t i = 0 ; i < MIXING_STEPPERS ; i + + )
2016-11-12 21:33:07 +01:00
mixing_factor [ i ] = ( i = = 0 ) ? 1.0 : 0.0 ;
2016-09-12 03:21:54 +02:00
for ( uint8_t t = 0 ; t < MIXING_VIRTUAL_TOOLS ; t + + )
for ( uint8_t i = 0 ; i < MIXING_STEPPERS ; i + + )
mixing_virtual_tool_mix [ t ] [ i ] = mixing_factor [ i ] ;
# endif
2017-01-22 01:10:02 +01:00
# if ENABLED(BLTOUCH)
2017-05-01 18:09:39 +02:00
// Make sure any BLTouch error condition is cleared
bltouch_command ( BLTOUCH_RESET ) ;
set_bltouch_deployed ( true ) ;
set_bltouch_deployed ( false ) ;
2017-01-22 01:10:02 +01:00
# endif
2017-06-09 14:06:23 +02:00
# if ENABLED(I2C_POSITION_ENCODERS)
I2CPEM . init ( ) ;
# endif
2016-09-12 03:21:54 +02:00
# if ENABLED(EXPERIMENTAL_I2CBUS) && I2C_SLAVE_ADDRESS > 0
i2c . onReceive ( i2c_on_receive ) ;
i2c . onRequest ( i2c_on_request ) ;
# endif
2016-11-05 22:38:48 +01:00
# if ENABLED(ENDSTOP_INTERRUPTS_FEATURE)
2016-11-19 04:53:45 +01:00
setup_endstop_interrupts ( ) ;
2016-11-05 22:38:48 +01:00
# endif
2017-06-06 00:41:38 +02:00
2017-08-18 11:11:00 +02:00
# if ENABLED(SWITCHING_EXTRUDER) && !DONT_SWITCH
2017-06-04 15:40:57 +02:00
move_extruder_servo ( 0 ) ; // Initialize extruder servo
# endif
# if ENABLED(SWITCHING_NOZZLE)
move_nozzle_servo ( 0 ) ; // Initialize nozzle servo
# endif
2017-08-21 23:30:08 +02:00
2017-08-13 23:35:59 +02:00
# if ENABLED(PARKING_EXTRUDER)
# if ENABLED(PARKING_EXTRUDER_SOLENOIDS_INVERT)
pe_activate_magnet ( 0 ) ;
pe_activate_magnet ( 1 ) ;
2017-08-21 23:30:08 +02:00
# else
2017-08-13 23:35:59 +02:00
pe_deactivate_magnet ( 0 ) ;
pe_deactivate_magnet ( 1 ) ;
# endif
# endif
2016-09-12 03:21:54 +02:00
}
/**
* The main Marlin program loop
*
* - Save or log commands to SD
* - Process available commands ( if not saving )
* - Call heater manager
* - Call inactivity manager
* - Call endstop manager
* - Call LCD update
*/
void loop ( ) {
if ( commands_in_queue < BUFSIZE ) get_available_commands ( ) ;
# if ENABLED(SDSUPPORT)
card . checkautostart ( false ) ;
# endif
if ( commands_in_queue ) {
# if ENABLED(SDSUPPORT)
if ( card . saving ) {
char * command = command_queue [ cmd_queue_index_r ] ;
if ( strstr_P ( command , PSTR ( " M29 " ) ) ) {
// M29 closes the file
card . closefile ( ) ;
SERIAL_PROTOCOLLNPGM ( MSG_FILE_SAVED ) ;
ok_to_send ( ) ;
}
else {
// Write the string from the read buffer to SD
card . write_command ( command ) ;
if ( card . logging )
process_next_command ( ) ; // The card is saving because it's logging
else
ok_to_send ( ) ;
}
}
else
process_next_command ( ) ;
# else
process_next_command ( ) ;
# endif // SDSUPPORT
// The queue may be reset by a command handler or by code invoked by idle() within a handler
if ( commands_in_queue ) {
- - commands_in_queue ;
2017-05-05 09:57:22 +02:00
if ( + + cmd_queue_index_r > = BUFSIZE ) cmd_queue_index_r = 0 ;
2016-09-12 03:21:54 +02:00
}
}
endstops . report_state ( ) ;
idle ( ) ;
}