1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/tests/candev/main.c
Wouter Symons a7880ab98d test/candev: add candev test app with native support
This test app bypasses candll and uses the candev abstraction directly.
This has the limitation that you can only use one can driver and one can
device, which is in most use cases sufficient.
2020-03-10 18:13:15 +01:00

215 lines
5.5 KiB
C

/*
* Copyright (C) 2020 Nalys
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @ingroup tests
* @{
*
* @file
* @brief Test application for the candev abstraction
*
* @author Toon Stegen <tstegen@nalys-group.com>
* @author Wouter Symons <wsymons@nalys-group.com>
*
* @}
*/
#define ENABLE_DEBUG (0)
#include <debug.h>
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <isrpipe.h>
#include "shell.h"
#include "can/device.h"
#ifdef BOARD_NATIVE
#include <candev_linux.h>
static candev_linux_t linux_dev;
#else
/* add other candev drivers here */
#endif
#define RX_RINGBUFFER_SIZE 128 /* Needs to be a power of 2! */
static isrpipe_t rxbuf;
static uint8_t rx_ringbuf[RX_RINGBUFFER_SIZE];
static candev_t *candev = NULL;
static int _send(int argc, char **argv)
{
int ret = 0;
struct can_frame frame = {
.can_id = 1,
.can_dlc = 3,
.data[0] = 0xAB,
.data[1] = 0xCD,
.data[2] = 0xEF,
};
if (argc > 1) {
if (argc > 1 + CAN_MAX_DLEN) {
printf("Could not send. Maximum CAN-bytes: %d\n", CAN_MAX_DLEN);
return -1;
}
for (int i = 1; i < argc; i++) {
frame.data[i - 1] = atoi(argv[i]);
}
frame.can_dlc = argc - 1;
}
ret = candev->driver->send(candev, &frame);
if (ret < 0) {
puts("Failed to send CAN-message!");
}
else {
DEBUG("sent using mailbox: %d\n", ret);
}
return 0;
}
static int _receive(int argc, char **argv)
{
uint8_t buf[CAN_MAX_DLEN];
int n = 1;
if (argc > 1) {
n = strtol(argv[1], NULL, 10);
if (n < 1) {
puts("Usage: receive <number>");
return -1;
}
}
for (int i = 0; i < n; i++) {
uint32_t can_id = 0;
uint8_t can_dlc = 0;
puts("Reading from Rxbuf...");
isrpipe_read(&rxbuf, buf, 4); /* can-id */
can_id = ((uint32_t)buf[0] << 24) |
((uint32_t)buf[1] << 16) |
((uint32_t)buf[2] << 8) |
((uint32_t)buf[3]);
isrpipe_read(&rxbuf, buf, 1); /* can-dlc */
can_dlc = buf[0];
isrpipe_read(&rxbuf, buf, can_dlc); /* data */
printf("id: %" PRIx32 " dlc: %" PRIx8 " Data: \n", can_id, can_dlc);
for (int i = 0; i < can_dlc; i++) {
printf("0x%X ", buf[i]);
}
puts("");
}
return 0;
}
static const shell_command_t shell_commands[] = {
{ "send", "send some data", _send },
{ "receive", "receive some data", _receive },
{ NULL, NULL, NULL }
};
static void _can_event_callback(candev_t *dev, candev_event_t event, void *arg)
{
(void)arg;
struct can_frame *frame;
switch (event) {
case CANDEV_EVENT_ISR:
DEBUG("_can_event: CANDEV_EVENT_ISR\n");
dev->driver->isr(candev);
break;
case CANDEV_EVENT_WAKE_UP:
DEBUG("_can_event: CANDEV_EVENT_WAKE_UP\n");
break;
case CANDEV_EVENT_TX_CONFIRMATION:
DEBUG("_can_event: CANDEV_EVENT_TX_CONFIRMATION\n");
break;
case CANDEV_EVENT_TX_ERROR:
DEBUG("_can_event: CANDEV_EVENT_TX_ERROR\n");
break;
case CANDEV_EVENT_RX_INDICATION:
DEBUG("_can_event: CANDEV_EVENT_RX_INDICATION\n");
frame = (struct can_frame *)arg;
DEBUG("\tid: %" PRIx32 " dlc: %" PRIx8 " Data: \n\t", frame->can_id,
frame->can_dlc);
for (uint8_t i = 0; i < frame->can_dlc; i++) {
DEBUG("0x%X ", frame->data[i]);
}
DEBUG(" ");
/* Store in buffer until user requests the data */
isrpipe_write_one(&rxbuf,
(uint8_t)((frame->can_id & 0x1FFFFFFF) >> 24));
isrpipe_write_one(&rxbuf,
(uint8_t)((frame->can_id & 0xFF0000) >> 16));
isrpipe_write_one(&rxbuf, (uint8_t)((frame->can_id & 0xFF00) >> 8));
isrpipe_write_one(&rxbuf, (uint8_t)((frame->can_id & 0xFF)));
isrpipe_write_one(&rxbuf, frame->can_dlc);
for (uint8_t i = 0; i < frame->can_dlc; i++) {
isrpipe_write_one(&rxbuf, frame->data[i]);
}
break;
case CANDEV_EVENT_RX_ERROR:
DEBUG("_can_event: CANDEV_EVENT_RX_ERROR\n");
break;
case CANDEV_EVENT_BUS_OFF:
dev->state = CAN_STATE_BUS_OFF;
break;
case CANDEV_EVENT_ERROR_PASSIVE:
dev->state = CAN_STATE_ERROR_PASSIVE;
break;
case CANDEV_EVENT_ERROR_WARNING:
dev->state = CAN_STATE_ERROR_WARNING;
break;
default:
DEBUG("_can_event: unknown event\n");
break;
}
}
int main(void)
{
puts("candev test application\n");
isrpipe_init(&rxbuf, (uint8_t *)rx_ringbuf, sizeof(rx_ringbuf));
#ifdef BOARD_NATIVE
puts("Initializing Linux Can device");
candev_linux_init( &linux_dev, &(candev_linux_conf[0])); /* vcan0 */
candev = (candev_t *)&linux_dev;
#else
/* add initialization for other candev drivers here */
#endif
assert(candev);
candev->event_callback = _can_event_callback;
candev->isr_arg = NULL;
candev->driver->init(candev);
char line_buf[SHELL_DEFAULT_BUFSIZE];
shell_run(shell_commands, line_buf, SHELL_DEFAULT_BUFSIZE);
return 0;
}