Browse Source

Quad Z stepper support (#16277)

InsanityAutomation 5 years ago
parent
commit
0fcf2b1110
52 changed files with 1198 additions and 415 deletions
  1. 1
    0
      Marlin/Configuration.h
  2. 46
    20
      Marlin/Configuration_adv.h
  3. 16
    0
      Marlin/src/HAL/HAL_AVR/endstop_interrupts.h
  4. 6
    0
      Marlin/src/HAL/HAL_DUE/endstop_interrupts.h
  5. 6
    0
      Marlin/src/HAL/HAL_ESP32/endstop_interrupts.h
  6. 13
    1
      Marlin/src/HAL/HAL_LPC1768/endstop_interrupts.h
  7. 34
    11
      Marlin/src/HAL/HAL_SAMD51/endstop_interrupts.h
  8. 6
    0
      Marlin/src/HAL/HAL_STM32/endstop_interrupts.h
  9. 6
    0
      Marlin/src/HAL/HAL_STM32F1/endstop_interrupts.h
  10. 6
    0
      Marlin/src/HAL/HAL_STM32_F4_F7/endstop_interrupts.h
  11. 6
    0
      Marlin/src/HAL/HAL_TEENSY35_36/endstop_interrupts.h
  12. 5
    3
      Marlin/src/core/drivers.h
  13. 3
    0
      Marlin/src/core/language.h
  14. 3
    0
      Marlin/src/feature/controllerfan.cpp
  15. 22
    1
      Marlin/src/feature/tmc_util.cpp
  16. 1
    1
      Marlin/src/feature/tmc_util.h
  17. 28
    15
      Marlin/src/gcode/calibrate/G34_M422.cpp
  18. 22
    13
      Marlin/src/gcode/calibrate/M666.cpp
  19. 3
    0
      Marlin/src/gcode/feature/L6470/M122.cpp
  20. 7
    1
      Marlin/src/gcode/feature/L6470/M906.cpp
  21. 1
    0
      Marlin/src/gcode/feature/L6470/M916-918.cpp
  22. 10
    3
      Marlin/src/gcode/feature/trinamic/M569.cpp
  23. 8
    2
      Marlin/src/gcode/feature/trinamic/M906.cpp
  24. 21
    3
      Marlin/src/gcode/feature/trinamic/M911-M914.cpp
  25. 2
    2
      Marlin/src/gcode/gcode.cpp
  26. 3
    0
      Marlin/src/gcode/host/M114.cpp
  27. 1
    4
      Marlin/src/inc/Conditionals_LCD.h
  28. 5
    0
      Marlin/src/inc/Conditionals_adv.h
  29. 118
    71
      Marlin/src/inc/Conditionals_post.h
  30. 90
    62
      Marlin/src/inc/SanityCheck.h
  31. 9
    0
      Marlin/src/lcd/menu/menu_tmc.cpp
  32. 27
    7
      Marlin/src/libs/L64XX/L64XX_Marlin.cpp
  33. 1
    1
      Marlin/src/libs/L64XX/L64XX_Marlin.h
  34. 115
    69
      Marlin/src/module/configuration_store.cpp
  35. 89
    16
      Marlin/src/module/endstops.cpp
  36. 7
    3
      Marlin/src/module/endstops.h
  37. 103
    55
      Marlin/src/module/motion.cpp
  38. 95
    18
      Marlin/src/module/stepper.cpp
  39. 16
    9
      Marlin/src/module/stepper.h
  40. 3
    0
      Marlin/src/module/stepper/L64xx.cpp
  41. 17
    0
      Marlin/src/module/stepper/L64xx.h
  42. 6
    0
      Marlin/src/module/stepper/TMC26X.cpp
  43. 8
    0
      Marlin/src/module/stepper/TMC26X.h
  44. 35
    2
      Marlin/src/module/stepper/indirection.h
  45. 26
    0
      Marlin/src/module/stepper/trinamic.cpp
  46. 14
    0
      Marlin/src/module/stepper/trinamic.h
  47. 49
    2
      Marlin/src/pins/pins.h
  48. 36
    0
      Marlin/src/pins/pinsDebug_list.h
  49. 3
    7
      Marlin/src/pins/ramps/pins_RL200.h
  50. 31
    5
      Marlin/src/pins/sensitive_pins.h
  51. 5
    5
      buildroot/share/tests/DUE-tests
  52. 4
    3
      buildroot/share/tests/teensy35-tests

+ 1
- 0
Marlin/Configuration.h View File

670
 //#define Y2_DRIVER_TYPE A4988
670
 //#define Y2_DRIVER_TYPE A4988
671
 //#define Z2_DRIVER_TYPE A4988
671
 //#define Z2_DRIVER_TYPE A4988
672
 //#define Z3_DRIVER_TYPE A4988
672
 //#define Z3_DRIVER_TYPE A4988
673
+//#define Z4_DRIVER_TYPE A4988
673
 //#define E0_DRIVER_TYPE A4988
674
 //#define E0_DRIVER_TYPE A4988
674
 //#define E1_DRIVER_TYPE A4988
675
 //#define E1_DRIVER_TYPE A4988
675
 //#define E2_DRIVER_TYPE A4988
676
 //#define E2_DRIVER_TYPE A4988

+ 46
- 20
Marlin/Configuration_adv.h View File

479
   //#define X_DUAL_ENDSTOPS
479
   //#define X_DUAL_ENDSTOPS
480
   #if ENABLED(X_DUAL_ENDSTOPS)
480
   #if ENABLED(X_DUAL_ENDSTOPS)
481
     #define X2_USE_ENDSTOP _XMAX_
481
     #define X2_USE_ENDSTOP _XMAX_
482
-    #define X_DUAL_ENDSTOPS_ADJUSTMENT  0
482
+    #define X2_ENDSTOP_ADJUSTMENT  0
483
   #endif
483
   #endif
484
 #endif
484
 #endif
485
 
485
 
489
   //#define Y_DUAL_ENDSTOPS
489
   //#define Y_DUAL_ENDSTOPS
490
   #if ENABLED(Y_DUAL_ENDSTOPS)
490
   #if ENABLED(Y_DUAL_ENDSTOPS)
491
     #define Y2_USE_ENDSTOP _YMAX_
491
     #define Y2_USE_ENDSTOP _YMAX_
492
-    #define Y_DUAL_ENDSTOPS_ADJUSTMENT  0
492
+    #define Y2_ENDSTOP_ADJUSTMENT  0
493
   #endif
493
   #endif
494
 #endif
494
 #endif
495
 
495
 
496
-//#define Z_DUAL_STEPPER_DRIVERS
497
-#if ENABLED(Z_DUAL_STEPPER_DRIVERS)
498
-  //#define Z_DUAL_ENDSTOPS
499
-  #if ENABLED(Z_DUAL_ENDSTOPS)
500
-    #define Z2_USE_ENDSTOP _XMAX_
501
-    #define Z_DUAL_ENDSTOPS_ADJUSTMENT  0
502
-  #endif
503
-#endif
504
-
505
-//#define Z_TRIPLE_STEPPER_DRIVERS
506
-#if ENABLED(Z_TRIPLE_STEPPER_DRIVERS)
507
-  //#define Z_TRIPLE_ENDSTOPS
508
-  #if ENABLED(Z_TRIPLE_ENDSTOPS)
509
-    #define Z2_USE_ENDSTOP _XMAX_
510
-    #define Z3_USE_ENDSTOP _YMAX_
511
-    #define Z_TRIPLE_ENDSTOPS_ADJUSTMENT2  0
512
-    #define Z_TRIPLE_ENDSTOPS_ADJUSTMENT3  0
496
+//
497
+// For Z set the number of stepper drivers
498
+//
499
+#define NUM_Z_STEPPER_DRIVERS 1   // (1-4) Z options change based on how many
500
+
501
+#if NUM_Z_STEPPER_DRIVERS > 1
502
+  //#define Z_MULTI_ENDSTOPS
503
+  #if ENABLED(Z_MULTI_ENDSTOPS)
504
+    #define Z2_USE_ENDSTOP          _XMAX_
505
+    #define Z2_ENDSTOP_ADJUSTMENT   0
506
+    #if NUM_Z_STEPPER_DRIVERS >= 3
507
+      #define Z3_USE_ENDSTOP        _YMAX_
508
+      #define Z3_ENDSTOP_ADJUSTMENT 0
509
+    #endif
510
+    #if NUM_Z_STEPPER_DRIVERS >= 4
511
+      #define Z4_USE_ENDSTOP        _ZMAX_
512
+      #define Z4_ENDSTOP_ADJUSTMENT 0
513
+    #endif
513
   #endif
514
   #endif
514
 #endif
515
 #endif
515
 
516
 
1898
     #define Z3_MICROSTEPS       16
1899
     #define Z3_MICROSTEPS       16
1899
   #endif
1900
   #endif
1900
 
1901
 
1902
+  #if AXIS_DRIVER_TYPE_Z4(TMC26X)
1903
+    #define Z4_MAX_CURRENT    1000
1904
+    #define Z4_SENSE_RESISTOR   91
1905
+    #define Z4_MICROSTEPS       16
1906
+  #endif
1907
+
1901
   #if AXIS_DRIVER_TYPE_E0(TMC26X)
1908
   #if AXIS_DRIVER_TYPE_E0(TMC26X)
1902
     #define E0_MAX_CURRENT    1000
1909
     #define E0_MAX_CURRENT    1000
1903
     #define E0_SENSE_RESISTOR   91
1910
     #define E0_SENSE_RESISTOR   91
2015
     #define Z3_CHAIN_POS     -1
2022
     #define Z3_CHAIN_POS     -1
2016
   #endif
2023
   #endif
2017
 
2024
 
2025
+  #if AXIS_IS_TMC(Z4)
2026
+    #define Z4_CURRENT      800
2027
+    #define Z4_CURRENT_HOME Z4_CURRENT
2028
+    #define Z4_MICROSTEPS    16
2029
+    #define Z4_RSENSE         0.11
2030
+    #define Z4_CHAIN_POS     -1
2031
+  #endif
2032
+
2018
   #if AXIS_IS_TMC(E0)
2033
   #if AXIS_IS_TMC(E0)
2019
     #define E0_CURRENT      800
2034
     #define E0_CURRENT      800
2020
     #define E0_MICROSTEPS    16
2035
     #define E0_MICROSTEPS    16
2104
   #define Y2_SLAVE_ADDRESS 0
2119
   #define Y2_SLAVE_ADDRESS 0
2105
   #define Z2_SLAVE_ADDRESS 0
2120
   #define Z2_SLAVE_ADDRESS 0
2106
   #define Z3_SLAVE_ADDRESS 0
2121
   #define Z3_SLAVE_ADDRESS 0
2122
+  #define Z4_SLAVE_ADDRESS 0
2107
   #define E0_SLAVE_ADDRESS 0
2123
   #define E0_SLAVE_ADDRESS 0
2108
   #define E1_SLAVE_ADDRESS 0
2124
   #define E1_SLAVE_ADDRESS 0
2109
   #define E2_SLAVE_ADDRESS 0
2125
   #define E2_SLAVE_ADDRESS 0
2179
   #define Z_HYBRID_THRESHOLD       3
2195
   #define Z_HYBRID_THRESHOLD       3
2180
   #define Z2_HYBRID_THRESHOLD      3
2196
   #define Z2_HYBRID_THRESHOLD      3
2181
   #define Z3_HYBRID_THRESHOLD      3
2197
   #define Z3_HYBRID_THRESHOLD      3
2198
+  #define Z4_HYBRID_THRESHOLD      3
2182
   #define E0_HYBRID_THRESHOLD     30
2199
   #define E0_HYBRID_THRESHOLD     30
2183
   #define E1_HYBRID_THRESHOLD     30
2200
   #define E1_HYBRID_THRESHOLD     30
2184
   #define E2_HYBRID_THRESHOLD     30
2201
   #define E2_HYBRID_THRESHOLD     30
2344
     #define Z3_SLEW_RATE         1
2361
     #define Z3_SLEW_RATE         1
2345
   #endif
2362
   #endif
2346
 
2363
 
2364
+  #if AXIS_IS_L64XX(Z4)
2365
+    #define Z4_MICROSTEPS      128
2366
+    #define Z4_OVERCURRENT    2000
2367
+    #define Z4_STALLCURRENT   1500
2368
+    #define Z4_MAX_VOLTAGE     127
2369
+    #define Z4_CHAIN_POS        -1
2370
+    #define Z4_SLEW_RATE         1
2371
+  #endif
2372
+
2347
   #if AXIS_IS_L64XX(E0)
2373
   #if AXIS_IS_L64XX(E0)
2348
     #define E0_MICROSTEPS      128
2374
     #define E0_MICROSTEPS      128
2349
     #define E0_OVERCURRENT    2000
2375
     #define E0_OVERCURRENT    2000
2407
    *         I not present or I0 or I1 - X, Y, Z or E0
2433
    *         I not present or I0 or I1 - X, Y, Z or E0
2408
    *         I2 - X2, Y2, Z2 or E1
2434
    *         I2 - X2, Y2, Z2 or E1
2409
    *         I3 - Z3 or E3
2435
    *         I3 - Z3 or E3
2410
-   *         I4 - E4
2436
+   *         I4 - Z4 or E4
2411
    *         I5 - E5
2437
    *         I5 - E5
2412
    * M916 - Increase drive level until get thermal warning
2438
    * M916 - Increase drive level until get thermal warning
2413
    * M917 - Find minimum current thresholds
2439
    * M917 - Find minimum current thresholds

+ 16
- 0
Marlin/src/HAL/HAL_AVR/endstop_interrupts.h View File

232
       pciSetup(Z3_MIN_PIN);
232
       pciSetup(Z3_MIN_PIN);
233
     #endif
233
     #endif
234
   #endif
234
   #endif
235
+  #if HAS_Z4_MAX
236
+    #if (digitalPinToInterrupt(Z4_MAX_PIN) != NOT_AN_INTERRUPT)
237
+      _ATTACH(Z4_MAX_PIN);
238
+    #else
239
+      static_assert(digitalPinHasPCICR(Z4_MAX_PIN), "Z4_MAX_PIN is not interrupt-capable");
240
+      pciSetup(Z4_MAX_PIN);
241
+    #endif
242
+  #endif
243
+  #if HAS_Z4_MIN
244
+    #if (digitalPinToInterrupt(Z4_MIN_PIN) != NOT_AN_INTERRUPT)
245
+      _ATTACH(Z4_MIN_PIN);
246
+    #else
247
+      static_assert(digitalPinHasPCICR(Z4_MIN_PIN), "Z4_MIN_PIN is not interrupt-capable");
248
+      pciSetup(Z4_MIN_PIN);
249
+    #endif
250
+  #endif
235
   #if HAS_Z_MIN_PROBE_PIN
251
   #if HAS_Z_MIN_PROBE_PIN
236
     #if (digitalPinToInterrupt(Z_MIN_PROBE_PIN) != NOT_AN_INTERRUPT)
252
     #if (digitalPinToInterrupt(Z_MIN_PROBE_PIN) != NOT_AN_INTERRUPT)
237
       _ATTACH(Z_MIN_PROBE_PIN);
253
       _ATTACH(Z_MIN_PROBE_PIN);

+ 6
- 0
Marlin/src/HAL/HAL_DUE/endstop_interrupts.h View File

77
   #if HAS_Z3_MIN
77
   #if HAS_Z3_MIN
78
     _ATTACH(Z3_MIN_PIN);
78
     _ATTACH(Z3_MIN_PIN);
79
   #endif
79
   #endif
80
+  #if HAS_Z4_MAX
81
+    _ATTACH(Z4_MAX_PIN);
82
+  #endif
83
+  #if HAS_Z4_MIN
84
+    _ATTACH(Z4_MIN_PIN);
85
+  #endif
80
   #if HAS_Z_MIN_PROBE_PIN
86
   #if HAS_Z_MIN_PROBE_PIN
81
     _ATTACH(Z_MIN_PROBE_PIN);
87
     _ATTACH(Z_MIN_PROBE_PIN);
82
   #endif
88
   #endif

+ 6
- 0
Marlin/src/HAL/HAL_ESP32/endstop_interrupts.h View File

72
   #if HAS_Z3_MIN
72
   #if HAS_Z3_MIN
73
     _ATTACH(Z3_MIN_PIN);
73
     _ATTACH(Z3_MIN_PIN);
74
   #endif
74
   #endif
75
+  #if HAS_Z4_MAX
76
+    _ATTACH(Z4_MAX_PIN);
77
+  #endif
78
+  #if HAS_Z4_MIN
79
+    _ATTACH(Z4_MIN_PIN);
80
+  #endif
75
   #if HAS_Z_MIN_PROBE_PIN
81
   #if HAS_Z_MIN_PROBE_PIN
76
     _ATTACH(Z_MIN_PROBE_PIN);
82
     _ATTACH(Z_MIN_PROBE_PIN);
77
   #endif
83
   #endif

+ 13
- 1
Marlin/src/HAL/HAL_LPC1768/endstop_interrupts.h View File

93
     _ATTACH(Z2_MIN_PIN);
93
     _ATTACH(Z2_MIN_PIN);
94
   #endif
94
   #endif
95
   #if HAS_Z3_MAX
95
   #if HAS_Z3_MAX
96
-    #if !LPC1768_PIN_INTERRUPT_M(Z3_MIN_PIN)
96
+    #if !LPC1768_PIN_INTERRUPT_M(Z3_MAX_PIN)
97
       #error "Z3_MIN_PIN is not INTERRUPT-capable."
97
       #error "Z3_MIN_PIN is not INTERRUPT-capable."
98
     #endif
98
     #endif
99
     _ATTACH(Z3_MAX_PIN);
99
     _ATTACH(Z3_MAX_PIN);
104
     #endif
104
     #endif
105
     _ATTACH(Z3_MIN_PIN);
105
     _ATTACH(Z3_MIN_PIN);
106
   #endif
106
   #endif
107
+  #if HAS_Z4_MAX
108
+    #if !LPC1768_PIN_INTERRUPT_M(Z4_MAX_PIN)
109
+      #error "Z4_MIN_PIN is not INTERRUPT-capable."
110
+    #endif
111
+    _ATTACH(Z4_MAX_PIN);
112
+  #endif
113
+  #if HAS_Z4_MIN
114
+    #if !LPC1768_PIN_INTERRUPT_M(Z4_MIN_PIN)
115
+      #error "Z4_MIN_PIN is not INTERRUPT-capable."
116
+    #endif
117
+    _ATTACH(Z4_MIN_PIN);
118
+  #endif
107
   #if HAS_Z_MIN_PROBE_PIN
119
   #if HAS_Z_MIN_PROBE_PIN
108
     #if !LPC1768_PIN_INTERRUPT_M(Z_MIN_PROBE_PIN)
120
     #if !LPC1768_PIN_INTERRUPT_M(Z_MIN_PROBE_PIN)
109
       #error "Z_MIN_PROBE_PIN is not INTERRUPT-capable."
121
       #error "Z_MIN_PROBE_PIN is not INTERRUPT-capable."

+ 34
- 11
Marlin/src/HAL/HAL_SAMD51/endstop_interrupts.h View File

98
 #else
98
 #else
99
   #define MATCH_Z3_MIN_EILINE(P) false
99
   #define MATCH_Z3_MIN_EILINE(P) false
100
 #endif
100
 #endif
101
+#if HAS_Z4_MAX
102
+  #define MATCH_Z4_MAX_EILINE(P) MATCH_EILINE(P, Z4_MAX_PIN)
103
+#else
104
+  #define MATCH_Z4_MAX_EILINE(P) false
105
+#endif
106
+#if HAS_Z4_MIN
107
+  #define MATCH_Z4_MIN_EILINE(P) MATCH_EILINE(P, Z4_MIN_PIN)
108
+#else
109
+  #define MATCH_Z4_MIN_EILINE(P) false
110
+#endif
101
 #if HAS_Z_MIN_PROBE_PIN
111
 #if HAS_Z_MIN_PROBE_PIN
102
   #define MATCH_Z_MIN_PROBE_EILINE(P)   MATCH_EILINE(P, Z_MIN_PROBE_PIN)
112
   #define MATCH_Z_MIN_PROBE_EILINE(P)   MATCH_EILINE(P, Z_MIN_PROBE_PIN)
103
 #else
113
 #else
109
                                  && !MATCH_Z_MAX_EILINE(P) && !MATCH_Z_MIN_EILINE(P)    \
119
                                  && !MATCH_Z_MAX_EILINE(P) && !MATCH_Z_MIN_EILINE(P)    \
110
                                  && !MATCH_Z2_MAX_EILINE(P) && !MATCH_Z2_MIN_EILINE(P)  \
120
                                  && !MATCH_Z2_MAX_EILINE(P) && !MATCH_Z2_MIN_EILINE(P)  \
111
                                  && !MATCH_Z3_MAX_EILINE(P) && !MATCH_Z3_MIN_EILINE(P)  \
121
                                  && !MATCH_Z3_MAX_EILINE(P) && !MATCH_Z3_MIN_EILINE(P)  \
122
+                                 && !MATCH_Z4_MAX_EILINE(P) && !MATCH_Z4_MIN_EILINE(P)  \
112
                                  && !MATCH_Z_MIN_PROBE_EILINE(P))
123
                                  && !MATCH_Z_MIN_PROBE_EILINE(P))
113
 
124
 
114
 // One ISR for all EXT-Interrupts
125
 // One ISR for all EXT-Interrupts
117
 void setup_endstop_interrupts() {
128
 void setup_endstop_interrupts() {
118
   #if HAS_X_MAX
129
   #if HAS_X_MAX
119
     #if !AVAILABLE_EILINE(X_MAX_PIN)
130
     #if !AVAILABLE_EILINE(X_MAX_PIN)
120
-      static_assert(false, "X_MAX_PIN has no EXTINT line available.");
131
+      #error "X_MAX_PIN has no EXTINT line available."
121
     #endif
132
     #endif
122
     attachInterrupt(X_MAX_PIN, endstop_ISR, CHANGE);
133
     attachInterrupt(X_MAX_PIN, endstop_ISR, CHANGE);
123
   #endif
134
   #endif
124
   #if HAS_X_MIN
135
   #if HAS_X_MIN
125
     #if !AVAILABLE_EILINE(X_MIN_PIN)
136
     #if !AVAILABLE_EILINE(X_MIN_PIN)
126
-      static_assert(false, "X_MIN_PIN has no EXTINT line available.");
137
+      #error "X_MIN_PIN has no EXTINT line available."
127
     #endif
138
     #endif
128
     attachInterrupt(X_MIN_PIN, endstop_ISR, CHANGE);
139
     attachInterrupt(X_MIN_PIN, endstop_ISR, CHANGE);
129
   #endif
140
   #endif
130
   #if HAS_Y_MAX
141
   #if HAS_Y_MAX
131
     #if !AVAILABLE_EILINE(Y_MAX_PIN)
142
     #if !AVAILABLE_EILINE(Y_MAX_PIN)
132
-      static_assert(false, "Y_MAX_PIN has no EXTINT line available.");
143
+      #error "Y_MAX_PIN has no EXTINT line available."
133
     #endif
144
     #endif
134
     attachInterrupt(Y_MAX_PIN, endstop_ISR, CHANGE);
145
     attachInterrupt(Y_MAX_PIN, endstop_ISR, CHANGE);
135
   #endif
146
   #endif
136
   #if HAS_Y_MIN
147
   #if HAS_Y_MIN
137
     #if !AVAILABLE_EILINE(Y_MIN_PIN)
148
     #if !AVAILABLE_EILINE(Y_MIN_PIN)
138
-      static_assert(false, "Y_MIN_PIN has no EXTINT line available.");
149
+      #error "Y_MIN_PIN has no EXTINT line available."
139
     #endif
150
     #endif
140
     attachInterrupt(Y_MIN_PIN, endstop_ISR, CHANGE);
151
     attachInterrupt(Y_MIN_PIN, endstop_ISR, CHANGE);
141
   #endif
152
   #endif
142
   #if HAS_Z_MAX
153
   #if HAS_Z_MAX
143
     #if !AVAILABLE_EILINE(Z_MAX_PIN)
154
     #if !AVAILABLE_EILINE(Z_MAX_PIN)
144
-      static_assert(false, "Z_MAX_PIN has no EXTINT line available.");
155
+      #error "Z_MAX_PIN has no EXTINT line available."
145
     #endif
156
     #endif
146
     attachInterrupt(Z_MAX_PIN, endstop_ISR, CHANGE);
157
     attachInterrupt(Z_MAX_PIN, endstop_ISR, CHANGE);
147
   #endif
158
   #endif
148
   #if HAS_Z_MIN
159
   #if HAS_Z_MIN
149
     #if !AVAILABLE_EILINE(Z_MIN_PIN)
160
     #if !AVAILABLE_EILINE(Z_MIN_PIN)
150
-      static_assert(false, "Z_MIN_PIN has no EXTINT line available.");
161
+      #error "Z_MIN_PIN has no EXTINT line available."
151
     #endif
162
     #endif
152
     attachInterrupt(Z_MIN_PIN, endstop_ISR, CHANGE);
163
     attachInterrupt(Z_MIN_PIN, endstop_ISR, CHANGE);
153
   #endif
164
   #endif
154
   #if HAS_Z2_MAX
165
   #if HAS_Z2_MAX
155
     #if !AVAILABLE_EILINE(Z2_MAX_PIN)
166
     #if !AVAILABLE_EILINE(Z2_MAX_PIN)
156
-      static_assert(false, "Z2_MAX_PIN has no EXTINT line available.");
167
+      #error "Z2_MAX_PIN has no EXTINT line available."
157
     #endif
168
     #endif
158
     attachInterrupt(Z2_MAX_PIN, endstop_ISR, CHANGE);
169
     attachInterrupt(Z2_MAX_PIN, endstop_ISR, CHANGE);
159
   #endif
170
   #endif
160
   #if HAS_Z2_MIN
171
   #if HAS_Z2_MIN
161
     #if !AVAILABLE_EILINE(Z2_MIN_PIN)
172
     #if !AVAILABLE_EILINE(Z2_MIN_PIN)
162
-      static_assert(false, "Z2_MIN_PIN has no EXTINT line available.");
173
+      #error "Z2_MIN_PIN has no EXTINT line available."
163
     #endif
174
     #endif
164
     attachInterrupt(Z2_MIN_PIN, endstop_ISR, CHANGE);
175
     attachInterrupt(Z2_MIN_PIN, endstop_ISR, CHANGE);
165
   #endif
176
   #endif
166
   #if HAS_Z3_MAX
177
   #if HAS_Z3_MAX
167
     #if !AVAILABLE_EILINE(Z3_MAX_PIN)
178
     #if !AVAILABLE_EILINE(Z3_MAX_PIN)
168
-      static_assert(false, "Z3_MAX_PIN has no EXTINT line available.");
179
+      #error "Z3_MAX_PIN has no EXTINT line available."
169
     #endif
180
     #endif
170
     attachInterrupt(Z3_MAX_PIN, endstop_ISR, CHANGE);
181
     attachInterrupt(Z3_MAX_PIN, endstop_ISR, CHANGE);
171
   #endif
182
   #endif
172
   #if HAS_Z3_MIN
183
   #if HAS_Z3_MIN
173
     #if !AVAILABLE_EILINE(Z3_MIN_PIN)
184
     #if !AVAILABLE_EILINE(Z3_MIN_PIN)
174
-      static_assert(false, "Z3_MIN_PIN has no EXTINT line available.");
185
+      #error "Z3_MIN_PIN has no EXTINT line available."
175
     #endif
186
     #endif
176
     attachInterrupt(Z3_MIN_PIN, endstop_ISR, CHANGE);
187
     attachInterrupt(Z3_MIN_PIN, endstop_ISR, CHANGE);
177
   #endif
188
   #endif
189
+  #if HAS_Z4_MAX
190
+    #if !AVAILABLE_EILINE(Z4_MAX_PIN)
191
+      #error "Z4_MAX_PIN has no EXTINT line available."
192
+    #endif
193
+    attachInterrupt(Z4_MAX_PIN, endstop_ISR, CHANGE);
194
+  #endif
195
+  #if HAS_Z4_MIN
196
+    #if !AVAILABLE_EILINE(Z4_MIN_PIN)
197
+      #error "Z4_MIN_PIN has no EXTINT line available."
198
+    #endif
199
+    attachInterrupt(Z4_MIN_PIN, endstop_ISR, CHANGE);
200
+  #endif
178
   #if HAS_Z_MIN_PROBE_PIN
201
   #if HAS_Z_MIN_PROBE_PIN
179
     #if !AVAILABLE_EILINE(Z_MIN_PROBE_PIN)
202
     #if !AVAILABLE_EILINE(Z_MIN_PROBE_PIN)
180
-      static_assert(false, "Z_MIN_PROBE_PIN has no EXTINT line available.");
203
+      #error "Z_MIN_PROBE_PIN has no EXTINT line available."
181
     #endif
204
     #endif
182
     attachInterrupt(Z_MIN_PROBE_PIN, endstop_ISR, CHANGE);
205
     attachInterrupt(Z_MIN_PROBE_PIN, endstop_ISR, CHANGE);
183
   #endif
206
   #endif

+ 6
- 0
Marlin/src/HAL/HAL_STM32/endstop_interrupts.h View File

58
   #if HAS_Z3_MIN
58
   #if HAS_Z3_MIN
59
     attachInterrupt(Z3_MIN_PIN, endstop_ISR, CHANGE);
59
     attachInterrupt(Z3_MIN_PIN, endstop_ISR, CHANGE);
60
   #endif
60
   #endif
61
+  #if HAS_Z4_MAX
62
+    attachInterrupt(Z4_MAX_PIN, endstop_ISR, CHANGE);
63
+  #endif
64
+  #if HAS_Z4_MIN
65
+    attachInterrupt(Z4_MIN_PIN, endstop_ISR, CHANGE);
66
+  #endif
61
   #if HAS_Z_MIN_PROBE_PIN
67
   #if HAS_Z_MIN_PROBE_PIN
62
     attachInterrupt(Z_MIN_PROBE_PIN, endstop_ISR, CHANGE);
68
     attachInterrupt(Z_MIN_PROBE_PIN, endstop_ISR, CHANGE);
63
   #endif
69
   #endif

+ 6
- 0
Marlin/src/HAL/HAL_STM32F1/endstop_interrupts.h View File

83
   #if HAS_Z3_MIN
83
   #if HAS_Z3_MIN
84
     attachInterrupt(Z3_MIN_PIN, endstop_ISR, CHANGE);
84
     attachInterrupt(Z3_MIN_PIN, endstop_ISR, CHANGE);
85
   #endif
85
   #endif
86
+    #if HAS_Z4_MAX
87
+    attachInterrupt(Z4_MAX_PIN, endstop_ISR, CHANGE);
88
+  #endif
89
+  #if HAS_Z4_MIN
90
+    attachInterrupt(Z4_MIN_PIN, endstop_ISR, CHANGE);
91
+  #endif
86
   #if HAS_Z_MIN_PROBE_PIN
92
   #if HAS_Z_MIN_PROBE_PIN
87
     attachInterrupt(Z_MIN_PROBE_PIN, endstop_ISR, CHANGE);
93
     attachInterrupt(Z_MIN_PROBE_PIN, endstop_ISR, CHANGE);
88
   #endif
94
   #endif

+ 6
- 0
Marlin/src/HAL/HAL_STM32_F4_F7/endstop_interrupts.h View File

58
   #if HAS_Z3_MIN
58
   #if HAS_Z3_MIN
59
     attachInterrupt(Z3_MIN_PIN, endstop_ISR, CHANGE);
59
     attachInterrupt(Z3_MIN_PIN, endstop_ISR, CHANGE);
60
   #endif
60
   #endif
61
+  #if HAS_Z4_MAX
62
+    attachInterrupt(Z4_MAX_PIN, endstop_ISR, CHANGE);
63
+  #endif
64
+  #if HAS_Z4_MIN
65
+    attachInterrupt(Z4_MIN_PIN, endstop_ISR, CHANGE);
66
+  #endif
61
   #if HAS_Z_MIN_PROBE_PIN
67
   #if HAS_Z_MIN_PROBE_PIN
62
     attachInterrupt(Z_MIN_PROBE_PIN, endstop_ISR, CHANGE);
68
     attachInterrupt(Z_MIN_PROBE_PIN, endstop_ISR, CHANGE);
63
   #endif
69
   #endif

+ 6
- 0
Marlin/src/HAL/HAL_TEENSY35_36/endstop_interrupts.h View File

76
   #if HAS_Z3_MIN
76
   #if HAS_Z3_MIN
77
     _ATTACH(Z3_MIN_PIN);
77
     _ATTACH(Z3_MIN_PIN);
78
   #endif
78
   #endif
79
+  #if HAS_Z4_MAX
80
+    _ATTACH(Z4_MAX_PIN);
81
+  #endif
82
+  #if HAS_Z4_MIN
83
+    _ATTACH(Z4_MIN_PIN);
84
+  #endif
79
   #if HAS_Z_MIN_PROBE_PIN
85
   #if HAS_Z_MIN_PROBE_PIN
80
     _ATTACH(Z_MIN_PROBE_PIN);
86
     _ATTACH(Z_MIN_PROBE_PIN);
81
   #endif
87
   #endif

+ 5
- 3
Marlin/src/core/drivers.h View File

63
   #define AXIS_DRIVER_TYPE_X2(T) false
63
   #define AXIS_DRIVER_TYPE_X2(T) false
64
 #endif
64
 #endif
65
 #define AXIS_DRIVER_TYPE_Y2(T) (ENABLED(Y_DUAL_STEPPER_DRIVERS) && _AXIS_DRIVER_TYPE(Y2,T))
65
 #define AXIS_DRIVER_TYPE_Y2(T) (ENABLED(Y_DUAL_STEPPER_DRIVERS) && _AXIS_DRIVER_TYPE(Y2,T))
66
-#define AXIS_DRIVER_TYPE_Z2(T) (Z_MULTI_STEPPER_DRIVERS && _AXIS_DRIVER_TYPE(Z2,T))
67
-#define AXIS_DRIVER_TYPE_Z3(T) (ENABLED(Z_TRIPLE_STEPPER_DRIVERS) && _AXIS_DRIVER_TYPE(Z3,T))
66
+#define AXIS_DRIVER_TYPE_Z2(T) (NUM_Z_STEPPER_DRIVERS >= 2 && _AXIS_DRIVER_TYPE(Z2,T))
67
+#define AXIS_DRIVER_TYPE_Z3(T) (NUM_Z_STEPPER_DRIVERS >= 3 && _AXIS_DRIVER_TYPE(Z3,T))
68
+#define AXIS_DRIVER_TYPE_Z4(T) (NUM_Z_STEPPER_DRIVERS >= 4 && _AXIS_DRIVER_TYPE(Z4,T))
68
 #define AXIS_DRIVER_TYPE_E0(T) (E_STEPPERS > 0 && _AXIS_DRIVER_TYPE(E0,T))
69
 #define AXIS_DRIVER_TYPE_E0(T) (E_STEPPERS > 0 && _AXIS_DRIVER_TYPE(E0,T))
69
 #define AXIS_DRIVER_TYPE_E1(T) (E_STEPPERS > 1 && _AXIS_DRIVER_TYPE(E1,T))
70
 #define AXIS_DRIVER_TYPE_E1(T) (E_STEPPERS > 1 && _AXIS_DRIVER_TYPE(E1,T))
70
 #define AXIS_DRIVER_TYPE_E2(T) (E_STEPPERS > 2 && _AXIS_DRIVER_TYPE(E2,T))
71
 #define AXIS_DRIVER_TYPE_E2(T) (E_STEPPERS > 2 && _AXIS_DRIVER_TYPE(E2,T))
80
 
81
 
81
 #define HAS_DRIVER(T) (    AXIS_DRIVER_TYPE_X(T)  || AXIS_DRIVER_TYPE_X2(T) \
82
 #define HAS_DRIVER(T) (    AXIS_DRIVER_TYPE_X(T)  || AXIS_DRIVER_TYPE_X2(T) \
82
                         || AXIS_DRIVER_TYPE_Y(T)  || AXIS_DRIVER_TYPE_Y2(T) \
83
                         || AXIS_DRIVER_TYPE_Y(T)  || AXIS_DRIVER_TYPE_Y2(T) \
83
-                        || AXIS_DRIVER_TYPE_Z(T)  || AXIS_DRIVER_TYPE_Z2(T) || AXIS_DRIVER_TYPE_Z3(T) \
84
+                        || AXIS_DRIVER_TYPE_Z(T)  || AXIS_DRIVER_TYPE_Z2(T) \
85
+                        || AXIS_DRIVER_TYPE_Z3(T) || AXIS_DRIVER_TYPE_Z4(T) \
84
                         || HAS_E_DRIVER(T) )
86
                         || HAS_E_DRIVER(T) )
