Skip to content

Commit

Permalink
ads1x1x: refactoring to allow thermistor and adc_temperature as senso…
Browse files Browse the repository at this point in the history
…r instead of a custom ads1x1x sensor.

Signed-off-by: Konstantin Koch <[email protected]>
  • Loading branch information
korsarNek committed Oct 20, 2024
1 parent 15f2a93 commit 3105a5d
Show file tree
Hide file tree
Showing 3 changed files with 77 additions and 158 deletions.
45 changes: 14 additions & 31 deletions docs/Config_Reference.md
Original file line number Diff line number Diff line change
Expand Up @@ -2737,33 +2737,6 @@ sensor_type: temperature_host
# system file on a Raspberry Pi computer.
```

### ADS Analog to Digital Converter as temperature sensor

The sensors ADS1013, ADS1014, ADS1015, ADS1113, ADS1114 and ADS1115 are i2c based Analog to Digital Converters to which are thermistor can be connected in a voltage divider setup.
Needs an [ads1x1x] section to configure the chip. Afterwards chip specific sensor_pins can be used.

```
sensor_type: ads1x1x
# A section for ads1s1x has to specified before the sensor can be used.
# Look at the ads1x1x below in this documentation.
probe_type: Generic 3950
# The probe type indicates the calculation profile to use for the raw data
# from the ADC device. Check the thermistors list for possible values.
sensor_pin: my_ads1x1x:AIN0
# A combination of the name of the ads1x1x chip that has been defined and
# the pin. Possible pin values are AIN0, AIN1, AIN2 and AIN3 for single ended
# lines and DIFF01, DIFF03, DIFF13 and DIFF23 for differential between their
# correspoding lines. For example
# DIFF03 measures the differential between line 0 and 3. Only specific
# combinations for the differentials are allowed.
#pullup_resistor: 4700
# A setting that defines the physical wiring method of the sensor.
# Default is 4700.
#voltage_offset: -0.295
# The differential voltage from the supplied voltage to the maximum pull-up
# voltage that the device under test (thermistor) sees.
```

### DS18B20 temperature sensor

DS18B20 is a 1-wire (w1) digital temperature sensor. Note that this sensor is not intended for use with extruders and heater beds, but rather for monitoring ambient temperature (C). These sensors have range up to 125 C, so are usable for e.g. chamber temperature monitoring. They can also function as simple fan/heater controllers. DS18B20 sensors are only supported on the "host mcu", e.g. the Raspberry Pi. The w1-gpio Linux kernel module must be installed.
Expand Down Expand Up @@ -4858,7 +4831,7 @@ Digital Converters that can be used for temperature sensors. They provide 4
analog input pins either as single line or as differential input.

```
[ads1x1x my_name]
[ads1x1x my_ads1x1x]
chip: ADS1115
#pga: 6.144V
# Default value is 4.096V. The maximum voltage range used for the input. This
Expand All @@ -4876,14 +4849,24 @@ chip: ADS1115
# ADS111X's support 8, 16, 32, 64, 128, 250, 475, 860.
i2c_mcu: host
i2c_bus: i2c.1
#adc_voltage: 5
# The suppy voltage for the device.
# Sensor is reading this amount different from external measuring device.
#address_pin: GND
# Default value is GND. There can be up to four addressed devices depending
# upon wiring of the device. Check the datasheet for details. The i2_address
# can be specified directly instead of using the address_pin.
```

The chip provides pins that can be used on other sensors.

```
sensor_type: ...
# Can be any thermistor or adc_temperature.
sensor_pin: my_ads1x1x:AIN0
# A combination of the name of the ads1x1x chip that and the pin. Possible
# pin values are AIN0, AIN1, AIN2 and AIN3 for single ended lines and
# DIFF01, DIFF03, DIFF13 and DIFF23 for differential between their
# correspoding lines. For example
# DIFF03 measures the differential between line 0 and 3. Only specific
# combinations for the differentials are allowed.
```

### [replicape]
Expand Down
187 changes: 63 additions & 124 deletions klippy/extras/ads1x1x.py
Original file line number Diff line number Diff line change
Expand Up @@ -86,21 +86,23 @@ def isADS111X(chip):
'0.512V': 0x0800, # +/-0.512V range = Gain 8
'0.256V': 0x0A00 # +/-0.256V range = Gain 16
}
ADS111X_RESOLUTION = 32767.0
ADS111X_PGA_SCALAR = {
0x0000: 6.144 / 32767.0, # +/-6.144V range = Gain 2/3
0x0200: 4.096 / 32767.0, # +/-4.096V range = Gain 1
0x0400: 2.048 / 32767.0, # +/-2.048V range = Gain 2
0x0600: 1.024 / 32767.0, # +/-1.024V range = Gain 4
0x0800: 0.512 / 32767.0, # +/-0.512V range = Gain 8
0x0A00: 0.256 / 32767.0 # +/-0.256V range = Gain 16
0x0000: 6.144 / ADS111X_RESOLUTION, # +/-6.144V range = Gain 2/3
0x0200: 4.096 / ADS111X_RESOLUTION, # +/-4.096V range = Gain 1
0x0400: 2.048 / ADS111X_RESOLUTION, # +/-2.048V range = Gain 2
0x0600: 1.024 / ADS111X_RESOLUTION, # +/-1.024V range = Gain 4
0x0800: 0.512 / ADS111X_RESOLUTION, # +/-0.512V range = Gain 8
0x0A00: 0.256 / ADS111X_RESOLUTION # +/-0.256V range = Gain 16
}
ADS101X_RESOLUTION = 2047.0
ADS101X_PGA_SCALAR = {
0x0000: 6.144 / 2047.0, # +/-6.144V range = Gain 2/3
0x0200: 4.096 / 2047.0, # +/-4.096V range = Gain 1
0x0400: 2.048 / 2047.0, # +/-2.048V range = Gain 2
0x0600: 1.024 / 2047.0, # +/-1.024V range = Gain 4
0x0800: 0.512 / 2047.0, # +/-0.512V range = Gain 8
0x0A00: 0.256 / 2047.0 # +/-0.256V range = Gain 16
0x0000: 6.144 / ADS101X_RESOLUTION, # +/-6.144V range = Gain 2/3
0x0200: 4.096 / ADS101X_RESOLUTION, # +/-4.096V range = Gain 1
0x0400: 2.048 / ADS101X_RESOLUTION, # +/-2.048V range = Gain 2
0x0600: 1.024 / ADS101X_RESOLUTION, # +/-1.024V range = Gain 4
0x0800: 0.512 / ADS101X_RESOLUTION, # +/-0.512V range = Gain 8
0x0A00: 0.256 / ADS101X_RESOLUTION # +/-0.256V range = Gain 16
}
ADS1X1X_MODE = {
'continuous': 0x0000, # Continuous conversion mode
Expand Down Expand Up @@ -196,8 +198,6 @@ def __init__(self, config):
self.comp_queue = ADS1X1X_COMPARATOR_QUEUE['QUEUE_NONE']
self._i2c = bus.MCU_I2C_from_config(config, address)

self.adc_voltage = config.getfloat('adc_voltage', 5., above=0.)

self.mcu = self._i2c.get_mcu()

self._printer.add_object("ads1x1x " + self.name, self)
Expand Down Expand Up @@ -271,114 +271,10 @@ def reset_all_devices(self):
except Exception:
logging.exception("ADS1X1X: error while resetting device")

class ADS1X1X_sensor:
def __init__(self, config):
self._printer = config.get_printer()
self.name = config.get_name().split()[-1]
sensor_pin = config.get('sensor_pin')

self.pin = sensor_pin.split(':')[-1]
self.chip = self._printer.lookup_object("ads1x1x " +
sensor_pin.split(':')[0])

self.voltage_offset = config.getfloat('voltage_offset', 0.0)
self.pullup_resistor = config.getfloat('pullup_resistor', 4700.,
above=0.)
self.report_time = config.getfloat('report_time'
, 1.0 / self.chip.samples_per_second_numeric * 4 # 4 possible inputs
, minval= 1.0 / self.chip.samples_per_second_numeric)

# Find probe for voltage to temp conversion function/table
for probe_type, params in adc_temperature.DefaultVoltageSensors:
if probe_type == config.get('probe_type'):
self.converter = adc_temperature.LinearVoltage(config, params)
break
# Find probe for resistance to temp conversion function/table
if probe_type != config.get('probe_type'):
for probe_type, params in \
adc_temperature.DefaultResistanceSensors:
if (probe_type == config.get('probe_type')):
self.converter = adc_temperature.LinearResistance(config,
params)
break
# Attempt to match against existing thermistors, generate conversion
# table
if probe_type != config.get('probe_type'):
pheaters = config.get_printer().load_object(config, "heaters")
probe_config = \
pheaters.sensor_factories[config.get('probe_type')](config) \
.adc_convert
all_temps = [float(i) for i in range(1, 351)]
adcs = [(t, probe_config.calc_adc(t)) for t in all_temps]
rs = [(t,self.pullup_resistor * adc / (1.0 - adc)) \
for t, adc in adcs]
if probe_config is not None:
self.converter = adc_temperature.LinearResistance(config, rs)
probe_type = config.get('probe_type')

if probe_type != config.get('probe_type'):
logging.error("Probe '%s' not found in adc_temperature for %s" \
% (config.get('probe_type'), self.name))

self.temp = self.min_temp = self.max_temp = 0.0

self._printer.register_event_handler("klippy:connect", \
self._handle_connect)

def _handle_connect(self):
self._reactor = self._printer.get_reactor()
self._sample_timer = \
self._reactor.register_timer(self._timer)
self._reactor.update_timer(self._sample_timer, self._reactor.NOW)

def setup_minmax(self, min_temp, max_temp):
self.min_temp = min_temp
self.max_temp = max_temp

def setup_callback(self, cb):
self._callback = cb

def _timer(self, eventtime):
sample = self.chip.sample(self)
if sample is not None:
self._process_sample(sample)

return self._reactor.monotonic() + self.report_time

def _process_sample(self, sample):
# The sample is encoded in the top 12 or full 16 bits
# Value's meaning is defined by ADS1X1X_REG_CONFIG['PGA_MASK']
if isADS101X(self.chip.chip):
sample >>= 4
volts = sample * ADS101X_PGA_SCALAR[self.chip.pga]
else:
volts = sample * ADS111X_PGA_SCALAR[self.chip.pga]
self.temp = self.converter.calc_temp( (volts - self.voltage_offset) \
/ self.chip.adc_voltage )

# Check if result is within limits
if self.temp < self.min_temp or self.temp > self.max_temp:
self._printer.invoke_shutdown(
"ADS1X1X temperature %0.1f outside range of %0.1f:%.01f"
% (self.temp, self.min_temp, self.max_temp))

# Publish result
measured_time = self._reactor.monotonic()
self._callback(self.chip.mcu.estimated_print_time(measured_time),
self.temp)

def get_report_time_delta(self):
return self.report_time

def get_status(self, eventtime):
return {
'temperature': round(self.temp, 2),
}


class ADS1X1X_pin:
def __init__(self, chip, pin_params):
self.mcu = chip.mcu
self.chip = chip
self.pin = pin_params['pin']

if (self.pin not in ADS1X1X_MUX):
Expand All @@ -403,17 +299,60 @@ def __init__(self, chip, pin_params):
self.config |= (chip.comp_queue \
& ADS1X1X_REG_CONFIG['COMPARATOR_QUEUE_MASK'])

self.report_time = 1.0 / chip.samples_per_second_numeric * 4
self.invalid_count = 0

self.chip._printer.register_event_handler("klippy:connect", \
self._handle_connect)

def _handle_connect(self):
self._reactor = self.chip._printer.get_reactor()
self._sample_timer = \
self._reactor.register_timer(self._timer)
self._reactor.update_timer(self._sample_timer, self._reactor.NOW)

def _timer(self, eventtime):
sample = self.chip.sample(self)
if sample is not None:
self._process_sample(sample)

return self._reactor.monotonic() + self.report_time

def _process_sample(self, sample):
# The sample is encoded in the top 12 or full 16 bits
# Value's meaning is defined by ADS1X1X_REG_CONFIG['PGA_MASK']
if isADS101X(self.chip.chip):
sample >>= 4
target_value = sample / ADS101X_RESOLUTION
else:
target_value = sample / ADS111X_RESOLUTION

if self.maxval > self.minval:
if target_value > self.maxval or target_value < self.minval:
self.invalid_count = self.invalid_count + 1
if self.invalid_count > self.range_check_count:
self.chip._printer.invoke_shutdown(
"ADS1X1X temperature outside range")
else:
self.invalid_count = 0

# Publish result
measured_time = self._reactor.monotonic()
self.callback(self.chip.mcu.estimated_print_time(measured_time),
target_value)

def get_mcu(self):
return self.mcu

def setup_adc_callback(self, report_time, callback):
self.report_time = report_time
self.callback = callback

def setup_adc_sample(self, sample_time, sample_count,
minval=0., maxval=1., range_check_count=0):
self.minval = minval
self.maxval = maxval
self.range_check_count = range_check_count

def load_config_prefix(config):
return ADS1X1X_chip(config)

def load_config(config):
# Register sensor
pheaters = config.get_printer().load_object(config, "heaters")
pheaters.add_sensor_factory('ads1x1x', ADS1X1X_sensor)
3 changes: 0 additions & 3 deletions klippy/extras/temperature_sensors.cfg
Original file line number Diff line number Diff line change
Expand Up @@ -26,9 +26,6 @@
# Load "LM75" sensor
[lm75]

# Load "ADS1013", "ADS1014", "ADS1015", "ADS1113", "ADS1114" and "ADS1115" sensors
[ads1x1x]

# Load "MAX6675", "MAX31855", "MAX31856", and "MAX31865" sensors
[spi_temperature]

Expand Down

0 comments on commit 3105a5d

Please sign in to comment.