// Copyright (c) FIRST and other WPILib contributors. // Open Source Software; you can modify and/or share it under the terms of // the WPILib BSD license file in the root directory of this project. #pragma once #include #include #include #include #include "wpi/commands2/CommandPtr.hpp" #include "wpi/commands2/Subsystem.hpp" #include "wpi/sysid/SysIdRoutineLog.hpp" #include "wpi/system/Timer.hpp" namespace wpi::cmd::sysid { using ramp_rate_t = wpi::units::unit_t>>; /** Hardware-independent configuration for a SysId test routine. */ class Config { public: /// The voltage ramp rate used for quasistatic test routines. ramp_rate_t rampRate{1_V / 1_s}; /// The step voltage output used for dynamic test routines. wpi::units::volt_t stepVoltage{7_V}; /// Safety timeout for the test routine commands. wpi::units::second_t timeout{10_s}; /// Optional handle for recording test state in a third-party logging /// solution. std::function recordState; /** * Create a new configuration for a SysId test routine. * * @param rampRate The voltage ramp rate used for quasistatic test routines. * Defaults to 1 volt per second if left null. * @param stepVoltage The step voltage output used for dynamic test routines. * Defaults to 7 volts if left null. * @param timeout Safety timeout for the test routine commands. Defaults to 10 * seconds if left null. * @param recordState Optional handle for recording test state in a * third-party logging solution. If provided, the test routine state will be * passed to this callback instead of logged in WPILog. */ Config(std::optional rampRate, std::optional stepVoltage, std::optional timeout, std::function recordState) : recordState{std::move(recordState)} { if (rampRate) { this->rampRate = rampRate.value(); } if (stepVoltage) { this->stepVoltage = stepVoltage.value(); } if (timeout) { this->timeout = timeout.value(); } } }; class Mechanism { public: /// Sends the SysId-specified drive signal to the mechanism motors during test /// routines. std::function drive; /// Returns measured data (voltages, positions, velocities) of the mechanism /// motors during test routines. std::function log; /// The subsystem containing the motor(s) that is (or are) being /// characterized. wpi::cmd::Subsystem* subsystem; /// The name of the mechanism being tested. Will be appended to the log entry /// title for the routine's test state, e.g. "sysid-test-state-mechanism". std::string name; /** * Create a new mechanism specification for a SysId routine. * * @param drive Sends the SysId-specified drive signal to the mechanism motors * during test routines. * @param log Returns measured data of the mechanism motors during test * routines. To return data, call `Motor(string motorName)` on the supplied * `SysIdRoutineLog` instance, and then call one or more of the chainable * logging handles (e.g. `voltage`) on the returned `MotorLog`. Multiple * motors can be logged in a single callback by calling `Motor` multiple * times. * @param subsystem The subsystem containing the motor(s) that is (or are) * being characterized. Will be declared as a requirement for the returned * test commands. * @param name The name of the mechanism being tested. Will be appended to the * log entry * title for the routine's test state, e.g. * "sysid-test-state-mechanism". Defaults to the name of the subsystem if * left null. */ Mechanism(std::function drive, std::function log, wpi::cmd::Subsystem* subsystem, std::string_view name) : drive{std::move(drive)}, log{log ? std::move(log) : [](wpi::sysid::SysIdRoutineLog* log) {}}, subsystem{subsystem}, name{name} {} /** * Create a new mechanism specification for a SysId routine. Defaults the * mechanism name to the subsystem name. * * @param drive Sends the SysId-specified drive signal to the mechanism motors * during test routines. * @param log Returns measured data of the mechanism motors during test * routines. To return data, call `Motor(string motorName)` on the supplied * `SysIdRoutineLog` instance, and then call one or more of the chainable * logging handles (e.g. `voltage`) on the returned `MotorLog`. Multiple * motors can be logged in a single callback by calling `Motor` multiple * times. * @param subsystem The subsystem containing the motor(s) that is (or are) * being characterized. Will be declared as a requirement for the returned * test commands. The subsystem's `name` will be appended to the log entry * title for the routine's test state, e.g. "sysid-test-state-subsystem". */ Mechanism(std::function drive, std::function log, wpi::cmd::Subsystem* subsystem) : drive{std::move(drive)}, log{log ? std::move(log) : [](wpi::sysid::SysIdRoutineLog* log) {}}, subsystem{subsystem}, name{subsystem->GetName()} {} }; /** * Motor direction for a SysId test. */ enum Direction { /// Forward. kForward, /// Reverse. kReverse }; /** * A SysId characterization routine for a single mechanism. Mechanisms may have * multiple motors. * * A single subsystem may have multiple mechanisms, but mechanisms should not * share test routines. Each complete test of a mechanism should have its own * SysIdRoutine instance, since the log name of the recorded data is determined * by the mechanism name. * * The test state (e.g. "quasistatic-forward") is logged once per iteration * during test execution, and once with state "none" when a test ends. Motor * frames are logged every iteration during test execution. * * Timestamps are not coordinated across data, so motor frames and test state * tags may be recorded on different log frames. Because frame alignment is not * guaranteed, SysId parses the log by using the test state flag to determine * the timestamp range for each section of the test, and then extracts the motor * frames within the valid timestamp ranges. If a given test was run multiple * times in a single logfile, the user will need to select which of the tests to * use for the fit in the analysis tool. */ class SysIdRoutine : public wpi::sysid::SysIdRoutineLog { public: /** * Create a new SysId characterization routine. * * @param config Hardware-independent parameters for the SysId routine. * @param mechanism Hardware interface for the SysId routine. */ SysIdRoutine(Config config, Mechanism mechanism) : SysIdRoutineLog(mechanism.name), m_config(config), m_mechanism(mechanism), m_recordState(config.recordState ? config.recordState : [this](wpi::sysid::State state) { this->RecordState(state); }) {} wpi::cmd::CommandPtr Quasistatic(Direction direction); wpi::cmd::CommandPtr Dynamic(Direction direction); private: Config m_config; Mechanism m_mechanism; wpi::units::volt_t m_outputVolts{0}; std::function m_recordState; wpi::Timer timer; }; } // namespace wpi::cmd::sysid