// 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 #include #include #include #include #include #include "frc/IterativeRobotBase.h" #include "frc/RobotController.h" namespace frc { /** * TimedRobot implements the IterativeRobotBase robot program framework. * * The TimedRobot class is intended to be subclassed by a user creating a * robot program. * * Periodic() functions from the base class are called on an interval by a * Notifier instance. */ class TimedRobot : public IterativeRobotBase { public: /// Default loop period. static constexpr auto kDefaultPeriod = 20_ms; /** * Provide an alternate "main loop" via StartCompetition(). */ void StartCompetition() override; /** * Ends the main loop in StartCompetition(). */ void EndCompetition() override; /** * Constructor for TimedRobot. * * @param period The period of the robot loop function. */ explicit TimedRobot(units::second_t period = kDefaultPeriod); /** * Constructor for TimedRobot. * * @param frequency The frequency of the robot loop function. */ explicit TimedRobot(units::hertz_t frequency); TimedRobot(TimedRobot&&) = default; TimedRobot& operator=(TimedRobot&&) = default; ~TimedRobot() override; /** * Return the system clock time in micrseconds for the start of the current * periodic loop. This is in the same time base as Timer.GetFPGATimestamp(), * but is stable through a loop. It is updated at the beginning of every * periodic callback (including the normal periodic loop). * * @return Robot running time in microseconds, as of the start of the current * periodic function. */ uint64_t GetLoopStartTime(); /** * Add a callback to run at a specific period with a starting time offset. * * This is scheduled on TimedRobot's Notifier, so TimedRobot and the callback * run synchronously. Interactions between them are thread-safe. * * @param callback The callback to run. * @param period The period at which to run the callback. * @param offset The offset from the common starting time. This is useful * for scheduling a callback in a different timeslot relative * to TimedRobot. */ void AddPeriodic(std::function callback, units::second_t period, units::second_t offset = 0_s); private: class Callback { public: std::function func; std::chrono::microseconds period; std::chrono::microseconds expirationTime; /** * Construct a callback container. * * @param func The callback to run. * @param startTime The common starting point for all callback scheduling. * @param period The period at which to run the callback. * @param offset The offset from the common starting time. */ Callback(std::function func, std::chrono::microseconds startTime, std::chrono::microseconds period, std::chrono::microseconds offset) : func{std::move(func)}, period{period}, expirationTime( startTime + offset + period + (std::chrono::microseconds{frc::RobotController::GetFPGATime()} - startTime) / period * period) {} bool operator>(const Callback& rhs) const { return expirationTime > rhs.expirationTime; } }; hal::Handle m_notifier; std::chrono::microseconds m_startTime; uint64_t m_loopStartTimeUs = 0; wpi::priority_queue, std::greater> m_callbacks; }; } // namespace frc