Skip to content

Commit

Permalink
Upgrade parameters (#274)
Browse files Browse the repository at this point in the history
* Initial parameters upgrade

* Fix memory init/fini

* Delete update

* Fix name on describe service

* Fix var name

* Update parameter example

* Fix add parameter undeclared

* Update test cases

* Uncrustify and lint tests

* Update tests teardown

* Update tests and add low mem mode

* Update example with new API

* Uncrustify

* Fix low mem fini

* Add sleep for event pubsub match

* Update notify_changed_over_dds

* Update string init and deleted event pub

* Disable set API on callback

* Aux param review (#275)

* Update

Signed-off-by: Pablo Garrido <[email protected]>

* Fix typos

Co-authored-by: acuadros95 <[email protected]>

* Fix int tests types

* Update readme

* Minor readme fix

* Apply suggestions from code review

Co-authored-by: Ralph Lange <[email protected]>

* Apply suggestions from code review

* Fix example

* Requested changes

* Apply suggestions from code review

Co-authored-by: Jan Staschulat <[email protected]>

* Add NULL checks

* Apply review comments

* Apply review comments

* Fix line break

* Add test cases

Co-authored-by: Pablo Garrido <[email protected]>
Co-authored-by: Ralph Lange <[email protected]>
Co-authored-by: Jan Staschulat <[email protected]>
  • Loading branch information
4 people authored Apr 14, 2022
1 parent 09de417 commit 9e4fc05
Show file tree
Hide file tree
Showing 8 changed files with 2,716 additions and 368 deletions.
5 changes: 3 additions & 2 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,11 +1,12 @@
# The rclc repository
This repository provides the rclc package, which complements the [ROS Client Support Library (rcl)](https://github.com/ros2/rcl/) to make up a complete ROS 2 client library for the C programming language. That is, rclc does not add a new layer of types on top of rcl (like rclcpp and rclpy do) but only provides convenience functions that ease the programming with the rcl types. New types are introduced only for concepts that are missing in rcl, most important an Executor and a Lifecycle Node.
This repository provides the rclc package, which complements the [ROS Client Support Library (rcl)](https://github.com/ros2/rcl/) to make up a complete ROS 2 client library for the C programming language. That is, rclc does not add a new layer of types on top of rcl (like rclcpp and rclpy do) but only provides convenience functions that ease the programming with the rcl types. New types are introduced only for concepts that are missing in rcl, most important an Executor, Lifecycle Node and the Parameter server.

In detail, this repository contains three packages:
In detail, this repository contains four packages:

- [rclc](rclc/) provides the mentioned convenience functions for creating instances of publishers, subscriptions, nodes, etc. with the corresponding [rcl types](https://github.com/ros2/rcl/tree/master/rcl/include/rcl). Furthermore, it provides the rclc Executor for C, analogously to rclcpp's [Executor class](https://github.com/ros2/rclcpp/blob/master/rclcpp/include/rclcpp/executor.hpp) for C++. A key feature compared to the rclcpp Executor is that it includes features for implementing deterministic timing behavior.
- [rclc_lifecycle](rclc_lifecycle/) introduces an rclc Lifecycle Node, bundling an rcl Node and the [lifecycle state machine](http://design.ros2.org/articles/node_lifecycle.html) from the [rcl_lifecycle package](https://github.com/ros2/rcl/tree/master/rcl_lifecycle).
- [rclc_examples](rclc_examples/) provides small examples for the use of the convenience functions and the rclc Executor, as well as a small example for the use of the rclc Lifecycle Node.
- [rclc_parameter](rclc_parameter/) provides convenience functions for creating parameter server instances with full ROS2 parameters client compatibility.

Technical information on the interfaces and the usage of these packages is given in the README.md files in the corresponding subfolders.

Expand Down
60 changes: 44 additions & 16 deletions rclc_examples/src/example_parameter_server.c
Original file line number Diff line number Diff line change
Expand Up @@ -36,23 +36,44 @@ void timer_callback(rcl_timer_t * timer, int64_t last_call_time)
rclc_parameter_set_int(&param_server, "param2", value);
}

void on_parameter_changed(Parameter * param)
bool on_parameter_changed(const Parameter * old_param, const Parameter * new_param, void * context)
{
printf("Parameter %s modified.", param->name.data);
switch (param->value.type) {
case RCLC_PARAMETER_BOOL:
printf(" New value: %d (bool)", param->value.bool_value);
break;
case RCLC_PARAMETER_INT:
printf(" New value: %ld (int)", param->value.integer_value);
break;
case RCLC_PARAMETER_DOUBLE:
printf(" New value: %f (double)", param->value.double_value);
break;
default:
break;
(void) context;

if (old_param == NULL && new_param == NULL) {
printf("Callback error, both parameters are NULL\n");
return false;
}

if (old_param == NULL) {
printf("Creating new parameter %s\n", new_param->name.data);
} else if (new_param == NULL) {
printf("Deleting parameter %s\n", old_param->name.data);
} else {
printf("Parameter %s modified.", old_param->name.data);
switch (old_param->value.type) {
case RCLC_PARAMETER_BOOL:
printf(
" Old value: %d, New value: %d (bool)", old_param->value.bool_value,
new_param->value.bool_value);
break;
case RCLC_PARAMETER_INT:
printf(
" Old value: %ld, New value: %ld (int)", old_param->value.integer_value,
new_param->value.integer_value);
break;
case RCLC_PARAMETER_DOUBLE:
printf(
" Old value: %f, New value: %f (double)", old_param->value.double_value,
new_param->value.double_value);
break;
default:
break;
}
printf("\n");
}
printf("\n");

return true;
}

int main()
Expand Down Expand Up @@ -82,7 +103,7 @@ int main()
// Create executor
rclc_executor_t executor;
rclc_executor_init(
&executor, &support.context, RCLC_PARAMETER_EXECUTOR_HANDLES_NUMBER + 1,
&executor, &support.context, RCLC_EXECUTOR_PARAMETER_SERVER_HANDLES + 1,
&allocator);
rclc_executor_add_parameter_server(&executor, &param_server, on_parameter_changed);
rclc_executor_add_timer(&executor, &timer);
Expand All @@ -96,6 +117,13 @@ int main()
rclc_parameter_set_int(&param_server, "param2", 10);
rclc_parameter_set_double(&param_server, "param3", 0.01);

// Add parameters constraints
rclc_add_parameter_description(&param_server, "param2", "Second parameter", "Only even numbers");
rclc_add_parameter_constraint_integer(&param_server, "param2", -10, 120, 2);

rclc_add_parameter_description(&param_server, "param3", "Third parameter", "");
rclc_set_parameter_read_only(&param_server, "param3", true);

bool param1;
int64_t param2;
double param3;
Expand Down
297 changes: 297 additions & 0 deletions rclc_parameter/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,297 @@
# The rclc parameter package

ROS 2 parameters allow the user to create variables on a node and manipulate/read them with different ROS 2 commands. Further information about ROS 2 parameters can be found [here](https://docs.ros.org/en/rolling/Tutorials/Parameters/Understanding-ROS2-Parameters.html)

This package provides the rclc API with parameter server instances with full ROS 2 parameter client compatibility. A parameter client has not been implemented for rclc (yet).
Ready to use code examples related to this tutorial can be found in [`rclc/rclc_examples/src/example_parameter_server.c`](../rclc_examples/src/example_parameter_server.c).

## Table of contents
* [Initialization](#initialization)
* [Memory requirements](#memory-requirements)
* [Callback](#callback)
* [Add a parameter](#add-a-parameter)
* [Delete a parameter](#delete-a-parameter)
* [Parameter description](#parameter-description)
* [Cleaning up](#cleaning-up)

## Initialization

- Default initialization:
```c
// Parameter server object
rclc_parameter_server_t param_server;
// Initialize parameter server with default configuration
rcl_ret_t rc = rclc_parameter_server_init_default(&param_server, &node);

if (RCL_RET_OK != rc) {
... // Handle error
return -1;
}
```

- Custom options:

The following options can be configured:
- notify_changed_over_dds: Publish parameter events to other ROS 2 nodes as well.
- max_params: Maximum number of parameters allowed on the `rclc_parameter_server_t` object.
- allow_undeclared_parameters: Allows creation of parameters from external parameter clients. A new parameter will be created if a `set` operation is requested on a non-existing parameter.
- low_mem_mode: Reduces the memory used by the parameter server, functionality constrains are applied.

```c
// Parameter server object
rclc_parameter_server_t param_server;

// Initialize parameter server options
const rclc_parameter_options_t options = {
.notify_changed_over_dds = true,
.max_params = 4,
.allow_undeclared_parameters = true,
.low_mem_mode = false; };

// Initialize parameter server with configured options
rcl_ret_t rc = rclc_parameter_server_init_with_option(&param_server, &node, &options);
if (RCL_RET_OK != rc) {
... // Handle error
return -1;
}
```

- Low memory mode:

This mode ports the parameter functionality to memory constrained devices. The following constrains are applied:
- Request size limited to one parameter on Set, Get, Get types and Describe services.
- List parameter request has no prefixes enabled nor depth.
- Parameter description strings not allowed, `rclc_add_parameter_description` is disabled.

Memory benchmark results on `STM32F4` for 7 parameters with `RCLC_PARAMETER_MAX_STRING_LENGTH = 50` and `notify_changed_over_dds = true`:
- Full mode: 11736 B
- Low memory mode: 4160 B

## Memory requirements

The parameter server uses five services and an optional publisher. These need to be taken into account on the `rmw-microxrcedds` package memory configuration:

```yaml
# colcon.meta example with memory requirements to use a parameter server
{
"names": {
"rmw_microxrcedds": {
"cmake-args": [
"-DRMW_UXRCE_MAX_NODES=1",
"-DRMW_UXRCE_MAX_PUBLISHERS=1",
"-DRMW_UXRCE_MAX_SUBSCRIPTIONS=0",
"-DRMW_UXRCE_MAX_SERVICES=5",
"-DRMW_UXRCE_MAX_CLIENTS=0"
]
}
}
}
```

At runtime, the variable `RCLC_EXECUTOR_PARAMETER_SERVER_HANDLES` defines the necessary number of handles required by a parameter server for the rclc Executor:

```c
// Executor init example with the minimum RCLC executor handles required
rclc_executor_t executor = rclc_executor_get_zero_initialized_executor();
rc = rclc_executor_init(
&executor, &support.context,
RCLC_EXECUTOR_PARAMETER_SERVER_HANDLES, &allocator);
```

## Callback

When adding the parameter server to the executor, a callback could to be configured. This callback would then be executed on the following events:
- Parameter value change: Internal and external parameter set on existing parameters.
- Parameter creation: External parameter set on unexisting parameter if `allow_undeclared_parameters` is set.
- Parameter delete: External parameter delete on existing parameter.
- The user can allow or reject this operations using the `bool` return value.

Callback parameters:
- `old_param`: Parameter actual value, `NULL` for new parameter request.
- `new_param`: Parameter new value, `NULL` for parameter removal request.
- `context`: User context, configured on `rclc_executor_add_parameter_server_with_context`.

```c
bool on_parameter_changed(const Parameter * old_param, const Parameter * new_param, void * context)
{
(void) context;

if (old_param == NULL && new_param == NULL) {
printf("Callback error, both parameters are NULL\n");
return false;
}

if (old_param == NULL) {
printf("Creating new parameter %s\n", new_param->name.data);
} else if (new_param == NULL) {
printf("Deleting parameter %s\n", old_param->name.data);
} else {
printf("Parameter %s modified.", old_param->name.data);
switch (old_param->value.type) {
case RCLC_PARAMETER_BOOL:
printf(
" Old value: %d, New value: %d (bool)", old_param->value.bool_value,
new_param->value.bool_value);
break;
case RCLC_PARAMETER_INT:
printf(
" Old value: %ld, New value: %ld (int)", old_param->value.integer_value,
new_param->value.integer_value);
break;
case RCLC_PARAMETER_DOUBLE:
printf(
" Old value: %f, New value: %f (double)", old_param->value.double_value,
new_param->value.double_value);
break;
default:
break;
}
printf("\n");
}

return true;
}
```
Parameters modifications are disabled while the callback `on_parameter_changed` is executed, causing the following methods to return `RCLC_PARAMETER_DISABLED_ON_CALLBACK` if they are invoked:
- rclc_add_parameter
- rclc_delete_parameter
- rclc_parameter_set_bool
- rclc_parameter_set_int
- rclc_parameter_set_double
- rclc_set_parameter_read_only
- rclc_add_parameter_constraint_double
- rclc_add_parameter_constraint_integer
Once the parameter server and the executor are initialized, the parameter server must be added to the executor in order to accept parameter commands from ROS 2:
```c
// Add parameter server to the executor including defined callback
rc = rclc_executor_add_parameter_server(&executor, &param_server, on_parameter_changed);
```

Note that this callback is optional as its just an event information for the user. To use the parameter server without a callback:
```c
// Add parameter server to the executor without a callback
rc = rclc_executor_add_parameter_server(&executor, &param_server, NULL);
```

Configuration of the callback context:
```c
// Add parameter server to the executor including defined callback and a context
rc = rclc_executor_add_parameter_server_with_context(&executor, &param_server, on_parameter_changed, &context);
```

## Add a parameter

The micro-ROS parameter server supports the following parameter types:

- Boolean:
```c
const char * parameter_name = "parameter_bool";
bool param_value = true;

// Add parameter to the server
rcl_ret_t rc = rclc_add_parameter(&param_server, parameter_name, RCLC_PARAMETER_BOOL);

// Set parameter value (Triggers `on_parameter_changed` callback, if defined)
rc = rclc_parameter_set_bool(&param_server, parameter_name, param_value);

// Get parameter value and store it in "param_value"
rc = rclc_parameter_get_bool(&param_server, "param1", &param_value);

if (RCL_RET_OK != rc) {
... // Handle error
return -1;
}
```

- Integer:
```c
const char * parameter_name = "parameter_int";
int param_value = 100;

// Add parameter to the server
rcl_ret_t rc = rclc_add_parameter(&param_server, parameter_name, RCLC_PARAMETER_INT);

// Set parameter value
rc = rclc_parameter_set_int(&param_server, parameter_name, param_value);

// Get parameter value on param_value
rc = rclc_parameter_get_int(&param_server, parameter_name, &param_value);
```

- Double:
```c
const char * parameter_name = "parameter_double";
double param_value = 0.15;

// Add parameter to the server
rcl_ret_t rc = rclc_add_parameter(&param_server, parameter_name, RCLC_PARAMETER_DOUBLE);

// Set parameter value
rc = rclc_parameter_set_double(&param_server, parameter_name, param_value);

// Get parameter value on param_value
rc = rclc_parameter_get_double(&param_server, parameter_name, &param_value);
```

Parameters can also be created by external clients if the `allow_undeclared_parameters` flag is set.
The client just needs to set a value on a non-existing parameter. Then this parameter will be created if the server has still capacity and the callback allows the operation.

*Max name size is controlled by the compile-time option `RCLC_PARAMETER_MAX_STRING_LENGTH`, default value is 50.*

## Delete a parameter
Parameters can be deleted by both, the parameter server and external clients:
```c
rclc_delete_parameter(&param_server, "param2");
```
Note that for external delete requests, the server callback will be executed, allowing the node to reject the operation.
## Parameter description
- Parameter description
Adds a description of a parameter and its constrains, which will be returned on a describe parameter requests:
```c
rclc_add_parameter_description(&param_server, "param2", "Second parameter", "Only even numbers");
```
*The maximum string size is controlled by the compilation time option `RCLC_PARAMETER_MAX_STRING_LENGTH`, default value is 50.*
- Parameter constraints
Informative numeric constraints can be added to int and double parameters, returning this values on describe parameter requests:
- `from_value`: Start value for valid values, inclusive.
- `to_value`: End value for valid values, inclusive.
- `step`: Size of valid steps between the from and to bound.
```c
int64_t int_from = 0;
int64_t int_to = 20;
uint64_t int_step = 2;
rclc_add_parameter_constraint_integer(&param_server, "param2", int_from, int_to, int_step);
double double_from = -0.5;
double double_to = 0.5;
double double_step = 0.01;
rclc_add_parameter_constraint_double(&param_server, "param3", double_from, double_to, double_step);
```
*This constrains will not be applied by the parameter server, leaving values filtering to the user callback.*
- Read-only parameters:
The new API offers a read-only flag. This flag blocks parameter changes from external clients, but allows changes on the server side:
```c
bool read_only = true;
rclc_set_parameter_read_only(&param_server, "param3", read_only);
```
## Cleaning up
To destroy an initialized parameter server:
```c
// Delete parameter server
rclc_parameter_server_fini(&param_server, &node);
```

This will delete any automatically created infrastructure on the agent (if possible) and deallocate used memory on the parameter server side.
Loading

0 comments on commit 9e4fc05

Please sign in to comment.