Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Implement filtering for multiple CAN ids #20

Open
wants to merge 3 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
14 changes: 14 additions & 0 deletions API.md
Original file line number Diff line number Diff line change
Expand Up @@ -247,6 +247,20 @@ if ((packetId & mask) == id) {

Returns `1` on success, `0` on failure.

### Advanced Filtering

```
CAN.multiFilter(ids, count);
```

* `ids` - array of 11-bit ids
* `count` - count of entries in `ids`

Only packets with an id in `ids` are acknowleged and received, other packets are ignored.
If `count` is larger than the filters supported by your CAN hardware additional frame ids might be received.

Returns `1` on success, `0` on failure.

## Other modes

### Loopback mode
Expand Down
5 changes: 5 additions & 0 deletions src/CANController.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -195,6 +195,11 @@ int CANControllerClass::filterExtended(long /*id*/, long /*mask*/)
return 0;
}

int CANControllerClass::multiFilter(int * /*ids*/, unsigned /*mask*/)
{
return 0;
}

int CANControllerClass::observe()
{
return 0;
Expand Down
2 changes: 2 additions & 0 deletions src/CANController.h
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,8 @@ class CANControllerClass : public Stream {
virtual int filterExtended(long id) { return filterExtended(id, 0x1fffffff); }
virtual int filterExtended(long id, long mask);

virtual int multiFilter(int *ids, unsigned count);

virtual int observe();
virtual int loopback();
virtual int sleep();
Expand Down
135 changes: 131 additions & 4 deletions src/MCP2515.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -21,10 +21,10 @@
#define FLAG_RXnIF(n) (0x01 << n)
#define FLAG_TXnIF(n) (0x04 << n)

#define REG_RXFnSIDH(n) (0x00 + (n * 4))
#define REG_RXFnSIDL(n) (0x01 + (n * 4))
#define REG_RXFnEID8(n) (0x02 + (n * 4))
#define REG_RXFnEID0(n) (0x03 + (n * 4))
#define REG_RXFnSIDH(n) ((n) < 3 ? (0x00 + (n) * 4) : (0x10 + ((n) - 3) * 4))
#define REG_RXFnSIDL(n) ((n) < 3 ? (0x01 + (n) * 4) : (0x11 + ((n) - 3) * 4))
#define REG_RXFnEID8(n) ((n) < 3 ? (0x02 + (n) * 4) : (0x12 + ((n) - 3) * 4))
#define REG_RXFnEID0(n) ((n) < 3 ? (0x03 + (n) * 4) : (0x13 + ((n) - 3) * 4))

#define REG_RXMnSIDH(n) (0x20 + (n * 0x04))
#define REG_RXMnSIDL(n) (0x21 + (n * 0x04))
Expand Down Expand Up @@ -313,6 +313,133 @@ int MCP2515Class::filter(int id, int mask)
return 1;
}

unsigned char hammingDistance(int x, int y)
{
unsigned char distance = 0;
int xorXY = x ^ y;

for(int i = 0; i < 8 * sizeof(int); i++) {
if(xorXY & (1 << i))
distance++;
}

return distance;
}

int MCP2515Class::multiFilter(int *ids, unsigned count)
{
long filterMasks[2];
long filterIds[6];

if(count <= 6) {
// the MCP2515 has enough registers to filter all ids
for (int n = 0; n < 2; n++)
filterMasks[n] = 0x7ff;

for (int n = 0; n < 6; n++) {
if (n < count)
filterIds[n] = ids[n];
else
filterIds[n] = 0;
}
}
else {
// we have more ids than filters, we find the two most distant ids and use
// them as base for anding together all other ids
unsigned char maxDistance = 0;

for(int i = 0; i < count; i++) {
for(int j = i; j < count; j++) {
unsigned char distance = hammingDistance(ids[i], ids[j]);

if(distance > maxDistance) {
filterIds[0] = ids[i];
filterIds[2] = ids[j];
maxDistance = distance;
}
}
}

filterMasks[0] = 0x7ff;
filterMasks[1] = 0x7ff;
filterIds[1] = 0;
filterIds[3] = 0;
filterIds[4] = 0;
filterIds[5] = 0;

for(int i = 0; i < count; i++) {
unsigned char distance0 = hammingDistance(ids[i], filterMasks[0]);
unsigned char distance1 = hammingDistance(ids[i], filterMasks[1]);

#if 0
Serial.print("id ");
Serial.print(ids[i]);
Serial.print(" distance0: ");
Serial.print(distance0);
Serial.print(" distance1: ");
Serial.print(distance1);
Serial.println();
#endif

if(distance0 < distance1) {
filterMasks[0] = filterMasks[0] & ~(filterIds[0] ^ ids[i]);
filterIds[0] &= ids[i];
}
else {
filterMasks[1] = filterMasks[1] & ~(filterIds[2] ^ ids[i]);
filterIds[2] &= ids[i];
}
}

#if 0
Serial.print("filterMasks[0] = ");
Serial.print(filterMasks[0], 16);
Serial.print(" filterIds[0] = ");
Serial.println(filterIds[0], 16);

Serial.print("filterMasks[1] = ");
Serial.print(filterMasks[1], 16);
Serial.print(" filterIds[2] = ");
Serial.println(filterIds[2], 16);
#endif
}



// config mode
writeRegister(REG_CANCTRL, 0x80);
if (readRegister(REG_CANCTRL) != 0x80) {
return 0;
}

for (int n = 0; n < 2; n++) {
// for now standard only (TODO: extended)
writeRegister(REG_RXBnCTRL(n), FLAG_RXM0);
writeRegister(REG_RXBnCTRL(n), FLAG_RXM0);

writeRegister(REG_RXMnSIDH(n), filterMasks[n] >> 3);
writeRegister(REG_RXMnSIDL(n), filterMasks[n] << 5);
writeRegister(REG_RXMnEID8(n), 0);
writeRegister(REG_RXMnEID0(n), 0);
}

for (int n = 0; n < 6; n++) {
writeRegister(REG_RXFnSIDH(n), filterIds[n] >> 3);
writeRegister(REG_RXFnSIDL(n), filterIds[n] << 5);

writeRegister(REG_RXFnEID8(n), 0);
writeRegister(REG_RXFnEID0(n), 0);
}

// normal mode
writeRegister(REG_CANCTRL, 0x00);
if (readRegister(REG_CANCTRL) != 0x00) {
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You should probably put this into the same mode as it previously was (read mode before setting to config?). Or document that this function explicitely puts the controller into normal mode, so users are not surprised.

Copy link

@timurrrr timurrrr Jul 29, 2020

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The datasheet says

In Listen-Only mode, both valid and invalid messages will
be received, regardless of filters and masks or the
Receive Buffer Operating Mode bits, RXMn.

This code matches the existing behavior implemented in filter() that resets the mode to "normal" on lines 307-311.

return 0;
}

return 1;
}

int MCP2515Class::filterExtended(long id, long mask)
{
id &= 0x1FFFFFFF;
Expand Down
2 changes: 2 additions & 0 deletions src/MCP2515.h
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,8 @@ class MCP2515Class : public CANControllerClass {
virtual int filter(int id, int mask);
using CANControllerClass::filterExtended;
virtual int filterExtended(long id, long mask);
using CANControllerClass::multiFilter;
virtual int multiFilter(int *ids, unsigned count);

virtual int observe();
virtual int loopback();
Expand Down