diff --git a/applications/zpc/components/zpc_attribute_store/include/attribute_store_defined_attribute_types.h b/applications/zpc/components/zpc_attribute_store/include/attribute_store_defined_attribute_types.h index 78ca2909e..f9ed19a13 100644 --- a/applications/zpc/components/zpc_attribute_store/include/attribute_store_defined_attribute_types.h +++ b/applications/zpc/components/zpc_attribute_store/include/attribute_store_defined_attribute_types.h @@ -690,6 +690,9 @@ DEFINE_ATTRIBUTE(ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_MAX_VALUE, DEFINE_ATTRIBUTE(ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_MAX_VALUE_SCALE, ((COMMAND_CLASS_THERMOSTAT_SETPOINT << 8) | 0x09)) + +DEFINE_ATTRIBUTE(ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_VALUE_PRECISION, + ((COMMAND_CLASS_THERMOSTAT_SETPOINT << 8) | 0x0A)) ///////////////////////////////////////////////// // Wakeup command class diff --git a/applications/zpc/components/zpc_attribute_store/src/zpc_attribute_store_type_registration.cpp b/applications/zpc/components/zpc_attribute_store/src/zpc_attribute_store_type_registration.cpp index 774ba2882..6f1815427 100644 --- a/applications/zpc/components/zpc_attribute_store/src/zpc_attribute_store_type_registration.cpp +++ b/applications/zpc/components/zpc_attribute_store/src/zpc_attribute_store_type_registration.cpp @@ -298,6 +298,8 @@ static const std::vector attribute_schema = { {ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_TYPE, "Thermostat Setpoint Type", ATTRIBUTE_ENDPOINT_ID, I8_STORAGE_TYPE}, {ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_VALUE, "Value", ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_TYPE, I32_STORAGE_TYPE}, {ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_VALUE_SCALE, "Value Scale", ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_TYPE, U32_STORAGE_TYPE}, + {ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_VALUE_PRECISION, "Value Precision", ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_TYPE, U8_STORAGE_TYPE}, + {ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_MIN_VALUE, "Min Value", ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_TYPE, I32_STORAGE_TYPE}, {ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_MIN_VALUE_SCALE, "Min Value Scale", ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_TYPE, U32_STORAGE_TYPE}, {ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_MAX_VALUE, "Max Value", ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_TYPE, I32_STORAGE_TYPE}, diff --git a/applications/zpc/components/zwave_command_classes/src/zwave_command_class_thermostat_setpoint.c b/applications/zpc/components/zwave_command_classes/src/zwave_command_class_thermostat_setpoint.c index d4629b54a..6c304586b 100644 --- a/applications/zpc/components/zwave_command_classes/src/zwave_command_class_thermostat_setpoint.c +++ b/applications/zpc/components/zwave_command_classes/src/zwave_command_class_thermostat_setpoint.c @@ -229,14 +229,14 @@ static attribute_store_node_t // Attribute resolution functions /////////////////////////////////////////////////////////////////////////////// sl_status_t zwave_command_class_thermostat_setpoint_supported_get( - attribute_store_node_t node, uint8_t *frame, uint16_t *frame_len) + attribute_store_node_t node, uint8_t *frame, uint16_t *frame_length) { // Check that we have the right type of attribute. assert(ATTRIBUTE(SUPPORTED_SETPOINT_TYPES) == attribute_store_get_node_type(node)); // Default frame length in case of error - *frame_len = 0; + *frame_length = 0; // Supported Get is the same for all versions. ZW_THERMOSTAT_SETPOINT_SUPPORTED_GET_V3_FRAME *supported_get_frame @@ -245,18 +245,18 @@ sl_status_t zwave_command_class_thermostat_setpoint_supported_get( supported_get_frame->cmdClass = COMMAND_CLASS_THERMOSTAT_SETPOINT_V3; supported_get_frame->cmd = THERMOSTAT_SETPOINT_SUPPORTED_GET_V3; - *frame_len = sizeof(ZW_THERMOSTAT_SETPOINT_SUPPORTED_GET_V3_FRAME); + *frame_length = sizeof(ZW_THERMOSTAT_SETPOINT_SUPPORTED_GET_V3_FRAME); return SL_STATUS_OK; } // FIXME: This will be failing CL:0043.01.21.02.1 sl_status_t zwave_command_class_thermostat_setpoint_capabilities_get( - attribute_store_node_t node, uint8_t *frame, uint16_t *frame_len) + attribute_store_node_t node, uint8_t *frame, uint16_t *frame_length) { // Check that we have the right type of attribute. assert(ATTRIBUTE(MIN_VALUE) == attribute_store_get_node_type(node)); // Default frame length in case of error - *frame_len = 0; + *frame_length = 0; attribute_store_node_t type_node = attribute_store_get_node_parent(node); attribute_store_node_t endpoint_node @@ -297,20 +297,20 @@ sl_status_t zwave_command_class_thermostat_setpoint_capabilities_get( &capabilities_get_frame->properties1, sizeof(capabilities_get_frame->properties1)); - *frame_len = sizeof(ZW_THERMOSTAT_SETPOINT_CAPABILITIES_GET_V3_FRAME); + *frame_length = sizeof(ZW_THERMOSTAT_SETPOINT_CAPABILITIES_GET_V3_FRAME); return SL_STATUS_OK; } } sl_status_t zwave_command_class_thermostat_setpoint_get( - attribute_store_node_t node, uint8_t *frame, uint16_t *frame_len) + attribute_store_node_t node, uint8_t *frame, uint16_t *frame_length) { // Check that we have the right type of attribute. assert(ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_VALUE == attribute_store_get_node_type(node)); // Default frame length in case of error - *frame_len = 0; + *frame_length = 0; attribute_store_node_t type_node = attribute_store_get_node_parent(node); @@ -325,57 +325,94 @@ sl_status_t zwave_command_class_thermostat_setpoint_get( &get_frame->level, sizeof(get_frame->level)); - *frame_len = sizeof(ZW_THERMOSTAT_SETPOINT_GET_V3_FRAME); + *frame_length = sizeof(ZW_THERMOSTAT_SETPOINT_GET_V3_FRAME); return SL_STATUS_OK; } sl_status_t zwave_command_class_thermostat_setpoint_set( - attribute_store_node_t node, uint8_t *frame, uint16_t *frame_len) + attribute_store_node_t node, uint8_t *frame, uint16_t *frame_length) { // Check that we have the right type of attribute. assert(ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_VALUE == attribute_store_get_node_type(node)); // Default frame length in case of error - *frame_len = 0; + *frame_length = 0; - attribute_store_node_t type_node = attribute_store_get_node_parent(node); - attribute_store_node_t scale_node = attribute_store_get_first_child_by_type( - type_node, - ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_VALUE_SCALE); - - // We will just always use 4 bytes, precision 2. - ZW_THERMOSTAT_SETPOINT_SET_4BYTE_V3_FRAME *set_frame - = (ZW_THERMOSTAT_SETPOINT_SET_4BYTE_V3_FRAME *)frame; + attribute_store_node_t type_node = attribute_store_get_node_parent(node); - set_frame->cmdClass = COMMAND_CLASS_THERMOSTAT_SETPOINT_V3; - set_frame->cmd = THERMOSTAT_SETPOINT_SET_V3; - // set_frame->level ? This is 4 bits reserved / 4 bits setpoint type. - set_frame->level = 0; + // Get setpoint type + uint8_t setpoint_type = 0; attribute_store_get_reported(type_node, - &set_frame->level, - sizeof(set_frame->level)); + &setpoint_type, + sizeof(setpoint_type)); - // set_frame->level2 ? This is Precision (3 bits) / Scale (2 bits) / size (3 bits) - uint32_t setpoint_value_scale = 0; // Reuse the same scale as current value. - attribute_store_get_reported(scale_node, - &setpoint_value_scale, - sizeof(setpoint_value_scale)); + uint32_t setpoint_value_scale = 0; + attribute_store_get_child_reported( + type_node, + ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_VALUE_SCALE, + &setpoint_value_scale, + sizeof(setpoint_value_scale)); - set_frame->level2 = SET_DEFAULT_PRECISION; - set_frame->level2 |= SET_DEFAULT_SIZE; - set_frame->level2 |= ((setpoint_value_scale << 3) & SCALE_MASK); + // Reuse the same precision as current value. + uint8_t setpoint_value_precision = 0; + attribute_store_get_child_reported( + type_node, + ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_VALUE_PRECISION, + &setpoint_value_precision, + sizeof(setpoint_value_precision)); + uint8_t level2_properties_field + = (setpoint_value_precision << 5) | (setpoint_value_scale << 3); int32_t setpoint_value_integer = thermostat_setpoint_get_valid_desired_setpoint_value(node); - set_frame->value1 = (setpoint_value_integer & 0xFF000000) >> 24; // MSB - set_frame->value2 = (setpoint_value_integer & 0x00FF0000) >> 16; - set_frame->value3 = (setpoint_value_integer & 0x0000FF00) >> 8; - set_frame->value4 = (setpoint_value_integer & 0x000000FF); // LSB + if (setpoint_value_integer >= INT8_MIN + && setpoint_value_integer <= INT8_MAX) { + ZW_THERMOSTAT_SETPOINT_SET_1BYTE_V3_FRAME *set_frame + = (ZW_THERMOSTAT_SETPOINT_SET_1BYTE_V3_FRAME *)frame; + *frame_length = sizeof(ZW_THERMOSTAT_SETPOINT_SET_1BYTE_V3_FRAME); + + set_frame->cmdClass = COMMAND_CLASS_THERMOSTAT_SETPOINT_V3; + set_frame->cmd = THERMOSTAT_SETPOINT_SET_V3; + set_frame->level = setpoint_type; + set_frame->level2 = level2_properties_field | 1; + set_frame->value1 = (uint8_t)(setpoint_value_integer & 0x000000FF); + } else if (setpoint_value_integer >= INT16_MIN + && setpoint_value_integer <= INT16_MAX) { + ZW_THERMOSTAT_SETPOINT_SET_2BYTE_V3_FRAME *set_frame + = (ZW_THERMOSTAT_SETPOINT_SET_2BYTE_V3_FRAME *)frame; + *frame_length = sizeof(ZW_THERMOSTAT_SETPOINT_SET_2BYTE_V3_FRAME); + + set_frame->cmdClass = COMMAND_CLASS_THERMOSTAT_SETPOINT_V3; + set_frame->cmd = THERMOSTAT_SETPOINT_SET_V3; + set_frame->level = setpoint_type; + set_frame->level2 = level2_properties_field | 2; + + set_frame->value1 = (setpoint_value_integer & 0x0000FF00) >> 8; // MSB + set_frame->value2 = (setpoint_value_integer & 0x000000FF); // LSB + + } else if (setpoint_value_integer >= INT32_MIN + && setpoint_value_integer <= INT32_MAX) { + ZW_THERMOSTAT_SETPOINT_SET_4BYTE_V3_FRAME *set_frame + = (ZW_THERMOSTAT_SETPOINT_SET_4BYTE_V3_FRAME *)frame; + *frame_length = sizeof(ZW_THERMOSTAT_SETPOINT_SET_4BYTE_V3_FRAME); + + set_frame->cmdClass = COMMAND_CLASS_THERMOSTAT_SETPOINT_V3; + set_frame->cmd = THERMOSTAT_SETPOINT_SET_V3; + set_frame->level = setpoint_type; + set_frame->level2 = level2_properties_field | 4; + + set_frame->value1 = (setpoint_value_integer & 0xFF000000) >> 24; // MSB + set_frame->value2 = (setpoint_value_integer & 0x00FF0000) >> 16; + set_frame->value3 = (setpoint_value_integer & 0x0000FF00) >> 8; + set_frame->value4 = (setpoint_value_integer & 0x000000FF); // LSB + } else { + sl_log_error(LOG_TAG, "Invalid desired value size"); + return SL_STATUS_NOT_SUPPORTED; + } - *frame_len = sizeof(ZW_THERMOSTAT_SETPOINT_SET_4BYTE_V3_FRAME); return SL_STATUS_OK; } @@ -389,7 +426,7 @@ static sl_status_t zwave_command_class_thermostat_setpoint_handle_report( { // We expect to have at least 1 byte of value if (frame_length <= REPORT_VALUE_INDEX) { - return SL_STATUS_FAIL; + return SL_STATUS_NOT_SUPPORTED; } attribute_store_node_t endpoint_node @@ -406,6 +443,12 @@ static sl_status_t zwave_command_class_thermostat_setpoint_handle_report( sizeof(uint8_t), 0); + // Add guard in case we don't find it + if (type_node == ATTRIBUTE_STORE_INVALID_NODE) { + sl_log_warning(LOG_TAG, "Can't find setpoint type %d", received_type); + return SL_STATUS_NOT_SUPPORTED; + } + uint8_t size = frame_data[REPORT_PRECISION_SCALE_SIZE_INDEX] & SIZE_MASK; int32_t scale = (frame_data[REPORT_PRECISION_SCALE_SIZE_INDEX] & SCALE_MASK) >> 3; @@ -419,9 +462,15 @@ static sl_status_t zwave_command_class_thermostat_setpoint_handle_report( attribute_store_node_t scale_node = attribute_store_get_first_child_by_type( type_node, ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_VALUE_SCALE); - attribute_store_set_reported(scale_node, &scale, sizeof(scale)); + // Save precision + attribute_store_set_child_reported( + type_node, + ATTRIBUTE_COMMAND_CLASS_THERMOSTAT_SETPOINT_VALUE_PRECISION, + &precision, + sizeof(precision)); + int32_t setpoint_value = command_class_get_int32_value(size, precision, @@ -688,9 +737,7 @@ sl_status_t zwave_command_class_thermostat_setpoint_init() handler.command_class_name = "Thermostat Setpoint"; handler.comments = "Partial Control:
" "1. No discovery of ambiguous types in v1-v2
" - "2. Only a few setpoints can be configured.
" - "3. Precision/size fields in the set are determined
" - "automatically by the controller. "; + "2. Only a few setpoints can be configured.
"; zwave_command_handler_register_handler(handler); diff --git a/applications/zpc/components/zwave_command_classes/src/zwave_command_class_thermostat_setpoint.h b/applications/zpc/components/zwave_command_classes/src/zwave_command_class_thermostat_setpoint.h index 8be0d09da..bed016ed7 100644 --- a/applications/zpc/components/zwave_command_classes/src/zwave_command_class_thermostat_setpoint.h +++ b/applications/zpc/components/zwave_command_classes/src/zwave_command_class_thermostat_setpoint.h @@ -31,7 +31,6 @@ #define SCALE_MASK 0x18 #define PRECISION_MASK 0xE0 -#define SET_DEFAULT_PRECISION (3 << 5) #define SET_DEFAULT_SIZE 4 #define REPORT_SETPOINT_TYPE_INDEX 2 diff --git a/applications/zpc/components/zwave_command_classes/test/zwave_command_class_thermostat_setpoint_test.c b/applications/zpc/components/zwave_command_classes/test/zwave_command_class_thermostat_setpoint_test.c index 1042aee5a..7d08abab0 100644 --- a/applications/zpc/components/zwave_command_classes/test/zwave_command_class_thermostat_setpoint_test.c +++ b/applications/zpc/components/zwave_command_classes/test/zwave_command_class_thermostat_setpoint_test.c @@ -160,6 +160,7 @@ void test_zwave_command_class_thermostat_setpoint_set_v1() &type, sizeof(type)); + int32_t desired_value = 123456; attribute_store_node_t value_node = attribute_store_add_node(ATTRIBUTE(VALUE), setpoint_type_node); @@ -167,6 +168,18 @@ void test_zwave_command_class_thermostat_setpoint_set_v1() &desired_value, sizeof(desired_value)); + // Set precision + uint8_t precision = 0b111; + attribute_store_set_child_reported(setpoint_type_node, + ATTRIBUTE(VALUE_PRECISION), + &precision, + sizeof(precision)); + int32_t scale = 0b11; + attribute_store_set_child_reported(setpoint_type_node, + ATTRIBUTE(VALUE_SCALE), + &scale, + sizeof(scale)); + TEST_ASSERT_NOT_NULL(thermostat_setpoint_set); TEST_ASSERT_EQUAL( SL_STATUS_OK, @@ -175,7 +188,7 @@ void test_zwave_command_class_thermostat_setpoint_set_v1() const uint8_t expected_frame[] = {COMMAND_CLASS_THERMOSTAT_SETPOINT_V3, THERMOSTAT_SETPOINT_SET_V3, type, - SET_DEFAULT_PRECISION | SET_DEFAULT_SIZE, + (precision << 5) | (scale << 3) | 4, (desired_value & 0xFF000000) >> 24, // MSB (desired_value & 0x00FF0000) >> 16, (desired_value & 0x0000FF00) >> 8, @@ -185,6 +198,58 @@ void test_zwave_command_class_thermostat_setpoint_set_v1() received_frame, sizeof(expected_frame)); } +void test_zwave_command_class_thermostat_setpoint_set_v2() +{ + TEST_ASSERT_NOT_NULL(thermostat_setpoint_set); + + // Version 2 node + const zwave_cc_version_t version = 2; + attribute_store_set_child_reported(endpoint_id_node, + ATTRIBUTE(VERSION), + &version, + sizeof(version)); + const uint8_t type = 1; + attribute_store_node_t setpoint_type_node + = attribute_store_emplace(endpoint_id_node, + ATTRIBUTE(TYPE), + &type, + sizeof(type)); + + + int32_t desired_value = 12; + attribute_store_node_t value_node + = attribute_store_add_node(ATTRIBUTE(VALUE), setpoint_type_node); + attribute_store_set_desired(value_node, + &desired_value, + sizeof(desired_value)); + + // Set precision + uint8_t precision = 0b101; + attribute_store_set_child_reported(setpoint_type_node, + ATTRIBUTE(VALUE_PRECISION), + &precision, + sizeof(precision)); + int32_t scale = 0b1; + attribute_store_set_child_reported(setpoint_type_node, + ATTRIBUTE(VALUE_SCALE), + &scale, + sizeof(scale)); + + TEST_ASSERT_NOT_NULL(thermostat_setpoint_set); + TEST_ASSERT_EQUAL( + SL_STATUS_OK, + thermostat_setpoint_set(value_node, received_frame, &received_frame_size)); + + const uint8_t expected_frame[] = {COMMAND_CLASS_THERMOSTAT_SETPOINT_V3, + THERMOSTAT_SETPOINT_SET_V3, + type, + (precision << 5) | (scale << 3) | 1, + (desired_value & 0x000000FF)}; + TEST_ASSERT_EQUAL(sizeof(expected_frame), received_frame_size); + TEST_ASSERT_EQUAL_UINT8_ARRAY(expected_frame, + received_frame, + sizeof(expected_frame)); +} void test_zwave_command_class_thermostat_setpoint_set_v3_clip_lower_bound() { @@ -210,6 +275,18 @@ void test_zwave_command_class_thermostat_setpoint_set_v3_clip_lower_bound() &desired_value, sizeof(desired_value)); + // Set precision + uint8_t precision = 0b010; + attribute_store_set_child_reported(setpoint_type_node, + ATTRIBUTE(VALUE_PRECISION), + &precision, + sizeof(precision)); + int32_t scale = 0b10; + attribute_store_set_child_reported(setpoint_type_node, + ATTRIBUTE(VALUE_SCALE), + &scale, + sizeof(scale)); + const int32_t min_value = DEFAULT_MIN_VALUE; attribute_store_set_child_reported(setpoint_type_node, ATTRIBUTE(MIN_VALUE), @@ -224,9 +301,7 @@ void test_zwave_command_class_thermostat_setpoint_set_v3_clip_lower_bound() const uint8_t expected_frame[] = {COMMAND_CLASS_THERMOSTAT_SETPOINT_V3, THERMOSTAT_SETPOINT_SET_V3, type, - SET_DEFAULT_PRECISION | SET_DEFAULT_SIZE, - (min_value & 0xFF000000) >> 24, // MSB - (min_value & 0x00FF0000) >> 16, + (precision << 5) | (scale << 3) | 2, (min_value & 0x0000FF00) >> 8, (min_value & 0x000000FF)}; TEST_ASSERT_EQUAL(sizeof(expected_frame), received_frame_size); @@ -259,6 +334,18 @@ void test_zwave_command_class_thermostat_setpoint_set_v3_clip_upper_bound() &desired_value, sizeof(desired_value)); + // Set precision + uint8_t precision = 0b001; + attribute_store_set_child_reported(setpoint_type_node, + ATTRIBUTE(VALUE_PRECISION), + &precision, + sizeof(precision)); + int32_t scale = 0b00; + attribute_store_set_child_reported(setpoint_type_node, + ATTRIBUTE(VALUE_SCALE), + &scale, + sizeof(scale)); + const int32_t max_value = DEFAULT_MAX_VALUE; attribute_store_set_child_reported(setpoint_type_node, ATTRIBUTE(MAX_VALUE), @@ -273,7 +360,7 @@ void test_zwave_command_class_thermostat_setpoint_set_v3_clip_upper_bound() const uint8_t expected_frame[] = {COMMAND_CLASS_THERMOSTAT_SETPOINT_V3, THERMOSTAT_SETPOINT_SET_V3, type, - SET_DEFAULT_PRECISION | SET_DEFAULT_SIZE, + (precision << 5) | (scale << 3) | 4, (max_value & 0xFF000000) >> 24, // MSB (max_value & 0x00FF0000) >> 16, (max_value & 0x0000FF00) >> 8, @@ -282,4 +369,71 @@ void test_zwave_command_class_thermostat_setpoint_set_v3_clip_upper_bound() TEST_ASSERT_EQUAL_UINT8_ARRAY(expected_frame, received_frame, sizeof(expected_frame)); +} + +void test_zwave_command_class_thermostat_setpoint_report_happy_case() { + zwave_controller_connection_info_t info = {}; + info.remote.node_id = node_id; + info.remote.endpoint_id = endpoint_id; + info.local.is_multicast = false; + + uint8_t setpoint_type = 1; + uint8_t precision = 0b010; + uint8_t scale = 0b00; + uint8_t size = 2; + int32_t value = 1212; + + uint8_t report_frame[] = {COMMAND_CLASS_THERMOSTAT_SETPOINT, + THERMOSTAT_SETPOINT_REPORT, + setpoint_type, + (precision << 5) | (scale << 3) | size, + (value & 0x0000FF00) >> 8, + (value & 0x000000FF)}; + + // Report should fail since the setpoint_type doesn't exists + TEST_ASSERT_EQUAL( + SL_STATUS_NOT_SUPPORTED, + thermostat_handler.control_handler(&info, report_frame, sizeof(report_frame))); + + // Create setpoint_type + attribute_store_node_t setpoint_type_node + = attribute_store_emplace(endpoint_id_node, + ATTRIBUTE(TYPE), + &setpoint_type, + sizeof(setpoint_type)); + + // Need to create an empty value node first since it isn't created automaticaly + attribute_store_add_node(ATTRIBUTE(VALUE), setpoint_type_node); + + // Now we can do it + TEST_ASSERT_EQUAL( + SL_STATUS_OK, + thermostat_handler.control_handler(&info, report_frame, sizeof(report_frame))); + + int32_t reported_scale = 0; + attribute_store_get_child_reported(setpoint_type_node, + ATTRIBUTE(VALUE_SCALE), + &reported_scale, + sizeof(reported_scale)); + TEST_ASSERT_EQUAL_MESSAGE(scale, + reported_scale, + "Scale value should match"); + + uint8_t reported_precision = 0; + attribute_store_get_child_reported(setpoint_type_node, + ATTRIBUTE(VALUE_PRECISION), + &reported_precision, + sizeof(reported_precision)); + TEST_ASSERT_EQUAL_MESSAGE(precision, + reported_precision, + "Precision value should match"); + + int32_t reported_value; + attribute_store_get_child_reported(setpoint_type_node, + ATTRIBUTE(VALUE), + &reported_value, + sizeof(reported_value)); + TEST_ASSERT_EQUAL_MESSAGE(value * 10, // Precision offset + reported_value, + "Value should match"); } \ No newline at end of file