2023-10-24 05:14:31 +02:00
|
|
|
#include "timer.h"
|
|
|
|
|
|
|
|
#include <errno.h>
|
2023-12-19 05:21:41 +01:00
|
|
|
#include <stdbool.h>
|
2023-10-24 05:14:31 +02:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include "block.h"
|
|
|
|
#include "util.h"
|
|
|
|
|
2023-10-25 16:30:08 +02:00
|
|
|
static unsigned int compute_tick(const block *const blocks,
|
|
|
|
const unsigned short block_count) {
|
2023-10-24 05:14:31 +02:00
|
|
|
unsigned int tick = 0;
|
|
|
|
|
2023-10-25 16:30:08 +02:00
|
|
|
for (unsigned short i = 0; i < block_count; ++i) {
|
2023-10-24 05:14:31 +02:00
|
|
|
const block *const block = &blocks[i];
|
|
|
|
tick = gcd(block->interval, tick);
|
|
|
|
}
|
|
|
|
|
|
|
|
return tick;
|
|
|
|
}
|
|
|
|
|
2023-10-25 16:30:08 +02:00
|
|
|
static unsigned int compute_reset_value(const block *const blocks,
|
|
|
|
const unsigned short block_count) {
|
2023-10-24 05:14:31 +02:00
|
|
|
unsigned int reset_value = 1;
|
|
|
|
|
2023-10-25 16:30:08 +02:00
|
|
|
for (unsigned short i = 0; i < block_count; ++i) {
|
2023-10-24 05:14:31 +02:00
|
|
|
const block *const block = &blocks[i];
|
|
|
|
reset_value = MAX(block->interval, reset_value);
|
|
|
|
}
|
|
|
|
|
|
|
|
return reset_value;
|
|
|
|
}
|
|
|
|
|
2023-10-25 16:30:08 +02:00
|
|
|
timer timer_new(const block *const blocks, const unsigned short block_count) {
|
2023-12-19 05:21:41 +01:00
|
|
|
const unsigned int reset_value = compute_reset_value(blocks, block_count);
|
|
|
|
|
2023-10-24 05:14:31 +02:00
|
|
|
timer timer = {
|
2023-12-19 05:21:41 +01:00
|
|
|
.time = reset_value, // Initial value to execute all blocks.
|
2023-10-25 16:30:08 +02:00
|
|
|
.tick = compute_tick(blocks, block_count),
|
2023-12-19 05:21:41 +01:00
|
|
|
.reset_value = reset_value,
|
2023-10-24 05:14:31 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
return timer;
|
|
|
|
}
|
|
|
|
|
|
|
|
int timer_arm(timer *const timer) {
|
|
|
|
errno = 0;
|
|
|
|
(void)alarm(timer->tick);
|
|
|
|
|
|
|
|
if (errno != 0) {
|
|
|
|
(void)fprintf(stderr, "error: could not arm timer\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wrap `time` to the interval [1, reset_value].
|
2023-12-19 05:21:41 +01:00
|
|
|
timer->time = (timer->time + timer->tick) % timer->reset_value;
|
2023-10-24 05:14:31 +02:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2023-12-19 05:21:41 +01:00
|
|
|
|
|
|
|
bool timer_must_run_block(const timer *const timer, const block *const block) {
|
|
|
|
if (timer->time == timer->reset_value) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (block->interval == 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return timer->time % block->interval == 0;
|
|
|
|
}
|