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:
parent
2b8dca5dbf
commit
74cc005ff3
|
@ -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, "
|
||||||
|
|
Loading…
Reference in New Issue