@stoprocent/bluetooth-hci-socket
Version:
Bluetooth HCI socket binding for Node.js
193 lines (148 loc) • 5.84 kB
JavaScript
const BluetoothHciSocket = require('../index');
const bluetoothHciSocket = new BluetoothHciSocket();
bluetoothHciSocket.on('data', function (data) {
console.log('data: ' + data.toString('hex') + ', ' + data.length + ' bytes');
if (data.readUInt8(0) === HCI_EVENT_PKT) {
if (data.readUInt8(1) === EVT_DISCONN_COMPLETE) {
const status = data.readUInt8(3);
const handle = data.readUInt16LE(4);
const reason = data.readUInt8(6);
console.log('Disconn Complete');
console.log('\t' + status);
console.log('\t' + handle);
console.log('\t' + reason);
process.exit(0);
} else if (data.readUInt8(1) === EVT_LE_META_EVENT) {
if (data.readUInt8(3) === EVT_LE_CONN_COMPLETE) { // subevent
const status = data.readUInt8(4);
const handle = data.readUInt16LE(5);
const role = data.readUInt8(7);
const peerBdAddrType = data.readUInt8(8);
const peerBdAddr = data.slice(9, 15);
const interval = data.readUInt16LE(15);
const latency = data.readUInt16LE(17);
const supervisionTimeout = data.readUInt16LE(19);
const masterClockAccuracy = data.readUInt8(21);
console.log('LE Connection Complete');
console.log('\t' + status);
console.log('\t' + handle);
console.log('\t' + role);
console.log('\t' + ['PUBLIC', 'RANDOM'][peerBdAddrType]);
console.log('\t' + peerBdAddr.toString('hex').match(/.{1,2}/g).reverse().join(':'));
console.log('\t' + interval * 1.25);
console.log('\t' + latency);
console.log('\t' + supervisionTimeout * 10);
console.log('\t' + masterClockAccuracy);
} else if (data.readUInt8(3) === EVT_LE_CONN_UPDATE_COMPLETE) {
const status = data.readUInt8(4);
const handle = data.readUInt16LE(5);
const interval = data.readUInt16LE(7);
const latency = data.readUInt16LE(9);
const supervisionTimeout = data.readUInt16LE(11);
console.log('LE Connection Update Complete');
console.log('\t' + status);
console.log('\t' + handle);
console.log('\t' + interval * 1.25);
console.log('\t' + latency);
console.log('\t' + supervisionTimeout * 10);
writeHandle(handle, Buffer.from('020001', 'hex'));
}
}
} else if (data.readUInt8(0) === HCI_ACLDATA_PKT) {
if (((data.readUInt16LE(1) >> 12) === ACL_START) &&
(data.readUInt16LE(7) === ATT_CID)) {
const handle = data.readUInt16LE(1) & 0x0fff;
const result = data.slice(9);
console.log('ACL data');
console.log('\t' + handle);
console.log('\t' + result.toString('hex'));
disconnectConnection(handle, HCI_OE_USER_ENDED_CONNECTION);
}
}
});
bluetoothHciSocket.on('error', function (error) {
// TODO: non-BLE adaptor
if (error.message === 'Network is down') {
console.log('state = powered off');
} else {
console.error(error);
}
});
const HCI_COMMAND_PKT = 0x01;
const HCI_ACLDATA_PKT = 0x02;
const HCI_EVENT_PKT = 0x04;
const ACL_START = 0x02;
const ATT_CID = 0x0004;
const EVT_DISCONN_COMPLETE = 0x05;
const EVT_CMD_COMPLETE = 0x0e;
const EVT_CMD_STATUS = 0x0f;
const EVT_LE_META_EVENT = 0x3e;
const EVT_LE_CONN_COMPLETE = 0x01;
const EVT_LE_CONN_UPDATE_COMPLETE = 0x03;
const OGF_LE_CTL = 0x08;
const OCF_LE_CREATE_CONN = 0x000d;
const OGF_LINK_CTL = 0x01;
const OCF_DISCONNECT = 0x0006;
const LE_CREATE_CONN_CMD = OCF_LE_CREATE_CONN | OGF_LE_CTL << 10;
const DISCONNECT_CMD = OCF_DISCONNECT | OGF_LINK_CTL << 10;
const HCI_OE_USER_ENDED_CONNECTION = 0x13;
function setFilter () {
const filter = Buffer.alloc(14);
const typeMask = (1 << HCI_EVENT_PKT) | (1 << HCI_ACLDATA_PKT);
const eventMask1 = (1 << EVT_DISCONN_COMPLETE) | (1 << EVT_CMD_COMPLETE) | (1 << EVT_CMD_STATUS);
const eventMask2 = (1 << (EVT_LE_META_EVENT - 32));
filter.writeUInt32LE(typeMask, 0);
filter.writeUInt32LE(eventMask1, 4);
filter.writeUInt32LE(eventMask2, 8);
filter.writeUInt16LE(typeMask, 12);
bluetoothHciSocket.setFilter(filter);
}
bluetoothHciSocket.bindRaw();
setFilter();
bluetoothHciSocket.start();
function createConnection (address, addressType) {
const cmd = Buffer.alloc(29);
// header
cmd.writeUInt8(HCI_COMMAND_PKT, 0);
cmd.writeUInt16LE(LE_CREATE_CONN_CMD, 1);
// length
cmd.writeUInt8(0x19, 3);
// data
cmd.writeUInt16LE(0x0060, 4); // interval
cmd.writeUInt16LE(0x0030, 6); // window
cmd.writeUInt8(0x00, 8); // initiator filter
cmd.writeUInt8(addressType === 'random' ? 0x01 : 0x00, 9); // peer address type
(Buffer.from(address.split(':').reverse().join(''), 'hex')).copy(cmd, 10); // peer address
cmd.writeUInt8(0x00, 16); // own address type
cmd.writeUInt16LE(0x0028, 17); // min interval
cmd.writeUInt16LE(0x0038, 19); // max interval
cmd.writeUInt16LE(0x0000, 21); // latency
cmd.writeUInt16LE(0x002a, 23); // supervision timeout
cmd.writeUInt16LE(0x0000, 25); // min ce length
cmd.writeUInt16LE(0x0000, 27); // max ce length
bluetoothHciSocket.write(cmd);
}
function writeHandle (handle, data) {
const cmd = Buffer.alloc(9 + data.length);
// header
cmd.writeUInt8(HCI_ACLDATA_PKT, 0);
cmd.writeUInt16LE(handle, 1);
cmd.writeUInt16LE(data.length + 4, 3); // data length 1
cmd.writeUInt16LE(data.length, 5); // data length 2
cmd.writeUInt16LE(ATT_CID, 7);
data.copy(cmd, 9);
bluetoothHciSocket.write(cmd);
}
function disconnectConnection (handle, reason) {
const cmd = Buffer.alloc(7);
// header
cmd.writeUInt8(HCI_COMMAND_PKT, 0);
cmd.writeUInt16LE(DISCONNECT_CMD, 1);
// length
cmd.writeUInt8(0x03, 3);
// data
cmd.writeUInt16LE(handle, 4); // handle
cmd.writeUInt8(reason, 6); // reason
bluetoothHciSocket.write(cmd);
}
createConnection('fc:d1:75:16:7b:fc', 'random');