UNPKG

modbus-connect

Version:

Modbus RTU over Web Serial and Node.js SerialPort

89 lines (73 loc) 2.66 kB
// src/function-codes/read-holding-registers.ts import { ReadHoldingRegistersResponse } from '../types/modbus-types.js'; const FUNCTION_CODE = 0x03; const MIN_QUANTITY = 1; const MAX_QUANTITY = 125; const REQUEST_SIZE = 5; const RESPONSE_HEADER_SIZE = 2; const UINT16_SIZE = 2; /** * Строит PDU-запрос для чтения holding регистров * @param startAddress - начальный адрес * @param quantity - количество регистров (1-125) * @returns Uint8Array */ export function buildReadHoldingRegistersRequest( startAddress: number, quantity: number ): Uint8Array { if ((quantity | 0) !== quantity || quantity < MIN_QUANTITY || quantity > MAX_QUANTITY) { throw new RangeError(`Quantity must be integer ${MIN_QUANTITY}-${MAX_QUANTITY}`); } const buffer = new ArrayBuffer(REQUEST_SIZE); const view = new DataView(buffer); view.setUint8(0, FUNCTION_CODE); view.setUint16(1, startAddress, false); // Big-endian view.setUint16(3, quantity, false); // Big-endian return new Uint8Array(buffer); } /** * Разбирает PDU-ответ с holding регистрами * @param pdu * @returns number[] - массив значений регистров */ export function parseReadHoldingRegistersResponse(pdu: Uint8Array): ReadHoldingRegistersResponse { 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 0x03, got 0x${pdu[0]?.toString(16)}`); } const byteCount = pdu[1]!; const expectedLength = byteCount + RESPONSE_HEADER_SIZE; if (pduLength !== expectedLength) { throw new Error(`Invalid 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: number[] = new Array(registerCount); if (byteOffset % UINT16_SIZE === 0) { const uint16View = new Uint16Array(buffer, byteOffset, registerCount); for (let i = 0; i < registerCount; i++) { registers[i] = uint16View[i]!; } } else { const view = new DataView( buffer, byteOffset - RESPONSE_HEADER_SIZE, byteCount + RESPONSE_HEADER_SIZE ); for (let i = 0; i < registerCount; i++) { registers[i] = view.getUint16(RESPONSE_HEADER_SIZE + i * UINT16_SIZE, false); } } return registers; }