Example
A simple example of a program that reads from the CAN bus and another program that writes to the CAN Bus is in directory “example”.
There are two Makefiles there that can be used to compile the example programs.
How to build
If you don’t have installed socan, use local libraries:
(cd .. && make)
make -f Makefile.local
If you have installed socan in /usr/local, use global libraries:
make -f Makefile.installed
How to run
Enter:
./runtest.sh
The output on the console should look like this:
Start 'read' program...
read: initialize CAN object for reading.
COB: 1 Port: 0 Length: 4 Timeout: 10000
read: wait for data
Start 'write' program...
write: initialize CAN object for writing.
COB: 1 Port: 1 Length: 4 Timeout: 10000
write: send data:
00 01 02 03
read: socan_read returned: SOCAN_OK
write: socan_write returned: SOCAN_OK
read: data received:
00 01 02 03
How it works
Here is the Makefile, “Makefile.installed” that uses socan libraries and headers installed in directory “/usr/local”:
PROGRAMS=read write
PREFIX=/usr/local
INC_DIR=$(PREFIX)/include
LIB_DIR=$(PREFIX)/lib64
LD_OPT=-Wl,--no-as-needed,-rpath,'$(PREFIX)/lib64'
all: $(PROGRAMS)
clean:
rm -f *.o $(PROGRAMS)
$(PROGRAMS): %: %.o $(LIB_DIR)/libsocan.so
gcc -L $(LIB_DIR) $(LD_OPT) -ggdb -o $@ $< -lsocan -lrt
%.o: %.c
gcc -Wall -ggdb -DB_LINUX -I$(INC_DIR) -o $@ -c $<
Here is the complete source of program “read.c”:
/**
* socan - SocketCAN higher level library
*
* Copyright 2025 Helmholtz-Zentrum Berlin für Materialien und Energie GmbH
* <https://www.helmholtz-berlin.de>
* Author: Goetz Pfeiffer <Goetz.Pfeiffer@helmholtz-berlin.de>
* socan is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* socan is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with socan. If not, see <https://www.gnu.org/licenses/>.
*/
#include <stdio.h>
#include <socan.h>
#define DEVICE0 "vcan0"
#define DEVICE1 "vcan1"
#define PORT 0
#define COB 1
#define OBJLEN 4
#define TIMEOUT_MS 10000
int main( int argc, char *argv[])
{
socan_hdl hdl;
socan_rc rc;
unsigned char buffer[SOCAN_FRAME_LENGTH];
socan_realtime_setup(false, false, 0, 0);
socan_add_port(DEVICE0);
socan_add_port(DEVICE1);
if (!socan_init())
{
printf("read: socan_init failed\n");
return(1);
}
hdl= socan_open();
if (hdl==NULL)
{
printf("read: socan_open failed\n");
return(1);
}
printf("read: initialize CAN object for reading.\n"
"COB: %d Port: %d Length: %d Timeout: %d\n",
COB, PORT, OBJLEN, TIMEOUT_MS);
rc= socan_add_obj(hdl,
PORT,
COB,
OBJLEN,
TIMEOUT_MS,
SOCAN_READ);
if (socan_rc_is_error(rc))
{
printf("read: socan_add_obj failed\n");
return(1);
}
printf("read: wait for data\n");
rc= socan_read(hdl,
PORT,
COB,
buffer);
if (socan_rc_is_error(rc))
{
printf("read: socan_read returned an error: %d (%s)\n",
rc, socan_str_rc(rc));
return(1);
}
printf("read: socan_read returned: %s\n", socan_str_rc(rc));
printf("read: data received:\n");
for(int i=0; i<OBJLEN; i++)
printf(" %02x", buffer[i]);
printf("\n");
}
Here is the complete source of program “write.c”:
/**
* socan - SocketCAN higher level library
*
* Copyright 2025 Helmholtz-Zentrum Berlin für Materialien und Energie GmbH
* <https://www.helmholtz-berlin.de>
* Author: Goetz Pfeiffer <Goetz.Pfeiffer@helmholtz-berlin.de>
* socan is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* socan is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with socan. If not, see <https://www.gnu.org/licenses/>.
*/
#include <stdio.h>
#include <unistd.h>
#include <socan.h>
#define DEVICE0 "vcan0"
#define DEVICE1 "vcan1"
#define PORT 1
#define COB 1
#define OBJLEN 4
#define TIMEOUT_MS 10000
int main( int argc, char *argv[])
{
socan_hdl hdl;
socan_rc rc;
unsigned char buffer[SOCAN_FRAME_LENGTH];
int i;
for(i=0; i<SOCAN_FRAME_LENGTH; buffer[i]= i, i++);
socan_realtime_setup(false, false, 0, 0);
socan_add_port(DEVICE0);
socan_add_port(DEVICE1);
if (!socan_init())
{
printf("write: socan_init failed\n");
return(1);
}
hdl= socan_open();
if (hdl==NULL)
{
printf("write: socan_open failed\n");
return(1);
}
printf("write: initialize CAN object for writing.\n"
"COB: %d Port: %d Length: %d Timeout: %d\n",
COB, PORT, OBJLEN, TIMEOUT_MS);
rc= socan_add_obj(hdl,
PORT,
COB,
OBJLEN,
TIMEOUT_MS,
SOCAN_WRITE);
if (socan_rc_is_error(rc))
{
printf("write: socan_add_obj failed\n");
return(1);
}
printf("write: send data:\n");
for(int i=0; i<OBJLEN; i++)
printf(" %02x", buffer[i]);
printf("\n");
rc= socan_write(hdl,
PORT,
COB,
buffer);
if (socan_rc_is_error(rc))
{
printf("write: socan_write returned an error: %d (%s)\n",
rc, socan_str_rc(rc));
return(1);
}
printf("write: socan_write returned: %s\n", socan_str_rc(rc));
#if 0
printf("write: Sleep for 5 seconds before terminating...\n");
sleep(5);
#endif
}
Here is the shell script “runtest.sh”:
#!/bin/bash
SCRIPT_FULL_NAME=$(readlink -e "$0")
MYDIR=$(dirname "$SCRIPT_FULL_NAME")
MYNAME=$(basename "$SCRIPT_FULL_NAME")
if [ "$1" = "-h" ] || [ "$1" = "--help" ]; then
echo "$MYNAME : run a simple read/write test."
echo
echo "Usage:"
echo " $MYNAME [OPTIONS]"
echo
echo "OPTIONS may be:"
echo
echo " -h --help: This help"
exit 0
fi
cd "$MYDIR" || exit 1
echo "Set up virtual CAN interfaces:"
../bin/init-can.sh -d vcan -p 0 3
echo
echo "Link vcan0 to vcan1"
../bin/link-vcan-devices.sh link 0 1
echo
echo "Start 'read' program..."
(nice ./read &)
echo
sleep 1
echo
echo "Start 'write' program..."
./write