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;
42 atomic_uint_least64_t period;
44 atomic_int_least64_t last_call_time;
46 atomic_int_least64_t next_call_time;
48 atomic_int_least64_t time_credit;
62 void _rcl_timer_time_jump(
75 RCUTILS_LOG_ERROR_NAMED(ROS_PACKAGE_NAME,
"Failed to get current time in jump callback");
84 const int64_t next_call_time = rcutils_atomic_load_int64_t(&timer->
impl->next_call_time);
85 rcutils_atomic_store(&timer->
impl->time_credit, next_call_time - now);
90 RCUTILS_LOG_ERROR_NAMED(ROS_PACKAGE_NAME,
"Failed to get current time in jump callback");
93 const int64_t last_call_time = rcutils_atomic_load_int64_t(&timer->
impl->last_call_time);
94 const int64_t next_call_time = rcutils_atomic_load_int64_t(&timer->
impl->next_call_time);
95 const int64_t period = rcutils_atomic_load_uint64_t(&timer->
impl->period);
104 int64_t time_credit = rcutils_atomic_exchange_int64_t(&timer->
impl->time_credit, 0);
107 rcutils_atomic_store(&timer->
impl->next_call_time, now - time_credit + period);
108 rcutils_atomic_store(&timer->
impl->last_call_time, now - time_credit);
110 }
else if (next_call_time <= now) {
113 RCUTILS_LOG_ERROR_NAMED(
114 ROS_PACKAGE_NAME,
"Failed to get trigger guard condition in jump callback");
116 }
else if (now < last_call_time) {
119 rcutils_atomic_store(&timer->
impl->next_call_time, now + period);
120 rcutils_atomic_store(&timer->
impl->last_call_time, now);
139 RCL_SET_ERROR_MSG(
"timer period must be non-negative");
142 RCUTILS_LOG_DEBUG_NAMED(
143 ROS_PACKAGE_NAME,
"Initializing timer with period: %" PRIu64
"ns", period);
145 RCL_SET_ERROR_MSG(
"timer already initialized, or memory was uninitialized");
155 impl.context = context;
171 RCUTILS_LOG_ERROR_NAMED(
172 ROS_PACKAGE_NAME,
"Failed to fini guard condition after failing to add jump callback");
177 atomic_init(&impl.callback, (uintptr_t)callback);
178 atomic_init(&impl.period, period);
179 atomic_init(&impl.time_credit, 0);
180 atomic_init(&impl.last_call_time, now);
181 atomic_init(&impl.next_call_time, now + period);
182 atomic_init(&impl.canceled,
false);
183 impl.allocator = allocator;
185 if (NULL == timer->
impl) {
188 RCUTILS_LOG_ERROR_NAMED(ROS_PACKAGE_NAME,
"Failed to fini guard condition after bad alloc");
192 RCUTILS_LOG_ERROR_NAMED(ROS_PACKAGE_NAME,
"Failed to remove callback after bad alloc");
195 RCL_SET_ERROR_MSG(
"allocating memory failed");
206 if (!timer || !timer->
impl) {
218 RCUTILS_LOG_ERROR_NAMED(ROS_PACKAGE_NAME,
"Failed to remove timer jump callback");
223 RCL_SET_ERROR_MSG(
"Failure to fini guard condition");
225 allocator.deallocate(timer->
impl, allocator.state);
238 *clock = timer->
impl->clock;
245 RCUTILS_LOG_DEBUG_NAMED(ROS_PACKAGE_NAME,
"Calling timer");
248 if (rcutils_atomic_load_bool(&timer->
impl->canceled)) {
249 RCL_SET_ERROR_MSG(
"timer is canceled");
258 RCL_SET_ERROR_MSG(
"clock now returned negative time point value");
262 rcutils_atomic_exchange_int64_t(&timer->
impl->last_call_time, now);
266 int64_t next_call_time = rcutils_atomic_load_int64_t(&timer->
impl->next_call_time);
267 int64_t period = rcutils_atomic_load_uint64_t(&timer->
impl->period);
271 next_call_time += period;
273 if (next_call_time < now) {
276 next_call_time = now;
279 int64_t now_ahead = now - next_call_time;
281 int64_t periods_ahead = 1 + (now_ahead - 1) / period;
282 next_call_time += periods_ahead * period;
285 rcutils_atomic_store(&timer->
impl->next_call_time, next_call_time);
287 if (typed_callback != NULL) {
288 int64_t since_last_call = now - previous_ns;
289 typed_callback(timer, since_last_call);
300 int64_t time_until_next_call;
308 *is_ready = (time_until_next_call <= 0);
318 if (rcutils_atomic_load_bool(&timer->
impl->canceled)) {
326 *time_until_next_call =
327 rcutils_atomic_load_int64_t(&timer->
impl->next_call_time) - now;
344 *time_since_last_call =
345 now - rcutils_atomic_load_int64_t(&timer->
impl->last_call_time);
355 *period = rcutils_atomic_load_uint64_t(&timer->
impl->period);
367 *old_period = rcutils_atomic_exchange_uint64_t(&timer->
impl->period, new_period);
368 RCUTILS_LOG_DEBUG_NAMED(
369 ROS_PACKAGE_NAME,
"Updated timer period from '%" PRIu64
"ns' to '%" PRIu64
"ns'",
370 *old_period, new_period);
377 RCL_CHECK_ARGUMENT_FOR_NULL(timer, NULL);
378 RCL_CHECK_FOR_NULL_WITH_MSG(timer->
impl,
"timer is invalid",
return NULL);
385 RCUTILS_LOG_DEBUG_NAMED(ROS_PACKAGE_NAME,
"Updating timer callback");
386 RCL_CHECK_ARGUMENT_FOR_NULL(timer, NULL);
387 RCL_CHECK_FOR_NULL_WITH_MSG(timer->
impl,
"timer is invalid",
return NULL);
389 &timer->
impl->callback, (uintptr_t)new_callback);
400 rcutils_atomic_store(&timer->
impl->canceled,
true);
401 RCUTILS_LOG_DEBUG_NAMED(ROS_PACKAGE_NAME,
"Timer canceled");
411 *is_canceled = rcutils_atomic_load_bool(&timer->
impl->canceled);
427 int64_t period = rcutils_atomic_load_uint64_t(&timer->
impl->period);
428 rcutils_atomic_store(&timer->
impl->next_call_time, now + period);
429 rcutils_atomic_store(&timer->
impl->canceled,
false);
432 RCUTILS_LOG_ERROR_NAMED(ROS_PACKAGE_NAME,
"Failed to trigger timer guard condition");
434 RCUTILS_LOG_DEBUG_NAMED(ROS_PACKAGE_NAME,
"Timer successfully reset");
441 RCL_CHECK_ARGUMENT_FOR_NULL(timer, NULL);
442 RCL_CHECK_FOR_NULL_WITH_MSG(timer->
impl,
"timer is invalid",
return NULL);
443 return &timer->
impl->allocator;
449 if (NULL == timer || NULL == timer->
impl || NULL == timer->
impl->guard_condition.
impl) {
452 return &timer->
impl->guard_condition;
#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 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_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_init(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)
Initialize a timer.
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_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_ret_t rcl_timer_clock(rcl_timer_t *timer, rcl_clock_t **clock)
Retrieve the clock of the timer.
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_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_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.