From 298f057ec741c41d3851f591c8dcc90ea86f946c Mon Sep 17 00:00:00 2001
From: Maciej Bednarczyk <83034299+mcbed@users.noreply.github.com>
Date: Tue, 28 Mar 2023 18:56:36 +0200
Subject: [PATCH] Added Sphinx documentation (#48)
* added documentation workflow
* added metapackage with doc
* updated doc
---
.github/workflows/documentation.yml | 60 +++++
README.md | 70 +-----
ethercat_driver_ros2/CMakeLists.txt | 4 +
ethercat_driver_ros2/doxygen/Doxyfile | 23 ++
ethercat_driver_ros2/package.xml | 18 ++
ethercat_driver_ros2/sphinx/Makefile | 24 ++
.../sphinx/_static/css/custom.css | 17 ++
ethercat_driver_ros2/sphinx/conf.py | 100 ++++++++
.../sphinx/developer_guide/cia402_drive.rst | 220 ++++++++++++++++++
.../sphinx/developer_guide/coe.rst | 37 +++
.../sphinx/developer_guide/new_plugin.rst | 166 +++++++++++++
.../sphinx/images/logo-icube.png | Bin 0 -> 204123 bytes
ethercat_driver_ros2/sphinx/index.rst | 31 +++
ethercat_driver_ros2/sphinx/make.bat | 35 +++
.../sphinx/quickstart/configuration.rst | 76 ++++++
.../sphinx/quickstart/installation.rst | 140 +++++++++++
.../sphinx/quickstart/usage.rst | 40 ++++
.../sphinx/user_guide/config_cia402_drive.rst | 104 +++++++++
.../user_guide/config_generic_slave.rst | 217 +++++++++++++++++
19 files changed, 1315 insertions(+), 67 deletions(-)
create mode 100644 .github/workflows/documentation.yml
create mode 100644 ethercat_driver_ros2/CMakeLists.txt
create mode 100644 ethercat_driver_ros2/doxygen/Doxyfile
create mode 100644 ethercat_driver_ros2/package.xml
create mode 100644 ethercat_driver_ros2/sphinx/Makefile
create mode 100644 ethercat_driver_ros2/sphinx/_static/css/custom.css
create mode 100644 ethercat_driver_ros2/sphinx/conf.py
create mode 100644 ethercat_driver_ros2/sphinx/developer_guide/cia402_drive.rst
create mode 100644 ethercat_driver_ros2/sphinx/developer_guide/coe.rst
create mode 100644 ethercat_driver_ros2/sphinx/developer_guide/new_plugin.rst
create mode 100644 ethercat_driver_ros2/sphinx/images/logo-icube.png
create mode 100644 ethercat_driver_ros2/sphinx/index.rst
create mode 100644 ethercat_driver_ros2/sphinx/make.bat
create mode 100644 ethercat_driver_ros2/sphinx/quickstart/configuration.rst
create mode 100644 ethercat_driver_ros2/sphinx/quickstart/installation.rst
create mode 100644 ethercat_driver_ros2/sphinx/quickstart/usage.rst
create mode 100644 ethercat_driver_ros2/sphinx/user_guide/config_cia402_drive.rst
create mode 100644 ethercat_driver_ros2/sphinx/user_guide/config_generic_slave.rst
diff --git a/.github/workflows/documentation.yml b/.github/workflows/documentation.yml
new file mode 100644
index 00000000..074e24ae
--- /dev/null
+++ b/.github/workflows/documentation.yml
@@ -0,0 +1,60 @@
+name: Documentation
+
+# Controls when the workflow will run
+on:
+ # Triggers the workflow on push or pull request events but only for the master branch
+ push:
+ branches: [ main ]
+
+ # Allows you to run this workflow manually from the Actions tab
+ workflow_dispatch:
+
+# A workflow run is made up of one or more jobs that can run sequentially or in parallel
+jobs:
+ # This workflow contains a single job called "build"
+ build_documentation:
+ # The type of runner that the job will run on
+ runs-on: ubuntu-latest
+
+ # Steps represent a sequence of tasks that will be executed as part of the job
+ steps:
+ # Checks-out your repository under $GITHUB_WORKSPACE, so your job can access it
+ - uses: actions/checkout@v3
+
+ - name: Install dependencies
+ run: |
+ sudo apt-get update -qq
+ sudo apt-get install -y -qq doxygen graphviz plantuml
+ pip install sphinx-rtd-theme
+ pip install sphinxcontrib-plantuml
+ pip install sphinx-mdinclude
+ pip install breathe
+ pip install exhale
+ - name: Build doxygen
+ run: |
+ cd ./documentation/doxygen/
+ doxygen
+ cd ../..
+ - name: Build documentation
+ run: |
+ cd ./documentation/sphinx/
+ make html
+ cd ../..
+ - name: Create commit
+ run: |
+ git clone https://github.com/ICube-Robotics/ethercat_driver_ros2.git --branch gh-pages --single-branch gh-pages
+ mkdir -p gh-pages/docs/
+ mkdir -p gh-pages/api/
+ cp -r ./documentation/sphinx/_build/html/* gh-pages/
+ cp -r ./documentation/doxygen/_build/html/* gh-pages/api/
+ cd gh-pages
+ git config --local user.email "action@github.com"
+ git config --local user.name "GitHub Action"
+ git add .
+ git commit -m "Update documentation" -a || true
+ - name: Push changes
+ uses: ad-m/github-push-action@master
+ with:
+ branch: gh-pages
+ directory: gh-pages
+ github_token: ${{ secrets.GITHUB_TOKEN }}
diff --git a/README.md b/README.md
index d10f6d20..c58edcbc 100644
--- a/README.md
+++ b/README.md
@@ -7,71 +7,7 @@ Implementation of a `Hardware Interface` for simple Ethercat module integration
## About
[EtherCAT](https://www.ethercat.org/default.htm) provides applications with the capacity of reliable, real-time communication between systems and is therefore a common industrial standard. In order to simplify the development/deployment of new application using EtherCAT modules, the `ethercat_driver` allows to combine them with [`ros2_control`](https://github.com/ros-controls/ros2_control). This driver proposes a generic ways to parametrise and assemble `Hardware Interfaces` based on EtherCAT modules that can be defined using parameter files.
-## Usage
-### 1. Installation
-Installation steps for [IgH EtherCAT Master for Linux](https://etherlab.org/en/ethercat/) and the driver can be found [here](INSTALL.md).
-
-### 2. Description
-#### Hardwre Interfaces and EtherCAT Master
-The `ethercat_driver` is designed in such a way that each EtherCAT Master is defined in `ros2_control` as a specific `Hardware Interface`. This is done in the `ros2_control` description file, where the EtherCAT driver is loaded as `Hardware Interface` and linked to an EtherCAT Master by its ID:
-```xml
-
-
- ethercat_driver/EthercatDriver
- 0
- 100
-
-
-```
-**NOTE**: As in the current implementation of `ros2_control` there is no information about the system update frequency, it needs to be passed here as parameter. This is only needed systems that include EtherCAT modules that use the Distributed Clock.
-
-#### EtherCAT Slave modules as Plugins
-In this driver, the EtherCAT Slave modules are defined as [Plugins](https://docs.ros.org/en/foxy/Tutorials/Pluginlib.html) and can be parametrized in the `ros2_control` description file :
-```xml
-
- ethercat_plugins/ECModule
- 0
- 1
-
-```
-All modules have `alias` and `position` parameters that specify their address in the EtherCAT Bus topology. Additional parameters can be specified depending on the purpose of the module. A list of implemented modules and their parameters can be found [here](ethercat_plugins/available_plugins.md).
-
-#### Interfacing controllers with EtherCAT Slave modules
-In `ros2_control` the access to resources within a system from a controller is done by means of [`Hardware Resources`](https://github.com/ros-controls/roadmap/blob/master/design_drafts/hardware_access.md). For this purpose `state_interface` and `command_interface` tags need to be defined and associated with the module functionalities.
-Also, for better understanding of the overall system, the pupropose of the used modules need to be clearly identified and sorted into the following types:
-- ``: logical component actuated by at least one actuator with read-write capacity.
-- ``: logical component to read-only states from system.
-- ``: logical component for general purpose IO systems.
-
-**NOTE**: These components have the possibility to include parameters that will be used to link particular states and commands to the slave module input/outputs.
-
-Here is an example of a `gpio` resource built using 2 EtherCAT IO modules, where digital commands are mapped on ports 4 and 6 of the digital output module and analog states are read from ports 1 and 4 of the analog input module:
-```xml
-
-
-
-
-
-
-
-
- ethercat_plugins/Beckhoff_EL3104
- 0
- 1
- ana_input.1
- ana_input.2
-
-
- ethercat_plugins/Beckhoff_EL2008
- 0
- 2
- dig_output.2
- dig_output.1
-
-
-```
-
-**NOTE** : To send commands to `gpio` resources, a generic controller was developed and can be found [here](https://github.com/mcbed/ros2_controllers/tree/gpio_controllers).
+**For more information, please check the [documentation](https://ICube-Robotics.github.io/ethercat_driver_ros2/).**
## Acknowledgments
Parts of the driver are based on the implementation of [`SimplECAT`](https://bitbucket.org/bsoe/simplecat/src/master/).
@@ -79,6 +15,6 @@ Parts of the driver are based on the implementation of [`SimplECAT`](https://bit
## Contacts ##
![icube](https://icube.unistra.fr/fileadmin/templates/DUN/icube/images/logo.png)
-[ICube Laboratory](https://plateforme.icube.unistra.fr), [University of Strasbourg](https://www.unistra.fr/), France
+[ICube Laboratory](https://icube.unistra.fr), [University of Strasbourg](https://www.unistra.fr/), France
-__Maciej Bednarczyk:__ [m.bednarczyk@unistra.fr](mailto:m.bednarczyk@unistra.fr), @github: [mcbed](mailto:macbednarczyk@gmail.com)
+__Maciej Bednarczyk:__ [m.bednarczyk@unistra.fr](mailto:m.bednarczyk@unistra.fr), @github: [mcbed](https://github.com/mcbed)
diff --git a/ethercat_driver_ros2/CMakeLists.txt b/ethercat_driver_ros2/CMakeLists.txt
new file mode 100644
index 00000000..f372d947
--- /dev/null
+++ b/ethercat_driver_ros2/CMakeLists.txt
@@ -0,0 +1,4 @@
+cmake_minimum_required(VERSION 3.8)
+project(ethercat_driver_ros2 NONE)
+find_package(ament_cmake REQUIRED)
+ament_package()
diff --git a/ethercat_driver_ros2/doxygen/Doxyfile b/ethercat_driver_ros2/doxygen/Doxyfile
new file mode 100644
index 00000000..7a3e19f4
--- /dev/null
+++ b/ethercat_driver_ros2/doxygen/Doxyfile
@@ -0,0 +1,23 @@
+# All settings not listed here will use the Doxygen default values.
+
+PROJECT_NAME = "ethercat_driver_ros2"
+PROJECT_NUMBER = main
+PROJECT_BRIEF = "C++ ROS test"
+
+# Use these lines to include the generated logging.hpp (update install path if needed)
+INPUT = ../../ethercat_driver/include ../../ethercat_interface/include
+
+RECURSIVE = YES
+OUTPUT_DIRECTORY = _build
+
+EXTRACT_ALL = YES
+SORT_MEMBER_DOCS = NO
+
+GENERATE_LATEX = NO
+GENERATE_XML = YES
+
+ENABLE_PREPROCESSING = YES
+
+DOT_GRAPH_MAX_NODES = 101
+
+FILE_PATTERNS = *.cpp *.h *.hpp *.md
diff --git a/ethercat_driver_ros2/package.xml b/ethercat_driver_ros2/package.xml
new file mode 100644
index 00000000..b6fdf8c6
--- /dev/null
+++ b/ethercat_driver_ros2/package.xml
@@ -0,0 +1,18 @@
+
+
+
+ ethercat_driver_ros2
+ 1.0.0
+ Meta-package aggregating the ethercat_driver_ros2 packages and documentation
+ Maciej Bednarczyk
+ Apache License 2.0
+
+ ament_cmake
+ ethercat_driver
+ ethercat_interface
+ ethercat_plugins
+
+
+ ament_cmake
+
+
diff --git a/ethercat_driver_ros2/sphinx/Makefile b/ethercat_driver_ros2/sphinx/Makefile
new file mode 100644
index 00000000..7c644899
--- /dev/null
+++ b/ethercat_driver_ros2/sphinx/Makefile
@@ -0,0 +1,24 @@
+# Minimal makefile for Sphinx documentation
+#
+
+# You can set these variables from the command line, and also
+# from the environment for the first two.
+SPHINXOPTS ?=
+SPHINXBUILD ?= sphinx-build
+SOURCEDIR = .
+BUILDDIR = _build
+
+# Put it first so that "make" without argument is like "make help".
+help:
+ @$(SPHINXBUILD) -M help "$(SOURCEDIR)" "$(BUILDDIR)" $(SPHINXOPTS) $(O)
+
+.PHONY: help Makefile clean
+
+clean:
+ rm -rf "_doxygen/" "api/"
+ @$(SPHINXBUILD) -M clean "$(SOURCEDIR)" "$(BUILDDIR)" $(SPHINXOPTS) $(O)
+
+# Catch-all target: route all unknown targets to Sphinx using the new
+# "make mode" option. $(O) is meant as a shortcut for $(SPHINXOPTS).
+%: Makefile
+ @$(SPHINXBUILD) -M $@ "$(SOURCEDIR)" "$(BUILDDIR)" $(SPHINXOPTS) $(O)
diff --git a/ethercat_driver_ros2/sphinx/_static/css/custom.css b/ethercat_driver_ros2/sphinx/_static/css/custom.css
new file mode 100644
index 00000000..54d2f2e7
--- /dev/null
+++ b/ethercat_driver_ros2/sphinx/_static/css/custom.css
@@ -0,0 +1,17 @@
+
+.wy-table-responsive table td,
+.wy-table-responsive table th {
+ white-space:normal;
+}
+
+tr {
+ white-space: normal;
+}
+
+thead {
+ white-space: normal;
+}
+
+table {
+ white-space: normal;
+}
diff --git a/ethercat_driver_ros2/sphinx/conf.py b/ethercat_driver_ros2/sphinx/conf.py
new file mode 100644
index 00000000..5b6a976a
--- /dev/null
+++ b/ethercat_driver_ros2/sphinx/conf.py
@@ -0,0 +1,100 @@
+# Copyright 2022 ICube Laboratory, University of Strasbourg
+#
+# 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.
+
+# Configuration file for the Sphinx documentation builder.
+#
+# This file only contains a selection of the most common options. For a full
+# list see the documentation:
+# https://www.sphinx-doc.org/en/master/usage/configuration.html
+
+# -- Path setup --------------------------------------------------------------
+
+# If extensions (or modules to document with autodoc) are in another directory,
+# add these directories to sys.path here. If the directory is relative to the
+# documentation root, use os.path.abspath to make it absolute, like shown here.
+#
+from pathlib import Path
+
+# -- Project information -----------------------------------------------------
+
+project = "ethercat_driver_ros2"
+copyright = "2023, ICUBE Laboratory, University of Strasbourg"
+author = "Maciej Bednarczyk, Philippe Zanne, Laurent Barbé"
+
+# The full version, including alpha/beta/rc tags
+release = "1.0.0"
+
+
+# -- General configuration ---------------------------------------------------
+
+# Add any Sphinx extension module names here, as strings. They can be
+# extensions coming with Sphinx (named 'sphinx.ext.*') or your custom
+# ones.
+extensions = [
+ "sphinx_rtd_theme",
+ "sphinx_mdinclude",
+ "sphinx.ext.imgmath",
+ "sphinx.ext.todo",
+ "sphinx.ext.graphviz",
+ "sphinxcontrib.plantuml",
+ "breathe",
+]
+
+breathe_default_project = "ethercat_driver_ros2"
+
+
+def get_package(package: str):
+ path = Path(__file__).parent.parent.parent.joinpath(f"{package}/include/{package}")
+ files_gen = path.glob("*.hpp")
+ files = []
+ for file in files_gen:
+ files.append(file.name)
+ return (path, files)
+
+
+breathe_projects = {
+ "ethercat_driver_ros2": "../doxygen/_build/xml/",
+}
+
+
+# Tell sphinx what the primary language being documented is.
+primary_domain = "cpp"
+
+# Tell sphinx what the pygments highlight language should be.
+highlight_language = "cpp"
+
+
+# Add any paths that contain templates here, relative to this directory.
+templates_path = ["_templates"]
+
+# List of patterns, relative to source directory, that match files and
+# directories to ignore when looking for source files.
+# This pattern also affects html_static_path and html_extra_path.
+exclude_patterns = ["_build", "Thumbs.db", ".DS_Store"]
+
+
+# -- Options for HTML output -------------------------------------------------
+
+# The theme to use for HTML and HTML Help pages. See the documentation for
+# a list of builtin themes.
+#
+html_theme = "sphinx_rtd_theme"
+
+# Add any paths that contain custom static files (such as style sheets) here,
+# relative to this directory. They are copied after the builtin static files,
+# so a file named "default.css" will overwrite the builtin "default.css".
+html_static_path = ["_static"]
+html_logo = "images/logo-icube.png"
+html_css_files = ["css/custom.css"]
+pygments_style = "sphinx"
diff --git a/ethercat_driver_ros2/sphinx/developer_guide/cia402_drive.rst b/ethercat_driver_ros2/sphinx/developer_guide/cia402_drive.rst
new file mode 100644
index 00000000..ee180445
--- /dev/null
+++ b/ethercat_driver_ros2/sphinx/developer_guide/cia402_drive.rst
@@ -0,0 +1,220 @@
+CANopen over EtherCAT for electrical drives
+===========================================
+
+CANopen also defines several device profiles by establishing standard behavior and communication objects for similar devices. In the case of electrical drives, the profile is given by the **CiA402** ("CANopen device profile for drives and motion control") norm compliant with the `IEC `_ standard that defines programming languages for programmable control systems. The CiA402 norm specifies specific objects in OD in range :code:`0x6000` to :code:`0x67fe`, what guarantees, that for a drive compatible with CiA402, typical process values such as position, velocity, or torque set-points and actual values will always be defined in the same objects with the same address regardless of the manufacturer and by such simplifies the commissioning of different drives. CIA402 also defines a state machine that corresponds to the drive’s actual operating state (:code:`Disabled`, :code:`Fault`, :code:`Switch ON`, etc.). This state machine is controlled and monitored by two mandatory PDO entries: :code:`Control Word` and :code:`Status Word`. The drive cannot be started until the state machine has been put into the appropriate state.
+
+The device states and possible control sequences of the drive are described by the CANopen state machine as follows:
+
+.. image:: https://webhelp.kollmorgen.com/kas/Content/Resources/Images/CoE%20status%20machine.png
+ :width: 400
+ :alt: ecm_lifecycle
+ :align: center
+
+The possible drive states are :
+
+.. list-table:: CiA402 Drive States
+ :widths: 15 35
+ :header-rows: 1
+
+ * - State
+ - Description
+ * - :code:`NOT_READY_TO_SWITCH_ON`
+ - The drive is not ready to switch on. The drive is in boot phase. Drive motion functions are disabled. No communication established.
+ * - :code:`SWITCH_ON_DISABLED`
+ - The drive cannot be enabled via the EtherCAT interface. Drive motion functions are disabled.
+ * - :code:`READY_TO_SWITCH_ON`
+ - The drive can be enabled. Parameters can be transferred. Power may be applied. Drive motion functions are disabled.
+ * - :code:`SWITCHED_ON`
+ - The drive is enabled but idle No fault detected. Power is enabled Drive motion functions are disabled. Parameters can be transferred. No set-points are transferred from the EtherCAT interface.
+ * - :code:`OPERATION_ENABLED`
+ - Normal operation mode No fault detected. Drive motion functions and power are enabled. Power enabled. Set-points are transferred from the EtherCAT interface.
+ * - :code:`QUICK_STOP_ACTIVE`
+ - The quick stop process is executed. Drive motion functions are disabled. Power is enabled.
+ * - :code:`FAULT_REACTION_ACTIVE`
+ - A fault has occurred, fault reaction processes are executed.
+ * - :code:`FAULT`
+ - A fault is active, Drive is stopped and its functions are disabled.
+
+State transitions are the result of events that can be either internal or triggers through the :code:`Control Word`. When a command trigger for state change is received, the transition is performed before any other command is processed.
+
+The possible transition states, their trigger events and performed actions are :
+
+.. list-table:: CiA402 Drive State Transitions
+ :widths: 1 20 24
+ :header-rows: 1
+
+ * - State Transition
+ - Event
+ - Action
+ * - :code:`0`
+ - **Automatic transition** after power-on or reset application
+ - Drive device self-test and/or self initialization has to be performed
+ * - :code:`1`
+ - **Automatic transition**
+ - Communication has to be activated
+ * - :code:`2`
+ - Shutdown command from control device or local signal
+ - None
+ * - :code:`3`
+ - Switch on command received from control device or local signal
+ - The high-level power has to be switched ON, if possible
+ * - :code:`4`
+ - Enable operation command received from control device or local signal
+ - The drive function has to be enabled. All internal set-points cleared.
+ * - :code:`5`
+ - Disable operation command received from control device or local signal
+ - The drive function has to be disabled.
+ * - :code:`6`
+ - Shutdown command received from control device or local signal
+ - The high-level power has to be switched OFF, if possible.
+ * - :code:`7`
+ - Quick stop or disable voltage command from control device or local signal
+ - None
+ * - :code:`8`
+ - Shutdown command from control device or local signal
+ - The drive function has to be disabled. The high-level power has to be switched OFF, if possible.
+ * - :code:`9`
+ - Disable voltage command from control device or local signal
+ - The drive function has to be disabled. The high-level power has to be switched OFF, if possible.
+ * - :code:`10`
+ - Disable voltage or quick stop command from control device or local signal
+ - The high-level power has to be switched OFF, if possible.
+ * - :code:`11`
+ - Quick stop command from control device or local signal
+ - The quick stop function has to be started.
+ * - :code:`12`
+ - Either: - **Automatic transition** when the quick stop function is completed and quick stop option code is 1, 2, 3 or 4. - Disable voltage command received from control device (depends on the quick stop option code)
+ - The drive function has to be disabled. The high-level power has to be switched OFF, if possible.
+ * - :code:`13`
+ - Fault signal
+ - The configured fault reaction function has to be executed.
+ * - :code:`14`
+ - **Automatic transition**
+ - The drive function has to be disabled. The high-level power has to be switched OFF, if possible.
+ * - :code:`15`
+ - Fault reset command from control device or local signal.
+ - A reset of the fault condition is performed if no fault exists currently on the drive device. After leaving the Fault state, the Fault reset bit in the control word has to be cleared by the control device.
+ * - :code:`16`
+ - If the quick stop option code is 5, 6, 7, or 8, enable operation command from control device
+ - The drive function has to be enabled
+
+CANopen uses 16-bits :code:`Status Word` and :code:`Control Word` to monitor and control the state machine. These PDO entries have the following bit assignment :
+
+.. list-table:: CiA402 Drive Words
+ :widths: 1 20 24
+ :header-rows: 1
+
+ * - Bit
+ - :code:`Status Word` Name
+ - :code:`Control Word` Name
+ * - :code:`0`
+ - Ready to switch on
+ - Switch On
+ * - :code:`1`
+ - Switched on
+ - Disable Voltage
+ * - :code:`2`
+ - Operation Enabled
+ - Quick Stop
+ * - :code:`3`
+ - Fault
+ - Enable Operation
+ * - :code:`4`
+ - Voltage Enabled
+ - Operation-mode Specific
+ * - :code:`5`
+ - Quick Stop
+ - Operation-mode Specific
+ * - :code:`6`
+ - Switch On Disabled
+ - Operation-mode Specific
+ * - :code:`7`
+ - Warning
+ - Reset Fault (only effective for faults)
+ * - :code:`8`
+ - Manufacturer-specific (reserved)
+ - Pause/halt
+ * - :code:`9`
+ - Remote (always 1)
+ - Reserved
+ * - :code:`10`
+ - Target Reached
+ - Reserved
+ * - :code:`11`
+ - Internal Limit Active
+ - Reserved
+ * - :code:`12`
+ - Operation-mode Specific (reserved)
+ - Reserved
+ * - :code:`13`
+ - Operation-mode Specific (reserved)
+ - Manufacturer-specific
+ * - :code:`14`
+ - Manufacturer-specific (reserved)
+ - Manufacturer-specific
+ * - :code:`15`
+ - Manufacturer-specific (reserved)
+ - Manufacturer-specific
+
+.. image:: https://infosys.beckhoff.com/content/1033/ax2000-b110/Images/StateMachine04.gif
+ :width: 700
+ :alt: words
+
+The :code:`Status Word` is only updated and written by the drive in :code:`Safe-Op` and :code:`Operational` states. The current drive state can be decoded from the logical combination of the bits in the :code:`Status Word`:
+
+.. list-table:: CiA402 Drive State form :code:`Status Word`
+ :widths: 50 50
+ :header-rows: 1
+
+ * - :code:`Status Word`
+ - State
+ * - :code:`xxxx xxxx x0xx 0000`
+ - Not ready to switch on
+ * - :code:`xxxx xxxx x1xx 0000`
+ - Switch on disabled
+ * - :code:`xxxx xxxx x01x 0001`
+ - Ready to switch on
+ * - :code:`xxxx xxxx x01x 0011`
+ - Switched on
+ * - :code:`xxxx xxxx x01x 0111`
+ - Operation enabled
+ * - :code:`xxxx xxxx x00x 0111`
+ - Quick stop active
+ * - :code:`xxxx xxxx x0xx 1111`
+ - Fault reaction active
+ * - :code:`xxxx xxxx x0xx 1000`
+ - Fault
+
+The control commands allow the manipulation of the state of a drive by setting its control word. Commands are built up from the logical combination of the bits in the :code:`Control Word`:
+
+.. list-table:: CiA402 Drive Commands to :code:`Control Word`
+ :widths: 25 25 25
+ :header-rows: 1
+
+ * - :code:`Control Word`
+ - Command
+ - State Transitions
+ * - :code:`xxxx 0xxx x110`
+ - Shutdown
+ - 2, 6, 8
+ * - :code:`xxxx 0xxx 0111`
+ - Switch On
+ - 3
+ * - :code:`xxxx 0xxx 1111`
+ - Switch On + Enable Operation
+ - 3 + 4
+ * - :code:`xxxx 0xxx xx0x`
+ - Disable
+ - 7, 9, 10, 12
+ * - :code:`xxxx 0xxx x01x`
+ - Quick Stop
+ - 7, 10, 11
+ * - :code:`xxxx 0xxx 0111`
+ - Disable Operation
+ - 5
+ * - :code:`xxxx 0xxx 1111`
+ - Enable Operation
+ - 4, 16
+ * - :code:`xxxx 1xxx xxxx`
+ - Fault Reset
+ - 15
diff --git a/ethercat_driver_ros2/sphinx/developer_guide/coe.rst b/ethercat_driver_ros2/sphinx/developer_guide/coe.rst
new file mode 100644
index 00000000..c1b745ad
--- /dev/null
+++ b/ethercat_driver_ros2/sphinx/developer_guide/coe.rst
@@ -0,0 +1,37 @@
+CANopen over EtherCAT
+=====================
+
+The CANopen over EtherCAT (CoE) protocol allows devices equipped with CANopen to be used on EtherCAT-based Industrial Ethernet networks.
+
+CANopen
+-------
+
+CANopen is a communication protocol based on the `CAN (Controller Area Network) `_ physical communication standard. In the `OSI communication model `_, CAN specifies the physical and data link layers, while CANopen addresses the higher layers — network, transport, session, presentation, and application.
+
+The CANopen protocol defines how automation devices are configured and accessed, and how messages are exchanged between them. CANopen is object-based, meaning that each node (drives, controllers, encoders, I/O, and other devices) in the network has an Object Dictionary (OD), which contains communication objects. These communication objects cover network management data; special functions; acyclic configuration data, which is handled by Service Data Objects (SDOs); cyclic real-time data, which is handled by Process Data Objects (PDOs):
+
+- **Process Data Objects (PDO)** contain OD entries that are cyclically transferred as process variables. Before cyclic communication is started during the configuration phase, particular OD objects are mapped to this structure. Each PDO entry has a defined offset in the exchanged dataset, encapsulated in the Ethernet frame, so that, during the cyclic phase, the slaves’ hardware can locate the relevant data. After cyclic communication is started, PDO entries are exchanged between master and slaves in every cycle and cannot be changed without reconfiguring the communication configuration of the network.
+- **Service Data Objects (SDO)** contain object dictionary entries that can be exchanged acyclically. SDO works as a mailbox sending and buffering received data. This communication is acyclic and is dependent on available bandwidth in the communication cycle. This communication is not deterministic and is best suited for transmitting configuration data.
+
+The use of object dictionaries, Service Data Objects, and Process Data Objects is a key component of the CANopen protocol, with SDOs being the mechanism for read-write access to the object dictionary.
+
+Every entry of the OD objects is given an index address and sometimes a sub-index sub-address and each OD object consists of 16-bits and data indexes. In this context, addresses between :code:`0x1000` and :code:`0x1fff` contain communication objects, between :code:`0x2000` and :code:`0x5999` manufacturer specific objects and from :code:`0x6000` device profile objects.
+
+CANopen is widely used thanks to its low hardware cost, wide range of device and application profiles, and simple implementation. It’s also extremely reliable and provides real-time communication, making it suitable for industrial applications.
+
+EtherCAT
+--------
+
+EtherCAT is an Industrial Ethernet network. It’s based on standard Ethernet hardware but uses a “processing-on-the-fly” method for transporting and routing messages. In addition to being a real-time networking protocol, EtherCAT is deterministic, meaning it guarantees that a message will be transmitted (or an event will occur) in a specified, predictable period of time — not slower or faster. EtherCAT allows up to 100 meters between nodes (devices) and can provide data transmission rates up to 100 Mbps, with cycle times of less than 100 μs and extremely low jitter, thanks to distributed synchronized clocks.
+
+CANopen over EtherCAT (CoE) protocol
+------------------------------------
+
+CANopen over EtherCAT (CoE) allows the CANopen communication protocol to be implemented over an EtherCAT network, providing a user-friendly, cost-effective solution that provides deterministic data delivery along with faster transmission speeds over longer network lengths.
+
+The use of CANopen over EtherCAT is possible in significant part because EtherCAT implements the same communication system, including object dictionaries, SDOs (the SDO protocol is implemented directly from CANopen, without changes) and PDOs. And on an EtherCAT network, PDO frames are sent deterministically and without the 8-byte limit imposed by CANopen. CANopen over EtherCAT also supports the CANopen device profiles, which specify the parameters and behavior of the device, as well as the device class-specific state machines.
+
+.. image:: https://b2600047.smushcdn.com/2600047/wp-content/uploads/2021/12/Applied-Motion-CANopen-over-EtherCAT.png
+ :width: 400
+ :alt: coe
+ :align: center
diff --git a/ethercat_driver_ros2/sphinx/developer_guide/new_plugin.rst b/ethercat_driver_ros2/sphinx/developer_guide/new_plugin.rst
new file mode 100644
index 00000000..145550cd
--- /dev/null
+++ b/ethercat_driver_ros2/sphinx/developer_guide/new_plugin.rst
@@ -0,0 +1,166 @@
+Creating a new device driver
+============================
+
+Creating a new device driver to work with the :code:`ethercat_driver_ros2` is fairly easy. You should do this if you need to create a driver for a specific device or a specific device profile that is not yet supported. If you create a driver for a device profile we are happy to integrate the package into this repository - simply create a PR.
+
+What you need to do
+-------------------
+
+To create a dedicated driver for an EtherCAT module, you need to create a new package containing the module driver plugin and register it so that it can be called by the :code:`ethercat_driver_ros2`.
+
+How to do it
+------------
+
+Create a package
+~~~~~~~~~~~~~~~~
+
+Create a new package using the standard ros2 pkg commands. Make sure you add the following dependencies:
+
+* :code:`ethercat_interface`
+* :code:`pluginlib`
+
+Create your plugin class
+~~~~~~~~~~~~~~~~~~~~~~~~
+
+To be loaded by the :code:`ethercat_driver_ros2`, the new module plugin needs to inherit from the :code:`EcSlave` class and implement some of its virtual functions. To do so create in your package :code:`src` folder a new file :code:`my_ec_device_driver.cpp`:
+
+.. code-block:: cpp
+
+ #include "ethercat_interface/ec_slave.hpp"
+
+ namespace ethercat_plugins
+ {
+
+ class MyEcDeviceDriver : public ethercat_interface::EcSlave
+ {
+ public:
+ MyEcDeviceDriver()
+ : EcSlave(, ) {}
+ virtual ~MyEcDeviceDriver() {}
+ // data processing method that will be called cyclically for every channel
+ // the index corresponds to the values registered in domains_
+ virtual void processData(size_t index, uint8_t * domain_address)
+ {
+ // Your process data logic goes here
+ }
+ virtual const ec_sync_info_t * syncs() {return &syncs_[0];}
+ virtual size_t syncSize()
+ {
+ return sizeof(syncs_) / sizeof(ec_sync_info_t);
+ }
+ virtual const ec_pdo_entry_info_t * channels()
+ {
+ return channels_;
+ }
+ virtual void domains(DomainMap & domains) const
+ {
+ domains = domains_;
+ }
+ // configure the slave module with urdf parameters
+ // and link ros2_control command and stat interface
+ virtual bool setupSlave(
+ std::unordered_map slave_paramters,
+ std::vector * state_interface,
+ std::vector * command_interface)
+ {
+ state_interface_ptr_ = state_interface;
+ command_interface_ptr_ = command_interface;
+ paramters_ = slave_paramters;
+
+ // Your module setup logic goes here
+
+ return true;
+ }
+
+ private:
+ // configure module channels
+ ec_pdo_entry_info_t channels_[X] = {
+ {, , },
+ };
+ // configure module pdos
+ ec_pdo_info_t pdos_[X] = {
+ {, , },
+ };
+ // configure module syncs
+ ec_sync_info_t syncs_[X] = {
+ {, , , , },
+ {0xff}
+ };
+ // configure module domain
+ DomainMap domains_ = {
+ {0, {0}} // index of channels that should call processData()
+ };
+ };
+
+ } // namespace ethercat_plugins
+
+ #include
+
+ PLUGINLIB_EXPORT_CLASS(ethercat_plugins::MyEcDeviceDriver, ethercat_interface::EcSlave)
+
+Export your plugin
+~~~~~~~~~~~~~~~~~~
+
+In the package root directory create a plugin description file :code:`ethercat_plugins.xml` :
+
+.. code-block:: xml
+
+
+
+ Description of the device driver.
+
+
+
+Modify your :code:`CMakeLists.txt` file so that it looks like this:
+
+.. code-block:: cmake
+
+ cmake_minimum_required(VERSION 3.8)
+ project()
+
+ if(CMAKE_COMPILER_IS_GNUCXX OR CMAKE_CXX_COMPILER_ID MATCHES "Clang")
+ add_compile_options(-Wall -Wextra -Wpedantic)
+ endif()
+
+ # find dependencies
+ find_package(ament_cmake REQUIRED)
+ find_package(ament_cmake_ros REQUIRED)
+ find_package(ethercat_interface REQUIRED)
+ find_package(pluginlib REQUIRED)
+
+ file(GLOB_RECURSE PLUGINS_SRC src/*.cpp)
+ add_library(${PROJECT_NAME} ${PLUGINS_SRC})
+ target_compile_features(${PROJECT_NAME} PUBLIC c_std_99 cxx_std_17) # Require C99 and C++17
+ target_include_directories(${PROJECT_NAME} PUBLIC
+ $
+ $
+ )
+ ament_target_dependencies(
+ ${PROJECT_NAME}
+ "ethercat_interface"
+ "pluginlib"
+ )
+ pluginlib_export_plugin_description_file(ethercat_interface ethercat_plugins.xml)
+ install(
+ DIRECTORY include/
+ DESTINATION include
+ )
+ install(
+ TARGETS ${PROJECT_NAME}
+ EXPORT export_${PROJECT_NAME}
+ ARCHIVE DESTINATION lib
+ LIBRARY DESTINATION lib
+ RUNTIME DESTINATION bin
+ )
+ ament_export_include_directories(
+ include
+ )
+ ament_export_libraries(
+ ethercat_plugins
+ )
+ ament_export_targets(
+ export_${PROJECT_NAME}
+ )
+ ament_package()
diff --git a/ethercat_driver_ros2/sphinx/images/logo-icube.png b/ethercat_driver_ros2/sphinx/images/logo-icube.png
new file mode 100644
index 0000000000000000000000000000000000000000..cf22f48a3abdcb1f5d91dba020b68c3a425acde5
GIT binary patch
literal 204123
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 DgGOk<`ziTLg#!xm
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