CAPLin stands for CAN Application Programming for Linux. Think of it as an application framework for easily developing SocketCAN applications in the C programming language. It is the perfect tool for quickly programming CAN nodes for prototyping, testing, simulation and even production purposes.
CAPLin resembles Vector CAPL scripts. Just like CAPL, it's an event driven programming environment. As an extra benefit, you code directly in C, enabling you to:
- Include and link all your favorite C libraries.
- Use development tools and IDEs that you are already familiar with.
- Debug your code without limitations.
- Reuse your code on your microcontroller based CAN node.
The only real downside, compared to a CAPL script, is that you don't have direct access to the CAN messages and signals from a CANdb DBC database file. However, you can circumvent this by using the cantools package to generate a C-source and -header file from your DBC database file, and then integrate these into your CAPLin application.
The PragmaticLinux blog dedicated an entire article towards getting started with the CAPLin framework:
To obtain and build CAPLin applications, make sure that your Linux system has C development and Git related packages installed:
- Debian/Ubuntu:
sudo apt install git gcc g++ gdb make cmake
- Fedora:
sudo dnf install git gcc g++ gdb make cmake
- openSUSE:
sudo zypper install git gcc gcc-c++ gdb make cmake
Additionally, it is assumed that you already connected and configured a SocketCAN supported CAN interface on your Linux system. The following tutorial explains in detail how to do this:
You could even use CAPLin on a Raspberry PI in combination with a low-cost Microchip MCP2515 CAN controller based add-on board:
In case you do not have a SocketCAN supported CAN interface and/or a physical CAN bus to interact with, you can still experiment with CAPLin using a virtual CAN interface on Linux:
To get the code, clone the Git repository to a subdirectory, for example inside your own home directory:
git clone https://github.com/pragmaticlinuxblog/caplin.git ~/CAPLin
After cloning the Git repository, you can prepare the build environment with the help of CMake:
cd ~/CAPLin/build
cmake ..
Note that this only has to be done once. From now on you can build the application by running this command from inside the build
subdirectory:
make all
Alternatively, you can leverage the build-in functionality of an IDE such as Visual Studio Code to perform all these steps, including running and debugging your CAPLin application:
After building your CAPLin application, you can run it directly from the build
subdirectory:
./canapp
Note that your CAPLin application automatically detects and connects to the first SocketCAN network interface it finds on your system. When multiple SocketCAN network interfaces are available, you can select the one to use by specifying its name as a command-line argument, e.g. ./canapp can1
.
Once your application runs, you can press ESC or CTRL+C to exit.
Refer to CAPLin's help info for additional details:
Optionally, you can install your CAPLin application system-wide, making it available to all users. The CMakeLists.txt contains details on how to perform this step. To install the application on your Linux system, run this command from the build
subdirectory:
sudo make install
What does such an event driven CAPLin application look like? Here is a basic demonstration application to whet your appetite. It:
- Displays information when the application starts and stops.
- Displays the CAN identifier of all received CAN messages.
- Transmits a CAN message when the t key is pressed on the keyboard.
- Creates a timer that generates an event every 1000 milliseconds.
#include "caplin.h"
tTimer myTimer;
void OnTimer(void)
{
static uint32_t eventCounter = 0;
printf("Timer event triggered %lu times\n", ++eventCounter);
TimerRestart(myTimer);
}
void OnStart(void)
{
printf("CAPlin application started at: %llu\n", UtilSystemTime());
printf("Currently connected to CAN network interface: %s\n", canDevice);
myTimer = TimerCreate(OnTimer);
TimerStart(myTimer, 1000);
}
void OnStop(void)
{
printf("CAPlin application stopped at: %llu\n", UtilSystemTime());
}
void OnMessage(tCanMsg const * msg)
{
printf("Received message with ID: %x\n", msg->id);
}
void OnKey(char key)
{
tCanMsg txMsg =
{
.id = 0x201, .ext = false, .len = 2, .data = { 1, 2 }
};
if (key == 't')
{
printf("Sending message with ID: %x\n", txMsg.id);
CanTransmit(&txMsg);
}
}
That's all you need. As you can see, thanks to the CAPLin framework, you do not have to dive deep into low-level SocketCAN, timer or thread programming. It's all built into the framework. Batteries included so to say.
The screenshot on the left shows the CAPLin demonstration application in action. The screenshot on the right shows the traffic on the CAN bus as a reference, using candump
from the can-utils package:
The CAPLin framework includes several example applications to help you further understand how to code with the framework. You can find these in the examples
subdirectory. You can build and run each example, just like any other CAPLin application:
- Example 1 - Ping Pong. Echoes all received CAN messages back with an incremented CAN identifier.
- Example 2 - On Key Transmit. Transmits a CAN message each time a specific key is pressed.
- Example 3 - Periodic Transmit. Transmits a CAN message periodically with the help of a timer.
- Example 4 - CAN Logger. Logs all received CAN messages to the screen.
- Example 5 - Interface Override. Programmatically sets the SocketCAN network interface to connect to.