rcl/rcl_lifecycle/test/test_transition_map.cpp
Chris Lalancette 885fa21a89 Make sure to check the return value of rcl APIs. (#838)
This is just a further check to ensure the test is correct,
and also gets rid of a slew of dead store warnings from
clang static analysis.

Signed-off-by: Chris Lalancette <clalancette@openrobotics.org>
2020-11-02 16:22:11 +01:00

164 lines
5.9 KiB
C++

// Copyright 2016 Open Source Robotics Foundation, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// testing default transition sequence.
// This test requires that the transitions are set
// as depicted in design.ros2.org
#include <gtest/gtest.h>
#include "rcl/error_handling.h"
#include "rcl_lifecycle/rcl_lifecycle.h"
#include "rcl_lifecycle/transition_map.h"
class TestTransitionMap : public ::testing::Test
{
protected:
void SetUp()
{
}
void TearDown()
{
}
};
static void * bad_malloc(size_t, void *)
{
return nullptr;
}
static void * bad_realloc(void *, size_t, void *)
{
return nullptr;
}
TEST_F(TestTransitionMap, zero_initialized) {
rcl_lifecycle_transition_map_t transition_map =
rcl_lifecycle_get_zero_initialized_transition_map();
EXPECT_EQ(RCL_RET_ERROR, rcl_lifecycle_transition_map_is_initialized(&transition_map));
rcutils_reset_error();
rcl_allocator_t allocator = rcl_get_default_allocator();
EXPECT_EQ(RCL_RET_OK, rcl_lifecycle_transition_map_fini(&transition_map, &allocator));
}
TEST_F(TestTransitionMap, initialized) {
rcl_lifecycle_transition_map_t transition_map =
rcl_lifecycle_get_zero_initialized_transition_map();
rcl_allocator_t allocator = rcl_get_default_allocator();
rcl_lifecycle_state_t state0 = {"my_state_0", 0, NULL, 0};
rcl_ret_t ret = rcl_lifecycle_register_state(&transition_map, state0, &allocator);
EXPECT_EQ(RCL_RET_OK, ret);
EXPECT_EQ(RCL_RET_OK, rcl_lifecycle_transition_map_is_initialized(&transition_map));
ret = rcl_lifecycle_register_state(&transition_map, state0, &allocator);
EXPECT_EQ(RCL_RET_ERROR, ret);
rcutils_reset_error();
rcl_lifecycle_state_t state1 = {"my_state_1", 1, NULL, 0};
ret = rcl_lifecycle_register_state(&transition_map, state1, &allocator);
ASSERT_EQ(RCL_RET_OK, ret);
rcl_lifecycle_state_t unregistered = {"my_state_2", 2, NULL, 0};
rcl_allocator_t bad_allocator = rcl_get_default_allocator();
bad_allocator.allocate = bad_malloc;
bad_allocator.reallocate = bad_realloc;
rcl_lifecycle_state_t * original_ptr = transition_map.states;
size_t original_size = transition_map.states_size;
ret = rcl_lifecycle_register_state(&transition_map, unregistered, &bad_allocator);
EXPECT_EQ(RCL_RET_ERROR, ret);
rcutils_reset_error();
EXPECT_EQ(transition_map.states, original_ptr);
EXPECT_EQ(original_size, transition_map.states_size);
rcl_lifecycle_state_t * start_state =
rcl_lifecycle_get_state(&transition_map, state0.id);
rcl_lifecycle_state_t * goal_state =
rcl_lifecycle_get_state(&transition_map, state1.id);
EXPECT_EQ(0u, start_state->id);
EXPECT_EQ(1u, goal_state->id);
rcl_lifecycle_transition_t transition01 = {"from0to1", 0,
start_state, goal_state};
ret = rcl_lifecycle_register_transition(
&transition_map, transition01, &allocator);
EXPECT_EQ(RCL_RET_OK, ret);
EXPECT_EQ(1u, transition_map.transitions_size);
rcl_lifecycle_transition_t transition10 = {"from1to0", 1,
goal_state, start_state};
ret = rcl_lifecycle_register_transition(
&transition_map, transition10, &allocator);
EXPECT_EQ(RCL_RET_OK, ret);
EXPECT_EQ(2u, transition_map.transitions_size);
rcl_lifecycle_transition_t * old_transitions_ptr = transition_map.transitions;
original_size = transition_map.transitions_size;
rcl_lifecycle_transition_t transition_bad1 = {"from0tobad", 2,
start_state, &unregistered};
ret = rcl_lifecycle_register_transition(
&transition_map, transition_bad1, &allocator);
EXPECT_EQ(RCL_RET_ERROR, ret);
rcutils_reset_error();
EXPECT_EQ(old_transitions_ptr, transition_map.transitions);
EXPECT_EQ(original_size, transition_map.transitions_size);
old_transitions_ptr = transition_map.transitions;
original_size = transition_map.transitions_size;
rcl_lifecycle_transition_t transition_bad2 = {"frombadto1", 3,
&unregistered, goal_state};
ret = rcl_lifecycle_register_transition(
&transition_map, transition_bad2, &allocator);
EXPECT_EQ(RCL_RET_ERROR, ret);
rcutils_reset_error();
EXPECT_EQ(old_transitions_ptr, transition_map.transitions);
EXPECT_EQ(original_size, transition_map.transitions_size);
old_transitions_ptr = transition_map.transitions;
original_size = transition_map.transitions_size;
rcl_lifecycle_transition_t transition00 = {"from0to0", 4,
start_state, start_state};
ret = rcl_lifecycle_register_transition(
&transition_map, transition00, &bad_allocator);
EXPECT_EQ(RCL_RET_BAD_ALLOC, ret);
rcutils_reset_error();
EXPECT_EQ(old_transitions_ptr, transition_map.transitions);
EXPECT_EQ(original_size, transition_map.transitions_size);
const rcl_lifecycle_transition_t * trans =
rcl_lifecycle_get_transition_by_id(start_state, 0);
EXPECT_EQ(0u, trans->id);
trans = rcl_lifecycle_get_transitions(&transition_map, 0);
EXPECT_EQ(0u, trans->id);
trans = rcl_lifecycle_get_transition_by_label(start_state, "from0to1");
EXPECT_EQ(0u, trans->id);
trans = rcl_lifecycle_get_transition_by_id(goal_state, 1);
EXPECT_EQ(1u, trans->id);
trans = rcl_lifecycle_get_transitions(&transition_map, 1);
EXPECT_EQ(1u, trans->id);
trans = rcl_lifecycle_get_transition_by_label(goal_state, "from1to0");
EXPECT_EQ(1u, trans->id);
// Check nonexistent transition
trans = rcl_lifecycle_get_transitions(&transition_map, 2);
EXPECT_EQ(nullptr, trans);
rcutils_reset_error();
EXPECT_EQ(RCL_RET_OK, rcl_lifecycle_transition_map_fini(&transition_map, &allocator));
}