blob: 6b5af2baf5c4a94839283581c6bf254c82843c85 [file] [log] [blame]
Austin Schuh812d0d12021-11-04 20:16:48 -07001// 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
13TEST(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
32TEST(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
72TEST(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}