4cc8067c86
When decoding the optional priority context tag in wp_decode_service_request(), the code passes apdu_len - apdu_size to bacnet_unsigned_context_decode instead of apdu_size - apdu_len. Similar typo in bacnet_action_command_decode().
553 lines
18 KiB
C
553 lines
18 KiB
C
/**
|
||
* @file
|
||
* @brief BACnet WriteProperty service encoder and decoder
|
||
* @author Steve Karg <skarg@users.sourceforge.net>
|
||
* @date 2005
|
||
* @copyright SPDX-License-Identifier: GPL-2.0-or-later WITH GCC-exception-2.0
|
||
*/
|
||
#include <stdbool.h>
|
||
#include <stdint.h>
|
||
/* BACnet Stack defines - first */
|
||
#include "bacnet/bacdef.h"
|
||
/* BACnet Stack API */
|
||
#include "bacnet/bacdcode.h"
|
||
#include "bacnet/proplist.h"
|
||
#include "bacnet/wp.h"
|
||
|
||
/** @file wp.c Encode/Decode BACnet Write Property APDUs */
|
||
|
||
#if BACNET_SVC_WP_A
|
||
/**
|
||
* @brief Encode the WriteProperty service request
|
||
*
|
||
* WriteProperty-Request ::= SEQUENCE {
|
||
* object-identifier [0] BACnetObjectIdentifier,
|
||
* property-identifier [1] BACnetPropertyIdentifier,
|
||
* property-array-index [2] Unsigned OPTIONAL,
|
||
* -- used only with array datatype
|
||
* -- if omitted with an array the entire
|
||
* -- array is referenced
|
||
* property-value [3] ABSTRACT-SYNTAX.&Type,
|
||
* priority [4] Unsigned (1..16) OPTIONAL
|
||
* -– used only when property is commandable
|
||
* }
|
||
*
|
||
* @param apdu Pointer to the buffer, or NULL for length
|
||
* @param invoke_id ID of service invoked.
|
||
* @param data Pointer to the service data used for encoding values
|
||
* @return number of bytes encoded, or zero if unable to encode
|
||
*/
|
||
size_t
|
||
writeproperty_apdu_encode(uint8_t *apdu, const BACNET_WRITE_PROPERTY_DATA *data)
|
||
{
|
||
size_t apdu_len = 0; /* total length of the apdu, return value */
|
||
size_t len = 0; /* total length of the apdu, return value */
|
||
|
||
if (!data) {
|
||
return 0;
|
||
}
|
||
len = encode_context_object_id(
|
||
apdu, 0, data->object_type, data->object_instance);
|
||
apdu_len += len;
|
||
if (apdu) {
|
||
apdu += len;
|
||
}
|
||
len = encode_context_enumerated(apdu, 1, data->object_property);
|
||
apdu_len += len;
|
||
if (apdu) {
|
||
apdu += len;
|
||
}
|
||
/* optional array index; ALL is -1 which is assumed when missing */
|
||
if (data->array_index != BACNET_ARRAY_ALL) {
|
||
len = encode_context_unsigned(apdu, 2, data->array_index);
|
||
apdu_len += len;
|
||
if (apdu) {
|
||
apdu += len;
|
||
}
|
||
}
|
||
/* propertyValue */
|
||
len = encode_opening_tag(apdu, 3);
|
||
apdu_len += len;
|
||
if (apdu) {
|
||
apdu += len;
|
||
}
|
||
for (len = 0; len < data->application_data_len; len++) {
|
||
if (apdu) {
|
||
*apdu = data->application_data[len];
|
||
apdu += 1;
|
||
}
|
||
apdu_len++;
|
||
}
|
||
len = encode_closing_tag(apdu, 3);
|
||
apdu_len += len;
|
||
if (apdu) {
|
||
apdu += len;
|
||
}
|
||
/* optional priority - 0 if not set, 1..16 if set */
|
||
if (data->priority != BACNET_NO_PRIORITY) {
|
||
len = encode_context_unsigned(apdu, 4, data->priority);
|
||
apdu_len += len;
|
||
}
|
||
|
||
return apdu_len;
|
||
}
|
||
|
||
/**
|
||
* @brief Initialize the APDU for encode service.
|
||
* @param apdu Pointer to the buffer, or NULL for length
|
||
* @param apdu Pointer to the buffer for encoding into
|
||
* @param apdu_size number of bytes available in the buffer
|
||
* @param data Pointer to the service data used for encoding values
|
||
* @return number of bytes encoded, or zero if unable to encode or too large
|
||
*/
|
||
size_t writeproperty_service_request_encode(
|
||
uint8_t *apdu, size_t apdu_size, const BACNET_WRITE_PROPERTY_DATA *data)
|
||
{
|
||
size_t apdu_len = 0; /* total length of the apdu, return value */
|
||
|
||
apdu_len = writeproperty_apdu_encode(NULL, data);
|
||
if (apdu_len > apdu_size) {
|
||
apdu_len = 0;
|
||
} else {
|
||
apdu_len = writeproperty_apdu_encode(apdu, data);
|
||
}
|
||
|
||
return apdu_len;
|
||
}
|
||
|
||
/**
|
||
* @brief Initialize the APDU for encode service.
|
||
*
|
||
* WriteProperty-Request ::= SEQUENCE {
|
||
* object-identifier [0] BACnetObjectIdentifier,
|
||
* property-identifier [1] BACnetPropertyIdentifier,
|
||
* property-array-index [2] Unsigned OPTIONAL,
|
||
* -- used only with array datatype
|
||
* -- if omitted with an array the entire
|
||
* -- array is referenced
|
||
* property-value [3] ABSTRACT-SYNTAX.&Type,
|
||
* priority [4] Unsigned (1..16) OPTIONAL
|
||
* -– used only when property is commandable
|
||
* }
|
||
*
|
||
* @param apdu Pointer to the buffer, or NULL for length
|
||
* @param invoke_id ID of service invoked.
|
||
* @param wpdata Pointer to the write property data.
|
||
*
|
||
* @return Bytes encoded
|
||
*/
|
||
int wp_encode_apdu(
|
||
uint8_t *apdu, uint8_t invoke_id, const BACNET_WRITE_PROPERTY_DATA *wpdata)
|
||
{
|
||
int apdu_len = 0; /* total length of the apdu, return value */
|
||
int len = 0; /* total length of the apdu, return value */
|
||
|
||
if (!wpdata) {
|
||
return BACNET_STATUS_ERROR;
|
||
}
|
||
if (apdu) {
|
||
apdu[0] = PDU_TYPE_CONFIRMED_SERVICE_REQUEST;
|
||
apdu[1] = encode_max_segs_max_apdu(0, MAX_APDU);
|
||
apdu[2] = invoke_id;
|
||
apdu[3] = SERVICE_CONFIRMED_WRITE_PROPERTY; /* service choice */
|
||
}
|
||
len = 4;
|
||
apdu_len += len;
|
||
if (apdu) {
|
||
apdu += len;
|
||
}
|
||
len = writeproperty_apdu_encode(apdu, wpdata);
|
||
apdu_len += len;
|
||
|
||
return apdu_len;
|
||
}
|
||
#endif
|
||
|
||
/**
|
||
* @brief Decode the service request only
|
||
*
|
||
* WriteProperty-Request ::= SEQUENCE {
|
||
* object-identifier [0] BACnetObjectIdentifier,
|
||
* property-identifier [1] BACnetPropertyIdentifier,
|
||
* property-array-index [2] Unsigned OPTIONAL,
|
||
* -- used only with array datatype
|
||
* -- if omitted with an array the entire
|
||
* -- array is referenced
|
||
* property-value [3] ABSTRACT-SYNTAX.&Type,
|
||
* priority [4] Unsigned (1..16) OPTIONAL
|
||
* -– used only when property is commandable
|
||
* }
|
||
*
|
||
* @param apdu Pointer to the buffer.
|
||
* @param apdu_size Valid bytes in the buffer
|
||
* @param wpdata Pointer to the write property data.
|
||
*
|
||
* @return number of bytes decoded, or #BACNET_STATUS_ERROR
|
||
*/
|
||
int wp_decode_service_request(
|
||
const uint8_t *apdu, unsigned apdu_size, BACNET_WRITE_PROPERTY_DATA *wpdata)
|
||
{
|
||
int len = 0;
|
||
int apdu_len = 0;
|
||
uint32_t instance = 0;
|
||
BACNET_OBJECT_TYPE type = OBJECT_NONE; /* for decoding */
|
||
uint32_t property = 0; /* for decoding */
|
||
BACNET_UNSIGNED_INTEGER unsigned_value = 0;
|
||
int i = 0; /* loop counter */
|
||
int imax = 0; /* max application data length */
|
||
|
||
/* check for value pointers */
|
||
if (!apdu) {
|
||
return BACNET_STATUS_ERROR;
|
||
}
|
||
/* object-identifier [0] BACnetObjectIdentifier */
|
||
len = bacnet_object_id_context_decode(
|
||
&apdu[apdu_len], apdu_size - apdu_len, 0, &type, &instance);
|
||
if (len > 0) {
|
||
if (instance > BACNET_MAX_INSTANCE) {
|
||
return BACNET_STATUS_ERROR;
|
||
}
|
||
apdu_len += len;
|
||
if (wpdata) {
|
||
wpdata->object_type = type;
|
||
wpdata->object_instance = instance;
|
||
}
|
||
} else {
|
||
return BACNET_STATUS_ERROR;
|
||
}
|
||
/* property-identifier [1] BACnetPropertyIdentifier */
|
||
len = bacnet_enumerated_context_decode(
|
||
&apdu[apdu_len], apdu_size - apdu_len, 1, &property);
|
||
if (len > 0) {
|
||
apdu_len += len;
|
||
if (wpdata) {
|
||
wpdata->object_property = (BACNET_PROPERTY_ID)property;
|
||
}
|
||
} else {
|
||
return BACNET_STATUS_ERROR;
|
||
}
|
||
/* property-array-index [2] Unsigned OPTIONAL */
|
||
len = bacnet_unsigned_context_decode(
|
||
&apdu[apdu_len], apdu_size - apdu_len, 2, &unsigned_value);
|
||
if (len > 0) {
|
||
apdu_len += len;
|
||
if (wpdata) {
|
||
wpdata->array_index = unsigned_value;
|
||
}
|
||
} else {
|
||
/* wrong tag - skip apdu_len increment and go to next field */
|
||
if (wpdata) {
|
||
wpdata->array_index = BACNET_ARRAY_ALL;
|
||
}
|
||
}
|
||
/* property-value [3] ABSTRACT-SYNTAX.&Type */
|
||
if (!bacnet_is_opening_tag_number(
|
||
&apdu[apdu_len], apdu_size - apdu_len, 3, &len)) {
|
||
return BACNET_STATUS_ERROR;
|
||
}
|
||
/* determine the length of the data blob */
|
||
imax = bacnet_enclosed_data_length(&apdu[apdu_len], apdu_size - apdu_len);
|
||
if (imax == BACNET_STATUS_ERROR) {
|
||
return BACNET_STATUS_ERROR;
|
||
}
|
||
/* count the opening tag number length */
|
||
apdu_len += len;
|
||
/* copy the data from the APDU */
|
||
if (imax > MAX_APDU) {
|
||
/* not enough size in application_data to store the data chunk */
|
||
return BACNET_STATUS_ERROR;
|
||
} else if (wpdata) {
|
||
for (i = 0; i < imax; i++) {
|
||
wpdata->application_data[i] = apdu[apdu_len + i];
|
||
}
|
||
wpdata->application_data_len = imax;
|
||
}
|
||
/* add on the data length */
|
||
apdu_len += imax;
|
||
if (!bacnet_is_closing_tag_number(
|
||
&apdu[apdu_len], apdu_size - apdu_len, 3, &len)) {
|
||
return BACNET_STATUS_ERROR;
|
||
}
|
||
/* count the closing tag number length */
|
||
apdu_len += len;
|
||
/* priority [4] Unsigned (1..16) OPTIONAL */
|
||
/* assumed MAX priority if not explicitly set */
|
||
if (wpdata) {
|
||
wpdata->priority = BACNET_MAX_PRIORITY;
|
||
}
|
||
if ((unsigned)apdu_len < apdu_size) {
|
||
len = bacnet_unsigned_context_decode(
|
||
&apdu[apdu_len], apdu_size - apdu_len, 4, &unsigned_value);
|
||
if (len > 0) {
|
||
apdu_len += len;
|
||
if ((unsigned_value >= BACNET_MIN_PRIORITY) &&
|
||
(unsigned_value <= BACNET_MAX_PRIORITY)) {
|
||
if (wpdata) {
|
||
wpdata->priority = (uint8_t)unsigned_value;
|
||
}
|
||
} else {
|
||
return BACNET_STATUS_ERROR;
|
||
}
|
||
} else {
|
||
return BACNET_STATUS_ERROR;
|
||
}
|
||
}
|
||
|
||
return apdu_len;
|
||
}
|
||
|
||
/**
|
||
* @brief simple validation of value tag for Write Property argument
|
||
* @param wp_data - #BACNET_WRITE_PROPERTY_DATA data, including
|
||
* requested data and space for the reply, or error response.
|
||
* @param value - #BACNET_APPLICATION_DATA_VALUE data, for the tag
|
||
* @param expected_tag - the tag that is expected for this property value
|
||
* @return true if the expected tag matches the value tag
|
||
*/
|
||
bool write_property_type_valid(
|
||
BACNET_WRITE_PROPERTY_DATA *wp_data,
|
||
const BACNET_APPLICATION_DATA_VALUE *value,
|
||
uint8_t expected_tag)
|
||
{
|
||
/* assume success */
|
||
bool valid = true;
|
||
|
||
if (value && (value->tag != expected_tag)) {
|
||
valid = false;
|
||
if (wp_data) {
|
||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||
wp_data->error_code = ERROR_CODE_INVALID_DATA_TYPE;
|
||
}
|
||
}
|
||
|
||
return (valid);
|
||
}
|
||
|
||
/**
|
||
* @brief simple validation of character string value for Write Property
|
||
* @param wp_data - #BACNET_WRITE_PROPERTY_DATA data, including
|
||
* requested data and space for the reply, or error response.
|
||
* @param value - #BACNET_APPLICATION_DATA_VALUE data, for the tag
|
||
* @param len_max - max length accepted for a character string, or 0=unchecked
|
||
* @return true if the character string value is valid
|
||
*/
|
||
bool write_property_string_valid(
|
||
BACNET_WRITE_PROPERTY_DATA *wp_data,
|
||
const BACNET_APPLICATION_DATA_VALUE *value,
|
||
size_t len_max)
|
||
{
|
||
bool valid = false;
|
||
|
||
if (value && (value->tag == BACNET_APPLICATION_TAG_CHARACTER_STRING)) {
|
||
if (characterstring_encoding(&value->type.Character_String) ==
|
||
CHARACTER_ANSI_X34) {
|
||
if (characterstring_length(&value->type.Character_String) == 0) {
|
||
if (wp_data) {
|
||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||
wp_data->error_code = ERROR_CODE_VALUE_OUT_OF_RANGE;
|
||
}
|
||
} else if (!characterstring_printable(
|
||
&value->type.Character_String)) {
|
||
/* assumption: non-empty also means must be "printable" */
|
||
if (wp_data) {
|
||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||
wp_data->error_code = ERROR_CODE_VALUE_OUT_OF_RANGE;
|
||
}
|
||
} else if (
|
||
(len_max > 0) &&
|
||
(characterstring_length(&value->type.Character_String) >
|
||
len_max)) {
|
||
if (wp_data) {
|
||
wp_data->error_class = ERROR_CLASS_RESOURCES;
|
||
wp_data->error_code = ERROR_CODE_NO_SPACE_TO_WRITE_PROPERTY;
|
||
}
|
||
} else {
|
||
/* It's all good! */
|
||
valid = true;
|
||
}
|
||
} else {
|
||
if (wp_data) {
|
||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||
wp_data->error_code = ERROR_CODE_CHARACTER_SET_NOT_SUPPORTED;
|
||
}
|
||
}
|
||
} else {
|
||
if (wp_data) {
|
||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||
wp_data->error_code = ERROR_CODE_INVALID_DATA_TYPE;
|
||
}
|
||
}
|
||
|
||
return (valid);
|
||
}
|
||
|
||
/**
|
||
* @brief simple validation of character string value for Write Property
|
||
* for character strings which can be empty
|
||
* @param wp_data - #BACNET_WRITE_PROPERTY_DATA data, including
|
||
* requested data and space for the reply, or error response.
|
||
* @param value - #BACNET_APPLICATION_DATA_VALUE data, for the tag
|
||
* @param len_max - max length accepted for a character string, or 0=unchecked
|
||
* @return true if the character string value is valid
|
||
*/
|
||
bool write_property_empty_string_valid(
|
||
BACNET_WRITE_PROPERTY_DATA *wp_data,
|
||
const BACNET_APPLICATION_DATA_VALUE *value,
|
||
size_t len_max)
|
||
{
|
||
bool valid = false;
|
||
|
||
if (value && (value->tag == BACNET_APPLICATION_TAG_CHARACTER_STRING)) {
|
||
if (characterstring_encoding(&value->type.Character_String) ==
|
||
CHARACTER_ANSI_X34) {
|
||
if ((len_max > 0) &&
|
||
(characterstring_length(&value->type.Character_String) >
|
||
len_max)) {
|
||
if (wp_data) {
|
||
wp_data->error_class = ERROR_CLASS_RESOURCES;
|
||
wp_data->error_code = ERROR_CODE_NO_SPACE_TO_WRITE_PROPERTY;
|
||
}
|
||
} else {
|
||
/* It's all good! */
|
||
valid = true;
|
||
}
|
||
} else {
|
||
if (wp_data) {
|
||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||
wp_data->error_code = ERROR_CODE_CHARACTER_SET_NOT_SUPPORTED;
|
||
}
|
||
}
|
||
} else {
|
||
if (wp_data) {
|
||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||
wp_data->error_code = ERROR_CODE_INVALID_DATA_TYPE;
|
||
}
|
||
}
|
||
|
||
return (valid);
|
||
}
|
||
|
||
/**
|
||
* @brief simple validation of BACnetARRAY for Write Property
|
||
* @param data - #BACNET_WRITE_PROPERTY_DATA data, including
|
||
* requested data and space for the reply, or error response.
|
||
* @return true if the property is an array and the request uses array
|
||
* indices.
|
||
*/
|
||
bool write_property_bacnet_array_valid(BACNET_WRITE_PROPERTY_DATA *data)
|
||
{
|
||
bool is_array;
|
||
|
||
/* only array properties can have array options */
|
||
is_array = property_list_bacnet_array_member(
|
||
data->object_type, data->object_property);
|
||
if (!is_array && (data->array_index != BACNET_ARRAY_ALL)) {
|
||
data->error_class = ERROR_CLASS_PROPERTY;
|
||
data->error_code = ERROR_CODE_PROPERTY_IS_NOT_AN_ARRAY;
|
||
return false;
|
||
}
|
||
|
||
return true;
|
||
}
|
||
|
||
/**
|
||
* @brief Helper to decode a WriteProperty unsigned integer and set a property
|
||
* @param wp_data - #BACNET_WRITE_PROPERTY_DATA data including any
|
||
* error response.
|
||
* @param value - #BACNET_APPLICATION_DATA_VALUE data
|
||
* @param setter - function to set the property
|
||
* @param maximum - maximum value allowed for the property
|
||
* @return true if the value was decoded and set, else false
|
||
*/
|
||
bool write_property_unsigned_decode(
|
||
BACNET_WRITE_PROPERTY_DATA *wp_data,
|
||
BACNET_APPLICATION_DATA_VALUE *value,
|
||
bacnet_property_unsigned_setter setter,
|
||
BACNET_UNSIGNED_INTEGER maximum)
|
||
{
|
||
bool status = write_property_type_valid(
|
||
wp_data, value, BACNET_APPLICATION_TAG_UNSIGNED_INT);
|
||
if (status) {
|
||
if (value->type.Unsigned_Int <= maximum) {
|
||
status =
|
||
(setter)(wp_data->object_instance, value->type.Unsigned_Int);
|
||
if (status) {
|
||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||
wp_data->error_code = ERROR_CODE_SUCCESS;
|
||
} else {
|
||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||
wp_data->error_code = ERROR_CODE_VALUE_OUT_OF_RANGE;
|
||
}
|
||
} else {
|
||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||
wp_data->error_code = ERROR_CODE_VALUE_OUT_OF_RANGE;
|
||
status = false;
|
||
}
|
||
}
|
||
|
||
return status;
|
||
}
|
||
|
||
/**
|
||
* @brief Handler for a WriteProperty Service request when the
|
||
* property is a NULL type and the property is not commandable
|
||
*
|
||
* 15.9.2 WriteProperty Service Procedure
|
||
*
|
||
* If an attempt is made to relinquish a property that is
|
||
* not commandable and for which Null is not a supported
|
||
* datatype, if no other error conditions exist,
|
||
* the property shall not be changed, and the write shall
|
||
* be considered successful.
|
||
*
|
||
* @note There was an interpretation request in April 2025 that clarifies
|
||
* that the NULL bypass is only for present-value property of objects that
|
||
* optionally support a priority array but don't implement it.
|
||
* See 135-2024-19.2.1.1 Commandable Properties for the list of commandable
|
||
* properties of specific objects.
|
||
*
|
||
* @param wp_data [in] The WriteProperty data structure
|
||
* @param member_of_object [in] Function to check if a property is a member
|
||
of an object instance
|
||
* @return true if the write shall be considered successful
|
||
*/
|
||
bool write_property_relinquish_bypass(
|
||
BACNET_WRITE_PROPERTY_DATA *wp_data,
|
||
write_property_member_of_object member_of_object)
|
||
{
|
||
bool bypass = false;
|
||
bool has_priority_array = false;
|
||
int len = 0;
|
||
|
||
if (!wp_data) {
|
||
return false;
|
||
}
|
||
len = bacnet_null_application_decode(
|
||
wp_data->application_data, wp_data->application_data_len);
|
||
if ((len > 0) && (len == wp_data->application_data_len)) {
|
||
/* single NULL */
|
||
if (property_list_commandable_member(
|
||
wp_data->object_type, wp_data->object_property)) {
|
||
if (member_of_object) {
|
||
/* check to see if this object property is commandable.
|
||
Does the property list contain a priority-array? */
|
||
has_priority_array = member_of_object(
|
||
wp_data->object_type, wp_data->object_instance,
|
||
PROP_PRIORITY_ARRAY);
|
||
}
|
||
if (has_priority_array ||
|
||
(wp_data->object_type == OBJECT_CHANNEL)) {
|
||
/* this property is commandable and shall not be bypassed */
|
||
} else {
|
||
/* this property that is optionally commandable
|
||
is not commandable for this object instance,
|
||
so it "shall not be changed, and
|
||
the write shall be considered successful." */
|
||
bypass = true;
|
||
}
|
||
}
|
||
}
|
||
|
||
return bypass;
|
||
}
|