Skip to content

Commit

Permalink
Merge pull request #299 from zivid/MISC-2024-12-03-hand-eye-low-dof
Browse files Browse the repository at this point in the history
Add hand-eye calibration for low degrees-of-freedom (DOF) robots
  • Loading branch information
johningve authored Dec 6, 2024
2 parents 3dfa872 + dd8035f commit f11e9b6
Show file tree
Hide file tree
Showing 15 changed files with 699 additions and 30 deletions.
269 changes: 269 additions & 0 deletions modules/zivid/experimental/hand_eye_low_dof.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,269 @@
"""Experimental implementation of hand-eye calibration for robots with low degrees-of-freedom.
This API may change in the future.
"""

import collections.abc
import _zivid
from zivid.calibration import Pose, HandEyeOutput, MarkerDictionary


class FixedPlacementOfFiducialMarker:
"""Specifies the fixed placement of a fiducial marker for low degrees-of-freedom hand-eye calibration."""

def __init__(self, marker_id, position):
"""Construct a FixedPlacementOfFiducialMarker.
For eye-in-hand calibration, positions should be given in the robot's base frame. For eye-to-hand calibration,
positions should be given in the robot's end-effector frame.
Note: the units of the input robot poses must be consistent with the units of the point clouds used to create
the detection result. Zivid point clouds are, by default, in millimeters.
Args:
marker_id: The ID of the fiducial marker to specify a position for.
position: The position of the fiducial marker as a three-element list, specified at the center of the
marker.
Raises:
TypeError: If one of the input arguments is of the wrong type.
"""
if not isinstance(marker_id, int):
raise TypeError(
"Unsupported type for argument marker_id. Expected int but got {}".format(
type(marker_id)
)
)

if not isinstance(
position, (collections.abc.Iterable, _zivid.data_model.PointXYZ)
):
raise TypeError(
"Unsupported type for argument position. Expected: (collections.abc.Iterable, _zivid.data_model.PointXYZ), got {value_type}".format(
value_type=type(position)
)
)

self.__impl = _zivid.calibration.FixedPlacementOfFiducialMarker(
marker_id, # pylint: disable=protected-access
_zivid.data_model.PointXYZ(position), # pylint: disable=protected-access
)

@property
def id(self):
"""Get ID of fiducial marker.
Returns:
An integer representing the ID
"""
return self.__impl.id

@property
def position(self):
"""Get position of fiducial marker.
Returns:
A three-element list of floats
"""
return self.__impl.position.to_array()

def __str__(self):
return str(self.__impl)


class FixedPlacementOfFiducialMarkers: # pylint: disable=too-few-public-methods
"""Specifies the fixed placement of a list of fiducial markers for low degrees-of-freedom hand-eye calibration."""

def __init__(self, marker_dictionary, markers):
"""Construct a FixedPlacementOfFiducialMarkers instance.
Args:
marker_dictionary: The dictionary that describes the appearance of the given markers. The name must be one
of the values returned by MarkerDictionary.valid_values()
markers: A list of FixedPlacementOfFiducialMarker describing the fixed placement of fiducial markers.
Raises:
ValueError: If marker_dictionary is not one of the valid values returned by MarkerDictionary.valid_values()
TypeError: If one of the input arguments are of the wrong type
"""

if marker_dictionary not in MarkerDictionary.valid_values():
raise ValueError(
"Invalid marker dictionary '{}'. Valid values are {}".format(
marker_dictionary, MarkerDictionary.valid_values()
)
)

dictionary = (
MarkerDictionary._valid_values.get( # pylint: disable=protected-access
marker_dictionary
)
)

if not (
isinstance(markers, list)
and all(
isinstance(marker, FixedPlacementOfFiducialMarker) for marker in markers
)
):
raise TypeError(
"Unsupported type for argument position. Expected list of FixedPlacementOfFiducialMarker but got {}".format(
type(markers)
)
)

self.__impl = _zivid.calibration.FixedPlacementOfFiducialMarkers(
dictionary, # pylint: disable=protected-access
[
marker._FixedPlacementOfFiducialMarker__impl # pylint: disable=protected-access
for marker in markers
],
)

def __str__(self):
return str(self.__impl)


class FixedPlacementOfCalibrationBoard: # pylint: disable=too-few-public-methods
"""Specifies the fixed placement of a Zivid calibration board for low degrees-of-freedom hand-eye calibration."""

