| 
									
										
										
										
											2019-02-08 20:11:05 -05:00
										 |  |  | # Support for a manual controlled stepper | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2021-03-29 13:34:25 -04:00
										 |  |  | # Copyright (C) 2019-2021  Kevin O'Connor <kevin@koconnor.net> | 
					
						
							| 
									
										
										
										
											2019-02-08 20:11:05 -05:00
										 |  |  | # | 
					
						
							|  |  |  | # This file may be distributed under the terms of the GNU GPLv3 license. | 
					
						
							| 
									
										
										
										
											2020-09-04 11:49:43 -04:00
										 |  |  | import stepper, chelper | 
					
						
							| 
									
										
										
										
											2020-06-12 09:55:57 -04:00
										 |  |  | from . import force_move | 
					
						
							| 
									
										
										
										
											2019-02-08 20:11:05 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | class ManualStepper: | 
					
						
							|  |  |  |     def __init__(self, config): | 
					
						
							|  |  |  |         self.printer = config.get_printer() | 
					
						
							|  |  |  |         if config.get('endstop_pin', None) is not None: | 
					
						
							|  |  |  |             self.can_home = True | 
					
						
							| 
									
										
										
										
											2025-05-07 00:06:36 +02:00
										 |  |  |             self.rail = stepper.LookupRail( | 
					
						
							| 
									
										
										
										
											2019-02-08 20:11:05 -05:00
										 |  |  |                 config, need_position_minmax=False, default_position_endstop=0.) | 
					
						
							| 
									
										
										
										
											2019-11-12 12:49:21 -05:00
										 |  |  |             self.steppers = self.rail.get_steppers() | 
					
						
							| 
									
										
										
										
											2019-02-08 20:11:05 -05:00
										 |  |  |         else: | 
					
						
							|  |  |  |             self.can_home = False | 
					
						
							| 
									
										
										
										
											2019-11-12 12:49:21 -05:00
										 |  |  |             self.rail = stepper.PrinterStepper(config) | 
					
						
							|  |  |  |             self.steppers = [self.rail] | 
					
						
							| 
									
										
										
										
											2019-03-03 13:23:45 -05:00
										 |  |  |         self.velocity = config.getfloat('velocity', 5., above=0.) | 
					
						
							| 
									
										
										
										
											2021-03-29 13:34:25 -04:00
										 |  |  |         self.accel = self.homing_accel = config.getfloat('accel', 0., minval=0.) | 
					
						
							| 
									
										
										
										
											2019-02-08 20:11:05 -05:00
										 |  |  |         self.next_cmd_time = 0. | 
					
						
							|  |  |  |         # Setup iterative solver | 
					
						
							|  |  |  |         ffi_main, ffi_lib = chelper.get_ffi() | 
					
						
							| 
									
										
										
										
											2019-10-29 11:34:52 -04:00
										 |  |  |         self.trapq = ffi_main.gc(ffi_lib.trapq_alloc(), ffi_lib.trapq_free) | 
					
						
							| 
									
										
										
										
											2019-10-29 12:44:39 -04:00
										 |  |  |         self.trapq_append = ffi_lib.trapq_append | 
					
						
							| 
									
										
										
										
											2021-07-19 11:12:58 -04:00
										 |  |  |         self.trapq_finalize_moves = ffi_lib.trapq_finalize_moves | 
					
						
							| 
									
										
										
										
											2021-10-01 19:09:37 -04:00
										 |  |  |         self.rail.setup_itersolve('cartesian_stepper_alloc', b'x') | 
					
						
							| 
									
										
										
										
											2019-11-12 12:49:21 -05:00
										 |  |  |         self.rail.set_trapq(self.trapq) | 
					
						
							| 
									
										
										
										
											2019-02-08 20:11:05 -05:00
										 |  |  |         # Register commands | 
					
						
							|  |  |  |         stepper_name = config.get_name().split()[1] | 
					
						
							| 
									
										
										
										
											2020-04-25 00:29:15 -04:00
										 |  |  |         gcode = self.printer.lookup_object('gcode') | 
					
						
							|  |  |  |         gcode.register_mux_command('MANUAL_STEPPER', "STEPPER", | 
					
						
							|  |  |  |                                    stepper_name, self.cmd_MANUAL_STEPPER, | 
					
						
							|  |  |  |                                    desc=self.cmd_MANUAL_STEPPER_help) | 
					
						
							| 
									
										
										
										
											2019-02-08 20:11:05 -05:00
										 |  |  |     def sync_print_time(self): | 
					
						
							|  |  |  |         toolhead = self.printer.lookup_object('toolhead') | 
					
						
							|  |  |  |         print_time = toolhead.get_last_move_time() | 
					
						
							|  |  |  |         if self.next_cmd_time > print_time: | 
					
						
							|  |  |  |             toolhead.dwell(self.next_cmd_time - print_time) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.next_cmd_time = print_time | 
					
						
							|  |  |  |     def do_enable(self, enable): | 
					
						
							|  |  |  |         self.sync_print_time() | 
					
						
							| 
									
										
										
										
											2019-11-12 12:49:21 -05:00
										 |  |  |         stepper_enable = self.printer.lookup_object('stepper_enable') | 
					
						
							|  |  |  |         if enable: | 
					
						
							|  |  |  |             for s in self.steppers: | 
					
						
							| 
									
										
										
										
											2019-11-12 13:55:50 -05:00
										 |  |  |                 se = stepper_enable.lookup_enable(s.get_name()) | 
					
						
							|  |  |  |                 se.motor_enable(self.next_cmd_time) | 
					
						
							| 
									
										
										
										
											2019-11-12 12:49:21 -05:00
										 |  |  |         else: | 
					
						
							|  |  |  |             for s in self.steppers: | 
					
						
							| 
									
										
										
										
											2019-11-12 13:55:50 -05:00
										 |  |  |                 se = stepper_enable.lookup_enable(s.get_name()) | 
					
						
							|  |  |  |                 se.motor_disable(self.next_cmd_time) | 
					
						
							| 
									
										
										
										
											2019-02-08 20:11:05 -05:00
										 |  |  |         self.sync_print_time() | 
					
						
							|  |  |  |     def do_set_position(self, setpos): | 
					
						
							| 
									
										
										
										
											2019-11-12 12:49:21 -05:00
										 |  |  |         self.rail.set_position([setpos, 0., 0.]) | 
					
						
							| 
									
										
										
										
											2025-04-12 16:54:49 -04:00
										 |  |  |     def _submit_move(self, movetime, movepos, speed, accel): | 
					
						
							| 
									
										
										
										
											2019-11-12 12:49:21 -05:00
										 |  |  |         cp = self.rail.get_commanded_position() | 
					
						
							| 
									
										
										
										
											2019-02-08 20:11:05 -05:00
										 |  |  |         dist = movepos - cp | 
					
						
							| 
									
										
										
										
											2019-11-06 08:41:50 -05:00
										 |  |  |         axis_r, accel_t, cruise_t, cruise_v = force_move.calc_move_time( | 
					
						
							| 
									
										
										
										
											2019-03-03 13:23:45 -05:00
										 |  |  |             dist, speed, accel) | 
					
						
							| 
									
										
										
										
											2025-04-12 16:54:49 -04:00
										 |  |  |         self.trapq_append(self.trapq, movetime, | 
					
						
							| 
									
										
										
										
											2019-10-29 12:44:39 -04:00
										 |  |  |                           accel_t, cruise_t, accel_t, | 
					
						
							| 
									
										
										
										
											2019-11-06 08:41:50 -05:00
										 |  |  |                           cp, 0., 0., axis_r, 0., 0., | 
					
						
							| 
									
										
										
										
											2019-10-29 12:44:39 -04:00
										 |  |  |                           0., cruise_v, accel) | 
					
						
							| 
									
										
										
										
											2025-04-12 16:54:49 -04:00
										 |  |  |         return movetime + accel_t + cruise_t + accel_t | 
					
						
							|  |  |  |     def do_move(self, movepos, speed, accel, sync=True): | 
					
						
							|  |  |  |         self.sync_print_time() | 
					
						
							|  |  |  |         self.next_cmd_time = self._submit_move(self.next_cmd_time, movepos, | 
					
						
							|  |  |  |                                                speed, accel) | 
					
						
							| 
									
										
										
										
											2019-11-12 12:49:21 -05:00
										 |  |  |         self.rail.generate_steps(self.next_cmd_time) | 
					
						
							| 
									
										
										
										
											2023-12-30 11:34:21 -05:00
										 |  |  |         self.trapq_finalize_moves(self.trapq, self.next_cmd_time + 99999.9, | 
					
						
							|  |  |  |                                   self.next_cmd_time + 99999.9) | 
					
						
							| 
									
										
										
										
											2019-11-24 10:59:13 -05:00
										 |  |  |         toolhead = self.printer.lookup_object('toolhead') | 
					
						
							| 
									
										
										
										
											2024-01-18 12:20:44 -05:00
										 |  |  |         toolhead.note_mcu_movequeue_activity(self.next_cmd_time) | 
					
						
							| 
									
										
										
										
											2020-03-21 10:54:01 +01:00
										 |  |  |         if sync: | 
					
						
							|  |  |  |             self.sync_print_time() | 
					
						
							| 
									
										
										
										
											2020-02-19 13:34:06 -05:00
										 |  |  |     def do_homing_move(self, movepos, speed, accel, triggered, check_trigger): | 
					
						
							| 
									
										
										
										
											2019-02-08 20:11:05 -05:00
										 |  |  |         if not self.can_home: | 
					
						
							| 
									
										
										
										
											2020-04-25 00:29:15 -04:00
										 |  |  |             raise self.printer.command_error( | 
					
						
							|  |  |  |                 "No endstop for this manual stepper") | 
					
						
							| 
									
										
										
										
											2021-03-29 13:34:25 -04:00
										 |  |  |         self.homing_accel = accel | 
					
						
							|  |  |  |         pos = [movepos, 0., 0., 0.] | 
					
						
							| 
									
										
										
										
											2020-02-12 13:03:42 -05:00
										 |  |  |         endstops = self.rail.get_endstops() | 
					
						
							| 
									
										
										
										
											2021-03-29 13:34:25 -04:00
										 |  |  |         phoming = self.printer.lookup_object('homing') | 
					
						
							|  |  |  |         phoming.manual_home(self, endstops, pos, speed, | 
					
						
							|  |  |  |                             triggered, check_trigger) | 
					
						
							| 
									
										
										
										
											2019-02-08 20:11:05 -05:00
										 |  |  |     cmd_MANUAL_STEPPER_help = "Command a manually configured stepper" | 
					
						
							| 
									
										
										
										
											2020-04-25 00:29:15 -04:00
										 |  |  |     def cmd_MANUAL_STEPPER(self, gcmd): | 
					
						
							|  |  |  |         enable = gcmd.get_int('ENABLE', None) | 
					
						
							|  |  |  |         if enable is not None: | 
					
						
							|  |  |  |             self.do_enable(enable) | 
					
						
							|  |  |  |         setpos = gcmd.get_float('SET_POSITION', None) | 
					
						
							|  |  |  |         if setpos is not None: | 
					
						
							| 
									
										
										
										
											2019-02-08 20:11:05 -05:00
										 |  |  |             self.do_set_position(setpos) | 
					
						
							| 
									
										
										
										
											2020-04-25 00:29:15 -04:00
										 |  |  |         speed = gcmd.get_float('SPEED', self.velocity, above=0.) | 
					
						
							|  |  |  |         accel = gcmd.get_float('ACCEL', self.accel, minval=0.) | 
					
						
							|  |  |  |         homing_move = gcmd.get_int('STOP_ON_ENDSTOP', 0) | 
					
						
							| 
									
										
										
										
											2019-02-08 20:11:05 -05:00
										 |  |  |         if homing_move: | 
					
						
							| 
									
										
										
										
											2020-04-25 00:29:15 -04:00
										 |  |  |             movepos = gcmd.get_float('MOVE') | 
					
						
							| 
									
										
										
										
											2020-02-19 13:34:06 -05:00
										 |  |  |             self.do_homing_move(movepos, speed, accel, | 
					
						
							|  |  |  |                                 homing_move > 0, abs(homing_move) == 1) | 
					
						
							| 
									
										
										
										
											2020-04-25 00:29:15 -04:00
										 |  |  |         elif gcmd.get_float('MOVE', None) is not None: | 
					
						
							|  |  |  |             movepos = gcmd.get_float('MOVE') | 
					
						
							|  |  |  |             sync = gcmd.get_int('SYNC', 1) | 
					
						
							| 
									
										
										
										
											2020-03-21 10:54:01 +01:00
										 |  |  |             self.do_move(movepos, speed, accel, sync) | 
					
						
							| 
									
										
										
										
											2020-04-25 00:29:15 -04:00
										 |  |  |         elif gcmd.get_int('SYNC', 0): | 
					
						
							| 
									
										
										
										
											2020-03-21 10:54:01 +01:00
										 |  |  |             self.sync_print_time() | 
					
						
							| 
									
										
										
										
											2021-03-29 13:34:25 -04:00
										 |  |  |     # Toolhead wrappers to support homing | 
					
						
							|  |  |  |     def flush_step_generation(self): | 
					
						
							|  |  |  |         self.sync_print_time() | 
					
						
							|  |  |  |     def get_position(self): | 
					
						
							|  |  |  |         return [self.rail.get_commanded_position(), 0., 0., 0.] | 
					
						
							| 
									
										
										
										
											2025-01-10 11:27:30 -05:00
										 |  |  |     def set_position(self, newpos, homing_axes=""): | 
					
						
							| 
									
										
										
										
											2021-03-29 13:34:25 -04:00
										 |  |  |         self.do_set_position(newpos[0]) | 
					
						
							|  |  |  |     def get_last_move_time(self): | 
					
						
							|  |  |  |         self.sync_print_time() | 
					
						
							|  |  |  |         return self.next_cmd_time | 
					
						
							|  |  |  |     def dwell(self, delay): | 
					
						
							|  |  |  |         self.next_cmd_time += max(0., delay) | 
					
						
							|  |  |  |     def drip_move(self, newpos, speed, drip_completion): | 
					
						
							| 
									
										
										
										
											2025-04-12 16:54:49 -04:00
										 |  |  |         # Submit move to trapq | 
					
						
							|  |  |  |         self.sync_print_time() | 
					
						
							|  |  |  |         maxtime = self._submit_move(self.next_cmd_time, newpos[0], | 
					
						
							|  |  |  |                                     speed, self.homing_accel) | 
					
						
							|  |  |  |         # Drip updates to motors | 
					
						
							|  |  |  |         toolhead = self.printer.lookup_object('toolhead') | 
					
						
							|  |  |  |         toolhead.drip_update_time(maxtime, drip_completion, self.steppers) | 
					
						
							|  |  |  |         # Clear trapq of any remaining parts of movement | 
					
						
							|  |  |  |         reactor = self.printer.get_reactor() | 
					
						
							|  |  |  |         self.trapq_finalize_moves(self.trapq, reactor.NEVER, 0) | 
					
						
							|  |  |  |         self.rail.set_position([newpos[0], 0., 0.]) | 
					
						
							|  |  |  |         self.sync_print_time() | 
					
						
							| 
									
										
										
										
											2021-03-29 13:34:25 -04:00
										 |  |  |     def get_kinematics(self): | 
					
						
							|  |  |  |         return self | 
					
						
							|  |  |  |     def get_steppers(self): | 
					
						
							|  |  |  |         return self.steppers | 
					
						
							| 
									
										
										
										
											2021-08-26 12:23:12 -04:00
										 |  |  |     def calc_position(self, stepper_positions): | 
					
						
							|  |  |  |         return [stepper_positions[self.rail.get_name()], 0., 0.] | 
					
						
							| 
									
										
										
										
											2019-02-08 20:11:05 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | def load_config_prefix(config): | 
					
						
							|  |  |  |     return ManualStepper(config) |