| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | # Printer stepper support | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Copyright (C) 2016  Kevin O'Connor <kevin@koconnor.net> | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # This file may be distributed under the terms of the GNU GPLv3 license. | 
					
						
							|  |  |  | import math, logging | 
					
						
							| 
									
										
										
										
											2016-11-18 14:35:31 -05:00
										 |  |  | import homing | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | class PrinterStepper: | 
					
						
							| 
									
										
										
										
											2016-11-17 17:24:03 -05:00
										 |  |  |     def __init__(self, printer, config, name): | 
					
						
							|  |  |  |         self.name = name | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.step_dist = config.getfloat('step_distance') | 
					
						
							|  |  |  |         self.inv_step_dist = 1. / self.step_dist | 
					
						
							| 
									
										
										
										
											2016-12-01 15:29:26 -05:00
										 |  |  |         self.min_stop_interval = 0. | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         self.homing_speed = config.getfloat('homing_speed', 5.0) | 
					
						
							|  |  |  |         self.homing_positive_dir = config.getboolean( | 
					
						
							|  |  |  |             'homing_positive_dir', False) | 
					
						
							|  |  |  |         self.homing_retract_dist = config.getfloat('homing_retract_dist', 5.) | 
					
						
							| 
									
										
										
										
											2016-11-30 15:39:36 -05:00
										 |  |  |         self.homing_stepper_phases = config.getint('homing_stepper_phases', None) | 
					
						
							|  |  |  |         self.homing_endstop_phase = config.getint('homing_endstop_phase', None) | 
					
						
							|  |  |  |         endstop_accuracy = config.getfloat('homing_endstop_accuracy', None) | 
					
						
							| 
									
										
										
										
											2016-07-26 22:06:14 -04:00
										 |  |  |         self.homing_endstop_accuracy = None | 
					
						
							|  |  |  |         if self.homing_stepper_phases: | 
					
						
							|  |  |  |             if endstop_accuracy is None: | 
					
						
							|  |  |  |                 self.homing_endstop_accuracy = self.homing_stepper_phases//2 - 1 | 
					
						
							|  |  |  |             elif self.homing_endstop_phase is not None: | 
					
						
							|  |  |  |                 self.homing_endstop_accuracy = int(math.ceil( | 
					
						
							|  |  |  |                     endstop_accuracy * self.inv_step_dist / 2.)) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self.homing_endstop_accuracy = int(math.ceil( | 
					
						
							|  |  |  |                     endstop_accuracy * self.inv_step_dist)) | 
					
						
							|  |  |  |             if self.homing_endstop_accuracy >= self.homing_stepper_phases/2: | 
					
						
							|  |  |  |                 logging.info("Endstop for %s is not accurate enough for stepper" | 
					
						
							| 
									
										
										
										
											2017-03-12 22:43:05 -04:00
										 |  |  |                              " phase adjustment" % (name,)) | 
					
						
							| 
									
										
										
										
											2016-07-26 22:06:14 -04:00
										 |  |  |                 self.homing_stepper_phases = None | 
					
						
							| 
									
										
										
										
											2017-03-31 14:38:09 -04:00
										 |  |  |             if printer.mcu.is_fileoutput(): | 
					
						
							|  |  |  |                 self.homing_endstop_accuracy = self.homing_stepper_phases | 
					
						
							| 
									
										
										
										
											2016-11-30 15:39:36 -05:00
										 |  |  |         self.position_min = self.position_endstop = self.position_max = None | 
					
						
							| 
									
										
										
										
											2017-03-12 22:43:05 -04:00
										 |  |  |         endstop_pin = config.get('endstop_pin', None) | 
					
						
							|  |  |  |         step_pin = config.get('step_pin') | 
					
						
							|  |  |  |         dir_pin = config.get('dir_pin') | 
					
						
							|  |  |  |         mcu = printer.mcu | 
					
						
							|  |  |  |         self.mcu_stepper = mcu.create_stepper(step_pin, dir_pin) | 
					
						
							|  |  |  |         enable_pin = config.get('enable_pin', None) | 
					
						
							|  |  |  |         if enable_pin is not None: | 
					
						
							|  |  |  |             self.mcu_enable = mcu.create_digital_out(enable_pin, 0) | 
					
						
							|  |  |  |         if endstop_pin is not None: | 
					
						
							| 
									
										
										
										
											2017-03-05 15:30:04 -05:00
										 |  |  |             self.mcu_endstop = mcu.create_endstop(endstop_pin) | 
					
						
							|  |  |  |             self.mcu_endstop.add_stepper(self.mcu_stepper) | 
					
						
							| 
									
										
										
										
											2016-11-30 15:39:36 -05:00
										 |  |  |             self.position_min = config.getfloat('position_min', 0.) | 
					
						
							|  |  |  |             self.position_endstop = config.getfloat('position_endstop') | 
					
						
							|  |  |  |             self.position_max = config.getfloat('position_max', 0.) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |         self.need_motor_enable = True | 
					
						
							| 
									
										
										
										
											2017-03-15 23:08:29 -04:00
										 |  |  |     def _dist_to_time(self, dist, start_velocity, accel): | 
					
						
							|  |  |  |         # Calculate the time it takes to travel a distance with constant accel | 
					
						
							|  |  |  |         time_offset = start_velocity / accel | 
					
						
							|  |  |  |         return math.sqrt(2. * dist / accel + time_offset**2) - time_offset | 
					
						
							| 
									
										
										
										
											2016-12-01 15:29:26 -05:00
										 |  |  |     def set_max_jerk(self, max_halt_velocity, max_accel): | 
					
						
							| 
									
										
										
										
											2017-03-15 23:08:29 -04:00
										 |  |  |         # Calculate the firmware's maximum halt interval time | 
					
						
							|  |  |  |         last_step_time = self._dist_to_time( | 
					
						
							|  |  |  |             self.step_dist, max_halt_velocity, max_accel) | 
					
						
							|  |  |  |         second_last_step_time = self._dist_to_time( | 
					
						
							|  |  |  |             2. * self.step_dist, max_halt_velocity, max_accel) | 
					
						
							|  |  |  |         min_stop_interval = second_last_step_time - last_step_time | 
					
						
							| 
									
										
										
										
											2017-03-12 22:43:05 -04:00
										 |  |  |         self.mcu_stepper.set_min_stop_interval(min_stop_interval) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |     def motor_enable(self, move_time, enable=0): | 
					
						
							| 
									
										
										
										
											2016-11-14 13:40:35 -05:00
										 |  |  |         if enable and self.need_motor_enable: | 
					
						
							|  |  |  |             mcu_time = self.mcu_stepper.print_to_mcu_time(move_time) | 
					
						
							|  |  |  |             self.mcu_stepper.reset_step_clock(mcu_time) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |         if (self.mcu_enable is not None | 
					
						
							|  |  |  |             and self.mcu_enable.get_last_setting() != enable): | 
					
						
							| 
									
										
										
										
											2016-08-24 16:42:25 -04:00
										 |  |  |             mcu_time = self.mcu_enable.print_to_mcu_time(move_time) | 
					
						
							|  |  |  |             self.mcu_enable.set_digital(mcu_time, enable) | 
					
						
							| 
									
										
										
										
											2016-11-14 13:40:35 -05:00
										 |  |  |         self.need_motor_enable = not enable | 
					
						
							| 
									
										
										
										
											2016-08-24 16:42:25 -04:00
										 |  |  |     def enable_endstop_checking(self, move_time, step_time): | 
					
						
							|  |  |  |         mcu_time = self.mcu_endstop.print_to_mcu_time(move_time) | 
					
						
							| 
									
										
										
										
											2016-12-08 18:12:20 -05:00
										 |  |  |         self.mcu_endstop.home_start(mcu_time, step_time) | 
					
						
							| 
									
										
										
										
											2016-05-25 11:37:40 -04:00
										 |  |  |         return self.mcu_endstop | 
					
						
							| 
									
										
										
										
											2016-11-17 17:24:03 -05:00
										 |  |  |     def query_endstop(self, print_time): | 
					
						
							|  |  |  |         mcu_time = self.mcu_endstop.print_to_mcu_time(print_time) | 
					
						
							|  |  |  |         self.mcu_endstop.query_endstop(mcu_time) | 
					
						
							| 
									
										
										
										
											2016-09-22 11:09:20 -04:00
										 |  |  |         return self.mcu_endstop | 
					
						
							| 
									
										
										
										
											2016-10-13 10:04:30 -04:00
										 |  |  |     def get_homed_offset(self): | 
					
						
							| 
									
										
										
										
											2016-11-14 13:40:35 -05:00
										 |  |  |         if not self.homing_stepper_phases or self.need_motor_enable: | 
					
						
							| 
									
										
										
										
											2016-10-13 10:04:30 -04:00
										 |  |  |             return 0 | 
					
						
							| 
									
										
										
										
											2016-12-08 18:12:20 -05:00
										 |  |  |         pos = self.mcu_stepper.get_mcu_position() | 
					
						
							| 
									
										
										
										
											2016-07-26 22:06:14 -04:00
										 |  |  |         pos %= self.homing_stepper_phases | 
					
						
							|  |  |  |         if self.homing_endstop_phase is None: | 
					
						
							| 
									
										
										
										
											2017-03-12 22:43:05 -04:00
										 |  |  |             logging.info("Setting %s endstop phase to %d" % (self.name, pos)) | 
					
						
							| 
									
										
										
										
											2016-07-26 22:06:14 -04:00
										 |  |  |             self.homing_endstop_phase = pos | 
					
						
							| 
									
										
										
										
											2016-10-13 10:04:30 -04:00
										 |  |  |             return 0 | 
					
						
							| 
									
										
										
										
											2016-07-26 22:06:14 -04:00
										 |  |  |         delta = (pos - self.homing_endstop_phase) % self.homing_stepper_phases | 
					
						
							|  |  |  |         if delta >= self.homing_stepper_phases - self.homing_endstop_accuracy: | 
					
						
							|  |  |  |             delta -= self.homing_stepper_phases | 
					
						
							|  |  |  |         elif delta > self.homing_endstop_accuracy: | 
					
						
							| 
									
										
										
										
											2016-11-18 14:35:31 -05:00
										 |  |  |             raise homing.EndstopError( | 
					
						
							|  |  |  |                 "Endstop %s incorrect phase (got %d vs %d)" % ( | 
					
						
							|  |  |  |                     self.name, pos, self.homing_endstop_phase)) | 
					
						
							| 
									
										
										
										
											2016-10-13 10:04:30 -04:00
										 |  |  |         return delta |