filament_motion_sensor: Add filament_motion_sensor (#3857)
Add functionality to support a Filament Motion Sensor for detecting extruder jams as well as runouts. Works by an encoder toggling the switch_pin 0/1 as the filament is pulled through the sensor. Signed-off-by: Joshua Wherrett <thejoshw.code@gmail.com>
This commit is contained in:
parent
d77928b17b
commit
bf8f7133b4
|
@ -3461,6 +3461,34 @@ information.
|
||||||
# provided.
|
# provided.
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## [filament_motion_sensor]
|
||||||
|
|
||||||
|
Filament Motion Sensor. Support for filament insert and runout
|
||||||
|
detection using an encoder that toggles the output pin during filament
|
||||||
|
movement through the sensor.
|
||||||
|
|
||||||
|
See the [command reference](G-Codes.md#filament-sensor) for more
|
||||||
|
information.
|
||||||
|
|
||||||
|
```
|
||||||
|
[filament_motion_sensor my_sensor]
|
||||||
|
detection_length: 7.0
|
||||||
|
# The minimum length of filament pulled through the sensor to trigger
|
||||||
|
# a state change on the switch_pin
|
||||||
|
# Default is 7 mm.
|
||||||
|
extruder:
|
||||||
|
# The name of the extruder section this sensor is associated with.
|
||||||
|
# This parameter must be provided.
|
||||||
|
switch_pin:
|
||||||
|
#pause_on_runout:
|
||||||
|
#runout_gcode:
|
||||||
|
#insert_gcode:
|
||||||
|
#event_delay:
|
||||||
|
#pause_delay:
|
||||||
|
# See the "filament_switch_sensor" section for a description of the
|
||||||
|
# above parameters.
|
||||||
|
```
|
||||||
|
|
||||||
## [tsl1401cl_filament_width_sensor]
|
## [tsl1401cl_filament_width_sensor]
|
||||||
|
|
||||||
TSLl401CL Based Filament Width Sensor. See the
|
TSLl401CL Based Filament Width Sensor. See the
|
||||||
|
|
|
@ -574,7 +574,7 @@ enabled:
|
||||||
## Filament Sensor
|
## Filament Sensor
|
||||||
|
|
||||||
The following command is available when the
|
The following command is available when the
|
||||||
[filament_switch_sensor config section](Config_Reference.md#filament_switch_sensor)
|
[filament_switch_sensor or filament_motion_sensor config section](Config_Reference.md#filament_switch_sensor)
|
||||||
is enabled.
|
is enabled.
|
||||||
- `QUERY_FILAMENT_SENSOR SENSOR=<sensor_name>`: Queries the current
|
- `QUERY_FILAMENT_SENSOR SENSOR=<sensor_name>`: Queries the current
|
||||||
status of the filament sensor. The data displayed on the terminal
|
status of the filament sensor. The data displayed on the terminal
|
||||||
|
|
|
@ -0,0 +1,76 @@
|
||||||
|
# Filament Motion Sensor Module
|
||||||
|
#
|
||||||
|
# Copyright (C) 2021 Joshua Wherrett <thejoshw.code@gmail.com>
|
||||||
|
#
|
||||||
|
# This file may be distributed under the terms of the GNU GPLv3 license.
|
||||||
|
import logging
|
||||||
|
from . import filament_switch_sensor
|
||||||
|
|
||||||
|
CHECK_RUNOUT_TIMEOUT = .250
|
||||||
|
|
||||||
|
class EncoderSensor:
|
||||||
|
def __init__(self, config):
|
||||||
|
# Read config
|
||||||
|
self.printer = config.get_printer()
|
||||||
|
switch_pin = config.get('switch_pin')
|
||||||
|
self.extruder_name = config.get('extruder')
|
||||||
|
self.detection_length = config.getfloat(
|
||||||
|
'detection_length', 7., above=0.)
|
||||||
|
# Configure pins
|
||||||
|
buttons = self.printer.load_object(config, 'buttons')
|
||||||
|
buttons.register_buttons([switch_pin], self.encoder_event)
|
||||||
|
# Get printer objects
|
||||||
|
self.reactor = self.printer.get_reactor()
|
||||||
|
self.runout_helper = filament_switch_sensor.RunoutHelper(config)
|
||||||
|
self.extruder = None
|
||||||
|
self.estimated_print_time = None
|
||||||
|
# Initialise internal state
|
||||||
|
self.filament_runout_pos = None
|
||||||
|
# Register commands and event handlers
|
||||||
|
self.printer.register_event_handler('klippy:ready',
|
||||||
|
self._handle_ready)
|
||||||
|
self.printer.register_event_handler('idle_timeout:printing',
|
||||||
|
self._handle_printing)
|
||||||
|
self.printer.register_event_handler('idle_timeout:ready',
|
||||||
|
self._handle_not_printing)
|
||||||
|
self.printer.register_event_handler('idle_timeout:idle',
|
||||||
|
self._handle_not_printing)
|
||||||
|
def _update_filament_runout_pos(self, eventtime=None):
|
||||||
|
if eventtime is None:
|
||||||
|
eventtime = self.reactor.monotonic()
|
||||||
|
self.filament_runout_pos = (
|
||||||
|
self._get_extruder_pos(eventtime) +
|
||||||
|
self.detection_length)
|
||||||
|
def _handle_ready(self):
|
||||||
|
self.extruder = self.printer.lookup_object(self.extruder_name)
|
||||||
|
self.estimated_print_time = (
|
||||||
|
self.printer.lookup_object('mcu').estimated_print_time)
|
||||||
|
self._update_filament_runout_pos()
|
||||||
|
self._extruder_pos_update_timer = self.reactor.register_timer(
|
||||||
|
self._extruder_pos_update_event)
|
||||||
|
def _handle_printing(self, print_time):
|
||||||
|
self.reactor.update_timer(self._extruder_pos_update_timer,
|
||||||
|
self.reactor.NOW)
|
||||||
|
def _handle_not_printing(self, print_time):
|
||||||
|
self.reactor.update_timer(self._extruder_pos_update_timer,
|
||||||
|
self.reactor.NEVER)
|
||||||
|
def _get_extruder_pos(self, eventtime=None):
|
||||||
|
if eventtime is None:
|
||||||
|
eventtime = self.reactor.monotonic()
|
||||||
|
print_time = self.estimated_print_time(eventtime)
|
||||||
|
return self.extruder.find_past_position(print_time)
|
||||||
|
def _extruder_pos_update_event(self, eventtime):
|
||||||
|
extruder_pos = self._get_extruder_pos(eventtime)
|
||||||
|
# Check for filament runout
|
||||||
|
self.runout_helper.note_filament_present(
|
||||||
|
extruder_pos < self.filament_runout_pos)
|
||||||
|
return eventtime + CHECK_RUNOUT_TIMEOUT
|
||||||
|
def encoder_event(self, eventtime, state):
|
||||||
|
if self.extruder is not None:
|
||||||
|
self._update_filament_runout_pos(eventtime)
|
||||||
|
# Check for filament insertion
|
||||||
|
# Filament is always assumed to be present on an encoder event
|
||||||
|
self.runout_helper.note_filament_present(True)
|
||||||
|
|
||||||
|
def load_config_prefix(config):
|
||||||
|
return EncoderSensor(config)
|
Loading…
Reference in New Issue