us.ihmc.simulationconstructionset.LoopClosureSoftConstraint Maven / Gradle / Ivy
package us.ihmc.simulationconstructionset;
import us.ihmc.euclid.matrix.Matrix3D;
import us.ihmc.euclid.matrix.interfaces.Matrix3DReadOnly;
import us.ihmc.euclid.tools.TupleTools;
import us.ihmc.euclid.tuple3D.Vector3D;
import us.ihmc.euclid.tuple3D.interfaces.Tuple3DReadOnly;
import us.ihmc.euclid.tuple3D.interfaces.Vector3DReadOnly;
import us.ihmc.euclid.tuple4D.Quaternion;
import us.ihmc.log.LogTools;
import us.ihmc.yoVariables.euclid.YoVector3D;
import us.ihmc.yoVariables.registry.YoRegistry;
import us.ihmc.yoVariables.variable.YoDouble;
/**
* This class implements a constraint for enforcing the closure of a kinematic loop.
*
* This implementation uses a simple PD-control scheme to correct for violation of the loop closure.
*
*
* @author Sylvain Bertrand
*/
public class LoopClosureSoftConstraint
{
private String name;
private final YoVector3D proportionalGains;
private final YoVector3D derivativeGains;
private final ExternalForcePoint constraintA;
private final ExternalForcePoint constraintB;
private final YoDouble positionErrorMagnitude;
private final YoDouble rotationErrorMagnitude;
private final YoVector3D positionError;
private final YoVector3D rotationError;
private final YoVector3D linearVelocityError;
private final YoVector3D angularVelocityError;
private final YoVector3D feedForwardForce;
private final YoVector3D feedForwardMoment;
private final Vector3D offsetFromParentJoint = new Vector3D();
private final Vector3D offsetFromLinkParentJoint = new Vector3D();
private final Matrix3D constraintForceSubSpace = new Matrix3D();
private final Matrix3D constraintMomentSubSpace = new Matrix3D();
private final Vector3D proportionalTermLinear = new Vector3D();
private final Vector3D derivativeTermLinear = new Vector3D();
private final Vector3D proportionalTermAngular = new Vector3D();
private final Vector3D derivativeTermAngular = new Vector3D();
private Joint parentJoint;
private final Quaternion quaternionA = new Quaternion();
private final Quaternion quaternionB = new Quaternion();
private final Quaternion quaternionDifference = new Quaternion();
private final Vector3D forceA = new Vector3D();
private final Vector3D momentA = new Vector3D();
/**
* Creates a new constraint for closing a kinematic loop.
*
* This constraint is general and the sub-space in which it operates has to be provided. Here's a
* couple examples:
*
* - For a constraint that only allows rotation around the z-axis, the matrices defining the
* sub-space should be as follows:
*
*
* / 1 0 0 \
* constraintForceSubSpace = | 0 1 0 |
* \ 0 0 1 /
* / 1 0 0 \
* constraintMomentSubSpace = | 0 1 0 |
* \ 0 0 0 /
*
*
* Note that by having constraintForceSubSpace be identity, the entire linear space is
* constrained, while by having the last row of constraintMomentSubSpace be only zeros, the
* z-axis is not constrained.
* - For a constraint that only allows translation along the y-axis, the matrices defining the
* sub-space should be as follows:
*
*
* / 1 0 0 \
* constraintForceSubSpace = | 0 0 0 |
* \ 0 0 1 /
* / 1 0 0 \
* constraintMomentSubSpace = | 0 1 0 |
* \ 0 0 1 /
*
*
* In other words, these matrices can be seen as selection matrices used for selecting which forces
* and moments are to be applied.
*
*
* @param name the name of the constraint, {@code YoVariable}s will be created
* using this name.
* @param offsetFromParentJoint the position of the constraint with respect to the parent joint.
* @param offsetFromLinkParentJoint the position of the constraint with respect to the parent joint
* of the associated link. Note that the link's parent joint is
* expected to be different from this constraint's parent joint.
* @param robot the robot is used for getting access to its
* {@code YoRegistry}.
* @param constraintForceSubSpace defines the linear part of the sub-space in which the constraint
* is to be enforced.
* @param constraintMomentSubSpace defines the angular part of the sub-space in which the
* constraint is to be enforced.
*/
public LoopClosureSoftConstraint(String name, Tuple3DReadOnly offsetFromParentJoint, Tuple3DReadOnly offsetFromLinkParentJoint, Robot robot,
Matrix3DReadOnly constraintForceSubSpace, Matrix3DReadOnly constraintMomentSubSpace)
{
this.name = name;
this.offsetFromParentJoint.set(offsetFromParentJoint);
this.offsetFromLinkParentJoint.set(offsetFromLinkParentJoint);
this.constraintForceSubSpace.set(constraintForceSubSpace);
this.constraintMomentSubSpace.set(constraintMomentSubSpace);
YoRegistry registry = robot.getRobotsYoRegistry();
proportionalGains = new YoVector3D(name + "ProportionalGain", registry);
derivativeGains = new YoVector3D(name + "DerivativeGain", registry);
constraintA = new ExternalForcePoint(name + "A", offsetFromParentJoint, robot);
constraintB = new ExternalForcePoint(name + "B", offsetFromLinkParentJoint, robot);
positionErrorMagnitude = new YoDouble(name + "PositionErrorMagnitude", registry);
rotationErrorMagnitude = new YoDouble(name + "RotationErrorMagnitude", registry);
positionError = new YoVector3D(name + "PositionError", registry);
rotationError = new YoVector3D(name + "RotationError", registry);
linearVelocityError = new YoVector3D(name + "LinearVelocityError", registry);
angularVelocityError = new YoVector3D(name + "AngularVelocityError", registry);
feedForwardForce = new YoVector3D(name + "FeedForwardForce", registry);
feedForwardMoment = new YoVector3D(name + "FeedForwardMoment", registry);
}
public void setGains(double proportionalGain, double derivativeGain)
{
proportionalGains.set(proportionalGain, proportionalGain, proportionalGain);
derivativeGains.set(derivativeGain, derivativeGain, derivativeGain);
}
/**
* Sets the gains to use for enforcing this constraint.
*
* Note that the gains are applied on the error in the local coordinates of the parent joints.
*
*
* @param proportionalGains the gains to apply on the position and rotation errors.
* @param derivativeGains the gains to apply on the linear and angular velocity errors.
*/
public void setGains(Tuple3DReadOnly proportionalGains, Tuple3DReadOnly derivativeGains)
{
this.proportionalGains.set(proportionalGains);
this.derivativeGains.set(derivativeGains);
}
public void setParentJoint(Joint parentJoint)
{
this.parentJoint = parentJoint;
parentJoint.addExternalForcePoint(constraintA);
}
public void setLink(Link link)
{
link.getParentJoint().addExternalForcePoint(constraintB);
}
private boolean isFirstUpdate = true;
public void update()
{
if (isFirstUpdate)
{
if (proportionalGains.containsNaN() || derivativeGains.containsNaN())
throw new IllegalArgumentException("The gains for the loop closure constraint: " + name
+ " have not been configured. If created from description, see: " + LoopClosureSoftConstraint.class.getSimpleName());
if (TupleTools.isTupleZero(proportionalGains, 0.0) || TupleTools.isTupleZero(derivativeGains, 0.0))
LogTools.warn("The gains for the loop closure constraint: " + name + " have not been configured. If created from description, see: "
+ LoopClosureSoftConstraint.class.getSimpleName());
isFirstUpdate = false;
}
// Position error in world
positionError.sub(constraintB.getYoPosition(), constraintA.getYoPosition());
// Position error in A's local coordinates.
parentJoint.transformToNext.inverseTransform(positionError);
// Applying the sub-space on the error so the visualization is accurate, i.e. doesn't incorporate error that does not matter.
constraintForceSubSpace.transform(positionError);
positionErrorMagnitude.set(positionError.length());
constraintA.getParentJoint().getRotationToWorld(quaternionA);
constraintB.getParentJoint().getRotationToWorld(quaternionB);
// Rotation error in A's local coordinates.
quaternionDifference.difference(quaternionA, quaternionB); // This the orientation from B to A
quaternionDifference.normalizeAndLimitToPi();
quaternionDifference.getRotationVector(rotationError);
constraintMomentSubSpace.transform(rotationError);
rotationErrorMagnitude.set(rotationError.length());
// Linear velocity error in world
linearVelocityError.sub(constraintB.getYoVelocity(), constraintA.getYoVelocity());
// Linear velocity error in A's local coordinates.
parentJoint.transformToNext.inverseTransform(linearVelocityError);
constraintForceSubSpace.transform(linearVelocityError);
// Angular velocity in world
angularVelocityError.sub(constraintB.getYoAngularVelocity(), constraintA.getYoAngularVelocity());
// Angular velocity error in A's local coordinates.
parentJoint.transformToNext.inverseTransform(angularVelocityError);
constraintMomentSubSpace.transform(angularVelocityError);
proportionalTermLinear.set(positionError);
proportionalTermLinear.scale(proportionalGains.getX(), proportionalGains.getY(), proportionalGains.getZ());
proportionalTermAngular.set(rotationError);
proportionalTermAngular.scale(proportionalGains.getX(), proportionalGains.getY(), proportionalGains.getZ());
derivativeTermLinear.set(linearVelocityError);
derivativeTermLinear.scale(derivativeGains.getX(), derivativeGains.getY(), derivativeGains.getZ());
derivativeTermAngular.set(angularVelocityError);
derivativeTermAngular.scale(derivativeGains.getX(), derivativeGains.getY(), derivativeGains.getZ());
forceA.set(proportionalTermLinear);
forceA.add(derivativeTermLinear);
forceA.add(feedForwardForce);
momentA.set(proportionalTermAngular);
momentA.add(derivativeTermAngular);
momentA.add(feedForwardMoment);
// Need to switch back to world before applying these.
parentJoint.transformToNext.transform(forceA);
parentJoint.transformToNext.transform(momentA);
constraintA.getYoForce().set(forceA);
constraintA.getYoMoment().set(momentA);
constraintB.getYoForce().setAndNegate(forceA);
constraintB.getYoMoment().setAndNegate(momentA);
}
/**
* Set the feed-forward force and moment to be applied at the constraint in addition to the error
* terms.
*/
public void setFeedForward(Vector3DReadOnly force, Vector3DReadOnly moment)
{
constraintForceSubSpace.transform(force, feedForwardForce);
constraintMomentSubSpace.transform(moment, feedForwardMoment);
}
public String getName()
{
return name;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy