No Description
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

steppers.cpp 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451
  1. #include <Arduino.h>
  2. #include <AccelStepper.h>
  3. #include "config.h"
  4. #include "config_pins.h"
  5. #include "lcd.h"
  6. #include "data.h"
  7. #include "steppers.h"
  8. static AccelStepper stepper_x(AccelStepper::DRIVER, X_STEP_PIN, X_DIR_PIN, 0, 0, false);
  9. static AccelStepper stepper_y(AccelStepper::DRIVER, Y_STEP_PIN, Y_DIR_PIN, 0, 0, false);
  10. static AccelStepper stepper_z(AccelStepper::DRIVER, Z_STEP_PIN, Z_DIR_PIN, 0, 0, false);
  11. static AccelStepper stepper_e(AccelStepper::DRIVER, E0_STEP_PIN, E0_DIR_PIN, 0, 0, false);
  12. enum stepper_states {
  13. step_disabled,
  14. step_not_homed,
  15. step_homing_x_fast,
  16. step_homing_x_back,
  17. step_homing_x_slow,
  18. step_homing_y_fast,
  19. step_homing_y_back,
  20. step_homing_y_slow,
  21. step_homing_z_fast,
  22. step_homing_z_back,
  23. step_homing_z_slow,
  24. step_homing_e_fast,
  25. step_homing_e_back,
  26. step_homing_e_slow,
  27. step_homed
  28. };
  29. static stepper_states state = step_disabled;
  30. static unsigned long steppers_home_move_start_time = 0;
  31. void steppers_init(void) {
  32. pinMode(X_ENDSTOP_PIN, INPUT_PULLUP);
  33. pinMode(Y_ENDSTOP_PIN, INPUT_PULLUP);
  34. pinMode(Z_ENDSTOP_PIN, INPUT_PULLUP);
  35. pinMode(E_ENDSTOP_PIN, INPUT_PULLUP);
  36. pinMode(X_ENABLE_PIN, OUTPUT);
  37. pinMode(X_STEP_PIN, OUTPUT);
  38. pinMode(X_DIR_PIN, OUTPUT);
  39. pinMode(Y_ENABLE_PIN, OUTPUT);
  40. pinMode(Y_STEP_PIN, OUTPUT);
  41. pinMode(Y_DIR_PIN, OUTPUT);
  42. pinMode(Z_ENABLE_PIN, OUTPUT);
  43. pinMode(Z_STEP_PIN, OUTPUT);
  44. pinMode(Z_DIR_PIN, OUTPUT);
  45. pinMode(E0_ENABLE_PIN, OUTPUT);
  46. pinMode(E0_STEP_PIN, OUTPUT);
  47. pinMode(E0_DIR_PIN, OUTPUT);
  48. stepper_x.setEnablePin(X_ENABLE_PIN);
  49. stepper_x.setMaxSpeed(XY_MAX_SPEED * XY_STEPS_PER_MM);
  50. steppers_set_speed_x(data_options()->speed_x);
  51. steppers_set_accel_x(data_options()->accel_x);
  52. stepper_y.setEnablePin(Y_ENABLE_PIN);
  53. steppers_set_speed_y(data_options()->speed_y);
  54. steppers_set_accel_y(data_options()->accel_y);
  55. stepper_z.setEnablePin(Z_ENABLE_PIN);
  56. steppers_set_speed_z(data_options()->speed_z);
  57. steppers_set_accel_z(data_options()->accel_z);
  58. stepper_e.setEnablePin(E0_ENABLE_PIN);
  59. steppers_set_speed_e(data_options()->speed_e);
  60. steppers_set_accel_e(data_options()->accel_e);
  61. }
  62. static void steppers_initiate_home(int axis, int phase) {
  63. steppers_home_move_start_time = millis();
  64. if (axis == 0) {
  65. // x
  66. if (phase == 0) {
  67. state = step_homing_x_fast;
  68. stepper_x.setSpeed(X_HOMING_DIR * X_AXIS_MOVEMENT_DIR * XY_FAST_HOME_SPEED * XY_STEPS_PER_MM);
  69. } else if (phase == 1) {
  70. state = step_homing_x_back;
  71. stepper_x.setSpeed(-1.0 * X_HOMING_DIR * X_AXIS_MOVEMENT_DIR * XY_FAST_HOME_SPEED * XY_STEPS_PER_MM);
  72. } else if (phase == 2) {
  73. state = step_homing_x_slow;
  74. stepper_x.setSpeed(X_HOMING_DIR * X_AXIS_MOVEMENT_DIR * XY_SLOW_HOME_SPEED * XY_STEPS_PER_MM);
  75. }
  76. } else if (axis == 1) {
  77. // y
  78. if (phase == 0) {
  79. state = step_homing_y_fast;
  80. stepper_y.setSpeed(Y_HOMING_DIR * Y_AXIS_MOVEMENT_DIR * XY_FAST_HOME_SPEED * XY_STEPS_PER_MM);
  81. } else if (phase == 1) {
  82. state = step_homing_y_back;
  83. stepper_y.setSpeed(-1.0 * Y_HOMING_DIR * Y_AXIS_MOVEMENT_DIR * XY_FAST_HOME_SPEED * XY_STEPS_PER_MM);
  84. } else if (phase == 2) {
  85. state = step_homing_y_slow;
  86. stepper_y.setSpeed(Y_HOMING_DIR * Y_AXIS_MOVEMENT_DIR * XY_SLOW_HOME_SPEED * XY_STEPS_PER_MM);
  87. }
  88. } else if (axis == 2) {
  89. // z
  90. if (phase == 0) {
  91. state = step_homing_z_fast;
  92. stepper_z.setSpeed(Z_HOMING_DIR * Z_AXIS_MOVEMENT_DIR * Z_FAST_HOME_SPEED * Z_STEPS_PER_MM);
  93. } else if (phase == 1) {
  94. state = step_homing_z_back;
  95. stepper_z.setSpeed(-1.0 * Z_HOMING_DIR * Z_AXIS_MOVEMENT_DIR * Z_FAST_HOME_SPEED * Z_STEPS_PER_MM);
  96. } else if (phase == 2) {
  97. state = step_homing_z_slow;
  98. stepper_z.setSpeed(Z_HOMING_DIR * Z_AXIS_MOVEMENT_DIR * Z_SLOW_HOME_SPEED * Z_STEPS_PER_MM);
  99. }
  100. } else if (axis == 3) {
  101. // e
  102. if (phase == 0) {
  103. state = step_homing_e_fast;
  104. stepper_e.setSpeed(E_HOMING_DIR * E_AXIS_MOVEMENT_DIR * E_FAST_HOME_SPEED * E_STEPS_PER_PERCENT);
  105. } else if (phase == 1) {
  106. state = step_homing_e_back;
  107. stepper_e.setSpeed(-1.0 * E_HOMING_DIR * E_AXIS_MOVEMENT_DIR * E_FAST_HOME_SPEED * E_STEPS_PER_PERCENT);
  108. } else if (phase == 2) {
  109. state = step_homing_e_slow;
  110. stepper_e.setSpeed(E_HOMING_DIR * E_AXIS_MOVEMENT_DIR * E_SLOW_HOME_SPEED * E_STEPS_PER_PERCENT);
  111. }
  112. } else {
  113. Serial.println(F("home_init error: invalid axis"));
  114. }
  115. }
  116. static bool steppers_home_switch(int axis) {
  117. if (axis == 0) {
  118. return digitalRead(X_ENDSTOP_PIN) == LOW;
  119. } else if (axis == 1) {
  120. return digitalRead(Y_ENDSTOP_PIN) == LOW;
  121. } else if (axis == 2) {
  122. return digitalRead(Z_ENDSTOP_PIN) == LOW;
  123. } else if (axis == 3) {
  124. return digitalRead(E_ENDSTOP_PIN) == LOW;
  125. } else {
  126. Serial.println(F("home_switch error: invalid axis"));
  127. }
  128. return true;
  129. }
  130. bool steppers_run(void) {
  131. if (state == step_homing_x_fast) {
  132. if (steppers_home_switch(0)) {
  133. steppers_initiate_home(0, 1);
  134. } else {
  135. stepper_x.runSpeed();
  136. }
  137. } else if (state == step_homing_x_back) {
  138. unsigned long end_time = steppers_home_move_start_time + XY_HOME_BACK_OFF_TIME;
  139. if ((!steppers_home_switch(0)) && (millis() >= end_time)) {
  140. steppers_initiate_home(0, 2);
  141. } else {
  142. stepper_x.runSpeed();
  143. }
  144. } else if (state == step_homing_x_slow) {
  145. if (steppers_home_switch(0)) {
  146. stepper_x.setSpeed(0);
  147. steppers_initiate_home(1, 0);
  148. } else {
  149. stepper_x.runSpeed();
  150. }
  151. } else if (state == step_homing_y_fast) {
  152. if (steppers_home_switch(1)) {
  153. steppers_initiate_home(1, 1);
  154. } else {
  155. stepper_y.runSpeed();
  156. }
  157. } else if (state == step_homing_y_back) {
  158. unsigned long end_time = steppers_home_move_start_time + XY_HOME_BACK_OFF_TIME;
  159. if ((!steppers_home_switch(1)) && (millis() >= end_time)) {
  160. steppers_initiate_home(1, 2);
  161. } else {
  162. stepper_y.runSpeed();
  163. }
  164. } else if (state == step_homing_y_slow) {
  165. if (steppers_home_switch(1)) {
  166. stepper_y.setSpeed(0);
  167. steppers_initiate_home(3, 0);
  168. } else {
  169. stepper_y.runSpeed();
  170. }
  171. } else if (state == step_homing_z_fast) {
  172. if (steppers_home_switch(2)) {
  173. steppers_initiate_home(2, 1);
  174. } else {
  175. stepper_z.runSpeed();
  176. }
  177. } else if (state == step_homing_z_back) {
  178. unsigned long end_time = steppers_home_move_start_time + Z_HOME_BACK_OFF_TIME;
  179. if ((!steppers_home_switch(2)) && (millis() >= end_time)) {
  180. steppers_initiate_home(2, 2);
  181. } else {
  182. stepper_z.runSpeed();
  183. }
  184. } else if (state == step_homing_z_slow) {
  185. if (steppers_home_switch(2)) {
  186. stepper_z.setSpeed(0);
  187. steppers_initiate_home(0, 0);
  188. } else {
  189. stepper_z.runSpeed();
  190. }
  191. } else if (state == step_homing_e_fast) {
  192. if (steppers_home_switch(3)) {
  193. steppers_initiate_home(3, 1);
  194. } else {
  195. stepper_e.runSpeed();
  196. }
  197. } else if (state == step_homing_e_back) {
  198. unsigned long end_time = steppers_home_move_start_time + E_HOME_BACK_OFF_TIME;
  199. if ((!steppers_home_switch(3)) && (millis() >= end_time)) {
  200. steppers_initiate_home(3, 2);
  201. } else {
  202. stepper_e.runSpeed();
  203. }
  204. } else if (state == step_homing_e_slow) {
  205. if (steppers_home_switch(3)) {
  206. stepper_e.setSpeed(0);
  207. state = step_homed;
  208. return false;
  209. } else {
  210. stepper_e.runSpeed();
  211. }
  212. } else if (state != step_disabled) {
  213. for (int i = 0; i < 4; i++) {
  214. if (steppers_home_switch(i)) {
  215. Serial.print(F("ERROR: endstop hit on "));
  216. Serial.println(i);
  217. // TODO proper error handling?
  218. lcd_clear();
  219. lcd_set_heading("ERROR");
  220. lcd_set_text("Endstop hit. Aborting!");
  221. while (1) { }
  222. }
  223. }
  224. boolean x = stepper_x.run();
  225. boolean y = stepper_y.run();
  226. boolean z = stepper_z.run();
  227. boolean e = stepper_e.run();
  228. return x || y || z || e;
  229. }
  230. return true;
  231. }
  232. bool steppers_homed(void) {
  233. return (state == step_homed);
  234. }
  235. void steppers_start_homing(void) {
  236. steppers_initiate_home(2, 0);
  237. }
  238. static int steppers_move_axis(AccelStepper &axis, long pos) {
  239. if (state == step_disabled) {
  240. Serial.println(F("Enabling stepper drivers"));
  241. stepper_x.enableOutputs();
  242. stepper_y.enableOutputs();
  243. stepper_z.enableOutputs();
  244. stepper_e.enableOutputs();
  245. state = step_not_homed;
  246. }
  247. if (!steppers_homed()) {
  248. Serial.println(F("WARNING: un-homed move!"));
  249. }
  250. axis.moveTo(pos);
  251. return 0;
  252. }
  253. int steppers_move_x(long pos) {
  254. Serial.print(F("Moving X to "));
  255. Serial.print(pos);
  256. Serial.print(F(" mm ("));
  257. Serial.print(pos * XY_STEPS_PER_MM * X_AXIS_MOVEMENT_DIR);
  258. Serial.println(F(" steps)"));
  259. if (pos < X_AXIS_MIN) {
  260. Serial.println(F("Move below X_AXIS_MIN!"));
  261. return -1;
  262. }
  263. if (pos > X_AXIS_MAX) {
  264. Serial.println(F("Move above X_AXIS_MAX!"));
  265. return -1;
  266. }
  267. return steppers_move_axis(stepper_x, pos * XY_STEPS_PER_MM * X_AXIS_MOVEMENT_DIR);
  268. }
  269. int steppers_move_y(long pos) {
  270. Serial.print(F("Moving Y to "));
  271. Serial.print(pos);
  272. Serial.print(F(" mm ("));
  273. Serial.print(pos * XY_STEPS_PER_MM * Y_AXIS_MOVEMENT_DIR);
  274. Serial.println(F(" steps)"));
  275. if (pos < Y_AXIS_MIN) {
  276. Serial.println(F("Move below Y_AXIS_MIN!"));
  277. return -1;
  278. }
  279. if (pos > Y_AXIS_MAX) {
  280. Serial.println(F("Move above Y_AXIS_MAX!"));
  281. return -1;
  282. }
  283. return steppers_move_axis(stepper_y, pos * XY_STEPS_PER_MM * Y_AXIS_MOVEMENT_DIR);
  284. }
  285. int steppers_move_z(long pos) {
  286. Serial.print(F("Moving Z to "));
  287. Serial.print(pos);
  288. Serial.print(F(" mm ("));
  289. Serial.print(pos * Z_STEPS_PER_MM * Z_AXIS_MOVEMENT_DIR);
  290. Serial.println(F(" steps)"));
  291. if (pos < Z_AXIS_MIN) {
  292. Serial.println(F("Move below Z_AXIS_MIN!"));
  293. return -1;
  294. }
  295. if (pos > Z_AXIS_MAX) {
  296. Serial.println(F("Move above Z_AXIS_MAX!"));
  297. return -1;
  298. }
  299. return steppers_move_axis(stepper_z, pos * Z_STEPS_PER_MM * Z_AXIS_MOVEMENT_DIR);
  300. }
  301. int steppers_move_e(long pos) {
  302. Serial.print(F("Moving E to "));
  303. Serial.print(pos);
  304. Serial.print(F(" mm ("));
  305. Serial.print(E_MM_TO_PERCENT(pos));
  306. Serial.println(F("% = "));
  307. Serial.print(E_MM_TO_PERCENT(pos) * E_STEPS_PER_PERCENT * E_AXIS_MOVEMENT_DIR);
  308. Serial.println(F(" steps)"));
  309. if (pos < E_AXIS_MIN) {
  310. Serial.println(F("Move below E_AXIS_MIN!"));
  311. return -1;
  312. }
  313. if (pos > E_AXIS_MAX) {
  314. Serial.println(F("Move above E_AXIS_MAX!"));
  315. return -1;
  316. }
  317. return steppers_move_axis(stepper_e, E_MM_TO_PERCENT(pos) * E_STEPS_PER_PERCENT * E_AXIS_MOVEMENT_DIR);
  318. }
  319. void steppers_kill(void) {
  320. stepper_x.setCurrentPosition(0);
  321. stepper_y.setCurrentPosition(0);
  322. stepper_z.setCurrentPosition(0);
  323. stepper_e.setCurrentPosition(0);
  324. stepper_x.disableOutputs();
  325. stepper_y.disableOutputs();
  326. stepper_z.disableOutputs();
  327. stepper_e.disableOutputs();
  328. state = step_not_homed;
  329. }
  330. void steppers_set_speed_x(float speed) {
  331. if (speed < 0.0) {
  332. speed = 0.0;
  333. }
  334. if (speed > XY_MAX_SPEED) {
  335. speed = XY_MAX_SPEED;
  336. }
  337. stepper_x.setMaxSpeed(speed * XY_STEPS_PER_MM);
  338. }
  339. void steppers_set_speed_y(float speed) {
  340. if (speed < 0.0) {
  341. speed = 0.0;
  342. }
  343. if (speed > XY_MAX_SPEED) {
  344. speed = XY_MAX_SPEED;
  345. }
  346. stepper_y.setMaxSpeed(speed * XY_STEPS_PER_MM);
  347. }
  348. void steppers_set_speed_z(float speed) {
  349. if (speed < 0.0) {
  350. speed = 0.0;
  351. }
  352. if (speed > Z_MAX_SPEED) {
  353. speed = Z_MAX_SPEED;
  354. }
  355. stepper_z.setMaxSpeed(speed * Z_STEPS_PER_MM);
  356. }
  357. void steppers_set_speed_e(float speed) {
  358. if (speed < 0.0) {
  359. speed = 0.0;
  360. }
  361. if (speed > E_MAX_SPEED) {
  362. speed = E_MAX_SPEED;
  363. }
  364. stepper_e.setMaxSpeed(speed * E_STEPS_PER_PERCENT);
  365. }
  366. void steppers_set_accel_x(float accel) {
  367. if (accel < 0.0) {
  368. accel = 0.0;
  369. }
  370. if (accel > XY_MAX_ACCEL) {
  371. accel = XY_MAX_ACCEL;
  372. }
  373. stepper_x.setAcceleration(accel * XY_STEPS_PER_MM);
  374. }
  375. void steppers_set_accel_y(float accel) {
  376. if (accel < 0.0) {
  377. accel = 0.0;
  378. }
  379. if (accel > XY_MAX_ACCEL) {
  380. accel = XY_MAX_ACCEL;
  381. }
  382. stepper_y.setAcceleration(accel * XY_STEPS_PER_MM);
  383. }
  384. void steppers_set_accel_z(float accel) {
  385. if (accel < 0.0) {
  386. accel = 0.0;
  387. }
  388. if (accel > Z_MAX_ACCEL) {
  389. accel = Z_MAX_ACCEL;
  390. }
  391. stepper_z.setAcceleration(accel * Z_STEPS_PER_MM);
  392. }
  393. void steppers_set_accel_e(float accel) {
  394. if (accel < 0.0) {
  395. accel = 0.0;
  396. }
  397. if (accel > E_MAX_ACCEL) {
  398. accel = E_MAX_ACCEL;
  399. }
  400. stepper_e.setAcceleration(accel * E_STEPS_PER_PERCENT);
  401. }