mesh_bed_calibration.cpp 55 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362
  1. #include "Marlin.h"
  2. #include "Configuration.h"
  3. #include "language_all.h"
  4. #include "mesh_bed_calibration.h"
  5. #include "mesh_bed_leveling.h"
  6. #include "stepper.h"
  7. #include "ultralcd.h"
  8. // #include "qr_solve.h"
  9. extern float home_retract_mm_ext(int axis);
  10. float world2machine_rotation_and_skew[2][2];
  11. float world2machine_shift[2];
  12. #define BED_ZERO_REF_X (- 22.f + X_PROBE_OFFSET_FROM_EXTRUDER)
  13. #define BED_ZERO_REF_Y (- 0.6f + Y_PROBE_OFFSET_FROM_EXTRUDER)
  14. // Positions of the bed reference points in the machine coordinates, referenced to the P.I.N.D.A sensor.
  15. // The points are ordered in a zig-zag fashion to speed up the calibration.
  16. const float bed_ref_points[] PROGMEM = {
  17. 13.f - BED_ZERO_REF_X, 6.4f - BED_ZERO_REF_Y,
  18. 115.f - BED_ZERO_REF_X, 6.4f - BED_ZERO_REF_Y,
  19. 216.f - BED_ZERO_REF_X, 6.4f - BED_ZERO_REF_Y,
  20. 216.f - BED_ZERO_REF_X, 104.4f - BED_ZERO_REF_Y,
  21. 115.f - BED_ZERO_REF_X, 104.4f - BED_ZERO_REF_Y,
  22. 13.f - BED_ZERO_REF_X, 104.4f - BED_ZERO_REF_Y,
  23. 13.f - BED_ZERO_REF_X, 202.4f - BED_ZERO_REF_Y,
  24. 115.f - BED_ZERO_REF_X, 202.4f - BED_ZERO_REF_Y,
  25. 216.f - BED_ZERO_REF_X, 202.4f - BED_ZERO_REF_Y
  26. };
  27. // Positions of the bed reference points in the machine coordinates, referenced to the P.I.N.D.A sensor.
  28. // The points are the following: center front, center right, center rear, center left.
  29. const float bed_ref_points_4[] PROGMEM = {
  30. 115.f - BED_ZERO_REF_X, 6.4f - BED_ZERO_REF_Y,
  31. 216.f - BED_ZERO_REF_X, 104.4f - BED_ZERO_REF_Y,
  32. 115.f - BED_ZERO_REF_X, 202.4f - BED_ZERO_REF_Y,
  33. 13.f - BED_ZERO_REF_X, 104.4f - BED_ZERO_REF_Y
  34. };
  35. //#define Y_MIN_POS_FOR_BED_CALIBRATION (MANUAL_Y_HOME_POS-0.2f)
  36. #define Y_MIN_POS_FOR_BED_CALIBRATION (Y_MIN_POS)
  37. static inline float sqr(float x) { return x * x; }
  38. bool calculate_machine_skew_and_offset_LS(
  39. // Matrix of maximum 9 2D points (18 floats)
  40. const float *measured_pts,
  41. uint8_t npts,
  42. const float *true_pts,
  43. // Resulting correction matrix.
  44. float *vec_x,
  45. float *vec_y,
  46. float *cntr,
  47. // Temporary values, 49-18-(2*3)=25 floats
  48. // , float *temp
  49. int8_t verbosity_level
  50. )
  51. {
  52. if (verbosity_level >= 10) {
  53. // Show the initial state, before the fitting.
  54. SERIAL_ECHOPGM("X vector, initial: ");
  55. MYSERIAL.print(vec_x[0], 5);
  56. SERIAL_ECHOPGM(", ");
  57. MYSERIAL.print(vec_x[1], 5);
  58. SERIAL_ECHOLNPGM("");
  59. SERIAL_ECHOPGM("Y vector, initial: ");
  60. MYSERIAL.print(vec_y[0], 5);
  61. SERIAL_ECHOPGM(", ");
  62. MYSERIAL.print(vec_y[1], 5);
  63. SERIAL_ECHOLNPGM("");
  64. SERIAL_ECHOPGM("center, initial: ");
  65. MYSERIAL.print(cntr[0], 5);
  66. SERIAL_ECHOPGM(", ");
  67. MYSERIAL.print(cntr[1], 5);
  68. SERIAL_ECHOLNPGM("");
  69. for (uint8_t i = 0; i < npts; ++ i) {
  70. SERIAL_ECHOPGM("point #");
  71. MYSERIAL.print(int(i));
  72. SERIAL_ECHOPGM(" measured: (");
  73. MYSERIAL.print(measured_pts[i*2], 5);
  74. SERIAL_ECHOPGM(", ");
  75. MYSERIAL.print(measured_pts[i*2+1], 5);
  76. SERIAL_ECHOPGM("); target: (");
  77. MYSERIAL.print(pgm_read_float(true_pts+i*2 ), 5);
  78. SERIAL_ECHOPGM(", ");
  79. MYSERIAL.print(pgm_read_float(true_pts+i*2+1), 5);
  80. SERIAL_ECHOPGM("), error: ");
  81. MYSERIAL.print(sqrt(
  82. sqr(pgm_read_float(true_pts+i*2 ) - measured_pts[i*2 ]) +
  83. sqr(pgm_read_float(true_pts+i*2+1) - measured_pts[i*2+1])), 5);
  84. SERIAL_ECHOLNPGM("");
  85. }
  86. delay_keep_alive(100);
  87. }
  88. {
  89. // Create covariance matrix for A, collect the right hand side b.
  90. float A[3][3] = { 0.f };
  91. float b[3] = { 0.f };
  92. float acc;
  93. for (uint8_t r = 0; r < 3; ++ r) {
  94. for (uint8_t c = 0; c < 3; ++ c) {
  95. acc = 0;
  96. for (uint8_t i = 0; i < npts; ++ i) {
  97. float a = (r == 2) ? 1.f : measured_pts[2 * i + r];
  98. float b = (c == 2) ? 1.f : measured_pts[2 * i + c];
  99. acc += a * b;
  100. }
  101. A[r][c] = acc;
  102. }
  103. acc = 0.f;
  104. for (uint8_t i = 0; i < npts; ++ i) {
  105. float a = (r == 2) ? 1.f : measured_pts[2 * i + r];
  106. float b = pgm_read_float(true_pts+i*2);
  107. acc += a * b;
  108. }
  109. b[r] = acc;
  110. }
  111. // Solve the linear equation for ax, bx, cx.
  112. float x[3] = { 0.f };
  113. for (uint8_t iter = 0; iter < 100; ++ iter) {
  114. x[0] = (b[0] - A[0][1] * x[1] - A[0][2] * x[2]) / A[0][0];
  115. x[1] = (b[1] - A[1][0] * x[0] - A[1][2] * x[2]) / A[1][1];
  116. x[2] = (b[2] - A[2][0] * x[0] - A[2][1] * x[1]) / A[2][2];
  117. }
  118. // Store the result to the output variables.
  119. vec_x[0] = x[0];
  120. vec_y[0] = x[1];
  121. cntr[0] = x[2];
  122. // Recalculate A and b for the y values.
  123. // Note the weighting of the first row of values.
  124. // const float weight_1st_row = 0.5f;
  125. const float weight_1st_row = 0.2f;
  126. for (uint8_t r = 0; r < 3; ++ r) {
  127. for (uint8_t c = 0; c < 3; ++ c) {
  128. acc = 0;
  129. for (uint8_t i = 0; i < npts; ++ i) {
  130. float w = (i < 3) ? weight_1st_row : 1.f;
  131. float a = (r == 2) ? 1.f : measured_pts[2 * i + r];
  132. float b = (c == 2) ? 1.f : measured_pts[2 * i + c];
  133. acc += a * b * w;
  134. }
  135. A[r][c] = acc;
  136. }
  137. acc = 0.f;
  138. for (uint8_t i = 0; i < npts; ++ i) {
  139. float w = (i < 3) ? weight_1st_row : 1.f;
  140. float a = (r == 2) ? 1.f : measured_pts[2 * i + r];
  141. float b = pgm_read_float(true_pts+i*2+1);
  142. acc += w * a * b;
  143. }
  144. b[r] = acc;
  145. }
  146. // Solve the linear equation for ay, by, cy.
  147. x[0] = 0.f, x[1] = 0.f; x[2] = 0.f;
  148. for (uint8_t iter = 0; iter < 100; ++ iter) {
  149. x[0] = (b[0] - A[0][1] * x[1] - A[0][2] * x[2]) / A[0][0];
  150. x[1] = (b[1] - A[1][0] * x[0] - A[1][2] * x[2]) / A[1][1];
  151. x[2] = (b[2] - A[2][0] * x[0] - A[2][1] * x[1]) / A[2][2];
  152. }
  153. // Store the result to the output variables.
  154. vec_x[1] = x[0];
  155. vec_y[1] = x[1];
  156. cntr[1] = x[2];
  157. }
  158. if (verbosity_level >= 10) {
  159. // Show the adjusted state, before the fitting.
  160. SERIAL_ECHOPGM("X vector new, inverted: ");
  161. MYSERIAL.print(vec_x[0], 5);
  162. SERIAL_ECHOPGM(", ");
  163. MYSERIAL.print(vec_x[1], 5);
  164. SERIAL_ECHOLNPGM("");
  165. SERIAL_ECHOPGM("Y vector new, inverted: ");
  166. MYSERIAL.print(vec_y[0], 5);
  167. SERIAL_ECHOPGM(", ");
  168. MYSERIAL.print(vec_y[1], 5);
  169. SERIAL_ECHOLNPGM("");
  170. SERIAL_ECHOPGM("center new, inverted: ");
  171. MYSERIAL.print(cntr[0], 5);
  172. SERIAL_ECHOPGM(", ");
  173. MYSERIAL.print(cntr[1], 5);
  174. SERIAL_ECHOLNPGM("");
  175. delay_keep_alive(100);
  176. SERIAL_ECHOLNPGM("Error after correction: ");
  177. for (uint8_t i = 0; i < npts; ++ i) {
  178. float x = vec_x[0] * measured_pts[i*2] + vec_y[0] * measured_pts[i*2+1] + cntr[0];
  179. float y = vec_x[1] * measured_pts[i*2] + vec_y[1] * measured_pts[i*2+1] + cntr[1];
  180. SERIAL_ECHOPGM("point #");
  181. MYSERIAL.print(int(i));
  182. SERIAL_ECHOPGM(" measured: (");
  183. MYSERIAL.print(measured_pts[i*2], 5);
  184. SERIAL_ECHOPGM(", ");
  185. MYSERIAL.print(measured_pts[i*2+1], 5);
  186. SERIAL_ECHOPGM("); corrected: (");
  187. MYSERIAL.print(x, 5);
  188. SERIAL_ECHOPGM(", ");
  189. MYSERIAL.print(y, 5);
  190. SERIAL_ECHOPGM("); target: (");
  191. MYSERIAL.print(pgm_read_float(true_pts+i*2 ), 5);
  192. SERIAL_ECHOPGM(", ");
  193. MYSERIAL.print(pgm_read_float(true_pts+i*2+1), 5);
  194. SERIAL_ECHOPGM("), error: ");
  195. MYSERIAL.print(sqrt(sqr(pgm_read_float(true_pts+i*2)-x)+sqr(pgm_read_float(true_pts+i*2+1)-y)));
  196. SERIAL_ECHOLNPGM("");
  197. }
  198. }
  199. #if 0
  200. // Normalize the vectors. We expect, that the machine axes may be skewed a bit, but the distances are correct.
  201. // l shall be very close to 1 already.
  202. float l = sqrt(vec_x[0]*vec_x[0] + vec_x[1] * vec_x[1]);
  203. vec_x[0] /= l;
  204. vec_x[1] /= l;
  205. SERIAL_ECHOPGM("Length of the X vector: ");
  206. MYSERIAL.print(l, 5);
  207. SERIAL_ECHOLNPGM("");
  208. l = sqrt(vec_y[0]*vec_y[0] + vec_y[1] * vec_y[1]);
  209. vec_y[0] /= l;
  210. vec_y[1] /= l;
  211. SERIAL_ECHOPGM("Length of the Y vector: ");
  212. MYSERIAL.print(l, 5);
  213. SERIAL_ECHOLNPGM("");
  214. // Recalculate the center using the adjusted vec_x/vec_y
  215. {
  216. cntr[0] = 0.f;
  217. cntr[1] = 0.f;
  218. for (uint8_t i = 0; i < npts; ++ i) {
  219. cntr[0] += measured_pts[2 * i ] - pgm_read_float(true_pts+i*2) * vec_x[0] - pgm_read_float(true_pts+i*2+1) * vec_y[0];
  220. cntr[1] += measured_pts[2 * i + 1] - pgm_read_float(true_pts+i*2) * vec_x[1] - pgm_read_float(true_pts+i*2+1) * vec_y[1];
  221. }
  222. cntr[0] /= float(npts);
  223. cntr[1] /= float(npts);
  224. }
  225. SERIAL_ECHOPGM("X vector new, inverted, normalized: ");
  226. MYSERIAL.print(vec_x[0], 5);
  227. SERIAL_ECHOPGM(", ");
  228. MYSERIAL.print(vec_x[1], 5);
  229. SERIAL_ECHOLNPGM("");
  230. SERIAL_ECHOPGM("Y vector new, inverted, normalized: ");
  231. MYSERIAL.print(vec_y[0], 5);
  232. SERIAL_ECHOPGM(", ");
  233. MYSERIAL.print(vec_y[1], 5);
  234. SERIAL_ECHOLNPGM("");
  235. SERIAL_ECHOPGM("center new, inverted, normalized: ");
  236. MYSERIAL.print(cntr[0], 5);
  237. SERIAL_ECHOPGM(", ");
  238. MYSERIAL.print(cntr[1], 5);
  239. SERIAL_ECHOLNPGM("");
  240. #endif
  241. // Invert the transformation matrix made of vec_x, vec_y and cntr.
  242. {
  243. float d = vec_x[0] * vec_y[1] - vec_x[1] * vec_y[0];
  244. float Ainv[2][2] = {
  245. { vec_y[1] / d, - vec_y[0] / d },
  246. { - vec_x[1] / d, vec_x[0] / d }
  247. };
  248. float cntrInv[2] = {
  249. - Ainv[0][0] * cntr[0] - Ainv[0][1] * cntr[1],
  250. - Ainv[1][0] * cntr[0] - Ainv[1][1] * cntr[1]
  251. };
  252. vec_x[0] = Ainv[0][0];
  253. vec_x[1] = Ainv[1][0];
  254. vec_y[0] = Ainv[0][1];
  255. vec_y[1] = Ainv[1][1];
  256. cntr[0] = cntrInv[0];
  257. cntr[1] = cntrInv[1];
  258. }
  259. if (verbosity_level >= 1) {
  260. // Show the adjusted state, before the fitting.
  261. SERIAL_ECHOPGM("X vector, adjusted: ");
  262. MYSERIAL.print(vec_x[0], 5);
  263. SERIAL_ECHOPGM(", ");
  264. MYSERIAL.print(vec_x[1], 5);
  265. SERIAL_ECHOLNPGM("");
  266. SERIAL_ECHOPGM("Y vector, adjusted: ");
  267. MYSERIAL.print(vec_y[0], 5);
  268. SERIAL_ECHOPGM(", ");
  269. MYSERIAL.print(vec_y[1], 5);
  270. SERIAL_ECHOLNPGM("");
  271. SERIAL_ECHOPGM("center, adjusted: ");
  272. MYSERIAL.print(cntr[0], 5);
  273. SERIAL_ECHOPGM(", ");
  274. MYSERIAL.print(cntr[1], 5);
  275. SERIAL_ECHOLNPGM("");
  276. delay_keep_alive(100);
  277. }
  278. if (verbosity_level >= 2) {
  279. SERIAL_ECHOLNPGM("Difference after correction: ");
  280. for (uint8_t i = 0; i < npts; ++ i) {
  281. float x = vec_x[0] * pgm_read_float(true_pts+i*2) + vec_y[0] * pgm_read_float(true_pts+i*2+1) + cntr[0];
  282. float y = vec_x[1] * pgm_read_float(true_pts+i*2) + vec_y[1] * pgm_read_float(true_pts+i*2+1) + cntr[1];
  283. SERIAL_ECHOPGM("point #");
  284. MYSERIAL.print(int(i));
  285. SERIAL_ECHOPGM("measured: (");
  286. MYSERIAL.print(measured_pts[i*2], 5);
  287. SERIAL_ECHOPGM(", ");
  288. MYSERIAL.print(measured_pts[i*2+1], 5);
  289. SERIAL_ECHOPGM("); measured-corrected: (");
  290. MYSERIAL.print(x, 5);
  291. SERIAL_ECHOPGM(", ");
  292. MYSERIAL.print(y, 5);
  293. SERIAL_ECHOPGM("); target: (");
  294. MYSERIAL.print(pgm_read_float(true_pts+i*2 ), 5);
  295. SERIAL_ECHOPGM(", ");
  296. MYSERIAL.print(pgm_read_float(true_pts+i*2+1), 5);
  297. SERIAL_ECHOPGM("), error: ");
  298. MYSERIAL.print(sqrt(sqr(measured_pts[i*2]-x)+sqr(measured_pts[i*2+1]-y)));
  299. SERIAL_ECHOLNPGM("");
  300. }
  301. delay_keep_alive(100);
  302. }
  303. return true;
  304. }
  305. void reset_bed_offset_and_skew()
  306. {
  307. eeprom_update_dword((uint32_t*)(EEPROM_BED_CALIBRATION_CENTER+0), 0x0FFFFFFFF);
  308. eeprom_update_dword((uint32_t*)(EEPROM_BED_CALIBRATION_CENTER+4), 0x0FFFFFFFF);
  309. eeprom_update_dword((uint32_t*)(EEPROM_BED_CALIBRATION_VEC_X +0), 0x0FFFFFFFF);
  310. eeprom_update_dword((uint32_t*)(EEPROM_BED_CALIBRATION_VEC_X +4), 0x0FFFFFFFF);
  311. eeprom_update_dword((uint32_t*)(EEPROM_BED_CALIBRATION_VEC_Y +0), 0x0FFFFFFFF);
  312. eeprom_update_dword((uint32_t*)(EEPROM_BED_CALIBRATION_VEC_Y +4), 0x0FFFFFFFF);
  313. }
  314. void world2machine_reset()
  315. {
  316. // Identity transformation.
  317. world2machine_rotation_and_skew[0][0] = 1.f;
  318. world2machine_rotation_and_skew[0][1] = 0.f;
  319. world2machine_rotation_and_skew[1][0] = 0.f;
  320. world2machine_rotation_and_skew[1][1] = 1.f;
  321. // Zero shift.
  322. world2machine_shift[0] = 0.f;
  323. world2machine_shift[1] = 0.f;
  324. }
  325. static inline bool vec_undef(const float v[2])
  326. {
  327. const uint32_t *vx = (const uint32_t*)v;
  328. return vx[0] == 0x0FFFFFFFF || vx[1] == 0x0FFFFFFFF;
  329. }
  330. void world2machine_initialize()
  331. {
  332. float cntr[2] = {
  333. eeprom_read_float((float*)(EEPROM_BED_CALIBRATION_CENTER+0)),
  334. eeprom_read_float((float*)(EEPROM_BED_CALIBRATION_CENTER+4))
  335. };
  336. float vec_x[2] = {
  337. eeprom_read_float((float*)(EEPROM_BED_CALIBRATION_VEC_X +0)),
  338. eeprom_read_float((float*)(EEPROM_BED_CALIBRATION_VEC_X +4))
  339. };
  340. float vec_y[2] = {
  341. eeprom_read_float((float*)(EEPROM_BED_CALIBRATION_VEC_Y +0)),
  342. eeprom_read_float((float*)(EEPROM_BED_CALIBRATION_VEC_Y +4))
  343. };
  344. bool reset = false;
  345. if (vec_undef(cntr) || vec_undef(vec_x) || vec_undef(vec_y)) {
  346. SERIAL_ECHOLNPGM("Undefined bed correction matrix.");
  347. reset = true;
  348. }
  349. else {
  350. // Length of the vec_x shall be close to unity.
  351. float l = sqrt(vec_x[0] * vec_x[0] + vec_x[1] * vec_x[1]);
  352. if (l < 0.9 || l > 1.1) {
  353. SERIAL_ECHOLNPGM("Invalid bed correction matrix. Length of the X vector out of range.");
  354. reset = true;
  355. }
  356. // Length of the vec_y shall be close to unity.
  357. l = sqrt(vec_y[0] * vec_y[0] + vec_y[1] * vec_y[1]);
  358. if (l < 0.9 || l > 1.1) {
  359. SERIAL_ECHOLNPGM("Invalid bed correction matrix. Length of the X vector out of range.");
  360. reset = true;
  361. }
  362. // Correction of the zero point shall be reasonably small.
  363. l = sqrt(cntr[0] * cntr[0] + cntr[1] * cntr[1]);
  364. if (l > 15.f) {
  365. SERIAL_ECHOLNPGM("Invalid bed correction matrix. Shift out of range.");
  366. reset = true;
  367. }
  368. // vec_x and vec_y shall be nearly perpendicular.
  369. l = vec_x[0] * vec_y[0] + vec_x[1] * vec_y[1];
  370. if (fabs(l) > 0.1f) {
  371. SERIAL_ECHOLNPGM("Invalid bed correction matrix. X/Y axes are far from being perpendicular.");
  372. reset = true;
  373. }
  374. }
  375. if (reset) {
  376. // SERIAL_ECHOLNPGM("Invalid bed correction matrix. Resetting to identity.");
  377. reset_bed_offset_and_skew();
  378. world2machine_reset();
  379. } else {
  380. world2machine_rotation_and_skew[0][0] = vec_x[0];
  381. world2machine_rotation_and_skew[1][0] = vec_x[1];
  382. world2machine_rotation_and_skew[0][1] = vec_y[0];
  383. world2machine_rotation_and_skew[1][1] = vec_y[1];
  384. world2machine_shift[0] = cntr[0];
  385. world2machine_shift[1] = cntr[1];
  386. }
  387. }
  388. // When switching from absolute to corrected coordinates,
  389. // this will get the absolute coordinates from the servos,
  390. // applies the inverse world2machine transformation
  391. // and stores the result into current_position[x,y].
  392. void world2machine_update_current()
  393. {
  394. // Invert the transformation matrix made of vec_x, vec_y and cntr.
  395. float d = world2machine_rotation_and_skew[0][0] * world2machine_rotation_and_skew[1][1] - world2machine_rotation_and_skew[1][0] * world2machine_rotation_and_skew[0][1];
  396. float Ainv[2][2] = {
  397. { world2machine_rotation_and_skew[1][1] / d, - world2machine_rotation_and_skew[0][1] / d },
  398. { - world2machine_rotation_and_skew[1][0] / d, world2machine_rotation_and_skew[0][0] / d }
  399. };
  400. float x = current_position[X_AXIS] - world2machine_shift[0];
  401. float y = current_position[Y_AXIS] - world2machine_shift[1];
  402. current_position[X_AXIS] = Ainv[0][0] * x + Ainv[0][1] * y;
  403. current_position[Y_AXIS] = Ainv[1][0] * x + Ainv[1][1] * y;
  404. }
  405. static inline void go_xyz(float x, float y, float z, float fr)
  406. {
  407. plan_buffer_line(x, y, z, current_position[E_AXIS], fr, active_extruder);
  408. st_synchronize();
  409. }
  410. static inline void go_xy(float x, float y, float fr)
  411. {
  412. plan_buffer_line(x, y, current_position[Z_AXIS], current_position[E_AXIS], fr, active_extruder);
  413. st_synchronize();
  414. }
  415. static inline void go_to_current(float fr)
  416. {
  417. plan_buffer_line(current_position[X_AXIS], current_position[Y_AXIS], current_position[Z_AXIS], current_position[E_AXIS], fr, active_extruder);
  418. st_synchronize();
  419. }
  420. static inline void update_current_position_xyz()
  421. {
  422. current_position[X_AXIS] = st_get_position_mm(X_AXIS);
  423. current_position[Y_AXIS] = st_get_position_mm(Y_AXIS);
  424. current_position[Z_AXIS] = st_get_position_mm(Z_AXIS);
  425. plan_set_position(current_position[X_AXIS], current_position[Y_AXIS], current_position[Z_AXIS], current_position[E_AXIS]);
  426. }
  427. static inline void update_current_position_z()
  428. {
  429. current_position[Z_AXIS] = st_get_position_mm(Z_AXIS);
  430. plan_set_z_position(current_position[Z_AXIS]);
  431. }
  432. // At the current position, find the Z stop.
  433. inline void find_bed_induction_sensor_point_z()
  434. {
  435. bool endstops_enabled = enable_endstops(true);
  436. bool endstop_z_enabled = enable_z_endstop(false);
  437. // move down until you find the bed
  438. current_position[Z_AXIS] = -10;
  439. go_to_current(homing_feedrate[Z_AXIS]/60);
  440. // we have to let the planner know where we are right now as it is not where we said to go.
  441. update_current_position_z();
  442. // move up the retract distance
  443. current_position[Z_AXIS] += home_retract_mm_ext(Z_AXIS);
  444. go_to_current(homing_feedrate[Z_AXIS]/60);
  445. // move back down slowly to find bed
  446. current_position[Z_AXIS] -= home_retract_mm_ext(Z_AXIS) * 2;
  447. go_to_current(homing_feedrate[Z_AXIS]/(4*60));
  448. // we have to let the planner know where we are right now as it is not where we said to go.
  449. update_current_position_z();
  450. enable_endstops(endstops_enabled);
  451. enable_z_endstop(endstop_z_enabled);
  452. }
  453. // Search around the current_position[X,Y],
  454. // look for the induction sensor response.
  455. // Adjust the current_position[X,Y,Z] to the center of the target dot and its response Z coordinate.
  456. #define FIND_BED_INDUCTION_SENSOR_POINT_X_RADIUS (8.f)
  457. #define FIND_BED_INDUCTION_SENSOR_POINT_Y_RADIUS (6.f)
  458. #define FIND_BED_INDUCTION_SENSOR_POINT_XY_STEP (1.f)
  459. #define FIND_BED_INDUCTION_SENSOR_POINT_Z_STEP (0.5f)
  460. inline bool find_bed_induction_sensor_point_xy()
  461. {
  462. float feedrate = homing_feedrate[X_AXIS] / 60.f;
  463. bool found = false;
  464. {
  465. float x0 = current_position[X_AXIS] - FIND_BED_INDUCTION_SENSOR_POINT_X_RADIUS;
  466. float x1 = current_position[X_AXIS] + FIND_BED_INDUCTION_SENSOR_POINT_X_RADIUS;
  467. float y0 = current_position[Y_AXIS] - FIND_BED_INDUCTION_SENSOR_POINT_Y_RADIUS;
  468. float y1 = current_position[Y_AXIS] + FIND_BED_INDUCTION_SENSOR_POINT_Y_RADIUS;
  469. uint8_t nsteps_y;
  470. uint8_t i;
  471. if (x0 < X_MIN_POS)
  472. x0 = X_MIN_POS;
  473. if (x1 > X_MAX_POS)
  474. x1 = X_MAX_POS;
  475. if (y0 < Y_MIN_POS_FOR_BED_CALIBRATION)
  476. y0 = Y_MIN_POS_FOR_BED_CALIBRATION;
  477. if (y1 > Y_MAX_POS)
  478. y1 = Y_MAX_POS;
  479. nsteps_y = int(ceil((y1 - y0) / FIND_BED_INDUCTION_SENSOR_POINT_XY_STEP));
  480. enable_endstops(false);
  481. bool dir_positive = true;
  482. // go_xyz(current_position[X_AXIS], current_position[Y_AXIS], MESH_HOME_Z_SEARCH, homing_feedrate[Z_AXIS]/60);
  483. go_xyz(x0, y0, current_position[Z_AXIS], feedrate);
  484. // Continously lower the Z axis.
  485. endstops_hit_on_purpose();
  486. enable_z_endstop(true);
  487. while (current_position[Z_AXIS] > -10.f) {
  488. // Do nsteps_y zig-zag movements.
  489. current_position[Y_AXIS] = y0;
  490. for (i = 0; i < nsteps_y; current_position[Y_AXIS] += (y1 - y0) / float(nsteps_y - 1), ++ i) {
  491. // Run with a slightly decreasing Z axis, zig-zag movement. Stop at the Z end-stop.
  492. current_position[Z_AXIS] -= FIND_BED_INDUCTION_SENSOR_POINT_Z_STEP / float(nsteps_y);
  493. go_xyz(dir_positive ? x1 : x0, current_position[Y_AXIS], current_position[Z_AXIS], feedrate);
  494. dir_positive = ! dir_positive;
  495. if (endstop_z_hit_on_purpose())
  496. goto endloop;
  497. }
  498. for (i = 0; i < nsteps_y; current_position[Y_AXIS] -= (y1 - y0) / float(nsteps_y - 1), ++ i) {
  499. // Run with a slightly decreasing Z axis, zig-zag movement. Stop at the Z end-stop.
  500. current_position[Z_AXIS] -= FIND_BED_INDUCTION_SENSOR_POINT_Z_STEP / float(nsteps_y);
  501. go_xyz(dir_positive ? x1 : x0, current_position[Y_AXIS], current_position[Z_AXIS], feedrate);
  502. dir_positive = ! dir_positive;
  503. if (endstop_z_hit_on_purpose())
  504. goto endloop;
  505. }
  506. }
  507. endloop:
  508. // SERIAL_ECHOLN("First hit");
  509. // we have to let the planner know where we are right now as it is not where we said to go.
  510. update_current_position_xyz();
  511. // Search in this plane for the first hit. Zig-zag first in X, then in Y axis.
  512. for (int8_t iter = 0; iter < 3; ++ iter) {
  513. if (iter > 0) {
  514. // Slightly lower the Z axis to get a reliable trigger.
  515. current_position[Z_AXIS] -= 0.02f;
  516. go_xyz(current_position[X_AXIS], current_position[Y_AXIS], MESH_HOME_Z_SEARCH, homing_feedrate[Z_AXIS]/60);
  517. }
  518. // Do nsteps_y zig-zag movements.
  519. float a, b;
  520. enable_endstops(false);
  521. enable_z_endstop(false);
  522. current_position[Y_AXIS] = y0;
  523. go_xy(x0, current_position[Y_AXIS], feedrate);
  524. enable_z_endstop(true);
  525. found = false;
  526. for (i = 0, dir_positive = true; i < nsteps_y; current_position[Y_AXIS] += (y1 - y0) / float(nsteps_y - 1), ++ i, dir_positive = ! dir_positive) {
  527. go_xy(dir_positive ? x1 : x0, current_position[Y_AXIS], feedrate);
  528. if (endstop_z_hit_on_purpose()) {
  529. found = true;
  530. break;
  531. }
  532. }
  533. update_current_position_xyz();
  534. if (! found) {
  535. // SERIAL_ECHOLN("Search in Y - not found");
  536. continue;
  537. }
  538. // SERIAL_ECHOLN("Search in Y - found");
  539. a = current_position[Y_AXIS];
  540. enable_z_endstop(false);
  541. current_position[Y_AXIS] = y1;
  542. go_xy(x0, current_position[Y_AXIS], feedrate);
  543. enable_z_endstop(true);
  544. found = false;
  545. for (i = 0, dir_positive = true; i < nsteps_y; current_position[Y_AXIS] -= (y1 - y0) / float(nsteps_y - 1), ++ i, dir_positive = ! dir_positive) {
  546. go_xy(dir_positive ? x1 : x0, current_position[Y_AXIS], feedrate);
  547. if (endstop_z_hit_on_purpose()) {
  548. found = true;
  549. break;
  550. }
  551. }
  552. update_current_position_xyz();
  553. if (! found) {
  554. // SERIAL_ECHOLN("Search in Y2 - not found");
  555. continue;
  556. }
  557. // SERIAL_ECHOLN("Search in Y2 - found");
  558. b = current_position[Y_AXIS];
  559. current_position[Y_AXIS] = 0.5f * (a + b);
  560. // Search in the X direction along a cross.
  561. found = false;
  562. enable_z_endstop(false);
  563. go_xy(x0, current_position[Y_AXIS], feedrate);
  564. enable_z_endstop(true);
  565. go_xy(x1, current_position[Y_AXIS], feedrate);
  566. update_current_position_xyz();
  567. if (! endstop_z_hit_on_purpose()) {
  568. // SERIAL_ECHOLN("Search X span 0 - not found");
  569. continue;
  570. }
  571. // SERIAL_ECHOLN("Search X span 0 - found");
  572. a = current_position[X_AXIS];
  573. enable_z_endstop(false);
  574. go_xy(x1, current_position[Y_AXIS], feedrate);
  575. enable_z_endstop(true);
  576. go_xy(x0, current_position[Y_AXIS], feedrate);
  577. update_current_position_xyz();
  578. if (! endstop_z_hit_on_purpose()) {
  579. // SERIAL_ECHOLN("Search X span 1 - not found");
  580. continue;
  581. }
  582. // SERIAL_ECHOLN("Search X span 1 - found");
  583. b = current_position[X_AXIS];
  584. // Go to the center.
  585. enable_z_endstop(false);
  586. current_position[X_AXIS] = 0.5f * (a + b);
  587. go_xy(current_position[X_AXIS], current_position[Y_AXIS], feedrate);
  588. found = true;
  589. #if 1
  590. // Search in the Y direction along a cross.
  591. found = false;
  592. enable_z_endstop(false);
  593. go_xy(current_position[X_AXIS], y0, feedrate);
  594. enable_z_endstop(true);
  595. go_xy(current_position[X_AXIS], y1, feedrate);
  596. update_current_position_xyz();
  597. if (! endstop_z_hit_on_purpose()) {
  598. // SERIAL_ECHOLN("Search Y2 span 0 - not found");
  599. continue;
  600. }
  601. // SERIAL_ECHOLN("Search Y2 span 0 - found");
  602. a = current_position[Y_AXIS];
  603. enable_z_endstop(false);
  604. go_xy(current_position[X_AXIS], y1, feedrate);
  605. enable_z_endstop(true);
  606. go_xy(current_position[X_AXIS], y0, feedrate);
  607. update_current_position_xyz();
  608. if (! endstop_z_hit_on_purpose()) {
  609. // SERIAL_ECHOLN("Search Y2 span 1 - not found");
  610. continue;
  611. }
  612. // SERIAL_ECHOLN("Search Y2 span 1 - found");
  613. b = current_position[Y_AXIS];
  614. // Go to the center.
  615. enable_z_endstop(false);
  616. current_position[Y_AXIS] = 0.5f * (a + b);
  617. go_xy(current_position[X_AXIS], current_position[Y_AXIS], feedrate);
  618. found = true;
  619. #endif
  620. break;
  621. }
  622. }
  623. enable_z_endstop(false);
  624. return found;
  625. }
  626. // Search around the current_position[X,Y,Z].
  627. // It is expected, that the induction sensor is switched on at the current position.
  628. // Look around this center point by painting a star around the point.
  629. inline bool improve_bed_induction_sensor_point()
  630. {
  631. static const float search_radius = 8.f;
  632. bool endstops_enabled = enable_endstops(false);
  633. bool endstop_z_enabled = enable_z_endstop(false);
  634. bool found = false;
  635. float feedrate = homing_feedrate[X_AXIS] / 60.f;
  636. float center_old_x = current_position[X_AXIS];
  637. float center_old_y = current_position[Y_AXIS];
  638. float center_x = 0.f;
  639. float center_y = 0.f;
  640. for (uint8_t iter = 0; iter < 4; ++ iter) {
  641. switch (iter) {
  642. case 0:
  643. destination[X_AXIS] = center_old_x - search_radius * 0.707;
  644. destination[Y_AXIS] = center_old_y - search_radius * 0.707;
  645. break;
  646. case 1:
  647. destination[X_AXIS] = center_old_x + search_radius * 0.707;
  648. destination[Y_AXIS] = center_old_y + search_radius * 0.707;
  649. break;
  650. case 2:
  651. destination[X_AXIS] = center_old_x + search_radius * 0.707;
  652. destination[Y_AXIS] = center_old_y - search_radius * 0.707;
  653. break;
  654. case 3:
  655. default:
  656. destination[X_AXIS] = center_old_x - search_radius * 0.707;
  657. destination[Y_AXIS] = center_old_y + search_radius * 0.707;
  658. break;
  659. }
  660. // Trim the vector from center_old_[x,y] to destination[x,y] by the bed dimensions.
  661. float vx = destination[X_AXIS] - center_old_x;
  662. float vy = destination[Y_AXIS] - center_old_y;
  663. float l = sqrt(vx*vx+vy*vy);
  664. float t;
  665. if (destination[X_AXIS] < X_MIN_POS) {
  666. // Exiting the bed at xmin.
  667. t = (center_x - X_MIN_POS) / l;
  668. destination[X_AXIS] = X_MIN_POS;
  669. destination[Y_AXIS] = center_old_y + t * vy;
  670. } else if (destination[X_AXIS] > X_MAX_POS) {
  671. // Exiting the bed at xmax.
  672. t = (X_MAX_POS - center_x) / l;
  673. destination[X_AXIS] = X_MAX_POS;
  674. destination[Y_AXIS] = center_old_y + t * vy;
  675. }
  676. if (destination[Y_AXIS] < Y_MIN_POS_FOR_BED_CALIBRATION) {
  677. // Exiting the bed at ymin.
  678. t = (center_y - Y_MIN_POS_FOR_BED_CALIBRATION) / l;
  679. destination[X_AXIS] = center_old_x + t * vx;
  680. destination[Y_AXIS] = Y_MIN_POS_FOR_BED_CALIBRATION;
  681. } else if (destination[Y_AXIS] > Y_MAX_POS) {
  682. // Exiting the bed at xmax.
  683. t = (Y_MAX_POS - center_y) / l;
  684. destination[X_AXIS] = center_old_x + t * vx;
  685. destination[Y_AXIS] = Y_MAX_POS;
  686. }
  687. // Move away from the measurement point.
  688. enable_endstops(false);
  689. go_xy(destination[X_AXIS], destination[Y_AXIS], feedrate);
  690. // Move towards the measurement point, until the induction sensor triggers.
  691. enable_endstops(true);
  692. go_xy(center_old_x, center_old_y, feedrate);
  693. update_current_position_xyz();
  694. // if (! endstop_z_hit_on_purpose()) return false;
  695. center_x += current_position[X_AXIS];
  696. center_y += current_position[Y_AXIS];
  697. }
  698. // Calculate the new center, move to the new center.
  699. center_x /= 4.f;
  700. center_y /= 4.f;
  701. current_position[X_AXIS] = center_x;
  702. current_position[Y_AXIS] = center_y;
  703. enable_endstops(false);
  704. go_xy(current_position[X_AXIS], current_position[Y_AXIS], feedrate);
  705. enable_endstops(endstops_enabled);
  706. enable_z_endstop(endstop_z_enabled);
  707. return found;
  708. }
  709. // Search around the current_position[X,Y,Z].
  710. // It is expected, that the induction sensor is switched on at the current position.
  711. // Look around this center point by painting a star around the point.
  712. #define IMPROVE_BED_INDUCTION_SENSOR_SEARCH_RADIUS (8.f)
  713. inline bool improve_bed_induction_sensor_point2(bool lift_z_on_min_y)
  714. {
  715. float center_old_x = current_position[X_AXIS];
  716. float center_old_y = current_position[Y_AXIS];
  717. float a, b;
  718. enable_endstops(false);
  719. {
  720. float x0 = center_old_x - IMPROVE_BED_INDUCTION_SENSOR_SEARCH_RADIUS;
  721. float x1 = center_old_x + IMPROVE_BED_INDUCTION_SENSOR_SEARCH_RADIUS;
  722. if (x0 < X_MIN_POS)
  723. x0 = X_MIN_POS;
  724. if (x1 > X_MAX_POS)
  725. x1 = X_MAX_POS;
  726. // Search in the X direction along a cross.
  727. enable_z_endstop(false);
  728. go_xy(x0, current_position[Y_AXIS], homing_feedrate[X_AXIS] / 60.f);
  729. enable_z_endstop(true);
  730. go_xy(x1, current_position[Y_AXIS], homing_feedrate[X_AXIS] / 60.f);
  731. update_current_position_xyz();
  732. if (! endstop_z_hit_on_purpose()) {
  733. current_position[X_AXIS] = center_old_x;
  734. goto canceled;
  735. }
  736. a = current_position[X_AXIS];
  737. enable_z_endstop(false);
  738. go_xy(x1, current_position[Y_AXIS], homing_feedrate[X_AXIS] / 60.f);
  739. enable_z_endstop(true);
  740. go_xy(x0, current_position[Y_AXIS], homing_feedrate[X_AXIS] / 60.f);
  741. update_current_position_xyz();
  742. if (! endstop_z_hit_on_purpose()) {
  743. current_position[X_AXIS] = center_old_x;
  744. goto canceled;
  745. }
  746. b = current_position[X_AXIS];
  747. // Go to the center.
  748. enable_z_endstop(false);
  749. current_position[X_AXIS] = 0.5f * (a + b);
  750. go_xy(current_position[X_AXIS], current_position[Y_AXIS], homing_feedrate[X_AXIS] / 60.f);
  751. }
  752. {
  753. float y0 = center_old_y - IMPROVE_BED_INDUCTION_SENSOR_SEARCH_RADIUS;
  754. float y1 = center_old_y + IMPROVE_BED_INDUCTION_SENSOR_SEARCH_RADIUS;
  755. if (y0 < Y_MIN_POS_FOR_BED_CALIBRATION)
  756. y0 = Y_MIN_POS_FOR_BED_CALIBRATION;
  757. if (y1 > Y_MAX_POS)
  758. y1 = Y_MAX_POS;
  759. // Search in the Y direction along a cross.
  760. enable_z_endstop(false);
  761. go_xy(current_position[X_AXIS], y0, homing_feedrate[X_AXIS] / 60.f);
  762. if (lift_z_on_min_y) {
  763. // The first row of points are very close to the end stop.
  764. // Lift the sensor to disengage the trigger. This is necessary because of the sensor hysteresis.
  765. go_xyz(current_position[X_AXIS], y0, current_position[Z_AXIS]+1.5f, homing_feedrate[Z_AXIS] / 60.f);
  766. // and go back.
  767. go_xyz(current_position[X_AXIS], y0, current_position[Z_AXIS], homing_feedrate[Z_AXIS] / 60.f);
  768. }
  769. if (lift_z_on_min_y && (READ(Z_MIN_PIN) ^ Z_MIN_ENDSTOP_INVERTING) == 1) {
  770. // Already triggering before we started the move.
  771. // Shift the trigger point slightly outwards.
  772. // a = current_position[Y_AXIS] - 1.5f;
  773. a = current_position[Y_AXIS];
  774. } else {
  775. enable_z_endstop(true);
  776. go_xy(current_position[X_AXIS], y1, homing_feedrate[X_AXIS] / 60.f);
  777. update_current_position_xyz();
  778. if (! endstop_z_hit_on_purpose()) {
  779. current_position[Y_AXIS] = center_old_y;
  780. goto canceled;
  781. }
  782. a = current_position[Y_AXIS];
  783. }
  784. enable_z_endstop(false);
  785. go_xy(current_position[X_AXIS], y1, homing_feedrate[X_AXIS] / 60.f);
  786. enable_z_endstop(true);
  787. go_xy(current_position[X_AXIS], y0, homing_feedrate[X_AXIS] / 60.f);
  788. update_current_position_xyz();
  789. if (! endstop_z_hit_on_purpose()) {
  790. current_position[Y_AXIS] = center_old_y;
  791. goto canceled;
  792. }
  793. b = current_position[Y_AXIS];
  794. // Go to the center.
  795. enable_z_endstop(false);
  796. current_position[Y_AXIS] = 0.5f * (a + b);
  797. go_xy(current_position[X_AXIS], current_position[Y_AXIS], homing_feedrate[X_AXIS] / 60.f);
  798. }
  799. return true;
  800. canceled:
  801. // Go back to the center.
  802. enable_z_endstop(false);
  803. go_xy(current_position[X_AXIS], current_position[Y_AXIS], homing_feedrate[X_AXIS] / 60.f);
  804. return false;
  805. }
  806. // Searching the front points, where one cannot move the sensor head in front of the sensor point.
  807. // Searching in a zig-zag movement in a plane for the maximum width of the response.
  808. #define IMPROVE_BED_INDUCTION_SENSOR_POINT3_SEARCH_RADIUS (4.f)
  809. #define IMPROVE_BED_INDUCTION_SENSOR_POINT3_SEARCH_STEP_FINE_Y (0.1f)
  810. inline bool improve_bed_induction_sensor_point3()
  811. {
  812. float center_old_x = current_position[X_AXIS];
  813. float center_old_y = current_position[Y_AXIS];
  814. float x0 = center_old_x - IMPROVE_BED_INDUCTION_SENSOR_POINT3_SEARCH_RADIUS;
  815. float x1 = center_old_x + IMPROVE_BED_INDUCTION_SENSOR_POINT3_SEARCH_RADIUS;
  816. float y0 = center_old_y - IMPROVE_BED_INDUCTION_SENSOR_POINT3_SEARCH_RADIUS;
  817. float y1 = center_old_y + IMPROVE_BED_INDUCTION_SENSOR_POINT3_SEARCH_RADIUS;
  818. float y = y0;
  819. float a, b;
  820. if (x0 < X_MIN_POS)
  821. x0 = X_MIN_POS;
  822. if (x1 > X_MAX_POS)
  823. x1 = X_MAX_POS;
  824. if (y0 < Y_MIN_POS_FOR_BED_CALIBRATION)
  825. y0 = Y_MIN_POS_FOR_BED_CALIBRATION;
  826. if (y1 > Y_MAX_POS)
  827. y1 = Y_MAX_POS;
  828. #if 0
  829. SERIAL_ECHOPGM("Initial position: ");
  830. SERIAL_ECHO(center_old_x);
  831. SERIAL_ECHOPGM(", ");
  832. SERIAL_ECHO(center_old_y);
  833. SERIAL_ECHOLNPGM("");
  834. #endif
  835. // Search in the X direction along a cross.
  836. float dmax = 0.;
  837. float xmax1 = 0.;
  838. float xmax2 = 0.;
  839. for (float y = y0; y < y1; y += IMPROVE_BED_INDUCTION_SENSOR_POINT3_SEARCH_STEP_FINE_Y) {
  840. enable_z_endstop(false);
  841. go_xy(x0, y, homing_feedrate[X_AXIS] / 60.f);
  842. enable_z_endstop(true);
  843. go_xy(x1, y, homing_feedrate[X_AXIS] / 60.f);
  844. update_current_position_xyz();
  845. if (! endstop_z_hit_on_purpose()) {
  846. // SERIAL_PROTOCOLPGM("Failed 1\n");
  847. current_position[X_AXIS] = center_old_x;
  848. goto canceled;
  849. }
  850. a = current_position[X_AXIS];
  851. enable_z_endstop(false);
  852. go_xy(x1, y, homing_feedrate[X_AXIS] / 60.f);
  853. enable_z_endstop(true);
  854. go_xy(x0, y, homing_feedrate[X_AXIS] / 60.f);
  855. update_current_position_xyz();
  856. if (! endstop_z_hit_on_purpose()) {
  857. // SERIAL_PROTOCOLPGM("Failed 2\n");
  858. current_position[X_AXIS] = center_old_x;
  859. goto canceled;
  860. }
  861. b = current_position[X_AXIS];
  862. float d = b - a;
  863. if (d > dmax) {
  864. xmax1 = 0.5f * (a + b);
  865. dmax = d;
  866. } else {
  867. y0 = y - IMPROVE_BED_INDUCTION_SENSOR_POINT3_SEARCH_STEP_FINE_Y;
  868. break;
  869. }
  870. }
  871. if (dmax == 0.) {
  872. // SERIAL_PROTOCOLPGM("failed - not found\n");
  873. goto canceled;
  874. }
  875. // SERIAL_PROTOCOLPGM("ok 1\n");
  876. // Search in the X direction along a cross.
  877. dmax = 0.;
  878. if (y0 + 1.f < y1)
  879. y1 = y0 + 1.f;
  880. for (float y = y1; y >= y0; y -= IMPROVE_BED_INDUCTION_SENSOR_POINT3_SEARCH_STEP_FINE_Y) {
  881. enable_z_endstop(false);
  882. go_xy(x0, y, homing_feedrate[X_AXIS] / 60.f);
  883. enable_z_endstop(true);
  884. go_xy(x1, y, homing_feedrate[X_AXIS] / 60.f);
  885. update_current_position_xyz();
  886. if (! endstop_z_hit_on_purpose()) {
  887. continue;
  888. /*
  889. current_position[X_AXIS] = center_old_x;
  890. SERIAL_PROTOCOLPGM("Failed 3\n");
  891. goto canceled;
  892. */
  893. }
  894. a = current_position[X_AXIS];
  895. enable_z_endstop(false);
  896. go_xy(x1, y, homing_feedrate[X_AXIS] / 60.f);
  897. enable_z_endstop(true);
  898. go_xy(x0, y, homing_feedrate[X_AXIS] / 60.f);
  899. update_current_position_xyz();
  900. if (! endstop_z_hit_on_purpose()) {
  901. continue;
  902. /*
  903. current_position[X_AXIS] = center_old_x;
  904. SERIAL_PROTOCOLPGM("Failed 4\n");
  905. goto canceled;
  906. */
  907. }
  908. b = current_position[X_AXIS];
  909. float d = b - a;
  910. if (d > dmax) {
  911. xmax2 = 0.5f * (a + b);
  912. dmax = d;
  913. } else {
  914. y1 = y + IMPROVE_BED_INDUCTION_SENSOR_POINT3_SEARCH_STEP_FINE_Y;
  915. break;
  916. }
  917. }
  918. // SERIAL_PROTOCOLPGM("ok 2\n");
  919. // Go to the center.
  920. enable_z_endstop(false);
  921. if (dmax == 0.f) {
  922. // Found only the point going from ymin to ymax.
  923. current_position[X_AXIS] = xmax1;
  924. current_position[Y_AXIS] = y0;
  925. } else {
  926. // Both points found (from ymin to ymax and from ymax to ymin).
  927. float p = 0.5f;
  928. // If the first hit was on the machine boundary,
  929. // give it a higher weight.
  930. if (y0 == Y_MIN_POS_FOR_BED_CALIBRATION)
  931. p = 0.75f;
  932. current_position[X_AXIS] = p * xmax1 + (1.f - p) * xmax2;
  933. current_position[Y_AXIS] = p * y0 + (1.f - p) * y1;
  934. }
  935. /*
  936. SERIAL_ECHOPGM("Adjusted position: ");
  937. SERIAL_ECHO(current_position[X_AXIS]);
  938. SERIAL_ECHOPGM(", ");
  939. SERIAL_ECHO(current_position[Y_AXIS]);
  940. SERIAL_ECHOLNPGM("");
  941. */
  942. go_xy(current_position[X_AXIS], current_position[Y_AXIS], homing_feedrate[X_AXIS] / 60.f);
  943. // delay_keep_alive(3000);
  944. return true;
  945. canceled:
  946. // Go back to the center.
  947. enable_z_endstop(false);
  948. go_xy(current_position[X_AXIS], current_position[Y_AXIS], homing_feedrate[X_AXIS] / 60.f);
  949. return false;
  950. }
  951. #define MESH_BED_CALIBRATION_SHOW_LCD
  952. bool find_bed_offset_and_skew(int8_t verbosity_level)
  953. {
  954. // Don't let the manage_inactivity() function remove power from the motors.
  955. refresh_cmd_timeout();
  956. // Reusing the z_values memory for the measurement cache.
  957. // 7x7=49 floats, good for 16 (x,y,z) vectors.
  958. float *pts = &mbl.z_values[0][0];
  959. float *vec_x = pts + 2 * 4;
  960. float *vec_y = vec_x + 2;
  961. float *cntr = vec_y + 2;
  962. memset(pts, 0, sizeof(float) * 7 * 7);
  963. // SERIAL_ECHOLNPGM("find_bed_offset_and_skew verbosity level: ");
  964. // SERIAL_ECHO(int(verbosity_level));
  965. // SERIAL_ECHOPGM("");
  966. #ifdef MESH_BED_CALIBRATION_SHOW_LCD
  967. lcd_implementation_clear();
  968. lcd_print_at_PGM(0, 0, MSG_FIND_BED_OFFSET_AND_SKEW_LINE1);
  969. #endif /* MESH_BED_CALIBRATION_SHOW_LCD */
  970. // Collect the rear 2x3 points.
  971. current_position[Z_AXIS] = MESH_HOME_Z_SEARCH;
  972. for (int k = 0; k < 4; ++ k) {
  973. // Don't let the manage_inactivity() function remove power from the motors.
  974. refresh_cmd_timeout();
  975. #ifdef MESH_BED_CALIBRATION_SHOW_LCD
  976. lcd_print_at_PGM(0, 1, MSG_FIND_BED_OFFSET_AND_SKEW_LINE2);
  977. lcd_implementation_print_at(0, 2, k+1);
  978. lcd_printPGM(MSG_FIND_BED_OFFSET_AND_SKEW_LINE3);
  979. #endif /* MESH_BED_CALIBRATION_SHOW_LCD */
  980. float *pt = pts + k * 2;
  981. // Go up to z_initial.
  982. go_to_current(homing_feedrate[Z_AXIS] / 60.f);
  983. if (verbosity_level >= 20) {
  984. // Go to Y0, wait, then go to Y-4.
  985. current_position[Y_AXIS] = 0.f;
  986. go_to_current(homing_feedrate[X_AXIS] / 60.f);
  987. SERIAL_ECHOLNPGM("At Y0");
  988. delay_keep_alive(5000);
  989. current_position[Y_AXIS] = Y_MIN_POS;
  990. go_to_current(homing_feedrate[X_AXIS] / 60.f);
  991. SERIAL_ECHOLNPGM("At Y-4");
  992. delay_keep_alive(5000);
  993. }
  994. // Go to the measurement point position.
  995. current_position[X_AXIS] = pgm_read_float(bed_ref_points_4+k*2);
  996. current_position[Y_AXIS] = pgm_read_float(bed_ref_points_4+k*2+1);
  997. go_to_current(homing_feedrate[X_AXIS] / 60.f);
  998. if (verbosity_level >= 10)
  999. delay_keep_alive(3000);
  1000. if (! find_bed_induction_sensor_point_xy())
  1001. return false;
  1002. find_bed_induction_sensor_point_z();
  1003. #if 1
  1004. if (k == 0) {
  1005. int8_t i = 4;
  1006. for (;;) {
  1007. if (improve_bed_induction_sensor_point3())
  1008. break;
  1009. if (-- i == 0)
  1010. return false;
  1011. // Try to move the Z axis down a bit to increase a chance of the sensor to trigger.
  1012. current_position[Z_AXIS] -= 0.025f;
  1013. enable_endstops(false);
  1014. enable_z_endstop(false);
  1015. go_to_current(homing_feedrate[Z_AXIS]);
  1016. }
  1017. if (i == 0)
  1018. // not found
  1019. return false;
  1020. }
  1021. #endif
  1022. if (verbosity_level >= 10)
  1023. delay_keep_alive(3000);
  1024. pt[0] = current_position[X_AXIS];
  1025. pt[1] = current_position[Y_AXIS];
  1026. // Start searching for the other points at 3mm above the last point.
  1027. current_position[Z_AXIS] += 3.f;
  1028. cntr[0] += pt[0];
  1029. cntr[1] += pt[1];
  1030. if (verbosity_level >= 10 && k == 0) {
  1031. // Show the zero. Test, whether the Y motor skipped steps.
  1032. current_position[Y_AXIS] = MANUAL_Y_HOME_POS;
  1033. go_to_current(homing_feedrate[X_AXIS] / 60.f);
  1034. delay_keep_alive(3000);
  1035. }
  1036. }
  1037. if (verbosity_level >= 20) {
  1038. // Test the positions. Are the positions reproducible? Now the calibration is active in the planner.
  1039. delay_keep_alive(3000);
  1040. for (int8_t mesh_point = 0; mesh_point < 4; ++ mesh_point) {
  1041. // Don't let the manage_inactivity() function remove power from the motors.
  1042. refresh_cmd_timeout();
  1043. // Go to the measurement point.
  1044. // Use the coorrected coordinate, which is a result of find_bed_offset_and_skew().
  1045. current_position[X_AXIS] = pts[mesh_point*2];
  1046. current_position[Y_AXIS] = pts[mesh_point*2+1];
  1047. go_to_current(homing_feedrate[X_AXIS]/60);
  1048. delay_keep_alive(3000);
  1049. }
  1050. }
  1051. calculate_machine_skew_and_offset_LS(pts, 4, bed_ref_points_4, vec_x, vec_y, cntr, verbosity_level);
  1052. world2machine_rotation_and_skew[0][0] = vec_x[0];
  1053. world2machine_rotation_and_skew[1][0] = vec_x[1];
  1054. world2machine_rotation_and_skew[0][1] = vec_y[0];
  1055. world2machine_rotation_and_skew[1][1] = vec_y[1];
  1056. world2machine_shift[0] = cntr[0];
  1057. world2machine_shift[1] = cntr[1];
  1058. #if 1
  1059. // Fearlessly store the calibration values into the eeprom.
  1060. eeprom_update_float((float*)(EEPROM_BED_CALIBRATION_CENTER+0), cntr [0]);
  1061. eeprom_update_float((float*)(EEPROM_BED_CALIBRATION_CENTER+4), cntr [1]);
  1062. eeprom_update_float((float*)(EEPROM_BED_CALIBRATION_VEC_X +0), vec_x[0]);
  1063. eeprom_update_float((float*)(EEPROM_BED_CALIBRATION_VEC_X +4), vec_x[1]);
  1064. eeprom_update_float((float*)(EEPROM_BED_CALIBRATION_VEC_Y +0), vec_y[0]);
  1065. eeprom_update_float((float*)(EEPROM_BED_CALIBRATION_VEC_Y +4), vec_y[1]);
  1066. #endif
  1067. // Correct the current_position to match the transformed coordinate system after world2machine_rotation_and_skew and world2machine_shift were set.
  1068. world2machine_update_current();
  1069. if (verbosity_level >= 20) {
  1070. // Test the positions. Are the positions reproducible? Now the calibration is active in the planner.
  1071. delay_keep_alive(3000);
  1072. for (int8_t mesh_point = 0; mesh_point < 9; ++ mesh_point) {
  1073. // Don't let the manage_inactivity() function remove power from the motors.
  1074. refresh_cmd_timeout();
  1075. // Go to the measurement point.
  1076. // Use the coorrected coordinate, which is a result of find_bed_offset_and_skew().
  1077. current_position[X_AXIS] = pgm_read_float(bed_ref_points+mesh_point*2);
  1078. current_position[Y_AXIS] = pgm_read_float(bed_ref_points+mesh_point*2+1);
  1079. go_to_current(homing_feedrate[X_AXIS]/60);
  1080. delay_keep_alive(3000);
  1081. }
  1082. }
  1083. return true;
  1084. }
  1085. bool improve_bed_offset_and_skew(int8_t method, int8_t verbosity_level)
  1086. {
  1087. // Don't let the manage_inactivity() function remove power from the motors.
  1088. refresh_cmd_timeout();
  1089. // Reusing the z_values memory for the measurement cache.
  1090. // 7x7=49 floats, good for 16 (x,y,z) vectors.
  1091. float *pts = &mbl.z_values[0][0];
  1092. float *vec_x = pts + 2 * 9;
  1093. float *vec_y = vec_x + 2;
  1094. float *cntr = vec_y + 2;
  1095. memset(pts, 0, sizeof(float) * 7 * 7);
  1096. // Cache the current correction matrix.
  1097. world2machine_initialize();
  1098. vec_x[0] = world2machine_rotation_and_skew[0][0];
  1099. vec_x[1] = world2machine_rotation_and_skew[1][0];
  1100. vec_y[0] = world2machine_rotation_and_skew[0][1];
  1101. vec_y[1] = world2machine_rotation_and_skew[1][1];
  1102. cntr[0] = world2machine_shift[0];
  1103. cntr[1] = world2machine_shift[1];
  1104. // and reset the correction matrix, so the planner will not do anything.
  1105. world2machine_reset();
  1106. bool endstops_enabled = enable_endstops(false);
  1107. bool endstop_z_enabled = enable_z_endstop(false);
  1108. #ifdef MESH_BED_CALIBRATION_SHOW_LCD
  1109. lcd_implementation_clear();
  1110. lcd_print_at_PGM(0, 0, MSG_IMPROVE_BED_OFFSET_AND_SKEW_LINE1);
  1111. #endif /* MESH_BED_CALIBRATION_SHOW_LCD */
  1112. // Collect a matrix of 9x9 points.
  1113. for (int8_t mesh_point = 0; mesh_point < 9; ++ mesh_point) {
  1114. // Don't let the manage_inactivity() function remove power from the motors.
  1115. refresh_cmd_timeout();
  1116. // Print the decrasing ID of the measurement point.
  1117. #ifdef MESH_BED_CALIBRATION_SHOW_LCD
  1118. lcd_print_at_PGM(0, 1, MSG_IMPROVE_BED_OFFSET_AND_SKEW_LINE2);
  1119. lcd_implementation_print_at(0, 2, mesh_point+1);
  1120. lcd_printPGM(MSG_IMPROVE_BED_OFFSET_AND_SKEW_LINE3);
  1121. #endif /* MESH_BED_CALIBRATION_SHOW_LCD */
  1122. // Move up.
  1123. current_position[Z_AXIS] = MESH_HOME_Z_SEARCH;
  1124. enable_endstops(false);
  1125. enable_z_endstop(false);
  1126. go_to_current(homing_feedrate[Z_AXIS]/60);
  1127. if (verbosity_level >= 20) {
  1128. // Go to Y0, wait, then go to Y-4.
  1129. current_position[Y_AXIS] = 0.f;
  1130. go_to_current(homing_feedrate[X_AXIS] / 60.f);
  1131. SERIAL_ECHOLNPGM("At Y0");
  1132. delay_keep_alive(5000);
  1133. current_position[Y_AXIS] = Y_MIN_POS;
  1134. go_to_current(homing_feedrate[X_AXIS] / 60.f);
  1135. SERIAL_ECHOLNPGM("At Y-4");
  1136. delay_keep_alive(5000);
  1137. }
  1138. // Go to the measurement point.
  1139. // Use the coorrected coordinate, which is a result of find_bed_offset_and_skew().
  1140. current_position[X_AXIS] = vec_x[0] * pgm_read_float(bed_ref_points+mesh_point*2) + vec_y[0] * pgm_read_float(bed_ref_points+mesh_point*2+1) + cntr[0];
  1141. current_position[Y_AXIS] = vec_x[1] * pgm_read_float(bed_ref_points+mesh_point*2) + vec_y[1] * pgm_read_float(bed_ref_points+mesh_point*2+1) + cntr[1];
  1142. // The calibration points are very close to the min Y.
  1143. if (current_position[Y_AXIS] < Y_MIN_POS_FOR_BED_CALIBRATION)
  1144. current_position[Y_AXIS] = Y_MIN_POS_FOR_BED_CALIBRATION;
  1145. go_to_current(homing_feedrate[X_AXIS]/60);
  1146. // Find its Z position by running the normal vertical search.
  1147. if (verbosity_level >= 10)
  1148. delay_keep_alive(3000);
  1149. find_bed_induction_sensor_point_z();
  1150. if (verbosity_level >= 10)
  1151. delay_keep_alive(3000);
  1152. // Improve the point position by searching its center in a current plane.
  1153. int8_t n_errors = 3;
  1154. for (int8_t iter = 0; iter < 8; ) {
  1155. bool found = false;
  1156. if (mesh_point < 3) {
  1157. found = improve_bed_induction_sensor_point3();
  1158. } else {
  1159. switch (method) {
  1160. case 0: found = improve_bed_induction_sensor_point(); break;
  1161. case 1: found = improve_bed_induction_sensor_point2(mesh_point < 3); break;
  1162. default: break;
  1163. }
  1164. }
  1165. if (found) {
  1166. if (iter > 3) {
  1167. // Average the last 4 measurements.
  1168. pts[mesh_point*2 ] += current_position[X_AXIS];
  1169. pts[mesh_point*2+1] += current_position[Y_AXIS];
  1170. }
  1171. ++ iter;
  1172. } else if (n_errors -- == 0) {
  1173. // Give up.
  1174. goto canceled;
  1175. } else {
  1176. // Try to move the Z axis down a bit to increase a chance of the sensor to trigger.
  1177. current_position[Z_AXIS] -= 0.025f;
  1178. enable_endstops(false);
  1179. enable_z_endstop(false);
  1180. go_to_current(homing_feedrate[Z_AXIS]);
  1181. }
  1182. }
  1183. if (verbosity_level >= 10)
  1184. delay_keep_alive(3000);
  1185. }
  1186. // Don't let the manage_inactivity() function remove power from the motors.
  1187. refresh_cmd_timeout();
  1188. // Average the last 4 measurements.
  1189. for (int8_t i = 0; i < 18; ++ i)
  1190. pts[i] *= (1.f/4.f);
  1191. enable_endstops(false);
  1192. enable_z_endstop(false);
  1193. if (verbosity_level >= 10) {
  1194. // Test the positions. Are the positions reproducible?
  1195. for (int8_t mesh_point = 0; mesh_point < 9; ++ mesh_point) {
  1196. // Don't let the manage_inactivity() function remove power from the motors.
  1197. refresh_cmd_timeout();
  1198. // Go to the measurement point.
  1199. // Use the coorrected coordinate, which is a result of find_bed_offset_and_skew().
  1200. current_position[X_AXIS] = pts[mesh_point*2];
  1201. current_position[Y_AXIS] = pts[mesh_point*2+1];
  1202. go_to_current(homing_feedrate[X_AXIS]/60);
  1203. delay_keep_alive(3000);
  1204. }
  1205. }
  1206. calculate_machine_skew_and_offset_LS(pts, 9, bed_ref_points, vec_x, vec_y, cntr, verbosity_level);
  1207. world2machine_rotation_and_skew[0][0] = vec_x[0];
  1208. world2machine_rotation_and_skew[1][0] = vec_x[1];
  1209. world2machine_rotation_and_skew[0][1] = vec_y[0];
  1210. world2machine_rotation_and_skew[1][1] = vec_y[1];
  1211. world2machine_shift[0] = cntr[0];
  1212. world2machine_shift[1] = cntr[1];
  1213. #if 1
  1214. // Fearlessly store the calibration values into the eeprom.
  1215. eeprom_update_float((float*)(EEPROM_BED_CALIBRATION_CENTER+0), cntr [0]);
  1216. eeprom_update_float((float*)(EEPROM_BED_CALIBRATION_CENTER+4), cntr [1]);
  1217. eeprom_update_float((float*)(EEPROM_BED_CALIBRATION_VEC_X +0), vec_x[0]);
  1218. eeprom_update_float((float*)(EEPROM_BED_CALIBRATION_VEC_X +4), vec_x[1]);
  1219. eeprom_update_float((float*)(EEPROM_BED_CALIBRATION_VEC_Y +0), vec_y[0]);
  1220. eeprom_update_float((float*)(EEPROM_BED_CALIBRATION_VEC_Y +4), vec_y[1]);
  1221. #endif
  1222. // Correct the current_position to match the transformed coordinate system after world2machine_rotation_and_skew and world2machine_shift were set.
  1223. world2machine_update_current();
  1224. enable_endstops(false);
  1225. enable_z_endstop(false);
  1226. if (verbosity_level >= 10) {
  1227. // Test the positions. Are the positions reproducible? Now the calibration is active in the planner.
  1228. delay_keep_alive(3000);
  1229. for (int8_t mesh_point = 0; mesh_point < 9; ++ mesh_point) {
  1230. // Don't let the manage_inactivity() function remove power from the motors.
  1231. refresh_cmd_timeout();
  1232. // Go to the measurement point.
  1233. // Use the coorrected coordinate, which is a result of find_bed_offset_and_skew().
  1234. current_position[X_AXIS] = pgm_read_float(bed_ref_points+mesh_point*2);
  1235. current_position[Y_AXIS] = pgm_read_float(bed_ref_points+mesh_point*2+1);
  1236. go_to_current(homing_feedrate[X_AXIS]/60);
  1237. delay_keep_alive(3000);
  1238. }
  1239. }
  1240. // Don't let the manage_inactivity() function remove power from the motors.
  1241. refresh_cmd_timeout();
  1242. enable_endstops(endstops_enabled);
  1243. enable_z_endstop(endstop_z_enabled);
  1244. return true;
  1245. canceled:
  1246. // Don't let the manage_inactivity() function remove power from the motors.
  1247. refresh_cmd_timeout();
  1248. // Store the identity matrix to EEPROM.
  1249. reset_bed_offset_and_skew();
  1250. enable_endstops(endstops_enabled);
  1251. enable_z_endstop(endstop_z_enabled);
  1252. return false;
  1253. }