diff --git a/rclcpp/include/rclcpp/executor.hpp b/rclcpp/include/rclcpp/executor.hpp index b062915da8..cbe7e55397 100644 --- a/rclcpp/include/rclcpp/executor.hpp +++ b/rclcpp/include/rclcpp/executor.hpp @@ -35,6 +35,7 @@ #include "rclcpp/node_interfaces/node_base_interface.hpp" #include "rclcpp/utilities.hpp" #include "rclcpp/visibility_control.hpp" +#include "rclcpp/scope_exit.hpp" namespace rclcpp { @@ -243,9 +244,14 @@ class Executor } std::chrono::nanoseconds timeout_left = timeout_ns; - while (rclcpp::ok(this->context_)) { + if (spinning.exchange(true)) { + throw std::runtime_error("spin_until_future_complete() called while already spinning"); + } + RCLCPP_SCOPE_EXIT(this->spinning.store(false); ); + while (rclcpp::ok(this->context_) && spinning.load()) { // Do one item of work. - spin_once(timeout_left); + spin_once_impl(timeout_left); + // Check if the future is set, return SUCCESS if it is. status = future.wait_for(std::chrono::seconds(0)); if (status == std::future_status::ready) { @@ -368,6 +374,10 @@ class Executor private: RCLCPP_DISABLE_COPY(Executor) + RCLCPP_PUBLIC + void + spin_once_impl(std::chrono::nanoseconds timeout); + std::list weak_nodes_; std::list guard_conditions_; }; diff --git a/rclcpp/src/rclcpp/executor.cpp b/rclcpp/src/rclcpp/executor.cpp index 25ef632ae5..db58fff702 100644 --- a/rclcpp/src/rclcpp/executor.cpp +++ b/rclcpp/src/rclcpp/executor.cpp @@ -244,6 +244,15 @@ Executor::spin_some(std::chrono::nanoseconds max_duration) } } +void +Executor::spin_once_impl(std::chrono::nanoseconds timeout) +{ + AnyExecutable any_exec; + if (get_next_executable(any_exec, timeout)) { + execute_any_executable(any_exec); + } +} + void Executor::spin_once(std::chrono::nanoseconds timeout) { @@ -251,10 +260,7 @@ Executor::spin_once(std::chrono::nanoseconds timeout) throw std::runtime_error("spin_once() called while already spinning"); } RCLCPP_SCOPE_EXIT(this->spinning.store(false); ); - AnyExecutable any_exec; - if (get_next_executable(any_exec, timeout)) { - execute_any_executable(any_exec); - } + spin_once_impl(timeout); } void