tmc2208: move local set_field/get_field helpers to FieldHelper class
Move these helper functions to the FieldHelper class so that they can be used by other Trinamic drivers. Signed-off-by: Kevin O'Connor <kevin@koconnor.net>
This commit is contained in:
parent
17a17c5b77
commit
30b6414295
|
@ -42,22 +42,44 @@ def decode_signed_int(val, bits):
|
||||||
return val
|
return val
|
||||||
|
|
||||||
class FieldHelper:
|
class FieldHelper:
|
||||||
def __init__(self, all_fields, field_formatters={}):
|
def __init__(self, all_fields, field_formatters={}, registers=None):
|
||||||
self.all_fields = all_fields
|
self.all_fields = all_fields
|
||||||
self.field_formatters = field_formatters
|
self.field_formatters = field_formatters
|
||||||
|
self.registers = registers
|
||||||
|
if self.registers is None:
|
||||||
|
self.registers = {}
|
||||||
self.field_to_register = { f: r for r, fields in self.all_fields.items()
|
self.field_to_register = { f: r for r, fields in self.all_fields.items()
|
||||||
for f in fields }
|
for f in fields }
|
||||||
def lookup_register(self, field_name):
|
def get_field(self, field_name, reg_value=None, reg_name=None):
|
||||||
# Return the name of the register containing the given field
|
|
||||||
return self.field_to_register[field_name]
|
|
||||||
def get_field(self, reg_name, field_name, reg_value):
|
|
||||||
# Returns value of the register field
|
# Returns value of the register field
|
||||||
|
if reg_name is None:
|
||||||
|
reg_name = self.field_to_register[field_name]
|
||||||
|
if reg_value is None:
|
||||||
|
reg_value = self.registers[reg_name]
|
||||||
mask = self.all_fields[reg_name][field_name]
|
mask = self.all_fields[reg_name][field_name]
|
||||||
return (reg_value & mask) >> ffs(mask)
|
return (reg_value & mask) >> ffs(mask)
|
||||||
def set_field(self, reg_name, field_name, reg_value, field_value):
|
def set_field(self, field_name, field_value, reg_value=None, reg_name=None):
|
||||||
# Returns register value with field bits filled with supplied value
|
# Returns register value with field bits filled with supplied value
|
||||||
|
if reg_name is None:
|
||||||
|
reg_name = self.field_to_register[field_name]
|
||||||
|
if reg_value is None:
|
||||||
|
reg_value = self.registers.get(reg_name, 0)
|
||||||
mask = self.all_fields[reg_name][field_name]
|
mask = self.all_fields[reg_name][field_name]
|
||||||
return (reg_value & ~mask) | ((field_value << ffs(mask)) & mask)
|
new_value = (reg_value & ~mask) | ((field_value << ffs(mask)) & mask)
|
||||||
|
self.registers[reg_name] = new_value
|
||||||
|
return new_value
|
||||||
|
def set_config_field(self, config, field_name, default, config_name=None):
|
||||||
|
# Allow a field to be set from the config file
|
||||||
|
if config_name is None:
|
||||||
|
config_name = "driver_" + field_name.upper()
|
||||||
|
reg_name = self.field_to_register[field_name]
|
||||||
|
mask = self.all_fields[reg_name][field_name]
|
||||||
|
maxval = mask >> ffs(mask)
|
||||||
|
if maxval == 1:
|
||||||
|
val = config.getboolean(config_name, default)
|
||||||
|
else:
|
||||||
|
val = config.getint(config_name, default, minval=0, maxval=maxval)
|
||||||
|
return self.set_field(field_name, val)
|
||||||
def pretty_format(self, reg_name, value):
|
def pretty_format(self, reg_name, value):
|
||||||
# Provide a string description of a register
|
# Provide a string description of a register
|
||||||
reg_fields = self.all_fields.get(reg_name, {})
|
reg_fields = self.all_fields.get(reg_name, {})
|
||||||
|
|
|
@ -287,15 +287,16 @@ class TMC2208:
|
||||||
self.cmd_DUMP_TMC, desc=self.cmd_DUMP_TMC_help)
|
self.cmd_DUMP_TMC, desc=self.cmd_DUMP_TMC_help)
|
||||||
# Setup basic register values
|
# Setup basic register values
|
||||||
self.ifcnt = None
|
self.ifcnt = None
|
||||||
self.field_helper = tmc2130.FieldHelper(Fields, FieldFormatters)
|
|
||||||
self.regs = collections.OrderedDict()
|
self.regs = collections.OrderedDict()
|
||||||
self.set_field("pdn_disable", True)
|
self.field_helper = tmc2130.FieldHelper(
|
||||||
self.set_field("mstep_reg_select", True)
|
Fields, FieldFormatters, self.regs)
|
||||||
self.set_field("multistep_filt", True)
|
self.field_helper.set_field("pdn_disable", True)
|
||||||
|
self.field_helper.set_field("mstep_reg_select", True)
|
||||||
|
self.field_helper.set_field("multistep_filt", True)
|
||||||
steps = {'256': 0, '128': 1, '64': 2, '32': 3, '16': 4,
|
steps = {'256': 0, '128': 1, '64': 2, '32': 3, '16': 4,
|
||||||
'8': 5, '4': 6, '2': 7, '1': 8}
|
'8': 5, '4': 6, '2': 7, '1': 8}
|
||||||
mres = config.getchoice('microsteps', steps)
|
mres = config.getchoice('microsteps', steps)
|
||||||
self.set_field("MRES", mres)
|
self.field_helper.set_field("MRES", mres)
|
||||||
# Calculate current
|
# Calculate current
|
||||||
vsense = False
|
vsense = False
|
||||||
run_current = config.getfloat('run_current', above=0., maxval=2.)
|
run_current = config.getfloat('run_current', above=0., maxval=2.)
|
||||||
|
@ -308,29 +309,30 @@ class TMC2208:
|
||||||
vsense = True
|
vsense = True
|
||||||
irun = self.current_bits(run_current, sense_resistor, vsense)
|
irun = self.current_bits(run_current, sense_resistor, vsense)
|
||||||
ihold = self.current_bits(hold_current, sense_resistor, vsense)
|
ihold = self.current_bits(hold_current, sense_resistor, vsense)
|
||||||
self.set_field("vsense", vsense)
|
self.field_helper.set_field("vsense", vsense)
|
||||||
self.set_field("IHOLD", ihold)
|
self.field_helper.set_field("IHOLD", ihold)
|
||||||
self.set_field("IRUN", irun)
|
self.field_helper.set_field("IRUN", irun)
|
||||||
# Setup stealthchop
|
# Setup stealthchop
|
||||||
sc_velocity = config.getfloat('stealthchop_threshold', 0., minval=0.)
|
sc_velocity = config.getfloat('stealthchop_threshold', 0., minval=0.)
|
||||||
sc_threshold = self.velocity_to_clock(config, sc_velocity, mres)
|
sc_threshold = self.velocity_to_clock(config, sc_velocity, mres)
|
||||||
self.set_field("en_spreadCycle", not sc_velocity)
|
self.field_helper.set_field("en_spreadCycle", not sc_velocity)
|
||||||
self.set_field("TPWMTHRS", max(0, min(0xfffff, sc_threshold)))
|
self.field_helper.set_field("TPWMTHRS", sc_threshold)
|
||||||
# Allow other registers to be set from the config
|
# Allow other registers to be set from the config
|
||||||
self.set_config_field(config, "toff", 3)
|
set_config_field = self.field_helper.set_config_field
|
||||||
self.set_config_field(config, "hstrt", 5)
|
set_config_field(config, "toff", 3)
|
||||||
self.set_config_field(config, "hend", 0)
|
set_config_field(config, "hstrt", 5)
|
||||||
self.set_config_field(config, "TBL", 2, "driver_BLANK_TIME_SELECT")
|
set_config_field(config, "hend", 0)
|
||||||
self.set_config_field(config, "intpol", True, "interpolate")
|
set_config_field(config, "TBL", 2, "driver_BLANK_TIME_SELECT")
|
||||||
self.set_config_field(config, "IHOLDDELAY", 8)
|
set_config_field(config, "intpol", True, "interpolate")
|
||||||
self.set_config_field(config, "TPOWERDOWN", 20)
|
set_config_field(config, "IHOLDDELAY", 8)
|
||||||
self.set_config_field(config, "PWM_OFS", 36)
|
set_config_field(config, "TPOWERDOWN", 20)
|
||||||
self.set_config_field(config, "PWM_GRAD", 14)
|
set_config_field(config, "PWM_OFS", 36)
|
||||||
self.set_config_field(config, "pwm_freq", 1)
|
set_config_field(config, "PWM_GRAD", 14)
|
||||||
self.set_config_field(config, "pwm_autoscale", True)
|
set_config_field(config, "pwm_freq", 1)
|
||||||
self.set_config_field(config, "pwm_autograd", True)
|
set_config_field(config, "pwm_autoscale", True)
|
||||||
self.set_config_field(config, "PWM_REG", 8)
|
set_config_field(config, "pwm_autograd", True)
|
||||||
self.set_config_field(config, "PWM_LIM", 12)
|
set_config_field(config, "PWM_REG", 8)
|
||||||
|
set_config_field(config, "PWM_LIM", 12)
|
||||||
def current_bits(self, current, sense_resistor, vsense_on):
|
def current_bits(self, current, sense_resistor, vsense_on):
|
||||||
sense_resistor += 0.020
|
sense_resistor += 0.020
|
||||||
vsense = 0.32
|
vsense = 0.32
|
||||||
|
@ -346,7 +348,8 @@ class TMC2208:
|
||||||
stepper_config = config.getsection(stepper_name)
|
stepper_config = config.getsection(stepper_name)
|
||||||
step_dist = stepper_config.getfloat('step_distance')
|
step_dist = stepper_config.getfloat('step_distance')
|
||||||
step_dist_256 = step_dist / (1 << mres)
|
step_dist_256 = step_dist / (1 << mres)
|
||||||
return int(TMC_FREQUENCY * step_dist_256 / velocity + .5)
|
threshold = int(TMC_FREQUENCY * step_dist_256 / velocity + .5)
|
||||||
|
return max(0, min(0xfffff, threshold))
|
||||||
def build_config(self):
|
def build_config(self):
|
||||||
bit_ticks = int(self.mcu.get_adjusted_freq() / 9000.)
|
bit_ticks = int(self.mcu.get_adjusted_freq() / 9000.)
|
||||||
self.mcu.add_config_cmd(
|
self.mcu.add_config_cmd(
|
||||||
|
@ -355,27 +358,6 @@ class TMC2208:
|
||||||
cmd_queue = self.mcu.alloc_command_queue()
|
cmd_queue = self.mcu.alloc_command_queue()
|
||||||
self.tmcuart_send_cmd = self.mcu.lookup_command(
|
self.tmcuart_send_cmd = self.mcu.lookup_command(
|
||||||
"tmcuart_send oid=%c write=%*s read=%c", cq=cmd_queue)
|
"tmcuart_send oid=%c write=%*s read=%c", cq=cmd_queue)
|
||||||
def get_field(self, field_name):
|
|
||||||
# Return a field from the local cache of register values
|
|
||||||
reg_name = self.field_helper.lookup_register(field_name)
|
|
||||||
return self.field_helper.get_field(
|
|
||||||
reg_name, field_name, self.regs.get(reg_name, 0))
|
|
||||||
def set_field(self, field_name, field_value):
|
|
||||||
# Set a field in the local cache of register values
|
|
||||||
reg_name = self.field_helper.lookup_register(field_name)
|
|
||||||
self.regs[reg_name] = self.field_helper.set_field(
|
|
||||||
reg_name, field_name, self.regs.get(reg_name, 0), field_value)
|
|
||||||
def set_config_field(self, config, field_name, default, config_name=None):
|
|
||||||
# Allow a field to be set from the config file
|
|
||||||
reg_name = self.field_helper.lookup_register(field_name)
|
|
||||||
if config_name is None:
|
|
||||||
config_name = "driver_" + field_name.upper()
|
|
||||||
maxval = self.field_helper.get_field(reg_name, field_name, 0xffffffff)
|
|
||||||
if maxval == 1:
|
|
||||||
val = config.getboolean(config_name, default)
|
|
||||||
else:
|
|
||||||
val = config.getint(config_name, default, minval=0, maxval=maxval)
|
|
||||||
self.set_field(field_name, val)
|
|
||||||
def handle_connect(self):
|
def handle_connect(self):
|
||||||
for reg_name, val in self.regs.items():
|
for reg_name, val in self.regs.items():
|
||||||
self.set_register(reg_name, val)
|
self.set_register(reg_name, val)
|
||||||
|
@ -408,11 +390,10 @@ class TMC2208:
|
||||||
raise self.printer.config_error(
|
raise self.printer.config_error(
|
||||||
"Unable to write tmc2208 '%s' register %s" % (self.name, reg_name))
|
"Unable to write tmc2208 '%s' register %s" % (self.name, reg_name))
|
||||||
def get_microsteps(self):
|
def get_microsteps(self):
|
||||||
return 256 >> self.get_field("MRES")
|
return 256 >> self.field_helper.get_field("MRES")
|
||||||
def get_phase(self):
|
def get_phase(self):
|
||||||
mscnt = self.get_register("MSCNT")
|
mscnt = self.field_helper.get_field("MSCNT", self.get_register("MSCNT"))
|
||||||
mscnt = self.field_helper.get_field("MSCNT", "MSCNT", mscnt)
|
return mscnt >> self.field_helper.get_field("MRES")
|
||||||
return mscnt >> self.get_field("MRES")
|
|
||||||
cmd_DUMP_TMC_help = "Read and display TMC stepper driver registers"
|
cmd_DUMP_TMC_help = "Read and display TMC stepper driver registers"
|
||||||
def cmd_DUMP_TMC(self, params):
|
def cmd_DUMP_TMC(self, params):
|
||||||
self.printer.lookup_object('toolhead').get_last_move_time()
|
self.printer.lookup_object('toolhead').get_last_move_time()
|
||||||
|
@ -426,8 +407,7 @@ class TMC2208:
|
||||||
# IOIN has different mappings depending on the driver type
|
# IOIN has different mappings depending on the driver type
|
||||||
# (SEL_A field of IOIN reg)
|
# (SEL_A field of IOIN reg)
|
||||||
if reg_name == "IOIN":
|
if reg_name == "IOIN":
|
||||||
drv_type = self.field_helper.get_field("IOIN@TMC222x", "SEL_A",
|
drv_type = self.field_helper.get_field("SEL_A", val)
|
||||||
val)
|
|
||||||
reg_name = "IOIN@TMC220x" if drv_type else "IOIN@TMC222x"
|
reg_name = "IOIN@TMC220x" if drv_type else "IOIN@TMC222x"
|
||||||
msg = self.field_helper.pretty_format(reg_name, val)
|
msg = self.field_helper.pretty_format(reg_name, val)
|
||||||
logging.info(msg)
|
logging.info(msg)
|
||||||
|
|
Loading…
Reference in New Issue