Austin Schuh | 812d0d1 | 2021-11-04 20:16:48 -0700 | [diff] [blame^] | 1 | // Copyright (c) FIRST and other WPILib contributors. |
| 2 | // Open Source Software; you can modify and/or share it under the terms of |
| 3 | // the WPILib BSD license file in the root directory of this project. |
| 4 | |
| 5 | #include "frc/MathUtil.h" |
| 6 | #include "gtest/gtest.h" |
| 7 | #include "units/angle.h" |
| 8 | |
| 9 | #define EXPECT_UNITS_EQ(a, b) EXPECT_DOUBLE_EQ((a).value(), (b).value()) |
| 10 | |
| 11 | #define EXPECT_UNITS_NEAR(a, b, c) EXPECT_NEAR((a).value(), (b).value(), c) |
| 12 | |
| 13 | TEST(MathUtilTest, ApplyDeadband) { |
| 14 | // < 0 |
| 15 | EXPECT_DOUBLE_EQ(-1.0, frc::ApplyDeadband(-1.0, 0.02)); |
| 16 | EXPECT_DOUBLE_EQ((-0.03 + 0.02) / (1.0 - 0.02), |
| 17 | frc::ApplyDeadband(-0.03, 0.02)); |
| 18 | EXPECT_DOUBLE_EQ(0.0, frc::ApplyDeadband(-0.02, 0.02)); |
| 19 | EXPECT_DOUBLE_EQ(0.0, frc::ApplyDeadband(-0.01, 0.02)); |
| 20 | |
| 21 | // == 0 |
| 22 | EXPECT_DOUBLE_EQ(0.0, frc::ApplyDeadband(0.0, 0.02)); |
| 23 | |
| 24 | // > 0 |
| 25 | EXPECT_DOUBLE_EQ(0.0, frc::ApplyDeadband(0.01, 0.02)); |
| 26 | EXPECT_DOUBLE_EQ(0.0, frc::ApplyDeadband(0.02, 0.02)); |
| 27 | EXPECT_DOUBLE_EQ((0.03 - 0.02) / (1.0 - 0.02), |
| 28 | frc::ApplyDeadband(0.03, 0.02)); |
| 29 | EXPECT_DOUBLE_EQ(1.0, frc::ApplyDeadband(1.0, 0.02)); |
| 30 | } |
| 31 | |
| 32 | TEST(MathUtilTest, InputModulus) { |
| 33 | // These tests check error wrapping. That is, the result of wrapping the |
| 34 | // result of an angle reference minus the measurement. |
| 35 | |
| 36 | // Test symmetric range |
| 37 | EXPECT_DOUBLE_EQ(-20.0, frc::InputModulus(170.0 - (-170.0), -180.0, 180.0)); |
| 38 | EXPECT_DOUBLE_EQ(-20.0, |
| 39 | frc::InputModulus(170.0 + 360.0 - (-170.0), -180.0, 180.0)); |
| 40 | EXPECT_DOUBLE_EQ(-20.0, |
| 41 | frc::InputModulus(170.0 - (-170.0 + 360.0), -180.0, 180.0)); |
| 42 | EXPECT_DOUBLE_EQ(20.0, frc::InputModulus(-170.0 - 170.0, -180.0, 180.0)); |
| 43 | EXPECT_DOUBLE_EQ(20.0, |
| 44 | frc::InputModulus(-170.0 + 360.0 - 170.0, -180.0, 180.0)); |
| 45 | EXPECT_DOUBLE_EQ(20.0, |
| 46 | frc::InputModulus(-170.0 - (170.0 + 360.0), -180.0, 180.0)); |
| 47 | |
| 48 | // Test range starting at zero |
| 49 | EXPECT_DOUBLE_EQ(340.0, frc::InputModulus(170.0 - 190.0, 0.0, 360.0)); |
| 50 | EXPECT_DOUBLE_EQ(340.0, frc::InputModulus(170.0 + 360.0 - 190.0, 0.0, 360.0)); |
| 51 | EXPECT_DOUBLE_EQ(340.0, |
| 52 | frc::InputModulus(170.0 - (190.0 + 360.0), 0.0, 360.0)); |
| 53 | |
| 54 | // Test asymmetric range that doesn't start at zero |
| 55 | EXPECT_DOUBLE_EQ(-20.0, frc::InputModulus(170.0 - (-170.0), -170.0, 190.0)); |
| 56 | |
| 57 | // Test range with both positive endpoints |
| 58 | EXPECT_DOUBLE_EQ(2.0, frc::InputModulus(0.0, 1.0, 3.0)); |
| 59 | EXPECT_DOUBLE_EQ(3.0, frc::InputModulus(1.0, 1.0, 3.0)); |
| 60 | EXPECT_DOUBLE_EQ(2.0, frc::InputModulus(2.0, 1.0, 3.0)); |
| 61 | EXPECT_DOUBLE_EQ(3.0, frc::InputModulus(3.0, 1.0, 3.0)); |
| 62 | EXPECT_DOUBLE_EQ(2.0, frc::InputModulus(4.0, 1.0, 3.0)); |
| 63 | |
| 64 | // Test all supported types |
| 65 | EXPECT_DOUBLE_EQ(-20.0, |
| 66 | frc::InputModulus<double>(170.0 - (-170.0), -170.0, 190.0)); |
| 67 | EXPECT_EQ(-20, frc::InputModulus<int>(170 - (-170), -170, 190)); |
| 68 | EXPECT_EQ(-20_deg, frc::InputModulus<units::degree_t>(170_deg - (-170_deg), |
| 69 | -170_deg, 190_deg)); |
| 70 | } |
| 71 | |
| 72 | TEST(MathUtilTest, AngleModulus) { |
| 73 | EXPECT_UNITS_NEAR( |
| 74 | frc::AngleModulus(units::radian_t{-2000 * wpi::numbers::pi / 180}), |
| 75 | units::radian_t{160 * wpi::numbers::pi / 180}, 1e-10); |
| 76 | EXPECT_UNITS_NEAR( |
| 77 | frc::AngleModulus(units::radian_t{358 * wpi::numbers::pi / 180}), |
| 78 | units::radian_t{-2 * wpi::numbers::pi / 180}, 1e-10); |
| 79 | EXPECT_UNITS_NEAR(frc::AngleModulus(units::radian_t{2.0 * wpi::numbers::pi}), |
| 80 | 0_rad, 1e-10); |
| 81 | |
| 82 | EXPECT_UNITS_EQ(frc::AngleModulus(units::radian_t(5 * wpi::numbers::pi)), |
| 83 | units::radian_t(wpi::numbers::pi)); |
| 84 | EXPECT_UNITS_EQ(frc::AngleModulus(units::radian_t(-5 * wpi::numbers::pi)), |
| 85 | units::radian_t(wpi::numbers::pi)); |
| 86 | EXPECT_UNITS_EQ(frc::AngleModulus(units::radian_t(wpi::numbers::pi / 2)), |
| 87 | units::radian_t(wpi::numbers::pi / 2)); |
| 88 | EXPECT_UNITS_EQ(frc::AngleModulus(units::radian_t(-wpi::numbers::pi / 2)), |
| 89 | units::radian_t(-wpi::numbers::pi / 2)); |
| 90 | } |