aboutsummaryrefslogtreecommitdiff
path: root/Marlin/src/inc
diff options
context:
space:
mode:
Diffstat (limited to 'Marlin/src/inc')
-rw-r--r--Marlin/src/inc/Conditionals_LCD.h1209
-rw-r--r--Marlin/src/inc/Conditionals_adv.h549
-rw-r--r--Marlin/src/inc/Conditionals_post.h2751
-rw-r--r--Marlin/src/inc/MarlinConfig.h59
-rw-r--r--Marlin/src/inc/MarlinConfigPre.h62
-rw-r--r--Marlin/src/inc/SanityCheck.h3367
-rw-r--r--Marlin/src/inc/Version.h122
7 files changed, 8119 insertions, 0 deletions
diff --git a/Marlin/src/inc/Conditionals_LCD.h b/Marlin/src/inc/Conditionals_LCD.h
new file mode 100644
index 0000000..712ed39
--- /dev/null
+++ b/Marlin/src/inc/Conditionals_LCD.h
@@ -0,0 +1,1209 @@
+/**
+ * Marlin 3D Printer Firmware
+ * Copyright (c) 2020 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
+ *
+ * Based on Sprinter and grbl.
+ * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ *
+ */
+#pragma once
+
+/**
+ * Conditionals_LCD.h
+ * Conditionals that need to be set before Configuration_adv.h or pins.h
+ */
+
+// MKS_LCD12864 is a variant of MKS_MINI_12864
+#if ENABLED(MKS_LCD12864)
+ #define MKS_MINI_12864
+#endif
+
+/**
+ * General Flags that may be set below by specific LCDs
+ *
+ * DOGLCD : Run a Graphical LCD through U8GLib (with MarlinUI)
+ * IS_ULTIPANEL : Define LCD_PINS_D5/6/7 for direct-connected "Ultipanel" LCDs
+ * IS_ULTRA_LCD : Ultra LCD, not necessarily Ultipanel.
+ * IS_RRD_SC : Common RRD Smart Controller digital interface pins
+ * IS_RRD_FG_SC : Common RRD Full Graphical Smart Controller digital interface pins
+ * U8GLIB_ST7920 : Most common DOGM display SPI interface, supporting a "lightweight" display mode.
+ * U8GLIB_SH1106 : SH1106 OLED with I2C interface via U8GLib
+ * IS_U8GLIB_SSD1306 : SSD1306 OLED with I2C interface via U8GLib
+ * U8GLIB_SSD1309 : SSD1309 OLED with I2C interface via U8GLib
+ * U8GLIB_ST7565_64128N : ST7565 128x64 LCD with SPI interface via U8GLib
+ * U8GLIB_LM6059_AF : LM6059 with Hardware SPI via U8GLib
+ */
+#if EITHER(MKS_MINI_12864, ENDER2_STOCKDISPLAY)
+
+ #define MINIPANEL
+
+#elif ENABLED(CARTESIO_UI)
+
+ #define DOGLCD
+ #define IS_ULTIPANEL 1
+
+#elif EITHER(DWIN_MARLINUI_PORTRAIT, DWIN_MARLINUI_LANDSCAPE)
+
+ #define IS_DWIN_MARLINUI 1
+ #define IS_ULTIPANEL 1
+
+#elif ENABLED(ZONESTAR_LCD)
+
+ #define HAS_ADC_BUTTONS 1
+ #define REPRAPWORLD_KEYPAD_MOVE_STEP 10.0
+ #define ADC_KEY_NUM 8
+ #define IS_ULTIPANEL 1
+
+ // This helps to implement HAS_ADC_BUTTONS menus
+ #define REVERSE_MENU_DIRECTION
+ #define ENCODER_PULSES_PER_STEP 1
+ #define ENCODER_STEPS_PER_MENU_ITEM 1
+ #define ENCODER_FEEDRATE_DEADZONE 2
+
+#elif ENABLED(ZONESTAR_12864LCD)
+ #define DOGLCD
+ #define IS_RRD_SC 1
+ #define U8GLIB_ST7920
+
+#elif ENABLED(ZONESTAR_12864OLED)
+ #define IS_RRD_SC 1
+ #define U8GLIB_SH1106
+
+#elif ENABLED(ZONESTAR_12864OLED_SSD1306)
+ #define IS_RRD_SC 1
+ #define IS_U8GLIB_SSD1306
+
+#elif ENABLED(RADDS_DISPLAY)
+ #define IS_ULTIPANEL 1
+ #define ENCODER_PULSES_PER_STEP 2
+
+#elif ANY(miniVIKI, VIKI2, ELB_FULL_GRAPHIC_CONTROLLER, AZSMZ_12864)
+
+ #define DOGLCD
+ #define IS_ULTIPANEL 1
+
+ #if ENABLED(miniVIKI)
+ #define U8GLIB_ST7565_64128N
+ #elif ENABLED(VIKI2)
+ #define U8GLIB_ST7565_64128N
+ #elif ENABLED(ELB_FULL_GRAPHIC_CONTROLLER)
+ #define U8GLIB_LM6059_AF
+ #elif ENABLED(AZSMZ_12864)
+ #define U8GLIB_ST7565_64128N
+ #endif
+
+#elif ENABLED(OLED_PANEL_TINYBOY2)
+
+ #define IS_U8GLIB_SSD1306
+ #define IS_ULTIPANEL 1
+
+#elif ENABLED(RA_CONTROL_PANEL)
+
+ #define LCD_I2C_TYPE_PCA8574
+ #define LCD_I2C_ADDRESS 0x27 // I2C Address of the port expander
+ #define IS_ULTIPANEL 1
+
+#elif ENABLED(REPRAPWORLD_GRAPHICAL_LCD)
+
+ #define DOGLCD
+ #define U8GLIB_ST7920
+ #define IS_ULTIPANEL 1
+
+#elif ENABLED(MKS_12864OLED)
+
+ #define IS_RRD_SC 1
+ #define U8GLIB_SH1106
+
+#elif ENABLED(MKS_12864OLED_SSD1306)
+
+ #define IS_RRD_SC 1
+ #define IS_U8GLIB_SSD1306
+
+#elif ENABLED(FYSETC_242_OLED_12864)
+
+ #define IS_RRD_SC 1
+ #define U8GLIB_SH1106
+
+ #define LED_CONTROL_MENU
+ #define NEOPIXEL_LED
+ #undef NEOPIXEL_TYPE
+ #define NEOPIXEL_TYPE NEO_RGB
+ #if NEOPIXEL_PIXELS < 3
+ #undef NEOPIXELS_PIXELS
+ #define NEOPIXEL_PIXELS 3
+ #endif
+ #ifndef NEOPIXEL_BRIGHTNESS
+ #define NEOPIXEL_BRIGHTNESS 127
+ #endif
+
+ #if ENABLED(PSU_CONTROL)
+ #define LED_BACKLIGHT_TIMEOUT 10000
+ #endif
+
+#elif ANY(FYSETC_MINI_12864_X_X, FYSETC_MINI_12864_1_2, FYSETC_MINI_12864_2_0, FYSETC_MINI_12864_2_1, FYSETC_GENERIC_12864_1_1)
+
+ #define FYSETC_MINI_12864
+ #define DOGLCD
+ #define IS_ULTIPANEL 1
+ #define LED_COLORS_REDUCE_GREEN
+ #if ENABLED(PSU_CONTROL) && EITHER(FYSETC_MINI_12864_2_0, FYSETC_MINI_12864_2_1)
+ #define LED_BACKLIGHT_TIMEOUT 10000
+ #endif
+
+ // Require LED backlighting enabled
+ #if EITHER(FYSETC_MINI_12864_1_2, FYSETC_MINI_12864_2_0)
+ #define RGB_LED
+ #elif ENABLED(FYSETC_MINI_12864_2_1)
+ #define LED_CONTROL_MENU
+ #define NEOPIXEL_LED
+ #undef NEOPIXEL_TYPE
+ #define NEOPIXEL_TYPE NEO_RGB
+ #if NEOPIXEL_PIXELS < 3
+ #undef NEOPIXELS_PIXELS
+ #define NEOPIXEL_PIXELS 3
+ #endif
+ #ifndef NEOPIXEL_BRIGHTNESS
+ #define NEOPIXEL_BRIGHTNESS 127
+ #endif
+ //#define NEOPIXEL_STARTUP_TEST
+ #endif
+
+#elif ENABLED(ULTI_CONTROLLER)
+
+ #define IS_ULTIPANEL 1
+ #define U8GLIB_SSD1309
+ #define LCD_RESET_PIN LCD_PINS_D6 // This controller need a reset pin
+ #define ENCODER_PULSES_PER_STEP 2
+ #define ENCODER_STEPS_PER_MENU_ITEM 2
+
+#elif ENABLED(MAKEBOARD_MINI_2_LINE_DISPLAY_1602)
+
+ #define IS_RRD_SC 1
+ #define LCD_WIDTH 16
+ #define LCD_HEIGHT 2
+
+#elif EITHER(TFTGLCD_PANEL_SPI, TFTGLCD_PANEL_I2C)
+
+ #define IS_TFTGLCD_PANEL 1
+ #define IS_ULTIPANEL 1 // Note that IS_ULTIPANEL leads to HAS_WIRED_LCD
+
+ #if ENABLED(SDSUPPORT) && DISABLED(LCD_PROGRESS_BAR)
+ #define LCD_PROGRESS_BAR
+ #endif
+ #if ENABLED(TFTGLCD_PANEL_I2C)
+ #define LCD_I2C_ADDRESS 0x27 // Must be equal to panel's I2C slave addres
+ #endif
+ #define LCD_USE_I2C_BUZZER // Enable buzzer on LCD, used for both I2C and SPI buses (LiquidTWI2 not required)
+ #define STD_ENCODER_PULSES_PER_STEP 2
+ #define STD_ENCODER_STEPS_PER_MENU_ITEM 1
+ #define LCD_WIDTH 20 // 20 or 24 chars in line
+ #define LCD_HEIGHT 10 // Character lines
+ #define LCD_CONTRAST_MIN 127
+ #define LCD_CONTRAST_MAX 255
+ #define DEFAULT_LCD_CONTRAST 250
+ #define CONVERT_TO_EXT_ASCII // Use extended 128-255 symbols from ASCII table.
+ // At this time present conversion only for cyrillic - bg, ru and uk languages.
+ // First 7 ASCII symbols in panel font must be replaced with Marlin's special symbols.
+
+#elif ENABLED(CR10_STOCKDISPLAY)
+
+ #define IS_RRD_FG_SC 1
+ #define BOARD_ST7920_DELAY_1 DELAY_NS(125)
+ #define BOARD_ST7920_DELAY_2 DELAY_NS(125)
+ #define BOARD_ST7920_DELAY_3 DELAY_NS(125)
+
+#elif ANY(REPRAP_DISCOUNT_FULL_GRAPHIC_SMART_CONTROLLER, ANET_FULL_GRAPHICS_LCD, ANET_FULL_GRAPHICS_LCD_ALT_WIRING, BQ_LCD_SMART_CONTROLLER)
+
+ #define IS_RRD_FG_SC 1
+
+#elif ENABLED(REPRAP_DISCOUNT_SMART_CONTROLLER)
+
+ // RepRapDiscount LCD or Graphical LCD with rotary click encoder
+ #define IS_RRD_SC 1
+
+#endif
+
+// ST7920-based graphical displays
+#if ANY(IS_RRD_FG_SC, LCD_FOR_MELZI, SILVER_GATE_GLCD_CONTROLLER)
+ #define DOGLCD
+ #define U8GLIB_ST7920
+ #define IS_RRD_SC 1
+#endif
+
+// ST7565 / 64128N graphical displays
+#if EITHER(MAKRPANEL, MINIPANEL)
+ #define IS_ULTIPANEL 1
+ #define DOGLCD
+ #if ENABLED(MAKRPANEL)
+ #define U8GLIB_ST7565_64128N
+ #endif
+#endif
+
+#if ENABLED(IS_U8GLIB_SSD1306)
+ #define U8GLIB_SSD1306
+#endif
+
+#if ENABLED(OVERLORD_OLED)
+ #define IS_ULTIPANEL 1
+ #define U8GLIB_SH1106
+ /**
+ * PCA9632 for buzzer and LEDs via i2c
+ * No auto-inc, red and green leds switched, buzzer
+ */
+ #define PCA9632
+ #define PCA9632_NO_AUTO_INC
+ #define PCA9632_GRN 0x00
+ #define PCA9632_RED 0x02
+ #define PCA9632_BUZZER
+ #define PCA9632_BUZZER_DATA { 0x09, 0x02 }
+
+ #define ENCODER_PULSES_PER_STEP 1 // Overlord uses buttons
+ #define ENCODER_STEPS_PER_MENU_ITEM 1
+#endif
+
+// 128x64 I2C OLED LCDs - SSD1306/SSD1309/SH1106
+#if ANY(U8GLIB_SSD1306, U8GLIB_SSD1309, U8GLIB_SH1106)
+ #define HAS_U8GLIB_I2C_OLED 1
+ #define IS_ULTRA_LCD 1
+ #define DOGLCD
+#endif
+
+/**
+ * SPI Ultipanels
+ */
+
+// Basic Ultipanel-like displays
+#if ANY(ULTIMAKERCONTROLLER, IS_RRD_SC, G3D_PANEL, RIGIDBOT_PANEL, PANEL_ONE, U8GLIB_SH1106)
+ #define IS_ULTIPANEL 1
+#endif
+
+// Einstart OLED has Cardinal nav via pins defined in pins_EINSTART-S.h
+#if ENABLED(U8GLIB_SH1106_EINSTART)
+ #define DOGLCD
+ #define IS_ULTIPANEL 1
+#endif
+
+// TFT Compatibility
+#if ANY(FSMC_GRAPHICAL_TFT, SPI_GRAPHICAL_TFT, TFT_320x240, TFT_480x320, TFT_320x240_SPI, TFT_480x320_SPI, TFT_LVGL_UI_FSMC, TFT_LVGL_UI_SPI)
+ #define IS_LEGACY_TFT 1
+ #define TFT_GENERIC
+ #warning "Don't forget to update your TFT settings in Configuration.h."
+#endif
+
+#if ANY(FSMC_GRAPHICAL_TFT, TFT_320x240, TFT_480x320, TFT_LVGL_UI_FSMC)
+ #define TFT_INTERFACE_FSMC
+#elif ANY(SPI_GRAPHICAL_TFT, TFT_320x240_SPI, TFT_480x320_SPI, TFT_LVGL_UI_SPI)
+ #define TFT_INTERFACE_SPI
+#endif
+
+#if EITHER(FSMC_GRAPHICAL_TFT, SPI_GRAPHICAL_TFT)
+ #define TFT_CLASSIC_UI
+#elif ANY(TFT_320x240, TFT_480x320, TFT_320x240_SPI, TFT_480x320_SPI)
+ #define TFT_COLOR_UI
+#elif EITHER(TFT_LVGL_UI_FSMC, TFT_LVGL_UI_SPI)
+ #define TFT_LVGL_UI
+#endif
+
+// FSMC/SPI TFT Panels (LVGL)
+#if ENABLED(TFT_LVGL_UI)
+ #define HAS_TFT_LVGL_UI 1
+ #define SERIAL_RUNTIME_HOOK 1
+#endif
+
+// FSMC/SPI TFT Panels
+#if ENABLED(TFT_CLASSIC_UI)
+ #define TFT_SCALED_DOGLCD 1
+#endif
+
+#if TFT_SCALED_DOGLCD
+ #define DOGLCD
+ #define IS_ULTIPANEL 1
+ #define DELAYED_BACKLIGHT_INIT
+#elif HAS_TFT_LVGL_UI
+ #define DELAYED_BACKLIGHT_INIT
+#endif
+
+// Color UI
+#if ENABLED(TFT_COLOR_UI)
+ #define HAS_GRAPHICAL_TFT 1
+ #define IS_ULTIPANEL 1
+#endif
+
+/**
+ * I2C Panels
+ */
+
+#if EITHER(LCD_SAINSMART_I2C_1602, LCD_SAINSMART_I2C_2004)
+
+ #define LCD_I2C_TYPE_PCF8575
+ #define LCD_I2C_ADDRESS 0x27 // I2C Address of the port expander
+
+ #if ENABLED(LCD_SAINSMART_I2C_2004)
+ #define LCD_WIDTH 20
+ #define LCD_HEIGHT 4
+ #endif
+
+#elif ENABLED(LCD_I2C_PANELOLU2)
+
+ // PANELOLU2 LCD with status LEDs, separate encoder and click inputs
+
+ #define LCD_I2C_TYPE_MCP23017
+ #define LCD_I2C_ADDRESS 0x20 // I2C Address of the port expander
+ #define LCD_USE_I2C_BUZZER // Enable buzzer on LCD (optional)
+ #define IS_ULTIPANEL 1
+
+#elif ENABLED(LCD_I2C_VIKI)
+
+ /**
+ * Panucatt VIKI LCD with status LEDs, integrated click & L/R/U/P buttons, separate encoder inputs
+ *
+ * This uses the LiquidTWI2 library v1.2.3 or later ( https://github.com/lincomatic/LiquidTWI2 )
+ * Make sure the LiquidTWI2 directory is placed in the Arduino or Sketchbook libraries subdirectory.
+ * Note: The pause/stop/resume LCD button pin should be connected to the Arduino
+ * BTN_ENC pin (or set BTN_ENC to -1 if not used)
+ */
+ #define LCD_I2C_TYPE_MCP23017
+ #define LCD_I2C_ADDRESS 0x20 // I2C Address of the port expander
+ #define LCD_USE_I2C_BUZZER // Enable buzzer on LCD (requires LiquidTWI2 v1.2.3 or later)
+ #define IS_ULTIPANEL 1
+
+ #define ENCODER_FEEDRATE_DEADZONE 4
+
+ #define STD_ENCODER_PULSES_PER_STEP 1
+ #define STD_ENCODER_STEPS_PER_MENU_ITEM 2
+
+#elif ENABLED(G3D_PANEL)
+
+ #define STD_ENCODER_PULSES_PER_STEP 2
+ #define STD_ENCODER_STEPS_PER_MENU_ITEM 1
+
+#elif ANY(IS_RRD_SC, miniVIKI, VIKI2, ELB_FULL_GRAPHIC_CONTROLLER, AZSMZ_12864, OLED_PANEL_TINYBOY2, BQ_LCD_SMART_CONTROLLER, LCD_I2C_PANELOLU2)
+
+ #define STD_ENCODER_PULSES_PER_STEP 4
+ #define STD_ENCODER_STEPS_PER_MENU_ITEM 1
+
+#endif
+
+#ifndef STD_ENCODER_PULSES_PER_STEP
+ #if ENABLED(TOUCH_SCREEN)
+ #define STD_ENCODER_PULSES_PER_STEP 2
+ #else
+ #define STD_ENCODER_PULSES_PER_STEP 5
+ #endif
+#endif
+#ifndef STD_ENCODER_STEPS_PER_MENU_ITEM
+ #define STD_ENCODER_STEPS_PER_MENU_ITEM 1
+#endif
+#ifndef ENCODER_PULSES_PER_STEP
+ #define ENCODER_PULSES_PER_STEP STD_ENCODER_PULSES_PER_STEP
+#endif
+#ifndef ENCODER_STEPS_PER_MENU_ITEM
+ #define ENCODER_STEPS_PER_MENU_ITEM STD_ENCODER_STEPS_PER_MENU_ITEM
+#endif
+#ifndef ENCODER_FEEDRATE_DEADZONE
+ #define ENCODER_FEEDRATE_DEADZONE 6
+#endif
+
+// Shift register panels
+// ---------------------
+// 2 wire Non-latching LCD SR from:
+// https://bitbucket.org/fmalpartida/new-liquidcrystal/wiki/schematics#!shiftregister-connection
+#if ENABLED(FF_INTERFACEBOARD)
+ #define SR_LCD_3W_NL // Non latching 3 wire shift register
+ #define IS_ULTIPANEL 1
+#elif ENABLED(SAV_3DLCD)
+ #define SR_LCD_2W_NL // Non latching 2 wire shift register
+ #define IS_ULTIPANEL 1
+#elif ENABLED(ULTIPANEL)
+ #define IS_ULTIPANEL 1
+#endif
+
+#if EITHER(IS_ULTIPANEL, ULTRA_LCD)
+ #define IS_ULTRA_LCD 1
+#endif
+
+#if EITHER(IS_ULTIPANEL, REPRAPWORLD_KEYPAD)
+ #define IS_NEWPANEL 1
+#endif
+
+#if EITHER(ZONESTAR_LCD, REPRAPWORLD_KEYPAD)
+ #define IS_RRW_KEYPAD 1
+ #ifndef REPRAPWORLD_KEYPAD_MOVE_STEP
+ #define REPRAPWORLD_KEYPAD_MOVE_STEP 1.0
+ #endif
+#endif
+
+// Aliases for LCD features
+#if ANY(DGUS_LCD_UI_ORIGIN, DGUS_LCD_UI_FYSETC, DGUS_LCD_UI_HIPRECY)
+ #define HAS_DGUS_LCD 1
+#endif
+
+// Extensible UI serial touch screens. (See src/lcd/extui)
+#if ANY(HAS_DGUS_LCD, MALYAN_LCD, TOUCH_UI_FTDI_EVE, ANYCUBIC_LCD_I3MEGA, ANYCUBIC_LCD_CHIRON)
+ #define IS_EXTUI 1
+ #define EXTENSIBLE_UI
+#endif
+
+// Aliases for LCD features
+#if EITHER(IS_ULTRA_LCD, EXTENSIBLE_UI)
+ #define HAS_DISPLAY 1
+#endif
+
+#if IS_ULTRA_LCD
+ #define HAS_WIRED_LCD 1
+ #if ENABLED(DOGLCD)
+ #define HAS_MARLINUI_U8GLIB 1
+ #elif IS_TFTGLCD_PANEL
+ // Neither DOGM nor HD44780. Fully customized interface.
+ #elif DISABLED(HAS_GRAPHICAL_TFT)
+ #define HAS_MARLINUI_HD44780 1
+ #endif
+#endif
+
+#if EITHER(HAS_DISPLAY, GLOBAL_STATUS_MESSAGE)
+ #define HAS_STATUS_MESSAGE 1
+#endif
+
+#if IS_ULTIPANEL && DISABLED(NO_LCD_MENUS)
+ #define HAS_LCD_MENU 1
+#endif
+
+#if HAS_MARLINUI_U8GLIB
+ #ifndef LCD_PIXEL_WIDTH
+ #define LCD_PIXEL_WIDTH 128
+ #endif
+ #ifndef LCD_PIXEL_HEIGHT
+ #define LCD_PIXEL_HEIGHT 64
+ #endif
+#endif
+
+/**
+ * Multi-Material-Unit supported models
+ */
+#define PRUSA_MMU1 1
+#define PRUSA_MMU2 2
+#define PRUSA_MMU2S 3
+#define SMUFF_EMU_MMU2 12
+#define SMUFF_EMU_MMU2S 13
+
+#ifdef MMU_MODEL
+ #define HAS_MMU 1
+ #if MMU_MODEL == PRUSA_MMU1
+ #define HAS_PRUSA_MMU1 1
+ #elif MMU_MODEL % 10 == PRUSA_MMU2
+ #define HAS_PRUSA_MMU2 1
+ #elif MMU_MODEL % 10 == PRUSA_MMU2S
+ #define HAS_PRUSA_MMU2 1
+ #define HAS_PRUSA_MMU2S 1
+ #endif
+ #if MMU_MODEL >= SMUFF_EMU_MMU2
+ #define HAS_SMUFF 1
+ #endif
+#endif
+
+#undef PRUSA_MMU1
+#undef PRUSA_MMU2
+#undef PRUSA_MMU2S
+#undef SMUFF_EMU_MMU2
+#undef SMUFF_EMU_MMU2S
+
+/**
+ * Extruders have some combination of stepper motors and hotends
+ * so we separate these concepts into the defines:
+ *
+ * EXTRUDERS - Number of Selectable Tools
+ * HOTENDS - Number of hotends, whether connected or separate
+ * E_STEPPERS - Number of actual E stepper motors
+ * E_MANUAL - Number of E steppers for LCD move options
+ */
+
+#if EXTRUDERS == 0
+ #undef EXTRUDERS
+ #define EXTRUDERS 0
+ #undef SINGLENOZZLE
+ #undef SWITCHING_EXTRUDER
+ #undef SWITCHING_NOZZLE
+ #undef MIXING_EXTRUDER
+ #undef HOTEND_IDLE_TIMEOUT
+#elif EXTRUDERS > 1
+ #define HAS_MULTI_EXTRUDER 1
+#endif
+
+#if ENABLED(SWITCHING_EXTRUDER) // One stepper for every two EXTRUDERS
+ #if EXTRUDERS > 4
+ #define E_STEPPERS 3
+ #elif EXTRUDERS > 2
+ #define E_STEPPERS 2
+ #else
+ #define E_STEPPERS 1
+ #endif
+ #if DISABLED(SWITCHING_NOZZLE)
+ #define HOTENDS E_STEPPERS
+ #endif
+#elif ENABLED(MIXING_EXTRUDER)
+ #define E_STEPPERS MIXING_STEPPERS
+ #define E_MANUAL 1
+ #if MIXING_STEPPERS == 2
+ #define HAS_DUAL_MIXING 1
+ #endif
+#elif ENABLED(SWITCHING_TOOLHEAD)
+ #define E_STEPPERS EXTRUDERS
+ #define E_MANUAL EXTRUDERS
+#elif HAS_PRUSA_MMU2
+ #define E_STEPPERS 1
+#endif
+
+// No inactive extruders with SWITCHING_NOZZLE or Průša MMU1
+#if ENABLED(SWITCHING_NOZZLE) || HAS_PRUSA_MMU1
+ #undef DISABLE_INACTIVE_EXTRUDER
+#endif
+
+// Průša MMU1, MMU 2.0, MMUS 2.0 and SMUFF force SINGLENOZZLE
+#if HAS_MMU
+ #define SINGLENOZZLE
+#endif
+
+#if EITHER(SINGLENOZZLE, MIXING_EXTRUDER) // One hotend, one thermistor, no XY offset
+ #undef HOTENDS
+ #define HOTENDS 1
+ #undef HOTEND_OFFSET_X
+ #undef HOTEND_OFFSET_Y
+#endif
+
+#ifndef HOTENDS
+ #define HOTENDS EXTRUDERS
+#endif
+#ifndef E_STEPPERS
+ #define E_STEPPERS EXTRUDERS
+#endif
+#ifndef E_MANUAL
+ #define E_MANUAL EXTRUDERS
+#endif
+
+#if HOTENDS
+ #define HAS_HOTEND 1
+ #ifndef HOTEND_OVERSHOOT
+ #define HOTEND_OVERSHOOT 15
+ #endif
+ #if HOTENDS > 1
+ #define HAS_MULTI_HOTEND 1
+ #define HAS_HOTEND_OFFSET 1
+ #endif
+#else
+ #undef PID_PARAMS_PER_HOTEND
+#endif
+
+// Helper macros for extruder and hotend arrays
+#define HOTEND_LOOP() for (int8_t e = 0; e < HOTENDS; e++)
+#define ARRAY_BY_EXTRUDERS(V...) ARRAY_N(EXTRUDERS, V)
+#define ARRAY_BY_EXTRUDERS1(v1) ARRAY_BY_EXTRUDERS(v1, v1, v1, v1, v1, v1, v1, v1)
+#define ARRAY_BY_HOTENDS(V...) ARRAY_N(HOTENDS, V)
+#define ARRAY_BY_HOTENDS1(v1) ARRAY_BY_HOTENDS(v1, v1, v1, v1, v1, v1, v1, v1)
+
+#if ENABLED(SWITCHING_EXTRUDER) && (DISABLED(SWITCHING_NOZZLE) || SWITCHING_EXTRUDER_SERVO_NR != SWITCHING_NOZZLE_SERVO_NR)
+ #define DO_SWITCH_EXTRUDER 1
+#endif
+
+/**
+ * Default hotend offsets, if not defined
+ */
+#if HAS_HOTEND_OFFSET
+ #ifndef HOTEND_OFFSET_X
+ #define HOTEND_OFFSET_X { 0 } // X offsets for each extruder
+ #endif
+ #ifndef HOTEND_OFFSET_Y
+ #define HOTEND_OFFSET_Y { 0 } // Y offsets for each extruder
+ #endif
+ #ifndef HOTEND_OFFSET_Z
+ #define HOTEND_OFFSET_Z { 0 } // Z offsets for each extruder
+ #endif
+#endif
+
+/**
+ * DISTINCT_E_FACTORS affects how some E factors are accessed
+ */
+#if ENABLED(DISTINCT_E_FACTORS) && E_STEPPERS > 1
+ #define DISTINCT_E E_STEPPERS
+ #define XYZE_N (XYZ + E_STEPPERS)
+ #define E_INDEX_N(E) (E)
+ #define E_AXIS_N(E) AxisEnum(E_AXIS + E)
+ #define UNUSED_E(E) NOOP
+#else
+ #undef DISTINCT_E_FACTORS
+ #define DISTINCT_E 1
+ #define XYZE_N XYZE
+ #define E_INDEX_N(E) 0
+ #define E_AXIS_N(E) E_AXIS
+ #define UNUSED_E(E) UNUSED(E)
+#endif
+
+#if ENABLED(DWIN_CREALITY_LCD)
+ #define SERIAL_CATCHALL 0
+ #ifndef LCD_SERIAL_PORT
+ #define LCD_SERIAL_PORT 3 // Creality 4.x board
+ #endif
+#endif
+
+/**
+ * The BLTouch Probe emulates a servo probe
+ * and uses "special" angles for its state.
+ */
+#if ENABLED(BLTOUCH)
+ #ifndef Z_PROBE_SERVO_NR
+ #define Z_PROBE_SERVO_NR 0
+ #endif
+ #undef DEACTIVATE_SERVOS_AFTER_MOVE
+
+ // Always disable probe pin inverting for BLTouch
+ #undef Z_MIN_PROBE_ENDSTOP_INVERTING
+ #define Z_MIN_PROBE_ENDSTOP_INVERTING false
+ #if ENABLED(Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN)
+ #undef Z_MIN_ENDSTOP_INVERTING
+ #define Z_MIN_ENDSTOP_INVERTING false
+ #endif
+#endif
+
+/**
+ * Set a flag for a servo probe (or BLTouch)
+ */
+#ifdef Z_PROBE_SERVO_NR
+ #define HAS_Z_SERVO_PROBE 1
+#endif
+#if ANY(HAS_Z_SERVO_PROBE, SWITCHING_EXTRUDER, SWITCHING_NOZZLE)
+ #define HAS_SERVO_ANGLES 1
+#endif
+#if !HAS_SERVO_ANGLES
+ #undef EDITABLE_SERVO_ANGLES
+#endif
+
+/**
+ * Set flags for enabled probes
+ */
+#if ANY(HAS_Z_SERVO_PROBE, FIX_MOUNTED_PROBE, NOZZLE_AS_PROBE, TOUCH_MI_PROBE, Z_PROBE_ALLEN_KEY, Z_PROBE_SLED, SOLENOID_PROBE, SENSORLESS_PROBING, RACK_AND_PINION_PROBE)
+ #define HAS_BED_PROBE 1
+#endif
+
+#if ENABLED(FILAMENT_RUNOUT_SENSOR)
+ #if NUM_RUNOUT_SENSORS >= 1
+ #ifndef FIL_RUNOUT1_STATE
+ #define FIL_RUNOUT1_STATE FIL_RUNOUT_STATE
+ #endif
+ #ifndef FIL_RUNOUT1_PULLUP
+ #define FIL_RUNOUT1_PULLUP FIL_RUNOUT_PULLUP
+ #endif
+ #ifndef FIL_RUNOUT1_PULLDOWN
+ #define FIL_RUNOUT1_PULLDOWN FIL_RUNOUT_PULLDOWN
+ #endif
+ #endif
+ #if NUM_RUNOUT_SENSORS >= 2
+ #ifndef FIL_RUNOUT2_STATE
+ #define FIL_RUNOUT2_STATE FIL_RUNOUT_STATE
+ #endif
+ #ifndef FIL_RUNOUT2_PULLUP
+ #define FIL_RUNOUT2_PULLUP FIL_RUNOUT_PULLUP
+ #endif
+ #ifndef FIL_RUNOUT2_PULLDOWN
+ #define FIL_RUNOUT2_PULLDOWN FIL_RUNOUT_PULLDOWN
+ #endif
+ #endif
+ #if NUM_RUNOUT_SENSORS >= 3
+ #ifndef FIL_RUNOUT3_STATE
+ #define FIL_RUNOUT3_STATE FIL_RUNOUT_STATE
+ #endif
+ #ifndef FIL_RUNOUT3_PULLUP
+ #define FIL_RUNOUT3_PULLUP FIL_RUNOUT_PULLUP
+ #endif
+ #ifndef FIL_RUNOUT3_PULLDOWN
+ #define FIL_RUNOUT3_PULLDOWN FIL_RUNOUT_PULLDOWN
+ #endif
+ #endif
+ #if NUM_RUNOUT_SENSORS >= 4
+ #ifndef FIL_RUNOUT4_STATE
+ #define FIL_RUNOUT4_STATE FIL_RUNOUT_STATE
+ #endif
+ #ifndef FIL_RUNOUT4_PULLUP
+ #define FIL_RUNOUT4_PULLUP FIL_RUNOUT_PULLUP
+ #endif
+ #ifndef FIL_RUNOUT4_PULLDOWN
+ #define FIL_RUNOUT4_PULLDOWN FIL_RUNOUT_PULLDOWN
+ #endif
+ #endif
+ #if NUM_RUNOUT_SENSORS >= 5
+ #ifndef FIL_RUNOUT5_STATE
+ #define FIL_RUNOUT5_STATE FIL_RUNOUT_STATE
+ #endif
+ #ifndef FIL_RUNOUT5_PULLUP
+ #define FIL_RUNOUT5_PULLUP FIL_RUNOUT_PULLUP
+ #endif
+ #ifndef FIL_RUNOUT5_PULLDOWN
+ #define FIL_RUNOUT5_PULLDOWN FIL_RUNOUT_PULLDOWN
+ #endif
+ #endif
+ #if NUM_RUNOUT_SENSORS >= 6
+ #ifndef FIL_RUNOUT6_STATE
+ #define FIL_RUNOUT6_STATE FIL_RUNOUT_STATE
+ #endif
+ #ifndef FIL_RUNOUT6_PULLUP
+ #define FIL_RUNOUT6_PULLUP FIL_RUNOUT_PULLUP
+ #endif
+ #ifndef FIL_RUNOUT6_PULLDOWN
+ #define FIL_RUNOUT6_PULLDOWN FIL_RUNOUT_PULLDOWN
+ #endif
+ #endif
+ #if NUM_RUNOUT_SENSORS >= 7
+ #ifndef FIL_RUNOUT7_STATE
+ #define FIL_RUNOUT7_STATE FIL_RUNOUT_STATE
+ #endif
+ #ifndef FIL_RUNOUT7_PULLUP
+ #define FIL_RUNOUT7_PULLUP FIL_RUNOUT_PULLUP
+ #endif
+ #ifndef FIL_RUNOUT7_PULLDOWN
+ #define FIL_RUNOUT7_PULLDOWN FIL_RUNOUT_PULLDOWN
+ #endif
+ #endif
+ #if NUM_RUNOUT_SENSORS >= 8
+ #ifndef FIL_RUNOUT8_STATE
+ #define FIL_RUNOUT8_STATE FIL_RUNOUT_STATE
+ #endif
+ #ifndef FIL_RUNOUT8_PULLUP
+ #define FIL_RUNOUT8_PULLUP FIL_RUNOUT_PULLUP
+ #endif
+ #ifndef FIL_RUNOUT8_PULLDOWN
+ #define FIL_RUNOUT8_PULLDOWN FIL_RUNOUT_PULLDOWN
+ #endif
+ #endif
+#endif // FILAMENT_RUNOUT_SENSOR
+
+#if EITHER(MESH_BED_LEVELING, AUTO_BED_LEVELING_UBL)
+ #undef PROBE_MANUALLY
+#endif
+
+#if ANY(HAS_BED_PROBE, PROBE_MANUALLY, MESH_BED_LEVELING)
+ #define PROBE_SELECTED 1
+#endif
+
+#if HAS_BED_PROBE
+ #if DISABLED(NOZZLE_AS_PROBE)
+ #define HAS_PROBE_XY_OFFSET 1
+ #endif
+ #if DISABLED(Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN)
+ #define HAS_CUSTOM_PROBE_PIN 1
+ #endif
+ #if Z_HOME_DIR < 0 && (!HAS_CUSTOM_PROBE_PIN || ENABLED(USE_PROBE_FOR_Z_HOMING))
+ #define HOMING_Z_WITH_PROBE 1
+ #endif
+ #ifndef Z_PROBE_LOW_POINT
+ #define Z_PROBE_LOW_POINT -5
+ #endif
+ #if ENABLED(Z_PROBE_ALLEN_KEY)
+ #define PROBE_TRIGGERED_WHEN_STOWED_TEST 1 // Extra test for Allen Key Probe
+ #endif
+ #if MULTIPLE_PROBING > 1
+ #if EXTRA_PROBING > 0
+ #define TOTAL_PROBING (MULTIPLE_PROBING + EXTRA_PROBING)
+ #else
+ #define TOTAL_PROBING MULTIPLE_PROBING
+ #endif
+ #endif
+ #if ENABLED(PREHEAT_BEFORE_PROBING)
+ #ifndef PROBING_NOZZLE_TEMP
+ #define PROBING_NOZZLE_TEMP 0
+ #endif
+ #ifndef PROBING_BED_TEMP
+ #define PROBING_BED_TEMP 0
+ #endif
+ #endif
+ #if ENABLED(PREHEAT_BEFORE_LEVELING)
+ #ifndef LEVELING_NOZZLE_TEMP
+ #define LEVELING_NOZZLE_TEMP 0
+ #endif
+ #ifndef LEVELING_BED_TEMP
+ #define LEVELING_BED_TEMP 0
+ #endif
+ #endif
+#else
+ // Clear probe pin settings when no probe is selected
+ #undef Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN
+ #undef USE_PROBE_FOR_Z_HOMING
+#endif
+
+#if Z_HOME_DIR > 0
+ #define HOME_Z_FIRST // If homing away from BED do Z first
+#endif
+
+/**
+ * Set granular options based on the specific type of leveling
+ */
+#if ENABLED(AUTO_BED_LEVELING_UBL)
+ #undef LCD_BED_LEVELING
+ #if ENABLED(DELTA)
+ #define UBL_SEGMENTED 1
+ #endif
+#endif
+#if EITHER(AUTO_BED_LEVELING_LINEAR, AUTO_BED_LEVELING_3POINT)
+ #define ABL_PLANAR 1
+#endif
+#if EITHER(AUTO_BED_LEVELING_LINEAR, AUTO_BED_LEVELING_BILINEAR)
+ #define ABL_GRID 1
+#endif
+#if ANY(AUTO_BED_LEVELING_LINEAR, AUTO_BED_LEVELING_BILINEAR, AUTO_BED_LEVELING_3POINT)
+ #define HAS_ABL_NOT_UBL 1
+#endif
+#if ANY(AUTO_BED_LEVELING_BILINEAR, AUTO_BED_LEVELING_UBL, MESH_BED_LEVELING)
+ #define HAS_MESH 1
+#endif
+#if EITHER(AUTO_BED_LEVELING_UBL, AUTO_BED_LEVELING_3POINT)
+ #define NEEDS_THREE_PROBE_POINTS 1
+#endif
+#if EITHER(HAS_ABL_NOT_UBL, AUTO_BED_LEVELING_UBL)
+ #define HAS_ABL_OR_UBL 1
+ #if DISABLED(PROBE_MANUALLY)
+ #define HAS_AUTOLEVEL 1
+ #endif
+#endif
+#if EITHER(HAS_ABL_OR_UBL, MESH_BED_LEVELING)
+ #define HAS_LEVELING 1
+ #if DISABLED(AUTO_BED_LEVELING_UBL)
+ #define PLANNER_LEVELING 1
+ #endif
+#endif
+#if EITHER(HAS_ABL_OR_UBL, Z_MIN_PROBE_REPEATABILITY_TEST)
+ #define HAS_PROBING_PROCEDURE 1
+#endif
+#if !HAS_LEVELING
+ #undef PROBE_MANUALLY
+ #undef RESTORE_LEVELING_AFTER_G28
+ #undef ENABLE_LEVELING_AFTER_G28
+#endif
+
+#ifdef GRID_MAX_POINTS_X
+ #define GRID_MAX_POINTS ((GRID_MAX_POINTS_X) * (GRID_MAX_POINTS_Y))
+ #define GRID_LOOP(A,B) LOOP_L_N(A, GRID_MAX_POINTS_X) LOOP_L_N(B, GRID_MAX_POINTS_Y)
+#endif
+
+// Slim menu optimizations
+#if ENABLED(SLIM_LCD_MENUS)
+ #define BOOT_MARLIN_LOGO_SMALL
+#endif
+
+/**
+ * CoreXY, CoreXZ, and CoreYZ - and their reverse
+ */
+#if EITHER(COREXY, COREYX)
+ #define CORE_IS_XY 1
+#endif
+#if EITHER(COREXZ, COREZX)
+ #define CORE_IS_XZ 1
+#endif
+#if EITHER(COREYZ, COREZY)
+ #define CORE_IS_YZ 1
+#endif
+#if CORE_IS_XY || CORE_IS_XZ || CORE_IS_YZ
+ #define IS_CORE 1
+#endif
+#if IS_CORE
+ #if CORE_IS_XY
+ #define CORE_AXIS_1 A_AXIS
+ #define CORE_AXIS_2 B_AXIS
+ #define NORMAL_AXIS Z_AXIS
+ #elif CORE_IS_XZ
+ #define CORE_AXIS_1 A_AXIS
+ #define NORMAL_AXIS Y_AXIS
+ #define CORE_AXIS_2 C_AXIS
+ #elif CORE_IS_YZ
+ #define NORMAL_AXIS X_AXIS
+ #define CORE_AXIS_1 B_AXIS
+ #define CORE_AXIS_2 C_AXIS
+ #endif
+ #define CORESIGN(n) (ANY(COREYX, COREZX, COREZY) ? (-(n)) : (n))
+#elif ENABLED(MARKFORGED_XY)
+ // Markforged kinematics
+ #define CORE_AXIS_1 A_AXIS
+ #define CORE_AXIS_2 B_AXIS
+ #define NORMAL_AXIS Z_AXIS
+#endif
+
+#if ENABLED(MORGAN_SCARA)
+ #define IS_SCARA 1
+ #define IS_KINEMATIC 1
+#elif ENABLED(DELTA)
+ #define IS_KINEMATIC 1
+#else
+ #define IS_CARTESIAN 1
+ #if !IS_CORE
+ #define IS_FULL_CARTESIAN 1
+ #endif
+#endif
+
+// This flag indicates some kind of jerk storage is needed
+#if EITHER(CLASSIC_JERK, IS_KINEMATIC)
+ #define HAS_CLASSIC_JERK 1
+#endif
+
+#if DISABLED(CLASSIC_JERK)
+ #define HAS_JUNCTION_DEVIATION 1
+#endif
+
+// E jerk exists with JD disabled (of course) but also when Linear Advance is disabled on Delta/SCARA
+#if ENABLED(CLASSIC_JERK) || (IS_KINEMATIC && DISABLED(LIN_ADVANCE))
+ #define HAS_CLASSIC_E_JERK 1
+#endif
+
+#if SERIAL_PORT == -1 || SERIAL_PORT_2 == -1
+ #define HAS_USB_SERIAL 1
+#endif
+#if SERIAL_PORT_2 == -2
+ #define HAS_ETHERNET 1
+#endif
+
+// Fallback Stepper Driver types that don't depend on Configuration_adv.h
+#ifndef X_DRIVER_TYPE
+ #define X_DRIVER_TYPE A4988
+#endif
+#ifndef X2_DRIVER_TYPE
+ #define X2_DRIVER_TYPE A4988
+#endif
+#ifndef Y_DRIVER_TYPE
+ #define Y_DRIVER_TYPE A4988
+#endif
+#ifndef Y2_DRIVER_TYPE
+ #define Y2_DRIVER_TYPE A4988
+#endif
+#ifndef Z_DRIVER_TYPE
+ #define Z_DRIVER_TYPE A4988
+#endif
+#ifndef Z2_DRIVER_TYPE
+ #define Z2_DRIVER_TYPE A4988
+#endif
+#ifndef Z3_DRIVER_TYPE
+ #define Z3_DRIVER_TYPE A4988
+#endif
+#ifndef Z4_DRIVER_TYPE
+ #define Z4_DRIVER_TYPE A4988
+#endif
+#if E_STEPPERS <= 0
+ #undef E0_DRIVER_TYPE
+#elif !defined(E0_DRIVER_TYPE)
+ #define E0_DRIVER_TYPE A4988
+#endif
+#if E_STEPPERS <= 1
+ #undef E1_DRIVER_TYPE
+#elif !defined(E1_DRIVER_TYPE)
+ #define E1_DRIVER_TYPE A4988
+#endif
+#if E_STEPPERS <= 2
+ #undef E2_DRIVER_TYPE
+#elif !defined(E2_DRIVER_TYPE)
+ #define E2_DRIVER_TYPE A4988
+#endif
+#if E_STEPPERS <= 3
+ #undef E3_DRIVER_TYPE
+#elif !defined(E3_DRIVER_TYPE)
+ #define E3_DRIVER_TYPE A4988
+#endif
+#if E_STEPPERS <= 4
+ #undef E4_DRIVER_TYPE
+#elif !defined(E4_DRIVER_TYPE)
+ #define E4_DRIVER_TYPE A4988
+#endif
+#if E_STEPPERS <= 5
+ #undef E5_DRIVER_TYPE
+#elif !defined(E5_DRIVER_TYPE)
+ #define E5_DRIVER_TYPE A4988
+#endif
+#if E_STEPPERS <= 6
+ #undef E6_DRIVER_TYPE
+#elif !defined(E6_DRIVER_TYPE)
+ #define E6_DRIVER_TYPE A4988
+#endif
+#if E_STEPPERS <= 7
+ #undef E7_DRIVER_TYPE
+#elif !defined(E7_DRIVER_TYPE)
+ #define E7_DRIVER_TYPE A4988
+#endif
+
+// Fallback axis inverting
+#ifndef INVERT_X_DIR
+ #define INVERT_X_DIR false
+#endif
+#ifndef INVERT_Y_DIR
+ #define INVERT_Y_DIR false
+#endif
+#ifndef INVERT_Z_DIR
+ #define INVERT_Z_DIR false
+#endif
+#ifndef INVERT_E_DIR
+ #define INVERT_E_DIR false
+#endif
+
+/**
+ * This setting is also used by M109 when trying to calculate
+ * a ballpark safe margin to prevent wait-forever situation.
+ */
+#ifndef EXTRUDE_MINTEMP
+ #define EXTRUDE_MINTEMP 170
+#endif
+
+/**
+ * TFT Displays
+ *
+ * Configure parameters for TFT displays:
+ * - TFT_DEFAULT_ORIENTATION
+ * - TFT_DRIVER
+ * - TFT_WIDTH
+ * - TFT_HEIGHT
+ * - TFT_INTERFACE_(SPI|FSMC)
+ * - TFT_COLOR
+ * - GRAPHICAL_TFT_UPSCALE
+ */
+#if ENABLED(MKS_TS35_V2_0) // Most common: ST7796
+ #define TFT_DEFAULT_ORIENTATION (TFT_EXCHANGE_XY)
+ #define TFT_RES_480x320
+ #define TFT_INTERFACE_SPI
+#elif ENABLED(MKS_ROBIN_TFT24) // Most common: ST7789
+ #define TFT_DEFAULT_ORIENTATION (TFT_EXCHANGE_XY | TFT_INVERT_Y)
+ #define TFT_RES_320x240
+ #define TFT_INTERFACE_FSMC
+#elif ENABLED(MKS_ROBIN_TFT28) // Most common: ST7789
+ #define TFT_DEFAULT_ORIENTATION (TFT_EXCHANGE_XY | TFT_INVERT_Y)
+ #define TFT_RES_320x240
+ #define TFT_INTERFACE_FSMC
+#elif ENABLED(MKS_ROBIN_TFT32) // Most common: ST7789
+ #define TFT_DEFAULT_ORIENTATION (TFT_EXCHANGE_XY | TFT_INVERT_Y)
+ #define TFT_RES_320x240
+ #define TFT_INTERFACE_FSMC
+#elif ENABLED(MKS_ROBIN_TFT35) // Most common: ILI9488
+ #define TFT_DEFAULT_ORIENTATION (TFT_EXCHANGE_XY | TFT_INVERT_X | TFT_INVERT_Y)
+ #define TFT_RES_480x320
+ #define TFT_INTERFACE_FSMC
+#elif ENABLED(MKS_ROBIN_TFT43)
+ #define TFT_DEFAULT_ORIENTATION 0
+ #define TFT_DRIVER SSD1963
+ #define TFT_RES_480x272
+ #define TFT_INTERFACE_FSMC
+#elif ENABLED(MKS_ROBIN_TFT_V1_1R) // ILI9328 or R61505
+ #define TFT_DEFAULT_ORIENTATION (TFT_EXCHANGE_XY | TFT_INVERT_X | TFT_INVERT_Y)
+ #define TFT_RES_320x240
+ #define TFT_INTERFACE_FSMC
+#elif EITHER(TFT_TRONXY_X5SA, ANYCUBIC_TFT35) // ILI9488
+ #define TFT_DEFAULT_ORIENTATION (TFT_EXCHANGE_XY | TFT_INVERT_X | TFT_INVERT_Y)
+ #define TFT_DRIVER ILI9488
+ #define TFT_RES_480x320
+ #define TFT_INTERFACE_FSMC
+#elif ENABLED(LONGER_LK_TFT28)
+ #define TFT_DEFAULT_ORIENTATION (TFT_EXCHANGE_XY | TFT_INVERT_X | TFT_INVERT_Y)
+ #define TFT_RES_320x240
+ #define TFT_INTERFACE_FSMC
+#elif ENABLED(ANET_ET4_TFT28) // ST7789
+ #define TFT_DEFAULT_ORIENTATION (TFT_EXCHANGE_XY | TFT_INVERT_Y)
+ #define TFT_RES_320x240
+ #define TFT_INTERFACE_FSMC
+#elif ENABLED(ANET_ET5_TFT35) // ST7796
+ #define TFT_DEFAULT_ORIENTATION (TFT_EXCHANGE_XY)
+ #define TFT_RES_480x320
+ #define TFT_INTERFACE_FSMC
+#elif ENABLED(TFT_GENERIC)
+ #define TFT_DEFAULT_ORIENTATION (TFT_EXCHANGE_XY | TFT_INVERT_X | TFT_INVERT_Y)
+ #if NONE(TFT_RES_320x240, TFT_RES_480x272, TFT_RES_480x320)
+ #define TFT_RES_320x240
+ #endif
+ #if NONE(TFT_INTERFACE_FSMC, TFT_INTERFACE_SPI)
+ #define TFT_INTERFACE_SPI
+ #endif
+#endif
+
+#if ENABLED(TFT_RES_320x240)
+ #define TFT_WIDTH 320
+ #define TFT_HEIGHT 240
+ #define GRAPHICAL_TFT_UPSCALE 2
+#elif ENABLED(TFT_RES_480x272)
+ #define TFT_WIDTH 480
+ #define TFT_HEIGHT 272
+ #define GRAPHICAL_TFT_UPSCALE 2
+#elif ENABLED(TFT_RES_480x320)
+ #define TFT_WIDTH 480
+ #define TFT_HEIGHT 320
+ #define GRAPHICAL_TFT_UPSCALE 3
+#endif
+
+// FSMC/SPI TFT Panels using standard HAL/tft/tft_(fsmc|spi).h
+#if ENABLED(TFT_INTERFACE_FSMC)
+ #define HAS_FSMC_TFT 1
+ #if TFT_SCALED_DOGLCD
+ #define HAS_FSMC_GRAPHICAL_TFT 1
+ #elif HAS_TFT_LVGL_UI
+ #define HAS_TFT_LVGL_UI_FSMC 1
+ #endif
+#elif ENABLED(TFT_INTERFACE_SPI)
+ #define HAS_SPI_TFT 1
+ #if TFT_SCALED_DOGLCD
+ #define HAS_SPI_GRAPHICAL_TFT 1
+ #elif HAS_TFT_LVGL_UI
+ #define HAS_TFT_LVGL_UI_SPI 1
+ #endif
+#endif
+
+#if ENABLED(TFT_COLOR_UI)
+ #if TFT_HEIGHT == 240
+ #if ENABLED(TFT_INTERFACE_SPI)
+ #define TFT_320x240_SPI
+ #elif ENABLED(TFT_INTERFACE_FSMC)
+ #define TFT_320x240
+ #endif
+ #elif TFT_HEIGHT == 320
+ #if ENABLED(TFT_INTERFACE_SPI)
+ #define TFT_480x320_SPI
+ #elif ENABLED(TFT_INTERFACE_FSMC)
+ #define TFT_480x320
+ #endif
+ #elif TFT_HEIGHT == 272
+ #if ENABLED(TFT_INTERFACE_SPI)
+ #define TFT_480x272_SPI
+ #elif ENABLED(TFT_INTERFACE_FSMC)
+ #define TFT_480x272
+ #endif
+ #endif
+#endif
+
+#if EITHER(TFT_320x240, TFT_320x240_SPI)
+ #define HAS_UI_320x240 1
+#elif EITHER(TFT_480x320, TFT_480x320_SPI)
+ #define HAS_UI_480x320 1
+#elif EITHER(TFT_480x272, TFT_480x272_SPI)
+ #define HAS_UI_480x272 1
+#endif
+#if ANY(HAS_UI_320x240, HAS_UI_480x320, HAS_UI_480x272)
+ #define LCD_HEIGHT TERN(TOUCH_SCREEN, 6, 7) // Fewer lines with touch buttons onscreen
+#endif
+
+// This emulated DOGM has 'touch/xpt2046', not 'tft/xpt2046'
+#if ENABLED(TOUCH_SCREEN) && !HAS_GRAPHICAL_TFT
+ #undef TOUCH_SCREEN
+ #if ENABLED(TFT_CLASSIC_UI)
+ #define HAS_TOUCH_BUTTONS 1
+ #endif
+#endif
+
+// XPT2046_** Compatibility
+#if !(defined(TOUCH_CALIBRATION_X) || defined(TOUCH_CALIBRATION_Y) || defined(TOUCH_OFFSET_X) || defined(TOUCH_OFFSET_Y) || defined(TOUCH_ORIENTATION))
+ #if defined(XPT2046_X_CALIBRATION) && defined(XPT2046_Y_CALIBRATION) && defined(XPT2046_X_OFFSET) && defined(XPT2046_Y_OFFSET)
+ #define TOUCH_CALIBRATION_X XPT2046_X_CALIBRATION
+ #define TOUCH_CALIBRATION_Y XPT2046_Y_CALIBRATION
+ #define TOUCH_OFFSET_X XPT2046_X_OFFSET
+ #define TOUCH_OFFSET_Y XPT2046_Y_OFFSET
+ #define TOUCH_ORIENTATION TOUCH_LANDSCAPE
+ #endif
+#endif
diff --git a/Marlin/src/inc/Conditionals_adv.h b/Marlin/src/inc/Conditionals_adv.h
new file mode 100644
index 0000000..22a671c
--- /dev/null
+++ b/Marlin/src/inc/Conditionals_adv.h
@@ -0,0 +1,549 @@
+/**
+ * Marlin 3D Printer Firmware
+ * Copyright (c) 2020 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
+ *
+ * Based on Sprinter and grbl.
+ * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ *
+ */
+#pragma once
+
+/**
+ * Conditionals_adv.h
+ * Defines that depend on advanced configuration.
+ */
+
+#ifdef SWITCHING_NOZZLE_E1_SERVO_NR
+ #define SWITCHING_NOZZLE_TWO_SERVOS 1
+#endif
+
+// Determine NUM_SERVOS if none was supplied
+#ifndef NUM_SERVOS
+ #define NUM_SERVOS 0
+ #if ANY(HAS_Z_SERVO_PROBE, CHAMBER_VENT, SWITCHING_TOOLHEAD, SWITCHING_EXTRUDER, SWITCHING_NOZZLE, SPINDLE_SERVO)
+ #if NUM_SERVOS <= Z_PROBE_SERVO_NR
+ #undef NUM_SERVOS
+ #define NUM_SERVOS (Z_PROBE_SERVO_NR + 1)
+ #endif
+ #if NUM_SERVOS <= CHAMBER_VENT_SERVO_NR
+ #undef NUM_SERVOS
+ #define NUM_SERVOS (CHAMBER_VENT_SERVO_NR + 1)
+ #endif
+ #if NUM_SERVOS <= SWITCHING_TOOLHEAD_SERVO_NR
+ #undef NUM_SERVOS
+ #define NUM_SERVOS (SWITCHING_TOOLHEAD_SERVO_NR + 1)
+ #endif
+ #if NUM_SERVOS <= SWITCHING_NOZZLE_SERVO_NR
+ #undef NUM_SERVOS
+ #define NUM_SERVOS (SWITCHING_NOZZLE_SERVO_NR + 1)
+ #endif
+ #if NUM_SERVOS <= SWITCHING_NOZZLE_E1_SERVO_NR
+ #undef NUM_SERVOS
+ #define NUM_SERVOS (SWITCHING_NOZZLE_E1_SERVO_NR + 1)
+ #endif
+ #if NUM_SERVOS <= SWITCHING_EXTRUDER_SERVO_NR
+ #undef NUM_SERVOS
+ #define NUM_SERVOS (SWITCHING_EXTRUDER_SERVO_NR + 1)
+ #endif
+ #if NUM_SERVOS <= SWITCHING_EXTRUDER_E23_SERVO_NR
+ #undef NUM_SERVOS
+ #define NUM_SERVOS (SWITCHING_EXTRUDER_E23_SERVO_NR + 1)
+ #endif
+ #if NUM_SERVOS <= SPINDLE_SERVO_NR
+ #undef NUM_SERVOS
+ #define NUM_SERVOS (SPINDLE_SERVO_NR + 1)
+ #endif
+ #endif
+#endif
+
+// Convenience override for a BLTouch alone
+#if ENABLED(BLTOUCH) && NUM_SERVOS == 1
+ #undef SERVO_DELAY
+ #define SERVO_DELAY { 50 }
+#endif
+
+#if EXTRUDERS == 0
+ #define NO_VOLUMETRICS
+ #undef TEMP_SENSOR_0
+ #undef TEMP_SENSOR_1
+ #undef TEMP_SENSOR_2
+ #undef TEMP_SENSOR_3
+ #undef TEMP_SENSOR_4
+ #undef TEMP_SENSOR_5
+ #undef TEMP_SENSOR_6
+ #undef TEMP_SENSOR_7
+ #undef FWRETRACT
+ #undef PIDTEMP
+ #undef AUTOTEMP
+ #undef PID_EXTRUSION_SCALING
+ #undef LIN_ADVANCE
+ #undef FILAMENT_RUNOUT_SENSOR
+ #undef ADVANCED_PAUSE_FEATURE
+ #undef FILAMENT_RUNOUT_DISTANCE_MM
+ #undef FILAMENT_LOAD_UNLOAD_GCODES
+ #undef DISABLE_INACTIVE_EXTRUDER
+ #undef FILAMENT_LOAD_UNLOAD_GCODES
+ #undef EXTRUDER_RUNOUT_PREVENT
+ #undef PREVENT_COLD_EXTRUSION
+ #undef PREVENT_LENGTHY_EXTRUDE
+ #undef THERMAL_PROTECTION_HOTENDS
+ #undef THERMAL_PROTECTION_PERIOD
+ #undef WATCH_TEMP_PERIOD
+ #undef SHOW_TEMP_ADC_VALUES
+#endif
+
+#if TEMP_SENSOR_BED == 0
+ #undef THERMAL_PROTECTION_BED
+ #undef THERMAL_PROTECTION_BED_PERIOD
+#endif
+
+#if TEMP_SENSOR_CHAMBER == 0
+ #undef THERMAL_PROTECTION_CHAMBER
+#endif
+
+#if ENABLED(MIXING_EXTRUDER) && (ENABLED(RETRACT_SYNC_MIXING) || BOTH(FILAMENT_LOAD_UNLOAD_GCODES, FILAMENT_UNLOAD_ALL_EXTRUDERS))
+ #define HAS_MIXER_SYNC_CHANNEL 1
+#endif
+
+#if EITHER(DUAL_X_CARRIAGE, MULTI_NOZZLE_DUPLICATION)
+ #define HAS_DUPLICATION_MODE 1
+#endif
+
+#if ENABLED(PRINTCOUNTER) && (SERVICE_INTERVAL_1 > 0 || SERVICE_INTERVAL_2 > 0 || SERVICE_INTERVAL_3 > 0)
+ #define HAS_SERVICE_INTERVALS 1
+#endif
+
+#if ENABLED(FILAMENT_RUNOUT_SENSOR)
+ #define HAS_FILAMENT_SENSOR 1
+ #ifdef FILAMENT_RUNOUT_DISTANCE_MM
+ #define HAS_FILAMENT_RUNOUT_DISTANCE 1
+ #endif
+#endif
+
+// Let SD_FINISHED_RELEASECOMMAND stand in for SD_FINISHED_STEPPERRELEASE
+#if ENABLED(SD_FINISHED_STEPPERRELEASE)
+ #ifndef SD_FINISHED_RELEASECOMMAND
+ #define SD_FINISHED_RELEASECOMMAND "M84" // planner.finish_and_disable()
+ #endif
+#else
+ #undef SD_FINISHED_RELEASECOMMAND
+#endif
+
+#if ENABLED(NO_SD_AUTOSTART)
+ #undef MENU_ADDAUTOSTART
+#endif
+
+#if EITHER(SDSUPPORT, LCD_SET_PROGRESS_MANUALLY)
+ #define HAS_PRINT_PROGRESS 1
+#endif
+
+#if ENABLED(SDSUPPORT) && SD_PROCEDURE_DEPTH
+ #define HAS_MEDIA_SUBCALLS 1
+#endif
+
+#if HAS_PRINT_PROGRESS && EITHER(PRINT_PROGRESS_SHOW_DECIMALS, SHOW_REMAINING_TIME)
+ #define HAS_PRINT_PROGRESS_PERMYRIAD 1
+#endif
+
+#if ANY(MARLIN_BRICKOUT, MARLIN_INVADERS, MARLIN_SNAKE, MARLIN_MAZE)
+ #define HAS_GAMES 1
+ #if MANY(MARLIN_BRICKOUT, MARLIN_INVADERS, MARLIN_SNAKE, MARLIN_MAZE)
+ #define HAS_GAME_MENU 1
+ #endif
+#endif
+
+#if ANY(FWRETRACT, HAS_LEVELING, SKEW_CORRECTION)
+ #define HAS_POSITION_MODIFIERS 1
+#endif
+
+#if ANY(X_DUAL_ENDSTOPS, Y_DUAL_ENDSTOPS, Z_MULTI_ENDSTOPS)
+ #define HAS_EXTRA_ENDSTOPS 1
+#endif
+#if EITHER(MIN_SOFTWARE_ENDSTOPS, MAX_SOFTWARE_ENDSTOPS)
+ #define HAS_SOFTWARE_ENDSTOPS 1
+#endif
+#if ANY(EXTENSIBLE_UI, IS_NEWPANEL, EMERGENCY_PARSER, HAS_ADC_BUTTONS, DWIN_CREALITY_LCD)
+ #define HAS_RESUME_CONTINUE 1
+#endif
+
+#if ANY(BLINKM, RGB_LED, RGBW_LED, PCA9632, PCA9533, NEOPIXEL_LED)
+ #define HAS_COLOR_LEDS 1
+#endif
+#if ALL(HAS_RESUME_CONTINUE, PRINTER_EVENT_LEDS, SDSUPPORT)
+ #define HAS_LEDS_OFF_FLAG 1
+#endif
+
+#if EITHER(DIGIPOT_MCP4018, DIGIPOT_MCP4451)
+ #define HAS_MOTOR_CURRENT_I2C 1
+#endif
+
+#if ENABLED(Z_STEPPER_AUTO_ALIGN)
+ #if ENABLED(Z_STEPPER_ALIGN_KNOWN_STEPPER_POSITIONS)
+ #undef Z_STEPPER_ALIGN_AMP
+ #endif
+ #ifndef Z_STEPPER_ALIGN_AMP
+ #define Z_STEPPER_ALIGN_AMP 1.0
+ #endif
+#endif
+
+// Multiple Z steppers
+#ifndef NUM_Z_STEPPER_DRIVERS
+ #define NUM_Z_STEPPER_DRIVERS 1
+#endif
+
+// Fallback Stepper Driver types that depend on Configuration_adv.h
+#if NONE(DUAL_X_CARRIAGE, X_DUAL_STEPPER_DRIVERS)
+ #undef X2_DRIVER_TYPE
+#endif
+#if DISABLED(Y_DUAL_STEPPER_DRIVERS)
+ #undef Y2_DRIVER_TYPE
+#endif
+
+#if NUM_Z_STEPPER_DRIVERS < 4
+ #undef Z4_DRIVER_TYPE
+ #undef INVERT_Z4_VS_Z_DIR
+ #if NUM_Z_STEPPER_DRIVERS < 3
+ #undef Z3_DRIVER_TYPE
+ #undef INVERT_Z3_VS_Z_DIR
+ #if NUM_Z_STEPPER_DRIVERS < 2
+ #undef Z2_DRIVER_TYPE
+ #undef INVERT_Z2_VS_Z_DIR
+ #endif
+ #endif
+#endif
+
+//
+// Spindle/Laser power display types
+// Defined here so sanity checks can use them
+//
+#if EITHER(SPINDLE_FEATURE, LASER_FEATURE)
+ #define HAS_CUTTER 1
+ #define _CUTTER_POWER_PWM255 1
+ #define _CUTTER_POWER_PERCENT 2
+ #define _CUTTER_POWER_RPM 3
+ #define _CUTTER_POWER(V) _CAT(_CUTTER_POWER_, V)
+ #define CUTTER_UNIT_IS(V) (_CUTTER_POWER(CUTTER_POWER_UNIT) == _CUTTER_POWER(V))
+#endif
+
+// Add features that need hardware PWM here
+#if ANY(FAST_PWM_FAN, SPINDLE_LASER_PWM)
+ #define NEEDS_HARDWARE_PWM 1
+#endif
+
+#if !defined(__AVR__) || !defined(USBCON)
+ // Define constants and variables for buffering serial data.
+ // Use only 0 or powers of 2 greater than 1
+ // : [0, 4, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, ...]
+ #ifndef RX_BUFFER_SIZE
+ #define RX_BUFFER_SIZE 128
+ #endif
+ // 256 is the max TX buffer limit due to uint8_t head and tail
+ // : [0, 4, 8, 16, 32, 64, 128, 256]
+ #ifndef TX_BUFFER_SIZE
+ #define TX_BUFFER_SIZE 32
+ #endif
+#else
+ // SERIAL_XON_XOFF not supported on USB-native devices
+ #undef SERIAL_XON_XOFF
+#endif
+
+#if ENABLED(HOST_ACTION_COMMANDS)
+ #ifndef ACTION_ON_PAUSE
+ #define ACTION_ON_PAUSE "pause"
+ #endif
+ #ifndef ACTION_ON_PAUSED
+ #define ACTION_ON_PAUSED "paused"
+ #endif
+ #ifndef ACTION_ON_RESUME
+ #define ACTION_ON_RESUME "resume"
+ #endif
+ #ifndef ACTION_ON_RESUMED
+ #define ACTION_ON_RESUMED "resumed"
+ #endif
+ #ifndef ACTION_ON_CANCEL
+ #define ACTION_ON_CANCEL "cancel"
+ #endif
+ #ifndef ACTION_ON_START
+ #define ACTION_ON_START "start"
+ #endif
+ #ifndef ACTION_ON_KILL
+ #define ACTION_ON_KILL "poweroff"
+ #endif
+ #if HAS_FILAMENT_SENSOR
+ #ifndef ACTION_ON_FILAMENT_RUNOUT
+ #define ACTION_ON_FILAMENT_RUNOUT "filament_runout"
+ #endif
+ #ifndef ACTION_REASON_ON_FILAMENT_RUNOUT
+ #define ACTION_REASON_ON_FILAMENT_RUNOUT "filament_runout"
+ #endif
+ #endif
+ #if ENABLED(G29_RETRY_AND_RECOVER)
+ #ifndef ACTION_ON_G29_RECOVER
+ #define ACTION_ON_G29_RECOVER "probe_rewipe"
+ #endif
+ #ifndef ACTION_ON_G29_FAILURE
+ #define ACTION_ON_G29_FAILURE "probe_failed"
+ #endif
+ #endif
+#endif
+
+#if EITHER(FYSETC_MINI_12864_2_1, FYSETC_242_OLED_12864)
+ #define LED_CONTROL_MENU
+ #define LED_USER_PRESET_STARTUP
+ #define LED_COLOR_PRESETS
+ #ifndef LED_USER_PRESET_GREEN
+ #define LED_USER_PRESET_GREEN 128
+ #endif
+ #ifndef LED_USER_PRESET_BLUE
+ #define LED_USER_PRESET_BLUE 0
+ #endif
+ #ifndef LED_USER_PRESET_BRIGHTNESS
+ #define LED_USER_PRESET_BRIGHTNESS 255
+ #endif
+#endif
+
+// Set defaults for unspecified LED user colors
+#if ENABLED(LED_CONTROL_MENU)
+ #ifndef LED_USER_PRESET_RED
+ #define LED_USER_PRESET_RED 255
+ #endif
+ #ifndef LED_USER_PRESET_GREEN
+ #define LED_USER_PRESET_GREEN 255
+ #endif
+ #ifndef LED_USER_PRESET_BLUE
+ #define LED_USER_PRESET_BLUE 255
+ #endif
+ #ifndef LED_USER_PRESET_WHITE
+ #define LED_USER_PRESET_WHITE 0
+ #endif
+ #ifndef LED_USER_PRESET_BRIGHTNESS
+ #ifdef NEOPIXEL_BRIGHTNESS
+ #define LED_USER_PRESET_BRIGHTNESS NEOPIXEL_BRIGHTNESS
+ #else
+ #define LED_USER_PRESET_BRIGHTNESS 255
+ #endif
+ #endif
+#endif
+
+#if BOTH(LED_CONTROL_MENU, NEOPIXEL2_SEPARATE)
+ #ifndef LED2_USER_PRESET_RED
+ #define LED2_USER_PRESET_RED 255
+ #endif
+ #ifndef LED2_USER_PRESET_GREEN
+ #define LED2_USER_PRESET_GREEN 255
+ #endif
+ #ifndef LED2_USER_PRESET_BLUE
+ #define LED2_USER_PRESET_BLUE 255
+ #endif
+ #ifndef LED2_USER_PRESET_WHITE
+ #define LED2_USER_PRESET_WHITE 0
+ #endif
+ #ifndef LED2_USER_PRESET_BRIGHTNESS
+ #ifdef NEOPIXEL2_BRIGHTNESS
+ #define LED2_USER_PRESET_BRIGHTNESS NEOPIXEL2_BRIGHTNESS
+ #else
+ #define LED2_USER_PRESET_BRIGHTNESS 255
+ #endif
+ #endif
+#endif
+
+// If platform requires early initialization of watchdog to properly boot
+#if ENABLED(USE_WATCHDOG) && defined(ARDUINO_ARCH_SAM)
+ #define EARLY_WATCHDOG 1
+#endif
+
+// Full Touch Screen needs 'tft/xpt2046'
+#if EITHER(TOUCH_SCREEN, HAS_TFT_LVGL_UI)
+ #define HAS_TFT_XPT2046 1
+#endif
+
+// Touch Screen or "Touch Buttons" need XPT2046 pins
+// but they use different components
+#if EITHER(HAS_TFT_XPT2046, HAS_TOUCH_BUTTONS)
+ #define NEED_TOUCH_PINS 1
+#endif
+
+// Extensible UI pin mapping for RepRapDiscount
+#if ENABLED(TOUCH_UI_FTDI_EVE) && ANY(AO_EXP1_PINMAP, AO_EXP2_PINMAP, CR10_TFT_PINMAP)
+ #define TOUCH_UI_ULTIPANEL 1
+#endif
+
+// Poll-based jogging for joystick and other devices
+#if ENABLED(JOYSTICK)
+ #define POLL_JOG
+#endif
+
+#ifndef HOMING_BUMP_MM
+ #define HOMING_BUMP_MM { 0, 0, 0 }
+#endif
+
+#if ENABLED(USB_FLASH_DRIVE_SUPPORT) && NONE(USE_OTG_USB_HOST, USE_UHS3_USB)
+ #define USE_UHS2_USB
+#endif
+
+/**
+ * Driver Timings (in nanoseconds)
+ * NOTE: Driver timing order is longest-to-shortest duration.
+ * Preserve this ordering when adding new drivers.
+ */
+#ifndef MINIMUM_STEPPER_POST_DIR_DELAY
+ #if HAS_DRIVER(TB6560)
+ #define MINIMUM_STEPPER_POST_DIR_DELAY 15000
+ #elif HAS_DRIVER(TB6600)
+ #define MINIMUM_STEPPER_POST_DIR_DELAY 1500
+ #elif HAS_DRIVER(DRV8825)
+ #define MINIMUM_STEPPER_POST_DIR_DELAY 650
+ #elif HAS_DRIVER(LV8729)
+ #define MINIMUM_STEPPER_POST_DIR_DELAY 500
+ #elif HAS_DRIVER(A5984)
+ #define MINIMUM_STEPPER_POST_DIR_DELAY 400
+ #elif HAS_DRIVER(A4988)
+ #define MINIMUM_STEPPER_POST_DIR_DELAY 200
+ #elif HAS_TRINAMIC_CONFIG || HAS_TRINAMIC_STANDALONE
+ #define MINIMUM_STEPPER_POST_DIR_DELAY 60
+ #else
+ #define MINIMUM_STEPPER_POST_DIR_DELAY 0 // Expect at least 10µS since one Stepper ISR must transpire
+ #endif
+#endif
+
+#ifndef MINIMUM_STEPPER_PRE_DIR_DELAY
+ #define MINIMUM_STEPPER_PRE_DIR_DELAY MINIMUM_STEPPER_POST_DIR_DELAY
+#endif
+
+#ifndef MINIMUM_STEPPER_PULSE
+ #if HAS_DRIVER(TB6560)
+ #define MINIMUM_STEPPER_PULSE 30
+ #elif HAS_DRIVER(TB6600)
+ #define MINIMUM_STEPPER_PULSE 3
+ #elif HAS_DRIVER(DRV8825)
+ #define MINIMUM_STEPPER_PULSE 2
+ #elif HAS_DRIVER(A4988) || HAS_DRIVER(A5984)
+ #define MINIMUM_STEPPER_PULSE 1
+ #elif HAS_TRINAMIC_CONFIG || HAS_TRINAMIC_STANDALONE
+ #define MINIMUM_STEPPER_PULSE 0
+ #elif HAS_DRIVER(LV8729)
+ #define MINIMUM_STEPPER_PULSE 0
+ #else
+ #define MINIMUM_STEPPER_PULSE 2
+ #endif
+#endif
+
+#ifndef MAXIMUM_STEPPER_RATE
+ #if HAS_DRIVER(TB6560)
+ #define MAXIMUM_STEPPER_RATE 15000
+ #elif HAS_DRIVER(TB6600)
+ #define MAXIMUM_STEPPER_RATE 150000
+ #elif HAS_DRIVER(DRV8825)
+ #define MAXIMUM_STEPPER_RATE 250000
+ #elif HAS_DRIVER(A4988)
+ #define MAXIMUM_STEPPER_RATE 500000
+ #elif HAS_DRIVER(LV8729)
+ #define MAXIMUM_STEPPER_RATE 1000000
+ #elif HAS_TRINAMIC_CONFIG || HAS_TRINAMIC_STANDALONE
+ #define MAXIMUM_STEPPER_RATE 5000000
+ #else
+ #define MAXIMUM_STEPPER_RATE 250000
+ #endif
+#endif
+
+#if ENABLED(DIRECT_STEPPING)
+ #ifndef STEPPER_PAGES
+ #define STEPPER_PAGES 16
+ #endif
+ #ifndef STEPPER_PAGE_FORMAT
+ #define STEPPER_PAGE_FORMAT SP_4x2_256
+ #endif
+ #ifndef PAGE_MANAGER
+ #define PAGE_MANAGER SerialPageManager
+ #endif
+#endif
+
+//
+// SD Card connection methods
+// Defined here so pins and sanity checks can use them
+//
+#if ENABLED(SDSUPPORT)
+ #define _SDCARD_LCD 1
+ #define _SDCARD_ONBOARD 2
+ #define _SDCARD_CUSTOM_CABLE 3
+ #define _SDCARD_ID(V) _CAT(_SDCARD_, V)
+ #define SD_CONNECTION_IS(V) (_SDCARD_ID(SDCARD_CONNECTION) == _SDCARD_ID(V))
+#else
+ #define SD_CONNECTION_IS(...) 0
+#endif
+
+// Power Monitor sensors
+#if EITHER(POWER_MONITOR_CURRENT, POWER_MONITOR_VOLTAGE)
+ #define HAS_POWER_MONITOR 1
+#endif
+#if ENABLED(POWER_MONITOR_CURRENT) && defined(POWER_MONITOR_FIXED_VOLTAGE)
+ #define HAS_POWER_MONITOR_VREF 1
+#endif
+#if BOTH(HAS_POWER_MONITOR_VREF, POWER_MONITOR_CURRENT)
+ #define HAS_POWER_MONITOR_WATTS 1
+#endif
+
+// Flag if an EEPROM type is pre-selected
+#if ENABLED(EEPROM_SETTINGS) && NONE(I2C_EEPROM, SPI_EEPROM, QSPI_EEPROM, FLASH_EEPROM_EMULATION, SRAM_EEPROM_EMULATION, SDCARD_EEPROM_EMULATION)
+ #define NO_EEPROM_SELECTED 1
+#endif
+
+// Flag whether hex_print.cpp is used
+#if ANY(AUTO_BED_LEVELING_UBL, M100_FREE_MEMORY_WATCHER, DEBUG_GCODE_PARSER, TMC_DEBUG, MARLIN_DEV_MODE)
+ #define NEED_HEX_PRINT 1
+#endif
+
+// Flag whether least_squares_fit.cpp is used
+#if ANY(AUTO_BED_LEVELING_UBL, AUTO_BED_LEVELING_LINEAR, Z_STEPPER_ALIGN_KNOWN_STEPPER_POSITIONS)
+ #define NEED_LSF 1
+#endif
+
+// Flag the indexed serial ports that are in use
+#define ANY_SERIAL_IS(N) (defined(SERIAL_PORT) && SERIAL_PORT == (N)) || \
+ (defined(SERIAL_PORT_2) && SERIAL_PORT_2 == (N)) || \
+ (defined(MMU2_SERIAL_PORT) && MMU2_SERIAL_PORT == (N)) || \
+ (defined(LCD_SERIAL_PORT) && LCD_SERIAL_PORT == (N))
+#if ANY_SERIAL_IS(-1)
+ #define USING_SERIAL_DEFAULT
+#endif
+#if ANY_SERIAL_IS(0)
+ #define USING_SERIAL_0 1
+#endif
+#if ANY_SERIAL_IS(1)
+ #define USING_SERIAL_1 1
+#endif
+#if ANY_SERIAL_IS(2)
+ #define USING_SERIAL_2 1
+#endif
+#if ANY_SERIAL_IS(3)
+ #define USING_SERIAL_3 1
+#endif
+#if ANY_SERIAL_IS(4)
+ #define USING_SERIAL_4 1
+#endif
+#if ANY_SERIAL_IS(5)
+ #define USING_SERIAL_5 1
+#endif
+#if ANY_SERIAL_IS(6)
+ #define USING_SERIAL_6 1
+#endif
+#if ANY_SERIAL_IS(7)
+ #define USING_SERIAL_7 1
+#endif
+#if ANY_SERIAL_IS(8)
+ #define USING_SERIAL_8 1
+#endif
+#undef ANY_SERIAL_IS
diff --git a/Marlin/src/inc/Conditionals_post.h b/Marlin/src/inc/Conditionals_post.h
new file mode 100644
index 0000000..acf8f36
--- /dev/null
+++ b/Marlin/src/inc/Conditionals_post.h
@@ -0,0 +1,2751 @@
+/**
+ * Marlin 3D Printer Firmware
+ * Copyright (c) 2020 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
+ *
+ * Based on Sprinter and grbl.
+ * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ *
+ */
+#pragma once
+
+/**
+ * Conditionals_post.h
+ * Defines that depend on configuration but are not editable.
+ */
+
+#ifdef GITHUB_ACTIONS
+ // Extras for CI testing
+#endif
+
+// ADC
+#ifdef BOARD_ADC_VREF
+ #define ADC_VREF BOARD_ADC_VREF
+#else
+ #define ADC_VREF HAL_ADC_VREF
+#endif
+
+// Linear advance uses Jerk since E is an isolated axis
+#if BOTH(HAS_JUNCTION_DEVIATION, LIN_ADVANCE)
+ #define HAS_LINEAR_E_JERK 1
+#endif
+
+// Determine which type of 'EEPROM' is in use
+#if ENABLED(EEPROM_SETTINGS)
+ // EEPROM type may be defined by compile flags, configs, HALs, or pins
+ // Set additional flags to let HALs choose in their Conditionals_post.h
+ #if ANY(FLASH_EEPROM_EMULATION, SRAM_EEPROM_EMULATION, SDCARD_EEPROM_EMULATION, QSPI_EEPROM)
+ #define USE_EMULATED_EEPROM 1
+ #elif ANY(I2C_EEPROM, SPI_EEPROM)
+ #define USE_WIRED_EEPROM 1
+ #elif ENABLED(IIC_BL24CXX_EEPROM)
+ // nothing
+ #else
+ #define USE_FALLBACK_EEPROM 1
+ #endif
+#else
+ #undef I2C_EEPROM
+ #undef SPI_EEPROM
+ #undef QSPI_EEPROM
+ #undef SDCARD_EEPROM_EMULATION
+ #undef SRAM_EEPROM_EMULATION
+ #undef FLASH_EEPROM_EMULATION
+ #undef IIC_BL24CXX_EEPROM
+#endif
+
+#ifdef TEENSYDUINO
+ #undef max
+ #define max(a,b) ((a)>(b)?(a):(b))
+ #undef min
+ #define min(a,b) ((a)<(b)?(a):(b))
+
+ #undef NOT_A_PIN // Override Teensyduino legacy CapSense define work-around
+ #define NOT_A_PIN 0 // For PINS_DEBUGGING
+#endif
+
+/**
+ * Axis lengths and center
+ */
+#define X_MAX_LENGTH (X_MAX_POS - (X_MIN_POS))
+#define Y_MAX_LENGTH (Y_MAX_POS - (Y_MIN_POS))
+#define Z_MAX_LENGTH (Z_MAX_POS - (Z_MIN_POS))
+
+// Defined only if the sanity-check is bypassed
+#ifndef X_BED_SIZE
+ #define X_BED_SIZE X_MAX_LENGTH
+#endif
+#ifndef Y_BED_SIZE
+ #define Y_BED_SIZE Y_MAX_LENGTH
+#endif
+
+// Require 0,0 bed center for Delta and SCARA
+#if IS_KINEMATIC
+ #define BED_CENTER_AT_0_0
+#endif
+
+// Define center values for future use
+#define _X_HALF_BED ((X_BED_SIZE) / 2)
+#define _Y_HALF_BED ((Y_BED_SIZE) / 2)
+#define X_CENTER TERN(BED_CENTER_AT_0_0, 0, _X_HALF_BED)
+#define Y_CENTER TERN(BED_CENTER_AT_0_0, 0, _Y_HALF_BED)
+#define XY_CENTER { X_CENTER, Y_CENTER }
+
+// Get the linear boundaries of the bed
+#define X_MIN_BED (X_CENTER - _X_HALF_BED)
+#define X_MAX_BED (X_MIN_BED + X_BED_SIZE)
+#define Y_MIN_BED (Y_CENTER - _Y_HALF_BED)
+#define Y_MAX_BED (Y_MIN_BED + Y_BED_SIZE)
+
+/**
+ * Dual X Carriage
+ */
+#if ENABLED(DUAL_X_CARRIAGE)
+ #ifndef X1_MIN_POS
+ #define X1_MIN_POS X_MIN_POS
+ #endif
+ #ifndef X1_MAX_POS
+ #define X1_MAX_POS X_BED_SIZE
+ #endif
+#endif
+
+// Calibration codes only for non-core axes
+#if EITHER(BACKLASH_GCODE, CALIBRATION_GCODE)
+ #if EITHER(IS_CORE, MARKFORGED_XY)
+ #define CAN_CALIBRATE(A,B) (_AXIS(A) == B)
+ #else
+ #define CAN_CALIBRATE(A,B) 1
+ #endif
+#endif
+#define AXIS_CAN_CALIBRATE(A) CAN_CALIBRATE(A,NORMAL_AXIS)
+
+/**
+ * No adjustable bed on non-cartesians
+ */
+#if IS_KINEMATIC
+ #undef LEVEL_BED_CORNERS
+#endif
+
+/**
+ * SCARA cannot use SLOWDOWN and requires QUICKHOME
+ * Printable radius assumes joints can fully extend
+ */
+#if IS_SCARA
+ #undef SLOWDOWN
+ #define QUICK_HOME
+ #define SCARA_PRINTABLE_RADIUS (SCARA_LINKAGE_1 + SCARA_LINKAGE_2)
+#endif
+
+/**
+ * Set the home position based on settings or manual overrides
+ */
+#ifdef MANUAL_X_HOME_POS
+ #define X_HOME_POS MANUAL_X_HOME_POS
+#else
+ #define X_END_POS (X_HOME_DIR < 0 ? X_MIN_POS : X_MAX_POS)
+ #if ENABLED(BED_CENTER_AT_0_0)
+ #define X_HOME_POS TERN(DELTA, 0, X_END_POS)
+ #else
+ #define X_HOME_POS TERN(DELTA, X_MIN_POS + (X_BED_SIZE) * 0.5, X_END_POS)
+ #endif
+#endif
+
+#ifdef MANUAL_Y_HOME_POS
+ #define Y_HOME_POS MANUAL_Y_HOME_POS
+#else
+ #define Y_END_POS (Y_HOME_DIR < 0 ? Y_MIN_POS : Y_MAX_POS)
+ #if ENABLED(BED_CENTER_AT_0_0)
+ #define Y_HOME_POS TERN(DELTA, 0, Y_END_POS)
+ #else
+ #define Y_HOME_POS TERN(DELTA, Y_MIN_POS + (Y_BED_SIZE) * 0.5, Y_END_POS)
+ #endif
+#endif
+
+#ifdef MANUAL_Z_HOME_POS
+ #define Z_HOME_POS MANUAL_Z_HOME_POS
+#else
+ #define Z_HOME_POS (Z_HOME_DIR < 0 ? Z_MIN_POS : Z_MAX_POS)
+#endif
+
+/**
+ * If DELTA_HEIGHT isn't defined use the old setting
+ */
+#if ENABLED(DELTA) && !defined(DELTA_HEIGHT)
+ #define DELTA_HEIGHT Z_HOME_POS
+#endif
+
+/**
+ * Z Sled Probe requires Z_SAFE_HOMING
+ */
+#if ENABLED(Z_PROBE_SLED)
+ #define Z_SAFE_HOMING
+#endif
+
+/**
+ * DELTA should ignore Z_SAFE_HOMING and SLOWDOWN
+ */
+#if ENABLED(DELTA)
+ #undef Z_SAFE_HOMING
+ #undef SLOWDOWN
+#endif
+
+#ifndef MESH_INSET
+ #define MESH_INSET 0
+#endif
+
+/**
+ * Safe Homing Options
+ */
+#if ENABLED(Z_SAFE_HOMING)
+ #if ENABLED(AUTO_BED_LEVELING_UBL)
+ // Home close to center so grid points have z heights very close to 0
+ #define _SAFE_POINT(A) (((GRID_MAX_POINTS_##A) / 2) * (A##_BED_SIZE - 2 * (MESH_INSET)) / (GRID_MAX_POINTS_##A - 1) + MESH_INSET)
+ #else
+ #define _SAFE_POINT(A) A##_CENTER
+ #endif
+ #ifndef Z_SAFE_HOMING_X_POINT
+ #define Z_SAFE_HOMING_X_POINT _SAFE_POINT(X)
+ #endif
+ #ifndef Z_SAFE_HOMING_Y_POINT
+ #define Z_SAFE_HOMING_Y_POINT _SAFE_POINT(Y)
+ #endif
+#endif
+
+/**
+ * Host keep alive
+ */
+#ifndef DEFAULT_KEEPALIVE_INTERVAL
+ #define DEFAULT_KEEPALIVE_INTERVAL 2
+#endif
+
+/**
+ * Provide a MAX_AUTORETRACT for older configs
+ */
+#if ENABLED(FWRETRACT) && !defined(MAX_AUTORETRACT)
+ #define MAX_AUTORETRACT 99
+#endif
+
+/**
+ * Provide a DEFAULT_VOLUMETRIC_EXTRUDER_LIMIT in case NO_VOLUMETRICS is enabled
+ */
+#ifndef DEFAULT_VOLUMETRIC_EXTRUDER_LIMIT
+ #define DEFAULT_VOLUMETRIC_EXTRUDER_LIMIT 0.00
+#endif
+
+/**
+ * LCD Contrast for Graphical Displays
+ */
+#if ENABLED(CARTESIO_UI)
+ #define _LCD_CONTRAST_MIN 60
+ #define _LCD_CONTRAST_INIT 90
+ #define _LCD_CONTRAST_MAX 140
+#elif ENABLED(miniVIKI)
+ #define _LCD_CONTRAST_MIN 75
+ #define _LCD_CONTRAST_INIT 95
+ #define _LCD_CONTRAST_MAX 115
+#elif ENABLED(VIKI2)
+ #define _LCD_CONTRAST_INIT 140
+#elif ENABLED(ELB_FULL_GRAPHIC_CONTROLLER)
+ #define _LCD_CONTRAST_MIN 90
+ #define _LCD_CONTRAST_INIT 110
+ #define _LCD_CONTRAST_MAX 130
+#elif ENABLED(AZSMZ_12864)
+ #define _LCD_CONTRAST_MIN 120
+ #define _LCD_CONTRAST_INIT 190
+#elif ENABLED(MKS_LCD12864)
+ #define _LCD_CONTRAST_MIN 120
+ #define _LCD_CONTRAST_INIT 205
+#elif EITHER(MKS_MINI_12864, ENDER2_STOCKDISPLAY)
+ #define _LCD_CONTRAST_MIN 120
+ #define _LCD_CONTRAST_INIT 195
+#elif ENABLED(FYSETC_MINI_12864)
+ #define _LCD_CONTRAST_INIT 220
+#elif ENABLED(ULTI_CONTROLLER)
+ #define _LCD_CONTRAST_INIT 127
+ #define _LCD_CONTRAST_MAX 254
+#elif ENABLED(MAKRPANEL)
+ #define _LCD_CONTRAST_INIT 17
+#elif ENABLED(MINIPANEL)
+ #define _LCD_CONTRAST_INIT 150
+#elif ENABLED(ZONESTAR_12864OLED)
+ #define _LCD_CONTRAST_MIN 64
+ #define _LCD_CONTRAST_INIT 128
+ #define _LCD_CONTRAST_MAX 255
+#elif IS_TFTGLCD_PANEL
+ #define _LCD_CONTRAST_MIN 0
+ #define _LCD_CONTRAST_INIT 250
+ #define _LCD_CONTRAST_MAX 255
+#endif
+
+#ifdef _LCD_CONTRAST_INIT
+ #define HAS_LCD_CONTRAST 1
+#endif
+
+#if HAS_LCD_CONTRAST
+ #ifndef LCD_CONTRAST_MIN
+ #ifdef _LCD_CONTRAST_MIN
+ #define LCD_CONTRAST_MIN _LCD_CONTRAST_MIN
+ #else
+ #define LCD_CONTRAST_MIN 0
+ #endif
+ #endif
+ #ifndef LCD_CONTRAST_INIT
+ #define LCD_CONTRAST_INIT _LCD_CONTRAST_INIT
+ #endif
+ #ifndef LCD_CONTRAST_MAX
+ #ifdef _LCD_CONTRAST_MAX
+ #define LCD_CONTRAST_MAX _LCD_CONTRAST_MAX
+ #elif _LCD_CONTRAST_INIT > 63
+ #define LCD_CONTRAST_MAX 255
+ #else
+ #define LCD_CONTRAST_MAX 63 // ST7567 6-bits contrast
+ #endif
+ #endif
+ #ifndef DEFAULT_LCD_CONTRAST
+ #define DEFAULT_LCD_CONTRAST LCD_CONTRAST_INIT
+ #endif
+#endif
+
+/**
+ * Override the SD_DETECT_STATE set in Configuration_adv.h
+ * and enable sharing of onboard SD host drives (all platforms but AGCM4)
+ */
+#if ENABLED(SDSUPPORT)
+
+ // Extender cable doesn't support SD_DETECT_PIN
+ #if ENABLED(NO_SD_DETECT)
+ #undef SD_DETECT_PIN
+ #endif
+
+ #if HAS_SD_HOST_DRIVE && SD_CONNECTION_IS(ONBOARD)
+ //
+ // The external SD card is not used. Hardware SPI is used to access the card.
+ // When sharing the SD card with a PC we want the menu options to
+ // mount/unmount the card and refresh it. So we disable card detect.
+ //
+ #undef SD_DETECT_PIN
+ #define HAS_SHARED_MEDIA 1
+ #endif
+
+ #if PIN_EXISTS(SD_DETECT)
+ #if HAS_LCD_MENU && (SD_CONNECTION_IS(LCD) || !defined(SDCARD_CONNECTION))
+ #undef SD_DETECT_STATE
+ #if ENABLED(ELB_FULL_GRAPHIC_CONTROLLER)
+ #define SD_DETECT_STATE HIGH
+ #endif
+ #endif
+ #ifndef SD_DETECT_STATE
+ #define SD_DETECT_STATE LOW
+ #endif
+ #endif
+#endif
+
+#if ANY(HAS_GRAPHICAL_TFT, LCD_USE_DMA_FSMC, HAS_FSMC_GRAPHICAL_TFT, HAS_SPI_GRAPHICAL_TFT) || !PIN_EXISTS(SD_DETECT)
+ #define NO_LCD_REINIT 1 // Suppress LCD re-initialization
+#endif
+
+/**
+ * Set defaults for missing (newer) options
+ */
+#ifndef DISABLE_INACTIVE_X
+ #define DISABLE_INACTIVE_X DISABLE_X
+#endif
+#ifndef DISABLE_INACTIVE_Y
+ #define DISABLE_INACTIVE_Y DISABLE_Y
+#endif
+#ifndef DISABLE_INACTIVE_Z
+ #define DISABLE_INACTIVE_Z DISABLE_Z
+#endif
+#ifndef DISABLE_INACTIVE_E
+ #define DISABLE_INACTIVE_E DISABLE_E
+#endif
+
+/**
+ * Power Supply
+ */
+#ifndef PSU_NAME
+ #if DISABLED(PSU_CONTROL)
+ #define PSU_NAME "Generic" // No control
+ #elif PSU_ACTIVE_STATE
+ #define PSU_NAME "XBox" // X-Box 360 (203W)
+ #else
+ #define PSU_NAME "ATX" // ATX style
+ #endif
+#endif
+
+#if ENABLED(PSU_CONTROL)
+ #ifndef PSU_POWERUP_DELAY
+ #define PSU_POWERUP_DELAY 250
+ #endif
+ #ifndef POWER_OFF_DELAY
+ #define POWER_OFF_DELAY 0
+ #endif
+#endif
+
+/**
+ * Temp Sensor defines
+ */
+
+#define ANY_TEMP_SENSOR_IS(n) (TEMP_SENSOR_0 == (n) || TEMP_SENSOR_1 == (n) || TEMP_SENSOR_2 == (n) || TEMP_SENSOR_3 == (n) || TEMP_SENSOR_4 == (n) || TEMP_SENSOR_5 == (n) || TEMP_SENSOR_6 == (n) || TEMP_SENSOR_7 == (n) || TEMP_SENSOR_BED == (n) || TEMP_SENSOR_PROBE == (n) || TEMP_SENSOR_CHAMBER == (n))
+
+#if ANY_TEMP_SENSOR_IS(1000)
+ #define HAS_USER_THERMISTORS 1
+#endif
+
+#if TEMP_SENSOR_0 == -5 || TEMP_SENSOR_0 == -3 || TEMP_SENSOR_0 == -2
+ #define HEATER_0_USES_MAX6675 1
+ #if TEMP_SENSOR_0 == -3
+ #define HEATER_0_MAX6675_TMIN -270
+ #define HEATER_0_MAX6675_TMAX 1800
+ #else
+ #define HEATER_0_MAX6675_TMIN 0
+ #define HEATER_0_MAX6675_TMAX 1024
+ #endif
+ #if TEMP_SENSOR_0 == -5
+ #define MAX6675_0_IS_MAX31865 1
+ #elif TEMP_SENSOR_0 == -3
+ #define MAX6675_0_IS_MAX31855 1
+ #endif
+#elif TEMP_SENSOR_0 == -4
+ #define HEATER_0_USES_AD8495 1
+#elif TEMP_SENSOR_0 == -1
+ #define HEATER_0_USES_AD595 1
+#elif TEMP_SENSOR_0 > 0
+ #define THERMISTOR_HEATER_0 TEMP_SENSOR_0
+ #define HEATER_0_USES_THERMISTOR 1
+ #if TEMP_SENSOR_0 == 1000
+ #define HEATER_0_USER_THERMISTOR 1
+ #elif TEMP_SENSOR_0 == 998 || TEMP_SENSOR_0 == 999
+ #define HEATER_0_DUMMY_THERMISTOR 1
+ #endif
+#else
+ #undef HEATER_0_MINTEMP
+ #undef HEATER_0_MAXTEMP
+#endif
+
+#if TEMP_SENSOR_1 == -5 || TEMP_SENSOR_1 == -3 || TEMP_SENSOR_1 == -2
+ #define HEATER_1_USES_MAX6675 1
+ #if TEMP_SENSOR_1 == -3
+ #define HEATER_1_MAX6675_TMIN -270
+ #define HEATER_1_MAX6675_TMAX 1800
+ #else
+ #define HEATER_1_MAX6675_TMIN 0
+ #define HEATER_1_MAX6675_TMAX 1024
+ #endif
+ #if TEMP_SENSOR_1 == -5
+ #define MAX6675_1_IS_MAX31865 1
+ #elif TEMP_SENSOR_1 == -3
+ #define MAX6675_1_IS_MAX31855 1
+ #endif
+ #if TEMP_SENSOR_1 != TEMP_SENSOR_0
+ #if TEMP_SENSOR_1 == -5
+ #error "If MAX31865 Thermocouple (-5) is used for TEMP_SENSOR_1 then TEMP_SENSOR_0 must match."
+ #elif TEMP_SENSOR_1 == -3
+ #error "If MAX31855 Thermocouple (-3) is used for TEMP_SENSOR_1 then TEMP_SENSOR_0 must match."
+ #elif TEMP_SENSOR_1 == -2
+ #error "If MAX6675 Thermocouple (-2) is used for TEMP_SENSOR_1 then TEMP_SENSOR_0 must match."
+ #endif
+ #endif
+#elif TEMP_SENSOR_1 == -4
+ #define HEATER_1_USES_AD8495 1
+#elif TEMP_SENSOR_1 == -1
+ #define HEATER_1_USES_AD595 1
+#elif TEMP_SENSOR_1 > 0
+ #define THERMISTOR_HEATER_1 TEMP_SENSOR_1
+ #define HEATER_1_USES_THERMISTOR 1
+ #if TEMP_SENSOR_1 == 1000
+ #define HEATER_1_USER_THERMISTOR 1
+ #elif TEMP_SENSOR_1 == 998 || TEMP_SENSOR_1 == 999
+ #define HEATER_1_DUMMY_THERMISTOR 1
+ #endif
+#else
+ #undef HEATER_1_MINTEMP
+ #undef HEATER_1_MAXTEMP
+#endif
+
+#if TEMP_SENSOR_2 == -4
+ #define HEATER_2_USES_AD8495 1
+#elif TEMP_SENSOR_2 == -3
+ #error "MAX31855 Thermocouples (-3) not supported for TEMP_SENSOR_2."
+#elif TEMP_SENSOR_2 == -2
+ #error "MAX6675 Thermocouples (-2) not supported for TEMP_SENSOR_2."
+#elif TEMP_SENSOR_2 == -1
+ #define HEATER_2_USES_AD595 1
+#elif TEMP_SENSOR_2 > 0
+ #define THERMISTOR_HEATER_2 TEMP_SENSOR_2
+ #define HEATER_2_USES_THERMISTOR 1
+ #if TEMP_SENSOR_2 == 1000
+ #define HEATER_2_USER_THERMISTOR 1
+ #elif TEMP_SENSOR_2 == 998 || TEMP_SENSOR_2 == 999
+ #define HEATER_2_DUMMY_THERMISTOR 1
+ #endif
+#else
+ #undef HEATER_2_MINTEMP
+ #undef HEATER_2_MAXTEMP
+#endif
+
+#if TEMP_SENSOR_3 == -4
+ #define HEATER_3_USES_AD8495 1
+#elif TEMP_SENSOR_3 == -3
+ #error "MAX31855 Thermocouples (-3) not supported for TEMP_SENSOR_3."
+#elif TEMP_SENSOR_3 == -2
+ #error "MAX6675 Thermocouples (-2) not supported for TEMP_SENSOR_3."
+#elif TEMP_SENSOR_3 == -1
+ #define HEATER_3_USES_AD595 1
+#elif TEMP_SENSOR_3 > 0
+ #define THERMISTOR_HEATER_3 TEMP_SENSOR_3
+ #define HEATER_3_USES_THERMISTOR 1
+ #if TEMP_SENSOR_3 == 1000
+ #define HEATER_3_USER_THERMISTOR 1
+ #elif TEMP_SENSOR_3 == 998 || TEMP_SENSOR_3 == 999
+ #define HEATER_3_DUMMY_THERMISTOR 1
+ #endif
+#else
+ #undef HEATER_3_MINTEMP
+ #undef HEATER_3_MAXTEMP
+#endif
+
+#if TEMP_SENSOR_4 == -4
+ #define HEATER_4_USES_AD8495 1
+#elif TEMP_SENSOR_4 == -3
+ #error "MAX31855 Thermocouples (-3) not supported for TEMP_SENSOR_4."
+#elif TEMP_SENSOR_4 == -2
+ #error "MAX6675 Thermocouples (-2) not supported for TEMP_SENSOR_4."
+#elif TEMP_SENSOR_4 == -1
+ #define HEATER_4_USES_AD595 1
+#elif TEMP_SENSOR_4 > 0
+ #define THERMISTOR_HEATER_4 TEMP_SENSOR_4
+ #define HEATER_4_USES_THERMISTOR 1
+ #if TEMP_SENSOR_4 == 1000
+ #define HEATER_4_USER_THERMISTOR 1
+ #elif TEMP_SENSOR_4 == 998 || TEMP_SENSOR_4 == 999
+ #define HEATER_4_DUMMY_THERMISTOR 1
+ #endif
+#else
+ #undef HEATER_4_MINTEMP
+ #undef HEATER_4_MAXTEMP
+#endif
+
+#if TEMP_SENSOR_5 == -4
+ #define HEATER_5_USES_AD8495 1
+#elif TEMP_SENSOR_5 == -3
+ #error "MAX31855 Thermocouples (-3) not supported for TEMP_SENSOR_5."
+#elif TEMP_SENSOR_5 == -2
+ #error "MAX6675 Thermocouples (-2) not supported for TEMP_SENSOR_5."
+#elif TEMP_SENSOR_5 == -1
+ #define HEATER_5_USES_AD595 1
+#elif TEMP_SENSOR_5 > 0
+ #define THERMISTOR_HEATER_5 TEMP_SENSOR_5
+ #define HEATER_5_USES_THERMISTOR 1
+ #if TEMP_SENSOR_5 == 1000
+ #define HEATER_5_USER_THERMISTOR 1
+ #elif TEMP_SENSOR_5 == 998 || TEMP_SENSOR_5 == 999
+ #define HEATER_5_DUMMY_THERMISTOR 1
+ #endif
+#else
+ #undef HEATER_5_MINTEMP
+ #undef HEATER_5_MAXTEMP
+#endif
+
+#if TEMP_SENSOR_6 == -4
+ #define HEATER_6_USES_AD8495 1
+#elif TEMP_SENSOR_6 == -3
+ #error "MAX31855 Thermocouples (-3) not supported for TEMP_SENSOR_6."
+#elif TEMP_SENSOR_6 == -2
+ #error "MAX6675 Thermocouples (-2) not supported for TEMP_SENSOR_6."
+#elif TEMP_SENSOR_6 == -1
+ #define HEATER_6_USES_AD595 1
+#elif TEMP_SENSOR_6 > 0
+ #define THERMISTOR_HEATER_6 TEMP_SENSOR_6
+ #define HEATER_6_USES_THERMISTOR 1
+ #if TEMP_SENSOR_6 == 1000
+ #define HEATER_6_USER_THERMISTOR 1
+ #elif TEMP_SENSOR_6 == 998 || TEMP_SENSOR_6 == 999
+ #define HEATER_6_DUMMY_THERMISTOR 1
+ #endif
+#else
+ #undef HEATER_6_MINTEMP
+ #undef HEATER_6_MAXTEMP
+#endif
+
+#if TEMP_SENSOR_7 == -4
+ #define HEATER_7_USES_AD8495 1
+#elif TEMP_SENSOR_7 == -3
+ #error "MAX31855 Thermocouples (-3) not supported for TEMP_SENSOR_7."
+#elif TEMP_SENSOR_7 == -2
+ #error "MAX7775 Thermocouples (-2) not supported for TEMP_SENSOR_7."
+#elif TEMP_SENSOR_7 == -1
+ #define HEATER_7_USES_AD595 1
+#elif TEMP_SENSOR_7 > 0
+ #define THERMISTOR_HEATER_7 TEMP_SENSOR_7
+ #define HEATER_7_USES_THERMISTOR 1
+ #if TEMP_SENSOR_7 == 1000
+ #define HEATER_7_USER_THERMISTOR 1
+ #elif TEMP_SENSOR_7 == 998 || TEMP_SENSOR_7 == 999
+ #define HEATER_7_DUMMY_THERMISTOR 1
+ #endif
+#else
+ #undef HEATER_7_MINTEMP
+ #undef HEATER_7_MAXTEMP
+#endif
+
+#if TEMP_SENSOR_BED == -4
+ #define HEATER_BED_USES_AD8495 1
+#elif TEMP_SENSOR_BED == -3
+ #error "MAX31855 Thermocouples (-3) not supported for TEMP_SENSOR_BED."
+#elif TEMP_SENSOR_BED == -2
+ #error "MAX6675 Thermocouples (-2) not supported for TEMP_SENSOR_BED."
+#elif TEMP_SENSOR_BED == -1
+ #define HEATER_BED_USES_AD595 1
+#elif TEMP_SENSOR_BED > 0
+ #define THERMISTORBED TEMP_SENSOR_BED
+ #define HEATER_BED_USES_THERMISTOR 1
+ #if TEMP_SENSOR_BED == 1000
+ #define HEATER_BED_USER_THERMISTOR 1
+ #elif TEMP_SENSOR_BED == 998 || TEMP_SENSOR_BED == 999
+ #define HEATER_BED_DUMMY_THERMISTOR 1
+ #endif
+#else
+ #undef BED_MINTEMP
+ #undef BED_MAXTEMP
+#endif
+
+#if TEMP_SENSOR_CHAMBER == -4
+ #define HEATER_CHAMBER_USES_AD8495 1
+#elif TEMP_SENSOR_CHAMBER == -3
+ #error "MAX31855 Thermocouples (-3) not supported for TEMP_SENSOR_CHAMBER."
+#elif TEMP_SENSOR_CHAMBER == -2
+ #error "MAX6675 Thermocouples (-2) not supported for TEMP_SENSOR_CHAMBER."
+#elif TEMP_SENSOR_CHAMBER == -1
+ #define HEATER_CHAMBER_USES_AD595 1
+#elif TEMP_SENSOR_CHAMBER > 0
+ #define THERMISTORCHAMBER TEMP_SENSOR_CHAMBER
+ #define HEATER_CHAMBER_USES_THERMISTOR 1
+ #if TEMP_SENSOR_CHAMBER == 1000
+ #define HEATER_CHAMBER_USER_THERMISTOR 1
+ #elif TEMP_SENSOR_CHAMBER == 998 || TEMP_SENSOR_CHAMBER == 999
+ #define HEATER_CHAMBER_DUMMY_THERMISTOR 1
+ #endif
+#else
+ #undef CHAMBER_MINTEMP
+ #undef CHAMBER_MAXTEMP
+#endif
+
+#if TEMP_SENSOR_PROBE == -4
+ #define HEATER_PROBE_USES_AD8495 1
+#elif TEMP_SENSOR_PROBE == -3
+ #error "MAX31855 Thermocouples (-3) not supported for TEMP_SENSOR_PROBE."
+#elif TEMP_SENSOR_PROBE == -2
+ #error "MAX6675 Thermocouples (-2) not supported for TEMP_SENSOR_PROBE."
+#elif TEMP_SENSOR_PROBE == -1
+ #define HEATER_PROBE_USES_AD595 1
+#elif TEMP_SENSOR_PROBE > 0
+ #define THERMISTORPROBE TEMP_SENSOR_PROBE
+ #define HEATER_PROBE_USES_THERMISTOR 1
+ #if TEMP_SENSOR_PROBE == 1000
+ #define HEATER_PROBE_USER_THERMISTOR 1
+ #elif TEMP_SENSOR_PROBE == 998 || TEMP_SENSOR_PROBE == 999
+ #define HEATER_PROBE_DUMMY_THERMISTOR 1
+ #endif
+#endif
+
+/**
+ * X_DUAL_ENDSTOPS endstop reassignment
+ */
+#if ENABLED(X_DUAL_ENDSTOPS)
+ #if X_HOME_DIR > 0
+ #ifndef X2_MAX_ENDSTOP_INVERTING
+ #if X2_USE_ENDSTOP == _XMIN_
+ #define X2_MAX_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
+ #elif X2_USE_ENDSTOP == _XMAX_
+ #define X2_MAX_ENDSTOP_INVERTING X_MAX_ENDSTOP_INVERTING
+ #elif X2_USE_ENDSTOP == _YMIN_
+ #define X2_MAX_ENDSTOP_INVERTING Y_MIN_ENDSTOP_INVERTING
+ #elif X2_USE_ENDSTOP == _YMAX_
+ #define X2_MAX_ENDSTOP_INVERTING Y_MAX_ENDSTOP_INVERTING
+ #elif X2_USE_ENDSTOP == _ZMIN_
+ #define X2_MAX_ENDSTOP_INVERTING Z_MIN_ENDSTOP_INVERTING
+ #elif X2_USE_ENDSTOP == _ZMAX_
+ #define X2_MAX_ENDSTOP_INVERTING Z_MAX_ENDSTOP_INVERTING
+ #else
+ #define X2_MAX_ENDSTOP_INVERTING false
+ #endif
+ #endif
+ #ifndef X2_MAX_PIN
+ #if X2_USE_ENDSTOP == _XMIN_
+ #define X2_MAX_PIN X_MIN_PIN
+ #elif X2_USE_ENDSTOP == _XMAX_
+ #define X2_MAX_PIN X_MAX_PIN
+ #elif X2_USE_ENDSTOP == _YMIN_
+ #define X2_MAX_PIN Y_MIN_PIN
+ #elif X2_USE_ENDSTOP == _YMAX_
+ #define X2_MAX_PIN Y_MAX_PIN
+ #elif X2_USE_ENDSTOP == _ZMIN_
+ #define X2_MAX_PIN Z_MIN_PIN
+ #elif X2_USE_ENDSTOP == _ZMAX_
+ #define X2_MAX_PIN Z_MAX_PIN
+ #elif X2_USE_ENDSTOP == _XDIAG_
+ #define X2_MAX_PIN X_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _YDIAG_
+ #define X2_MAX_PIN Y_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _ZDIAG_
+ #define X2_MAX_PIN Z_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _E0DIAG_
+ #define X2_MAX_PIN E0_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _E1DIAG_
+ #define X2_MAX_PIN E1_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _E2DIAG_
+ #define X2_MAX_PIN E2_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _E3DIAG_
+ #define X2_MAX_PIN E3_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _E4DIAG_
+ #define X2_MAX_PIN E4_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _E5DIAG_
+ #define X2_MAX_PIN E5_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _E6DIAG_
+ #define X2_MAX_PIN E6_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _E7DIAG_
+ #define X2_MAX_PIN E7_DIAG_PIN
+ #endif
+ #endif
+ #ifndef X2_MIN_ENDSTOP_INVERTING
+ #define X2_MIN_ENDSTOP_INVERTING false
+ #endif
+ #else
+ #ifndef X2_MIN_ENDSTOP_INVERTING
+ #if X2_USE_ENDSTOP == _XMIN_
+ #define X2_MIN_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
+ #elif X2_USE_ENDSTOP == _XMAX_
+ #define X2_MIN_ENDSTOP_INVERTING X_MAX_ENDSTOP_INVERTING
+ #elif X2_USE_ENDSTOP == _YMIN_
+ #define X2_MIN_ENDSTOP_INVERTING Y_MIN_ENDSTOP_INVERTING
+ #elif X2_USE_ENDSTOP == _YMAX_
+ #define X2_MIN_ENDSTOP_INVERTING Y_MAX_ENDSTOP_INVERTING
+ #elif X2_USE_ENDSTOP == _ZMIN_
+ #define X2_MIN_ENDSTOP_INVERTING Z_MIN_ENDSTOP_INVERTING
+ #elif X2_USE_ENDSTOP == _ZMAX_
+ #define X2_MIN_ENDSTOP_INVERTING Z_MAX_ENDSTOP_INVERTING
+ #else
+ #define X2_MIN_ENDSTOP_INVERTING false
+ #endif
+ #endif
+ #ifndef X2_MIN_PIN
+ #if X2_USE_ENDSTOP == _XMIN_
+ #define X2_MIN_PIN X_MIN_PIN
+ #elif X2_USE_ENDSTOP == _XMAX_
+ #define X2_MIN_PIN X_MAX_PIN
+ #elif X2_USE_ENDSTOP == _YMIN_
+ #define X2_MIN_PIN Y_MIN_PIN
+ #elif X2_USE_ENDSTOP == _YMAX_
+ #define X2_MIN_PIN Y_MAX_PIN
+ #elif X2_USE_ENDSTOP == _ZMIN_
+ #define X2_MIN_PIN Z_MIN_PIN
+ #elif X2_USE_ENDSTOP == _ZMAX_
+ #define X2_MIN_PIN Z_MAX_PIN
+ #elif X2_USE_ENDSTOP == _XDIAG_
+ #define X2_MIN_PIN X_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _YDIAG_
+ #define X2_MIN_PIN Y_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _ZDIAG_
+ #define X2_MIN_PIN Z_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _E0DIAG_
+ #define X2_MIN_PIN E0_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _E1DIAG_
+ #define X2_MIN_PIN E1_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _E2DIAG_
+ #define X2_MIN_PIN E2_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _E3DIAG_
+ #define X2_MIN_PIN E3_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _E4DIAG_
+ #define X2_MIN_PIN E4_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _E5DIAG_
+ #define X2_MIN_PIN E5_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _E6DIAG_
+ #define X2_MIN_PIN E6_DIAG_PIN
+ #elif X2_USE_ENDSTOP == _E7DIAG_
+ #define X2_MIN_PIN E7_DIAG_PIN
+ #endif
+ #endif
+ #ifndef X2_MAX_ENDSTOP_INVERTING
+ #define X2_MAX_ENDSTOP_INVERTING false
+ #endif
+ #endif
+#endif
+
+/**
+ * Y_DUAL_ENDSTOPS endstop reassignment
+ */
+#if ENABLED(Y_DUAL_ENDSTOPS)
+ #if Y_HOME_DIR > 0
+ #ifndef Y2_MAX_ENDSTOP_INVERTING
+ #if Y2_USE_ENDSTOP == _XMIN_
+ #define Y2_MAX_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
+ #elif Y2_USE_ENDSTOP == _XMAX_
+ #define Y2_MAX_ENDSTOP_INVERTING X_MAX_ENDSTOP_INVERTING
+ #elif Y2_USE_ENDSTOP == _YMIN_
+ #define Y2_MAX_ENDSTOP_INVERTING Y_MIN_ENDSTOP_INVERTING
+ #elif Y2_USE_ENDSTOP == _YMAX_
+ #define Y2_MAX_ENDSTOP_INVERTING Y_MAX_ENDSTOP_INVERTING
+ #elif Y2_USE_ENDSTOP == _ZMIN_
+ #define Y2_MAX_ENDSTOP_INVERTING Z_MIN_ENDSTOP_INVERTING
+ #elif Y2_USE_ENDSTOP == _ZMAX_
+ #define Y2_MAX_ENDSTOP_INVERTING Z_MAX_ENDSTOP_INVERTING
+ #else
+ #define Y2_MAX_ENDSTOP_INVERTING false
+ #endif
+ #endif
+ #ifndef Y2_MAX_PIN
+ #if Y2_USE_ENDSTOP == _XMIN_
+ #define Y2_MAX_PIN X_MIN_PIN
+ #elif Y2_USE_ENDSTOP == _XMAX_
+ #define Y2_MAX_PIN X_MAX_PIN
+ #elif Y2_USE_ENDSTOP == _YMIN_
+ #define Y2_MAX_PIN Y_MIN_PIN
+ #elif Y2_USE_ENDSTOP == _YMAX_
+ #define Y2_MAX_PIN Y_MAX_PIN
+ #elif Y2_USE_ENDSTOP == _ZMIN_
+ #define Y2_MAX_PIN Z_MIN_PIN
+ #elif Y2_USE_ENDSTOP == _ZMAX_
+ #define Y2_MAX_PIN Z_MAX_PIN
+ #elif Y2_USE_ENDSTOP == _XDIAG_
+ #define Y2_MAX_PIN X_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _YDIAG_
+ #define Y2_MAX_PIN Y_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _ZDIAG_
+ #define Y2_MAX_PIN Z_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _E0DIAG_
+ #define Y2_MAX_PIN E0_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _E1DIAG_
+ #define Y2_MAX_PIN E1_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _E2DIAG_
+ #define Y2_MAX_PIN E2_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _E3DIAG_
+ #define Y2_MAX_PIN E3_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _E4DIAG_
+ #define Y2_MAX_PIN E4_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _E5DIAG_
+ #define Y2_MAX_PIN E5_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _E6DIAG_
+ #define Y2_MAX_PIN E6_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _E7DIAG_
+ #define Y2_MAX_PIN E7_DIAG_PIN
+ #endif
+ #endif
+ #ifndef Y2_MIN_ENDSTOP_INVERTING
+ #define Y2_MIN_ENDSTOP_INVERTING false
+ #endif
+ #else
+ #ifndef Y2_MIN_ENDSTOP_INVERTING
+ #if Y2_USE_ENDSTOP == _XMIN_
+ #define Y2_MIN_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
+ #elif Y2_USE_ENDSTOP == _XMAX_
+ #define Y2_MIN_ENDSTOP_INVERTING X_MAX_ENDSTOP_INVERTING
+ #elif Y2_USE_ENDSTOP == _YMIN_
+ #define Y2_MIN_ENDSTOP_INVERTING Y_MIN_ENDSTOP_INVERTING
+ #elif Y2_USE_ENDSTOP == _YMAX_
+ #define Y2_MIN_ENDSTOP_INVERTING Y_MAX_ENDSTOP_INVERTING
+ #elif Y2_USE_ENDSTOP == _ZMIN_
+ #define Y2_MIN_ENDSTOP_INVERTING Z_MIN_ENDSTOP_INVERTING
+ #elif Y2_USE_ENDSTOP == _ZMAX_
+ #define Y2_MIN_ENDSTOP_INVERTING Z_MAX_ENDSTOP_INVERTING
+ #else
+ #define Y2_MIN_ENDSTOP_INVERTING false
+ #endif
+ #endif
+ #ifndef Y2_MIN_PIN
+ #if Y2_USE_ENDSTOP == _XMIN_
+ #define Y2_MIN_PIN X_MIN_PIN
+ #elif Y2_USE_ENDSTOP == _XMAX_
+ #define Y2_MIN_PIN X_MAX_PIN
+ #elif Y2_USE_ENDSTOP == _YMIN_
+ #define Y2_MIN_PIN Y_MIN_PIN
+ #elif Y2_USE_ENDSTOP == _YMAX_
+ #define Y2_MIN_PIN Y_MAX_PIN
+ #elif Y2_USE_ENDSTOP == _ZMIN_
+ #define Y2_MIN_PIN Z_MIN_PIN
+ #elif Y2_USE_ENDSTOP == _ZMAX_
+ #define Y2_MIN_PIN Z_MAX_PIN
+ #elif Y2_USE_ENDSTOP == _XDIAG_
+ #define Y2_MIN_PIN X_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _YDIAG_
+ #define Y2_MIN_PIN Y_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _ZDIAG_
+ #define Y2_MIN_PIN Z_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _E0DIAG_
+ #define Y2_MIN_PIN E0_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _E1DIAG_
+ #define Y2_MIN_PIN E1_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _E2DIAG_
+ #define Y2_MIN_PIN E2_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _E3DIAG_
+ #define Y2_MIN_PIN E3_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _E4DIAG_
+ #define Y2_MIN_PIN E4_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _E5DIAG_
+ #define Y2_MIN_PIN E5_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _E6DIAG_
+ #define Y2_MIN_PIN E6_DIAG_PIN
+ #elif Y2_USE_ENDSTOP == _E7DIAG_
+ #define Y2_MIN_PIN E7_DIAG_PIN
+ #endif
+ #endif
+ #ifndef Y2_MAX_ENDSTOP_INVERTING
+ #define Y2_MAX_ENDSTOP_INVERTING false
+ #endif
+ #endif
+#endif
+
+/**
+ * Z_MULTI_ENDSTOPS endstop reassignment
+ */
+#if ENABLED(Z_MULTI_ENDSTOPS)
+
+ #if Z_HOME_DIR > 0
+ #ifndef Z2_MAX_ENDSTOP_INVERTING
+ #if Z2_USE_ENDSTOP == _XMIN_
+ #define Z2_MAX_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
+ #elif Z2_USE_ENDSTOP == _XMAX_
+ #define Z2_MAX_ENDSTOP_INVERTING X_MAX_ENDSTOP_INVERTING
+ #elif Z2_USE_ENDSTOP == _YMIN_
+ #define Z2_MAX_ENDSTOP_INVERTING Y_MIN_ENDSTOP_INVERTING
+ #elif Z2_USE_ENDSTOP == _YMAX_
+ #define Z2_MAX_ENDSTOP_INVERTING Y_MAX_ENDSTOP_INVERTING
+ #elif Z2_USE_ENDSTOP == _ZMIN_
+ #define Z2_MAX_ENDSTOP_INVERTING Z_MIN_ENDSTOP_INVERTING
+ #elif Z2_USE_ENDSTOP == _ZMAX_
+ #define Z2_MAX_ENDSTOP_INVERTING Z_MAX_ENDSTOP_INVERTING
+ #else
+ #define Z2_MAX_ENDSTOP_INVERTING false
+ #endif
+ #endif
+ #ifndef Z2_MAX_PIN
+ #if Z2_USE_ENDSTOP == _XMIN_
+ #define Z2_MAX_PIN X_MIN_PIN
+ #elif Z2_USE_ENDSTOP == _XMAX_
+ #define Z2_MAX_PIN X_MAX_PIN
+ #elif Z2_USE_ENDSTOP == _YMIN_
+ #define Z2_MAX_PIN Y_MIN_PIN
+ #elif Z2_USE_ENDSTOP == _YMAX_
+ #define Z2_MAX_PIN Y_MAX_PIN
+ #elif Z2_USE_ENDSTOP == _ZMIN_
+ #define Z2_MAX_PIN Z_MIN_PIN
+ #elif Z2_USE_ENDSTOP == _ZMAX_
+ #define Z2_MAX_PIN Z_MAX_PIN
+ #elif Z2_USE_ENDSTOP == _XDIAG_
+ #define Z2_MAX_PIN X_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _YDIAG_
+ #define Z2_MAX_PIN Y_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _ZDIAG_
+ #define Z2_MAX_PIN Z_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _E0DIAG_
+ #define Z2_MAX_PIN E0_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _E1DIAG_
+ #define Z2_MAX_PIN E1_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _E2DIAG_
+ #define Z2_MAX_PIN E2_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _E3DIAG_
+ #define Z2_MAX_PIN E3_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _E4DIAG_
+ #define Z2_MAX_PIN E4_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _E5DIAG_
+ #define Z2_MAX_PIN E5_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _E6DIAG_
+ #define Z2_MAX_PIN E6_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _E7DIAG_
+ #define Z2_MAX_PIN E7_DIAG_PIN
+ #endif
+ #endif
+ #ifndef Z2_MIN_ENDSTOP_INVERTING
+ #define Z2_MIN_ENDSTOP_INVERTING false
+ #endif
+ #else
+ #ifndef Z2_MIN_ENDSTOP_INVERTING
+ #if Z2_USE_ENDSTOP == _XMIN_
+ #define Z2_MIN_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
+ #elif Z2_USE_ENDSTOP == _XMAX_
+ #define Z2_MIN_ENDSTOP_INVERTING X_MAX_ENDSTOP_INVERTING
+ #elif Z2_USE_ENDSTOP == _YMIN_
+ #define Z2_MIN_ENDSTOP_INVERTING Y_MIN_ENDSTOP_INVERTING
+ #elif Z2_USE_ENDSTOP == _YMAX_
+ #define Z2_MIN_ENDSTOP_INVERTING Y_MAX_ENDSTOP_INVERTING
+ #elif Z2_USE_ENDSTOP == _ZMIN_
+ #define Z2_MIN_ENDSTOP_INVERTING Z_MIN_ENDSTOP_INVERTING
+ #elif Z2_USE_ENDSTOP == _ZMAX_
+ #define Z2_MIN_ENDSTOP_INVERTING Z_MAX_ENDSTOP_INVERTING
+ #else
+ #define Z2_MIN_ENDSTOP_INVERTING false
+ #endif
+ #endif
+ #ifndef Z2_MIN_PIN
+ #if Z2_USE_ENDSTOP == _XMIN_
+ #define Z2_MIN_PIN X_MIN_PIN
+ #elif Z2_USE_ENDSTOP == _XMAX_
+ #define Z2_MIN_PIN X_MAX_PIN
+ #elif Z2_USE_ENDSTOP == _YMIN_
+ #define Z2_MIN_PIN Y_MIN_PIN
+ #elif Z2_USE_ENDSTOP == _YMAX_
+ #define Z2_MIN_PIN Y_MAX_PIN
+ #elif Z2_USE_ENDSTOP == _ZMIN_
+ #define Z2_MIN_PIN Z_MIN_PIN
+ #elif Z2_USE_ENDSTOP == _ZMAX_
+ #define Z2_MIN_PIN Z_MAX_PIN
+ #elif Z2_USE_ENDSTOP == _XDIAG_
+ #define Z2_MIN_PIN X_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _YDIAG_
+ #define Z2_MIN_PIN Y_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _ZDIAG_
+ #define Z2_MIN_PIN Z_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _E0DIAG_
+ #define Z2_MIN_PIN E0_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _E1DIAG_
+ #define Z2_MIN_PIN E1_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _E2DIAG_
+ #define Z2_MIN_PIN E2_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _E3DIAG_
+ #define Z2_MIN_PIN E3_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _E4DIAG_
+ #define Z2_MIN_PIN E4_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _E5DIAG_
+ #define Z2_MIN_PIN E5_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _E6DIAG_
+ #define Z2_MIN_PIN E6_DIAG_PIN
+ #elif Z2_USE_ENDSTOP == _E7DIAG_
+ #define Z2_MIN_PIN E7_DIAG_PIN
+ #endif
+ #endif
+ #ifndef Z2_MAX_ENDSTOP_INVERTING
+ #define Z2_MAX_ENDSTOP_INVERTING false
+ #endif
+ #endif
+
+ #if NUM_Z_STEPPER_DRIVERS >= 3
+ #if Z_HOME_DIR > 0
+ #ifndef Z3_MAX_ENDSTOP_INVERTING
+ #if Z3_USE_ENDSTOP == _XMIN_
+ #define Z3_MAX_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
+ #elif Z3_USE_ENDSTOP == _XMAX_
+ #define Z3_MAX_ENDSTOP_INVERTING X_MAX_ENDSTOP_INVERTING
+ #elif Z3_USE_ENDSTOP == _YMIN_
+ #define Z3_MAX_ENDSTOP_INVERTING Y_MIN_ENDSTOP_INVERTING
+ #elif Z3_USE_ENDSTOP == _YMAX_
+ #define Z3_MAX_ENDSTOP_INVERTING Y_MAX_ENDSTOP_INVERTING
+ #elif Z3_USE_ENDSTOP == _ZMIN_
+ #define Z3_MAX_ENDSTOP_INVERTING Z_MIN_ENDSTOP_INVERTING
+ #elif Z3_USE_ENDSTOP == _ZMAX_
+ #define Z3_MAX_ENDSTOP_INVERTING Z_MAX_ENDSTOP_INVERTING
+ #else
+ #define Z3_MAX_ENDSTOP_INVERTING false
+ #endif
+ #endif
+ #ifndef Z3_MAX_PIN
+ #if Z3_USE_ENDSTOP == _XMIN_
+ #define Z3_MAX_PIN X_MIN_PIN
+ #elif Z3_USE_ENDSTOP == _XMAX_
+ #define Z3_MAX_PIN X_MAX_PIN
+ #elif Z3_USE_ENDSTOP == _YMIN_
+ #define Z3_MAX_PIN Y_MIN_PIN
+ #elif Z3_USE_ENDSTOP == _YMAX_
+ #define Z3_MAX_PIN Y_MAX_PIN
+ #elif Z3_USE_ENDSTOP == _ZMIN_
+ #define Z3_MAX_PIN Z_MIN_PIN
+ #elif Z3_USE_ENDSTOP == _ZMAX_
+ #define Z3_MAX_PIN Z_MAX_PIN
+ #elif Z3_USE_ENDSTOP == _XDIAG_
+ #define Z3_MAX_PIN X_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _YDIAG_
+ #define Z3_MAX_PIN Y_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _ZDIAG_
+ #define Z3_MAX_PIN Z_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _E0DIAG_
+ #define Z3_MAX_PIN E0_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _E1DIAG_
+ #define Z3_MAX_PIN E1_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _E2DIAG_
+ #define Z3_MAX_PIN E2_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _E3DIAG_
+ #define Z3_MAX_PIN E3_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _E4DIAG_
+ #define Z3_MAX_PIN E4_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _E5DIAG_
+ #define Z3_MAX_PIN E5_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _E6DIAG_
+ #define Z3_MAX_PIN E6_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _E7DIAG_
+ #define Z3_MAX_PIN E7_DIAG_PIN
+ #endif
+ #endif
+ #ifndef Z3_MIN_ENDSTOP_INVERTING
+ #define Z3_MIN_ENDSTOP_INVERTING false
+ #endif
+ #else
+ #ifndef Z3_MIN_ENDSTOP_INVERTING
+ #if Z3_USE_ENDSTOP == _XMIN_
+ #define Z3_MIN_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
+ #elif Z3_USE_ENDSTOP == _XMAX_
+ #define Z3_MIN_ENDSTOP_INVERTING X_MAX_ENDSTOP_INVERTING
+ #elif Z3_USE_ENDSTOP == _YMIN_
+ #define Z3_MIN_ENDSTOP_INVERTING Y_MIN_ENDSTOP_INVERTING
+ #elif Z3_USE_ENDSTOP == _YMAX_
+ #define Z3_MIN_ENDSTOP_INVERTING Y_MAX_ENDSTOP_INVERTING
+ #elif Z3_USE_ENDSTOP == _ZMIN_
+ #define Z3_MIN_ENDSTOP_INVERTING Z_MIN_ENDSTOP_INVERTING
+ #elif Z3_USE_ENDSTOP == _ZMAX_
+ #define Z3_MIN_ENDSTOP_INVERTING Z_MAX_ENDSTOP_INVERTING
+ #else
+ #define Z3_MIN_ENDSTOP_INVERTING false
+ #endif
+ #endif
+ #ifndef Z3_MIN_PIN
+ #if Z3_USE_ENDSTOP == _XMIN_
+ #define Z3_MIN_PIN X_MIN_PIN
+ #elif Z3_USE_ENDSTOP == _XMAX_
+ #define Z3_MIN_PIN X_MAX_PIN
+ #elif Z3_USE_ENDSTOP == _YMIN_
+ #define Z3_MIN_PIN Y_MIN_PIN
+ #elif Z3_USE_ENDSTOP == _YMAX_
+ #define Z3_MIN_PIN Y_MAX_PIN
+ #elif Z3_USE_ENDSTOP == _ZMIN_
+ #define Z3_MIN_PIN Z_MIN_PIN
+ #elif Z3_USE_ENDSTOP == _ZMAX_
+ #define Z3_MIN_PIN Z_MAX_PIN
+ #elif Z3_USE_ENDSTOP == _XDIAG_
+ #define Z3_MIN_PIN X_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _YDIAG_
+ #define Z3_MIN_PIN Y_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _ZDIAG_
+ #define Z3_MIN_PIN Z_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _E0DIAG_
+ #define Z3_MIN_PIN E0_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _E1DIAG_
+ #define Z3_MIN_PIN E1_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _E2DIAG_
+ #define Z3_MIN_PIN E2_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _E3DIAG_
+ #define Z3_MIN_PIN E3_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _E4DIAG_
+ #define Z3_MIN_PIN E4_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _E5DIAG_
+ #define Z3_MIN_PIN E5_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _E6DIAG_
+ #define Z3_MIN_PIN E6_DIAG_PIN
+ #elif Z3_USE_ENDSTOP == _E7DIAG_
+ #define Z3_MIN_PIN E7_DIAG_PIN
+ #endif
+ #endif
+ #ifndef Z3_MAX_ENDSTOP_INVERTING
+ #define Z3_MAX_ENDSTOP_INVERTING false
+ #endif
+ #endif
+ #endif
+
+ #if NUM_Z_STEPPER_DRIVERS >= 4
+ #if Z_HOME_DIR > 0
+ #ifndef Z4_MAX_ENDSTOP_INVERTING
+ #if Z4_USE_ENDSTOP == _XMIN_
+ #define Z4_MAX_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
+ #elif Z4_USE_ENDSTOP == _XMAX_
+ #define Z4_MAX_ENDSTOP_INVERTING X_MAX_ENDSTOP_INVERTING
+ #elif Z4_USE_ENDSTOP == _YMIN_
+ #define Z4_MAX_ENDSTOP_INVERTING Y_MIN_ENDSTOP_INVERTING
+ #elif Z4_USE_ENDSTOP == _YMAX_
+ #define Z4_MAX_ENDSTOP_INVERTING Y_MAX_ENDSTOP_INVERTING
+ #elif Z4_USE_ENDSTOP == _ZMIN_
+ #define Z4_MAX_ENDSTOP_INVERTING Z_MIN_ENDSTOP_INVERTING
+ #elif Z4_USE_ENDSTOP == _ZMAX_
+ #define Z4_MAX_ENDSTOP_INVERTING Z_MAX_ENDSTOP_INVERTING
+ #else
+ #define Z4_MAX_ENDSTOP_INVERTING false
+ #endif
+ #endif
+ #ifndef Z4_MAX_PIN
+ #if Z4_USE_ENDSTOP == _XMIN_
+ #define Z4_MAX_PIN X_MIN_PIN
+ #elif Z4_USE_ENDSTOP == _XMAX_
+ #define Z4_MAX_PIN X_MAX_PIN
+ #elif Z4_USE_ENDSTOP == _YMIN_
+ #define Z4_MAX_PIN Y_MIN_PIN
+ #elif Z4_USE_ENDSTOP == _YMAX_
+ #define Z4_MAX_PIN Y_MAX_PIN
+ #elif Z4_USE_ENDSTOP == _ZMIN_
+ #define Z4_MAX_PIN Z_MIN_PIN
+ #elif Z4_USE_ENDSTOP == _ZMAX_
+ #define Z4_MAX_PIN Z_MAX_PIN
+ #elif Z4_USE_ENDSTOP == _XDIAG_
+ #define Z4_MAX_PIN X_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _YDIAG_
+ #define Z4_MAX_PIN Y_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _ZDIAG_
+ #define Z4_MAX_PIN Z_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _E0DIAG_
+ #define Z4_MAX_PIN E0_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _E1DIAG_
+ #define Z4_MAX_PIN E1_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _E2DIAG_
+ #define Z4_MAX_PIN E2_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _E3DIAG_
+ #define Z4_MAX_PIN E3_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _E4DIAG_
+ #define Z4_MAX_PIN E4_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _E5DIAG_
+ #define Z4_MAX_PIN E5_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _E6DIAG_
+ #define Z4_MAX_PIN E6_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _E7DIAG_
+ #define Z4_MAX_PIN E7_DIAG_PIN
+ #endif
+ #endif
+ #ifndef Z4_MIN_ENDSTOP_INVERTING
+ #define Z4_MIN_ENDSTOP_INVERTING false
+ #endif
+ #else
+ #ifndef Z4_MIN_ENDSTOP_INVERTING
+ #if Z4_USE_ENDSTOP == _XMIN_
+ #define Z4_MIN_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
+ #elif Z4_USE_ENDSTOP == _XMAX_
+ #define Z4_MIN_ENDSTOP_INVERTING X_MAX_ENDSTOP_INVERTING
+ #elif Z4_USE_ENDSTOP == _YMIN_
+ #define Z4_MIN_ENDSTOP_INVERTING Y_MIN_ENDSTOP_INVERTING
+ #elif Z4_USE_ENDSTOP == _YMAX_
+ #define Z4_MIN_ENDSTOP_INVERTING Y_MAX_ENDSTOP_INVERTING
+ #elif Z4_USE_ENDSTOP == _ZMIN_
+ #define Z4_MIN_ENDSTOP_INVERTING Z_MIN_ENDSTOP_INVERTING
+ #elif Z4_USE_ENDSTOP == _ZMAX_
+ #define Z4_MIN_ENDSTOP_INVERTING Z_MAX_ENDSTOP_INVERTING
+ #else
+ #define Z4_MIN_ENDSTOP_INVERTING false
+ #endif
+ #endif
+ #ifndef Z4_MIN_PIN
+ #if Z4_USE_ENDSTOP == _XMIN_
+ #define Z4_MIN_PIN X_MIN_PIN
+ #elif Z4_USE_ENDSTOP == _XMAX_
+ #define Z4_MIN_PIN X_MAX_PIN
+ #elif Z4_USE_ENDSTOP == _YMIN_
+ #define Z4_MIN_PIN Y_MIN_PIN
+ #elif Z4_USE_ENDSTOP == _YMAX_
+ #define Z4_MIN_PIN Y_MAX_PIN
+ #elif Z4_USE_ENDSTOP == _ZMIN_
+ #define Z4_MIN_PIN Z_MIN_PIN
+ #elif Z4_USE_ENDSTOP == _ZMAX_
+ #define Z4_MIN_PIN Z_MAX_PIN
+ #elif Z4_USE_ENDSTOP == _XDIAG_
+ #define Z4_MIN_PIN X_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _YDIAG_
+ #define Z4_MIN_PIN Y_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _ZDIAG_
+ #define Z4_MIN_PIN Z_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _E0DIAG_
+ #define Z4_MIN_PIN E0_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _E1DIAG_
+ #define Z4_MIN_PIN E1_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _E2DIAG_
+ #define Z4_MIN_PIN E2_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _E3DIAG_
+ #define Z4_MIN_PIN E3_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _E4DIAG_
+ #define Z4_MIN_PIN E4_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _E5DIAG_
+ #define Z4_MIN_PIN E5_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _E6DIAG_
+ #define Z4_MIN_PIN E6_DIAG_PIN
+ #elif Z4_USE_ENDSTOP == _E7DIAG_
+ #define Z4_MIN_PIN E7_DIAG_PIN
+ #endif
+ #endif
+ #ifndef Z4_MAX_ENDSTOP_INVERTING
+ #define Z4_MAX_ENDSTOP_INVERTING false
+ #endif
+ #endif
+ #endif
+
+#endif // Z_MULTI_ENDSTOPS
+
+/**
+ * Set ENDSTOPPULLUPS for active endstop switches
+ */
+#if ENABLED(ENDSTOPPULLUPS)
+ #if ENABLED(USE_XMAX_PLUG)
+ #define ENDSTOPPULLUP_XMAX
+ #endif
+ #if ENABLED(USE_YMAX_PLUG)
+ #define ENDSTOPPULLUP_YMAX
+ #endif
+ #if ENABLED(USE_ZMAX_PLUG)
+ #define ENDSTOPPULLUP_ZMAX
+ #endif
+ #if ENABLED(USE_XMIN_PLUG)
+ #define ENDSTOPPULLUP_XMIN
+ #endif
+ #if ENABLED(USE_YMIN_PLUG)
+ #define ENDSTOPPULLUP_YMIN
+ #endif
+ #if ENABLED(USE_ZMIN_PLUG)
+ #define ENDSTOPPULLUP_ZMIN
+ #endif
+#endif
+
+/**
+ * Set ENDSTOPPULLDOWNS for active endstop switches
+ */
+#if ENABLED(ENDSTOPPULLDOWNS)
+ #if ENABLED(USE_XMAX_PLUG)
+ #define ENDSTOPPULLDOWN_XMAX
+ #endif
+ #if ENABLED(USE_YMAX_PLUG)
+ #define ENDSTOPPULLDOWN_YMAX
+ #endif
+ #if ENABLED(USE_ZMAX_PLUG)
+ #define ENDSTOPPULLDOWN_ZMAX
+ #endif
+ #if ENABLED(USE_XMIN_PLUG)
+ #define ENDSTOPPULLDOWN_XMIN
+ #endif
+ #if ENABLED(USE_YMIN_PLUG)
+ #define ENDSTOPPULLDOWN_YMIN
+ #endif
+ #if ENABLED(USE_ZMIN_PLUG)
+ #define ENDSTOPPULLDOWN_ZMIN
+ #endif
+#endif
+
+/**
+ * Shorthand for pin tests, used wherever needed
+ */
+
+// Steppers
+#if PIN_EXISTS(X_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(X))
+ #define HAS_X_ENABLE 1
+#endif
+#if PIN_EXISTS(X_DIR)
+ #define HAS_X_DIR 1
+#endif
+#if PIN_EXISTS(X_STEP)
+ #define HAS_X_STEP 1
+#endif
+#if PIN_EXISTS(X_MS1)
+ #define HAS_X_MS_PINS 1
+#endif
+
+#if PIN_EXISTS(X2_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(X2))
+ #define HAS_X2_ENABLE 1
+#endif
+#if PIN_EXISTS(X2_DIR)
+ #define HAS_X2_DIR 1
+#endif
+#if PIN_EXISTS(X2_STEP)
+ #define HAS_X2_STEP 1
+#endif
+#if PIN_EXISTS(X2_MS1)
+ #define HAS_X2_MS_PINS 1
+#endif
+
+#if PIN_EXISTS(Y_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(Y))
+ #define HAS_Y_ENABLE 1
+#endif
+#if PIN_EXISTS(Y_DIR)
+ #define HAS_Y_DIR 1
+#endif
+#if PIN_EXISTS(Y_STEP)
+ #define HAS_Y_STEP 1
+#endif
+#if PIN_EXISTS(Y_MS1)
+ #define HAS_Y_MS_PINS 1
+#endif
+
+#if PIN_EXISTS(Y2_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(Y2))
+ #define HAS_Y2_ENABLE 1
+#endif
+#if PIN_EXISTS(Y2_DIR)
+ #define HAS_Y2_DIR 1
+#endif
+#if PIN_EXISTS(Y2_STEP)
+ #define HAS_Y2_STEP 1
+#endif
+#if PIN_EXISTS(Y2_MS1)
+ #define HAS_Y2_MS_PINS 1
+#endif
+
+#if PIN_EXISTS(Z_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(Z))
+ #define HAS_Z_ENABLE 1
+#endif
+#if PIN_EXISTS(Z_DIR)
+ #define HAS_Z_DIR 1
+#endif
+#if PIN_EXISTS(Z_STEP)
+ #define HAS_Z_STEP 1
+#endif
+#if PIN_EXISTS(Z_MS1)
+ #define HAS_Z_MS_PINS 1
+#endif
+
+#if PIN_EXISTS(Z2_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(Z2))
+ #define HAS_Z2_ENABLE 1
+#endif
+#if PIN_EXISTS(Z2_DIR)
+ #define HAS_Z2_DIR 1
+#endif
+#if PIN_EXISTS(Z2_STEP)
+ #define HAS_Z2_STEP 1
+#endif
+#if PIN_EXISTS(Z2_MS1)
+ #define HAS_Z2_MS_PINS 1
+#endif
+
+#if PIN_EXISTS(Z3_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(Z3))
+ #define HAS_Z3_ENABLE 1
+#endif
+#if PIN_EXISTS(Z3_DIR)
+ #define HAS_Z3_DIR 1
+#endif
+#if PIN_EXISTS(Z3_STEP)
+ #define HAS_Z3_STEP 1
+#endif
+#if PIN_EXISTS(Z3_MS1)
+ #define HAS_Z3_MS_PINS 1
+#endif
+
+#if PIN_EXISTS(Z4_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(Z4))
+ #define HAS_Z4_ENABLE 1
+#endif
+#if PIN_EXISTS(Z4_DIR)
+ #define HAS_Z4_DIR 1
+#endif
+#if PIN_EXISTS(Z4_STEP)
+ #define HAS_Z4_STEP 1
+#endif
+#if PIN_EXISTS(Z4_MS1)
+ #define HAS_Z4_MS_PINS 1
+#endif
+
+// Extruder steppers and solenoids
+#if PIN_EXISTS(E0_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(E0))
+ #define HAS_E0_ENABLE 1
+#endif
+#if PIN_EXISTS(E0_DIR)
+ #define HAS_E0_DIR 1
+#endif
+#if PIN_EXISTS(E0_STEP)
+ #define HAS_E0_STEP 1
+#endif
+#if PIN_EXISTS(E0_MS1)
+ #define HAS_E0_MS_PINS 1
+#endif
+#if PIN_EXISTS(SOL0)
+ #define HAS_SOLENOID_0 1
+#endif
+
+#if PIN_EXISTS(E1_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(E1))
+ #define HAS_E1_ENABLE 1
+#endif
+#if PIN_EXISTS(E1_DIR)
+ #define HAS_E1_DIR 1
+#endif
+#if PIN_EXISTS(E1_STEP)
+ #define HAS_E1_STEP 1
+#endif
+#if PIN_EXISTS(E1_MS1)
+ #define HAS_E1_MS_PINS 1
+#endif
+#if PIN_EXISTS(SOL1)
+ #define HAS_SOLENOID_1 1
+#endif
+
+#if PIN_EXISTS(E2_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(E2))
+ #define HAS_E2_ENABLE 1
+#endif
+#if PIN_EXISTS(E2_DIR)
+ #define HAS_E2_DIR 1
+#endif
+#if PIN_EXISTS(E2_STEP)
+ #define HAS_E2_STEP 1
+#endif
+#if PIN_EXISTS(E2_MS1)
+ #define HAS_E2_MS_PINS 1
+#endif
+#if PIN_EXISTS(SOL2)
+ #define HAS_SOLENOID_2 1
+#endif
+
+#if PIN_EXISTS(E3_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(E3))
+ #define HAS_E3_ENABLE 1
+#endif
+#if PIN_EXISTS(E3_DIR)
+ #define HAS_E3_DIR 1
+#endif
+#if PIN_EXISTS(E3_STEP)
+ #define HAS_E3_STEP 1
+#endif
+#if PIN_EXISTS(E3_MS1)
+ #define HAS_E3_MS_PINS 1
+#endif
+#if PIN_EXISTS(SOL3)
+ #define HAS_SOLENOID_3 1
+#endif
+
+#if PIN_EXISTS(E4_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(E4))
+ #define HAS_E4_ENABLE 1
+#endif
+#if PIN_EXISTS(E4_DIR)
+ #define HAS_E4_DIR 1
+#endif
+#if PIN_EXISTS(E4_STEP)
+ #define HAS_E4_STEP 1
+#endif
+#if PIN_EXISTS(E4_MS1)
+ #define HAS_E4_MS_PINS 1
+#endif
+#if PIN_EXISTS(SOL4)
+ #define HAS_SOLENOID_4 1
+#endif
+
+#if PIN_EXISTS(E5_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(E5))
+ #define HAS_E5_ENABLE 1
+#endif
+#if PIN_EXISTS(E5_DIR)
+ #define HAS_E5_DIR 1
+#endif
+#if PIN_EXISTS(E5_STEP)
+ #define HAS_E5_STEP 1
+#endif
+#if PIN_EXISTS(E5_MS1)
+ #define HAS_E5_MS_PINS 1
+#endif
+#if PIN_EXISTS(SOL5)
+ #define HAS_SOLENOID_5 1
+#endif
+
+#if PIN_EXISTS(E6_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(E6))
+ #define HAS_E6_ENABLE 1
+#endif
+#if PIN_EXISTS(E6_DIR)
+ #define HAS_E6_DIR 1
+#endif
+#if PIN_EXISTS(E6_STEP)
+ #define HAS_E6_STEP 1
+#endif
+#if PIN_EXISTS(E6_MS1)
+ #define HAS_E6_MS_PINS 1
+#endif
+#if PIN_EXISTS(SOL6)
+ #define HAS_SOLENOID_6 1
+#endif
+
+#if PIN_EXISTS(E7_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(E7))
+ #define HAS_E7_ENABLE 1
+#endif
+#if PIN_EXISTS(E7_DIR)
+ #define HAS_E7_DIR 1
+#endif
+#if PIN_EXISTS(E7_STEP)
+ #define HAS_E7_STEP 1
+#endif
+#if PIN_EXISTS(E7_MS1)
+ #define HAS_E7_MS_PINS 1
+#endif
+#if PIN_EXISTS(SOL7)
+ #define HAS_SOLENOID_7 1
+#endif
+
+//
+// Trinamic Stepper Drivers
+//
+
+#if HAS_TRINAMIC_CONFIG
+ #if ANY(STEALTHCHOP_XY, STEALTHCHOP_Z, STEALTHCHOP_E)
+ #define STEALTHCHOP_ENABLED 1
+ #endif
+ #if EITHER(SENSORLESS_HOMING, SENSORLESS_PROBING)
+ #define USE_SENSORLESS 1
+ #endif
+ // Disable Z axis sensorless homing if a probe is used to home the Z axis
+ #if HOMING_Z_WITH_PROBE
+ #undef Z_STALL_SENSITIVITY
+ #undef Z2_STALL_SENSITIVITY
+ #undef Z3_STALL_SENSITIVITY
+ #undef Z4_STALL_SENSITIVITY
+ #endif
+ #if defined(X_STALL_SENSITIVITY) && AXIS_HAS_STALLGUARD(X)
+ #define X_SENSORLESS 1
+ #endif
+ #if defined(X2_STALL_SENSITIVITY) && AXIS_HAS_STALLGUARD(X2)
+ #define X2_SENSORLESS 1
+ #endif
+ #if defined(Y_STALL_SENSITIVITY) && AXIS_HAS_STALLGUARD(Y)
+ #define Y_SENSORLESS 1
+ #endif
+ #if defined(Y2_STALL_SENSITIVITY) && AXIS_HAS_STALLGUARD(Y2)
+ #define Y2_SENSORLESS 1
+ #endif
+ #if defined(Z_STALL_SENSITIVITY) && AXIS_HAS_STALLGUARD(Z)
+ #define Z_SENSORLESS 1
+ #endif
+ #if defined(Z2_STALL_SENSITIVITY) && AXIS_HAS_STALLGUARD(Z2)
+ #define Z2_SENSORLESS 1
+ #endif
+ #if defined(Z3_STALL_SENSITIVITY) && AXIS_HAS_STALLGUARD(Z3)
+ #define Z3_SENSORLESS 1
+ #endif
+ #if defined(Z4_STALL_SENSITIVITY) && AXIS_HAS_STALLGUARD(Z4)
+ #define Z4_SENSORLESS 1
+ #endif
+ #if ENABLED(SPI_ENDSTOPS)
+ #define X_SPI_SENSORLESS X_SENSORLESS
+ #define Y_SPI_SENSORLESS Y_SENSORLESS
+ #define Z_SPI_SENSORLESS Z_SENSORLESS
+ #endif
+ #ifndef X_INTERPOLATE
+ #define X_INTERPOLATE INTERPOLATE
+ #endif
+ #ifndef X2_INTERPOLATE
+ #define X2_INTERPOLATE INTERPOLATE
+ #endif
+ #ifndef Y_INTERPOLATE
+ #define Y_INTERPOLATE INTERPOLATE
+ #endif
+ #ifndef Y2_INTERPOLATE
+ #define Y2_INTERPOLATE INTERPOLATE
+ #endif
+ #ifndef Z_INTERPOLATE
+ #define Z_INTERPOLATE INTERPOLATE
+ #endif
+ #ifndef Z2_INTERPOLATE
+ #define Z2_INTERPOLATE INTERPOLATE
+ #endif
+ #ifndef Z3_INTERPOLATE
+ #define Z3_INTERPOLATE INTERPOLATE
+ #endif
+ #ifndef Z4_INTERPOLATE
+ #define Z4_INTERPOLATE INTERPOLATE
+ #endif
+ #ifndef E0_INTERPOLATE
+ #define E0_INTERPOLATE INTERPOLATE
+ #endif
+ #ifndef E1_INTERPOLATE
+ #define E1_INTERPOLATE INTERPOLATE
+ #endif
+ #ifndef E2_INTERPOLATE
+ #define E2_INTERPOLATE INTERPOLATE
+ #endif
+ #ifndef E3_INTERPOLATE
+ #define E3_INTERPOLATE INTERPOLATE
+ #endif
+ #ifndef E4_INTERPOLATE
+ #define E4_INTERPOLATE INTERPOLATE
+ #endif
+ #ifndef E5_INTERPOLATE
+ #define E5_INTERPOLATE INTERPOLATE
+ #endif
+ #ifndef E6_INTERPOLATE
+ #define E6_INTERPOLATE INTERPOLATE
+ #endif
+ #ifndef E7_INTERPOLATE
+ #define E7_INTERPOLATE INTERPOLATE
+ #endif
+ #ifndef X_SLAVE_ADDRESS
+ #define X_SLAVE_ADDRESS 0
+ #endif
+ #ifndef Y_SLAVE_ADDRESS
+ #define Y_SLAVE_ADDRESS 0
+ #endif
+ #ifndef Z_SLAVE_ADDRESS
+ #define Z_SLAVE_ADDRESS 0
+ #endif
+ #ifndef X2_SLAVE_ADDRESS
+ #define X2_SLAVE_ADDRESS 0
+ #endif
+ #ifndef Y2_SLAVE_ADDRESS
+ #define Y2_SLAVE_ADDRESS 0
+ #endif
+ #ifndef Z2_SLAVE_ADDRESS
+ #define Z2_SLAVE_ADDRESS 0
+ #endif
+ #ifndef Z3_SLAVE_ADDRESS
+ #define Z3_SLAVE_ADDRESS 0
+ #endif
+ #ifndef Z4_SLAVE_ADDRESS
+ #define Z4_SLAVE_ADDRESS 0
+ #endif
+ #ifndef E0_SLAVE_ADDRESS
+ #define E0_SLAVE_ADDRESS 0
+ #endif
+ #ifndef E1_SLAVE_ADDRESS
+ #define E1_SLAVE_ADDRESS 0
+ #endif
+ #ifndef E2_SLAVE_ADDRESS
+ #define E2_SLAVE_ADDRESS 0
+ #endif
+ #ifndef E3_SLAVE_ADDRESS
+ #define E3_SLAVE_ADDRESS 0
+ #endif
+ #ifndef E4_SLAVE_ADDRESS
+ #define E4_SLAVE_ADDRESS 0
+ #endif
+ #ifndef E5_SLAVE_ADDRESS
+ #define E5_SLAVE_ADDRESS 0
+ #endif
+ #ifndef E6_SLAVE_ADDRESS
+ #define E6_SLAVE_ADDRESS 0
+ #endif
+ #ifndef E7_SLAVE_ADDRESS
+ #define E7_SLAVE_ADDRESS 0
+ #endif
+#endif
+
+#if (HAS_E_DRIVER(TMC2660) \
+ || ( E0_ENABLE_PIN != X_ENABLE_PIN && E1_ENABLE_PIN != X_ENABLE_PIN \
+ && E0_ENABLE_PIN != Y_ENABLE_PIN && E1_ENABLE_PIN != Y_ENABLE_PIN ) )
+ #define HAS_E_STEPPER_ENABLE 1
+#endif
+
+#if ANY_AXIS_HAS(HW_SERIAL)
+ #define HAS_TMC_HW_SERIAL 1
+#endif
+#if ANY_AXIS_HAS(SW_SERIAL)
+ #define HAS_TMC_SW_SERIAL 1
+#endif
+
+//
+// Endstops and bed probe
+//
+
+// Is an endstop plug used for extra Z endstops or the probe?
+#define IS_PROBE_PIN(A,M) (HAS_CUSTOM_PROBE_PIN && Z_MIN_PROBE_PIN == A##_##M##_PIN)
+#define IS_X2_ENDSTOP(A,M) (ENABLED(X_DUAL_ENDSTOPS) && X2_USE_ENDSTOP == _##A##M##_)
+#define IS_Y2_ENDSTOP(A,M) (ENABLED(Y_DUAL_ENDSTOPS) && Y2_USE_ENDSTOP == _##A##M##_)
+#define IS_Z2_ENDSTOP(A,M) (ENABLED(Z_MULTI_ENDSTOPS) && Z2_USE_ENDSTOP == _##A##M##_)
+#define IS_Z3_ENDSTOP(A,M) (ENABLED(Z_MULTI_ENDSTOPS) && NUM_Z_STEPPER_DRIVERS >= 3 && Z3_USE_ENDSTOP == _##A##M##_)
+#define IS_Z4_ENDSTOP(A,M) (ENABLED(Z_MULTI_ENDSTOPS) && NUM_Z_STEPPER_DRIVERS >= 4 && Z4_USE_ENDSTOP == _##A##M##_)
+
+#define _HAS_STOP(A,M) (PIN_EXISTS(A##_##M) && !IS_PROBE_PIN(A,M) && !IS_X2_ENDSTOP(A,M) && !IS_Y2_ENDSTOP(A,M) && !IS_Z2_ENDSTOP(A,M) && !IS_Z3_ENDSTOP(A,M) && !IS_Z4_ENDSTOP(A,M))
+#if _HAS_STOP(X,MIN)
+ #define HAS_X_MIN 1
+#endif
+#if _HAS_STOP(X,MAX)
+ #define HAS_X_MAX 1
+#endif
+#if _HAS_STOP(Y,MIN)
+ #define HAS_Y_MIN 1
+#endif
+#if _HAS_STOP(Y,MAX)
+ #define HAS_Y_MAX 1
+#endif
+#if _HAS_STOP(Z,MIN)
+ #define HAS_Z_MIN 1
+#endif
+#if _HAS_STOP(Z,MAX)
+ #define HAS_Z_MAX 1
+#endif
+#if _HAS_STOP(X,STOP)
+ #define HAS_X_STOP 1
+#endif
+#if _HAS_STOP(Y,STOP)
+ #define HAS_Y_STOP 1
+#endif
+#if _HAS_STOP(Z,STOP)
+ #define HAS_Z_STOP 1
+#endif
+#if PIN_EXISTS(X2_MIN)
+ #define HAS_X2_MIN 1
+#endif
+#if PIN_EXISTS(X2_MAX)
+ #define HAS_X2_MAX 1
+#endif
+#if PIN_EXISTS(Y2_MIN)
+ #define HAS_Y2_MIN 1
+#endif
+#if PIN_EXISTS(Y2_MAX)
+ #define HAS_Y2_MAX 1
+#endif
+#if PIN_EXISTS(Z2_MIN)
+ #define HAS_Z2_MIN 1
+#endif
+#if PIN_EXISTS(Z2_MAX)
+ #define HAS_Z2_MAX 1
+#endif
+#if PIN_EXISTS(Z3_MIN)
+ #define HAS_Z3_MIN 1
+#endif
+#if PIN_EXISTS(Z3_MAX)
+ #define HAS_Z3_MAX 1
+#endif
+#if PIN_EXISTS(Z4_MIN)
+ #define HAS_Z4_MIN 1
+#endif
+#if PIN_EXISTS(Z4_MAX)
+ #define HAS_Z4_MAX 1
+#endif
+#if HAS_CUSTOM_PROBE_PIN && PIN_EXISTS(Z_MIN_PROBE)
+ #define HAS_Z_MIN_PROBE_PIN 1
+#endif
+
+//
+// ADC Temp Sensors (Thermistor or Thermocouple with amplifier ADC interface)
+//
+#define HAS_ADC_TEST(P) (PIN_EXISTS(TEMP_##P) && TEMP_SENSOR_##P != 0 && NONE(HEATER_##P##_USES_MAX6675, HEATER_##P##_DUMMY_THERMISTOR))
+#if HAS_ADC_TEST(0)
+ #define HAS_TEMP_ADC_0 1
+#endif
+#if HAS_ADC_TEST(1)
+ #define HAS_TEMP_ADC_1 1
+#endif
+#if HAS_ADC_TEST(2)
+ #define HAS_TEMP_ADC_2 1
+#endif
+#if HAS_ADC_TEST(3)
+ #define HAS_TEMP_ADC_3 1
+#endif
+#if HAS_ADC_TEST(4)
+ #define HAS_TEMP_ADC_4 1
+#endif
+#if HAS_ADC_TEST(5)
+ #define HAS_TEMP_ADC_5 1
+#endif
+#if HAS_ADC_TEST(6)
+ #define HAS_TEMP_ADC_6 1
+#endif
+#if HAS_ADC_TEST(7)
+ #define HAS_TEMP_ADC_7 1
+#endif
+#if HAS_ADC_TEST(BED)
+ #define HAS_TEMP_ADC_BED 1
+#endif
+#if HAS_ADC_TEST(PROBE)
+ #define HAS_TEMP_ADC_PROBE 1
+#endif
+#if HAS_ADC_TEST(CHAMBER)
+ #define HAS_TEMP_ADC_CHAMBER 1
+#endif
+
+#define HAS_TEMP(N) ANY(HAS_TEMP_ADC_##N, HEATER_##N##_USES_MAX6675, HEATER_##N##_DUMMY_THERMISTOR)
+#if HAS_HOTEND && HAS_TEMP(0)
+ #define HAS_TEMP_HOTEND 1
+#endif
+#if HAS_TEMP(BED)
+ #define HAS_TEMP_BED 1
+#endif
+#if HAS_TEMP(PROBE)
+ #define HAS_TEMP_PROBE 1
+#endif
+#if HAS_TEMP(CHAMBER)
+ #define HAS_TEMP_CHAMBER 1
+#endif
+
+#if ENABLED(JOYSTICK)
+ #if PIN_EXISTS(JOY_X)
+ #define HAS_JOY_ADC_X 1
+ #endif
+ #if PIN_EXISTS(JOY_Y)
+ #define HAS_JOY_ADC_Y 1
+ #endif
+ #if PIN_EXISTS(JOY_Z)
+ #define HAS_JOY_ADC_Z 1
+ #endif
+ #if PIN_EXISTS(JOY_EN)
+ #define HAS_JOY_ADC_EN 1
+ #endif
+#endif
+
+// Heaters
+#if PIN_EXISTS(HEATER_0)
+ #define HAS_HEATER_0 1
+#endif
+#if PIN_EXISTS(HEATER_1)
+ #define HAS_HEATER_1 1
+#endif
+#if PIN_EXISTS(HEATER_2)
+ #define HAS_HEATER_2 1
+#endif
+#if PIN_EXISTS(HEATER_3)
+ #define HAS_HEATER_3 1
+#endif
+#if PIN_EXISTS(HEATER_4)
+ #define HAS_HEATER_4 1
+#endif
+#if PIN_EXISTS(HEATER_5)
+ #define HAS_HEATER_5 1
+#endif
+#if PIN_EXISTS(HEATER_6)
+ #define HAS_HEATER_6 1
+#endif
+#if PIN_EXISTS(HEATER_7)
+ #define HAS_HEATER_7 1
+#endif
+#if PIN_EXISTS(HEATER_BED)
+ #define HAS_HEATER_BED 1
+#endif
+
+// Shorthand for common combinations
+#if HAS_TEMP_BED && HAS_HEATER_BED
+ #define HAS_HEATED_BED 1
+ #ifndef BED_OVERSHOOT
+ #define BED_OVERSHOOT 10
+ #endif
+ #define BED_MAX_TARGET (BED_MAXTEMP - (BED_OVERSHOOT))
+#endif
+#if HAS_HEATED_BED || HAS_TEMP_CHAMBER
+ #define BED_OR_CHAMBER 1
+#endif
+#if HAS_TEMP_HOTEND || BED_OR_CHAMBER || HAS_TEMP_PROBE
+ #define HAS_TEMP_SENSOR 1
+#endif
+#if HAS_TEMP_CHAMBER && PIN_EXISTS(HEATER_CHAMBER)
+ #define HAS_HEATED_CHAMBER 1
+#endif
+
+// PID heating
+#if !HAS_HEATED_BED
+ #undef PIDTEMPBED
+#endif
+#if EITHER(PIDTEMP, PIDTEMPBED)
+ #define HAS_PID_HEATING 1
+#endif
+#if BOTH(PIDTEMP, PIDTEMPBED)
+ #define HAS_PID_FOR_BOTH 1
+#endif
+
+// Thermal protection
+#if BOTH(HAS_HEATED_BED, THERMAL_PROTECTION_BED)
+ #define HAS_THERMALLY_PROTECTED_BED 1
+#endif
+#if ENABLED(THERMAL_PROTECTION_HOTENDS) && WATCH_TEMP_PERIOD > 0
+ #define WATCH_HOTENDS 1
+#endif
+#if HAS_THERMALLY_PROTECTED_BED && WATCH_BED_TEMP_PERIOD > 0
+ #define WATCH_BED 1
+#endif
+#if BOTH(HAS_HEATED_CHAMBER, THERMAL_PROTECTION_CHAMBER) && WATCH_CHAMBER_TEMP_PERIOD > 0
+ #define WATCH_CHAMBER 1
+#endif
+#if (ENABLED(THERMAL_PROTECTION_HOTENDS) || !EXTRUDERS) \
+ && (ENABLED(THERMAL_PROTECTION_BED) || !HAS_HEATED_BED) \
+ && (ENABLED(THERMAL_PROTECTION_CHAMBER) || !HAS_HEATED_CHAMBER)
+ #define THERMALLY_SAFE 1
+#endif
+
+// Auto fans
+#if HAS_HOTEND && PIN_EXISTS(E0_AUTO_FAN)
+ #define HAS_AUTO_FAN_0 1
+#endif
+#if HAS_MULTI_HOTEND && PIN_EXISTS(E1_AUTO_FAN)
+ #define HAS_AUTO_FAN_1 1
+#endif
+#if HOTENDS > 2 && PIN_EXISTS(E2_AUTO_FAN)
+ #define HAS_AUTO_FAN_2 1
+#endif
+#if HOTENDS > 3 && PIN_EXISTS(E3_AUTO_FAN)
+ #define HAS_AUTO_FAN_3 1
+#endif
+#if HOTENDS > 4 && PIN_EXISTS(E4_AUTO_FAN)
+ #define HAS_AUTO_FAN_4 1
+#endif
+#if HOTENDS > 5 && PIN_EXISTS(E5_AUTO_FAN)
+ #define HAS_AUTO_FAN_5 1
+#endif
+#if HOTENDS > 6 && PIN_EXISTS(E6_AUTO_FAN)
+ #define HAS_AUTO_FAN_6 1
+#endif
+#if HOTENDS > 7 && PIN_EXISTS(E7_AUTO_FAN)
+ #define HAS_AUTO_FAN_7 1
+#endif
+#if HAS_TEMP_CHAMBER && PIN_EXISTS(CHAMBER_AUTO_FAN)
+ #define HAS_AUTO_CHAMBER_FAN 1
+#endif
+
+#if ANY(HAS_AUTO_FAN_0, HAS_AUTO_FAN_1, HAS_AUTO_FAN_2, HAS_AUTO_FAN_3, HAS_AUTO_FAN_4, HAS_AUTO_FAN_5, HAS_AUTO_FAN_6, HAS_AUTO_FAN_7, HAS_AUTO_CHAMBER_FAN)
+ #define HAS_AUTO_FAN 1
+#endif
+#define _FANOVERLAP(A,B) (A##_AUTO_FAN_PIN == E##B##_AUTO_FAN_PIN)
+#if HAS_AUTO_FAN && (_FANOVERLAP(CHAMBER,0) || _FANOVERLAP(CHAMBER,1) || _FANOVERLAP(CHAMBER,2) || _FANOVERLAP(CHAMBER,3) || _FANOVERLAP(CHAMBER,4) || _FANOVERLAP(CHAMBER,5) || _FANOVERLAP(CHAMBER,6) || _FANOVERLAP(CHAMBER,7))
+ #define AUTO_CHAMBER_IS_E 1
+#endif
+
+#if !HAS_TEMP_SENSOR
+ #undef AUTO_REPORT_TEMPERATURES
+#endif
+#if EITHER(AUTO_REPORT_TEMPERATURES, AUTO_REPORT_SD_STATUS)
+ #define HAS_AUTO_REPORTING 1
+#endif
+
+#if !HAS_AUTO_CHAMBER_FAN || AUTO_CHAMBER_IS_E
+ #undef AUTO_POWER_CHAMBER_FAN
+#endif
+
+// Print Cooling fans (limit)
+#ifdef NUM_M106_FANS
+ #define MAX_FANS NUM_M106_FANS
+#else
+ #define MAX_FANS 8 // Max supported fans
+#endif
+
+#define _NOT_E_AUTO(N,F) (E##N##_AUTO_FAN_PIN != FAN##F##_PIN)
+#define _HAS_FAN(F) (PIN_EXISTS(FAN##F) \
+ && CONTROLLER_FAN_PIN != FAN##F##_PIN \
+ && _NOT_E_AUTO(0,F) \
+ && _NOT_E_AUTO(1,F) \
+ && _NOT_E_AUTO(2,F) \
+ && _NOT_E_AUTO(3,F) \
+ && _NOT_E_AUTO(4,F) \
+ && _NOT_E_AUTO(5,F) \
+ && _NOT_E_AUTO(6,F) \
+ && _NOT_E_AUTO(7,F) \
+ && F < MAX_FANS)
+#if PIN_EXISTS(FAN)
+ #define HAS_FAN0 1
+#endif
+#if _HAS_FAN(1)
+ #define HAS_FAN1 1
+#endif
+#if _HAS_FAN(2)
+ #define HAS_FAN2 1
+#endif
+#if _HAS_FAN(3)
+ #define HAS_FAN3 1
+#endif
+#if _HAS_FAN(4)
+ #define HAS_FAN4 1
+#endif
+#if _HAS_FAN(5)
+ #define HAS_FAN5 1
+#endif
+#if _HAS_FAN(6)
+ #define HAS_FAN6 1
+#endif
+#if _HAS_FAN(7)
+ #define HAS_FAN7 1
+#endif
+#undef _NOT_E_AUTO
+#undef _HAS_FAN
+#if PIN_EXISTS(CONTROLLER_FAN)
+ #define HAS_CONTROLLER_FAN 1
+#endif
+
+#if BED_OR_CHAMBER || HAS_FAN0
+ #define BED_OR_CHAMBER_OR_FAN 1
+#endif
+
+// Servos
+#if PIN_EXISTS(SERVO0) && NUM_SERVOS > 0
+ #define HAS_SERVO_0 1
+#endif
+#if PIN_EXISTS(SERVO1) && NUM_SERVOS > 1
+ #define HAS_SERVO_1 1
+#endif
+#if PIN_EXISTS(SERVO2) && NUM_SERVOS > 2
+ #define HAS_SERVO_2 1
+#endif
+#if PIN_EXISTS(SERVO3) && NUM_SERVOS > 3
+ #define HAS_SERVO_3 1
+#endif
+#if NUM_SERVOS > 0
+ #define HAS_SERVOS 1
+#endif
+#if HAS_SERVOS && defined(PAUSE_SERVO_OUTPUT) && defined(RESUME_SERVO_OUTPUT)
+ #define HAS_PAUSE_SERVO_OUTPUT 1
+#endif
+
+// Sensors
+#if PIN_EXISTS(FILWIDTH)
+ #define HAS_FILAMENT_WIDTH_SENSOR 1
+#endif
+
+// User Interface
+#if PIN_EXISTS(HOME)
+ #define HAS_HOME 1
+#endif
+#if PIN_EXISTS(KILL)
+ #define HAS_KILL 1
+#endif
+#if PIN_EXISTS(SUICIDE)
+ #define HAS_SUICIDE 1
+#endif
+#if PIN_EXISTS(PHOTOGRAPH)
+ #define HAS_PHOTOGRAPH 1
+#endif
+
+// Digital control
+#if PIN_EXISTS(STEPPER_RESET)
+ #define HAS_STEPPER_RESET 1
+#endif
+#if PIN_EXISTS(DIGIPOTSS)
+ #define HAS_MOTOR_CURRENT_SPI 1
+#endif
+#if ANY_PIN(MOTOR_CURRENT_PWM_X, MOTOR_CURRENT_PWM_Y, MOTOR_CURRENT_PWM_XY, MOTOR_CURRENT_PWM_Z, MOTOR_CURRENT_PWM_E)
+ #define HAS_MOTOR_CURRENT_PWM 1
+#endif
+
+#if ANY(HAS_Z_MS_PINS, HAS_Z2_MS_PINS, HAS_Z3_MS_PINS, HAS_Z4_MS_PINS)
+ #define HAS_SOME_Z_MS_PINS 1
+#endif
+#if ANY(HAS_E0_MS_PINS, HAS_E1_MS_PINS, HAS_E2_MS_PINS, HAS_E3_MS_PINS, HAS_E4_MS_PINS, HAS_E5_MS_PINS, HAS_E6_MS_PINS, HAS_E7_MS_PINS)
+ #define HAS_SOME_E_MS_PINS 1
+#endif
+#if ANY(HAS_X_MS_PINS, HAS_X2_MS_PINS, HAS_Y_MS_PINS, HAS_Y2_MS_PINS, HAS_SOME_Z_MS_PINS, HAS_SOME_E_MS_PINS)
+ #define HAS_MICROSTEPS 1
+#endif
+
+#if HAS_MICROSTEPS
+
+ // MS1 MS2 MS3 Stepper Driver Microstepping mode table
+ #ifndef MICROSTEP1
+ #define MICROSTEP1 LOW,LOW,LOW
+ #endif
+ #if ENABLED(HEROIC_STEPPER_DRIVERS)
+ #ifndef MICROSTEP128
+ #define MICROSTEP128 LOW,HIGH,LOW
+ #endif
+ #else
+ #ifndef MICROSTEP2
+ #define MICROSTEP2 HIGH,LOW,LOW
+ #endif
+ #ifndef MICROSTEP4
+ #define MICROSTEP4 LOW,HIGH,LOW
+ #endif
+ #endif
+ #ifndef MICROSTEP8
+ #define MICROSTEP8 HIGH,HIGH,LOW
+ #endif
+ #ifdef __SAM3X8E__
+ #if MB(ALLIGATOR)
+ #ifndef MICROSTEP16
+ #define MICROSTEP16 LOW,LOW,LOW
+ #endif
+ #ifndef MICROSTEP32
+ #define MICROSTEP32 HIGH,HIGH,LOW
+ #endif
+ #else
+ #ifndef MICROSTEP16
+ #define MICROSTEP16 HIGH,HIGH,LOW
+ #endif
+ #endif
+ #else
+ #ifndef MICROSTEP16
+ #define MICROSTEP16 HIGH,HIGH,LOW
+ #endif
+ #endif
+
+ #ifdef MICROSTEP1
+ #define HAS_MICROSTEP1 1
+ #endif
+ #ifdef MICROSTEP2
+ #define HAS_MICROSTEP2 1
+ #endif
+ #ifdef MICROSTEP4
+ #define HAS_MICROSTEP4 1
+ #endif
+ #ifdef MICROSTEP8
+ #define HAS_MICROSTEP8 1
+ #endif
+ #ifdef MICROSTEP16
+ #define HAS_MICROSTEP16 1
+ #endif
+ #ifdef MICROSTEP32
+ #define HAS_MICROSTEP32 1
+ #endif
+ #ifdef MICROSTEP64
+ #define HAS_MICROSTEP64 1
+ #endif
+ #ifdef MICROSTEP128
+ #define HAS_MICROSTEP128 1
+ #endif
+
+#endif // HAS_MICROSTEPS
+
+/**
+ * Heater signal inversion defaults
+ */
+
+#if HAS_HEATER_0 && !defined(HEATER_0_INVERTING)
+ #define HEATER_0_INVERTING false
+#endif
+#if HAS_HEATER_1 && !defined(HEATER_1_INVERTING)
+ #define HEATER_1_INVERTING false
+#endif
+#if HAS_HEATER_2 && !defined(HEATER_2_INVERTING)
+ #define HEATER_2_INVERTING false
+#endif
+#if HAS_HEATER_3 && !defined(HEATER_3_INVERTING)
+ #define HEATER_3_INVERTING false
+#endif
+#if HAS_HEATER_4 && !defined(HEATER_4_INVERTING)
+ #define HEATER_4_INVERTING false
+#endif
+#if HAS_HEATER_5 && !defined(HEATER_5_INVERTING)
+ #define HEATER_5_INVERTING false
+#endif
+#if HAS_HEATER_6 && !defined(HEATER_6_INVERTING)
+ #define HEATER_6_INVERTING false
+#endif
+#if HAS_HEATER_7 && !defined(HEATER_7_INVERTING)
+ #define HEATER_7_INVERTING false
+#endif
+
+/**
+ * Helper Macros for heaters and extruder fan
+ */
+
+#define WRITE_HEATER_0P(v) WRITE(HEATER_0_PIN, (v) ^ HEATER_0_INVERTING)
+#if EITHER(HAS_MULTI_HOTEND, HEATERS_PARALLEL)
+ #define WRITE_HEATER_1(v) WRITE(HEATER_1_PIN, (v) ^ HEATER_1_INVERTING)
+ #if HOTENDS > 2
+ #define WRITE_HEATER_2(v) WRITE(HEATER_2_PIN, (v) ^ HEATER_2_INVERTING)
+ #if HOTENDS > 3
+ #define WRITE_HEATER_3(v) WRITE(HEATER_3_PIN, (v) ^ HEATER_3_INVERTING)
+ #if HOTENDS > 4
+ #define WRITE_HEATER_4(v) WRITE(HEATER_4_PIN, (v) ^ HEATER_4_INVERTING)
+ #if HOTENDS > 5
+ #define WRITE_HEATER_5(v) WRITE(HEATER_5_PIN, (v) ^ HEATER_5_INVERTING)
+ #if HOTENDS > 6
+ #define WRITE_HEATER_6(v) WRITE(HEATER_6_PIN, (v) ^ HEATER_6_INVERTING)
+ #if HOTENDS > 7
+ #define WRITE_HEATER_7(v) WRITE(HEATER_7_PIN, (v) ^ HEATER_7_INVERTING)
+ #endif // HOTENDS > 7
+ #endif // HOTENDS > 6
+ #endif // HOTENDS > 5
+ #endif // HOTENDS > 4
+ #endif // HOTENDS > 3
+ #endif // HOTENDS > 2
+#endif // HAS_MULTI_HOTEND || HEATERS_PARALLEL
+#if ENABLED(HEATERS_PARALLEL)
+ #define WRITE_HEATER_0(v) { WRITE_HEATER_0P(v); WRITE_HEATER_1(v); }
+#else
+ #define WRITE_HEATER_0(v) WRITE_HEATER_0P(v)
+#endif
+
+#ifndef MIN_POWER
+ #define MIN_POWER 0
+#endif
+
+/**
+ * Heated bed requires settings
+ */
+#if HAS_HEATED_BED
+ #ifndef MIN_BED_POWER
+ #define MIN_BED_POWER 0
+ #endif
+ #ifndef MAX_BED_POWER
+ #define MAX_BED_POWER 255
+ #endif
+ #ifndef HEATER_BED_INVERTING
+ #define HEATER_BED_INVERTING false
+ #endif
+ #define WRITE_HEATER_BED(v) WRITE(HEATER_BED_PIN, (v) ^ HEATER_BED_INVERTING)
+#endif
+
+/**
+ * Heated chamber requires settings
+ */
+#if HAS_HEATED_CHAMBER
+ #ifndef MAX_CHAMBER_POWER
+ #define MAX_CHAMBER_POWER 255
+ #endif
+ #ifndef HEATER_CHAMBER_INVERTING
+ #define HEATER_CHAMBER_INVERTING false
+ #endif
+ #define WRITE_HEATER_CHAMBER(v) WRITE(HEATER_CHAMBER_PIN, (v) ^ HEATER_CHAMBER_INVERTING)
+#endif
+
+#if HAS_HOTEND || HAS_HEATED_BED || HAS_HEATED_CHAMBER
+ #define HAS_TEMPERATURE 1
+#endif
+
+#if HAS_TEMPERATURE && EITHER(HAS_LCD_MENU, DWIN_CREALITY_LCD)
+ #ifdef PREHEAT_5_LABEL
+ #define PREHEAT_COUNT 5
+ #elif defined(PREHEAT_4_LABEL)
+ #define PREHEAT_COUNT 4
+ #elif defined(PREHEAT_3_LABEL)
+ #define PREHEAT_COUNT 3
+ #elif defined(PREHEAT_2_LABEL)
+ #define PREHEAT_COUNT 2
+ #elif defined(PREHEAT_1_LABEL)
+ #define PREHEAT_COUNT 1
+ #endif
+#endif
+
+/**
+ * Up to 3 PWM fans
+ */
+#ifndef FAN_INVERTING
+ #define FAN_INVERTING false
+#endif
+
+#if HAS_FAN7
+ #define FAN_COUNT 8
+#elif HAS_FAN6
+ #define FAN_COUNT 7
+#elif HAS_FAN5
+ #define FAN_COUNT 6
+#elif HAS_FAN4
+ #define FAN_COUNT 5
+#elif HAS_FAN3
+ #define FAN_COUNT 4
+#elif HAS_FAN2
+ #define FAN_COUNT 3
+#elif HAS_FAN1
+ #define FAN_COUNT 2
+#elif HAS_FAN0
+ #define FAN_COUNT 1
+#else
+ #define FAN_COUNT 0
+#endif
+
+#if FAN_COUNT > 0
+ #define HAS_FAN 1
+#endif
+
+/**
+ * Part Cooling fan multipliexer
+ */
+#if PIN_EXISTS(FANMUX0)
+ #define HAS_FANMUX 1
+#endif
+
+/**
+ * MIN/MAX fan PWM scaling
+ */
+#ifndef FAN_OFF_PWM
+ #define FAN_OFF_PWM 0
+#endif
+#ifndef FAN_MIN_PWM
+ #if FAN_OFF_PWM > 0
+ #define FAN_MIN_PWM (FAN_OFF_PWM + 1)
+ #else
+ #define FAN_MIN_PWM 0
+ #endif
+#endif
+#ifndef FAN_MAX_PWM
+ #define FAN_MAX_PWM 255
+#endif
+#if FAN_MIN_PWM < 0 || FAN_MIN_PWM > 255
+ #error "FAN_MIN_PWM must be a value from 0 to 255."
+#elif FAN_MAX_PWM < 0 || FAN_MAX_PWM > 255
+ #error "FAN_MAX_PWM must be a value from 0 to 255."
+#elif FAN_MIN_PWM > FAN_MAX_PWM
+ #error "FAN_MIN_PWM must be less than or equal to FAN_MAX_PWM."
+#elif FAN_OFF_PWM > FAN_MIN_PWM
+ #error "FAN_OFF_PWM must be less than or equal to FAN_MIN_PWM."
+#endif
+
+/**
+ * FAST PWM FAN Settings
+ */
+#if ENABLED(FAST_PWM_FAN) && !defined(FAST_PWM_FAN_FREQUENCY)
+ #define FAST_PWM_FAN_FREQUENCY ((F_CPU) / (2 * 255 * 1)) // Fan frequency default
+#endif
+
+/**
+ * MIN/MAX case light PWM scaling
+ */
+#if ENABLED(CASE_LIGHT_ENABLE)
+ #ifndef CASE_LIGHT_MAX_PWM
+ #define CASE_LIGHT_MAX_PWM 255
+ #elif !WITHIN(CASE_LIGHT_MAX_PWM, 1, 255)
+ #error "CASE_LIGHT_MAX_PWM must be a value from 1 to 255."
+ #endif
+#endif
+
+/**
+ * Bed Probe dependencies
+ */
+#if HAS_BED_PROBE
+ #if BOTH(ENDSTOPPULLUPS, HAS_Z_MIN_PROBE_PIN)
+ #define ENDSTOPPULLUP_ZMIN_PROBE
+ #endif
+ #ifndef Z_PROBE_OFFSET_RANGE_MIN
+ #define Z_PROBE_OFFSET_RANGE_MIN -20
+ #endif
+ #ifndef Z_PROBE_OFFSET_RANGE_MAX
+ #define Z_PROBE_OFFSET_RANGE_MAX 20
+ #endif
+ #ifndef XY_PROBE_SPEED
+ #define XY_PROBE_SPEED ((homing_feedrate_mm_m.x + homing_feedrate_mm_m.y) / 2)
+ #endif
+ #ifndef NOZZLE_TO_PROBE_OFFSET
+ #define NOZZLE_TO_PROBE_OFFSET { 0, 0, 0 }
+ #endif
+#else
+ #undef NOZZLE_TO_PROBE_OFFSET
+#endif
+
+/**
+ * XYZ Bed Skew Correction
+ */
+#if ENABLED(SKEW_CORRECTION)
+ #define SKEW_FACTOR_MIN -1
+ #define SKEW_FACTOR_MAX 1
+
+ #define _GET_SIDE(a,b,c) (SQRT(2*sq(a)+2*sq(b)-4*sq(c))*0.5)
+ #define _SKEW_SIDE(a,b,c) tan(M_PI*0.5-acos((sq(a)-sq(b)-sq(c))/(2*c*b)))
+ #define _SKEW_FACTOR(a,b,c) _SKEW_SIDE(float(a),_GET_SIDE(float(a),float(b),float(c)),float(c))
+
+ #ifndef XY_SKEW_FACTOR
+ #if defined(XY_DIAG_AC) && defined(XY_DIAG_BD) && defined(XY_SIDE_AD)
+ #define XY_SKEW_FACTOR _SKEW_FACTOR(XY_DIAG_AC, XY_DIAG_BD, XY_SIDE_AD)
+ #else
+ #define XY_SKEW_FACTOR 0.0
+ #endif
+ #endif
+ #ifndef XZ_SKEW_FACTOR
+ #if defined(XY_SIDE_AD) && !defined(XZ_SIDE_AD)
+ #define XZ_SIDE_AD XY_SIDE_AD
+ #endif
+ #if defined(XZ_DIAG_AC) && defined(XZ_DIAG_BD) && defined(XZ_SIDE_AD)
+ #define XZ_SKEW_FACTOR _SKEW_FACTOR(XZ_DIAG_AC, XZ_DIAG_BD, XZ_SIDE_AD)
+ #else
+ #define XZ_SKEW_FACTOR 0.0
+ #endif
+ #endif
+ #ifndef YZ_SKEW_FACTOR
+ #if defined(YZ_DIAG_AC) && defined(YZ_DIAG_BD) && defined(YZ_SIDE_AD)
+ #define YZ_SKEW_FACTOR _SKEW_FACTOR(YZ_DIAG_AC, YZ_DIAG_BD, YZ_SIDE_AD)
+ #else
+ #define YZ_SKEW_FACTOR 0.0
+ #endif
+ #endif
+#endif // SKEW_CORRECTION
+
+/**
+ * Heater, Fan, and Probe interactions
+ */
+#if FAN_COUNT == 0
+ #undef PROBING_FANS_OFF
+ #undef ADAPTIVE_FAN_SLOWING
+ #undef NO_FAN_SLOWING_IN_PID_TUNING
+#endif
+
+#if HAS_BED_PROBE && (EITHER(PROBING_HEATERS_OFF, PROBING_FANS_OFF) || DELAY_BEFORE_PROBING > 0)
+ #define HAS_QUIET_PROBING 1
+#endif
+#if EITHER(ADVANCED_PAUSE_FEATURE, PROBING_HEATERS_OFF)
+ #define HEATER_IDLE_HANDLER 1
+#endif
+
+#if ENABLED(ADVANCED_PAUSE_FEATURE) && !defined(FILAMENT_CHANGE_SLOW_LOAD_LENGTH)
+ #define FILAMENT_CHANGE_SLOW_LOAD_LENGTH 0
+#endif
+
+#if HAS_MULTI_EXTRUDER && !defined(TOOLCHANGE_FS_EXTRA_PRIME)
+ #define TOOLCHANGE_FS_EXTRA_PRIME 0
+#endif
+
+/**
+ * Only constrain Z on DELTA / SCARA machines
+ */
+#if IS_KINEMATIC
+ #undef MIN_SOFTWARE_ENDSTOP_X
+ #undef MIN_SOFTWARE_ENDSTOP_Y
+ #undef MAX_SOFTWARE_ENDSTOP_X
+ #undef MAX_SOFTWARE_ENDSTOP_Y
+#endif
+
+/**
+ * Bed Probing bounds
+ */
+
+#ifndef PROBING_MARGIN
+ #define PROBING_MARGIN 0
+#endif
+
+#if IS_KINEMATIC
+ #undef PROBING_MARGIN_LEFT
+ #undef PROBING_MARGIN_RIGHT
+ #undef PROBING_MARGIN_FRONT
+ #undef PROBING_MARGIN_BACK
+ #define PROBING_MARGIN_LEFT 0
+ #define PROBING_MARGIN_RIGHT 0
+ #define PROBING_MARGIN_FRONT 0
+ #define PROBING_MARGIN_BACK 0
+#else
+ #ifndef PROBING_MARGIN_LEFT
+ #define PROBING_MARGIN_LEFT PROBING_MARGIN
+ #endif
+ #ifndef PROBING_MARGIN_RIGHT
+ #define PROBING_MARGIN_RIGHT PROBING_MARGIN
+ #endif
+ #ifndef PROBING_MARGIN_FRONT
+ #define PROBING_MARGIN_FRONT PROBING_MARGIN
+ #endif
+ #ifndef PROBING_MARGIN_BACK
+ #define PROBING_MARGIN_BACK PROBING_MARGIN
+ #endif
+#endif
+
+#if ENABLED(DELTA)
+ /**
+ * Delta radius/rod trimmers/angle trimmers
+ */
+ #ifndef DELTA_ENDSTOP_ADJ
+ #define DELTA_ENDSTOP_ADJ { 0, 0, 0 }
+ #endif
+ #ifndef DELTA_TOWER_ANGLE_TRIM
+ #define DELTA_TOWER_ANGLE_TRIM { 0, 0, 0 }
+ #endif
+ #ifndef DELTA_RADIUS_TRIM_TOWER
+ #define DELTA_RADIUS_TRIM_TOWER { 0, 0, 0 }
+ #endif
+ #ifndef DELTA_DIAGONAL_ROD_TRIM_TOWER
+ #define DELTA_DIAGONAL_ROD_TRIM_TOWER { 0, 0, 0 }
+ #endif
+#endif
+
+#ifndef DEFAULT_LEVELING_FADE_HEIGHT
+ #define DEFAULT_LEVELING_FADE_HEIGHT 0.0
+#endif
+
+#if ENABLED(SEGMENT_LEVELED_MOVES) && !defined(LEVELED_SEGMENT_LENGTH)
+ #define LEVELED_SEGMENT_LENGTH 5
+#endif
+
+/**
+ * Default mesh area is an area with an inset margin on the print area.
+ */
+#if EITHER(MESH_BED_LEVELING, AUTO_BED_LEVELING_UBL)
+ #if IS_KINEMATIC
+ // Probing points may be verified at compile time within the radius
+ // using static_assert(HYPOT2(X2-X1,Y2-Y1)<=sq(DELTA_PRINTABLE_RADIUS),"bad probe point!")
+ // so that may be added to SanityCheck.h in the future.
+ #define _MESH_MIN_X (X_MIN_BED + MESH_INSET)
+ #define _MESH_MIN_Y (Y_MIN_BED + MESH_INSET)
+ #define _MESH_MAX_X (X_MAX_BED - (MESH_INSET))
+ #define _MESH_MAX_Y (Y_MAX_BED - (MESH_INSET))
+ #else
+ // Boundaries for Cartesian probing based on set limits
+ #define _MESH_MIN_X (_MAX(X_MIN_BED + MESH_INSET, X_MIN_POS)) // UBL is careful not to probe off the bed. It does not
+ #define _MESH_MIN_Y (_MAX(Y_MIN_BED + MESH_INSET, Y_MIN_POS)) // need NOZZLE_TO_PROBE_OFFSET in the mesh dimensions
+ #define _MESH_MAX_X (_MIN(X_MAX_BED - (MESH_INSET), X_MAX_POS))
+ #define _MESH_MAX_Y (_MIN(Y_MAX_BED - (MESH_INSET), Y_MAX_POS))
+ #endif
+
+ // These may be overridden in Configuration.h if a smaller area is desired
+ #ifndef MESH_MIN_X
+ #define MESH_MIN_X _MESH_MIN_X
+ #endif
+ #ifndef MESH_MIN_Y
+ #define MESH_MIN_Y _MESH_MIN_Y
+ #endif
+ #ifndef MESH_MAX_X
+ #define MESH_MAX_X _MESH_MAX_X
+ #endif
+ #ifndef MESH_MAX_Y
+ #define MESH_MAX_Y _MESH_MAX_Y
+ #endif
+#else
+ #undef MESH_MIN_X
+ #undef MESH_MIN_Y
+ #undef MESH_MAX_X
+ #undef MESH_MAX_Y
+#endif
+
+#define _POINT_COUNT (defined(PROBE_PT_1_X) + defined(PROBE_PT_2_X) + defined(PROBE_PT_3_X) + defined(PROBE_PT_1_Y) + defined(PROBE_PT_2_Y) + defined(PROBE_PT_3_Y))
+#if _POINT_COUNT == 6
+ #define HAS_FIXED_3POINT 1
+#elif _POINT_COUNT > 0
+ #error "For 3-Point Leveling all XY points must be defined (or none for the defaults)."
+#endif
+#undef _POINT_COUNT
+
+/**
+ * Buzzer/Speaker
+ */
+#if PIN_EXISTS(BEEPER)
+ #define USE_BEEPER 1
+#endif
+#if USE_BEEPER || ANY(LCD_USE_I2C_BUZZER, PCA9632_BUZZER)
+ #define HAS_BUZZER 1
+#endif
+
+#if ENABLED(LCD_USE_I2C_BUZZER)
+ #ifndef LCD_FEEDBACK_FREQUENCY_HZ
+ #define LCD_FEEDBACK_FREQUENCY_HZ 1000
+ #endif
+ #ifndef LCD_FEEDBACK_FREQUENCY_DURATION_MS
+ #define LCD_FEEDBACK_FREQUENCY_DURATION_MS 100
+ #endif
+#elif HAS_BUZZER
+ #ifndef LCD_FEEDBACK_FREQUENCY_HZ
+ #define LCD_FEEDBACK_FREQUENCY_HZ 5000
+ #endif
+ #ifndef LCD_FEEDBACK_FREQUENCY_DURATION_MS
+ #define LCD_FEEDBACK_FREQUENCY_DURATION_MS 2
+ #endif
+#endif
+
+#if HAS_BUZZER
+ #if LCD_FEEDBACK_FREQUENCY_DURATION_MS && LCD_FEEDBACK_FREQUENCY_HZ
+ #define HAS_CHIRP 1
+ #endif
+#else
+ #undef SOUND_MENU_ITEM // No buzzer menu item without a buzzer
+#endif
+
+/**
+ * Make sure DOGLCD_SCK and DOGLCD_MOSI are defined.
+ */
+#if HAS_MARLINUI_U8GLIB
+ #ifndef DOGLCD_SCK
+ #define DOGLCD_SCK SD_SCK_PIN
+ #endif
+ #ifndef DOGLCD_MOSI
+ #define DOGLCD_MOSI SD_MOSI_PIN
+ #endif
+#endif
+
+/**
+ * Z_HOMING_HEIGHT / Z_CLEARANCE_BETWEEN_PROBES
+ */
+#ifndef Z_HOMING_HEIGHT
+ #ifdef Z_CLEARANCE_BETWEEN_PROBES
+ #define Z_HOMING_HEIGHT Z_CLEARANCE_BETWEEN_PROBES
+ #else
+ #define Z_HOMING_HEIGHT 0
+ #endif
+#endif
+
+#if PROBE_SELECTED
+ #ifndef Z_CLEARANCE_BETWEEN_PROBES
+ #define Z_CLEARANCE_BETWEEN_PROBES Z_HOMING_HEIGHT
+ #endif
+ #if Z_CLEARANCE_BETWEEN_PROBES > Z_HOMING_HEIGHT
+ #define MANUAL_PROBE_HEIGHT Z_CLEARANCE_BETWEEN_PROBES
+ #else
+ #define MANUAL_PROBE_HEIGHT Z_HOMING_HEIGHT
+ #endif
+ #ifndef Z_CLEARANCE_MULTI_PROBE
+ #define Z_CLEARANCE_MULTI_PROBE Z_CLEARANCE_BETWEEN_PROBES
+ #endif
+ #if ENABLED(BLTOUCH) && !defined(BLTOUCH_DELAY)
+ #define BLTOUCH_DELAY 500
+ #endif
+#endif
+
+#if !defined(MANUAL_PROBE_START_Z) && defined(Z_CLEARANCE_BETWEEN_PROBES)
+ #define MANUAL_PROBE_START_Z Z_CLEARANCE_BETWEEN_PROBES
+#endif
+
+#ifndef __SAM3X8E__ //todo: hal: broken hal encapsulation
+ #undef UI_VOLTAGE_LEVEL
+ #undef RADDS_DISPLAY
+ #undef MOTOR_CURRENT
+#endif
+
+// Updated G92 behavior shifts the workspace
+#if DISABLED(NO_WORKSPACE_OFFSETS)
+ #define HAS_POSITION_SHIFT 1
+ #if IS_CARTESIAN
+ #define HAS_HOME_OFFSET 1 // The home offset also shifts the coordinate space
+ #define HAS_WORKSPACE_OFFSET 1 // Cumulative offset to workspace to save some calculation
+ #define HAS_M206_COMMAND 1 // M206 sets the home offset for Cartesian machines
+ #elif IS_SCARA
+ #define HAS_SCARA_OFFSET 1 // The SCARA home offset applies only on G28
+ #endif
+#endif
+
+// LCD timeout to status screen default is 15s
+#ifndef LCD_TIMEOUT_TO_STATUS
+ #define LCD_TIMEOUT_TO_STATUS 15000
+#endif
+
+// Add commands that need sub-codes to this list
+#if ANY(G38_PROBE_TARGET, CNC_COORDINATE_SYSTEMS, POWER_LOSS_RECOVERY)
+ #define USE_GCODE_SUBCODES
+#endif
+
+// Parking Extruder
+#if ENABLED(PARKING_EXTRUDER)
+ #ifndef PARKING_EXTRUDER_GRAB_DISTANCE
+ #define PARKING_EXTRUDER_GRAB_DISTANCE 0
+ #endif
+ #ifndef PARKING_EXTRUDER_SOLENOIDS_PINS_ACTIVE
+ #define PARKING_EXTRUDER_SOLENOIDS_PINS_ACTIVE HIGH
+ #endif
+#endif
+
+// Number of VFAT entries used. Each entry has 13 UTF-16 characters
+#if EITHER(SCROLL_LONG_FILENAMES, DWIN_CREALITY_LCD)
+ #define MAX_VFAT_ENTRIES (5)
+#else
+ #define MAX_VFAT_ENTRIES (2)
+#endif
+
+// Nozzle park for Delta
+#if BOTH(NOZZLE_PARK_FEATURE, DELTA)
+ #undef NOZZLE_PARK_Z_FEEDRATE
+ #define NOZZLE_PARK_Z_FEEDRATE NOZZLE_PARK_XY_FEEDRATE
+#endif
+
+// Force SDCARD_SORT_ALPHA to be enabled for Graphical LCD on LPC1768
+// on boards where SD card and LCD display share the same SPI bus
+// because of a bug in the shared SPI implementation. (See #8122)
+#if defined(TARGET_LPC1768) && IS_RRD_FG_SC && (SD_SCK_PIN == LCD_PINS_D4)
+ #define SDCARD_SORT_ALPHA // Keep one directory level in RAM. Changing directory levels
+ // may still glitch the screen, but LCD updates clean it up.
+ #undef SDSORT_LIMIT
+ #undef SDSORT_USES_RAM
+ #undef SDSORT_USES_STACK
+ #undef SDSORT_CACHE_NAMES
+ #define SDSORT_LIMIT 64
+ #define SDSORT_USES_RAM true
+ #define SDSORT_USES_STACK false
+ #define SDSORT_CACHE_NAMES true
+ #ifndef FOLDER_SORTING
+ #define FOLDER_SORTING -1
+ #endif
+ #ifndef SDSORT_GCODE
+ #define SDSORT_GCODE false
+ #endif
+ #ifndef SDSORT_DYNAMIC_RAM
+ #define SDSORT_DYNAMIC_RAM false
+ #endif
+ #ifndef SDSORT_CACHE_VFATS
+ #define SDSORT_CACHE_VFATS 2
+ #endif
+#endif
+
+// Fallback SPI Speed for SD
+#if ENABLED(SDSUPPORT) && !defined(SD_SPI_SPEED)
+ #define SD_SPI_SPEED SPI_FULL_SPEED
+#endif
+
+// Defined here to catch the above defines
+#if ENABLED(SDCARD_SORT_ALPHA) && (FOLDER_SORTING || ENABLED(SDSORT_GCODE))
+ #define HAS_FOLDER_SORTING 1
+#endif
+
+#if HAS_WIRED_LCD
+ // Get LCD character width/height, which may be overridden by pins, configs, etc.
+ #ifndef LCD_WIDTH
+ #if HAS_MARLINUI_U8GLIB
+ #define LCD_WIDTH 21
+ #else
+ #define LCD_WIDTH TERN(IS_ULTIPANEL, 20, 16)
+ #endif
+ #endif
+ #ifndef LCD_HEIGHT
+ #if HAS_MARLINUI_U8GLIB
+ #define LCD_HEIGHT 5
+ #else
+ #define LCD_HEIGHT TERN(IS_ULTIPANEL, 4, 2)
+ #endif
+ #endif
+#endif
+
+#if BUTTONS_EXIST(EN1, EN2, ENC)
+ #define HAS_ROTARY_ENCODER 1
+#endif
+
+#if !NUM_SERIAL
+ #undef BAUD_RATE_GCODE
+#elif NUM_SERIAL > 1
+ #define HAS_MULTI_SERIAL 1
+#endif
diff --git a/Marlin/src/inc/MarlinConfig.h b/Marlin/src/inc/MarlinConfig.h
new file mode 100644
index 0000000..8fdb4b9
--- /dev/null
+++ b/Marlin/src/inc/MarlinConfig.h
@@ -0,0 +1,59 @@
+/**
+ * Marlin 3D Printer Firmware
+ * Copyright (c) 2020 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
+ *
+ * Based on Sprinter and grbl.
+ * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ *
+ */
+#pragma once
+
+//
+// Prefix header for all Marlin sources
+//
+
+#include "MarlinConfigPre.h"
+
+#ifndef __MARLIN_DEPS__
+ #include "../HAL/HAL.h"
+#endif
+
+#include "../pins/pins.h"
+
+#ifndef __MARLIN_DEPS__
+ #include HAL_PATH(../HAL, timers.h)
+ #include HAL_PATH(../HAL, spi_pins.h)
+#endif
+
+#include "Conditionals_post.h"
+
+#ifndef __MARLIN_DEPS__
+
+ #include HAL_PATH(../HAL, inc/Conditionals_post.h)
+
+ #include "../core/types.h" // Ahead of sanity-checks
+
+ #include "SanityCheck.h"
+ #include HAL_PATH(../HAL, inc/SanityCheck.h)
+
+ // Include all core headers
+ #include "../core/language.h"
+ #include "../core/utility.h"
+ #include "../core/serial.h"
+
+#endif
+
+#include "../core/multi_language.h"
diff --git a/Marlin/src/inc/MarlinConfigPre.h b/Marlin/src/inc/MarlinConfigPre.h
new file mode 100644
index 0000000..dfa0adb
--- /dev/null
+++ b/Marlin/src/inc/MarlinConfigPre.h
@@ -0,0 +1,62 @@
+/**
+ * Marlin 3D Printer Firmware
+ * Copyright (c) 2020 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
+ *
+ * Based on Sprinter and grbl.
+ * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ *
+ */
+#pragma once
+
+#ifndef __MARLIN_FIRMWARE__
+#define __MARLIN_FIRMWARE__
+#endif
+
+//
+// Prefix header to acquire configurations
+//
+#include <stdint.h>
+
+#ifndef __MARLIN_DEPS__
+ #include "../HAL/platforms.h"
+#endif
+
+#include "../core/boards.h"
+#include "../core/macros.h"
+#include "../../Configuration.h"
+
+#ifdef CUSTOM_VERSION_FILE
+ #if __has_include(STRINGIFY(../../CUSTOM_VERSION_FILE))
+ #include STRINGIFY(../../CUSTOM_VERSION_FILE)
+ #endif
+#endif
+
+#include "Version.h"
+
+#include "Conditionals_LCD.h"
+
+#ifndef __MARLIN_DEPS__
+ #include HAL_PATH(../HAL, inc/Conditionals_LCD.h)
+#endif
+
+#include "../core/drivers.h"
+#include "../../Configuration_adv.h"
+
+#include "Conditionals_adv.h"
+
+#ifndef __MARLIN_DEPS__
+ #include HAL_PATH(../HAL, inc/Conditionals_adv.h)
+#endif
diff --git a/Marlin/src/inc/SanityCheck.h b/Marlin/src/inc/SanityCheck.h
new file mode 100644
index 0000000..b3beedf
--- /dev/null
+++ b/Marlin/src/inc/SanityCheck.h
@@ -0,0 +1,3367 @@
+/**
+ * Marlin 3D Printer Firmware
+ * Copyright (c) 2020 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
+ *
+ * Based on Sprinter and grbl.
+ * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ *
+ */
+#pragma once
+
+/**
+ * SanityCheck.h
+ *
+ * Test configuration values for errors at compile-time.
+ */
+
+/**
+ * Require gcc 4.7 or newer (first included with Arduino 1.6.8) for C++11 features.
+ */
+#if __cplusplus < 201103L
+ #error "Marlin requires C++11 support (gcc >= 4.7, Arduino IDE >= 1.6.8). Please upgrade your toolchain."
+#endif
+
+// Make sure macros aren't borked
+#define TEST1
+#define TEST2 1
+#define TEST3 0
+#define TEST4 true
+#if ENABLED(TEST0) || !ENABLED(TEST2) || ENABLED(TEST3)
+ #error "ENABLED is borked!"
+#endif
+#if BOTH(TEST0, TEST1)
+ #error "BOTH is borked!"
+#endif
+#if DISABLED(TEST1) || !DISABLED(TEST3) || DISABLED(TEST4) || DISABLED(TEST0, TEST1, TEST2, TEST4) || !DISABLED(TEST0, TEST3)
+ #error "DISABLED is borked!"
+#endif
+#if !ANY(TEST1, TEST2, TEST3, TEST4) || ANY(TEST0, TEST3)
+ #error "ANY is borked!"
+#endif
+#if NONE(TEST0, TEST1, TEST2, TEST4) || !NONE(TEST0, TEST3)
+ #error "NONE is borked!"
+#endif
+#undef TEST1
+#undef TEST2
+#undef TEST3
+#undef TEST4
+
+/**
+ * We try our best to include sanity checks for all changed configuration
+ * directives because users have a tendency to use outdated config files with
+ * the bleeding-edge source code, but sometimes this is not enough. This check
+ * forces a minimum config file revision. Otherwise Marlin will not build.
+ */
+#define HEXIFY(H) _CAT(0x,H)
+#if !defined(CONFIGURATION_H_VERSION) || HEXIFY(CONFIGURATION_H_VERSION) < HEXIFY(REQUIRED_CONFIGURATION_H_VERSION)
+ #error "You are using an old Configuration.h file, update it before building Marlin."
+#endif
+
+#if !defined(CONFIGURATION_ADV_H_VERSION) || HEXIFY(CONFIGURATION_ADV_H_VERSION) < HEXIFY(REQUIRED_CONFIGURATION_ADV_H_VERSION)
+ #error "You are using an old Configuration_adv.h file, update it before building Marlin."
+#endif
+#undef HEXIFY
+
+/**
+ * Warnings for old configurations
+ */
+#ifndef MOTHERBOARD
+ #error "MOTHERBOARD is required."
+#elif !defined(X_BED_SIZE) || !defined(Y_BED_SIZE)
+ #error "X_BED_SIZE and Y_BED_SIZE are now required!"
+#elif WATCH_TEMP_PERIOD > 500
+ #error "WATCH_TEMP_PERIOD now uses seconds instead of milliseconds."
+#elif DISABLED(THERMAL_PROTECTION_HOTENDS) && (defined(WATCH_TEMP_PERIOD) || defined(THERMAL_PROTECTION_PERIOD))
+ #error "Thermal Runaway Protection for hotends is now enabled with THERMAL_PROTECTION_HOTENDS."
+#elif DISABLED(THERMAL_PROTECTION_BED) && defined(THERMAL_PROTECTION_BED_PERIOD)
+ #error "Thermal Runaway Protection for the bed is now enabled with THERMAL_PROTECTION_BED."
+#elif (CORE_IS_XZ || CORE_IS_YZ) && ENABLED(Z_LATE_ENABLE)
+ #error "Z_LATE_ENABLE can't be used with COREXZ, COREZX, COREYZ, or COREZY."
+#elif defined(X_HOME_RETRACT_MM)
+ #error "[XYZ]_HOME_RETRACT_MM settings have been renamed [XYZ]_HOME_BUMP_MM."
+#elif defined(SDCARDDETECTINVERTED)
+ #error "SDCARDDETECTINVERTED is now SD_DETECT_STATE (HIGH)."
+#elif defined(SD_DETECT_INVERTED)
+ #error "SD_DETECT_INVERTED is now SD_DETECT_STATE (HIGH)."
+#elif defined(BTENABLED)
+ #error "BTENABLED is now BLUETOOTH."
+#elif defined(CUSTOM_MENDEL_NAME)
+ #error "CUSTOM_MENDEL_NAME is now CUSTOM_MACHINE_NAME."
+#elif defined(HAS_AUTOMATIC_VERSIONING)
+ #error "HAS_AUTOMATIC_VERSIONING is now CUSTOM_VERSION_FILE."
+#elif defined(USE_AUTOMATIC_VERSIONING)
+ #error "USE_AUTOMATIC_VERSIONING is now CUSTOM_VERSION_FILE."
+#elif defined(SDSLOW)
+ #error "SDSLOW deprecated. Set SD_SPI_SPEED to SPI_HALF_SPEED instead."
+#elif defined(SDEXTRASLOW)
+ #error "SDEXTRASLOW deprecated. Set SD_SPI_SPEED to SPI_QUARTER_SPEED instead."
+#elif defined(FILAMENT_SENSOR)
+ #error "FILAMENT_SENSOR is now FILAMENT_WIDTH_SENSOR."
+#elif defined(ENDSTOPPULLUP_FIL_RUNOUT)
+ #error "ENDSTOPPULLUP_FIL_RUNOUT is now FIL_RUNOUT_PULLUP."
+#elif defined(DISABLE_MAX_ENDSTOPS) || defined(DISABLE_MIN_ENDSTOPS)
+ #error "DISABLE_MAX_ENDSTOPS and DISABLE_MIN_ENDSTOPS deprecated. Use individual USE_*_PLUG options instead."
+#elif defined(LANGUAGE_INCLUDE)
+ #error "LANGUAGE_INCLUDE has been replaced by LCD_LANGUAGE."
+#elif defined(EXTRUDER_OFFSET_X) || defined(EXTRUDER_OFFSET_Y)
+ #error "EXTRUDER_OFFSET_[XY] is deprecated. Use HOTEND_OFFSET_[XY] instead."
+#elif defined(PID_PARAMS_PER_EXTRUDER)
+ #error "PID_PARAMS_PER_EXTRUDER is deprecated. Use PID_PARAMS_PER_HOTEND instead."
+#elif defined(EXTRUDER_WATTS) || defined(BED_WATTS)
+ #error "EXTRUDER_WATTS and BED_WATTS are deprecated and should be removed."
+#elif defined(SERVO_ENDSTOP_ANGLES)
+ #error "SERVO_ENDSTOP_ANGLES is deprecated. Use Z_SERVO_ANGLES instead."
+#elif defined(X_ENDSTOP_SERVO_NR) || defined(Y_ENDSTOP_SERVO_NR)
+ #error "X_ENDSTOP_SERVO_NR and Y_ENDSTOP_SERVO_NR are deprecated and should be removed."
+#elif defined(Z_ENDSTOP_SERVO_NR)
+ #error "Z_ENDSTOP_SERVO_NR is now Z_PROBE_SERVO_NR."
+#elif defined(DEFAULT_XYJERK)
+ #error "DEFAULT_XYJERK is deprecated. Use DEFAULT_XJERK and DEFAULT_YJERK instead."
+#elif defined(XY_TRAVEL_SPEED)
+ #error "XY_TRAVEL_SPEED is deprecated. Use XY_PROBE_SPEED instead."
+#elif defined(PROBE_SERVO_DEACTIVATION_DELAY)
+ #error "PROBE_SERVO_DEACTIVATION_DELAY is deprecated. Use DEACTIVATE_SERVOS_AFTER_MOVE instead."
+#elif defined(SERVO_DEACTIVATION_DELAY)
+ #error "SERVO_DEACTIVATION_DELAY is now SERVO_DELAY."
+#elif ENABLED(FILAMENTCHANGEENABLE)
+ #error "FILAMENTCHANGEENABLE is now ADVANCED_PAUSE_FEATURE."
+#elif ENABLED(FILAMENT_CHANGE_FEATURE)
+ #error "FILAMENT_CHANGE_FEATURE is now ADVANCED_PAUSE_FEATURE."
+#elif defined(FILAMENT_CHANGE_X_POS) || defined(FILAMENT_CHANGE_Y_POS)
+ #error "FILAMENT_CHANGE_[XY]_POS is now set with NOZZLE_PARK_POINT."
+#elif defined(FILAMENT_CHANGE_Z_ADD)
+ #error "FILAMENT_CHANGE_Z_ADD is now set with NOZZLE_PARK_POINT."
+#elif defined(FILAMENT_CHANGE_XY_FEEDRATE)
+ #error "FILAMENT_CHANGE_XY_FEEDRATE is now NOZZLE_PARK_XY_FEEDRATE."
+#elif defined(FILAMENT_CHANGE_Z_FEEDRATE)
+ #error "FILAMENT_CHANGE_Z_FEEDRATE is now NOZZLE_PARK_Z_FEEDRATE."
+#elif defined(PAUSE_PARK_X_POS) || defined(PAUSE_PARK_Y_POS)
+ #error "PAUSE_PARK_[XY]_POS is now set with NOZZLE_PARK_POINT."
+#elif defined(PAUSE_PARK_Z_ADD)
+ #error "PAUSE_PARK_Z_ADD is now set with NOZZLE_PARK_POINT."
+#elif defined(PAUSE_PARK_XY_FEEDRATE)
+ #error "PAUSE_PARK_XY_FEEDRATE is now NOZZLE_PARK_XY_FEEDRATE."
+#elif defined(PAUSE_PARK_Z_FEEDRATE)
+ #error "PAUSE_PARK_Z_FEEDRATE is now NOZZLE_PARK_Z_FEEDRATE."
+#elif defined(FILAMENT_CHANGE_RETRACT_FEEDRATE)
+ #error "FILAMENT_CHANGE_RETRACT_FEEDRATE is now PAUSE_PARK_RETRACT_FEEDRATE."
+#elif defined(FILAMENT_CHANGE_RETRACT_LENGTH)
+ #error "FILAMENT_CHANGE_RETRACT_LENGTH is now PAUSE_PARK_RETRACT_LENGTH."
+#elif defined(FILAMENT_CHANGE_EXTRUDE_FEEDRATE)
+ #error "FILAMENT_CHANGE_EXTRUDE_FEEDRATE is now ADVANCED_PAUSE_PURGE_FEEDRATE."
+#elif defined(ADVANCED_PAUSE_EXTRUDE_FEEDRATE)
+ #error "ADVANCED_PAUSE_EXTRUDE_FEEDRATE is now ADVANCED_PAUSE_PURGE_FEEDRATE."
+#elif defined(FILAMENT_CHANGE_EXTRUDE_LENGTH)
+ #error "FILAMENT_CHANGE_EXTRUDE_LENGTH is now ADVANCED_PAUSE_PURGE_LENGTH."
+#elif defined(ADVANCED_PAUSE_EXTRUDE_LENGTH)
+ #error "ADVANCED_PAUSE_EXTRUDE_LENGTH is now ADVANCED_PAUSE_PURGE_LENGTH."
+#elif defined(FILAMENT_CHANGE_NOZZLE_TIMEOUT)
+ #error "FILAMENT_CHANGE_NOZZLE_TIMEOUT is now PAUSE_PARK_NOZZLE_TIMEOUT."
+#elif defined(FILAMENT_CHANGE_NUMBER_OF_ALERT_BEEPS)
+ #error "FILAMENT_CHANGE_NUMBER_OF_ALERT_BEEPS is now FILAMENT_CHANGE_ALERT_BEEPS."
+#elif defined(FILAMENT_CHANGE_NO_STEPPER_TIMEOUT)
+ #error "FILAMENT_CHANGE_NO_STEPPER_TIMEOUT is now PAUSE_PARK_NO_STEPPER_TIMEOUT."
+#elif defined(PLA_PREHEAT_HOTEND_TEMP)
+ #error "PLA_PREHEAT_HOTEND_TEMP is now PREHEAT_1_TEMP_HOTEND."
+#elif defined(PLA_PREHEAT_HPB_TEMP)
+ #error "PLA_PREHEAT_HPB_TEMP is now PREHEAT_1_TEMP_BED."
+#elif defined(PLA_PREHEAT_FAN_SPEED)
+ #error "PLA_PREHEAT_FAN_SPEED is now PREHEAT_1_FAN_SPEED."
+#elif defined(ABS_PREHEAT_HOTEND_TEMP)
+ #error "ABS_PREHEAT_HOTEND_TEMP is now PREHEAT_2_TEMP_HOTEND."
+#elif defined(ABS_PREHEAT_HPB_TEMP)
+ #error "ABS_PREHEAT_HPB_TEMP is now PREHEAT_2_TEMP_BED."
+#elif defined(ABS_PREHEAT_FAN_SPEED)
+ #error "ABS_PREHEAT_FAN_SPEED is now PREHEAT_2_FAN_SPEED."
+#elif defined(ENDSTOPS_ONLY_FOR_HOMING)
+ #error "ENDSTOPS_ONLY_FOR_HOMING is deprecated. Use (disable) ENDSTOPS_ALWAYS_ON_DEFAULT instead."
+#elif defined(HOMING_FEEDRATE)
+ #error "HOMING_FEEDRATE is now set using the HOMING_FEEDRATE_MM_M array instead."
+#elif (defined(HOMING_FEEDRATE_XY) || defined(HOMING_FEEDRATE_Z)) && !defined(HOMING_FEEDRATE_MM_M)
+ #error "HOMING_FEEDRATE_XY and HOMING_FEEDRATE_Z are now set using the HOMING_FEEDRATE_MM_M array instead."
+#elif defined(MANUAL_HOME_POSITIONS)
+ #error "MANUAL_HOME_POSITIONS is deprecated. Set MANUAL_[XYZ]_HOME_POS as-needed instead."
+#elif defined(PID_ADD_EXTRUSION_RATE)
+ #error "PID_ADD_EXTRUSION_RATE is now PID_EXTRUSION_SCALING and is DISABLED by default."
+#elif defined(Z_RAISE_BEFORE_HOMING)
+ #error "Z_RAISE_BEFORE_HOMING is now Z_HOMING_HEIGHT."
+#elif defined(MIN_Z_HEIGHT_FOR_HOMING)
+ #error "MIN_Z_HEIGHT_FOR_HOMING is now Z_HOMING_HEIGHT."
+#elif defined(Z_RAISE_BEFORE_PROBING) || defined(Z_RAISE_AFTER_PROBING)
+ #error "Z_RAISE_(BEFORE|AFTER)_PROBING are deprecated. Use Z_CLEARANCE_DEPLOY_PROBE and Z_AFTER_PROBING instead."
+#elif defined(Z_RAISE_PROBE_DEPLOY_STOW) || defined(Z_RAISE_BETWEEN_PROBINGS)
+ #error "Z_RAISE_PROBE_DEPLOY_STOW and Z_RAISE_BETWEEN_PROBINGS are now Z_CLEARANCE_DEPLOY_PROBE and Z_CLEARANCE_BETWEEN_PROBES."
+#elif defined(Z_PROBE_DEPLOY_HEIGHT) || defined(Z_PROBE_TRAVEL_HEIGHT)
+ #error "Z_PROBE_DEPLOY_HEIGHT and Z_PROBE_TRAVEL_HEIGHT are now Z_CLEARANCE_DEPLOY_PROBE and Z_CLEARANCE_BETWEEN_PROBES."
+#elif defined(MANUAL_BED_LEVELING)
+ #error "MANUAL_BED_LEVELING is now LCD_BED_LEVELING."
+#elif defined(MESH_HOME_SEARCH_Z)
+ #error "MESH_HOME_SEARCH_Z is now LCD_PROBE_Z_RANGE."
+#elif defined(MANUAL_PROBE_Z_RANGE)
+ #error "MANUAL_PROBE_Z_RANGE is now LCD_PROBE_Z_RANGE."
+#elif !defined(MIN_STEPS_PER_SEGMENT)
+ #error "Please replace 'const int dropsegments' with '#define MIN_STEPS_PER_SEGMENT' (and increase by 1)."
+#elif MIN_STEPS_PER_SEGMENT <= 0
+ #error "MIN_STEPS_PER_SEGMENT must be at least 1."
+#elif defined(PREVENT_DANGEROUS_EXTRUDE)
+ #error "PREVENT_DANGEROUS_EXTRUDE is now PREVENT_COLD_EXTRUSION."
+#elif defined(SCARA)
+ #error "SCARA is now MORGAN_SCARA."
+#elif defined(ENABLE_AUTO_BED_LEVELING)
+ #error "ENABLE_AUTO_BED_LEVELING is deprecated. Specify AUTO_BED_LEVELING_LINEAR, AUTO_BED_LEVELING_BILINEAR, or AUTO_BED_LEVELING_3POINT."
+#elif defined(AUTO_BED_LEVELING_FEATURE)
+ #error "AUTO_BED_LEVELING_FEATURE is deprecated. Specify AUTO_BED_LEVELING_LINEAR, AUTO_BED_LEVELING_BILINEAR, or AUTO_BED_LEVELING_3POINT."
+#elif defined(ABL_GRID_POINTS)
+ #error "ABL_GRID_POINTS is now GRID_MAX_POINTS_X and GRID_MAX_POINTS_Y."
+#elif defined(ABL_GRID_POINTS_X) || defined(ABL_GRID_POINTS_Y)
+ #error "ABL_GRID_POINTS_[XY] is now GRID_MAX_POINTS_[XY]."
+#elif defined(ABL_GRID_MAX_POINTS_X) || defined(ABL_GRID_MAX_POINTS_Y)
+ #error "ABL_GRID_MAX_POINTS_[XY] is now GRID_MAX_POINTS_[XY]."
+#elif defined(MESH_NUM_X_POINTS) || defined(MESH_NUM_Y_POINTS)
+ #error "MESH_NUM_[XY]_POINTS is now GRID_MAX_POINTS_[XY]."
+#elif defined(UBL_MESH_NUM_X_POINTS) || defined(UBL_MESH_NUM_Y_POINTS)
+ #error "UBL_MESH_NUM_[XY]_POINTS is now GRID_MAX_POINTS_[XY]."
+#elif defined(UBL_G26_MESH_VALIDATION)
+ #error "UBL_G26_MESH_VALIDATION is now G26_MESH_VALIDATION."
+#elif defined(UBL_MESH_EDIT_ENABLED)
+ #error "UBL_MESH_EDIT_ENABLED is now G26_MESH_VALIDATION."
+#elif defined(UBL_MESH_EDITING)
+ #error "UBL_MESH_EDITING is now G26_MESH_VALIDATION."
+#elif defined(BLTOUCH_HEATERS_OFF)
+ #error "BLTOUCH_HEATERS_OFF is now PROBING_HEATERS_OFF."
+#elif defined(BLTOUCH_V3)
+ #error "BLTOUCH_V3 is obsolete."
+#elif defined(BLTOUCH_FORCE_OPEN_DRAIN_MODE)
+ #error "BLTOUCH_FORCE_OPEN_DRAIN_MODE is obsolete."
+#elif defined(BEEPER)
+ #error "BEEPER is now BEEPER_PIN."
+#elif defined(SDCARDDETECT)
+ #error "SDCARDDETECT is now SD_DETECT_PIN."
+#elif defined(STAT_LED_RED) || defined(STAT_LED_BLUE)
+ #error "STAT_LED_RED/STAT_LED_BLUE are now STAT_LED_RED_PIN/STAT_LED_BLUE_PIN."
+#elif defined(LCD_PIN_BL)
+ #error "LCD_PIN_BL is now LCD_BACKLIGHT_PIN."
+#elif defined(LCD_PIN_RESET)
+ #error "LCD_PIN_RESET is now LCD_RESET_PIN."
+#elif defined(EXTRUDER_0_AUTO_FAN_PIN) || defined(EXTRUDER_1_AUTO_FAN_PIN) || defined(EXTRUDER_2_AUTO_FAN_PIN) || defined(EXTRUDER_3_AUTO_FAN_PIN)
+ #error "EXTRUDER_[0123]_AUTO_FAN_PIN is now E[0123]_AUTO_FAN_PIN."
+#elif defined(PID_FAN_SCALING) && !HAS_FAN
+ #error "PID_FAN_SCALING needs at least one fan enabled."
+#elif defined(min_software_endstops) || defined(max_software_endstops)
+ #error "(min|max)_software_endstops are now (MIN|MAX)_SOFTWARE_ENDSTOPS."
+#elif ENABLED(Z_PROBE_SLED) && defined(SLED_PIN)
+ #error "Replace SLED_PIN with SOL1_PIN (applies to both Z_PROBE_SLED and SOLENOID_PROBE)."
+#elif defined(CONTROLLERFAN_PIN)
+ #error "CONTROLLERFAN_PIN is now CONTROLLER_FAN_PIN, enabled with USE_CONTROLLER_FAN."
+#elif defined(CONTROLLERFAN_SPEED)
+ #error "CONTROLLERFAN_SPEED is now CONTROLLERFAN_SPEED_ACTIVE."
+#elif defined(CONTROLLERFAN_SECS)
+ #error "CONTROLLERFAN_SECS is now CONTROLLERFAN_IDLE_TIME."
+#elif defined(MIN_RETRACT)
+ #error "MIN_RETRACT is now MIN_AUTORETRACT and MAX_AUTORETRACT."
+#elif defined(ADVANCE)
+ #error "ADVANCE is now LIN_ADVANCE."
+#elif defined(LIN_ADVANCE_E_D_RATIO)
+ #error "LIN_ADVANCE (1.5) no longer uses LIN_ADVANCE_E_D_RATIO."
+#elif defined(NEOPIXEL_RGBW_LED)
+ #error "NEOPIXEL_RGBW_LED is now NEOPIXEL_LED."
+#elif ENABLED(DELTA) && defined(DELTA_PROBEABLE_RADIUS)
+ #error "Remove DELTA_PROBEABLE_RADIUS and use PROBING_MARGIN to inset the probe area instead."
+#elif ENABLED(DELTA) && defined(DELTA_CALIBRATION_RADIUS)
+ #error "Remove DELTA_CALIBRATION_RADIUS and use PROBING_MARGIN to inset the probe area instead."
+#elif defined(UBL_MESH_INSET)
+ #error "UBL_MESH_INSET is now just MESH_INSET."
+#elif defined(UBL_MESH_MIN_X) || defined(UBL_MESH_MIN_Y) || defined(UBL_MESH_MAX_X) || defined(UBL_MESH_MAX_Y)
+ #error "UBL_MESH_(MIN|MAX)_[XY] is now just MESH_(MIN|MAX)_[XY]."
+#elif defined(ABL_PROBE_PT_1_X) || defined(ABL_PROBE_PT_1_Y) || defined(ABL_PROBE_PT_2_X) || defined(ABL_PROBE_PT_2_Y) || defined(ABL_PROBE_PT_3_X) || defined(ABL_PROBE_PT_3_Y)
+ #error "ABL_PROBE_PT_[123]_[XY] is no longer required. Please remove it."
+#elif defined(UBL_PROBE_PT_1_X) || defined(UBL_PROBE_PT_1_Y) || defined(UBL_PROBE_PT_2_X) || defined(UBL_PROBE_PT_2_Y) || defined(UBL_PROBE_PT_3_X) || defined(UBL_PROBE_PT_3_Y)
+ #error "UBL_PROBE_PT_[123]_[XY] is no longer required. Please remove it."
+#elif defined(MIN_PROBE_EDGE)
+ #error "MIN_PROBE_EDGE is now called PROBING_MARGIN."
+#elif defined(MIN_PROBE_EDGE_LEFT)
+ #error "MIN_PROBE_EDGE_LEFT is now called PROBING_MARGIN_LEFT."
+#elif defined(MIN_PROBE_EDGE_RIGHT)
+ #error "MIN_PROBE_EDGE_RIGHT is now called PROBING_MARGIN_RIGHT."
+#elif defined(MIN_PROBE_EDGE_FRONT)
+ #error "MIN_PROBE_EDGE_FRONT is now called PROBING_MARGIN_FRONT."
+#elif defined(MIN_PROBE_EDGE_BACK)
+ #error "MIN_PROBE_EDGE_BACK is now called PROBING_MARGIN_BACK."
+#elif defined(LEFT_PROBE_BED_POSITION)
+ #error "LEFT_PROBE_BED_POSITION is obsolete. Set a margin with PROBING_MARGIN or PROBING_MARGIN_LEFT instead."
+#elif defined(RIGHT_PROBE_BED_POSITION)
+ #error "RIGHT_PROBE_BED_POSITION is obsolete. Set a margin with PROBING_MARGIN or PROBING_MARGIN_RIGHT instead."
+#elif defined(FRONT_PROBE_BED_POSITION)
+ #error "FRONT_PROBE_BED_POSITION is obsolete. Set a margin with PROBING_MARGIN or PROBING_MARGIN_FRONT instead."
+#elif defined(BACK_PROBE_BED_POSITION)
+ #error "BACK_PROBE_BED_POSITION is obsolete. Set a margin with PROBING_MARGIN or PROBING_MARGIN_BACK instead."
+#elif defined(ENABLE_MESH_EDIT_GFX_OVERLAY)
+ #error "ENABLE_MESH_EDIT_GFX_OVERLAY is now MESH_EDIT_GFX_OVERLAY."
+#elif defined(BABYSTEP_ZPROBE_GFX_REVERSE)
+ #error "BABYSTEP_ZPROBE_GFX_REVERSE is now set by OVERLAY_GFX_REVERSE."
+#elif defined(UBL_GRANULAR_SEGMENTATION_FOR_CARTESIAN)
+ #error "UBL_GRANULAR_SEGMENTATION_FOR_CARTESIAN is now SEGMENT_LEVELED_MOVES."
+#elif HAS_PID_HEATING && (defined(K1) || !defined(PID_K1))
+ #error "K1 is now PID_K1."
+#elif defined(PROBE_DOUBLE_TOUCH)
+ #error "PROBE_DOUBLE_TOUCH is now MULTIPLE_PROBING."
+#elif defined(ANET_KEYPAD_LCD)
+ #error "ANET_KEYPAD_LCD is now ZONESTAR_LCD."
+#elif defined(LCD_I2C_SAINSMART_YWROBOT)
+ #error "LCD_I2C_SAINSMART_YWROBOT is now LCD_SAINSMART_I2C_(1602|2004)."
+#elif defined(MEASURED_LOWER_LIMIT) || defined(MEASURED_UPPER_LIMIT)
+ #error "MEASURED_(UPPER|LOWER)_LIMIT is now FILWIDTH_ERROR_MARGIN."
+#elif defined(HAVE_TMCDRIVER)
+ #error "HAVE_TMCDRIVER is now [AXIS]_DRIVER_TYPE TMC26X."
+#elif defined(STEALTHCHOP)
+ #error "STEALTHCHOP is now STEALTHCHOP_(XY|Z|E)."
+#elif defined(HAVE_TMC26X)
+ #error "HAVE_TMC26X is now [AXIS]_DRIVER_TYPE TMC26X."
+#elif defined(HAVE_TMC2130)
+ #error "HAVE_TMC2130 is now [AXIS]_DRIVER_TYPE TMC2130."
+#elif defined(HAVE_TMC2208)
+ #error "HAVE_TMC2208 is now [AXIS]_DRIVER_TYPE TMC2208."
+#elif defined(HAVE_L6470DRIVER)
+ #error "HAVE_L6470DRIVER is now [AXIS]_DRIVER_TYPE L6470."
+#elif defined(X_IS_TMC) || defined(X2_IS_TMC) || defined(Y_IS_TMC) || defined(Y2_IS_TMC) || defined(Z_IS_TMC) || defined(Z2_IS_TMC) || defined(Z3_IS_TMC) \
+ || defined(E0_IS_TMC) || defined(E1_IS_TMC) || defined(E2_IS_TMC) || defined(E3_IS_TMC) || defined(E4_IS_TMC) || defined(E5_IS_TMC) || defined(E6_IS_TMC) || defined(E7_IS_TMC)
+ #error "[AXIS]_IS_TMC is now [AXIS]_DRIVER_TYPE TMC26X."
+#elif defined(X_IS_TMC26X) || defined(X2_IS_TMC26X) || defined(Y_IS_TMC26X) || defined(Y2_IS_TMC26X) || defined(Z_IS_TMC26X) || defined(Z2_IS_TMC26X) || defined(Z3_IS_TMC26X) \
+ || defined(E0_IS_TMC26X) || defined(E1_IS_TMC26X) || defined(E2_IS_TMC26X) || defined(E3_IS_TMC26X) || defined(E4_IS_TMC26X) || defined(E5_IS_TMC26X) || defined(E6_IS_TMC26X) || defined(E7_IS_TMC26X)
+ #error "[AXIS]_IS_TMC26X is now [AXIS]_DRIVER_TYPE TMC26X."
+#elif defined(X_IS_TMC2130) || defined(X2_IS_TMC2130) || defined(Y_IS_TMC2130) || defined(Y2_IS_TMC2130) || defined(Z_IS_TMC2130) || defined(Z2_IS_TMC2130) || defined(Z3_IS_TMC2130) \
+ || defined(E0_IS_TMC2130) || defined(E1_IS_TMC2130) || defined(E2_IS_TMC2130) || defined(E3_IS_TMC2130) || defined(E4_IS_TMC2130) || defined(E5_IS_TMC2130) || defined(E6_IS_TMC2130) || defined(E7_IS_TMC2130)
+ #error "[AXIS]_IS_TMC2130 is now [AXIS]_DRIVER_TYPE TMC2130."
+#elif defined(X_IS_TMC2208) || defined(X2_IS_TMC2208) || defined(Y_IS_TMC2208) || defined(Y2_IS_TMC2208) || defined(Z_IS_TMC2208) || defined(Z2_IS_TMC2208) || defined(Z3_IS_TMC2208) \
+ || defined(E0_IS_TMC2208) || defined(E1_IS_TMC2208) || defined(E2_IS_TMC2208) || defined(E3_IS_TMC2208) || defined(E4_IS_TMC2208) || defined(E5_IS_TMC2208) || defined(E6_IS_TMC2208) || defined(E7_IS_TMC2208)
+ #error "[AXIS]_IS_TMC2208 is now [AXIS]_DRIVER_TYPE TMC2208."
+#elif defined(X_IS_L6470) || defined(X2_IS_L6470) || defined(Y_IS_L6470) || defined(Y2_IS_L6470) || defined(Z_IS_L6470) || defined(Z2_IS_L6470) || defined(Z3_IS_L6470) \
+ || defined(E0_IS_L6470) || defined(E1_IS_L6470) || defined(E2_IS_L6470) || defined(E3_IS_L6470) || defined(E4_IS_L6470) || defined(E5_IS_L6470) || defined(E6_IS_L6470) || defined(E7_IS_L6470)
+ #error "[AXIS]_IS_L6470 is now [AXIS]_DRIVER_TYPE L6470."
+#elif defined(AUTOMATIC_CURRENT_CONTROL)
+ #error "AUTOMATIC_CURRENT_CONTROL is now MONITOR_DRIVER_STATUS."
+#elif defined(FILAMENT_CHANGE_LOAD_LENGTH)
+ #error "FILAMENT_CHANGE_LOAD_LENGTH is now FILAMENT_CHANGE_FAST_LOAD_LENGTH."
+#elif defined(LEVEL_CORNERS_INSET)
+ #error "LEVEL_CORNERS_INSET is now LEVEL_CORNERS_INSET_LFRB."
+#elif defined(BEZIER_JERK_CONTROL)
+ #error "BEZIER_JERK_CONTROL is now S_CURVE_ACCELERATION."
+#elif HAS_JUNCTION_DEVIATION && defined(JUNCTION_DEVIATION_FACTOR)
+ #error "JUNCTION_DEVIATION_FACTOR is now JUNCTION_DEVIATION_MM."
+#elif defined(JUNCTION_ACCELERATION_FACTOR)
+ #error "JUNCTION_ACCELERATION_FACTOR is obsolete. Delete it from Configuration_adv.h."
+#elif defined(JUNCTION_ACCELERATION)
+ #error "JUNCTION_ACCELERATION is obsolete. Delete it from Configuration_adv.h."
+#elif defined(MAX7219_DEBUG_STEPPER_HEAD)
+ #error "MAX7219_DEBUG_STEPPER_HEAD is now MAX7219_DEBUG_PLANNER_HEAD."
+#elif defined(MAX7219_DEBUG_STEPPER_TAIL)
+ #error "MAX7219_DEBUG_STEPPER_TAIL is now MAX7219_DEBUG_PLANNER_TAIL."
+#elif defined(MAX7219_DEBUG_STEPPER_QUEUE)
+ #error "MAX7219_DEBUG_STEPPER_QUEUE is now MAX7219_DEBUG_PLANNER_QUEUE."
+#elif defined(ENDSTOP_NOISE_FILTER)
+ #error "ENDSTOP_NOISE_FILTER is now ENDSTOP_NOISE_THRESHOLD [2-7]."
+#elif defined(RETRACT_ZLIFT)
+ #error "RETRACT_ZLIFT is now RETRACT_ZRAISE."
+#elif defined(TOOLCHANGE_PARK_ZLIFT) || defined(TOOLCHANGE_UNPARK_ZLIFT)
+ #error "TOOLCHANGE_PARK_ZLIFT and TOOLCHANGE_UNPARK_ZLIFT are now TOOLCHANGE_ZRAISE."
+#elif defined(SINGLENOZZLE_TOOLCHANGE_ZRAISE)
+ #error "SINGLENOZZLE_TOOLCHANGE_ZRAISE is now TOOLCHANGE_ZRAISE."
+#elif defined(SINGLENOZZLE_SWAP_LENGTH)
+ #error "SINGLENOZZLE_SWAP_LENGTH is now TOOLCHANGE_FIL_SWAP_LENGTH."
+#elif defined(SINGLENOZZLE_SWAP_RETRACT_SPEED)
+ #error "SINGLENOZZLE_SWAP_RETRACT_SPEED is now TOOLCHANGE_FIL_SWAP_RETRACT_SPEED."
+#elif defined(SINGLENOZZLE_SWAP_PRIME_SPEED)
+ #error "SINGLENOZZLE_SWAP_PRIME_SPEED is now TOOLCHANGE_FIL_SWAP_PRIME_SPEED."
+#elif defined(SINGLENOZZLE_SWAP_PARK)
+ #error "SINGLENOZZLE_SWAP_PARK is now TOOLCHANGE_PARK."
+#elif defined(SINGLENOZZLE_TOOLCHANGE_XY)
+ #error "SINGLENOZZLE_TOOLCHANGE_XY is now TOOLCHANGE_PARK_XY."
+#elif defined(SINGLENOZZLE_PARK_XY_FEEDRATE)
+ #error "SINGLENOZZLE_PARK_XY_FEEDRATE is now TOOLCHANGE_PARK_XY_FEEDRATE."
+#elif defined(PARKING_EXTRUDER_SECURITY_RAISE)
+ #error "PARKING_EXTRUDER_SECURITY_RAISE is now TOOLCHANGE_ZRAISE."
+#elif defined(SWITCHING_TOOLHEAD_SECURITY_RAISE)
+ #error "SWITCHING_TOOLHEAD_SECURITY_RAISE is now TOOLCHANGE_ZRAISE."
+#elif defined(G0_FEEDRATE) && G0_FEEDRATE == 0
+ #error "G0_FEEDRATE is now used to set the G0 feedrate."
+#elif defined(MBL_Z_STEP)
+ #error "MBL_Z_STEP is now MESH_EDIT_Z_STEP."
+#elif defined(CHDK)
+ #error "CHDK is now CHDK_PIN."
+#elif defined(MAX6675_SS) || defined(MAX6675_SS2)
+ #error "MAX6675_SS / MAX6675_SS2 is now MAX6675_SS_PIN / MAX6675_SS2_PIN."
+#elif defined(MAX31865_SENSOR_OHMS)
+ #error "MAX31865_SENSOR_OHMS is now MAX31865_SENSOR_OHMS_0."
+#elif defined(MAX31865_CALIBRATION_OHMS)
+ #error "MAX31865_CALIBRATION_OHMS is now MAX31865_CALIBRATION_OHMS_0."
+#elif defined(SPINDLE_LASER_ENABLE)
+ #error "SPINDLE_LASER_ENABLE is now SPINDLE_FEATURE or LASER_FEATURE."
+#elif defined(SPINDLE_LASER_ENABLE_PIN)
+ #error "SPINDLE_LASER_ENABLE_PIN is now SPINDLE_LASER_ENA_PIN."
+#elif defined(SPINDLE_DIR_CHANGE)
+ #error "SPINDLE_DIR_CHANGE is now SPINDLE_CHANGE_DIR."
+#elif defined(SPINDLE_STOP_ON_DIR_CHANGE)
+ #error "SPINDLE_STOP_ON_DIR_CHANGE is now SPINDLE_CHANGE_DIR_STOP."
+#elif defined(SPINDLE_LASER_ACTIVE_HIGH)
+ #error "SPINDLE_LASER_ACTIVE_HIGH is now SPINDLE_LASER_ACTIVE_STATE."
+#elif defined(SPINDLE_LASER_ENABLE_INVERT)
+ #error "SPINDLE_LASER_ENABLE_INVERT is now SPINDLE_LASER_ACTIVE_STATE."
+#elif defined(CUTTER_POWER_DISPLAY)
+ #error "CUTTER_POWER_DISPLAY is now CUTTER_POWER_UNIT."
+#elif defined(CHAMBER_HEATER_PIN)
+ #error "CHAMBER_HEATER_PIN is now HEATER_CHAMBER_PIN."
+#elif defined(TMC_Z_CALIBRATION)
+ #error "TMC_Z_CALIBRATION has been deprecated in favor of MECHANICAL_GANTRY_CALIBRATION."
+#elif defined(Z_MIN_PROBE_ENDSTOP)
+ #error "Z_MIN_PROBE_ENDSTOP is no longer required. Please remove it."
+#elif defined(DUAL_NOZZLE_DUPLICATION_MODE)
+ #error "DUAL_NOZZLE_DUPLICATION_MODE is now MULTI_NOZZLE_DUPLICATION."
+#elif defined(MENU_ITEM_CASE_LIGHT)
+ #error "MENU_ITEM_CASE_LIGHT is now CASE_LIGHT_MENU."
+#elif defined(ABORT_ON_ENDSTOP_HIT_FEATURE_ENABLED)
+ #error "ABORT_ON_ENDSTOP_HIT_FEATURE_ENABLED is now SD_ABORT_ON_ENDSTOP_HIT."
+#elif defined(LPC_SD_LCD) || defined(LPC_SD_ONBOARD) || defined(LPC_SD_CUSTOM_CABLE)
+ #error "LPC_SD_(LCD|ONBOARD|CUSTOM_CABLE) are now SDCARD_CONNECTION."
+#elif defined(USB_SD_DISABLED)
+ #error "USB_SD_DISABLED is now NO_SD_HOST_DRIVE."
+#elif defined(USB_SD_ONBOARD)
+ #error "USB_SD_ONBOARD is obsolete. Disable NO_SD_HOST_DRIVE instead."
+#elif defined(PSU_ACTIVE_HIGH)
+ #error "PSU_ACTIVE_HIGH is now PSU_ACTIVE_STATE."
+#elif POWER_SUPPLY == 1
+ #error "Replace POWER_SUPPLY 1 by enabling PSU_CONTROL and setting PSU_ACTIVE_STATE to 'LOW'."
+#elif POWER_SUPPLY == 2
+ #error "Replace POWER_SUPPLY 2 by enabling PSU_CONTROL and setting PSU_ACTIVE_STATE to 'HIGH'."
+#elif defined(POWER_SUPPLY)
+ #error "POWER_SUPPLY is now obsolete. Please remove it."
+#elif defined(MKS_ROBIN_TFT)
+ #error "MKS_ROBIN_TFT is now FSMC_GRAPHICAL_TFT."
+#elif defined(SDPOWER)
+ #error "SDPOWER is now SDPOWER_PIN."
+#elif defined(STRING_SPLASH_LINE1) || defined(STRING_SPLASH_LINE2)
+ #error "STRING_SPLASH_LINE[12] are now obsolete. Please remove them."
+#elif defined(Z_PROBE_ALLEN_KEY_DEPLOY_1_X) || defined(Z_PROBE_ALLEN_KEY_STOW_1_X)
+ #error "Z_PROBE_ALLEN_KEY_(DEPLOY|STOW) coordinates are now a single setting."
+#elif defined(X_PROBE_OFFSET_FROM_EXTRUDER) || defined(Y_PROBE_OFFSET_FROM_EXTRUDER) || defined(Z_PROBE_OFFSET_FROM_EXTRUDER)
+ #error "[XYZ]_PROBE_OFFSET_FROM_EXTRUDER is now NOZZLE_TO_PROBE_OFFSET."
+#elif defined(MIN_PROBE_X) || defined(MIN_PROBE_Y) || defined(MAX_PROBE_X) || defined(MAX_PROBE_Y)
+ #error "(MIN|MAX)_PROBE_[XY] are now calculated at runtime. Please remove them."
+#elif defined(Z_STEPPER_ALIGN_X) || defined(Z_STEPPER_ALIGN_X)
+ #error "Z_STEPPER_ALIGN_X and Z_STEPPER_ALIGN_Y are now combined as Z_STEPPER_ALIGN_XY."
+#elif defined(JUNCTION_DEVIATION)
+ #error "JUNCTION_DEVIATION is no longer required. (See CLASSIC_JERK). Please remove it."
+#elif defined(BABYSTEP_MULTIPLICATOR)
+ #error "BABYSTEP_MULTIPLICATOR is now BABYSTEP_MULTIPLICATOR_[XY|Z]."
+#elif defined(LULZBOT_TOUCH_UI)
+ #error "LULZBOT_TOUCH_UI is now TOUCH_UI_FTDI_EVE."
+#elif defined(PS_DEFAULT_OFF)
+ #error "PS_DEFAULT_OFF is now PSU_DEFAULT_OFF."
+#elif defined(FILAMENT_UNLOAD_RETRACT_LENGTH)
+ #error "FILAMENT_UNLOAD_RETRACT_LENGTH is now FILAMENT_UNLOAD_PURGE_RETRACT."
+#elif defined(FILAMENT_UNLOAD_DELAY)
+ #error "FILAMENT_UNLOAD_DELAY is now FILAMENT_UNLOAD_PURGE_DELAY."
+#elif defined(HOME_USING_SPREADCYCLE)
+ #error "HOME_USING_SPREADCYCLE is now obsolete. Please remove it."
+#elif defined(DGUS_LCD)
+ #error "DGUS_LCD is now DGUS_LCD_UI_(ORIGIN|FYSETC|HIPRECY)."
+#elif defined(DGUS_SERIAL_PORT)
+ #error "DGUS_SERIAL_PORT is now LCD_SERIAL_PORT."
+#elif defined(DGUS_BAUDRATE)
+ #error "DGUS_BAUDRATE is now LCD_BAUDRATE."
+#elif defined(DGUS_STATS_RX_BUFFER_OVERRUNS)
+ #error "DGUS_STATS_RX_BUFFER_OVERRUNS is now STATS_RX_BUFFER_OVERRUNS."
+#elif defined(ANYCUBIC_LCD_SERIAL_PORT)
+ #error "ANYCUBIC_LCD_SERIAL_PORT is now LCD_SERIAL_PORT."
+#elif defined(INTERNAL_SERIAL_PORT)
+ #error "INTERNAL_SERIAL_PORT is now MMU2_SERIAL_PORT."
+#elif defined(X_DUAL_ENDSTOPS_ADJUSTMENT) || defined(Y_DUAL_ENDSTOPS_ADJUSTMENT) || defined(Z_DUAL_ENDSTOPS_ADJUSTMENT)
+ #error "[XYZ]_DUAL_ENDSTOPS_ADJUSTMENT is now [XYZ]2_ENDSTOP_ADJUSTMENT."
+#elif defined(Z_TRIPLE_ENDSTOPS_ADJUSTMENT2) || defined(Z_TRIPLE_ENDSTOPS_ADJUSTMENT3)
+ #error "Z_TRIPLE_ENDSTOPS_ADJUSTMENT[23] is now Z[23]_ENDSTOP_ADJUSTMENT."
+#elif defined(Z_QUAD_ENDSTOPS_ADJUSTMENT2) || defined(Z_QUAD_ENDSTOPS_ADJUSTMENT3) || defined(Z_QUAD_ENDSTOPS_ADJUSTMENT4)
+ #error "Z_QUAD_ENDSTOPS_ADJUSTMENT[234] is now Z[234]_ENDSTOP_ADJUSTMENT."
+#elif defined(Z_DUAL_STEPPER_DRIVERS)
+ #error "Z_DUAL_STEPPER_DRIVERS is now NUM_Z_STEPPER_DRIVERS with a value of 2."
+#elif defined(Z_TRIPLE_STEPPER_DRIVERS)
+ #error "Z_TRIPLE_STEPPER_DRIVERS is now NUM_Z_STEPPER_DRIVERS with a value of 3."
+#elif defined(Z_QUAD_STEPPER_DRIVERS)
+ #error "Z_QUAD_STEPPER_DRIVERS is now NUM_Z_STEPPER_DRIVERS with a value of 4."
+#elif defined(Z_DUAL_ENDSTOPS) || defined(Z_TRIPLE_ENDSTOPS) || defined(Z_QUAD_ENDSTOPS)
+ #error "Z_(DUAL|TRIPLE|QUAD)_ENDSTOPS is now Z_MULTI_ENDSTOPS."
+#elif defined(DUGS_UI_MOVE_DIS_OPTION)
+ #error "DUGS_UI_MOVE_DIS_OPTION is spelled DGUS_UI_MOVE_DIS_OPTION."
+#elif defined(ORIG_E0_AUTO_FAN_PIN) || defined(ORIG_E1_AUTO_FAN_PIN) || defined(ORIG_E2_AUTO_FAN_PIN) || defined(ORIG_E3_AUTO_FAN_PIN) || defined(ORIG_E4_AUTO_FAN_PIN) || defined(ORIG_E5_AUTO_FAN_PIN) || defined(ORIG_E6_AUTO_FAN_PIN) || defined(ORIG_E7_AUTO_FAN_PIN)
+ #error "ORIG_Ex_AUTO_FAN_PIN is now just Ex_AUTO_FAN_PIN."
+#elif defined(ORIG_CHAMBER_AUTO_FAN_PIN)
+ #error "ORIG_CHAMBER_AUTO_FAN_PIN is now just CHAMBER_AUTO_FAN_PIN."
+#elif defined(HOMING_BACKOFF_MM)
+ #error "HOMING_BACKOFF_MM is now HOMING_BACKOFF_POST_MM."
+#elif defined(X_HOME_BUMP_MM) || defined(Y_HOME_BUMP_MM) || defined(Z_HOME_BUMP_MM)
+ #error "[XYZ]_HOME_BUMP_MM is now HOMING_BUMP_MM."
+#elif defined(DIGIPOT_I2C)
+ #error "DIGIPOT_I2C is now DIGIPOT_MCP4451 (or DIGIPOT_MCP4018)."
+#elif defined(TOUCH_BUTTONS)
+ #error "TOUCH_BUTTONS is now TOUCH_SCREEN."
+#elif defined(LCD_FULL_PIXEL_HEIGHT) || defined(LCD_FULL_PIXEL_WIDTH)
+ #error "LCD_FULL_PIXEL_(WIDTH|HEIGHT) is deprecated and should be removed."
+#elif defined(FSMC_UPSCALE)
+ #error "FSMC_UPSCALE is now GRAPHICAL_TFT_UPSCALE."
+#elif defined(ANYCUBIC_TFT_MODEL)
+ #error "ANYCUBIC_TFT_MODEL is now ANYCUBIC_LCD_I3MEGA."
+#elif defined(EVENT_GCODE_SD_STOP)
+ #error "EVENT_GCODE_SD_STOP is now EVENT_GCODE_SD_ABORT."
+#elif defined(GRAPHICAL_TFT_ROTATE_180)
+ #error "GRAPHICAL_TFT_ROTATE_180 is now TFT_ROTATION set to TFT_ROTATE_180."
+#elif defined(PROBE_OFFSET_START)
+ #error "PROBE_OFFSET_START is now PROBE_OFFSET_WIZARD_START_Z."
+#elif defined(POWER_LOSS_PULL)
+ #error "POWER_LOSS_PULL is now specifically POWER_LOSS_PULL(UP|DOWN)."
+#elif defined(SHORT_MANUAL_Z_MOVE)
+ #error "SHORT_MANUAL_Z_MOVE is now FINE_MANUAL_MOVE, applying to Z on most printers."
+#elif defined(FIL_RUNOUT_INVERTING)
+ #if FIL_RUNOUT_INVERTING
+ #error "FIL_RUNOUT_INVERTING true is now FIL_RUNOUT_STATE HIGH."
+ #else
+ #error "FIL_RUNOUT_INVERTING false is now FIL_RUNOUT_STATE LOW."
+ #endif
+#elif defined(ASSISTED_TRAMMING_MENU_ITEM)
+ #error "ASSISTED_TRAMMING_MENU_ITEM is deprecated and should be removed."
+#endif
+
+/**
+ * Probe temp compensation requirements
+ */
+#if ENABLED(PROBE_TEMP_COMPENSATION)
+ #if defined(PTC_PARK_POS_X) || defined(PTC_PARK_POS_Y) || defined(PTC_PARK_POS_Z)
+ #error "PTC_PARK_POS_[XYZ] is now PTC_PARK_POS (array)."
+ #elif !defined(PTC_PARK_POS)
+ #error "PROBE_TEMP_COMPENSATION requires PTC_PARK_POS."
+ #elif defined(PTC_PROBE_POS_X) || defined(PTC_PROBE_POS_Y)
+ #error "PTC_PROBE_POS_[XY] is now PTC_PROBE_POS (array)."
+ #elif !defined(PTC_PROBE_POS)
+ #error "PROBE_TEMP_COMPENSATION requires PTC_PROBE_POS."
+ #endif
+#endif
+
+/**
+ * Marlin release, version and default string
+ */
+#ifndef SHORT_BUILD_VERSION
+ #error "SHORT_BUILD_VERSION must be specified."
+#elif !defined(DETAILED_BUILD_VERSION)
+ #error "BUILD_VERSION must be specified."
+#elif !defined(STRING_DISTRIBUTION_DATE)
+ #error "STRING_DISTRIBUTION_DATE must be specified."
+#elif !defined(PROTOCOL_VERSION)
+ #error "PROTOCOL_VERSION must be specified."
+#elif !defined(MACHINE_NAME)
+ #error "MACHINE_NAME must be specified."
+#elif !defined(SOURCE_CODE_URL)
+ #error "SOURCE_CODE_URL must be specified."
+#elif !defined(DEFAULT_MACHINE_UUID)
+ #error "DEFAULT_MACHINE_UUID must be specified."
+#elif !defined(WEBSITE_URL)
+ #error "WEBSITE_URL must be specified."
+#endif
+
+/**
+ * Serial
+ */
+#ifndef SERIAL_PORT
+ #error "SERIAL_PORT must be defined."
+#elif defined(SERIAL_PORT_2) && SERIAL_PORT_2 == SERIAL_PORT
+ #error "SERIAL_PORT_2 cannot be the same as SERIAL_PORT."
+#endif
+#if !(defined(__AVR__) && defined(USBCON))
+ #if ENABLED(SERIAL_XON_XOFF) && RX_BUFFER_SIZE < 1024
+ #error "SERIAL_XON_XOFF requires RX_BUFFER_SIZE >= 1024 for reliable transfers without drops."
+ #elif RX_BUFFER_SIZE && (RX_BUFFER_SIZE < 2 || !IS_POWER_OF_2(RX_BUFFER_SIZE))
+ #error "RX_BUFFER_SIZE must be a power of 2 greater than 1."
+ #elif TX_BUFFER_SIZE && (TX_BUFFER_SIZE < 2 || TX_BUFFER_SIZE > 256 || !IS_POWER_OF_2(TX_BUFFER_SIZE))
+ #error "TX_BUFFER_SIZE must be 0 or a power of 2 between 1 and 256."
+ #endif
+#elif ANY(SERIAL_XON_XOFF, SERIAL_STATS_MAX_RX_QUEUED, SERIAL_STATS_DROPPED_RX)
+ #error "SERIAL_XON_XOFF and SERIAL_STATS_* features not supported on USB-native AVR devices."
+#endif
+
+/**
+ * Multiple Stepper Drivers Per Axis
+ */
+#define GOOD_AXIS_PINS(A) (HAS_##A##_ENABLE && HAS_##A##_STEP && HAS_##A##_DIR)
+#if ENABLED(X_DUAL_STEPPER_DRIVERS)
+ #if ENABLED(DUAL_X_CARRIAGE)
+ #error "DUAL_X_CARRIAGE is not compatible with X_DUAL_STEPPER_DRIVERS."
+ #elif !GOOD_AXIS_PINS(X)
+ #error "X_DUAL_STEPPER_DRIVERS requires X2 pins to be defined."
+ #endif
+#endif
+
+#if ENABLED(Y_DUAL_STEPPER_DRIVERS) && !GOOD_AXIS_PINS(Y)
+ #error "Y_DUAL_STEPPER_DRIVERS requires Y2 pins to be defined."
+#elif !WITHIN(NUM_Z_STEPPER_DRIVERS, 1, 4)
+ #error "NUM_Z_STEPPER_DRIVERS must be an integer from 1 to 4."
+#elif NUM_Z_STEPPER_DRIVERS == 2 && !GOOD_AXIS_PINS(Z2)
+ #error "If NUM_Z_STEPPER_DRIVERS is 2, you must define stepper pins for Z2."
+#elif NUM_Z_STEPPER_DRIVERS == 3 && !(GOOD_AXIS_PINS(Z2) && GOOD_AXIS_PINS(Z3))
+ #error "If NUM_Z_STEPPER_DRIVERS is 3, you must define stepper pins for Z2 and Z3."
+#elif NUM_Z_STEPPER_DRIVERS == 4 && !(GOOD_AXIS_PINS(Z2) && GOOD_AXIS_PINS(Z3) && GOOD_AXIS_PINS(Z4))
+ #error "If NUM_Z_STEPPER_DRIVERS is 4, you must define stepper pins for Z2, Z3, and Z4."
+#endif
+
+/**
+ * Validate that the bed size fits
+ */
+static_assert(X_MAX_LENGTH >= X_BED_SIZE, "Movement bounds (X_MIN_POS, X_MAX_POS) are too narrow to contain X_BED_SIZE.");
+static_assert(Y_MAX_LENGTH >= Y_BED_SIZE, "Movement bounds (Y_MIN_POS, Y_MAX_POS) are too narrow to contain Y_BED_SIZE.");
+
+/**
+ * Granular software endstops (Marlin >= 1.1.7)
+ */
+#if ENABLED(MIN_SOFTWARE_ENDSTOPS) && DISABLED(MIN_SOFTWARE_ENDSTOP_Z)
+ #if IS_KINEMATIC
+ #error "MIN_SOFTWARE_ENDSTOPS on DELTA/SCARA also requires MIN_SOFTWARE_ENDSTOP_Z."
+ #elif NONE(MIN_SOFTWARE_ENDSTOP_X, MIN_SOFTWARE_ENDSTOP_Y)
+ #error "MIN_SOFTWARE_ENDSTOPS requires at least one of the MIN_SOFTWARE_ENDSTOP_[XYZ] options."
+ #endif
+#endif
+
+#if ENABLED(MAX_SOFTWARE_ENDSTOPS) && DISABLED(MAX_SOFTWARE_ENDSTOP_Z)
+ #if IS_KINEMATIC
+ #error "MAX_SOFTWARE_ENDSTOPS on DELTA/SCARA also requires MAX_SOFTWARE_ENDSTOP_Z."
+ #elif NONE(MAX_SOFTWARE_ENDSTOP_X, MAX_SOFTWARE_ENDSTOP_Y)
+ #error "MAX_SOFTWARE_ENDSTOPS requires at least one of the MAX_SOFTWARE_ENDSTOP_[XYZ] options."
+ #endif
+#endif
+
+#if !defined(TARGET_LPC1768) && ANY( \
+ ENDSTOPPULLDOWNS, \
+ ENDSTOPPULLDOWN_XMAX, ENDSTOPPULLDOWN_YMAX, \
+ ENDSTOPPULLDOWN_ZMAX, ENDSTOPPULLDOWN_XMIN, \
+ ENDSTOPPULLDOWN_YMIN, ENDSTOPPULLDOWN_ZMIN \
+ )
+ #error "PULLDOWN pin mode is not available on the selected board."
+#endif
+
+#if BOTH(ENDSTOPPULLUPS, ENDSTOPPULLDOWNS)
+ #error "Enable only one of ENDSTOPPULLUPS or ENDSTOPPULLDOWNS."
+#elif BOTH(FIL_RUNOUT_PULLUP, FIL_RUNOUT_PULLDOWN)
+ #error "Enable only one of FIL_RUNOUT_PULLUP or FIL_RUNOUT_PULLDOWN."
+#elif BOTH(ENDSTOPPULLUP_XMAX, ENDSTOPPULLDOWN_XMAX)
+ #error "Enable only one of ENDSTOPPULLUP_X_MAX or ENDSTOPPULLDOWN_X_MAX."
+#elif BOTH(ENDSTOPPULLUP_YMAX, ENDSTOPPULLDOWN_YMAX)
+ #error "Enable only one of ENDSTOPPULLUP_Y_MAX or ENDSTOPPULLDOWN_Y_MAX."
+#elif BOTH(ENDSTOPPULLUP_ZMAX, ENDSTOPPULLDOWN_ZMAX)
+ #error "Enable only one of ENDSTOPPULLUP_Z_MAX or ENDSTOPPULLDOWN_Z_MAX."
+#elif BOTH(ENDSTOPPULLUP_XMIN, ENDSTOPPULLDOWN_XMIN)
+ #error "Enable only one of ENDSTOPPULLUP_X_MIN or ENDSTOPPULLDOWN_X_MIN."
+#elif BOTH(ENDSTOPPULLUP_YMIN, ENDSTOPPULLDOWN_YMIN)
+ #error "Enable only one of ENDSTOPPULLUP_Y_MIN or ENDSTOPPULLDOWN_Y_MIN."
+#elif BOTH(ENDSTOPPULLUP_ZMIN, ENDSTOPPULLDOWN_ZMIN)
+ #error "Enable only one of ENDSTOPPULLUP_Z_MIN or ENDSTOPPULLDOWN_Z_MIN."
+#endif
+
+/**
+ * LCD Info Screen Style
+ */
+#if LCD_INFO_SCREEN_STYLE > 0
+ #if HAS_MARLINUI_U8GLIB || LCD_WIDTH < 20 || LCD_HEIGHT < 4
+ #error "Alternative LCD_INFO_SCREEN_STYLE requires 20x4 Character LCD."
+ #elif LCD_INFO_SCREEN_STYLE > 1
+ #error "LCD_INFO_SCREEN_STYLE only has options 0 and 1 at this time."
+ #endif
+#endif
+
+/**
+ * Progress Bar
+ */
+#if ENABLED(LCD_PROGRESS_BAR)
+ #if NONE(SDSUPPORT, LCD_SET_PROGRESS_MANUALLY)
+ #error "LCD_PROGRESS_BAR requires SDSUPPORT or LCD_SET_PROGRESS_MANUALLY."
+ #elif NONE(HAS_MARLINUI_HD44780, IS_TFTGLCD_PANEL)
+ #error "LCD_PROGRESS_BAR only applies to HD44780 character LCD and TFTGLCD_PANEL_(SPI|I2C)."
+ #elif HAS_MARLINUI_U8GLIB
+ #error "LCD_PROGRESS_BAR does not apply to graphical displays."
+ #elif ENABLED(FILAMENT_LCD_DISPLAY)
+ #error "LCD_PROGRESS_BAR and FILAMENT_LCD_DISPLAY are not fully compatible. Comment out this line to use both."
+ #elif PROGRESS_MSG_EXPIRE < 0
+ #error "PROGRESS_MSG_EXPIRE must be greater than or equal to 0."
+ #endif
+#elif ENABLED(LCD_SET_PROGRESS_MANUALLY)
+ #if NONE(HAS_MARLINUI_U8GLIB, HAS_GRAPHICAL_TFT, HAS_MARLINUI_HD44780, EXTENSIBLE_UI)
+ #error "LCD_SET_PROGRESS_MANUALLY requires LCD_PROGRESS_BAR, Character LCD, Graphical LCD, TFT, or EXTENSIBLE_UI."
+ #endif
+#endif
+
+#if ENABLED(USE_M73_REMAINING_TIME) && DISABLED(LCD_SET_PROGRESS_MANUALLY)
+ #error "USE_M73_REMAINING_TIME requires LCD_SET_PROGRESS_MANUALLY"
+#endif
+
+#if !HAS_LCD_MENU && ENABLED(SD_REPRINT_LAST_SELECTED_FILE)
+ #error "SD_REPRINT_LAST_SELECTED_FILE currently requires a Marlin-native LCD menu."
+#endif
+
+/**
+ * Custom Boot and Status screens
+ */
+#if ENABLED(SHOW_CUSTOM_BOOTSCREEN) && NONE(HAS_MARLINUI_U8GLIB, TOUCH_UI_FTDI_EVE)
+ #error "SHOW_CUSTOM_BOOTSCREEN requires Graphical LCD or TOUCH_UI_FTDI_EVE."
+#elif ENABLED(CUSTOM_STATUS_SCREEN_IMAGE) && !HAS_MARLINUI_U8GLIB
+ #error "CUSTOM_STATUS_SCREEN_IMAGE requires a 128x64 DOGM B/W Graphical LCD."
+#endif
+
+/**
+ * LCD Lightweight Screen Style
+ */
+#if ENABLED(LIGHTWEIGHT_UI) && DISABLED(U8GLIB_ST7920)
+ #error "LIGHTWEIGHT_UI requires a U8GLIB_ST7920-based display."
+#endif
+
+/**
+ * SD File Sorting
+ */
+#if ENABLED(SDCARD_SORT_ALPHA)
+ #if SDSORT_LIMIT > 256
+ #error "SDSORT_LIMIT must be 256 or smaller."
+ #elif SDSORT_LIMIT < 10
+ #error "SDSORT_LIMIT should be greater than 9 to be useful."
+ #elif DISABLED(SDSORT_USES_RAM)
+ #if ENABLED(SDSORT_DYNAMIC_RAM)
+ #error "SDSORT_DYNAMIC_RAM requires SDSORT_USES_RAM (which reads the directory into RAM)."
+ #elif ENABLED(SDSORT_CACHE_NAMES)
+ #error "SDSORT_CACHE_NAMES requires SDSORT_USES_RAM (which reads the directory into RAM)."
+ #endif
+ #endif
+
+ #if ENABLED(SDSORT_CACHE_NAMES) && DISABLED(SDSORT_DYNAMIC_RAM)
+ #if SDSORT_CACHE_VFATS < 2
+ #error "SDSORT_CACHE_VFATS must be 2 or greater!"
+ #elif SDSORT_CACHE_VFATS > MAX_VFAT_ENTRIES
+ #undef SDSORT_CACHE_VFATS
+ #define SDSORT_CACHE_VFATS MAX_VFAT_ENTRIES
+ #warning "SDSORT_CACHE_VFATS was reduced to MAX_VFAT_ENTRIES!"
+ #endif
+ #endif
+#endif
+
+#if defined(EVENT_GCODE_SD_ABORT) && DISABLED(NOZZLE_PARK_FEATURE)
+ static_assert(nullptr == strstr(EVENT_GCODE_SD_ABORT, "G27"), "NOZZLE_PARK_FEATURE is required to use G27 in EVENT_GCODE_SD_ABORT.");
+#endif
+
+/**
+ * I2C Position Encoders
+ */
+#if ENABLED(I2C_POSITION_ENCODERS)
+ #if !BOTH(BABYSTEPPING, BABYSTEP_XY)
+ #error "I2C_POSITION_ENCODERS requires BABYSTEPPING and BABYSTEP_XY."
+ #elif !WITHIN(I2CPE_ENCODER_CNT, 1, 5)
+ #error "I2CPE_ENCODER_CNT must be between 1 and 5."
+ #endif
+#endif
+
+/**
+ * Babystepping
+ */
+#if ENABLED(BABYSTEPPING)
+ #if ENABLED(SCARA)
+ #error "BABYSTEPPING is not implemented for SCARA yet."
+ #elif BOTH(MARKFORGED_XY, BABYSTEP_XY)
+ #error "BABYSTEPPING only implemented for Z axis on MarkForged."
+ #elif BOTH(DELTA, BABYSTEP_XY)
+ #error "BABYSTEPPING only implemented for Z axis on deltabots."
+ #elif BOTH(BABYSTEP_ZPROBE_OFFSET, MESH_BED_LEVELING)
+ #error "MESH_BED_LEVELING and BABYSTEP_ZPROBE_OFFSET is not a valid combination"
+ #elif ENABLED(BABYSTEP_ZPROBE_OFFSET) && !HAS_BED_PROBE
+ #error "BABYSTEP_ZPROBE_OFFSET requires a probe."
+ #elif ENABLED(BABYSTEP_ZPROBE_GFX_OVERLAY) && !HAS_MARLINUI_U8GLIB
+ #error "BABYSTEP_ZPROBE_GFX_OVERLAY requires a Graphical LCD."
+ #elif ENABLED(BABYSTEP_ZPROBE_GFX_OVERLAY) && DISABLED(BABYSTEP_ZPROBE_OFFSET)
+ #error "BABYSTEP_ZPROBE_GFX_OVERLAY requires a BABYSTEP_ZPROBE_OFFSET."
+ #elif ENABLED(BABYSTEP_HOTEND_Z_OFFSET) && !HAS_HOTEND_OFFSET
+ #error "BABYSTEP_HOTEND_Z_OFFSET requires 2 or more HOTENDS."
+ #elif BOTH(BABYSTEP_ALWAYS_AVAILABLE, MOVE_Z_WHEN_IDLE)
+ #error "BABYSTEP_ALWAYS_AVAILABLE and MOVE_Z_WHEN_IDLE are incompatible."
+ #elif !defined(BABYSTEP_MULTIPLICATOR_Z)
+ #error "BABYSTEPPING requires BABYSTEP_MULTIPLICATOR_Z."
+ #elif ENABLED(BABYSTEP_XY) && !defined(BABYSTEP_MULTIPLICATOR_XY)
+ #error "BABYSTEP_XY requires BABYSTEP_MULTIPLICATOR_XY."
+ #elif ENABLED(BABYSTEP_MILLIMETER_UNITS)
+ static_assert(BABYSTEP_MULTIPLICATOR_Z <= 0.1f, "BABYSTEP_MULTIPLICATOR_Z must be less or equal to 0.1mm.");
+ #if ENABLED(BABYSTEP_XY)
+ static_assert(BABYSTEP_MULTIPLICATOR_XY <= 0.25f, "BABYSTEP_MULTIPLICATOR_XY must be less than or equal to 0.25mm.");
+ #endif
+ #endif
+#endif
+
+/**
+ * Filament Runout needs one or more pins and either SD Support or Auto print start detection
+ */
+#if HAS_FILAMENT_SENSOR
+ #if !PIN_EXISTS(FIL_RUNOUT)
+ #error "FILAMENT_RUNOUT_SENSOR requires FIL_RUNOUT_PIN."
+ #elif NUM_RUNOUT_SENSORS > E_STEPPERS
+ #if HAS_PRUSA_MMU2
+ #error "NUM_RUNOUT_SENSORS must be 1 with MMU2 / MMU2S."
+ #else
+ #error "NUM_RUNOUT_SENSORS cannot exceed the number of E steppers."
+ #endif
+ #elif NUM_RUNOUT_SENSORS >= 2 && !PIN_EXISTS(FIL_RUNOUT2)
+ #error "FIL_RUNOUT2_PIN is required with NUM_RUNOUT_SENSORS >= 2."
+ #elif NUM_RUNOUT_SENSORS >= 3 && !PIN_EXISTS(FIL_RUNOUT3)
+ #error "FIL_RUNOUT3_PIN is required with NUM_RUNOUT_SENSORS >= 3."
+ #elif NUM_RUNOUT_SENSORS >= 4 && !PIN_EXISTS(FIL_RUNOUT4)
+ #error "FIL_RUNOUT4_PIN is required with NUM_RUNOUT_SENSORS >= 4."
+ #elif NUM_RUNOUT_SENSORS >= 5 && !PIN_EXISTS(FIL_RUNOUT5)
+ #error "FIL_RUNOUT5_PIN is required with NUM_RUNOUT_SENSORS >= 5."
+ #elif NUM_RUNOUT_SENSORS >= 6 && !PIN_EXISTS(FIL_RUNOUT6)
+ #error "FIL_RUNOUT6_PIN is required with NUM_RUNOUT_SENSORS >= 6."
+ #elif NUM_RUNOUT_SENSORS >= 7 && !PIN_EXISTS(FIL_RUNOUT7)
+ #error "FIL_RUNOUT7_PIN is required with NUM_RUNOUT_SENSORS >= 7."
+ #elif NUM_RUNOUT_SENSORS >= 8 && !PIN_EXISTS(FIL_RUNOUT8)
+ #error "FIL_RUNOUT8_PIN is required with NUM_RUNOUT_SENSORS >= 8."
+ #elif BOTH(FIL_RUNOUT1_PULLUP, FIL_RUNOUT1_PULLDOWN)
+ #error "You can't enable FIL_RUNOUT1_PULLUP and FIL_RUNOUT1_PULLDOWN at the same time."
+ #elif BOTH(FIL_RUNOUT2_PULLUP, FIL_RUNOUT2_PULLDOWN)
+ #error "You can't enable FIL_RUNOUT2_PULLUP and FIL_RUNOUT2_PULLDOWN at the same time."
+ #elif BOTH(FIL_RUNOUT3_PULLUP, FIL_RUNOUT3_PULLDOWN)
+ #error "You can't enable FIL_RUNOUT3_PULLUP and FIL_RUNOUT3_PULLDOWN at the same time."
+ #elif BOTH(FIL_RUNOUT4_PULLUP, FIL_RUNOUT4_PULLDOWN)
+ #error "You can't enable FIL_RUNOUT4_PULLUP and FIL_RUNOUT4_PULLDOWN at the same time."
+ #elif BOTH(FIL_RUNOUT5_PULLUP, FIL_RUNOUT5_PULLDOWN)
+ #error "You can't enable FIL_RUNOUT5_PULLUP and FIL_RUNOUT5_PULLDOWN at the same time."
+ #elif BOTH(FIL_RUNOUT6_PULLUP, FIL_RUNOUT6_PULLDOWN)
+ #error "You can't enable FIL_RUNOUT6_PULLUP and FIL_RUNOUT6_PULLDOWN at the same time."
+ #elif BOTH(FIL_RUNOUT7_PULLUP, FIL_RUNOUT7_PULLDOWN)
+ #error "You can't enable FIL_RUNOUT7_PULLUP and FIL_RUNOUT7_PULLDOWN at the same time."
+ #elif BOTH(FIL_RUNOUT8_PULLUP, FIL_RUNOUT8_PULLDOWN)
+ #error "You can't enable FIL_RUNOUT8_PULLUP and FIL_RUNOUT8_PULLDOWN at the same time."
+ #elif FILAMENT_RUNOUT_DISTANCE_MM < 0
+ #error "FILAMENT_RUNOUT_DISTANCE_MM must be greater than or equal to zero."
+ #elif DISABLED(ADVANCED_PAUSE_FEATURE)
+ static_assert(nullptr == strstr(FILAMENT_RUNOUT_SCRIPT, "M600"), "ADVANCED_PAUSE_FEATURE is required to use M600 with FILAMENT_RUNOUT_SENSOR.");
+ #endif
+#endif
+
+/**
+ * Advanced Pause
+ */
+#if ENABLED(ADVANCED_PAUSE_FEATURE)
+ #if !HAS_RESUME_CONTINUE
+ #error "ADVANCED_PAUSE_FEATURE requires a supported LCD controller (or EMERGENCY_PARSER)."
+ #elif DISABLED(NOZZLE_PARK_FEATURE)
+ #error "ADVANCED_PAUSE_FEATURE requires NOZZLE_PARK_FEATURE."
+ #elif !defined(FILAMENT_UNLOAD_PURGE_FEEDRATE)
+ #error "ADVANCED_PAUSE_FEATURE requires FILAMENT_UNLOAD_PURGE_FEEDRATE."
+ #elif ENABLED(EXTRUDER_RUNOUT_PREVENT)
+ #error "EXTRUDER_RUNOUT_PREVENT is incompatible with ADVANCED_PAUSE_FEATURE."
+ #elif ENABLED(PARK_HEAD_ON_PAUSE) && NONE(SDSUPPORT, IS_NEWPANEL, EMERGENCY_PARSER)
+ #error "PARK_HEAD_ON_PAUSE requires SDSUPPORT, EMERGENCY_PARSER, or an LCD controller."
+ #elif ENABLED(HOME_BEFORE_FILAMENT_CHANGE) && DISABLED(PAUSE_PARK_NO_STEPPER_TIMEOUT)
+ #error "HOME_BEFORE_FILAMENT_CHANGE requires PAUSE_PARK_NO_STEPPER_TIMEOUT."
+ #elif ENABLED(PREVENT_LENGTHY_EXTRUDE) && FILAMENT_CHANGE_UNLOAD_LENGTH > EXTRUDE_MAXLENGTH
+ #error "FILAMENT_CHANGE_UNLOAD_LENGTH must be less than or equal to EXTRUDE_MAXLENGTH."
+ #elif ENABLED(PREVENT_LENGTHY_EXTRUDE) && FILAMENT_CHANGE_SLOW_LOAD_LENGTH > EXTRUDE_MAXLENGTH
+ #error "FILAMENT_CHANGE_SLOW_LOAD_LENGTH must be less than or equal to EXTRUDE_MAXLENGTH."
+ #elif ENABLED(PREVENT_LENGTHY_EXTRUDE) && FILAMENT_CHANGE_FAST_LOAD_LENGTH > EXTRUDE_MAXLENGTH
+ #error "FILAMENT_CHANGE_FAST_LOAD_LENGTH must be less than or equal to EXTRUDE_MAXLENGTH."
+ #endif
+#endif
+
+#if ENABLED(NOZZLE_PARK_FEATURE)
+ constexpr float npp[] = NOZZLE_PARK_POINT;
+ static_assert(COUNT(npp) == XYZ, "NOZZLE_PARK_POINT requires X, Y, and Z values.");
+ constexpr xyz_pos_t npp_xyz = NOZZLE_PARK_POINT;
+ static_assert(WITHIN(npp_xyz.x, X_MIN_POS, X_MAX_POS), "NOZZLE_PARK_POINT.X is out of bounds (X_MIN_POS, X_MAX_POS).");
+ static_assert(WITHIN(npp_xyz.y, Y_MIN_POS, Y_MAX_POS), "NOZZLE_PARK_POINT.Y is out of bounds (Y_MIN_POS, Y_MAX_POS).");
+ static_assert(WITHIN(npp_xyz.z, Z_MIN_POS, Z_MAX_POS), "NOZZLE_PARK_POINT.Z is out of bounds (Z_MIN_POS, Z_MAX_POS).");
+#endif
+
+/**
+ * Individual axis homing is useless for DELTAS
+ */
+#if BOTH(INDIVIDUAL_AXIS_HOMING_MENU, DELTA)
+ #error "INDIVIDUAL_AXIS_HOMING_MENU is incompatible with DELTA kinematics."
+#endif
+
+/**
+ * Sanity checking for all Průša MMU
+ */
+#ifdef SNMM
+ #error "SNMM is obsolete. Define MMU_MODEL as PRUSA_MMU1 instead."
+#elif ENABLED(MK2_MULTIPLEXER)
+ #error "MK2_MULTIPLEXER is obsolete. Define MMU_MODEL as PRUSA_MMU1 instead."
+#elif ENABLED(PRUSA_MMU2)
+ #error "PRUSA_MMU2 is obsolete. Define MMU_MODEL as PRUSA_MMU2 instead."
+#elif ENABLED(PRUSA_MMU2_S_MODE)
+ #error "PRUSA_MMU2_S_MODE is obsolete. Define MMU_MODEL as PRUSA_MMU2S instead."
+#endif
+
+/**
+ * Multi-Material-Unit 2 / SMUFF requirements
+ */
+#if HAS_PRUSA_MMU2
+ #if EXTRUDERS != 5
+ #undef SINGLENOZZLE
+ #error "PRUSA_MMU2(S) requires exactly 5 EXTRUDERS. Please update your Configuration."
+ #elif DISABLED(NOZZLE_PARK_FEATURE)
+ #error "PRUSA_MMU2(S) requires NOZZLE_PARK_FEATURE. Enable it to continue."
+ #elif HAS_PRUSA_MMU2S && DISABLED(FILAMENT_RUNOUT_SENSOR)
+ #error "PRUSA_MMU2S requires FILAMENT_RUNOUT_SENSOR. Enable it to continue."
+ #elif ENABLED(MMU_EXTRUDER_SENSOR) && DISABLED(FILAMENT_RUNOUT_SENSOR)
+ #error "MMU_EXTRUDER_SENSOR requires FILAMENT_RUNOUT_SENSOR. Enable it to continue."
+ #elif ENABLED(MMU_EXTRUDER_SENSOR) && !HAS_LCD_MENU
+ #error "MMU_EXTRUDER_SENSOR requires an LCD supporting MarlinUI to be enabled."
+ #elif DISABLED(ADVANCED_PAUSE_FEATURE)
+ static_assert(nullptr == strstr(MMU2_FILAMENT_RUNOUT_SCRIPT, "M600"), "ADVANCED_PAUSE_FEATURE is required to use M600 with PRUSA_MMU2(S) / SMUFF_EMU_MMU2(S).");
+ #endif
+#endif
+#if HAS_SMUFF && EXTRUDERS > 12
+ #error "Too many extruders for SMUFF_EMU_MMU2(S). (12 maximum)."
+#endif
+
+/**
+ * Options only for EXTRUDERS > 1
+ */
+#if HAS_MULTI_EXTRUDER
+
+ #if EXTRUDERS > 8
+ #error "Marlin supports a maximum of 8 EXTRUDERS."
+ #endif
+
+ #if ENABLED(HEATERS_PARALLEL)
+ #error "EXTRUDERS must be 1 with HEATERS_PARALLEL."
+ #endif
+
+ #if ENABLED(TOOLCHANGE_FILAMENT_SWAP)
+ #ifndef TOOLCHANGE_FS_LENGTH
+ #error "TOOLCHANGE_FILAMENT_SWAP requires TOOLCHANGE_FS_LENGTH."
+ #elif !defined(TOOLCHANGE_FS_RETRACT_SPEED)
+ #error "TOOLCHANGE_FILAMENT_SWAP requires TOOLCHANGE_FS_RETRACT_SPEED."
+ #elif !defined(TOOLCHANGE_FS_PRIME_SPEED)
+ #error "TOOLCHANGE_FILAMENT_SWAP requires TOOLCHANGE_FS_PRIME_SPEED."
+ #endif
+ #endif
+
+ #if ENABLED(TOOLCHANGE_PARK)
+ #ifndef TOOLCHANGE_PARK_XY
+ #error "TOOLCHANGE_PARK requires TOOLCHANGE_PARK_XY."
+ #elif !defined(TOOLCHANGE_PARK_XY_FEEDRATE)
+ #error "TOOLCHANGE_PARK requires TOOLCHANGE_PARK_XY_FEEDRATE."
+ #endif
+ #endif
+
+ #ifndef TOOLCHANGE_ZRAISE
+ #error "TOOLCHANGE_ZRAISE required for EXTRUDERS > 1."
+ #endif
+
+#elif HAS_PRUSA_MMU1 || HAS_SMUFF
+
+ #error "Multi-Material-Unit requires 2 or more EXTRUDERS."
+
+#elif ENABLED(SINGLENOZZLE)
+
+ #error "SINGLENOZZLE requires 2 or more EXTRUDERS."
+
+#endif
+
+/**
+ * A Dual Nozzle carriage with switching servo
+ */
+#if ENABLED(SWITCHING_NOZZLE)
+ #if ENABLED(DUAL_X_CARRIAGE)
+ #error "SWITCHING_NOZZLE and DUAL_X_CARRIAGE are incompatible."
+ #elif ENABLED(SINGLENOZZLE)
+ #error "SWITCHING_NOZZLE and SINGLENOZZLE are incompatible."
+ #elif EXTRUDERS != 2
+ #error "SWITCHING_NOZZLE requires exactly 2 EXTRUDERS."
+ #elif NUM_SERVOS < 1
+ #error "SWITCHING_NOZZLE requires NUM_SERVOS >= 1."
+ #endif
+
+ #ifndef SWITCHING_NOZZLE_SERVO_NR
+ #error "SWITCHING_NOZZLE requires SWITCHING_NOZZLE_SERVO_NR."
+ #elif SWITCHING_NOZZLE_SERVO_NR == 0 && !PIN_EXISTS(SERVO0)
+ #error "SERVO0_PIN must be defined for your SWITCHING_NOZZLE."
+ #elif SWITCHING_NOZZLE_SERVO_NR == 1 && !PIN_EXISTS(SERVO1)
+ #error "SERVO1_PIN must be defined for your SWITCHING_NOZZLE."
+ #elif SWITCHING_NOZZLE_SERVO_NR == 2 && !PIN_EXISTS(SERVO2)
+ #error "SERVO2_PIN must be defined for your SWITCHING_NOZZLE."
+ #elif SWITCHING_NOZZLE_SERVO_NR == 3 && !PIN_EXISTS(SERVO3)
+ #error "SERVO3_PIN must be defined for your SWITCHING_NOZZLE."
+ #endif
+
+ #ifdef SWITCHING_NOZZLE_E1_SERVO_NR
+ #if SWITCHING_NOZZLE_E1_SERVO_NR == SWITCHING_NOZZLE_SERVO_NR
+ #error "SWITCHING_NOZZLE_E1_SERVO_NR must be different from SWITCHING_NOZZLE_SERVO_NR."
+ #elif SWITCHING_NOZZLE_E1_SERVO_NR == 0 && !PIN_EXISTS(SERVO0)
+ #error "SERVO0_PIN must be defined for your SWITCHING_NOZZLE."
+ #elif SWITCHING_NOZZLE_E1_SERVO_NR == 1 && !PIN_EXISTS(SERVO1)
+ #error "SERVO1_PIN must be defined for your SWITCHING_NOZZLE."
+ #elif SWITCHING_NOZZLE_E1_SERVO_NR == 2 && !PIN_EXISTS(SERVO2)
+ #error "SERVO2_PIN must be defined for your SWITCHING_NOZZLE."
+ #elif SWITCHING_NOZZLE_E1_SERVO_NR == 3 && !PIN_EXISTS(SERVO3)
+ #error "SERVO3_PIN must be defined for your SWITCHING_NOZZLE."
+ #endif
+ #endif
+#endif
+
+/**
+ * Single Stepper Dual Extruder with switching servo
+ */
+#if ENABLED(SWITCHING_EXTRUDER)
+ #if NUM_SERVOS < 1
+ #error "SWITCHING_EXTRUDER requires NUM_SERVOS >= 1."
+ #elif SWITCHING_EXTRUDER_SERVO_NR == 0 && !PIN_EXISTS(SERVO0)
+ #error "SERVO0_PIN must be defined for your SWITCHING_EXTRUDER."
+ #elif SWITCHING_EXTRUDER_SERVO_NR == 1 && !PIN_EXISTS(SERVO1)
+ #error "SERVO1_PIN must be defined for your SWITCHING_EXTRUDER."
+ #elif SWITCHING_EXTRUDER_SERVO_NR == 2 && !PIN_EXISTS(SERVO2)
+ #error "SERVO2_PIN must be defined for your SWITCHING_EXTRUDER."
+ #elif SWITCHING_EXTRUDER_SERVO_NR == 3 && !PIN_EXISTS(SERVO3)
+ #error "SERVO3_PIN must be defined for your SWITCHING_EXTRUDER."
+ #endif
+ #if EXTRUDERS > 3
+ #if NUM_SERVOS < 2
+ #error "SWITCHING_EXTRUDER with 4 extruders requires NUM_SERVOS >= 2."
+ #elif SWITCHING_EXTRUDER_E23_SERVO_NR == 0 && !PIN_EXISTS(SERVO0)
+ #error "SERVO0_PIN must be defined for your SWITCHING_EXTRUDER."
+ #elif SWITCHING_EXTRUDER_E23_SERVO_NR == 1 && !PIN_EXISTS(SERVO1)
+ #error "SERVO1_PIN must be defined for your SWITCHING_EXTRUDER."
+ #elif SWITCHING_EXTRUDER_E23_SERVO_NR == 2 && !PIN_EXISTS(SERVO2)
+ #error "SERVO2_PIN must be defined for your SWITCHING_EXTRUDER."
+ #elif SWITCHING_EXTRUDER_E23_SERVO_NR == 3 && !PIN_EXISTS(SERVO3)
+ #error "SERVO3_PIN must be defined for your SWITCHING_EXTRUDER."
+ #elif SWITCHING_EXTRUDER_E23_SERVO_NR == SWITCHING_EXTRUDER_SERVO_NR
+ #error "SWITCHING_EXTRUDER_E23_SERVO_NR should be a different extruder from SWITCHING_EXTRUDER_SERVO_NR."
+ #endif
+ #endif
+#endif
+
+/**
+ * Mixing Extruder requirements
+ */
+#if ENABLED(MIXING_EXTRUDER)
+ #if HAS_MULTI_EXTRUDER
+ #error "For MIXING_EXTRUDER set MIXING_STEPPERS > 1 instead of EXTRUDERS > 1."
+ #elif MIXING_STEPPERS < 2
+ #error "You must set MIXING_STEPPERS >= 2 for a mixing extruder."
+ #elif ENABLED(FILAMENT_SENSOR)
+ #error "MIXING_EXTRUDER is incompatible with FILAMENT_SENSOR. Comment out this line to use it anyway."
+ #elif ENABLED(SWITCHING_EXTRUDER)
+ #error "Please select either MIXING_EXTRUDER or SWITCHING_EXTRUDER, not both."
+ #elif ENABLED(SINGLENOZZLE)
+ #error "MIXING_EXTRUDER is incompatible with SINGLENOZZLE."
+ #endif
+#endif
+
+/**
+ * Linear Advance 1.5 - Check K value range
+ */
+#if ENABLED(LIN_ADVANCE)
+ static_assert(
+ WITHIN(LIN_ADVANCE_K, 0, 10),
+ "LIN_ADVANCE_K must be a value from 0 to 10 (Changed in LIN_ADVANCE v1.5, Marlin 1.1.9)."
+ );
+ #if ENABLED(S_CURVE_ACCELERATION) && DISABLED(EXPERIMENTAL_SCURVE)
+ #error "LIN_ADVANCE and S_CURVE_ACCELERATION may not play well together! Enable EXPERIMENTAL_SCURVE to continue."
+ #endif
+#endif
+
+/**
+ * Special tool-changing options
+ */
+#if MANY(SINGLENOZZLE, DUAL_X_CARRIAGE, PARKING_EXTRUDER, MAGNETIC_PARKING_EXTRUDER, SWITCHING_TOOLHEAD, MAGNETIC_SWITCHING_TOOLHEAD, ELECTROMAGNETIC_SWITCHING_TOOLHEAD)
+ #error "Please select only one of SINGLENOZZLE, DUAL_X_CARRIAGE, PARKING_EXTRUDER, MAGNETIC_PARKING_EXTRUDER, SWITCHING_TOOLHEAD, MAGNETIC_SWITCHING_TOOLHEAD, or ELECTROMAGNETIC_SWITCHING_TOOLHEAD."
+#endif
+
+/**
+ * (Magnetic) Parking Extruder requirements
+ */
+#if ANY(PARKING_EXTRUDER, MAGNETIC_PARKING_EXTRUDER)
+ #if ENABLED(EXT_SOLENOID)
+ #error "(MAGNETIC_)PARKING_EXTRUDER and EXT_SOLENOID are incompatible. (Pins are used twice.)"
+ #elif EXTRUDERS != 2
+ #error "(MAGNETIC_)PARKING_EXTRUDER requires exactly 2 EXTRUDERS."
+ #elif !defined(PARKING_EXTRUDER_PARKING_X)
+ #error "(MAGNETIC_)PARKING_EXTRUDER requires PARKING_EXTRUDER_PARKING_X."
+ #elif !defined(TOOLCHANGE_ZRAISE)
+ #error "(MAGNETIC_)PARKING_EXTRUDER requires TOOLCHANGE_ZRAISE."
+ #elif TOOLCHANGE_ZRAISE < 0
+ #error "TOOLCHANGE_ZRAISE must be 0 or higher."
+ #elif ENABLED(PARKING_EXTRUDER)
+ #if !PINS_EXIST(SOL0, SOL1)
+ #error "PARKING_EXTRUDER requires SOL0_PIN and SOL1_PIN."
+ #elif !defined(PARKING_EXTRUDER_SOLENOIDS_PINS_ACTIVE) || !WITHIN(PARKING_EXTRUDER_SOLENOIDS_PINS_ACTIVE, LOW, HIGH)
+ #error "PARKING_EXTRUDER_SOLENOIDS_PINS_ACTIVE must be defined as HIGH or LOW."
+ #elif !defined(PARKING_EXTRUDER_SOLENOIDS_DELAY) || !WITHIN(PARKING_EXTRUDER_SOLENOIDS_DELAY, 0, 2000)
+ #error "PARKING_EXTRUDER_SOLENOIDS_DELAY must be between 0 and 2000 (ms)."
+ #endif
+ #endif
+#endif
+
+/**
+ * Switching Toolhead requirements
+ */
+#if ENABLED(SWITCHING_TOOLHEAD)
+ #ifndef SWITCHING_TOOLHEAD_SERVO_NR
+ #error "SWITCHING_TOOLHEAD requires SWITCHING_TOOLHEAD_SERVO_NR."
+ #elif EXTRUDERS < 2
+ #error "SWITCHING_TOOLHEAD requires at least 2 EXTRUDERS."
+ #elif NUM_SERVOS < (SWITCHING_TOOLHEAD_SERVO_NR - 1)
+ #if SWITCHING_TOOLHEAD_SERVO_NR == 0
+ #error "A SWITCHING_TOOLHEAD_SERVO_NR of 0 requires NUM_SERVOS >= 1."
+ #elif SWITCHING_TOOLHEAD_SERVO_NR == 1
+ #error "A SWITCHING_TOOLHEAD_SERVO_NR of 1 requires NUM_SERVOS >= 2."
+ #elif SWITCHING_TOOLHEAD_SERVO_NR == 2
+ #error "A SWITCHING_TOOLHEAD_SERVO_NR of 2 requires NUM_SERVOS >= 3."
+ #elif SWITCHING_TOOLHEAD_SERVO_NR == 3
+ #error "A SWITCHING_TOOLHEAD_SERVO_NR of 3 requires NUM_SERVOS >= 4."
+ #endif
+ #elif !defined(TOOLCHANGE_ZRAISE)
+ #error "SWITCHING_TOOLHEAD requires TOOLCHANGE_ZRAISE."
+ #elif TOOLCHANGE_ZRAISE < 0
+ #error "TOOLCHANGE_ZRAISE must be 0 or higher."
+ #endif
+#endif
+
+/**
+ * (Electro)magnetic Switching Toolhead requirements
+ */
+#if EITHER(MAGNETIC_SWITCHING_TOOLHEAD, ELECTROMAGNETIC_SWITCHING_TOOLHEAD)
+ #ifndef SWITCHING_TOOLHEAD_Y_POS
+ #error "(ELECTRO)MAGNETIC_SWITCHING_TOOLHEAD requires SWITCHING_TOOLHEAD_Y_POS"
+ #elif !defined(SWITCHING_TOOLHEAD_X_POS)
+ #error "(ELECTRO)MAGNETIC_SWITCHING_TOOLHEAD requires SWITCHING_TOOLHEAD_X_POS"
+ #elif !defined(SWITCHING_TOOLHEAD_Z_HOP)
+ #error "(ELECTRO)MAGNETIC_SWITCHING_TOOLHEAD requires SWITCHING_TOOLHEAD_Z_HOP."
+ #elif !defined(SWITCHING_TOOLHEAD_Y_CLEAR)
+ #error "(ELECTRO)MAGNETIC_SWITCHING_TOOLHEAD requires SWITCHING_TOOLHEAD_Y_CLEAR."
+ #elif ENABLED(ELECTROMAGNETIC_SWITCHING_TOOLHEAD)
+ #if ENABLED(EXT_SOLENOID)
+ #error "(ELECTRO)MAGNETIC_SWITCHING_TOOLHEAD and EXT_SOLENOID are incompatible. (Pins are used twice.)"
+ #elif !PIN_EXISTS(SOL0)
+ #error "(ELECTRO)MAGNETIC_SWITCHING_TOOLHEAD requires SOL0_PIN."
+ #endif
+ #endif
+#endif
+
+/**
+ * Part-Cooling Fan Multiplexer requirements
+ */
+#if PIN_EXISTS(FANMUX1)
+ #if !HAS_FANMUX
+ #error "FANMUX0_PIN must be set before FANMUX1_PIN can be set."
+ #endif
+#elif PIN_EXISTS(FANMUX2)
+ #error "FANMUX0_PIN and FANMUX1_PIN must be set before FANMUX2_PIN can be set."
+#endif
+
+/**
+ * Limited user-controlled fans
+ */
+#if NUM_M106_FANS > FAN_COUNT
+ #error "The selected board doesn't support enough user-controlled fans. Reduce NUM_M106_FANS."
+#endif
+
+/**
+ * Limited number of servos
+ */
+#if NUM_SERVOS > NUM_SERVO_PLUGS
+ #error "The selected board doesn't support enough servos for your configuration. Reduce NUM_SERVOS."
+#endif
+
+/**
+ * Servo deactivation depends on servo endstops, switching nozzle, or switching extruder
+ */
+#if ENABLED(DEACTIVATE_SERVOS_AFTER_MOVE) && !HAS_Z_SERVO_PROBE && !defined(SWITCHING_NOZZLE_SERVO_NR) && !defined(SWITCHING_EXTRUDER_SERVO_NR) && !defined(SWITCHING_TOOLHEAD_SERVO_NR)
+ #error "Z_PROBE_SERVO_NR, switching nozzle, switching toolhead or switching extruder is required for DEACTIVATE_SERVOS_AFTER_MOVE."
+#endif
+
+/**
+ * Required LCD language
+ */
+#if HAS_MARLINUI_HD44780 && !defined(DISPLAY_CHARSET_HD44780)
+ #error "You must set DISPLAY_CHARSET_HD44780 to JAPANESE, WESTERN or CYRILLIC for your LCD controller."
+#endif
+
+/**
+ * Bed Heating Options - PID vs Limit Switching
+ */
+#if BOTH(PIDTEMPBED, BED_LIMIT_SWITCHING)
+ #error "To use BED_LIMIT_SWITCHING you must disable PIDTEMPBED."
+#endif
+
+/**
+ * Kinematics
+ */
+
+/**
+ * Allow only one kinematic type to be defined
+ */
+#if MANY(DELTA, MORGAN_SCARA, COREXY, COREXZ, COREYZ, COREYX, COREZX, COREZY, MARKFORGED_XY)
+ #error "Please enable only one of DELTA, MORGAN_SCARA, COREXY, COREYX, COREXZ, COREZX, COREYZ, COREZY, or MARKFORGED_XY."
+#endif
+
+/**
+ * Delta requirements
+ */
+#if ENABLED(DELTA)
+ #if NONE(USE_XMAX_PLUG, USE_YMAX_PLUG, USE_ZMAX_PLUG)
+ #error "You probably want to use Max Endstops for DELTA!"
+ #elif ENABLED(ENABLE_LEVELING_FADE_HEIGHT) && DISABLED(AUTO_BED_LEVELING_BILINEAR) && !UBL_SEGMENTED
+ #error "ENABLE_LEVELING_FADE_HEIGHT on DELTA requires AUTO_BED_LEVELING_BILINEAR or AUTO_BED_LEVELING_UBL."
+ #elif ENABLED(DELTA_AUTO_CALIBRATION) && !(HAS_BED_PROBE || HAS_LCD_MENU)
+ #error "DELTA_AUTO_CALIBRATION requires a probe or LCD Controller."
+ #elif ENABLED(DELTA_CALIBRATION_MENU) && !HAS_LCD_MENU
+ #error "DELTA_CALIBRATION_MENU requires an LCD Controller."
+ #elif ABL_GRID
+ #if (GRID_MAX_POINTS_X & 1) == 0 || (GRID_MAX_POINTS_Y & 1) == 0
+ #error "DELTA requires GRID_MAX_POINTS_X and GRID_MAX_POINTS_Y to be odd numbers."
+ #elif GRID_MAX_POINTS_X < 3
+ #error "DELTA requires GRID_MAX_POINTS_X and GRID_MAX_POINTS_Y to be 3 or higher."
+ #endif
+ #endif
+#endif
+
+/**
+ * Junction deviation is incompatible with kinematic systems.
+ */
+#if HAS_JUNCTION_DEVIATION && IS_KINEMATIC
+ #error "CLASSIC_JERK is required for DELTA and SCARA."
+#endif
+
+/**
+ * Probes
+ */
+
+/**
+ * Allow only one probe option to be defined
+ */
+#if 1 < 0 \
+ + (DISABLED(BLTOUCH) && HAS_Z_SERVO_PROBE) \
+ + COUNT_ENABLED(PROBE_MANUALLY, BLTOUCH, FIX_MOUNTED_PROBE, NOZZLE_AS_PROBE, TOUCH_MI_PROBE, SOLENOID_PROBE, Z_PROBE_ALLEN_KEY, Z_PROBE_SLED, RACK_AND_PINION_PROBE, SENSORLESS_PROBING)
+ #error "Please enable only one probe option: PROBE_MANUALLY, SENSORLESS_PROBING, BLTOUCH, FIX_MOUNTED_PROBE, NOZZLE_AS_PROBE, TOUCH_MI_PROBE, SOLENOID_PROBE, Z_PROBE_ALLEN_KEY, Z_PROBE_SLED, or Z Servo."
+#endif
+
+#if HAS_BED_PROBE
+
+ /**
+ * Z_PROBE_SLED is incompatible with DELTA
+ */
+ #if BOTH(Z_PROBE_SLED, DELTA)
+ #error "You cannot use Z_PROBE_SLED with DELTA."
+ #endif
+
+ /**
+ * SOLENOID_PROBE requirements
+ */
+ #if ENABLED(SOLENOID_PROBE)
+ #if ENABLED(EXT_SOLENOID)
+ #error "SOLENOID_PROBE is incompatible with EXT_SOLENOID."
+ #elif !HAS_SOLENOID_1
+ #error "SOLENOID_PROBE requires SOL1_PIN."
+ #endif
+ #endif
+
+ /**
+ * NUM_SERVOS is required for a Z servo probe
+ */
+ #if HAS_Z_SERVO_PROBE
+ #if !NUM_SERVOS
+ #error "NUM_SERVOS is required for a Z servo probe (Z_PROBE_SERVO_NR)."
+ #elif Z_PROBE_SERVO_NR >= NUM_SERVOS
+ #error "Z_PROBE_SERVO_NR must be smaller than NUM_SERVOS."
+ #elif Z_PROBE_SERVO_NR == 0 && !PIN_EXISTS(SERVO0)
+ #error "SERVO0_PIN must be defined for your servo or BLTOUCH probe."
+ #elif Z_PROBE_SERVO_NR == 1 && !PIN_EXISTS(SERVO1)
+ #error "SERVO1_PIN must be defined for your servo or BLTOUCH probe."
+ #elif Z_PROBE_SERVO_NR == 2 && !PIN_EXISTS(SERVO2)
+ #error "SERVO2_PIN must be defined for your servo or BLTOUCH probe."
+ #elif Z_PROBE_SERVO_NR == 3 && !PIN_EXISTS(SERVO3)
+ #error "SERVO3_PIN must be defined for your servo or BLTOUCH probe."
+ #endif
+ #endif
+
+ #if ENABLED(BLTOUCH)
+ #if BLTOUCH_DELAY < 200
+ #error "BLTOUCH_DELAY less than 200 is unsafe and is not supported."
+ #elif DISABLED(BLTOUCH_SET_5V_MODE) && NONE(ONBOARD_ENDSTOPPULLUPS, ENDSTOPPULLUPS, ENDSTOPPULLUP_ZMIN, ENDSTOPPULLUP_ZMIN_PROBE)
+ #error "BLTOUCH without BLTOUCH_SET_5V_MODE requires ENDSTOPPULLUPS, ENDSTOPPULLUP_ZMIN or ENDSTOPPULLUP_ZMIN_PROBE."
+ #endif
+ #endif
+
+ #if ENABLED(RACK_AND_PINION_PROBE) && !(defined(Z_PROBE_DEPLOY_X) && defined(Z_PROBE_RETRACT_X))
+ #error "RACK_AND_PINION_PROBE requires Z_PROBE_DEPLOY_X and Z_PROBE_RETRACT_X."
+ #endif
+
+ /**
+ * Touch-MI probe requirements
+ */
+ #if ENABLED(TOUCH_MI_PROBE)
+ #if DISABLED(Z_SAFE_HOMING)
+ #error "TOUCH_MI_PROBE requires Z_SAFE_HOMING."
+ #elif !defined(TOUCH_MI_RETRACT_Z)
+ #error "TOUCH_MI_PROBE requires TOUCH_MI_RETRACT_Z."
+ #elif defined(Z_AFTER_PROBING)
+ #error "TOUCH_MI_PROBE requires Z_AFTER_PROBING to be disabled."
+ #elif Z_HOMING_HEIGHT < 10
+ #error "TOUCH_MI_PROBE requires Z_HOMING_HEIGHT >= 10."
+ #elif Z_MIN_PROBE_ENDSTOP_INVERTING
+ #error "TOUCH_MI_PROBE requires Z_MIN_PROBE_ENDSTOP_INVERTING to be set to false."
+ #elif DISABLED(BABYSTEP_ZPROBE_OFFSET)
+ #error "TOUCH_MI_PROBE requires BABYSTEPPING with BABYSTEP_ZPROBE_OFFSET."
+ #elif !HAS_RESUME_CONTINUE
+ #error "TOUCH_MI_PROBE currently requires an LCD controller or EMERGENCY_PARSER."
+ #endif
+ #endif
+
+ /**
+ * Require pin options and pins to be defined
+ */
+ #if ENABLED(SENSORLESS_PROBING)
+ #if ENABLED(DELTA) && !(AXIS_HAS_STALLGUARD(X) && AXIS_HAS_STALLGUARD(Y) && AXIS_HAS_STALLGUARD(Z))
+ #error "SENSORLESS_PROBING requires TMC2130/2160/2209/5130/5160 drivers on X, Y, and Z."
+ #elif !AXIS_HAS_STALLGUARD(Z)
+ #error "SENSORLESS_PROBING requires a TMC2130/2160/2209/5130/5160 driver on Z."
+ #endif
+ #elif ENABLED(Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN)
+ #if DISABLED(USE_ZMIN_PLUG)
+ #error "Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN requires USE_ZMIN_PLUG to be enabled."
+ #elif !HAS_Z_MIN
+ #error "Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN requires the Z_MIN_PIN to be defined."
+ #elif Z_MIN_PROBE_ENDSTOP_INVERTING != Z_MIN_ENDSTOP_INVERTING
+ #error "Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN requires Z_MIN_ENDSTOP_INVERTING to match Z_MIN_PROBE_ENDSTOP_INVERTING."
+ #endif
+ #elif !HAS_Z_MIN_PROBE_PIN
+ #error "Z_MIN_PROBE_PIN must be defined if Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN is not enabled."
+ #endif
+
+ /**
+ * Check for improper NOZZLE_TO_PROBE_OFFSET
+ */
+ constexpr xyz_pos_t sanity_nozzle_to_probe_offset = NOZZLE_TO_PROBE_OFFSET;
+ #if ENABLED(NOZZLE_AS_PROBE)
+ static_assert(sanity_nozzle_to_probe_offset.x == 0 && sanity_nozzle_to_probe_offset.y == 0,
+ "NOZZLE_AS_PROBE requires the XY offsets in NOZZLE_TO_PROBE_OFFSET to both be 0.");
+ #else
+ static_assert(PROBING_MARGIN >= 0, "PROBING_MARGIN must be >= 0.");
+ static_assert(PROBING_MARGIN_BACK >= 0, "PROBING_MARGIN_BACK must be >= 0.");
+ static_assert(PROBING_MARGIN_FRONT >= 0, "PROBING_MARGIN_FRONT must be >= 0.");
+ static_assert(PROBING_MARGIN_LEFT >= 0, "PROBING_MARGIN_LEFT must be >= 0.");
+ static_assert(PROBING_MARGIN_RIGHT >= 0, "PROBING_MARGIN_RIGHT must be >= 0.");
+ #endif
+
+ #define _MARGIN(A) TERN(IS_SCARA, SCARA_PRINTABLE_RADIUS, TERN(DELTA, DELTA_PRINTABLE_RADIUS, ((A##_BED_SIZE) / 2)))
+ static_assert(PROBING_MARGIN < _MARGIN(X), "PROBING_MARGIN is too large.");
+ static_assert(PROBING_MARGIN_BACK < _MARGIN(Y), "PROBING_MARGIN_BACK is too large.");
+ static_assert(PROBING_MARGIN_FRONT < _MARGIN(Y), "PROBING_MARGIN_FRONT is too large.");
+ static_assert(PROBING_MARGIN_LEFT < _MARGIN(X), "PROBING_MARGIN_LEFT is too large.");
+ static_assert(PROBING_MARGIN_RIGHT < _MARGIN(X), "PROBING_MARGIN_RIGHT is too large.");
+ #undef _MARGIN
+
+ /**
+ * Make sure Z raise values are set
+ */
+ #ifndef Z_CLEARANCE_DEPLOY_PROBE
+ #error "You must define Z_CLEARANCE_DEPLOY_PROBE in your configuration."
+ #elif !defined(Z_CLEARANCE_BETWEEN_PROBES)
+ #error "You must define Z_CLEARANCE_BETWEEN_PROBES in your configuration."
+ #elif Z_CLEARANCE_DEPLOY_PROBE < 0
+ #error "Probes need Z_CLEARANCE_DEPLOY_PROBE >= 0."
+ #elif Z_CLEARANCE_BETWEEN_PROBES < 0
+ #error "Probes need Z_CLEARANCE_BETWEEN_PROBES >= 0."
+ #elif Z_AFTER_PROBING < 0
+ #error "Probes need Z_AFTER_PROBING >= 0."
+ #endif
+
+ #if MULTIPLE_PROBING > 0 || EXTRA_PROBING > 0
+ #if MULTIPLE_PROBING == 0
+ #error "EXTRA_PROBING requires MULTIPLE_PROBING."
+ #elif MULTIPLE_PROBING < 2
+ #error "MULTIPLE_PROBING must be 2 or more."
+ #elif MULTIPLE_PROBING <= EXTRA_PROBING
+ #error "EXTRA_PROBING must be less than MULTIPLE_PROBING."
+ #endif
+ #endif
+
+ #if Z_PROBE_LOW_POINT > 0
+ #error "Z_PROBE_LOW_POINT must be less than or equal to 0."
+ #endif
+
+ #if HOMING_Z_WITH_PROBE && IS_CARTESIAN && DISABLED(Z_SAFE_HOMING)
+ #error "Z_SAFE_HOMING is recommended when homing with a probe. Enable it or comment out this line to continue."
+ #endif
+
+ #if ENABLED(PROBE_ACTIVATION_SWITCH)
+ #ifndef PROBE_ACTIVATION_SWITCH_STATE
+ #error "PROBE_ACTIVATION_SWITCH_STATE is required for PROBE_ACTIVATION_SWITCH."
+ #elif !PIN_EXISTS(PROBE_ACTIVATION_SWITCH)
+ #error "A PROBE_ACTIVATION_SWITCH_PIN is required for PROBE_ACTIVATION_SWITCH."
+ #endif
+ #endif
+
+#else
+
+ /**
+ * Require some kind of probe for bed leveling and probe testing
+ */
+ #if HAS_ABL_NOT_UBL && !PROBE_SELECTED
+ #error "Auto Bed Leveling requires one of these: PROBE_MANUALLY, SENSORLESS_PROBING, BLTOUCH, FIX_MOUNTED_PROBE, NOZZLE_AS_PROBE, TOUCH_MI_PROBE, SOLENOID_PROBE, Z_PROBE_ALLEN_KEY, Z_PROBE_SLED, or a Z Servo."
+ #endif
+
+ #if ENABLED(Z_MIN_PROBE_REPEATABILITY_TEST)
+ #error "Z_MIN_PROBE_REPEATABILITY_TEST requires a probe: FIX_MOUNTED_PROBE, NOZZLE_AS_PROBE, BLTOUCH, SOLENOID_PROBE, Z_PROBE_ALLEN_KEY, Z_PROBE_SLED, or Z Servo."
+ #endif
+
+#endif
+
+#if ENABLED(LEVEL_BED_CORNERS)
+ #ifndef LEVEL_CORNERS_INSET_LFRB
+ #error "LEVEL_BED_CORNERS requires LEVEL_CORNERS_INSET_LFRB values."
+ #elif ENABLED(LEVEL_CORNERS_USE_PROBE)
+ #if !HAS_BED_PROBE
+ #error "LEVEL_CORNERS_USE_PROBE requires a real probe."
+ #elif ENABLED(SENSORLESS_PROBING)
+ #error "LEVEL_CORNERS_USE_PROBE is incompatible with SENSORLESS_PROBING."
+ #endif
+ #endif
+#endif
+
+/**
+ * Allow only one bed leveling option to be defined
+ */
+#if MANY(AUTO_BED_LEVELING_LINEAR, AUTO_BED_LEVELING_3POINT, AUTO_BED_LEVELING_BILINEAR, AUTO_BED_LEVELING_UBL, MESH_BED_LEVELING)
+ #error "Select only one of: MESH_BED_LEVELING, AUTO_BED_LEVELING_LINEAR, AUTO_BED_LEVELING_3POINT, AUTO_BED_LEVELING_BILINEAR or AUTO_BED_LEVELING_UBL."
+#endif
+
+/**
+ * Bed Leveling Requirements
+ */
+
+#if ENABLED(AUTO_BED_LEVELING_UBL)
+
+ /**
+ * Unified Bed Leveling
+ */
+
+ #if IS_SCARA
+ #error "AUTO_BED_LEVELING_UBL does not yet support SCARA printers."
+ #elif DISABLED(EEPROM_SETTINGS)
+ #error "AUTO_BED_LEVELING_UBL requires EEPROM_SETTINGS."
+ #elif !WITHIN(GRID_MAX_POINTS_X, 3, 15) || !WITHIN(GRID_MAX_POINTS_Y, 3, 15)
+ #error "GRID_MAX_POINTS_[XY] must be a whole number between 3 and 15."
+ #endif
+
+#elif HAS_ABL_NOT_UBL
+
+ /**
+ * Auto Bed Leveling
+ */
+
+ /**
+ * Delta and SCARA have limited bed leveling options
+ */
+ #if IS_SCARA && DISABLED(AUTO_BED_LEVELING_BILINEAR)
+ #error "SCARA machines can only use the AUTO_BED_LEVELING_BILINEAR leveling option."
+ #endif
+
+#elif ENABLED(MESH_BED_LEVELING)
+
+ // Mesh Bed Leveling
+ #if ENABLED(DELTA)
+ #error "MESH_BED_LEVELING is not compatible with DELTA printers."
+ #elif GRID_MAX_POINTS_X > 9 || GRID_MAX_POINTS_Y > 9
+ #error "GRID_MAX_POINTS_X and GRID_MAX_POINTS_Y must be less than 10 for MBL."
+ #endif
+
+#endif
+
+#if ALL(HAS_LEVELING, RESTORE_LEVELING_AFTER_G28, ENABLE_LEVELING_AFTER_G28)
+ #error "Only enable RESTORE_LEVELING_AFTER_G28 or ENABLE_LEVELING_AFTER_G28, but not both."
+#endif
+
+#if HAS_MESH && HAS_CLASSIC_JERK
+ static_assert(DEFAULT_ZJERK > 0.1, "Low DEFAULT_ZJERK values are incompatible with mesh-based leveling.");
+#endif
+
+#if ENABLED(G26_MESH_VALIDATION)
+ #if !EXTRUDERS
+ #error "G26_MESH_VALIDATION requires at least one extruder."
+ #elif !HAS_MESH
+ #error "G26_MESH_VALIDATION requires MESH_BED_LEVELING, AUTO_BED_LEVELING_BILINEAR, or AUTO_BED_LEVELING_UBL."
+ #endif
+#endif
+
+#if ENABLED(MESH_EDIT_GFX_OVERLAY) && !BOTH(AUTO_BED_LEVELING_UBL, HAS_MARLINUI_U8GLIB)
+ #error "MESH_EDIT_GFX_OVERLAY requires AUTO_BED_LEVELING_UBL and a Graphical LCD."
+#endif
+
+#if ENABLED(G29_RETRY_AND_RECOVER)
+ #if ENABLED(AUTO_BED_LEVELING_UBL)
+ #error "G29_RETRY_AND_RECOVER is not compatible with UBL."
+ #elif ENABLED(MESH_BED_LEVELING)
+ #error "G29_RETRY_AND_RECOVER is not compatible with MESH_BED_LEVELING."
+ #endif
+#endif
+
+/**
+ * LCD_BED_LEVELING requirements
+ */
+#if ENABLED(LCD_BED_LEVELING)
+ #if !HAS_LCD_MENU
+ #error "LCD_BED_LEVELING requires a programmable LCD controller."
+ #elif !(ENABLED(MESH_BED_LEVELING) || HAS_ABL_NOT_UBL)
+ #error "LCD_BED_LEVELING requires MESH_BED_LEVELING or AUTO_BED_LEVELING."
+ #endif
+#endif
+
+#if BOTH(PREHEAT_BEFORE_PROBING, PREHEAT_BEFORE_LEVELING)
+ #error "Disable PREHEAT_BEFORE_LEVELING when using PREHEAT_BEFORE_PROBING."
+#endif
+
+/**
+ * Homing
+ */
+constexpr float hbm[] = HOMING_BUMP_MM;
+static_assert(COUNT(hbm) == XYZ, "HOMING_BUMP_MM requires X, Y, and Z elements.");
+static_assert(hbm[X_AXIS] >= 0, "HOMING_BUMP_MM.X must be greater than or equal to 0.");
+static_assert(hbm[Y_AXIS] >= 0, "HOMING_BUMP_MM.Y must be greater than or equal to 0.");
+static_assert(hbm[Z_AXIS] >= 0, "HOMING_BUMP_MM.Z must be greater than or equal to 0.");
+
+#if ENABLED(CODEPENDENT_XY_HOMING)
+ #if ENABLED(QUICK_HOME)
+ #error "QUICK_HOME is incompatible with CODEPENDENT_XY_HOMING."
+ #elif IS_KINEMATIC
+ #error "CODEPENDENT_XY_HOMING requires a Cartesian setup."
+ #endif
+#endif
+
+/**
+ * Make sure Z_SAFE_HOMING point is reachable
+ */
+#if ENABLED(Z_SAFE_HOMING)
+ static_assert(WITHIN(Z_SAFE_HOMING_X_POINT, X_MIN_POS, X_MAX_POS), "Z_SAFE_HOMING_X_POINT can't be reached by the nozzle.");
+ static_assert(WITHIN(Z_SAFE_HOMING_Y_POINT, Y_MIN_POS, Y_MAX_POS), "Z_SAFE_HOMING_Y_POINT can't be reached by the nozzle.");
+#endif
+
+/**
+ * Make sure DISABLE_[XYZ] compatible with selected homing options
+ */
+#if ANY(DISABLE_X, DISABLE_Y, DISABLE_Z)
+ #if EITHER(HOME_AFTER_DEACTIVATE, Z_SAFE_HOMING)
+ #error "DISABLE_[XYZ] is not compatible with HOME_AFTER_DEACTIVATE or Z_SAFE_HOMING."
+ #endif
+#endif
+
+/**
+ * Filament Width Sensor
+ */
+#if ENABLED(FILAMENT_WIDTH_SENSOR)
+ #if !HAS_FILAMENT_WIDTH_SENSOR
+ #error "FILAMENT_WIDTH_SENSOR requires a FILWIDTH_PIN to be defined."
+ #elif ENABLED(NO_VOLUMETRICS)
+ #error "FILAMENT_WIDTH_SENSOR requires NO_VOLUMETRICS to be disabled."
+ #endif
+#endif
+
+/**
+ * System Power Sensor
+ */
+#if ENABLED(POWER_MONITOR_CURRENT) && !PIN_EXISTS(POWER_MONITOR_CURRENT)
+ #error "POWER_MONITOR_CURRENT requires a valid POWER_MONITOR_CURRENT_PIN."
+#elif ENABLED(POWER_MONITOR_VOLTAGE) && !PIN_EXISTS(POWER_MONITOR_VOLTAGE)
+ #error "POWER_MONITOR_VOLTAGE requires POWER_MONITOR_VOLTAGE_PIN to be defined."
+#elif BOTH(POWER_MONITOR_CURRENT, POWER_MONITOR_VOLTAGE) && POWER_MONITOR_CURRENT_PIN == POWER_MONITOR_VOLTAGE_PIN
+ #error "POWER_MONITOR_CURRENT_PIN and POWER_MONITOR_VOLTAGE_PIN must be different."
+#endif
+
+/**
+ * Volumetric Extruder Limit
+ */
+#if ENABLED(VOLUMETRIC_EXTRUDER_LIMIT)
+ #if ENABLED(NO_VOLUMETRICS)
+ #error "VOLUMETRIC_EXTRUDER_LIMIT requires NO_VOLUMETRICS to be disabled."
+ #elif MIN_STEPS_PER_SEGMENT > 1
+ #error "VOLUMETRIC_EXTRUDER_LIMIT is not compatible with MIN_STEPS_PER_SEGMENT greater than 1."
+ #endif
+#endif
+
+/**
+ * ULTIPANEL encoder
+ */
+#if IS_ULTIPANEL && NONE(IS_NEWPANEL, SR_LCD_2W_NL) && !PIN_EXISTS(SHIFT_CLK)
+ #error "ULTIPANEL controllers require some kind of encoder."
+#endif
+
+#if ENCODER_PULSES_PER_STEP < 0
+ #error "ENCODER_PULSES_PER_STEP should not be negative, use REVERSE_MENU_DIRECTION instead."
+#endif
+
+/**
+ * SAV_3DGLCD display options
+ */
+#if ENABLED(SAV_3DGLCD)
+ #if NONE(U8GLIB_SSD1306, U8GLIB_SH1106)
+ #error "Enable a SAV_3DGLCD display type: U8GLIB_SSD1306 or U8GLIB_SH1106."
+ #elif BOTH(U8GLIB_SSD1306, U8GLIB_SH1106)
+ #error "Only enable one SAV_3DGLCD display type: U8GLIB_SSD1306 or U8GLIB_SH1106."
+ #endif
+#endif
+
+/**
+ * Allen Key
+ * Deploying the Allen Key probe uses big moves in z direction. Too dangerous for an unhomed z-axis.
+ */
+#if BOTH(Z_PROBE_ALLEN_KEY, Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN) && (Z_HOME_DIR < 0)
+ #error "You can't home to a Z min endstop with a Z_PROBE_ALLEN_KEY."
+#endif
+
+/**
+ * Dual X Carriage requirements
+ */
+#if ENABLED(DUAL_X_CARRIAGE)
+ #if EXTRUDERS < 2
+ #error "DUAL_X_CARRIAGE requires 2 (or more) extruders."
+ #elif ANY(CORE_IS_XY, CORE_IS_XZ, MARKFORGED_XY)
+ #error "DUAL_X_CARRIAGE cannot be used with COREXY, COREYX, COREXZ, COREZX, or MARKFORGED_XY."
+ #elif !GOOD_AXIS_PINS(X2)
+ #error "DUAL_X_CARRIAGE requires X2 stepper pins to be defined."
+ #elif !HAS_X_MAX
+ #error "DUAL_X_CARRIAGE requires USE_XMAX_PLUG and an X Max Endstop."
+ #elif !defined(X2_HOME_POS) || !defined(X2_MIN_POS) || !defined(X2_MAX_POS)
+ #error "DUAL_X_CARRIAGE requires X2_HOME_POS, X2_MIN_POS, and X2_MAX_POS."
+ #elif X_HOME_DIR != -1 || X2_HOME_DIR != 1
+ #error "DUAL_X_CARRIAGE requires X_HOME_DIR -1 and X2_HOME_DIR 1."
+ #endif
+#endif
+
+#undef GOOD_AXIS_PINS
+
+/**
+ * Make sure auto fan pins don't conflict with the fan pin
+ */
+#if HAS_AUTO_FAN
+ #if HAS_FAN0
+ #if E0_AUTO_FAN_PIN == FAN_PIN
+ #error "You cannot set E0_AUTO_FAN_PIN equal to FAN_PIN."
+ #elif E1_AUTO_FAN_PIN == FAN_PIN
+ #error "You cannot set E1_AUTO_FAN_PIN equal to FAN_PIN."
+ #elif E2_AUTO_FAN_PIN == FAN_PIN
+ #error "You cannot set E2_AUTO_FAN_PIN equal to FAN_PIN."
+ #elif E3_AUTO_FAN_PIN == FAN_PIN
+ #error "You cannot set E3_AUTO_FAN_PIN equal to FAN_PIN."
+ #endif
+ #endif
+#endif
+
+#if HAS_FAN0 && CONTROLLER_FAN_PIN == FAN_PIN
+ #error "You cannot set CONTROLLER_FAN_PIN equal to FAN_PIN."
+#endif
+
+#if ENABLED(USE_CONTROLLER_FAN)
+ #if !HAS_CONTROLLER_FAN
+ #error "USE_CONTROLLER_FAN requires a CONTROLLER_FAN_PIN. Define in Configuration_adv.h."
+ #elif E0_AUTO_FAN_PIN == CONTROLLER_FAN_PIN
+ #error "You cannot set E0_AUTO_FAN_PIN equal to CONTROLLER_FAN_PIN."
+ #elif E1_AUTO_FAN_PIN == CONTROLLER_FAN_PIN
+ #error "You cannot set E1_AUTO_FAN_PIN equal to CONTROLLER_FAN_PIN."
+ #elif E2_AUTO_FAN_PIN == CONTROLLER_FAN_PIN
+ #error "You cannot set E2_AUTO_FAN_PIN equal to CONTROLLER_FAN_PIN."
+ #elif E3_AUTO_FAN_PIN == CONTROLLER_FAN_PIN
+ #error "You cannot set E3_AUTO_FAN_PIN equal to CONTROLLER_FAN_PIN."
+ #endif
+#endif
+
+/**
+ * Case Light requirements
+ */
+#if ENABLED(CASE_LIGHT_ENABLE)
+ #if !PIN_EXISTS(CASE_LIGHT)
+ #error "CASE_LIGHT_ENABLE requires CASE_LIGHT_PIN to be defined."
+ #elif CASE_LIGHT_PIN == FAN_PIN
+ #error "CASE_LIGHT_PIN conflicts with FAN_PIN. Resolve before continuing."
+ #endif
+#endif
+
+/**
+ * Required custom thermistor settings
+ */
+#if HEATER_0_USER_THERMISTOR && !(defined(HOTEND0_PULLUP_RESISTOR_OHMS) && defined(HOTEND0_RESISTANCE_25C_OHMS) && defined(HOTEND0_BETA))
+ #error "TEMP_SENSOR_0 1000 requires HOTEND0_PULLUP_RESISTOR_OHMS, HOTEND0_RESISTANCE_25C_OHMS and HOTEND0_BETA in Configuration_adv.h."
+#elif HEATER_1_USER_THERMISTOR && !(defined(HOTEND1_PULLUP_RESISTOR_OHMS) && defined(HOTEND1_RESISTANCE_25C_OHMS) && defined(HOTEND1_BETA))
+ #error "TEMP_SENSOR_1 1000 requires HOTEND1_PULLUP_RESISTOR_OHMS, HOTEND1_RESISTANCE_25C_OHMS and HOTEND1_BETA in Configuration_adv.h."
+#elif HEATER_2_USER_THERMISTOR && !(defined(HOTEND2_PULLUP_RESISTOR_OHMS) && defined(HOTEND2_RESISTANCE_25C_OHMS) && defined(HOTEND2_BETA))
+ #error "TEMP_SENSOR_2 1000 requires HOTEND2_PULLUP_RESISTOR_OHMS, HOTEND2_RESISTANCE_25C_OHMS and HOTEND2_BETA in Configuration_adv.h."
+#elif HEATER_3_USER_THERMISTOR && !(defined(HOTEND3_PULLUP_RESISTOR_OHMS) && defined(HOTEND3_RESISTANCE_25C_OHMS) && defined(HOTEND3_BETA))
+ #error "TEMP_SENSOR_3 1000 requires HOTEND3_PULLUP_RESISTOR_OHMS, HOTEND3_RESISTANCE_25C_OHMS and HOTEND3_BETA in Configuration_adv.h."
+#elif HEATER_4_USER_THERMISTOR && !(defined(HOTEND4_PULLUP_RESISTOR_OHMS) && defined(HOTEND4_RESISTANCE_25C_OHMS) && defined(HOTEND4_BETA))
+ #error "TEMP_SENSOR_4 1000 requires HOTEND4_PULLUP_RESISTOR_OHMS, HOTEND4_RESISTANCE_25C_OHMS and HOTEND4_BETA in Configuration_adv.h."
+#elif HEATER_5_USER_THERMISTOR && !(defined(HOTEND5_PULLUP_RESISTOR_OHMS) && defined(HOTEND5_RESISTANCE_25C_OHMS) && defined(HOTEND5_BETA))
+ #error "TEMP_SENSOR_5 1000 requires HOTEND5_PULLUP_RESISTOR_OHMS, HOTEND5_RESISTANCE_25C_OHMS and HOTEND5_BETA in Configuration_adv.h."
+#elif HEATER_6_USER_THERMISTOR && !(defined(HOTEND6_PULLUP_RESISTOR_OHMS) && defined(HOTEND6_RESISTANCE_25C_OHMS) && defined(HOTEND6_BETA))
+ #error "TEMP_SENSOR_6 1000 requires HOTEND6_PULLUP_RESISTOR_OHMS, HOTEND6_RESISTANCE_25C_OHMS and HOTEND6_BETA in Configuration_adv.h."
+#elif HEATER_7_USER_THERMISTOR && !(defined(HOTEND7_PULLUP_RESISTOR_OHMS) && defined(HOTEND7_RESISTANCE_25C_OHMS) && defined(HOTEND7_BETA))
+ #error "TEMP_SENSOR_7 1000 requires HOTEND7_PULLUP_RESISTOR_OHMS, HOTEND7_RESISTANCE_25C_OHMS and HOTEND7_BETA in Configuration_adv.h."
+#elif HEATER_BED_USER_THERMISTOR && !(defined(BED_PULLUP_RESISTOR_OHMS) && defined(BED_RESISTANCE_25C_OHMS) && defined(BED_BETA))
+ #error "TEMP_SENSOR_BED 1000 requires BED_PULLUP_RESISTOR_OHMS, BED_RESISTANCE_25C_OHMS and BED_BETA in Configuration_adv.h."
+#elif HEATER_CHAMBER_USER_THERMISTOR && !(defined(CHAMBER_PULLUP_RESISTOR_OHMS) && defined(CHAMBER_RESISTANCE_25C_OHMS) && defined(CHAMBER_BETA))
+ #error "TEMP_SENSOR_CHAMBER 1000 requires CHAMBER_PULLUP_RESISTOR_OHMS, CHAMBER_RESISTANCE_25C_OHMS and CHAMBER_BETA in Configuration_adv.h."
+#endif
+
+/**
+ * Pins and Sensor IDs must be set for each heater
+ */
+#if HEATER_0_USES_MAX6675 && !PIN_EXISTS(MAX6675_SS)
+ #error "MAX6675_SS_PIN (required for TEMP_SENSOR_0) not defined for this board."
+#elif HAS_HOTEND && !HAS_TEMP_HOTEND && !HEATER_0_DUMMY_THERMISTOR
+ #error "TEMP_0_PIN (required for TEMP_SENSOR_0) not defined for this board."
+#elif EITHER(HAS_MULTI_HOTEND, HEATERS_PARALLEL) && !HAS_HEATER_1
+ #error "HEATER_1_PIN is not defined. TEMP_SENSOR_1 might not be set, or the board (not EEB / EEF?) doesn't define a pin."
+#endif
+
+#if HAS_MULTI_HOTEND
+ #if HEATER_1_USES_MAX6675 && !PIN_EXISTS(MAX6675_SS2)
+ #error "MAX6675_SS2_PIN (required for TEMP_SENSOR_1) not defined for this board."
+ #elif TEMP_SENSOR_1 == 0
+ #error "TEMP_SENSOR_1 is required with 2 or more HOTENDS."
+ #elif !ANY_PIN(TEMP_1, MAX6675_SS2) && !HEATER_1_DUMMY_THERMISTOR
+ #error "TEMP_1_PIN not defined for this board."
+ #elif ENABLED(TEMP_SENSOR_1_AS_REDUNDANT)
+ #error "HOTENDS must be 1 with TEMP_SENSOR_1_AS_REDUNDANT."
+ #endif
+ #if HOTENDS > 2
+ #if TEMP_SENSOR_2 == 0
+ #error "TEMP_SENSOR_2 is required with 3 or more HOTENDS."
+ #elif !HAS_HEATER_2
+ #error "HEATER_2_PIN not defined for this board."
+ #elif !PIN_EXISTS(TEMP_2) && !HEATER_2_DUMMY_THERMISTOR
+ #error "TEMP_2_PIN not defined for this board."
+ #endif
+ #if HOTENDS > 3
+ #if TEMP_SENSOR_3 == 0
+ #error "TEMP_SENSOR_3 is required with 4 or more HOTENDS."
+ #elif !HAS_HEATER_3
+ #error "HEATER_3_PIN not defined for this board."
+ #elif !PIN_EXISTS(TEMP_3) && !HEATER_3_DUMMY_THERMISTOR
+ #error "TEMP_3_PIN not defined for this board."
+ #endif
+ #if HOTENDS > 4
+ #if TEMP_SENSOR_4 == 0
+ #error "TEMP_SENSOR_4 is required with 5 or more HOTENDS."
+ #elif !HAS_HEATER_4
+ #error "HEATER_4_PIN not defined for this board."
+ #elif !PIN_EXISTS(TEMP_4) && !HEATER_4_DUMMY_THERMISTOR
+ #error "TEMP_4_PIN not defined for this board."
+ #endif
+ #if HOTENDS > 5
+ #if TEMP_SENSOR_5 == 0
+ #error "TEMP_SENSOR_5 is required with 6 HOTENDS."
+ #elif !HAS_HEATER_5
+ #error "HEATER_5_PIN not defined for this board."
+ #elif !PIN_EXISTS(TEMP_5) && !HEATER_5_DUMMY_THERMISTOR
+ #error "TEMP_5_PIN not defined for this board."
+ #endif
+ #if HOTENDS > 6
+ #if TEMP_SENSOR_6 == 0
+ #error "TEMP_SENSOR_6 is required with 6 HOTENDS."
+ #elif !HAS_HEATER_6
+ #error "HEATER_6_PIN not defined for this board."
+ #elif !PIN_EXISTS(TEMP_6) && !HEATER_6_DUMMY_THERMISTOR
+ #error "TEMP_6_PIN not defined for this board."
+ #endif
+ #if HOTENDS > 7
+ #if TEMP_SENSOR_7 == 0
+ #error "TEMP_SENSOR_7 is required with 7 HOTENDS."
+ #elif !HAS_HEATER_7
+ #error "HEATER_7_PIN not defined for this board."
+ #elif !PIN_EXISTS(TEMP_7) && !HEATER_7_DUMMY_THERMISTOR
+ #error "TEMP_7_PIN not defined for this board."
+ #endif
+ #elif TEMP_SENSOR_7 != 0
+ #error "TEMP_SENSOR_7 shouldn't be set with only 7 HOTENDS."
+ #endif
+ #elif TEMP_SENSOR_6 != 0
+ #error "TEMP_SENSOR_6 shouldn't be set with only 6 HOTENDS."
+ #elif TEMP_SENSOR_7 != 0
+ #error "TEMP_SENSOR_7 shouldn't be set with only 6 HOTENDS."
+ #endif
+ #elif TEMP_SENSOR_5 != 0
+ #error "TEMP_SENSOR_5 shouldn't be set with only 5 HOTENDS."
+ #elif TEMP_SENSOR_6 != 0
+ #error "TEMP_SENSOR_6 shouldn't be set with only 5 HOTENDS."
+ #elif TEMP_SENSOR_7 != 0
+ #error "TEMP_SENSOR_7 shouldn't be set with only 5 HOTENDS."
+ #endif
+ #elif TEMP_SENSOR_4 != 0
+ #error "TEMP_SENSOR_4 shouldn't be set with only 4 HOTENDS."
+ #elif TEMP_SENSOR_5 != 0
+ #error "TEMP_SENSOR_5 shouldn't be set with only 4 HOTENDS."
+ #elif TEMP_SENSOR_6 != 0
+ #error "TEMP_SENSOR_6 shouldn't be set with only 4 HOTENDS."
+ #elif TEMP_SENSOR_7 != 0
+ #error "TEMP_SENSOR_7 shouldn't be set with only 4 HOTENDS."
+ #endif
+ #elif TEMP_SENSOR_3 != 0
+ #error "TEMP_SENSOR_3 shouldn't be set with only 3 HOTENDS."
+ #elif TEMP_SENSOR_4 != 0
+ #error "TEMP_SENSOR_4 shouldn't be set with only 3 HOTENDS."
+ #elif TEMP_SENSOR_5 != 0
+ #error "TEMP_SENSOR_5 shouldn't be set with only 3 HOTENDS."
+ #elif TEMP_SENSOR_6 != 0
+ #error "TEMP_SENSOR_6 shouldn't be set with only 3 HOTENDS."
+ #elif TEMP_SENSOR_7 != 0
+ #error "TEMP_SENSOR_7 shouldn't be set with only 3 HOTENDS."
+ #endif
+ #elif TEMP_SENSOR_2 != 0
+ #error "TEMP_SENSOR_2 shouldn't be set with only 2 HOTENDS."
+ #elif TEMP_SENSOR_3 != 0
+ #error "TEMP_SENSOR_3 shouldn't be set with only 2 HOTENDS."
+ #elif TEMP_SENSOR_4 != 0
+ #error "TEMP_SENSOR_4 shouldn't be set with only 2 HOTENDS."
+ #elif TEMP_SENSOR_5 != 0
+ #error "TEMP_SENSOR_5 shouldn't be set with only 2 HOTENDS."
+ #elif TEMP_SENSOR_6 != 0
+ #error "TEMP_SENSOR_6 shouldn't be set with only 2 HOTENDS."
+ #elif TEMP_SENSOR_7 != 0
+ #error "TEMP_SENSOR_7 shouldn't be set with only 2 HOTENDS."
+ #endif
+#elif TEMP_SENSOR_1 != 0 && DISABLED(TEMP_SENSOR_1_AS_REDUNDANT)
+ #error "TEMP_SENSOR_1 shouldn't be set with only 1 HOTEND."
+#elif TEMP_SENSOR_2 != 0
+ #error "TEMP_SENSOR_2 shouldn't be set with only 1 HOTEND."
+#elif TEMP_SENSOR_3 != 0
+ #error "TEMP_SENSOR_3 shouldn't be set with only 1 HOTEND."
+#elif TEMP_SENSOR_4 != 0
+ #error "TEMP_SENSOR_4 shouldn't be set with only 1 HOTEND."
+#elif TEMP_SENSOR_5 != 0
+ #error "TEMP_SENSOR_5 shouldn't be set with only 1 HOTEND."
+#elif TEMP_SENSOR_6 != 0
+ #error "TEMP_SENSOR_6 shouldn't be set with only 1 HOTEND."
+#elif TEMP_SENSOR_7 != 0
+ #error "TEMP_SENSOR_7 shouldn't be set with only 1 HOTEND."
+#endif
+
+#if TEMP_SENSOR_CHAMBER && !PIN_EXISTS(TEMP_CHAMBER)
+ #error "TEMP_SENSOR_CHAMBER requires TEMP_CHAMBER_PIN."
+#endif
+
+#if ENABLED(CHAMBER_FAN) && !(defined(CHAMBER_FAN_MODE) && WITHIN(CHAMBER_FAN_MODE, 0, 2))
+ #error "CHAMBER_FAN_MODE must be between 0 and 2."
+#endif
+
+#if ENABLED(CHAMBER_VENT)
+ #ifndef CHAMBER_VENT_SERVO_NR
+ #error "CHAMBER_VENT_SERVO_NR is required for CHAMBER SERVO."
+ #elif !NUM_SERVOS
+ #error "NUM_SERVOS is required for a Heated Chamber vent servo (CHAMBER_VENT_SERVO_NR)."
+ #elif CHAMBER_VENT_SERVO_NR >= NUM_SERVOS
+ #error "CHAMBER_VENT_SERVO_NR must be smaller than NUM_SERVOS."
+ #elif HAS_Z_SERVO_PROBE && CHAMBER_VENT_SERVO_NR == Z_PROBE_SERVO_NR
+ #error "CHAMBER SERVO is already used by BLTOUCH."
+ #elif CHAMBER_VENT_SERVO_NR == 0 && !PIN_EXISTS(SERVO0)
+ #error "SERVO0_PIN must be defined for your Heated Chamber vent servo."
+ #elif CHAMBER_VENT_SERVO_NR == 1 && !PIN_EXISTS(SERVO1)
+ #error "SERVO1_PIN must be defined for your Heated Chamber vent servo."
+ #elif CHAMBER_VENT_SERVO_NR == 2 && !PIN_EXISTS(SERVO2)
+ #error "SERVO2_PIN must be defined for your Heated Chamber vent servo."
+ #elif CHAMBER_VENT_SERVO_NR == 3 && !PIN_EXISTS(SERVO3)
+ #error "SERVO3_PIN must be defined for your Heated Chamber vent servo."
+ #endif
+#endif
+
+#if TEMP_SENSOR_PROBE
+ #if !PIN_EXISTS(TEMP_PROBE)
+ #error "TEMP_SENSOR_PROBE requires TEMP_PROBE_PIN."
+ #elif !HAS_TEMP_ADC_PROBE
+ #error "TEMP_PROBE_PIN must be an ADC pin."
+ #elif DISABLED(FIX_MOUNTED_PROBE)
+ #error "TEMP_SENSOR_PROBE shouldn't be set without FIX_MOUNTED_PROBE."
+ #endif
+#endif
+
+#if ENABLED(TEMP_SENSOR_1_AS_REDUNDANT) && TEMP_SENSOR_1 == 0
+ #error "TEMP_SENSOR_1 is required with TEMP_SENSOR_1_AS_REDUNDANT."
+#endif
+
+#if MAX6675_0_IS_MAX31865 && !(defined(MAX31865_SENSOR_OHMS_0) && defined(MAX31865_CALIBRATION_OHMS_0))
+ #error "MAX31865_SENSOR_OHMS_0 and MAX31865_CALIBRATION_OHMS_0 must be set if TEMP_SENSOR_0 is MAX31865."
+#elif MAX6675_1_IS_MAX31865 && !(defined(MAX31865_SENSOR_OHMS_1) && defined(MAX31865_CALIBRATION_OHMS_1))
+ #error "MAX31865_SENSOR_OHMS_1 and MAX31865_CALIBRATION_OHMS_1 must be set if TEMP_SENSOR_1 is MAX31865."
+#endif
+
+/**
+ * Test Heater, Temp Sensor, and Extruder Pins
+ */
+#if !HAS_HEATER_0 && EXTRUDERS
+ #error "HEATER_0_PIN not defined for this board."
+#elif !ANY_PIN(TEMP_0, MAX6675_SS)
+ #error "TEMP_0_PIN not defined for this board."
+#elif ((defined(__AVR_ATmega644P__) || defined(__AVR_ATmega1284P__)) && !PINS_EXIST(E0_STEP, E0_DIR))
+ #error "E0_STEP_PIN or E0_DIR_PIN not defined for this board."
+#elif ( !(defined(__AVR_ATmega644P__) || defined(__AVR_ATmega1284P__)) && (!PINS_EXIST(E0_STEP, E0_DIR) || !HAS_E0_ENABLE))
+ #error "E0_STEP_PIN, E0_DIR_PIN, or E0_ENABLE_PIN not defined for this board."
+#elif EXTRUDERS && TEMP_SENSOR_0 == 0
+ #error "TEMP_SENSOR_0 is required if there are any extruders."
+#endif
+
+/**
+ * Temperature status LEDs
+ */
+#if ENABLED(TEMP_STAT_LEDS) && !ANY_PIN(STAT_LED_RED, STAT_LED_BLUE)
+ #error "TEMP_STAT_LEDS requires STAT_LED_RED_PIN or STAT_LED_BLUE_PIN, preferably both."
+#endif
+
+/**
+ * LED Control Menu
+ */
+#if ENABLED(LED_CONTROL_MENU) && !HAS_COLOR_LEDS
+ #error "LED_CONTROL_MENU requires BLINKM, RGB_LED, RGBW_LED, PCA9533, PCA9632, or NEOPIXEL_LED."
+#endif
+
+/**
+ * LED Backlight Timeout
+ */
+#if defined(LED_BACKLIGHT_TIMEOUT) && !(ENABLED(PSU_CONTROL) && ANY(FYSETC_MINI_12864_2_0, FYSETC_MINI_12864_2_1, FYSETC_242_OLED_12864))
+ #error "LED_BACKLIGHT_TIMEOUT requires a FYSETC Mini Panel and a Power Switch."
+#endif
+
+/**
+ * Basic multi hotend duplication mode
+ */
+#if ENABLED(MULTI_NOZZLE_DUPLICATION)
+ #if ENABLED(SINGLENOZZLE)
+ #error "MULTI_NOZZLE_DUPLICATION is incompatible with SINGLENOZZLE."
+ #elif ENABLED(DUAL_X_CARRIAGE)
+ #error "MULTI_NOZZLE_DUPLICATION is incompatible with DUAL_X_CARRIAGE."
+ #elif ENABLED(MIXING_EXTRUDER)
+ #error "MULTI_NOZZLE_DUPLICATION is incompatible with MIXING_EXTRUDER."
+ #elif ENABLED(SWITCHING_EXTRUDER)
+ #error "MULTI_NOZZLE_DUPLICATION is incompatible with SWITCHING_EXTRUDER."
+ #elif HOTENDS < 2
+ #error "MULTI_NOZZLE_DUPLICATION requires 2 or more hotends."
+ #endif
+#endif
+
+/**
+ * Test Extruder Stepper Pins
+ */
+#if E_STEPPERS
+ #if !(PINS_EXIST(E0_STEP, E0_DIR) && HAS_E0_ENABLE)
+ #error "E0_STEP_PIN, E0_DIR_PIN, or E0_ENABLE_PIN not defined for this board."
+ #endif
+ #if E_STEPPERS > 1
+ #if !(PINS_EXIST(E1_STEP, E1_DIR) && HAS_E1_ENABLE)
+ #error "E1_STEP_PIN, E1_DIR_PIN, or E1_ENABLE_PIN not defined for this board."
+ #endif
+ #if E_STEPPERS > 2
+ #if !(PINS_EXIST(E2_STEP, E2_DIR) && HAS_E2_ENABLE)
+ #error "E2_STEP_PIN, E2_DIR_PIN, or E2_ENABLE_PIN not defined for this board."
+ #endif
+ #if E_STEPPERS > 3
+ #if !(PINS_EXIST(E3_STEP, E3_DIR) && HAS_E3_ENABLE)
+ #error "E3_STEP_PIN, E3_DIR_PIN, or E3_ENABLE_PIN not defined for this board."
+ #endif
+ #if E_STEPPERS > 4
+ #if !(PINS_EXIST(E4_STEP, E4_DIR) && HAS_E4_ENABLE)
+ #error "E4_STEP_PIN, E4_DIR_PIN, or E4_ENABLE_PIN not defined for this board."
+ #endif
+ #if E_STEPPERS > 5
+ #if !(PINS_EXIST(E5_STEP, E5_DIR) && HAS_E5_ENABLE)
+ #error "E5_STEP_PIN, E5_DIR_PIN, or E5_ENABLE_PIN not defined for this board."
+ #endif
+ #if E_STEPPERS > 6
+ #if !(PINS_EXIST(E6_STEP, E6_DIR) && HAS_E6_ENABLE)
+ #error "E6_STEP_PIN, E6_DIR_PIN, or E6_ENABLE_PIN not defined for this board."
+ #endif
+ #if E_STEPPERS > 7
+ #if !(PINS_EXIST(E7_STEP, E7_DIR) && HAS_E7_ENABLE)
+ #error "E7_STEP_PIN, E7_DIR_PIN, or E7_ENABLE_PIN not defined for this board."
+ #endif
+ #endif // E_STEPPERS > 7
+ #endif // E_STEPPERS > 6
+ #endif // E_STEPPERS > 5
+ #endif // E_STEPPERS > 4
+ #endif // E_STEPPERS > 3
+ #endif // E_STEPPERS > 2
+ #endif // E_STEPPERS > 1
+#endif // E_STEPPERS
+
+/**
+ * Endstop Tests
+ */
+
+#define _PLUG_UNUSED_TEST(A,P) (DISABLED(USE_##P##MIN_PLUG, USE_##P##MAX_PLUG) \
+ && !(ENABLED(A##_DUAL_ENDSTOPS) && WITHIN(A##2_USE_ENDSTOP, _##P##MAX_, _##P##MIN_)) \
+ && !(ENABLED(A##_MULTI_ENDSTOPS) && WITHIN(A##2_USE_ENDSTOP, _##P##MAX_, _##P##MIN_)) )
+#define _AXIS_PLUG_UNUSED_TEST(A) (_PLUG_UNUSED_TEST(A,X) && _PLUG_UNUSED_TEST(A,Y) && _PLUG_UNUSED_TEST(A,Z))
+
+// At least 3 endstop plugs must be used
+#if _AXIS_PLUG_UNUSED_TEST(X)
+ #error "You must enable USE_XMIN_PLUG or USE_XMAX_PLUG."
+#endif
+#if _AXIS_PLUG_UNUSED_TEST(Y)
+ #error "You must enable USE_YMIN_PLUG or USE_YMAX_PLUG."
+#endif
+#if _AXIS_PLUG_UNUSED_TEST(Z)
+ #error "You must enable USE_ZMIN_PLUG or USE_ZMAX_PLUG."
+#endif
+
+// Delta and Cartesian use 3 homing endstops
+#if NONE(IS_SCARA, SPI_ENDSTOPS)
+ #if X_HOME_DIR < 0 && DISABLED(USE_XMIN_PLUG)
+ #error "Enable USE_XMIN_PLUG when homing X to MIN."
+ #elif X_HOME_DIR > 0 && DISABLED(USE_XMAX_PLUG)
+ #error "Enable USE_XMAX_PLUG when homing X to MAX."
+ #elif Y_HOME_DIR < 0 && DISABLED(USE_YMIN_PLUG)
+ #error "Enable USE_YMIN_PLUG when homing Y to MIN."
+ #elif Y_HOME_DIR > 0 && DISABLED(USE_YMAX_PLUG)
+ #error "Enable USE_YMAX_PLUG when homing Y to MAX."
+ #endif
+#endif
+
+// Z homing direction and plug usage flags
+#if Z_HOME_DIR < 0 && NONE(USE_ZMIN_PLUG, HOMING_Z_WITH_PROBE)
+ #error "Enable USE_ZMIN_PLUG when homing Z to MIN."
+#elif Z_HOME_DIR > 0 && ENABLED(USE_PROBE_FOR_Z_HOMING)
+ #error "Z_HOME_DIR must be -1 when homing Z with the probe."
+#elif BOTH(HOMING_Z_WITH_PROBE, Z_MULTI_ENDSTOPS)
+ #error "Z_MULTI_ENDSTOPS is incompatible with USE_PROBE_FOR_Z_HOMING."
+#elif Z_HOME_DIR > 0 && DISABLED(USE_ZMAX_PLUG)
+ #error "Enable USE_ZMAX_PLUG when homing Z to MAX."
+#endif
+
+#if BOTH(HOME_Z_FIRST, USE_PROBE_FOR_Z_HOMING)
+ #error "HOME_Z_FIRST can't be used when homing Z with a probe."
+#endif
+
+// Dual/multiple endstops requirements
+#if ENABLED(X_DUAL_ENDSTOPS)
+ #if !X2_USE_ENDSTOP
+ #error "You must set X2_USE_ENDSTOP with X_DUAL_ENDSTOPS."
+ #elif X2_USE_ENDSTOP == _XMIN_ && DISABLED(USE_XMIN_PLUG)
+ #error "USE_XMIN_PLUG is required when X2_USE_ENDSTOP is _XMIN_."
+ #elif X2_USE_ENDSTOP == _XMAX_ && DISABLED(USE_XMAX_PLUG)
+ #error "USE_XMAX_PLUG is required when X2_USE_ENDSTOP is _XMAX_."
+ #elif X2_USE_ENDSTOP == _YMIN_ && DISABLED(USE_YMIN_PLUG)
+ #error "USE_YMIN_PLUG is required when X2_USE_ENDSTOP is _YMIN_."
+ #elif X2_USE_ENDSTOP == _YMAX_ && DISABLED(USE_YMAX_PLUG)
+ #error "USE_YMAX_PLUG is required when X2_USE_ENDSTOP is _YMAX_."
+ #elif X2_USE_ENDSTOP == _ZMIN_ && DISABLED(USE_ZMIN_PLUG)
+ #error "USE_ZMIN_PLUG is required when X2_USE_ENDSTOP is _ZMIN_."
+ #elif X2_USE_ENDSTOP == _ZMAX_ && DISABLED(USE_ZMAX_PLUG)
+ #error "USE_ZMAX_PLUG is required when X2_USE_ENDSTOP is _ZMAX_."
+ #elif !HAS_X2_MIN && !HAS_X2_MAX
+ #error "X2_USE_ENDSTOP has been assigned to a nonexistent endstop!"
+ #elif ENABLED(DELTA)
+ #error "X_DUAL_ENDSTOPS is not compatible with DELTA."
+ #endif
+#endif
+#if ENABLED(Y_DUAL_ENDSTOPS)
+ #if !Y2_USE_ENDSTOP
+ #error "You must set Y2_USE_ENDSTOP with Y_DUAL_ENDSTOPS."
+ #elif Y2_USE_ENDSTOP == _XMIN_ && DISABLED(USE_XMIN_PLUG)
+ #error "USE_XMIN_PLUG is required when Y2_USE_ENDSTOP is _XMIN_."
+ #elif Y2_USE_ENDSTOP == _XMAX_ && DISABLED(USE_XMAX_PLUG)
+ #error "USE_XMAX_PLUG is required when Y2_USE_ENDSTOP is _XMAX_."
+ #elif Y2_USE_ENDSTOP == _YMIN_ && DISABLED(USE_YMIN_PLUG)
+ #error "USE_YMIN_PLUG is required when Y2_USE_ENDSTOP is _YMIN_."
+ #elif Y2_USE_ENDSTOP == _YMAX_ && DISABLED(USE_YMAX_PLUG)
+ #error "USE_YMAX_PLUG is required when Y2_USE_ENDSTOP is _YMAX_."
+ #elif Y2_USE_ENDSTOP == _ZMIN_ && DISABLED(USE_ZMIN_PLUG)
+ #error "USE_ZMIN_PLUG is required when Y2_USE_ENDSTOP is _ZMIN_."
+ #elif Y2_USE_ENDSTOP == _ZMAX_ && DISABLED(USE_ZMAX_PLUG)
+ #error "USE_ZMAX_PLUG is required when Y2_USE_ENDSTOP is _ZMAX_."
+ #elif !HAS_Y2_MIN && !HAS_Y2_MAX
+ #error "Y2_USE_ENDSTOP has been assigned to a nonexistent endstop!"
+ #elif ENABLED(DELTA)
+ #error "Y_DUAL_ENDSTOPS is not compatible with DELTA."
+ #endif
+#endif
+
+#if ENABLED(Z_MULTI_ENDSTOPS)
+ #if !Z2_USE_ENDSTOP
+ #error "You must set Z2_USE_ENDSTOP with Z_MULTI_ENDSTOPS when NUM_Z_STEPPER_DRIVERS >= 2."
+ #elif Z2_USE_ENDSTOP == _XMIN_ && DISABLED(USE_XMIN_PLUG)
+ #error "USE_XMIN_PLUG is required when Z2_USE_ENDSTOP is _XMIN_."
+ #elif Z2_USE_ENDSTOP == _XMAX_ && DISABLED(USE_XMAX_PLUG)
+ #error "USE_XMAX_PLUG is required when Z2_USE_ENDSTOP is _XMAX_."
+ #elif Z2_USE_ENDSTOP == _YMIN_ && DISABLED(USE_YMIN_PLUG)
+ #error "USE_YMIN_PLUG is required when Z2_USE_ENDSTOP is _YMIN_."
+ #elif Z2_USE_ENDSTOP == _YMAX_ && DISABLED(USE_YMAX_PLUG)
+ #error "USE_YMAX_PLUG is required when Z2_USE_ENDSTOP is _YMAX_."
+ #elif Z2_USE_ENDSTOP == _ZMIN_ && DISABLED(USE_ZMIN_PLUG)
+ #error "USE_ZMIN_PLUG is required when Z2_USE_ENDSTOP is _ZMIN_."
+ #elif Z2_USE_ENDSTOP == _ZMAX_ && DISABLED(USE_ZMAX_PLUG)
+ #error "USE_ZMAX_PLUG is required when Z2_USE_ENDSTOP is _ZMAX_."
+ #elif !HAS_Z2_MIN && !HAS_Z2_MAX
+ #error "Z2_USE_ENDSTOP has been assigned to a nonexistent endstop!"
+ #elif ENABLED(DELTA)
+ #error "Z_MULTI_ENDSTOPS is not compatible with DELTA."
+ #endif
+ #if NUM_Z_STEPPER_DRIVERS >= 3
+ #if !Z3_USE_ENDSTOP
+ #error "You must set Z3_USE_ENDSTOP with Z_MULTI_ENDSTOPS when NUM_Z_STEPPER_DRIVERS >= 3."
+ #elif Z3_USE_ENDSTOP == _XMIN_ && DISABLED(USE_XMIN_PLUG)
+ #error "USE_XMIN_PLUG is required when Z3_USE_ENDSTOP is _XMIN_."
+ #elif Z3_USE_ENDSTOP == _XMAX_ && DISABLED(USE_XMAX_PLUG)
+ #error "USE_XMAX_PLUG is required when Z3_USE_ENDSTOP is _XMAX_."
+ #elif Z3_USE_ENDSTOP == _YMIN_ && DISABLED(USE_YMIN_PLUG)
+ #error "USE_YMIN_PLUG is required when Z3_USE_ENDSTOP is _YMIN_."
+ #elif Z3_USE_ENDSTOP == _YMAX_ && DISABLED(USE_YMAX_PLUG)
+ #error "USE_YMAX_PLUG is required when Z3_USE_ENDSTOP is _YMAX_."
+ #elif Z3_USE_ENDSTOP == _ZMIN_ && DISABLED(USE_ZMIN_PLUG)
+ #error "USE_ZMIN_PLUG is required when Z3_USE_ENDSTOP is _ZMIN_."
+ #elif Z3_USE_ENDSTOP == _ZMAX_ && DISABLED(USE_ZMAX_PLUG)
+ #error "USE_ZMAX_PLUG is required when Z3_USE_ENDSTOP is _ZMAX_."
+ #elif !HAS_Z3_MIN && !HAS_Z3_MAX
+ #error "Z3_USE_ENDSTOP has been assigned to a nonexistent endstop!"
+ #endif
+ #endif
+ #if NUM_Z_STEPPER_DRIVERS >= 4
+ #if !Z4_USE_ENDSTOP
+ #error "You must set Z4_USE_ENDSTOP with Z_MULTI_ENDSTOPS when NUM_Z_STEPPER_DRIVERS >= 4."
+ #elif Z4_USE_ENDSTOP == _XMIN_ && DISABLED(USE_XMIN_PLUG)
+ #error "USE_XMIN_PLUG is required when Z4_USE_ENDSTOP is _XMIN_."
+ #elif Z4_USE_ENDSTOP == _XMAX_ && DISABLED(USE_XMAX_PLUG)
+ #error "USE_XMAX_PLUG is required when Z4_USE_ENDSTOP is _XMAX_."
+ #elif Z4_USE_ENDSTOP == _YMIN_ && DISABLED(USE_YMIN_PLUG)
+ #error "USE_YMIN_PLUG is required when Z4_USE_ENDSTOP is _YMIN_."
+ #elif Z4_USE_ENDSTOP == _YMAX_ && DISABLED(USE_YMAX_PLUG)
+ #error "USE_YMAX_PLUG is required when Z4_USE_ENDSTOP is _YMAX_."
+ #elif Z4_USE_ENDSTOP == _ZMIN_ && DISABLED(USE_ZMIN_PLUG)
+ #error "USE_ZMIN_PLUG is required when Z4_USE_ENDSTOP is _ZMIN_."
+ #elif Z4_USE_ENDSTOP == _ZMAX_ && DISABLED(USE_ZMAX_PLUG)
+ #error "USE_ZMAX_PLUG is required when Z4_USE_ENDSTOP is _ZMAX_."
+ #elif !HAS_Z4_MIN && !HAS_Z4_MAX
+ #error "Z4_USE_ENDSTOP has been assigned to a nonexistent endstop!"
+ #endif
+ #endif
+#endif
+
+#if defined(ENDSTOP_NOISE_THRESHOLD) && !WITHIN(ENDSTOP_NOISE_THRESHOLD, 2, 7)
+ #error "ENDSTOP_NOISE_THRESHOLD must be an integer from 2 to 7."
+#endif
+
+/**
+ * emergency-command parser
+ */
+#if ENABLED(EMERGENCY_PARSER) && defined(__AVR__) && defined(USBCON)
+ #error "EMERGENCY_PARSER does not work on boards with AT90USB processors (USBCON)."
+#endif
+
+/**
+ * I2C bus
+ */
+#if ENABLED(EXPERIMENTAL_I2CBUS) && I2C_SLAVE_ADDRESS > 0
+ #if I2C_SLAVE_ADDRESS < 8
+ #error "I2C_SLAVE_ADDRESS can't be less than 8. (Addresses 0 - 7 are reserved.)"
+ #elif I2C_SLAVE_ADDRESS > 127
+ #error "I2C_SLAVE_ADDRESS can't be over 127. (Only 7 bits allowed.)"
+ #endif
+#endif
+
+/**
+ * G35 Assisted Tramming
+ */
+#if ENABLED(ASSISTED_TRAMMING) && !HAS_BED_PROBE
+ #error "ASSISTED_TRAMMING requires a bed probe."
+#endif
+
+/**
+ * G38 Probe Target
+ */
+#if ENABLED(G38_PROBE_TARGET)
+ #if !HAS_BED_PROBE
+ #error "G38_PROBE_TARGET requires a bed probe."
+ #elif !IS_CARTESIAN
+ #error "G38_PROBE_TARGET requires a Cartesian machine."
+ #endif
+#endif
+
+/**
+ * RGB_LED Requirements
+ */
+#define _RGB_TEST (PINS_EXIST(RGB_LED_R, RGB_LED_G, RGB_LED_B))
+#if ENABLED(PRINTER_EVENT_LEDS) && !HAS_COLOR_LEDS
+ #error "PRINTER_EVENT_LEDS requires BLINKM, PCA9533, PCA9632, RGB_LED, RGBW_LED or NEOPIXEL_LED."
+#elif ENABLED(RGB_LED)
+ #if !_RGB_TEST
+ #error "RGB_LED requires RGB_LED_R_PIN, RGB_LED_G_PIN, and RGB_LED_B_PIN."
+ #elif ENABLED(RGBW_LED)
+ #error "Please enable only one of RGB_LED and RGBW_LED."
+ #endif
+#elif ENABLED(RGBW_LED)
+ #if !(_RGB_TEST && PIN_EXISTS(RGB_LED_W))
+ #error "RGBW_LED requires RGB_LED_R_PIN, RGB_LED_G_PIN, RGB_LED_B_PIN, and RGB_LED_W_PIN."
+ #endif
+#endif
+#undef _RGB_TEST
+
+// NeoPixel requirements
+#if ENABLED(NEOPIXEL_LED)
+ #if !PIN_EXISTS(NEOPIXEL) || NEOPIXEL_PIXELS == 0
+ #error "NEOPIXEL_LED requires NEOPIXEL_PIN and NEOPIXEL_PIXELS."
+ #elif ENABLED(NEOPIXEL2_SEPARATE) && !(defined(NEOPIXEL2_TYPE) && PIN_EXISTS(NEOPIXEL2) && NEOPIXEL2_PIXELS > 0)
+ #error "NEOPIXEL2_SEPARATE requires NEOPIXEL2_TYPE, NEOPIXEL2_PIN and NEOPIXEL2_PIXELS."
+ #elif ENABLED(NEO2_COLOR_PRESETS) && DISABLED(NEOPIXEL2_SEPARATE)
+ #error "NEO2_COLOR_PRESETS requires NEOPIXEL2_SEPARATE to be enabled."
+ #endif
+#endif
+
+#if DISABLED(NO_COMPILE_TIME_PWM)
+ #define _TEST_PWM(P) PWM_PIN(P)
+#else
+ #define _TEST_PWM(P) 1 // pass
+#endif
+
+/**
+ * Auto Fan check for PWM pins
+ */
+#if HAS_AUTO_FAN && EXTRUDER_AUTO_FAN_SPEED != 255
+ #define AF_ERR_SUFF "_AUTO_FAN_PIN is not a PWM pin. Set EXTRUDER_AUTO_FAN_SPEED to 255."
+ #if HAS_AUTO_FAN_0
+ static_assert(_TEST_PWM(E0_AUTO_FAN_PIN), "E0" AF_ERR_SUFF);
+ #elif HAS_AUTO_FAN_1
+ static_assert(_TEST_PWM(E1_AUTO_FAN_PIN), "E1" AF_ERR_SUFF);
+ #elif HAS_AUTO_FAN_2
+ static_assert(_TEST_PWM(E2_AUTO_FAN_PIN), "E2" AF_ERR_SUFF);
+ #elif HAS_AUTO_FAN_3
+ static_assert(_TEST_PWM(E3_AUTO_FAN_PIN), "E3" AF_ERR_SUFF);
+ #elif HAS_AUTO_FAN_4
+ static_assert(_TEST_PWM(E4_AUTO_FAN_PIN), "E4" AF_ERR_SUFF);
+ #elif HAS_AUTO_FAN_5
+ static_assert(_TEST_PWM(E5_AUTO_FAN_PIN), "E5" AF_ERR_SUFF);
+ #elif HAS_AUTO_FAN_6
+ static_assert(_TEST_PWM(E6_AUTO_FAN_PIN), "E6" AF_ERR_SUFF);
+ #elif HAS_AUTO_FAN_7
+ static_assert(_TEST_PWM(E7_AUTO_FAN_PIN), "E7" AF_ERR_SUFF);
+ #endif
+#endif
+
+/**
+ * Make sure only one EEPROM type is enabled
+ */
+#if ENABLED(EEPROM_SETTINGS)
+ #if 1 < 0 \
+ + ENABLED(I2C_EEPROM) \
+ + ENABLED(SPI_EEPROM) \
+ + ENABLED(QSPI_EEPROM) \
+ + ENABLED(SDCARD_EEPROM_EMULATION) \
+ + ENABLED(FLASH_EEPROM_EMULATION) \
+ + ENABLED(SRAM_EEPROM_EMULATION) \
+ + ENABLED(IIC_BL24CXX_EEPROM)
+ #error "Please select only one method of EEPROM Persistent Storage."
+ #endif
+#endif
+
+/**
+ * Make sure features that need to write to the SD card can
+ */
+#if ENABLED(SDCARD_READONLY) && ANY(POWER_LOSS_RECOVERY, BINARY_FILE_TRANSFER, SDCARD_EEPROM_EMULATION)
+ #undef SDCARD_READONLY
+ #if ENABLED(POWER_LOSS_RECOVERY)
+ #warning "Either disable SDCARD_READONLY or disable POWER_LOSS_RECOVERY."
+ #elif ENABLED(BINARY_FILE_TRANSFER)
+ #warning "Either disable SDCARD_READONLY or disable BINARY_FILE_TRANSFER."
+ #elif ENABLED(SDCARD_EEPROM_EMULATION)
+ #warning "Either disable SDCARD_READONLY or disable SDCARD_EEPROM_EMULATION."
+ #endif
+#endif
+
+#if ENABLED(SD_IGNORE_AT_STARTUP)
+ #if ENABLED(POWER_LOSS_RECOVERY)
+ #error "SD_IGNORE_AT_STARTUP is incompatible with POWER_LOSS_RECOVERY."
+ #elif ENABLED(SDCARD_EEPROM_EMULATION)
+ #error "SD_IGNORE_AT_STARTUP is incompatible with SDCARD_EEPROM_EMULATION."
+ #endif
+#endif
+
+/**
+ * Make sure only one display is enabled
+ */
+#if 1 < 0 \
+ + ENABLED(REPRAP_DISCOUNT_SMART_CONTROLLER) \
+ + ENABLED(REPRAP_DISCOUNT_FULL_GRAPHIC_SMART_CONTROLLER) \
+ + ENABLED(ULTIPANEL) \
+ + ENABLED(ULTRA_LCD) \
+ + (ENABLED(U8GLIB_SSD1306) && DISABLED(IS_U8GLIB_SSD1306)) \
+ + (ENABLED(MINIPANEL) && NONE(MKS_MINI_12864, ENDER2_STOCKDISPLAY)) \
+ + (ENABLED(MKS_MINI_12864) && DISABLED(MKS_LCD12864)) \
+ + (ENABLED(EXTENSIBLE_UI) && DISABLED(IS_EXTUI)) \
+ + (DISABLED(IS_LEGACY_TFT) && ENABLED(TFT_GENERIC)) \
+ + (ENABLED(IS_LEGACY_TFT) && COUNT_ENABLED(TFT_320x240, TFT_320x240_SPI, TFT_480x320, TFT_480x320_SPI)) \
+ + COUNT_ENABLED(ANYCUBIC_LCD_I3MEGA, ANYCUBIC_LCD_CHIRON, ANYCUBIC_TFT35) \
+ + COUNT_ENABLED(DGUS_LCD_UI_ORIGIN, DGUS_LCD_UI_FYSETC, DGUS_LCD_UI_HIPRECY) \
+ + COUNT_ENABLED(ENDER2_STOCKDISPLAY, CR10_STOCKDISPLAY, DWIN_CREALITY_LCD) \
+ + COUNT_ENABLED(FYSETC_MINI_12864_X_X, FYSETC_MINI_12864_1_2, FYSETC_MINI_12864_2_0, FYSETC_MINI_12864_2_1, FYSETC_GENERIC_12864_1_1) \
+ + COUNT_ENABLED(LCD_SAINSMART_I2C_1602, LCD_SAINSMART_I2C_2004) \
+ + COUNT_ENABLED(MKS_12864OLED, MKS_12864OLED_SSD1306) \
+ + COUNT_ENABLED(MKS_TS35_V2_0, MKS_ROBIN_TFT24, MKS_ROBIN_TFT28, MKS_ROBIN_TFT32, MKS_ROBIN_TFT35, MKS_ROBIN_TFT43, MKS_ROBIN_TFT_V1_1R, ANET_ET4_TFT28, ANET_ET5_TFT35) \
+ + COUNT_ENABLED(TFTGLCD_PANEL_SPI, TFTGLCD_PANEL_I2C) \
+ + COUNT_ENABLED(VIKI2, miniVIKI) \
+ + COUNT_ENABLED(ZONESTAR_12864LCD, ZONESTAR_12864OLED, ZONESTAR_12864OLED_SSD1306) \
+ + COUNT_ENABLED(ANET_FULL_GRAPHICS_LCD, ANET_FULL_GRAPHICS_LCD_ALT_WIRING) \
+ + ENABLED(AZSMZ_12864) \
+ + ENABLED(BQ_LCD_SMART_CONTROLLER) \
+ + ENABLED(CARTESIO_UI) \
+ + ENABLED(ELB_FULL_GRAPHIC_CONTROLLER) \
+ + ENABLED(FF_INTERFACEBOARD) \
+ + ENABLED(FYSETC_242_OLED_12864) \
+ + ENABLED(G3D_PANEL) \
+ + ENABLED(LCD_FOR_MELZI) \
+ + ENABLED(LCD_I2C_PANELOLU2) \
+ + ENABLED(LCD_I2C_VIKI) \
+ + ENABLED(LCM1602) \
+ + ENABLED(LONGER_LK_TFT28) \
+ + ENABLED(MAKEBOARD_MINI_2_LINE_DISPLAY_1602) \
+ + ENABLED(MAKRPANEL) \
+ + ENABLED(MALYAN_LCD) \
+ + ENABLED(MKS_LCD12864) \
+ + ENABLED(OLED_PANEL_TINYBOY2) \
+ + ENABLED(OVERLORD_OLED) \
+ + ENABLED(PANEL_ONE) \
+ + ENABLED(RA_CONTROL_PANEL) \
+ + ENABLED(RADDS_DISPLAY) \
+ + ENABLED(REPRAPWORLD_GRAPHICAL_LCD) \
+ + ENABLED(RIGIDBOT_PANEL) \
+ + ENABLED(SAV_3DGLCD) \
+ + ENABLED(SAV_3DLCD) \
+ + ENABLED(SILVER_GATE_GLCD_CONTROLLER) \
+ + ENABLED(TFT_TRONXY_X5SA) \
+ + ENABLED(TOUCH_UI_FTDI_EVE) \
+ + ENABLED(U8GLIB_SH1106_EINSTART) \
+ + ENABLED(ULTI_CONTROLLER) \
+ + ENABLED(ULTIMAKERCONTROLLER) \
+ + ENABLED(ZONESTAR_LCD)
+ #error "Please select only one LCD controller option."
+#endif
+
+#undef IS_U8GLIB_SSD1306
+#undef IS_EXTUI
+#undef IS_LEGACY_TFT
+
+#if ANY(TFT_GENERIC, MKS_TS35_V2_0, MKS_ROBIN_TFT24, MKS_ROBIN_TFT28, MKS_ROBIN_TFT32, MKS_ROBIN_TFT35, MKS_ROBIN_TFT43, MKS_ROBIN_TFT_V1_1R, TFT_TRONXY_X5SA, ANYCUBIC_TFT35, ANYCUBIC_TFT35, LONGER_LK_TFT28, ANET_ET4_TFT28, ANET_ET5_TFT35)
+ #if NONE(TFT_COLOR_UI, TFT_CLASSIC_UI, TFT_LVGL_UI)
+ #error "TFT_COLOR_UI, TFT_CLASSIC_UI, TFT_LVGL_UI is required for your TFT. Please enable one."
+ #elif 1 < ENABLED(TFT_COLOR_UI) + ENABLED(TFT_CLASSIC_UI) + ENABLED(TFT_LVGL_UI)
+ #error "Please select only one of TFT_COLOR_UI, TFT_CLASSIC_UI, or TFT_LVGL_UI."
+ #endif
+#elif ANY(TFT_COLOR_UI, TFT_CLASSIC_UI, TFT_LVGL_UI)
+ #error "TFT_(COLOR|CLASSIC|LVGL)_UI requires a TFT display to be enabled."
+#endif
+
+#if ENABLED(TFT_GENERIC) && NONE(TFT_INTERFACE_FSMC, TFT_INTERFACE_SPI)
+ #error "TFT_GENERIC requires either TFT_INTERFACE_FSMC or TFT_INTERFACE_SPI interface."
+#endif
+
+#if BOTH(TFT_INTERFACE_FSMC, TFT_INTERFACE_SPI)
+ #error "Please enable only one of TFT_INTERFACE_SPI or TFT_INTERFACE_SPI."
+#endif
+
+#if MANY(LCD_SCREEN_ROT_0, LCD_SCREEN_ROT_90, LCD_SCREEN_ROT_180, LCD_SCREEN_ROT_270)
+ #error "Please enable only one LCD_SCREEN_ROT_* option: 0, 90, 180, or 270."
+#endif
+
+#if MANY(TFT_RES_320x240, TFT_RES_480x272, TFT_RES_480x320)
+ #error "Please select only one of TFT_RES_480x320, TFT_RES_480x320, or TFT_RES_480x272."
+#endif
+
+#if HAS_TFT_LVGL_UI && DISABLED(TFT_RES_480x320)
+ #error "(FMSC|SPI)TFT_LVGL_UI requires TFT_RES_480x320."
+#endif
+
+#if defined(GRAPHICAL_TFT_UPSCALE) && !WITHIN(GRAPHICAL_TFT_UPSCALE, 2, 3)
+ #error "GRAPHICAL_TFT_UPSCALE must be set to 2 or 3."
+#endif
+
+/**
+ * Some boards forbid the use of -1 Native USB
+ */
+#if ENABLED(BOARD_NO_NATIVE_USB)
+ #undef BOARD_NO_NATIVE_USB
+ #if SERIAL_PORT == -1
+ #error "SERIAL_PORT is set to -1, but the MOTHERBOARD has no native USB support. Set SERIAL_PORT to a valid value for your board."
+ #elif SERIAL_PORT_2 == -1
+ #error "SERIAL_PORT_2 is set to -1, but the MOTHERBOARD has no native USB support. Set SERIAL_PORT_2 to a valid value for your board."
+ #elif MMU2_SERIAL_PORT == -1
+ #error "MMU2_SERIAL_PORT is set to -1, but the MOTHERBOARD has no native USB support. Set MMU2_SERIAL_PORT to a valid value for your board."
+ #elif LCD_SERIAL_PORT == -1
+ #error "LCD_SERIAL_PORT is set to -1, but the MOTHERBOARD has no native USB support. Set LCD_SERIAL_PORT to a valid value for your board."
+ #endif
+#endif
+
+/**
+ * MMU2 require a dedicated serial port
+ */
+#ifdef MMU2_SERIAL_PORT
+ #if MMU2_SERIAL_PORT == SERIAL_PORT
+ #error "MMU2_SERIAL_PORT cannot be the same as SERIAL_PORT."
+ #elif defined(SERIAL_PORT_2) && MMU2_SERIAL_PORT == SERIAL_PORT_2
+ #error "MMU2_SERIAL_PORT cannot be the same as SERIAL_PORT_2."
+ #elif defined(LCD_SERIAL_PORT) && MMU2_SERIAL_PORT == LCD_SERIAL_PORT
+ #error "MMU2_SERIAL_PORT cannot be the same as LCD_SERIAL_PORT."
+ #endif
+#endif
+
+/**
+ * Serial displays require a dedicated serial port
+ */
+#ifdef LCD_SERIAL_PORT
+ #if LCD_SERIAL_PORT == SERIAL_PORT
+ #error "LCD_SERIAL_PORT cannot be the same as SERIAL_PORT."
+ #elif defined(SERIAL_PORT_2) && LCD_SERIAL_PORT == SERIAL_PORT_2
+ #error "LCD_SERIAL_PORT cannot be the same as SERIAL_PORT_2."
+ #endif
+#else
+ #if HAS_DGUS_LCD
+ #error "The DGUS LCD requires LCD_SERIAL_PORT to be defined."
+ #elif EITHER(ANYCUBIC_LCD_I3MEGA, ANYCUBIC_LCD_CHIRON)
+ #error "The ANYCUBIC LCD requires LCD_SERIAL_PORT to be defined."
+ #elif ENABLED(MALYAN_LCD)
+ #error "MALYAN_LCD requires LCD_SERIAL_PORT to be defined."
+ #endif
+#endif
+
+/**
+ * FYSETC Mini 12864 RGB backlighting required
+ */
+#if EITHER(FYSETC_MINI_12864_1_2, FYSETC_MINI_12864_2_0) && DISABLED(RGB_LED)
+ #error "RGB_LED is required for FYSETC_MINI_12864 1.2 and 2.0."
+#elif EITHER(FYSETC_MINI_12864_2_0, FYSETC_MINI_12864_2_1) && DISABLED(LED_USER_PRESET_STARTUP)
+ #error "LED_USER_PRESET_STARTUP is required for FYSETC_MINI_12864 2.x displays."
+#endif
+
+/**
+ * Check existing CS pins against enabled TMC SPI drivers.
+ */
+#define INVALID_TMC_SPI(ST) (AXIS_HAS_SPI(ST) && !PIN_EXISTS(ST##_CS))
+#if INVALID_TMC_SPI(X)
+ #error "An SPI driven TMC driver on X requires X_CS_PIN."
+#elif INVALID_TMC_SPI(X2)
+ #error "An SPI driven TMC driver on X2 requires X2_CS_PIN."
+#elif INVALID_TMC_SPI(Y)
+ #error "An SPI driven TMC driver on Y requires Y_CS_PIN."
+#elif INVALID_TMC_SPI(Y2)
+ #error "An SPI driven TMC driver on Y2 requires Y2_CS_PIN."
+#elif INVALID_TMC_SPI(Z)
+ #error "An SPI driven TMC driver on Z requires Z_CS_PIN."
+#elif INVALID_TMC_SPI(Z2)
+ #error "An SPI driven TMC driver on Z2 requires Z2_CS_PIN."
+#elif INVALID_TMC_SPI(Z3)
+ #error "An SPI driven TMC driver on Z3 requires Z3_CS_PIN."
+#elif INVALID_TMC_SPI(Z4)
+ #error "An SPI driven TMC driver on Z4 requires Z4_CS_PIN."
+#elif INVALID_TMC_SPI(E0)
+ #error "An SPI driven TMC driver on E0 requires E0_CS_PIN."
+#elif INVALID_TMC_SPI(E1)
+ #error "An SPI driven TMC driver on E1 requires E1_CS_PIN."
+#elif INVALID_TMC_SPI(E2)
+ #error "An SPI driven TMC driver on E2 requires E2_CS_PIN."
+#elif INVALID_TMC_SPI(E3)
+ #error "An SPI driven TMC driver on E3 requires E3_CS_PIN."
+#elif INVALID_TMC_SPI(E4)
+ #error "An SPI driven TMC driver on E4 requires E4_CS_PIN."
+#elif INVALID_TMC_SPI(E5)
+ #error "An SPI driven TMC driver on E5 requires E5_CS_PIN."
+#elif INVALID_TMC_SPI(E6)
+ #error "An SPI driven TMC driver on E6 requires E6_CS_PIN."
+#elif INVALID_TMC_SPI(E7)
+ #error "An SPI driven TMC driver on E7 requires E7_CS_PIN."
+#endif
+#undef INVALID_TMC_SPI
+
+/**
+ * Check existing RX/TX pins against enable TMC UART drivers.
+ */
+#define INVALID_TMC_UART(ST) (AXIS_HAS_UART(ST) && !(defined(ST##_HARDWARE_SERIAL) || (PINS_EXIST(ST##_SERIAL_RX, ST##_SERIAL_TX))))
+#if INVALID_TMC_UART(X)
+ #error "TMC2208 or TMC2209 on X requires X_HARDWARE_SERIAL or X_SERIAL_(RX|TX)_PIN."
+#elif INVALID_TMC_UART(X2)
+ #error "TMC2208 or TMC2209 on X2 requires X2_HARDWARE_SERIAL or X2_SERIAL_(RX|TX)_PIN."
+#elif INVALID_TMC_UART(Y)
+ #error "TMC2208 or TMC2209 on Y requires Y_HARDWARE_SERIAL or Y_SERIAL_(RX|TX)_PIN."
+#elif INVALID_TMC_UART(Y2)
+ #error "TMC2208 or TMC2209 on Y2 requires Y2_HARDWARE_SERIAL or Y2_SERIAL_(RX|TX)_PIN."
+#elif INVALID_TMC_UART(Z)
+ #error "TMC2208 or TMC2209 on Z requires Z_HARDWARE_SERIAL or Z_SERIAL_(RX|TX)_PIN."
+#elif INVALID_TMC_UART(Z2)
+ #error "TMC2208 or TMC2209 on Z2 requires Z2_HARDWARE_SERIAL or Z2_SERIAL_(RX|TX)_PIN."
+#elif INVALID_TMC_UART(Z3)
+ #error "TMC2208 or TMC2209 on Z3 requires Z3_HARDWARE_SERIAL or Z3_SERIAL_(RX|TX)_PIN."
+#elif INVALID_TMC_UART(Z4)
+ #error "TMC2208 or TMC2209 on Z4 requires Z4_HARDWARE_SERIAL or Z4_SERIAL_(RX|TX)_PIN."
+#elif INVALID_TMC_UART(E0)
+ #error "TMC2208 or TMC2209 on E0 requires E0_HARDWARE_SERIAL or E0_SERIAL_(RX|TX)_PIN."
+#elif INVALID_TMC_UART(E1)
+ #error "TMC2208 or TMC2209 on E1 requires E1_HARDWARE_SERIAL or E1_SERIAL_(RX|TX)_PIN."
+#elif INVALID_TMC_UART(E2)
+ #error "TMC2208 or TMC2209 on E2 requires E2_HARDWARE_SERIAL or E2_SERIAL_(RX|TX)_PIN."
+#elif INVALID_TMC_UART(E3)
+ #error "TMC2208 or TMC2209 on E3 requires E3_HARDWARE_SERIAL or E3_SERIAL_(RX|TX)_PIN."
+#elif INVALID_TMC_UART(E4)
+ #error "TMC2208 or TMC2209 on E4 requires E4_HARDWARE_SERIAL or E4_SERIAL_(RX|TX)_PIN."
+#elif INVALID_TMC_UART(E5)
+ #error "TMC2208 or TMC2209 on E5 requires E5_HARDWARE_SERIAL or E5_SERIAL_(RX|TX)_PIN."
+#elif INVALID_TMC_UART(E6)
+ #error "TMC2208 or TMC2209 on E6 requires E6_HARDWARE_SERIAL or E6_SERIAL_(RX|TX)_PIN."
+#elif INVALID_TMC_UART(E7)
+ #error "TMC2208 or TMC2209 on E7 requires E7_HARDWARE_SERIAL or E7_SERIAL_(RX|TX)_PIN."
+#endif
+#undef INVALID_TMC_UART
+
+/**
+ * TMC2209 slave address values
+ */
+#define INVALID_TMC_ADDRESS(ST) static_assert(0 <= ST##_SLAVE_ADDRESS && ST##_SLAVE_ADDRESS <= 3, "TMC2209 slave address must be 0, 1, 2 or 3")
+#if AXIS_DRIVER_TYPE_X(TMC2209)
+ INVALID_TMC_ADDRESS(X);
+#elif AXIS_DRIVER_TYPE_X2(TMC2209)
+ INVALID_TMC_ADDRESS(X2);
+#elif AXIS_DRIVER_TYPE_Y(TMC2209)
+ INVALID_TMC_ADDRESS(Y);
+#elif AXIS_DRIVER_TYPE_Y2(TMC2209)
+ INVALID_TMC_ADDRESS(Y2);
+#elif AXIS_DRIVER_TYPE_Z(TMC2209)
+ INVALID_TMC_ADDRESS(Z);
+#elif AXIS_DRIVER_TYPE_Z2(TMC2209)
+ INVALID_TMC_ADDRESS(Z2);
+#elif AXIS_DRIVER_TYPE_Z3(TMC2209)
+ INVALID_TMC_ADDRESS(Z3);
+#elif AXIS_DRIVER_TYPE_Z4(TMC2209)
+ INVALID_TMC_ADDRESS(Z4);
+#elif AXIS_DRIVER_TYPE_E0(TMC2209)
+ INVALID_TMC_ADDRESS(E0);
+#elif AXIS_DRIVER_TYPE_E1(TMC2209)
+ INVALID_TMC_ADDRESS(E1);
+#elif AXIS_DRIVER_TYPE_E2(TMC2209)
+ INVALID_TMC_ADDRESS(E2);
+#elif AXIS_DRIVER_TYPE_E3(TMC2209)
+ INVALID_TMC_ADDRESS(E3);
+#elif AXIS_DRIVER_TYPE_E4(TMC2209)
+ INVALID_TMC_ADDRESS(E4);
+#elif AXIS_DRIVER_TYPE_E5(TMC2209)
+ INVALID_TMC_ADDRESS(E5);
+#elif AXIS_DRIVER_TYPE_E6(TMC2209)
+ INVALID_TMC_ADDRESS(E6);
+#elif AXIS_DRIVER_TYPE_E7(TMC2209)
+ INVALID_TMC_ADDRESS(E7);
+#endif
+#undef INVALID_TMC_ADDRESS
+
+#define _TMC_MICROSTEP_IS_VALID(MS) (MS == 0 || MS == 2 || MS == 4 || MS == 8 || MS == 16 || MS == 32 || MS == 64 || MS == 128 || MS == 256)
+#define TMC_MICROSTEP_IS_VALID(M) (!AXIS_IS_TMC(M) || _TMC_MICROSTEP_IS_VALID(M##_MICROSTEPS))
+#define INVALID_TMC_MS(ST) static_assert(0, "Invalid " STRINGIFY(ST) "_MICROSTEPS. Valid values are 0, 2, 4, 8, 16, 32, 64, 128, and 256.")
+
+#if !TMC_MICROSTEP_IS_VALID(X)
+ INVALID_TMC_MS(X);
+#elif !TMC_MICROSTEP_IS_VALID(Y)
+ INVALID_TMC_MS(Y)
+#elif !TMC_MICROSTEP_IS_VALID(Z)
+ INVALID_TMC_MS(Z)
+#elif !TMC_MICROSTEP_IS_VALID(X2)
+ INVALID_TMC_MS(X2);
+#elif !TMC_MICROSTEP_IS_VALID(Y2)
+ INVALID_TMC_MS(Y2)
+#elif !TMC_MICROSTEP_IS_VALID(Z2)
+ INVALID_TMC_MS(Z2)
+#elif !TMC_MICROSTEP_IS_VALID(Z3)
+ INVALID_TMC_MS(Z3)
+#elif !TMC_MICROSTEP_IS_VALID(Z4)
+ INVALID_TMC_MS(Z4)
+#elif !TMC_MICROSTEP_IS_VALID(E0)
+ INVALID_TMC_MS(E0)
+#elif !TMC_MICROSTEP_IS_VALID(E1)
+ INVALID_TMC_MS(E1)
+#elif !TMC_MICROSTEP_IS_VALID(E2)
+ INVALID_TMC_MS(E2)
+#elif !TMC_MICROSTEP_IS_VALID(E3)
+ INVALID_TMC_MS(E3)
+#elif !TMC_MICROSTEP_IS_VALID(E4)
+ INVALID_TMC_MS(E4)
+#elif !TMC_MICROSTEP_IS_VALID(E5)
+ INVALID_TMC_MS(E5)
+#elif !TMC_MICROSTEP_IS_VALID(E6)
+ INVALID_TMC_MS(E6)
+#elif !TMC_MICROSTEP_IS_VALID(E7)
+ INVALID_TMC_MS(E7)
+#endif
+#undef INVALID_TMC_MS
+#undef TMC_MICROSTEP_IS_VALID
+#undef _TMC_MICROSTEP_IS_VALID
+
+#if ENABLED(DELTA) && (ENABLED(STEALTHCHOP_XY) != ENABLED(STEALTHCHOP_Z))
+ #error "STEALTHCHOP_XY and STEALTHCHOP_Z must be the same on DELTA."
+#endif
+
+#if ENABLED(SENSORLESS_HOMING)
+ // Require STEALTHCHOP for SENSORLESS_HOMING on DELTA as the transition from spreadCycle to stealthChop
+ // is necessary in order to reset the stallGuard indication between the initial movement of all three
+ // towers to +Z and the individual homing of each tower. This restriction can be removed once a means of
+ // clearing the stallGuard activated status is found.
+
+ // Stall detection DIAG = HIGH : TMC2209
+ // Stall detection DIAG = LOW : TMC2130/TMC2160/TMC2660/TMC5130/TMC5160
+ #define X_ENDSTOP_INVERTING !AXIS_DRIVER_TYPE(X,TMC2209)
+ #define Y_ENDSTOP_INVERTING !AXIS_DRIVER_TYPE(Y,TMC2209)
+ #define Z_ENDSTOP_INVERTING !AXIS_DRIVER_TYPE(Z,TMC2209)
+
+ #if NONE(SPI_ENDSTOPS, ONBOARD_ENDSTOPPULLUPS, ENDSTOPPULLUPS)
+ #if X_SENSORLESS && X_HOME_DIR < 0 && DISABLED(ENDSTOPPULLUP_XMIN)
+ #error "SENSORLESS_HOMING requires ENDSTOPPULLUP_XMIN (or ENDSTOPPULLUPS) when homing to X_MIN."
+ #elif X_SENSORLESS && X_HOME_DIR > 0 && DISABLED(ENDSTOPPULLUP_XMAX)
+ #error "SENSORLESS_HOMING requires ENDSTOPPULLUP_XMAX (or ENDSTOPPULLUPS) when homing to X_MAX."
+ #elif Y_SENSORLESS && Y_HOME_DIR < 0 && DISABLED(ENDSTOPPULLUP_YMIN)
+ #error "SENSORLESS_HOMING requires ENDSTOPPULLUP_YMIN (or ENDSTOPPULLUPS) when homing to Y_MIN."
+ #elif Y_SENSORLESS && Y_HOME_DIR > 0 && DISABLED(ENDSTOPPULLUP_YMAX)
+ #error "SENSORLESS_HOMING requires ENDSTOPPULLUP_YMAX (or ENDSTOPPULLUPS) when homing to Y_MAX."
+ #elif Z_SENSORLESS && Z_HOME_DIR < 0 && DISABLED(ENDSTOPPULLUP_ZMIN)
+ #error "SENSORLESS_HOMING requires ENDSTOPPULLUP_ZMIN (or ENDSTOPPULLUPS) when homing to Z_MIN."
+ #elif Z_SENSORLESS && Z_HOME_DIR > 0 && DISABLED(ENDSTOPPULLUP_ZMAX)
+ #error "SENSORLESS_HOMING requires ENDSTOPPULLUP_ZMAX (or ENDSTOPPULLUPS) when homing to Z_MAX."
+ #endif
+ #endif
+
+ #if ENABLED(SPI_ENDSTOPS)
+ #if ENABLED(QUICK_HOME)
+ #warning "SPI_ENDSTOPS may be unreliable with QUICK_HOME. Adjust back-offs for better results."
+ #endif
+ #else
+ #if X_SENSORLESS && X_HOME_DIR < 0 && X_MIN_ENDSTOP_INVERTING != X_ENDSTOP_INVERTING
+ #if X_ENDSTOP_INVERTING
+ #error "SENSORLESS_HOMING requires X_MIN_ENDSTOP_INVERTING = true when homing to X_MIN."
+ #else
+ #error "SENSORLESS_HOMING requires X_MIN_ENDSTOP_INVERTING = false when homing TMC2209 to X_MIN."
+ #endif
+ #elif X_SENSORLESS && X_HOME_DIR > 0 && X_MAX_ENDSTOP_INVERTING != X_ENDSTOP_INVERTING
+ #if X_ENDSTOP_INVERTING
+ #error "SENSORLESS_HOMING requires X_MAX_ENDSTOP_INVERTING = true when homing to X_MAX."
+ #else
+ #error "SENSORLESS_HOMING requires X_MAX_ENDSTOP_INVERTING = false when homing TMC2209 to X_MAX."
+ #endif
+ #elif Y_SENSORLESS && Y_HOME_DIR < 0 && Y_MIN_ENDSTOP_INVERTING != Y_ENDSTOP_INVERTING
+ #if Y_ENDSTOP_INVERTING
+ #error "SENSORLESS_HOMING requires Y_MIN_ENDSTOP_INVERTING = true when homing to Y_MIN."
+ #else
+ #error "SENSORLESS_HOMING requires Y_MIN_ENDSTOP_INVERTING = false when homing TMC2209 to Y_MIN."
+ #endif
+ #elif Y_SENSORLESS && Y_HOME_DIR > 0 && Y_MAX_ENDSTOP_INVERTING != Y_ENDSTOP_INVERTING
+ #if Y_ENDSTOP_INVERTING
+ #error "SENSORLESS_HOMING requires Y_MAX_ENDSTOP_INVERTING = true when homing to Y_MAX."
+ #else
+ #error "SENSORLESS_HOMING requires Y_MAX_ENDSTOP_INVERTING = false when homing TMC2209 to Y_MAX."
+ #endif
+ #elif Z_SENSORLESS && Z_HOME_DIR < 0 && Z_MIN_ENDSTOP_INVERTING != Z_ENDSTOP_INVERTING
+ #if Z_ENDSTOP_INVERTING
+ #error "SENSORLESS_HOMING requires Z_MIN_ENDSTOP_INVERTING = true when homing to Z_MIN."
+ #else
+ #error "SENSORLESS_HOMING requires Z_MIN_ENDSTOP_INVERTING = false when homing TMC2209 to Z_MIN."
+ #endif
+ #elif Z_SENSORLESS && Z_HOME_DIR > 0 && Z_MAX_ENDSTOP_INVERTING != Z_ENDSTOP_INVERTING
+ #if Z_ENDSTOP_INVERTING
+ #error "SENSORLESS_HOMING requires Z_MAX_ENDSTOP_INVERTING = true when homing to Z_MAX."
+ #else
+ #error "SENSORLESS_HOMING requires Z_MAX_ENDSTOP_INVERTING = false when homing TMC2209 to Z_MAX."
+ #endif
+ #endif
+ #endif
+
+ #if ENABLED(DELTA) && !BOTH(STEALTHCHOP_XY, STEALTHCHOP_Z)
+ #error "SENSORLESS_HOMING on DELTA currently requires STEALTHCHOP_XY and STEALTHCHOP_Z."
+ #elif ENDSTOP_NOISE_THRESHOLD
+ #error "SENSORLESS_HOMING is incompatible with ENDSTOP_NOISE_THRESHOLD."
+ #elif !(X_SENSORLESS || Y_SENSORLESS || Z_SENSORLESS)
+ #error "SENSORLESS_HOMING requires a TMC stepper driver with StallGuard on X, Y, or Z axes."
+ #endif
+
+ #undef X_ENDSTOP_INVERTING
+ #undef Y_ENDSTOP_INVERTING
+ #undef Z_ENDSTOP_INVERTING
+#endif
+
+// Sensorless probing requirements
+#if ENABLED(SENSORLESS_PROBING)
+ #if ENABLED(DELTA) && !(X_SENSORLESS && Y_SENSORLESS && Z_SENSORLESS)
+ #error "SENSORLESS_PROBING for DELTA requires TMC stepper drivers with StallGuard on X, Y, and Z axes."
+ #elif ENABLED(Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN)
+ #error "SENSORLESS_PROBING cannot be used with Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN."
+ #elif ENABLED(USE_PROBE_FOR_Z_HOMING)
+ #error "SENSORLESS_PROBING cannot be used with USE_PROBE_FOR_Z_HOMING."
+ #elif !Z_SENSORLESS
+ #error "SENSORLESS_PROBING requires a TMC stepper driver with StallGuard on Z."
+ #endif
+#endif
+
+// Sensorless homing is required for both combined steppers in an H-bot
+#if CORE_IS_XY && X_SENSORLESS != Y_SENSORLESS
+ #error "CoreXY requires both X and Y to use sensorless homing if either one does."
+#elif CORE_IS_XZ && X_SENSORLESS != Z_SENSORLESS && !HOMING_Z_WITH_PROBE
+ #error "CoreXZ requires both X and Z to use sensorless homing if either one does."
+#elif CORE_IS_YZ && Y_SENSORLESS != Z_SENSORLESS && !HOMING_Z_WITH_PROBE
+ #error "CoreYZ requires both Y and Z to use sensorless homing if either one does."
+#elif ENABLED(MARKFORGED_XY) && X_SENSORLESS != Y_SENSORLESS
+ #error "MARKFORGED_XY requires both X and Y to use sensorless homing if either one does."
+#endif
+
+// Other TMC feature requirements
+#if ENABLED(HYBRID_THRESHOLD) && !STEALTHCHOP_ENABLED
+ #error "Enable STEALTHCHOP_(XY|Z|E) to use HYBRID_THRESHOLD."
+#elif ENABLED(SENSORLESS_HOMING) && !HAS_STALLGUARD
+ #error "SENSORLESS_HOMING requires TMC2130, TMC2160, TMC2209, TMC2660, or TMC5160 stepper drivers."
+#elif ENABLED(SENSORLESS_PROBING) && !HAS_STALLGUARD
+ #error "SENSORLESS_PROBING requires TMC2130, TMC2160, TMC2209, TMC2660, or TMC5160 stepper drivers."
+#elif STEALTHCHOP_ENABLED && !HAS_STEALTHCHOP
+ #error "STEALTHCHOP requires TMC2130, TMC2160, TMC2208, TMC2209, or TMC5160 stepper drivers."
+#endif
+
+/**
+ * TMC SPI Chaining
+ */
+#define IN_CHAIN(A) ((A##_CHAIN_POS > 0) && !HAS_L64XX)
+#if IN_CHAIN(X ) || IN_CHAIN(Y ) || IN_CHAIN(Z ) || IN_CHAIN(X2) || IN_CHAIN(Y2) || IN_CHAIN(Z2) || IN_CHAIN(Z3) || IN_CHAIN(Z4) \
+ || IN_CHAIN(E0) || IN_CHAIN(E1) || IN_CHAIN(E2) || IN_CHAIN(E3) || IN_CHAIN(E4) || IN_CHAIN(E5) || IN_CHAIN(E6) || IN_CHAIN(E7)
+ #define BAD_CHAIN(A) (IN_CHAIN(A) && !PIN_EXISTS(A##_CS))
+ #if BAD_CHAIN(X ) || BAD_CHAIN(Y ) || BAD_CHAIN(Z ) || BAD_CHAIN(X2) || BAD_CHAIN(Y2) || BAD_CHAIN(Z2) || BAD_CHAIN(Z3) || BAD_CHAIN(Z4) \
+ || BAD_CHAIN(E0) || BAD_CHAIN(E1) || BAD_CHAIN(E2) || BAD_CHAIN(E3) || BAD_CHAIN(E4) || BAD_CHAIN(E5) || BAD_CHAIN(E6) || BAD_CHAIN(E7)
+ #error "All chained TMC drivers need a CS pin."
+ #else
+ #if IN_CHAIN(X)
+ #define CS_COMPARE X_CS_PIN
+ #elif IN_CHAIN(Y)
+ #define CS_COMPARE Y_CS_PIN
+ #elif IN_CHAIN(Z)
+ #define CS_COMPARE Z_CS_PIN
+ #elif IN_CHAIN(X2)
+ #define CS_COMPARE X2_CS_PIN
+ #elif IN_CHAIN(Y2)
+ #define CS_COMPARE Y2_CS_PIN
+ #elif IN_CHAIN(Z2)
+ #define CS_COMPARE Z2_CS_PIN
+ #elif IN_CHAIN(Z3)
+ #define CS_COMPARE Z3_CS_PIN
+ #elif IN_CHAIN(E0)
+ #define CS_COMPARE E0_CS_PIN
+ #elif IN_CHAIN(E1)
+ #define CS_COMPARE E1_CS_PIN
+ #elif IN_CHAIN(E2)
+ #define CS_COMPARE E2_CS_PIN
+ #elif IN_CHAIN(E3)
+ #define CS_COMPARE E3_CS_PIN
+ #elif IN_CHAIN(E4)
+ #define CS_COMPARE E4_CS_PIN
+ #elif IN_CHAIN(E5)
+ #define CS_COMPARE E5_CS_PIN
+ #elif IN_CHAIN(E6)
+ #define CS_COMPARE E6_CS_PIN
+ #elif IN_CHAIN(E7)
+ #define CS_COMPARE E7_CS_PIN
+ #endif
+ #define BAD_CS_PIN(A) (IN_CHAIN(A) && A##_CS_PIN != CS_COMPARE)
+ #if BAD_CS_PIN(X ) || BAD_CS_PIN(Y ) || BAD_CS_PIN(Z ) || BAD_CS_PIN(X2) || BAD_CS_PIN(Y2) || BAD_CS_PIN(Z2) || BAD_CS_PIN(Z3) || BAD_CS_PIN(Z4) \
+ || BAD_CS_PIN(E0) || BAD_CS_PIN(E1) || BAD_CS_PIN(E2) || BAD_CS_PIN(E3) || BAD_CS_PIN(E4) || BAD_CS_PIN(E5) || BAD_CS_PIN(E6) || BAD_CS_PIN(E7)
+ #error "All chained TMC drivers must use the same CS pin."
+ #endif
+ #undef BAD_CS_PIN
+ #undef CS_COMPARE
+ #endif
+ #undef BAD_CHAIN
+#endif
+#undef IN_CHAIN
+
+/**
+ * Digipot requirement
+ */
+#if HAS_MOTOR_CURRENT_I2C
+ #if BOTH(DIGIPOT_MCP4018, DIGIPOT_MCP4451)
+ #error "Enable only one of DIGIPOT_MCP4018 or DIGIPOT_MCP4451."
+ #elif !MB(MKS_SBASE) \
+ && (!defined(DIGIPOTS_I2C_SDA_X) || !defined(DIGIPOTS_I2C_SDA_Y) || !defined(DIGIPOTS_I2C_SDA_Z) || !defined(DIGIPOTS_I2C_SDA_E0) || !defined(DIGIPOTS_I2C_SDA_E1))
+ #error "DIGIPOT_MCP4018/4451 requires DIGIPOTS_I2C_SDA_* pins to be defined."
+ #endif
+#endif
+
+/**
+ * Check per-axis initializers for errors
+ */
+constexpr float sanity_arr_1[] = DEFAULT_AXIS_STEPS_PER_UNIT,
+ sanity_arr_2[] = DEFAULT_MAX_FEEDRATE,
+ sanity_arr_3[] = DEFAULT_MAX_ACCELERATION;
+
+#define _ARR_TEST(N,I) (sanity_arr_##N[_MIN(I,int(COUNT(sanity_arr_##N))-1)] > 0)
+
+static_assert(COUNT(sanity_arr_1) >= XYZE, "DEFAULT_AXIS_STEPS_PER_UNIT requires X, Y, Z and E elements.");
+static_assert(COUNT(sanity_arr_1) <= XYZE_N, "DEFAULT_AXIS_STEPS_PER_UNIT has too many elements. (Did you forget to enable DISTINCT_E_FACTORS?)");
+static_assert( _ARR_TEST(1,0) && _ARR_TEST(1,1) && _ARR_TEST(1,2)
+ && _ARR_TEST(1,3) && _ARR_TEST(1,4) && _ARR_TEST(1,5)
+ && _ARR_TEST(1,6) && _ARR_TEST(1,7) && _ARR_TEST(1,8),
+ "DEFAULT_AXIS_STEPS_PER_UNIT values must be positive.");
+
+static_assert(COUNT(sanity_arr_2) >= XYZE, "DEFAULT_MAX_FEEDRATE requires X, Y, Z and E elements.");
+static_assert(COUNT(sanity_arr_2) <= XYZE_N, "DEFAULT_MAX_FEEDRATE has too many elements. (Did you forget to enable DISTINCT_E_FACTORS?)");
+static_assert( _ARR_TEST(2,0) && _ARR_TEST(2,1) && _ARR_TEST(2,2)
+ && _ARR_TEST(2,3) && _ARR_TEST(2,4) && _ARR_TEST(2,5)
+ && _ARR_TEST(2,6) && _ARR_TEST(2,7) && _ARR_TEST(2,8),
+ "DEFAULT_MAX_FEEDRATE values must be positive.");
+
+static_assert(COUNT(sanity_arr_3) >= XYZE, "DEFAULT_MAX_ACCELERATION requires X, Y, Z and E elements.");
+static_assert(COUNT(sanity_arr_3) <= XYZE_N, "DEFAULT_MAX_ACCELERATION has too many elements. (Did you forget to enable DISTINCT_E_FACTORS?)");
+static_assert( _ARR_TEST(3,0) && _ARR_TEST(3,1) && _ARR_TEST(3,2)
+ && _ARR_TEST(3,3) && _ARR_TEST(3,4) && _ARR_TEST(3,5)
+ && _ARR_TEST(3,6) && _ARR_TEST(3,7) && _ARR_TEST(3,8),
+ "DEFAULT_MAX_ACCELERATION values must be positive.");
+
+#if ENABLED(LIMITED_MAX_ACCEL_EDITING)
+ #ifdef MAX_ACCEL_EDIT_VALUES
+ constexpr float sanity_arr_4[] = MAX_ACCEL_EDIT_VALUES;
+ static_assert(COUNT(sanity_arr_4) >= XYZE, "MAX_ACCEL_EDIT_VALUES requires X, Y, Z and E elements.");
+ static_assert(COUNT(sanity_arr_4) <= XYZE, "MAX_ACCEL_EDIT_VALUES has too many elements. X, Y, Z and E elements only.");
+ static_assert( _ARR_TEST(4,0) && _ARR_TEST(4,1) && _ARR_TEST(4,2)
+ && _ARR_TEST(4,3) && _ARR_TEST(4,4) && _ARR_TEST(4,5)
+ && _ARR_TEST(4,6) && _ARR_TEST(4,7) && _ARR_TEST(4,8),
+ "MAX_ACCEL_EDIT_VALUES values must be positive.");
+ #endif
+#endif
+
+#if ENABLED(LIMITED_MAX_FR_EDITING)
+ #ifdef MAX_FEEDRATE_EDIT_VALUES
+ constexpr float sanity_arr_5[] = MAX_FEEDRATE_EDIT_VALUES;
+ static_assert(COUNT(sanity_arr_5) >= XYZE, "MAX_FEEDRATE_EDIT_VALUES requires X, Y, Z and E elements.");
+ static_assert(COUNT(sanity_arr_5) <= XYZE, "MAX_FEEDRATE_EDIT_VALUES has too many elements. X, Y, Z and E elements only.");
+ static_assert( _ARR_TEST(5,0) && _ARR_TEST(5,1) && _ARR_TEST(5,2)
+ && _ARR_TEST(5,3) && _ARR_TEST(5,4) && _ARR_TEST(5,5)
+ && _ARR_TEST(5,6) && _ARR_TEST(5,7) && _ARR_TEST(5,8),
+ "MAX_FEEDRATE_EDIT_VALUES values must be positive.");
+ #endif
+#endif
+
+#if ENABLED(LIMITED_JERK_EDITING)
+ #ifdef MAX_JERK_EDIT_VALUES
+ constexpr float sanity_arr_6[] = MAX_JERK_EDIT_VALUES;
+ static_assert(COUNT(sanity_arr_6) >= XYZE, "MAX_JERK_EDIT_VALUES requires X, Y, Z and E elements.");
+ static_assert(COUNT(sanity_arr_6) <= XYZE, "MAX_JERK_EDIT_VALUES has too many elements. X, Y, Z and E elements only.");
+ static_assert( _ARR_TEST(6,0) && _ARR_TEST(6,1) && _ARR_TEST(6,2)
+ && _ARR_TEST(6,3) && _ARR_TEST(6,4) && _ARR_TEST(6,5)
+ && _ARR_TEST(6,6) && _ARR_TEST(6,7) && _ARR_TEST(6,8),
+ "MAX_JERK_EDIT_VALUES values must be positive.");
+ #endif
+#endif
+
+#undef _ARR_TEST
+
+#if BOTH(CNC_COORDINATE_SYSTEMS, NO_WORKSPACE_OFFSETS)
+ #error "CNC_COORDINATE_SYSTEMS is incompatible with NO_WORKSPACE_OFFSETS."
+#endif
+
+#if !BLOCK_BUFFER_SIZE || !IS_POWER_OF_2(BLOCK_BUFFER_SIZE)
+ #error "BLOCK_BUFFER_SIZE must be a power of 2."
+#elif BLOCK_BUFFER_SIZE > 64
+ #error "A very large BLOCK_BUFFER_SIZE is not needed and takes longer to drain the buffer on pause / cancel."
+#endif
+
+#if ENABLED(LED_CONTROL_MENU) && !IS_ULTIPANEL
+ #error "LED_CONTROL_MENU requires an LCD controller."
+#endif
+
+#if ENABLED(CASE_LIGHT_USE_NEOPIXEL) && DISABLED(NEOPIXEL_LED)
+ #error "CASE_LIGHT_USE_NEOPIXEL requires NEOPIXEL_LED."
+#endif
+
+#if ENABLED(SKEW_CORRECTION)
+ #if !defined(XY_SKEW_FACTOR) && !(defined(XY_DIAG_AC) && defined(XY_DIAG_BD) && defined(XY_SIDE_AD))
+ #error "SKEW_CORRECTION requires XY_SKEW_FACTOR or XY_DIAG_AC, XY_DIAG_BD, XY_SIDE_AD."
+ #endif
+ #if ENABLED(SKEW_CORRECTION_FOR_Z)
+ #if !defined(XZ_SKEW_FACTOR) && !(defined(XZ_DIAG_AC) && defined(XZ_DIAG_BD) && defined(XZ_SIDE_AD))
+ #error "SKEW_CORRECTION requires XZ_SKEW_FACTOR or XZ_DIAG_AC, XZ_DIAG_BD, XZ_SIDE_AD."
+ #endif
+ #if !defined(YZ_SKEW_FACTOR) && !(defined(YZ_DIAG_AC) && defined(YZ_DIAG_BD) && defined(YZ_SIDE_AD))
+ #error "SKEW_CORRECTION requires YZ_SKEW_FACTOR or YZ_DIAG_AC, YZ_DIAG_BD, YZ_SIDE_AD."
+ #endif
+ #endif
+#endif
+
+#if ENABLED(BACKUP_POWER_SUPPLY) && !PIN_EXISTS(POWER_LOSS)
+ #error "BACKUP_POWER_SUPPLY requires a POWER_LOSS_PIN."
+#endif
+
+#if BOTH(POWER_LOSS_PULLUP, POWER_LOSS_PULLDOWN)
+ #error "You can't enable POWER_LOSS_PULLUP and POWER_LOSS_PULLDOWN at the same time."
+#endif
+
+#if ENABLED(Z_STEPPER_AUTO_ALIGN)
+ #if NUM_Z_STEPPER_DRIVERS <= 1
+ #error "Z_STEPPER_AUTO_ALIGN requires NUM_Z_STEPPER_DRIVERS greater than 1."
+ #elif !HAS_BED_PROBE
+ #error "Z_STEPPER_AUTO_ALIGN requires a Z-bed probe."
+ #elif ENABLED(Z_STEPPER_ALIGN_KNOWN_STEPPER_POSITIONS)
+ static_assert(WITHIN(Z_STEPPER_ALIGN_AMP, 0.5, 2.0), "Z_STEPPER_ALIGN_AMP must be between 0.5 and 2.0.");
+ #if NUM_Z_STEPPER_DRIVERS < 3
+ #error "Z_STEPPER_ALIGN_KNOWN_STEPPER_POSITIONS requires NUM_Z_STEPPER_DRIVERS to be 3 or 4."
+ #endif
+ #endif
+#endif
+
+#if ENABLED(MECHANICAL_GANTRY_CALIBRATION)
+ #if NONE(HAS_MOTOR_CURRENT_DAC, HAS_MOTOR_CURRENT_SPI, HAS_MOTOR_CURRENT_DAC, HAS_TRINAMIC_CONFIG, HAS_MOTOR_CURRENT_PWM)
+ #error "It is highly recommended to have adjustable current drivers to prevent damage. Disable this line to continue anyway."
+ #elif !defined(GANTRY_CALIBRATION_CURRENT)
+ #error "MECHANICAL_GANTRY_CALIBRATION Requires GANTRY_CALIBRATION_CURRENT to be set."
+ #elif !defined(GANTRY_CALIBRATION_EXTRA_HEIGHT)
+ #error "MECHANICAL_GANTRY_CALIBRATION Requires GANTRY_CALIBRATION_EXTRA_HEIGHT to be set."
+ #elif !defined(GANTRY_CALIBRATION_FEEDRATE)
+ #error "MECHANICAL_GANTRY_CALIBRATION Requires GANTRY_CALIBRATION_FEEDRATE to be set."
+ #endif
+ #if defined(GANTRY_CALIBRATION_SAFE_POSITION) && !defined(GANTRY_CALIBRATION_XY_PARK_FEEDRATE)
+ #error "GANTRY_CALIBRATION_SAFE_POSITION Requires GANTRY_CALIBRATION_XY_PARK_FEEDRATE to be set."
+ #endif
+#endif
+
+#if BOTH(Z_STEPPER_AUTO_ALIGN, MECHANICAL_GANTRY_CALIBRATION)
+ #error "You cannot use Z_STEPPER_AUTO_ALIGN and MECHANICAL_GANTRY_CALIBRATION at the same time."
+#endif
+
+#if ENABLED(PRINTCOUNTER) && DISABLED(EEPROM_SETTINGS)
+ #error "PRINTCOUNTER requires EEPROM_SETTINGS."
+#endif
+
+#if ENABLED(USB_FLASH_DRIVE_SUPPORT) && !PINS_EXIST(USB_CS, USB_INTR) && DISABLED(USE_OTG_USB_HOST)
+ #error "USB_CS_PIN and USB_INTR_PIN are required for USB_FLASH_DRIVE_SUPPORT."
+#endif
+
+#if ENABLED(USE_OTG_USB_HOST) && !defined(HAS_OTG_USB_HOST_SUPPORT)
+ #error "The current board does not support USE_OTG_USB_HOST."
+#endif
+
+#if ENABLED(SD_FIRMWARE_UPDATE) && !defined(__AVR_ATmega2560__)
+ #error "SD_FIRMWARE_UPDATE requires an ATmega2560-based (Arduino Mega) board."
+#endif
+
+#if ENABLED(GCODE_MACROS) && !WITHIN(GCODE_MACROS_SLOTS, 1, 10)
+ #error "GCODE_MACROS_SLOTS must be a number from 1 to 10."
+#endif
+
+#if ENABLED(CUSTOM_USER_MENUS)
+ #ifdef USER_GCODE_1
+ constexpr char _chr1 = USER_GCODE_1[strlen(USER_GCODE_1) - 1];
+ static_assert(_chr1 != '\n' && _chr1 != '\r', "USER_GCODE_1 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_2
+ constexpr char _chr2 = USER_GCODE_2[strlen(USER_GCODE_2) - 1];
+ static_assert(_chr2 != '\n' && _chr2 != '\r', "USER_GCODE_2 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_3
+ constexpr char _chr3 = USER_GCODE_3[strlen(USER_GCODE_3) - 1];
+ static_assert(_chr3 != '\n' && _chr3 != '\r', "USER_GCODE_3 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_4
+ constexpr char _chr4 = USER_GCODE_4[strlen(USER_GCODE_4) - 1];
+ static_assert(_chr4 != '\n' && _chr4 != '\r', "USER_GCODE_4 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_5
+ constexpr char _chr5 = USER_GCODE_5[strlen(USER_GCODE_5) - 1];
+ static_assert(_chr5 != '\n' && _chr5 != '\r', "USER_GCODE_5 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_6
+ constexpr char _chr6 = USER_GCODE_6[strlen(USER_GCODE_6) - 1];
+ static_assert(_chr6 != '\n' && _chr6 != '\r', "USER_GCODE_6 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_7
+ constexpr char _chr7 = USER_GCODE_7[strlen(USER_GCODE_7) - 1];
+ static_assert(_chr7 != '\n' && _chr7 != '\r', "USER_GCODE_7 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_8
+ constexpr char _chr8 = USER_GCODE_8[strlen(USER_GCODE_8) - 1];
+ static_assert(_chr8 != '\n' && _chr8 != '\r', "USER_GCODE_8 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_9
+ constexpr char _chr9 = USER_GCODE_9[strlen(USER_GCODE_9) - 1];
+ static_assert(_chr9 != '\n' && _chr9 != '\r', "USER_GCODE_9 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_10
+ constexpr char _chr10 = USER_GCODE_10[strlen(USER_GCODE_10) - 1];
+ static_assert(_chr10 != '\n' && _chr10 != '\r', "USER_GCODE_10 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_11
+ constexpr char _chr11 = USER_GCODE_11[strlen(USER_GCODE_11) - 1];
+ static_assert(_chr11 != '\n' && _chr11 != '\r', "USER_GCODE_11 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_12
+ constexpr char _chr12 = USER_GCODE_12[strlen(USER_GCODE_12) - 1];
+ static_assert(_chr12 != '\n' && _chr12 != '\r', "USER_GCODE_12 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_13
+ constexpr char _chr13 = USER_GCODE_13[strlen(USER_GCODE_13) - 1];
+ static_assert(_chr13 != '\n' && _chr13 != '\r', "USER_GCODE_13 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_14
+ constexpr char _chr14 = USER_GCODE_14[strlen(USER_GCODE_14) - 1];
+ static_assert(_chr14 != '\n' && _chr14 != '\r', "USER_GCODE_14 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_15
+ constexpr char _chr15 = USER_GCODE_15[strlen(USER_GCODE_15) - 1];
+ static_assert(_chr15 != '\n' && _chr15 != '\r', "USER_GCODE_15 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_16
+ constexpr char _chr16 = USER_GCODE_16[strlen(USER_GCODE_16) - 1];
+ static_assert(_chr16 != '\n' && _chr16 != '\r', "USER_GCODE_16 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_17
+ constexpr char _chr17 = USER_GCODE_17[strlen(USER_GCODE_17) - 1];
+ static_assert(_chr17 != '\n' && _chr17 != '\r', "USER_GCODE_17 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_18
+ constexpr char _chr18 = USER_GCODE_18[strlen(USER_GCODE_18) - 1];
+ static_assert(_chr18 != '\n' && _chr18 != '\r', "USER_GCODE_18 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_19
+ constexpr char _chr19 = USER_GCODE_19[strlen(USER_GCODE_19) - 1];
+ static_assert(_chr19 != '\n' && _chr19 != '\r', "USER_GCODE_19 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_20
+ constexpr char _chr20 = USER_GCODE_20[strlen(USER_GCODE_20) - 1];
+ static_assert(_chr20 != '\n' && _chr20 != '\r', "USER_GCODE_20 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_21
+ constexpr char _chr21 = USER_GCODE_21[strlen(USER_GCODE_21) - 1];
+ static_assert(_chr21 != '\n' && _chr21 != '\r', "USER_GCODE_21 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_22
+ constexpr char _chr22 = USER_GCODE_22[strlen(USER_GCODE_22) - 1];
+ static_assert(_chr22 != '\n' && _chr22 != '\r', "USER_GCODE_22 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_23
+ constexpr char _chr23 = USER_GCODE_23[strlen(USER_GCODE_23) - 1];
+ static_assert(_chr23 != '\n' && _chr23 != '\r', "USER_GCODE_23 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_24
+ constexpr char _chr24 = USER_GCODE_24[strlen(USER_GCODE_24) - 1];
+ static_assert(_chr24 != '\n' && _chr24 != '\r', "USER_GCODE_24 cannot have a newline at the end. Please remove it.");
+ #endif
+ #ifdef USER_GCODE_25
+ constexpr char _chr25 = USER_GCODE_25[strlen(USER_GCODE_25) - 1];
+ static_assert(_chr25 != '\n' && _chr25 != '\r', "USER_GCODE_25 cannot have a newline at the end. Please remove it.");
+ #endif
+#endif
+
+#if ENABLED(BACKLASH_COMPENSATION)
+ #ifndef BACKLASH_DISTANCE_MM
+ #error "BACKLASH_COMPENSATION requires BACKLASH_DISTANCE_MM."
+ #elif !defined(BACKLASH_CORRECTION)
+ #error "BACKLASH_COMPENSATION requires BACKLASH_CORRECTION."
+ #elif ENABLED(MARKFORGED_XY)
+ constexpr float backlash_arr[] = BACKLASH_DISTANCE_MM;
+ static_assert(!backlash_arr[CORE_AXIS_1] && !backlash_arr[CORE_AXIS_2],
+ "BACKLASH_COMPENSATION can only apply to " STRINGIFY(NORMAL_AXIS) " on a MarkForged system.");
+ #elif IS_CORE
+ constexpr float backlash_arr[] = BACKLASH_DISTANCE_MM;
+ static_assert(!backlash_arr[CORE_AXIS_1] && !backlash_arr[CORE_AXIS_2],
+ "BACKLASH_COMPENSATION can only apply to " STRINGIFY(NORMAL_AXIS) " with your CORE system.");
+ #endif
+#endif
+
+#if ENABLED(GRADIENT_MIX) && MIXING_VIRTUAL_TOOLS < 2
+ #error "GRADIENT_MIX requires 2 or more MIXING_VIRTUAL_TOOLS."
+#endif
+
+/**
+ * Photo G-code requirements
+ */
+#if ENABLED(PHOTO_GCODE)
+ #if (PIN_EXISTS(CHDK) + PIN_EXISTS(PHOTOGRAPH) + defined(PHOTO_SWITCH_POSITION)) > 1
+ #error "Please define only one of CHDK_PIN, PHOTOGRAPH_PIN, or PHOTO_SWITCH_POSITION."
+ #elif defined(PHOTO_SWITCH_POSITION) && !defined(PHOTO_POSITION)
+ #error "PHOTO_SWITCH_POSITION requires PHOTO_POSITION."
+ #elif PIN_EXISTS(CHDK) && defined(CHDK_DELAY)
+ #error "CHDK_DELAY has been replaced by PHOTO_SWITCH_MS."
+ #elif PIN_EXISTS(CHDK) && !defined(PHOTO_SWITCH_MS)
+ #error "PHOTO_SWITCH_MS is required with CHDK_PIN."
+ #elif defined(PHOTO_RETRACT_MM)
+ static_assert(PHOTO_RETRACT_MM + 0 >= 0, "PHOTO_RETRACT_MM must be >= 0.");
+ #endif
+#endif
+
+/**
+ * Advanced PRINTCOUNTER settings
+ */
+#if ENABLED(PRINTCOUNTER)
+ #if defined(SERVICE_INTERVAL_1) != defined(SERVICE_NAME_1)
+ #error "Both SERVICE_NAME_1 and SERVICE_INTERVAL_1 are required."
+ #elif defined(SERVICE_INTERVAL_2) != defined(SERVICE_NAME_2)
+ #error "Both SERVICE_NAME_2 and SERVICE_INTERVAL_2 are required."
+ #elif defined(SERVICE_INTERVAL_3) != defined(SERVICE_NAME_3)
+ #error "Both SERVICE_NAME_3 and SERVICE_INTERVAL_3 are required."
+ #endif
+#endif
+
+/**
+ * Require soft endstops for certain setups
+ */
+#if !BOTH(MIN_SOFTWARE_ENDSTOPS, MAX_SOFTWARE_ENDSTOPS)
+ #if ENABLED(DUAL_X_CARRIAGE)
+ #error "DUAL_X_CARRIAGE requires both MIN_ and MAX_SOFTWARE_ENDSTOPS."
+ #elif HAS_HOTEND_OFFSET
+ #error "MIN_ and MAX_SOFTWARE_ENDSTOPS are both required with offset hotends."
+ #endif
+#endif
+
+/**
+ * Ensure this option is set intentionally
+ */
+#if ENABLED(PSU_CONTROL)
+ #ifndef PSU_ACTIVE_STATE
+ #error "PSU_CONTROL requires PSU_ACTIVE_STATE to be defined as 'HIGH' or 'LOW'."
+ #elif !PIN_EXISTS(PS_ON)
+ #error "PSU_CONTROL requires PS_ON_PIN."
+ #elif POWER_OFF_DELAY < 0
+ #error "POWER_OFF_DELAY must be a positive value."
+ #endif
+#endif
+
+#if HAS_CUTTER
+ #ifndef CUTTER_POWER_UNIT
+ #error "CUTTER_POWER_UNIT is required with a spindle or laser."
+ #elif !CUTTER_UNIT_IS(PWM255) && !CUTTER_UNIT_IS(PERCENT) && !CUTTER_UNIT_IS(RPM) && !CUTTER_UNIT_IS(SERVO)
+ #error "CUTTER_POWER_UNIT must be PWM255, PERCENT, RPM, or SERVO."
+ #endif
+
+ #if ENABLED(LASER_POWER_INLINE)
+ #if ENABLED(SPINDLE_CHANGE_DIR)
+ #error "SPINDLE_CHANGE_DIR and LASER_POWER_INLINE are incompatible."
+ #elif ENABLED(LASER_MOVE_G0_OFF) && DISABLED(LASER_MOVE_POWER)
+ #error "LASER_MOVE_G0_OFF requires LASER_MOVE_POWER."
+ #endif
+ #if ENABLED(LASER_POWER_INLINE_TRAPEZOID)
+ #if DISABLED(SPINDLE_LASER_PWM)
+ #error "LASER_POWER_INLINE_TRAPEZOID requires SPINDLE_LASER_PWM to function."
+ #elif ENABLED(S_CURVE_ACCELERATION)
+ //#ifndef LASER_POWER_INLINE_S_CURVE_ACCELERATION_WARN
+ // #define LASER_POWER_INLINE_S_CURVE_ACCELERATION_WARN
+ // #warning "Combining LASER_POWER_INLINE_TRAPEZOID with S_CURVE_ACCELERATION may result in unintended behavior."
+ //#endif
+ #endif
+ #endif
+ #if ENABLED(LASER_POWER_INLINE_INVERT)
+ //#ifndef LASER_POWER_INLINE_INVERT_WARN
+ // #define LASER_POWER_INLINE_INVERT_WARN
+ // #warning "Enabling LASER_POWER_INLINE_INVERT means that `M5` won't kill the laser immediately; use `M5 I` instead."
+ //#endif
+ #endif
+ #else
+ #if SPINDLE_LASER_POWERUP_DELAY < 1
+ #error "SPINDLE_LASER_POWERUP_DELAY must be greater than 0."
+ #elif SPINDLE_LASER_POWERDOWN_DELAY < 1
+ #error "SPINDLE_LASER_POWERDOWN_DELAY must be greater than 0."
+ #elif ENABLED(LASER_MOVE_POWER)
+ #error "LASER_MOVE_POWER requires LASER_POWER_INLINE."
+ #elif ANY(LASER_POWER_INLINE_TRAPEZOID, LASER_POWER_INLINE_INVERT, LASER_MOVE_G0_OFF, LASER_MOVE_POWER)
+ #error "Enabled an inline laser feature without inline laser power being enabled."
+ #endif
+ #endif
+ #define _PIN_CONFLICT(P) (PIN_EXISTS(P) && P##_PIN == SPINDLE_LASER_PWM_PIN)
+ #if BOTH(SPINDLE_FEATURE, LASER_FEATURE)
+ #error "Enable only one of SPINDLE_FEATURE or LASER_FEATURE."
+ #elif !PIN_EXISTS(SPINDLE_LASER_ENA) && DISABLED(SPINDLE_SERVO)
+ #error "(SPINDLE|LASER)_FEATURE requires SPINDLE_LASER_ENA_PIN or SPINDLE_SERVO to control the power."
+ #elif ENABLED(SPINDLE_CHANGE_DIR) && !PIN_EXISTS(SPINDLE_DIR)
+ #error "SPINDLE_DIR_PIN is required for SPINDLE_CHANGE_DIR."
+ #elif ENABLED(SPINDLE_LASER_PWM)
+ #if !defined(SPINDLE_LASER_PWM_PIN) || SPINDLE_LASER_PWM_PIN < 0
+ #error "SPINDLE_LASER_PWM_PIN is required for SPINDLE_LASER_PWM."
+ #elif !_TEST_PWM(SPINDLE_LASER_PWM_PIN)
+ #error "SPINDLE_LASER_PWM_PIN not assigned to a PWM pin."
+ #elif !defined(SPINDLE_LASER_PWM_INVERT)
+ #error "SPINDLE_LASER_PWM_INVERT is required for (SPINDLE|LASER)_FEATURE."
+ #elif !(defined(SPEED_POWER_INTERCEPT) && defined(SPEED_POWER_MIN) && defined(SPEED_POWER_MAX) && defined(SPEED_POWER_STARTUP))
+ #error "SPINDLE_LASER_PWM equation constant(s) missing."
+ #elif _PIN_CONFLICT(X_MIN)
+ #error "SPINDLE_LASER_PWM pin conflicts with X_MIN_PIN."
+ #elif _PIN_CONFLICT(X_MAX)
+ #error "SPINDLE_LASER_PWM pin conflicts with X_MAX_PIN."
+ #elif _PIN_CONFLICT(Z_STEP)
+ #error "SPINDLE_LASER_PWM pin conflicts with Z_STEP_PIN."
+ #elif _PIN_CONFLICT(CASE_LIGHT)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with CASE_LIGHT_PIN."
+ #elif _PIN_CONFLICT(E0_AUTO_FAN)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with E0_AUTO_FAN_PIN."
+ #elif _PIN_CONFLICT(E1_AUTO_FAN)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with E1_AUTO_FAN_PIN."
+ #elif _PIN_CONFLICT(E2_AUTO_FAN)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with E2_AUTO_FAN_PIN."
+ #elif _PIN_CONFLICT(E3_AUTO_FAN)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with E3_AUTO_FAN_PIN."
+ #elif _PIN_CONFLICT(E4_AUTO_FAN)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with E4_AUTO_FAN_PIN."
+ #elif _PIN_CONFLICT(E5_AUTO_FAN)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with E5_AUTO_FAN_PIN."
+ #elif _PIN_CONFLICT(E6_AUTO_FAN)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with E6_AUTO_FAN_PIN."
+ #elif _PIN_CONFLICT(E7_AUTO_FAN)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with E7_AUTO_FAN_PIN."
+ #elif _PIN_CONFLICT(FAN)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with FAN_PIN."
+ #elif _PIN_CONFLICT(FAN1)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with FAN1_PIN."
+ #elif _PIN_CONFLICT(FAN2)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with FAN2_PIN."
+ #elif _PIN_CONFLICT(FAN3)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with FAN3_PIN."
+ #elif _PIN_CONFLICT(FAN4)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with FAN4_PIN."
+ #elif _PIN_CONFLICT(FAN5)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with FAN5_PIN."
+ #elif _PIN_CONFLICT(FAN6)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with FAN6_PIN."
+ #elif _PIN_CONFLICT(FAN7)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with FAN7_PIN."
+ #elif _PIN_CONFLICT(CONTROLLERFAN)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with CONTROLLERFAN_PIN."
+ #elif _PIN_CONFLICT(MOTOR_CURRENT_PWM_XY)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with MOTOR_CURRENT_PWM_XY."
+ #elif _PIN_CONFLICT(MOTOR_CURRENT_PWM_Z)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with MOTOR_CURRENT_PWM_Z."
+ #elif _PIN_CONFLICT(MOTOR_CURRENT_PWM_E)
+ #error "SPINDLE_LASER_PWM_PIN conflicts with MOTOR_CURRENT_PWM_E."
+ #endif
+ #endif
+ #undef _PIN_CONFLICT
+#endif
+
+#if NONE(HAS_MARLINUI_U8GLIB, EXTENSIBLE_UI) && ENABLED(PRINT_PROGRESS_SHOW_DECIMALS)
+ #error "PRINT_PROGRESS_SHOW_DECIMALS currently requires a Graphical LCD."
+#endif
+
+#if HAS_ADC_BUTTONS && defined(ADC_BUTTON_DEBOUNCE_DELAY) && ADC_BUTTON_DEBOUNCE_DELAY < 16
+ #error "ADC_BUTTON_DEBOUNCE_DELAY must be greater than 16."
+#endif
+
+/**
+ * Check to make sure MONITOR_DRIVER_STATUS isn't enabled
+ * on boards where TMC drivers share the SPI bus with SD.
+ */
+#if HAS_TMC_SPI && ALL(MONITOR_DRIVER_STATUS, SDSUPPORT, USES_SHARED_SPI)
+ #error "MONITOR_DRIVER_STATUS and SDSUPPORT cannot be used together on boards with shared SPI."
+#endif
+
+// G60/G61 Position Save
+#if SAVED_POSITIONS > 256
+ #error "SAVED_POSITIONS must be an integer from 0 to 256."
+#endif
+
+/**
+ * Stepper Chunk support
+ */
+#if BOTH(DIRECT_STEPPING, LIN_ADVANCE)
+ #error "DIRECT_STEPPING is incompatible with LIN_ADVANCE. Enable in external planner if possible."
+#endif
+
+/**
+ * Touch Buttons
+ */
+#if ENABLED(TOUCH_SCREEN) && DISABLED(TOUCH_SCREEN_CALIBRATION)
+ #ifndef TOUCH_CALIBRATION_X
+ #error "TOUCH_CALIBRATION_X must be defined with TOUCH_SCREEN."
+ #endif
+ #ifndef TOUCH_CALIBRATION_Y
+ #error "TOUCH_CALIBRATION_Y must be defined with TOUCH_SCREEN."
+ #endif
+ #ifndef TOUCH_OFFSET_X
+ #error "TOUCH_OFFSET_X must be defined with TOUCH_SCREEN."
+ #endif
+ #ifndef TOUCH_OFFSET_Y
+ #error "TOUCH_OFFSET_Y must be defined with TOUCH_SCREEN."
+ #endif
+#endif
+
+/**
+ * Sanity check for WIFI
+ */
+#if EITHER(ESP3D_WIFISUPPORT, WIFISUPPORT) && DISABLED(ARDUINO_ARCH_ESP32)
+ #error "ESP3D_WIFISUPPORT or WIFISUPPORT requires an ESP32 MOTHERBOARD."
+#endif
+
+/**
+ * Sanity Check for Password Feature
+ */
+#if ENABLED(PASSWORD_FEATURE)
+ #if NONE(HAS_LCD_MENU, PASSWORD_UNLOCK_GCODE, PASSWORD_CHANGE_GCODE)
+ #error "Without PASSWORD_UNLOCK_GCODE, PASSWORD_CHANGE_GCODE, or a supported LCD there's no way to unlock the printer or set a password."
+ #elif DISABLED(EEPROM_SETTINGS)
+ #warning "PASSWORD_FEATURE settings will be lost on power-off without EEPROM_SETTINGS."
+ #endif
+#endif
+
+
+/**
+ * Sanity Check for MEATPACK and BINARY_FILE_TRANSFER Features
+ */
+#if BOTH(MEATPACK, BINARY_FILE_TRANSFER)
+ #error "Either enable MEATPACK or enable BINARY_FILE_TRANSFER."
+#endif
+
+/**
+ * Sanity check for valid stepper driver types
+ */
+#define _BAD_DRIVER(A) (defined(A##_DRIVER_TYPE) && !_DRIVER_ID(A##_DRIVER_TYPE))
+#if _BAD_DRIVER(X)
+ #error "X_DRIVER_TYPE is not recognized."
+#endif
+#if _BAD_DRIVER(Y)
+ #error "Y_DRIVER_TYPE is not recognized."
+#endif
+#if _BAD_DRIVER(Z)
+ #error "Z_DRIVER_TYPE is not recognized."
+#endif
+#if _BAD_DRIVER(X2)
+ #error "X2_DRIVER_TYPE is not recognized."
+#endif
+#if _BAD_DRIVER(Y2)
+ #error "Y2_DRIVER_TYPE is not recognized."
+#endif
+#if _BAD_DRIVER(Z2)
+ #error "Z2_DRIVER_TYPE is not recognized."
+#endif
+#if _BAD_DRIVER(Z3)
+ #error "Z3_DRIVER_TYPE is not recognized."
+#endif
+#if _BAD_DRIVER(Z4)
+ #error "Z4_DRIVER_TYPE is not recognized."
+#endif
+#if _BAD_DRIVER(E0)
+ #error "E0_DRIVER_TYPE is not recognized."
+#endif
+#if _BAD_DRIVER(E1)
+ #error "E1_DRIVER_TYPE is not recognized."
+#endif
+#if _BAD_DRIVER(E2)
+ #error "E2_DRIVER_TYPE is not recognized."
+#endif
+#if _BAD_DRIVER(E3)
+ #error "E3_DRIVER_TYPE is not recognized."
+#endif
+#if _BAD_DRIVER(E4)
+ #error "E4_DRIVER_TYPE is not recognized."
+#endif
+#if _BAD_DRIVER(E5)
+ #error "E5_DRIVER_TYPE is not recognized."
+#endif
+#if _BAD_DRIVER(E6)
+ #error "E6_DRIVER_TYPE is not recognized."
+#endif
+#if _BAD_DRIVER(E7)
+ #error "E7_DRIVER_TYPE is not recognized."
+#endif
+#undef _BAD_DRIVER
+
+// Misc. Cleanup
+#undef _TEST_PWM
diff --git a/Marlin/src/inc/Version.h b/Marlin/src/inc/Version.h
new file mode 100644
index 0000000..488e570
--- /dev/null
+++ b/Marlin/src/inc/Version.h
@@ -0,0 +1,122 @@
+/**
+ * Marlin 3D Printer Firmware
+ * Copyright (c) 2020 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
+ *
+ * Based on Sprinter and grbl.
+ * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+ *
+ */
+#pragma once
+
+/**
+ * Release version. Leave the Marlin version or apply a custom scheme.
+ */
+#ifndef SHORT_BUILD_VERSION
+ #define SHORT_BUILD_VERSION "bugfix-2.0.x"
+#endif
+
+/**
+ * Verbose version identifier containing a unique identifier, such as the
+ * vendor name, download location, GitHub account, etc.
+ */
+#ifndef DETAILED_BUILD_VERSION
+ #define DETAILED_BUILD_VERSION SHORT_BUILD_VERSION
+#endif
+
+/**
+ * The STRING_DISTRIBUTION_DATE represents when the binary file was built,
+ * here we define this default string as the date where the latest release
+ * version was tagged.
+ */
+#ifndef STRING_DISTRIBUTION_DATE
+ #define STRING_DISTRIBUTION_DATE "2021-02-01"
+#endif
+
+/**
+ * Minimum Configuration.h and Configuration_adv.h file versions.
+ * Set based on the release version number. Used to catch an attempt to use
+ * older configurations. Override these if using a custom versioning scheme
+ * to alert users to major changes.
+ */
+
+#define MARLIN_HEX_VERSION 020008
+#ifndef REQUIRED_CONFIGURATION_H_VERSION
+ #define REQUIRED_CONFIGURATION_H_VERSION MARLIN_HEX_VERSION
+#endif
+#ifndef REQUIRED_CONFIGURATION_ADV_H_VERSION
+ #define REQUIRED_CONFIGURATION_ADV_H_VERSION MARLIN_HEX_VERSION
+#endif
+
+/**
+ * The protocol for communication to the host. Protocol indicates communication
+ * standards such as the use of ASCII, "echo:" and "error:" line prefixes, etc.
+ * (Other behaviors are given by the firmware version and capabilities report.)
+ */
+#ifndef PROTOCOL_VERSION
+ #define PROTOCOL_VERSION "1.0"
+#endif
+
+/**
+ * Define a generic printer name to be output to the LCD after booting Marlin.
+ */
+#ifndef MACHINE_NAME
+ #define MACHINE_NAME "3D Printer"
+#endif
+
+/**
+ * Website where users can find Marlin source code for the binary installed on the
+ * device. Override this if you provide public source code download. (GPLv3 requires
+ * providing the source code to your customers.)
+ */
+#ifndef SOURCE_CODE_URL
+ #define SOURCE_CODE_URL "github.com/MarlinFirmware/Marlin"
+#endif
+
+/**
+ * Default generic printer UUID.
+ */
+#ifndef DEFAULT_MACHINE_UUID
+ #define DEFAULT_MACHINE_UUID "cede2a2f-41a2-4748-9b12-c55c62f367ff"
+#endif
+
+ /**
+ * The WEBSITE_URL is the location where users can get more information such as
+ * documentation about a specific Marlin release. Displayed in the Info Menu.
+ */
+#ifndef WEBSITE_URL
+ #define WEBSITE_URL "marlinfw.org"
+#endif
+
+/**
+ * Set the vendor info the serial USB interface, if changable
+ * Currently only supported by DUE platform
+ */
+#ifndef USB_DEVICE_VENDOR_ID
+ #define USB_DEVICE_VENDOR_ID 0x03EB /* ATMEL VID */
+#endif
+#ifndef USB_DEVICE_PRODUCT_ID
+ #define USB_DEVICE_PRODUCT_ID 0x2424 /* MSC / CDC */
+#endif
+//! USB Device string definitions (Optional)
+#ifndef USB_DEVICE_MANUFACTURE_NAME
+ #define USB_DEVICE_MANUFACTURE_NAME WEBSITE_URL
+#endif
+#ifdef CUSTOM_MACHINE_NAME
+ #define USB_DEVICE_PRODUCT_NAME CUSTOM_MACHINE_NAME
+#else
+ #define USB_DEVICE_PRODUCT_NAME MACHINE_NAME
+#endif
+#define USB_DEVICE_SERIAL_NAME "123985739853"