Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: support Z-Wave Long Range #6401

Merged
merged 45 commits into from
Jan 19, 2024
Merged
Show file tree
Hide file tree
Changes from 37 commits
Commits
Show all changes
45 commits
Select commit Hold shift + click to select a range
666257e
feat(workaround): zst39 workaround for corrupted soft reset ack
jtbraun Oct 15, 2023
acfb9ea
feat(longrange): add FunctionType values
jtbraun Oct 13, 2023
5f81ca9
fix(functype): add comment about source of FunctionType=0x28
jtbraun Oct 16, 2023
b4ddd89
feat(longrange): support encoding/parsing node information
jtbraun Oct 14, 2023
5eb0762
feat(longrange): handle node ids > 256 in S2 auth data
jtbraun Oct 16, 2023
fcaacb3
feat(longrange): add long range command class
jtbraun Oct 14, 2023
29ef9af
feat(squash): with other ZWaveLRProtocolCC.ts changes
jtbraun Oct 16, 2023
1b56690
feat(squash): with other index.ts changes
jtbraun Oct 16, 2023
63d142b
feat(squash): with other Constants.ts
jtbraun Oct 16, 2023
6005dfa
feat(squash): with other Protocols.ts
jtbraun Oct 16, 2023
cd27d02
feat(longrange): the bulk of long range support
jtbraun Oct 16, 2023
8f15e3a
fix(lint): lint errors
jtbraun Oct 16, 2023
d1e23fc
fix(lint): lint errors
jtbraun Oct 16, 2023
70038ab
Merge remote-tracking branch 'upstream/master' into feature/longrange
jtbraun Nov 15, 2023
9b679c3
feat(longrange): remove corrupted ack workaround
jtbraun Nov 15, 2023
9637127
feat(longrange): remove ZWaveLRProtocolCC
jtbraun Nov 15, 2023
8223ee9
feat(longrange): remove "Protocol" from the LR CC name
jtbraun Nov 15, 2023
e8f9023
feat(longrange): move NodeInfo parsing/encoding changes one function …
jtbraun Nov 15, 2023
02052ef
feat(longrange): expand LR to LongRange in smart start messages
jtbraun Nov 15, 2023
c9184ef
feat(longrange): remove comment about long range CC
jtbraun Nov 15, 2023
d00b084
feat(longrange): undo changes to GetSerialApiCapabilitiesRequest
jtbraun Nov 15, 2023
15bc808
feat(longrange): use segment for GetLongRangeNodes*
jtbraun Nov 15, 2023
7b268b4
feat(longrange): array.push, not array.concat to append
jtbraun Nov 15, 2023
1bb54b2
feat(longrange): move fetching LR nodes to a helper method
jtbraun Nov 15, 2023
9b5c7ba
feat(longrange): move LongRangeChannel to core..Protocols.ts
jtbraun Nov 15, 2023
32c0c34
feat(longrange): remove validate length comments
jtbraun Nov 15, 2023
25908de
feat(longrange): remove ResponseStatus
jtbraun Nov 15, 2023
e8c3134
feat(longrange): fix erronous boolean instead of bitwise and
jtbraun Nov 15, 2023
6e52505
feat(longrange): fix lint errors
jtbraun Nov 15, 2023
e55e479
feat(longrange): lint fixes
jtbraun Nov 15, 2023
f0d0dde
feat(longrange): change LR=>LongRange, condition some serial api setu…
jtbraun Nov 15, 2023
2e7971d
fix: only pass Z-Wave LR protocol flag to SmartStart inclusion
AlCalzone Jan 16, 2024
8715fce
refactor: cleanup, prevent managing routes for LR, highlight some FIXMEs
AlCalzone Jan 16, 2024
f9585f0
chore: more cleanup
AlCalzone Jan 16, 2024
fe446eb
fix: default to the first supported protocol
AlCalzone Jan 17, 2024
79b1a6a
chore: rework controller info message
AlCalzone Jan 17, 2024
8f22fd1
refactor: rework initialization sequence
AlCalzone Jan 17, 2024
a42243d
fix: restore original encoding of GetSerialApiInitDataResponse
AlCalzone Jan 17, 2024
683db90
fix: typo
AlCalzone Jan 17, 2024
ea170ce
fix: split GetLongRangeNodes into own file, fix query loop
AlCalzone Jan 17, 2024
f226a82
fix: node info parsing
AlCalzone Jan 17, 2024
1e09888
fix: remember node ID when requesting protocol info
AlCalzone Jan 18, 2024
098b8d1
fix: cleanup
AlCalzone Jan 19, 2024
c58c2fa
docs: add documentation
AlCalzone Jan 19, 2024
01039de
Merge branch 'master' into pr/jtbraun/6401-1
AlCalzone Jan 19, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
24 changes: 18 additions & 6 deletions packages/cc/src/cc/Security2CC.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ import {
encryptAES128CCM,
getCCName,
highResTimestamp,
isLongRangeNodeId,
isTransmissionError,
isZWaveError,
parseBitMask,
Expand Down Expand Up @@ -94,13 +95,24 @@ function getAuthenticationData(
commandLength: number,
unencryptedPayload: Buffer,
): Buffer {
const ret = Buffer.allocUnsafe(8 + unencryptedPayload.length);
ret[0] = sendingNodeId;
ret[1] = destination;
ret.writeUInt32BE(homeId, 2);
ret.writeUInt16BE(commandLength, 6);
const nodeIdSize =
isLongRangeNodeId(sendingNodeId) || isLongRangeNodeId(destination)
? 2
: 1;
const ret = Buffer.allocUnsafe(
2 * nodeIdSize + 6 + unencryptedPayload.length,
);
let offset = 0;
ret.writeUIntBE(sendingNodeId, offset, nodeIdSize);
offset += nodeIdSize;
ret.writeUIntBE(destination, offset, nodeIdSize);
offset += nodeIdSize;
ret.writeUInt32BE(homeId, offset);
offset += 4;
ret.writeUInt16BE(commandLength, offset);
offset += 2;
// This includes the sequence number and all unencrypted extensions
unencryptedPayload.copy(ret, 8, 0);
unencryptedPayload.copy(ret, offset, 0);
return ret;
}

Expand Down
3 changes: 2 additions & 1 deletion packages/core/src/capabilities/CommandClasses.ts
Original file line number Diff line number Diff line change
Expand Up @@ -128,8 +128,9 @@ export enum CommandClasses {
"Z/IP ND" = 0x58,
"Z/IP Portal" = 0x61,
"Z-Wave Plus Info" = 0x5e,
// Internal CC which is not used directly by applications
// Internal CCs which are not used directly by applications
"Z-Wave Protocol" = 0x01,
"Z-Wave Long Range" = 0x04,
}

export function getCCName(cc: number): string {
Expand Down
179 changes: 129 additions & 50 deletions packages/core/src/capabilities/NodeInfo.ts
Original file line number Diff line number Diff line change
Expand Up @@ -212,37 +212,53 @@ export type NodeInformationFrame =
& NodeProtocolInfoAndDeviceClass
& ApplicationNodeInformation;

// FIXME: Split these methods into two, one each for long range and one each for classic Z-Wave
export function parseNodeProtocolInfo(
buffer: Buffer,
offset: number,
isLongRange: boolean = false,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Where did you find the information you used to update this method? I'm not sure a change is necessary, as the controller should forward a properly formatted NIF, no matter if the protocol-level frame it received is using the Long Range CC or not.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

See specs\Z-Wave Stack Specifications\Z-Wave and Z-Wave Long Range Network Layer Specification.pdf. Long Range added their own Node Information Frame Command in section 6.3.1.2. The non-long-range frame (section 4.3.2.1) has 3 bytes for Basic/Generic/Specific device classes, and then a list of command classes for the remainder of the packet.

The long range frame has Generic and Specific device class bytes (no basic), and then a byte that is the "command list length" (in bytes, as I recall), and then the list of command classes.

Why they felt the need to replace these fields and add the seemingly redundant length, I'm not sure.

If we don't need/want the long range command class in this change (it wasn't needed to get pairing and comms to work). I think this stuff can be backed out.

): NodeProtocolInfo {
validatePayload(buffer.length >= offset + 3);

const isListening = !!(buffer[offset] & 0b10_000_000);
const isRouting = !!(buffer[offset] & 0b01_000_000);
let isRouting = false;
if (!isLongRange) {
isRouting = !!(buffer[offset] & 0b01_000_000);
}

const supportedDataRates: DataRate[] = [];
const maxSpeed = buffer[offset] & 0b00_011_000;
const speedExtension = buffer[offset + 2] & 0b111;
if (maxSpeed & 0b00_010_000) {
supportedDataRates.push(40000);
}
if (maxSpeed & 0b00_001_000) {
supportedDataRates.push(9600);
}
if (speedExtension & 0b001) {
supportedDataRates.push(100000);
}
if (supportedDataRates.length === 0) {
supportedDataRates.push(9600);
if (isLongRange) {
const speedExtension = buffer[offset + 2] & 0b111;
if (speedExtension & 0b010) {
supportedDataRates.push(100000);
}
} else {
const maxSpeed = buffer[offset] & 0b00_011_000;
const speedExtension = buffer[offset + 2] & 0b111;
if (maxSpeed & 0b00_010_000) {
supportedDataRates.push(40000);
}
if (maxSpeed & 0b00_001_000) {
supportedDataRates.push(9600);
}
if (speedExtension & 0b001) {
supportedDataRates.push(100000);
}
if (supportedDataRates.length === 0) {
supportedDataRates.push(9600);
}
}

const protocolVersion = buffer[offset] & 0b111;
// BUGBUG: what's the correct protocol version here for long range?
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I couldn't find this in the specs.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Given that the NIF for Long Range lists these bits as reserved, I'd guess it's either 1 or not interesting. What do the frames received from your controller contain here?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good question, I'm not sure. I'll have to check.

const protocolVersion = isLongRange
? 0
: buffer[offset] & 0b111;

const capability = buffer[offset + 1];
const optionalFunctionality = !!(capability & 0b1000_0000);
const optionalFunctionality = (!isLongRange)
&& !!(capability & 0b1000_0000);
let isFrequentListening: FLiRS;
switch (capability & 0b0110_0000) {
switch (capability & (isLongRange ? 0b0100_0000 : 0b0110_0000)) {
case 0b0100_0000:
isFrequentListening = "1000ms";
break;
Expand All @@ -252,21 +268,29 @@ export function parseNodeProtocolInfo(
default:
isFrequentListening = false;
}
const supportsBeaming = !!(capability & 0b0001_0000);
const supportsBeaming = (!isLongRange) && !!(capability & 0b0001_0000);

let nodeType: NodeType;
switch (capability & 0b1010) {
case 0b1000:

switch (
isLongRange
? (0b1_0000_0000 | (capability & 0b0010))
: (capability & 0b1010)
) {
case 0b0_0000_1000:
case 0b1_0000_0000:
nodeType = NodeType["End Node"];
break;
case 0b0010:
case 0b0_0000_0010:
case 0b1_0000_0010:
default:
// BUGBUG: is Controller correct for default && isLongRange?
nodeType = NodeType.Controller;
break;
}

const hasSpecificDeviceClass = !!(capability & 0b100);
const supportsSecurity = !!(capability & 0b1);
const hasSpecificDeviceClass = isLongRange || !!(capability & 0b100);
const supportsSecurity = isLongRange || !!(capability & 0b1);

return {
isListening,
Expand All @@ -282,39 +306,61 @@ export function parseNodeProtocolInfo(
};
}

export function encodeNodeProtocolInfo(info: NodeProtocolInfo): Buffer {
export function encodeNodeProtocolInfo(
info: NodeProtocolInfo,
isLongRange: boolean = false,
): Buffer {
const ret = Buffer.alloc(3, 0);
// Byte 0 and 2
if (info.isListening) ret[0] |= 0b10_000_000;
if (info.isRouting) ret[0] |= 0b01_000_000;
if (info.supportedDataRates.includes(40000)) ret[0] |= 0b00_010_000;
if (info.supportedDataRates.includes(9600)) ret[0] |= 0b00_001_000;
if (info.supportedDataRates.includes(100000)) ret[2] |= 0b001;
ret[0] |= info.protocolVersion & 0b111;
if (isLongRange) {
if (info.supportedDataRates.includes(100000)) ret[2] |= 0b010;
} else {
if (info.isRouting) ret[0] |= 0b01_000_000;
if (info.supportedDataRates.includes(40000)) ret[0] |= 0b00_010_000;
if (info.supportedDataRates.includes(9600)) ret[0] |= 0b00_001_000;
if (info.supportedDataRates.includes(100000)) ret[2] |= 0b001;
ret[0] |= info.protocolVersion & 0b111;
}

// Byte 1
if (info.optionalFunctionality) ret[1] |= 0b1000_0000;
if (!isLongRange) {
if (info.optionalFunctionality) ret[1] |= 0b1000_0000;
}
if (info.isFrequentListening === "1000ms") ret[1] |= 0b0100_0000;
else if (info.isFrequentListening === "250ms") ret[1] |= 0b0010_0000;
else if (!isLongRange && info.isFrequentListening === "250ms") {
ret[1] |= 0b0010_0000;
}

if (!isLongRange) {
if (info.supportsBeaming) ret[1] |= 0b0001_0000;
if (info.supportsSecurity) ret[1] |= 0b1;
}

if (info.supportsBeaming) ret[1] |= 0b0001_0000;
if (info.supportsSecurity) ret[1] |= 0b1;
if (info.nodeType === NodeType["End Node"]) ret[1] |= 0b1000;
else ret[1] |= 0b0010; // Controller
if (info.nodeType === NodeType["End Node"]) {
if (!isLongRange) ret[1] |= 0b1000;
} else ret[1] |= 0b0010; // Controller

if (info.hasSpecificDeviceClass) ret[1] |= 0b100;
if (!isLongRange && info.hasSpecificDeviceClass) ret[1] |= 0b100;

return ret;
}

export function parseNodeProtocolInfoAndDeviceClass(buffer: Buffer): {
export function parseNodeProtocolInfoAndDeviceClass(
buffer: Buffer,
isLongRange: boolean = false,
): {
info: NodeProtocolInfoAndDeviceClass;
bytesRead: number;
} {
validatePayload(buffer.length >= 5);
const protocolInfo = parseNodeProtocolInfo(buffer, 0);
const protocolInfo = parseNodeProtocolInfo(buffer, 0, isLongRange);
let offset = 3;
const basic = buffer[offset++];
// BUGBUG: 4.3.2.1.1.14 says this is omitted if the Controller field is set to 0, yet we always parse it?
let basic = 0x100; // BUGBUG: is there an assume one here, or...?
if (!isLongRange) {
basic = buffer[offset++];
}
const generic = buffer[offset++];
let specific = 0;
if (protocolInfo.hasSpecificDeviceClass) {
Expand All @@ -334,36 +380,69 @@ export function parseNodeProtocolInfoAndDeviceClass(buffer: Buffer): {

export function encodeNodeProtocolInfoAndDeviceClass(
info: NodeProtocolInfoAndDeviceClass,
isLongRange: boolean = false,
): Buffer {
return Buffer.concat([
encodeNodeProtocolInfo({ ...info, hasSpecificDeviceClass: true }),
Buffer.from([
const deviceClasses = isLongRange
? Buffer.from([
info.genericDeviceClass,
info.specificDeviceClass,
])
: Buffer.from([
info.basicDeviceClass,
info.genericDeviceClass,
info.specificDeviceClass,
]),
]);
return Buffer.concat([
encodeNodeProtocolInfo({ ...info, hasSpecificDeviceClass: true }),
deviceClasses,
]);
}

export function parseNodeInformationFrame(
buffer: Buffer,
isLongRange: boolean = false,
): NodeInformationFrame {
const { info, bytesRead: offset } = parseNodeProtocolInfoAndDeviceClass(
const result = parseNodeProtocolInfoAndDeviceClass(
buffer,
isLongRange,
);
const supportedCCs = parseCCList(buffer.subarray(offset)).supportedCCs;
const info = result.info;
let offset = result.bytesRead;
let ccListLength;
if (isLongRange) {
ccListLength = buffer[offset];
offset += 1;
} else {
ccListLength = buffer.length - offset;
}
const supportedCCs =
parseCCList(buffer.subarray(offset, ccListLength)).supportedCCs;

return {
...info,
supportedCCs,
};
}

export function encodeNodeInformationFrame(info: NodeInformationFrame): Buffer {
return Buffer.concat([
encodeNodeProtocolInfoAndDeviceClass(info),
encodeCCList(info.supportedCCs, []),
]);
export function encodeNodeInformationFrame(
info: NodeInformationFrame,
isLongRange: boolean = false,
): Buffer {
const protocolInfo = encodeNodeProtocolInfoAndDeviceClass(
info,
isLongRange,
);
const ccList = encodeCCList(info.supportedCCs, []);

const buffers = [protocolInfo];
if (isLongRange) {
const ccListLength = Buffer.allocUnsafe(1);
ccListLength[0] = ccList.length;
buffers.push(ccListLength);
}
buffers.push(ccList);

return Buffer.concat(buffers);
}

export function parseNodeID(
Expand Down
11 changes: 11 additions & 0 deletions packages/core/src/capabilities/Protocols.ts
Original file line number Diff line number Diff line change
Expand Up @@ -87,3 +87,14 @@ export function isEmptyRoute(route: Route): boolean {
&& route.routeSpeed === ZWaveDataRate["9k6"]
);
}

export enum LongRangeChannel {
Unknown = 0x00, // Reserved
A = 0x01,
B = 0x02,
// 0x03..0xFF are reserved and must not be used
}

export function isLongRangeNodeId(nodeId: number): boolean {
return nodeId > 255;
}
6 changes: 6 additions & 0 deletions packages/core/src/consts/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,12 @@ export const NODE_ID_MAX = MAX_NODES;
/** The number of bytes in a node bit mask */
export const NUM_NODEMASK_BYTES = MAX_NODES / 8;

/** The number of node ids in a long range "segment" (GetLongRangeNodes response) */
export const NUM_LR_NODES_PER_SEGMENT = 128;

/** The number of bytes in a long range node bit mask segment */
export const NUM_LR_NODEMASK_SEGMENT_BYTES = NUM_LR_NODES_PER_SEGMENT / 8;
jtbraun marked this conversation as resolved.
Show resolved Hide resolved

export enum NodeIDType {
Short = 0x01,
Long = 0x02,
Expand Down
Loading
Loading