def __init__(self, position_or_pose):
"""Construct a FixedPlacementOfCalibrationBoard instance.
For eye-in-hand calibration, the position or pose should be given in the robot's base frame. For eye-to-hand
calibration, the position or pose should be given in the robot's end-effector frame.
The origin is the top left inner corner of the calibration board. Using a pose instead of a position can improve
accuracy of the hand-eye calibration in some situations.
Note: the units of the input robot poses must be consistent with the units of the point clouds used to create
the detection result. Zivid point clouds are, by default, in millimeters.
Args:
position_or_pose: A position specifying the origin of the calibration board as a three-element list, or the
pose of the calibration board specified using the Pose type.
Raises:
TypeError: If the input argument is of the wrong type.
"""
if isinstance(position_or_pose, Pose):
self.__impl = _zivid.calibration.FixedPlacementOfCalibrationBoard(
position_or_pose._Pose__impl # pylint: disable=protected-access
)
elif isinstance(
position_or_pose, (collections.abc.Iterable, _zivid.data_model.PointXYZ)
):
self.__impl = _zivid.calibration.FixedPlacementOfCalibrationBoard(
_zivid.data_model.PointXYZ(
position_or_pose
), # pylint: disable=protected-access
)
else:
raise TypeError(
"Unsupported type for argument id. Expected zivid.calibration.Pose "
"or a three-element list, but got {}".format(type(position_or_pose))
)

def __str__(self):
return str(self.__impl)


class FixedPlacementOfCalibrationObjects: # pylint: disable=too-few-public-methods
"""Specifies the fixed placement of calibration objects for low degrees-of-freedom hand-eye calibration."""

def __init__(self, fixed_objects):
"""Construct a FixedPlacementOfCalibrationObjects instance from fiducial markers or a calibration board.
Args:
fixed_objects: An instance of FixedPlacementOfFiducialMarkers or FixedPlacementOfCalibrationBoard.
Raises:
TypeError: If the input argument is of the wrong type.
"""
if isinstance(fixed_objects, FixedPlacementOfFiducialMarkers):
self.__impl = _zivid.calibration.FixedPlacementOfCalibrationObjects(
fixed_objects._FixedPlacementOfFiducialMarkers__impl # pylint: disable=protected-access
)
elif isinstance(fixed_objects, FixedPlacementOfCalibrationBoard):
self.__impl = _zivid.calibration.FixedPlacementOfCalibrationObjects(
fixed_objects._FixedPlacementOfCalibrationBoard__impl # pylint: disable=protected-access
)
else:
raise TypeError(
"Unsupported type for argument fixed_objects. Got {}, expected {} or {}".format(
type(fixed_objects),
FixedPlacementOfFiducialMarkers,
FixedPlacementOfCalibrationBoard,
)
)

def __str__(self):
return str(self.__impl)


def calibrate_eye_in_hand_low_dof(calibration_inputs, fixed_objects):
"""Perform eye-in-hand calibration for low degrees-of-freedom robots.
For robots with low degrees-of-freedom (DOF), that is, less than 6 DOF, the robot pose and capture inputs are not
alone sufficient to uniquely identify the solution to the hand-eye calibration. This procedure additionally takes
knowledge about the fixed placement of the calibration objects in the scene to provide a unique solution. For 6 DOF
robots, consider using the `calibrate_eye_in_hand` function instead.
The procedure requires all robot poses to be different. At least 2 poses are required when using a calibration
board, or 6 poses when using fiducial markers. For fiducial markers, each marker must be detected across 2 poses at
minimum. An exception will be thrown if the preceding requirements are not fulfilled.
Note: the units of the input robot poses must be consistent with the units of the point clouds used to create the
detection results. Zivid point clouds are, by default, in millimeters.
Args:
calibration_inputs: List of HandEyeInput
fixed_objects: Specifies the fixed placement of calibration objects in the robot's base frame, using an instance
of FixedPlacementOfCalibrationObjects.
Returns:
A HandEyeOutput instance containing the eye-in-hand transform (camera pose in robot end-effector frame)
"""
return HandEyeOutput(
_zivid.calibration.calibrate_eye_in_hand_low_dof(
[
calibration_input._HandEyeInput__impl # pylint: disable=protected-access
for calibration_input in calibration_inputs
],
fixed_objects._FixedPlacementOfCalibrationObjects__impl, # pylint: disable=protected-access
)
)