85
 
87
 
86
 // Test for supported TMC drivers that require advanced configuration
88
 // Test for supported TMC drivers that require advanced configuration

+ 3
- 0
Marlin/src/core/language.h View File

195
 #define MSG_Z2_MAX                          "z2_max"
195
 #define MSG_Z2_MAX                          "z2_max"
196
 #define MSG_Z3_MIN                          "z3_min"
196
 #define MSG_Z3_MIN                          "z3_min"
197
 #define MSG_Z3_MAX                          "z3_max"
197
 #define MSG_Z3_MAX                          "z3_max"
198
+#define MSG_Z4_MIN                          "z4_min"
199
+#define MSG_Z4_MAX                          "z4_max"
198
 #define MSG_Z_PROBE                         "z_probe"
200
 #define MSG_Z_PROBE                         "z_probe"
199
 #define MSG_FILAMENT_RUNOUT_SENSOR          "filament"
201
 #define MSG_FILAMENT_RUNOUT_SENSOR          "filament"
200
 #define MSG_PROBE_OFFSET                    "Probe Offset"
202
 #define MSG_PROBE_OFFSET                    "Probe Offset"
333
 #define MSG_Y2 "Y2"
335
 #define MSG_Y2 "Y2"
334
 #define MSG_Z2 "Z2"
336
 #define MSG_Z2 "Z2"
335
 #define MSG_Z3 "Z3"
337
 #define MSG_Z3 "Z3"
338
+#define MSG_Z4 "Z4"
336
 
339
 
337
 #define LCD_STR_A MSG_A
340
 #define LCD_STR_A MSG_A
338
 #define LCD_STR_B MSG_B
341
 #define LCD_STR_B MSG_B

+ 3
- 0
Marlin/src/feature/controllerfan.cpp View File

55
       #if HAS_Z3_ENABLE
55
       #if HAS_Z3_ENABLE
56
         || Z3_ENABLE_READ() == bool(Z_ENABLE_ON)
56
         || Z3_ENABLE_READ() == bool(Z_ENABLE_ON)
57
       #endif
57
       #endif
58
+      #if HAS_Z4_ENABLE
59
+        || Z4_ENABLE_READ() == bool(Z_ENABLE_ON)
60
+      #endif
58
       #if E_STEPPERS
61
       #if E_STEPPERS
59
         #define _OR_ENABLED_E(N) || E##N##_ENABLE_READ() == bool(E_ENABLE_ON)
62
         #define _OR_ENABLED_E(N) || E##N##_ENABLE_READ() == bool(E_ENABLE_ON)
60
         REPEAT(E_STEPPERS, _OR_ENABLED_E)
63
         REPEAT(E_STEPPERS, _OR_ENABLED_E)

+ 22
- 1
Marlin/src/feature/tmc_util.cpp View File

387
       }
387
       }
388
       #endif
388
       #endif
389
 
389
 
390
-      #if AXIS_IS_TMC(Z) || AXIS_IS_TMC(Z2) || AXIS_IS_TMC(Z3)
390
+      #if AXIS_IS_TMC(Z) || AXIS_IS_TMC(Z2) || AXIS_IS_TMC(Z3) || AXIS_IS_TMC(Z4)
391
       {
391
       {
392
         bool result = false;
392
         bool result = false;
393
         #if AXIS_IS_TMC(Z)
393
         #if AXIS_IS_TMC(Z)
399
         #if AXIS_IS_TMC(Z3)
399
         #if AXIS_IS_TMC(Z3)
400
           if (monitor_tmc_driver(stepperZ3, need_update_error_counters, need_debug_reporting)) result = true;
400
           if (monitor_tmc_driver(stepperZ3, need_update_error_counters, need_debug_reporting)) result = true;
401
         #endif
401
         #endif
402
+        #if AXIS_IS_TMC(Z4)
403
+          if (monitor_tmc_driver(stepperZ4, need_update_error_counters, need_debug_reporting)) result = true;
404
+        #endif
402
         if (result) {
405
         if (result) {
403
           #if AXIS_IS_TMC(Z)
406
           #if AXIS_IS_TMC(Z)
404
             step_current_down(stepperZ);
407
             step_current_down(stepperZ);
409
           #if AXIS_IS_TMC(Z3)
412
           #if AXIS_IS_TMC(Z3)
410
             step_current_down(stepperZ3);
413
             step_current_down(stepperZ3);
411
           #endif
414
           #endif
415
+          #if AXIS_IS_TMC(Z4)
416
+            step_current_down(stepperZ4);
417
+          #endif
412
         }
418
         }
413
       }
419
       }
414
       #endif
420
       #endif
750
       #if AXIS_IS_TMC(Z3)
756
       #if AXIS_IS_TMC(Z3)
751
         tmc_status(stepperZ3, i);
757
         tmc_status(stepperZ3, i);
752
       #endif
758
       #endif
759
+      #if AXIS_IS_TMC(Z4)
760
+        tmc_status(stepperZ4, i);
761
+      #endif
753
     }
762
     }
754
 
763
 
755
     if (print_e) {
764
     if (print_e) {
805
       #if AXIS_IS_TMC(Z3)
814
       #if AXIS_IS_TMC(Z3)
806
         tmc_parse_drv_status(stepperZ3, i);
815
         tmc_parse_drv_status(stepperZ3, i);
807
       #endif
816
       #endif
817
+      #if AXIS_IS_TMC(Z4)
818
+        tmc_parse_drv_status(stepperZ4, i);
819
+      #endif
808
     }
820
     }
809
 
821
 
810
     if (print_e) {
822
     if (print_e) {
980
       #if AXIS_IS_TMC(Z3)
992
       #if AXIS_IS_TMC(Z3)
981
         tmc_get_registers(stepperZ3, i);
993
         tmc_get_registers(stepperZ3, i);
982
       #endif
994
       #endif
995
+      #if AXIS_IS_TMC(Z4)
996
+        tmc_get_registers(stepperZ4, i);
997
+      #endif
983
     }
998
     }
984
 
999
 
985
     if (print_e) {
1000
     if (print_e) {
1086
     #if AXIS_HAS_SPI(Z3)
1101
     #if AXIS_HAS_SPI(Z3)
1087
       SET_CS_PIN(Z3);
1102
       SET_CS_PIN(Z3);
1088
     #endif
1103
     #endif
1104
+    #if AXIS_HAS_SPI(Z4)
1105
+      SET_CS_PIN(Z4);
1106
+    #endif
1089
     #if AXIS_HAS_SPI(E0)
1107
     #if AXIS_HAS_SPI(E0)
1090
       SET_CS_PIN(E0);
1108
       SET_CS_PIN(E0);
1091
     #endif
1109
     #endif
1160
     #if AXIS_IS_TMC(Z3)
1178
     #if AXIS_IS_TMC(Z3)
1161
       axis_connection += test_connection(stepperZ3);
1179
       axis_connection += test_connection(stepperZ3);
1162
     #endif
1180
     #endif
1181
+    #if AXIS_IS_TMC(Z4)
1182
+      axis_connection += test_connection(stepperZ4);
1183
+    #endif
1163
   }
1184
   }
1164
 
1185
 
