Skip to content

Commit 4d819b1

Browse files
committed
chore: internal-release->edge for 0.13.0
2 parents ae317ed + e2b6b8f commit 4d819b1

File tree

28 files changed

+2931
-469
lines changed

28 files changed

+2931
-469
lines changed

api/release-notes-internal.md

+5-2
Original file line numberDiff line numberDiff line change
@@ -20,11 +20,13 @@ For more details about this release, please see the full [technical change log][
2020
- The ODD should prompt you to update your pipettes when you attach one that needs it (!)
2121
- The 96 shouldn't fall down when you attach it now (!)
2222
- More little stall detection fixes
23+
- The ODD will now display a spinner until the robot server has started
24+
- Homing now goes to the limit switch then backs off until the limit switch is released
25+
- Support for PVT (v3.5) pipettes
2326

2427
## Big Things That Don't Work Yet So Don't Report Bugs About Them
2528

2629
### ODD
27-
- The ODD doesn't really tell you if the robot server hasn't started yet; if a robot looks on but has the name "opentrons", or says it's not network-connected when you know it is, probably the server isn't up yet, give it another little bit
2830
- It can take a while for the robot to start after installing an update (it's the firmware updates happening on boot). Allow 10 minutes after an update that has a firmware change.
2931

3032
### Robot Control
@@ -33,14 +35,15 @@ For more details about this release, please see the full [technical change log][
3335

3436
## Big Things That Do Work Please Do Report Bugs About Them
3537
### Robot Control
36-
- Liquid handling protocols with 1 and 8 channel pipettes
38+
- Protocol behavior
3739
- Labware movement between slots/modules, both manual and with gripper, from python protocols
3840
- Labware drop/gripper crash errors, but they're very insensitive
3941
- Pipette and gripper automated offset calibration
4042
- Network connectivity and discoverability
4143
- Firmware update for all devices
4244
- Cancelling a protocol run. We're even more sure we fixed this so definitely tell us if it's not.
4345
- USB connectivity
46+
- Stall detection firing basically ever unless you clearly ran into something
4447

4548
### ODD
4649
- Protocol execution including end-of-protocol screen

api/src/opentrons/hardware_control/backends/ot3controller.py

+59-78
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,7 @@
3131
create_move_group,
3232
axis_to_node,
3333
get_current_settings,
34-
create_home_group,
34+
create_home_groups,
3535
node_to_axis,
3636
sensor_node_for_mount,
3737
sensor_node_for_pipette,
@@ -40,6 +40,7 @@
4040
create_gripper_jaw_home_group,
4141
create_gripper_jaw_hold_group,
4242
create_tip_action_group,
43+
create_tip_action_home_group,
4344
PipetteAction,
4445
motor_nodes,
4546
LIMIT_SWITCH_OVERTRAVEL_DISTANCE,
@@ -115,6 +116,7 @@
115116
AionotifyEvent,
116117
OT3Mount,
117118
OT3AxisMap,
119+
OT3AxisKind,
118120
CurrentConfig,
119121
MotorStatus,
120122
InstrumentProbeType,
@@ -495,85 +497,69 @@ def _get_axis_home_distance(self, axis: Axis) -> float:
495497
else:
496498
return -1 * self.axis_bounds[axis][1] - self.axis_bounds[axis][0]
497499

500+
def _build_axes_home_groups(
501+
self, axes: Sequence[Axis], speed_settings: Dict[OT3AxisKind, float]
502+
) -> List[MoveGroup]:
503+
present_axes = [ax for ax in axes if self.axis_is_present(ax)]
504+
if not present_axes:
505+
return []
506+
else:
507+
distances = {ax: self._get_axis_home_distance(ax) for ax in present_axes}
508+
velocities = {
509+
ax: -1 * speed_settings[Axis.to_kind(ax)] for ax in present_axes
510+
}
511+
return create_home_groups(distances, velocities)
512+
498513
def _build_home_pipettes_runner(
499514
self,
500515
axes: Sequence[Axis],
501516
gantry_load: GantryLoad,
502517
) -> Optional[MoveGroupRunner]:
518+
pipette_axes = [ax for ax in axes if ax in Axis.pipette_axes()]
519+
if not pipette_axes:
520+
return None
521+
503522
speed_settings = self._configuration.motion_settings.max_speed_discontinuity[
504523
gantry_load
505524
]
506-
507-
distances_pipette = {
508-
ax: self._get_axis_home_distance(ax)
509-
for ax in axes
510-
if ax in Axis.pipette_axes()
511-
}
512-
velocities_pipette = {
513-
ax: -1 * speed_settings[Axis.to_kind(ax)]
514-
for ax in axes
515-
if ax in Axis.pipette_axes()
516-
}
517-
518-
move_group_pipette = []
519-
if distances_pipette and velocities_pipette:
520-
pipette_move = self._filter_move_group(
521-
create_home_group(distances_pipette, velocities_pipette)
522-
)
523-
move_group_pipette.append(pipette_move)
524-
525-
if move_group_pipette:
526-
return MoveGroupRunner(move_groups=move_group_pipette, start_at_index=2)
527-
return None
525+
move_groups: List[MoveGroup] = self._build_axes_home_groups(
526+
pipette_axes, speed_settings
527+
)
528+
return MoveGroupRunner(move_groups=move_groups)
528529

529530
def _build_home_gantry_z_runner(
530531
self,
531532
axes: Sequence[Axis],
532533
gantry_load: GantryLoad,
533534
) -> Optional[MoveGroupRunner]:
535+
gantry_axes = [ax for ax in axes if ax in Axis.gantry_axes()]
536+
if not gantry_axes:
537+
return None
538+
534539
speed_settings = self._configuration.motion_settings.max_speed_discontinuity[
535540
gantry_load
536541
]
537542

538-
distances_gantry = {
539-
ax: self._get_axis_home_distance(ax)
540-
for ax in axes
541-
if ax in Axis.gantry_axes() and ax not in Axis.ot3_mount_axes()
542-
}
543-
velocities_gantry = {
544-
ax: -1 * speed_settings[Axis.to_kind(ax)]
545-
for ax in axes
546-
if ax in Axis.gantry_axes() and ax not in Axis.ot3_mount_axes()
547-
}
548-
distances_z = {
549-
ax: self._get_axis_home_distance(ax)
550-
for ax in axes
551-
if ax in Axis.ot3_mount_axes()
552-
}
553-
velocities_z = {
554-
ax: -1 * speed_settings[Axis.to_kind(ax)]
555-
for ax in axes
556-
if ax in Axis.ot3_mount_axes()
557-
}
558-
move_group_gantry_z = []
559-
if distances_z and velocities_z:
560-
z_move = self._filter_move_group(
561-
create_home_group(distances_z, velocities_z)
562-
)
563-
move_group_gantry_z.append(z_move)
564-
if distances_gantry and velocities_gantry:
565-
# home X axis before Y axis, to avoid collision with thermo-cycler lid
566-
# that could be in the back-left corner
567-
for ax in [Axis.X, Axis.Y]:
568-
if ax in axes:
569-
gantry_move = self._filter_move_group(
570-
create_home_group(
571-
{ax: distances_gantry[ax]}, {ax: velocities_gantry[ax]}
572-
)
573-
)
574-
move_group_gantry_z.append(gantry_move)
575-
if move_group_gantry_z:
576-
return MoveGroupRunner(move_groups=move_group_gantry_z)
543+
# first home all the present mount axes
544+
z_axes = list(filter(lambda ax: ax in Axis.ot3_mount_axes(), gantry_axes))
545+
z_groups = self._build_axes_home_groups(z_axes, speed_settings)
546+
547+
# home X axis before Y axis, to avoid collision with thermo-cycler lid
548+
# that could be in the back-left corner
549+
x_groups = (
550+
self._build_axes_home_groups([Axis.X], speed_settings)
551+
if Axis.X in gantry_axes
552+
else []
553+
)
554+
y_groups = (
555+
self._build_axes_home_groups([Axis.Y], speed_settings)
556+
if Axis.Y in gantry_axes
557+
else []
558+
)
559+
560+
move_groups = [*z_groups, *x_groups, *y_groups]
561+
if move_groups:
562+
return MoveGroupRunner(move_groups=move_groups)
577563
return None
578564

579565
@requires_update
@@ -619,18 +605,6 @@ async def home(
619605
self._handle_motor_status_response(position)
620606
return axis_convert(self._position, 0.0)
621607

622-
def _filter_move_group(self, move_group: MoveGroup) -> MoveGroup:
623-
new_group: MoveGroup = []
624-
for step in move_group:
625-
new_group.append(
626-
{
627-
node: axis_step
628-
for node, axis_step in step.items()
629-
if node in self._motor_nodes()
630-
}
631-
)
632-
return new_group
633-
634608
async def tip_action(
635609
self,
636610
axes: Sequence[Axis],
@@ -640,10 +614,17 @@ async def tip_action(
640614
) -> None:
641615
if tip_action == "home":
642616
speed = speed * -1
643-
move_group = create_tip_action_group(
644-
axes, distance, speed, cast(PipetteAction, tip_action)
645-
)
646-
runner = MoveGroupRunner(move_groups=[move_group])
617+
runner = MoveGroupRunner(
618+
move_groups=create_tip_action_home_group(axes, distance, speed)
619+
)
620+
else:
621+
runner = MoveGroupRunner(
622+
move_groups=[
623+
create_tip_action_group(
624+
axes, distance, speed, cast(PipetteAction, tip_action)
625+
)
626+
]
627+
)
647628
positions = await runner.run(can_messenger=self._messenger)
648629
for axis, point in positions.items():
649630
self._position.update({axis: point[0]})

api/src/opentrons/hardware_control/backends/ot3utils.py

+37-9
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
"""Shared utilities for ot3 hardware control."""
22
from typing import Dict, Iterable, List, Set, Tuple, TypeVar, Sequence, cast
33
from typing_extensions import Literal
4+
from logging import getLogger
45
from opentrons.config.defaults_ot3 import DEFAULT_CALIBRATION_AXIS_MAX_SPEED
56
from opentrons.config.types import OT3MotionSettings, OT3CurrentSettings, GantryLoad
67
from opentrons.hardware_control.types import (
@@ -46,12 +47,14 @@
4647
NodeIdMotionValues,
4748
create_home_step,
4849
create_backoff_step,
50+
create_tip_action_backoff_step,
4951
MoveGroup,
5052
MoveType,
5153
MoveStopCondition,
5254
create_gripper_jaw_step,
5355
create_tip_action_step,
5456
)
57+
from opentrons_hardware.hardware_control.constants import interrupts_per_sec
5558

5659
GRIPPER_JAW_HOME_TIME: float = 10
5760
GRIPPER_JAW_GRIP_TIME: float = 10
@@ -83,6 +86,8 @@
8386

8487
USB_SUBSYSTEM = {target: subsystem for subsystem, target in SUBSYSTEM_USB.items()}
8588

89+
LOG = getLogger(__name__)
90+
8691

8792
def axis_nodes() -> List["NodeId"]:
8893
return [
@@ -331,6 +336,11 @@ def create_move_group(
331336
for move in moves:
332337
unit_vector = move.unit_vector
333338
for block in move.blocks:
339+
if block.time < (3.0 / interrupts_per_sec):
340+
LOG.info(
341+
f"Skipping move block with time {block.time} (<{3.0/interrupts_per_sec})"
342+
)
343+
continue
334344
distances = unit_vector_multiplication(unit_vector, block.distance)
335345
node_id_distances = _convert_to_node_id_dict(distances)
336346
velocities = unit_vector_multiplication(unit_vector, block.initial_speed)
@@ -349,21 +359,39 @@ def create_move_group(
349359
return move_group, {k: float(v) for k, v in pos.items()}
350360

351361

352-
def create_home_group(
362+
def create_home_groups(
353363
distance: Dict[Axis, float], velocity: Dict[Axis, float]
354-
) -> MoveGroup:
364+
) -> List[MoveGroup]:
355365
node_id_distances = _convert_to_node_id_dict(distance)
356366
node_id_velocities = _convert_to_node_id_dict(velocity)
357-
home = create_home_step(
358-
distance=node_id_distances,
359-
velocity=node_id_velocities,
360-
)
367+
home_group = [
368+
create_home_step(distance=node_id_distances, velocity=node_id_velocities)
369+
]
361370
# halve the homing speed for backoff
362371
backoff_velocities = {k: v / 2 for k, v in node_id_velocities.items()}
363-
backoff = create_backoff_step(backoff_velocities)
372+
backoff_group = [create_backoff_step(backoff_velocities)]
373+
return [home_group, backoff_group]
364374

365-
move_group: MoveGroup = [home, backoff]
366-
return move_group
375+
376+
def create_tip_action_home_group(
377+
axes: Sequence[Axis], distance: float, velocity: float
378+
) -> List[MoveGroup]:
379+
current_nodes = [axis_to_node(ax) for ax in axes]
380+
home_group = [
381+
create_tip_action_step(
382+
velocity={node_id: np.float64(velocity) for node_id in current_nodes},
383+
distance={node_id: np.float64(distance) for node_id in current_nodes},
384+
present_nodes=current_nodes,
385+
action=PipetteTipActionType.home,
386+
)
387+
]
388+
389+
backoff_group = [
390+
create_tip_action_backoff_step(
391+
velocity={node_id: np.float64(velocity / 2) for node_id in current_nodes}
392+
)
393+
]
394+
return [home_group, backoff_group]
367395

368396

369397
def create_tip_action_group(

api/src/opentrons/hardware_control/instruments/ot3/gripper.py

+15-12
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
""" Classes and functions for gripper state tracking
44
"""
55
import logging
6-
from typing import Any, Optional, Set, Dict
6+
from typing import Any, Optional, Set, Dict, Tuple
77

88
from opentrons.types import Point
99
from opentrons.config import gripper_config
@@ -237,7 +237,7 @@ def _reload_gripper(
237237
new_config: GripperDefinition,
238238
attached_instr: Gripper,
239239
cal_offset: GripperCalibrationOffset,
240-
) -> Gripper:
240+
) -> Tuple[Gripper, bool]:
241241
# Once we have determined that the new and attached grippers
242242
# are similar enough that we might skip, see if the configs
243243
# match closely enough.
@@ -247,7 +247,7 @@ def _reload_gripper(
247247
and cal_offset == attached_instr._calibration_offset
248248
):
249249
# Same config, good enough
250-
return attached_instr
250+
return attached_instr, True
251251
else:
252252
newdict = new_config.dict()
253253
olddict = attached_instr.config.dict()
@@ -257,22 +257,25 @@ def _reload_gripper(
257257
changed.add(k)
258258
if changed.intersection(RECONFIG_KEYS):
259259
# Something has changed that requires reconfig
260-
return Gripper(
261-
new_config,
262-
cal_offset,
263-
attached_instr._gripper_id,
260+
return (
261+
Gripper(
262+
new_config,
263+
cal_offset,
264+
attached_instr._gripper_id,
265+
),
266+
False,
264267
)
265268
else:
266269
# update just the cal offset and update info
267270
attached_instr._calibration_offset = cal_offset
268-
return attached_instr
271+
return attached_instr, True
269272

270273

271274
def compare_gripper_config_and_check_skip(
272275
freshly_detected: AttachedGripper,
273276
attached: Optional[Gripper],
274277
cal_offset: GripperCalibrationOffset,
275-
) -> Optional[Gripper]:
278+
) -> Tuple[Optional[Gripper], bool]:
276279
"""
277280
Given the gripper config for an attached gripper (if any) freshly read
278281
from disk, and any attached instruments,
@@ -288,7 +291,7 @@ def compare_gripper_config_and_check_skip(
288291
if not config and not attached:
289292
# nothing attached now, nothing used to be attached, nothing
290293
# to reconfigure
291-
return attached
294+
return attached, True
292295

293296
if config and attached:
294297
# something was attached and something is attached. are they
@@ -298,6 +301,6 @@ def compare_gripper_config_and_check_skip(
298301
return _reload_gripper(config, attached, cal_offset)
299302

300303
if config:
301-
return Gripper(config, cal_offset, serial)
304+
return Gripper(config, cal_offset, serial), False
302305
else:
303-
return None
306+
return None, False

0 commit comments

Comments
 (0)