modbus-connect
Version:
Modbus RTU over Web Serial and Node.js SerialPort
88 lines (74 loc) • 3.4 kB
JavaScript
// function-codes/read-input-registers.js
const FUNCTION_CODE = 0x04; // READ_INPUT_REGISTERS
const MIN_QUANTITY = 1;
const MAX_QUANTITY = 125;
const REQUEST_SIZE = 5; // 1 (FC) + 2 (Addr) + 2 (Qty)
const RESPONSE_HEADER_SIZE = 2; // FC (1) + ByteCount (1)
const UINT16_SIZE = 2;
/**
* Строит PDU-запрос для чтения input-регистров (FC 0x04)
* @param {number} startAddress - начальный адрес (0x0000–0xFFFF)
* @param {number} quantity - количество регистров (1–125)
* @returns {Uint8Array}
*/
function buildReadInputRegistersRequest(startAddress, quantity) {
// Быстрая проверка через побитовые операции
if ((quantity | 0) !== quantity || quantity < MIN_QUANTITY || quantity > MAX_QUANTITY) {
throw new RangeError(`Quantity must be integer ${MIN_QUANTITY}-${MAX_QUANTITY}`);
}
const buffer = new Uint8Array(REQUEST_SIZE);
buffer[0] = FUNCTION_CODE;
buffer[1] = startAddress >>> 8; // Старший байт адреса
buffer[2] = startAddress & 0xFF; // Младший байт адреса
buffer[3] = quantity >>> 8; // Старший байт количества
buffer[4] = quantity & 0xFF; // Младший байт количества
return buffer;
}
/**
* Разбирает PDU-ответ с input-регистрами (FC 0x04)
* @param {Uint8Array} pdu - принятый PDU-пакет
* @returns {number[]} - массив значений регистров
* @throws {TypeError|Error} - при неверных данных
*/
function parseReadInputRegistersResponse(pdu) {
if (!(pdu instanceof Uint8Array)) {
throw new TypeError('PDU must be Uint8Array');
}
const pduLength = pdu.length;
if (pduLength < RESPONSE_HEADER_SIZE) {
throw new Error('PDU too short');
}
if (pdu[0] !== FUNCTION_CODE) {
throw new Error(`Invalid function code: expected 0x04, got 0x${pdu[0].toString(16).padStart(2, '0')}`);
}
const byteCount = pdu[1];
if (byteCount % UINT16_SIZE !== 0) {
throw new Error(`Invalid byte count: must be multiple of ${UINT16_SIZE}`);
}
const expectedLength = RESPONSE_HEADER_SIZE + byteCount;
if (pduLength !== expectedLength) {
throw new Error(`Invalid PDU length: expected ${expectedLength}, got ${pduLength}`);
}
if (byteCount === 0) {
return [];
}
const buffer = pdu.buffer || pdu;
const byteOffset = (pdu.byteOffset || 0) + RESPONSE_HEADER_SIZE;
const registerCount = byteCount / UINT16_SIZE;
const registers = new Array(registerCount);
// Используем Uint16Array, если данные выровнены, иначе DataView
if (byteOffset % UINT16_SIZE === 0) {
const uint16View = new Uint16Array(buffer, byteOffset, registerCount);
registers.set(uint16View); // Быстрее, чем цикл for
} else {
const view = new DataView(buffer, byteOffset, byteCount);
for (let i = 0; i < registerCount; i++) {
registers[i] = view.getUint16(i * UINT16_SIZE, false);
}
}
return registers;
}
module.exports = {
buildReadInputRegistersRequest,
parseReadInputRegistersResponse
};