Files
lerobot-clone/lerobot/scripts/calibration_visualization_so100.py

106 lines
3.5 KiB
Python
Raw Normal View History

"""
usage:
```python
python lerobot/scripts/calibration_visualization_so100.py \
--teleop.type=so100 \
--teleop.port=/dev/tty.usbmodem58760430541
python lerobot/scripts/calibration_visualization_so100.py \
--robot.type=so100 \
--robot.port=/dev/tty.usbmodem585A0084711
```
"""
import time
from dataclasses import dataclass
import draccus
2025-03-20 14:33:36 +01:00
from lerobot.common.motors import MotorsBus
from lerobot.common.motors.feetech.feetech_calibration import (
adjusted_to_homing_ticks,
adjusted_to_motor_ticks,
convert_degrees_to_ticks,
convert_ticks_to_degrees,
)
from lerobot.common.robots import RobotConfig
from lerobot.common.robots.so100 import SO100Robot, SO100RobotConfig # noqa: F401
from lerobot.common.teleoperators import TeleoperatorConfig
from lerobot.common.teleoperators.so100 import SO100Teleop, SO100TeleopConfig # noqa: F401
@dataclass
2025-03-20 14:33:36 +01:00
class DebugConfig:
teleop: TeleoperatorConfig | None = None
robot: RobotConfig | None = None
def __post_init__(self):
if bool(self.teleop) == bool(self.robot):
raise ValueError("Select a single device.")
@draccus.wrap()
2025-03-20 14:33:36 +01:00
def debug_device_calibration(cfg: DebugConfig):
# TODO(aliberts): make device and automatically get its motors_bus
device = SO100Teleop(cfg.teleop) if cfg.teleop else SO100Robot(cfg.robot)
visualize_motors_bus(device.arm)
def visualize_motors_bus(motor_bus: MotorsBus):
"""
Reads each joint's (1) raw ticks, (2) homed ticks, (3) degrees, and (4) invert-adjusted ticks.
"""
2025-03-20 14:33:36 +01:00
motor_bus.connect()
# Disable torque on all motors so you can move them freely by hand
2025-03-20 14:33:36 +01:00
# values_dict = {idx: 0 for idx in motors_bus.motor_ids}
# motor_bus.write("Torque_Enable", values_dict)
# print("Torque disabled on all joints.")
try:
print("\nPress Ctrl+C to quit.\n")
while True:
# Read *raw* positions (no calibration).
2025-03-20 14:33:36 +01:00
raw_positions = motor_bus.read("Present_Position")
2025-03-20 14:33:36 +01:00
# # Read *already-homed* positions
# homed_positions = motor_bus.read("Present_Position")
2025-03-20 14:33:36 +01:00
for name, raw_ticks in raw_positions.items():
idx = motor_bus.motors[name][0]
model = motor_bus.motors[name][1]
2025-03-20 14:33:36 +01:00
# homed_val = homed_positions[i] # degrees or % if linear
# Manually compute "adjusted ticks" from raw ticks
2025-03-20 14:33:36 +01:00
manual_adjusted = adjusted_to_homing_ticks(raw_ticks, model, motor_bus, idx)
# Convert to degrees
manual_degs = convert_ticks_to_degrees(manual_adjusted, model)
# Convert that deg back to ticks
manual_ticks = convert_degrees_to_ticks(manual_degs, model)
# Then invert them using offset & bus drive mode
2025-03-20 14:33:36 +01:00
inv_ticks = adjusted_to_motor_ticks(manual_ticks, model, motor_bus, idx)
print(
f"{name:15s} | "
f"RAW={raw_ticks:4d} | "
2025-03-20 14:33:36 +01:00
# f"HOMED_FROM_READ={homed_val:7.2f} | "
f"HOMED_TICKS={manual_adjusted:6d} | "
f"MANUAL_ADJ_DEG={manual_degs:7.2f} | "
f"MANUAL_ADJ_TICKS={manual_ticks:6d} | "
f"INV_TICKS={inv_ticks:4d} "
)
print("----------------------------------------------------")
time.sleep(0.25)
except KeyboardInterrupt:
pass
finally:
print("\nExiting. Disconnecting bus...")
2025-03-20 14:33:36 +01:00
motor_bus.disconnect()
if __name__ == "__main__":
2025-03-20 14:33:36 +01:00
debug_device_calibration()