Ported bacnet-basic from zephyr project to create basis for mini example. (#933)
This commit is contained in:
+22
-1
@@ -25,6 +25,11 @@ option(
|
||||
ON)
|
||||
|
||||
option(
|
||||
BACNET_BUILD_BACMINI_APP
|
||||
"compile the bacmini app"
|
||||
ON)
|
||||
|
||||
option(
|
||||
BACNET_BUILD_PIFACE_APP
|
||||
"compile the piface app"
|
||||
OFF)
|
||||
@@ -84,7 +89,7 @@ option(
|
||||
"build with uci"
|
||||
OFF)
|
||||
|
||||
set(BACNET_PROTOCOL_REVISION 19)
|
||||
set(BACNET_PROTOCOL_REVISION 24)
|
||||
|
||||
if(NOT CMAKE_BUILD_TYPE)
|
||||
set(CMAKE_BUILD_TYPE Release)
|
||||
@@ -912,6 +917,22 @@ if(BACNET_STACK_BUILD_APPS)
|
||||
target_link_libraries(piface PRIVATE ${PROJECT_NAME})
|
||||
endif(BACNET_BUILD_PIFACE_APP)
|
||||
|
||||
if(BACNET_BUILD_BACMINI_APP)
|
||||
add_executable(bacmini
|
||||
apps/server-basic/main.c
|
||||
src/bacnet/basic/server/bacnet_basic.c
|
||||
src/bacnet/basic/server/bacnet_device.c
|
||||
src/bacnet/basic/server/bacnet_port.c
|
||||
src/bacnet/basic/server/bacnet_port_ipv4.c
|
||||
src/bacnet/basic/server/bacnet_port_ipv6.c
|
||||
)
|
||||
target_link_libraries(bacmini PRIVATE ${PROJECT_NAME})
|
||||
target_compile_options(bacmini PRIVATE
|
||||
# Unreachable code because we have endless loop.
|
||||
$<$<C_COMPILER_ID:MSVC>:/wd4702>
|
||||
)
|
||||
endif(BACNET_BUILD_BACMINI_APP)
|
||||
|
||||
if(BACNET_BUILD_BACPOLL_APP)
|
||||
add_executable(bacpoll
|
||||
apps/server-client/main.c
|
||||
|
||||
@@ -202,6 +202,10 @@ netnumis:
|
||||
server:
|
||||
$(MAKE) -s -C apps $@
|
||||
|
||||
.PHONY: server-basic
|
||||
server-basic:
|
||||
$(MAKE) LEGACY=true NOTIFY=false -s -C apps $@
|
||||
|
||||
.PHONY: server-client
|
||||
server-client:
|
||||
$(MAKE) LEGACY=true -s -C apps $@
|
||||
|
||||
+11
-2
@@ -184,9 +184,14 @@ ifeq (${LEGACY},true)
|
||||
BACNET_DEFINES += -DBACNET_STACK_DEPRECATED_DISABLE
|
||||
endif
|
||||
|
||||
ifeq (${NOTIFY},false)
|
||||
# disable intrinsic reporting
|
||||
else
|
||||
BACNET_DEFINES += -DINTRINSIC_REPORTING
|
||||
endif
|
||||
|
||||
BACNET_DEFINES += -DPRINT_ENABLED=1
|
||||
BACNET_DEFINES += -DBACAPP_ALL
|
||||
BACNET_DEFINES += -DINTRINSIC_REPORTING
|
||||
BACNET_DEFINES += -DBACNET_TIME_MASTER
|
||||
BACNET_DEFINES += -DBACNET_PROPERTY_LISTS=1
|
||||
BACNET_DEFINES += -DBACNET_PROTOCOL_REVISION=24
|
||||
@@ -217,7 +222,7 @@ SUBDIRS = lib readprop writeprop readfile writefile reinit server dcc \
|
||||
whohas whois iam ucov scov timesync epics readpropm readrange \
|
||||
writepropm uptransfer getevent uevent abort error event ack-alarm \
|
||||
server-client add-list-element remove-list-element create-object \
|
||||
delete-object server-discover apdu writegroup
|
||||
delete-object server-discover apdu writegroup server-basic
|
||||
|
||||
ifeq (${BACDL_DEFINE},-DBACDL_BIP=1)
|
||||
SUBDIRS += whoisrouter iamrouter initrouter whatisnetnum netnumis
|
||||
@@ -406,6 +411,10 @@ scov: $(BACNET_LIB_TARGET)
|
||||
server: $(BACNET_LIB_TARGET)
|
||||
$(MAKE) -B -C $@
|
||||
|
||||
.PHONY: server-basic
|
||||
server-basic: $(BACNET_LIB_TARGET)
|
||||
$(MAKE) -B -C $@
|
||||
|
||||
.PHONY: server-client
|
||||
server-client: $(BACNET_LIB_TARGET)
|
||||
$(MAKE) -B -C $@
|
||||
|
||||
@@ -0,0 +1,71 @@
|
||||
#Makefile to build BACnet Application using GCC compiler
|
||||
|
||||
# Executable file name
|
||||
TARGET = bacmini
|
||||
# BACnet objects supporting CreateObject that are used with this app
|
||||
BACNET_OBJECT_DIR = $(BACNET_SRC_DIR)/bacnet/basic/object
|
||||
BACNET_SERVER_DIR = $(BACNET_SRC_DIR)/bacnet/basic/server
|
||||
SRC = main.c \
|
||||
$(BACNET_SERVER_DIR)/bacnet_basic.c \
|
||||
$(BACNET_SERVER_DIR)/bacnet_device.c \
|
||||
$(BACNET_SERVER_DIR)/bacnet_port.c \
|
||||
$(BACNET_SERVER_DIR)/bacnet_port_ipv4.c \
|
||||
$(BACNET_SERVER_DIR)/bacnet_port_ipv6.c \
|
||||
$(BACNET_OBJECT_DIR)/ai.c \
|
||||
$(BACNET_OBJECT_DIR)/ao.c \
|
||||
$(BACNET_OBJECT_DIR)/av.c \
|
||||
$(BACNET_OBJECT_DIR)/bacfile.c \
|
||||
$(BACNET_OBJECT_DIR)/bi.c \
|
||||
$(BACNET_OBJECT_DIR)/bitstring_value.c \
|
||||
$(BACNET_OBJECT_DIR)/blo.c \
|
||||
$(BACNET_OBJECT_DIR)/bo.c \
|
||||
$(BACNET_OBJECT_DIR)/bv.c \
|
||||
$(BACNET_OBJECT_DIR)/calendar.c \
|
||||
$(BACNET_OBJECT_DIR)/channel.c \
|
||||
$(BACNET_OBJECT_DIR)/color_object.c \
|
||||
$(BACNET_OBJECT_DIR)/color_temperature.c \
|
||||
$(BACNET_OBJECT_DIR)/command.c \
|
||||
$(BACNET_OBJECT_DIR)/csv.c \
|
||||
$(BACNET_OBJECT_DIR)/iv.c \
|
||||
$(BACNET_OBJECT_DIR)/lc.c \
|
||||
$(BACNET_OBJECT_DIR)/lo.c \
|
||||
$(BACNET_OBJECT_DIR)/lsp.c \
|
||||
$(BACNET_OBJECT_DIR)/lsz.c \
|
||||
$(BACNET_OBJECT_DIR)/ms-input.c \
|
||||
$(BACNET_OBJECT_DIR)/mso.c \
|
||||
$(BACNET_OBJECT_DIR)/msv.c \
|
||||
$(BACNET_OBJECT_DIR)/nc.c \
|
||||
$(BACNET_OBJECT_DIR)/netport.c \
|
||||
$(BACNET_OBJECT_DIR)/osv.c \
|
||||
$(BACNET_OBJECT_DIR)/structured_view.c \
|
||||
$(BACNET_OBJECT_DIR)/time_value.c
|
||||
|
||||
# TARGET_EXT is defined in apps/Makefile as .exe or nothing
|
||||
TARGET_BIN = ${TARGET}$(TARGET_EXT)
|
||||
|
||||
OBJS += ${SRC:.c=.o}
|
||||
|
||||
all: ${BACNET_LIB_TARGET} Makefile ${TARGET_BIN}
|
||||
|
||||
${TARGET_BIN}: ${OBJS} Makefile ${BACNET_LIB_TARGET}
|
||||
${CC} ${PFLAGS} ${OBJS} ${LFLAGS} -o $@
|
||||
size $@
|
||||
cp $@ ../../bin
|
||||
|
||||
${BACNET_LIB_TARGET}:
|
||||
( cd ${BACNET_LIB_DIR} ; $(MAKE) clean ; $(MAKE) -s )
|
||||
|
||||
.c.o:
|
||||
${CC} -c ${CFLAGS} $*.c -o $@
|
||||
|
||||
.PHONY: depend
|
||||
depend:
|
||||
rm -f .depend
|
||||
${CC} -MM ${CFLAGS} *.c >> .depend
|
||||
|
||||
.PHONY: clean
|
||||
clean:
|
||||
rm -f core ${TARGET_BIN} ${OBJS} $(TARGET).map ${BACNET_LIB_TARGET}
|
||||
|
||||
.PHONY: include
|
||||
include: .depend
|
||||
@@ -0,0 +1,186 @@
|
||||
/**
|
||||
* @file
|
||||
* @brief BACnet Stack sample Smart Sensor (B-SS) main file
|
||||
* @author Steve Karg <skarg@users.sourceforge.net>
|
||||
* @date October 2024
|
||||
* @copyright SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
#include <stddef.h>
|
||||
#include <stdint.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
/* BACnet Stack defines - first */
|
||||
#include "bacnet/bacdef.h"
|
||||
/* BACnet Stack API */
|
||||
#include "bacnet/bactext.h"
|
||||
#include "bacnet/version.h"
|
||||
/* some demo stuff needed */
|
||||
#include "bacnet/basic/sys/filename.h"
|
||||
#include "bacnet/basic/sys/debug.h"
|
||||
#include "bacnet/basic/sys/mstimer.h"
|
||||
#include "bacnet/basic/server/bacnet_basic.h"
|
||||
#include "bacnet/basic/server/bacnet_port.h"
|
||||
#include "bacnet/basic/object/device.h"
|
||||
#include "bacnet/basic/object/ai.h"
|
||||
#include "bacnet/basic/object/ao.h"
|
||||
#include "bacnet/datalink/datalink.h"
|
||||
#include "bacnet/datalink/dlenv.h"
|
||||
|
||||
static const char *Device_Name = "BACnet Smart Sensor (B-SS)";
|
||||
#define SENSOR_ID 1
|
||||
|
||||
/* table of default point names and units */
|
||||
struct BACnetObjectTable {
|
||||
const uint8_t type;
|
||||
const uint8_t instance;
|
||||
const uint16_t units;
|
||||
const float value;
|
||||
const char *name;
|
||||
const char *description;
|
||||
};
|
||||
static struct BACnetObjectTable Object_Table[] = {
|
||||
{ OBJECT_ANALOG_INPUT, SENSOR_ID, UNITS_DEGREES_CELSIUS, 25.0f,
|
||||
"Indoor Air Temperature", "indoor air temperature" },
|
||||
{ OBJECT_ANALOG_OUTPUT, 1, UNITS_DEGREES_ANGULAR, 0.0f, "VAV Actuator",
|
||||
"variable air valve actuator" },
|
||||
{ OBJECT_ANALOG_OUTPUT, 2, UNITS_DEGREES_CELSIUS, 25.0f,
|
||||
"Temperature Setpoint", "indoor air temperature setpoint" },
|
||||
};
|
||||
/* timer for Sensor Update Interval */
|
||||
static struct mstimer Sensor_Update_Timer;
|
||||
|
||||
/**
|
||||
* @brief BACnet Project Initialization Handler
|
||||
* @param context [in] The context to pass to the callback function
|
||||
* @note This is called from the BACnet task
|
||||
*/
|
||||
static void BACnet_Object_Table_Init(void *context)
|
||||
{
|
||||
unsigned i;
|
||||
|
||||
(void)context;
|
||||
/* initialize child objects for this basic sample */
|
||||
for (i = 0; i < ARRAY_SIZE(Object_Table); i++) {
|
||||
switch (Object_Table[i].type) {
|
||||
case OBJECT_ANALOG_INPUT:
|
||||
Analog_Input_Create(Object_Table[i].instance);
|
||||
Analog_Input_Name_Set(
|
||||
Object_Table[i].instance, Object_Table[i].name);
|
||||
Analog_Input_Present_Value_Set(
|
||||
Object_Table[i].instance, Object_Table[i].value);
|
||||
Analog_Input_Units_Set(
|
||||
Object_Table[i].instance, Object_Table[i].units);
|
||||
Analog_Input_Description_Set(
|
||||
Object_Table[i].instance, Object_Table[i].description);
|
||||
break;
|
||||
case OBJECT_ANALOG_OUTPUT:
|
||||
Analog_Output_Create(Object_Table[i].instance);
|
||||
Analog_Output_Name_Set(
|
||||
Object_Table[i].instance, Object_Table[i].name);
|
||||
Analog_Output_Relinquish_Default_Set(
|
||||
Object_Table[i].instance, Object_Table[i].value);
|
||||
Analog_Output_Units_Set(
|
||||
Object_Table[i].instance, Object_Table[i].units);
|
||||
Analog_Output_Description_Set(
|
||||
Object_Table[i].instance, Object_Table[i].description);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
/* start the seconds cyclic timer */
|
||||
mstimer_set(&Sensor_Update_Timer, 1000);
|
||||
srand(0);
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief BACnet Project Task Handler
|
||||
* @param context [in] The context to pass to the callback function
|
||||
* @note This is called from the BACnet task
|
||||
*/
|
||||
static void BACnet_Object_Task(void *context)
|
||||
{
|
||||
float temperature = 0.0f, change = 0.0f;
|
||||
|
||||
(void)context;
|
||||
if (mstimer_expired(&Sensor_Update_Timer)) {
|
||||
mstimer_reset(&Sensor_Update_Timer);
|
||||
/* simulate a sensor reading, and update the BACnet object values */
|
||||
if (Analog_Input_Out_Of_Service(SENSOR_ID)) {
|
||||
return;
|
||||
}
|
||||
temperature = Analog_Input_Present_Value(SENSOR_ID);
|
||||
change = -1.0f + 2.0f * ((float)rand()) / RAND_MAX;
|
||||
temperature += change;
|
||||
Analog_Input_Present_Value_Set(SENSOR_ID, temperature);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Store the BACnet data after a WriteProperty for object property
|
||||
* @param object_type - BACnet object type
|
||||
* @param object_instance - BACnet object instance
|
||||
* @param object_property - BACnet object property
|
||||
* @param array_index - BACnet array index
|
||||
* @param application_data - pointer to the data
|
||||
* @param application_data_len - length of the data
|
||||
*/
|
||||
static void BACnet_Basic_Store(
|
||||
BACNET_OBJECT_TYPE object_type,
|
||||
uint32_t object_instance,
|
||||
BACNET_PROPERTY_ID object_property,
|
||||
BACNET_ARRAY_INDEX array_index,
|
||||
uint8_t *application_data,
|
||||
int application_data_len)
|
||||
{
|
||||
(void)array_index;
|
||||
(void)application_data;
|
||||
(void)application_data_len;
|
||||
debug_printf_stdout(
|
||||
"BACnet Store: %s[%lu]-%s\n", bactext_object_type_name(object_type),
|
||||
(unsigned long)object_instance, bactext_property_name(object_property));
|
||||
}
|
||||
|
||||
/** Main function of server demo.
|
||||
*
|
||||
* @see Device_Set_Object_Instance_Number, dlenv_init, Send_I_Am,
|
||||
* datalink_receive, npdu_handler,
|
||||
* dcc_timer_seconds, datalink_maintenance_timer,
|
||||
* handler_cov_task, tsm_timer_milliseconds
|
||||
*
|
||||
* @param argc [in] Arg count.
|
||||
* @param argv [in] Takes one argument: the Device Instance #.
|
||||
* @return 0 on success.
|
||||
*/
|
||||
int main(int argc, char *argv[])
|
||||
{
|
||||
if (argc > 1) {
|
||||
/* allow the device ID to be set */
|
||||
Device_Set_Object_Instance_Number(strtol(argv[1], NULL, 0));
|
||||
}
|
||||
if (argc > 2) {
|
||||
/* allow the device name to be set */
|
||||
Device_Name = argv[2];
|
||||
}
|
||||
Device_Object_Name_ANSI_Init(Device_Name);
|
||||
debug_printf_stdout("BACnet Device: %s\n", Device_Name);
|
||||
debug_printf_stdout("BACnet Stack Version %s\n", BACNET_VERSION_TEXT);
|
||||
debug_printf_stdout("BACnet Stack Max APDU: %d\n", MAX_APDU);
|
||||
bacnet_basic_init_callback_set(BACnet_Object_Table_Init, NULL);
|
||||
bacnet_basic_task_callback_set(BACnet_Object_Task, NULL);
|
||||
bacnet_basic_store_callback_set(BACnet_Basic_Store);
|
||||
bacnet_basic_init();
|
||||
if (bacnet_port_init()) {
|
||||
/* OS based apps use DLENV for environment variables */
|
||||
dlenv_init();
|
||||
atexit(datalink_cleanup);
|
||||
}
|
||||
debug_printf_stdout("Server: initialized\n");
|
||||
for (;;) {
|
||||
bacnet_basic_task();
|
||||
bacnet_port_task();
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
@@ -3,7 +3,6 @@
|
||||
# Executable file name - BACnet Server-Client Polling Application
|
||||
TARGET = bacpoll
|
||||
# BACnet objects that are used with this app
|
||||
# BACnet objects that are used with this app
|
||||
BACNET_OBJECT_DIR = $(BACNET_SRC_DIR)/bacnet/basic/object
|
||||
BACNET_CLIENT_DIR = $(BACNET_SRC_DIR)/bacnet/basic/client
|
||||
SRC = main.c \
|
||||
|
||||
@@ -57,7 +57,8 @@ static analog_output_write_present_value_callback
|
||||
|
||||
/* These three arrays are used by the ReadPropertyMultiple handler */
|
||||
|
||||
static const int Analog_Output_Properties_Required[] = {
|
||||
static const int Properties_Required[] = {
|
||||
/* unordered list of required properties */
|
||||
PROP_OBJECT_IDENTIFIER,
|
||||
PROP_OBJECT_NAME,
|
||||
PROP_OBJECT_TYPE,
|
||||
@@ -74,12 +75,13 @@ static const int Analog_Output_Properties_Required[] = {
|
||||
-1
|
||||
};
|
||||
|
||||
static const int Analog_Output_Properties_Optional[] = {
|
||||
static const int Properties_Optional[] = {
|
||||
/* unordered list of optional properties */
|
||||
PROP_RELIABILITY, PROP_DESCRIPTION, PROP_COV_INCREMENT,
|
||||
PROP_MIN_PRES_VALUE, PROP_MAX_PRES_VALUE, -1
|
||||
};
|
||||
|
||||
static const int Analog_Output_Properties_Proprietary[] = { -1 };
|
||||
static const int Properties_Proprietary[] = { -1 };
|
||||
|
||||
/**
|
||||
* @brief Returns the list of required, optional, and proprietary properties.
|
||||
@@ -95,13 +97,13 @@ void Analog_Output_Property_Lists(
|
||||
const int **pRequired, const int **pOptional, const int **pProprietary)
|
||||
{
|
||||
if (pRequired) {
|
||||
*pRequired = Analog_Output_Properties_Required;
|
||||
*pRequired = Properties_Required;
|
||||
}
|
||||
if (pOptional) {
|
||||
*pOptional = Analog_Output_Properties_Optional;
|
||||
*pOptional = Properties_Optional;
|
||||
}
|
||||
if (pProprietary) {
|
||||
*pProprietary = Analog_Output_Properties_Proprietary;
|
||||
*pProprietary = Properties_Proprietary;
|
||||
}
|
||||
|
||||
return;
|
||||
@@ -985,7 +987,13 @@ int Analog_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
(rpdata->application_data_len == 0)) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (!property_lists_member(
|
||||
Properties_Required, Properties_Optional, Properties_Proprietary,
|
||||
rpdata->object_property)) {
|
||||
rpdata->error_class = ERROR_CLASS_PROPERTY;
|
||||
rpdata->error_code = ERROR_CODE_UNKNOWN_PROPERTY;
|
||||
return BACNET_STATUS_ERROR;
|
||||
}
|
||||
apdu = rpdata->application_data;
|
||||
apdu_size = rpdata->application_data_len;
|
||||
switch (rpdata->object_property) {
|
||||
@@ -1069,7 +1077,6 @@ int Analog_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
apdu_len = encode_application_real(
|
||||
&apdu[0], Analog_Output_COV_Increment(rpdata->object_instance));
|
||||
break;
|
||||
#if (BACNET_PROTOCOL_REVISION >= 17)
|
||||
case PROP_CURRENT_COMMAND_PRIORITY:
|
||||
i = Analog_Output_Present_Value_Priority(rpdata->object_instance);
|
||||
if ((i >= BACNET_MIN_PRIORITY) && (i <= BACNET_MAX_PRIORITY)) {
|
||||
@@ -1078,7 +1085,6 @@ int Analog_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
apdu_len = encode_application_null(&apdu[0]);
|
||||
}
|
||||
break;
|
||||
#endif
|
||||
default:
|
||||
rpdata->error_class = ERROR_CLASS_PROPERTY;
|
||||
rpdata->error_code = ERROR_CODE_UNKNOWN_PROPERTY;
|
||||
@@ -1139,28 +1145,16 @@ bool Analog_Output_Write_Property(BACNET_WRITE_PROPERTY_DATA *wp_data)
|
||||
wp_data->object_instance, value.type.Boolean);
|
||||
}
|
||||
break;
|
||||
case PROP_COV_INCREMENT:
|
||||
case PROP_OBJECT_IDENTIFIER:
|
||||
case PROP_OBJECT_TYPE:
|
||||
case PROP_OBJECT_NAME:
|
||||
case PROP_STATUS_FLAGS:
|
||||
case PROP_EVENT_STATE:
|
||||
case PROP_UNITS:
|
||||
case PROP_RELIABILITY:
|
||||
case PROP_PRIORITY_ARRAY:
|
||||
case PROP_RELINQUISH_DEFAULT:
|
||||
case PROP_MAX_PRES_VALUE:
|
||||
case PROP_MIN_PRES_VALUE:
|
||||
case PROP_DESCRIPTION:
|
||||
#if (BACNET_PROTOCOL_REVISION >= 17)
|
||||
case PROP_CURRENT_COMMAND_PRIORITY:
|
||||
#endif
|
||||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||||
wp_data->error_code = ERROR_CODE_WRITE_ACCESS_DENIED;
|
||||
break;
|
||||
default:
|
||||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||||
wp_data->error_code = ERROR_CODE_UNKNOWN_PROPERTY;
|
||||
if (property_lists_member(
|
||||
Properties_Required, Properties_Optional,
|
||||
Properties_Proprietary, wp_data->object_property)) {
|
||||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||||
wp_data->error_code = ERROR_CODE_WRITE_ACCESS_DENIED;
|
||||
} else {
|
||||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||||
wp_data->error_code = ERROR_CODE_UNKNOWN_PROPERTY;
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
|
||||
@@ -59,7 +59,8 @@ static binary_lighting_output_blink_warn_callback
|
||||
|
||||
/* These arrays are used by the ReadPropertyMultiple handler and
|
||||
property-list property (as of protocol-revision 14) */
|
||||
static const int Binary_Lighting_Output_Properties_Required[] = {
|
||||
static const int Properties_Required[] = {
|
||||
/* unordered list of required properties */
|
||||
PROP_OBJECT_IDENTIFIER,
|
||||
PROP_OBJECT_NAME,
|
||||
PROP_OBJECT_TYPE,
|
||||
@@ -76,11 +77,12 @@ static const int Binary_Lighting_Output_Properties_Required[] = {
|
||||
#endif
|
||||
-1
|
||||
};
|
||||
static const int Binary_Lighting_Output_Properties_Optional[] = {
|
||||
static const int Properties_Optional[] = {
|
||||
/* unordered list of optional properties */
|
||||
PROP_DESCRIPTION, PROP_RELIABILITY, PROP_FEEDBACK_VALUE, -1
|
||||
};
|
||||
|
||||
static const int Binary_Lighting_Output_Properties_Proprietary[] = { -1 };
|
||||
static const int Properties_Proprietary[] = { -1 };
|
||||
|
||||
/**
|
||||
* Returns the list of required, optional, and proprietary properties.
|
||||
@@ -97,13 +99,13 @@ void Binary_Lighting_Output_Property_Lists(
|
||||
const int **pRequired, const int **pOptional, const int **pProprietary)
|
||||
{
|
||||
if (pRequired) {
|
||||
*pRequired = Binary_Lighting_Output_Properties_Required;
|
||||
*pRequired = Properties_Required;
|
||||
}
|
||||
if (pOptional) {
|
||||
*pOptional = Binary_Lighting_Output_Properties_Optional;
|
||||
*pOptional = Properties_Optional;
|
||||
}
|
||||
if (pProprietary) {
|
||||
*pProprietary = Binary_Lighting_Output_Properties_Proprietary;
|
||||
*pProprietary = Properties_Proprietary;
|
||||
}
|
||||
|
||||
return;
|
||||
@@ -1263,6 +1265,13 @@ int Binary_Lighting_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
(rpdata->application_data_len == 0)) {
|
||||
return 0;
|
||||
}
|
||||
if (!property_lists_member(
|
||||
Properties_Required, Properties_Optional, Properties_Proprietary,
|
||||
rpdata->object_property)) {
|
||||
rpdata->error_class = ERROR_CLASS_PROPERTY;
|
||||
rpdata->error_code = ERROR_CODE_UNKNOWN_PROPERTY;
|
||||
return BACNET_STATUS_ERROR;
|
||||
}
|
||||
apdu = rpdata->application_data;
|
||||
apdu_size = rpdata->application_data_len;
|
||||
switch (rpdata->object_property) {
|
||||
@@ -1338,7 +1347,6 @@ int Binary_Lighting_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
rpdata->object_instance);
|
||||
apdu_len = encode_application_enumerated(&apdu[0], lighting_value);
|
||||
break;
|
||||
#if (BACNET_PROTOCOL_REVISION >= 17)
|
||||
case PROP_CURRENT_COMMAND_PRIORITY:
|
||||
i = Binary_Lighting_Output_Present_Value_Priority(
|
||||
rpdata->object_instance);
|
||||
@@ -1348,7 +1356,6 @@ int Binary_Lighting_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
apdu_len = encode_application_null(&apdu[0]);
|
||||
}
|
||||
break;
|
||||
#endif
|
||||
case PROP_DESCRIPTION:
|
||||
characterstring_init_ansi(
|
||||
&char_string,
|
||||
@@ -1424,30 +1431,16 @@ bool Binary_Lighting_Output_Write_Property(BACNET_WRITE_PROPERTY_DATA *wp_data)
|
||||
wp_data->object_instance, value.type.Boolean);
|
||||
}
|
||||
break;
|
||||
case PROP_OBJECT_IDENTIFIER:
|
||||
case PROP_OBJECT_NAME:
|
||||
case PROP_OBJECT_TYPE:
|
||||
case PROP_TRACKING_VALUE:
|
||||
case PROP_IN_PROGRESS:
|
||||
case PROP_STATUS_FLAGS:
|
||||
case PROP_BLINK_WARN_ENABLE:
|
||||
case PROP_EGRESS_TIME:
|
||||
case PROP_EGRESS_ACTIVE:
|
||||
case PROP_PRIORITY_ARRAY:
|
||||
case PROP_RELINQUISH_DEFAULT:
|
||||
case PROP_LIGHTING_COMMAND_DEFAULT_PRIORITY:
|
||||
#if (BACNET_PROTOCOL_REVISION >= 17)
|
||||
case PROP_CURRENT_COMMAND_PRIORITY:
|
||||
#endif
|
||||
case PROP_DESCRIPTION:
|
||||
case PROP_RELIABILITY:
|
||||
case PROP_FEEDBACK_VALUE:
|
||||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||||
wp_data->error_code = ERROR_CODE_WRITE_ACCESS_DENIED;
|
||||
break;
|
||||
default:
|
||||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||||
wp_data->error_code = ERROR_CODE_UNKNOWN_PROPERTY;
|
||||
if (property_lists_member(
|
||||
Properties_Required, Properties_Optional,
|
||||
Properties_Proprietary, wp_data->object_property)) {
|
||||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||||
wp_data->error_code = ERROR_CODE_WRITE_ACCESS_DENIED;
|
||||
} else {
|
||||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||||
wp_data->error_code = ERROR_CODE_UNKNOWN_PROPERTY;
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
|
||||
@@ -56,24 +56,28 @@ static binary_output_write_present_value_callback
|
||||
Binary_Output_Write_Present_Value_Callback;
|
||||
|
||||
/* These three arrays are used by the ReadPropertyMultiple handler */
|
||||
static const int Properties_Required[] = { PROP_OBJECT_IDENTIFIER,
|
||||
PROP_OBJECT_NAME,
|
||||
PROP_OBJECT_TYPE,
|
||||
PROP_PRESENT_VALUE,
|
||||
PROP_STATUS_FLAGS,
|
||||
PROP_EVENT_STATE,
|
||||
PROP_OUT_OF_SERVICE,
|
||||
PROP_POLARITY,
|
||||
PROP_PRIORITY_ARRAY,
|
||||
PROP_RELINQUISH_DEFAULT,
|
||||
static const int Properties_Required[] = {
|
||||
/* unordered list of required properties */
|
||||
PROP_OBJECT_IDENTIFIER,
|
||||
PROP_OBJECT_NAME,
|
||||
PROP_OBJECT_TYPE,
|
||||
PROP_PRESENT_VALUE,
|
||||
PROP_STATUS_FLAGS,
|
||||
PROP_EVENT_STATE,
|
||||
PROP_OUT_OF_SERVICE,
|
||||
PROP_POLARITY,
|
||||
PROP_PRIORITY_ARRAY,
|
||||
PROP_RELINQUISH_DEFAULT,
|
||||
#if (BACNET_PROTOCOL_REVISION >= 17)
|
||||
PROP_CURRENT_COMMAND_PRIORITY,
|
||||
PROP_CURRENT_COMMAND_PRIORITY,
|
||||
#endif
|
||||
-1 };
|
||||
-1
|
||||
};
|
||||
|
||||
static const int Properties_Optional[] = { PROP_RELIABILITY, PROP_DESCRIPTION,
|
||||
PROP_ACTIVE_TEXT, PROP_INACTIVE_TEXT,
|
||||
-1 };
|
||||
static const int Properties_Optional[] = {
|
||||
/* unordered list of optional properties */
|
||||
PROP_RELIABILITY, PROP_DESCRIPTION, PROP_ACTIVE_TEXT, PROP_INACTIVE_TEXT, -1
|
||||
};
|
||||
|
||||
static const int Properties_Proprietary[] = { -1 };
|
||||
|
||||
@@ -987,6 +991,13 @@ int Binary_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
(rpdata->application_data_len == 0)) {
|
||||
return 0;
|
||||
}
|
||||
if (!property_lists_member(
|
||||
Properties_Required, Properties_Optional, Properties_Proprietary,
|
||||
rpdata->object_property)) {
|
||||
rpdata->error_class = ERROR_CLASS_PROPERTY;
|
||||
rpdata->error_code = ERROR_CODE_UNKNOWN_PROPERTY;
|
||||
return BACNET_STATUS_ERROR;
|
||||
}
|
||||
apdu = rpdata->application_data;
|
||||
apdu_size = rpdata->application_data_len;
|
||||
switch (rpdata->object_property) {
|
||||
@@ -1073,7 +1084,6 @@ int Binary_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
apdu_len =
|
||||
encode_application_character_string(&apdu[0], &char_string);
|
||||
break;
|
||||
#if (BACNET_PROTOCOL_REVISION >= 17)
|
||||
case PROP_CURRENT_COMMAND_PRIORITY:
|
||||
i = Binary_Output_Present_Value_Priority(rpdata->object_instance);
|
||||
if ((i >= BACNET_MIN_PRIORITY) && (i <= BACNET_MAX_PRIORITY)) {
|
||||
@@ -1082,7 +1092,6 @@ int Binary_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
apdu_len = encode_application_null(&apdu[0]);
|
||||
}
|
||||
break;
|
||||
#endif
|
||||
default:
|
||||
rpdata->error_class = ERROR_CLASS_PROPERTY;
|
||||
rpdata->error_code = ERROR_CODE_UNKNOWN_PROPERTY;
|
||||
|
||||
@@ -125,7 +125,7 @@ uint32_t CharacterString_Value_Create(uint32_t object_instance)
|
||||
* @param object_instance - object-instance number of the object
|
||||
* @return true if the object is deleted
|
||||
*/
|
||||
bool Characterstring_Value_Delete(uint32_t object_instance)
|
||||
bool CharacterString_Value_Delete(uint32_t object_instance)
|
||||
{
|
||||
bool status = false;
|
||||
struct object_data *pObject = NULL;
|
||||
@@ -142,7 +142,7 @@ bool Characterstring_Value_Delete(uint32_t object_instance)
|
||||
/**
|
||||
* @brief Cleans up the object list and its data
|
||||
*/
|
||||
void Characterstring_Value_Cleanup(void)
|
||||
void CharacterString_Value_Cleanup(void)
|
||||
{
|
||||
struct object_data *pObject = NULL;
|
||||
|
||||
|
||||
@@ -27,9 +27,9 @@ void CharacterString_Value_Property_Lists(
|
||||
BACNET_STACK_EXPORT
|
||||
uint32_t CharacterString_Value_Create(uint32_t object_instance);
|
||||
BACNET_STACK_EXPORT
|
||||
bool Characterstring_Value_Delete(uint32_t object_instance);
|
||||
bool CharacterString_Value_Delete(uint32_t object_instance);
|
||||
BACNET_STACK_EXPORT
|
||||
void Characterstring_Value_Cleanup(void);
|
||||
void CharacterString_Value_Cleanup(void);
|
||||
BACNET_STACK_EXPORT
|
||||
bool CharacterString_Value_Valid_Instance(uint32_t object_instance);
|
||||
BACNET_STACK_EXPORT
|
||||
|
||||
@@ -52,28 +52,18 @@
|
||||
#if defined(BACFILE)
|
||||
#include "bacnet/basic/object/bacfile.h"
|
||||
#endif /* defined(BACFILE) */
|
||||
#if (BACNET_PROTOCOL_REVISION >= 10)
|
||||
#include "bacnet/basic/object/bitstring_value.h"
|
||||
#include "bacnet/basic/object/csv.h"
|
||||
#include "bacnet/basic/object/iv.h"
|
||||
#include "bacnet/basic/object/osv.h"
|
||||
#include "bacnet/basic/object/piv.h"
|
||||
#include "bacnet/basic/object/time_value.h"
|
||||
#endif
|
||||
#if (BACNET_PROTOCOL_REVISION >= 14)
|
||||
#include "bacnet/basic/object/channel.h"
|
||||
#include "bacnet/basic/object/lo.h"
|
||||
#endif
|
||||
#if (BACNET_PROTOCOL_REVISION >= 16)
|
||||
#include "bacnet/basic/object/blo.h"
|
||||
#endif
|
||||
#if (BACNET_PROTOCOL_REVISION >= 17)
|
||||
#include "bacnet/basic/object/netport.h"
|
||||
#endif
|
||||
#if (BACNET_PROTOCOL_REVISION >= 24)
|
||||
#include "bacnet/basic/object/color_object.h"
|
||||
#include "bacnet/basic/object/color_temperature.h"
|
||||
#endif
|
||||
|
||||
/* external prototypes */
|
||||
extern int Routed_Device_Read_Property_Local(BACNET_READ_PROPERTY_DATA *rpdata);
|
||||
@@ -187,7 +177,8 @@ static object_functions_t My_Object_Table[] = {
|
||||
CharacterString_Value_Change_Of_Value,
|
||||
CharacterString_Value_Change_Of_Value_Clear,
|
||||
NULL /* Intrinsic Reporting */, NULL /* Add_List_Element */,
|
||||
NULL /* Remove_List_Element */, NULL /* Create */, NULL /* Delete */,
|
||||
NULL /* Remove_List_Element */, CharacterString_Value_Create,
|
||||
CharacterString_Value_Delete,
|
||||
NULL /* Timer */ },
|
||||
{ OBJECT_OCTETSTRING_VALUE, OctetString_Value_Init, OctetString_Value_Count,
|
||||
OctetString_Value_Index_To_Instance, OctetString_Value_Valid_Instance,
|
||||
@@ -215,14 +206,6 @@ static object_functions_t My_Object_Table[] = {
|
||||
NULL /* COV */, NULL /* COV Clear */, NULL /* Intrinsic Reporting */,
|
||||
NULL /* Add_List_Element */, NULL /* Remove_List_Element */,
|
||||
NULL /* Create */, NULL /* Delete */, NULL /* Timer */ },
|
||||
#endif
|
||||
{ OBJECT_COMMAND, Command_Init, Command_Count, Command_Index_To_Instance,
|
||||
Command_Valid_Instance, Command_Object_Name, Command_Read_Property,
|
||||
Command_Write_Property, Command_Property_Lists,
|
||||
NULL /* ReadRangeInfo */, NULL /* Iterator */, NULL /* Value_Lists */,
|
||||
NULL /* COV */, NULL /* COV Clear */, NULL /* Intrinsic Reporting */,
|
||||
NULL /* Add_List_Element */, NULL /* Remove_List_Element */,
|
||||
NULL /* Create */, NULL /* Delete */, NULL /* Timer */ },
|
||||
{ OBJECT_INTEGER_VALUE, Integer_Value_Init, Integer_Value_Count,
|
||||
Integer_Value_Index_To_Instance, Integer_Value_Valid_Instance,
|
||||
Integer_Value_Object_Name, Integer_Value_Read_Property,
|
||||
@@ -230,8 +213,16 @@ static object_functions_t My_Object_Table[] = {
|
||||
NULL /* ReadRangeInfo */, NULL /* Iterator */, NULL /* Value_Lists */,
|
||||
NULL /* COV */, NULL /* COV Clear */, NULL /* Intrinsic Reporting */,
|
||||
NULL /* Add_List_Element */, NULL /* Remove_List_Element */,
|
||||
NULL /* Create */, NULL /* Delete */, NULL /* Timer */ },
|
||||
#if defined(INTRINSIC_REPORTING)
|
||||
Integer_Value_Create, Integer_Value_Delete, NULL /* Timer */ },
|
||||
#endif
|
||||
{ OBJECT_COMMAND, Command_Init, Command_Count, Command_Index_To_Instance,
|
||||
Command_Valid_Instance, Command_Object_Name, Command_Read_Property,
|
||||
Command_Write_Property, Command_Property_Lists,
|
||||
NULL /* ReadRangeInfo */, NULL /* Iterator */, NULL /* Value_Lists */,
|
||||
NULL /* COV */, NULL /* COV Clear */, NULL /* Intrinsic Reporting */,
|
||||
NULL /* Add_List_Element */, NULL /* Remove_List_Element */,
|
||||
NULL /* Create */, NULL /* Delete */, NULL /* Timer */ },
|
||||
#if defined(INTRINSIC_REPORTING)
|
||||
{ OBJECT_NOTIFICATION_CLASS, Notification_Class_Init,
|
||||
Notification_Class_Count, Notification_Class_Index_To_Instance,
|
||||
Notification_Class_Valid_Instance, Notification_Class_Object_Name,
|
||||
|
||||
@@ -63,7 +63,8 @@ static lighting_command_tracking_value_callback
|
||||
|
||||
/* These arrays are used by the ReadPropertyMultiple handler and
|
||||
property-list property (as of protocol-revision 14) */
|
||||
static const int Lighting_Output_Properties_Required[] = {
|
||||
static const int Properties_Required[] = {
|
||||
/* unordered list of required properties */
|
||||
PROP_OBJECT_IDENTIFIER,
|
||||
PROP_OBJECT_NAME,
|
||||
PROP_OBJECT_TYPE,
|
||||
@@ -87,7 +88,8 @@ static const int Lighting_Output_Properties_Required[] = {
|
||||
#endif
|
||||
-1
|
||||
};
|
||||
static const int Lighting_Output_Properties_Optional[] = {
|
||||
static const int Properties_Optional[] = {
|
||||
/* unordered list of optional properties */
|
||||
PROP_DESCRIPTION,
|
||||
PROP_TRANSITION,
|
||||
#if (BACNET_PROTOCOL_REVISION >= 24)
|
||||
@@ -98,7 +100,7 @@ static const int Lighting_Output_Properties_Optional[] = {
|
||||
-1
|
||||
};
|
||||
|
||||
static const int Lighting_Output_Properties_Proprietary[] = { -1 };
|
||||
static const int Properties_Proprietary[] = { -1 };
|
||||
|
||||
/**
|
||||
* Returns the list of required, optional, and proprietary properties.
|
||||
@@ -115,13 +117,13 @@ void Lighting_Output_Property_Lists(
|
||||
const int **pRequired, const int **pOptional, const int **pProprietary)
|
||||
{
|
||||
if (pRequired) {
|
||||
*pRequired = Lighting_Output_Properties_Required;
|
||||
*pRequired = Properties_Required;
|
||||
}
|
||||
if (pOptional) {
|
||||
*pOptional = Lighting_Output_Properties_Optional;
|
||||
*pOptional = Properties_Optional;
|
||||
}
|
||||
if (pProprietary) {
|
||||
*pProprietary = Lighting_Output_Properties_Proprietary;
|
||||
*pProprietary = Properties_Proprietary;
|
||||
}
|
||||
|
||||
return;
|
||||
@@ -2004,9 +2006,7 @@ int Lighting_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
BACNET_BIT_STRING bit_string;
|
||||
BACNET_CHARACTER_STRING char_string;
|
||||
BACNET_LIGHTING_COMMAND lighting_command;
|
||||
#if (BACNET_PROTOCOL_REVISION >= 24)
|
||||
BACNET_OBJECT_ID object_id = { 0 };
|
||||
#endif
|
||||
float real_value = (float)1.414;
|
||||
uint32_t unsigned_value = 0;
|
||||
unsigned i = 0;
|
||||
@@ -2017,6 +2017,13 @@ int Lighting_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
(rpdata->application_data_len == 0)) {
|
||||
return 0;
|
||||
}
|
||||
if (!property_lists_member(
|
||||
Properties_Required, Properties_Optional, Properties_Proprietary,
|
||||
rpdata->object_property)) {
|
||||
rpdata->error_class = ERROR_CLASS_PROPERTY;
|
||||
rpdata->error_code = ERROR_CODE_UNKNOWN_PROPERTY;
|
||||
return BACNET_STATUS_ERROR;
|
||||
}
|
||||
apdu = rpdata->application_data;
|
||||
apdu_size = rpdata->application_data_len;
|
||||
switch (rpdata->object_property) {
|
||||
@@ -2120,7 +2127,6 @@ int Lighting_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
Lighting_Output_Default_Priority(rpdata->object_instance);
|
||||
apdu_len = encode_application_unsigned(&apdu[0], unsigned_value);
|
||||
break;
|
||||
#if (BACNET_PROTOCOL_REVISION >= 17)
|
||||
case PROP_CURRENT_COMMAND_PRIORITY:
|
||||
i = Lighting_Output_Present_Value_Priority(rpdata->object_instance);
|
||||
if ((i >= BACNET_MIN_PRIORITY) && (i <= BACNET_MAX_PRIORITY)) {
|
||||
@@ -2129,8 +2135,6 @@ int Lighting_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
apdu_len = encode_application_null(&apdu[0]);
|
||||
}
|
||||
break;
|
||||
#endif
|
||||
#if (BACNET_PROTOCOL_REVISION >= 24)
|
||||
case PROP_COLOR_OVERRIDE:
|
||||
apdu_len = encode_application_boolean(
|
||||
&apdu[0],
|
||||
@@ -2148,7 +2152,6 @@ int Lighting_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
apdu_len = encode_application_object_id(
|
||||
&apdu[0], object_id.type, object_id.instance);
|
||||
break;
|
||||
#endif
|
||||
case PROP_DESCRIPTION:
|
||||
characterstring_init_ansi(
|
||||
&char_string,
|
||||
@@ -2274,33 +2277,16 @@ bool Lighting_Output_Write_Property(BACNET_WRITE_PROPERTY_DATA *wp_data)
|
||||
&wp_data->error_code);
|
||||
}
|
||||
break;
|
||||
case PROP_OBJECT_IDENTIFIER:
|
||||
case PROP_OBJECT_NAME:
|
||||
case PROP_OBJECT_TYPE:
|
||||
case PROP_TRACKING_VALUE:
|
||||
case PROP_IN_PROGRESS:
|
||||
case PROP_STATUS_FLAGS:
|
||||
case PROP_BLINK_WARN_ENABLE:
|
||||
case PROP_EGRESS_TIME:
|
||||
case PROP_EGRESS_ACTIVE:
|
||||
case PROP_PRIORITY_ARRAY:
|
||||
case PROP_RELINQUISH_DEFAULT:
|
||||
case PROP_LIGHTING_COMMAND_DEFAULT_PRIORITY:
|
||||
#if (BACNET_PROTOCOL_REVISION >= 17)
|
||||
case PROP_CURRENT_COMMAND_PRIORITY:
|
||||
#endif
|
||||
#if (BACNET_PROTOCOL_REVISION >= 24)
|
||||
case PROP_COLOR_OVERRIDE:
|
||||
case PROP_COLOR_REFERENCE:
|
||||
case PROP_OVERRIDE_COLOR_REFERENCE:
|
||||
#endif
|
||||
case PROP_DESCRIPTION:
|
||||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||||
wp_data->error_code = ERROR_CODE_WRITE_ACCESS_DENIED;
|
||||
break;
|
||||
default:
|
||||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||||
wp_data->error_code = ERROR_CODE_UNKNOWN_PROPERTY;
|
||||
if (property_lists_member(
|
||||
Properties_Required, Properties_Optional,
|
||||
Properties_Proprietary, wp_data->object_property)) {
|
||||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||||
wp_data->error_code = ERROR_CODE_WRITE_ACCESS_DENIED;
|
||||
} else {
|
||||
wp_data->error_class = ERROR_CLASS_PROPERTY;
|
||||
wp_data->error_code = ERROR_CODE_UNKNOWN_PROPERTY;
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
|
||||
@@ -934,6 +934,13 @@ int Multistate_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
(rpdata->application_data_len == 0)) {
|
||||
return 0;
|
||||
}
|
||||
if (!property_lists_member(
|
||||
Properties_Required, Properties_Optional, Properties_Proprietary,
|
||||
rpdata->object_property)) {
|
||||
rpdata->error_class = ERROR_CLASS_PROPERTY;
|
||||
rpdata->error_code = ERROR_CODE_UNKNOWN_PROPERTY;
|
||||
return BACNET_STATUS_ERROR;
|
||||
}
|
||||
apdu = rpdata->application_data;
|
||||
apdu_size = rpdata->application_data_len;
|
||||
switch (rpdata->object_property) {
|
||||
@@ -1024,7 +1031,6 @@ int Multistate_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
apdu_len =
|
||||
encode_application_character_string(&apdu[0], &char_string);
|
||||
break;
|
||||
#if (BACNET_PROTOCOL_REVISION >= 17)
|
||||
case PROP_CURRENT_COMMAND_PRIORITY:
|
||||
i = Multistate_Output_Present_Value_Priority(
|
||||
rpdata->object_instance);
|
||||
@@ -1034,7 +1040,6 @@ int Multistate_Output_Read_Property(BACNET_READ_PROPERTY_DATA *rpdata)
|
||||
apdu_len = encode_application_null(&apdu[0]);
|
||||
}
|
||||
break;
|
||||
#endif
|
||||
default:
|
||||
rpdata->error_class = ERROR_CLASS_PROPERTY;
|
||||
rpdata->error_code = ERROR_CODE_UNKNOWN_PROPERTY;
|
||||
|
||||
@@ -0,0 +1,276 @@
|
||||
/**
|
||||
* @file
|
||||
* @brief BACnet Stack initialization and task handler
|
||||
* @author Steve Karg <skarg@users.sourceforge.net>
|
||||
* @date March 2024
|
||||
* @copyright SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
#include <stdint.h>
|
||||
#include <stdbool.h>
|
||||
/* BACnet Stack defines - first */
|
||||
#include "bacnet/bacdef.h"
|
||||
/* BACnet Stack core API */
|
||||
#include "bacnet/npdu.h"
|
||||
#include "bacnet/dcc.h"
|
||||
#include "bacnet/iam.h"
|
||||
/* BACnet Stack basic services */
|
||||
#include "bacnet/basic/sys/mstimer.h"
|
||||
#include "bacnet/basic/services.h"
|
||||
#include "bacnet/basic/tsm/tsm.h"
|
||||
#include "bacnet/datalink/datalink.h"
|
||||
/* BACnet Stack basic objects */
|
||||
#include "bacnet/basic/object/device.h"
|
||||
/* objects */
|
||||
#if (BACNET_PROTOCOL_REVISION >= 17)
|
||||
#include "bacnet/basic/object/netport.h"
|
||||
#endif
|
||||
#include "bacnet/basic/object/device.h"
|
||||
/* me */
|
||||
#include "bacnet/basic/server/bacnet_basic.h"
|
||||
#include "bacnet/basic/server/bacnet_port.h"
|
||||
|
||||
/* 1s timer for basic non-critical timed tasks */
|
||||
static struct mstimer BACnet_Task_Timer;
|
||||
/* task timer for object functionality */
|
||||
static struct mstimer BACnet_Object_Timer;
|
||||
/* uptimer for BACnet task */
|
||||
static unsigned long BACnet_Uptime_Seconds;
|
||||
/* packet counter for BACnet task */
|
||||
static unsigned long BACnet_Packet_Count;
|
||||
/* local Device ID to track changes */
|
||||
static uint32_t Device_ID = 0xFFFFFFFF;
|
||||
/* callbacks for custom features in BACnet thread */
|
||||
static bacnet_basic_callback BACnet_Init_Callback;
|
||||
static void *BACnet_Init_Context;
|
||||
static bacnet_basic_callback BACnet_Task_Callback;
|
||||
static void *BACnet_Task_Context;
|
||||
static bacnet_basic_store_callback BACnet_Store_Callback;
|
||||
|
||||
/**
|
||||
* @brief Set the callback for the BACnet initialization
|
||||
* @param callback [in] The callback function called after initialization
|
||||
* @param context [in] The context to pass to the callback function
|
||||
*/
|
||||
void bacnet_basic_init_callback_set(
|
||||
bacnet_basic_callback callback, void *context)
|
||||
{
|
||||
BACnet_Init_Callback = callback;
|
||||
BACnet_Init_Context = context;
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief BACnet Task Callback Handler
|
||||
*/
|
||||
static void bacnet_init_callback_handler(void)
|
||||
{
|
||||
if (BACnet_Init_Callback) {
|
||||
BACnet_Init_Callback(BACnet_Init_Context);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Set the callback for the BACnet Task
|
||||
* @param callback [in] The callback function to call during the task
|
||||
* @param context [in] The context to pass to the callback function
|
||||
*/
|
||||
void bacnet_basic_task_callback_set(
|
||||
bacnet_basic_callback callback, void *context)
|
||||
{
|
||||
BACnet_Task_Callback = callback;
|
||||
BACnet_Task_Context = context;
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief BACnet Task Callback Handler
|
||||
*/
|
||||
static void bacnet_task_callback_handler(void)
|
||||
{
|
||||
if (BACnet_Task_Callback) {
|
||||
BACnet_Task_Callback(BACnet_Task_Context);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Set the callback for the BACnet WriteProperty Store
|
||||
* @param callback [in] The callback function to call after a successful
|
||||
* WriteProperty with the data in BACnet binary encoded format (small!)
|
||||
*/
|
||||
void bacnet_basic_store_callback_set(bacnet_basic_store_callback callback)
|
||||
{
|
||||
BACnet_Store_Callback = callback;
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Store the BACnet data after a WriteProperty for object property
|
||||
* @param object_type - BACnet object type
|
||||
* @param object_instance - BACnet object instance
|
||||
* @param object_property - BACnet object property
|
||||
* @param array_index - BACnet array index
|
||||
* @param application_data - pointer to the data
|
||||
* @param application_data_len - length of the data
|
||||
*/
|
||||
static void bacnet_store_callback_handler(
|
||||
BACNET_OBJECT_TYPE object_type,
|
||||
uint32_t object_instance,
|
||||
BACNET_PROPERTY_ID object_property,
|
||||
BACNET_ARRAY_INDEX array_index,
|
||||
uint8_t *application_data,
|
||||
int application_data_len)
|
||||
{
|
||||
if (BACnet_Store_Callback) {
|
||||
BACnet_Store_Callback(
|
||||
object_type, object_instance, object_property, array_index,
|
||||
application_data, application_data_len);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Get the BACnet device uptime in seconds
|
||||
* @return The number of seconds the BACnet device has been running
|
||||
*/
|
||||
unsigned long bacnet_basic_uptime_seconds(void)
|
||||
{
|
||||
return BACnet_Uptime_Seconds;
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Get the BACnet device uptime in seconds
|
||||
* @return The number of seconds the BACnet device has been running
|
||||
*/
|
||||
unsigned long bacnet_basic_packet_count(void)
|
||||
{
|
||||
return BACnet_Packet_Count;
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Set the BACnet task device object timer interval
|
||||
* @param milliseconds [in] The number of milliseconds for the timer interval
|
||||
*/
|
||||
void bacnet_basic_task_object_timer_set(unsigned long milliseconds)
|
||||
{
|
||||
mstimer_set(&BACnet_Object_Timer, milliseconds);
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Store the BACnet data after a successful WriteProperty for
|
||||
* an object property
|
||||
* @param wp_data - pointer to the write property data
|
||||
*/
|
||||
bool bacnet_basic_write_property_store(BACNET_WRITE_PROPERTY_DATA *wp_data)
|
||||
{
|
||||
BACNET_ARRAY_INDEX array_index = BACNET_ARRAY_ALL;
|
||||
|
||||
if (property_list_bacnet_array_member(
|
||||
wp_data->object_type, wp_data->object_property)) {
|
||||
array_index = wp_data->array_index;
|
||||
} else if (wp_data->object_property == PROP_PRESENT_VALUE) {
|
||||
/* indirect Priority_Array write */
|
||||
if (Device_Objects_Property_List_Member(
|
||||
wp_data->object_type, wp_data->object_instance,
|
||||
PROP_PRIORITY_ARRAY)) {
|
||||
/* store the priority as an array index to be used on restore */
|
||||
array_index = wp_data->priority;
|
||||
}
|
||||
} else {
|
||||
array_index = wp_data->array_index;
|
||||
}
|
||||
bacnet_store_callback_handler(
|
||||
wp_data->object_type, wp_data->object_instance,
|
||||
wp_data->object_property, array_index, wp_data->application_data,
|
||||
wp_data->application_data_len);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Initialize the BACnet device object, the service handlers, and timers
|
||||
*/
|
||||
void bacnet_basic_init(void)
|
||||
{
|
||||
/* set up our confirmed service unrecognized service handler - required! */
|
||||
apdu_set_unrecognized_service_handler_handler(handler_unrecognized_service);
|
||||
/* we need to handle who-is to support dynamic device binding */
|
||||
apdu_set_unconfirmed_handler(SERVICE_UNCONFIRMED_WHO_IS, handler_who_is);
|
||||
apdu_set_unconfirmed_handler(SERVICE_UNCONFIRMED_WHO_HAS, handler_who_has);
|
||||
/* Set the handlers for any confirmed services that we support. */
|
||||
/* We must implement read property - it's required! */
|
||||
apdu_set_confirmed_handler(
|
||||
SERVICE_CONFIRMED_READ_PROPERTY, handler_read_property);
|
||||
apdu_set_confirmed_handler(
|
||||
SERVICE_CONFIRMED_READ_PROP_MULTIPLE, handler_read_property_multiple);
|
||||
apdu_set_confirmed_handler(
|
||||
SERVICE_CONFIRMED_WRITE_PROPERTY, handler_write_property);
|
||||
apdu_set_confirmed_handler(
|
||||
SERVICE_CONFIRMED_WRITE_PROP_MULTIPLE, handler_write_property_multiple);
|
||||
apdu_set_confirmed_handler(
|
||||
SERVICE_CONFIRMED_SUBSCRIBE_COV, handler_cov_subscribe);
|
||||
/* handle communication so we can shutup when asked, or restart */
|
||||
apdu_set_confirmed_handler(
|
||||
SERVICE_CONFIRMED_DEVICE_COMMUNICATION_CONTROL,
|
||||
handler_device_communication_control);
|
||||
apdu_set_confirmed_handler(
|
||||
SERVICE_CONFIRMED_REINITIALIZE_DEVICE, handler_reinitialize_device);
|
||||
/* start the 1 second timer for non-critical cyclic tasks */
|
||||
mstimer_set(&BACnet_Task_Timer, 1000L);
|
||||
/* start the timer for more time sensitive object specific cyclic tasks */
|
||||
if (mstimer_interval(&BACnet_Object_Timer) == 0) {
|
||||
mstimer_set(&BACnet_Object_Timer, 100UL);
|
||||
}
|
||||
Device_Write_Property_Store_Callback_Set(bacnet_basic_write_property_store);
|
||||
Device_Init(NULL);
|
||||
/* initialize user data in this thread */
|
||||
bacnet_init_callback_handler();
|
||||
}
|
||||
|
||||
/* local buffer for incoming PDUs to process */
|
||||
static uint8_t PDUBuffer[MAX_MPDU];
|
||||
|
||||
/**
|
||||
* @brief non-blocking BACnet task
|
||||
*/
|
||||
void bacnet_basic_task(void)
|
||||
{
|
||||
bool hello_world = false;
|
||||
uint16_t pdu_len = 0;
|
||||
BACNET_ADDRESS src = { 0 };
|
||||
uint32_t elapsed_milliseconds = 0;
|
||||
uint32_t elapsed_seconds = 0;
|
||||
|
||||
/* hello, World! */
|
||||
if (Device_ID != Device_Object_Instance_Number()) {
|
||||
Device_ID = Device_Object_Instance_Number();
|
||||
hello_world = true;
|
||||
}
|
||||
if (hello_world) {
|
||||
Send_I_Am(&Handler_Transmit_Buffer[0]);
|
||||
}
|
||||
/* handle non-time-critical cyclic tasks */
|
||||
if (mstimer_expired(&BACnet_Task_Timer)) {
|
||||
/* 1 second tasks */
|
||||
mstimer_reset(&BACnet_Task_Timer);
|
||||
/* presume that the elapsed time is the interval time */
|
||||
elapsed_milliseconds = mstimer_interval(&BACnet_Task_Timer);
|
||||
elapsed_seconds = elapsed_milliseconds / 1000;
|
||||
BACnet_Uptime_Seconds += elapsed_seconds;
|
||||
dcc_timer_seconds(elapsed_seconds);
|
||||
datalink_maintenance_timer(elapsed_seconds);
|
||||
handler_cov_timer_seconds(elapsed_seconds);
|
||||
}
|
||||
while (!handler_cov_fsm()) {
|
||||
/* waiting for COV processing to be IDLE */
|
||||
}
|
||||
/* object specific cyclic tasks */
|
||||
if (mstimer_expired(&BACnet_Object_Timer)) {
|
||||
elapsed_milliseconds = mstimer_elapsed(&BACnet_Object_Timer);
|
||||
mstimer_restart(&BACnet_Object_Timer);
|
||||
Device_Timer(elapsed_milliseconds);
|
||||
}
|
||||
/* handle the messaging */
|
||||
pdu_len = datalink_receive(&src, &PDUBuffer[0], sizeof(PDUBuffer), 0);
|
||||
if (pdu_len) {
|
||||
npdu_handler(&src, &PDUBuffer[0], pdu_len);
|
||||
BACnet_Packet_Count++;
|
||||
}
|
||||
/* call user task in this thread */
|
||||
bacnet_task_callback_handler();
|
||||
}
|
||||
@@ -0,0 +1,59 @@
|
||||
/**
|
||||
* @file
|
||||
* @brief BACnet Basic Stack initialization and basic task handler
|
||||
* @author Steve Karg <skarg@users.sourceforge.net>
|
||||
* @date April 2024
|
||||
* @copyright SPDX-License-Identifier: MIT
|
||||
*/
|
||||
#ifndef BACNET_BASIC_H
|
||||
#define BACNET_BASIC_H
|
||||
|
||||
#include <stdint.h>
|
||||
#include <stdbool.h>
|
||||
#include "bacnet/bacdef.h"
|
||||
|
||||
/**
|
||||
* @brief Callback function for BACnet initialization and task
|
||||
* @param context - pointer to the context
|
||||
*/
|
||||
typedef void (*bacnet_basic_callback)(void *context);
|
||||
|
||||
/**
|
||||
* @brief Store the BACnet data after a WriteProperty for object property
|
||||
* @param object_type - BACnet object type
|
||||
* @param object_instance - BACnet object instance
|
||||
* @param object_property - BACnet object property
|
||||
* @param array_index - BACnet array index
|
||||
* @param application_data - pointer to the data
|
||||
* @param application_data_len - length of the data
|
||||
*/
|
||||
typedef void (*bacnet_basic_store_callback)(
|
||||
BACNET_OBJECT_TYPE object_type,
|
||||
uint32_t object_instance,
|
||||
BACNET_PROPERTY_ID object_property,
|
||||
BACNET_ARRAY_INDEX array_index,
|
||||
uint8_t *application_data,
|
||||
int application_data_len);
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif /* __cplusplus */
|
||||
|
||||
void bacnet_basic_init(void);
|
||||
void bacnet_basic_init_callback_set(
|
||||
bacnet_basic_callback callback, void *context);
|
||||
|
||||
void bacnet_basic_task(void);
|
||||
void bacnet_basic_task_callback_set(
|
||||
bacnet_basic_callback callback, void *context);
|
||||
void bacnet_basic_task_object_timer_set(unsigned long milliseconds);
|
||||
|
||||
void bacnet_basic_store_callback_set(bacnet_basic_store_callback callback);
|
||||
|
||||
unsigned long bacnet_basic_uptime_seconds(void);
|
||||
unsigned long bacnet_basic_packet_count(void);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif /* __cplusplus */
|
||||
#endif
|
||||
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,66 @@
|
||||
/**
|
||||
* @file
|
||||
* @brief The BACnet/IPv4 datalink tasks for handling the device specific
|
||||
* data link layer
|
||||
* @author Steve Karg <skarg@users.sourceforge.net>
|
||||
* @date April 2024
|
||||
* @copyright SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
#include <stdint.h>
|
||||
#include <stdbool.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
/* BACnet definitions */
|
||||
#include "bacnet/bacdef.h"
|
||||
/* BACnet library API */
|
||||
#include "bacnet/basic/sys/mstimer.h"
|
||||
#include "bacnet/datalink/datalink.h"
|
||||
#include "bacnet/basic/object/netport.h"
|
||||
#if defined(BACDL_BIP)
|
||||
#include "bacnet/basic/server/bacnet_port_ipv4.h"
|
||||
#elif defined(BACDL_BIP6)
|
||||
#include "bacnet/basic/server/bacnet_port_ipv6.h"
|
||||
#endif
|
||||
/* me! */
|
||||
#include "bacnet/basic/server/bacnet_port.h"
|
||||
|
||||
/* timer used to renew Foreign Device Registration */
|
||||
static struct mstimer BACnet_Task_Timer;
|
||||
|
||||
/**
|
||||
* @brief Periodic tasks for the BACnet datalink layer
|
||||
*/
|
||||
void bacnet_port_task(void)
|
||||
{
|
||||
uint32_t elapsed_milliseconds = 0;
|
||||
uint32_t elapsed_seconds = 0;
|
||||
|
||||
if (mstimer_expired(&BACnet_Task_Timer)) {
|
||||
/* 1 second tasks */
|
||||
mstimer_reset(&BACnet_Task_Timer);
|
||||
/* presume that the elapsed time is the interval time */
|
||||
elapsed_milliseconds = mstimer_interval(&BACnet_Task_Timer);
|
||||
elapsed_seconds = elapsed_milliseconds / 1000;
|
||||
#if defined(BACDL_BIP)
|
||||
bacnet_port_ipv4_task(elapsed_seconds);
|
||||
#elif defined(BACDL_BIP6)
|
||||
bacnet_port_ipv6_task(elapsed_seconds);
|
||||
#endif
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Initialize the datalink network port
|
||||
*/
|
||||
bool bacnet_port_init(void)
|
||||
{
|
||||
bool status = false;
|
||||
/* start the 1 second timer for non-critical cyclic tasks */
|
||||
mstimer_set(&BACnet_Task_Timer, 1000L);
|
||||
#if defined(BACDL_BIP)
|
||||
status = bacnet_port_ipv4_init();
|
||||
#elif defined(BACDL_BIP6)
|
||||
status = bacnet_port_ipv6_init();
|
||||
#endif
|
||||
return status;
|
||||
}
|
||||
@@ -0,0 +1,27 @@
|
||||
/**
|
||||
* @file
|
||||
* @brief The BACnet datalink tasks for handling the device specific
|
||||
* data link network port layer
|
||||
* @author Steve Karg <skarg@users.sourceforge.net>
|
||||
* @date April 2024
|
||||
* @copyright SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
#ifndef BACNET_PORT_H
|
||||
#define BACNET_PORT_H
|
||||
|
||||
#include <stdint.h>
|
||||
#include <stdint.h>
|
||||
/* BACnet Stack defines - first */
|
||||
#include "bacnet/bacdef.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif /* __cplusplus */
|
||||
|
||||
bool bacnet_port_init(void);
|
||||
void bacnet_port_task(void);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif /* __cplusplus */
|
||||
#endif
|
||||
@@ -0,0 +1,101 @@
|
||||
/**
|
||||
* @file
|
||||
* @brief The BACnet/IPv4 datalink tasks for handling the device specific
|
||||
* data link layer
|
||||
* @author Steve Karg <skarg@users.sourceforge.net>
|
||||
* @date April 2024
|
||||
* @copyright SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
#include <stdint.h>
|
||||
#include <stdbool.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
/* BACnet definitions */
|
||||
#include "bacnet/bacdef.h"
|
||||
/* BACnet library API */
|
||||
#include "bacnet/basic/object/netport.h"
|
||||
#include "bacnet/basic/bbmd/h_bbmd.h"
|
||||
#include "bacnet/datalink/bip.h"
|
||||
#include "bacnet/datalink/bvlc.h"
|
||||
#include "bacnet/datalink/datalink.h"
|
||||
/* me! */
|
||||
#include "bacnet/basic/server/bacnet_port_ipv4.h"
|
||||
|
||||
#if defined(BACDL_BIP)
|
||||
|
||||
/* timer used to renew Foreign Device Registration */
|
||||
static uint16_t BBMD_Timer_Seconds;
|
||||
static uint16_t BBMD_TTL_Seconds = 60000;
|
||||
static BACNET_IP_ADDRESS BBMD_Address;
|
||||
|
||||
/**
|
||||
* @brief Initialize the datalink network port
|
||||
* @param ttl_seconds [in] The time-to-live in seconds for the Foreign Device
|
||||
* Registration
|
||||
* @param bbmd_address [in] The address of the BBMD
|
||||
*/
|
||||
void bacnet_port_ipv4_foreign_device_init(
|
||||
const uint16_t ttl_seconds, const BACNET_IP_ADDRESS *bbmd_address)
|
||||
{
|
||||
BBMD_TTL_Seconds = ttl_seconds;
|
||||
if (bbmd_address) {
|
||||
memcpy(&BBMD_Address, bbmd_address, sizeof(BACNET_IP_ADDRESS));
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Renew the Foreign Device Registration
|
||||
*/
|
||||
void bacnet_port_ipv4_task(uint16_t elapsed_seconds)
|
||||
{
|
||||
if (BBMD_Timer_Seconds) {
|
||||
if (BBMD_Timer_Seconds <= elapsed_seconds) {
|
||||
BBMD_Timer_Seconds = 0;
|
||||
} else {
|
||||
BBMD_Timer_Seconds -= elapsed_seconds;
|
||||
}
|
||||
if (BBMD_Timer_Seconds == 0) {
|
||||
if (BBMD_Address.port > 0) {
|
||||
(void)bvlc_register_with_bbmd(&BBMD_Address, BBMD_TTL_Seconds);
|
||||
}
|
||||
BBMD_Timer_Seconds = (uint16_t)BBMD_TTL_Seconds;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Initialize the network port object.
|
||||
*/
|
||||
bool bacnet_port_ipv4_init(void)
|
||||
{
|
||||
const uint32_t instance = 1;
|
||||
BACNET_IP_ADDRESS addr = { 0 };
|
||||
uint8_t prefix = 0;
|
||||
|
||||
if (!bip_init(NULL)) {
|
||||
return false;
|
||||
}
|
||||
Network_Port_Object_Instance_Number_Set(0, instance);
|
||||
Network_Port_Name_Set(instance, "BACnet/IP Port");
|
||||
Network_Port_Type_Set(instance, PORT_TYPE_BIP);
|
||||
bip_get_addr(&addr);
|
||||
prefix = bip_get_subnet_prefix();
|
||||
Network_Port_BIP_Port_Set(instance, addr.port);
|
||||
Network_Port_IP_Address_Set(
|
||||
instance, addr.address[0], addr.address[1], addr.address[2],
|
||||
addr.address[3]);
|
||||
Network_Port_IP_Subnet_Prefix_Set(instance, prefix);
|
||||
Network_Port_Link_Speed_Set(instance, 0.0);
|
||||
/* common NP data */
|
||||
Network_Port_Reliability_Set(instance, RELIABILITY_NO_FAULT_DETECTED);
|
||||
Network_Port_Out_Of_Service_Set(instance, false);
|
||||
Network_Port_Quality_Set(instance, PORT_QUALITY_UNKNOWN);
|
||||
Network_Port_APDU_Length_Set(instance, MAX_APDU);
|
||||
Network_Port_Network_Number_Set(instance, 0);
|
||||
/* last thing - clear pending changes - we don't want to set these
|
||||
since they are already set */
|
||||
Network_Port_Changes_Pending_Set(instance, false);
|
||||
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
@@ -0,0 +1,30 @@
|
||||
/**
|
||||
* @file
|
||||
* @brief API for the BACnet datalink tasks for handling the device specific
|
||||
* data link network port layer
|
||||
* @author Steve Karg <skarg@users.sourceforge.net>
|
||||
* @date April 2024
|
||||
* @copyright SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
#ifndef BACNET_PORT_IPV4_H
|
||||
#define BACNET_PORT_IPV4_H
|
||||
|
||||
#include <stdbool.h>
|
||||
#include <stdint.h>
|
||||
/* BACnet Stack defines - first */
|
||||
#include "bacnet/bacdef.h"
|
||||
#include "bacnet/datalink/bvlc.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif /* __cplusplus */
|
||||
|
||||
void bacnet_port_ipv4_foreign_device_init(
|
||||
const uint16_t ttl_seconds, const BACNET_IP_ADDRESS *bbmd_address);
|
||||
void bacnet_port_ipv4_task(uint16_t elapsed_seconds);
|
||||
bool bacnet_port_ipv4_init(void);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif /* __cplusplus */
|
||||
#endif
|
||||
@@ -0,0 +1,104 @@
|
||||
/**
|
||||
* @file
|
||||
* @brief The BACnet datalink tasks for handling the device specific
|
||||
* data link layer
|
||||
* @author Steve Karg <skarg@users.sourceforge.net>
|
||||
* @date April 2024
|
||||
* @copyright SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
#include <stdint.h>
|
||||
#include <stdbool.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
/* BACnet definitions */
|
||||
#include "bacnet/bacdef.h"
|
||||
/* BACnet library API */
|
||||
#include "bacnet/basic/object/netport.h"
|
||||
#include "bacnet/basic/bbmd6/h_bbmd6.h"
|
||||
#include "bacnet/datalink/bip6.h"
|
||||
#include "bacnet/datalink/bvlc6.h"
|
||||
#include "bacnet/datalink/datalink.h"
|
||||
/* me! */
|
||||
#include "bacnet/basic/server/bacnet_port_ipv6.h"
|
||||
|
||||
#if defined(BACDL_BIP6)
|
||||
|
||||
/* timer used to renew Foreign Device Registration */
|
||||
static uint16_t BBMD_Timer_Seconds;
|
||||
static uint16_t BBMD_TTL_Seconds = 60000;
|
||||
static BACNET_IP6_ADDRESS BBMD_Address;
|
||||
|
||||
/**
|
||||
* @brief Initialize the datalink network port
|
||||
* @param ttl_seconds [in] The time-to-live in seconds for the Foreign Device
|
||||
* Registration
|
||||
* @param bbmd_address [in] The address of the BBMD
|
||||
*/
|
||||
void bacnet_port_ipv6_foreign_device_init(
|
||||
const uint16_t ttl_seconds, const BACNET_IP6_ADDRESS *bbmd_address)
|
||||
{
|
||||
BBMD_TTL_Seconds = ttl_seconds;
|
||||
if (bbmd_address) {
|
||||
memcpy(&BBMD_Address, bbmd_address, sizeof(BACNET_IP6_ADDRESS));
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Renew the Foreign Device Registration
|
||||
*/
|
||||
void bacnet_port_ipv6_task(uint16_t elapsed_seconds)
|
||||
{
|
||||
if (BBMD_Timer_Seconds) {
|
||||
if (BBMD_Timer_Seconds <= elapsed_seconds) {
|
||||
BBMD_Timer_Seconds = 0;
|
||||
} else {
|
||||
BBMD_Timer_Seconds -= elapsed_seconds;
|
||||
}
|
||||
if (BBMD_Timer_Seconds == 0) {
|
||||
if (BBMD_Address.port > 0) {
|
||||
(void)bvlc6_register_with_bbmd(&BBMD_Address, BBMD_TTL_Seconds);
|
||||
}
|
||||
BBMD_Timer_Seconds = BBMD_TTL_Seconds;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Initialize the network port object.
|
||||
* @return true if successful
|
||||
*/
|
||||
bool bacnet_port_ipv6_init(void)
|
||||
{
|
||||
uint32_t instance = 1;
|
||||
uint8_t prefix = 0;
|
||||
BACNET_ADDRESS addr = { 0 };
|
||||
BACNET_IP6_ADDRESS addr6 = { 0 };
|
||||
|
||||
if (!bip6_init(NULL)) {
|
||||
return false;
|
||||
}
|
||||
Network_Port_Object_Instance_Number_Set(0, instance);
|
||||
Network_Port_Name_Set(instance, "BACnet/IPv6 Port");
|
||||
Network_Port_Type_Set(instance, PORT_TYPE_BIP6);
|
||||
Network_Port_BIP6_Port_Set(instance, bip6_get_port());
|
||||
bip6_get_my_address(&addr);
|
||||
Network_Port_MAC_Address_Set(instance, &addr.mac[0], addr.mac_len);
|
||||
bip6_get_addr(&addr6);
|
||||
Network_Port_IPv6_Address_Set(instance, &addr6.address[0]);
|
||||
bip6_get_broadcast_addr(&addr6);
|
||||
Network_Port_IPv6_Multicast_Address_Set(instance, &addr6.address[0]);
|
||||
Network_Port_IPv6_Subnet_Prefix_Set(instance, prefix);
|
||||
|
||||
Network_Port_Reliability_Set(instance, RELIABILITY_NO_FAULT_DETECTED);
|
||||
Network_Port_Link_Speed_Set(instance, 0.0);
|
||||
Network_Port_Out_Of_Service_Set(instance, false);
|
||||
Network_Port_Quality_Set(instance, PORT_QUALITY_UNKNOWN);
|
||||
Network_Port_APDU_Length_Set(instance, MAX_APDU);
|
||||
Network_Port_Network_Number_Set(instance, 0);
|
||||
/* last thing - clear pending changes - we don't want to set these
|
||||
since they are already set */
|
||||
Network_Port_Changes_Pending_Set(instance, false);
|
||||
|
||||
return true;
|
||||
}
|
||||
#endif
|
||||
@@ -0,0 +1,30 @@
|
||||
/**
|
||||
* @file
|
||||
* @brief The BACnet/IPv6 datalink tasks for handling the device specific
|
||||
* data link network port layer
|
||||
* @author Steve Karg <skarg@users.sourceforge.net>
|
||||
* @date April 2024
|
||||
* @copyright SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
#ifndef BACNET_PORT_IPV6_H
|
||||
#define BACNET_PORT_IPV6_H
|
||||
|
||||
#include <stdbool.h>
|
||||
#include <stdint.h>
|
||||
/* BACnet Stack defines - first */
|
||||
#include "bacnet/bacdef.h"
|
||||
#include "bacnet/datalink/bvlc6.h"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif /* __cplusplus */
|
||||
|
||||
void bacnet_port_ipv6_foreign_device_init(
|
||||
const uint16_t ttl_seconds, const BACNET_IP6_ADDRESS *bbmd_address);
|
||||
void bacnet_port_ipv6_task(uint16_t elapsed_seconds);
|
||||
bool bacnet_port_ipv6_init(void);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif /* __cplusplus */
|
||||
#endif
|
||||
Reference in New Issue
Block a user