Skip to content

Commit b0b8305

Browse files
committed
Add fault injection macros and unit tests to rcl_lifecycle
Signed-off-by: Stephen Brawner <[email protected]>
1 parent 72389f8 commit b0b8305

File tree

4 files changed

+72
-1
lines changed

4 files changed

+72
-1
lines changed

rcl_lifecycle/src/rcl_lifecycle.c

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -80,6 +80,8 @@ rcl_lifecycle_state_fini(
8080
rcl_lifecycle_state_t * state,
8181
const rcl_allocator_t * allocator)
8282
{
83+
RCUTILS_CAN_RETURN_WITH_ERROR_OF(RCL_RET_ERROR);
84+
8385
if (!allocator) {
8486
RCL_SET_ERROR_MSG("can't free state, no allocator given\n");
8587
return RCL_RET_ERROR;

rcl_lifecycle/src/transition_map.c

Lines changed: 6 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -53,6 +53,7 @@ rcl_lifecycle_transition_map_fini(
5353
rcl_lifecycle_transition_map_t * transition_map,
5454
const rcutils_allocator_t * allocator)
5555
{
56+
RCUTILS_CAN_RETURN_WITH_ERROR_OF(RCL_RET_ERROR);
5657
if (!allocator) {
5758
RCL_SET_ERROR_MSG("can't free transition map, no allocator given\n");
5859
return RCL_RET_ERROR;
@@ -85,6 +86,8 @@ rcl_lifecycle_register_state(
8586
rcl_lifecycle_state_t state,
8687
const rcutils_allocator_t * allocator)
8788
{
89+
RCUTILS_CAN_RETURN_WITH_ERROR_OF(RCL_RET_ERROR);
90+
8891
if (rcl_lifecycle_get_state(transition_map, state.id) != NULL) {
8992
RCL_SET_ERROR_MSG_WITH_FORMAT_STRING("state %u is already registered\n", state.id);
9093
return RCL_RET_ERROR;
@@ -116,6 +119,9 @@ rcl_lifecycle_register_transition(
116119
rcl_lifecycle_transition_t transition,
117120
const rcutils_allocator_t * allocator)
118121
{
122+
RCUTILS_CAN_RETURN_WITH_ERROR_OF(RCL_RET_ERROR);
123+
RCUTILS_CAN_RETURN_WITH_ERROR_OF(RCL_RET_BAD_ALLOC);
124+
119125
RCUTILS_CHECK_ALLOCATOR_WITH_MSG(
120126
allocator, "invalid allocator", return RCL_RET_ERROR)
121127

rcl_lifecycle/test/test_default_state_machine.cpp

Lines changed: 14 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@
2828
#include "rcl/rcl.h"
2929

3030
#include "rcutils/logging_macros.h"
31+
#include "rcutils/testing/fault_injection.h"
3132

3233
#include "rcl_lifecycle/rcl_lifecycle.h"
3334
#include "rcl_lifecycle/default_state_machine.h"
@@ -833,3 +834,16 @@ TEST_F(TestDefaultStateMachine, default_sequence_error_unresolved) {
833834
rcl_lifecycle_state_machine_fini(&state_machine, this->node_ptr, this->allocator));
834835
}
835836
}
837+
838+
TEST_F(TestDefaultStateMachine, init_fini_maybe_fail) {
839+
RCUTILS_FAULT_INJECTION_TEST(
840+
{
841+
// If this in zero initialized outside of this loop, there is potential for a segfault
842+
// when attempting to init a second time
843+
rcl_lifecycle_state_machine_t sm = rcl_lifecycle_get_zero_initialized_state_machine();
844+
rcl_ret_t ret = rcl_lifecycle_init_default_state_machine(&sm, this->allocator);
845+
if (RCL_RET_OK == ret) {
846+
ret = rcl_lifecycle_state_machine_fini(&sm, this->node_ptr, this->allocator);
847+
}
848+
});
849+
}

rcl_lifecycle/test/test_rcl_lifecycle.cpp

Lines changed: 50 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -19,9 +19,12 @@
1919
#include <gtest/gtest.h>
2020

2121
#include "rcl_lifecycle/rcl_lifecycle.h"
22+
2223
#include "osrf_testing_tools_cpp/memory_tools/memory_tools.hpp"
2324
#include "osrf_testing_tools_cpp/scope_exit.hpp"
2425
#include "rcl/error_handling.h"
26+
#include "rcutils/testing/fault_injection.h"
27+
2528
#include "lifecycle_msgs/msg/transition_event.h"
2629
#include "lifecycle_msgs/srv/change_state.h"
2730
#include "lifecycle_msgs/srv/get_available_states.h"
@@ -337,7 +340,6 @@ TEST(TestRclLifecycle, state_machine) {
337340
ret = rcl_lifecycle_state_machine_fini(&state_machine, nullptr, &allocator);
338341
EXPECT_EQ(ret, RCL_RET_ERROR);
339342
rcutils_reset_error();
340-
std::cout << "state_machine: " << __LINE__ << std::endl;
341343
}
342344

343345
TEST(TestRclLifecycle, state_transitions) {
@@ -437,3 +439,50 @@ TEST(TestRclLifecycle, state_transitions) {
437439
ret = rcl_lifecycle_state_machine_fini(&state_machine, &node, &allocator);
438440
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
439441
}
442+
443+
TEST(TestRclLifecycle, init_fini_maybe_fail) {
444+
rcl_node_t node = rcl_get_zero_initialized_node();
445+
rcl_allocator_t allocator = rcl_get_default_allocator();
446+
rcl_context_t context = rcl_get_zero_initialized_context();
447+
rcl_node_options_t options = rcl_node_get_default_options();
448+
rcl_init_options_t init_options = rcl_get_zero_initialized_init_options();
449+
rcl_ret_t ret = rcl_init_options_init(&init_options, allocator);
450+
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
451+
452+
ret = rcl_init(0, nullptr, &init_options, &context);
453+
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
454+
455+
OSRF_TESTING_TOOLS_CPP_SCOPE_EXIT(
456+
{
457+
ASSERT_EQ(RCL_RET_OK, rcl_shutdown(&context));
458+
ASSERT_EQ(RCL_RET_OK, rcl_context_fini(&context));
459+
});
460+
461+
ret = rcl_node_init(&node, "node", "namespace", &context, &options);
462+
EXPECT_EQ(ret, RCL_RET_OK) << rcl_get_error_string().str;
463+
464+
const rosidl_message_type_support_t * pn =
465+
ROSIDL_GET_MSG_TYPE_SUPPORT(lifecycle_msgs, msg, TransitionEvent);
466+
const rosidl_service_type_support_t * cs =
467+
ROSIDL_GET_SRV_TYPE_SUPPORT(lifecycle_msgs, srv, ChangeState);
468+
const rosidl_service_type_support_t * gs =
469+
ROSIDL_GET_SRV_TYPE_SUPPORT(lifecycle_msgs, srv, GetState);
470+
const rosidl_service_type_support_t * gas =
471+
ROSIDL_GET_SRV_TYPE_SUPPORT(lifecycle_msgs, srv, GetAvailableStates);
472+
const rosidl_service_type_support_t * gat =
473+
ROSIDL_GET_SRV_TYPE_SUPPORT(lifecycle_msgs, srv, GetAvailableTransitions);
474+
const rosidl_service_type_support_t * gtg =
475+
ROSIDL_GET_SRV_TYPE_SUPPORT(lifecycle_msgs, srv, GetAvailableTransitions);
476+
477+
RCUTILS_FAULT_INJECTION_TEST(
478+
{
479+
// Init segfaults if this is not zero initialized
480+
rcl_lifecycle_state_machine_t sm = rcl_lifecycle_get_zero_initialized_state_machine();
481+
482+
ret = rcl_lifecycle_state_machine_init(
483+
&sm, &node, pn, cs, gs, gas, gat, gtg, true, &allocator);
484+
if (RCL_RET_OK == ret) {
485+
ret = rcl_lifecycle_state_machine_fini(&sm, &node, &allocator);
486+
}
487+
});
488+
}

0 commit comments

Comments
 (0)