Goals • Requirements • Getting Started • Features • TODOs • References
C++ Robotics has the following goals:
- Implement as many robotics algorithms as possible, without sacrificing quality. These include, e.g., control, path planning, and estimation algorithms.
- Be easy to use and to get started with, thus the header-only format and minimal external dependencies.
- Be fast: by making use of templates, most algorithms use static-size data structures whose size is known at compilation time.
This project is inspired by PythonRobotics. Instead of being just an educational repo, this library aims at implementing fast algorithms with a consistent API, to guarantee runtime performance and ease of use.
While this is still a work in progress, I would appreciate it if you left any suggestions or starred the repo. Any help is greatly appreciated!
-
C++11 compiler
-
CMake 3.14+ (if using Visual Studio on Windows you should be able to import the project as a CMake project)
-
The C++ dependencies will be obtained automatically by CPM.cmake. Note that the library only depends on
Eigen
, butmatplotplusplus
is used in theexamples
folder to plot the results.
Following are some examples to get started. The examples
folder contains several examples that are useful to get started.
CppRobotics aims to be modular, which means:
-
Once you define a dynamical system, most algorithms will be readily available for it to use
-
Data should flow seamlessly between objects (e.g. estimator -> controller)
-
Once you setup an algorithm, you should be able to change the dynamical system and integrate it directly
Given a generic EXAMPLE
that you want to run, the following commands build it:
cmake -S examples/EXAMPLE -B build/EXAMPLE
cmake --build build/EXAMPLE
On Windows, this will default to a Debug configuration. To build the project in release mode, you can add --config=Release
after the first command.
To run the example on Linux, macOS, and most Unix-based systems:
./build/EXAMPLE/main
On Windows:
./build/EXAMPLE/CONFIG_TYPE/main
where CONFIG_TYPE
is either Debug
or Release
, depending on how you configured the project.
#include <robotics/robotics.hpp>
SystemBase
represents a generic dynamical system. In most cases, you will be using either a LinearSystem
or NonlinearSystem
.
Since the library is templated, to define a system you need to define:
-
The number of states
-
The number of inputs (control actions)
-
The number of outputs
e.g.:
static constexpr int N = 2; // Number of states
static constexpr int M = 1; // Number of inputs
static constexpr int P = 2; // Number of outputs
Type aliases can come handy, and prevent the coder from mixing up the wrong dimensions:
using State = Robotics::ColumnVector<N>;
using Input = Robotics::ColumnVector<M>;
using Output = Robotics::ColumnVector<P>;
using StateMatrix = Robotics::SquareMatrix<N>;
using InputMatrix = Robotics::Matrix<N, M>;
using OutputMatrix = Robotics::Matrix<P, N>;
using FeedthroughMatrix = Robotics::Matrix<P, N>;
Let's define a linear system whose state form is
x' = A * x + B * u
y = C * x + D * u
To set up a LinearSystem
:
StateMatrix A;
A << 1, 0,
0, 1;
InputMatrix B;
B << 1, 0;
OutputMatrix C;
C << 1, 0,
0, 1;
Note that having templates not only improves runtime performance, but also enforces compile-time checking. If you initialize a matrix with the wrong number of elements, the code will not compile.
Matrices C and D are not required: they are null by default if not provided. In this case, D is null. To define the system:
Robotics::Model::LinearSystem<N, M, P> system(A, B, C);
The initial state is zero by default. You can set a custom initial state as follows:
system.SetInitialState(State(1.0, -1.0));
Check out TheLartians/ModernCppStarter if you want to include these features in your project.
From the root directory:
cmake -S test -B build/test
cmake --build build/test
CTEST_OUTPUT_ON_FAILURE=1 cmake --build build/test --target test
# or simply call the executable:
./build/test/RoboticsTests
To also collect code coverage information, run CMake with the -DENABLE_TEST_COVERAGE=1
option.
This requires clang-format, cmake-format and pyyaml to be installed on the current system.
cmake -S test -B build/test
# view changes
cmake --build build/test --target format
# apply changes
cmake --build build/test --target fix-format
See Format.cmake for details.
The documentation is automatically built and published whenever a GitHub Release is created. To manually build documentation, call the following command.
cmake -S documentation -B build/doc
cmake --build build/doc --target GenerateDocs
# view the docs
open build/doc/doxygen/html/index.html
To build the documentation locally, you will need Doxygen, jinja2 and Pygments installed in your system.
The project also includes an all
directory that allows building all targets at the same time.
This is useful during development, as it exposes all subprojects to your IDE and avoids redundant builds of the library.
cmake -S all -B build
cmake --build build
# run tests
./build/test/RoboticsTests
# format code
cmake --build build --target fix-format
# run standalone
./build/standalone/Robotics --help
# build docs
cmake --build build --target GenerateDocs
The test and standalone subprojects include the tools.cmake file which is used to import additional tools on-demand through CMake configuration arguments. The following are currently supported.
Sanitizers can be enabled by configuring CMake with -DUSE_SANITIZER=<Address | Memory | MemoryWithOrigins | Undefined | Thread | Leak | 'Address;Undefined'>
.
Static Analyzers can be enabled by setting -DUSE_STATIC_ANALYZER=<clang-tidy | iwyu | cppcheck>
, or a combination of those in quotation marks, separated by semicolons.
By default, analyzers will automatically find configuration files such as .clang-format
.
Additional arguments can be passed to the analyzers by setting the CLANG_TIDY_ARGS
, IWYU_ARGS
, or CPPCHECK_ARGS
variables.
Ccache can be enabled by configuring with -DUSE_CCACHE=<ON | OFF>
.
- Add more algorithms
- Add support for nonlinear systems automatic differentiation, so that the Jacobians are automatically computed (see autodiff)
- Add a README.md to each example folder, to explain the theory
- Cache the packages downloaded by CPM.CMake (currently everything is re-downloaded every time a new example is built)
- Many more, feel free to add your ideas!
As mentioned above, this repo was originally inspired by AtsushiSakai/PythonRobotics. So go check it out if you want to see more algorithms (or if you want to help port a few of them!).