/* packet-zbee-zcl-closures.c * Dissector routines for the ZigBee ZCL Closures clusters * Shade configuration, Door Lock * By * * Wireshark - Network traffic analyzer * By Gerald Combs * Copyright 1998 Gerald Combs * * SPDX-License-Identifier: GPL-2.0-or-later */ /* Include Files */ #include "config.h" #include #include #include "packet-zbee.h" #include "packet-zbee-aps.h" #include "packet-zbee-zcl.h" /* ########################################################################## */ /* #### (0x0100) SHADE CONFIGURATION CLUSTER ################################ */ /* ########################################################################## */ /*************************/ /* Defines */ /*************************/ #define ZBEE_ZCL_SHADE_CONFIGURATION_NUM_ETT 2 /*Attributes*/ #define ZBEE_ZCL_ATTR_ID_SHADE_CONFIGURATION_PHYSICAL_CLOSED_LIMIT 0x0000 /* Physical Closed Limit */ #define ZBEE_ZCL_ATTR_ID_SHADE_CONFIGURATION_MOTOR_STEP_SIZE 0x0001 /* Motor Step Size */ #define ZBEE_ZCL_ATTR_ID_SHADE_CONFIGURATION_STATUS 0x0002 /* Status */ #define ZBEE_ZCL_ATTR_ID_SHADE_CONFIGURATION_CLOSED_LIMIT 0x0010 /* Closed Limit */ #define ZBEE_ZCL_ATTR_ID_SHADE_CONFIGURATION_MODE 0x0011 /* Mode */ /*Server commands received - none*/ /*Server commands generated - none*/ /*Status Mask Value*/ #define ZBEE_ZCL_SHADE_CONFIGURATION_STATUS_SHADE_OPERATIONAL 0x01 /* Shade Operational */ #define ZBEE_ZCL_SHADE_CONFIGURATION_STATUS_SHADE_ADJUSTING 0x02 /* Shade Adjusting */ #define ZBEE_ZCL_SHADE_CONFIGURATION_STATUS_SHADE_DIRECTION 0x04 /* Shade Direction */ #define ZBEE_ZCL_SHADE_CONFIGURATION_STATUS_MOTOR_FORWARD_DIRECTION 0x08 /* Motor Forward Direction */ /*************************/ /* Function Declarations */ /*************************/ void proto_register_zbee_zcl_shade_configuration(void); void proto_reg_handoff_zbee_zcl_shade_configuration(void); /* Command Dissector Helpers */ static void dissect_zcl_shade_configuration_attr_data (proto_tree *tree, tvbuff_t *tvb, unsigned *offset, uint16_t attr_id, unsigned data_type, bool client_attr); /* Private functions prototype */ /*************************/ /* Global Variables */ /*************************/ /* Initialize the protocol and registered fields */ static int proto_zbee_zcl_shade_configuration; static int hf_zbee_zcl_shade_configuration_attr_id; static int hf_zbee_zcl_shade_configuration_status; static int hf_zbee_zcl_shade_configuration_status_shade_operational; static int hf_zbee_zcl_shade_configuration_status_shade_adjusting; static int hf_zbee_zcl_shade_configuration_status_shade_direction; static int hf_zbee_zcl_shade_configuration_status_motor_forward_direction; static int hf_zbee_zcl_shade_configuration_mode; /* Initialize the subtree pointers */ static int ett_zbee_zcl_shade_configuration; static int ett_zbee_zcl_shade_configuration_status; /* Attributes */ static const value_string zbee_zcl_shade_configuration_attr_names[] = { { ZBEE_ZCL_ATTR_ID_SHADE_CONFIGURATION_PHYSICAL_CLOSED_LIMIT, "Physical Closed Limit" }, { ZBEE_ZCL_ATTR_ID_SHADE_CONFIGURATION_MOTOR_STEP_SIZE, "Motor Step Size" }, { ZBEE_ZCL_ATTR_ID_SHADE_CONFIGURATION_STATUS, "Status" }, { ZBEE_ZCL_ATTR_ID_SHADE_CONFIGURATION_CLOSED_LIMIT, "Closed Limit" }, { ZBEE_ZCL_ATTR_ID_SHADE_CONFIGURATION_MODE, "Mode" }, { 0, NULL } }; /*Shade and motor direction values*/ static const value_string zbee_zcl_shade_configuration_shade_motor_direction_names[] = { {0, "Closing"}, {1, "Opening"}, {0, NULL} }; /*Mode Values*/ static const value_string zbee_zcl_shade_configuration_mode_names[] = { {0, "Normal"}, {1, "Configure"}, {0, NULL} }; /*************************/ /* Function Bodies */ /*************************/ /** *ZigBee ZCL Shade Configuration cluster dissector for wireshark. * *@param tvb pointer to buffer containing raw packet. *@param pinfo pointer to packet information fields *@param tree pointer to data tree Wireshark uses to display packet. */ static int dissect_zbee_zcl_shade_configuration(tvbuff_t *tvb _U_, packet_info *pinfo _U_, proto_tree *tree _U_, void* data _U_) { return tvb_captured_length(tvb); } /*dissect_zbee_zcl_shade_configuration*/ /** *This function is called by ZCL foundation dissector in order to decode * *@param tree pointer to data tree Wireshark uses to display packet. *@param tvb pointer to buffer containing raw packet. *@param offset pointer to buffer offset *@param attr_id attribute identifier *@param data_type attribute data type *@param client_attr ZCL client */ void dissect_zcl_shade_configuration_attr_data(proto_tree *tree, tvbuff_t *tvb, unsigned *offset, uint16_t attr_id, unsigned data_type, bool client_attr) { static int * const shade_config_status[] = { &hf_zbee_zcl_shade_configuration_status_shade_operational, &hf_zbee_zcl_shade_configuration_status_shade_adjusting, &hf_zbee_zcl_shade_configuration_status_shade_direction, &hf_zbee_zcl_shade_configuration_status_motor_forward_direction, NULL }; /* Dissect attribute data type and data */ switch (attr_id) { case ZBEE_ZCL_ATTR_ID_SHADE_CONFIGURATION_STATUS: proto_tree_add_bitmask(tree, tvb, *offset, hf_zbee_zcl_shade_configuration_status, ett_zbee_zcl_shade_configuration_status, shade_config_status, ENC_LITTLE_ENDIAN); *offset += 1; break; case ZBEE_ZCL_ATTR_ID_SHADE_CONFIGURATION_MODE: proto_tree_add_item(tree, hf_zbee_zcl_shade_configuration_mode, tvb, *offset, 1, ENC_NA); *offset += 1; break; case ZBEE_ZCL_ATTR_ID_SHADE_CONFIGURATION_PHYSICAL_CLOSED_LIMIT: case ZBEE_ZCL_ATTR_ID_SHADE_CONFIGURATION_MOTOR_STEP_SIZE: case ZBEE_ZCL_ATTR_ID_SHADE_CONFIGURATION_CLOSED_LIMIT: default: dissect_zcl_attr_data(tvb, tree, offset, data_type, client_attr); break; } } /*dissect_zcl_shade_configuration_attr_data*/ /** *ZigBee ZCL Shade Configuration cluster protocol registration routine. * */ void proto_register_zbee_zcl_shade_configuration(void) { /* Setup list of header fields */ static hf_register_info hf[] = { { &hf_zbee_zcl_shade_configuration_attr_id, { "Attribute", "zbee_zcl_closures.shade_configuration.attr_id", FT_UINT16, BASE_HEX, VALS(zbee_zcl_shade_configuration_attr_names), 0x00, NULL, HFILL } }, /* start Shade Configuration Status fields */ { &hf_zbee_zcl_shade_configuration_status, { "Shade Configuration Status", "zbee_zcl_closures.shade_configuration.attr.status", FT_UINT8, BASE_HEX, NULL, 0x00, NULL, HFILL } }, { &hf_zbee_zcl_shade_configuration_status_shade_operational, { "Shade Operational", "zbee_zcl_closures.shade_configuration.attr.status.shade_operational", FT_BOOLEAN, 8, TFS(&tfs_no_yes), ZBEE_ZCL_SHADE_CONFIGURATION_STATUS_SHADE_OPERATIONAL, NULL, HFILL } }, { &hf_zbee_zcl_shade_configuration_status_shade_adjusting, { "Shade Adjusting", "zbee_zcl_closures.shade_configuration.attr.status.shade_adjusting", FT_BOOLEAN, 8, TFS(&tfs_no_yes), ZBEE_ZCL_SHADE_CONFIGURATION_STATUS_SHADE_ADJUSTING, NULL, HFILL } }, { &hf_zbee_zcl_shade_configuration_status_shade_direction, { "Shade Direction", "zbee_zcl_closures.shade_configuration.attr.status.shade_direction", FT_UINT8, BASE_DEC, VALS(zbee_zcl_shade_configuration_shade_motor_direction_names), ZBEE_ZCL_SHADE_CONFIGURATION_STATUS_SHADE_DIRECTION, NULL, HFILL } }, { &hf_zbee_zcl_shade_configuration_status_motor_forward_direction, { "Motor Forward Direction", "zbee_zcl_closures.shade_configuration.attr.status.motor_forward_direction", FT_UINT8, BASE_DEC, VALS(zbee_zcl_shade_configuration_shade_motor_direction_names), ZBEE_ZCL_SHADE_CONFIGURATION_STATUS_MOTOR_FORWARD_DIRECTION, NULL, HFILL } }, /* end Shade Configuration Status fields */ { &hf_zbee_zcl_shade_configuration_mode, { "Mode", "zbee_zcl_closures.shade_configuration.attr.mode", FT_UINT8, BASE_HEX, VALS(zbee_zcl_shade_configuration_mode_names), 0x00, NULL, HFILL } } }; /* ZCL Shade Configuration subtrees */ static int *ett[ZBEE_ZCL_SHADE_CONFIGURATION_NUM_ETT]; ett[0] = &ett_zbee_zcl_shade_configuration; ett[1] = &ett_zbee_zcl_shade_configuration_status; /* Register the ZigBee ZCL Shade Configuration cluster protocol name and description */ proto_zbee_zcl_shade_configuration = proto_register_protocol("ZigBee ZCL Shade Configuration", "ZCL Shade Configuration", ZBEE_PROTOABBREV_ZCL_SHADE_CONFIG); proto_register_field_array(proto_zbee_zcl_shade_configuration, hf, array_length(hf)); proto_register_subtree_array(ett, array_length(ett)); /* Register the ZigBee ZCL Shade Configuration dissector. */ register_dissector(ZBEE_PROTOABBREV_ZCL_SHADE_CONFIG, dissect_zbee_zcl_shade_configuration, proto_zbee_zcl_shade_configuration); } /*proto_register_zbee_zcl_shade_configuration*/ /** *Hands off the ZCL Shade Configuration dissector. * */ void proto_reg_handoff_zbee_zcl_shade_configuration(void) { zbee_zcl_init_cluster( ZBEE_PROTOABBREV_ZCL_SHADE_CONFIG, proto_zbee_zcl_shade_configuration, ett_zbee_zcl_shade_configuration, ZBEE_ZCL_CID_SHADE_CONFIG, ZBEE_MFG_CODE_NONE, hf_zbee_zcl_shade_configuration_attr_id, hf_zbee_zcl_shade_configuration_attr_id, -1, -1, (zbee_zcl_fn_attr_data)dissect_zcl_shade_configuration_attr_data ); } /*proto_reg_handoff_zbee_zcl_shade_configuration*/ /* ########################################################################## */ /* #### (0x0101) DOOR LOCK CLUSTER ########################################## */ /* ########################################################################## */ /*************************/ /* Defines */ /*************************/ #define ZBEE_ZCL_DOOR_LOCK_NUM_ETT 1 /* Attributes */ #define ZBEE_ZCL_ATTR_ID_DOOR_LOCK_LOCK_STATE 0x0000 /* Lock State */ #define ZBEE_ZCL_ATTR_ID_DOOR_LOCK_LOCK_TYPE 0x0001 /* Lock Type */ #define ZBEE_ZCL_ATTR_ID_DOOR_LOCK_ACTUATOR_ENABLED 0x0002 /* Actuator Enabled */ #define ZBEE_ZCL_ATTR_ID_DOOR_LOCK_DOOR_STATE 0x0003 /* Door State */ #define ZBEE_ZCL_ATTR_ID_DOOR_LOCK_DOOR_OPEN_EVENTS 0x0004 /* Door Open Events */ #define ZBEE_ZCL_ATTR_ID_DOOR_LOCK_DOOR_CLOSED_EVENTS 0x0005 /* Door Closed Events */ #define ZBEE_ZCL_ATTR_ID_DOOR_LOCK_OPEN_PERIOD 0x0006 /* Open Period */ /* Server Commands Received */ #define ZBEE_ZCL_CMD_ID_DOOR_LOCK_LOCK_DOOR 0x00 /* Lock Door */ #define ZBEE_ZCL_CMD_ID_DOOR_LOCK_UNLOCK_DOOR 0x01 /* Unlock Door */ /* Server Commands Generated */ #define ZBEE_ZCL_CMD_ID_DOOR_LOCK_LOCK_DOOR_RESPONSE 0x00 /* Lock Door Response */ #define ZBEE_ZCL_CMD_ID_DOOR_LOCK_UNLOCK_DOOR_RESPONSE 0x01 /* Unlock Door Response */ /*************************/ /* Function Declarations */ /*************************/ void proto_register_zbee_zcl_door_lock(void); void proto_reg_handoff_zbee_zcl_door_lock(void); /* Command Dissector Helpers */ static void dissect_zcl_door_lock_lock_unlock_door_response (tvbuff_t *tvb, proto_tree *tree, unsigned *offset); static void dissect_zcl_door_lock_attr_data (proto_tree *tree, tvbuff_t *tvb, unsigned *offset, uint16_t attr_id, unsigned data_type, bool client_attr); /* Private functions prototype */ /*************************/ /* Global Variables */ /*************************/ /* Initialize the protocol and registered fields */ static int proto_zbee_zcl_door_lock; static int hf_zbee_zcl_door_lock_attr_id; static int hf_zbee_zcl_door_lock_lock_state; static int hf_zbee_zcl_door_lock_lock_type; static int hf_zbee_zcl_door_lock_door_state; static int hf_zbee_zcl_door_lock_actuator_enabled; static int hf_zbee_zcl_door_lock_status; static int hf_zbee_zcl_door_lock_srv_rx_cmd_id; static int hf_zbee_zcl_door_lock_srv_tx_cmd_id; /* Initialize the subtree pointers */ static int ett_zbee_zcl_door_lock; /* Attributes */ static const value_string zbee_zcl_door_lock_attr_names[] = { { ZBEE_ZCL_ATTR_ID_DOOR_LOCK_LOCK_STATE, "Lock State" }, { ZBEE_ZCL_ATTR_ID_DOOR_LOCK_LOCK_TYPE, "Lock Type" }, { ZBEE_ZCL_ATTR_ID_DOOR_LOCK_ACTUATOR_ENABLED, "Actuator Enabled" }, { ZBEE_ZCL_ATTR_ID_DOOR_LOCK_DOOR_STATE, "Door State" }, { ZBEE_ZCL_ATTR_ID_DOOR_LOCK_DOOR_OPEN_EVENTS, "Door Open Events" }, { ZBEE_ZCL_ATTR_ID_DOOR_LOCK_DOOR_CLOSED_EVENTS, "Door Closed Events" }, { ZBEE_ZCL_ATTR_ID_DOOR_LOCK_OPEN_PERIOD, "Open Period" }, { 0, NULL } }; /* Server Commands Received */ static const value_string zbee_zcl_door_lock_srv_rx_cmd_names[] = { { ZBEE_ZCL_CMD_ID_DOOR_LOCK_LOCK_DOOR, "Lock Door" }, { ZBEE_ZCL_CMD_ID_DOOR_LOCK_UNLOCK_DOOR, "Unlock Door" }, { 0, NULL } }; /* Server Commands Generated */ static const value_string zbee_zcl_door_lock_srv_tx_cmd_names[] = { { ZBEE_ZCL_CMD_ID_DOOR_LOCK_LOCK_DOOR_RESPONSE, "Lock Door Response" }, { ZBEE_ZCL_CMD_ID_DOOR_LOCK_UNLOCK_DOOR_RESPONSE, "Unlock Door Response" }, { 0, NULL } }; /* Lock State Values */ static const value_string zbee_zcl_door_lock_lock_state_values[] = { { 0x00, "Not Fully Locked" }, { 0x01, "Locked" }, { 0x02, "Unlocked" }, { 0, NULL } }; /* Lock Type Values */ static const value_string zbee_zcl_door_lock_lock_type_values[] = { { 0x00, "Deadbolt" }, { 0x01, "Magnetic" }, { 0x02, "Other" }, { 0, NULL } }; /* Door State Values */ static const value_string zbee_zcl_door_lock_door_state_values[] = { { 0x00, "Open" }, { 0x01, "Closed" }, { 0x02, "Error (Jammed)" }, { 0x03, "Error (Forced Open)" }, { 0x04, "Error (Unspecified)" }, { 0, NULL } }; /*************************/ /* Function Bodies */ /*************************/ /** *ZigBee ZCL Door Lock cluster dissector for wireshark. * *@param tvb pointer to buffer containing raw packet. *@param pinfo pointer to packet information fields *@param tree pointer to data tree Wireshark uses to display packet. */ static int dissect_zbee_zcl_door_lock(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void* data) { proto_tree *payload_tree; zbee_zcl_packet *zcl; unsigned offset = 0; uint8_t cmd_id; int rem_len; /* Reject the packet if data is NULL */ if (data == NULL) return 0; zcl = (zbee_zcl_packet *)data; cmd_id = zcl->cmd_id; /* Create a subtree for the ZCL Command frame, and add the command ID to it. */ if (zcl->direction == ZBEE_ZCL_FCF_TO_SERVER) { /* Append the command name to the info column. */ col_append_fstr(pinfo->cinfo, COL_INFO, "%s, Seq: %u", val_to_str_const(cmd_id, zbee_zcl_door_lock_srv_rx_cmd_names, "Unknown Command"), zcl->tran_seqno); /* Add the command ID. */ proto_tree_add_item(tree, hf_zbee_zcl_door_lock_srv_rx_cmd_id, tvb, offset, 1, ENC_LITTLE_ENDIAN); /* Check if this command has a payload, then add the payload tree */ rem_len = tvb_reported_length_remaining(tvb, ++offset); if (rem_len > 0) { /*payload_tree =*/ proto_tree_add_subtree(tree, tvb, offset, rem_len, ett_zbee_zcl_door_lock, NULL, "Payload"); /* Call the appropriate command dissector */ switch (cmd_id) { case ZBEE_ZCL_CMD_ID_DOOR_LOCK_LOCK_DOOR: case ZBEE_ZCL_CMD_ID_DOOR_LOCK_UNLOCK_DOOR: /* No Payload */ default: break; } } } else { /* ZBEE_ZCL_FCF_TO_CLIENT */ /* Append the command name to the info column. */ col_append_fstr(pinfo->cinfo, COL_INFO, "%s, Seq: %u", val_to_str_const(cmd_id, zbee_zcl_door_lock_srv_tx_cmd_names, "Unknown Command"), zcl->tran_seqno); /* Add the command ID. */ proto_tree_add_item(tree, hf_zbee_zcl_door_lock_srv_tx_cmd_id, tvb, offset, 1, ENC_LITTLE_ENDIAN); /* Check if this command has a payload, then add the payload tree */ rem_len = tvb_reported_length_remaining(tvb, ++offset); if (rem_len > 0) { payload_tree = proto_tree_add_subtree(tree, tvb, offset, rem_len, ett_zbee_zcl_door_lock, NULL, "Payload"); /* Call the appropriate command dissector */ switch (cmd_id) { case ZBEE_ZCL_CMD_ID_DOOR_LOCK_LOCK_DOOR_RESPONSE: case ZBEE_ZCL_CMD_ID_DOOR_LOCK_UNLOCK_DOOR_RESPONSE: dissect_zcl_door_lock_lock_unlock_door_response(tvb, payload_tree, &offset); break; default: break; } } } return tvb_captured_length(tvb); } /*dissect_zbee_zcl_door_lock*/ /** *This function decodes the lock and unlock door response * *@param tvb the tv buffer of the current data_type *@param tree the tree to append this item to *@param offset offset of data in tvb */ static void dissect_zcl_door_lock_lock_unlock_door_response(tvbuff_t *tvb, proto_tree *tree, unsigned *offset) { /* Retrieve "Status" field */ proto_tree_add_item(tree, hf_zbee_zcl_door_lock_status, tvb, *offset, 1, ENC_LITTLE_ENDIAN); *offset += 1; } /*dissect_zcl_door_lock_lock_unlock_door_response*/ /** *This function is called by ZCL foundation dissector in order to decode * *@param tree pointer to data tree Wireshark uses to display packet. *@param tvb pointer to buffer containing raw packet. *@param offset pointer to buffer offset *@param attr_id attribute identifier *@param data_type attribute data type *@param client_attr ZCL client */ void dissect_zcl_door_lock_attr_data(proto_tree *tree, tvbuff_t *tvb, unsigned *offset, uint16_t attr_id, unsigned data_type, bool client_attr) { /* Dissect attribute data type and data */ switch ( attr_id ) { case ZBEE_ZCL_ATTR_ID_DOOR_LOCK_LOCK_STATE: proto_tree_add_item(tree, hf_zbee_zcl_door_lock_lock_state, tvb, *offset, 1, ENC_LITTLE_ENDIAN); *offset += 1; break; case ZBEE_ZCL_ATTR_ID_DOOR_LOCK_LOCK_TYPE: proto_tree_add_item(tree, hf_zbee_zcl_door_lock_lock_type, tvb, *offset, 1, ENC_LITTLE_ENDIAN); *offset += 1; break; case ZBEE_ZCL_ATTR_ID_DOOR_LOCK_ACTUATOR_ENABLED: proto_tree_add_item(tree, hf_zbee_zcl_door_lock_actuator_enabled, tvb, *offset, 1, ENC_LITTLE_ENDIAN); *offset += 1; break; case ZBEE_ZCL_ATTR_ID_DOOR_LOCK_DOOR_STATE: proto_tree_add_item(tree, hf_zbee_zcl_door_lock_door_state, tvb, *offset, 1, ENC_LITTLE_ENDIAN); *offset += 1; break; case ZBEE_ZCL_ATTR_ID_DOOR_LOCK_DOOR_OPEN_EVENTS: case ZBEE_ZCL_ATTR_ID_DOOR_LOCK_DOOR_CLOSED_EVENTS: case ZBEE_ZCL_ATTR_ID_DOOR_LOCK_OPEN_PERIOD: default: dissect_zcl_attr_data(tvb, tree, offset, data_type, client_attr); break; } } /*dissect_zcl_door_lock_attr_data*/ /** *ZigBee ZCL Door Lock cluster protocol registration routine. * */ void proto_register_zbee_zcl_door_lock(void) { /* Setup list of header fields */ static hf_register_info hf[] = { { &hf_zbee_zcl_door_lock_attr_id, { "Attribute", "zbee_zcl_closures.door_lock.attr_id", FT_UINT16, BASE_HEX, VALS(zbee_zcl_door_lock_attr_names), 0x00, NULL, HFILL } }, { &hf_zbee_zcl_door_lock_lock_state, { "Lock State", "zbee_zcl_closures.door_lock.attr.lock_state", FT_UINT8, BASE_HEX, VALS(zbee_zcl_door_lock_lock_state_values), 0x00, NULL, HFILL } }, { &hf_zbee_zcl_door_lock_lock_type, { "Lock Type", "zbee_zcl_closures.door_lock.attr.lock_type", FT_UINT8, BASE_HEX, VALS(zbee_zcl_door_lock_lock_type_values), 0x00, NULL, HFILL } }, { &hf_zbee_zcl_door_lock_door_state, { "Door State", "zbee_zcl_closures.door_lock.attr.door_state", FT_UINT8, BASE_HEX, VALS(zbee_zcl_door_lock_door_state_values), 0x00, NULL, HFILL } }, { &hf_zbee_zcl_door_lock_actuator_enabled, { "Actuator enabled", "zbee_zcl_closures.door_lock.attr.actuator_enabled", FT_BOOLEAN, 8, TFS(&tfs_disabled_enabled), 0x01, NULL, HFILL } }, { &hf_zbee_zcl_door_lock_status, { "Lock Status", "zbee_zcl_closures.door_lock.status", FT_UINT8, BASE_HEX, VALS(zbee_zcl_status_names), 0x00, NULL, HFILL } }, { &hf_zbee_zcl_door_lock_srv_rx_cmd_id, { "Command", "zbee_zcl_closures.door_lock.cmd.srv_rx.id", FT_UINT8, BASE_HEX, VALS(zbee_zcl_door_lock_srv_rx_cmd_names), 0x00, NULL, HFILL } }, { &hf_zbee_zcl_door_lock_srv_tx_cmd_id, { "Command", "zbee_zcl_closures.door_lock.cmd.srv_tx.id", FT_UINT8, BASE_HEX, VALS(zbee_zcl_door_lock_srv_tx_cmd_names), 0x00, NULL, HFILL } } }; /* ZCL Door Lock subtrees */ static int *ett[ZBEE_ZCL_DOOR_LOCK_NUM_ETT]; ett[0] = &ett_zbee_zcl_door_lock; /* Register the ZigBee ZCL Door Lock cluster protocol name and description */ proto_zbee_zcl_door_lock = proto_register_protocol("ZigBee ZCL Door Lock", "ZCL Door Lock", ZBEE_PROTOABBREV_ZCL_DOOR_LOCK); proto_register_field_array(proto_zbee_zcl_door_lock, hf, array_length(hf)); proto_register_subtree_array(ett, array_length(ett)); /* Register the ZigBee ZCL Door Lock dissector. */ register_dissector(ZBEE_PROTOABBREV_ZCL_DOOR_LOCK, dissect_zbee_zcl_door_lock, proto_zbee_zcl_door_lock); } /*proto_register_zbee_zcl_door_lock*/ /** *Hands off the ZCL Door Lock dissector. * */ void proto_reg_handoff_zbee_zcl_door_lock(void) { zbee_zcl_init_cluster( ZBEE_PROTOABBREV_ZCL_DOOR_LOCK, proto_zbee_zcl_door_lock, ett_zbee_zcl_door_lock, ZBEE_ZCL_CID_DOOR_LOCK, ZBEE_MFG_CODE_NONE, hf_zbee_zcl_door_lock_attr_id, hf_zbee_zcl_door_lock_attr_id, hf_zbee_zcl_door_lock_srv_rx_cmd_id, hf_zbee_zcl_door_lock_srv_tx_cmd_id, (zbee_zcl_fn_attr_data)dissect_zcl_door_lock_attr_data ); } /*proto_reg_handoff_zbee_zcl_door_lock*/ /* ########################################################################## */ /* #### (0x0100) WINDOW COVERING CLUSTER ################################ */ /* ########################################################################## */ /*************************/ /* Defines */ /*************************/ #define ZBEE_ZCL_WINDOW_COVERING_NUM_ETT 1 /*Attributes*/ #define ZBEE_ZCL_ATTR_ID_WINDOW_COVERING_TYPE 0x0000 /* Type of shutter */ #define ZBEE_ZCL_ATTR_ID_WINDOW_COVERING_CURRENT_POSITION_LIFT_PERCENTAGE 0x0008 /* Current position lift */ #define ZBEE_ZCL_ATTR_ID_WINDOW_COVERING_CURRENT_POSITION_TILT_PERCENTAGE 0x0009 /* Current position tilt */ /* Server commands received */ #define ZBEE_ZCL_CMD_ID_WINDOW_COVERING_UP_OPEN 0x00 /* Open the shutter */ #define ZBEE_ZCL_CMD_ID_WINDOW_COVERING_DOWN_CLOSE 0x01 /* Close the shutter */ #define ZBEE_ZCL_CMD_ID_WINDOW_COVERING_STOP 0x02 /* Stop the shutter */ #define ZBEE_ZCL_CMD_ID_WINDOW_COVERING_GO_TO_LIFT_PERCENTAGE 0x05 /* Go to lift percentage */ #define ZBEE_ZCL_CMD_ID_WINDOW_COVERING_GO_TO_TILT_PERCENTAGE 0x08 /* Go to tilt percentage */ /*Server commands generated - none*/ /*Status Mask Value - none */ /*************************/ /* Function Declarations */ /*************************/ void proto_register_zbee_zcl_window_covering(void); /* Command Dissector Helpers */ static void dissect_zcl_window_covering_go_to_percentage(tvbuff_t *tvb, proto_tree *tree, unsigned *offset); /* Private functions prototype */ /*************************/ /* Global Variables */ /*************************/ /* Initialize the protocol and registered fields */ static int proto_zbee_zcl_window_covering; static int hf_zbee_zcl_window_covering_attr_id; static int hf_zbee_zcl_window_covering_type; static int hf_zbee_zcl_window_covering_current_position_lift_percentage; static int hf_zbee_zcl_window_covering_current_position_tilt_percentage; static int hf_zbee_zcl_window_covering_go_to_percentage; static int hf_zbee_zcl_window_covering_srv_rx_cmd_id; /* Initialize the subtree pointers */ static int ett_zbee_zcl_window_covering; /* Attributes */ static const value_string zbee_zcl_window_covering_attr_names[] = { { ZBEE_ZCL_ATTR_ID_WINDOW_COVERING_TYPE, "Window covering type" }, { ZBEE_ZCL_ATTR_ID_WINDOW_COVERING_CURRENT_POSITION_LIFT_PERCENTAGE, "Current position lift percentage" }, { ZBEE_ZCL_ATTR_ID_WINDOW_COVERING_CURRENT_POSITION_TILT_PERCENTAGE, "Current position tilt percentage" }, { 0, NULL } }; /* Server Commands Received */ static const value_string zbee_zcl_window_covering_srv_rx_cmd_names[] = { { ZBEE_ZCL_CMD_ID_WINDOW_COVERING_UP_OPEN, "Up / Open" }, { ZBEE_ZCL_CMD_ID_WINDOW_COVERING_DOWN_CLOSE, "Down / Close" }, { ZBEE_ZCL_CMD_ID_WINDOW_COVERING_STOP, "Stop" }, { ZBEE_ZCL_CMD_ID_WINDOW_COVERING_GO_TO_LIFT_PERCENTAGE, "Go to lift closed percentage" }, { ZBEE_ZCL_CMD_ID_WINDOW_COVERING_GO_TO_TILT_PERCENTAGE, "Go to tilt percentage" }, { 0, NULL } }; /*************************/ /* Function Bodies */ /*************************/ /** *ZigBee ZCL Window Covering cluster dissector for wireshark. * *@param tvb pointer to buffer containing raw packet. *@param pinfo pointer to packet information fields *@param tree pointer to data tree Wireshark uses to display packet. */ static int dissect_zbee_zcl_window_covering(tvbuff_t *tvb, packet_info *pinfo, proto_tree *tree, void* data) { proto_tree *payload_tree; zbee_zcl_packet *zcl; unsigned offset = 0; uint8_t cmd_id; int rem_len; /* Reject the packet if data is NULL */ if (data == NULL) return 0; zcl = (zbee_zcl_packet *)data; cmd_id = zcl->cmd_id; /* Create a subtree for the ZCL Command frame, and add the command ID to it. */ if (zcl->direction == ZBEE_ZCL_FCF_TO_SERVER) { /* Append the command name to the info column. */ col_append_fstr(pinfo->cinfo, COL_INFO, "%s, Seq: %u", val_to_str_const(cmd_id, zbee_zcl_window_covering_srv_rx_cmd_names, "Unknown Command"), zcl->tran_seqno); /* Add the command ID. */ proto_tree_add_item(tree, hf_zbee_zcl_window_covering_srv_rx_cmd_id, tvb, offset, 1, ENC_LITTLE_ENDIAN); /* Check if this command has a payload, then add the payload tree */ rem_len = tvb_reported_length_remaining(tvb, ++offset); if (rem_len > 0) { payload_tree = proto_tree_add_subtree(tree, tvb, offset, rem_len, ett_zbee_zcl_window_covering, NULL, "Payload"); /* Call the appropriate command dissector */ switch (cmd_id) { case ZBEE_ZCL_CMD_ID_WINDOW_COVERING_GO_TO_LIFT_PERCENTAGE: case ZBEE_ZCL_CMD_ID_WINDOW_COVERING_GO_TO_TILT_PERCENTAGE: dissect_zcl_window_covering_go_to_percentage(tvb, payload_tree, &offset); break; case ZBEE_ZCL_CMD_ID_WINDOW_COVERING_UP_OPEN: case ZBEE_ZCL_CMD_ID_WINDOW_COVERING_DOWN_CLOSE: case ZBEE_ZCL_CMD_ID_WINDOW_COVERING_STOP: /* No Payload */ default: break; } } } return tvb_captured_length(tvb); } /*dissect_zbee_zcl_window_covering */ /** *This function decodes the go to lift/tilt percentage * *@param tvb the tv buffer of the current data_type *@param tree the tree to append this item to *@param offset offset of data in tvb */ static void dissect_zcl_window_covering_go_to_percentage(tvbuff_t *tvb, proto_tree *tree, unsigned *offset) { /* Retrieve "go to lift/tilt percentage" field */ proto_tree_add_item(tree, hf_zbee_zcl_window_covering_go_to_percentage, tvb, *offset, 1, ENC_LITTLE_ENDIAN); *offset += 1; } /*dissect_zcl_window_covering_go_to_percentage*/ /** *This function is called by ZCL foundation dissector in order to decode * *@param tree pointer to data tree Wireshark uses to display packet. *@param tvb pointer to buffer containing raw packet. *@param offset pointer to buffer offset *@param attr_id attribute identifier *@param data_type attribute data type *@param client_attr ZCL client */ static void dissect_zcl_window_covering_attr_data(proto_tree *tree, tvbuff_t *tvb, unsigned *offset, uint16_t attr_id, unsigned data_type, bool client_attr) { /* Dissect attribute data type and data */ switch ( attr_id ) { case ZBEE_ZCL_ATTR_ID_WINDOW_COVERING_TYPE: proto_tree_add_item(tree, hf_zbee_zcl_window_covering_type, tvb, *offset, 1, ENC_LITTLE_ENDIAN); *offset += 1; break; case ZBEE_ZCL_ATTR_ID_WINDOW_COVERING_CURRENT_POSITION_LIFT_PERCENTAGE: proto_tree_add_item(tree, hf_zbee_zcl_window_covering_current_position_lift_percentage, tvb, *offset, 1, ENC_LITTLE_ENDIAN); *offset += 1; break; case ZBEE_ZCL_ATTR_ID_WINDOW_COVERING_CURRENT_POSITION_TILT_PERCENTAGE: proto_tree_add_item(tree, hf_zbee_zcl_window_covering_current_position_tilt_percentage, tvb, *offset, 1, ENC_LITTLE_ENDIAN); *offset += 1; break; default: dissect_zcl_attr_data(tvb, tree, offset, data_type, client_attr); break; } } /*dissect_zcl_window_covering_attr_data*/ /** *ZigBee ZCL Window Covering cluster protocol registration routine. * */ void proto_register_zbee_zcl_window_covering(void) { /* Setup list of header fields */ static hf_register_info hf[] = { { &hf_zbee_zcl_window_covering_attr_id, { "Attribute", "zbee_zcl_closures.window_covering.attr_id", FT_UINT16, BASE_HEX, VALS(zbee_zcl_window_covering_attr_names), 0x00, NULL, HFILL } }, { &hf_zbee_zcl_window_covering_type, { "Type", "zbee_zcl_closures.window_covering.attr.type", FT_UINT8, BASE_HEX, NULL, 0x00, NULL, HFILL } }, { &hf_zbee_zcl_window_covering_current_position_lift_percentage, { "Current position lift percentage", "zbee_zcl_closures.window_covering.attr.current_position_lift_percentage", FT_UINT8, BASE_DEC, NULL, 0x00, NULL, HFILL } }, { &hf_zbee_zcl_window_covering_current_position_tilt_percentage, { "Current position tilt percentage", "zbee_zcl_closures.window_covering.attr.current_position_tilt_percentage", FT_UINT8, BASE_DEC, NULL, 0x00, NULL, HFILL } }, { &hf_zbee_zcl_window_covering_go_to_percentage, { "Go to", "zbee_zcl_closures.window_covering.go_to", FT_UINT8, BASE_DEC, NULL, 0x00, NULL, HFILL } }, }; /* ZCL Window Covering subtrees */ static int *ett[ZBEE_ZCL_WINDOW_COVERING_NUM_ETT]; ett[0] = &ett_zbee_zcl_window_covering; /* Register the ZigBee ZCL Window Covering cluster protocol name and description */ proto_zbee_zcl_window_covering = proto_register_protocol("ZigBee ZCL Window Covering", "ZCL Window Covering", ZBEE_PROTOABBREV_ZCL_WINDOW_COVERING); proto_register_field_array(proto_zbee_zcl_window_covering, hf, array_length(hf)); proto_register_subtree_array(ett, array_length(ett)); /* Register the ZigBee ZCL Window Covering dissector. */ register_dissector(ZBEE_PROTOABBREV_ZCL_WINDOW_COVERING, dissect_zbee_zcl_window_covering, proto_zbee_zcl_window_covering); } /*proto_register_zbee_zcl_window_covering*/ /** *Hands off the ZCL Window Covering dissector. * */ void proto_reg_handoff_zbee_zcl_window_covering(void) { zbee_zcl_init_cluster( ZBEE_PROTOABBREV_ZCL_WINDOW_COVERING, proto_zbee_zcl_window_covering, ett_zbee_zcl_window_covering, ZBEE_ZCL_CID_WINDOW_COVERING, ZBEE_MFG_CODE_NONE, hf_zbee_zcl_window_covering_attr_id, hf_zbee_zcl_window_covering_attr_id, hf_zbee_zcl_window_covering_srv_rx_cmd_id, -1, (zbee_zcl_fn_attr_data)dissect_zcl_window_covering_attr_data ); } /*proto_reg_handoff_zbee_zcl_window_covering*/ /* * Editor modelines - https://www.wireshark.org/tools/modelines.html * * Local variables: * c-basic-offset: 4 * tab-width: 8 * indent-tabs-mode: nil * End: * * vi: set shiftwidth=4 tabstop=8 expandtab: * :indentSize=4:tabSize=8:noTabs=true: */