3ea710f92f
Added segmentation support for server devices for some services. Configure BACNET_SEGMENTATION_ENABLED=1 to include in the library, and adjust BACNET_MAX_SEGMENTS_ACCEPTED for maximum number of segments.
231 lines
9.5 KiB
C
231 lines
9.5 KiB
C
/**************************************************************************
|
|
*
|
|
* Copyright (C) 2005 Steve Karg <skarg@users.sourceforge.net>
|
|
*
|
|
* SPDX-License-Identifier: MIT
|
|
*
|
|
*********************************************************************/
|
|
#include <stddef.h>
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
/* BACnet Stack defines - first */
|
|
#include "bacnet/bacdef.h"
|
|
/* BACnet Stack API */
|
|
#include "bacnet/bacdcode.h"
|
|
#include "bacnet/bacerror.h"
|
|
#include "bacnet/bacdevobjpropref.h"
|
|
#include "bacnet/apdu.h"
|
|
#include "bacnet/npdu.h"
|
|
#include "bacnet/abort.h"
|
|
#include "bacnet/reject.h"
|
|
#include "bacnet/rp.h"
|
|
/* basic objects, services, TSM, and datalink */
|
|
#include "bacnet/basic/object/device.h"
|
|
#if (BACNET_PROTOCOL_REVISION >= 17)
|
|
#include "bacnet/basic/object/netport.h"
|
|
#endif
|
|
#include "bacnet/basic/tsm/tsm.h"
|
|
#include "bacnet/basic/services.h"
|
|
#include "bacnet/basic/sys/debug.h"
|
|
#include "bacnet/datalink/datalink.h"
|
|
|
|
#if BACNET_SEGMENTATION_ENABLED
|
|
#define BACNET_RP_BUFFER_OVERFLOW ERROR_CODE_ABORT_BUFFER_OVERFLOW
|
|
#else
|
|
#define BACNET_RP_BUFFER_OVERFLOW ERROR_CODE_ABORT_SEGMENTATION_NOT_SUPPORTED
|
|
#endif
|
|
|
|
/** @file h_rp.c Handles Read Property requests. */
|
|
|
|
/** Handler for a ReadProperty Service request.
|
|
* @ingroup DSRP
|
|
* This handler will be invoked by apdu_handler() if it has been enabled
|
|
* by a call to apdu_set_confirmed_handler().
|
|
* This handler builds a response packet, which is
|
|
* - an Abort if
|
|
* - the message is segmented and segmentation is not supported
|
|
* - if decoding fails
|
|
* - if the response would be too large
|
|
* - the result from Device_Read_Property(), if it succeeds
|
|
* - an Error if Device_Read_Property() fails
|
|
* or there isn't enough room in the APDU to fit the data.
|
|
*
|
|
* @param service_request [in] The contents of the service request.
|
|
* @param service_len [in] The length of the service_request.
|
|
* @param src [in] BACNET_ADDRESS of the source of the message
|
|
* @param service_data [in] The BACNET_CONFIRMED_SERVICE_DATA information
|
|
* decoded from the APDU header of this message.
|
|
*/
|
|
void handler_read_property(
|
|
uint8_t *service_request,
|
|
uint16_t service_len,
|
|
BACNET_ADDRESS *src,
|
|
BACNET_CONFIRMED_SERVICE_DATA *service_data)
|
|
{
|
|
BACNET_READ_PROPERTY_DATA rpdata;
|
|
int len = 0;
|
|
int max_resp = 0;
|
|
int pdu_len = 0;
|
|
int apdu_len = -1;
|
|
int npdu_len = -1;
|
|
BACNET_NPDU_DATA npdu_data;
|
|
bool error = true; /* assume that there is an error */
|
|
int bytes_sent = 0;
|
|
BACNET_ADDRESS my_address;
|
|
#if BACNET_SEGMENTATION_ENABLED
|
|
BACNET_APDU_FIXED_HEADER apdu_fixed_header;
|
|
int apdu_header_len = 3;
|
|
#endif
|
|
|
|
/* configure default error code as an abort since it is common */
|
|
rpdata.error_code = BACNET_RP_BUFFER_OVERFLOW;
|
|
/* encode the NPDU portion of the packet */
|
|
datalink_get_my_address(&my_address);
|
|
npdu_encode_npdu_data(&npdu_data, false, service_data->priority);
|
|
npdu_len = npdu_encode_pdu(
|
|
&Handler_Transmit_Buffer[0], src, &my_address, &npdu_data);
|
|
if (npdu_len <= 0) {
|
|
/* If 0 or negative, there were problems with the data or encoding. */
|
|
len = BACNET_STATUS_ABORT;
|
|
debug_print("RP: npdu_encode_pdu error. Sending Abort!\n");
|
|
} else if (service_len == 0) {
|
|
len = BACNET_STATUS_REJECT;
|
|
rpdata.error_code = ERROR_CODE_REJECT_MISSING_REQUIRED_PARAMETER;
|
|
debug_print("RP: Missing Required Parameter. Sending Reject!\n");
|
|
#if !BACNET_SEGMENTATION_ENABLED
|
|
} else if (service_data->segmented_message) {
|
|
/* we don't support segmentation - send an abort */
|
|
len = BACNET_STATUS_ABORT;
|
|
debug_print("RP: Segmented message. Sending Abort!\n");
|
|
#endif
|
|
} else {
|
|
len = rp_decode_service_request(service_request, service_len, &rpdata);
|
|
if (len <= 0) {
|
|
debug_print("RP: Unable to decode Request!\n");
|
|
} else {
|
|
/* When the object-type in the Object Identifier parameter
|
|
contains the value DEVICE and the instance in the 'Object
|
|
Identifier' parameter contains the value 4194303, the responding
|
|
BACnet-user shall treat the Object Identifier as if it correctly
|
|
matched the local Device object. This allows the device instance
|
|
of a device that does not generate I-Am messages to be
|
|
determined. */
|
|
if ((rpdata.object_type == OBJECT_DEVICE) &&
|
|
(rpdata.object_instance == BACNET_MAX_INSTANCE)) {
|
|
rpdata.object_instance = Device_Object_Instance_Number();
|
|
}
|
|
#if (BACNET_PROTOCOL_REVISION >= 17)
|
|
/* When the object-type in the Object Identifier parameter
|
|
contains the value NETWORK_PORT and the instance in the 'Object
|
|
Identifier' parameter contains the value 4194303, the responding
|
|
BACnet-user shall treat the Object Identifier as if it correctly
|
|
matched the local Network Port object representing the network
|
|
port through which the request was received. This allows the
|
|
network port instance of the network port that was used to
|
|
receive the request to be determined. */
|
|
if ((rpdata.object_type == OBJECT_NETWORK_PORT) &&
|
|
(rpdata.object_instance == BACNET_MAX_INSTANCE)) {
|
|
rpdata.object_instance = Network_Port_Index_To_Instance(0);
|
|
}
|
|
#endif
|
|
|
|
#if BACNET_SEGMENTATION_ENABLED
|
|
apdu_init_fixed_header(
|
|
&apdu_fixed_header, PDU_TYPE_COMPLEX_ACK,
|
|
service_data->invoke_id, SERVICE_CONFIRMED_READ_PROPERTY,
|
|
service_data->max_resp);
|
|
#endif
|
|
apdu_len = rp_ack_encode_apdu_init(
|
|
&Handler_Transmit_Buffer[npdu_len], service_data->invoke_id,
|
|
&rpdata);
|
|
/* configure our storage */
|
|
rpdata.application_data =
|
|
&Handler_Transmit_Buffer[npdu_len + apdu_len];
|
|
rpdata.application_data_len =
|
|
sizeof(Handler_Transmit_Buffer) - (npdu_len + apdu_len);
|
|
if (!read_property_bacnet_array_valid(&rpdata)) {
|
|
len = BACNET_STATUS_ERROR;
|
|
} else {
|
|
len = Device_Read_Property(&rpdata);
|
|
}
|
|
if (len >= 0) {
|
|
apdu_len += len;
|
|
len = rp_ack_encode_apdu_object_property_end(
|
|
&Handler_Transmit_Buffer[npdu_len + apdu_len]);
|
|
apdu_len += len;
|
|
/* pick the smaller response packet: ours or theirs */
|
|
max_resp = min(service_data->max_resp, MAX_APDU);
|
|
if (apdu_len > max_resp) {
|
|
#if BACNET_SEGMENTATION_ENABLED
|
|
if (service_data->segmented_response_accepted) {
|
|
npdu_encode_npdu_data(
|
|
&npdu_data, true, MESSAGE_PRIORITY_NORMAL);
|
|
npdu_len = npdu_encode_pdu(
|
|
&Handler_Transmit_Buffer[0], src, &my_address,
|
|
&npdu_data);
|
|
|
|
tsm_set_complexack_transaction(
|
|
src, &npdu_data, &apdu_fixed_header, service_data,
|
|
&Handler_Transmit_Buffer
|
|
[npdu_len + apdu_header_len],
|
|
(apdu_len - apdu_header_len));
|
|
return;
|
|
}
|
|
#else
|
|
/* too big for the sender - send an abort!
|
|
Setting of error code needed here as read property
|
|
processing may have overridden the default set at start
|
|
*/
|
|
rpdata.error_code =
|
|
ERROR_CODE_ABORT_SEGMENTATION_NOT_SUPPORTED;
|
|
len = BACNET_STATUS_ABORT;
|
|
debug_print("RP: Message too large.\n");
|
|
#endif
|
|
} else {
|
|
debug_print("RP: Sending Ack!\n");
|
|
error = false;
|
|
}
|
|
} else {
|
|
debug_print("RP: Device_Read_Property: ");
|
|
if (len == BACNET_STATUS_ABORT) {
|
|
debug_print("Abort!\n");
|
|
} else if (len == BACNET_STATUS_ERROR) {
|
|
debug_print("Error!\n");
|
|
} else if (len == BACNET_STATUS_REJECT) {
|
|
debug_print("Reject!\n");
|
|
} else {
|
|
debug_print("Unknown Len!\n");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (error) {
|
|
if (len == BACNET_STATUS_ABORT) {
|
|
apdu_len = abort_encode_apdu(
|
|
&Handler_Transmit_Buffer[npdu_len], service_data->invoke_id,
|
|
abort_convert_error_code(rpdata.error_code), true);
|
|
debug_print("RP: Sending Abort!\n");
|
|
} else if (len == BACNET_STATUS_ERROR) {
|
|
apdu_len = bacerror_encode_apdu(
|
|
&Handler_Transmit_Buffer[npdu_len], service_data->invoke_id,
|
|
SERVICE_CONFIRMED_READ_PROPERTY, rpdata.error_class,
|
|
rpdata.error_code);
|
|
debug_print("RP: Sending Error!\n");
|
|
} else if (len == BACNET_STATUS_REJECT) {
|
|
apdu_len = reject_encode_apdu(
|
|
&Handler_Transmit_Buffer[npdu_len], service_data->invoke_id,
|
|
reject_convert_error_code(rpdata.error_code));
|
|
debug_print("RP: Sending Reject!\n");
|
|
}
|
|
}
|
|
pdu_len = npdu_len + apdu_len;
|
|
bytes_sent = datalink_send_pdu(
|
|
src, &npdu_data, &Handler_Transmit_Buffer[0], pdu_len);
|
|
if (bytes_sent <= 0) {
|
|
debug_perror("RP: Failed to send PDU");
|
|
}
|
|
|
|
return;
|
|
}
|