probe: Unify mean and median code between run_probe() and PROBE_ACCURACY

Factor out _calc_mean() and _calc_median() functions and call from
both run_probe() and cmd_PROBE_ACCURACY().

This also fixes a subtle error in the run_probe() median function - on
some kinematics the x and y position can change on a z move so the x
and y should be taken from the z probe values actually used.

Signed-off-by: Kevin O'Connor <kevin@koconnor.net>
This commit is contained in:
Kevin O'Connor 2019-05-31 14:16:59 -04:00 committed by KevinOConnor
parent 2b8dca5dbf
commit 74cc005ff3
1 changed files with 25 additions and 42 deletions

View File

@ -33,9 +33,9 @@ class PrinterProbe:
self.samples = config.getint('samples', 1, minval=1) self.samples = config.getint('samples', 1, minval=1)
self.sample_retract_dist = config.getfloat( self.sample_retract_dist = config.getfloat(
'sample_retract_dist', 2., above=0.) 'sample_retract_dist', 2., above=0.)
self.samples_result = config.getchoice('samples_result', atypes = {'median': 'median', 'average': 'average'}
{'median': 0, 'average': 1}, self.samples_result = config.getchoice('samples_result', atypes,
default='average') 'average')
# Register z_virtual_endstop pin # Register z_virtual_endstop pin
self.printer.lookup_object('pins').register_chip('probe', self) self.printer.lookup_object('pins').register_chip('probe', self)
# Register PROBE/QUERY_PROBE commands # Register PROBE/QUERY_PROBE commands
@ -86,6 +86,18 @@ class PrinterProbe:
toolhead.move(curpos, speed) toolhead.move(curpos, speed)
except homing.EndstopError as e: except homing.EndstopError as e:
raise self.gcode.error(str(e)) raise self.gcode.error(str(e))
def _calc_mean(self, positions):
count = float(len(positions))
return [sum([pos[i] for pos in positions]) / count
for i in range(3)]
def _calc_median(self, positions):
z_sorted = sorted(positions, key=(lambda p: p[2]))
middle = len(positions) // 2
if (len(positions) & 1) == 1:
# odd number of samples
return z_sorted[middle]
# even number of samples
return self._calc_mean(z_sorted[middle-1:middle+1])
def run_probe(self): def run_probe(self):
positions = [] positions = []
for i in range(self.samples): for i in range(self.samples):
@ -95,26 +107,9 @@ class PrinterProbe:
# retract # retract
liftpos = [None, None, pos[2] + self.sample_retract_dist] liftpos = [None, None, pos[2] + self.sample_retract_dist]
self._move(liftpos, self.speed) self._move(liftpos, self.speed)
if self.samples_result == 1: if self.samples_result == 'median':
# Calculate Average return self._calc_median(positions)
calculated_value = [sum([pos[i] for pos in positions]) / return self._calc_mean(positions)
self.samples for i in range(3)]
else:
# Calculate Median
sorted_z_positions = sorted([position[2]
for position in positions])
middle = self.samples // 2
if (self.samples & 1) == 1:
# odd number of samples
median = sorted_z_positions[middle]
else:
# even number of samples
median = (sorted_z_positions[middle] +
sorted_z_positions[middle - 1]) / 2
calculated_value = [positions[0][0],
positions[0][1],
median]
return calculated_value
cmd_PROBE_help = "Probe Z-height at current XY position" cmd_PROBE_help = "Probe Z-height at current XY position"
def cmd_PROBE(self, params): def cmd_PROBE(self, params):
pos = self.run_probe() pos = self.run_probe()
@ -130,7 +125,6 @@ class PrinterProbe:
cmd_PROBE_ACCURACY_help = "Probe Z-height accuracy at current XY position" cmd_PROBE_ACCURACY_help = "Probe Z-height accuracy at current XY position"
def cmd_PROBE_ACCURACY(self, params): def cmd_PROBE_ACCURACY(self, params):
toolhead = self.printer.lookup_object('toolhead') toolhead = self.printer.lookup_object('toolhead')
probes = []
pos = toolhead.get_position() pos = toolhead.get_position()
number_of_reads = self.gcode.get_int('REPEAT', params, default=10, number_of_reads = self.gcode.get_int('REPEAT', params, default=10,
minval=4, maxval=50) minval=4, maxval=50)
@ -147,36 +141,25 @@ class PrinterProbe:
x_start_position, y_start_position, x_start_position, y_start_position,
z_start_position, number_of_reads, speed)) z_start_position, number_of_reads, speed))
# Probe bed "number_of_reads" times # Probe bed "number_of_reads" times
sum_reads = 0 positions = []
for i in range(number_of_reads): for i in range(number_of_reads):
# Move Z to start reading position # Move Z to start reading position
self._move(start_pos, speed) self._move(start_pos, speed)
# Probe # Probe
pos = self._probe(speed) pos = self._probe(speed)
# Get Z value, accumulate value to calculate average positions.append(pos)
# and save it to calculate standard deviation
sum_reads += pos[2]
probes.append(pos[2])
# Move Z to start reading position # Move Z to start reading position
self._move(start_pos, speed) self._move(start_pos, speed)
# Calculate maximum, minimum and average values # Calculate maximum, minimum and average values
max_value = max(probes) max_value = max([p[2] for p in positions])
min_value = min(probes) min_value = min([p[2] for p in positions])
avg_value = sum(probes) / number_of_reads avg_value = self._calc_mean(positions)[2]
median = self._calc_median(positions)[2]
# calculate the standard deviation # calculate the standard deviation
deviation_sum = 0 deviation_sum = 0
for i in range(number_of_reads): for i in range(number_of_reads):
deviation_sum += pow(probes[i] - avg_value, 2) deviation_sum += pow(positions[i][2] - avg_value, 2)
sigma = (deviation_sum / number_of_reads) ** 0.5 sigma = (deviation_sum / number_of_reads) ** 0.5
# Median
sorted_probes = sorted(probes)
middle = number_of_reads//2
if (number_of_reads & 1) == 1:
# odd number of reads
median = sorted_probes[middle]
else:
# even number of reads
median = (sorted_probes[middle]+sorted_probes[middle-1])/2
# Show information # Show information
self.gcode.respond_info( self.gcode.respond_info(
"probe accuracy results: maximum %.6f, minimum %.6f, " "probe accuracy results: maximum %.6f, minimum %.6f, "