/*----------------------------------------------------------------------------*/ /* Copyright (c) FIRST 2008-2016. All Rights Reserved. */ /* Open Source Software - may be modified and shared by FRC teams. The code */ /* must be accompanied by the FIRST BSD license file in the root directory of */ /* the project. */ /*----------------------------------------------------------------------------*/ #pragma once #include #include #include #include #include "ErrorBase.h" #include "HAL/cpp/priority_mutex.h" typedef std::function TimerEventHandler; class Notifier : public ErrorBase { public: explicit Notifier(TimerEventHandler handler); template Notifier(Callable&& f, Arg&& arg, Args&&... args) : Notifier(std::bind(std::forward(f), std::forward(arg), std::forward(args)...)) {} virtual ~Notifier(); Notifier(const Notifier&) = delete; Notifier& operator=(const Notifier&) = delete; void StartSingle(double delay); void StartPeriodic(double period); void Stop(); private: static std::list timerQueue; static priority_recursive_mutex queueMutex; static priority_mutex halMutex; static void *m_notifier; static std::atomic refcount; // Process the timer queue on a timer event static void ProcessQueue(uint32_t mask, void *params); // Update the FPGA alarm since the queue has changed static void UpdateAlarm(); // Insert the Notifier in the timer queue void InsertInQueue(bool reschedule); // Delete this Notifier from the timer queue void DeleteFromQueue(); // Address of the handler TimerEventHandler m_handler; // The relative time (either periodic or single) double m_period = 0; // Absolute expiration time for the current event double m_expirationTime = 0; // True if this is a periodic event bool m_periodic = false; // Indicates if this entry is queued bool m_queued = false; // Held by interrupt manager task while handler call is in progress priority_mutex m_handlerMutex; static std::thread m_task; static std::atomic m_stopped; static void Run(); };