UNPKG

@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
"use strict"; 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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"}