mirror of
https://github.com/apache/nuttx.git
synced 2026-05-20 04:16:35 +08:00
3538d805fc
Several code comments in the hrtimer subsystem have become outdated following recent implementation changes. This patch updates them to accurately describe the current code behavior and algorithms. Signed-off-by: Chengdong Wang <wangcd91@gmail.com>
208 lines
6.2 KiB
C
208 lines
6.2 KiB
C
/****************************************************************************
|
|
* sched/hrtimer/hrtimer_cancel.c
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
* contributor license agreements. See the NOTICE file distributed with
|
|
* this work for additional information regarding copyright ownership. The
|
|
* ASF licenses this file under the Apache License, Version 2.0 (the
|
|
* "License"); you may not use this file except in compliance with the
|
|
* License. You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
* License for the specific language governing permissions and limitations
|
|
* under the License.
|
|
*
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Included Files
|
|
****************************************************************************/
|
|
|
|
#include <nuttx/config.h>
|
|
#include <nuttx/arch.h>
|
|
#include <nuttx/clock.h>
|
|
#include <errno.h>
|
|
|
|
#include "sched/sched.h"
|
|
#include "hrtimer/hrtimer.h"
|
|
|
|
/****************************************************************************
|
|
* Pre-processor Definitions
|
|
****************************************************************************/
|
|
|
|
/* Delay used while waiting for a running hrtimer callback to complete */
|
|
|
|
#define HRTIMER_CANCEL_SYNC_DELAY_MS 5
|
|
|
|
/****************************************************************************
|
|
* Private Functions
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Name: hrtimer_is_active
|
|
*
|
|
* Description:
|
|
* Check whether a high-resolution timer is currently running on any CPU.
|
|
*
|
|
* Input Parameters:
|
|
* hrtimer - Pointer to the high-resolution timer to check.
|
|
*
|
|
* Returned Value:
|
|
* true - The timer is active on at least one CPU.
|
|
* false - The timer is not active.
|
|
****************************************************************************/
|
|
#ifdef CONFIG_SMP
|
|
static inline_function bool hrtimer_is_active(FAR hrtimer_t *hrtimer)
|
|
{
|
|
bool is_active = false;
|
|
|
|
for (int i = 0; i < CONFIG_SMP_NCPUS; i++)
|
|
{
|
|
if (g_hrtimer_running[i] == hrtimer)
|
|
{
|
|
is_active = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return is_active;
|
|
}
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
* Public Functions
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Name: hrtimer_cancel
|
|
*
|
|
* Description:
|
|
* Cancel a high-resolution timer.
|
|
*
|
|
* If the timer is currently armed, it will be removed from the active
|
|
* hrtimer container (tree or list) and will not be executed.
|
|
*
|
|
* If the timer callback is currently executing, the timer will be marked
|
|
* as canceled. The running callback is allowed to complete, but the timer
|
|
* will not be re-armed or executed again.
|
|
*
|
|
* If the canceled timer was the earliest (head) timer in the container,
|
|
* the expiration of the underlying hardware timer will be updated to the
|
|
* expiration time of the next earliest timer
|
|
*
|
|
* This function is non-blocking and does not wait for a running callback
|
|
* to finish.
|
|
*
|
|
* Input Parameters:
|
|
* hrtimer - Pointer to the high-resolution timer instance to cancel.
|
|
*
|
|
* Returned Value:
|
|
* OK (0) on success; a negated errno value on failure.
|
|
*
|
|
* Assumptions/Notes:
|
|
* - This function acquires the global hrtimer spinlock to protect
|
|
* the container.
|
|
* - The caller must ensure that the timer structure is not freed until
|
|
* it is guaranteed that any running callback has returned.
|
|
*
|
|
****************************************************************************/
|
|
|
|
int hrtimer_cancel(FAR hrtimer_t *hrtimer)
|
|
{
|
|
FAR hrtimer_t *first;
|
|
irqstate_t flags;
|
|
int ret = OK;
|
|
|
|
DEBUGASSERT(hrtimer != NULL);
|
|
|
|
/* Enter critical section to protect the hrtimer container */
|
|
|
|
flags = spin_lock_irqsave(&g_hrtimer_spinlock);
|
|
|
|
if (hrtimer_is_armed(hrtimer))
|
|
{
|
|
hrtimer_remove(hrtimer);
|
|
}
|
|
|
|
/* If the timer was running, increment its expiration count to prevent
|
|
* it from being re-armed by the callback.
|
|
*/
|
|
|
|
hrtimer->expired++;
|
|
|
|
/* If the canceled timer was the earliest one, update the hardware timer */
|
|
|
|
if (hrtimer_is_first(hrtimer))
|
|
{
|
|
first = hrtimer_get_first();
|
|
if (first != NULL)
|
|
{
|
|
ret = hrtimer_starttimer(first->expired);
|
|
}
|
|
}
|
|
|
|
/* Leave critical section */
|
|
|
|
spin_unlock_irqrestore(&g_hrtimer_spinlock, flags);
|
|
return ret;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Name: hrtimer_cancel_sync
|
|
*
|
|
* Description:
|
|
* Cancel a high-resolution timer and wait until it becomes inactive.
|
|
*
|
|
* - Calls hrtimer_cancel() to request timer cancellation.
|
|
* - If the timer callback is running, waits until it completes.
|
|
* - If sleeping is allowed (normal task context), yields CPU briefly
|
|
* to avoid busy-waiting.
|
|
* - Otherwise (interrupt or idle task context), spins until completion.
|
|
*
|
|
* Input Parameters:
|
|
* hrtimer - Pointer to the high-resolution timer instance to cancel.
|
|
*
|
|
* Returned Value:
|
|
* OK (0) on success; a negated errno value on failure.
|
|
*
|
|
****************************************************************************/
|
|
|
|
int hrtimer_cancel_sync(FAR hrtimer_t *hrtimer)
|
|
{
|
|
int ret = OK;
|
|
|
|
DEBUGASSERT(hrtimer != NULL);
|
|
|
|
/* Request cancellation of the timer */
|
|
|
|
ret = hrtimer_cancel(hrtimer);
|
|
if (ret < 0)
|
|
{
|
|
return ret;
|
|
}
|
|
|
|
/* Wait until all cpu finish running the timer callback.
|
|
*
|
|
* If sleeping is permitted, yield the CPU briefly to avoid
|
|
* busy-waiting. Otherwise, spin until the callback completes.
|
|
*/
|
|
#ifdef CONFIG_SMP
|
|
while (hrtimer_is_active(hrtimer))
|
|
{
|
|
if (!up_interrupt_context() &&
|
|
!is_idle_task(this_task()))
|
|
{
|
|
nxsched_msleep(HRTIMER_CANCEL_SYNC_DELAY_MS);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
return ret;
|
|
}
|