1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/sys/can/router.c
2020-11-02 21:49:39 +01:00

413 lines
12 KiB
C

/*
* Copyright (C) 2016-2018 OTA keys S.A.
*
* 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 sys_can_dll
* @{
* @file
* @brief Functions for routing RX can frames
*
* @author Toon Stegen <toon.stegen@altran.com>
* @author Vincent Dupont <vincent@otakeys.com>
* @}
*/
#include <stdint.h>
#include <errno.h>
#include "kernel_defines.h"
#include "can/router.h"
#include "can/pkt.h"
#include "can/device.h"
#include "utlist.h"
#include "mutex.h"
#include "assert.h"
#include "memarray.h"
#include "mbox.h"
#define ENABLE_DEBUG 0
#include "debug.h"
#if ENABLE_DEBUG
/* For PRIu16 etc. */
#include <inttypes.h>
#endif
/**
* This is a can_id element
*/
typedef struct filter_el {
can_reg_entry_t entry; /**< filter entry */
canid_t can_id; /**< CAN ID of the element */
canid_t mask; /**< Mask of the element */
void *data; /**< Private data */
} filter_el_t;
/**
* This table contains @p CAN_ROUTER_APP_MAX lists of CAN IDs per interface
*/
static can_reg_entry_t *table[CAN_DLL_NUMOF];
#ifndef CAN_ROUTER_MAX_FILTER
#define CAN_ROUTER_MAX_FILTER 64
#endif
static filter_el_t _filter_buf[CAN_ROUTER_MAX_FILTER];
static memarray_t _filter_array;
static mutex_t lock = MUTEX_INIT;
static filter_el_t *_alloc_filter_el(canid_t can_id, canid_t mask, void *data);
static void _free_filter_el(filter_el_t *el);
static void _insert_to_list(can_reg_entry_t **list, filter_el_t *el);
static filter_el_t *_find_filter_el(can_reg_entry_t *list, can_reg_entry_t *entry, canid_t can_id, canid_t mask, void *data);
static int _filter_is_used(unsigned int ifnum, canid_t can_id, canid_t mask);
#if IS_ACTIVE(ENABLE_DEBUG)
static void _print_filters(void)
{
for (int i = 0; i < (int)CAN_DLL_NUMOF; i++) {
DEBUG("--- Ifnum: %d ---\n", i);
can_reg_entry_t *entry;
LL_FOREACH(table[i], entry) {
filter_el_t *el = container_of(entry, filter_el_t, entry);
DEBUG("App pid=%" PRIkernel_pid ", el=%p, can_id=0x%" PRIx32 ", mask=0x%" PRIx32 ", data=%p\n",
el->entry.target.pid, (void*)el, el->can_id, el->mask, el->data);
}
}
}
#define PRINT_FILTERS() _print_filters()
#else
#define PRINT_FILTERS()
#endif
void can_router_init(void)
{
mutex_init(&lock);
memarray_init(&_filter_array, _filter_buf, sizeof(filter_el_t), CAN_ROUTER_MAX_FILTER);
}
static filter_el_t *_alloc_filter_el(canid_t can_id, canid_t mask, void *data)
{
filter_el_t *el;
el = memarray_alloc(&_filter_array);
if (!el) {
DEBUG("can_router: _alloc_canid_el: out of memory\n");
return NULL;
}
el->can_id = can_id;
el->mask = mask;
el->data = data;
el->entry.next = NULL;
DEBUG("_alloc_canid_el: el allocated with can_id=0x%" PRIx32 ", mask=0x%" PRIx32
", data=%p\n", can_id, mask, data);
return el;
}
static void _free_filter_el(filter_el_t *el)
{
assert(el);
DEBUG("_free_canid_el: el freed with can_id=0x%" PRIx32 ", mask=0x%" PRIx32
", data=%p\n", el->can_id, el->mask, el->data);
memarray_free(&_filter_array, el);
}
/* Insert to the list in a sorted way
* Lower CAN IDs are inserted first */
static void _insert_to_list(can_reg_entry_t **list, filter_el_t *el)
{
can_reg_entry_t *next_entry = *list;
filter_el_t *next_el = container_of(next_entry, filter_el_t, entry);
DEBUG("_insert_to_list: list=%p, el=%p\n", (void *)list, (void *)el);
if (!(*list) || (next_el->can_id > el->can_id)) {
LL_PREPEND(*list, &el->entry);
DEBUG("_insert_to_list: inserting first el, list=%p\n", (void *)list);
}
else {
do {
if (el->can_id <= next_el->can_id) {
DEBUG("_insert_to_list: found next_el can_id:0x%" PRIx32
"\n", next_el->can_id);
LL_PREPEND_ELEM(*list, next_entry, &el->entry);
return;
}
else if (next_el->entry.next == NULL) {
DEBUG("_insert_to_list: insert at the end\n");
LL_APPEND(next_entry, &el->entry);
return;
}
else {
next_entry = next_entry->next;
next_el = container_of(next_entry, filter_el_t, entry);
DEBUG("_insert_to_list: going to next el: %p\n", (void*) next_el);
}
} while (next_el);
}
}
#ifdef MODULE_CAN_MBOX
#define ENTRY_MATCHES(e1, e2) (((e1)->type == (e2)->type) && \
(((e1)->type == CAN_TYPE_DEFAULT && (e1)->target.pid == (e2)->target.pid) ||\
((e1)->type == CAN_TYPE_MBOX && (e1)->target.mbox == (e2)->target.mbox)))
#else
#define ENTRY_MATCHES(e1, e2) ((e1)->target.pid == (e2)->target.pid)
#endif
static filter_el_t *_find_filter_el(can_reg_entry_t *list, can_reg_entry_t *entry, canid_t can_id, canid_t mask, void *data)
{
filter_el_t *el = container_of(list, filter_el_t, entry);
if (!el) {
return el;
}
do {
if ((el->can_id == can_id) && (el->mask == mask) && (el->data == data) &&
ENTRY_MATCHES(&el->entry, entry)) {
DEBUG("_find_filter_el: found el=%p, can_id=%" PRIx32 ", mask=%" PRIx32 ", data=%p\n",
(void *)el, el->can_id, el->mask, el->data);
return el;
}
el = container_of(el->entry.next, filter_el_t, entry);
} while (el);
return NULL;
}
static int _filter_is_used(unsigned int ifnum, canid_t can_id, canid_t mask)
{
filter_el_t *el = container_of(table[ifnum], filter_el_t, entry);
if (!el) {
DEBUG("_filter_is_used: empty list\n");
return 0;
}
do {
if ((el->can_id == can_id) && (el->mask == mask)) {
DEBUG("_filter_is_used: found el=%p, can_id=%" PRIx32 ", mask=%" PRIx32 ", data=%p\n",
(void *)el, el->can_id, el->mask, el->data);
return 1;
}
el = container_of(el->entry.next, filter_el_t, entry);
} while (el);
DEBUG("_filter_is_used: filter not found\n");
return 0;
}
/* register interested users */
int can_router_register(can_reg_entry_t *entry, canid_t can_id, canid_t mask, void *param)
{
filter_el_t *filter;
int ret;
#ifdef MODULE_CAN_MBOX
if (IS_ACTIVE(ENABLE_DEBUG)) {
if (entry->type == CAN_TYPE_DEFAULT) {
DEBUG("can_router_register: ifnum=%d, pid=%" PRIkernel_pid ", can_id=0x%" PRIx32
", mask=0x%" PRIx32 ", data=%p\n", entry->ifnum, entry->target.pid, can_id, mask, param);
} else if (entry->type == CAN_TYPE_MBOX) {
DEBUG("can_router_register: ifnum=%d, mbox=%p, can_id=0x%" PRIx32
", mask=0x%" PRIx32 ", data=%p\n", entry->ifnum, (void *)entry->target.mbox, can_id, mask, param);
}
}
#endif
mutex_lock(&lock);
ret = _filter_is_used(entry->ifnum, can_id, mask);
filter = _alloc_filter_el(can_id, mask, param);
if (!filter) {
mutex_unlock(&lock);
return -ENOMEM;
}
#ifdef MODULE_CAN_MBOX
filter->entry.type = entry->type;
switch (entry->type) {
case CAN_TYPE_DEFAULT:
filter->entry.target.pid = entry->target.pid;
break;
case CAN_TYPE_MBOX:
filter->entry.target.mbox = entry->target.mbox;
break;
}
#else
filter->entry.target.pid = entry->target.pid;
#endif
filter->entry.ifnum = entry->ifnum;
_insert_to_list(&table[entry->ifnum], filter);
mutex_unlock(&lock);
PRINT_FILTERS();
return ret;
}
/* unregister interested users */
int can_router_unregister(can_reg_entry_t *entry, canid_t can_id,
canid_t mask, void *param)
{
filter_el_t *el;
int ret;
#ifdef MODULE_CAN_MBOX
if (IS_ACTIVE(ENABLE_DEBUG)) {
if (entry->type == CAN_TYPE_DEFAULT) {
DEBUG("can_router_unregister: ifnum=%d, pid=%" PRIkernel_pid ", can_id=0x%" PRIx32
", mask=0x%" PRIx32 ", data=%p", entry->ifnum, entry->target.pid, can_id, mask, param);
} else if (entry->type == CAN_TYPE_MBOX) {
DEBUG("can_router_unregister: ifnum=%d, mbox=%p, can_id=0x%" PRIx32
", mask=0x%" PRIx32 ", data=%p\n", entry->ifnum, (void *)entry->target.mbox, can_id, mask, param);
}
}
#endif
mutex_lock(&lock);
el = _find_filter_el(table[entry->ifnum], entry, can_id, mask, param);
if (!el) {
mutex_unlock(&lock);
return -EINVAL;
}
LL_DELETE(table[entry->ifnum], &el->entry);
_free_filter_el(el);
ret = _filter_is_used(entry->ifnum, can_id, mask);
mutex_unlock(&lock);
PRINT_FILTERS();
return ret;
}
static int _send_msg(msg_t *msg, can_reg_entry_t *entry)
{
#ifdef MODULE_CAN_MBOX
switch (entry->type) {
case CAN_TYPE_DEFAULT:
return msg_try_send(msg, entry->target.pid);
case CAN_TYPE_MBOX:
DEBUG("_send_msg: sending msg=%p to mbox=%p\n", (void *)msg, (void *)entry->target.mbox);
return mbox_try_put(entry->target.mbox, msg);
default:
return -ENOTSUP;
}
#else
return msg_try_send(msg, entry->target.pid);
#endif
}
/* send received pkt to all interested users */
int can_router_dispatch_rx_indic(can_pkt_t *pkt)
{
if (!pkt) {
DEBUG("can_router_dispatch_rx_indic: invalid pkt\n");
return -EINVAL;
}
int res = 0;
msg_t msg;
msg.type = CAN_MSG_RX_INDICATION;
int msg_cnt = 0;
DEBUG("can_router_dispatch_rx_indic: pkt=%p, ifnum=%d, can_id=%" PRIx32 "\n",
(void *)pkt, pkt->entry.ifnum, pkt->frame.can_id);
mutex_lock(&lock);
can_reg_entry_t *entry = NULL;
filter_el_t *el;
LL_FOREACH(table[pkt->entry.ifnum], entry) {
el = container_of(entry, filter_el_t, entry);
if ((pkt->frame.can_id & el->mask) == el->can_id) {
DEBUG("can_router_dispatch_rx_indic: found el=%p, data=%p\n",
(void *)el, (void *)el->data);
DEBUG("can_router_dispatch_rx_indic: rx_ind to pid: %"
PRIkernel_pid "\n", entry->target.pid);
atomic_fetch_add(&pkt->ref_count, 1);
msg.content.ptr = can_pkt_alloc_rx_data(&pkt->frame, sizeof(pkt->frame), el->data);
if (IS_ACTIVE(ENABLE_DEBUG)) {
msg_cnt++;
}
if (!msg.content.ptr || (_send_msg(&msg, entry) <= 0)) {
can_pkt_free_rx_data(msg.content.ptr);
atomic_fetch_sub(&pkt->ref_count, 1);
DEBUG("can_router_dispatch_rx_indic: failed to send msg to "
"pid=%" PRIkernel_pid "\n", entry->target.pid);
res = -EBUSY;
break;
}
}
}
mutex_unlock(&lock);
DEBUG("can_router_dispatch_rx: msg send to %d threads\n", msg_cnt);
if (atomic_load(&pkt->ref_count) == 0) {
can_pkt_free(pkt);
}
return res;
}
int can_router_dispatch_tx_conf(can_pkt_t *pkt)
{
msg_t msg;
msg.type = CAN_MSG_TX_CONFIRMATION;
msg.content.value = pkt->handle;
DEBUG("can_router_dispatch_tx_conf: frame=%p, pid=%" PRIkernel_pid "\n",
(void *)&pkt->frame, pkt->entry.target.pid);
if (_send_msg(&msg, &pkt->entry) <= 0) {
return -1;
}
return 0;
}
int can_router_dispatch_tx_error(can_pkt_t *pkt)
{
msg_t msg;
msg.type = CAN_MSG_TX_ERROR;
msg.content.value = pkt->handle;
DEBUG("can_router_dispatch_tx_error: frame=%p, pid=%" PRIkernel_pid "\n",
(void *)&pkt->frame, pkt->entry.target.pid);
if (_send_msg(&msg, &pkt->entry) <= 0) {
return -1;
}
return 0;
}
int can_router_free_frame(struct can_frame *frame)
{
can_pkt_t *pkt = container_of(frame, can_pkt_t, frame);
DEBUG("can_router_free_frame: pkt=%p\n", (void*) pkt);
if (!pkt || (atomic_load(&pkt->ref_count) <= 0)) {
return -1;
}
atomic_fetch_sub(&pkt->ref_count, 1);
if (atomic_load(&pkt->ref_count) == 0) {
can_pkt_free(pkt);
}
return 0;
}