Skip to content

Commit

Permalink
protect state_handle_ with mutex lock.
Browse files Browse the repository at this point in the history
Signed-off-by: Tomoya Fujita <Tomoya.Fujita@sony.com>
  • Loading branch information
fujitatomoya committed Aug 25, 2021
1 parent 8a568d6 commit a9de1b6
Show file tree
Hide file tree
Showing 3 changed files with 52 additions and 37 deletions.
2 changes: 2 additions & 0 deletions rclcpp_lifecycle/include/rclcpp_lifecycle/state.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
#ifndef RCLCPP_LIFECYCLE__STATE_HPP_
#define RCLCPP_LIFECYCLE__STATE_HPP_

#include <mutex>
#include <string>

#include "rclcpp_lifecycle/visibility_control.h"
Expand Down Expand Up @@ -93,6 +94,7 @@ class State

bool owns_rcl_state_handle_;

mutable std::recursive_mutex state_handle_mutex_;
rcl_lifecycle_state_t * state_handle_;
};

Expand Down
17 changes: 8 additions & 9 deletions rclcpp_lifecycle/src/lifecycle_node_interface_impl.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -182,9 +182,9 @@ class LifecycleNode::LifecycleNodeInterfaceImpl
std::lock_guard<std::recursive_mutex> lock(state_machine_mutex_);
srv_get_available_transitions_ =
std::make_shared<rclcpp::Service<GetAvailableTransitionsSrv>>(
node_base_interface_->get_shared_rcl_node_handle(),
&state_machine_.com_interface.srv_get_available_transitions,
any_cb);
node_base_interface_->get_shared_rcl_node_handle(),
&state_machine_.com_interface.srv_get_available_transitions,
any_cb);
}
node_services_interface_->add_service(
std::dynamic_pointer_cast<rclcpp::ServiceBase>(srv_get_available_transitions_),
Expand All @@ -202,9 +202,9 @@ class LifecycleNode::LifecycleNodeInterfaceImpl
std::lock_guard<std::recursive_mutex> lock(state_machine_mutex_);
srv_get_transition_graph_ =
std::make_shared<rclcpp::Service<GetAvailableTransitionsSrv>>(
node_base_interface_->get_shared_rcl_node_handle(),
&state_machine_.com_interface.srv_get_transition_graph,
any_cb);
node_base_interface_->get_shared_rcl_node_handle(),
&state_machine_.com_interface.srv_get_transition_graph,
any_cb);
}
node_services_interface_->add_service(
std::dynamic_pointer_cast<rclcpp::ServiceBase>(srv_get_transition_graph_),
Expand Down Expand Up @@ -233,8 +233,7 @@ class LifecycleNode::LifecycleNodeInterfaceImpl
{
std::lock_guard<std::recursive_mutex> lock(state_machine_mutex_);
if (rcl_lifecycle_state_machine_is_initialized(&state_machine_) != RCL_RET_OK) {
throw std::runtime_error(
"Can't get state. State machine is not initialized.");
throw std::runtime_error("Can't get state. State machine is not initialized.");
}

transition_id = req->transition.id;
Expand Down Expand Up @@ -522,7 +521,7 @@ class LifecycleNode::LifecycleNodeInterfaceImpl
const State & trigger_transition(
const char * transition_label, LifecycleNodeInterface::CallbackReturn & cb_return_code)
{
const rcl_lifecycle_transition_t *transition;
const rcl_lifecycle_transition_t * transition;
{
std::lock_guard<std::recursive_mutex> lock(state_machine_mutex_);
transition =
Expand Down
70 changes: 42 additions & 28 deletions rclcpp_lifecycle/src/state.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -44,19 +44,22 @@ State::State(
throw std::runtime_error("Lifecycle State cannot have an empty label.");
}

state_handle_ = static_cast<rcl_lifecycle_state_t *>(
allocator_.allocate(sizeof(rcl_lifecycle_state_t), allocator_.state));
if (!state_handle_) {
throw std::runtime_error("failed to allocate memory for rcl_lifecycle_state_t");
}
// zero initialize
state_handle_->id = 0;
state_handle_->label = nullptr;

auto ret = rcl_lifecycle_state_init(state_handle_, id, label.c_str(), &allocator_);
if (ret != RCL_RET_OK) {
reset();
rclcpp::exceptions::throw_from_rcl_error(ret);
{
std::lock_guard<std::recursive_mutex> lock(state_handle_mutex_);
state_handle_ = static_cast<rcl_lifecycle_state_t *>(
allocator_.allocate(sizeof(rcl_lifecycle_state_t), allocator_.state));
if (!state_handle_) {
throw std::runtime_error("failed to allocate memory for rcl_lifecycle_state_t");
}
// zero initialize
state_handle_->id = 0;
state_handle_->label = nullptr;

auto ret = rcl_lifecycle_state_init(state_handle_, id, label.c_str(), &allocator_);
if (ret != RCL_RET_OK) {
reset();
rclcpp::exceptions::throw_from_rcl_error(ret);
}
}
}

Expand All @@ -70,7 +73,11 @@ State::State(
if (!rcl_lifecycle_state_handle) {
throw std::runtime_error("rcl_lifecycle_state_handle is null");
}
state_handle_ = const_cast<rcl_lifecycle_state_t *>(rcl_lifecycle_state_handle);

{
std::lock_guard<std::recursive_mutex> lock(state_handle_mutex_);
state_handle_ = const_cast<rcl_lifecycle_state_t *>(rcl_lifecycle_state_handle);
}
}

State::State(const State & rhs)
Expand Down Expand Up @@ -101,25 +108,29 @@ State::operator=(const State & rhs)

// we don't own the handle, so we can return straight ahead
if (!owns_rcl_state_handle_) {
std::lock_guard<std::recursive_mutex> lock(state_handle_mutex_);
state_handle_ = rhs.state_handle_;
return *this;
}

// we own the handle, so we have to deep-copy the rhs object
state_handle_ = static_cast<rcl_lifecycle_state_t *>(
allocator_.allocate(sizeof(rcl_lifecycle_state_t), allocator_.state));
if (!state_handle_) {
throw std::runtime_error("failed to allocate memory for rcl_lifecycle_state_t");
}
// zero initialize
state_handle_->id = 0;
state_handle_->label = nullptr;

auto ret = rcl_lifecycle_state_init(
state_handle_, rhs.id(), rhs.label().c_str(), &allocator_);
if (ret != RCL_RET_OK) {
reset();
throw std::runtime_error("failed to duplicate label for rcl_lifecycle_state_t");
{
std::lock_guard<std::recursive_mutex> lock(state_handle_mutex_);
state_handle_ = static_cast<rcl_lifecycle_state_t *>(
allocator_.allocate(sizeof(rcl_lifecycle_state_t), allocator_.state));
if (!state_handle_) {
throw std::runtime_error("failed to allocate memory for rcl_lifecycle_state_t");
}
// zero initialize
state_handle_->id = 0;
state_handle_->label = nullptr;

auto ret = rcl_lifecycle_state_init(
state_handle_, rhs.id(), rhs.label().c_str(), &allocator_);
if (ret != RCL_RET_OK) {
reset();
throw std::runtime_error("failed to duplicate label for rcl_lifecycle_state_t");
}
}

return *this;
Expand All @@ -128,6 +139,7 @@ State::operator=(const State & rhs)
uint8_t
State::id() const
{
std::lock_guard<std::recursive_mutex> lock(state_handle_mutex_);
if (!state_handle_) {
throw std::runtime_error("Error in state! Internal state_handle is NULL.");
}
Expand All @@ -137,6 +149,7 @@ State::id() const
std::string
State::label() const
{
std::lock_guard<std::recursive_mutex> lock(state_handle_mutex_);
if (!state_handle_) {
throw std::runtime_error("Error in state! Internal state_handle is NULL.");
}
Expand All @@ -146,6 +159,7 @@ State::label() const
void
State::reset() noexcept
{
std::lock_guard<std::recursive_mutex> lock(state_handle_mutex_);
if (!owns_rcl_state_handle_) {
state_handle_ = nullptr;
}
Expand Down

0 comments on commit a9de1b6

Please sign in to comment.