mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
Merge pull request #18131 from firas-hamdi/feat/apply_filters_to_target_mailbox
drivers/mcp2515: apply filters to target mailbox
This commit is contained in:
commit
54544c0a36
@ -421,6 +421,8 @@ static int _get(candev_t *candev, canopt_t opt, void *value, size_t max_len)
|
||||
static int _set_filter(candev_t *dev, const struct can_filter *filter)
|
||||
{
|
||||
DEBUG("inside _set_filter of MCP2515\n");
|
||||
assert(filter->target_mailbox < MCP2515_RX_MAILBOXES);
|
||||
|
||||
struct can_filter f = *filter;
|
||||
int res = -1;
|
||||
enum mcp2515_mode mode;
|
||||
@ -449,58 +451,66 @@ static int _set_filter(candev_t *dev, const struct can_filter *filter)
|
||||
f.can_mask &= CAN_SFF_MASK;
|
||||
}
|
||||
|
||||
/* Browse on each mailbox to find an empty space */
|
||||
int mailbox_index = 0;
|
||||
|
||||
while (mailbox_index < MCP2515_RX_MAILBOXES) {
|
||||
/* mask unused */
|
||||
if (dev_mcp->masks[mailbox_index] == 0) {
|
||||
/* mask unused */
|
||||
if (dev_mcp->masks[f.target_mailbox] == 0) {
|
||||
if (mutex_trylock(&_mcp_mutex)) {
|
||||
/* set mask */
|
||||
mcp2515_set_mask(dev_mcp, mailbox_index, f.can_mask);
|
||||
mcp2515_set_mask(dev_mcp, f.target_mailbox, f.can_mask);
|
||||
/* set filter */
|
||||
mcp2515_set_filter(dev_mcp, MCP2515_FILTERS_MB0 * mailbox_index,
|
||||
f.can_id);
|
||||
mcp2515_set_filter(dev_mcp, MCP2515_FILTERS_MB0 * f.target_mailbox,
|
||||
f.can_id);
|
||||
mutex_unlock(&_mcp_mutex);
|
||||
}
|
||||
else {
|
||||
DEBUG("setfilt2_Failed to lock mutex\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* save filter */
|
||||
dev_mcp->masks[f.target_mailbox] = f.can_mask;
|
||||
dev_mcp->filter_ids[f.target_mailbox][0] = f.can_id;
|
||||
}
|
||||
|
||||
/* mask existed and same mask */
|
||||
else if (dev_mcp->masks[f.target_mailbox] == f.can_mask) {
|
||||
/* find an empty space if it exists */
|
||||
int filter_pos = 1; /* first one is already filled */
|
||||
/* stop at the end of mailbox or an empty space found */
|
||||
while (filter_pos < _max_filters(f.target_mailbox) &&
|
||||
dev_mcp->filter_ids[f.target_mailbox][filter_pos] != 0) {
|
||||
filter_pos++;
|
||||
}
|
||||
|
||||
/* an empty space is found */
|
||||
if (filter_pos < _max_filters(f.target_mailbox)) {
|
||||
/* set filter on this memory space */
|
||||
if (mutex_trylock(&_mcp_mutex)) {
|
||||
mcp2515_set_filter(dev_mcp,
|
||||
MCP2515_FILTERS_MB0 * f.target_mailbox + filter_pos,
|
||||
f.can_id);
|
||||
mutex_unlock(&_mcp_mutex);
|
||||
}
|
||||
else {
|
||||
DEBUG("setfilt3_Failed to lock mutex");
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* save filter */
|
||||
dev_mcp->masks[mailbox_index] = f.can_mask;
|
||||
dev_mcp->filter_ids[mailbox_index][0] = f.can_id;
|
||||
|
||||
/* function succeeded */
|
||||
break;
|
||||
dev_mcp->filter_ids[f.target_mailbox][filter_pos] = f.can_id;
|
||||
}
|
||||
|
||||
/* mask existed and same mask */
|
||||
else if (dev_mcp->masks[mailbox_index] == f.can_mask) {
|
||||
/* find en empty space if it exist */
|
||||
int filter_pos = 1; /* first one is already filled */
|
||||
/* stop at the end of mailbox or an empty space found */
|
||||
while (filter_pos < _max_filters(mailbox_index) &&
|
||||
dev_mcp->filter_ids[mailbox_index][filter_pos] != 0) {
|
||||
filter_pos++;
|
||||
/* No empty space is found */
|
||||
else {
|
||||
DEBUG_PUTS("no empty space found");
|
||||
if (mutex_trylock(&_mcp_mutex)) {
|
||||
mcp2515_set_mode(dev_mcp, mode);
|
||||
mutex_unlock(&_mcp_mutex);
|
||||
}
|
||||
|
||||
/* an empty space is found */
|
||||
if (filter_pos < _max_filters(mailbox_index)) {
|
||||
/* set filter on this memory space */
|
||||
if (mutex_trylock(&_mcp_mutex)) {
|
||||
mcp2515_set_filter(dev_mcp,
|
||||
MCP2515_FILTERS_MB0 * mailbox_index + filter_pos,
|
||||
f.can_id);
|
||||
mutex_unlock(&_mcp_mutex);
|
||||
}
|
||||
else {
|
||||
DEBUG("setfilt2_Failed to lock mutex");
|
||||
return -1;
|
||||
}
|
||||
|
||||
/* save filter */
|
||||
dev_mcp->filter_ids[mailbox_index][filter_pos] = f.can_id;
|
||||
|
||||
/* function succeeded */
|
||||
break;
|
||||
else {
|
||||
DEBUG_PUTS("setfilt4_Failed to lock mutex");
|
||||
return -1;
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
mailbox_index++;
|
||||
}
|
||||
|
||||
if (mutex_trylock(&_mcp_mutex)) {
|
||||
@ -508,7 +518,7 @@ static int _set_filter(candev_t *dev, const struct can_filter *filter)
|
||||
mutex_unlock(&_mcp_mutex);
|
||||
}
|
||||
else {
|
||||
DEBUG("setfilt3_Failed to lock mutex");
|
||||
DEBUG("setfilt5_Failed to lock mutex");
|
||||
return -1;
|
||||
}
|
||||
|
||||
|
@ -303,9 +303,6 @@ extern "C" {
|
||||
#define MCP2515_RXB0CTRL_MODE_RECV_FILTER 0x00
|
||||
#define MCP2515_RXB0CTRL_RXM0 0x20
|
||||
#define MCP2515_RXB0CTRL_RXM1 0x40
|
||||
#define MCP2515_RXB0CTRL_MODE_RECV_STD_OR_EXT 0x00
|
||||
#define MCP2515_RXB0CTRL_MODE_RECV_STD MCP2515_RXB0CTRL_RXM0
|
||||
#define MCP2515_RXB0CTRL_MODE_RECV_EXT MCP2515_RXB0CTRL_RXM1
|
||||
#define MCP2515_RXB0CTRL_MODE_RECV_ALL (MCP2515_RXB0CTRL_RXM1 | \
|
||||
MCP2515_RXB0CTRL_RXM0)
|
||||
|
||||
@ -316,9 +313,6 @@ extern "C" {
|
||||
#define MCP2515_RXB1CTRL_MODE_RECV_FILTER 0x00
|
||||
#define MCP2515_RXB1CTRL_RXM0 0x20
|
||||
#define MCP2515_RXB1CTRL_RXM1 0x40
|
||||
#define MCP2515_RXB1CTRL_MODE_RECV_STD_OR_EXT 0x00
|
||||
#define MCP2515_RXB1CTRL_MODE_RECV_STD MCP2515_RXB1CTRL_RXM0
|
||||
#define MCP2515_RXB1CTRL_MODE_RECV_EXT MCP2515_RXB1CTRL_RXM1
|
||||
#define MCP2515_RXB1CTRL_MODE_RECV_ALL (MCP2515_RXB1CTRL_RXM1 | \
|
||||
MCP2515_RXB1CTRL_RXM0)
|
||||
/** @} */
|
||||
|
@ -101,6 +101,9 @@ struct can_frame {
|
||||
struct can_filter {
|
||||
canid_t can_id; /**< CAN ID */
|
||||
canid_t can_mask; /**< Mask */
|
||||
#if defined(MODULE_MCP2515)
|
||||
uint8_t target_mailbox; /**< The mailbox to apply the filter to */
|
||||
#endif
|
||||
};
|
||||
|
||||
/**
|
||||
|
@ -223,14 +223,28 @@ if (IS_ACTIVE(CONFIG_USE_LOOPBACK_MODE)) {
|
||||
|
||||
if (IS_ACTIVE(MCP2515_RECV_FILTER_EN)) {
|
||||
/* CAN filters examples */
|
||||
struct can_filter filter[3];
|
||||
struct can_filter filter[4];
|
||||
filter[0].can_mask = 0x7FF;
|
||||
filter[0].can_id = 0x001; /* messages with CAN ID 0x001 will be received in mailbox 0 */
|
||||
filter[0].can_id = 0x001;
|
||||
#if defined(MODULE_MCP2515)
|
||||
filter[0].target_mailbox = 0; /* messages with CAN ID 0x001 will be received in mailbox 0 */
|
||||
#endif
|
||||
filter[1].can_mask = 0x7FF;
|
||||
filter[1].can_id = 0x003; /* messages with CAN ID 0x003 will be received in mailbox 0 */
|
||||
filter[1].can_id = 0x002;
|
||||
#if defined(MODULE_MCP2515)
|
||||
filter[1].target_mailbox = 1; /* messages with CAN ID 0x002 will be received in mailbox 1 */
|
||||
#endif
|
||||
filter[2].can_mask = 0x7FF;
|
||||
filter[2].can_id = 0x002; /* messages with CAN ID 0x002 will be received in mailbox 1 */
|
||||
for (uint8_t i = 0; i < 3; i++) {
|
||||
filter[2].can_id = 0x003;
|
||||
#if defined(MODULE_MCP2515)
|
||||
filter[2].target_mailbox = 0; /* messages with CAN ID 0x003 will be received in mailbox 0 */
|
||||
#endif
|
||||
filter[3].can_mask = 0x7FF;
|
||||
filter[3].can_id = 0x004;
|
||||
#if defined(MODULE_MCP2515)
|
||||
filter[3].target_mailbox = 0; /* this filter won't be applied. Reason is no space found in the first mailbox as it supports only two filters */
|
||||
#endif
|
||||
for (uint8_t i = 0; i < 4; i++) {
|
||||
candev->driver->set_filter(candev, &filter[i]);
|
||||
}
|
||||
/* All other messages won't be received */
|
||||
|
Loading…
Reference in New Issue
Block a user