Driver for ThermoFisher A 10
Change-Id: Ic19ae444c3b4242f3bb1fe83852d4521326d0b9d
This commit is contained in:
parent
7a3cfe9836
commit
5fb1e649ab
@ -1,24 +1,24 @@
|
|||||||
Node('QnwTC1test.psi.ch',
|
Node('QnwTC1test.psi.ch',
|
||||||
'QnwTC1 test',
|
'QnwTC1 test',
|
||||||
'tcp://5000',
|
'tcp://5000',
|
||||||
)
|
)
|
||||||
|
|
||||||
Mod('io',
|
Mod('io',
|
||||||
'frappy_psi.qnw.QnwIO',
|
'frappy_psi.qnw.QnwIO',
|
||||||
'connection for Quantum northwest',
|
'connection for Quantum northwest',
|
||||||
uri= 'tcp://ldmcc01-ts:3004',
|
uri='tcp://ldmcc01-ts:3004',
|
||||||
)
|
)
|
||||||
|
|
||||||
Mod('T',
|
Mod('T',
|
||||||
'frappy_psi.qnw.TemperatureLoopTC1',
|
'frappy_psi.qnw.TemperatureLoopTC1',
|
||||||
'holder temperature',
|
'holder temperature',
|
||||||
channel='CT',
|
channel='CT',
|
||||||
io='io',
|
io='io',
|
||||||
)
|
)
|
||||||
|
|
||||||
Mod('Th',
|
Mod('Th',
|
||||||
'frappy_psi.qnw.SensorTC1',
|
'frappy_psi.qnw.SensorTC1',
|
||||||
'heat exch. temperature',
|
'heat exch. temperature',
|
||||||
channel='HT',
|
channel='HT',
|
||||||
io='io',
|
io='io',
|
||||||
)
|
)
|
||||||
|
@ -30,7 +30,7 @@ Mod('Heater',
|
|||||||
'heater output',
|
'heater output',
|
||||||
channel='B',
|
channel='B',
|
||||||
io='io',
|
io='io',
|
||||||
resistance=50,
|
resistance=25,
|
||||||
max_power=50,
|
max_power=50,
|
||||||
current=1
|
current=1
|
||||||
)
|
)
|
||||||
|
22
frappy_psi/TFA10_cfg.py
Normal file
22
frappy_psi/TFA10_cfg.py
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
Node('TFA10.psi.ch',
|
||||||
|
'TFA10',
|
||||||
|
'tcp://5000',
|
||||||
|
)
|
||||||
|
|
||||||
|
Mod('io',
|
||||||
|
'frappy_psi.thermofisher.ThermFishIO',
|
||||||
|
'connection for ThermoFisher A10',
|
||||||
|
uri='tcp://ldmse-d910-ts:3001',
|
||||||
|
)
|
||||||
|
|
||||||
|
Mod('T',
|
||||||
|
'frappy_psi.qnw.TemperatureLoopA10',
|
||||||
|
'holder temperature',
|
||||||
|
io='io',
|
||||||
|
)
|
||||||
|
|
||||||
|
Mod('Th',
|
||||||
|
'frappy_psi.qnw.SensorA10',
|
||||||
|
'heat exch. temperature',
|
||||||
|
io='io',
|
||||||
|
)
|
@ -18,11 +18,11 @@
|
|||||||
# Module authors:
|
# Module authors:
|
||||||
# Oksana Shliakhtun <oksana.shliakhtun@psi.ch>
|
# Oksana Shliakhtun <oksana.shliakhtun@psi.ch>
|
||||||
# *****************************************************************************
|
# *****************************************************************************
|
||||||
|
import math
|
||||||
|
|
||||||
from frappy.core import Readable, Parameter, IntRange, EnumType, FloatRange, \
|
from frappy.core import Readable, Parameter, IntRange, EnumType, FloatRange, \
|
||||||
StringIO, HasIO, StringType, Property, Writable, Drivable, IDLE, ERROR, \
|
StringIO, HasIO, StringType, Property, Writable, Drivable, IDLE, ERROR, \
|
||||||
Attached, StructOf, WARN, Done, BoolType
|
Attached, StructOf, WARN, Done, BoolType, Enum
|
||||||
|
|
||||||
from frappy_psi.convergence import HasConvergence
|
from frappy_psi.convergence import HasConvergence
|
||||||
from frappy_psi.mixins import HasOutputModule, HasControlledBy
|
from frappy_psi.mixins import HasOutputModule, HasControlledBy
|
||||||
@ -37,20 +37,30 @@ class Ls340IO(StringIO):
|
|||||||
|
|
||||||
class LakeShore(HasIO):
|
class LakeShore(HasIO):
|
||||||
def set_par(self, cmd, *args):
|
def set_par(self, cmd, *args):
|
||||||
head = ','.join([cmd] + [f'{a:g}' for a in args])
|
head = ','.join([cmd] + [a if isinstance(a, str) else f'{a:g}' for a in args])
|
||||||
tail = cmd.replace(' ', '? ')
|
tail = cmd.replace(' ', '? ')
|
||||||
reply = self.communicate(f'{head};{tail}')
|
reply = self.communicate(f'{head};{tail}')
|
||||||
reply = [float(num) for num in reply.split(',')]
|
result = []
|
||||||
if len(reply) == 1:
|
for num in reply.split(','):
|
||||||
return reply[0]
|
try:
|
||||||
return reply
|
result.append(float(num))
|
||||||
|
except ValueError:
|
||||||
|
result.append(num)
|
||||||
|
if len(result) == 1:
|
||||||
|
return result[0]
|
||||||
|
return result
|
||||||
|
|
||||||
def get_par(self, cmd):
|
def get_par(self, cmd):
|
||||||
reply = self.communicate(cmd)
|
reply = self.communicate(cmd)
|
||||||
reply = [float(num) for num in reply.split(',')]
|
result = []
|
||||||
if len(reply) == 1:
|
for num in reply.split(','):
|
||||||
return reply[0]
|
try:
|
||||||
return reply
|
result.append(float(num))
|
||||||
|
except ValueError:
|
||||||
|
result.append(num)
|
||||||
|
if len(result) == 1:
|
||||||
|
return result[0]
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
class Sensor340(LakeShore, Readable):
|
class Sensor340(LakeShore, Readable):
|
||||||
@ -60,7 +70,7 @@ class Sensor340(LakeShore, Readable):
|
|||||||
ioClass = Ls340IO
|
ioClass = Ls340IO
|
||||||
channel = Property('lakeshore channel', StringType())
|
channel = Property('lakeshore channel', StringType())
|
||||||
alarm = Parameter('alarm limit', FloatRange(unit='K'), readonly=False)
|
alarm = Parameter('alarm limit', FloatRange(unit='K'), readonly=False)
|
||||||
# # define or alter the parameters
|
# define or alter the parameters
|
||||||
# as Readable.value exists already, we give only the modified property 'unit'
|
# as Readable.value exists already, we give only the modified property 'unit'
|
||||||
value = Parameter(unit='K')
|
value = Parameter(unit='K')
|
||||||
|
|
||||||
@ -96,17 +106,16 @@ class HeaterOutput(LakeShore, HasControlledBy, HasIO, Writable):
|
|||||||
max_power = Parameter('max heater power', datatype=FloatRange(0, 100), unit='W', readonly=False)
|
max_power = Parameter('max heater power', datatype=FloatRange(0, 100), unit='W', readonly=False)
|
||||||
value = Parameter('heater output', datatype=FloatRange(0, 100), unit='W')
|
value = Parameter('heater output', datatype=FloatRange(0, 100), unit='W')
|
||||||
target = Parameter('manual heater output', datatype=FloatRange(0, 100), unit='W')
|
target = Parameter('manual heater output', datatype=FloatRange(0, 100), unit='W')
|
||||||
loop = Property('lakeshore loop', IntRange(1, 2), default=1)
|
loop = Property('lakeshore loop', IntRange(1, 2), default=1) # output
|
||||||
channel = Property('attached channel', StringType())
|
channel = Property('attached channel', StringType()) # input
|
||||||
resistance = Property('heater resistance', datatype=FloatRange(10, 1000))
|
resistance = Property('heater resistance', datatype=FloatRange(10, 100))
|
||||||
current = Property('heater current', datatype=FloatRange(0, 2))
|
|
||||||
_range = 0
|
_range = 0
|
||||||
|
_max_power = 50
|
||||||
|
|
||||||
MAXCURRENTS = {1: 0.25, 2: 0.5, 3: 1.0, 4: 2.0}
|
MAXCURRENTS = {1: 0.25, 2: 0.5, 3: 1.0, 4: 2.0}
|
||||||
RANGES = {1: 1e4, 2: 1e3, 3: 1e2, 4: 1e1, 5: 1}
|
RANGES = {1: 1e4, 2: 1e3, 3: 1e2, 4: 1e1, 5: 1}
|
||||||
|
|
||||||
SETPOINTLIMS = 999
|
SETPOINTLIMS = 999
|
||||||
max_current = 0
|
|
||||||
|
|
||||||
STATUS_MAP = {
|
STATUS_MAP = {
|
||||||
0: (IDLE, ''),
|
0: (IDLE, ''),
|
||||||
@ -136,8 +145,7 @@ class HeaterOutput(LakeShore, HasControlledBy, HasIO, Writable):
|
|||||||
self._range = irange
|
self._range = irange
|
||||||
self.set_par(f'CLIMIT {self.loop}', self.SETPOINTLIMS, 0, 0, icurrent, irange)
|
self.set_par(f'CLIMIT {self.loop}', self.SETPOINTLIMS, 0, 0, icurrent, irange)
|
||||||
self.set_par(f'RANGE {irange}')
|
self.set_par(f'RANGE {irange}')
|
||||||
self.set_par(f'CDISP {self.loop}', 1, self.resistance, 0)
|
self.set_par(f'CDISP {self.loop}', 1, self.resistance, 1, 0)
|
||||||
return self.read_max_power()
|
|
||||||
|
|
||||||
def read_max_power(self):
|
def read_max_power(self):
|
||||||
setplimit, _, _, icurrent, irange = self.get_par(f'CLIMIT? {self.loop}')
|
setplimit, _, _, icurrent, irange = self.get_par(f'CLIMIT? {self.loop}')
|
||||||
@ -163,7 +171,7 @@ class HeaterOutput(LakeShore, HasControlledBy, HasIO, Writable):
|
|||||||
def write_target(self, target):
|
def write_target(self, target):
|
||||||
self.self_controlled()
|
self.self_controlled()
|
||||||
self.write_max_power(self.max_power)
|
self.write_max_power(self.max_power)
|
||||||
self.set_heater_mode(3)
|
self.set_heater_mode(3) # 3=open loop
|
||||||
self.set_range()
|
self.set_range()
|
||||||
percent = self.power_to_percent(target)
|
percent = self.power_to_percent(target)
|
||||||
reply = self.set_par(f'MOUT {self.loop}', percent)
|
reply = self.set_par(f'MOUT {self.loop}', percent)
|
||||||
@ -175,30 +183,91 @@ class HeaterOutput(LakeShore, HasControlledBy, HasIO, Writable):
|
|||||||
return self.get_par(f'RANGE?')
|
return self.get_par(f'RANGE?')
|
||||||
|
|
||||||
def read_value(self):
|
def read_value(self):
|
||||||
return self.percent_to_power(self.get_par(f'HTR?'))
|
return self.percent_to_power(self.get_par(f'HTR?{self.loop}'))
|
||||||
|
|
||||||
|
|
||||||
|
class HeaterOutput340(HeaterOutput):
|
||||||
|
resistance = Property('heater resistance', datatype=FloatRange(10, 100))
|
||||||
|
|
||||||
|
MAXCURRENTS = {1: 0.25, 2: 0.5, 3: 1.0, 4: 2.0}
|
||||||
|
RANGES = {1: 1e4, 2: 1e3, 3: 1e2, 4: 1e1, 5: 1}
|
||||||
|
|
||||||
|
STATUS_MAP = {
|
||||||
|
0: (IDLE, ''),
|
||||||
|
1: (ERROR, 'Power supply over voltage'),
|
||||||
|
2: (ERROR, 'Power supply under voltage'),
|
||||||
|
3: (ERROR, 'Output digital-to-analog Converter error'),
|
||||||
|
4: (ERROR, 'Current limit digital-to-analog converter error'),
|
||||||
|
5: (ERROR, 'Open heater load'),
|
||||||
|
6: (ERROR, 'Heater load less than 10 ohms')
|
||||||
|
}
|
||||||
|
|
||||||
|
def read_value(self):
|
||||||
|
return self.percent_to_power(self.get_par(f'HTR?')) # no loop to be given on 340
|
||||||
|
|
||||||
|
|
||||||
|
class HeaterOutput336(HeaterOutput):
|
||||||
|
|
||||||
|
power = 20
|
||||||
|
|
||||||
|
STATUS_MAP = {
|
||||||
|
0: (IDLE, ''),
|
||||||
|
1: (ERROR, 'Open heater load'),
|
||||||
|
2: (ERROR, 'Heater short')
|
||||||
|
}
|
||||||
|
|
||||||
|
def write_max_power(self, max_power):
|
||||||
|
max_current = min(math.sqrt(self.power / self.resistance), 2500 / self.resistance)
|
||||||
|
if self.loop == 1:
|
||||||
|
max_current_limit = 2
|
||||||
|
else:
|
||||||
|
max_current_limit = 1.414
|
||||||
|
if max_current > max_current_limit:
|
||||||
|
raise RangeError('max_power above limit')
|
||||||
|
if max_current >= max_current_limit / math.sqrt(10):
|
||||||
|
self._range = 3
|
||||||
|
user_current = max_current
|
||||||
|
elif max_current >= max_current_limit / 10:
|
||||||
|
self._range = 2
|
||||||
|
user_current = max_current * math.sqrt(10)
|
||||||
|
else:
|
||||||
|
self._range = 1
|
||||||
|
user_current = max_current * math.sqrt(100)
|
||||||
|
self.set_par(f'HTRSET {self.loop}', <1 or 2>, 0, user_current, 1)
|
||||||
|
max_power = max_current ** 2 * self.resistance
|
||||||
|
self._max_power = max_power
|
||||||
|
self.set_range()
|
||||||
|
return max_power
|
||||||
|
|
||||||
|
|
||||||
class TemperatureLoop340(HasConvergence, HasOutputModule, Sensor340, Drivable, LakeShore):
|
class TemperatureLoop340(HasConvergence, HasOutputModule, Sensor340, Drivable, LakeShore):
|
||||||
|
Status = Enum(
|
||||||
|
Drivable.Status,
|
||||||
|
RAMPING=370,
|
||||||
|
STABILIZING=380,
|
||||||
|
)
|
||||||
|
|
||||||
target = Parameter(unit='K')
|
target = Parameter(unit='K')
|
||||||
ctrlpars = Parameter('PID parameters',
|
ctrlpars = Parameter('PID parameters',
|
||||||
StructOf(p=FloatRange(0, 1000), i=FloatRange(0, 1000), d=FloatRange(0, 1000)),
|
StructOf(p=FloatRange(0, 1000), i=FloatRange(0, 1000), d=FloatRange(0, 1000)),
|
||||||
readonly=False)
|
readonly=False)
|
||||||
loop = Property('lakeshore loop', IntRange(1, 2), default=1)
|
loop = Property('lakeshore loop', IntRange(1, 2), default=1)
|
||||||
ramp = Parameter('ramp rate', FloatRange(min=0, max=1000), unit='K/min', readonly=False)
|
ramp = Parameter('ramp rate', FloatRange(min=0, max=100), unit='K/min', readonly=False)
|
||||||
ramp_used = Parameter('whether ramp is used or not', BoolType(), readonly=False)
|
ramp_used = Parameter('whether ramp is used or not', BoolType(), readonly=False)
|
||||||
setpoint = Parameter('setpoint', datatype=FloatRange, unit='K')
|
setpoint = Parameter('setpoint', datatype=FloatRange, unit='K')
|
||||||
|
|
||||||
def doPoll(self):
|
|
||||||
super().doPoll()
|
|
||||||
self.read_setpoint()
|
|
||||||
|
|
||||||
def write_target(self, target):
|
def write_target(self, target):
|
||||||
out = self.output_module
|
out = self.output_module
|
||||||
if out.controlled_by != self.name:
|
if not self.control_active:
|
||||||
|
if self.ramp_used:
|
||||||
|
self.set_par(f'RAMP {self.loop}', 0, self.ramp)
|
||||||
|
self.set_par(f'SETP {self.loop}', self.value)
|
||||||
|
self.set_par(f'RAMP {self.loop}', 1, self.ramp)
|
||||||
out.write_target(0)
|
out.write_target(0)
|
||||||
out.set_heater_mode(1)
|
|
||||||
out.write_max_power(out.max_power)
|
out.write_max_power(out.max_power)
|
||||||
|
out.set_heater_mode(1) # closed loop
|
||||||
self.activate_output()
|
self.activate_output()
|
||||||
|
self.start_state() # start the convergence check
|
||||||
out.set_range()
|
out.set_range()
|
||||||
self.set_par(f'SETP {self.loop}', target)
|
self.set_par(f'SETP {self.loop}', target)
|
||||||
return target
|
return target
|
||||||
@ -218,22 +287,37 @@ class TemperatureLoop340(HasConvergence, HasOutputModule, Sensor340, Drivable, L
|
|||||||
p, i, d = self.get_par(f'PID? {self.loop}')
|
p, i, d = self.get_par(f'PID? {self.loop}')
|
||||||
return {'p': p, 'i': i, 'd': d}
|
return {'p': p, 'i': i, 'd': d}
|
||||||
|
|
||||||
def write_ramp(self, ramp):
|
|
||||||
return self.set_par(f'RAMP {self.loop}', self.ramp_used, ramp)[1]
|
|
||||||
|
|
||||||
def write_ramp_used(self, ramp_used):
|
|
||||||
return self.set_par(f'RAMP {self.loop}', ramp_used, self.ramp)[0]
|
|
||||||
|
|
||||||
def read_ramp(self):
|
def read_ramp(self):
|
||||||
self.ramp_used, rate = self.get_par(f'RAMP? {self.loop}')
|
self.ramp_used, rate = self.get_par(f'RAMP? {self.loop}')
|
||||||
return rate
|
return rate
|
||||||
|
|
||||||
|
def write_ramp(self, ramp):
|
||||||
|
self.ramp_used = True
|
||||||
|
ramp = self.set_par(f'RAMP {self.loop}', self.ramp_used, ramp)[1]
|
||||||
|
# if self.control:
|
||||||
|
# self.ramp = ramp
|
||||||
|
# self.write_target(self.target)
|
||||||
|
# return Done
|
||||||
|
return ramp
|
||||||
|
|
||||||
|
def write_ramp_used(self, ramp_used):
|
||||||
|
ramp_used = self.set_par(f'RAMP {self.loop}', ramp_used, self.ramp)[0]
|
||||||
|
if self.ramp_used and not ramp_used:
|
||||||
|
self.write_target(self.target)
|
||||||
|
return ramp_used
|
||||||
|
|
||||||
def read_status(self):
|
def read_status(self):
|
||||||
statuscode, statustext = super().read_status()
|
statuscode, statustext = self.status
|
||||||
if statuscode == IDLE:
|
if self.ramp_used:
|
||||||
|
if self.read_setpoint() == self.target:
|
||||||
|
statuscode = self.Status.STABILIZING
|
||||||
|
else:
|
||||||
|
statuscode = self.Status.RAMPING
|
||||||
|
statustext = 'ramping'
|
||||||
|
if statuscode != ERROR:
|
||||||
return Done
|
return Done
|
||||||
if self.convergence_state.is_active():
|
if self.convergence_state.is_active:
|
||||||
self.convergence_state.stop_machine((statuscode, statustext))
|
self.stop_machine((statuscode, statustext))
|
||||||
return ERROR, statustext
|
return ERROR, statustext
|
||||||
|
|
||||||
|
|
||||||
|
109
frappy_psi/thermofisher.py
Normal file
109
frappy_psi/thermofisher.py
Normal file
@ -0,0 +1,109 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# *****************************************************************************
|
||||||
|
# This program is free software; you can redistribute it and/or modify it under
|
||||||
|
# the terms of the GNU General Public License as published by the Free Software
|
||||||
|
# Foundation; either version 2 of the License, or (at your option) any later
|
||||||
|
# version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
||||||
|
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
|
||||||
|
# details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License along with
|
||||||
|
# this program; if not, write to the Free Software Foundation, Inc.,
|
||||||
|
# 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
#
|
||||||
|
# Module authors:
|
||||||
|
# Oksana Shliakhtun <oksana.shliakhtun@psi.ch>
|
||||||
|
# *****************************************************************************
|
||||||
|
|
||||||
|
from frappy.core import StringIO, Parameter, Readable, HasIO, \
|
||||||
|
Drivable, FloatRange
|
||||||
|
|
||||||
|
|
||||||
|
class ThermFishIO(StringIO):
|
||||||
|
end_of_line = '\r'
|
||||||
|
identification = [('RVER', r'.[.*')] # Firmware Version
|
||||||
|
|
||||||
|
|
||||||
|
class SensorA10(HasIO, Readable):
|
||||||
|
ClassIO = ThermFishIO
|
||||||
|
value = Parameter('internal temperature', unit='degC')
|
||||||
|
|
||||||
|
def read_value(self):
|
||||||
|
return self.communicate('RT') # return the value and the units without space
|
||||||
|
|
||||||
|
def set_par(self, cmd):
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
class TemperatureLoopA10(SensorSC, Drivable):
|
||||||
|
value = Parameter('temperature', unit='degC')
|
||||||
|
target = Parameter('setpoint', FloatRange, readonly=False)
|
||||||
|
p_heat = Parameter('proportional heat parameter', FloatRange(), unit='degC', readonly=False)
|
||||||
|
i_heat = Parameter('integral heat parameter', FloatRange(), readonly=False)
|
||||||
|
d_heat = Parameter('derivative heat parameter', FloatRange(), readonly=False)
|
||||||
|
p_cool = Parameter('proportional cool parameter', FloatRange(), readonly=False)
|
||||||
|
i_cool = Parameter('integral cool parameter', FloatRange(), readonly=False)
|
||||||
|
d_cool = Parameter('derivative cool parameter', FloatRange(), readonly=False)
|
||||||
|
|
||||||
|
setpoint_num = ['', 1, 2, 3, 4, 5]
|
||||||
|
|
||||||
|
def read_target(self):
|
||||||
|
return self.communicate(f'RS{self.setpoint_num}')
|
||||||
|
|
||||||
|
def write_target(self):
|
||||||
|
target = self.communicate(f'SS{self.setpoint_num} {self.target}')
|
||||||
|
return target
|
||||||
|
|
||||||
|
## heat PID
|
||||||
|
def read_p_heat(self):
|
||||||
|
p_heat = self.communicate(f'RPH')
|
||||||
|
return p_heat
|
||||||
|
|
||||||
|
def write_p_heat(self, p_heat):
|
||||||
|
self.communicate(f'SPH {p_heat}')
|
||||||
|
return self.read_p_heat()
|
||||||
|
|
||||||
|
def read_i_heat(self):
|
||||||
|
i_heat = self.communicate(f'RIH')
|
||||||
|
return i_heat
|
||||||
|
|
||||||
|
def write_i_heat(self, i_heat):
|
||||||
|
self.communicate(f'SIH {i_heat}')
|
||||||
|
return self.read_i_heat()
|
||||||
|
|
||||||
|
def read_d_heat(self):
|
||||||
|
d_heat = self.communicate(f'RDH')
|
||||||
|
return d_heat
|
||||||
|
|
||||||
|
def write_d_heat(self, d_heat):
|
||||||
|
self.communicate(f'SDH {d_heat}')
|
||||||
|
return self.read_d_heat()
|
||||||
|
|
||||||
|
## cool PID
|
||||||
|
def read_p_cool(self):
|
||||||
|
p_cool = self.communicate(f'RPC')
|
||||||
|
return p_cool
|
||||||
|
|
||||||
|
def write_p_cool(self, p_cool):
|
||||||
|
self.communicate(f'SPC {p_cool}')
|
||||||
|
return self.read_p_cool()
|
||||||
|
|
||||||
|
def read_i_cool(self):
|
||||||
|
i_cool = self.communicate(f'RIC')
|
||||||
|
return i_cool
|
||||||
|
|
||||||
|
def write_i_cool(self, i_cool):
|
||||||
|
self.communicate(f'SIC {i_cool}')
|
||||||
|
return self.read_i_cool()
|
||||||
|
|
||||||
|
def read_d_cool(self):
|
||||||
|
d_cool = self.communicate(f'RDC')
|
||||||
|
return d_cool
|
||||||
|
|
||||||
|
def write_d_cool(self, d_cool):
|
||||||
|
self.communicate(f'SDC {d_cool}')
|
||||||
|
return self.read_d_cool()
|
Loading…
x
Reference in New Issue
Block a user