
* (refactor) add rcl_lifecycle package * (refactor) cleanup free calls * (fix) initialize state machine with external node handle * (dev) use external typesupport * (fix) use external typesupport * (fix) cleanup states * (fix) no pointer comparison in state machine * (test) refactor for rcl testing purpose * (test) test suite for lifecycle sequence * (dev) change to rcl_ret_t * (test) wrong transition test * (fix) dependency for isolated build * (clean) remove std_msgs as a dep * (fix) enable correct visibility control * (fix) correct test for initialization * (fix) correct visibility attributes * (dev) change default value to lifecycle_msgs * (clean) style and lifecycle prefix * (dev) cmake macro get_rcl_lifecycle_information * (cleanup) remove unused files' * (cleanup) remove callback pointer * (debug) add print state machine function * (review) address review comments * (bugfix) correct export in information.cmake * (fix) correct visibility control * (vcs) fix convertion from size_t to unsigned int * (typo) fix internal struct name * const correctness * get_available_states service * new service msgs * tes for multiple instances * (dev) return codes * initial refactor * test default sequence * refactor state machine * apply upstream changes * c++14 * disable state machine print * address review comments * uncrustify * fix comparison with unsigned warning
178 lines
5.6 KiB
C
178 lines
5.6 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.
|
|
|
|
#if __cplusplus
|
|
extern "C"
|
|
{
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include "rcl/rcl.h"
|
|
#include "rcl/error_handling.h"
|
|
|
|
#include "rcl_lifecycle/rcl_lifecycle.h"
|
|
#include "rcl_lifecycle/transition_map.h"
|
|
|
|
#include "com_interface.h" // NOLINT
|
|
#include "default_state_machine.h" // NOLINT
|
|
#include "states.h" // NOLINT
|
|
|
|
// get zero initialized state machine here
|
|
rcl_lifecycle_state_machine_t
|
|
rcl_lifecycle_get_zero_initialized_state_machine()
|
|
{
|
|
rcl_lifecycle_state_machine_t state_machine;
|
|
state_machine.transition_map.states = NULL;
|
|
state_machine.transition_map.states_size = 0;
|
|
state_machine.transition_map.transitions = NULL;
|
|
state_machine.transition_map.transitions_size = 0;
|
|
state_machine.com_interface = rcl_lifecycle_get_zero_initialized_com_interface();
|
|
return state_machine;
|
|
}
|
|
|
|
rcl_ret_t
|
|
rcl_lifecycle_state_machine_init(
|
|
rcl_lifecycle_state_machine_t * state_machine,
|
|
rcl_node_t * node_handle,
|
|
const rosidl_message_type_support_t * ts_pub_notify,
|
|
const rosidl_service_type_support_t * ts_srv_change_state,
|
|
const rosidl_service_type_support_t * ts_srv_get_state,
|
|
const rosidl_service_type_support_t * ts_srv_get_available_states,
|
|
const rosidl_service_type_support_t * ts_srv_get_available_transitions,
|
|
bool default_states)
|
|
{
|
|
if (rcl_lifecycle_com_interface_init(
|
|
&state_machine->com_interface, node_handle,
|
|
ts_pub_notify,
|
|
ts_srv_change_state, ts_srv_get_state,
|
|
ts_srv_get_available_states, ts_srv_get_available_transitions) != RCL_RET_OK)
|
|
{
|
|
return RCL_RET_ERROR;
|
|
}
|
|
|
|
if (default_states) {
|
|
rcl_lifecycle_init_default_state_machine(state_machine);
|
|
}
|
|
return RCL_RET_OK;
|
|
}
|
|
|
|
rcl_ret_t
|
|
rcl_lifecycle_state_machine_fini(
|
|
rcl_lifecycle_state_machine_t * state_machine,
|
|
rcl_node_t * node_handle)
|
|
{
|
|
rcl_ret_t fcn_ret = RCL_RET_OK;
|
|
|
|
fcn_ret = rcl_lifecycle_com_interface_fini(&state_machine->com_interface, node_handle);
|
|
|
|
rcl_lifecycle_transition_map_t * transition_map = &state_machine->transition_map;
|
|
|
|
// for each state free the allocations for their keys/transitions
|
|
for (unsigned int i = 0; i < transition_map->states_size; ++i) {
|
|
free(transition_map->states[i].valid_transition_keys);
|
|
free(transition_map->states[i].valid_transitions);
|
|
}
|
|
// free the primary states
|
|
free(transition_map->states);
|
|
transition_map->states = NULL;
|
|
// free the tansitions
|
|
free(transition_map->transitions);
|
|
transition_map->transitions = NULL;
|
|
|
|
return fcn_ret;
|
|
}
|
|
|
|
rcl_ret_t
|
|
rcl_lifecycle_state_machine_is_initialized(const rcl_lifecycle_state_machine_t * state_machine)
|
|
{
|
|
if (!state_machine->com_interface.srv_get_state.impl) {
|
|
RCL_SET_ERROR_MSG("get_state service is null");
|
|
return RCL_RET_ERROR;
|
|
}
|
|
if (!state_machine->com_interface.srv_change_state.impl) {
|
|
RCL_SET_ERROR_MSG("change_state service is null");
|
|
return RCL_RET_ERROR;
|
|
}
|
|
return RCL_RET_OK;
|
|
}
|
|
|
|
const rcl_lifecycle_transition_t *
|
|
rcl_lifecycle_is_valid_transition(
|
|
rcl_lifecycle_state_machine_t * state_machine,
|
|
rcl_lifecycle_ret_t key)
|
|
{
|
|
unsigned int current_id = state_machine->current_state->id;
|
|
const rcl_lifecycle_state_t * current_state = rcl_lifecycle_get_state(
|
|
&state_machine->transition_map, current_id);
|
|
|
|
for (unsigned int i = 0; i < current_state->valid_transition_size; ++i) {
|
|
if (current_state->valid_transition_keys[i] == key) {
|
|
return ¤t_state->valid_transitions[i];
|
|
}
|
|
}
|
|
fprintf(stderr, "%s:%u, No callback transition matching %u found for current state %s\n",
|
|
__FILE__, __LINE__, key, state_machine->current_state->label);
|
|
return NULL;
|
|
}
|
|
|
|
rcl_ret_t
|
|
rcl_lifecycle_trigger_transition(
|
|
rcl_lifecycle_state_machine_t * state_machine,
|
|
rcl_lifecycle_ret_t key, bool publish_notification)
|
|
{
|
|
const rcl_lifecycle_transition_t * transition =
|
|
rcl_lifecycle_is_valid_transition(state_machine, key);
|
|
|
|
// If we have a faulty transition pointer
|
|
if (!transition) {
|
|
fprintf(stderr, "No transition found for node %s with key %d\n",
|
|
state_machine->current_state->label, key);
|
|
RCL_SET_ERROR_MSG("Transition is not registered.");
|
|
return RCL_RET_ERROR;
|
|
}
|
|
|
|
if (!transition->goal) {
|
|
fprintf(stderr, "No valid goal is set\n");
|
|
}
|
|
state_machine->current_state = transition->goal;
|
|
if (publish_notification) {
|
|
rcl_lifecycle_com_interface_publish_notification(
|
|
&state_machine->com_interface, transition->start, state_machine->current_state);
|
|
}
|
|
|
|
return RCL_RET_OK;
|
|
}
|
|
|
|
|
|
void
|
|
rcl_print_state_machine(const rcl_lifecycle_state_machine_t * state_machine)
|
|
{
|
|
const rcl_lifecycle_transition_map_t * map = &state_machine->transition_map;
|
|
for (size_t i = 0; i < map->states_size; ++i) {
|
|
printf("Primary State: %s(%u)\n", map->states[i].label, map->states[i].id);
|
|
printf("# of valid transitions: %u\n", map->states[i].valid_transition_size);
|
|
for (size_t j = 0; j < map->states[i].valid_transition_size; ++j) {
|
|
printf("\tNode %s: Key %d: Transition: %s\n",
|
|
map->states[i].label,
|
|
map->states[i].valid_transition_keys[j],
|
|
map->states[i].valid_transitions[j].label);
|
|
}
|
|
}
|
|
}
|
|
#if __cplusplus
|
|
}
|
|
#endif // extern "C"
|