Solves the inverse kinematics problem as a mixed integer convex optimization problem.
We use a mixed-integer convex relaxation of the rotation matrix. So if this global inverse kinematics problem says the solution is infeasible, then it is guaranteed that the kinematics constraints are not satisfiable. If the global inverse kinematics returns a solution, the posture should approximately satisfy the kinematics constraints, with some error. The approach is described in Global Inverse Kinematics via Mixed-integer Convex Optimization by Hongkai Dai, Gregory Izatt and Russ Tedrake, International Journal of Robotics Research, 2019.
|
| GlobalInverseKinematics (const MultibodyPlant< double > &plant, const Options &options=Options()) |
| Parses the robot kinematics tree. More...
|
|
| ~GlobalInverseKinematics () |
|
const solvers::MathematicalProgram & | prog () const |
|
solvers::MathematicalProgram * | get_mutable_prog () |
|
const solvers::MatrixDecisionVariable< 3, 3 > & | body_rotation_matrix (BodyIndex body_index) const |
| Getter for the decision variables on the rotation matrix R_WB for a body with the specified index. More...
|
|
const solvers::VectorDecisionVariable< 3 > & | body_position (BodyIndex body_index) const |
| Getter for the decision variables on the position p_WBo of the body B's origin measured and expressed in the world frame. More...
|
|
Eigen::VectorXd | ReconstructGeneralizedPositionSolution (const solvers::MathematicalProgramResult &result) const |
| After solving the inverse kinematics problem and finding out the pose of each body, reconstruct the robot generalized position (joint angles, etc) that matches with the body poses. More...
|
|
solvers::Binding< solvers::LinearConstraint > | AddWorldPositionConstraint (BodyIndex body_index, const Eigen::Vector3d &p_BQ, const Eigen::Vector3d &box_lb_F, const Eigen::Vector3d &box_ub_F, const math::RigidTransformd &X_WF=math::RigidTransformd()) |
| Adds the constraint that the position of a point Q on a body B (whose index is body_index ), is within a box in a specified frame F . More...
|
|
solvers::Binding< solvers::LinearConstraint > | AddWorldRelativePositionConstraint (BodyIndex body_index_B, const Eigen::Vector3d &p_BQ, BodyIndex body_index_A, const Eigen::Vector3d &p_AP, const Eigen::Vector3d &box_lb_F, const Eigen::Vector3d &box_ub_F, const math::RigidTransformd &X_WF=math::RigidTransformd()) |
| Adds the constraint that the position of a point Q on a body B relative to a point P on body A , is within a box in a specified frame F . More...
|
|
solvers::Binding< solvers::LinearConstraint > | AddWorldOrientationConstraint (BodyIndex body_index, const Eigen::Quaterniond &desired_orientation, double angle_tol) |
| Adds a constraint that the angle between the body orientation and the desired orientation should not be larger than angle_tol . More...
|
|
void | AddPostureCost (const Eigen::Ref< const Eigen::VectorXd > &q_desired, const Eigen::Ref< const Eigen::VectorXd > &body_position_cost, const Eigen::Ref< const Eigen::VectorXd > &body_orientation_cost) |
| Penalizes the deviation to the desired posture. More...
|
|
solvers::VectorXDecisionVariable | BodyPointInOneOfRegions (BodyIndex body_index, const Eigen::Ref< const Eigen::Vector3d > &p_BQ, const std::vector< Eigen::Matrix3Xd > ®ion_vertices) |
| Constrain the point Q lying within one of the convex polytopes. More...
|
|
solvers::VectorXDecisionVariable | BodySphereInOneOfPolytopes (BodyIndex body_index, const Eigen::Ref< const Eigen::Vector3d > &p_BQ, double radius, const std::vector< Polytope3D > &polytopes) |
| Adds the constraint that a sphere rigidly attached to a body has to be within at least one of the given bounded polytopes. More...
|
|
void | AddJointLimitConstraint (BodyIndex body_index, double joint_lower_bound, double joint_upper_bound, bool linear_constraint_approximation=false) |
| Adds joint limits on a specified joint. More...
|
|
void | SetInitialGuess (const Eigen::Ref< const Eigen::VectorXd > &q) |
| Sets an initial guess for all variables (including the binary variables) by evaluating the kinematics of the plant at q . More...
|
|
|
| GlobalInverseKinematics (const GlobalInverseKinematics &)=delete |
|
GlobalInverseKinematics & | operator= (const GlobalInverseKinematics &)=delete |
|
| GlobalInverseKinematics (GlobalInverseKinematics &&)=delete |
|
GlobalInverseKinematics & | operator= (GlobalInverseKinematics &&)=delete |
|
void AddPostureCost |
( |
const Eigen::Ref< const Eigen::VectorXd > & |
q_desired, |
|
|
const Eigen::Ref< const Eigen::VectorXd > & |
body_position_cost, |
|
|
const Eigen::Ref< const Eigen::VectorXd > & |
body_orientation_cost |
|
) |
| |
Penalizes the deviation to the desired posture.
For each body (except the world) in the kinematic tree, we add the cost
∑ᵢ body_position_cost(i) * body_position_error(i) +
body_orientation_cost(i) * body_orientation_error(i)
where body_position_error(i)
is computed as the Euclidean distance error |p_WBo(i) - p_WBo_desired(i)| where
- p_WBo(i) : position of body i'th origin
Bo
in the world frame W
.
- p_WBo_desired(i): position of body i'th origin
Bo
in the world frame W
, computed from the desired posture q_desired
.
body_orientation_error(i) is computed as (1 - cos(θ)), where θ is the angle between the orientation of body i'th frame and body i'th frame using the desired posture. Notice that 1 - cos(θ) = θ²/2 + O(θ⁴), so this cost is on the square of θ, when θ is small. Notice that since body 0 is the world, the cost on that body is always 0, no matter what value body_position_cost(0)
and body_orientation_cost(0)
take.
- Parameters
-
q_desired | The desired posture. |
body_position_cost | The cost for each body's position error. Unit is [1/m] (one over meters). |
- Precondition
- body_position_cost.rows() == plant.num_bodies(), where
plant
is the input argument in the constructor of the class.
- body_position_cost(i) is non-negative.
- Exceptions
-
std::exception | if the precondition is not satisfied. |
- Parameters
-
body_orientation_cost | The cost for each body's orientation error. |
- Precondition
- body_orientation_cost.rows() == plant.num_bodies() , where
plant
is the input argument in the constructor of the class.
- body_position_cost(i) is non-negative.
- Exceptions
-
std::exception | if the precondition is not satisfied. |
Adds a constraint that the angle between the body orientation and the desired orientation should not be larger than angle_tol
.
If we denote the angle between two rotation matrices R1
and R2
as θ
, i.e. θ is the angle of the angle-axis representation of the rotation matrix R1ᵀ * R2
, we then know
trace(R1ᵀ * R2) = 2 * cos(θ) + 1
as in http://www.euclideanspace.com/maths/geometry/rotations/conversions/matrixToAngle/ To constraint θ < angle_tol
, we can impose the following constraint
2 * cos(angle_tol) + 1 <= trace(R1ᵀ * R2) <= 3
- Parameters
-
body_index | The index of the body whose orientation will be constrained. |
desired_orientation | The desired orientation of the body. |
angle_tol | The tolerance on the angle between the body orientation and the desired orientation. Unit is radians. |
- Return values
-
binding | The newly added constraint, together with the bound variables. |
solvers::Binding<solvers::LinearConstraint> AddWorldPositionConstraint |
( |
BodyIndex |
body_index, |
|
|
const Eigen::Vector3d & |
p_BQ, |
|
|
const Eigen::Vector3d & |
box_lb_F, |
|
|
const Eigen::Vector3d & |
box_ub_F, |
|
|
const math::RigidTransformd & |
X_WF = math::RigidTransformd() |
|
) |
| |
Adds the constraint that the position of a point Q
on a body B
(whose index is body_index
), is within a box in a specified frame F
.
The constraint is that the point Q
's position should lie within a bounding box in the frame F
. Namely
box_lb_F <= p_FQ <= box_ub_F
where p_FQ is the position of the point Q measured and expressed in the F
, computed as
p_FQ = X_FW * (p_WBo + R_WB * p_BQ)
hence this is a linear constraint on the decision variables p_WBo and R_WB. The inequality is imposed elementwise.
- Note
- since the rotation matrix
R_WB
does not lie exactly on the SO(3), due to the McCormick envelope relaxation, this constraint is subject to the accumulated error from the root of the kinematics tree.
- Parameters
-
body_index | The index of the body on which the position of a point is constrained. |
p_BQ | The position of the point Q measured and expressed in the body frame B. |
box_lb_F | The lower bound of the box in frame F . |
box_ub_F | The upper bound of the box in frame F . |
X_WF | The frame in which the box is specified. This frame is represented by a RigidTransform X_WF, the transform from the constraint frame F to the world frame W. Namely if the position of the point Q in the world frame is p_WQ , then the constraint is box_lb_F <= R_FW * (p_WQ-p_WFo) <= box_ub_F
where
- R_FW is the rotation matrix of frame
W expressed and measured in frame F . R_FW = X_WF.linear().transpose() .
- p_WFo is the position of frame
F 's origin, expressed and measured in frame W . p_WFo = X_WF.translation() .
Default: is the identity transform.
|
- Return values
-
binding | The newly added constraint, together with the bound variables. |
solvers::Binding<solvers::LinearConstraint> AddWorldRelativePositionConstraint |
( |
BodyIndex |
body_index_B, |
|
|
const Eigen::Vector3d & |
p_BQ, |
|
|
BodyIndex |
body_index_A, |
|
|
const Eigen::Vector3d & |
p_AP, |
|
|
const Eigen::Vector3d & |
box_lb_F, |
|
|
const Eigen::Vector3d & |
box_ub_F, |
|
|
const math::RigidTransformd & |
X_WF = math::RigidTransformd() |
|
) |
| |
Adds the constraint that the position of a point Q
on a body B
relative to a point P
on body A
, is within a box in a specified frame F
.
Using monogram notation we have:
box_lb_F <= p_FQ - p_FP <= box_ub_F
where p_FQ and p_FP are the position of the points measured and expressed in F
. The inequality is imposed elementwise. See AddWorldPositionConstraint for more details.
- Parameters
-
body_index_B | The index of the body B. |
p_BQ | The position of the point Q measured and expressed in the body frame B. |
body_index_A | The index of the body A. |
p_AP | The position of the point P measured and expressed in the body frame A. |
box_lb_F | The lower bound of the box in frame F . |
box_ub_F | The upper bound of the box in frame F . |
X_WF | Defines the frame in which the box is specified.
Default: is the identity transform. |
- Return values
-
binding | The newly added constraint, together with the bound variables. |
Constrain the point Q
lying within one of the convex polytopes.
Each convex polytope Pᵢ is represented by its vertices as Pᵢ = ConvexHull(v_i1, v_i2, ... v_in). Mathematically we want to impose the constraint that the p_WQ, i.e., the position of point Q
in world frame W
, satisfies
p_WQ ∈ Pᵢ for one i.
To impose this constraint, we consider to introduce binary variable zᵢ, and continuous variables w_i1, w_i2, ..., w_in for each vertex of Pᵢ, with the following constraints
p_WQ = sum_i (w_i1 * v_i1 + w_i2 * v_i2 + ... + w_in * v_in)
w_ij >= 0, ∀i,j
w_i1 + w_i2 + ... + w_in = zᵢ
sum_i zᵢ = 1
zᵢ ∈ {0, 1}
Notice that if zᵢ = 0, then w_i1 * v_i1 + w_i2 * v_i2 + ... + w_in * v_in is just 0. This function can be used for collision avoidance, where each region Pᵢ is a free space region. It can also be used for grasping, where each region Pᵢ is a surface patch on the grasped object. Note this approach also works if the region Pᵢ overlaps with each other.
- Parameters
-
body_index | The index of the body to on which point Q is attached. |
p_BQ | The position of point Q in the body frame B . |
region_vertices | region_vertices[i] is the vertices for the i'th region. |
- Return values
-
z | The newly added binary variables. If point Q is in the i'th region, z(i) = 1. |
- Precondition
- region_vertices[i] has at least 3 columns. Throw a std::runtime_error if the precondition is not satisfied.
Adds the constraint that a sphere rigidly attached to a body has to be within at least one of the given bounded polytopes.
If the polytopes don't intersect, then the sphere is in one and only one polytope. Otherwise the sphere is in at least one of the polytopes (could be in the intersection of multiple polytopes.) If the i'th polytope is described as
Aᵢ * x ≤ bᵢ
where Aᵢ ∈ ℝⁿ ˣ ³, bᵢ ∈ ℝⁿ. Then a sphere with center position p_WQ and radius r is within the i'th polytope, if
Aᵢ * p_WQ ≤ bᵢ - aᵢr
where aᵢ(j) = Aᵢ.row(j).norm() To constrain that the sphere is in one of the n polytopes, we introduce the binary variable z ∈{0, 1}ⁿ, together with continuous variables yᵢ ∈ ℝ³, i = 1, ..., n, such that p_WQ = y₁ + ... + yₙ Aᵢ * yᵢ ≤ (bᵢ - aᵢr)zᵢ z₁ + ... +zₙ = 1 Notice that when zᵢ = 0, Aᵢ * yᵢ ≤ 0 implies that yᵢ = 0. This is due to the boundedness of the polytope. If Aᵢ * yᵢ ≤ 0 has a non-zero solution y̅, that y̅ ≠ 0 and Aᵢ * y̅ ≤ 0. Then for any point x̂ in the polytope satisfying Aᵢ * x̂ ≤ bᵢ, we know the ray x̂ + ty̅, ∀ t ≥ 0 also satisfies Aᵢ * (x̂ + ty̅) ≤ bᵢ, thus the ray is within the polytope, violating the boundedness assumption.
- Parameters
-
body_index | The index of the body to which the sphere is attached. |
p_BQ | The position of the sphere center in the body frame B. |
radius | The radius of the sphere. |
polytopes | polytopes[i] = (Aᵢ, bᵢ). We assume that Aᵢx≤ bᵢ is a bounded polytope. It is the user's responsibility to guarantee the boundedness. |
- Return values
-
z | The newly added binary variables. If z(i) = 1, then the sphere is in the i'th polytope. If two or more polytopes are intersecting, and the sphere is in the intersection region, then it is up to the solver to choose one of z(i) to be 1. |
After solving the inverse kinematics problem and finding out the pose of each body, reconstruct the robot generalized position (joint angles, etc) that matches with the body poses.
Notice that since the rotation matrix is approximated, that the solution of body_rotmat() might not be on SO(3) exactly, the reconstructed body posture might not match with the body poses exactly, and the kinematics constraint might not be satisfied exactly with this reconstructed posture.
- Warning
- Do not call this method if the problem is not solved successfully! The returned value can be NaN or meaningless number if the problem is not solved.
- Return values
-
q | The reconstructed posture of the robot of the generalized coordinates, corresponding to the RigidBodyTree on which the inverse kinematics problem is solved. |