24 #include "rcl/error_handling.h"
25 #include "rcutils/logging_macros.h"
26 #include "rcutils/stdatomic_helper.h"
27 #include "rcutils/time.h"
28 #include "tracetools/tracetools.h"
40 atomic_uintptr_t callback;
43 atomic_int_least64_t period;
45 atomic_int_least64_t last_call_time;
47 atomic_int_least64_t next_call_time;
49 atomic_int_least64_t time_credit;
66 void _rcl_timer_time_jump(
79 RCUTILS_LOG_ERROR_NAMED(ROS_PACKAGE_NAME,
"Failed to get current time in jump callback");
88 const int64_t next_call_time = rcutils_atomic_load_int64_t(&impl->next_call_time);
89 rcutils_atomic_store(&impl->time_credit, next_call_time - now);
94 RCUTILS_LOG_ERROR_NAMED(ROS_PACKAGE_NAME,
"Failed to get current time in jump callback");
97 const int64_t last_call_time = rcutils_atomic_load_int64_t(&impl->last_call_time);
98 const int64_t next_call_time = rcutils_atomic_load_int64_t(&impl->next_call_time);
99 const int64_t period = rcutils_atomic_load_int64_t(&impl->period);
108 int64_t time_credit = rcutils_atomic_exchange_int64_t(&impl->time_credit, 0);
111 rcutils_atomic_store(&impl->next_call_time, now - time_credit + period);
112 rcutils_atomic_store(&impl->last_call_time, now - time_credit);
114 }
else if (next_call_time <= now) {
117 RCUTILS_LOG_ERROR_NAMED(
118 ROS_PACKAGE_NAME,
"Failed to get trigger guard condition in jump callback");
120 }
else if (now < last_call_time) {
123 rcutils_atomic_store(&impl->next_call_time, now + period);
124 rcutils_atomic_store(&impl->last_call_time, now);
144 RCL_SET_ERROR_MSG(
"timer period must be non-negative");
147 RCUTILS_LOG_DEBUG_NAMED(
148 ROS_PACKAGE_NAME,
"Initializing timer with period: %" PRIu64
"ns", period);
150 RCL_SET_ERROR_MSG(
"timer already initialized, or memory was uninitialized");
160 impl.context = context;
168 atomic_init(&impl.callback, (uintptr_t)callback);
169 atomic_init(&impl.period, period);
170 atomic_init(&impl.time_credit, 0);
171 atomic_init(&impl.last_call_time, now);
172 atomic_init(&impl.next_call_time, now + period);
173 atomic_init(&impl.canceled, !autostart);
174 impl.allocator = allocator;
177 impl.callback_data.on_reset_callback = NULL;
178 impl.callback_data.user_data = NULL;
179 impl.callback_data.reset_counter = 0;
182 if (NULL == timer->
impl) {
185 RCUTILS_LOG_ERROR_NAMED(ROS_PACKAGE_NAME,
"Failed to fini guard condition after bad alloc");
190 RCUTILS_LOG_ERROR_NAMED(ROS_PACKAGE_NAME,
"Failed to remove callback after bad alloc");
194 RCL_SET_ERROR_MSG(
"allocating memory failed");
208 RCUTILS_LOG_ERROR_NAMED(
209 ROS_PACKAGE_NAME,
"Failed to fini guard condition after failing to add jump callback");
212 allocator.deallocate(timer->
impl, allocator.state);
219 TRACETOOLS_TRACEPOINT(rcl_timer_init, (
const void *)timer, period);
226 if (!timer || !timer->
impl) {
239 RCUTILS_LOG_ERROR_NAMED(ROS_PACKAGE_NAME,
"Failed to remove timer jump callback");
244 RCL_SET_ERROR_MSG(
"Failure to fini guard condition");
246 allocator.deallocate(timer->
impl, allocator.state);
257 *clock = timer->
impl->clock;
271 RCUTILS_LOG_DEBUG_NAMED(ROS_PACKAGE_NAME,
"Calling timer");
275 if (rcutils_atomic_load_bool(&timer->
impl->canceled)) {
276 RCL_SET_ERROR_MSG(
"timer is canceled");
285 RCL_SET_ERROR_MSG(
"clock now returned negative time point value");
289 rcutils_atomic_exchange_int64_t(&timer->
impl->last_call_time, now);
293 int64_t next_call_time = rcutils_atomic_load_int64_t(&timer->
impl->next_call_time);
294 call_info->expected_call_time = next_call_time;
295 call_info->actual_call_time = now;
296 int64_t period = rcutils_atomic_load_int64_t(&timer->
impl->period);
300 next_call_time += period;
302 if (next_call_time <= now) {
305 next_call_time = now;
308 int64_t now_ahead = now - next_call_time;
310 int64_t periods_ahead = 1 + now_ahead / period;
311 next_call_time += periods_ahead * period;
314 rcutils_atomic_store(&timer->
impl->next_call_time, next_call_time);
316 if (typed_callback != NULL) {
317 int64_t since_last_call = now - previous_ns;
318 typed_callback(timer, since_last_call);
329 int64_t time_until_next_call;
337 *is_ready = (time_until_next_call <= 0);
348 if (rcutils_atomic_load_bool(&timer->
impl->canceled)) {
353 rcutils_atomic_load_int64_t(&timer->
impl->next_call_time);
363 if (rcutils_atomic_load_bool(&timer->
impl->canceled)) {
371 *time_until_next_call =
372 rcutils_atomic_load_int64_t(&timer->
impl->next_call_time) - now;
389 *time_since_last_call =
390 now - rcutils_atomic_load_int64_t(&timer->
impl->last_call_time);
400 *period = rcutils_atomic_load_int64_t(&timer->
impl->period);
412 *old_period = rcutils_atomic_exchange_int64_t(&timer->
impl->period, new_period);
413 RCUTILS_LOG_DEBUG_NAMED(
414 ROS_PACKAGE_NAME,
"Updated timer period from '%" PRIu64
"ns' to '%" PRIu64
"ns'",
415 *old_period, new_period);
422 RCL_CHECK_ARGUMENT_FOR_NULL(timer, NULL);
423 RCL_CHECK_FOR_NULL_WITH_MSG(timer->
impl,
"timer is invalid",
return NULL);
430 RCUTILS_LOG_DEBUG_NAMED(ROS_PACKAGE_NAME,
"Updating timer callback");
431 RCL_CHECK_ARGUMENT_FOR_NULL(timer, NULL);
432 RCL_CHECK_FOR_NULL_WITH_MSG(timer->
impl,
"timer is invalid",
return NULL);
434 &timer->
impl->callback, (uintptr_t)new_callback);
445 rcutils_atomic_store(&timer->
impl->canceled,
true);
446 RCUTILS_LOG_DEBUG_NAMED(ROS_PACKAGE_NAME,
"Timer canceled");
456 *is_canceled = rcutils_atomic_load_bool(&timer->
impl->canceled);
472 int64_t period = rcutils_atomic_load_int64_t(&timer->
impl->period);
473 rcutils_atomic_store(&timer->
impl->next_call_time, now + period);
474 rcutils_atomic_store(&timer->
impl->canceled,
false);
479 if (cb_data->on_reset_callback) {
480 cb_data->on_reset_callback(cb_data->user_data, 1);
482 cb_data->reset_counter++;
486 RCUTILS_LOG_ERROR_NAMED(ROS_PACKAGE_NAME,
"Failed to trigger timer guard condition");
488 RCUTILS_LOG_DEBUG_NAMED(ROS_PACKAGE_NAME,
"Timer successfully reset");
495 RCL_CHECK_ARGUMENT_FOR_NULL(timer, NULL);
496 RCL_CHECK_FOR_NULL_WITH_MSG(timer->
impl,
"timer is invalid",
return NULL);
497 return &timer->
impl->allocator;
503 if (NULL == timer || NULL == timer->
impl || NULL == timer->
impl->guard_condition.
impl) {
506 return &timer->
impl->guard_condition;
512 rcl_event_callback_t on_reset_callback,
513 const void * user_data)
519 if (on_reset_callback) {
520 cb_data->on_reset_callback = on_reset_callback;
521 cb_data->user_data = user_data;
522 if (cb_data->reset_counter) {
523 cb_data->on_reset_callback(user_data, cb_data->reset_counter);
524 cb_data->reset_counter = 0;
527 cb_data->on_reset_callback = NULL;
528 cb_data->user_data = NULL;
#define RCL_CHECK_ALLOCATOR_WITH_MSG(allocator, msg, fail_statement)
Check that the given allocator is initialized, or fail with a message.
rcutils_allocator_t rcl_allocator_t
Encapsulation of an allocator.
RCL_PUBLIC RCL_WARN_UNUSED rcl_guard_condition_t rcl_get_zero_initialized_guard_condition(void)
Return a rcl_guard_condition_t struct with members set to NULL.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_trigger_guard_condition(rcl_guard_condition_t *guard_condition)
Trigger a rcl guard condition.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_guard_condition_init(rcl_guard_condition_t *guard_condition, rcl_context_t *context, const rcl_guard_condition_options_t options)
Initialize a rcl guard_condition.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_guard_condition_fini(rcl_guard_condition_t *guard_condition)
Finalize a rcl_guard_condition_t.
RCL_PUBLIC RCL_WARN_UNUSED rcl_guard_condition_options_t rcl_guard_condition_get_default_options(void)
Return the default options in a rcl_guard_condition_options_t struct.
Encapsulation of a time source.
rcl_clock_type_t type
Clock type.
Encapsulates the non-global state of an init/shutdown cycle.
rcl_duration_value_t nanoseconds
Duration in nanoseconds and its source.
Options available for a rcl guard condition.
Handle for a rcl guard condition.
rcl_guard_condition_impl_t * impl
Pointer to the guard condition implementation.
Describe the prerequisites for calling a time jump callback.
rcl_duration_t min_forward
bool on_clock_change
True to call callback when the clock type changes.
rcl_duration_t min_backward
Struct to describe a jump in time.
rcl_clock_change_t clock_change
Indicate whether or not the source of time changed.
Structure which encapsulates timer information when called.
Structure which encapsulates the on reset callback data.
Structure which encapsulates a ROS Timer.
rcl_timer_impl_t * impl
Private implementation pointer.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_clock_get_now(rcl_clock_t *clock, rcl_time_point_value_t *time_point_value)
Fill the time point value with the current value of the associated clock.
@ RCL_ROS_TIME_DEACTIVATED
The source switched to SYSTEM_TIME from ROS_TIME.
@ RCL_ROS_TIME_ACTIVATED
The source switched to ROS_TIME from SYSTEM_TIME.
rcutils_time_point_value_t rcl_time_point_value_t
A single point in time, measured in nanoseconds since the Unix epoch.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_clock_add_jump_callback(rcl_clock_t *clock, rcl_jump_threshold_t threshold, rcl_jump_callback_t callback, void *user_data)
Add a callback to be called when a time jump exceeds a threshold.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_clock_remove_jump_callback(rcl_clock_t *clock, rcl_jump_callback_t callback, void *user_data)
Remove a previously added time jump callback.
@ RCL_ROS_TIME
Use ROS time.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_timer_get_next_call_time(const rcl_timer_t *timer, int64_t *next_call_time)
Retrieve the time when the next call to rcl_timer_call() shall occur.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_timer_get_time_since_last_call(const rcl_timer_t *timer, int64_t *time_since_last_call)
Retrieve the time since the previous call to rcl_timer_call() occurred.
void(* rcl_timer_callback_t)(rcl_timer_t *, int64_t)
User callback signature for timers.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_timer_get_time_until_next_call(const rcl_timer_t *timer, int64_t *time_until_next_call)
Calculate and retrieve the time until the next call in nanoseconds.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_timer_call(rcl_timer_t *timer)
Call the timer's callback and set the last call time.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_timer_reset(rcl_timer_t *timer)
Reset a timer.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_timer_get_period(const rcl_timer_t *timer, int64_t *period)
Retrieve the period of the timer.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_timer_set_on_reset_callback(const rcl_timer_t *timer, rcl_event_callback_t on_reset_callback, const void *user_data)
Set the on reset callback function for the timer.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_timer_exchange_period(const rcl_timer_t *timer, int64_t new_period, int64_t *old_period)
Exchange the period of the timer and return the previous period.
RCL_PUBLIC RCL_WARN_UNUSED rcl_timer_callback_t rcl_timer_get_callback(const rcl_timer_t *timer)
Return the current timer callback.
RCL_PUBLIC RCL_WARN_UNUSED rcl_timer_callback_t rcl_timer_exchange_callback(rcl_timer_t *timer, const rcl_timer_callback_t new_callback)
Exchange the current timer callback and return the current callback.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_timer_init2(rcl_timer_t *timer, rcl_clock_t *clock, rcl_context_t *context, int64_t period, const rcl_timer_callback_t callback, rcl_allocator_t allocator, bool autostart)
Initialize a timer.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_timer_cancel(rcl_timer_t *timer)
Cancel a timer.
RCL_PUBLIC RCL_WARN_UNUSED rcl_guard_condition_t * rcl_timer_get_guard_condition(const rcl_timer_t *timer)
Retrieve a guard condition used by the timer to wake the waitset when using ROSTime.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_timer_clock(const rcl_timer_t *timer, rcl_clock_t **clock)
Retrieve the clock of the timer.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_timer_call_with_info(rcl_timer_t *timer, rcl_timer_call_info_t *call_info)
Same as rcl_timer_call() except that it also retrieves the actual and expected call time.
RCL_PUBLIC RCL_WARN_UNUSED rcl_timer_t rcl_get_zero_initialized_timer(void)
Return a zero initialized timer.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_timer_fini(rcl_timer_t *timer)
Finalize a timer.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_timer_is_canceled(const rcl_timer_t *timer, bool *is_canceled)
Retrieve the canceled state of a timer.
RCL_PUBLIC RCL_WARN_UNUSED rcl_ret_t rcl_timer_is_ready(const rcl_timer_t *timer, bool *is_ready)
Calculates whether or not the timer should be called.
RCL_PUBLIC RCL_WARN_UNUSED const rcl_allocator_t * rcl_timer_get_allocator(const rcl_timer_t *timer)
Return the allocator for the timer.
#define RCL_RET_TIMER_INVALID
Invalid rcl_timer_t given return code.
#define RCL_RET_ALREADY_INIT
rcl_init() already called return code.
#define RCL_RET_OK
Success return code.
#define RCL_RET_BAD_ALLOC
Failed to allocate memory return code.
#define RCL_RET_INVALID_ARGUMENT
Invalid argument return code.
#define RCL_RET_ERROR
Unspecified error return code.
#define RCL_RET_TIMER_CANCELED
Given timer was canceled return code.
rmw_ret_t rcl_ret_t
The type that holds an rcl return code.