def calibrate_eye_to_hand_low_dof(calibration_inputs, fixed_objects):
"""Perform eye-to-hand calibration for low degrees-of-freedom robots.
For robots with low degrees-of-freedom (DOF), that is, less than 6 DOF, the robot pose and capture inputs are not
alone sufficient to uniquely identify the solution to the hand-eye calibration. This procedure additionally takes
knowledge about the fixed placement of the calibration objects in the scene to provide a unique solution. For 6 DOF
robots, consider using the `calibrate_eye_to_hand` function instead.
The procedure requires all robot poses to be different. At least 2 poses are required when using a calibration
board, or 6 poses when using fiducial markers. For fiducial markers, each marker must be detected across 2 poses at
minimum. An exception will be thrown if the preceding requirements are not fulfilled.
Note: the units of the input robot poses must be consistent with the units of the point clouds used to create the
detection results. Zivid point clouds are, by default, in millimeters.
Args:
calibration_inputs: List of HandEyeInput
fixed_objects: Specifies the fixed placement of calibration objects in the robot's end-effector frame, using an
instance of FixedPlacementOfCalibrationObjects.
Returns:
A HandEyeOutput instance containing the eye-to-hand transform (camera pose in robot base frame)
"""
return HandEyeOutput(
_zivid.calibration.calibrate_eye_to_hand_low_dof(
[
calibration_input._HandEyeInput__impl # pylint: disable=protected-access
for calibration_input in calibration_inputs
],
fixed_objects._FixedPlacementOfCalibrationObjects__impl, # pylint: disable=protected-access
)
)
23 changes: 22 additions & 1 deletion src/Calibration/Calibration.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
#include <Zivid/Calibration/MultiCamera.h>
#include <Zivid/Calibration/Pose.h>
#include <Zivid/Experimental/Calibration.h>
#include <Zivid/Experimental/Calibration/HandEyeLowDOF.h>
#include <Zivid/PointCloud.h>

#include <ZividPython/Calibration/Detector.h>
Expand All @@ -25,6 +26,7 @@ namespace ZividPython::Calibration
void wrapAsSubmodule(py::module &dest)
{
using namespace Zivid::Calibration;
using namespace Zivid::Experimental::Calibration::HandEyeLowDOF;

ZIVID_PYTHON_WRAP_CLASS(dest, Pose);
ZIVID_PYTHON_WRAP_CLASS(dest, HandEyeOutput);
Expand All @@ -35,6 +37,11 @@ namespace ZividPython::Calibration
ZIVID_PYTHON_WRAP_CLASS(dest, DetectionResultFiducialMarkers);
ZIVID_PYTHON_WRAP_CLASS(dest, HandEyeResidual);

ZIVID_PYTHON_WRAP_CLASS(dest, FixedPlacementOfFiducialMarker);
ZIVID_PYTHON_WRAP_CLASS(dest, FixedPlacementOfFiducialMarkers);
ZIVID_PYTHON_WRAP_CLASS(dest, FixedPlacementOfCalibrationBoard);
ZIVID_PYTHON_WRAP_CLASS(dest, FixedPlacementOfCalibrationObjects);

ZIVID_PYTHON_WRAP_CLASS(dest, MultiCameraResidual);
ZIVID_PYTHON_WRAP_CLASS(dest, MultiCameraOutput);

Expand Down Expand Up @@ -62,6 +69,20 @@ namespace ZividPython::Calibration
})
.def("calibrate_eye_in_hand", &Zivid::Calibration::calibrateEyeInHand)
.def("calibrate_eye_to_hand", &Zivid::Calibration::calibrateEyeToHand)
.def(
"calibrate_eye_in_hand_low_dof",
[](const std::vector<HandEyeInput> &inputs, const FixedPlacementOfCalibrationObjects &fixedObjects) {
return Zivid::Experimental::Calibration::calibrateEyeInHandLowDOF(inputs, fixedObjects);
},
py::arg("inputs"),
py::arg("fixed_objects"))
.def(
"calibrate_eye_to_hand_low_dof",
[](const std::vector<HandEyeInput> &inputs, const FixedPlacementOfCalibrationObjects &fixedObjects) {
return Zivid::Experimental::Calibration::calibrateEyeToHandLowDOF(inputs, fixedObjects);
},
py::arg("inputs"),
py::arg("fixed_objects"))
.def("calibrate_multi_camera", &Zivid::Calibration::calibrateMultiCamera)
.def(
"intrinsics",
Expand Down Expand Up @@ -95,4 +116,4 @@ namespace ZividPython::Calibration
py::arg("camera"),
py::arg("settings"));
}
} // namespace ZividPython::Calibration
} // namespace ZividPython::Calibration
34 changes: 34 additions & 0 deletions src/Calibration/HandEye.cpp
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
#include <Zivid/Calibration/HandEye.h>

