|
|
|
/*
|
|
|
|
* Copyright 2013-2019 Software Radio Systems Limited
|
|
|
|
*
|
|
|
|
* This file is part of srsLTE.
|
|
|
|
*
|
|
|
|
* srsLTE is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License as
|
|
|
|
* published by the Free Software Foundation, either version 3 of
|
|
|
|
* the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* srsLTE is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU Affero General Public License for more details.
|
|
|
|
*
|
|
|
|
* A copy of the GNU Affero General Public License can be found in
|
|
|
|
* the LICENSE file in the top-level directory of this distribution
|
|
|
|
* and at http://www.gnu.org/licenses/.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
* File: timers.h
|
|
|
|
* Description: Manually incremented timers. Call a callback function upon
|
|
|
|
* expiry.
|
|
|
|
* Reference:
|
|
|
|
*****************************************************************************/
|
|
|
|
|
|
|
|
#ifndef SRSLTE_TIMERS_H
|
|
|
|
#define SRSLTE_TIMERS_H
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <vector>
|
|
|
|
#include <time.h>
|
|
|
|
|
|
|
|
#include "srslte/srslte.h"
|
|
|
|
|
|
|
|
namespace srslte {
|
|
|
|
|
|
|
|
class timer_callback
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual void timer_expired(uint32_t timer_id) = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
class timers
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
class timer
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
timer(uint32_t id_=0) {id = id_; counter = 0; timeout = 0; running = false; callback = NULL; }
|
|
|
|
void set(timer_callback *callback_, uint32_t timeout_) {
|
|
|
|
callback = callback_;
|
|
|
|
timeout = timeout_;
|
|
|
|
reset();
|
|
|
|
}
|
|
|
|
bool is_running() {
|
|
|
|
return (counter < timeout) && running;
|
|
|
|
}
|
|
|
|
bool is_expired() {
|
|
|
|
return (timeout > 0) && (counter >= timeout);
|
|
|
|
}
|
|
|
|
uint32_t get_timeout() {
|
|
|
|
return timeout;
|
|
|
|
}
|
|
|
|
void reset() {
|
|
|
|
counter = 0;
|
|
|
|
}
|
|
|
|
uint32_t value() {
|
|
|
|
return counter;
|
|
|
|
}
|
|
|
|
void step() {
|
|
|
|
if (running) {
|
|
|
|
counter++;
|
|
|
|
if (is_expired()) {
|
|
|
|
running = false;
|
|
|
|
if (callback) {
|
|
|
|
callback->timer_expired(id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void stop() {
|
|
|
|
running = false;
|
|
|
|
}
|
|
|
|
void run() {
|
|
|
|
running = true;
|
|
|
|
}
|
|
|
|
uint32_t id;
|
|
|
|
private:
|
|
|
|
timer_callback *callback;
|
|
|
|
uint32_t timeout;
|
|
|
|
uint32_t counter;
|
|
|
|
bool running;
|
|
|
|
};
|
|
|
|
|
|
|
|
timers(uint32_t nof_timers_) : timer_list(nof_timers_),used_timers(nof_timers_) {
|
|
|
|
nof_timers = nof_timers_;
|
|
|
|
next_timer = 0;
|
|
|
|
nof_used_timers = 0;
|
|
|
|
for (uint32_t i=0;i<nof_timers;i++) {
|
|
|
|
timer_list[i].id = i;
|
|
|
|
used_timers[i] = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void step_all() {
|
|
|
|
for (uint32_t i=0;i<nof_timers;i++) {
|
|
|
|
get(i)->step();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void stop_all() {
|
|
|
|
for (uint32_t i=0;i<nof_timers;i++) {
|
|
|
|
get(i)->stop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void run_all() {
|
|
|
|
for (uint32_t i=0;i<nof_timers;i++) {
|
|
|
|
get(i)->run();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void reset_all() {
|
|
|
|
for (uint32_t i=0;i<nof_timers;i++) {
|
|
|
|
get(i)->reset();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
timer *get(uint32_t i) {
|
|
|
|
if (i < nof_timers) {
|
|
|
|
return &timer_list[i];
|
|
|
|
} else {
|
|
|
|
printf("Error accessing invalid timer %d (Only %d timers available)\n", i, nof_timers);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
void release_id(uint32_t i) {
|
|
|
|
if (nof_used_timers > 0 && i < nof_timers) {
|
|
|
|
used_timers[i] = false;
|
|
|
|
nof_used_timers--;
|
|
|
|
} else {
|
|
|
|
ERROR("Error releasing timer id=%d: nof_used_timers=%d, nof_timers=%d\n", i, nof_used_timers, nof_timers);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
uint32_t get_unique_id() {
|
|
|
|
if (nof_used_timers >= nof_timers) {
|
|
|
|
ERROR("Error getting unique timer id: no more timers available\n");
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
for (uint32_t i=0;i<nof_timers;i++) {
|
|
|
|
if (!used_timers[i]) {
|
|
|
|
used_timers[i] = true;
|
|
|
|
nof_used_timers++;
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ERROR("Error getting unique timer id: no more timers available but nof_used_timers=%d, nof_timers=%d\n",
|
|
|
|
nof_used_timers,
|
|
|
|
nof_timers);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
uint32_t next_timer;
|
|
|
|
uint32_t nof_used_timers;
|
|
|
|
uint32_t nof_timers;
|
|
|
|
std::vector<timer> timer_list;
|
|
|
|
std::vector<bool> used_timers;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace srslte
|
|
|
|
|
|
|
|
#endif // SRSLTE_TIMERS_H
|