@essense/iso-claim
Version:
Library & background program that implements the ISO address claim procedure for CANbus (i.e. for NMEA2000 devices)
288 lines • 24.7 kB
JavaScript
;
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
const debug_1 = __importDefault(require("debug"));
const events_1 = require("events");
const canboatjs_1 = require("@canboat/canboatjs");
// @ts-ignore
const socketcan_1 = require("socketcan");
const debug = debug_1.default('CANBus');
class CANBus extends events_1.EventEmitter {
constructor(bus) {
super();
this.onChannelMessage = (msg) => {
const message = this.CANtoN2Kmessage(msg);
this.emit('N2KMessage', message);
try {
this.parser.parsePgnData(message, message.length, message.data);
}
catch (e) {
debug('Error whilst parsing message: ' + e.message);
}
};
this.onChannelStopped = () => {
this.open = false;
this.channel = null;
this.emit('disconnected');
debug(`Disconnected from ${this.bus}`);
};
this.network = {};
this.address = -1;
this.bus = bus || 'can0';
this.channel = null;
this.open = false;
this.parser = new canboatjs_1.FromPgn();
this.parser.on('pgn', (pgn) => {
this.handlePGN(pgn);
});
this.parser.on('error', (pgn, err) => {
debug(`Error for pgn ${pgn}: ${err}`);
this.emit('error', err);
});
this.parser.on('warning', (pgn, err) => {
debug(`Warning for pgn ${pgn}: ${err}`);
this.emit('warning', err);
});
this.connect();
}
getAddress() {
return this.address;
}
setAddress(address) {
this.address = address;
}
addDevice(src, device) {
this.network = Object.assign(Object.assign({}, this.network), { [src]: Object.assign(Object.assign({}, (this.network[src] || {})), device) });
// Clean up raw IDs & raw can data
Object.keys(this.network).forEach(source => {
Object.keys(this.network[source]).forEach(key => {
let node = this.network[source][key];
if (!node || typeof node !== 'object') {
return;
}
// Remove raw CAN data
if (node.hasOwnProperty('data')) {
delete node.data;
}
// Remove raw CAN ID
if (node.hasOwnProperty('id')) {
delete node.id;
}
// Flatten fields
if (node.hasOwnProperty('fields')) {
node = Object.assign(Object.assign({}, node), node.fields);
delete node.fields;
}
this.network[source][key] = Object.assign({}, node);
});
});
debug(`[addDevice] adding/modifying device ${src}. Network has changed: ${JSON.stringify(this.network, null, 2)}`);
}
getDevices(array = false) {
const devices = Object.assign({}, this.network);
if (array === true) {
return Object.keys(devices).reduce((list, src) => {
list.push({
src,
device: devices[src]
});
return list;
}, []);
}
return devices;
}
reconnect() {
// @FIXME: do we need to do cleanup?
// if (this.channel) {
// this.channel.removeListener('onStopped')
// this.channel.removeListener('onMessage')
// this.open = false
// this.channel = null
// }
// Reconnect after 200 ms; assuming the CAN bus is set up with "reconnect-ms 100".
setTimeout(() => this.connect(), 200);
}
connect() {
try {
this.channel = socketcan_1.createRawChannel(this.bus, true);
this.open = true;
this.emit('connected');
debug(`Connected to ${this.bus}`);
this.channel.addListener('onStopped', this.onChannelStopped);
this.channel.addListener('onMessage', this.onChannelMessage);
this.channel.start();
}
catch (err) {
debug(`Error opening channel: ${err.message}`);
this.emit('error', err);
}
}
send(message) {
if (this.open === false ||
!this.channel ||
typeof this.channel.send !== 'function') {
this.emit('error', new Error(`Can't send message before bus is open: ${JSON.stringify(message)}`));
}
// 60928 = address claim
if (this.address === -1 && message.pgn !== 60928) {
this.emit('error', new Error(`Can't send messages before an address is claimed`));
}
// Fix discrepancies between canboatjs & internal format
if (message.hasOwnProperty('dest') && !message.hasOwnProperty('dst')) {
message.dst = message.dest;
}
// Fix discrepancies between canboatjs & internal format
if (message.hasOwnProperty('dst') && !message.hasOwnProperty('dest')) {
message.dest = message.dst;
}
if (message.dest === 0 || message.dst === 0) {
message.dest = 255;
message.dst = 255;
}
const frame = {
id: this.PGNtoID(message.pgn, message.src, message.dst),
ext: true,
data: canboatjs_1.toPgn(Object.assign(Object.assign({}, message), message.data))
};
let frames = [frame];
// If it's actually a fast-packet
if (frame.data.length > 8) {
frames = this.generateCANFrames(frame);
}
try {
debug(`[send] Sending ${frames.length} CAN frames...`);
frames.forEach((canframe, index) => {
debug(`[send] (${index + 1} / ${frames.length}) Sending CAN frame with ID ${canframe.id} - ${JSON.stringify(canframe.data)}`);
this.channel.send(canframe);
});
}
catch (e) {
debug(`[send] Error sending PGN: ${e.message}`);
this.emit('error', e);
}
}
sendRawPgn(pgn, data, dest = 255) {
if (this.open === false ||
!this.channel ||
typeof this.channel.send !== 'function') {
this.emit('error', new Error(`Can't send message before bus is open: ${JSON.stringify({ pgn, data, dest })}`));
}
// 60928 = address claim
if (this.address === -1 && pgn !== 60928) {
this.emit('error', new Error(`Can't send messages before an address is claimed`));
}
const frame = {
id: this.PGNtoID(pgn, this.address, dest),
ext: true,
data
};
let frames = [frame];
// If it's actually a fast-packet
if (frame.data.length > 8) {
frames = this.generateCANFrames(frame);
}
try {
debug(`[send] Sending ${frames.length} CAN frames...`);
frames.forEach((canframe, index) => {
debug(`[send] (${index + 1} / ${frames.length}) Sending CAN frame with ID ${canframe.id} - ${JSON.stringify(canframe.data)}`);
this.channel.send(canframe);
});
}
catch (e) {
debug(`[send] Error sending PGN: ${e.message}`);
this.emit('error', e);
}
}
isConnected() {
return this.open;
}
// @FIXME prio is always 6 atm
PGNtoID(pgn, src, dest = 255, prio = 6) {
let id = src | 0x80000000;
if ((pgn & 0xff) === 0) {
// PDU 1
id += dest << 8;
id += pgn << 8;
id += prio << 26;
}
else {
// PDU 2
id += pgn << 8;
id += prio << 26;
}
return id;
}
IDtoPGN(id) {
let idString = id.toString(2);
const missingZeroes = 32 - idString.length;
for (let i = 0; i < missingZeroes; i += 1) {
idString = `0${idString}`;
}
const DP = `0000000${idString[7]}`;
const PF = idString.slice(8, 16);
const PS = idString.slice(16, 24);
const SA = idString.slice(24, 32);
// debug(`DP: ${parseInt(DP, 2)}, PF: ${parseInt(PF, 2)}, PS: ${parseInt(PS, 2)}, SA: ${parseInt(SA, 2)}`)
return {
id,
pgn: parseInt(PF, 2) < 240
? (parseInt(DP, 2) << 16) + (parseInt(PF, 2) << 8)
: (parseInt(DP, 2) << 16) + (parseInt(PF, 2) << 8) + parseInt(PS, 2),
dest: parseInt(PF, 2) < 240 ? parseInt(PS, 2) : 0xff,
src: parseInt(SA, 2)
};
}
CANtoN2Kmessage(message) {
const PGN = this.IDtoPGN(message.id);
return {
data: message.data,
pgn: PGN.pgn,
src: PGN.src,
dest: PGN.dest,
id: message.id,
timestamp: message.timestamp || new Date().toISOString(),
length: message.data.length
};
}
generateCANFrames(frame) {
const buffer = Buffer.from(frame.data);
const frames = [];
let cursor = 0x40;
const first = Buffer.alloc(8);
first.writeUInt8(cursor++, 0);
first.writeUInt8(buffer.length, 1);
// Copy from source "buffer" byte 0 - byte 6 into target "first", starting at target byte index 2
buffer.copy(first, 2, 0, 6);
frames.push(Object.assign(Object.assign({}, frame), { data: Buffer.from(first) }));
// loop over each byte (starting at 6) in "buffer", increasing the loop index by 7 bytes for each iteration
for (let i = 6; i < buffer.length; i += 7) {
const next = Buffer.alloc(8);
let end = i + 7; // end byte index of the slice to take from "buffer"
let fill = 0;
// If the end is beyond "buffer" length, set a variable "fill" with the number of padding bytes to add
if (end > buffer.length) {
fill = end - buffer.length;
end = buffer.length;
}
// Write the next cursor at byte 0 in the current frame
next.writeUInt8(cursor++, 0);
// Copy from source "buffer" byte i - byte end into target "next", starting at target byte index 1
buffer.copy(next, 1, i, end);
// pad the buffer if required
if (fill > 0) {
for (let j = end - i; j < 8; j++) {
next.writeUInt8(0xff, j);
}
}
frames.push(Object.assign(Object.assign({}, frame), { data: Buffer.from(next) }));
}
return frames;
}
handlePGN(message) {
this.emit('message', message);
}
}
exports.default = CANBus;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"can-bus.js","sourceRoot":"","sources":["../../../src/lib/can-bus.ts"],"names":[],"mappings":"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"}