1165
   if (test_e) {
1186
   if (test_e) {

+ 1
- 1
Marlin/src/feature/tmc_util.h View File

350
 #if USE_SENSORLESS
350
 #if USE_SENSORLESS
351
 
351
 
352
   // Track enabled status of stealthChop and only re-enable where applicable
352
   // Track enabled status of stealthChop and only re-enable where applicable
353
-  struct sensorless_t { bool x, y, z, x2, y2, z2, z3; };
353
+  struct sensorless_t { bool x, y, z, x2, y2, z2, z3, z4; };
354
 
354
 
355
   #if ENABLED(IMPROVE_HOMING_RELIABILITY)
355
   #if ENABLED(IMPROVE_HOMING_RELIABILITY)
356
     extern millis_t sg_guard_period;
356
     extern millis_t sg_guard_period;

+ 28
- 15
Marlin/src/gcode/calibrate/G34_M422.cpp View File

52
 
52
 
53
 #if ENABLED(Z_STEPPER_ALIGN_KNOWN_STEPPER_POSITIONS)
53
 #if ENABLED(Z_STEPPER_ALIGN_KNOWN_STEPPER_POSITIONS)
54
 
54
 
55
-  static_assert(COUNT(test_z_stepper_align_xy) >= Z_STEPPER_COUNT,
55
+  static_assert(COUNT(test_z_stepper_align_xy) >= NUM_Z_STEPPER_DRIVERS,
56
     "Z_STEPPER_ALIGN_XY requires at least three {X,Y} entries (Z, Z2, Z3, ...)."
56
     "Z_STEPPER_ALIGN_XY requires at least three {X,Y} entries (Z, Z2, Z3, ...)."
57
   );
57
   );
58
 
58
 
59
   constexpr float test_z_stepper_align_stepper_xy[][XY] = Z_STEPPER_ALIGN_STEPPER_XY;
59
   constexpr float test_z_stepper_align_stepper_xy[][XY] = Z_STEPPER_ALIGN_STEPPER_XY;
60
   static_assert(
60
   static_assert(
61
-    COUNT(test_z_stepper_align_stepper_xy) == Z_STEPPER_COUNT,
61
+    COUNT(test_z_stepper_align_stepper_xy) == NUM_Z_STEPPER_DRIVERS,
62
     "Z_STEPPER_ALIGN_STEPPER_XY requires three {X,Y} entries (one per Z stepper)."
62
     "Z_STEPPER_ALIGN_STEPPER_XY requires three {X,Y} entries (one per Z stepper)."
63
   );
63
   );
64
 
64
 
65
 #else
65
 #else
66
 
66
 
67
-  static_assert(COUNT(test_z_stepper_align_xy) == Z_STEPPER_COUNT,
68
-    #if ENABLED(Z_TRIPLE_STEPPER_DRIVERS)
67
+  static_assert(COUNT(test_z_stepper_align_xy) == NUM_Z_STEPPER_DRIVERS,
68
+    #if NUM_Z_STEPPER_DRIVERS == 4
69
+      "Z_STEPPER_ALIGN_XY requires four {X,Y} entries (Z, Z2, Z3, and Z4)."
70
+    #elif NUM_Z_STEPPER_DRIVERS == 3
69
       "Z_STEPPER_ALIGN_XY requires three {X,Y} entries (Z, Z2, and Z3)."
71
       "Z_STEPPER_ALIGN_XY requires three {X,Y} entries (Z, Z2, and Z3)."
70
     #else
72
     #else
71
       "Z_STEPPER_ALIGN_XY requires two {X,Y} entries (Z and Z2)."
73
       "Z_STEPPER_ALIGN_XY requires two {X,Y} entries (Z and Z2)."
85
 static_assert(FTEST(0) && BTEST(0), "The 1st Z_STEPPER_ALIGN_XY Y is unreachable with the default probe Y offset.");
87
 static_assert(FTEST(0) && BTEST(0), "The 1st Z_STEPPER_ALIGN_XY Y is unreachable with the default probe Y offset.");
86
 static_assert(LTEST(1) && RTEST(1), "The 2nd Z_STEPPER_ALIGN_XY X is unreachable with the default probe X offset.");
88
 static_assert(LTEST(1) && RTEST(1), "The 2nd Z_STEPPER_ALIGN_XY X is unreachable with the default probe X offset.");
87
 static_assert(FTEST(1) && BTEST(1), "The 2nd Z_STEPPER_ALIGN_XY Y is unreachable with the default probe Y offset.");
89
 static_assert(FTEST(1) && BTEST(1), "The 2nd Z_STEPPER_ALIGN_XY Y is unreachable with the default probe Y offset.");
88
-
89
-#if ENABLED(Z_TRIPLE_STEPPER_DRIVERS)
90
+#if NUM_Z_STEPPER_DRIVERS >= 3
90
   static_assert(LTEST(2) && RTEST(2), "The 3rd Z_STEPPER_ALIGN_XY X is unreachable with the default probe X offset.");
91
   static_assert(LTEST(2) && RTEST(2), "The 3rd Z_STEPPER_ALIGN_XY X is unreachable with the default probe X offset.");
91
   static_assert(FTEST(2) && BTEST(2), "The 3rd Z_STEPPER_ALIGN_XY Y is unreachable with the default probe Y offset.");
92
   static_assert(FTEST(2) && BTEST(2), "The 3rd Z_STEPPER_ALIGN_XY Y is unreachable with the default probe Y offset.");
93
+  #if NUM_Z_STEPPER_DRIVERS >= 4
94
+    static_assert(LTEST(3) && RTEST(3), "The 4th Z_STEPPER_ALIGN_XY X is unreachable with the default probe X offset.");
95
+    static_assert(FTEST(3) && BTEST(3), "The 4th Z_STEPPER_ALIGN_XY Y is unreachable with the default probe Y offset.");
96
+  #endif
92
 #endif
97
 #endif
93
 
98
 
94
 //
99
 //
105
 inline void set_all_z_lock(const bool lock) {
110
 inline void set_all_z_lock(const bool lock) {
106
   stepper.set_z_lock(lock);
111
   stepper.set_z_lock(lock);
107
   stepper.set_z2_lock(lock);
112
   stepper.set_z2_lock(lock);
108
-  #if ENABLED(Z_TRIPLE_STEPPER_DRIVERS)
113
+  #if NUM_Z_STEPPER_DRIVERS >= 3
109
     stepper.set_z3_lock(lock);
114
     stepper.set_z3_lock(lock);
115
+    #if NUM_Z_STEPPER_DRIVERS >= 4
116
+      stepper.set_z4_lock(lock);
117
+    #endif
110
   #endif
118
   #endif
111
 }
119
 }
112
 
120
 
125
 
133
 
126
   do { // break out on error
134
   do { // break out on error
127
 
135
 
136
+    #if NUM_Z_STEPPER_DRIVERS == 4
137
+      SERIAL_ECHOLNPGM("Quad Z Stepper Leveling not Yet Supported");
138
+      break;
139
+    #endif
140
+
128
     const int8_t z_auto_align_iterations = parser.intval('I', Z_STEPPER_ALIGN_ITERATIONS);
141
     const int8_t z_auto_align_iterations = parser.intval('I', Z_STEPPER_ALIGN_ITERATIONS);
129
     if (!WITHIN(z_auto_align_iterations, 1, 30)) {
142
     if (!WITHIN(z_auto_align_iterations, 1, 30)) {
130
       SERIAL_ECHOLNPGM("?(I)teration out of bounds (1-30).");
143
       SERIAL_ECHOLNPGM("?(I)teration out of bounds (1-30).");
187
     // Compute a worst-case clearance height to probe from. After the first
200
     // Compute a worst-case clearance height to probe from. After the first
188
     // iteration this will be re-calculated based on the actual bed position
201
     // iteration this will be re-calculated based on the actual bed position
189
     float z_probe = Z_BASIC_CLEARANCE + (G34_MAX_GRADE) * 0.01f * (
202
     float z_probe = Z_BASIC_CLEARANCE + (G34_MAX_GRADE) * 0.01f * (
190
-      #if ENABLED(Z_TRIPLE_STEPPER_DRIVERS)
203
+      #if NUM_Z_STEPPER_DRIVERS == 3
191
          SQRT(_MAX(HYPOT2(z_stepper_align_pos[0].x - z_stepper_align_pos[0].y, z_stepper_align_pos[1].x - z_stepper_align_pos[1].y),
204
          SQRT(_MAX(HYPOT2(z_stepper_align_pos[0].x - z_stepper_align_pos[0].y, z_stepper_align_pos[1].x - z_stepper_align_pos[1].y),
192
                    HYPOT2(z_stepper_align_pos[1].x - z_stepper_align_pos[1].y, z_stepper_align_pos[2].x - z_stepper_align_pos[2].y),
205
                    HYPOT2(z_stepper_align_pos[1].x - z_stepper_align_pos[1].y, z_stepper_align_pos[2].x - z_stepper_align_pos[2].y),
193
                    HYPOT2(z_stepper_align_pos[2].x - z_stepper_align_pos[2].y, z_stepper_align_pos[0].x - z_stepper_align_pos[0].y)))
206
                    HYPOT2(z_stepper_align_pos[2].x - z_stepper_align_pos[2].y, z_stepper_align_pos[0].x - z_stepper_align_pos[0].y)))
202
     // Move the Z coordinate realm towards the positive - dirty trick
215
     // Move the Z coordinate realm towards the positive - dirty trick
203
     current_position.z -= z_probe * 0.5f;
216
     current_position.z -= z_probe * 0.5f;
204
 
217
 
205
-    float last_z_align_move[Z_STEPPER_COUNT] = ARRAY_N(Z_STEPPER_COUNT, 10000.0f, 10000.0f, 10000.0f),
218
+    float last_z_align_move[NUM_Z_STEPPER_DRIVERS] = ARRAY_N(NUM_Z_STEPPER_DRIVERS, 10000.0f, 10000.0f, 10000.0f),
206
           z_measured[G34_PROBE_COUNT] = { 0 },
219
           z_measured[G34_PROBE_COUNT] = { 0 },
207
           z_maxdiff = 0.0f,
220
           z_maxdiff = 0.0f,
208
           amplification = z_auto_align_amplification;
221
           amplification = z_auto_align_amplification;
273
         finish_incremental_LSF(&lfd);
286
         finish_incremental_LSF(&lfd);
274
 
287
 
275
         z_measured_min = 100000.0f;
288
         z_measured_min = 100000.0f;
276
-        for (uint8_t i = 0; i < Z_STEPPER_COUNT; ++i) {
289
+        for (uint8_t i = 0; i < NUM_Z_STEPPER_DRIVERS; ++i) {
277
           z_measured[i] = -(lfd.A * z_stepper_align_stepper_pos[i].x + lfd.B * z_stepper_align_stepper_pos[i].y);
290
           z_measured[i] = -(lfd.A * z_stepper_align_stepper_pos[i].x + lfd.B * z_stepper_align_stepper_pos[i].y);
278
           z_measured_min = _MIN(z_measured_min, z_measured[i]);
291
           z_measured_min = _MIN(z_measured_min, z_measured[i]);
279
         }
292
         }
283
 
296
 
284
       SERIAL_ECHOLNPAIR("\n"
297
       SERIAL_ECHOLNPAIR("\n"
285
         "DIFFERENCE Z1-Z2=", ABS(z_measured[0] - z_measured[1])
298
         "DIFFERENCE Z1-Z2=", ABS(z_measured[0] - z_measured[1])
286
-        #if ENABLED(Z_TRIPLE_STEPPER_DRIVERS)
299
+        #if NUM_Z_STEPPER_DRIVERS == 3
287
           , " Z2-Z3=", ABS(z_measured[1] - z_measured[2])
300
           , " Z2-Z3=", ABS(z_measured[1] - z_measured[2])
288
           , " Z3-Z1=", ABS(z_measured[2] - z_measured[0])
301
           , " Z3-Z1=", ABS(z_measured[2] - z_measured[0])
289
         #endif
302
         #endif
294
 
307
 
295
       bool success_break = true;
308
       bool success_break = true;
296
       // Correct the individual stepper offsets
309
       // Correct the individual stepper offsets
297
-      for (uint8_t zstepper = 0; zstepper < Z_STEPPER_COUNT; ++zstepper) {
310
+      for (uint8_t zstepper = 0; zstepper < NUM_Z_STEPPER_DRIVERS; ++zstepper) {
298
         // Calculate current stepper move
311
         // Calculate current stepper move
299
         const float z_align_move = z_measured[zstepper] - z_measured_min,
312
         const float z_align_move = z_measured[zstepper] - z_measured_min,
300
                     z_align_abs = ABS(z_align_move);
313
                     z_align_abs = ABS(z_align_move);
324
         switch (zstepper) {
337
         switch (zstepper) {
325
           case 0: stepper.set_z_lock(false); break;
338
           case 0: stepper.set_z_lock(false); break;
326
           case 1: stepper.set_z2_lock(false); break;
339
           case 1: stepper.set_z2_lock(false); break;
327
-          #if ENABLED(Z_TRIPLE_STEPPER_DRIVERS)
340
+          #if NUM_Z_STEPPER_DRIVERS == 3
328
             case 2: stepper.set_z3_lock(false); break;
341
             case 2: stepper.set_z3_lock(false); break;
329
           #endif
342
           #endif
330
         }
343
         }
397
     for (uint8_t i = 0; i < G34_PROBE_COUNT; ++i)
410
     for (uint8_t i = 0; i < G34_PROBE_COUNT; ++i)
398
       SERIAL_ECHOLNPAIR_P(PSTR("M422 S"), i + 1, SP_X_STR, z_stepper_align_pos[i].x, SP_Y_STR, z_stepper_align_pos[i].y);
411
       SERIAL_ECHOLNPAIR_P(PSTR("M422 S"), i + 1, SP_X_STR, z_stepper_align_pos[i].x, SP_Y_STR, z_stepper_align_pos[i].y);
399
     #if ENABLED(Z_STEPPER_ALIGN_KNOWN_STEPPER_POSITIONS)
412
     #if ENABLED(Z_STEPPER_ALIGN_KNOWN_STEPPER_POSITIONS)
400
-      for (uint8_t i = 0; i < Z_STEPPER_COUNT; ++i)
413
+      for (uint8_t i = 0; i < NUM_Z_STEPPER_DRIVERS; ++i)
401
         SERIAL_ECHOLNPAIR_P(PSTR("M422 W"), i + 1, SP_X_STR, z_stepper_align_stepper_pos[i].x, SP_Y_STR, z_stepper_align_stepper_pos[i].y);
414
         SERIAL_ECHOLNPAIR_P(PSTR("M422 W"), i + 1, SP_X_STR, z_stepper_align_stepper_pos[i].x, SP_Y_STR, z_stepper_align_stepper_pos[i].y);
402
     #endif
415
     #endif
403
     return;
416
     return;
446
   else {
459
   else {
447
     #if ENABLED(Z_STEPPER_ALIGN_KNOWN_STEPPER_POSITIONS)
460
     #if ENABLED(Z_STEPPER_ALIGN_KNOWN_STEPPER_POSITIONS)
448
       position_index = parser.intval('W') - 1;
461
       position_index = parser.intval('W') - 1;
449
-      if (!WITHIN(position_index, 0, Z_STEPPER_COUNT - 1)) {
462
+      if (!WITHIN(position_index, 0, NUM_Z_STEPPER_DRIVERS - 1)) {
450
         SERIAL_ECHOLNPGM("?(W) Z-Stepper index invalid.");
463
         SERIAL_ECHOLNPGM("?(W) Z-Stepper index invalid.");
451
         return;
464
         return;
452
       }
465
       }

+ 22
- 13
Marlin/src/gcode/calibrate/M666.cpp View File

57
    * M666: Set Dual Endstops offsets for X, Y, and/or Z.
57
    * M666: Set Dual Endstops offsets for X, Y, and/or Z.
58
    *       With no parameters report current offsets.
58
    *       With no parameters report current offsets.
59
    *
59
    *
60
-   * For Triple Z Endstops:
60
+   * For Triple / Quad Z Endstops:
61
    *   Set Z2 Only: M666 S2 Z<offset>
61
    *   Set Z2 Only: M666 S2 Z<offset>
62
    *   Set Z3 Only: M666 S3 Z<offset>
62
    *   Set Z3 Only: M666 S3 Z<offset>
63
-   *      Set Both: M666 Z<offset>
63
+   *   Set Z4 Only: M666 S4 Z<offset>
64
+   *       Set All: M666 Z<offset>
64
    */
65
    */
65
   void GcodeSuite::M666() {
66
   void GcodeSuite::M666() {
66
     #if ENABLED(X_DUAL_ENDSTOPS)
67
     #if ENABLED(X_DUAL_ENDSTOPS)
69
     #if ENABLED(Y_DUAL_ENDSTOPS)
70
     #if ENABLED(Y_DUAL_ENDSTOPS)
70
       if (parser.seenval('Y')) endstops.y2_endstop_adj = parser.value_linear_units();
71
       if (parser.seenval('Y')) endstops.y2_endstop_adj = parser.value_linear_units();
71
     #endif
72
     #endif
72
-    #if ENABLED(Z_TRIPLE_ENDSTOPS)
73
+    #if ENABLED(Z_MULTI_ENDSTOPS)
73
       if (parser.seenval('Z')) {
74
       if (parser.seenval('Z')) {
74
-        const float z_adj = parser.value_linear_units();
75
-        const int ind = parser.intval('S');
76
-        if (!ind || ind == 2) endstops.z2_endstop_adj = z_adj;
77
-        if (!ind || ind == 3) endstops.z3_endstop_adj = z_adj;
75
+        #if NUM_Z_STEPPER_DRIVERS >= 3
76
+          const float z_adj = parser.value_linear_units();
77
+          const int ind = parser.intval('S');
78
+          if (!ind || ind == 2) endstops.z2_endstop_adj = z_adj;
79
+          if (!ind || ind == 3) endstops.z3_endstop_adj = z_adj;
80
+          #if NUM_Z_STEPPER_DRIVERS >= 4
81
+            if (!ind || ind == 4) endstops.z4_endstop_adj = z_adj;
82
+          #endif
83
+        #else
84
+          endstops.z2_endstop_adj = parser.value_linear_units();
85
+        #endif
78
       }
86
       }
79
-    #elif Z_MULTI_ENDSTOPS
80
-      if (parser.seen('Z')) endstops.z2_endstop_adj = parser.value_linear_units();
81
     #endif
87
     #endif
82
     if (!parser.seen("XYZ")) {
88
     if (!parser.seen("XYZ")) {
83
       SERIAL_ECHOPGM("Dual Endstop Adjustment (mm): ");
89
       SERIAL_ECHOPGM("Dual Endstop Adjustment (mm): ");
87
       #if ENABLED(Y_DUAL_ENDSTOPS)
93
       #if ENABLED(Y_DUAL_ENDSTOPS)
88
         SERIAL_ECHOPAIR(" Y2:", endstops.y2_endstop_adj);
94
         SERIAL_ECHOPAIR(" Y2:", endstops.y2_endstop_adj);
89
       #endif
95
       #endif
90
-      #if Z_MULTI_ENDSTOPS
96
+      #if ENABLED(Z_MULTI_ENDSTOPS)
91
         SERIAL_ECHOPAIR(" Z2:", endstops.z2_endstop_adj);
97
         SERIAL_ECHOPAIR(" Z2:", endstops.z2_endstop_adj);
92
-      #endif
93
-      #if ENABLED(Z_TRIPLE_ENDSTOPS)
94
-        SERIAL_ECHOPAIR(" Z3:", endstops.z3_endstop_adj);
98
+        #if NUM_Z_STEPPER_DRIVERS >= 3
99
+          SERIAL_ECHOPAIR(" Z3:", endstops.z3_endstop_adj);
100
+          #if NUM_Z_STEPPER_DRIVERS >= 4
101
+            SERIAL_ECHOPAIR(" Z4:", endstops.z4_endstop_adj);
102
+          #endif
103
+        #endif
95
       #endif
104
       #endif
96
       SERIAL_EOL();
105
       SERIAL_EOL();
97
     }
106
     }

+ 3
- 0
Marlin/src/gcode/feature/L6470/M122.cpp View File

115
   #if AXIS_IS_L64XX(Z3)
115
   #if AXIS_IS_L64XX(Z3)
116
     L6470_say_status(Z3);
116
     L6470_say_status(Z3);
117
   #endif
117
   #endif
118
+  #if AXIS_IS_L64XX(Z4)
119
+    L6470_say_status(Z4);
120
+  #endif
118
   #if AXIS_IS_L64XX(E0)
121
   #if AXIS_IS_L64XX(E0)
119
     L6470_say_status(E0);
122
     L6470_say_status(E0);
120
   #endif
123
   #endif

+ 7
- 1
Marlin/src/gcode/feature/L6470/M906.cpp View File

44
  *     1 - monitor only X, Y, Z or E1
44
  *     1 - monitor only X, Y, Z or E1
45
  *     2 - monitor only X2, Y2, Z2 or E2
45
  *     2 - monitor only X2, Y2, Z2 or E2
46
  *     3 - monitor only Z3 or E3
46
  *     3 - monitor only Z3 or E3
47
- *     4 - monitor only E4
47
+ *     4 - monitor only Z4 or E4
48
  *     5 - monitor only E5
48
  *     5 - monitor only E5
49
  * Xxxx, Yxxx, Zxxx, Exxx - axis to change (optional)
49
  * Xxxx, Yxxx, Zxxx, Exxx - axis to change (optional)
50
  *     L6474 - current in mA (4A max)
50
  *     L6474 - current in mA (4A max)
274
         #if AXIS_IS_L64XX(Z3)
274
         #if AXIS_IS_L64XX(Z3)
275
           if (index == 2) L6470_SET_KVAL_HOLD(Z3);
275
           if (index == 2) L6470_SET_KVAL_HOLD(Z3);
276
         #endif
276
         #endif
277
+        #if AXIS_DRIVER_TYPE_Z4(L6470)
278
+          if (index == 3) L6470_SET_KVAL_HOLD(Z4);
279
+        #endif
277
         break;
280
         break;
278
       case E_AXIS: {
281
       case E_AXIS: {
279
         const int8_t target_extruder = get_target_extruder_from_command();
282
         const int8_t target_extruder = get_target_extruder_from_command();
328
     #if AXIS_IS_L64XX(Z3)
331
     #if AXIS_IS_L64XX(Z3)
329
       L64XX_REPORT_CURRENT(Z3);
332
       L64XX_REPORT_CURRENT(Z3);
330
     #endif
333
     #endif
334
+    #if AXIS_IS_L64XX(Z4)
335
+      L64XX_REPORT_CURRENT(Z4);
336
+    #endif
331
     #if AXIS_IS_L64XX(E0)
337
     #if AXIS_IS_L64XX(E0)
332
       L64XX_REPORT_CURRENT(E0);
338
       L64XX_REPORT_CURRENT(E0);
333
     #endif
339
     #endif

+ 1
- 0
Marlin/src/gcode/feature/L6470/M916-918.cpp View File

46
  *     1 - monitor only X, Y, Z, E1
46
  *     1 - monitor only X, Y, Z, E1
47
  *     2 - monitor only X2, Y2, Z2, E2
47
  *     2 - monitor only X2, Y2, Z2, E2
48
  *     3 - monitor only Z3, E3
48
  *     3 - monitor only Z3, E3
49
+ *     4 - monitor only Z4, E4
49
  *
50
  *
50
  * Xxxx, Yxxx, Zxxx, Exxx - axis to be monitored with displacement
51
  * Xxxx, Yxxx, Zxxx, Exxx - axis to be monitored with displacement
51
  *     xxx (1-255) is distance moved on either side of current position
52
  *     xxx (1-255) is distance moved on either side of current position

+ 10
- 3
Marlin/src/gcode/feature/trinamic/M569.cpp View File

44
 static void set_stealth_status(const bool enable, const int8_t target_extruder) {
44
 static void set_stealth_status(const bool enable, const int8_t target_extruder) {
45
   #define TMC_SET_STEALTH(Q) tmc_set_stealthChop(stepper##Q, enable)
45
   #define TMC_SET_STEALTH(Q) tmc_set_stealthChop(stepper##Q, enable)
46
 
46
 
47
-  #if    AXIS_HAS_STEALTHCHOP(X) || AXIS_HAS_STEALTHCHOP(X2) \
48
-      || AXIS_HAS_STEALTHCHOP(Y) || AXIS_HAS_STEALTHCHOP(Y2) \
49
-      || AXIS_HAS_STEALTHCHOP(Z) || AXIS_HAS_STEALTHCHOP(Z2) || AXIS_HAS_STEALTHCHOP(Z3)
47
+  #if    AXIS_HAS_STEALTHCHOP(X)  || AXIS_HAS_STEALTHCHOP(X2) \
48
+      || AXIS_HAS_STEALTHCHOP(Y)  || AXIS_HAS_STEALTHCHOP(Y2) \
49
+      || AXIS_HAS_STEALTHCHOP(Z)  || AXIS_HAS_STEALTHCHOP(Z2) \
50
+      || AXIS_HAS_STEALTHCHOP(Z3) || AXIS_HAS_STEALTHCHOP(Z4)
50
     const uint8_t index = parser.byteval('I');
51
     const uint8_t index = parser.byteval('I');
51
   #endif
52
   #endif
52
 
53
 
78
         #if AXIS_HAS_STEALTHCHOP(Z3)
79
         #if AXIS_HAS_STEALTHCHOP(Z3)
79
           if (index == 2) TMC_SET_STEALTH(Z3);
80
           if (index == 2) TMC_SET_STEALTH(Z3);
80
         #endif
81
         #endif
82
+        #if AXIS_HAS_STEALTHCHOP(Z4)
83
+          if (index == 3) TMC_SET_STEALTH(Z4);
84
+        #endif
81
         break;
85
         break;
82
       case E_AXIS: {
86
       case E_AXIS: {
83
         if (target_extruder < 0) return;
87
         if (target_extruder < 0) return;
130
   #if AXIS_HAS_STEALTHCHOP(Z3)
134
   #if AXIS_HAS_STEALTHCHOP(Z3)
131
     TMC_SAY_STEALTH_STATUS(Z3);
135
     TMC_SAY_STEALTH_STATUS(Z3);
132
   #endif
136
   #endif
137
+  #if AXIS_HAS_STEALTHCHOP(Z4)
138
+    TMC_SAY_STEALTH_STATUS(Z4);
139
+  #endif
133
   #if AXIS_HAS_STEALTHCHOP(E0)
140
   #if AXIS_HAS_STEALTHCHOP(E0)
134
     TMC_SAY_STEALTH_STATUS(E0);
141
     TMC_SAY_STEALTH_STATUS(E0);
135
   #endif
142
   #endif

+ 8
- 2
Marlin/src/gcode/feature/trinamic/M906.cpp View File

37
  *   Z[current]  - Set mA current for Z driver(s)
37
  *   Z[current]  - Set mA current for Z driver(s)
38
  *   E[current]  - Set mA current for E driver(s)
38
  *   E[current]  - Set mA current for E driver(s)
39
  *
39
  *
40
- *   I[index]    - Axis sub-index (Omit or 0 for X, Y, Z; 1 for X2, Y2, Z2; 2 for Z3.)
40
+ *   I[index]    - Axis sub-index (Omit or 0 for X, Y, Z; 1 for X2, Y2, Z2; 2 for Z3; 3 for Z4.)
41
  *   T[index]    - Extruder index (Zero-based. Omit for E0 only.)
41
  *   T[index]    - Extruder index (Zero-based. Omit for E0 only.)
42
  *
42
  *
43
  * With no parameters report driver currents.
43
  * With no parameters report driver currents.
48
 
48
 
49
   bool report = true;
49
   bool report = true;
50
 
50
 
51
-  #if AXIS_IS_TMC(X) || AXIS_IS_TMC(X2) || AXIS_IS_TMC(Y) || AXIS_IS_TMC(Y2) || AXIS_IS_TMC(Z) || AXIS_IS_TMC(Z2) || AXIS_IS_TMC(Z3)
51
+  #if AXIS_IS_TMC(X) || AXIS_IS_TMC(X2) || AXIS_IS_TMC(Y) || AXIS_IS_TMC(Y2) || AXIS_IS_TMC(Z) || AXIS_IS_TMC(Z2) || AXIS_IS_TMC(Z3) || AXIS_IS_TMC(Z4)
52
     const uint8_t index = parser.byteval('I');
52
     const uint8_t index = parser.byteval('I');
53
   #endif
53
   #endif
54
 
54
 
81
         #if AXIS_IS_TMC(Z3)
81
         #if AXIS_IS_TMC(Z3)
82
           if (index == 2) TMC_SET_CURRENT(Z3);
82
           if (index == 2) TMC_SET_CURRENT(Z3);
83
         #endif
83
         #endif
84
+        #if AXIS_IS_TMC(Z4)
85
+          if (index == 3) TMC_SET_CURRENT(Z4);
86
+        #endif
84
         break;
87
         break;
85
       case E_AXIS: {
88
       case E_AXIS: {
86
         const int8_t target_extruder = get_target_extruder_from_command();
89
         const int8_t target_extruder = get_target_extruder_from_command();
131
     #if AXIS_IS_TMC(Z3)
134
     #if AXIS_IS_TMC(Z3)
132
       TMC_SAY_CURRENT(Z3);
135
       TMC_SAY_CURRENT(Z3);
133
     #endif
136
     #endif
137
+    #if AXIS_IS_TMC(Z4)
138
+      TMC_SAY_CURRENT(Z4);
139
+    #endif
134
     #if AXIS_IS_TMC(E0)
140
     #if AXIS_IS_TMC(E0)
135
       TMC_SAY_CURRENT(E0);
141
       TMC_SAY_CURRENT(E0);
136
     #endif
142
     #endif

+ 21
- 3
Marlin/src/gcode/feature/trinamic/M911-M914.cpp View File

37
 
37
 
38
   #define M91x_SOME_X (M91x_USE(X) || M91x_USE(X2))
38
   #define M91x_SOME_X (M91x_USE(X) || M91x_USE(X2))
39
   #define M91x_SOME_Y (M91x_USE(Y) || M91x_USE(Y2))
39
   #define M91x_SOME_Y (M91x_USE(Y) || M91x_USE(Y2))
40
-  #define M91x_SOME_Z (M91x_USE(Z) || M91x_USE(Z2) || M91x_USE(Z3))
40
+  #define M91x_SOME_Z (M91x_USE(Z) || M91x_USE(Z2) || M91x_USE(Z3) || M91x_USE(Z4))
41
   #define M91x_SOME_E (M91x_USE_E(0) || M91x_USE_E(1) || M91x_USE_E(2) || M91x_USE_E(3) || M91x_USE_E(4) || M91x_USE_E(5))
41
   #define M91x_SOME_E (M91x_USE_E(0) || M91x_USE_E(1) || M91x_USE_E(2) || M91x_USE_E(3) || M91x_USE_E(4) || M91x_USE_E(5))
42
 
42
 
43
   #if !M91x_SOME_X && !M91x_SOME_Y && !M91x_SOME_Z && !M91x_SOME_E
43
   #if !M91x_SOME_X && !M91x_SOME_Y && !M91x_SOME_Z && !M91x_SOME_E
70
     #if M91x_USE(Z3)
70
     #if M91x_USE(Z3)
71
       tmc_report_otpw(stepperZ3);
71
       tmc_report_otpw(stepperZ3);
72
     #endif
72
     #endif
73
+    #if M91x_USE(Z4)
74
+      tmc_report_otpw(stepperZ4);
75
+    #endif
73
     #if M91x_USE_E(0)
76
     #if M91x_USE_E(0)
74
       tmc_report_otpw(stepperE0);
77
       tmc_report_otpw(stepperE0);
75
     #endif
78
     #endif
92
 
95
 
93
   /**
96
   /**
94
    * M912: Clear TMC stepper driver overtemperature pre-warn flag held by the library
97
    * M912: Clear TMC stepper driver overtemperature pre-warn flag held by the library
95
-   *       Specify one or more axes with X, Y, Z, X1, Y1, Z1, X2, Y2, Z2, Z3 and E[index].
98
+   *       Specify one or more axes with X, Y, Z, X1, Y1, Z1, X2, Y2, Z2, Z3, Z4 and E[index].
96
    *       If no axes are given, clear all.
99
    *       If no axes are given, clear all.
97
    *
100
    *
98
    * Examples:
101
    * Examples:
160
       #if M91x_USE(Z3)
163
       #if M91x_USE(Z3)
161
         if (hasNone || zval == 3 || (hasZ && zval < 0)) tmc_clear_otpw(stepperZ3);
164
         if (hasNone || zval == 3 || (hasZ && zval < 0)) tmc_clear_otpw(stepperZ3);
162
       #endif
165
       #endif
166
+      #if M91x_USE(Z4)
167
+        if (hasNone || zval == 4 || (hasZ && zval < 0)) tmc_clear_otpw(stepperZ4);
168
+      #endif
163
     #endif
169
     #endif
164
 
170
 
165
     #if M91x_SOME_E
171
     #if M91x_SOME_E
198
     #define TMC_SET_PWMTHRS_E(E) stepperE##E.set_pwm_thrs(value)
204
     #define TMC_SET_PWMTHRS_E(E) stepperE##E.set_pwm_thrs(value)
199
 
205
 
200
     bool report = true;
206
     bool report = true;
201
-    #if AXIS_IS_TMC(X) || AXIS_IS_TMC(X2) || AXIS_IS_TMC(Y) || AXIS_IS_TMC(Y2) || AXIS_IS_TMC(Z) || AXIS_IS_TMC(Z2) || AXIS_IS_TMC(Z3)
207
+    #if AXIS_IS_TMC(X) || AXIS_IS_TMC(X2) || AXIS_IS_TMC(Y) || AXIS_IS_TMC(Y2) || AXIS_IS_TMC(Z) || AXIS_IS_TMC(Z2) || AXIS_IS_TMC(Z3) || AXIS_IS_TMC(Z4)
202
       const uint8_t index = parser.byteval('I');
208
       const uint8_t index = parser.byteval('I');
203
     #endif
209
     #endif
204
     LOOP_XYZE(i) if (int32_t value = parser.longval(axis_codes[i])) {
210
     LOOP_XYZE(i) if (int32_t value = parser.longval(axis_codes[i])) {
230
           #if AXIS_HAS_STEALTHCHOP(Z3)
236
           #if AXIS_HAS_STEALTHCHOP(Z3)
231
             if (index == 0 || index == 3) TMC_SET_PWMTHRS(Z,Z3);
237
             if (index == 0 || index == 3) TMC_SET_PWMTHRS(Z,Z3);
232
           #endif
238
           #endif
239
+          #if AXIS_HAS_STEALTHCHOP(Z4)
240
+            if (index == 0 || index == 4) TMC_SET_PWMTHRS(Z,Z4);
241
+          #endif
233
           break;
242
           break;
234
         case E_AXIS: {
243
         case E_AXIS: {
235
           #if E_STEPPERS
244
           #if E_STEPPERS
282
       #if AXIS_HAS_STEALTHCHOP(Z3)
291
       #if AXIS_HAS_STEALTHCHOP(Z3)
283
         TMC_SAY_PWMTHRS(Z,Z3);
292
         TMC_SAY_PWMTHRS(Z,Z3);
284
       #endif
293
       #endif
294
+      #if AXIS_HAS_STEALTHCHOP(Z4)
295
+        TMC_SAY_PWMTHRS(Z,Z4);
296
+      #endif
285
       #if E_STEPPERS && AXIS_HAS_STEALTHCHOP(E0)
297
       #if E_STEPPERS && AXIS_HAS_STEALTHCHOP(E0)
286
         TMC_SAY_PWMTHRS_E(0);
298
         TMC_SAY_PWMTHRS_E(0);
287
       #endif
299
       #endif
347
             #if AXIS_HAS_STALLGUARD(Z3)
359
             #if AXIS_HAS_STALLGUARD(Z3)
348
               if (index == 0 || index == 3) stepperZ3.homing_threshold(value);
360
               if (index == 0 || index == 3) stepperZ3.homing_threshold(value);
349
             #endif
361
             #endif
362
+            #if AXIS_HAS_STALLGUARD(Z4)
363
+              if (index == 0 || index == 4) stepperZ4.homing_threshold(value);
364
+            #endif
350
             break;
365
             break;
351
         #endif
366
         #endif
352
       }
367
       }
379
         #if AXIS_HAS_STALLGUARD(Z3)
394
         #if AXIS_HAS_STALLGUARD(Z3)
380
           tmc_print_sgt(stepperZ3);
395
           tmc_print_sgt(stepperZ3);
381
         #endif
396
         #endif
397
+        #if AXIS_HAS_STALLGUARD(Z4)
398
+          tmc_print_sgt(stepperZ4);
399
+        #endif
382
       #endif
400
       #endif
383
     }
401
     }
384
   }
402
   }

+ 2
- 2
Marlin/src/gcode/gcode.cpp View File

566
         case 665: M665(); break;                                  // M665: Set delta configurations
566
         case 665: M665(); break;                                  // M665: Set delta configurations
567
       #endif
567
       #endif
568
 
568
 
569
-      #if ANY(DELTA, X_DUAL_ENDSTOPS, Y_DUAL_ENDSTOPS, Z_DUAL_ENDSTOPS)
570
-        case 666: M666(); break;                                  // M666: Set delta or dual endstop adjustment
569
+      #if ENABLED(DELTA) || HAS_EXTRA_ENDSTOPS
570
+        case 666: M666(); break;                                  // M666: Set delta or multiple endstop adjustment
571
       #endif
571
       #endif
572
 
572
 
573
       #if ENABLED(FWRETRACT)
573
       #if ENABLED(FWRETRACT)

+ 3
- 0
Marlin/src/gcode/host/M114.cpp View File

121
       #if AXIS_IS_L64XX(Z3)
121
       #if AXIS_IS_L64XX(Z3)
122
         REPORT_ABSOLUTE_POS(Z3);
122
         REPORT_ABSOLUTE_POS(Z3);
123
       #endif
123
       #endif
124
+      #if AXIS_IS_L64XX(Z4)
125
+        REPORT_ABSOLUTE_POS(Z4);
126
+      #endif
124
       #if AXIS_IS_L64XX(E0)
127
       #if AXIS_IS_L64XX(E0)
125
         REPORT_ABSOLUTE_POS(E0);
128
         REPORT_ABSOLUTE_POS(E0);
126
       #endif
129
       #endif

+ 1
- 4
Marlin/src/inc/Conditionals_LCD.h View File

526
   #define GRID_MAX_POINTS ((GRID_MAX_POINTS_X) * (GRID_MAX_POINTS_Y))
526
   #define GRID_MAX_POINTS ((GRID_MAX_POINTS_X) * (GRID_MAX_POINTS_Y))
527
 #endif
527
 #endif
528
 
528
 
529
+#define HAS_EXTRA_ENDSTOPS           ANY(X_DUAL_ENDSTOPS, Y_DUAL_ENDSTOPS, Z_MULTI_ENDSTOPS)
529
 #define HAS_SOFTWARE_ENDSTOPS        EITHER(MIN_SOFTWARE_ENDSTOPS, MAX_SOFTWARE_ENDSTOPS)
530
 #define HAS_SOFTWARE_ENDSTOPS        EITHER(MIN_SOFTWARE_ENDSTOPS, MAX_SOFTWARE_ENDSTOPS)
530
 #define HAS_RESUME_CONTINUE          ANY(EXTENSIBLE_UI, NEWPANEL, EMERGENCY_PARSER)
531
 #define HAS_RESUME_CONTINUE          ANY(EXTENSIBLE_UI, NEWPANEL, EMERGENCY_PARSER)
531
 #define HAS_COLOR_LEDS               ANY(BLINKM, RGB_LED, RGBW_LED, PCA9632, PCA9533, NEOPIXEL_LED)
532
 #define HAS_COLOR_LEDS               ANY(BLINKM, RGB_LED, RGBW_LED, PCA9632, PCA9533, NEOPIXEL_LED)
535
 #define HAS_SERVICE_INTERVALS        (ENABLED(PRINTCOUNTER) && (SERVICE_INTERVAL_1 > 0 || SERVICE_INTERVAL_2 > 0 || SERVICE_INTERVAL_3 > 0))
536
 #define HAS_SERVICE_INTERVALS        (ENABLED(PRINTCOUNTER) && (SERVICE_INTERVAL_1 > 0 || SERVICE_INTERVAL_2 > 0 || SERVICE_INTERVAL_3 > 0))
536
 #define HAS_FILAMENT_SENSOR          ENABLED(FILAMENT_RUNOUT_SENSOR)
537
 #define HAS_FILAMENT_SENSOR          ENABLED(FILAMENT_RUNOUT_SENSOR)
537
 
538
 
538
-#define Z_MULTI_STEPPER_DRIVERS EITHER(Z_DUAL_STEPPER_DRIVERS, Z_TRIPLE_STEPPER_DRIVERS)
539
-#define Z_MULTI_ENDSTOPS        EITHER(Z_DUAL_ENDSTOPS, Z_TRIPLE_ENDSTOPS)
540
-#define HAS_EXTRA_ENDSTOPS     (EITHER(X_DUAL_ENDSTOPS, Y_DUAL_ENDSTOPS) || Z_MULTI_ENDSTOPS)
541
-
542
 #define HAS_GAMES     ANY(MARLIN_BRICKOUT, MARLIN_INVADERS, MARLIN_SNAKE, MARLIN_MAZE)
539
 #define HAS_GAMES     ANY(MARLIN_BRICKOUT, MARLIN_INVADERS, MARLIN_SNAKE, MARLIN_MAZE)
543
 #define HAS_GAME_MENU (1 < ENABLED(MARLIN_BRICKOUT) + ENABLED(MARLIN_INVADERS) + ENABLED(MARLIN_SNAKE) + ENABLED(MARLIN_MAZE))
540
 #define HAS_GAME_MENU (1 < ENABLED(MARLIN_BRICKOUT) + ENABLED(MARLIN_INVADERS) + ENABLED(MARLIN_SNAKE) + ENABLED(MARLIN_MAZE))
544
 
541
 

+ 5
- 0
Marlin/src/inc/Conditionals_adv.h View File

54
   #undef SHOW_TEMP_ADC_VALUES
54
   #undef SHOW_TEMP_ADC_VALUES
55
 #endif
55
 #endif
56
 
56
 
57
+// Multiple Z steppers
58
+#ifndef NUM_Z_STEPPER_DRIVERS
59
+  #define NUM_Z_STEPPER_DRIVERS 1
60
+#endif
61
+
57
 #define HAS_CUTTER EITHER(SPINDLE_FEATURE, LASER_FEATURE)
62
 #define HAS_CUTTER EITHER(SPINDLE_FEATURE, LASER_FEATURE)
58
 
63
 
59
 #if !defined(__AVR__) || !defined(USBCON)
64
 #if !defined(__AVR__) || !defined(USBCON)

+ 118
- 71
Marlin/src/inc/Conditionals_post.h View File

687
   #endif
687
   #endif
688
 #endif
688
 #endif
689
 
689
 
690
-// Is an endstop plug used for the X2 endstop?
691
-#define IS_X2_ENDSTOP(A,M) (ENABLED(X_DUAL_ENDSTOPS) && X2_USE_ENDSTOP == _##A##M##_)
692
-
693
 /**
690
 /**
694
  * Y_DUAL_ENDSTOPS endstop reassignment
691
  * Y_DUAL_ENDSTOPS endstop reassignment
695
  */
692
  */
743
   #endif
740
   #endif
744
 #endif
741
 #endif
745
 
742
 
746
-// Is an endstop plug used for the Y2 endstop or the bed probe?
747
-#define IS_Y2_ENDSTOP(A,M) (ENABLED(Y_DUAL_ENDSTOPS) && Y2_USE_ENDSTOP == _##A##M##_)
748
-
749
 /**
743
 /**
750
- * Z_DUAL_ENDSTOPS endstop reassignment
744
+ * Z_MULTI_ENDSTOPS endstop reassignment
751
  */
745
  */
752
-#if Z_MULTI_ENDSTOPS
746
+#if ENABLED(Z_MULTI_ENDSTOPS)
747
+
753
   #if Z_HOME_DIR > 0
748
   #if Z_HOME_DIR > 0
754
     #if Z2_USE_ENDSTOP == _XMIN_
749
     #if Z2_USE_ENDSTOP == _XMIN_
755
       #define Z2_MAX_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
750
       #define Z2_MAX_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
797
     #endif
792
     #endif
798
     #define Z2_MAX_ENDSTOP_INVERTING false
793
     #define Z2_MAX_ENDSTOP_INVERTING false
799
   #endif
794
   #endif
800
-#endif
801
 
795
 
802
-#if ENABLED(Z_TRIPLE_ENDSTOPS)
803
-  #if Z_HOME_DIR > 0
804
-    #if Z3_USE_ENDSTOP == _XMIN_
805
-      #define Z3_MAX_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
806
-      #define Z3_MAX_PIN X_MIN_PIN
807
-    #elif Z3_USE_ENDSTOP == _XMAX_
808
-      #define Z3_MAX_ENDSTOP_INVERTING X_MAX_ENDSTOP_INVERTING
809
-      #define Z3_MAX_PIN X_MAX_PIN
810
-    #elif Z3_USE_ENDSTOP == _YMIN_
811
-      #define Z3_MAX_ENDSTOP_INVERTING Y_MIN_ENDSTOP_INVERTING
812
-      #define Z3_MAX_PIN Y_MIN_PIN
813
-    #elif Z3_USE_ENDSTOP == _YMAX_
814
-      #define Z3_MAX_ENDSTOP_INVERTING Y_MAX_ENDSTOP_INVERTING
815
-      #define Z3_MAX_PIN Y_MAX_PIN
816
-    #elif Z3_USE_ENDSTOP == _ZMIN_
817
-      #define Z3_MAX_ENDSTOP_INVERTING Z_MIN_ENDSTOP_INVERTING
818
-      #define Z3_MAX_PIN Z_MIN_PIN
819
-    #elif Z3_USE_ENDSTOP == _ZMAX_
820
-      #define Z3_MAX_ENDSTOP_INVERTING Z_MAX_ENDSTOP_INVERTING
821
-      #define Z3_MAX_PIN Z_MAX_PIN
796
+  #if NUM_Z_STEPPER_DRIVERS >= 3
797
+    #if Z_HOME_DIR > 0
798
+      #if Z3_USE_ENDSTOP == _XMIN_
799
+        #define Z3_MAX_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
800
+        #define Z3_MAX_PIN X_MIN_PIN
801
+      #elif Z3_USE_ENDSTOP == _XMAX_
802
+        #define Z3_MAX_ENDSTOP_INVERTING X_MAX_ENDSTOP_INVERTING
803
+        #define Z3_MAX_PIN X_MAX_PIN
804
+      #elif Z3_USE_ENDSTOP == _YMIN_
805
+        #define Z3_MAX_ENDSTOP_INVERTING Y_MIN_ENDSTOP_INVERTING
806
+        #define Z3_MAX_PIN Y_MIN_PIN
807
+      #elif Z3_USE_ENDSTOP == _YMAX_
808
+        #define Z3_MAX_ENDSTOP_INVERTING Y_MAX_ENDSTOP_INVERTING
809
+        #define Z3_MAX_PIN Y_MAX_PIN
810
+      #elif Z3_USE_ENDSTOP == _ZMIN_
811
+        #define Z3_MAX_ENDSTOP_INVERTING Z_MIN_ENDSTOP_INVERTING
812
+        #define Z3_MAX_PIN Z_MIN_PIN
813
+      #elif Z3_USE_ENDSTOP == _ZMAX_
814
+        #define Z3_MAX_ENDSTOP_INVERTING Z_MAX_ENDSTOP_INVERTING
815
+        #define Z3_MAX_PIN Z_MAX_PIN
816
+      #else
817
+        #define Z3_MAX_ENDSTOP_INVERTING false
818
+      #endif
819
+      #define Z3_MIN_ENDSTOP_INVERTING false
822
     #else
820
     #else
821
+      #if Z3_USE_ENDSTOP == _XMIN_
822
+        #define Z3_MIN_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
823
+        #define Z3_MIN_PIN X_MIN_PIN
824
+      #elif Z3_USE_ENDSTOP == _XMAX_
825
+        #define Z3_MIN_ENDSTOP_INVERTING X_MAX_ENDSTOP_INVERTING
826
+        #define Z3_MIN_PIN X_MAX_PIN
827
+      #elif Z3_USE_ENDSTOP == _YMIN_
828
+        #define Z3_MIN_ENDSTOP_INVERTING Y_MIN_ENDSTOP_INVERTING
829
+        #define Z3_MIN_PIN Y_MIN_PIN
830
+      #elif Z3_USE_ENDSTOP == _YMAX_
831
+        #define Z3_MIN_ENDSTOP_INVERTING Y_MAX_ENDSTOP_INVERTING
832
+        #define Z3_MIN_PIN Y_MAX_PIN
833
+      #elif Z3_USE_ENDSTOP == _ZMIN_
834
+        #define Z3_MIN_ENDSTOP_INVERTING Z_MIN_ENDSTOP_INVERTING
835
+        #define Z3_MIN_PIN Z_MIN_PIN
836
+      #elif Z3_USE_ENDSTOP == _ZMAX_
837
+        #define Z3_MIN_ENDSTOP_INVERTING Z_MAX_ENDSTOP_INVERTING
838
+        #define Z3_MIN_PIN Z_MAX_PIN
839
+      #else
840
+        #define Z3_MIN_ENDSTOP_INVERTING false
841
+      #endif
823
       #define Z3_MAX_ENDSTOP_INVERTING false
842
       #define Z3_MAX_ENDSTOP_INVERTING false
824
     #endif
843
     #endif
825
-    #define Z3_MIN_ENDSTOP_INVERTING false
826
-  #else
827
-    #if Z3_USE_ENDSTOP == _XMIN_
828
-      #define Z3_MIN_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
829
-      #define Z3_MIN_PIN X_MIN_PIN
830
-    #elif Z3_USE_ENDSTOP == _XMAX_
831
-      #define Z3_MIN_ENDSTOP_INVERTING X_MAX_ENDSTOP_INVERTING
832
-      #define Z3_MIN_PIN X_MAX_PIN
833
-    #elif Z3_USE_ENDSTOP == _YMIN_
834
-      #define Z3_MIN_ENDSTOP_INVERTING Y_MIN_ENDSTOP_INVERTING
835
-      #define Z3_MIN_PIN Y_MIN_PIN
836
-    #elif Z3_USE_ENDSTOP == _YMAX_
837
-      #define Z3_MIN_ENDSTOP_INVERTING Y_MAX_ENDSTOP_INVERTING
838
-      #define Z3_MIN_PIN Y_MAX_PIN
839
-    #elif Z3_USE_ENDSTOP == _ZMIN_
840
-      #define Z3_MIN_ENDSTOP_INVERTING Z_MIN_ENDSTOP_INVERTING
841
-      #define Z3_MIN_PIN Z_MIN_PIN
842
-    #elif Z3_USE_ENDSTOP == _ZMAX_
843
-      #define Z3_MIN_ENDSTOP_INVERTING Z_MAX_ENDSTOP_INVERTING
844
-      #define Z3_MIN_PIN Z_MAX_PIN
844
+  #endif
845
+
846
+  #if NUM_Z_STEPPER_DRIVERS >= 4
847
+    #if Z_HOME_DIR > 0
848
+      #if Z4_USE_ENDSTOP == _XMIN_
849
+        #define Z4_MAX_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
850
+        #define Z4_MAX_PIN X_MIN_PIN
851
+      #elif Z4_USE_ENDSTOP == _XMAX_
852
+        #define Z4_MAX_ENDSTOP_INVERTING X_MAX_ENDSTOP_INVERTING
853
+        #define Z4_MAX_PIN X_MAX_PIN
854
+      #elif Z4_USE_ENDSTOP == _YMIN_
855
+        #define Z4_MAX_ENDSTOP_INVERTING Y_MIN_ENDSTOP_INVERTING
856
+        #define Z4_MAX_PIN Y_MIN_PIN
857
+      #elif Z4_USE_ENDSTOP == _YMAX_
858
+        #define Z4_MAX_ENDSTOP_INVERTING Y_MAX_ENDSTOP_INVERTING
859
+        #define Z4_MAX_PIN Y_MAX_PIN
860
+      #elif Z4_USE_ENDSTOP == _ZMIN_
861
+        #define Z4_MAX_ENDSTOP_INVERTING Z_MIN_ENDSTOP_INVERTING
862
+        #define Z4_MAX_PIN Z_MIN_PIN
863
+      #elif Z4_USE_ENDSTOP == _ZMAX_
864
+        #define Z4_MAX_ENDSTOP_INVERTING Z_MAX_ENDSTOP_INVERTING
865
+        #define Z4_MAX_PIN Z_MAX_PIN
866
+      #else
867
+        #define Z4_MAX_ENDSTOP_INVERTING false
868
+      #endif
869
+      #define Z4_MIN_ENDSTOP_INVERTING false
845
     #else
870
     #else
846
-      #define Z3_MIN_ENDSTOP_INVERTING false
871
+      #if Z4_USE_ENDSTOP == _XMIN_
872
+        #define Z4_MIN_ENDSTOP_INVERTING X_MIN_ENDSTOP_INVERTING
873
+        #define Z4_MIN_PIN X_MIN_PIN
874
+      #elif Z4_USE_ENDSTOP == _XMAX_
875
+        #define Z4_MIN_ENDSTOP_INVERTING X_MAX_ENDSTOP_INVERTING
876
+        #define Z4_MIN_PIN X_MAX_PIN
877
+      #elif Z4_USE_ENDSTOP == _YMIN_
878
+        #define Z4_MIN_ENDSTOP_INVERTING Y_MIN_ENDSTOP_INVERTING
879
+        #define Z4_MIN_PIN Y_MIN_PIN
880
+      #elif Z4_USE_ENDSTOP == _YMAX_
881
+        #define Z4_MIN_ENDSTOP_INVERTING Y_MAX_ENDSTOP_INVERTING
882
+        #define Z4_MIN_PIN Y_MAX_PIN
883
+      #elif Z4_USE_ENDSTOP == _ZMIN_
884
+        #define Z4_MIN_ENDSTOP_INVERTING Z_MIN_ENDSTOP_INVERTING
885
+        #define Z4_MIN_PIN Z_MIN_PIN
886
+      #elif Z4_USE_ENDSTOP == _ZMAX_
887
+        #define Z4_MIN_ENDSTOP_INVERTING Z_MAX_ENDSTOP_INVERTING
888
+        #define Z4_MIN_PIN Z_MAX_PIN
889
+      #else
890
+        #define Z4_MIN_ENDSTOP_INVERTING false
891
+      #endif
892
+      #define Z4_MAX_ENDSTOP_INVERTING false
847
     #endif
893
     #endif
848
-    #define Z3_MAX_ENDSTOP_INVERTING false
849
   #endif
894
   #endif
850
-#endif
851
-
852
-// Is an endstop plug used for the Z2 endstop or the bed probe?
853
-#define IS_Z2_OR_PROBE(A,M) ( \
854
-     (Z_MULTI_ENDSTOPS && Z2_USE_ENDSTOP == _##A##M##_) \
855
-  || (HAS_CUSTOM_PROBE_PIN && Z_MIN_PROBE_PIN == A##_##M##_PIN ) )
856
 
895
 
857
-// Is an endstop plug used for the Z3 endstop or the bed probe?
858
-#define IS_Z3_OR_PROBE(A,M) ( \
859
-     (ENABLED(Z_TRIPLE_ENDSTOPS) && Z3_USE_ENDSTOP == _##A##M##_) \
860
-  || (HAS_CUSTOM_PROBE_PIN && Z_MIN_PROBE_PIN == A##_##M##_PIN ) )
896
+#endif // Z_MULTI_ENDSTOPS
861
 
897
 
862
 /**
898
 /**
863
  * Set ENDSTOPPULLUPS for active endstop switches
899
  * Set ENDSTOPPULLUPS for active endstop switches
947
 #define HAS_Z3_STEP       (PIN_EXISTS(Z3_STEP))
983
 #define HAS_Z3_STEP       (PIN_EXISTS(Z3_STEP))
948
 #define HAS_Z3_MICROSTEPS (PIN_EXISTS(Z3_MS1))
984
 #define HAS_Z3_MICROSTEPS (PIN_EXISTS(Z3_MS1))
949
 
985
 
986
+#define HAS_Z4_ENABLE     (PIN_EXISTS(Z4_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(Z4)))
987
+#define HAS_Z4_DIR        (PIN_EXISTS(Z4_DIR))
988
+#define HAS_Z4_STEP       (PIN_EXISTS(Z4_STEP))
989
+#define HAS_Z4_MICROSTEPS (PIN_EXISTS(Z4_MS1))
990
+
950
 // Extruder steppers and solenoids
991
 // Extruder steppers and solenoids
951
 #define HAS_E0_ENABLE     (PIN_EXISTS(E0_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(E0)))
992
 #define HAS_E0_ENABLE     (PIN_EXISTS(E0_ENABLE) || (ENABLED(SOFTWARE_DRIVER_ENABLE) && AXIS_IS_TMC(E0)))
952
 #define HAS_E0_DIR        (PIN_EXISTS(E0_DIR))
993
 #define HAS_E0_DIR        (PIN_EXISTS(E0_DIR))
999
   #define Z_SENSORLESS  (AXIS_HAS_STALLGUARD(Z)  && defined(Z_STALL_SENSITIVITY))
1040
   #define Z_SENSORLESS  (AXIS_HAS_STALLGUARD(Z)  && defined(Z_STALL_SENSITIVITY))
1000
   #define Z2_SENSORLESS (AXIS_HAS_STALLGUARD(Z2) && defined(Z2_STALL_SENSITIVITY))
1041
   #define Z2_SENSORLESS (AXIS_HAS_STALLGUARD(Z2) && defined(Z2_STALL_SENSITIVITY))
1001
   #define Z3_SENSORLESS (AXIS_HAS_STALLGUARD(Z3) && defined(Z3_STALL_SENSITIVITY))
1042
   #define Z3_SENSORLESS (AXIS_HAS_STALLGUARD(Z3) && defined(Z3_STALL_SENSITIVITY))
1043
+  #define Z4_SENSORLESS (AXIS_HAS_STALLGUARD(Z4) && defined(Z4_STALL_SENSITIVITY))
1002
   #if ENABLED(SPI_ENDSTOPS)
1044
   #if ENABLED(SPI_ENDSTOPS)
1003
     #define X_SPI_SENSORLESS X_SENSORLESS
1045
     #define X_SPI_SENSORLESS X_SENSORLESS
1004
     #define Y_SPI_SENSORLESS Y_SENSORLESS
1046
     #define Y_SPI_SENSORLESS Y_SENSORLESS
1011
     && E0_ENABLE_PIN != Y_ENABLE_PIN && E1_ENABLE_PIN != Y_ENABLE_PIN ) \
1053
     && E0_ENABLE_PIN != Y_ENABLE_PIN && E1_ENABLE_PIN != Y_ENABLE_PIN ) \
1012
 )
1054
 )
1013
 
1055
 
1056
+//
1014
 // Endstops and bed probe
1057
 // Endstops and bed probe
1015
-#define _HAS_STOP(A,M) (PIN_EXISTS(A##_##M) && !IS_X2_ENDSTOP(A,M) && !IS_Y2_ENDSTOP(A,M) && !IS_Z2_OR_PROBE(A,M))
1058
+//
1059
+
1060
+// Is an endstop plug used for extra Z endstops or the probe?
1061
+#define IS_PROBE_PIN(A,M) (HAS_CUSTOM_PROBE_PIN && Z_MIN_PROBE_PIN == P)
1062
+#define IS_X2_ENDSTOP(A,M) (ENABLED(X_DUAL_ENDSTOPS) && X2_USE_ENDSTOP == _##A##M##_)
1063
+#define IS_Y2_ENDSTOP(A,M) (ENABLED(Y_DUAL_ENDSTOPS) && Y2_USE_ENDSTOP == _##A##M##_)
1064
+#define IS_Z2_ENDSTOP(A,M) (ENABLED(Z_MULTI_ENDSTOPS) && Z2_USE_ENDSTOP == _##A##M##_)
1065
+#define IS_Z3_ENDSTOP(A,M) (ENABLED(Z_MULTI_ENDSTOPS) && NUM_Z_STEPPER_DRIVERS >= 3 && Z3_USE_ENDSTOP == _##A##M##_)
1066
+#define IS_Z4_ENDSTOP(A,M) (ENABLED(Z_MULTI_ENDSTOPS) && NUM_Z_STEPPER_DRIVERS >= 4 && Z4_USE_ENDSTOP == _##A##M##_)
1067
+
1068
+#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))
1016
 #define HAS_X_MIN _HAS_STOP(X,MIN)
1069
 #define HAS_X_MIN _HAS_STOP(X,MIN)
1017
 #define HAS_X_MAX _HAS_STOP(X,MAX)
1070
 #define HAS_X_MAX _HAS_STOP(X,MAX)
1018
 #define HAS_Y_MIN _HAS_STOP(Y,MIN)
1071
 #define HAS_Y_MIN _HAS_STOP(Y,MIN)
1027
 #define HAS_Z2_MAX (PIN_EXISTS(Z2_MAX))
1080
 #define HAS_Z2_MAX (PIN_EXISTS(Z2_MAX))
1028
 #define HAS_Z3_MIN (PIN_EXISTS(Z3_MIN))
1081
 #define HAS_Z3_MIN (PIN_EXISTS(Z3_MIN))
1029
 #define HAS_Z3_MAX (PIN_EXISTS(Z3_MAX))
1082
 #define HAS_Z3_MAX (PIN_EXISTS(Z3_MAX))
1083
+#define HAS_Z4_MIN (PIN_EXISTS(Z4_MIN))
1084
+#define HAS_Z4_MAX (PIN_EXISTS(Z4_MAX))
1030
 #define HAS_Z_MIN_PROBE_PIN (HAS_CUSTOM_PROBE_PIN && PIN_EXISTS(Z_MIN_PROBE))
1085
 #define HAS_Z_MIN_PROBE_PIN (HAS_CUSTOM_PROBE_PIN && PIN_EXISTS(Z_MIN_PROBE))
1031
 #define HAS_CALIBRATION_PIN (PIN_EXISTS(CALIBRATION))
1086
 #define HAS_CALIBRATION_PIN (PIN_EXISTS(CALIBRATION))
1032
 
1087
 
1147
 #define HAS_DIGIPOTSS         (PIN_EXISTS(DIGIPOTSS))
1202
 #define HAS_DIGIPOTSS         (PIN_EXISTS(DIGIPOTSS))
1148
 #define HAS_MOTOR_CURRENT_PWM ANY_PIN(MOTOR_CURRENT_PWM_X, MOTOR_CURRENT_PWM_Y, MOTOR_CURRENT_PWM_XY, MOTOR_CURRENT_PWM_Z, MOTOR_CURRENT_PWM_E)
1203
 #define HAS_MOTOR_CURRENT_PWM ANY_PIN(MOTOR_CURRENT_PWM_X, MOTOR_CURRENT_PWM_Y, MOTOR_CURRENT_PWM_XY, MOTOR_CURRENT_PWM_Z, MOTOR_CURRENT_PWM_E)
1149
 
1204
 
1150
-#define HAS_SOME_Z_MICROSTEPS (HAS_Z_MICROSTEPS || HAS_Z2_MICROSTEPS || HAS_Z3_MICROSTEPS)
1205
+#define HAS_SOME_Z_MICROSTEPS (HAS_Z_MICROSTEPS || HAS_Z2_MICROSTEPS || HAS_Z3_MICROSTEPS || HAS_Z4_MICROSTEPS)
1151
 #define HAS_SOME_E_MICROSTEPS (HAS_E0_MICROSTEPS || HAS_E1_MICROSTEPS || HAS_E2_MICROSTEPS || HAS_E3_MICROSTEPS || HAS_E4_MICROSTEPS || HAS_E5_MICROSTEPS)
1206
 #define HAS_SOME_E_MICROSTEPS (HAS_E0_MICROSTEPS || HAS_E1_MICROSTEPS || HAS_E2_MICROSTEPS || HAS_E3_MICROSTEPS || HAS_E4_MICROSTEPS || HAS_E5_MICROSTEPS)
1152
 #define HAS_MICROSTEPS (HAS_X_MICROSTEPS || HAS_X2_MICROSTEPS || HAS_Y_MICROSTEPS || HAS_Y2_MICROSTEPS || HAS_SOME_Z_MICROSTEPS || HAS_SOME_E_MICROSTEPS)
1207
 #define HAS_MICROSTEPS (HAS_X_MICROSTEPS || HAS_X2_MICROSTEPS || HAS_Y_MICROSTEPS || HAS_Y2_MICROSTEPS || HAS_SOME_Z_MICROSTEPS || HAS_SOME_E_MICROSTEPS)
1153
 
1208
 
1753
 // If platform requires early initialization of watchdog to properly boot
1808
 // If platform requires early initialization of watchdog to properly boot
1754
 #define EARLY_WATCHDOG (ENABLED(USE_WATCHDOG) && defined(ARDUINO_ARCH_SAM))
1809
 #define EARLY_WATCHDOG (ENABLED(USE_WATCHDOG) && defined(ARDUINO_ARCH_SAM))
1755
 
1810
 
1756
-#if ENABLED(Z_TRIPLE_STEPPER_DRIVERS)
1757
-  #define Z_STEPPER_COUNT 3
1758
-#elif ENABLED(Z_DUAL_STEPPER_DRIVERS)
1759
-  #define Z_STEPPER_COUNT 2
1760
-#else
1761
-  #define Z_STEPPER_COUNT 1
1762
-#endif
1763
-
1764
 #if HAS_SPI_LCD
1811
 #if HAS_SPI_LCD
1765
   // Get LCD character width/height, which may be overridden by pins, configs, etc.
1812
   // Get LCD character width/height, which may be overridden by pins, configs, etc.
1766
   #ifndef LCD_WIDTH
1813
   #ifndef LCD_WIDTH

+ 90
- 62
Marlin/src/inc/SanityCheck.h View File

426
   #error "HOME_USING_SPREADCYCLE is now obsolete. Please remove it from Configuration_adv.h."
426
   #error "HOME_USING_SPREADCYCLE is now obsolete. Please remove it from Configuration_adv.h."
427
 #elif defined(DGUS_LCD)
427
 #elif defined(DGUS_LCD)
428
   #error "DGUS_LCD is now DGUS_LCD_UI_(ORIGIN|FYSETC|HIPRECY). Please update your configuration."
428
   #error "DGUS_LCD is now DGUS_LCD_UI_(ORIGIN|FYSETC|HIPRECY). Please update your configuration."
429
+#elif defined(X_DUAL_ENDSTOPS_ADJUSTMENT)
430
+  #error "X_DUAL_ENDSTOPS_ADJUSTMENT is now X2_ENDSTOP_ADJUSTMENT. Please update Configuration_adv.h."
431
+#elif defined(Y_DUAL_ENDSTOPS_ADJUSTMENT)
432
+  #error "Y_DUAL_ENDSTOPS_ADJUSTMENT is now Y2_ENDSTOP_ADJUSTMENT. Please update Configuration_adv.h."
433
+#elif defined(Z_DUAL_ENDSTOPS_ADJUSTMENT)
434
+  #error "Z_DUAL_ENDSTOPS_ADJUSTMENT is now Z2_ENDSTOP_ADJUSTMENT. Please update Configuration_adv.h."
435
+#elif defined(Z_TRIPLE_ENDSTOPS_ADJUSTMENT2) || defined(Z_TRIPLE_ENDSTOPS_ADJUSTMENT3)
436
+  #error "Z_TRIPLE_ENDSTOPS_ADJUSTMENT[23] is now Z[23]_ENDSTOP_ADJUSTMENT. Please update Configuration_adv.h."
437
+#elif defined(Z_QUAD_ENDSTOPS_ADJUSTMENT2) || defined(Z_QUAD_ENDSTOPS_ADJUSTMENT3) || defined(Z_QUAD_ENDSTOPS_ADJUSTMENT4)
438
+  #error "Z_QUAD_ENDSTOPS_ADJUSTMENT[234] is now Z[234]_ENDSTOP_ADJUSTMENT. Please update Configuration_adv.h."
439
+#elif defined(Z_DUAL_STEPPER_DRIVERS)
440
+  #error "Z_DUAL_STEPPER_DRIVERS is now NUM_Z_STEPPER_DRIVERS with a value of 2. Please update Configuration_adv.h."
441
+#elif defined(Z_TRIPLE_STEPPER_DRIVERS)
442
+  #error "Z_TRIPLE_STEPPER_DRIVERS is now NUM_Z_STEPPER_DRIVERS with a value of 3. Please update Configuration_adv.h."
443
+#elif defined(Z_QUAD_STEPPER_DRIVERS)
444
+  #error "Z_QUAD_STEPPER_DRIVERS is now NUM_Z_STEPPER_DRIVERS with a value of 4. Please update Configuration_adv.h."
445
+#elif defined(Z_DUAL_ENDSTOPS)
446
+  #error "Z_DUAL_ENDSTOPS is now Z_MULTI_ENDSTOPS. Please update Configuration_adv.h."
447
+#elif defined(Z_TRIPLE_ENDSTOPS)
448
+  #error "Z_TRIPLE_ENDSTOPS is now Z_MULTI_ENDSTOPS. Please update Configuration_adv.h."
449
+#elif defined(Z_QUAD_ENDSTOPS)
450
+  #error "Z_QUAD_ENDSTOPS is now Z_MULTI_ENDSTOPS. Please update Configuration_adv.h."
429
 #endif
451
 #endif
430
 
452
 
431
 /**
453
 /**
473
 #endif
495
 #endif
474
 
496
 
475
 /**
497
 /**
476
- * Dual / Triple Stepper Drivers
498
+ * Multiple Stepper Drivers Per Axis
477
  */
499
  */
478
-#if BOTH(X_DUAL_STEPPER_DRIVERS, DUAL_X_CARRIAGE)
479
-  #error "DUAL_X_CARRIAGE is not compatible with X_DUAL_STEPPER_DRIVERS."
480
-#elif ENABLED(X_DUAL_STEPPER_DRIVERS) && !(HAS_X2_ENABLE && HAS_X2_STEP && HAS_X2_DIR)
481
-  #error "X_DUAL_STEPPER_DRIVERS requires X2 pins (and an extra E plug)."
482
-#elif ENABLED(Y_DUAL_STEPPER_DRIVERS) && !(HAS_Y2_ENABLE && HAS_Y2_STEP && HAS_Y2_DIR)
483
-  #error "Y_DUAL_STEPPER_DRIVERS requires Y2 pins (and an extra E plug)."
484
-#elif ENABLED(Z_DUAL_STEPPER_DRIVERS)
485
-  #if ENABLED(Z_TRIPLE_STEPPER_DRIVERS)
486
-    #error "Please select either Z_TRIPLE_STEPPER_DRIVERS or Z_DUAL_STEPPER_DRIVERS, not both."
487
-  #elif !(HAS_Z2_ENABLE && HAS_Z2_STEP && HAS_Z2_DIR)
488
-    #error "Z_DUAL_STEPPER_DRIVERS requires Z2 pins (and an extra E plug)."
500
+#define GOOD_AXIS_PINS(A) (HAS_##A##_ENABLE && HAS_##A##_STEP && HAS_##A##_DIR)
501
+#if ENABLED(X_DUAL_STEPPER_DRIVERS)
502
+  #if ENABLED(DUAL_X_CARRIAGE)
503
+    #error "DUAL_X_CARRIAGE is not compatible with X_DUAL_STEPPER_DRIVERS."
504
+  #elif !GOOD_AXIS_PINS(X)
505
+    #error "X_DUAL_STEPPER_DRIVERS requires X2 pins to be defined."
489
   #endif
506
   #endif
490
-#elif ENABLED(Z_TRIPLE_STEPPER_DRIVERS) && !(HAS_Z2_ENABLE && HAS_Z2_STEP && HAS_Z2_DIR && HAS_Z3_ENABLE && HAS_Z3_STEP && HAS_Z3_DIR)
491
-  #error "Z_TRIPLE_STEPPER_DRIVERS requires Z3 pins (and two extra E plugs)."
507
+#endif
508
+
509
+#if ENABLED(Y_DUAL_STEPPER_DRIVERS) && !GOOD_AXIS_PINS(Y)
510
+  #error "Y_DUAL_STEPPER_DRIVERS requires Y2 pins to be defined."
511
+#elif !WITHIN(NUM_Z_STEPPER_DRIVERS, 1, 4)
512
+  #error "NUM_Z_STEPPER_DRIVERS must be an integer from 1 to 4."
513
+#elif NUM_Z_STEPPER_DRIVERS == 2 && !GOOD_AXIS_PINS(Z2)
514
+  #error "If NUM_Z_STEPPER_DRIVERS is 2, you must define stepper pins for Z2."
515
+#elif NUM_Z_STEPPER_DRIVERS == 3 && !(GOOD_AXIS_PINS(Z2) && GOOD_AXIS_PINS(Z3))
516
+  #error "If NUM_Z_STEPPER_DRIVERS is 3, you must define stepper pins for Z2 and Z3."
517
+#elif NUM_Z_STEPPER_DRIVERS == 4 && !(GOOD_AXIS_PINS(Z2) && GOOD_AXIS_PINS(Z3) && GOOD_AXIS_PINS(Z4))
518
+  #error "If NUM_Z_STEPPER_DRIVERS is 4, you must define stepper pins for Z2, Z3, and Z4."
492
 #endif
519
 #endif
493
 
520
 
494
 /**
521
 /**
1393
     #error "DUAL_X_CARRIAGE requires 2 (or more) extruders."
1420
     #error "DUAL_X_CARRIAGE requires 2 (or more) extruders."
1394
   #elif CORE_IS_XY || CORE_IS_XZ
1421
   #elif CORE_IS_XY || CORE_IS_XZ
1395
     #error "DUAL_X_CARRIAGE cannot be used with COREXY, COREYX, COREXZ, or COREZX."
1422
     #error "DUAL_X_CARRIAGE cannot be used with COREXY, COREYX, COREXZ, or COREZX."
1396
-  #elif !(HAS_X2_ENABLE && HAS_X2_STEP && HAS_X2_DIR)
1423
+  #elif !GOOD_AXIS_PINS(X2)
1397
     #error "DUAL_X_CARRIAGE requires X2 stepper pins to be defined."
1424
     #error "DUAL_X_CARRIAGE requires X2 stepper pins to be defined."
1398
   #elif !HAS_X_MAX
1425
   #elif !HAS_X_MAX
1399
     #error "DUAL_X_CARRIAGE requires USE_XMAX_PLUG and an X Max Endstop."
1426
     #error "DUAL_X_CARRIAGE requires USE_XMAX_PLUG and an X Max Endstop."
1404
   #endif
1431
   #endif
1405
 #endif
1432
 #endif
1406
 
1433
 
1434
+#undef GOOD_AXIS_PINS
1435
+
1407
 /**
1436
 /**
1408
  * Make sure auto fan pins don't conflict with the fan pin
1437
  * Make sure auto fan pins don't conflict with the fan pin
1409
  */
1438
  */
1695
   #error "Enable USE_ZMAX_PLUG when homing Z to MAX."
1724
   #error "Enable USE_ZMAX_PLUG when homing Z to MAX."
1696
 #endif
1725
 #endif
1697
 
1726
 
1698
-// Dual endstops requirements
1727
+// Dual/multiple endstops requirements
1699
 #if ENABLED(X_DUAL_ENDSTOPS)
1728
 #if ENABLED(X_DUAL_ENDSTOPS)
1700
   #if !X2_USE_ENDSTOP
1729
   #if !X2_USE_ENDSTOP
1701
     #error "You must set X2_USE_ENDSTOP with X_DUAL_ENDSTOPS."
1730
     #error "You must set X2_USE_ENDSTOP with X_DUAL_ENDSTOPS."
1738
     #error "Y_DUAL_ENDSTOPS is not compatible with DELTA."
1767
     #error "Y_DUAL_ENDSTOPS is not compatible with DELTA."
1739
   #endif
1768
   #endif
1740
 #endif
1769
 #endif
1741
-#if ENABLED(Z_DUAL_ENDSTOPS)
1770
+
1771
+#if ENABLED(Z_MULTI_ENDSTOPS)
1742
   #if !Z2_USE_ENDSTOP
1772
   #if !Z2_USE_ENDSTOP
1743
-    #error "You must set Z2_USE_ENDSTOP with Z_DUAL_ENDSTOPS."
1773
+    #error "You must set Z2_USE_ENDSTOP with Z_MULTI_ENDSTOPS when NUM_Z_STEPPER_DRIVERS >= 2."
1744
   #elif Z2_USE_ENDSTOP == _XMIN_ && DISABLED(USE_XMIN_PLUG)
1774
   #elif Z2_USE_ENDSTOP == _XMIN_ && DISABLED(USE_XMIN_PLUG)
1745
     #error "USE_XMIN_PLUG is required when Z2_USE_ENDSTOP is _XMIN_."
1775
     #error "USE_XMIN_PLUG is required when Z2_USE_ENDSTOP is _XMIN_."
1746
   #elif Z2_USE_ENDSTOP == _XMAX_ && DISABLED(USE_XMAX_PLUG)
1776
   #elif Z2_USE_ENDSTOP == _XMAX_ && DISABLED(USE_XMAX_PLUG)
1756
   #elif !HAS_Z2_MIN && !HAS_Z2_MAX
1786
   #elif !HAS_Z2_MIN && !HAS_Z2_MAX
1757
     #error "Z2_USE_ENDSTOP has been assigned to a nonexistent endstop!"
1787
     #error "Z2_USE_ENDSTOP has been assigned to a nonexistent endstop!"
1758
   #elif ENABLED(DELTA)
1788
   #elif ENABLED(DELTA)
1759
-    #error "Z_DUAL_ENDSTOPS is not compatible with DELTA."
1789
+    #error "Z_MULTI_ENDSTOPS is not compatible with DELTA."
1790
+  #endif
1791
+  #if NUM_Z_STEPPER_DRIVERS >= 3
1792
+    #if !Z3_USE_ENDSTOP
1793
+      #error "You must set Z3_USE_ENDSTOP with Z_MULTI_ENDSTOPS when NUM_Z_STEPPER_DRIVERS >= 3."
1794
+    #elif Z3_USE_ENDSTOP == _XMIN_ && DISABLED(USE_XMIN_PLUG)
1795
+      #error "USE_XMIN_PLUG is required when Z3_USE_ENDSTOP is _XMIN_."
1796
+    #elif Z3_USE_ENDSTOP == _XMAX_ && DISABLED(USE_XMAX_PLUG)
1797
+      #error "USE_XMAX_PLUG is required when Z3_USE_ENDSTOP is _XMAX_."
1798
+    #elif Z3_USE_ENDSTOP == _YMIN_ && DISABLED(USE_YMIN_PLUG)
1799
+      #error "USE_YMIN_PLUG is required when Z3_USE_ENDSTOP is _YMIN_."
1800
+    #elif Z3_USE_ENDSTOP == _YMAX_ && DISABLED(USE_YMAX_PLUG)
1801
+      #error "USE_YMAX_PLUG is required when Z3_USE_ENDSTOP is _YMAX_."
1802
+    #elif Z3_USE_ENDSTOP == _ZMIN_ && DISABLED(USE_ZMIN_PLUG)
1803
+      #error "USE_ZMIN_PLUG is required when Z3_USE_ENDSTOP is _ZMIN_."
1804
+    #elif Z3_USE_ENDSTOP == _ZMAX_ && DISABLED(USE_ZMAX_PLUG)
1805
+      #error "USE_ZMAX_PLUG is required when Z3_USE_ENDSTOP is _ZMAX_."
1806
+    #elif !HAS_Z3_MIN && !HAS_Z3_MAX
1807
+      #error "Z3_USE_ENDSTOP has been assigned to a nonexistent endstop!"
1808
+    #endif
1760
   #endif
1809
   #endif
1761
-#endif
1762
-#if ENABLED(Z_TRIPLE_ENDSTOPS)
1763
-  #if !Z2_USE_ENDSTOP
1764
-    #error "You must set Z2_USE_ENDSTOP with Z_TRIPLE_ENDSTOPS."
1765
-  #elif Z2_USE_ENDSTOP == _XMIN_ && DISABLED(USE_XMIN_PLUG)
1766
-    #error "USE_XMIN_PLUG is required when Z2_USE_ENDSTOP is _XMIN_."
1767
-  #elif Z2_USE_ENDSTOP == _XMAX_ && DISABLED(USE_XMAX_PLUG)
1768
-    #error "USE_XMAX_PLUG is required when Z2_USE_ENDSTOP is _XMAX_."
1769
-  #elif Z2_USE_ENDSTOP == _YMIN_ && DISABLED(USE_YMIN_PLUG)
1770
-    #error "USE_YMIN_PLUG is required when Z2_USE_ENDSTOP is _YMIN_."
1771
-  #elif Z2_USE_ENDSTOP == _YMAX_ && DISABLED(USE_YMAX_PLUG)
1772
-    #error "USE_YMAX_PLUG is required when Z2_USE_ENDSTOP is _YMAX_."
1773
-  #elif Z2_USE_ENDSTOP == _ZMIN_ && DISABLED(USE_ZMIN_PLUG)
1774
-    #error "USE_ZMIN_PLUG is required when Z2_USE_ENDSTOP is _ZMIN_."
1775
-  #elif Z2_USE_ENDSTOP == _ZMAX_ && DISABLED(USE_ZMAX_PLUG)
1776
-    #error "USE_ZMAX_PLUG is required when Z2_USE_ENDSTOP is _ZMAX_."
1777
-  #elif !HAS_Z2_MIN && !HAS_Z2_MAX
1778
-    #error "Z2_USE_ENDSTOP has been assigned to a nonexistent endstop!"
1779
-  #elif ENABLED(DELTA)
1780
-    #error "Z_TRIPLE_ENDSTOPS is not compatible with DELTA."
1781
-  #endif
1782
-  #if !Z3_USE_ENDSTOP
1783
-    #error "You must set Z3_USE_ENDSTOP with Z_TRIPLE_ENDSTOPS."
1784
-  #elif Z3_USE_ENDSTOP == _XMIN_ && DISABLED(USE_XMIN_PLUG)
1785
-    #error "USE_XMIN_PLUG is required when Z3_USE_ENDSTOP is _XMIN_."
1786
-  #elif Z3_USE_ENDSTOP == _XMAX_ && DISABLED(USE_XMAX_PLUG)
1787
-    #error "USE_XMAX_PLUG is required when Z3_USE_ENDSTOP is _XMAX_."
1788
-  #elif Z3_USE_ENDSTOP == _YMIN_ && DISABLED(USE_YMIN_PLUG)
1789
-    #error "USE_YMIN_PLUG is required when Z3_USE_ENDSTOP is _YMIN_."
1790
-  #elif Z3_USE_ENDSTOP == _YMAX_ && DISABLED(USE_YMAX_PLUG)
1791
-    #error "USE_YMAX_PLUG is required when Z3_USE_ENDSTOP is _YMAX_."
1792
-  #elif Z3_USE_ENDSTOP == _ZMIN_ && DISABLED(USE_ZMIN_PLUG)
1793
-    #error "USE_ZMIN_PLUG is required when Z3_USE_ENDSTOP is _ZMIN_."
1794
-  #elif Z3_USE_ENDSTOP == _ZMAX_ && DISABLED(USE_ZMAX_PLUG)
1795
-    #error "USE_ZMAX_PLUG is required when Z3_USE_ENDSTOP is _ZMAX_."
1796
-  #elif !HAS_Z3_MIN && !HAS_Z3_MAX
1797
-    #error "Z3_USE_ENDSTOP has been assigned to a nonexistent endstop!"
1798
-  #elif ENABLED(DELTA)
1799
-    #error "Z_TRIPLE_ENDSTOPS is not compatible with DELTA."
1810
+  #if NUM_Z_STEPPER_DRIVERS >= 4
1811
+    #if !Z4_USE_ENDSTOP
1812
+      #error "You must set Z4_USE_ENDSTOP with Z_MULTI_ENDSTOPS when NUM_Z_STEPPER_DRIVERS >= 4."
1813
+    #elif Z4_USE_ENDSTOP == _XMIN_ && DISABLED(USE_XMIN_PLUG)
1814
+      #error "USE_XMIN_PLUG is required when Z4_USE_ENDSTOP is _XMIN_."
1815
+    #elif Z4_USE_ENDSTOP == _XMAX_ && DISABLED(USE_XMAX_PLUG)
1816
+      #error "USE_XMAX_PLUG is required when Z4_USE_ENDSTOP is _XMAX_."
1817
+    #elif Z4_USE_ENDSTOP == _YMIN_ && DISABLED(USE_YMIN_PLUG)
1818
+      #error "USE_YMIN_PLUG is required when Z4_USE_ENDSTOP is _YMIN_."
1819
+    #elif Z4_USE_ENDSTOP == _YMAX_ && DISABLED(USE_YMAX_PLUG)
1820
+      #error "USE_YMAX_PLUG is required when Z4_USE_ENDSTOP is _YMAX_."
1821
+    #elif Z4_USE_ENDSTOP == _ZMIN_ && DISABLED(USE_ZMIN_PLUG)
1822
+      #error "USE_ZMIN_PLUG is required when Z4_USE_ENDSTOP is _ZMIN_."
1823
+    #elif Z4_USE_ENDSTOP == _ZMAX_ && DISABLED(USE_ZMAX_PLUG)
1824
+      #error "USE_ZMAX_PLUG is required when Z4_USE_ENDSTOP is _ZMAX_."
1825
+    #elif !HAS_Z4_MIN && !HAS_Z4_MAX
1826
+      #error "Z4_USE_ENDSTOP has been assigned to a nonexistent endstop!"
1827
+    #endif
1800
   #endif
1828
   #endif
1801
 #endif
1829
 #endif
1802
 
1830
 
2321
 #endif
2349
 #endif
2322
 
2350
 
2323
 #if ENABLED(Z_STEPPER_AUTO_ALIGN)
2351
 #if ENABLED(Z_STEPPER_AUTO_ALIGN)
2324
-  #if !Z_MULTI_STEPPER_DRIVERS
2325
-    #error "Z_STEPPER_AUTO_ALIGN requires Z_DUAL_STEPPER_DRIVERS or Z_TRIPLE_STEPPER_DRIVERS."
2352
+  #if NUM_Z_STEPPER_DRIVERS <= 1
2353
+    #error "Z_STEPPER_AUTO_ALIGN requires NUM_Z_STEPPER_DRIVERS greater than 1."
2326
   #elif !HAS_BED_PROBE
2354
   #elif !HAS_BED_PROBE
2327
     #error "Z_STEPPER_AUTO_ALIGN requires a Z-bed probe."
2355
     #error "Z_STEPPER_AUTO_ALIGN requires a Z-bed probe."
2328
-  #elif ENABLED(Z_STEPPER_ALIGN_KNOWN_STEPPER_POSITIONS) && DISABLED(Z_TRIPLE_STEPPER_DRIVERS)
2329
-    #error "Z_STEPPER_ALIGN_KNOWN_STEPPER_POSITIONS requires Z_TRIPLE_STEPPER_DRIVERS."
2356
+  #elif ENABLED(Z_STEPPER_ALIGN_KNOWN_STEPPER_POSITIONS) && NUM_Z_STEPPER_DRIVERS != 3
2357
+    #error "Z_STEPPER_ALIGN_KNOWN_STEPPER_POSITIONS requires NUM_Z_STEPPER_DRIVERS to be 3."
2330
   #endif
2358
   #endif
2331
 #endif
2359
 #endif
2332
 
2360
 

+ 9
- 0
Marlin/src/lcd/menu/menu_tmc.cpp View File

58
   #if AXIS_IS_TMC(Z3)
58
   #if AXIS_IS_TMC(Z3)
59
     TMC_EDIT_STORED_I_RMS(Z3, MSG_Z3);
59
     TMC_EDIT_STORED_I_RMS(Z3, MSG_Z3);
60
   #endif
60
   #endif
61
+  #if AXIS_IS_TMC(Z4)
62
+    TMC_EDIT_STORED_I_RMS(Z4, MSG_Z4);
63
+  #endif
61
   #if AXIS_IS_TMC(E0)
64
   #if AXIS_IS_TMC(E0)
62
     TMC_EDIT_STORED_I_RMS(E0, LCD_STR_E0);
65
     TMC_EDIT_STORED_I_RMS(E0, LCD_STR_E0);
63
   #endif
66
   #endif
107
     #if AXIS_HAS_STEALTHCHOP(Z3)
110
     #if AXIS_HAS_STEALTHCHOP(Z3)
108
       TMC_EDIT_STORED_HYBRID_THRS(Z3, MSG_Z3);
111
       TMC_EDIT_STORED_HYBRID_THRS(Z3, MSG_Z3);
109
     #endif
112
     #endif
113
+    #if AXIS_HAS_STEALTHCHOP(Z4)
114
+      TMC_EDIT_STORED_HYBRID_THRS(Z4, MSG_Z4);
115
+    #endif
110
     #if AXIS_HAS_STEALTHCHOP(E0)
116
     #if AXIS_HAS_STEALTHCHOP(E0)
111
       TMC_EDIT_STORED_HYBRID_THRS(E0, LCD_STR_E0);
117
       TMC_EDIT_STORED_HYBRID_THRS(E0, LCD_STR_E0);
112
     #endif
118
     #endif
183
     #if AXIS_HAS_STEALTHCHOP(Z3)
189
     #if AXIS_HAS_STEALTHCHOP(Z3)
184
       TMC_EDIT_STEP_MODE(Z3, MSG_Z3);
190
       TMC_EDIT_STEP_MODE(Z3, MSG_Z3);
185
     #endif
191
     #endif
192
+    #if AXIS_HAS_STEALTHCHOP(Z4)
193
+      TMC_EDIT_STEP_MODE(Z4, MSG_Z4);
194
+    #endif
186
     #if AXIS_HAS_STEALTHCHOP(E0)
195
     #if AXIS_HAS_STEALTHCHOP(E0)
187
       TMC_EDIT_STEP_MODE(E0, LCD_STR_E0);
196
       TMC_EDIT_STEP_MODE(E0, LCD_STR_E0);
188
     #endif
197
     #endif

+ 27
- 7
Marlin/src/libs/L64XX/L64XX_Marlin.cpp View File

39
 
39
 
40
 void echo_yes_no(const bool yes) { serialprintPGM(yes ? PSTR(" YES") : PSTR(" NO ")); }
40
 void echo_yes_no(const bool yes) { serialprintPGM(yes ? PSTR(" YES") : PSTR(" NO ")); }
41
 
41
 
42
-char L64XX_Marlin::index_to_axis[MAX_L64XX][3] = { "X ", "Y ", "Z ", "X2", "Y2", "Z2", "Z3", "E0", "E1", "E2", "E3", "E4", "E5" };
42
+char L64XX_Marlin::index_to_axis[MAX_L64XX][3] = { "X ", "Y ", "Z ", "X2", "Y2", "Z2", "Z3", "Z4", "E0", "E1", "E2", "E3", "E4", "E5" };
43
 
43
 
44
 #define DEBUG_OUT ENABLED(L6470_CHITCHAT)
44
 #define DEBUG_OUT ENABLED(L6470_CHITCHAT)
45
 #include "../../core/debug_out.h"
45
 #include "../../core/debug_out.h"
61
                                                   #endif
61
                                                   #endif
62
                                                   (INVERT_Z_DIR),                         //  5 Z2
62
                                                   (INVERT_Z_DIR),                         //  5 Z2
63
                                                   (INVERT_Z_DIR),                         //  6 Z3
63
                                                   (INVERT_Z_DIR),                         //  6 Z3
64
-                                                  (INVERT_E0_DIR),                        //  7 E0
65
-                                                  (INVERT_E1_DIR),                        //  8 E1
66
-                                                  (INVERT_E2_DIR),                        //  9 E2
67
-                                                  (INVERT_E3_DIR),                        // 10 E3
68
-                                                  (INVERT_E4_DIR),                        // 11 E4
69
-                                                  (INVERT_E5_DIR),                        // 12 E5
64
+                                                  (INVERT_Z_DIR),                         //  7 Z4
65
+
66
+                                                  (INVERT_E0_DIR),                        //  8 E0
67
+                                                  (INVERT_E1_DIR),                        //  9 E1
68
+                                                  (INVERT_E2_DIR),                        // 10 E2
69
+                                                  (INVERT_E3_DIR),                        // 11 E3
70
+                                                  (INVERT_E4_DIR),                        // 12 E4
71
+                                                  (INVERT_E5_DIR),                        // 13 E5
70
                                                 };
72
                                                 };
71
 
73
 
72
 volatile uint8_t L64XX_Marlin::spi_abort = false;
74
 volatile uint8_t L64XX_Marlin::spi_abort = false;
101
   #if AXIS_IS_L64XX(Z3)
103
   #if AXIS_IS_L64XX(Z3)
102
     _L6470_INIT_SPI(Z3);
104
     _L6470_INIT_SPI(Z3);
103
   #endif
105
   #endif
106
+  #if AXIS_IS_L64XX(Z4)
107
+    _L6470_INIT_SPI(Z4);
108
+  #endif
104
   #if AXIS_IS_L64XX(E0)
109
   #if AXIS_IS_L64XX(E0)
105
     _L6470_INIT_SPI(E0);
110
     _L6470_INIT_SPI(E0);
106
   #endif
111
   #endif
211
     #if AXIS_IS_L64XX(Z3)
216
     #if AXIS_IS_L64XX(Z3)
212
       case Z3: return STATUS_L6470(Z3);
217
       case Z3: return STATUS_L6470(Z3);
213
     #endif
218
     #endif
219
+    #if AXIS_IS_L64XX(Z4)
220
+      case Z4: return STATUS_L6470(Z4);
221
+    #endif
214
     #if AXIS_IS_L64XX(E0)
222
     #if AXIS_IS_L64XX(E0)
215
       case E0: return STATUS_L6470(E0);
223
       case E0: return STATUS_L6470(E0);
216
     #endif
224
     #endif
261
     #if AXIS_IS_L64XX(Z3)
269
     #if AXIS_IS_L64XX(Z3)
262
       case Z3: return GET_L6470_PARAM(Z3);
270
       case Z3: return GET_L6470_PARAM(Z3);
263
     #endif
271
     #endif
272
+    #if AXIS_IS_L64XX(Z4)
273
+      case Z4: return GET_L6470_PARAM(Z4);
274
+    #endif
264
     #if AXIS_IS_L64XX(E0)
275
     #if AXIS_IS_L64XX(E0)
265
       case E0: return GET_L6470_PARAM(E0);
276
       case E0: return GET_L6470_PARAM(E0);
266
     #endif
277
     #endif
311
     #if AXIS_IS_L64XX(Z3)
322
     #if AXIS_IS_L64XX(Z3)
312
       case Z3: SET_L6470_PARAM(Z3); break;
323
       case Z3: SET_L6470_PARAM(Z3); break;
313
     #endif
324
     #endif
325
+    #if AXIS_IS_L64XX(Z4)
326
+      case Z4: SET_L6470_PARAM(Z4); break;
327
+    #endif
314
     #if AXIS_IS_L64XX(E0)
328
     #if AXIS_IS_L64XX(E0)
315
       case E0: SET_L6470_PARAM(E0); break;
329
       case E0: SET_L6470_PARAM(E0); break;
316
     #endif
330
     #endif
684
     #if AXIS_IS_L64XX(Z3)
698
     #if AXIS_IS_L64XX(Z3)
685
       {  6, 0, 0, 0, 0, 0, 0 },
699
       {  6, 0, 0, 0, 0, 0, 0 },
686
     #endif
700
     #endif
701
+    #if AXIS_IS_L64XX(Z4)
702
+      {  6, 0, 0, 0, 0, 0, 0 },
703
+    #endif
687
     #if AXIS_IS_L64XX(E0)
704
     #if AXIS_IS_L64XX(E0)
688
       {  7, 0, 0, 0, 0, 0, 0 },
705
       {  7, 0, 0, 0, 0, 0, 0 },
689
     #endif
706
     #endif
858
         #if AXIS_IS_L64XX(Z3)
875
         #if AXIS_IS_L64XX(Z3)
859
           monitor_update(Z3);
876
           monitor_update(Z3);
860
         #endif
877
         #endif
878
+        #if AXIS_IS_L64XX(Z4)
879
+          monitor_update(Z4);
880
+        #endif
861
         #if AXIS_IS_L64XX(E0)
881
         #if AXIS_IS_L64XX(E0)
862
           monitor_update(E0);
882
           monitor_update(E0);
863
         #endif
883
         #endif

+ 1
- 1
Marlin/src/libs/L64XX/L64XX_Marlin.h View File

35
 #define dSPIN_STEP_CLOCK_REV dSPIN_STEP_CLOCK+1
35
 #define dSPIN_STEP_CLOCK_REV dSPIN_STEP_CLOCK+1
36
 #define HAS_L64XX_EXTRUDER (AXIS_IS_L64XX(E0) || AXIS_IS_L64XX(E1) || AXIS_IS_L64XX(E2) || AXIS_IS_L64XX(E3) || AXIS_IS_L64XX(E4) || AXIS_IS_L64XX(E5))
36
 #define HAS_L64XX_EXTRUDER (AXIS_IS_L64XX(E0) || AXIS_IS_L64XX(E1) || AXIS_IS_L64XX(E2) || AXIS_IS_L64XX(E3) || AXIS_IS_L64XX(E4) || AXIS_IS_L64XX(E5))
37
 
37
 
38
-enum L64XX_axis_t : uint8_t { X, Y, Z, X2, Y2, Z2, Z3, E0, E1, E2, E3, E4, E5, MAX_L64XX };
38
+enum L64XX_axis_t : uint8_t { X, Y, Z, X2, Y2, Z2, Z3, Z4, E0, E1, E2, E3, E4, E5, MAX_L64XX };
39
 
39
 
40
 class L64XX_Marlin : public L64XXHelper {
40
 class L64XX_Marlin : public L64XXHelper {
41
 public:
41
 public:

+ 115
- 69
Marlin/src/module/configuration_store.cpp View File

37
  */
37
  */
38
 
38
 
39
 // Change EEPROM version if the structure changes
39
 // Change EEPROM version if the structure changes
40
-#define EEPROM_VERSION "V74"
40
+#define EEPROM_VERSION "V75"
41
 #define EEPROM_OFFSET 100
41
 #define EEPROM_OFFSET 100
42
 
42
 
43
 // Check the integrity of data offsets.
43
 // Check the integrity of data offsets.
120
 
120
 
121
 #pragma pack(push, 1) // No padding between variables
121
 #pragma pack(push, 1) // No padding between variables
122
 
122
 
123
-typedef struct { uint16_t X, Y, Z, X2, Y2, Z2, Z3, E0, E1, E2, E3, E4, E5; } tmc_stepper_current_t;
124
-typedef struct { uint32_t X, Y, Z, X2, Y2, Z2, Z3, E0, E1, E2, E3, E4, E5; } tmc_hybrid_threshold_t;
123
+typedef struct { uint16_t X, Y, Z, X2, Y2, Z2, Z3, Z4, E0, E1, E2, E3, E4, E5; } tmc_stepper_current_t;
124
+typedef struct { uint32_t X, Y, Z, X2, Y2, Z2, Z3, Z4, E0, E1, E2, E3, E4, E5; } tmc_hybrid_threshold_t;
125
 typedef struct {  int16_t X, Y, Z, X2;                                     } tmc_sgt_t;
125
 typedef struct {  int16_t X, Y, Z, X2;                                     } tmc_sgt_t;
126
-typedef struct {     bool X, Y, Z, X2, Y2, Z2, Z3, E0, E1, E2, E3, E4, E5; } tmc_stealth_enabled_t;
126
+typedef struct {     bool X, Y, Z, X2, Y2, Z2, Z3, Z4, E0, E1, E2, E3, E4, E5; } tmc_stealth_enabled_t;
127
 
127
 
128
 // Limit an index to an array size
128
 // Limit an index to an array size
129
 #define ALIM(I,ARR) _MIN(I, COUNT(ARR) - 1)
129
 #define ALIM(I,ARR) _MIN(I, COUNT(ARR) - 1)
243
           delta_diagonal_rod,                           // M665 L
243
           delta_diagonal_rod,                           // M665 L
244
           delta_segments_per_second;                    // M665 S
244
           delta_segments_per_second;                    // M665 S
245
     abc_float_t delta_tower_angle_trim;                 // M665 XYZ
245
     abc_float_t delta_tower_angle_trim;                 // M665 XYZ
246
-  #elif EITHER(X_DUAL_ENDSTOPS, Y_DUAL_ENDSTOPS) || Z_MULTI_ENDSTOPS
246
+  #elif HAS_EXTRA_ENDSTOPS
247
     float x2_endstop_adj,                               // M666 X
247
     float x2_endstop_adj,                               // M666 X
248
           y2_endstop_adj,                               // M666 Y
248
           y2_endstop_adj,                               // M666 Y
249
-          z2_endstop_adj,                               // M666 Z (S2)
250
-          z3_endstop_adj;                               // M666 Z (S3)
249
+          z2_endstop_adj,                               // M666 (S2) Z
250
+          z3_endstop_adj,                               // M666 (S3) Z
251
+          z4_endstop_adj;                               // M666 (S4) Z
251
   #endif
252
   #endif
252
 
253
 
253
   //
254
   //
300
   //
301
   //
301
   // HAS_TRINAMIC
302
   // HAS_TRINAMIC
302
   //
303
   //
303
-  tmc_stepper_current_t tmc_stepper_current;            // M906 X Y Z X2 Y2 Z2 Z3 E0 E1 E2 E3 E4 E5
304
-  tmc_hybrid_threshold_t tmc_hybrid_threshold;          // M913 X Y Z X2 Y2 Z2 Z3 E0 E1 E2 E3 E4 E5
304
+  tmc_stepper_current_t tmc_stepper_current;            // M906 X Y Z X2 Y2 Z2 Z3 Z4 E0 E1 E2 E3 E4 E5
305
+  tmc_hybrid_threshold_t tmc_hybrid_threshold;          // M913 X Y Z X2 Y2 Z2 Z3 Z4 E0 E1 E2 E3 E4 E5
305
   tmc_sgt_t tmc_sgt;                                    // M914 X Y Z X2
306
   tmc_sgt_t tmc_sgt;                                    // M914 X Y Z X2
306
-  tmc_stealth_enabled_t tmc_stealth_enabled;            // M569 X Y Z X2 Y2 Z2 Z3 E0 E1 E2 E3 E4 E5
307
+  tmc_stealth_enabled_t tmc_stealth_enabled;            // M569 X Y Z X2 Y2 Z2 Z3 Z4 E0 E1 E2 E3 E4 E5
307
 
308
 
308
   //
309
   //
309
   // LIN_ADVANCE
310
   // LIN_ADVANCE
756
         EEPROM_WRITE(delta_segments_per_second); // 1 float
757
         EEPROM_WRITE(delta_segments_per_second); // 1 float
757
         EEPROM_WRITE(delta_tower_angle_trim);    // 3 floats
758
         EEPROM_WRITE(delta_tower_angle_trim);    // 3 floats
758
 
759
 
759
-      #elif EITHER(X_DUAL_ENDSTOPS, Y_DUAL_ENDSTOPS) || Z_MULTI_ENDSTOPS
760
+      #elif HAS_EXTRA_ENDSTOPS
760
 
761
 
761
         _FIELD_TEST(x2_endstop_adj);
762
         _FIELD_TEST(x2_endstop_adj);
762
 
763
 
774
           EEPROM_WRITE(dummy);
775
           EEPROM_WRITE(dummy);
775
         #endif
776
         #endif
776
 
777
 
777
-        #if Z_MULTI_ENDSTOPS
778
+        #if ENABLED(Z_MULTI_ENDSTOPS)
778
           EEPROM_WRITE(endstops.z2_endstop_adj);   // 1 float
779
           EEPROM_WRITE(endstops.z2_endstop_adj);   // 1 float
779
         #else
780
         #else
780
           EEPROM_WRITE(dummy);
781
           EEPROM_WRITE(dummy);
781
         #endif
782
         #endif
782
 
783
 
783
-        #if ENABLED(Z_TRIPLE_ENDSTOPS)
784
+        #if ENABLED(Z_MULTI_ENDSTOPS) && NUM_Z_STEPPER_DRIVERS >= 3
784
           EEPROM_WRITE(endstops.z3_endstop_adj);   // 1 float
785
           EEPROM_WRITE(endstops.z3_endstop_adj);   // 1 float
785
         #else
786
         #else
786
           EEPROM_WRITE(dummy);
787
           EEPROM_WRITE(dummy);
787
         #endif
788
         #endif
788
 
789
 
790
+        #if ENABLED(Z_MULTI_ENDSTOPS) && NUM_Z_STEPPER_DRIVERS >= 4
791
+          EEPROM_WRITE(endstops.z4_endstop_adj);   // 1 float
792
+        #else
793
+          EEPROM_WRITE(dummy);
794
+        #endif
795
+
789
       #endif
796
       #endif
790
     }
797
     }
791
 
798
 
974
         #if AXIS_IS_TMC(Z3)
981
         #if AXIS_IS_TMC(Z3)
975
           tmc_stepper_current.Z3 = stepperZ3.getMilliamps();
982
           tmc_stepper_current.Z3 = stepperZ3.getMilliamps();
976
         #endif
983
         #endif
984
+        #if AXIS_IS_TMC(Z4)
985
+          tmc_stepper_current.Z4 = stepperZ4.getMilliamps();
986
+        #endif
977
         #if MAX_EXTRUDERS
987
         #if MAX_EXTRUDERS
978
           #if AXIS_IS_TMC(E0)
988
           #if AXIS_IS_TMC(E0)
979
             tmc_stepper_current.E0 = stepperE0.getMilliamps();
989
             tmc_stepper_current.E0 = stepperE0.getMilliamps();
1037
         #if AXIS_HAS_STEALTHCHOP(Z3)
1047
         #if AXIS_HAS_STEALTHCHOP(Z3)
1038
           tmc_hybrid_threshold.Z3 = stepperZ3.get_pwm_thrs();
1048
           tmc_hybrid_threshold.Z3 = stepperZ3.get_pwm_thrs();
1039
         #endif
1049
         #endif
1050
+        #if AXIS_HAS_STEALTHCHOP(Z4)
1051
+          tmc_hybrid_threshold.Z4 = stepperZ4.get_pwm_thrs();
1052
+        #endif
1040
         #if MAX_EXTRUDERS
1053
         #if MAX_EXTRUDERS
1041
           #if AXIS_HAS_STEALTHCHOP(E0)
1054
           #if AXIS_HAS_STEALTHCHOP(E0)
1042
             tmc_hybrid_threshold.E0 = stepperE0.get_pwm_thrs();
1055
             tmc_hybrid_threshold.E0 = stepperE0.get_pwm_thrs();
1070
       #else
1083
       #else
1071
         const tmc_hybrid_threshold_t tmc_hybrid_threshold = {
1084
         const tmc_hybrid_threshold_t tmc_hybrid_threshold = {
1072
           .X  = 100, .Y  = 100, .Z  =   3,
1085
           .X  = 100, .Y  = 100, .Z  =   3,
1073
-          .X2 = 100, .Y2 = 100, .Z2 =   3, .Z3 =   3,
1086
+          .X2 = 100, .Y2 = 100, .Z2 =   3, .Z3 =   3, .Z4 = 3,
1074
           .E0 =  30, .E1 =  30, .E2 =  30,
1087
           .E0 =  30, .E1 =  30, .E2 =  30,
1075
           .E3 =  30, .E4 =  30, .E5 =  30
1088
           .E3 =  30, .E4 =  30, .E5 =  30
1076
         };
1089
         };
1130
         #if AXIS_HAS_STEALTHCHOP(Z3)
1143
         #if AXIS_HAS_STEALTHCHOP(Z3)
1131
           tmc_stealth_enabled.Z3 = stepperZ3.get_stealthChop_status();
1144
           tmc_stealth_enabled.Z3 = stepperZ3.get_stealthChop_status();
1132
         #endif
1145
         #endif
1146
+        #if AXIS_HAS_STEALTHCHOP(Z4)
1147
+          tmc_stealth_enabled.Z4 = stepperZ4.get_stealthChop_status();
1148
+        #endif
1133
         #if MAX_EXTRUDERS
1149
         #if MAX_EXTRUDERS
1134
           #if AXIS_HAS_STEALTHCHOP(E0)
1150
           #if AXIS_HAS_STEALTHCHOP(E0)
1135
             tmc_stealth_enabled.E0 = stepperE0.get_stealthChop_status();
1151
             tmc_stealth_enabled.E0 = stepperE0.get_stealthChop_status();
1585
           EEPROM_READ(delta_segments_per_second); // 1 float
1601
           EEPROM_READ(delta_segments_per_second); // 1 float
1586
           EEPROM_READ(delta_tower_angle_trim);    // 3 floats
1602
           EEPROM_READ(delta_tower_angle_trim);    // 3 floats
1587
 
1603
 
1588
-        #elif EITHER(X_DUAL_ENDSTOPS, Y_DUAL_ENDSTOPS) || Z_MULTI_ENDSTOPS
1604
+        #elif HAS_EXTRA_ENDSTOPS
1589
 
1605
 
1590
           _FIELD_TEST(x2_endstop_adj);
1606
           _FIELD_TEST(x2_endstop_adj);
1591
 
1607
 
1599
           #else
1615
           #else
1600
             EEPROM_READ(dummy);
1616
             EEPROM_READ(dummy);
1601
           #endif
1617
           #endif
1602
-          #if Z_MULTI_ENDSTOPS
1618
+          #if ENABLED(Z_MULTI_ENDSTOPS)
1603
             EEPROM_READ(endstops.z2_endstop_adj); // 1 float
1619
             EEPROM_READ(endstops.z2_endstop_adj); // 1 float
1604
           #else
1620
           #else
1605
             EEPROM_READ(dummy);
1621
             EEPROM_READ(dummy);
1606
           #endif
1622
           #endif
1607
-          #if ENABLED(Z_TRIPLE_ENDSTOPS)
1623
+          #if ENABLED(Z_MULTI_ENDSTOPS) && NUM_Z_STEPPER_DRIVERS >= 3
1608
             EEPROM_READ(endstops.z3_endstop_adj); // 1 float
1624
             EEPROM_READ(endstops.z3_endstop_adj); // 1 float
1609
           #else
1625
           #else
1610
             EEPROM_READ(dummy);
1626
             EEPROM_READ(dummy);
1611
           #endif
1627
           #endif
1628
+          #if ENABLED(Z_MULTI_ENDSTOPS) && NUM_Z_STEPPER_DRIVERS >= 4
1629
+            EEPROM_READ(endstops.z4_endstop_adj); // 1 float
1630
+          #else
1631
+            EEPROM_READ(dummy);
1632
+          #endif
1612
 
1633
 
1613
         #endif
1634
         #endif
1614
       }
1635
       }
1800
             #if AXIS_IS_TMC(Z3)
1821
             #if AXIS_IS_TMC(Z3)
1801
               SET_CURR(Z3);
1822
               SET_CURR(Z3);
1802
             #endif
1823
             #endif
1824
+            #if AXIS_IS_TMC(Z4)
1825
+              SET_CURR(Z4);
1826
+            #endif
1803
             #if AXIS_IS_TMC(E0)
1827
             #if AXIS_IS_TMC(E0)
1804
               SET_CURR(E0);
1828
               SET_CURR(E0);
1805
             #endif
1829
             #endif
1851
             #if AXIS_HAS_STEALTHCHOP(Z3)
1875
             #if AXIS_HAS_STEALTHCHOP(Z3)
1852
               stepperZ3.set_pwm_thrs(tmc_hybrid_threshold.Z3);
1876
               stepperZ3.set_pwm_thrs(tmc_hybrid_threshold.Z3);
1853
             #endif
1877
             #endif
1878
+            #if AXIS_HAS_STEALTHCHOP(Z4)
1879
+              stepperZ4.set_pwm_thrs(tmc_hybrid_threshold.Z4);
1880
+            #endif
1854
             #if AXIS_HAS_STEALTHCHOP(E0)
1881
             #if AXIS_HAS_STEALTHCHOP(E0)
1855
               stepperE0.set_pwm_thrs(tmc_hybrid_threshold.E0);
1882
               stepperE0.set_pwm_thrs(tmc_hybrid_threshold.E0);
1856
             #endif
1883
             #endif
1877
       // TMC StallGuard threshold.
1904
       // TMC StallGuard threshold.
1878
       // X and X2 use the same value
1905
       // X and X2 use the same value
1879
       // Y and Y2 use the same value
1906
       // Y and Y2 use the same value
1880
-      // Z, Z2 and Z3 use the same value
1907
+      // Z, Z2, Z3 and Z4 use the same value
1881
       //
1908
       //
1882
       {
1909
       {
1883
         tmc_sgt_t tmc_sgt;
1910
         tmc_sgt_t tmc_sgt;
1914
               #if AXIS_HAS_STALLGUARD(Z3)
1941
               #if AXIS_HAS_STALLGUARD(Z3)
1915
                 stepperZ3.homing_threshold(tmc_sgt.Z);
1942
                 stepperZ3.homing_threshold(tmc_sgt.Z);
1916
               #endif
1943
               #endif
1944
+              #if AXIS_HAS_STALLGUARD(Z4)
1945
+                stepperZ4.homing_threshold(tmc_sgt.Z);
1946
+              #endif
1917
             #endif
1947
             #endif
1918
           }
1948
           }
1919
         #endif
1949
         #endif
1951
             #if AXIS_HAS_STEALTHCHOP(Z3)
1981
             #if AXIS_HAS_STEALTHCHOP(Z3)
1952
               SET_STEPPING_MODE(Z3);
1982
               SET_STEPPING_MODE(Z3);
1953
             #endif
1983
             #endif
1984
+            #if AXIS_HAS_STEALTHCHOP(Z4)
1985
+              SET_STEPPING_MODE(Z4);
1986
+            #endif
1954
             #if AXIS_HAS_STEALTHCHOP(E0)
1987
             #if AXIS_HAS_STEALTHCHOP(E0)
1955
               SET_STEPPING_MODE(E0);
1988
               SET_STEPPING_MODE(E0);
1956
             #endif
1989
             #endif
2433
     delta_segments_per_second = DELTA_SEGMENTS_PER_SECOND;
2466
     delta_segments_per_second = DELTA_SEGMENTS_PER_SECOND;
2434
     delta_tower_angle_trim = dta;
2467
     delta_tower_angle_trim = dta;
2435
 
2468
 
2436
-  #elif EITHER(X_DUAL_ENDSTOPS, Y_DUAL_ENDSTOPS) || Z_MULTI_ENDSTOPS
2469
+  #endif
2437
 
2470
 
2438
-    #if ENABLED(X_DUAL_ENDSTOPS)
2439
-      endstops.x2_endstop_adj = (
2440
-        #ifdef X_DUAL_ENDSTOPS_ADJUSTMENT
2441
-          X_DUAL_ENDSTOPS_ADJUSTMENT
2442
-        #else
2443
-          0
2444
-        #endif
2445
-      );
2446
-    #endif
2447
-    #if ENABLED(Y_DUAL_ENDSTOPS)
2448
-      endstops.y2_endstop_adj = (
2449
-        #ifdef Y_DUAL_ENDSTOPS_ADJUSTMENT
2450
-          Y_DUAL_ENDSTOPS_ADJUSTMENT
2451
-        #else
2452
-          0
2453
-        #endif
2454
-      );
2471
+  #if ENABLED(X_DUAL_ENDSTOPS)
2472
+    #ifndef X2_ENDSTOP_ADJUSTMENT
2473
+      #define X2_ENDSTOP_ADJUSTMENT 0
2455
     #endif
2474
     #endif
2456
-    #if ENABLED(Z_DUAL_ENDSTOPS)
2457
-      endstops.z2_endstop_adj = (
2458
-        #ifdef Z_DUAL_ENDSTOPS_ADJUSTMENT
2459
-          Z_DUAL_ENDSTOPS_ADJUSTMENT
2460
-        #else
2461
-          0
2462
-        #endif
2463
-      );
2464
-    #elif ENABLED(Z_TRIPLE_ENDSTOPS)
2465
-      endstops.z2_endstop_adj = (
2466
-        #ifdef Z_TRIPLE_ENDSTOPS_ADJUSTMENT2
2467
-          Z_TRIPLE_ENDSTOPS_ADJUSTMENT2
2468
-        #else
2469
-          0
2470
-        #endif
2471
-      );
2472
-      endstops.z3_endstop_adj = (
2473
-        #ifdef Z_TRIPLE_ENDSTOPS_ADJUSTMENT3
2474
-          Z_TRIPLE_ENDSTOPS_ADJUSTMENT3
2475
-        #else
2476
-          0
2477
-        #endif
2478
-      );
2475
+    endstops.x2_endstop_adj = X2_ENDSTOP_ADJUSTMENT;
2476
+  #endif
2477
+
2478
+  #if ENABLED(Y_DUAL_ENDSTOPS)
2479
+    #ifndef Y2_ENDSTOP_ADJUSTMENT
2480
+      #define Y2_ENDSTOP_ADJUSTMENT 0
2479
     #endif
2481
     #endif
2482
+    endstops.y2_endstop_adj = Y2_ENDSTOP_ADJUSTMENT;
2483
+  #endif
2480
 
2484
 
2485
+  #if ENABLED(Z_MULTI_ENDSTOPS)
2486
+    #ifndef Z2_ENDSTOP_ADJUSTMENT
2487
+      #define Z2_ENDSTOP_ADJUSTMENT 0
2488
+    #endif
2489
+    endstops.z2_endstop_adj = Z2_ENDSTOP_ADJUSTMENT;
2490
+    #if NUM_Z_STEPPER_DRIVERS >= 3
2491
+      #ifndef Z3_ENDSTOP_ADJUSTMENT
2492
+        #define Z3_ENDSTOP_ADJUSTMENT 0
2493
+      #endif
2494
+      endstops.z3_endstop_adj = Z3_ENDSTOP_ADJUSTMENT;
2495
+    #endif
2496
+    #if NUM_Z_STEPPER_DRIVERS >= 4
2497
+      #ifndef Z4_ENDSTOP_ADJUSTMENT
2498
+        #define Z4_ENDSTOP_ADJUSTMENT 0
2499
+      #endif
2500
+      endstops.z4_endstop_adj = Z4_ENDSTOP_ADJUSTMENT;
2501
+    #endif
2481
   #endif
2502
   #endif
2482
 
2503
 
2483
   //
2504
   //
3016
         , SP_Z_STR, LINEAR_UNIT(delta_tower_angle_trim.c)
3037
         , SP_Z_STR, LINEAR_UNIT(delta_tower_angle_trim.c)
3017
       );
3038
       );
3018
 
3039
 
3019
-    #elif EITHER(X_DUAL_ENDSTOPS, Y_DUAL_ENDSTOPS) || Z_MULTI_ENDSTOPS
3040
+    #elif HAS_EXTRA_ENDSTOPS
3020
 
3041
 
3021
       CONFIG_ECHO_HEADING("Endstop adjustment:");
3042
       CONFIG_ECHO_HEADING("Endstop adjustment:");
3022
       CONFIG_ECHO_START();
3043
       CONFIG_ECHO_START();
3023
       SERIAL_ECHOPGM("  M666");
3044
       SERIAL_ECHOPGM("  M666");
3024
       #if ENABLED(X_DUAL_ENDSTOPS)
3045
       #if ENABLED(X_DUAL_ENDSTOPS)
3025
-        SERIAL_ECHOPAIR_P(SP_X_STR, LINEAR_UNIT(endstops.x2_endstop_adj));
3046
+        SERIAL_ECHOLNPAIR_P(SP_X_STR, LINEAR_UNIT(endstops.x2_endstop_adj));
3026
       #endif
3047
       #endif
3027
       #if ENABLED(Y_DUAL_ENDSTOPS)
3048
       #if ENABLED(Y_DUAL_ENDSTOPS)
3028
-        SERIAL_ECHOPAIR_P(SP_Y_STR, LINEAR_UNIT(endstops.y2_endstop_adj));
3049
+        SERIAL_ECHOLNPAIR_P(SP_Y_STR, LINEAR_UNIT(endstops.y2_endstop_adj));
3029
       #endif
3050
       #endif
3030
-      #if ENABLED(Z_TRIPLE_ENDSTOPS)
3031
-        SERIAL_ECHOLNPAIR("S1 Z", LINEAR_UNIT(endstops.z2_endstop_adj));
3032
-        CONFIG_ECHO_START();
3033
-        SERIAL_ECHOPAIR("  M666 S2 Z", LINEAR_UNIT(endstops.z3_endstop_adj));
3034
-      #elif ENABLED(Z_DUAL_ENDSTOPS)
3035
-        SERIAL_ECHOPAIR_P(SP_Z_STR, LINEAR_UNIT(endstops.z2_endstop_adj));
3051
+      #if ENABLED(Z_MULTI_ENDSTOPS)
3052
+        #if NUM_Z_STEPPER_DRIVERS >= 3
3053
+          SERIAL_ECHOPAIR(" S2 Z", LINEAR_UNIT(endstops.z3_endstop_adj));
3054
+          CONFIG_ECHO_START();
3055
+          SERIAL_ECHOPAIR("  M666 S3 Z", LINEAR_UNIT(endstops.z3_endstop_adj));
3056
+          #if NUM_Z_STEPPER_DRIVERS >= 4
3057
+            CONFIG_ECHO_START();
3058
+            SERIAL_ECHOPAIR("  M666 S4 Z", LINEAR_UNIT(endstops.z4_endstop_adj));
3059
+          #endif
3060
+        #else
3061
+          SERIAL_ECHOLNPAIR_P(SP_Z_STR, LINEAR_UNIT(endstops.z2_endstop_adj));
3062
+        #endif
3036
       #endif
3063
       #endif
3037
-      SERIAL_EOL();
3038
 
3064
 
3039
     #endif // [XYZ]_DUAL_ENDSTOPS
3065
     #endif // [XYZ]_DUAL_ENDSTOPS
3040
 
3066
 
3218
         SERIAL_ECHOLNPAIR(" I2 Z", stepperZ3.getMilliamps());
3244
         SERIAL_ECHOLNPAIR(" I2 Z", stepperZ3.getMilliamps());
3219
       #endif
3245
       #endif
3220
 
3246
 
3247
+      #if AXIS_IS_TMC(Z4)
3248
+        say_M906(forReplay);
3249
+        SERIAL_ECHOLNPAIR(" I3 Z", stepperZ4.getMilliamps());
3250
+      #endif
3251
+
3221
       #if AXIS_IS_TMC(E0)
3252
       #if AXIS_IS_TMC(E0)
3222
         say_M906(forReplay);
3253
         say_M906(forReplay);
3223
         SERIAL_ECHOLNPAIR(" T0 E", stepperE0.getMilliamps());
3254
         SERIAL_ECHOLNPAIR(" T0 E", stepperE0.getMilliamps());
3287
           SERIAL_ECHOLNPAIR(" I2 Z", stepperZ3.get_pwm_thrs());
3318
           SERIAL_ECHOLNPAIR(" I2 Z", stepperZ3.get_pwm_thrs());
3288
         #endif
3319
         #endif
3289
 
3320
 
3321
+        #if AXIS_HAS_STEALTHCHOP(Z4)
3322
+          say_M913(forReplay);
3323
+          SERIAL_ECHOLNPAIR(" I3 Z", stepperZ4.get_pwm_thrs());
3324
+        #endif
3325
+
3290
         #if AXIS_HAS_STEALTHCHOP(E0)
3326
         #if AXIS_HAS_STEALTHCHOP(E0)
3291
           say_M913(forReplay);
3327
           say_M913(forReplay);
3292
           SERIAL_ECHOLNPAIR(" T0 E", stepperE0.get_pwm_thrs());
3328
           SERIAL_ECHOLNPAIR(" T0 E", stepperE0.get_pwm_thrs());
3356
           SERIAL_ECHOLNPAIR(" I2 Z", stepperZ3.homing_threshold());
3392
           SERIAL_ECHOLNPAIR(" I2 Z", stepperZ3.homing_threshold());
3357
         #endif
3393
         #endif
3358
 
3394
 
3395
+        #if Z4_SENSORLESS
3396
+          CONFIG_ECHO_START();
3397
+          say_M914();
3398
+          SERIAL_ECHOLNPAIR(" I3 Z", stepperZ4.homing_threshold());
3399
+        #endif
3400
+
3359
       #endif // USE_SENSORLESS
3401
       #endif // USE_SENSORLESS
3360
 
3402
 
3361
       /**
3403
       /**
3415
           if (stepperZ3.get_stealthChop_status()) { say_M569(forReplay, PSTR("I2 Z"), true); }
3457
           if (stepperZ3.get_stealthChop_status()) { say_M569(forReplay, PSTR("I2 Z"), true); }
3416
         #endif
3458
         #endif
3417
 
3459
 
3460
+        #if AXIS_HAS_STEALTHCHOP(Z4)
3461
+          if (stepperZ4.get_stealthChop_status()) { say_M569(forReplay, PSTR("I3 Z"), true); }
3462
+        #endif
3463
+
3418
         #if AXIS_HAS_STEALTHCHOP(E0)
3464
         #if AXIS_HAS_STEALTHCHOP(E0)
3419
           if (stepperE0.get_stealthChop_status()) { say_M569(forReplay, PSTR("T0 E"), true); }
3465
           if (stepperE0.get_stealthChop_status()) { say_M569(forReplay, PSTR("T0 E"), true); }
3420
         #endif
3466
         #endif

+ 89
- 16
Marlin/src/module/endstops.cpp View File

73
 #if ENABLED(Y_DUAL_ENDSTOPS)
73
 #if ENABLED(Y_DUAL_ENDSTOPS)
74
   float Endstops::y2_endstop_adj;
74
   float Endstops::y2_endstop_adj;
75
 #endif
75
 #endif
76
-#if Z_MULTI_ENDSTOPS
76
+#if ENABLED(Z_MULTI_ENDSTOPS)
77
   float Endstops::z2_endstop_adj;
77
   float Endstops::z2_endstop_adj;
78
-#endif
79
-#if ENABLED(Z_TRIPLE_ENDSTOPS)
80
-  float Endstops::z3_endstop_adj;
78
+  #if NUM_Z_STEPPER_DRIVERS >= 3
79
+    float Endstops::z3_endstop_adj;
80
+    #if NUM_Z_STEPPER_DRIVERS >= 4
81
+      float Endstops::z4_endstop_adj;
82
+    #endif
83
+  #endif
81
 #endif
84
 #endif
82
 
85
 
83
 #if ENABLED(SPI_ENDSTOPS)
86
 #if ENABLED(SPI_ENDSTOPS)
163
     #endif
166
     #endif
164
   #endif
167
   #endif
165
 
168
 
169
+  #if HAS_Z4_MIN
170
+    #if ENABLED(ENDSTOPPULLUP_ZMIN)
171
+      SET_INPUT_PULLUP(Z4_MIN_PIN);
172
+    #elif ENABLED(ENDSTOPPULLDOWN_ZMIN)
173
+      SET_INPUT_PULLDOWN(Z4_MIN_PIN);
174
+    #else
175
+      SET_INPUT(Z4_MIN_PIN);
176
+    #endif
177
+  #endif
178
+
166
   #if HAS_X_MAX
179
   #if HAS_X_MAX
167
     #if ENABLED(ENDSTOPPULLUP_XMAX)
180
     #if ENABLED(ENDSTOPPULLUP_XMAX)
168
       SET_INPUT_PULLUP(X_MAX_PIN);
181
       SET_INPUT_PULLUP(X_MAX_PIN);
233
     #endif
246
     #endif
234
   #endif
247
   #endif
235
 
248
 
249
+  #if HAS_Z4_MAX
250
+    #if ENABLED(ENDSTOPPULLUP_ZMAX)
251
+      SET_INPUT_PULLUP(Z4_MAX_PIN);
252
+    #elif ENABLED(ENDSTOPPULLDOWN_ZMAX)
253
+      SET_INPUT_PULLDOWN(Z4_MAX_PIN);
254
+    #else
255
+      SET_INPUT(Z4_MAX_PIN);
256
+    #endif
257
+  #endif
258
+
236
   #if HAS_CALIBRATION_PIN
259
   #if HAS_CALIBRATION_PIN
237
     #if ENABLED(CALIBRATION_PIN_PULLUP)
260
     #if ENABLED(CALIBRATION_PIN_PULLUP)
238
       SET_INPUT_PULLUP(CALIBRATION_PIN);
261
       SET_INPUT_PULLUP(CALIBRATION_PIN);
435
   #if HAS_Z3_MIN
458
   #if HAS_Z3_MIN
436
     ES_REPORT(Z3_MIN);
459
     ES_REPORT(Z3_MIN);
437
   #endif
460
   #endif
461
+  #if HAS_Z4_MIN
462
+    ES_REPORT(Z4_MIN);
463
+  #endif
438
   #if HAS_Z_MAX
464
   #if HAS_Z_MAX
439
     ES_REPORT(Z_MAX);
465
     ES_REPORT(Z_MAX);
440
   #endif
466
   #endif
444
   #if HAS_Z3_MAX
470
   #if HAS_Z3_MAX
445
     ES_REPORT(Z3_MAX);
471
     ES_REPORT(Z3_MAX);
446
   #endif
472
   #endif
473
+  #if HAS_Z4_MAX
474
+    ES_REPORT(Z4_MAX);
475
+  #endif
447
   #if HAS_CUSTOM_PROBE_PIN
476
   #if HAS_CUSTOM_PROBE_PIN
448
     print_es_state(READ(Z_MIN_PROBE_PIN) != Z_MIN_PROBE_ENDSTOP_INVERTING, PSTR(MSG_Z_PROBE));
477
     print_es_state(READ(Z_MIN_PROBE_PIN) != Z_MIN_PROBE_ENDSTOP_INVERTING, PSTR(MSG_Z_PROBE));
449
   #endif
478
   #endif
584
 
613
 
585
   #if HAS_Z_MIN && !Z_SPI_SENSORLESS
614
   #if HAS_Z_MIN && !Z_SPI_SENSORLESS
586
     UPDATE_ENDSTOP_BIT(Z, MIN);
615
     UPDATE_ENDSTOP_BIT(Z, MIN);
587
-    #if Z_MULTI_ENDSTOPS
616
+    #if ENABLED(Z_MULTI_ENDSTOPS)
588
       #if HAS_Z2_MIN
617
       #if HAS_Z2_MIN
589
         UPDATE_ENDSTOP_BIT(Z2, MIN);
618
         UPDATE_ENDSTOP_BIT(Z2, MIN);
590
       #else
619
       #else
591
         COPY_LIVE_STATE(Z_MIN, Z2_MIN);
620
         COPY_LIVE_STATE(Z_MIN, Z2_MIN);
592
       #endif
621
       #endif
593
-      #if ENABLED(Z_TRIPLE_ENDSTOPS)
622
+      #if NUM_Z_STEPPER_DRIVERS >= 3
594
         #if HAS_Z3_MIN
623
         #if HAS_Z3_MIN
595
           UPDATE_ENDSTOP_BIT(Z3, MIN);
624
           UPDATE_ENDSTOP_BIT(Z3, MIN);
596
         #else
625
         #else
597
           COPY_LIVE_STATE(Z_MIN, Z3_MIN);
626
           COPY_LIVE_STATE(Z_MIN, Z3_MIN);
598
         #endif
627
         #endif
599
       #endif
628
       #endif
629
+      #if NUM_Z_STEPPER_DRIVERS >= 4
630
+        #if HAS_Z4_MIN
631
+          UPDATE_ENDSTOP_BIT(Z4, MIN);
632
+        #else
633
+          COPY_LIVE_STATE(Z_MIN, Z4_MIN);
634
+        #endif
635
+      #endif
600
     #endif
636
     #endif
601
   #endif
637
   #endif
602
 
638
 
607
 
643
 
608
   #if HAS_Z_MAX && !Z_SPI_SENSORLESS
644
   #if HAS_Z_MAX && !Z_SPI_SENSORLESS
609
     // Check both Z dual endstops
645
     // Check both Z dual endstops
610
-    #if Z_MULTI_ENDSTOPS
646
+    #if ENABLED(Z_MULTI_ENDSTOPS)
611
       UPDATE_ENDSTOP_BIT(Z, MAX);
647
       UPDATE_ENDSTOP_BIT(Z, MAX);
612
       #if HAS_Z2_MAX
648
       #if HAS_Z2_MAX
613
         UPDATE_ENDSTOP_BIT(Z2, MAX);
649
         UPDATE_ENDSTOP_BIT(Z2, MAX);
614
       #else
650
       #else
615
         COPY_LIVE_STATE(Z_MAX, Z2_MAX);
651
         COPY_LIVE_STATE(Z_MAX, Z2_MAX);
616
       #endif
652
       #endif
617
-      #if ENABLED(Z_TRIPLE_ENDSTOPS)
653
+      #if NUM_Z_STEPPER_DRIVERS >= 3
618
         #if HAS_Z3_MAX
654
         #if HAS_Z3_MAX
619
           UPDATE_ENDSTOP_BIT(Z3, MAX);
655
           UPDATE_ENDSTOP_BIT(Z3, MAX);
620
         #else
656
         #else
621
           COPY_LIVE_STATE(Z_MAX, Z3_MAX);
657
           COPY_LIVE_STATE(Z_MAX, Z3_MAX);
622
         #endif
658
         #endif
623
       #endif
659
       #endif
660
+      #if NUM_Z_STEPPER_DRIVERS >= 4
661
+        #if HAS_Z4_MAX
662
+          UPDATE_ENDSTOP_BIT(Z4, MAX);
663
+        #else
664
+          COPY_LIVE_STATE(Z_MAX, Z4_MAX);
665
+        #endif
666
+      #endif
624
     #elif !HAS_CUSTOM_PROBE_PIN || Z_MAX_PIN != Z_MIN_PROBE_PIN
667
     #elif !HAS_CUSTOM_PROBE_PIN || Z_MAX_PIN != Z_MIN_PROBE_PIN
625
       // If this pin isn't the bed probe it's the Z endstop
668
       // If this pin isn't the bed probe it's the Z endstop
626
       UPDATE_ENDSTOP_BIT(Z, MAX);
669
       UPDATE_ENDSTOP_BIT(Z, MAX);
686
     } \
729
     } \
687
   }while(0)
730
   }while(0)
688
 
731
 
732
+  #define PROCESS_QUAD_ENDSTOP(AXIS1, AXIS2, AXIS3, AXIS4, MINMAX) do { \
733
+    const byte quad_hit = TEST_ENDSTOP(_ENDSTOP(AXIS1, MINMAX)) | (TEST_ENDSTOP(_ENDSTOP(AXIS2, MINMAX)) << 1) | (TEST_ENDSTOP(_ENDSTOP(AXIS3, MINMAX)) << 2) | (TEST_ENDSTOP(_ENDSTOP(AXIS4, MINMAX)) << 3); \
734
+    if (quad_hit) { \
735
+      _ENDSTOP_HIT(AXIS1, MINMAX); \
736
+      /* if not performing home or if both endstops were trigged during homing... */ \
737
+      if (!stepper.separate_multi_axis || quad_hit == 0b1111) \
738
+        planner.endstop_triggered(_AXIS(AXIS1)); \
739
+    } \
740
+  }while(0)
741
+
689
   #if ENABLED(G38_PROBE_TARGET) && PIN_EXISTS(Z_MIN_PROBE) && !(CORE_IS_XY || CORE_IS_XZ)
742
   #if ENABLED(G38_PROBE_TARGET) && PIN_EXISTS(Z_MIN_PROBE) && !(CORE_IS_XY || CORE_IS_XZ)
690
     #if ENABLED(G38_PROBE_AWAY)
743
     #if ENABLED(G38_PROBE_AWAY)
691
       #define _G38_OPEN_STATE (G38_move >= 4)
744
       #define _G38_OPEN_STATE (G38_move >= 4)
747
   if (stepper.axis_is_moving(Z_AXIS)) {
800
   if (stepper.axis_is_moving(Z_AXIS)) {
748
     if (stepper.motor_direction(Z_AXIS_HEAD)) { // Z -direction. Gantry down, bed up.
801
     if (stepper.motor_direction(Z_AXIS_HEAD)) { // Z -direction. Gantry down, bed up.
749
       #if HAS_Z_MIN || (Z_SPI_SENSORLESS && Z_HOME_DIR < 0)
802
       #if HAS_Z_MIN || (Z_SPI_SENSORLESS && Z_HOME_DIR < 0)
750
-        #if ENABLED(Z_TRIPLE_ENDSTOPS)
751
-          PROCESS_TRIPLE_ENDSTOP(Z, Z2, Z3, MIN);
752
-        #elif ENABLED(Z_DUAL_ENDSTOPS)
753
-          PROCESS_DUAL_ENDSTOP(Z, Z2, MIN);
803
+        #if ENABLED(Z_MULTI_ENDSTOPS)
804
+          #if NUM_Z_STEPPER_DRIVERS == 4
805
+            PROCESS_QUAD_ENDSTOP(Z, Z2, Z3, Z4, MIN);
806
+          #elif NUM_Z_STEPPER_DRIVERS == 3
807
+            PROCESS_TRIPLE_ENDSTOP(Z, Z2, Z3, MIN);
808
+          #else
809
+            PROCESS_DUAL_ENDSTOP(Z, Z2, MIN);
810
+          #endif
754
         #else
811
         #else
755
           #if ENABLED(Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN)
812
           #if ENABLED(Z_MIN_PROBE_USES_Z_MIN_ENDSTOP_PIN)
756
             if (z_probe_enabled) PROCESS_ENDSTOP(Z, MIN);
813
             if (z_probe_enabled) PROCESS_ENDSTOP(Z, MIN);
769
     }
826
     }
770
     else { // Z +direction. Gantry up, bed down.
827
     else { // Z +direction. Gantry up, bed down.
771
       #if HAS_Z_MAX || (Z_SPI_SENSORLESS && Z_HOME_DIR > 0)
828
       #if HAS_Z_MAX || (Z_SPI_SENSORLESS && Z_HOME_DIR > 0)
772
-        #if ENABLED(Z_TRIPLE_ENDSTOPS)
773
-          PROCESS_TRIPLE_ENDSTOP(Z, Z2, Z3, MAX);
774
-        #elif ENABLED(Z_DUAL_ENDSTOPS)
775
-          PROCESS_DUAL_ENDSTOP(Z, Z2, MAX);
829
+        #if ENABLED(Z_MULTI_ENDSTOPS)
830
+          #if NUM_Z_STEPPER_DRIVERS == 4
831
+            PROCESS_QUAD_ENDSTOP(Z, Z2, Z3, Z4, MAX);
832
+          #elif NUM_Z_STEPPER_DRIVERS == 3
833
+            PROCESS_TRIPLE_ENDSTOP(Z, Z2, Z3, MAX);
834
+          #else
835
+            PROCESS_DUAL_ENDSTOP(Z, Z2, MAX);
836
+          #endif
776
         #elif !HAS_CUSTOM_PROBE_PIN || Z_MAX_PIN != Z_MIN_PROBE_PIN
837
         #elif !HAS_CUSTOM_PROBE_PIN || Z_MAX_PIN != Z_MIN_PROBE_PIN
777
           // If this pin is not hijacked for the bed probe
838
           // If this pin is not hijacked for the bed probe
778
           // then it belongs to the Z endstop
839
           // then it belongs to the Z endstop
892
     #if HAS_Z3_MAX
953
     #if HAS_Z3_MAX
893
       ES_GET_STATE(Z3_MAX);
954
       ES_GET_STATE(Z3_MAX);
894
     #endif
955
     #endif
956
+    #if HAS_Z4_MIN
957
+      ES_GET_STATE(Z4_MIN);
958
+    #endif
959
+    #if HAS_Z4_MAX
960
+      ES_GET_STATE(Z4_MAX);
961
+    #endif
895
 
962
 
896
     uint16_t endstop_change = live_state_local ^ old_live_state_local;
963
     uint16_t endstop_change = live_state_local ^ old_live_state_local;
897
     #define ES_REPORT_CHANGE(S) if (TEST(endstop_change, S)) SERIAL_ECHOPAIR("  " STRINGIFY(S) ":", TEST(live_state_local, S))
964
     #define ES_REPORT_CHANGE(S) if (TEST(endstop_change, S)) SERIAL_ECHOPAIR("  " STRINGIFY(S) ":", TEST(live_state_local, S))
942
       #if HAS_Z3_MAX
1009
       #if HAS_Z3_MAX
943
         ES_REPORT_CHANGE(Z3_MAX);
1010
         ES_REPORT_CHANGE(Z3_MAX);
944
       #endif
1011
       #endif
1012
+      #if HAS_Z4_MIN
1013
+        ES_REPORT_CHANGE(Z4_MIN);
1014
+      #endif
1015
+      #if HAS_Z4_MAX
1016
+        ES_REPORT_CHANGE(Z4_MAX);
1017
+      #endif
945
       SERIAL_ECHOLNPGM("\n");
1018
       SERIAL_ECHOLNPGM("\n");
946
       analogWrite(pin_t(LED_PIN), local_LED_status);
1019
       analogWrite(pin_t(LED_PIN), local_LED_status);
947
       local_LED_status ^= 255;
1020
       local_LED_status ^= 255;

+ 7
- 3
Marlin/src/module/endstops.h View File

34
   X2_MIN, X2_MAX,
34
   X2_MIN, X2_MAX,
35
   Y2_MIN, Y2_MAX,
35
   Y2_MIN, Y2_MAX,
36
   Z2_MIN, Z2_MAX,
36
   Z2_MIN, Z2_MAX,
37
-  Z3_MIN, Z3_MAX
37
+  Z3_MIN, Z3_MAX,
38
+  Z4_MIN, Z4_MAX
38
 };
39
 };
39
 
40
 
40
 class Endstops {
41
 class Endstops {
47
       #if ENABLED(Y_DUAL_ENDSTOPS)
48
       #if ENABLED(Y_DUAL_ENDSTOPS)
48
         static float y2_endstop_adj;
49
         static float y2_endstop_adj;
49
       #endif
50
       #endif
50
-      #if Z_MULTI_ENDSTOPS
51
+      #if ENABLED(Z_MULTI_ENDSTOPS)
51
         static float z2_endstop_adj;
52
         static float z2_endstop_adj;
52
       #endif
53
       #endif
53
-      #if ENABLED(Z_TRIPLE_ENDSTOPS)
54
+      #if ENABLED(Z_MULTI_ENDSTOPS) && NUM_Z_STEPPER_DRIVERS >= 3
54
         static float z3_endstop_adj;
55
         static float z3_endstop_adj;
55
       #endif
56
       #endif
57
+      #if ENABLED(Z_MULTI_ENDSTOPS) && NUM_Z_STEPPER_DRIVERS >= 4
58
+        static float z4_endstop_adj;
59
+      #endif
56
     #else
60
     #else
57
       typedef uint8_t esbits_t;
61
       typedef uint8_t esbits_t;
58
     #endif
62
     #endif

+ 103
- 55
Marlin/src/module/motion.cpp View File

1156
           #if AXIS_HAS_STALLGUARD(Z3)
1156
           #if AXIS_HAS_STALLGUARD(Z3)
1157
             stealth_states.z3 = tmc_enable_stallguard(stepperZ3);
1157
             stealth_states.z3 = tmc_enable_stallguard(stepperZ3);
1158
           #endif
1158
           #endif
1159
+          #if AXIS_HAS_STALLGUARD(Z4)
1160
+            stealth_states.z4 = tmc_enable_stallguard(stepperZ4);
1161
+          #endif
1159
           #if CORE_IS_XZ && X_SENSORLESS
1162
           #if CORE_IS_XZ && X_SENSORLESS
1160
             stealth_states.x = tmc_enable_stallguard(stepperX);
1163
             stealth_states.x = tmc_enable_stallguard(stepperX);
1161
           #elif CORE_IS_YZ && Y_SENSORLESS
1164
           #elif CORE_IS_YZ && Y_SENSORLESS
1225
           #if AXIS_HAS_STALLGUARD(Z3)
1228
           #if AXIS_HAS_STALLGUARD(Z3)
1226
             tmc_disable_stallguard(stepperZ3, enable_stealth.z3);
1229
             tmc_disable_stallguard(stepperZ3, enable_stealth.z3);
1227
           #endif
1230
           #endif
1231
+          #if AXIS_HAS_STALLGUARD(Z4)
1232
+            tmc_disable_stallguard(stepperZ4, enable_stealth.z4);
1233
+          #endif
1228
           #if CORE_IS_XZ && X_SENSORLESS
1234
           #if CORE_IS_XZ && X_SENSORLESS
1229
             tmc_disable_stallguard(stepperX, enable_stealth.x);
1235
             tmc_disable_stallguard(stepperX, enable_stealth.x);
1230
           #elif CORE_IS_YZ && Y_SENSORLESS
1236
           #elif CORE_IS_YZ && Y_SENSORLESS
1509
       #if ENABLED(Y_DUAL_ENDSTOPS)
1515
       #if ENABLED(Y_DUAL_ENDSTOPS)
1510
         case Y_AXIS:
1516
         case Y_AXIS:
1511
       #endif
1517
       #endif
1512
-      #if Z_MULTI_ENDSTOPS
1518
+      #if ENABLED(Z_MULTI_ENDSTOPS)
1513
         case Z_AXIS:
1519
         case Z_AXIS:
1514
       #endif
1520
       #endif
1515
       stepper.set_separate_multi_axis(true);
1521
       stepper.set_separate_multi_axis(true);
1593
         }
1599
         }
1594
       }
1600
       }
1595
     #endif
1601
     #endif
1596
-    #if ENABLED(Z_DUAL_ENDSTOPS)
1602
+
1603
+    #if ENABLED(Z_MULTI_ENDSTOPS)
1597
       if (axis == Z_AXIS) {
1604
       if (axis == Z_AXIS) {
1598
-        const float adj = ABS(endstops.z2_endstop_adj);
1599
-        if (adj) {
1600
-          if (pos_dir ? (endstops.z2_endstop_adj > 0) : (endstops.z2_endstop_adj < 0)) stepper.set_z_lock(true); else stepper.set_z2_lock(true);
1601
-          do_homing_move(axis, pos_dir ? -adj : adj);
1605
+
1606
+        #if NUM_Z_STEPPER_DRIVERS == 2
1607
+
1608
+          const float adj = ABS(endstops.z2_endstop_adj);
1609
+          if (adj) {
1610
+            if (pos_dir ? (endstops.z2_endstop_adj > 0) : (endstops.z2_endstop_adj < 0)) stepper.set_z_lock(true); else stepper.set_z2_lock(true);
1611
+            do_homing_move(axis, pos_dir ? -adj : adj);
1612
+            stepper.set_z_lock(false);
1613
+            stepper.set_z2_lock(false);
1614
+          }
1615
+
1616
+        #else
1617
+
1618
+          // Handy arrays of stepper lock function pointers
1619
+
1620
+          typedef void (*adjustFunc_t)(const bool);
1621
+
1622
+          adjustFunc_t lock[] = {
1623
+            stepper.set_z_lock, stepper.set_z2_lock, stepper.set_z3_lock
1624
+            #if NUM_Z_STEPPER_DRIVERS >= 4
1625
+              , stepper.set_z4_lock
1626
+            #endif
1627
+          };
1628
+          float adj[] = {
1629
+            0, endstops.z2_endstop_adj, endstops.z3_endstop_adj
1630
+            #if NUM_Z_STEPPER_DRIVERS >= 4
1631
+              , endstops.z4_endstop_adj
1632
+            #endif
1633
+          };
1634
+
1635
+          adjustFunc_t tempLock;
1636
+          float tempAdj;
1637
+
1638
+          // Manual bubble sort by adjust value
1639
+          if (adj[1] < adj[0]) {
1640
+            tempLock = lock[0], tempAdj = adj[0];
1641
+            lock[0] = lock[1], adj[0] = adj[1];
1642
+            lock[1] = tempLock, adj[1] = tempAdj;
1643
+          }
1644
+          if (adj[2] < adj[1]) {
1645
+            tempLock = lock[1], tempAdj = adj[1];
1646
+            lock[1] = lock[2], adj[1] = adj[2];
1647
+            lock[2] = tempLock, adj[2] = tempAdj;
1648
+          }
1649
+          #if NUM_Z_STEPPER_DRIVERS >= 4
1650
+            if (adj[3] < adj[2]) {
1651
+              tempLock = lock[2], tempAdj = adj[2];
1652
+              lock[2] = lock[3], adj[2] = adj[3];
1653
+              lock[3] = tempLock, adj[3] = tempAdj;
1654
+            }
1655
+            if (adj[2] < adj[1]) {
1656
+              tempLock = lock[1], tempAdj = adj[1];
1657
+              lock[1] = lock[2], adj[1] = adj[2];
1658
+              lock[2] = tempLock, adj[2] = tempAdj;
1659
+            }
1660
+          #endif
1661
+          if (adj[1] < adj[0]) {
1662
+            tempLock = lock[0], tempAdj = adj[0];
1663
+            lock[0] = lock[1], adj[0] = adj[1];
1664
+            lock[1] = tempLock, adj[1] = tempAdj;
1665
+          }
1666
+
1667
+          if (pos_dir) {
1668
+            // normalize adj to smallest value and do the first move
1669
+            (*lock[0])(true);
1670
+            do_homing_move(axis, adj[1] - adj[0]);
1671
+            // lock the second stepper for the final correction
1672
+            (*lock[1])(true);
1673
+            do_homing_move(axis, adj[2] - adj[1]);
1674
+            #if NUM_Z_STEPPER_DRIVERS >= 4
1675
+              // lock the third stepper for the final correction
1676
+              (*lock[2])(true);
1677
+              do_homing_move(axis, adj[3] - adj[2]);
1678
+            #endif
1679
+          }
1680
+          else {
1681
+            #if NUM_Z_STEPPER_DRIVERS >= 4
1682
+              (*lock[3])(true);
1683
+              do_homing_move(axis, adj[2] - adj[3]);
1684
+            #endif
1685
+            (*lock[2])(true);
1686
+            do_homing_move(axis, adj[1] - adj[2]);
1687
+            (*lock[1])(true);
1688
+            do_homing_move(axis, adj[0] - adj[1]);
1689
+          }
1690
+
1602
           stepper.set_z_lock(false);
1691
           stepper.set_z_lock(false);
1603
           stepper.set_z2_lock(false);
1692
           stepper.set_z2_lock(false);
1604
-        }
1605
-      }
1606
-    #endif
1607
-    #if ENABLED(Z_TRIPLE_ENDSTOPS)
1608
-      if (axis == Z_AXIS) {
1609
-        // we push the function pointers for the stepper lock function into an array
1610
-        void (*lock[3]) (bool)= {&stepper.set_z_lock, &stepper.set_z2_lock, &stepper.set_z3_lock};
1611
-        float adj[3] = {0, endstops.z2_endstop_adj, endstops.z3_endstop_adj};
1612
-
1613
-        void (*tempLock) (bool);
1614
-        float tempAdj;
1615
-
1616
-        // manual bubble sort by adjust value
1617
-        if (adj[1] < adj[0]) {
1618
-          tempLock = lock[0], tempAdj = adj[0];
1619
-          lock[0] = lock[1], adj[0] = adj[1];
1620
-          lock[1] = tempLock, adj[1] = tempAdj;
1621
-        }
1622
-        if (adj[2] < adj[1]) {
1623
-          tempLock = lock[1], tempAdj = adj[1];
1624
-          lock[1] = lock[2], adj[1] = adj[2];
1625
-          lock[2] = tempLock, adj[2] = tempAdj;
1626
-        }
1627
-        if (adj[1] < adj[0]) {
1628
-          tempLock = lock[0], tempAdj = adj[0];
1629
-          lock[0] = lock[1], adj[0] = adj[1];
1630
-          lock[1] = tempLock, adj[1] = tempAdj;
1631
-        }
1632
-
1633
-        if (pos_dir) {
1634
-          // normalize adj to smallest value and do the first move
1635
-          (*lock[0])(true);
1636
-          do_homing_move(axis, adj[1] - adj[0]);
1637
-          // lock the second stepper for the final correction
1638
-          (*lock[1])(true);
1639
-          do_homing_move(axis, adj[2] - adj[1]);
1640
-        }
1641
-        else {
1642
-          (*lock[2])(true);
1643
-          do_homing_move(axis, adj[1] - adj[2]);
1644
-          (*lock[1])(true);
1645
-          do_homing_move(axis, adj[0] - adj[1]);
1646
-        }
1693
+          stepper.set_z3_lock(false);
1694
+          #if NUM_Z_STEPPER_DRIVERS >= 4
1695
+            stepper.set_z4_lock(false);
1696
+          #endif
1647
 
1697
 
1648
-        stepper.set_z_lock(false);
1649
-        stepper.set_z2_lock(false);
1650
-        stepper.set_z3_lock(false);
1698
+        #endif
1651
       }
1699
       }
1652
     #endif
1700
     #endif
1653
 
1701
 
1654
     // Reset flags for X, Y, Z motor locking
1702
     // Reset flags for X, Y, Z motor locking
1655
     switch (axis) {
1703
     switch (axis) {
1704
+      default: break;
1656
       #if ENABLED(X_DUAL_ENDSTOPS)
1705
       #if ENABLED(X_DUAL_ENDSTOPS)
1657
         case X_AXIS:
1706
         case X_AXIS:
1658
       #endif
1707
       #endif
1659
       #if ENABLED(Y_DUAL_ENDSTOPS)
1708
       #if ENABLED(Y_DUAL_ENDSTOPS)
1660
         case Y_AXIS:
1709
         case Y_AXIS:
1661
       #endif
1710
       #endif
1662
-      #if Z_MULTI_ENDSTOPS
1711
+      #if ENABLED(Z_MULTI_ENDSTOPS)
1663
         case Z_AXIS:
1712
         case Z_AXIS:
1664
       #endif
1713
       #endif
1665
-      stepper.set_separate_multi_axis(false);
1666
-      default: break;
1714
+          stepper.set_separate_multi_axis(false);
1667
     }
1715
     }
1668
   #endif
1716
   #endif
1669
 
1717
 

+ 95
- 18
Marlin/src/module/stepper.cpp View File

156
 #if ENABLED(Y_DUAL_ENDSTOPS)
156
 #if ENABLED(Y_DUAL_ENDSTOPS)
157
   bool Stepper::locked_Y_motor = false, Stepper::locked_Y2_motor = false;
157
   bool Stepper::locked_Y_motor = false, Stepper::locked_Y2_motor = false;
158
 #endif
158
 #endif
159
-#if Z_MULTI_ENDSTOPS || ENABLED(Z_STEPPER_AUTO_ALIGN)
160
-  bool Stepper::locked_Z_motor = false, Stepper::locked_Z2_motor = false;
161
-#endif
162
-#if ENABLED(Z_TRIPLE_ENDSTOPS) || BOTH(Z_STEPPER_AUTO_ALIGN, Z_TRIPLE_STEPPER_DRIVERS)
163
-  bool Stepper::locked_Z3_motor = false;
159
+
160
+#if EITHER(Z_MULTI_ENDSTOPS, Z_STEPPER_AUTO_ALIGN)
161
+  bool Stepper::locked_Z_motor = false, Stepper::locked_Z2_motor = false
162
+       #if NUM_Z_STEPPER_DRIVERS >= 3
163
+         , Stepper::locked_Z3_motor = false
164
+         #if NUM_Z_STEPPER_DRIVERS >= 4
165
+           , Stepper::locked_Z4_motor = false
166
+         #endif
167
+       #endif
168
+       ;
164
 #endif
169
 #endif
165
 
170
 
166
 uint32_t Stepper::acceleration_time, Stepper::deceleration_time;
171
 uint32_t Stepper::acceleration_time, Stepper::deceleration_time;
281
     A##3_STEP_WRITE(V);                           \
286
     A##3_STEP_WRITE(V);                           \
282
   }
287
   }
283
 
288
 
289
+#define QUAD_ENDSTOP_APPLY_STEP(A,V)                                                                                        \
290
+  if (separate_multi_axis) {                                                                                                \
291
+    if (A##_HOME_DIR < 0) {                                                                                                 \
292
+      if (!(TEST(endstops.state(), A##_MIN) && count_direction[_AXIS(A)] < 0) && !locked_##A##_motor) A##_STEP_WRITE(V);    \
293
+      if (!(TEST(endstops.state(), A##2_MIN) && count_direction[_AXIS(A)] < 0) && !locked_##A##2_motor) A##2_STEP_WRITE(V); \
294
+      if (!(TEST(endstops.state(), A##3_MIN) && count_direction[_AXIS(A)] < 0) && !locked_##A##3_motor) A##3_STEP_WRITE(V); \
295
+      if (!(TEST(endstops.state(), A##4_MIN) && count_direction[_AXIS(A)] < 0) && !locked_##A##4_motor) A##4_STEP_WRITE(V); \
296
+    }                                                                                                                       \
297
+    else {                                                                                                                  \
298
+      if (!(TEST(endstops.state(), A##_MAX) && count_direction[_AXIS(A)] > 0) && !locked_##A##_motor) A##_STEP_WRITE(V);    \
299
+      if (!(TEST(endstops.state(), A##2_MAX) && count_direction[_AXIS(A)] > 0) && !locked_##A##2_motor) A##2_STEP_WRITE(V); \
300
+      if (!(TEST(endstops.state(), A##3_MAX) && count_direction[_AXIS(A)] > 0) && !locked_##A##3_motor) A##3_STEP_WRITE(V); \
301
+      if (!(TEST(endstops.state(), A##4_MAX) && count_direction[_AXIS(A)] > 0) && !locked_##A##4_motor) A##4_STEP_WRITE(V); \
302
+    }                                                                                                                       \
303
+  }                                                                                                                         \
304
+  else {                                                                                                                    \
305
+    A##_STEP_WRITE(V);                                                                                                      \
306
+    A##2_STEP_WRITE(V);                                                                                                     \
307
+    A##3_STEP_WRITE(V);                                                                                                     \
308
+    A##4_STEP_WRITE(V);                                                                                                     \
309
+  }
310
+
311
+#define QUAD_SEPARATE_APPLY_STEP(A,V)             \
312
+  if (separate_multi_axis) {                      \
313
+    if (!locked_##A##_motor) A##_STEP_WRITE(V);   \
314
+    if (!locked_##A##2_motor) A##2_STEP_WRITE(V); \
315
+    if (!locked_##A##3_motor) A##3_STEP_WRITE(V); \
316
+    if (!locked_##A##4_motor) A##4_STEP_WRITE(V); \
317
+  }                                               \
318
+  else {                                          \
319
+    A##_STEP_WRITE(V);                            \
320
+    A##2_STEP_WRITE(V);                           \
321
+    A##3_STEP_WRITE(V);                           \
322
+    A##4_STEP_WRITE(V);                           \
323
+  }
324
+
284
 #if ENABLED(X_DUAL_STEPPER_DRIVERS)
325
 #if ENABLED(X_DUAL_STEPPER_DRIVERS)
285
   #define X_APPLY_DIR(v,Q) do{ X_DIR_WRITE(v); X2_DIR_WRITE((v) != INVERT_X2_VS_X_DIR); }while(0)
326
   #define X_APPLY_DIR(v,Q) do{ X_DIR_WRITE(v); X2_DIR_WRITE((v) != INVERT_X2_VS_X_DIR); }while(0)
286
   #if ENABLED(X_DUAL_ENDSTOPS)
327
   #if ENABLED(X_DUAL_ENDSTOPS)
314
   #define Y_APPLY_STEP(v,Q) Y_STEP_WRITE(v)
355
   #define Y_APPLY_STEP(v,Q) Y_STEP_WRITE(v)
315
 #endif
356
 #endif
316
 
357
 
317
-#if ENABLED(Z_TRIPLE_STEPPER_DRIVERS)
358
+#if NUM_Z_STEPPER_DRIVERS == 4
359
+  #define Z_APPLY_DIR(v,Q) do{ Z_DIR_WRITE(v); Z2_DIR_WRITE(v); Z3_DIR_WRITE(v); Z4_DIR_WRITE(v); }while(0)
360
+  #if ENABLED(Z_MULTI_ENDSTOPS)
361
+    #define Z_APPLY_STEP(v,Q) QUAD_ENDSTOP_APPLY_STEP(Z,v)
362
+  #elif ENABLED(Z_STEPPER_AUTO_ALIGN)
363
+    #define Z_APPLY_STEP(v,Q) QUAD_SEPARATE_APPLY_STEP(Z,v)
364
+  #else
365
+    #define Z_APPLY_STEP(v,Q) do{ Z_STEP_WRITE(v); Z2_STEP_WRITE(v); Z3_STEP_WRITE(v); Z4_STEP_WRITE(v); }while(0)
366
+  #endif
367
+#elif NUM_Z_STEPPER_DRIVERS == 3
318
   #define Z_APPLY_DIR(v,Q) do{ Z_DIR_WRITE(v); Z2_DIR_WRITE(v); Z3_DIR_WRITE(v); }while(0)
368
   #define Z_APPLY_DIR(v,Q) do{ Z_DIR_WRITE(v); Z2_DIR_WRITE(v); Z3_DIR_WRITE(v); }while(0)
319
-  #if ENABLED(Z_TRIPLE_ENDSTOPS)
369
+  #if ENABLED(Z_MULTI_ENDSTOPS)
320
     #define Z_APPLY_STEP(v,Q) TRIPLE_ENDSTOP_APPLY_STEP(Z,v)
370
     #define Z_APPLY_STEP(v,Q) TRIPLE_ENDSTOP_APPLY_STEP(Z,v)
321
   #elif ENABLED(Z_STEPPER_AUTO_ALIGN)
371
   #elif ENABLED(Z_STEPPER_AUTO_ALIGN)
322
     #define Z_APPLY_STEP(v,Q) TRIPLE_SEPARATE_APPLY_STEP(Z,v)
372
     #define Z_APPLY_STEP(v,Q) TRIPLE_SEPARATE_APPLY_STEP(Z,v)
323
   #else
373
   #else
324
     #define Z_APPLY_STEP(v,Q) do{ Z_STEP_WRITE(v); Z2_STEP_WRITE(v); Z3_STEP_WRITE(v); }while(0)
374
     #define Z_APPLY_STEP(v,Q) do{ Z_STEP_WRITE(v); Z2_STEP_WRITE(v); Z3_STEP_WRITE(v); }while(0)
325
   #endif
375
   #endif
326
-#elif ENABLED(Z_DUAL_STEPPER_DRIVERS)
376
+#elif NUM_Z_STEPPER_DRIVERS == 2
327
   #define Z_APPLY_DIR(v,Q) do{ Z_DIR_WRITE(v); Z2_DIR_WRITE(v); }while(0)
377
   #define Z_APPLY_DIR(v,Q) do{ Z_DIR_WRITE(v); Z2_DIR_WRITE(v); }while(0)
328
-  #if ENABLED(Z_DUAL_ENDSTOPS)
378
+  #if ENABLED(Z_MULTI_ENDSTOPS)
329
     #define Z_APPLY_STEP(v,Q) DUAL_ENDSTOP_APPLY_STEP(Z,v)
379
     #define Z_APPLY_STEP(v,Q) DUAL_ENDSTOP_APPLY_STEP(Z,v)
330
   #elif ENABLED(Z_STEPPER_AUTO_ALIGN)
380
   #elif ENABLED(Z_STEPPER_AUTO_ALIGN)
331
     #define Z_APPLY_STEP(v,Q) DUAL_SEPARATE_APPLY_STEP(Z,v)
381
     #define Z_APPLY_STEP(v,Q) DUAL_SEPARATE_APPLY_STEP(Z,v)
2062
   #endif
2112
   #endif
2063
   #if HAS_Z_DIR
2113
   #if HAS_Z_DIR
2064
     Z_DIR_INIT();
2114
     Z_DIR_INIT();
2065
-    #if Z_MULTI_STEPPER_DRIVERS && HAS_Z2_DIR
2115
+    #if NUM_Z_STEPPER_DRIVERS >= 2 && HAS_Z2_DIR
2066
       Z2_DIR_INIT();
2116
       Z2_DIR_INIT();
2067
     #endif
2117
     #endif
2068
-    #if ENABLED(Z_TRIPLE_STEPPER_DRIVERS) && HAS_Z3_DIR
2118
+    #if NUM_Z_STEPPER_DRIVERS >= 3 && HAS_Z3_DIR
2069
       Z3_DIR_INIT();
2119
       Z3_DIR_INIT();
2070
     #endif
2120
     #endif
2121
+    #if NUM_Z_STEPPER_DRIVERS >= 4 && HAS_Z4_DIR
2122
+      Z4_DIR_INIT();
2123
+    #endif
2071
   #endif
2124
   #endif
2072
   #if HAS_E0_DIR
2125
   #if HAS_E0_DIR
2073
     E0_DIR_INIT();
2126
     E0_DIR_INIT();
2108
   #if HAS_Z_ENABLE
2161
   #if HAS_Z_ENABLE
2109
     Z_ENABLE_INIT();
2162
     Z_ENABLE_INIT();
2110
     if (!Z_ENABLE_ON) Z_ENABLE_WRITE(HIGH);
2163
     if (!Z_ENABLE_ON) Z_ENABLE_WRITE(HIGH);
2111
-    #if Z_MULTI_STEPPER_DRIVERS && HAS_Z2_ENABLE
2164
+    #if NUM_Z_STEPPER_DRIVERS >= 2 && HAS_Z2_ENABLE
2112
       Z2_ENABLE_INIT();
2165
       Z2_ENABLE_INIT();
2113
       if (!Z_ENABLE_ON) Z2_ENABLE_WRITE(HIGH);
2166
       if (!Z_ENABLE_ON) Z2_ENABLE_WRITE(HIGH);
2114
     #endif
2167
     #endif
2115
-    #if ENABLED(Z_TRIPLE_STEPPER_DRIVERS) && HAS_Z3_ENABLE
2168
+    #if NUM_Z_STEPPER_DRIVERS >= 3 && HAS_Z3_ENABLE
2116
       Z3_ENABLE_INIT();
2169
       Z3_ENABLE_INIT();
2117
       if (!Z_ENABLE_ON) Z3_ENABLE_WRITE(HIGH);
2170
       if (!Z_ENABLE_ON) Z3_ENABLE_WRITE(HIGH);
2118
     #endif
2171
     #endif
2172
+    #if NUM_Z_STEPPER_DRIVERS >= 4 && HAS_Z4_ENABLE
2173
+      Z4_ENABLE_INIT();
2174
+      if (!Z_ENABLE_ON) Z4_ENABLE_WRITE(HIGH);
2175
+    #endif
2119
   #endif
2176
   #endif
2120
   #if HAS_E0_ENABLE
2177
   #if HAS_E0_ENABLE
2121
     E0_ENABLE_INIT();
2178
     E0_ENABLE_INIT();
2171
   #endif
2228
   #endif
2172
 
2229
 
2173
   #if HAS_Z_STEP
2230
   #if HAS_Z_STEP
2174
-    #if Z_MULTI_STEPPER_DRIVERS
2231
+    #if NUM_Z_STEPPER_DRIVERS >= 2
2175
       Z2_STEP_INIT();
2232
       Z2_STEP_INIT();
2176
       Z2_STEP_WRITE(INVERT_Z_STEP_PIN);
2233
       Z2_STEP_WRITE(INVERT_Z_STEP_PIN);
2177
     #endif
2234
     #endif
2178
-    #if ENABLED(Z_TRIPLE_STEPPER_DRIVERS)
2235
+    #if NUM_Z_STEPPER_DRIVERS >= 3
2179
       Z3_STEP_INIT();
2236
       Z3_STEP_INIT();
2180
       Z3_STEP_WRITE(INVERT_Z_STEP_PIN);
2237
       Z3_STEP_WRITE(INVERT_Z_STEP_PIN);
2181
     #endif
2238
     #endif
2239
+    #if NUM_Z_STEPPER_DRIVERS >= 4
2240
+      Z4_STEP_INIT();
2241
+      Z4_STEP_WRITE(INVERT_Z_STEP_PIN);
2242
+    #endif
2182
     AXIS_INIT(Z, Z);
2243
     AXIS_INIT(Z, Z);
2183
   #endif
2244
   #endif
2184
 
2245
 
2692
         SET_OUTPUT(Z3_MS3_PIN);
2753
         SET_OUTPUT(Z3_MS3_PIN);
2693
       #endif
2754
       #endif
2694
     #endif
2755
     #endif
2756
+    #if HAS_Z4_MICROSTEPS
2757
+      SET_OUTPUT(Z4_MS1_PIN);
2758
+      SET_OUTPUT(Z4_MS2_PIN);
2759
+      #if PIN_EXISTS(Z4_MS3)
2760
+        SET_OUTPUT(Z4_MS3_PIN);
2761
+      #endif
2762
+    #endif
2695
     #if HAS_E0_MICROSTEPS
2763
     #if HAS_E0_MICROSTEPS
2696
       SET_OUTPUT(E0_MS1_PIN);
2764
       SET_OUTPUT(E0_MS1_PIN);
2697
       SET_OUTPUT(E0_MS2_PIN);
2765
       SET_OUTPUT(E0_MS2_PIN);
2762
           #endif
2830
           #endif
2763
           break;
2831
           break;
2764
       #endif
2832
       #endif
2765
-      #if HAS_Z_MICROSTEPS || HAS_Z2_MICROSTEPS || HAS_Z3_MICROSTEPS
2833
+      #if HAS_SOME_Z_MICROSTEPS
2766
         case 2:
2834
         case 2:
2767
           #if HAS_Z_MICROSTEPS
2835
           #if HAS_Z_MICROSTEPS
2768
             WRITE(Z_MS1_PIN, ms1);
2836
             WRITE(Z_MS1_PIN, ms1);
2773
           #if HAS_Z3_MICROSTEPS
2841
           #if HAS_Z3_MICROSTEPS
2774
             WRITE(Z3_MS1_PIN, ms1);
2842
             WRITE(Z3_MS1_PIN, ms1);
2775
           #endif
2843
           #endif
2844
+          #if HAS_Z4_MICROSTEPS
2845
+            WRITE(Z4_MS1_PIN, ms1);
2846
+          #endif
2776
           break;
2847
           break;
2777
       #endif
2848
       #endif
2778
       #if HAS_E0_MICROSTEPS
2849
       #if HAS_E0_MICROSTEPS
2815
           #endif
2886
           #endif
2816
           break;
2887
           break;
2817
       #endif
2888
       #endif
2818
-      #if HAS_Z_MICROSTEPS || HAS_Z2_MICROSTEPS || HAS_Z3_MICROSTEPS
2889
+      #if HAS_SOME_Z_MICROSTEPS
2819
         case 2:
2890
         case 2:
2820
           #if HAS_Z_MICROSTEPS
2891
           #if HAS_Z_MICROSTEPS
2821
             WRITE(Z_MS2_PIN, ms2);
2892
             WRITE(Z_MS2_PIN, ms2);
2826
           #if HAS_Z3_MICROSTEPS
2897
           #if HAS_Z3_MICROSTEPS
2827
             WRITE(Z3_MS2_PIN, ms2);
2898
             WRITE(Z3_MS2_PIN, ms2);
2828
           #endif
2899
           #endif
2900
+          #if HAS_Z4_MICROSTEPS
2901
+            WRITE(Z4_MS2_PIN, ms2);
2902
+          #endif
2829
           break;
2903
           break;
2830
       #endif
2904
       #endif
2831
       #if HAS_E0_MICROSTEPS
2905
       #if HAS_E0_MICROSTEPS
2868
           #endif
2942
           #endif
2869
           break;
2943
           break;
2870
       #endif
2944
       #endif
2871
-      #if HAS_Z_MICROSTEPS || HAS_Z2_MICROSTEPS || HAS_Z3_MICROSTEPS
2945
+      #if HAS_SOME_Z_MICROSTEPS
2872
         case 2:
2946
         case 2:
2873
           #if HAS_Z_MICROSTEPS && PIN_EXISTS(Z_MS3)
2947
           #if HAS_Z_MICROSTEPS && PIN_EXISTS(Z_MS3)
2874
             WRITE(Z_MS3_PIN, ms3);
2948
             WRITE(Z_MS3_PIN, ms3);
2879
           #if HAS_Z3_MICROSTEPS && PIN_EXISTS(Z3_MS3)
2953
           #if HAS_Z3_MICROSTEPS && PIN_EXISTS(Z3_MS3)
2880
             WRITE(Z3_MS3_PIN, ms3);
2954
             WRITE(Z3_MS3_PIN, ms3);
2881
           #endif
2955
           #endif
2956
+          #if HAS_Z4_MICROSTEPS && PIN_EXISTS(Z4_MS3)
2957
+            WRITE(Z4_MS3_PIN, ms3);
2958
+          #endif
2882
           break;
2959
           break;
2883
       #endif
2960
       #endif
2884
       #if HAS_E0_MICROSTEPS && PIN_EXISTS(E0_MS3)
2961
       #if HAS_E0_MICROSTEPS && PIN_EXISTS(E0_MS3)

+ 16
- 9
Marlin/src/module/stepper.h View File

274
     #if ENABLED(Y_DUAL_ENDSTOPS)
274
     #if ENABLED(Y_DUAL_ENDSTOPS)
275
       static bool locked_Y_motor, locked_Y2_motor;
275
       static bool locked_Y_motor, locked_Y2_motor;
276
     #endif
276
     #endif
277
-    #if Z_MULTI_ENDSTOPS || ENABLED(Z_STEPPER_AUTO_ALIGN)
278
-      static bool locked_Z_motor, locked_Z2_motor;
279
-    #endif
280
-    #if ENABLED(Z_TRIPLE_ENDSTOPS) || BOTH(Z_STEPPER_AUTO_ALIGN, Z_TRIPLE_STEPPER_DRIVERS)
281
-      static bool locked_Z3_motor;
277
+    #if EITHER(Z_MULTI_ENDSTOPS, Z_STEPPER_AUTO_ALIGN)
278
+      static bool locked_Z_motor, locked_Z2_motor
279
+                  #if NUM_Z_STEPPER_DRIVERS >= 3
280
+                    , locked_Z3_motor
281
+                    #if NUM_Z_STEPPER_DRIVERS >= 4
282
+                      , locked_Z4_motor
283
+                    #endif
284
+                  #endif
285
+                  ;
282
     #endif
286
     #endif
283
 
287
 
284
     static uint32_t acceleration_time, deceleration_time; // time measured in Stepper Timer ticks
288
     static uint32_t acceleration_time, deceleration_time; // time measured in Stepper Timer ticks
430
       FORCE_INLINE static void set_y_lock(const bool state) { locked_Y_motor = state; }
434
       FORCE_INLINE static void set_y_lock(const bool state) { locked_Y_motor = state; }
431
       FORCE_INLINE static void set_y2_lock(const bool state) { locked_Y2_motor = state; }
435
       FORCE_INLINE static void set_y2_lock(const bool state) { locked_Y2_motor = state; }
432
     #endif
436
     #endif
433
-    #if Z_MULTI_ENDSTOPS || (ENABLED(Z_STEPPER_AUTO_ALIGN) && Z_MULTI_STEPPER_DRIVERS)
437
+    #if EITHER(Z_MULTI_ENDSTOPS, Z_STEPPER_AUTO_ALIGN)
434
       FORCE_INLINE static void set_z_lock(const bool state) { locked_Z_motor = state; }
438
       FORCE_INLINE static void set_z_lock(const bool state) { locked_Z_motor = state; }
435
       FORCE_INLINE static void set_z2_lock(const bool state) { locked_Z2_motor = state; }
439
       FORCE_INLINE static void set_z2_lock(const bool state) { locked_Z2_motor = state; }
436
-    #endif
437
-    #if ENABLED(Z_TRIPLE_ENDSTOPS) || BOTH(Z_STEPPER_AUTO_ALIGN, Z_TRIPLE_STEPPER_DRIVERS)
438
-      FORCE_INLINE static void set_z3_lock(const bool state) { locked_Z3_motor = state; }
440
+      #if NUM_Z_STEPPER_DRIVERS >= 3
441
+        FORCE_INLINE static void set_z3_lock(const bool state) { locked_Z3_motor = state; }
442
+        #if NUM_Z_STEPPER_DRIVERS >= 4
443
+          FORCE_INLINE static void set_z4_lock(const bool state) { locked_Z4_motor = state; }
444
+        #endif
445
+      #endif
439
     #endif
446
     #endif
440
 
447
 
441
     #if ENABLED(BABYSTEPPING)
448
     #if ENABLED(BABYSTEPPING)

+ 3
- 0
Marlin/src/module/stepper/L64xx.cpp View File

52
 #if AXIS_IS_L64XX(Z3)
52
 #if AXIS_IS_L64XX(Z3)
53
   L64XX_CLASS(Z3) stepperZ3(L6470_CHAIN_SS_PIN);
53
   L64XX_CLASS(Z3) stepperZ3(L6470_CHAIN_SS_PIN);
54
 #endif
54
 #endif
55
+#if AXIS_IS_L64XX(Z4)
56
+  L64XX_CLASS(Z4) stepperZ4(L6470_CHAIN_SS_PIN);
57
+#endif
55
 #if AXIS_IS_L64XX(E0)
58
 #if AXIS_IS_L64XX(E0)
56
   L64XX_CLASS(E0) stepperE0(L6470_CHAIN_SS_PIN);
59
   L64XX_CLASS(E0) stepperE0(L6470_CHAIN_SS_PIN);
57
 #endif
60
 #endif

+ 17
- 0
Marlin/src/module/stepper/L64xx.h View File

160
   #endif
160
   #endif
161
 #endif
161
 #endif
162
 
162
 
163
+// Z4 Stepper
164
+#if HAS_Z4_ENABLE && AXIS_IS_L64XX(Z4)
165
+  extern L64XX_CLASS(Z4)         stepperZ4;
166
+  #define Z4_ENABLE_INIT()       NOOP
167
+  #define Z4_ENABLE_WRITE(STATE) (STATE ? NOOP : stepperZ4.free())
168
+  #define Z4_ENABLE_READ()       (stepperZ4.getStatus() & STATUS_HIZ)
169
+  #if AXIS_DRIVER_TYPE_Z4(L6474)
170
+    #define Z4_DIR_INIT()        SET_OUTPUT(Z4_DIR_PIN)
171
+    #define Z4_DIR_WRITE(STATE)  L6474_DIR_WRITE(Z4, STATE)
172
+    #define Z4_DIR_READ()        READ(Z4_DIR_PIN)
173
+  #else
174
+    #define Z4_DIR_INIT()        NOOP
175
+    #define Z4_DIR_WRITE(STATE)  L64XX_DIR_WRITE(Z4, STATE)
176
+    #define Z4_DIR_READ()        (stepper##Z4.getStatus() & STATUS_DIR);
177
+  #endif
178
+#endif
179
+
163
 // E0 Stepper
180
 // E0 Stepper
164
 #if AXIS_IS_L64XX(E0)
181
 #if AXIS_IS_L64XX(E0)
165
   extern L64XX_CLASS(E0)         stepperE0;
182
   extern L64XX_CLASS(E0)         stepperE0;

+ 6
- 0
Marlin/src/module/stepper/TMC26X.cpp View File

57
 #if AXIS_DRIVER_TYPE_Z3(TMC26X)
57
 #if AXIS_DRIVER_TYPE_Z3(TMC26X)
58
   _TMC26X_DEFINE(Z3);
58
   _TMC26X_DEFINE(Z3);
59
 #endif
59
 #endif
60
+#if AXIS_DRIVER_TYPE_Z4(TMC26X)
61
+  _TMC26X_DEFINE(Z4);
62
+#endif
60
 #if AXIS_DRIVER_TYPE_E0(TMC26X)
63
 #if AXIS_DRIVER_TYPE_E0(TMC26X)
61
   _TMC26X_DEFINE(E0);
64
   _TMC26X_DEFINE(E0);
62
 #endif
65
 #endif
103
   #if AXIS_DRIVER_TYPE_Z3(TMC26X)
106
   #if AXIS_DRIVER_TYPE_Z3(TMC26X)
104
     _TMC26X_INIT(Z3);
107
     _TMC26X_INIT(Z3);
105
   #endif
108
   #endif
109
+  #if AXIS_DRIVER_TYPE_Z4(TMC26X)
110
+    _TMC26X_INIT(Z4);
111
+  #endif
106
   #if AXIS_DRIVER_TYPE_E0(TMC26X)
112
   #if AXIS_DRIVER_TYPE_E0(TMC26X)
107
     _TMC26X_INIT(E0);
113
     _TMC26X_INIT(E0);
108
   #endif
114
   #endif

+ 8
- 0
Marlin/src/module/stepper/TMC26X.h View File

95
   #define Z3_ENABLE_READ() stepperZ3.isEnabled()
95
   #define Z3_ENABLE_READ() stepperZ3.isEnabled()
96
 #endif
96
 #endif
97
 
97
 
98
+// Z4 Stepper
99
+#if HAS_Z4_ENABLE && AXIS_DRIVER_TYPE_Z4(TMC26X)
100
+  extern TMC26XStepper stepperZ4;
101
+  #define Z4_ENABLE_INIT() NOOP
102
+  #define Z4_ENABLE_WRITE(STATE) stepperZ4.setEnabled(STATE)
103
+  #define Z4_ENABLE_READ() stepperZ4.isEnabled()
104
+#endif
105
+
98
 // E0 Stepper
106
 // E0 Stepper
99
 #if AXIS_DRIVER_TYPE_E0(TMC26X)
107
 #if AXIS_DRIVER_TYPE_E0(TMC26X)
100
   extern TMC26XStepper stepperE0;
108
   extern TMC26XStepper stepperE0;

+ 35
- 2
Marlin/src/module/stepper/indirection.h View File

180
   #define Z3_DIR_WRITE(STATE) NOOP
180
   #define Z3_DIR_WRITE(STATE) NOOP
181
 #endif
181
 #endif
182
 
182
 
183
+// Z4 Stepper
184
+#if HAS_Z4_ENABLE
185
+  #ifndef Z4_ENABLE_INIT
186
+    #define Z4_ENABLE_INIT() SET_OUTPUT(Z4_ENABLE_PIN)
187
+    #define Z4_ENABLE_WRITE(STATE) WRITE(Z4_ENABLE_PIN,STATE)
188
+    #define Z4_ENABLE_READ() READ(Z4_ENABLE_PIN)
189
+  #endif
190
+  #ifndef Z4_DIR_INIT
191
+    #define Z4_DIR_INIT() SET_OUTPUT(Z4_DIR_PIN)
192
+    #define Z4_DIR_WRITE(STATE) WRITE(Z4_DIR_PIN,STATE)
193
+    #define Z4_DIR_READ() READ(Z4_DIR_PIN)
194
+  #endif
195
+  #define Z4_STEP_INIT SET_OUTPUT(Z4_STEP_PIN)
196
+  #ifndef Z4_STEP_WRITE
197
+    #define Z4_STEP_WRITE(STATE) WRITE(Z4_STEP_PIN,STATE)
198
+  #endif
199
+  #define Z4_STEP_READ READ(Z4_STEP_PIN)
200
+#else
201
+  #define Z4_DIR_WRITE(STATE) NOOP
202
+#endif
203
+
183
 // E0 Stepper
204
 // E0 Stepper
184
 #ifndef E0_ENABLE_INIT
205
 #ifndef E0_ENABLE_INIT
185
   #define E0_ENABLE_INIT() SET_OUTPUT(E0_ENABLE_PIN)
206
   #define E0_ENABLE_INIT() SET_OUTPUT(E0_ENABLE_PIN)
491
   #define Z3_disable() NOOP
512
   #define Z3_disable() NOOP
492
 #endif
513
 #endif
493
 
514
 
494
-#define  enable_Z() do{ Z_enable();  Z2_enable();  Z3_enable();  }while(0)
495
-#define disable_Z() do{ Z_disable(); Z2_disable(); Z3_disable(); CBI(axis_known_position, Z_AXIS); }while(0)
515
+#if AXIS_DRIVER_TYPE_Z4(L6470)
516
+  extern L6470 stepperZ4;
517
+  #define Z4_enable()  NOOP
518
+  #define Z4_disable() stepperZ4.free()
519
+#elif HAS_Z4_ENABLE
520
+  #define Z4_enable()  Z4_ENABLE_WRITE( Z_ENABLE_ON)
521
+  #define Z4_disable() Z4_ENABLE_WRITE(!Z_ENABLE_ON)
522
+#else
523
+  #define Z4_enable()  NOOP
524
+  #define Z4_disable() NOOP
525
+#endif
526
+
527
+#define  enable_Z() do{ Z_enable();  Z2_enable();  Z3_enable();  Z4_enable(); }while(0)
528
+#define disable_Z() do{ Z_disable(); Z2_disable(); Z3_disable(); Z4_disable(); CBI(axis_known_position, Z_AXIS); }while(0)
496
 
529
 
497
 //
530
 //
498
 // Extruder Stepper enable / disable
531
 // Extruder Stepper enable / disable

+ 26
- 0
Marlin/src/module/stepper/trinamic.cpp View File

88
 #if AXIS_HAS_SPI(Z3)
88
 #if AXIS_HAS_SPI(Z3)
89
   TMC_SPI_DEFINE(Z3, Z);
89
   TMC_SPI_DEFINE(Z3, Z);
90
 #endif
90
 #endif
91
+#if AXIS_HAS_SPI(Z4)
92
+  TMC_SPI_DEFINE(Z4, Z);
93
+#endif
91
 #if AXIS_HAS_SPI(E0)
94
 #if AXIS_HAS_SPI(E0)
92
   TMC_SPI_DEFINE_E(0);
95
   TMC_SPI_DEFINE_E(0);
93
 #endif
96
 #endif
249
       TMC_UART_DEFINE(SW, Z3, Z);
252
       TMC_UART_DEFINE(SW, Z3, Z);
250
     #endif
253
     #endif
251
   #endif
254
   #endif
255
+  #if AXIS_HAS_UART(Z4)
256
+    #ifdef Z4_HARDWARE_SERIAL
257
+      TMC_UART_DEFINE(HW, Z4, Z);
258
+    #else
259
+      TMC_UART_DEFINE(SW, Z4, Z);
260
+    #endif
261
+  #endif
252
   #if AXIS_HAS_UART(E0)
262
   #if AXIS_HAS_UART(E0)
253
     #ifdef E0_HARDWARE_SERIAL
263
     #ifdef E0_HARDWARE_SERIAL
254
       TMC_UART_DEFINE_E(HW, 0);
264
       TMC_UART_DEFINE_E(HW, 0);
342
         stepperZ3.beginSerial(TMC_BAUD_RATE);
352
         stepperZ3.beginSerial(TMC_BAUD_RATE);
343
       #endif
353
       #endif
344
     #endif
354
     #endif
355
+    #if AXIS_HAS_UART(Z4)
356
+      #ifdef Z4_HARDWARE_SERIAL
357
+        Z4_HARDWARE_SERIAL.begin(TMC_BAUD_RATE);
358
+      #else
359
+        stepperZ4.beginSerial(TMC_BAUD_RATE);
360
+      #endif
361
+    #endif
345
     #if AXIS_HAS_UART(E0)
362
     #if AXIS_HAS_UART(E0)
346
       #ifdef E0_HARDWARE_SERIAL
363
       #ifdef E0_HARDWARE_SERIAL
347
         E0_HARDWARE_SERIAL.begin(TMC_BAUD_RATE);
364
         E0_HARDWARE_SERIAL.begin(TMC_BAUD_RATE);
616
   #if AXIS_IS_TMC(Z3)
633
   #if AXIS_IS_TMC(Z3)
617
     stepperZ3.push();
634
     stepperZ3.push();
618
   #endif
635
   #endif
636
+  #if AXIS_IS_TMC(Z4)
637
+    stepperZ4.push();
638
+  #endif
619
   #if AXIS_IS_TMC(E0)
639
   #if AXIS_IS_TMC(E0)
620
     stepperE0.push();
640
     stepperE0.push();
621
   #endif
641
   #endif
678
   #if AXIS_IS_TMC(Z3)
698
   #if AXIS_IS_TMC(Z3)
679
     _TMC_INIT(Z3, STEALTH_AXIS_Z);
699
     _TMC_INIT(Z3, STEALTH_AXIS_Z);
680
   #endif
700
   #endif
701
+  #if AXIS_IS_TMC(Z4)
702
+    _TMC_INIT(Z4, STEALTH_AXIS_Z);
703
+  #endif
681
   #if AXIS_IS_TMC(E0)
704
   #if AXIS_IS_TMC(E0)
682
     _TMC_INIT(E0, STEALTH_AXIS_E);
705
     _TMC_INIT(E0, STEALTH_AXIS_E);
683
   #endif
706
   #endif
727
       #if AXIS_HAS_STALLGUARD(Z3)
750
       #if AXIS_HAS_STALLGUARD(Z3)
728
         stepperZ3.homing_threshold(Z_STALL_SENSITIVITY);
751
         stepperZ3.homing_threshold(Z_STALL_SENSITIVITY);
729
       #endif
752
       #endif
753
+      #if AXIS_HAS_STALLGUARD(Z4)
754
+        stepperZ4.homing_threshold(Z_STALL_SENSITIVITY);
755
+      #endif
730
     #endif
756
     #endif
731
   #endif
757
   #endif
732
 
758
 

+ 14
- 0
Marlin/src/module/stepper/trinamic.h View File

50
 #define TMC_Y2_LABEL 'Y', '2'
50
 #define TMC_Y2_LABEL 'Y', '2'
51
 #define TMC_Z2_LABEL 'Z', '2'
51
 #define TMC_Z2_LABEL 'Z', '2'
52
 #define TMC_Z3_LABEL 'Z', '3'
52
 #define TMC_Z3_LABEL 'Z', '3'
53
+#define TMC_Z4_LABEL 'Z', '4'
53
 
54
 
54
 #define TMC_E0_LABEL 'E', '0'
55
 #define TMC_E0_LABEL 'E', '0'
55
 #define TMC_E1_LABEL 'E', '1'
56
 #define TMC_E1_LABEL 'E', '1'
175
   #endif
176
   #endif
176
 #endif
177
 #endif
177
 
178
 
179
+// Z4 Stepper
180
+#if HAS_Z4_ENABLE && AXIS_IS_TMC(Z4)
181
+  extern TMC_CLASS(Z4, Z) stepperZ4;
182
+  #if ENABLED(SOFTWARE_DRIVER_ENABLE)
183
+    #define Z4_ENABLE_INIT() NOOP
184
+    #define Z4_ENABLE_WRITE(STATE) stepperZ4.toff((STATE)==Z_ENABLE_ON ? chopper_timing.toff : 0)
185
+    #define Z4_ENABLE_READ() stepperZ4.isEnabled()
186
+  #endif
187
+  #if AXIS_HAS_SQUARE_WAVE(Z4)
188
+    #define Z4_STEP_WRITE(STATE) do{ if(STATE) TOGGLE(Z4_STEP_PIN); }while(0)
189
+  #endif
190
+#endif
191
+
178
 // E0 Stepper
192
 // E0 Stepper
179
 #if AXIS_IS_TMC(E0)
193
 #if AXIS_IS_TMC(E0)
180
   extern TMC_CLASS_E(0) stepperE0;
194
   extern TMC_CLASS_E(0) stepperE0;

+ 49
- 2
Marlin/src/pins/pins.h View File

1153
 #endif
1153
 #endif
1154
 
1154
 
1155
 // The Z2 axis, if any, should be the next open extruder port
1155
 // The Z2 axis, if any, should be the next open extruder port
1156
-#if Z_MULTI_STEPPER_DRIVERS
1156
+#if NUM_Z_STEPPER_DRIVERS >= 2
1157
   #ifndef Z2_STEP_PIN
1157
   #ifndef Z2_STEP_PIN
1158
     #define Z2_STEP_PIN   _EPIN(Z2_E_INDEX, STEP)
1158
     #define Z2_STEP_PIN   _EPIN(Z2_E_INDEX, STEP)
1159
     #define Z2_DIR_PIN    _EPIN(Z2_E_INDEX, DIR)
1159
     #define Z2_DIR_PIN    _EPIN(Z2_E_INDEX, DIR)
1200
   #define Z2_MS3_PIN -1
1200
   #define Z2_MS3_PIN -1
1201
 #endif
1201
 #endif
1202
 
1202
 
1203
-#if ENABLED(Z_TRIPLE_STEPPER_DRIVERS)
1203
+#if NUM_Z_STEPPER_DRIVERS >= 3
1204
   #ifndef Z3_STEP_PIN
1204
   #ifndef Z3_STEP_PIN
1205
     #define Z3_STEP_PIN   _EPIN(Z3_E_INDEX, STEP)
1205
     #define Z3_STEP_PIN   _EPIN(Z3_E_INDEX, STEP)
1206
     #define Z3_DIR_PIN    _EPIN(Z3_E_INDEX, DIR)
1206
     #define Z3_DIR_PIN    _EPIN(Z3_E_INDEX, DIR)
1231
       #define Z3_SERIAL_RX_PIN _EPIN(Z3_E_INDEX, SERIAL_RX)
1231
       #define Z3_SERIAL_RX_PIN _EPIN(Z3_E_INDEX, SERIAL_RX)
1232
     #endif
1232
     #endif
1233
   #endif
1233
   #endif
1234
+  #define Z4_E_INDEX INCREMENT(Z3_E_INDEX)
1234
 #endif
1235
 #endif
1235
 
1236
 
1236
 #ifndef Z3_CS_PIN
1237
 #ifndef Z3_CS_PIN
1246
   #define Z3_MS3_PIN -1
1247
   #define Z3_MS3_PIN -1
1247
 #endif
1248
 #endif
1248
 
1249
 
1250
+#if NUM_Z_STEPPER_DRIVERS >= 4
1251
+  #ifndef Z4_STEP_PIN
1252
+    #define Z4_STEP_PIN   _EPIN(Z4_E_INDEX, STEP)
1253
+    #define Z4_DIR_PIN    _EPIN(Z4_E_INDEX, DIR)
1254
+    #define Z4_ENABLE_PIN _EPIN(Z4_E_INDEX, ENABLE)
1255
+    #if Z4_E_INDEX >= MAX_EXTRUDERS || !PIN_EXISTS(Z4_STEP)
1256
+      #error "No E stepper plug left for Z4!"
1257
+    #endif
1258
+  #endif
1259
+  #if AXIS_HAS_SPI(Z4)
1260
+    #ifndef Z4_CS_PIN
1261
+      #define Z4_CS_PIN     _EPIN(Z4_E_INDEX, CS)
1262
+    #endif
1263
+  #endif
1264
+  #ifndef Z4_MS1_PIN
1265
+    #define Z4_MS1_PIN    _EPIN(Z4_E_INDEX, MS1)
1266
+  #endif
1267
+  #ifndef Z4_MS2_PIN
1268
+    #define Z4_MS2_PIN    _EPIN(Z4_E_INDEX, MS2)
1269
+  #endif
1270
+  #ifndef Z4_MS3_PIN
1271
+    #define Z4_MS3_PIN    _EPIN(Z4_E_INDEX, MS3)
1272
+  #endif
1273
+  #if AXIS_HAS_UART(Z4)
1274
+    #ifndef Z4_SERIAL_TX_PIN
1275
+      #define Z4_SERIAL_TX_PIN _EPIN(Z4_E_INDEX, SERIAL_TX)
1276
+    #endif
1277
+    #ifndef Z4_SERIAL_RX_PIN
1278
+      #define Z4_SERIAL_RX_PIN _EPIN(Z4_E_INDEX, SERIAL_RX)
1279
+    #endif
1280
+  #endif
1281
+#endif
1282
+
1283
+#ifndef Z4_CS_PIN
1284
+  #define Z4_CS_PIN  -1
1285
+#endif
1286
+#ifndef Z4_MS1_PIN
1287
+  #define Z4_MS1_PIN -1
1288
+#endif
1289
+#ifndef Z4_MS2_PIN
1290
+  #define Z4_MS2_PIN -1
1291
+#endif
1292
+#ifndef Z4_MS3_PIN
1293
+  #define Z4_MS3_PIN -1
1294
+#endif
1295
+
1249
 #if HAS_GRAPHICAL_LCD
1296
 #if HAS_GRAPHICAL_LCD
1250
   #if !defined(ST7920_DELAY_1) && defined(BOARD_ST7920_DELAY_1)
1297
   #if !defined(ST7920_DELAY_1) && defined(BOARD_ST7920_DELAY_1)
1251
     #define ST7920_DELAY_1 BOARD_ST7920_DELAY_1
1298
     #define ST7920_DELAY_1 BOARD_ST7920_DELAY_1

+ 36
- 0
Marlin/src/pins/pinsDebug_list.h View File

90
 #if !PIN_EXISTS(Z3_MS3)
90
 #if !PIN_EXISTS(Z3_MS3)
91
   #undef Z3_MS3_PIN
91
   #undef Z3_MS3_PIN
92
 #endif
92
 #endif
93
+#if !PIN_EXISTS(Z4_MS1)
94
+  #undef Z4_MS1_PIN
95
+#endif
96
+#if !PIN_EXISTS(Z4_MS2)
97
+  #undef Z4_MS2_PIN
98
+#endif
99
+#if !PIN_EXISTS(Z4_MS3)
100
+  #undef Z4_MS3_PIN
101
+#endif
93
 #if !PIN_EXISTS(E0_MS1)
102
 #if !PIN_EXISTS(E0_MS1)
94
   #undef E0_MS1_PIN
103
   #undef E0_MS1_PIN
95
 #endif
104
 #endif
1335
 #if PIN_EXISTS(Z3_STEP)
1344
 #if PIN_EXISTS(Z3_STEP)
1336
   REPORT_NAME_DIGITAL(__LINE__, Z3_STEP_PIN)
1345
   REPORT_NAME_DIGITAL(__LINE__, Z3_STEP_PIN)
1337
 #endif
1346
 #endif
1347
+#if PIN_EXISTS(Z4_CS)
1348
+  REPORT_NAME_DIGITAL(__LINE__, Z4_CS_PIN)
1349
+#endif
1350
+#if PIN_EXISTS(Z4_DIR)
1351
+  REPORT_NAME_DIGITAL(__LINE__, Z4_DIR_PIN)
1352
+#endif
1353
+#if PIN_EXISTS(Z4_ENABLE)
1354
+  REPORT_NAME_DIGITAL(__LINE__, Z4_ENABLE_PIN)
1355
+#endif
1356
+#if PIN_EXISTS(Z4_MS1)
1357
+  REPORT_NAME_DIGITAL(__LINE__, Z4_MS1_PIN)
1358
+#endif
1359
+#if PIN_EXISTS(Z4_MS2)
1360
+  REPORT_NAME_DIGITAL(__LINE__, Z4_MS2_PIN)
1361
+#endif
1362
+#if PIN_EXISTS(Z4_MS3)
1363
+  REPORT_NAME_DIGITAL(__LINE__, Z4_MS3_PIN)
1364
+#endif
1365
+#if PIN_EXISTS(Z4_STEP)
1366
+  REPORT_NAME_DIGITAL(__LINE__, Z4_STEP_PIN)
1367
+#endif
1338
 #if PIN_EXISTS(ZRIB_V20_D6)
1368
 #if PIN_EXISTS(ZRIB_V20_D6)
1339
   REPORT_NAME_DIGITAL(__LINE__, ZRIB_V20_D6_PIN)
1369
   REPORT_NAME_DIGITAL(__LINE__, ZRIB_V20_D6_PIN)
1340
 #endif
1370
 #endif
1383
 #if PIN_EXISTS(Z3_SERIAL_RX)
1413
 #if PIN_EXISTS(Z3_SERIAL_RX)
1384
   REPORT_NAME_DIGITAL(__LINE__, Z3_SERIAL_RX_PIN)
1414
   REPORT_NAME_DIGITAL(__LINE__, Z3_SERIAL_RX_PIN)
1385
 #endif
1415
 #endif
1416
+#if PIN_EXISTS(Z4_SERIAL_TX)
1417
+  REPORT_NAME_DIGITAL(__LINE__, Z4_SERIAL_TX_PIN)
1418
+#endif
1419
+#if PIN_EXISTS(Z4_SERIAL_RX)
1420
+  REPORT_NAME_DIGITAL(__LINE__, Z4_SERIAL_RX_PIN)
1421
+#endif
1386
 #if PIN_EXISTS(E0_SERIAL_TX)
1422
 #if PIN_EXISTS(E0_SERIAL_TX)
1387
   REPORT_NAME_DIGITAL(__LINE__, E0_SERIAL_TX_PIN)
1423
   REPORT_NAME_DIGITAL(__LINE__, E0_SERIAL_TX_PIN)
1388
 #endif
1424
 #endif

+ 3
- 7
Marlin/src/pins/ramps/pins_RL200.h View File

31
 
31
 
32
 #if HOTENDS > 2 || E_STEPPERS > 2
32
 #if HOTENDS > 2 || E_STEPPERS > 2
33
   #error "RL200v1 supports up to 2 hotends / E-steppers. Comment out this line to continue."
33
   #error "RL200v1 supports up to 2 hotends / E-steppers. Comment out this line to continue."
34
-#endif
35
-
36
-#if DISABLED(Z_DUAL_STEPPER_DRIVERS)
37
-  #error "RL200 uses dual Z stepper motors. Update Configuration_adv.h or comment out this line to continue."
38
-#endif
39
-
40
-#if !(AXIS_DRIVER_TYPE_X(DRV8825) && AXIS_DRIVER_TYPE_Y(DRV8825) && AXIS_DRIVER_TYPE_Z(DRV8825) && AXIS_DRIVER_TYPE_Z2(DRV8825) && AXIS_DRIVER_TYPE_E0(DRV8825))
34
+#elif NUM_Z_STEPPER_DRIVERS != 2
35
+  #error "RL200 uses dual Z stepper motors. Set NUM_Z_STEPPER_DRIVERS to 2 or comment out this line to continue."
36
+#elif !(AXIS_DRIVER_TYPE_X(DRV8825) && AXIS_DRIVER_TYPE_Y(DRV8825) && AXIS_DRIVER_TYPE_Z(DRV8825) && AXIS_DRIVER_TYPE_Z2(DRV8825) && AXIS_DRIVER_TYPE_E0(DRV8825))
41
   #error "You must set ([XYZ]|Z2|E0)_DRIVER_TYPE to DRV8825 in Configuration.h for RL200."
37
   #error "You must set ([XYZ]|Z2|E0)_DRIVER_TYPE to DRV8825 in Configuration.h for RL200."
42
 #endif
38
 #endif
43
 
39
 

+ 31
- 5
Marlin/src/pins/sensitive_pins.h View File

423
   #define _Y2_PINS
423
   #define _Y2_PINS
424
 #endif
424
 #endif
425
 
425
 
426
-#if Z_MULTI_STEPPER_DRIVERS
426
+#if NUM_Z_STEPPER_DRIVERS >= 2
427
   #if PIN_EXISTS(Z2_CS) && AXIS_HAS_SPI(Z2)
427
   #if PIN_EXISTS(Z2_CS) && AXIS_HAS_SPI(Z2)
428
     #define _Z2_CS Z2_CS_PIN,
428
     #define _Z2_CS Z2_CS_PIN,
429
   #else
429
   #else
449
   #define _Z2_PINS
449
   #define _Z2_PINS
450
 #endif
450
 #endif
451
 
451
 
452
-#if ENABLED(Z_TRIPLE_STEPPER_DRIVERS)
452
+#if NUM_Z_STEPPER_DRIVERS >= 3
453
   #if PIN_EXISTS(Z3_CS) && AXIS_HAS_SPI(Z3)
453
   #if PIN_EXISTS(Z3_CS) && AXIS_HAS_SPI(Z3)
454
     #define _Z3_CS Z3_CS_PIN,
454
     #define _Z3_CS Z3_CS_PIN,
455
   #else
455
   #else
475
   #define _Z3_PINS
475
   #define _Z3_PINS
476
 #endif
476
 #endif
477
 
477
 
478
+#if NUM_Z_STEPPER_DRIVERS >= 4
479
+  #if PIN_EXISTS(Z4_CS) && AXIS_HAS_SPI(Z4)
480
+    #define _Z4_CS Z4_CS_PIN,
481
+  #else
482
+    #define _Z4_CS
483
+  #endif
484
+  #if PIN_EXISTS(Z4_MS1)
485
+    #define _Z4_MS1 Z4_MS1_PIN,
486
+  #else
487
+    #define _Z4_MS1
488
+  #endif
489
+  #if PIN_EXISTS(Z4_MS2)
490
+    #define _Z4_MS2 Z4_MS2_PIN,
491
+  #else
492
+    #define _Z4_MS2
493
+  #endif
494
+  #if PIN_EXISTS(Z4_MS3)
495
+    #define _Z4_MS3 Z4_MS3_PIN,
496
+  #else
497
+    #define _Z4_MS3
498
+  #endif
499
+  #define _Z4_PINS Z4_STEP_PIN, Z4_DIR_PIN, Z4_ENABLE_PIN, _Z4_CS _Z4_MS1 _Z4_MS2 _Z4_MS3
500
+#else
501
+  #define _Z4_PINS
502
+#endif
503
+
478
 //
504
 //
479
 // Generate the final Sensitive Pins array,
505
 // Generate the final Sensitive Pins array,
480
 // keeping the array as small as possible.
506
 // keeping the array as small as possible.
524
 #endif
550
 #endif
525
 
551
 
526
 #define SENSITIVE_PINS { \
552
 #define SENSITIVE_PINS { \
527
-  _X_PINS _Y_PINS _Z_PINS _X2_PINS _Y2_PINS _Z2_PINS _Z3_PINS _Z_PROBE \
528
-  _E0_PINS _E1_PINS _E2_PINS _E3_PINS _E4_PINS _E5_PINS _BED_PINS \
529
-  _H0_PINS _H1_PINS _H2_PINS _H3_PINS _H4_PINS _H5_PINS \
553
+  _X_PINS _Y_PINS _Z_PINS _X2_PINS _Y2_PINS _Z2_PINS _Z3_PINS _Z4_PINS \
554
+  _Z_PROBE _E0_PINS _E1_PINS _E2_PINS _E3_PINS _E4_PINS _E5_PINS \
555
+  _BED_PINS _H0_PINS _H1_PINS _H2_PINS _H3_PINS _H4_PINS _H5_PINS \
530
   _PS_ON _HEATER_BED _FAN0 _FAN1 _FAN2 _FANC \
556
   _PS_ON _HEATER_BED _FAN0 _FAN1 _FAN2 _FANC \
531
   HAL_SENSITIVE_PINS \
557
   HAL_SENSITIVE_PINS \
532
 }
558
 }

+ 5
- 5
buildroot/share/tests/DUE-tests View File

37
 restore_configs
37
 restore_configs
38
 opt_set MOTHERBOARD BOARD_RADDS
38
 opt_set MOTHERBOARD BOARD_RADDS
39
 opt_enable USE_XMAX_PLUG USE_YMAX_PLUG ENDSTOPPULLUPS BLTOUCH AUTO_BED_LEVELING_BILINEAR \
39
 opt_enable USE_XMAX_PLUG USE_YMAX_PLUG ENDSTOPPULLUPS BLTOUCH AUTO_BED_LEVELING_BILINEAR \
40
-           Z_TRIPLE_STEPPER_DRIVERS Z_TRIPLE_ENDSTOPS Z_STEPPER_AUTO_ALIGN \
41
-           Z_STEPPER_ALIGN_KNOWN_STEPPER_POSITIONS
40
+           Z_MULTI_ENDSTOPS Z_STEPPER_AUTO_ALIGN Z_STEPPER_ALIGN_KNOWN_STEPPER_POSITIONS
42
            #TOUCH_UI_FTDI_EVE LCD_ALEPHOBJECTS_CLCD_UI OTHER_PIN_LAYOUT
41
            #TOUCH_UI_FTDI_EVE LCD_ALEPHOBJECTS_CLCD_UI OTHER_PIN_LAYOUT
42
+opt_set NUM_Z_STEPPER_DRIVERS 3
43
 opt_add Z2_MAX_ENDSTOP_INVERTING false
43
 opt_add Z2_MAX_ENDSTOP_INVERTING false
44
 opt_add Z3_MAX_ENDSTOP_INVERTING false
44
 opt_add Z3_MAX_ENDSTOP_INVERTING false
45
-pins_set ramps/RAMPS X_MAX_PIN -1
46
-pins_set ramps/RAMPS Y_MAX_PIN -1
47
 opt_add Z2_MAX_PIN 2
45
 opt_add Z2_MAX_PIN 2
48
 opt_add Z3_MAX_PIN 3
46
 opt_add Z3_MAX_PIN 3
49
-exec_test $1 $2 "RADDS with ABL (Bilinear), Z_TRIPLE_STEPPER_DRIVERS and Z_STEPPER_AUTO_ALIGN"
47
+pins_set ramps/RAMPS X_MAX_PIN -1
48
+pins_set ramps/RAMPS Y_MAX_PIN -1
49
+exec_test $1 $2 "RADDS with ABL (Bilinear), Triple Z Axis, Z_STEPPER_AUTO_ALIGN"
50
 
50
 
51
 #
51
 #
52
 # Test SWITCHING_EXTRUDER
52
 # Test SWITCHING_EXTRUDER

+ 4
- 3
buildroot/share/tests/teensy35-tests View File

100
 exec_test $1 $2 "COREXZ"
100
 exec_test $1 $2 "COREXZ"
101
 
101
 
102
 #
102
 #
103
-# Enable Z_DUAL_STEPPER_DRIVERS, Z_DUAL_ENDSTOPS
103
+# Enable Dual Z with Dual Z endstops
104
 #
104
 #
105
 restore_configs
105
 restore_configs
106
 opt_set MOTHERBOARD BOARD_TEENSY35_36
106
 opt_set MOTHERBOARD BOARD_TEENSY35_36
107
-opt_enable Z_DUAL_STEPPER_DRIVERS Z_DUAL_ENDSTOPS
107
+opt_enable Z_MULTI_ENDSTOPS
108
+opt_set NUM_Z_STEPPER_DRIVERS 2
108
 pins_set ramps/RAMPS X_MAX_PIN -1
109
 pins_set ramps/RAMPS X_MAX_PIN -1
109
 opt_add Z2_MAX_PIN 2
110
 opt_add Z2_MAX_PIN 2
110
 opt_enable USE_XMAX_PLUG
111
 opt_enable USE_XMAX_PLUG
111
-exec_test $1 $2 "Z_DUAL_STEPPER_DRIVERS, Z_DUAL_ENDSTOPS"
112
+exec_test $1 $2 "Dual Z with Dual Z endstops"
112
 
113
 
113
 # Clean up
114
 # Clean up
114
 restore_configs
115
 restore_configs

Loading…
Cancel
Save