/*----------------------------------------------------------------------------*/ /* Copyright (c) 2019 FIRST. All Rights Reserved. */ /* Open Source Software - may be modified and shared by FRC teams. The code */ /* must be accompanied by the FIRST BSD license file in the root directory of */ /* the project. */ /*----------------------------------------------------------------------------*/ #include #include "frc/kinematics/DifferentialDriveKinematics.h" #include "frc/kinematics/DifferentialDriveOdometry.h" #include "gtest/gtest.h" static constexpr double kEpsilon = 1E-9; using namespace frc; TEST(DifferentialDriveOdometry, OneIteration) { DifferentialDriveKinematics kinematics{0.381_m * 2}; DifferentialDriveOdometry odometry{kinematics, 0_rad}; odometry.ResetPosition(Pose2d(), 0_rad); DifferentialDriveWheelSpeeds wheelSpeeds{0.02_mps, 0.02_mps}; odometry.UpdateWithTime(0_s, Rotation2d(), DifferentialDriveWheelSpeeds()); const auto& pose = odometry.UpdateWithTime(1_s, Rotation2d(), wheelSpeeds); EXPECT_NEAR(pose.Translation().X().to(), 0.02, kEpsilon); EXPECT_NEAR(pose.Translation().Y().to(), 0.0, kEpsilon); EXPECT_NEAR(pose.Rotation().Radians().to(), 0.0, kEpsilon); } TEST(DifferentialDriveOdometry, QuarterCircle) { DifferentialDriveKinematics kinematics{0.381_m * 2}; DifferentialDriveOdometry odometry{kinematics, 0_rad}; odometry.ResetPosition(Pose2d(), 0_rad); DifferentialDriveWheelSpeeds wheelSpeeds{ 0.0_mps, units::meters_per_second_t(5 * wpi::math::pi)}; odometry.UpdateWithTime(0_s, Rotation2d(), DifferentialDriveWheelSpeeds()); const auto& pose = odometry.UpdateWithTime(1_s, Rotation2d(90_deg), wheelSpeeds); EXPECT_NEAR(pose.Translation().X().to(), 5.0, kEpsilon); EXPECT_NEAR(pose.Translation().Y().to(), 5.0, kEpsilon); EXPECT_NEAR(pose.Rotation().Degrees().to(), 90.0, kEpsilon); } TEST(DifferentialDriveWheelSpeeds, GyroAngleReset) { DifferentialDriveKinematics kinematics{0.381_m * 2}; DifferentialDriveOdometry odometry{kinematics, Rotation2d(90_deg)}; odometry.UpdateWithTime(0_s, Rotation2d(90_deg), {}); const auto& pose = odometry.UpdateWithTime(1_s, Rotation2d(90_deg), {1_mps, 1_mps}); EXPECT_NEAR(pose.Translation().X().to(), 1.0, kEpsilon); EXPECT_NEAR(pose.Translation().Y().to(), 0.0, kEpsilon); EXPECT_NEAR(pose.Rotation().Degrees().to(), 0.0, kEpsilon); } TEST(DifferentialDriveOdometry, EncoderDistances) { DifferentialDriveKinematics kinematics{0.381_m * 2}; DifferentialDriveOdometry odometry{kinematics, Rotation2d(45_deg)}; const auto& pose = odometry.Update(Rotation2d(135_deg), 0_m, units::meter_t(5 * wpi::math::pi)); EXPECT_NEAR(pose.Translation().X().to(), 5.0, kEpsilon); EXPECT_NEAR(pose.Translation().Y().to(), 5.0, kEpsilon); EXPECT_NEAR(pose.Rotation().Degrees().to(), 90.0, kEpsilon); }