Skip to content

xdaco/kacanopen-fork

 
 

Repository files navigation

Licence drivers_lgpl Licence Build Status

KaCanOpen

KaCanOpen is an easy-to-use CANopen stack, which consists of four parts:

  • Drivers: A wide range of hardware is supported using different CAN drivers. They have been developed by the CanFestival project. Read this for details.

  • Core: This is a library which implements basic CANopen protocols like NMT, SDO and PDO. As an example, you can easily fetch a value from a device (uploading in CANopen terminology) via core.sdo.upload(node_id, index, subindex). It furthermore allows you to register callbacks on certain events or any incoming messages, so one can build arbitrary CANopen nodes (master or slave) using this library.

  • Master: This library is intended to be used for a master node. It detects all nodes in a network and allows to access them via standardized CiA® profiles. For example, on a motor device (profile CiA® 402) you can simply call mymotor.set_entry("Target velocity", 500). A main feature of this library is transparent SDO/PDO access to dictionary entries: By default a value is fetched and set via SDO, but you can configure PDO mappings to instead keep the value up-to-date in background via (more lightweight) PDO messages. The call itself (mymotor.set_entry("Target velocity", 500)) keeps unchanged. The dictionary structure of a device can be loaded automatically from a set of generic, CiA® profile-specific, or device-specific EDS files which is distributed along with KaCanOpen. This allows you to write a meaningful Plug and Play master node with only a few lines of code.

  • ROS Bridge: This library provides a bridge to a ROS network, which makes KaCanOpen especially interesting for robotics. After setting up the CANopen network, the library can publish slave nodes so they are accessible through ROS messages. Special effort is put into the use of standardized message types which allows interaction with other software from the ROS universe. For example, motors can be operated using JointState messages.

KaCanOpen is designed to make use of modern C++11/14 features and to avoid redundant code on the user side wherever possible. This makes the interface neater than it is in other products on the market.

Quick start

First make sure you've got a recent C++ compiler with C++14 support (GCC >= 4.9, Clang >= 3.6), as well as CMake >= 3.2 and Boost >= 1.46.1.

KaCanOpen without the ROS part can be built easily using CMake:

git clone https://github.com/KITmedical/kacanopen.git
cd kacanopen
mkdir build
cd build
cmake -DDRIVER=<driver> -DNO_ROS=On ..
make

<driver> can be one of the following: serial, socket, virtual, lincan, peak_linux.

The examples directory lists some examples on how to use KaCanOpen libraries. You can run them from the build/examples directory.

KaCanOpen including the ROS part must be built using Catkin. Make sure you have ROS Jade installed. Go to your workspace and clone the repository into src:

cd your_catkin_workspace/src
git clone https://github.com/KITmedical/kacanopen.git
cd ..
catkin_make -DDRIVER=<driver>

When building with Catkin, you can excute example programs like that:

cd your_catkin_workspace
source devel/setup.bash
rosrun kacanopen kacanopen_example_motor_and_io_bridge # roscore needs to be running

Complete build instructions can be found here.

Examples

There are several examples has been implemented in the "example" folder.

  • master.cpp: This is an example which shows the usage of the Master library.

  • core.cpp: This is an example which shows the usage of the Core library.

  • pdo.cpp: This example runs a counter completely without SDO transfers.There must be a CiA 401 device which is configured to send 'Read input 8-bit/Digital Inputs 1-8'and 'Read input 8-bit/Digital Inputs 9-16' via TPDO1 and to receive 'Write output 8-bit/Digital Outputs 1-8' via RPDO1.

  • simple_sdo_rw.cpp: This example shows how to access a slave device with SDO read write. Alsmost all the device parameters can be read and or write by SDO transfer only. But this will not be time optimised as SDOs are not meant to be used for real time/ time critical parameters. A Roboteq CANOpen motor driver was used to test the program.

  • simple_pdo_rw.cpp: This example shows how to access a slave device with PDOs only. The slave device must be preconfigured to transmit TPDOs and receive RPDOs. Dynamic pdo mapping for the slave device has not been shown in this exmple, though this can be done by simple SDO transfer.A Roboteq CANOpen motor driver was used to test the program.

  • periodic_tpdo_write.cpp: This example shows how to use periodic PDO transfer by the master. A transmit pdo has been configured which will be received by the device rpdo1 periodically at every 250ms. A CANopenSocket simulated/emulated CiA401 slave.

  • simple_pdo_rw_dynamic_mapping.cpp: This example shows how to access a slave device with PDOs only. The slave device must be preconfigured to transmit TPDOs and receive RPDOs. Dynamic pdo mapping for the slave device has been implemented using SDO transfer.A CANopenSocket simulated/emulated CiA401 slave.

