001// Copyright (c) FIRST and other WPILib contributors. 002// Open Source Software; you can modify and/or share it under the terms of 003// the WPILib BSD license file in the root directory of this project. 004 005package edu.wpi.first.math.kinematics; 006 007import edu.wpi.first.math.MathSharedStore; 008import edu.wpi.first.math.MathUsageId; 009import edu.wpi.first.math.geometry.Translation2d; 010import edu.wpi.first.math.geometry.Twist2d; 011import edu.wpi.first.math.kinematics.proto.MecanumDriveKinematicsProto; 012import edu.wpi.first.math.kinematics.struct.MecanumDriveKinematicsStruct; 013import edu.wpi.first.util.protobuf.ProtobufSerializable; 014import edu.wpi.first.util.struct.StructSerializable; 015import org.ejml.simple.SimpleMatrix; 016 017/** 018 * Helper class that converts a chassis velocity (dx, dy, and dtheta components) into individual 019 * wheel speeds. 020 * 021 * <p>The inverse kinematics (converting from a desired chassis velocity to individual wheel speeds) 022 * uses the relative locations of the wheels with respect to the center of rotation. The center of 023 * rotation for inverse kinematics is also variable. This means that you can set your center of 024 * rotation in a corner of the robot to perform special evasion maneuvers. 025 * 026 * <p>Forward kinematics (converting an array of wheel speeds into the overall chassis motion) is 027 * performs the exact opposite of what inverse kinematics does. Since this is an overdetermined 028 * system (more equations than variables), we use a least-squares approximation. 029 * 030 * <p>The inverse kinematics: [wheelSpeeds] = [wheelLocations] * [chassisSpeeds] We take the 031 * Moore-Penrose pseudoinverse of [wheelLocations] and then multiply by [wheelSpeeds] to get our 032 * chassis speeds. 033 * 034 * <p>Forward kinematics is also used for odometry -- determining the position of the robot on the 035 * field using encoders and a gyro. 036 */ 037public class MecanumDriveKinematics 038 implements Kinematics<MecanumDriveWheelSpeeds, MecanumDriveWheelPositions>, 039 ProtobufSerializable, 040 StructSerializable { 041 private final SimpleMatrix m_inverseKinematics; 042 private final SimpleMatrix m_forwardKinematics; 043 044 private final Translation2d m_frontLeftWheelMeters; 045 private final Translation2d m_frontRightWheelMeters; 046 private final Translation2d m_rearLeftWheelMeters; 047 private final Translation2d m_rearRightWheelMeters; 048 049 private Translation2d m_prevCoR = Translation2d.kZero; 050 051 /** MecanumDriveKinematics protobuf for serialization. */ 052 public static final MecanumDriveKinematicsProto proto = new MecanumDriveKinematicsProto(); 053 054 /** MecanumDriveKinematics struct for serialization. */ 055 public static final MecanumDriveKinematicsStruct struct = new MecanumDriveKinematicsStruct(); 056 057 /** 058 * Constructs a mecanum drive kinematics object. 059 * 060 * @param frontLeftWheelMeters The location of the front-left wheel relative to the physical 061 * center of the robot. 062 * @param frontRightWheelMeters The location of the front-right wheel relative to the physical 063 * center of the robot. 064 * @param rearLeftWheelMeters The location of the rear-left wheel relative to the physical center 065 * of the robot. 066 * @param rearRightWheelMeters The location of the rear-right wheel relative to the physical 067 * center of the robot. 068 */ 069 public MecanumDriveKinematics( 070 Translation2d frontLeftWheelMeters, 071 Translation2d frontRightWheelMeters, 072 Translation2d rearLeftWheelMeters, 073 Translation2d rearRightWheelMeters) { 074 m_frontLeftWheelMeters = frontLeftWheelMeters; 075 m_frontRightWheelMeters = frontRightWheelMeters; 076 m_rearLeftWheelMeters = rearLeftWheelMeters; 077 m_rearRightWheelMeters = rearRightWheelMeters; 078 079 m_inverseKinematics = new SimpleMatrix(4, 3); 080 081 setInverseKinematics( 082 frontLeftWheelMeters, frontRightWheelMeters, rearLeftWheelMeters, rearRightWheelMeters); 083 m_forwardKinematics = m_inverseKinematics.pseudoInverse(); 084 085 MathSharedStore.reportUsage(MathUsageId.kKinematics_MecanumDrive, 1); 086 } 087 088 /** 089 * Performs inverse kinematics to return the wheel speeds from a desired chassis velocity. This 090 * method is often used to convert joystick values into wheel speeds. 091 * 092 * <p>This function also supports variable centers of rotation. During normal operations, the 093 * center of rotation is usually the same as the physical center of the robot; therefore, the 094 * argument is defaulted to that use case. However, if you wish to change the center of rotation 095 * for evasive maneuvers, vision alignment, or for any other use case, you can do so. 096 * 097 * @param chassisSpeeds The desired chassis speed. 098 * @param centerOfRotationMeters The center of rotation. For example, if you set the center of 099 * rotation at one corner of the robot and provide a chassis speed that only has a dtheta 100 * component, the robot will rotate around that corner. 101 * @return The wheel speeds. Use caution because they are not normalized. Sometimes, a user input 102 * may cause one of the wheel speeds to go above the attainable max velocity. Use the {@link 103 * MecanumDriveWheelSpeeds#desaturate(double)} function to rectify this issue. 104 */ 105 public MecanumDriveWheelSpeeds toWheelSpeeds( 106 ChassisSpeeds chassisSpeeds, Translation2d centerOfRotationMeters) { 107 // We have a new center of rotation. We need to compute the matrix again. 108 if (!centerOfRotationMeters.equals(m_prevCoR)) { 109 var fl = m_frontLeftWheelMeters.minus(centerOfRotationMeters); 110 var fr = m_frontRightWheelMeters.minus(centerOfRotationMeters); 111 var rl = m_rearLeftWheelMeters.minus(centerOfRotationMeters); 112 var rr = m_rearRightWheelMeters.minus(centerOfRotationMeters); 113 114 setInverseKinematics(fl, fr, rl, rr); 115 m_prevCoR = centerOfRotationMeters; 116 } 117 118 var chassisSpeedsVector = new SimpleMatrix(3, 1); 119 chassisSpeedsVector.setColumn( 120 0, 121 0, 122 chassisSpeeds.vxMetersPerSecond, 123 chassisSpeeds.vyMetersPerSecond, 124 chassisSpeeds.omegaRadiansPerSecond); 125 126 var wheelsVector = m_inverseKinematics.mult(chassisSpeedsVector); 127 return new MecanumDriveWheelSpeeds( 128 wheelsVector.get(0, 0), 129 wheelsVector.get(1, 0), 130 wheelsVector.get(2, 0), 131 wheelsVector.get(3, 0)); 132 } 133 134 /** 135 * Performs inverse kinematics. See {@link #toWheelSpeeds(ChassisSpeeds, Translation2d)} for more 136 * information. 137 * 138 * @param chassisSpeeds The desired chassis speed. 139 * @return The wheel speeds. 140 */ 141 @Override 142 public MecanumDriveWheelSpeeds toWheelSpeeds(ChassisSpeeds chassisSpeeds) { 143 return toWheelSpeeds(chassisSpeeds, Translation2d.kZero); 144 } 145 146 /** 147 * Performs forward kinematics to return the resulting chassis state from the given wheel speeds. 148 * This method is often used for odometry -- determining the robot's position on the field using 149 * data from the real-world speed of each wheel on the robot. 150 * 151 * @param wheelSpeeds The current mecanum drive wheel speeds. 152 * @return The resulting chassis speed. 153 */ 154 @Override 155 public ChassisSpeeds toChassisSpeeds(MecanumDriveWheelSpeeds wheelSpeeds) { 156 var wheelSpeedsVector = new SimpleMatrix(4, 1); 157 wheelSpeedsVector.setColumn( 158 0, 159 0, 160 wheelSpeeds.frontLeftMetersPerSecond, 161 wheelSpeeds.frontRightMetersPerSecond, 162 wheelSpeeds.rearLeftMetersPerSecond, 163 wheelSpeeds.rearRightMetersPerSecond); 164 var chassisSpeedsVector = m_forwardKinematics.mult(wheelSpeedsVector); 165 166 return new ChassisSpeeds( 167 chassisSpeedsVector.get(0, 0), 168 chassisSpeedsVector.get(1, 0), 169 chassisSpeedsVector.get(2, 0)); 170 } 171 172 @Override 173 public Twist2d toTwist2d(MecanumDriveWheelPositions start, MecanumDriveWheelPositions end) { 174 var wheelDeltasVector = new SimpleMatrix(4, 1); 175 wheelDeltasVector.setColumn( 176 0, 177 0, 178 end.frontLeftMeters - start.frontLeftMeters, 179 end.frontRightMeters - start.frontRightMeters, 180 end.rearLeftMeters - start.rearLeftMeters, 181 end.rearRightMeters - start.rearRightMeters); 182 var twist = m_forwardKinematics.mult(wheelDeltasVector); 183 return new Twist2d(twist.get(0, 0), twist.get(1, 0), twist.get(2, 0)); 184 } 185 186 /** 187 * Performs forward kinematics to return the resulting Twist2d from the given wheel deltas. This 188 * method is often used for odometry -- determining the robot's position on the field using 189 * changes in the distance driven by each wheel on the robot. 190 * 191 * @param wheelDeltas The distances driven by each wheel. 192 * @return The resulting Twist2d. 193 */ 194 public Twist2d toTwist2d(MecanumDriveWheelPositions wheelDeltas) { 195 var wheelDeltasVector = new SimpleMatrix(4, 1); 196 wheelDeltasVector.setColumn( 197 0, 198 0, 199 wheelDeltas.frontLeftMeters, 200 wheelDeltas.frontRightMeters, 201 wheelDeltas.rearLeftMeters, 202 wheelDeltas.rearRightMeters); 203 var twist = m_forwardKinematics.mult(wheelDeltasVector); 204 return new Twist2d(twist.get(0, 0), twist.get(1, 0), twist.get(2, 0)); 205 } 206 207 /** 208 * Construct inverse kinematics matrix from wheel locations. 209 * 210 * @param fl The location of the front-left wheel relative to the physical center of the robot. 211 * @param fr The location of the front-right wheel relative to the physical center of the robot. 212 * @param rl The location of the rear-left wheel relative to the physical center of the robot. 213 * @param rr The location of the rear-right wheel relative to the physical center of the robot. 214 */ 215 private void setInverseKinematics( 216 Translation2d fl, Translation2d fr, Translation2d rl, Translation2d rr) { 217 m_inverseKinematics.setRow(0, 0, 1, -1, -(fl.getX() + fl.getY())); 218 m_inverseKinematics.setRow(1, 0, 1, 1, fr.getX() - fr.getY()); 219 m_inverseKinematics.setRow(2, 0, 1, 1, rl.getX() - rl.getY()); 220 m_inverseKinematics.setRow(3, 0, 1, -1, -(rr.getX() + rr.getY())); 221 } 222 223 /** 224 * Returns the front-left wheel translation. 225 * 226 * @return The front-left wheel translation. 227 */ 228 public Translation2d getFrontLeft() { 229 return m_frontLeftWheelMeters; 230 } 231 232 /** 233 * Returns the front-right wheel translation. 234 * 235 * @return The front-right wheel translation. 236 */ 237 public Translation2d getFrontRight() { 238 return m_frontRightWheelMeters; 239 } 240 241 /** 242 * Returns the rear-left wheel translation. 243 * 244 * @return The rear-left wheel translation. 245 */ 246 public Translation2d getRearLeft() { 247 return m_rearLeftWheelMeters; 248 } 249 250 /** 251 * Returns the rear-right wheel translation. 252 * 253 * @return The rear-right wheel translation. 254 */ 255 public Translation2d getRearRight() { 256 return m_rearRightWheelMeters; 257 } 258 259 @Override 260 public MecanumDriveWheelPositions copy(MecanumDriveWheelPositions positions) { 261 return new MecanumDriveWheelPositions( 262 positions.frontLeftMeters, 263 positions.frontRightMeters, 264 positions.rearLeftMeters, 265 positions.rearRightMeters); 266 } 267 268 @Override 269 public void copyInto(MecanumDriveWheelPositions positions, MecanumDriveWheelPositions output) { 270 output.frontLeftMeters = positions.frontLeftMeters; 271 output.frontRightMeters = positions.frontRightMeters; 272 output.rearLeftMeters = positions.rearLeftMeters; 273 output.rearRightMeters = positions.rearRightMeters; 274 } 275 276 @Override 277 public MecanumDriveWheelPositions interpolate( 278 MecanumDriveWheelPositions startValue, MecanumDriveWheelPositions endValue, double t) { 279 return startValue.interpolate(endValue, t); 280 } 281}