Files
allwpilib/commandsv2/src/main/native/include/wpi/commands2/sysid/SysIdRoutine.hpp

198 lines
7.6 KiB
C++
Raw Normal View History

// 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 <functional>
#include <string>
#include <string_view>
#include <utility>
2025-11-07 19:56:21 -05:00
#include "wpi/commands2/CommandPtr.hpp"
#include "wpi/commands2/Subsystem.hpp"
2025-11-07 19:57:55 -05:00
#include "wpi/sysid/SysIdRoutineLog.hpp"
#include "wpi/system/Timer.hpp"
2025-11-07 20:00:05 -05:00
namespace wpi::cmd::sysid {
2025-11-07 20:01:58 -05:00
using ramp_rate_t = wpi::units::unit_t<wpi::units::compound_unit<
wpi::units::volt, wpi::units::inverse<wpi::units::second>>>;
/** 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<void(wpi::sysid::State)> 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<ramp_rate_t> rampRate,
2025-11-07 20:00:05 -05:00
std::optional<wpi::units::volt_t> stepVoltage,
std::optional<wpi::units::second_t> timeout,
std::function<void(wpi::sysid::State)> 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<void(wpi::units::volt_t)> drive;
/// Returns measured data (voltages, positions, velocities) of the mechanism
/// motors during test routines.
std::function<void(wpi::sysid::SysIdRoutineLog*)> 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.
*/
2025-11-07 20:00:05 -05:00
Mechanism(std::function<void(wpi::units::volt_t)> drive,
std::function<void(wpi::sysid::SysIdRoutineLog*)> 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".
*/
2025-11-07 20:00:05 -05:00
Mechanism(std::function<void(wpi::units::volt_t)> drive,
std::function<void(wpi::sysid::SysIdRoutineLog*)> 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.
*/
2025-11-07 20:00:05 -05:00
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);
}) {}
2025-11-07 20:00:05 -05:00
wpi::cmd::CommandPtr Quasistatic(Direction direction);
wpi::cmd::CommandPtr Dynamic(Direction direction);
private:
Config m_config;
Mechanism m_mechanism;
2025-11-07 20:00:05 -05:00
wpi::units::volt_t m_outputVolts{0};
std::function<void(wpi::sysid::State)> m_recordState;
wpi::Timer timer;
};
2025-11-07 20:00:05 -05:00
} // namespace wpi::cmd::sysid