mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
215 lines
5.5 KiB
C
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;
|
||
|
}
|