// 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 "frc/IterativeRobotBase.h" #include "frc/Timer.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: 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 Period. */ explicit TimedRobot(units::second_t period = kDefaultPeriod); ~TimedRobot() override; TimedRobot(TimedRobot&&) = default; TimedRobot& operator=(TimedRobot&&) = default; /** * 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; units::second_t period; units::second_t 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, units::second_t startTime, units::second_t period, units::second_t offset) : func{std::move(func)}, period{period}, expirationTime{startTime + offset + units::math::floor( (Timer::GetFPGATimestamp() - startTime) / period) * period + period} {} bool operator>(const Callback& rhs) const { return expirationTime > rhs.expirationTime; } }; hal::Handle m_notifier; units::second_t m_startTime; wpi::priority_queue, std::greater> m_callbacks; }; } // namespace frc