Documentation

Full documentation can be found at https://kitmedical.github.io/kacanopen/.

License

Core, Master and ROS Bridge are licensed under the BSD-3-Clause license. Drivers from CanFestival are licensed under the LGPLv2.1+ license.

TODO

Roadmap

This is a yet incomplete list of features which are planned for the future. Feel free to contact me you have any further ideas or if you want to contribute code.

Short-term

  • Bridge: Rename ros_bridge to something that is not as easily confused with the rosbridge (http://wiki.ros.org/rosbridge_suite), e.g. kacanopen_ros
  • Bridge: Implement a ROS service for accessing any dictionary entry of a slave by name.
  • Bridge: Subscribers / Publishers: Allow configuration of queue_size.
  • Bridge: Trigger publishing on entry change.
  • Core: Automatically map PDOs like they are configured in slave's dictionary.
  • Core: Add methods for manipulating PDO mapping configuration in slave's dictionary.
  • Core: Add ability to remove/change existing PDO transmitters / receivers / mappings.
  • Master: Device: Add methods print_operations(), print_constants() and read_complete_dictionary().

Long-term

  • Master: Make master a CiA 301 compliant node. This means it has to implement some slave functionality. Changes in core library are necessary in order to parse SDO request messages.
  • Project: Implement a slave node library.
  • Project: Check thread safety. Should be achievable with very few changes (mutexes on driver and callback vectors).
  • Bridge: Have a look into ros_control. Can we use it for controlling CiA 402 devices?
  • Core: Support multiple device profiles per slave.
  • Core: Improve error handling.
  • Core: Implement missing protocol parts like LSS, EMERGENCY and SYNC master.

Ideas for a CanOpen slave library

  • Where? Fits best in master library (rename to node e.g.), because it shares many things like dictionary and eds_reader, and master could inherit from the slave class (-> inlining).

  • Core::received_message:

    • implement two missing cases in receive loop (simple delegation to sdo/pdo class)
  • PDO class:

    • Not much to do here. Just provide callbacks for RPDOs. These will be registered by Slave class.
    • Rename process_incoming_message() to process_incoming_tpdo() and add process_incoming_rpdo()
    • Rename add_pdo_received_callback() to add_tpdo_received_callback() and add add_rpdo_received_callback()
  • SDO class:

    • Just like in PDO class, we need to implement client SDOs.
    • Slave class listens for requests and (add_request_callback()) and can react using send_response()
    • add add_client_sdo_callback(SDOReceivedCallback) (callback signature void(const SDOResponse&)) and/or add_request_callback(node_id, SDORequestCallback) (listening only for SDOs with slave's own node_id, callback signature void(index, subindex))
    • add send_response(node_id, index, subindex, vector<uint8_t> data) (chooses segmented/expedited transfer on it's own)
    • add abort_transfer(node_id, index, subindex, errorcode)
  • Slave class:

    • has a node_id! (never used this until now...)
    • m_dictionary of type map<name, Entry>
    • m_index_to_name of type map<index,subindex,name>
    • add_entry(Entry)
    • read_dictionary_from_eds(filename) -> store default values
    • set_value(name,value) -> user should call this for all entries without default value
    • get_value(name) (internal use)
    • register sdo_request_callback -> get name from index, do get_value() and send result using core.sdo.send_response(node_id, index, subindex, value.get_bytes())
    • add_pdo_mapping(cob_id, name) -> core.pdo.add_rpdo_received_callback() with this cob_id -> set_value()...
  • Master class:

    • inherit from slave
    • add at least mandatory entries using add_entry() in constructor
  • EDSReader class:

    • handle default values but only if used by Slave class, otherwise dictionary could contain outdated values.

About

Easy-to-use CanOpen stack and CanOpen-to-ROS bridge.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • C++ 90.1%
  • Python 4.6%
  • C 4.0%
  • Other 1.3%