2018-02-01 09:53:04 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2020 Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
* 2020 Freie Universität Berlin
|
|
|
|
* 2020 Inria
|
|
|
|
* 2018 Joakim Nohlgård <joakim.nohlgard@eistec.se>
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser General
|
|
|
|
* Public License v2.1. See the file LICENSE in the top level directory for more
|
|
|
|
* details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup sys_ztimer_convert_frac
|
|
|
|
*
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief ztimer conversion using frac implementation
|
|
|
|
*
|
|
|
|
* @author Joakim Nohlgård <joakim.nohlgard@eistec.se>
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdint.h>
|
2020-11-26 13:21:07 +01:00
|
|
|
#include <inttypes.h>
|
2018-02-01 09:53:04 +01:00
|
|
|
|
|
|
|
#include "frac.h"
|
|
|
|
#include "assert.h"
|
|
|
|
#include "irq.h"
|
|
|
|
#include "ztimer/convert.h"
|
|
|
|
#include "ztimer/convert_frac.h"
|
|
|
|
|
2020-10-22 11:35:22 +02:00
|
|
|
#define ENABLE_DEBUG 0
|
2018-02-01 09:53:04 +01:00
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Compute the scaling parameters for the given two frequencies
|
|
|
|
*
|
|
|
|
* @param[in] self pointer to instance to operate on
|
|
|
|
* @param[in] freq_self desired frequency of this clock
|
|
|
|
* @param[in] freq_lower frequency of the underlying clock
|
|
|
|
*/
|
2020-05-07 14:59:40 +02:00
|
|
|
static void ztimer_convert_frac_compute_scale(ztimer_convert_frac_t *self,
|
|
|
|
uint32_t freq_self,
|
|
|
|
uint32_t freq_lower);
|
2018-02-01 09:53:04 +01:00
|
|
|
|
|
|
|
static void ztimer_convert_frac_op_set(ztimer_clock_t *z, uint32_t val)
|
|
|
|
{
|
|
|
|
ztimer_convert_frac_t *self = (ztimer_convert_frac_t *)z;
|
|
|
|
uint32_t target_lower = frac_scale(&self->scale_set, val + self->round);
|
2020-05-07 14:59:40 +02:00
|
|
|
|
|
|
|
DEBUG("ztimer_convert_frac_op_set(%" PRIu32 ")=%" PRIu32 "\n", val,
|
2018-02-01 09:53:04 +01:00
|
|
|
target_lower);
|
|
|
|
ztimer_set(self->super.lower, &self->super.lower_entry, target_lower);
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t ztimer_convert_frac_op_now(ztimer_clock_t *z)
|
|
|
|
{
|
|
|
|
ztimer_convert_frac_t *self = (ztimer_convert_frac_t *)z;
|
|
|
|
uint32_t lower_now = ztimer_now(self->super.lower);
|
2020-05-07 14:59:40 +02:00
|
|
|
|
2018-02-01 09:53:04 +01:00
|
|
|
if (lower_now == 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
uint32_t scaled = frac_scale(&self->scale_now, lower_now);
|
2021-01-19 17:48:14 +01:00
|
|
|
|
2020-05-07 14:59:40 +02:00
|
|
|
DEBUG("ztimer_convert_frac_op_now() %" PRIu32 "->%" PRIu32 "\n", lower_now,
|
|
|
|
scaled);
|
2018-02-01 09:53:04 +01:00
|
|
|
return scaled;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const ztimer_ops_t ztimer_convert_frac_ops = {
|
2020-05-07 14:59:40 +02:00
|
|
|
.set = ztimer_convert_frac_op_set,
|
|
|
|
.now = ztimer_convert_frac_op_now,
|
2018-02-01 09:53:04 +01:00
|
|
|
.cancel = ztimer_convert_cancel,
|
2022-02-04 18:51:21 +01:00
|
|
|
#if MODULE_ZTIMER_ONDEMAND
|
|
|
|
.start = ztimer_convert_start,
|
|
|
|
.stop = ztimer_convert_stop,
|
|
|
|
#endif
|
2018-02-01 09:53:04 +01:00
|
|
|
};
|
|
|
|
|
2020-05-07 14:59:40 +02:00
|
|
|
static void ztimer_convert_frac_compute_scale(ztimer_convert_frac_t *self,
|
|
|
|
uint32_t freq_self,
|
|
|
|
uint32_t freq_lower)
|
2018-02-01 09:53:04 +01:00
|
|
|
{
|
|
|
|
assert(freq_self);
|
|
|
|
assert(freq_lower);
|
|
|
|
frac_init(&self->scale_now, freq_self, freq_lower);
|
|
|
|
frac_init(&self->scale_set, freq_lower, freq_self);
|
|
|
|
}
|
|
|
|
|
2020-05-07 14:59:40 +02:00
|
|
|
void ztimer_convert_frac_init(ztimer_convert_frac_t *self,
|
|
|
|
ztimer_clock_t *lower, uint32_t freq_self,
|
|
|
|
uint32_t freq_lower)
|
2018-02-01 09:53:04 +01:00
|
|
|
{
|
|
|
|
DEBUG("ztimer_convert_frac_init: %p->%p fs=%" PRIu32 " fl=%" PRIu32 "\n",
|
2020-05-07 14:59:40 +02:00
|
|
|
(void *)self, (void *)lower, freq_self, freq_lower);
|
2018-02-01 09:53:04 +01:00
|
|
|
|
|
|
|
*self = (ztimer_convert_frac_t) {
|
|
|
|
.super.super = { .ops = &ztimer_convert_frac_ops, },
|
|
|
|
.super.lower = lower,
|
2020-05-07 14:59:40 +02:00
|
|
|
.super.lower_entry =
|
2023-03-15 15:13:48 +01:00
|
|
|
{ .callback = (void (*)(void *)) ztimer_handler, .arg = &self->super, },
|
2018-02-01 09:53:04 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
ztimer_convert_frac_compute_scale(self, freq_self, freq_lower);
|
|
|
|
if (freq_self < freq_lower) {
|
2020-06-16 12:24:11 +02:00
|
|
|
self->super.super.max_value = frac_scale(&self->scale_now, UINT32_MAX);
|
2022-02-02 20:21:35 +01:00
|
|
|
#if !MODULE_ZTIMER_ONDEMAND
|
|
|
|
/* extend lower clock only if the ondemand driver isn't selected
|
|
|
|
* otherwise, the clock extension will be called with the first
|
|
|
|
* ztimer_acquire() call */
|
2020-05-04 15:47:37 +02:00
|
|
|
ztimer_init_extend(&self->super.super);
|
2022-02-02 20:21:35 +01:00
|
|
|
#endif
|
2018-02-01 09:53:04 +01:00
|
|
|
}
|
|
|
|
else {
|
2020-05-07 14:59:40 +02:00
|
|
|
DEBUG("ztimer_convert_frac_init: rounding up val:%" PRIu32 "\n",
|
|
|
|
(uint32_t)(freq_self / freq_lower));
|
2018-02-01 09:53:04 +01:00
|
|
|
self->round = freq_self / freq_lower;
|
|
|
|
self->super.super.max_value = UINT32_MAX;
|
|
|
|
}
|
2022-02-02 17:31:28 +01:00
|
|
|
#if MODULE_PM_LAYERED && !MODULE_ZTIMER_ONDEMAND
|
2021-06-18 18:24:22 +02:00
|
|
|
self->super.super.block_pm_mode = ZTIMER_CLOCK_NO_REQUIRED_PM_MODE;
|
|
|
|
#endif
|
2018-02-01 09:53:04 +01:00
|
|
|
}
|