#include <ZividPython/Calibration/Detector.h>
#include <ZividPython/Calibration/HandEye.h>
#include <ZividPython/Matrix.h>

Expand Down Expand Up @@ -36,4 +37,37 @@ namespace ZividPython
})
.def("detection_result", &Zivid::Calibration::HandEyeInput::detectionResult);
}

void wrapClass(
pybind11::class_<Zivid::Experimental::Calibration::HandEyeLowDOF::FixedPlacementOfFiducialMarker> pyClass)
{
using T = Zivid::Experimental::Calibration::HandEyeLowDOF::FixedPlacementOfFiducialMarker;
pyClass.def(py::init<int, const Zivid::PointXYZ &>())
.def_property_readonly("id", &T::id)
.def_property_readonly("position", &T::position);
}

void wrapClass(
pybind11::class_<Zivid::Experimental::Calibration::HandEyeLowDOF::FixedPlacementOfFiducialMarkers> pyClass)
{
using T = Zivid::Experimental::Calibration::HandEyeLowDOF::FixedPlacementOfFiducialMarkers;
pyClass.def(
py::init<const Zivid::Calibration::MarkerDictionary &,
const std::vector<Zivid::Experimental::Calibration::HandEyeLowDOF::FixedPlacementOfFiducialMarker>
&>());
}

void wrapClass(
pybind11::class_<Zivid::Experimental::Calibration::HandEyeLowDOF::FixedPlacementOfCalibrationBoard> pyClass)
{
pyClass.def(py::init<const Zivid::PointXYZ &>()).def(py::init<const Zivid::Calibration::Pose &>());
}

void wrapClass(
pybind11::class_<Zivid::Experimental::Calibration::HandEyeLowDOF::FixedPlacementOfCalibrationObjects> pyClass)
{
pyClass
.def(py::init<const Zivid::Experimental::Calibration::HandEyeLowDOF::FixedPlacementOfFiducialMarkers &>())
.def(py::init<const Zivid::Experimental::Calibration::HandEyeLowDOF::FixedPlacementOfCalibrationBoard &>());
}
} // namespace ZividPython
2 changes: 0 additions & 2 deletions src/PixelMapping.cpp
Original file line number Diff line number Diff line change
@@ -1,5 +1,3 @@
#pragma once

#include <Zivid/Experimental/PixelMapping.h>

#include <pybind11/pybind11.h>
Expand Down
9 changes: 9 additions & 0 deletions src/include/ZividPython/Calibration/HandEye.h
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
#pragma once

#include <Zivid/Calibration/HandEye.h>
#include <Zivid/Experimental/Calibration/HandEyeLowDOF.h>

#include <pybind11/pybind11.h>

Expand All @@ -9,4 +10,12 @@ namespace ZividPython
void wrapClass(pybind11::class_<Zivid::Calibration::HandEyeResidual> pyClass);
void wrapClass(pybind11::class_<Zivid::Calibration::HandEyeOutput> pyClass);
void wrapClass(pybind11::class_<Zivid::Calibration::HandEyeInput> pyClass);
void wrapClass(
pybind11::class_<Zivid::Experimental::Calibration::HandEyeLowDOF::FixedPlacementOfFiducialMarker> pyClass);
void wrapClass(
pybind11::class_<Zivid::Experimental::Calibration::HandEyeLowDOF::FixedPlacementOfFiducialMarkers> pyClass);
void wrapClass(
pybind11::class_<Zivid::Experimental::Calibration::HandEyeLowDOF::FixedPlacementOfCalibrationBoard> pyClass);
void wrapClass(
pybind11::class_<Zivid::Experimental::Calibration::HandEyeLowDOF::FixedPlacementOfCalibrationObjects> pyClass);
} // namespace ZividPython
Loading

0 comments on commit f11e9b6

Please sign in to comment.