mirror of
https://gitee.com/mirrors_PX4/PX4-Autopilot.git
synced 2026-04-14 10:07:39 +08:00
522 lines
13 KiB
C
522 lines
13 KiB
C
/****************************************************************************
|
|
*
|
|
* Copyright (c) 2016-2020 PX4 Development Team. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in
|
|
* the documentation and/or other materials provided with the
|
|
* distribution.
|
|
* 3. Neither the name PX4 nor the names of its contributors may be
|
|
* used to endorse or promote products derived from this software
|
|
* without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
|
|
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
|
|
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
****************************************************************************/
|
|
|
|
/**
|
|
* @file hrt.c
|
|
* Author: Peter van der Perk <peter.vanderperk@nxp.com>
|
|
* David Sidrane <david_s5@nscdg.com>
|
|
*
|
|
* High-resolution timer callouts and timekeeping.
|
|
*
|
|
* This driver uses the S32K3 System Time Module STM
|
|
*
|
|
* Note that really, this could use systick too, but that's
|
|
* monopolised by NuttX and stealing it would just be awkward.
|
|
*/
|
|
|
|
#include <px4_platform_common/px4_config.h>
|
|
#include <systemlib/px4_macros.h>
|
|
#include <lib/perf/perf_counter.h>
|
|
#include <nuttx/arch.h>
|
|
#include <nuttx/irq.h>
|
|
|
|
#include <sys/types.h>
|
|
#include <stdbool.h>
|
|
|
|
#include <assert.h>
|
|
#include <debug.h>
|
|
#include <time.h>
|
|
#include <queue.h>
|
|
#include <errno.h>
|
|
#include <string.h>
|
|
|
|
#include <board_config.h>
|
|
#include <drivers/drv_hrt.h>
|
|
|
|
#include "hardware/s32k3xx_stm.h"
|
|
|
|
|
|
#ifdef CONFIG_DEBUG_HRT
|
|
# define hrtinfo _info
|
|
#else
|
|
# define hrtinfo(x...)
|
|
#endif
|
|
|
|
#ifdef HRT_TIMER
|
|
|
|
#define HRT_TIMER_FREQ 1000000 //FXOSC 16Mhz / 2 / 8
|
|
|
|
#define HRT_TIMER_VECTOR S32K3XX_IRQ_STM0
|
|
|
|
/**
|
|
* Minimum/maximum deadlines.
|
|
*
|
|
* These are suitable for use with a 16-bit timer/counter clocked
|
|
* at 1MHz. The high-resolution timer need only guarantee that it
|
|
* not wrap more than once in the 50ms period for absolute time to
|
|
* be consistently maintained.
|
|
*
|
|
* The minimum deadline must be such that the time taken between
|
|
* reading a time and writing a deadline to the timer cannot
|
|
* result in missing the deadline.
|
|
*/
|
|
#define HRT_INTERVAL_MIN 50
|
|
#define HRT_INTERVAL_MAX 50000
|
|
|
|
/*
|
|
* Period of the free-running counter, in microseconds.
|
|
*/
|
|
#define HRT_COUNTER_PERIOD UINT32_MAX
|
|
|
|
/*
|
|
* Scaling factor(s) for the free-running counter; convert an input
|
|
* in counts to a time in microseconds.
|
|
*/
|
|
#define HRT_COUNTER_SCALE(_c) (_c)
|
|
|
|
/* Register accessors */
|
|
|
|
#define _REG(_addr) (*(volatile uint32_t *)(_addr))
|
|
|
|
/*
|
|
* Specific registers and bits used by HRT sub-functions
|
|
*/
|
|
|
|
#define STATUS_HRT STM_CIR_CIF
|
|
|
|
/*
|
|
* Queue of callout entries.
|
|
*/
|
|
static struct sq_queue_s callout_queue;
|
|
|
|
/* latency baseline (last compare value applied) */
|
|
static uint32_t latency_baseline;
|
|
|
|
/* timer count at interrupt (for latency purposes) */
|
|
static uint32_t latency_actual;
|
|
|
|
/* latency histogram */
|
|
const uint16_t latency_bucket_count = LATENCY_BUCKET_COUNT;
|
|
const uint16_t latency_buckets[LATENCY_BUCKET_COUNT] = { 1, 2, 5, 10, 20, 50, 100, 1000 };
|
|
__EXPORT uint32_t latency_counters[LATENCY_BUCKET_COUNT + 1];
|
|
|
|
/* timer-specific functions */
|
|
static void hrt_tim_init(void);
|
|
static int hrt_tim_isr(int irq, void *context, void *args);
|
|
static void hrt_latency_update(void);
|
|
|
|
/* callout list manipulation */
|
|
static void hrt_call_internal(struct hrt_call *entry, hrt_abstime deadline, hrt_abstime interval, hrt_callout callout,
|
|
void *arg);
|
|
static void hrt_call_enter(struct hrt_call *entry);
|
|
static void hrt_call_reschedule(void);
|
|
static void hrt_call_invoke(void);
|
|
|
|
|
|
/**
|
|
* Initialize the timer we are going to use.
|
|
*/
|
|
static void hrt_tim_init(void)
|
|
{
|
|
/* FTM clock should be configured in s32k1xx_periphclocks.c */
|
|
|
|
/* claim our interrupt vector */
|
|
|
|
irq_attach(HRT_TIMER_VECTOR, hrt_tim_isr, NULL);
|
|
|
|
/* Reset the timer counter */
|
|
|
|
putreg32(0, S32K3XX_STM0_CNT);
|
|
|
|
/* Timer enable and prescaler 8 */
|
|
|
|
putreg32(STM_CR_TEN | STM_CR_CPS(8), S32K3XX_STM0_CR);
|
|
|
|
/* Channel 0 HRT */
|
|
|
|
putreg32(1000, S32K3XX_STM0_CMP0);
|
|
|
|
putreg32(STM_CCR_CEN, S32K3XX_STM0_CCR0);
|
|
|
|
/* enable interrupts */
|
|
up_enable_irq(HRT_TIMER_VECTOR);
|
|
}
|
|
|
|
/**
|
|
* Handle the compare interrupt by calling the callout dispatcher
|
|
* and then re-scheduling the next deadline.
|
|
*/
|
|
static int
|
|
hrt_tim_isr(int irq, void *context, void *arg)
|
|
{
|
|
/* grab the timer for latency tracking purposes */
|
|
|
|
latency_actual = getreg32(S32K3XX_STM0_CNT);
|
|
|
|
/* copy interrupt status */
|
|
uint32_t status = getreg32(S32K3XX_STM0_CIR0);
|
|
|
|
/* was this a timer tick? */
|
|
if (status & STATUS_HRT) {
|
|
putreg32(STATUS_HRT, S32K3XX_STM0_CIR0);
|
|
|
|
/* do latency calculations */
|
|
hrt_latency_update();
|
|
|
|
/* run any callouts that have met their deadline */
|
|
hrt_call_invoke();
|
|
|
|
/* and schedule the next interrupt */
|
|
hrt_call_reschedule();
|
|
}
|
|
|
|
return OK;
|
|
}
|
|
|
|
/**
|
|
* Fetch a never-wrapping absolute time value in microseconds from
|
|
* some arbitrary epoch shortly after system start.
|
|
*/
|
|
hrt_abstime
|
|
hrt_absolute_time(void)
|
|
{
|
|
hrt_abstime abstime = 0;
|
|
|
|
uint32_t count;
|
|
irqstate_t flags;
|
|
|
|
/*
|
|
* Counter state. Marked volatile as they may change
|
|
* inside this routine but outside the irqsave/restore
|
|
* pair. Discourage the compiler from moving loads/stores
|
|
* to these outside of the protected range.
|
|
*/
|
|
static volatile hrt_abstime base_time;
|
|
static volatile uint32_t last_count;
|
|
|
|
/* prevent re-entry */
|
|
flags = px4_enter_critical_section();
|
|
|
|
/* get the current counter value */
|
|
count = getreg32(S32K3XX_STM0_CNT);
|
|
|
|
/*
|
|
* Determine whether the counter has wrapped since the
|
|
* last time we're called.
|
|
*
|
|
* This simple test is sufficient due to the guarantee that
|
|
* we are always called at least once per counter period.
|
|
*/
|
|
if (count < last_count) {
|
|
base_time += HRT_COUNTER_PERIOD;
|
|
}
|
|
|
|
/* save the count for next time */
|
|
last_count = count;
|
|
|
|
/* compute the current time */
|
|
abstime = HRT_COUNTER_SCALE(base_time + count);
|
|
|
|
px4_leave_critical_section(flags);
|
|
|
|
return abstime;
|
|
}
|
|
|
|
/**
|
|
* Store the absolute time in an interrupt-safe fashion
|
|
*/
|
|
void
|
|
hrt_store_absolute_time(volatile hrt_abstime *t)
|
|
{
|
|
irqstate_t flags = px4_enter_critical_section();
|
|
*t = hrt_absolute_time();
|
|
px4_leave_critical_section(flags);
|
|
}
|
|
|
|
/**
|
|
* Initialize the high-resolution timing module.
|
|
*/
|
|
void
|
|
hrt_init(void)
|
|
{
|
|
sq_init(&callout_queue);
|
|
hrt_tim_init();
|
|
}
|
|
|
|
/**
|
|
* Call callout(arg) after interval has elapsed.
|
|
*/
|
|
void
|
|
hrt_call_after(struct hrt_call *entry, hrt_abstime delay, hrt_callout callout, void *arg)
|
|
{
|
|
hrt_call_internal(entry,
|
|
hrt_absolute_time() + delay,
|
|
0,
|
|
callout,
|
|
arg);
|
|
}
|
|
|
|
/**
|
|
* Call callout(arg) at calltime.
|
|
*/
|
|
void
|
|
hrt_call_at(struct hrt_call *entry, hrt_abstime calltime, hrt_callout callout, void *arg)
|
|
{
|
|
hrt_call_internal(entry, calltime, 0, callout, arg);
|
|
}
|
|
|
|
/**
|
|
* Call callout(arg) every period.
|
|
*/
|
|
void
|
|
hrt_call_every(struct hrt_call *entry, hrt_abstime delay, hrt_abstime interval, hrt_callout callout, void *arg)
|
|
{
|
|
hrt_call_internal(entry,
|
|
hrt_absolute_time() + delay,
|
|
interval,
|
|
callout,
|
|
arg);
|
|
}
|
|
|
|
static void
|
|
hrt_call_internal(struct hrt_call *entry, hrt_abstime deadline, hrt_abstime interval, hrt_callout callout, void *arg)
|
|
{
|
|
irqstate_t flags = px4_enter_critical_section();
|
|
|
|
/* if the entry is currently queued, remove it */
|
|
/* note that we are using a potentially uninitialized
|
|
entry->link here, but it is safe as sq_rem() doesn't
|
|
dereference the passed node unless it is found in the
|
|
list. So we potentially waste a bit of time searching the
|
|
queue for the uninitialized entry->link but we don't do
|
|
anything actually unsafe.
|
|
*/
|
|
if (entry->deadline != 0) {
|
|
sq_rem(&entry->link, &callout_queue);
|
|
}
|
|
|
|
entry->deadline = deadline;
|
|
entry->period = interval;
|
|
entry->callout = callout;
|
|
entry->arg = arg;
|
|
|
|
hrt_call_enter(entry);
|
|
|
|
px4_leave_critical_section(flags);
|
|
}
|
|
|
|
/**
|
|
* If this returns true, the call has been invoked and removed from the callout list.
|
|
*
|
|
* Always returns false for repeating callouts.
|
|
*/
|
|
bool
|
|
hrt_called(struct hrt_call *entry)
|
|
{
|
|
return (entry->deadline == 0);
|
|
}
|
|
|
|
/**
|
|
* Remove the entry from the callout list.
|
|
*/
|
|
void
|
|
hrt_cancel(struct hrt_call *entry)
|
|
{
|
|
irqstate_t flags = px4_enter_critical_section();
|
|
|
|
sq_rem(&entry->link, &callout_queue);
|
|
entry->deadline = 0;
|
|
|
|
/* if this is a periodic call being removed by the callout, prevent it from
|
|
* being re-entered when the callout returns.
|
|
*/
|
|
entry->period = 0;
|
|
|
|
px4_leave_critical_section(flags);
|
|
}
|
|
|
|
static void
|
|
hrt_call_enter(struct hrt_call *entry)
|
|
{
|
|
struct hrt_call *call, *next;
|
|
|
|
call = (struct hrt_call *)sq_peek(&callout_queue);
|
|
|
|
if ((call == NULL) || (entry->deadline < call->deadline)) {
|
|
sq_addfirst(&entry->link, &callout_queue);
|
|
hrtinfo("call enter at head, reschedule\n");
|
|
/* we changed the next deadline, reschedule the timer event */
|
|
hrt_call_reschedule();
|
|
|
|
} else {
|
|
do {
|
|
next = (struct hrt_call *)sq_next(&call->link);
|
|
|
|
if ((next == NULL) || (entry->deadline < next->deadline)) {
|
|
hrtinfo("call enter after head\n");
|
|
sq_addafter(&call->link, &entry->link, &callout_queue);
|
|
break;
|
|
}
|
|
} while ((call = next) != NULL);
|
|
}
|
|
|
|
hrtinfo("scheduled\n");
|
|
}
|
|
|
|
static void
|
|
hrt_call_invoke(void)
|
|
{
|
|
struct hrt_call *call;
|
|
hrt_abstime deadline;
|
|
|
|
while (true) {
|
|
/* get the current time */
|
|
hrt_abstime now = hrt_absolute_time();
|
|
|
|
call = (struct hrt_call *)sq_peek(&callout_queue);
|
|
|
|
if (call == NULL) {
|
|
break;
|
|
}
|
|
|
|
if (call->deadline > now) {
|
|
break;
|
|
}
|
|
|
|
sq_rem(&call->link, &callout_queue);
|
|
hrtinfo("call pop\n");
|
|
|
|
/* save the intended deadline for periodic calls */
|
|
deadline = call->deadline;
|
|
|
|
/* zero the deadline, as the call has occurred */
|
|
call->deadline = 0;
|
|
|
|
/* invoke the callout (if there is one) */
|
|
if (call->callout) {
|
|
hrtinfo("call %p: %p(%p)\n", call, call->callout, call->arg);
|
|
call->callout(call->arg);
|
|
}
|
|
|
|
/* if the callout has a non-zero period, it has to be re-entered */
|
|
if (call->period != 0) {
|
|
// re-check call->deadline to allow for
|
|
// callouts to re-schedule themselves
|
|
// using hrt_call_delay()
|
|
if (call->deadline <= now) {
|
|
call->deadline = deadline + call->period;
|
|
}
|
|
|
|
hrt_call_enter(call);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Reschedule the next timer interrupt.
|
|
*
|
|
* This routine must be called with interrupts disabled.
|
|
*/
|
|
static void
|
|
hrt_call_reschedule()
|
|
{
|
|
hrt_abstime now = hrt_absolute_time();
|
|
struct hrt_call *next = (struct hrt_call *)sq_peek(&callout_queue);
|
|
hrt_abstime deadline = now + HRT_INTERVAL_MAX;
|
|
|
|
/*
|
|
* Determine what the next deadline will be.
|
|
*
|
|
* Note that we ensure that this will be within the counter
|
|
* period, so that when we truncate all but the low 16 bits
|
|
* the next time the compare matches it will be the deadline
|
|
* we want.
|
|
*
|
|
* It is important for accurate timekeeping that the compare
|
|
* interrupt fires sufficiently often that the base_time update in
|
|
* hrt_absolute_time runs at least once per timer period.
|
|
*/
|
|
if (next != NULL) {
|
|
hrtinfo("entry in queue\n");
|
|
|
|
if (next->deadline <= (now + HRT_INTERVAL_MIN)) {
|
|
hrtinfo("pre-expired\n");
|
|
/* set a minimal deadline so that we call ASAP */
|
|
deadline = now + HRT_INTERVAL_MIN;
|
|
|
|
} else if (next->deadline < deadline) {
|
|
hrtinfo("due soon\n");
|
|
deadline = next->deadline;
|
|
}
|
|
}
|
|
|
|
hrtinfo("schedule for %u at %u\n", (unsigned)(deadline & 0xffffffff), (unsigned)(now & 0xffffffff));
|
|
|
|
/* set the new compare value and remember it for latency tracking */
|
|
latency_baseline = deadline & 0xffffffff;
|
|
|
|
putreg32(latency_baseline, S32K3XX_STM0_CMP0);
|
|
}
|
|
|
|
static void
|
|
hrt_latency_update(void)
|
|
{
|
|
uint32_t latency = latency_actual - latency_baseline;
|
|
unsigned index;
|
|
|
|
/* bounded buckets */
|
|
for (index = 0; index < LATENCY_BUCKET_COUNT; index++) {
|
|
if (latency <= latency_buckets[index]) {
|
|
latency_counters[index]++;
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* catch-all at the end */
|
|
latency_counters[index]++;
|
|
}
|
|
|
|
void
|
|
hrt_call_init(struct hrt_call *entry)
|
|
{
|
|
memset(entry, 0, sizeof(*entry));
|
|
}
|
|
|
|
void
|
|
hrt_call_delay(struct hrt_call *entry, hrt_abstime delay)
|
|
{
|
|
entry->deadline = hrt_absolute_time() + delay;
|
|
}
|
|
|
|
#endif /* HRT_TIMER */
|