| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657 |
- const {
- bytesToBase64,
- toByteArray
- } = require('./binary-utils.js')
- const {
- clampInteger
- } = require('./base-utils.js')
- const CRC16_MODBUS_TABLE = [
- 0x0000, 0xC0C1, 0xC181, 0x0140, 0xC301, 0x03C0, 0x0280, 0xC241,
- 0xC601, 0x06C0, 0x0780, 0xC741, 0x0500, 0xC5C1, 0xC481, 0x0440,
- 0xCC01, 0x0CC0, 0x0D80, 0xCD41, 0x0F00, 0xCFC1, 0xCE81, 0x0E40,
- 0x0A00, 0xCAC1, 0xCB81, 0x0B40, 0xC901, 0x09C0, 0x0880, 0xC841,
- 0xD801, 0x18C0, 0x1980, 0xD941, 0x1B00, 0xDBC1, 0xDA81, 0x1A40,
- 0x1E00, 0xDEC1, 0xDF81, 0x1F40, 0xDD01, 0x1DC0, 0x1C80, 0xDC41,
- 0x1400, 0xD4C1, 0xD581, 0x1540, 0xD701, 0x17C0, 0x1680, 0xD641,
- 0xD201, 0x12C0, 0x1380, 0xD341, 0x1100, 0xD1C1, 0xD081, 0x1040,
- 0xF001, 0x30C0, 0x3180, 0xF141, 0x3300, 0xF3C1, 0xF281, 0x3240,
- 0x3600, 0xF6C1, 0xF781, 0x3740, 0xF501, 0x35C0, 0x3480, 0xF441,
- 0x3C00, 0xFCC1, 0xFD81, 0x3D40, 0xFF01, 0x3FC0, 0x3E80, 0xFE41,
- 0xFA01, 0x3AC0, 0x3B80, 0xFB41, 0x3900, 0xF9C1, 0xF881, 0x3840,
- 0x2800, 0xE8C1, 0xE981, 0x2940, 0xEB01, 0x2BC0, 0x2A80, 0xEA41,
- 0xEE01, 0x2EC0, 0x2F80, 0xEF41, 0x2D00, 0xEDC1, 0xEC81, 0x2C40,
- 0xE401, 0x24C0, 0x2580, 0xE541, 0x2700, 0xE7C1, 0xE681, 0x2640,
- 0x2200, 0xE2C1, 0xE381, 0x2340, 0xE101, 0x21C0, 0x2080, 0xE041,
- 0xA001, 0x60C0, 0x6180, 0xA141, 0x6300, 0xA3C1, 0xA281, 0x6240,
- 0x6600, 0xA6C1, 0xA781, 0x6740, 0xA501, 0x65C0, 0x6480, 0xA441,
- 0x6C00, 0xACC1, 0xAD81, 0x6D40, 0xAF01, 0x6FC0, 0x6E80, 0xAE41,
- 0xAA01, 0x6AC0, 0x6B80, 0xAB41, 0x6900, 0xA9C1, 0xA881, 0x6840,
- 0x7800, 0xB8C1, 0xB981, 0x7940, 0xBB01, 0x7BC0, 0x7A80, 0xBA41,
- 0xBE01, 0x7EC0, 0x7F80, 0xBF41, 0x7D00, 0xBDC1, 0xBC81, 0x7C40,
- 0xB401, 0x74C0, 0x7580, 0xB541, 0x7700, 0xB7C1, 0xB681, 0x7640,
- 0x7200, 0xB2C1, 0xB381, 0x7340, 0xB101, 0x71C0, 0x7080, 0xB041,
- 0x5000, 0x90C1, 0x9181, 0x5140, 0x9301, 0x53C0, 0x5280, 0x9241,
- 0x9601, 0x56C0, 0x5780, 0x9741, 0x5500, 0x95C1, 0x9481, 0x5440,
- 0x9C01, 0x5CC0, 0x5D80, 0x9D41, 0x5F00, 0x9FC1, 0x9E81, 0x5E40,
- 0x5A00, 0x9AC1, 0x9B81, 0x5B40, 0x9901, 0x59C0, 0x5880, 0x9841,
- 0x8801, 0x48C0, 0x4980, 0x8941, 0x4B00, 0x8BC1, 0x8A81, 0x4A40,
- 0x4E00, 0x8EC1, 0x8F81, 0x4F40, 0x8D01, 0x4DC0, 0x4C80, 0x8C41,
- 0x4400, 0x84C1, 0x8581, 0x4540, 0x8701, 0x47C0, 0x4680, 0x8641,
- 0x8201, 0x42C0, 0x4380, 0x8341, 0x4100, 0x81C1, 0x8081, 0x4040
- ]
- const CRC16_CCITT_INIT = 0xFFFF
- const CRC16_CCITT_POLY = 0x1021
- const BYTE_ORDER_HIGH = 'high'
- const BYTE_ORDER_LOW = 'low'
- const CRC_TABLE_CACHE = {}
- function createReflectByteTable() {
- const table = []
- for (let index = 0; index < 256; index += 1) {
- let source = index
- let reflected = 0
- for (let bit = 0; bit < 8; bit += 1) {
- reflected = (reflected << 1) | (source & 0x01)
- source >>= 1
- }
- table[index] = reflected & 0xFF
- }
- return table
- }
- const REFLECT_BYTE_TABLE = createReflectByteTable()
- function getCrcNumberMask(width) {
- if (width === 8) return 0xFF
- if (width === 16) return 0xFFFF
- return 0
- }
- function createMsbCrcTable(width, polynomial) {
- const table = []
- const mask = getCrcNumberMask(width)
- const topBit = 1 << (width - 1)
- const shift = width - 8
- const poly = polynomial & mask
- for (let index = 0; index < 256; index += 1) {
- let crc = shift > 0 ? (index << shift) : index
- for (let bit = 0; bit < 8; bit += 1) {
- crc = (crc & topBit)
- ? ((crc << 1) ^ poly)
- : (crc << 1)
- crc &= mask
- }
- table[index] = crc
- }
- return table
- }
- function getMsbCrcTable(width, polynomial) {
- const mask = getCrcNumberMask(width)
- if (!mask) return null
- const key = `${width}:${polynomial & mask}`
- if (!CRC_TABLE_CACHE[key]) {
- CRC_TABLE_CACHE[key] = createMsbCrcTable(width, polynomial)
- }
- return CRC_TABLE_CACHE[key]
- }
- function reflectNumberBits(value, width) {
- if (width === 8) return REFLECT_BYTE_TABLE[value & 0xFF]
- if (width === 16) {
- return ((REFLECT_BYTE_TABLE[value & 0xFF] << 8) | REFLECT_BYTE_TABLE[(value >> 8) & 0xFF]) & 0xFFFF
- }
- return Number(reflectBits(BigInt(value), width))
- }
- const CRC16_CCITT_TABLE = getMsbCrcTable(16, CRC16_CCITT_POLY)
- function hasOwnOption(options, key) {
- return Object.prototype.hasOwnProperty.call(options, key)
- }
- function getSourceOptions(options) {
- return typeof options === 'number'
- ? { initialValue: options }
- : (options || {})
- }
- function normalizeChecksumOptions(options = {}, defaultByteOrder = BYTE_ORDER_HIGH, requireByteOrder = false) {
- if (typeof options === 'number') {
- if (requireByteOrder) {
- throw new Error("16位校验需要指定 byteOrder: 'high' 或 'low'")
- }
- return {
- byteOrder: defaultByteOrder,
- initialValue: options
- }
- }
- const source = options || {}
- const hasByteOrder = hasOwnOption(source, 'byteOrder') || hasOwnOption(source, 'lowByteFirst')
- if (requireByteOrder && !hasByteOrder) {
- throw new Error("16位校验需要指定 byteOrder: 'high' 或 'low'")
- }
- let byteOrder = source.byteOrder || defaultByteOrder
- if (source.lowByteFirst === true || byteOrder === 'low' || byteOrder === 'little' || byteOrder === 'le') {
- byteOrder = BYTE_ORDER_LOW
- } else {
- byteOrder = BYTE_ORDER_HIGH
- }
- return {
- ...source,
- byteOrder
- }
- }
- function getOptionNumber(options, key, fallback, mask) {
- const value = hasOwnOption(options, key) ? Number(options[key]) : fallback
- if (!Number.isFinite(value)) return fallback & mask
- return value & mask
- }
- function appendChecksum8Value(bytes, checksum) {
- const frame = toByteArray(bytes)
- return frame.concat([checksum & 0xFF])
- }
- function hasValidChecksum8By(bytes, checksumFunction, options = {}) {
- const frame = toByteArray(bytes)
- if (frame.length < 2) return false
- const expected = checksumFunction(frame.slice(0, -1), options) & 0xFF
- const received = frame[frame.length - 1] & 0xFF
- return expected === received
- }
- function appendChecksum16(bytes, checksum, options = {}) {
- const frame = toByteArray(bytes)
- const normalized = normalizeChecksumOptions(options)
- const highByte = (checksum >> 8) & 0xFF
- const lowByte = checksum & 0xFF
- return normalized.byteOrder === BYTE_ORDER_LOW
- ? frame.concat([lowByte, highByte])
- : frame.concat([highByte, lowByte])
- }
- function readChecksum16(bytes, options = {}) {
- const frame = toByteArray(bytes)
- const normalized = normalizeChecksumOptions(options, BYTE_ORDER_HIGH, true)
- const firstByte = frame[frame.length - 2] & 0xFF
- const secondByte = frame[frame.length - 1] & 0xFF
- return normalized.byteOrder === BYTE_ORDER_LOW
- ? (firstByte | (secondByte << 8))
- : ((firstByte << 8) | secondByte)
- }
- function hasValidChecksum16(bytes, checksumFunction, options = {}) {
- const frame = toByteArray(bytes)
- if (frame.length < 2) return false
- const expected = checksumFunction(frame.slice(0, -2), options) & 0xFFFF
- const received = readChecksum16(frame, options)
- return expected === received
- }
- function checksum8(bytes, options = {}) {
- const normalized = normalizeChecksumOptions(options)
- const initialValue = getOptionNumber(normalized, 'initialValue', 0x00, 0xFF)
- const finalXor = getOptionNumber(normalized, 'finalXor', 0x00, 0xFF)
- let checksum = initialValue
- toByteArray(bytes).forEach((byte) => {
- checksum = (checksum + (byte & 0xFF)) & 0xFF
- })
- return (checksum ^ finalXor) & 0xFF
- }
- function appendChecksum8(bytes, options = {}) {
- return appendChecksum8Value(bytes, checksum8(bytes, options))
- }
- function hasValidChecksum8Value(bytes, options = {}) {
- return hasValidChecksum8By(bytes, checksum8, options)
- }
- function xor8(bytes, options = {}) {
- const normalized = normalizeChecksumOptions(options)
- const initialValue = getOptionNumber(normalized, 'initialValue', 0x00, 0xFF)
- const finalXor = getOptionNumber(normalized, 'finalXor', 0x00, 0xFF)
- let checksum = initialValue
- toByteArray(bytes).forEach((byte) => {
- checksum = (checksum ^ (byte & 0xFF)) & 0xFF
- })
- return (checksum ^ finalXor) & 0xFF
- }
- function appendXor8(bytes, options = {}) {
- return appendChecksum8Value(bytes, xor8(bytes, options))
- }
- function hasValidXor8(bytes, options = {}) {
- return hasValidChecksum8By(bytes, xor8, options)
- }
- function lrc8(bytes, options = {}) {
- const normalized = normalizeChecksumOptions(options)
- const sum = checksum8(bytes, {
- ...normalized,
- finalXor: 0x00
- })
- const finalXor = getOptionNumber(normalized, 'finalXor', 0x00, 0xFF)
- return (((-sum) & 0xFF) ^ finalXor) & 0xFF
- }
- function appendLrc8(bytes, options = {}) {
- return appendChecksum8Value(bytes, lrc8(bytes, options))
- }
- function hasValidLrc8(bytes, options = {}) {
- return hasValidChecksum8By(bytes, lrc8, options)
- }
- function crc8(bytes, options = {}) {
- const normalized = normalizeChecksumOptions(options)
- const polynomial = getOptionNumber(normalized, 'polynomial', 0x07, 0xFF)
- const initialValue = getOptionNumber(normalized, 'initialValue', 0x00, 0xFF)
- const finalXor = getOptionNumber(normalized, 'finalXor', 0x00, 0xFF)
- const table = getMsbCrcTable(8, polynomial)
- let crc = initialValue
- toByteArray(bytes).forEach((byte) => {
- crc = table[(crc ^ (byte & 0xFF)) & 0xFF]
- })
- return (crc ^ finalXor) & 0xFF
- }
- function appendCrc8(bytes, options = {}) {
- return appendChecksum8Value(bytes, crc8(bytes, options))
- }
- function hasValidCrc8(bytes, options = {}) {
- return hasValidChecksum8By(bytes, crc8, options)
- }
- function checksum16(bytes, options = {}) {
- const normalized = normalizeChecksumOptions(options)
- const initialValue = getOptionNumber(normalized, 'initialValue', 0x0000, 0xFFFF)
- const finalXor = getOptionNumber(normalized, 'finalXor', 0x0000, 0xFFFF)
- let checksum = initialValue
- toByteArray(bytes).forEach((byte) => {
- checksum = (checksum + (byte & 0xFF)) & 0xFFFF
- })
- return (checksum ^ finalXor) & 0xFFFF
- }
- function appendChecksum16Value(bytes, options = {}) {
- const normalized = normalizeChecksumOptions(options, BYTE_ORDER_HIGH, true)
- return appendChecksum16(bytes, checksum16(bytes, normalized), normalized)
- }
- function hasValidChecksum16Value(bytes, options = {}) {
- const normalized = normalizeChecksumOptions(options, BYTE_ORDER_HIGH, true)
- return hasValidChecksum16(bytes, checksum16, normalized)
- }
- function crc16Ibm(bytes, options = {}) {
- const normalized = normalizeChecksumOptions(options, BYTE_ORDER_LOW)
- const initialValue = getOptionNumber(normalized, 'initialValue', 0x0000, 0xFFFF)
- const finalXor = getOptionNumber(normalized, 'finalXor', 0x0000, 0xFFFF)
- let crc = initialValue
- toByteArray(bytes).forEach((byte) => {
- crc = (crc >> 8) ^ CRC16_MODBUS_TABLE[(crc ^ byte) & 0xFF]
- })
- return (crc ^ finalXor) & 0xFFFF
- }
- function appendCrc16Ibm(bytes, options = {}) {
- const normalized = normalizeChecksumOptions(options, BYTE_ORDER_LOW, true)
- return appendChecksum16(bytes, crc16Ibm(bytes, normalized), normalized)
- }
- function hasValidCrc16Ibm(bytes, options = {}) {
- const frame = toByteArray(bytes)
- if (frame.length < 4) return false
- return hasValidChecksum16(frame, crc16Ibm, normalizeChecksumOptions(options, BYTE_ORDER_LOW, true))
- }
- function crc16Modbus(bytes, options = {}) {
- const source = getSourceOptions(options)
- const normalized = {
- ...normalizeChecksumOptions(options, BYTE_ORDER_LOW),
- initialValue: hasOwnOption(source, 'initialValue') ? source.initialValue : 0xFFFF
- }
- return crc16Ibm(bytes, normalized)
- }
- function appendCrc16Modbus(bytes, options = {}) {
- const normalized = normalizeChecksumOptions(options, BYTE_ORDER_LOW, true)
- return appendChecksum16(bytes, crc16Modbus(bytes, normalized), normalized)
- }
- function hasValidCrc16Modbus(bytes, options = {}) {
- const frame = toByteArray(bytes)
- if (frame.length < 4) return false
- return hasValidChecksum16(frame, crc16Modbus, normalizeChecksumOptions(options, BYTE_ORDER_LOW, true))
- }
- function crc16Ccitt(bytes, options = {}) {
- const normalized = normalizeChecksumOptions(options, BYTE_ORDER_HIGH)
- const initialValue = getOptionNumber(normalized, 'initialValue', CRC16_CCITT_INIT, 0xFFFF)
- const finalXor = getOptionNumber(normalized, 'finalXor', 0x0000, 0xFFFF)
- let crc = initialValue
- toByteArray(bytes).forEach((byte) => {
- crc = ((crc << 8) ^ CRC16_CCITT_TABLE[((crc >> 8) ^ byte) & 0xFF]) & 0xFFFF
- })
- return (crc ^ finalXor) & 0xFFFF
- }
- function appendCrc16Ccitt(bytes, options = {}) {
- const normalized = normalizeChecksumOptions(options, BYTE_ORDER_HIGH, true)
- return appendChecksum16(bytes, crc16Ccitt(bytes, normalized), normalized)
- }
- function hasValidCrc16Ccitt(bytes, options = {}) {
- const frame = toByteArray(bytes)
- if (frame.length < 4) return false
- return hasValidChecksum16(frame, crc16Ccitt, normalizeChecksumOptions(options, BYTE_ORDER_HIGH, true))
- }
- function crc16Xmodem(bytes, options = {}) {
- const source = getSourceOptions(options)
- const normalized = {
- ...normalizeChecksumOptions(source, BYTE_ORDER_HIGH),
- initialValue: hasOwnOption(source, 'initialValue') ? source.initialValue : 0x0000
- }
- return crc16Ccitt(bytes, normalized)
- }
- function appendCrc16Xmodem(bytes, options = {}) {
- const normalized = normalizeChecksumOptions(options, BYTE_ORDER_HIGH, true)
- return appendChecksum16(bytes, crc16Xmodem(bytes, normalized), normalized)
- }
- function hasValidCrc16Xmodem(bytes, options = {}) {
- const frame = toByteArray(bytes)
- if (frame.length < 4) return false
- return hasValidChecksum16(frame, crc16Xmodem, normalizeChecksumOptions(options, BYTE_ORDER_HIGH, true))
- }
- const CRC_ALGORITHM_PRESETS = [
- { key: 'crc-8', label: 'CRC-8', width: 8, poly: '07', init: '00', xorOut: '00', reflectIn: false, reflectOut: false },
- { key: 'crc-8-itu', label: 'CRC-8-ITU', width: 8, poly: '07', init: '00', xorOut: '55', reflectIn: false, reflectOut: false },
- { key: 'crc-8-rohc', label: 'CRC-8-ROHC', width: 8, poly: '07', init: 'FF', xorOut: '00', reflectIn: true, reflectOut: true },
- { key: 'crc-8-maxim', label: 'CRC-8-MAXIM', width: 8, poly: '31', init: '00', xorOut: '00', reflectIn: true, reflectOut: true },
- { key: 'crc-16-ibm', label: 'CRC-16-IBM', width: 16, poly: '8005', init: '0000', xorOut: '0000', reflectIn: true, reflectOut: true },
- { key: 'crc-16-usb', label: 'CRC-16-USB', width: 16, poly: '8005', init: 'FFFF', xorOut: 'FFFF', reflectIn: true, reflectOut: true },
- { key: 'crc-16-maxim', label: 'CRC-16-MAXIM', width: 16, poly: '8005', init: '0000', xorOut: 'FFFF', reflectIn: true, reflectOut: true },
- { key: 'crc-16-ccitt', label: 'CRC-16-CCITT', width: 16, poly: '1021', init: '0000', xorOut: '0000', reflectIn: true, reflectOut: true },
- { key: 'crc-16-ccitt-false', label: 'CRC-16-CCITT-FALSE', width: 16, poly: '1021', init: 'FFFF', xorOut: '0000', reflectIn: false, reflectOut: false },
- { key: 'crc-16-x25', label: 'CRC-16-X25', width: 16, poly: '1021', init: 'FFFF', xorOut: 'FFFF', reflectIn: true, reflectOut: true },
- { key: 'crc-16-xmodem', label: 'CRC-16-XMODEM', width: 16, poly: '1021', init: '0000', xorOut: '0000', reflectIn: false, reflectOut: false },
- { key: 'crc-16-xmodem2', label: 'CRC-16-XMODEM2', width: 16, poly: '8408', init: '0000', xorOut: '0000', reflectIn: true, reflectOut: true },
- { key: 'crc-16-dnp', label: 'CRC-16-DNP', width: 16, poly: '3D65', init: '0000', xorOut: 'FFFF', reflectIn: true, reflectOut: true },
- { key: 'crc-24-q', label: 'CRC-24-Q', width: 24, poly: '864CFB', init: '000000', xorOut: '000000', reflectIn: false, reflectOut: false },
- { key: 'crc-32', label: 'CRC-32', width: 32, poly: '04C11DB7', init: 'FFFFFFFF', xorOut: 'FFFFFFFF', reflectIn: true, reflectOut: true },
- { key: 'crc-32-c', label: 'CRC-32-C', width: 32, poly: '1EDC6F41', init: 'FFFFFFFF', xorOut: 'FFFFFFFF', reflectIn: true, reflectOut: true },
- { key: 'crc-32-koopman', label: 'CRC-32-KOOPMAN', width: 32, poly: '741B8CD7', init: 'FFFFFFFF', xorOut: 'FFFFFFFF', reflectIn: true, reflectOut: true },
- { key: 'crc-32-mpeg-2', label: 'CRC-32-MPEG-2', width: 32, poly: '04C11DB7', init: 'FFFFFFFF', xorOut: '00000000', reflectIn: false, reflectOut: false },
- { key: 'crc-64-iso', label: 'CRC-64-ISO', width: 64, poly: '000000000000001B', init: 'FFFFFFFFFFFFFFFF', xorOut: 'FFFFFFFFFFFFFFFF', reflectIn: true, reflectOut: true },
- { key: 'crc-64-ecma', label: 'CRC-64-ECMA', width: 64, poly: '42F0E1EBA9EA3693', init: 'FFFFFFFFFFFFFFFF', xorOut: 'FFFFFFFFFFFFFFFF', reflectIn: true, reflectOut: true },
- { key: 'custom', label: '自定义', width: 16, poly: '1021', init: 'FFFF', xorOut: '0000', reflectIn: false, reflectOut: false, custom: true }
- ]
- function maskForWidth(width) {
- const bitWidth = Number(width)
- if (!Number.isInteger(bitWidth) || bitWidth < 1 || bitWidth > 64) {
- throw new Error('CRC 位宽需为 1 - 64')
- }
- return (1n << BigInt(bitWidth)) - 1n
- }
- function normalizeHexText(value, fallback = '0') {
- const text = String(value === undefined || value === null ? '' : value).trim()
- if (!text) return fallback
- return text.toUpperCase().startsWith('0X') ? text.slice(2) : text
- }
- function parseHexBigInt(value, label, fallback = '0') {
- if (typeof value === 'bigint') return value
- if (typeof value === 'number' && Number.isFinite(value)) return BigInt(Math.max(0, Math.trunc(value)))
- const hexText = normalizeHexText(value, fallback)
- if (!/^[0-9A-F]+$/i.test(hexText)) {
- throw new Error(`${label}需为十六进制`)
- }
- return BigInt(`0x${hexText}`)
- }
- function reflectBits(value, width) {
- let source = BigInt(value)
- let reflected = 0n
- for (let index = 0; index < width; index += 1) {
- reflected = (reflected << 1n) | (source & 1n)
- source >>= 1n
- }
- return reflected
- }
- function normalizeCrcConfig(config = {}) {
- const width = clampInteger(config.width, 1, 64, 16)
- const mask = maskForWidth(width)
- const polyValue = config.poly !== undefined ? config.poly : config.polynomial
- const initValue = config.init !== undefined ? config.init : config.initialValue
- const xorValue = config.xorOut !== undefined ? config.xorOut : config.finalXor
- const presetKey = String(config.key || config.presetKey || '')
- return {
- finalXor: parseHexBigInt(xorValue, '结果异或值', '0') & mask,
- initialValue: parseHexBigInt(initValue, '初始值', '0') & mask,
- mask,
- presetKey,
- polynomial: parseHexBigInt(polyValue, 'Poly', '0') & mask,
- reflectIn: !!(config.reflectIn || config.refin),
- reflectOut: !!(config.reflectOut || config.refout),
- useLookupTable: config.useLookupTable === true || (
- !!presetKey && presetKey !== 'custom' && (width === 8 || width === 16)
- ),
- width
- }
- }
- function computeCrcTable(bytes, normalized) {
- if (!normalized.useLookupTable) return null
- const width = normalized.width
- const mask = getCrcNumberMask(width)
- if (!mask) return null
- const table = getMsbCrcTable(width, Number(normalized.polynomial & BigInt(mask)))
- const shift = width - 8
- let crc = Number(normalized.initialValue & BigInt(mask))
- toByteArray(bytes).forEach((sourceByte) => {
- const byte = normalized.reflectIn
- ? REFLECT_BYTE_TABLE[sourceByte & 0xFF]
- : (sourceByte & 0xFF)
- const tableIndex = shift > 0
- ? (((crc >> shift) ^ byte) & 0xFF)
- : ((crc ^ byte) & 0xFF)
- crc = shift > 0
- ? (((crc << 8) & mask) ^ table[tableIndex])
- : table[tableIndex]
- crc &= mask
- })
- if (normalized.reflectOut) {
- crc = reflectNumberBits(crc, width) & mask
- }
- return BigInt((crc ^ Number(normalized.finalXor & BigInt(mask))) & mask)
- }
- function computeCrc(bytes, config = {}) {
- const normalized = normalizeCrcConfig(config)
- const tableValue = computeCrcTable(bytes, normalized)
- if (tableValue !== null) return tableValue
- const topBit = 1n << BigInt(normalized.width - 1)
- let crc = normalized.initialValue
- toByteArray(bytes).forEach((sourceByte) => {
- const byte = normalized.reflectIn
- ? Number(reflectBits(BigInt(sourceByte & 0xFF), 8))
- : (sourceByte & 0xFF)
- for (let bitMask = 0x80; bitMask > 0; bitMask >>= 1) {
- let bitSet = (crc & topBit) !== 0n
- crc = (crc << 1n) & normalized.mask
- if (byte & bitMask) {
- bitSet = !bitSet
- }
- if (bitSet) {
- crc = (crc ^ normalized.polynomial) & normalized.mask
- }
- }
- })
- if (normalized.reflectOut) {
- crc = reflectBits(crc, normalized.width) & normalized.mask
- }
- return (crc ^ normalized.finalXor) & normalized.mask
- }
- function formatCrcHex(value, width) {
- const hexLength = Math.max(1, Math.ceil(Number(width || 1) / 4))
- return `0x${BigInt(value).toString(16).toUpperCase().padStart(hexLength, '0')}`
- }
- function formatCrcBin(value, width) {
- return BigInt(value).toString(2).padStart(Number(width || 1), '0')
- }
- function crcValueToBytes(value, width) {
- const byteLength = Math.max(1, Math.ceil(Number(width || 1) / 8))
- const result = []
- let source = BigInt(value)
- for (let index = byteLength - 1; index >= 0; index -= 1) {
- result[index] = Number(source & 0xFFn)
- source >>= 8n
- }
- return result
- }
- function calculateCrc(bytes, config = {}) {
- const normalized = normalizeCrcConfig(config)
- const value = computeCrc(bytes, normalized)
- const resultBytes = crcValueToBytes(value, normalized.width)
- return {
- base64: bytesToBase64(resultBytes),
- bin: formatCrcBin(value, normalized.width),
- bytes: resultBytes,
- hex: formatCrcHex(value, normalized.width),
- value,
- width: normalized.width
- }
- }
- module.exports = {
- BYTE_ORDER_HIGH,
- BYTE_ORDER_LOW,
- CRC_ALGORITHM_PRESETS,
- appendChecksum16: appendChecksum16Value,
- appendChecksum8,
- appendCrc16Ccitt,
- appendCrc16Ibm,
- appendCrc16Modbus,
- appendCrc16Xmodem,
- appendCrc8,
- appendLrc8,
- appendXor8,
- bytesToBase64,
- calculateCrc,
- checksum16,
- checksum8,
- computeCrc,
- crc16Ccitt,
- crc16Ibm,
- crc16Modbus,
- crc16Xmodem,
- crc8,
- crcValueToBytes,
- formatCrcBin,
- formatCrcHex,
- hasValidChecksum16: hasValidChecksum16Value,
- hasValidChecksum8: hasValidChecksum8Value,
- hasValidCrc16Ccitt,
- hasValidCrc16Ibm,
- hasValidCrc16Modbus,
- hasValidCrc16Xmodem,
- hasValidCrc8,
- hasValidLrc8,
- hasValidXor8,
- lrc8,
- normalizeCrcConfig,
- readChecksum16,
- xor8
- }
|