2018-04-04 21:53:47 +03:00
|
|
|
# Temperature measurements with thermistors
|
|
|
|
#
|
2019-01-20 22:52:40 +03:00
|
|
|
# Copyright (C) 2016-2019 Kevin O'Connor <kevin@koconnor.net>
|
2018-04-04 21:53:47 +03:00
|
|
|
#
|
|
|
|
# This file may be distributed under the terms of the GNU GPLv3 license.
|
2018-04-09 19:35:18 +03:00
|
|
|
import math, logging
|
2020-06-12 16:55:57 +03:00
|
|
|
from . import adc_temperature
|
2018-04-04 21:53:47 +03:00
|
|
|
|
2020-02-27 19:31:51 +03:00
|
|
|
KELVIN_TO_CELSIUS = -273.15
|
2018-04-04 21:53:47 +03:00
|
|
|
|
|
|
|
# Analog voltage to temperature converter for thermistors
|
|
|
|
class Thermistor:
|
2019-05-29 18:11:07 +03:00
|
|
|
def __init__(self, pullup, inline_resistor):
|
2019-01-20 22:52:40 +03:00
|
|
|
self.pullup = pullup
|
2019-05-29 18:11:07 +03:00
|
|
|
self.inline_resistor = inline_resistor
|
2018-04-04 21:53:47 +03:00
|
|
|
self.c1 = self.c2 = self.c3 = 0.
|
2019-01-20 22:52:40 +03:00
|
|
|
def setup_coefficients(self, t1, r1, t2, r2, t3, r3, name=""):
|
2018-04-04 21:53:47 +03:00
|
|
|
# Calculate Steinhart-Hart coefficents from temp measurements.
|
|
|
|
# Arrange samples as 3 linear equations and solve for c1, c2, and c3.
|
2020-02-27 19:31:51 +03:00
|
|
|
inv_t1 = 1. / (t1 - KELVIN_TO_CELSIUS)
|
|
|
|
inv_t2 = 1. / (t2 - KELVIN_TO_CELSIUS)
|
|
|
|
inv_t3 = 1. / (t3 - KELVIN_TO_CELSIUS)
|
2019-01-20 22:52:40 +03:00
|
|
|
ln_r1 = math.log(r1)
|
|
|
|
ln_r2 = math.log(r2)
|
|
|
|
ln_r3 = math.log(r3)
|
2018-04-04 21:53:47 +03:00
|
|
|
ln3_r1, ln3_r2, ln3_r3 = ln_r1**3, ln_r2**3, ln_r3**3
|
|
|
|
|
|
|
|
inv_t12, inv_t13 = inv_t1 - inv_t2, inv_t1 - inv_t3
|
|
|
|
ln_r12, ln_r13 = ln_r1 - ln_r2, ln_r1 - ln_r3
|
|
|
|
ln3_r12, ln3_r13 = ln3_r1 - ln3_r2, ln3_r1 - ln3_r3
|
|
|
|
|
|
|
|
self.c3 = ((inv_t12 - inv_t13 * ln_r12 / ln_r13)
|
|
|
|
/ (ln3_r12 - ln3_r13 * ln_r12 / ln_r13))
|
2018-04-09 19:35:18 +03:00
|
|
|
if self.c3 <= 0.:
|
|
|
|
beta = ln_r13 / inv_t13
|
2019-01-20 22:52:40 +03:00
|
|
|
logging.warn("Using thermistor beta %.3f in heater %s", beta, name)
|
|
|
|
self.setup_coefficients_beta(t1, r1, beta)
|
2018-04-09 19:35:18 +03:00
|
|
|
return
|
2018-04-04 21:53:47 +03:00
|
|
|
self.c2 = (inv_t12 - self.c3 * ln3_r12) / ln_r12
|
|
|
|
self.c1 = inv_t1 - self.c2 * ln_r1 - self.c3 * ln3_r1
|
2019-01-20 22:52:40 +03:00
|
|
|
def setup_coefficients_beta(self, t1, r1, beta):
|
2018-04-04 21:53:47 +03:00
|
|
|
# Calculate equivalent Steinhart-Hart coefficents from beta
|
2020-02-27 19:31:51 +03:00
|
|
|
inv_t1 = 1. / (t1 - KELVIN_TO_CELSIUS)
|
2019-01-20 22:52:40 +03:00
|
|
|
ln_r1 = math.log(r1)
|
2018-04-04 21:53:47 +03:00
|
|
|
self.c3 = 0.
|
2018-04-09 19:35:18 +03:00
|
|
|
self.c2 = 1. / beta
|
2018-04-04 21:53:47 +03:00
|
|
|
self.c1 = inv_t1 - self.c2 * ln_r1
|
2019-01-20 22:52:40 +03:00
|
|
|
def calc_temp(self, adc):
|
2018-04-04 21:53:47 +03:00
|
|
|
# Calculate temperature from adc
|
2019-01-20 22:52:40 +03:00
|
|
|
adc = max(.00001, min(.99999, adc))
|
2018-04-04 21:53:47 +03:00
|
|
|
r = self.pullup * adc / (1.0 - adc)
|
2019-05-29 18:11:07 +03:00
|
|
|
ln_r = math.log(r - self.inline_resistor)
|
2018-04-04 21:53:47 +03:00
|
|
|
inv_t = self.c1 + self.c2 * ln_r + self.c3 * ln_r**3
|
2020-02-27 19:31:51 +03:00
|
|
|
return 1.0/inv_t + KELVIN_TO_CELSIUS
|
2018-04-04 21:53:47 +03:00
|
|
|
def calc_adc(self, temp):
|
2019-01-20 22:52:40 +03:00
|
|
|
# Calculate adc reading from a temperature
|
2020-02-27 19:31:51 +03:00
|
|
|
if temp <= KELVIN_TO_CELSIUS:
|
2019-01-04 20:30:20 +03:00
|
|
|
return 1.
|
2020-02-27 19:31:51 +03:00
|
|
|
inv_t = 1. / (temp - KELVIN_TO_CELSIUS)
|
2018-04-04 21:53:47 +03:00
|
|
|
if self.c3:
|
|
|
|
# Solve for ln_r using Cardano's formula
|
|
|
|
y = (self.c1 - inv_t) / (2. * self.c3)
|
|
|
|
x = math.sqrt((self.c2 / (3. * self.c3))**3 + y**2)
|
|
|
|
ln_r = math.pow(x - y, 1./3.) - math.pow(x + y, 1./3.)
|
|
|
|
else:
|
|
|
|
ln_r = (inv_t - self.c1) / self.c2
|
2019-05-29 18:11:07 +03:00
|
|
|
r = math.exp(ln_r) + self.inline_resistor
|
2018-04-04 21:53:47 +03:00
|
|
|
return r / (self.pullup + r)
|
|
|
|
|
2019-01-22 02:11:09 +03:00
|
|
|
# Create an ADC converter with a thermistor
|
|
|
|
def PrinterThermistor(config, params):
|
|
|
|
pullup = config.getfloat('pullup_resistor', 4700., above=0.)
|
2019-05-29 18:11:07 +03:00
|
|
|
inline_resistor = config.getfloat('inline_resistor', 0., minval=0.)
|
|
|
|
thermistor = Thermistor(pullup, inline_resistor)
|
2019-01-22 02:11:09 +03:00
|
|
|
if 'beta' in params:
|
|
|
|
thermistor.setup_coefficients_beta(
|
|
|
|
params['t1'], params['r1'], params['beta'])
|
|
|
|
else:
|
|
|
|
thermistor.setup_coefficients(
|
|
|
|
params['t1'], params['r1'], params['t2'], params['r2'],
|
|
|
|
params['t3'], params['r3'], name=config.get_name())
|
2019-01-22 02:29:08 +03:00
|
|
|
return adc_temperature.PrinterADCtoTemperature(config, thermistor)
|
2019-01-22 02:11:09 +03:00
|
|
|
|
2018-04-09 19:14:20 +03:00
|
|
|
# Custom defined thermistors from the config file
|
|
|
|
class CustomThermistor:
|
|
|
|
def __init__(self, config):
|
|
|
|
self.name = " ".join(config.get_name().split()[1:])
|
2020-02-27 19:31:51 +03:00
|
|
|
t1 = config.getfloat("temperature1", minval=KELVIN_TO_CELSIUS)
|
2018-04-09 19:14:20 +03:00
|
|
|
r1 = config.getfloat("resistance1", minval=0.)
|
|
|
|
beta = config.getfloat("beta", None, above=0.)
|
|
|
|
if beta is not None:
|
|
|
|
self.params = {'t1': t1, 'r1': r1, 'beta': beta}
|
|
|
|
return
|
2020-02-27 19:31:51 +03:00
|
|
|
t2 = config.getfloat("temperature2", minval=KELVIN_TO_CELSIUS)
|
2018-04-09 19:14:20 +03:00
|
|
|
r2 = config.getfloat("resistance2", minval=0.)
|
2020-02-27 19:31:51 +03:00
|
|
|
t3 = config.getfloat("temperature3", minval=KELVIN_TO_CELSIUS)
|
2018-04-09 19:14:20 +03:00
|
|
|
r3 = config.getfloat("resistance3", minval=0.)
|
|
|
|
(t1, r1), (t2, r2), (t3, r3) = sorted([(t1, r1), (t2, r2), (t3, r3)])
|
|
|
|
self.params = {'t1': t1, 'r1': r1, 't2': t2, 'r2': r2,
|
|
|
|
't3': t3, 'r3': r3}
|
|
|
|
def create(self, config):
|
2019-01-20 22:52:40 +03:00
|
|
|
return PrinterThermistor(config, self.params)
|
2018-04-09 19:14:20 +03:00
|
|
|
|
|
|
|
def load_config_prefix(config):
|
|
|
|
thermistor = CustomThermistor(config)
|
2020-05-05 21:10:30 +03:00
|
|
|
pheaters = config.get_printer().load_object(config, "heaters")
|
2020-04-25 20:27:41 +03:00
|
|
|
pheaters.add_sensor_factory(thermistor.name, thermistor.create)
|