My Marlin configs for Fabrikator Mini and CTC i3 Pro B
Você não pode selecionar mais de 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.

L64XX_Marlin.cpp 32KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914
  1. /**
  2. * Marlin 3D Printer Firmware
  3. * Copyright (c) 2019 MarlinFirmware [https://github.com/MarlinFirmware/Marlin]
  4. *
  5. * Based on Sprinter and grbl.
  6. * Copyright (c) 2011 Camiel Gubbels / Erik van der Zalm
  7. *
  8. * This program is free software: you can redistribute it and/or modify
  9. * it under the terms of the GNU General Public License as published by
  10. * the Free Software Foundation, either version 3 of the License, or
  11. * (at your option) any later version.
  12. *
  13. * This program is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU General Public License
  19. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  20. *
  21. */
  22. /**
  23. * The monitor_driver routines are a close copy of the TMC code
  24. */
  25. #include "../../inc/MarlinConfig.h"
  26. #if HAS_L64XX
  27. #include "L64XX_Marlin.h"
  28. L64XX_Marlin L64xxManager;
  29. #include "../../module/stepper/indirection.h"
  30. #include "../../gcode/gcode.h"
  31. #include "../../module/planner.h"
  32. #include "../../HAL/shared/Delay.h"
  33. void echo_yes_no(const bool yes) { serialprintPGM(yes ? PSTR(" YES") : PSTR(" NO ")); }
  34. char L64XX_Marlin::index_to_axis[MAX_L64XX][3] = { "X ", "Y ", "Z ", "X2", "Y2", "Z2", "Z3", "Z4", "E0", "E1", "E2", "E3", "E4", "E5", "E6", "E7" };
  35. #define DEBUG_OUT ENABLED(L6470_CHITCHAT)
  36. #include "../../core/debug_out.h"
  37. uint8_t L64XX_Marlin::dir_commands[MAX_L64XX]; // array to hold direction command for each driver
  38. uint8_t L64XX_Marlin::index_to_dir[MAX_L64XX] = { (INVERT_X_DIR), // 0 X
  39. (INVERT_Y_DIR), // 1 Y
  40. (INVERT_Z_DIR), // 2 Z
  41. #if ENABLED(X_DUAL_STEPPER_DRIVERS) // 3 X2
  42. (INVERT_X_DIR) ^ (INVERT_X2_VS_X_DIR),
  43. #else
  44. (INVERT_X_DIR),
  45. #endif
  46. #if ENABLED(Y_DUAL_STEPPER_DRIVERS) // 4 Y2
  47. (INVERT_Y_DIR) ^ (INVERT_Y2_VS_Y_DIR),
  48. #else
  49. (INVERT_Y_DIR),
  50. #endif
  51. (INVERT_Z_DIR), // 5 Z2
  52. (INVERT_Z_DIR), // 6 Z3
  53. (INVERT_Z_DIR), // 7 Z4
  54. (INVERT_E0_DIR), // 8 E0
  55. (INVERT_E1_DIR), // 9 E1
  56. (INVERT_E2_DIR), // 10 E2
  57. (INVERT_E3_DIR), // 11 E3
  58. (INVERT_E4_DIR), // 12 E4
  59. (INVERT_E5_DIR), // 13 E5
  60. (INVERT_E6_DIR), // 14 E6
  61. (INVERT_E7_DIR) // 15 E7
  62. };
  63. volatile uint8_t L64XX_Marlin::spi_abort = false;
  64. uint8_t L64XX_Marlin::spi_active = false;
  65. L64XX_Marlin::L64XX_shadow_t L64XX_Marlin::shadow;
  66. //uint32_t UVLO_ADC = 0x0400; // ADC undervoltage event
  67. void L6470_populate_chain_array() {
  68. #define _L6470_INIT_SPI(Q) do{ stepper##Q.set_chain_info(Q, Q##_CHAIN_POS); }while(0)
  69. #if AXIS_IS_L64XX(X)
  70. _L6470_INIT_SPI(X);
  71. #endif
  72. #if AXIS_IS_L64XX(X2)
  73. _L6470_INIT_SPI(X2);
  74. #endif
  75. #if AXIS_IS_L64XX(Y)
  76. _L6470_INIT_SPI(Y);
  77. #endif
  78. #if AXIS_IS_L64XX(Y2)
  79. _L6470_INIT_SPI(Y2);
  80. #endif
  81. #if AXIS_IS_L64XX(Z)
  82. _L6470_INIT_SPI(Z);
  83. #endif
  84. #if AXIS_IS_L64XX(Z2)
  85. _L6470_INIT_SPI(Z2);
  86. #endif
  87. #if AXIS_IS_L64XX(Z3)
  88. _L6470_INIT_SPI(Z3);
  89. #endif
  90. #if AXIS_IS_L64XX(Z4)
  91. _L6470_INIT_SPI(Z4);
  92. #endif
  93. #if AXIS_IS_L64XX(E0)
  94. _L6470_INIT_SPI(E0);
  95. #endif
  96. #if AXIS_IS_L64XX(E1)
  97. _L6470_INIT_SPI(E1);
  98. #endif
  99. #if AXIS_IS_L64XX(E2)
  100. _L6470_INIT_SPI(E2);
  101. #endif
  102. #if AXIS_IS_L64XX(E3)
  103. _L6470_INIT_SPI(E3);
  104. #endif
  105. #if AXIS_IS_L64XX(E4)
  106. _L6470_INIT_SPI(E4);
  107. #endif
  108. #if AXIS_IS_L64XX(E5)
  109. _L6470_INIT_SPI(E5);
  110. #endif
  111. }
  112. /**
  113. * Some status bit positions & definitions differ per driver.
  114. * Copy info to known locations to simplfy check/display logic.
  115. * 1. Copy stepper status
  116. * 2. Copy status bit definitions
  117. * 3. Copy status layout
  118. * 4. Make all error bits active low (as needed)
  119. */
  120. uint16_t L64XX_Marlin::get_stepper_status(L64XX &st) {
  121. shadow.STATUS_AXIS_RAW = st.getStatus();
  122. shadow.STATUS_AXIS = shadow.STATUS_AXIS_RAW;
  123. shadow.STATUS_AXIS_LAYOUT = st.L6470_status_layout;
  124. shadow.AXIS_OCD_TH_MAX = st.OCD_TH_MAX;
  125. shadow.AXIS_STALL_TH_MAX = st.STALL_TH_MAX;
  126. shadow.AXIS_OCD_CURRENT_CONSTANT_INV = st.OCD_CURRENT_CONSTANT_INV;
  127. shadow.AXIS_STALL_CURRENT_CONSTANT_INV = st.STALL_CURRENT_CONSTANT_INV;
  128. shadow.L6470_AXIS_CONFIG = st.L64XX_CONFIG;
  129. shadow.L6470_AXIS_STATUS = st.L64XX_STATUS;
  130. shadow.STATUS_AXIS_OCD = st.STATUS_OCD;
  131. shadow.STATUS_AXIS_SCK_MOD = st.STATUS_SCK_MOD;
  132. shadow.STATUS_AXIS_STEP_LOSS_A = st.STATUS_STEP_LOSS_A;
  133. shadow.STATUS_AXIS_STEP_LOSS_B = st.STATUS_STEP_LOSS_B;
  134. shadow.STATUS_AXIS_TH_SD = st.STATUS_TH_SD;
  135. shadow.STATUS_AXIS_TH_WRN = st.STATUS_TH_WRN;
  136. shadow.STATUS_AXIS_UVLO = st.STATUS_UVLO;
  137. shadow.STATUS_AXIS_WRONG_CMD = st.STATUS_WRONG_CMD;
  138. shadow.STATUS_AXIS_CMD_ERR = st.STATUS_CMD_ERR;
  139. shadow.STATUS_AXIS_NOTPERF_CMD = st.STATUS_NOTPERF_CMD;
  140. switch (shadow.STATUS_AXIS_LAYOUT) {
  141. case L6470_STATUS_LAYOUT: { // L6470
  142. shadow.L6470_ERROR_MASK = shadow.STATUS_AXIS_UVLO | shadow.STATUS_AXIS_TH_WRN | shadow.STATUS_AXIS_TH_SD | shadow.STATUS_AXIS_OCD | shadow.STATUS_AXIS_STEP_LOSS_A | shadow.STATUS_AXIS_STEP_LOSS_B;
  143. shadow.STATUS_AXIS ^= (shadow.STATUS_AXIS_WRONG_CMD | shadow.STATUS_AXIS_NOTPERF_CMD); // invert just error bits that are active high
  144. break;
  145. }
  146. case L6474_STATUS_LAYOUT: { // L6474
  147. shadow.L6470_ERROR_MASK = shadow.STATUS_AXIS_UVLO | shadow.STATUS_AXIS_TH_WRN | shadow.STATUS_AXIS_TH_SD | shadow.STATUS_AXIS_OCD ;
  148. shadow.STATUS_AXIS ^= (shadow.STATUS_AXIS_WRONG_CMD | shadow.STATUS_AXIS_NOTPERF_CMD); // invert just error bits that are active high
  149. break;
  150. }
  151. case L6480_STATUS_LAYOUT: { // L6480 & powerSTEP01
  152. shadow.L6470_ERROR_MASK = shadow.STATUS_AXIS_UVLO | shadow.STATUS_AXIS_TH_WRN | shadow.STATUS_AXIS_TH_SD | shadow.STATUS_AXIS_OCD | shadow.STATUS_AXIS_STEP_LOSS_A | shadow.STATUS_AXIS_STEP_LOSS_B;
  153. shadow.STATUS_AXIS ^= (shadow.STATUS_AXIS_CMD_ERR | shadow.STATUS_AXIS_TH_WRN | shadow.STATUS_AXIS_TH_SD); // invert just error bits that are active high
  154. break;
  155. }
  156. }
  157. return shadow.STATUS_AXIS;
  158. }
  159. void L64XX_Marlin::init() { // Set up SPI and then init chips
  160. ENABLE_RESET_L64XX_CHIPS(LOW); // hardware reset of drivers
  161. DELAY_US(100);
  162. ENABLE_RESET_L64XX_CHIPS(HIGH);
  163. DELAY_US(1000); // need about 650µs for the chip(s) to fully start up
  164. L6470_populate_chain_array(); // Set up array to control where in the SPI transfer sequence a particular stepper's data goes
  165. spi_init(); // Since L64XX SPI pins are unset we must init SPI here
  166. init_to_defaults(); // init the chips
  167. }
  168. uint16_t L64XX_Marlin::get_status(const L64XX_axis_t axis) {
  169. #define STATUS_L6470(Q) get_stepper_status(stepper##Q)
  170. switch (axis) {
  171. default: break;
  172. #if AXIS_IS_L64XX(X)
  173. case X : return STATUS_L6470(X);
  174. #endif
  175. #if AXIS_IS_L64XX(Y)
  176. case Y : return STATUS_L6470(Y);
  177. #endif
  178. #if AXIS_IS_L64XX(Z)
  179. case Z : return STATUS_L6470(Z);
  180. #endif
  181. #if AXIS_IS_L64XX(X2)
  182. case X2: return STATUS_L6470(X2);
  183. #endif
  184. #if AXIS_IS_L64XX(Y2)
  185. case Y2: return STATUS_L6470(Y2);
  186. #endif
  187. #if AXIS_IS_L64XX(Z2)
  188. case Z2: return STATUS_L6470(Z2);
  189. #endif
  190. #if AXIS_IS_L64XX(Z3)
  191. case Z3: return STATUS_L6470(Z3);
  192. #endif
  193. #if AXIS_IS_L64XX(Z4)
  194. case Z4: return STATUS_L6470(Z4);
  195. #endif
  196. #if AXIS_IS_L64XX(E0)
  197. case E0: return STATUS_L6470(E0);
  198. #endif
  199. #if AXIS_IS_L64XX(E1)
  200. case E1: return STATUS_L6470(E1);
  201. #endif
  202. #if AXIS_IS_L64XX(E2)
  203. case E2: return STATUS_L6470(E2);
  204. #endif
  205. #if AXIS_IS_L64XX(E3)
  206. case E3: return STATUS_L6470(E3);
  207. #endif
  208. #if AXIS_IS_L64XX(E4)
  209. case E4: return STATUS_L6470(E4);
  210. #endif
  211. #if AXIS_IS_L64XX(E5)
  212. case E5: return STATUS_L6470(E5);
  213. #endif
  214. }
  215. return 0; // Not needed but kills a compiler warning
  216. }
  217. uint32_t L64XX_Marlin::get_param(const L64XX_axis_t axis, const uint8_t param) {
  218. #define GET_L6470_PARAM(Q) L6470_GETPARAM(param, Q)
  219. switch (axis) {
  220. default: break;
  221. #if AXIS_IS_L64XX(X)
  222. case X : return GET_L6470_PARAM(X);
  223. #endif
  224. #if AXIS_IS_L64XX(Y)
  225. case Y : return GET_L6470_PARAM(Y);
  226. #endif
  227. #if AXIS_IS_L64XX(Z)
  228. case Z : return GET_L6470_PARAM(Z);
  229. #endif
  230. #if AXIS_IS_L64XX(X2)
  231. case X2: return GET_L6470_PARAM(X2);
  232. #endif
  233. #if AXIS_IS_L64XX(Y2)
  234. case Y2: return GET_L6470_PARAM(Y2);
  235. #endif
  236. #if AXIS_IS_L64XX(Z2)
  237. case Z2: return GET_L6470_PARAM(Z2);
  238. #endif
  239. #if AXIS_IS_L64XX(Z3)
  240. case Z3: return GET_L6470_PARAM(Z3);
  241. #endif
  242. #if AXIS_IS_L64XX(Z4)
  243. case Z4: return GET_L6470_PARAM(Z4);
  244. #endif
  245. #if AXIS_IS_L64XX(E0)
  246. case E0: return GET_L6470_PARAM(E0);
  247. #endif
  248. #if AXIS_IS_L64XX(E1)
  249. case E1: return GET_L6470_PARAM(E1);
  250. #endif
  251. #if AXIS_IS_L64XX(E2)
  252. case E2: return GET_L6470_PARAM(E2);
  253. #endif
  254. #if AXIS_IS_L64XX(E3)
  255. case E3: return GET_L6470_PARAM(E3);
  256. #endif
  257. #if AXIS_IS_L64XX(E4)
  258. case E4: return GET_L6470_PARAM(E4);
  259. #endif
  260. #if AXIS_IS_L64XX(E5)
  261. case E5: return GET_L6470_PARAM(E5);
  262. #endif
  263. }
  264. return 0; // not needed but kills a compiler warning
  265. }
  266. void L64XX_Marlin::set_param(const L64XX_axis_t axis, const uint8_t param, const uint32_t value) {
  267. #define SET_L6470_PARAM(Q) stepper##Q.SetParam(param, value)
  268. switch (axis) {
  269. default: break;
  270. #if AXIS_IS_L64XX(X)
  271. case X : SET_L6470_PARAM(X); break;
  272. #endif
  273. #if AXIS_IS_L64XX(Y)
  274. case Y : SET_L6470_PARAM(Y); break;
  275. #endif
  276. #if AXIS_IS_L64XX(Z)
  277. case Z : SET_L6470_PARAM(Z); break;
  278. #endif
  279. #if AXIS_IS_L64XX(X2)
  280. case X2: SET_L6470_PARAM(X2); break;
  281. #endif
  282. #if AXIS_IS_L64XX(Y2)
  283. case Y2: SET_L6470_PARAM(Y2); break;
  284. #endif
  285. #if AXIS_IS_L64XX(Z2)
  286. case Z2: SET_L6470_PARAM(Z2); break;
  287. #endif
  288. #if AXIS_IS_L64XX(Z3)
  289. case Z3: SET_L6470_PARAM(Z3); break;
  290. #endif
  291. #if AXIS_IS_L64XX(Z4)
  292. case Z4: SET_L6470_PARAM(Z4); break;
  293. #endif
  294. #if AXIS_IS_L64XX(E0)
  295. case E0: SET_L6470_PARAM(E0); break;
  296. #endif
  297. #if AXIS_IS_L64XX(E1)
  298. case E1: SET_L6470_PARAM(E1); break;
  299. #endif
  300. #if AXIS_IS_L64XX(E2)
  301. case E2: SET_L6470_PARAM(E2); break;
  302. #endif
  303. #if AXIS_IS_L64XX(E3)
  304. case E3: SET_L6470_PARAM(E3); break;
  305. #endif
  306. #if AXIS_IS_L64XX(E4)
  307. case E4: SET_L6470_PARAM(E4); break;
  308. #endif
  309. #if AXIS_IS_L64XX(E5)
  310. case E5: SET_L6470_PARAM(E5); break;
  311. #endif
  312. }
  313. }
  314. inline void echo_min_max(const char a, const float &min, const float &max) {
  315. DEBUG_CHAR(' '); DEBUG_CHAR(a);
  316. DEBUG_ECHOPAIR(" min = ", min);
  317. DEBUG_ECHOLNPAIR(" max = ", max);
  318. }
  319. inline void echo_oct_used(const float &oct, const uint8_t stall) {
  320. DEBUG_ECHOPAIR("over_current_threshold used : ", oct);
  321. serialprintPGM(stall ? PSTR(" (Stall") : PSTR(" (OCD"));
  322. DEBUG_ECHOLNPGM(" threshold)");
  323. }
  324. inline void err_out_of_bounds() { DEBUG_ECHOLNPGM("Test aborted - motion out of bounds"); }
  325. uint8_t L64XX_Marlin::get_user_input(uint8_t &driver_count, L64XX_axis_t axis_index[3], char axis_mon[3][3],
  326. float &position_max, float &position_min, float &final_feedrate, uint8_t &kval_hold,
  327. uint8_t over_current_flag, uint8_t &OCD_TH_val, uint8_t &STALL_TH_val, uint16_t &over_current_threshold
  328. ) {
  329. // Return TRUE if the calling routine needs to abort/kill
  330. uint16_t displacement = 0; // " = 0" to eliminate compiler warning
  331. uint8_t j; // general purpose counter
  332. if (!all_axes_homed()) {
  333. DEBUG_ECHOLNPGM("Test aborted - home all before running this command");
  334. return true;
  335. }
  336. uint8_t found_displacement = false;
  337. LOOP_XYZE(i) if (uint16_t _displacement = parser.intval(axis_codes[i])) {
  338. found_displacement = true;
  339. displacement = _displacement;
  340. uint8_t axis_offset = parser.byteval('J');
  341. axis_mon[0][0] = axis_codes[i]; // axis ASCII value (target character)
  342. uint8_t driver_count_local = 0; // Can't use "driver_count" directly as a subscript because it's passed by reference
  343. if (axis_offset >= 2 || axis_mon[0][0] == 'E') { // Single axis, E0, or E1
  344. axis_mon[0][1] = axis_offset + '0';
  345. for (j = 0; j < MAX_L64XX; j++) { // See how many drivers on this axis
  346. const char * const str = index_to_axis[j];
  347. if (axis_mon[0][0] == str[0]) {
  348. char * const mon = axis_mon[driver_count_local];
  349. mon[0] = str[0];
  350. mon[1] = str[1];
  351. mon[2] = str[2]; // append end of string
  352. axis_index[driver_count_local] = (L64XX_axis_t)j; // set axis index
  353. driver_count_local++;
  354. }
  355. }
  356. }
  357. else if (axis_offset == 0) { // One or more axes
  358. for (j = 0; j < MAX_L64XX; j++) { // See how many drivers on this axis
  359. const char * const str = index_to_axis[j];
  360. if (axis_mon[0][0] == str[0]) {
  361. char * const mon = axis_mon[driver_count_local];
  362. mon[0] = str[0];
  363. mon[1] = str[1];
  364. mon[2] = str[2]; // append end of string
  365. axis_index[driver_count_local] = (L64XX_axis_t)j; // set axis index
  366. driver_count_local++;
  367. }
  368. }
  369. driver_count = driver_count_local;
  370. }
  371. break; // only take first axis found
  372. }
  373. if (!found_displacement) {
  374. DEBUG_ECHOLNPGM("Test aborted - AXIS with displacement is required");
  375. return true;
  376. }
  377. //
  378. // Position calcs & checks
  379. //
  380. const float X_center = LOGICAL_X_POSITION(current_position.x),
  381. Y_center = LOGICAL_Y_POSITION(current_position.y),
  382. Z_center = LOGICAL_Z_POSITION(current_position.z),
  383. E_center = current_position.e;
  384. switch (axis_mon[0][0]) {
  385. default: position_max = position_min = 0; break;
  386. case 'X': {
  387. position_min = X_center - displacement;
  388. position_max = X_center + displacement;
  389. echo_min_max('X', position_min, position_max);
  390. if (false
  391. #ifdef X_MIN_POS
  392. || position_min < (X_MIN_POS)
  393. #endif
  394. #ifdef X_MAX_POS
  395. || position_max > (X_MAX_POS)
  396. #endif
  397. ) {
  398. err_out_of_bounds();
  399. return true;
  400. }
  401. } break;
  402. case 'Y': {
  403. position_min = Y_center - displacement;
  404. position_max = Y_center + displacement;
  405. echo_min_max('Y', position_min, position_max);
  406. if (false
  407. #ifdef Y_MIN_POS
  408. || position_min < (Y_MIN_POS)
  409. #endif
  410. #ifdef Y_MAX_POS
  411. || position_max > (Y_MAX_POS)
  412. #endif
  413. ) {
  414. err_out_of_bounds();
  415. return true;
  416. }
  417. } break;
  418. case 'Z': {
  419. position_min = Z_center - displacement;
  420. position_max = Z_center + displacement;
  421. echo_min_max('Z', position_min, position_max);
  422. if (false
  423. #ifdef Z_MIN_POS
  424. || position_min < (Z_MIN_POS)
  425. #endif
  426. #ifdef Z_MAX_POS
  427. || position_max > (Z_MAX_POS)
  428. #endif
  429. ) {
  430. err_out_of_bounds();
  431. return true;
  432. }
  433. } break;
  434. case 'E': {
  435. position_min = E_center - displacement;
  436. position_max = E_center + displacement;
  437. echo_min_max('E', position_min, position_max);
  438. } break;
  439. }
  440. //
  441. // Work on the drivers
  442. //
  443. for (uint8_t k = 0; k < driver_count; k++) {
  444. uint8_t not_found = true;
  445. for (j = 1; j <= L64XX::chain[0]; j++) {
  446. const char * const ind_axis = index_to_axis[L64XX::chain[j]];
  447. if (ind_axis[0] == axis_mon[k][0] && ind_axis[1] == axis_mon[k][1]) { // See if a L6470 driver
  448. not_found = false;
  449. break;
  450. }
  451. }
  452. if (not_found) {
  453. driver_count = k;
  454. axis_mon[k][0] = ' '; // mark this entry invalid
  455. break;
  456. }
  457. }
  458. if (driver_count == 0) {
  459. DEBUG_ECHOLNPGM("Test aborted - not a L6470 axis");
  460. return true;
  461. }
  462. DEBUG_ECHOPGM("Monitoring:");
  463. for (j = 0; j < driver_count; j++) DEBUG_ECHOPAIR(" ", axis_mon[j]);
  464. DEBUG_EOL();
  465. // now have a list of driver(s) to monitor
  466. //
  467. // TVAL & kVAL_HOLD checks & settings
  468. //
  469. const L64XX_shadow_t &sh = shadow;
  470. get_status(axis_index[0]); // populate shadow array
  471. if (sh.STATUS_AXIS_LAYOUT == L6474_STATUS_LAYOUT) { // L6474 - use TVAL
  472. uint16_t TVAL_current = parser.ushortval('T');
  473. if (TVAL_current) {
  474. uint8_t TVAL_count = (TVAL_current / sh.AXIS_STALL_CURRENT_CONSTANT_INV) - 1;
  475. LIMIT(TVAL_count, 0, sh.AXIS_STALL_TH_MAX);
  476. for (j = 0; j < driver_count; j++)
  477. set_param(axis_index[j], L6474_TVAL, TVAL_count);
  478. }
  479. // only print the tval from one of the drivers
  480. kval_hold = get_param(axis_index[0], L6474_TVAL);
  481. DEBUG_ECHOLNPAIR("TVAL current (mA) = ", (kval_hold + 1) * sh.AXIS_STALL_CURRENT_CONSTANT_INV);
  482. }
  483. else {
  484. kval_hold = parser.byteval('K');
  485. if (kval_hold) {
  486. DEBUG_ECHOLNPAIR("kval_hold = ", kval_hold);
  487. for (j = 0; j < driver_count; j++)
  488. set_param(axis_index[j], L6470_KVAL_HOLD, kval_hold);
  489. }
  490. else {
  491. // only print the KVAL_HOLD from one of the drivers
  492. kval_hold = get_param(axis_index[0], L6470_KVAL_HOLD);
  493. DEBUG_ECHOLNPAIR("KVAL_HOLD = ", kval_hold);
  494. }
  495. }
  496. //
  497. // Overcurrent checks & settings
  498. //
  499. if (over_current_flag) {
  500. uint8_t OCD_TH_val_local = 0, // compiler thinks OCD_TH_val is unused if use it directly
  501. STALL_TH_val_local = 0; // just in case ...
  502. over_current_threshold = parser.intval('I');
  503. if (over_current_threshold) {
  504. OCD_TH_val_local = over_current_threshold/375;
  505. LIMIT(OCD_TH_val_local, 0, 15);
  506. STALL_TH_val_local = over_current_threshold/31.25;
  507. LIMIT(STALL_TH_val_local, 0, 127);
  508. uint16_t OCD_TH_actual = (OCD_TH_val_local + 1) * 375,
  509. STALL_TH_actual = (STALL_TH_val_local + 1) * 31.25;
  510. if (OCD_TH_actual < STALL_TH_actual) {
  511. OCD_TH_val_local++;
  512. OCD_TH_actual = (OCD_TH_val_local + 1) * 375;
  513. }
  514. DEBUG_ECHOLNPAIR("over_current_threshold specified: ", over_current_threshold);
  515. if (!(sh.STATUS_AXIS_LAYOUT == L6474_STATUS_LAYOUT)) echo_oct_used((STALL_TH_val_local + 1) * 31.25, true);
  516. echo_oct_used((OCD_TH_val_local + 1) * 375, false);
  517. #define SET_OVER_CURRENT(Q) do { stepper##Q.SetParam(L6470_STALL_TH, STALL_TH_val_local); stepper##Q.SetParam(L6470_OCD_TH, OCD_TH_val_local);} while (0)
  518. for (j = 0; j < driver_count; j++) {
  519. set_param(axis_index[j], L6470_STALL_TH, STALL_TH_val_local);
  520. set_param(axis_index[j], L6470_OCD_TH, OCD_TH_val_local);
  521. }
  522. }
  523. else {
  524. // only get & print the OVER_CURRENT values from one of the drivers
  525. STALL_TH_val_local = get_param(axis_index[0], L6470_STALL_TH);
  526. OCD_TH_val_local = get_param(axis_index[0], L6470_OCD_TH);
  527. if (!(sh.STATUS_AXIS_LAYOUT == L6474_STATUS_LAYOUT)) echo_oct_used((STALL_TH_val_local + 1) * 31.25, true);
  528. echo_oct_used((OCD_TH_val_local + 1) * 375, false);
  529. } // over_current_threshold
  530. for (j = 0; j < driver_count; j++) { // set all drivers on axis the same
  531. set_param(axis_index[j], L6470_STALL_TH, STALL_TH_val_local);
  532. set_param(axis_index[j], L6470_OCD_TH, OCD_TH_val_local);
  533. }
  534. OCD_TH_val = OCD_TH_val_local; // force compiler to update the main routine's copy
  535. STALL_TH_val = STALL_TH_val_local; // force compiler to update the main routine's copy
  536. } // end of overcurrent
  537. //
  538. // Feedrate
  539. //
  540. final_feedrate = parser.floatval('F');
  541. if (final_feedrate == 0) {
  542. static constexpr float default_max_feedrate[] = DEFAULT_MAX_FEEDRATE;
  543. const uint8_t num_feedrates = COUNT(default_max_feedrate);
  544. for (j = 0; j < num_feedrates; j++) {
  545. if (axis_codes[j] == axis_mon[0][0]) {
  546. final_feedrate = default_max_feedrate[j];
  547. break;
  548. }
  549. }
  550. if (j == 3 && num_feedrates > 4) { // have more than one extruder feedrate
  551. uint8_t extruder_num = axis_mon[0][1] - '0';
  552. if (j <= num_feedrates - extruder_num) // have a feedrate specifically for this extruder
  553. final_feedrate = default_max_feedrate[j + extruder_num];
  554. else
  555. final_feedrate = default_max_feedrate[3]; // use E0 feedrate for this extruder
  556. }
  557. final_feedrate *= 60; // convert to mm/minute
  558. } // end of feedrate
  559. return false; // FALSE indicates no user input problems
  560. }
  561. void L64XX_Marlin::say_axis(const L64XX_axis_t axis, const uint8_t label/*=true*/) {
  562. if (label) SERIAL_ECHOPGM("AXIS:");
  563. const char * const str = L64xxManager.index_to_axis[axis];
  564. SERIAL_CHAR(' ', str[0], str[1], ' ');
  565. }
  566. #if ENABLED(L6470_CHITCHAT)
  567. // Assumes status bits have been inverted
  568. void L64XX_Marlin::error_status_decode(const uint16_t status, const L64XX_axis_t axis,
  569. const uint16_t _status_axis_th_sd, const uint16_t _status_axis_th_wrn,
  570. const uint16_t _status_axis_step_loss_a, const uint16_t _status_axis_step_loss_b,
  571. const uint16_t _status_axis_ocd, const uint8_t _status_axis_layout
  572. ) {
  573. say_axis(axis);
  574. DEBUG_ECHOPGM(" THERMAL: ");
  575. serialprintPGM((status & _status_axis_th_sd) ? PSTR("SHUTDOWN") : (status & _status_axis_th_wrn) ? PSTR("WARNING ") : PSTR("OK "));
  576. DEBUG_ECHOPGM(" OVERCURRENT: ");
  577. echo_yes_no((status & _status_axis_ocd) != 0);
  578. if (!(_status_axis_layout == L6474_STATUS_LAYOUT)) { // L6474 doesn't have these bits
  579. DEBUG_ECHOPGM(" STALL: ");
  580. echo_yes_no((status & (_status_axis_step_loss_a | _status_axis_step_loss_b)) != 0);
  581. }
  582. DEBUG_EOL();
  583. }
  584. #endif
  585. //////////////////////////////////////////////////////////////////////////////////////////////////
  586. ////
  587. //// MONITOR_L6470_DRIVER_STATUS routines
  588. ////
  589. //////////////////////////////////////////////////////////////////////////////////////////////////
  590. #if ENABLED(MONITOR_L6470_DRIVER_STATUS)
  591. bool L64XX_Marlin::monitor_paused = false; // Flag to skip monitor during M122, M906, M916, M917, M918, etc.
  592. struct L6470_driver_data {
  593. uint8_t driver_index;
  594. uint32_t driver_status;
  595. uint8_t is_otw;
  596. uint8_t otw_counter;
  597. uint8_t is_ot;
  598. uint8_t is_hi_Z;
  599. uint8_t com_counter;
  600. };
  601. L6470_driver_data driver_L6470_data[] = {
  602. #if AXIS_IS_L64XX(X)
  603. { 0, 0, 0, 0, 0, 0, 0 },
  604. #endif
  605. #if AXIS_IS_L64XX(Y)
  606. { 1, 0, 0, 0, 0, 0, 0 },
  607. #endif
  608. #if AXIS_IS_L64XX(Z)
  609. { 2, 0, 0, 0, 0, 0, 0 },
  610. #endif
  611. #if AXIS_IS_L64XX(X2)
  612. { 3, 0, 0, 0, 0, 0, 0 },
  613. #endif
  614. #if AXIS_IS_L64XX(Y2)
  615. { 4, 0, 0, 0, 0, 0, 0 },
  616. #endif
  617. #if AXIS_IS_L64XX(Z2)
  618. { 5, 0, 0, 0, 0, 0, 0 },
  619. #endif
  620. #if AXIS_IS_L64XX(Z3)
  621. { 6, 0, 0, 0, 0, 0, 0 },
  622. #endif
  623. #if AXIS_IS_L64XX(Z4)
  624. { 6, 0, 0, 0, 0, 0, 0 },
  625. #endif
  626. #if AXIS_IS_L64XX(E0)
  627. { 7, 0, 0, 0, 0, 0, 0 },
  628. #endif
  629. #if AXIS_IS_L64XX(E1)
  630. { 8, 0, 0, 0, 0, 0, 0 },
  631. #endif
  632. #if AXIS_IS_L64XX(E2)
  633. { 9, 0, 0, 0, 0, 0, 0 },
  634. #endif
  635. #if AXIS_IS_L64XX(E3)
  636. { 10, 0, 0, 0, 0, 0, 0 },
  637. #endif
  638. #if AXIS_IS_L64XX(E4)
  639. { 11, 0, 0, 0, 0, 0, 0 },
  640. #endif
  641. #if AXIS_IS_L64XX(E5)
  642. { 12, 0, 0, 0, 0, 0, 0 }
  643. #endif
  644. };
  645. void L64XX_Marlin::append_stepper_err(char* &p, const uint8_t stepper_index, const char * const err/*=nullptr*/) {
  646. p += sprintf_P(p, PSTR("Stepper %c%c "), index_to_axis[stepper_index][0], index_to_axis[stepper_index][1]);
  647. if (err) p += sprintf_P(p, err);
  648. }
  649. void L64XX_Marlin::monitor_update(L64XX_axis_t stepper_index) {
  650. if (spi_abort) return; // don't do anything if set_directions() has occurred
  651. const L64XX_shadow_t &sh = shadow;
  652. get_status(stepper_index); // get stepper status and details
  653. uint16_t status = sh.STATUS_AXIS;
  654. uint8_t kval_hold, tval;
  655. char temp_buf[120], *p = temp_buf;
  656. uint8_t j;
  657. for (j = 0; j < L64XX::chain[0]; j++) // find the table for this stepper
  658. if (driver_L6470_data[j].driver_index == stepper_index) break;
  659. driver_L6470_data[j].driver_status = status;
  660. uint16_t _status = ~status; // all error bits are active low
  661. if (status == 0 || status == 0xFFFF) { // com problem
  662. if (driver_L6470_data[j].com_counter == 0) { // warn user when it first happens
  663. driver_L6470_data[j].com_counter++;
  664. append_stepper_err(p, stepper_index, PSTR(" - communications lost\n"));
  665. DEBUG_ECHO(temp_buf);
  666. }
  667. else {
  668. driver_L6470_data[j].com_counter++;
  669. if (driver_L6470_data[j].com_counter > 240) { // remind of com problem about every 2 minutes
  670. driver_L6470_data[j].com_counter = 1;
  671. append_stepper_err(p, stepper_index, PSTR(" - still no communications\n"));
  672. DEBUG_ECHO(temp_buf);
  673. }
  674. }
  675. }
  676. else {
  677. if (driver_L6470_data[j].com_counter) { // comms re-established
  678. driver_L6470_data[j].com_counter = 0;
  679. append_stepper_err(p, stepper_index, PSTR(" - communications re-established\n.. setting all drivers to default values\n"));
  680. DEBUG_ECHO(temp_buf);
  681. init_to_defaults();
  682. }
  683. else {
  684. // no com problems - do the usual checks
  685. if (_status & sh.L6470_ERROR_MASK) {
  686. append_stepper_err(p, stepper_index);
  687. if (status & STATUS_HIZ) { // The driver has shut down. HiZ is active high
  688. driver_L6470_data[j].is_hi_Z = true;
  689. p += sprintf_P(p, PSTR("%cIS SHUT DOWN"), ' ');
  690. //if (_status & sh.STATUS_AXIS_TH_SD) { // strange - TH_SD never seems to go active, must be implied by the HiZ and TH_WRN
  691. if (_status & sh.STATUS_AXIS_TH_WRN) { // over current shutdown
  692. p += sprintf_P(p, PSTR("%cdue to over temperature"), ' ');
  693. driver_L6470_data[j].is_ot = true;
  694. if (sh.STATUS_AXIS_LAYOUT == L6474_STATUS_LAYOUT) { // L6474
  695. tval = get_param(stepper_index, L6474_TVAL) - 2 * KVAL_HOLD_STEP_DOWN;
  696. set_param(stepper_index, L6474_TVAL, tval); // reduce TVAL
  697. p += sprintf_P(p, PSTR(" - TVAL reduced by %d to %d mA"), uint16_t (2 * KVAL_HOLD_STEP_DOWN * sh.AXIS_STALL_CURRENT_CONSTANT_INV), uint16_t ((tval + 1) * sh.AXIS_STALL_CURRENT_CONSTANT_INV)); // let user know
  698. }
  699. else {
  700. kval_hold = get_param(stepper_index, L6470_KVAL_HOLD) - 2 * KVAL_HOLD_STEP_DOWN;
  701. set_param(stepper_index, L6470_KVAL_HOLD, kval_hold); // reduce KVAL_HOLD
  702. p += sprintf_P(p, PSTR(" - KVAL_HOLD reduced by %d to %d"), 2 * KVAL_HOLD_STEP_DOWN, kval_hold); // let user know
  703. }
  704. }
  705. else
  706. driver_L6470_data[j].is_ot = false;
  707. }
  708. else {
  709. driver_L6470_data[j].is_hi_Z = false;
  710. if (_status & sh.STATUS_AXIS_TH_WRN) { // have an over temperature warning
  711. driver_L6470_data[j].is_otw = true;
  712. driver_L6470_data[j].otw_counter++;
  713. kval_hold = get_param(stepper_index, L6470_KVAL_HOLD);
  714. if (driver_L6470_data[j].otw_counter > 4) { // otw present for 2 - 2.5 seconds, reduce KVAL_HOLD
  715. driver_L6470_data[j].otw_counter = 0;
  716. driver_L6470_data[j].is_otw = true;
  717. if (sh.STATUS_AXIS_LAYOUT == L6474_STATUS_LAYOUT) { // L6474
  718. tval = get_param(stepper_index, L6474_TVAL) - KVAL_HOLD_STEP_DOWN;
  719. set_param(stepper_index, L6474_TVAL, tval); // reduce TVAL
  720. p += sprintf_P(p, PSTR(" - TVAL reduced by %d to %d mA"), uint16_t (KVAL_HOLD_STEP_DOWN * sh.AXIS_STALL_CURRENT_CONSTANT_INV), uint16_t ((tval + 1) * sh.AXIS_STALL_CURRENT_CONSTANT_INV)); // let user know
  721. }
  722. else {
  723. kval_hold = get_param(stepper_index, L6470_KVAL_HOLD) - KVAL_HOLD_STEP_DOWN;
  724. set_param(stepper_index, L6470_KVAL_HOLD, kval_hold); // reduce KVAL_HOLD
  725. p += sprintf_P(p, PSTR(" - KVAL_HOLD reduced by %d to %d"), KVAL_HOLD_STEP_DOWN, kval_hold); // let user know
  726. }
  727. }
  728. else if (driver_L6470_data[j].otw_counter)
  729. p += sprintf_P(p, PSTR("%c- thermal warning"), ' '); // warn user
  730. }
  731. }
  732. #if ENABLED(L6470_STOP_ON_ERROR)
  733. if (_status & (sh.STATUS_AXIS_UVLO | sh.STATUS_AXIS_TH_WRN | sh.STATUS_AXIS_TH_SD))
  734. kill(temp_buf);
  735. #endif
  736. #if ENABLED(L6470_CHITCHAT)
  737. if (_status & sh.STATUS_AXIS_OCD)
  738. p += sprintf_P(p, PSTR("%c over current"), ' ');
  739. if (_status & (sh.STATUS_AXIS_STEP_LOSS_A | sh.STATUS_AXIS_STEP_LOSS_B))
  740. p += sprintf_P(p, PSTR("%c stall"), ' ');
  741. if (_status & sh.STATUS_AXIS_UVLO)
  742. p += sprintf_P(p, PSTR("%c under voltage lock out"), ' ');
  743. p += sprintf_P(p, PSTR("%c\n"), ' ');
  744. #endif
  745. DEBUG_ECHOLN(temp_buf); // print the error message
  746. }
  747. else {
  748. driver_L6470_data[j].is_ot = false;
  749. driver_L6470_data[j].otw_counter = 0; //clear out warning indicators
  750. driver_L6470_data[j].is_otw = false;
  751. } // end usual checks
  752. } // comms established but have errors
  753. } // comms re-established
  754. } // end monitor_update()
  755. void L64XX_Marlin::monitor_driver() {
  756. static millis_t next_cOT = 0;
  757. if (ELAPSED(millis(), next_cOT)) {
  758. next_cOT = millis() + 500;
  759. if (!monitor_paused) { // Skip during M122, M906, M916, M917 or M918 (could steal status result from test)
  760. spi_active = true; // Tell set_directions() a series of SPI transfers is underway
  761. #if AXIS_IS_L64XX(X)
  762. monitor_update(X);
  763. #endif
  764. #if AXIS_IS_L64XX(Y)
  765. monitor_update(Y);
  766. #endif
  767. #if AXIS_IS_L64XX(Z)
  768. monitor_update(Z);
  769. #endif
  770. #if AXIS_IS_L64XX(X2)
  771. monitor_update(X2);
  772. #endif
  773. #if AXIS_IS_L64XX(Y2)
  774. monitor_update(Y2);
  775. #endif
  776. #if AXIS_IS_L64XX(Z2)
  777. monitor_update(Z2);
  778. #endif
  779. #if AXIS_IS_L64XX(Z3)
  780. monitor_update(Z3);
  781. #endif
  782. #if AXIS_IS_L64XX(Z4)
  783. monitor_update(Z4);
  784. #endif
  785. #if AXIS_IS_L64XX(E0)
  786. monitor_update(E0);
  787. #endif
  788. #if AXIS_IS_L64XX(E1)
  789. monitor_update(E1);
  790. #endif
  791. #if AXIS_IS_L64XX(E2)
  792. monitor_update(E2);
  793. #endif
  794. #if AXIS_IS_L64XX(E3)
  795. monitor_update(E3);
  796. #endif
  797. #if AXIS_IS_L64XX(E4)
  798. monitor_update(E4);
  799. #endif
  800. #if AXIS_IS_L64XX(E5)
  801. monitor_update(E5);
  802. #endif
  803. #if ENABLED(L6470_DEBUG)
  804. if (report_L6470_status) DEBUG_EOL();
  805. #endif
  806. spi_active = false; // done with all SPI transfers - clear handshake flags
  807. spi_abort = false;
  808. }
  809. }
  810. }
  811. #endif // MONITOR_L6470_DRIVER_STATUS
  812. #endif // HAS_L64XX