Bugfix/lighting-output-internal-process-api (#1086)

* Fixed Lighting_Command to ignore write priority and use its own. 

* Fixed BACnetLightingOperation reserved range.

* Refactor overridden into the lighting command module. Added Overridden status flags API.  Added Lighting Output API to implement a momentary override to the output that is cleared at the next lighting command.  Integrated lighting command overridden behavior into the lighting output object.

* Added Trim_Fade_Time, High_End_Trim, Low_End_Trim, Last_On_Value and Default_On_Value properties to lighting output.
Added TRIM_ACTIVE flag to lighting command. Added Last_On_Value and Default_On_Value to lighting command for restore and toggle.
This commit is contained in:
Steve Karg
2025-09-04 15:04:42 -05:00
committed by GitHub
parent 8a9c808b76
commit 71c03b29e8
10 changed files with 1854 additions and 514 deletions
@@ -23,6 +23,7 @@ set(ZTST_DIR "${TST_DIR}/ztest/src")
add_compile_definitions(
BIG_ENDIAN=0
CONFIG_ZTEST=1
BACNET_PROTOCOL_REVISION=28
)
include_directories(
+327 -9
View File
@@ -1,15 +1,12 @@
/*
* Copyright (c) 2020 Legrand North America, LLC.
*
/* @file
* @brief test BACnet Lighting Output object APIs
* @date 2013
* @author Steve Karg <skarg@users.sourceforge.net>
* @copyright SPDX-License-Identifier: MIT
*/
/* @file
* @brief test BACnet integer encode/decode APIs
*/
#include <zephyr/ztest.h>
#include <bacnet/bactext.h>
#include <bacnet/proplist.h>
#include <bacnet/basic/object/lo.h>
/**
@@ -17,6 +14,33 @@
* @{
*/
/**
* @brief compare two floating point values to 3 decimal places
*
* @param x1 - first comparison value
* @param x2 - second comparison value
* @return true if the value is the same to 3 decimal points
*/
static bool is_float_equal(float x1, float x2)
{
return fabs(x1 - x2) < 0.001;
}
static float Test_Tracking_Value;
/**
* @brief Callback for tracking value updates
* @param key - key used to link to specific light
* @param old_value - value prior to write
* @param value - value of the write
*/
static void lighting_command_tracking_value_observer(
uint32_t key, float old_value, float value)
{
(void)key;
(void)old_value;
Test_Tracking_Value = value;
}
/**
* @brief Test
*/
@@ -36,17 +60,30 @@ static void testLightingOutput(void)
const uint32_t instance = 123;
BACNET_WRITE_PROPERTY_DATA wpdata = { 0 };
bool status = false;
unsigned index;
unsigned index, count;
uint16_t milliseconds = 10;
uint32_t test_instance;
const char *test_name = NULL;
char *sample_name = "sample";
BACNET_LIGHTING_COMMAND lighting_command = { 0 };
BACNET_LIGHTING_IN_PROGRESS in_progress;
float real_value, test_real;
uint32_t unsigned_value, test_unsigned;
unsigned priority, test_priority;
BACNET_OBJECT_ID object_id, test_object_id;
Lighting_Output_Init();
Lighting_Output_Create(instance);
status = Lighting_Output_Valid_Instance(instance);
zassert_true(status, NULL);
status = Lighting_Output_Valid_Instance(BACNET_MAX_INSTANCE);
zassert_false(status, NULL);
index = Lighting_Output_Instance_To_Index(instance);
zassert_equal(index, 0, NULL);
count = Lighting_Output_Count();
zassert_equal(count, 1, NULL);
test_instance = Lighting_Output_Index_To_Instance(0);
zassert_equal(test_instance, instance, NULL);
rpdata.application_data = &apdu[0];
rpdata.application_data_len = sizeof(apdu);
@@ -88,6 +125,27 @@ static void testLightingOutput(void)
"property '%s': WriteProperty Unknown!\n",
bactext_property_name(rpdata.object_property));
}
if (property_list_commandable_member(
wpdata.object_type, wpdata.object_property)) {
wpdata.priority = 16;
status = Lighting_Output_Write_Property(&wpdata);
zassert_true(status, NULL);
wpdata.application_data_len =
encode_application_null(wpdata.application_data);
wpdata.priority = 16;
status = Lighting_Output_Write_Property(&wpdata);
zassert_true(status, NULL);
wpdata.priority = 6;
status = Lighting_Output_Write_Property(&wpdata);
zassert_false(status, NULL);
zassert_equal(
wpdata.error_code, ERROR_CODE_WRITE_ACCESS_DENIED, NULL);
wpdata.priority = 0;
status = Lighting_Output_Write_Property(&wpdata);
zassert_false(status, NULL);
zassert_equal(
wpdata.error_code, ERROR_CODE_VALUE_OUT_OF_RANGE, NULL);
}
}
pRequired++;
}
@@ -129,6 +187,7 @@ static void testLightingOutput(void)
rpdata.object_property = PROP_ALL;
len = Lighting_Output_Read_Property(&rpdata);
zassert_equal(len, BACNET_STATUS_ERROR, NULL);
wpdata.object_property = PROP_ALL;
status = Lighting_Output_Write_Property(&wpdata);
zassert_false(status, NULL);
/* check the dimming/ramping/stepping engine*/
@@ -142,9 +201,268 @@ static void testLightingOutput(void)
zassert_true(status, NULL);
test_name = Lighting_Output_Name_ASCII(instance);
zassert_equal(test_name, NULL, NULL);
/* test the ASCII description get/set */
status = Lighting_Output_Description_Set(instance, sample_name);
zassert_true(status, NULL);
test_name = Lighting_Output_Description(instance);
zassert_equal(test_name, sample_name, NULL);
/* test local control API */
lighting_command.operation = BACNET_LIGHTS_NONE;
do {
status =
Lighting_Output_Lighting_Command_Set(instance, &lighting_command);
if (status) {
zassert_true(status, NULL);
} else {
printf(
"lighting-command operation[%d] %s not supported.\n",
lighting_command.operation,
bactext_lighting_operation_name(lighting_command.operation));
}
if (lighting_command.operation == BACNET_LIGHTS_PROPRIETARY_MIN) {
/* skip to make testing faster */
lighting_command.operation = BACNET_LIGHTS_PROPRIETARY_MAX;
} else if (lighting_command.operation == BACNET_LIGHTS_RESERVED_MIN) {
/* skip to make testing faster */
lighting_command.operation = BACNET_LIGHTS_RESERVED_MAX;
} else {
lighting_command.operation++;
}
} while (lighting_command.operation <= BACNET_LIGHTS_PROPRIETARY_MAX);
/* test the present-value APIs */
Lighting_Output_Present_Value_Relinquish_All(instance);
real_value = 1.0;
priority = 8;
Lighting_Output_Present_Value_Set(instance, real_value, priority);
test_real = Lighting_Output_Present_Value(instance);
zassert_true(is_float_equal(test_real, real_value), NULL);
test_priority = Lighting_Output_Present_Value_Priority(instance);
zassert_equal(
priority, test_priority, "priority=%u test_priority=%u", priority,
test_priority);
Lighting_Output_Present_Value_Relinquish(instance, priority);
real_value = Lighting_Output_Relinquish_Default(instance);
test_real = Lighting_Output_Present_Value(instance);
zassert_true(is_float_equal(test_real, real_value), NULL);
/* test the present-value special values */
status = Lighting_Output_Default_Fade_Time_Set(instance, 100);
zassert_true(status, NULL);
status = Lighting_Output_Default_Ramp_Rate_Set(instance, 100.0f);
zassert_true(status, NULL);
status = Lighting_Output_Egress_Time_Set(instance, 0);
zassert_true(status, NULL);
status = Lighting_Output_Default_Step_Increment_Set(instance, 1.0f);
zassert_true(status, NULL);
status = Lighting_Output_Transition_Set(
instance, BACNET_LIGHTING_TRANSITION_NONE);
zassert_true(status, NULL);
/* WARN - start with lights on */
real_value = -1.0;
priority = 8;
status = Lighting_Output_Present_Value_Set(instance, 100.0f, priority);
zassert_true(status, NULL);
Lighting_Output_Timer(instance, 10);
in_progress = Lighting_Output_In_Progress(instance);
zassert_equal(
in_progress, BACNET_LIGHTING_IDLE, "in_progress=%s",
bactext_lighting_in_progress(in_progress));
status = Lighting_Output_Present_Value_Set(instance, real_value, priority);
zassert_true(status, NULL);
in_progress = Lighting_Output_In_Progress(instance);
zassert_equal(
in_progress, BACNET_LIGHTING_OTHER, "in_progress=%s",
bactext_lighting_in_progress(in_progress));
Lighting_Output_Timer(instance, 10);
in_progress = Lighting_Output_In_Progress(instance);
zassert_equal(
in_progress, BACNET_LIGHTING_IDLE, "in_progress=%s",
bactext_lighting_in_progress(in_progress));
/* WARN RELINQUISH */
real_value = -2.0;
priority = 8;
Lighting_Output_Present_Value_Set(instance, real_value, priority);
Lighting_Output_Timer(instance, 10);
in_progress = Lighting_Output_In_Progress(instance);
zassert_equal(
in_progress, BACNET_LIGHTING_IDLE, "in_progress=%s",
bactext_lighting_in_progress(in_progress));
real_value = -3.0;
priority = 8;
Lighting_Output_Present_Value_Set(instance, real_value, priority);
Lighting_Output_Timer(instance, 10);
in_progress = Lighting_Output_In_Progress(instance);
zassert_equal(
in_progress, BACNET_LIGHTING_IDLE, "in_progress=%s",
bactext_lighting_in_progress(in_progress));
real_value = -4.0;
priority = 8;
Lighting_Output_Present_Value_Set(instance, real_value, priority);
Lighting_Output_Timer(instance, 10);
in_progress = Lighting_Output_In_Progress(instance);
zassert_equal(
in_progress, BACNET_LIGHTING_IDLE, "in_progress=%s",
bactext_lighting_in_progress(in_progress));
real_value = -5.0;
priority = 8;
Lighting_Output_Present_Value_Set(instance, real_value, priority);
Lighting_Output_Timer(instance, 10);
in_progress = Lighting_Output_In_Progress(instance);
zassert_equal(
in_progress, BACNET_LIGHTING_IDLE, "in_progress=%s",
bactext_lighting_in_progress(in_progress));
real_value = -6.0;
priority = 8;
Lighting_Output_Present_Value_Set(instance, real_value, priority);
Lighting_Output_Timer(instance, 10);
in_progress = Lighting_Output_In_Progress(instance);
zassert_equal(
in_progress, BACNET_LIGHTING_IDLE, "in_progress=%s",
bactext_lighting_in_progress(in_progress));
real_value = -7.0;
priority = 8;
Lighting_Output_Present_Value_Set(instance, real_value, priority);
Lighting_Output_Timer(instance, 10);
in_progress = Lighting_Output_In_Progress(instance);
zassert_equal(
in_progress, BACNET_LIGHTING_IDLE, "in_progress=%s",
bactext_lighting_in_progress(in_progress));
status = Lighting_Output_In_Progress_Set(
instance, BACNET_LIGHTING_NOT_CONTROLLED);
in_progress = Lighting_Output_In_Progress(instance);
zassert_equal(
in_progress, BACNET_LIGHTING_NOT_CONTROLLED, "in_progress=%s",
bactext_lighting_in_progress(in_progress));
/* tracking-value */
real_value = 100.0f;
status = Lighting_Output_Tracking_Value_Set(instance, real_value);
zassert_true(status, NULL);
test_real = Lighting_Output_Tracking_Value(instance);
zassert_true(is_float_equal(test_real, real_value), NULL);
/* blink-warn features */
status = Lighting_Output_Blink_Warn_Feature_Set(instance, 0.0f, 0, 65535);
zassert_true(status, NULL);
status = Lighting_Output_Blink_Warn_Feature_Set(instance, -1.0f, 0, 65535);
zassert_true(status, NULL);
status = Lighting_Output_Blink_Warn_Feature_Set(instance, 101.0f, 0, 65535);
zassert_true(status, NULL);
/* egress-time */
unsigned_value = 5 * 60;
status = Lighting_Output_Egress_Time_Set(instance, unsigned_value);
zassert_true(status, NULL);
test_unsigned = Lighting_Output_Egress_Time(instance);
zassert_equal(unsigned_value, test_unsigned, NULL);
/* default-fade-time */
unsigned_value = 5 * 60 * 1000;
status = Lighting_Output_Default_Fade_Time_Set(instance, unsigned_value);
zassert_true(status, NULL);
test_unsigned = Lighting_Output_Default_Fade_Time(instance);
zassert_equal(unsigned_value, test_unsigned, NULL);
/* default-ramp-rate */
real_value = 1.0f;
status = Lighting_Output_Default_Ramp_Rate_Set(instance, real_value);
zassert_true(status, NULL);
test_real = Lighting_Output_Default_Ramp_Rate(instance);
zassert_true(is_float_equal(test_real, real_value), NULL);
/* default-step-increment */
real_value = 2.0f;
status = Lighting_Output_Default_Step_Increment_Set(instance, real_value);
zassert_true(status, NULL);
test_real = Lighting_Output_Default_Step_Increment(instance);
zassert_true(is_float_equal(test_real, real_value), NULL);
/* relinquish-default */
real_value = 0.0f;
status = Lighting_Output_Relinquish_Default_Set(instance, real_value);
zassert_true(status, NULL);
test_real = Lighting_Output_Relinquish_Default(instance);
zassert_true(is_float_equal(test_real, real_value), NULL);
/* overridden */
real_value = 88.0f;
status = Lighting_Output_Overridden_Set(instance, real_value);
zassert_true(status, NULL);
status = Lighting_Output_Overridden_Status(instance);
zassert_true(status, NULL);
test_real = Lighting_Output_Tracking_Value(instance);
zassert_true(
is_float_equal(test_real, real_value), "value=%f test_value=%f",
real_value, test_real);
status = Lighting_Output_Present_Value_Set(instance, 99.0f, priority);
zassert_true(status, NULL);
Lighting_Output_Timer(instance, 10);
test_real = Lighting_Output_Tracking_Value(instance);
zassert_true(
is_float_equal(test_real, real_value), "value=%f test_value=%f",
real_value, test_real);
real_value = Lighting_Output_Present_Value(instance);
status = Lighting_Output_Overridden_Clear(instance);
zassert_true(status, NULL);
Lighting_Output_Timer(instance, 10);
test_real = Lighting_Output_Tracking_Value(instance);
zassert_true(is_float_equal(test_real, real_value), NULL);
/* overridden-momentary */
real_value = 77.0f;
status = Lighting_Output_Overridden_Momentary(instance, real_value);
zassert_true(status, NULL);
status = Lighting_Output_Overridden_Status(instance);
zassert_true(status, NULL);
test_real = Lighting_Output_Tracking_Value(instance);
zassert_true(
is_float_equal(test_real, real_value), "value=%f test_value=%f",
real_value, test_real);
status = Lighting_Output_Present_Value_Set(instance, 98.0f, priority);
Lighting_Output_Timer(instance, 10);
real_value = Lighting_Output_Present_Value(instance);
test_real = Lighting_Output_Tracking_Value(instance);
zassert_true(
is_float_equal(test_real, real_value), "value=%f test_value=%f",
real_value, test_real);
/* color-override */
status = Lighting_Output_Color_Override_Set(instance, true);
zassert_true(status, NULL);
status = Lighting_Output_Color_Override(instance);
zassert_true(status, NULL);
status = Lighting_Output_Color_Override_Set(instance, false);
zassert_true(status, NULL);
status = Lighting_Output_Color_Override(instance);
zassert_false(status, NULL);
/* color-reference */
object_id.instance = 1;
object_id.type = OBJECT_COLOR;
status = Lighting_Output_Color_Reference_Set(instance, &object_id);
zassert_true(status, NULL);
status = Lighting_Output_Color_Reference(instance, &test_object_id);
zassert_true(status, NULL);
zassert_equal(object_id.instance, test_object_id.instance, NULL);
zassert_equal(object_id.type, test_object_id.type, NULL);
/* override-color-reference */
object_id.instance = 2;
object_id.type = OBJECT_COLOR;
status = Lighting_Output_Override_Color_Reference_Set(instance, &object_id);
zassert_true(status, NULL);
status =
Lighting_Output_Override_Color_Reference(instance, &test_object_id);
zassert_true(status, NULL);
zassert_equal(object_id.instance, test_object_id.instance, NULL);
zassert_equal(object_id.type, test_object_id.type, NULL);
/* tracking-value observer */
real_value = 95.0f;
Lighting_Output_Write_Present_Value_Callback_Set(
lighting_command_tracking_value_observer);
status = Lighting_Output_Present_Value_Set(instance, real_value, priority);
zassert_true(status, NULL);
Lighting_Output_Timer(instance, 10);
test_real = Lighting_Output_Tracking_Value(instance);
zassert_true(is_float_equal(test_real, real_value), NULL);
zassert_true(is_float_equal(Test_Tracking_Value, real_value), NULL);
/* out-of-bounds */
test_instance = Lighting_Output_Create(BACNET_MAX_INSTANCE + 1);
zassert_equal(test_instance, BACNET_MAX_INSTANCE, NULL);
/* check the delete function */
status = Lighting_Output_Delete(instance);
zassert_true(status, NULL);
test_instance = Lighting_Output_Create(BACNET_MAX_INSTANCE);
zassert_equal(test_instance, 1, NULL);
Lighting_Output_Cleanup();
return;
}
+204 -12
View File
@@ -1,11 +1,7 @@
/* @file
* @brief test BACnet integer encode/decode APIs
* @brief test BACnet Lighting Command encode/decode APIs
* @date June 2022
* @brief tests sRGB to and from from CIE xy and brightness API
*
* @section LICENSE
* Copyright (c) 2022 Steve Karg <skarg@users.sourceforge.net>
*
* @author Steve Karg <skarg@users.sourceforge.net>
* @copyright SPDX-License-Identifier: MIT
*/
#include <stdint.h>
@@ -20,6 +16,7 @@
*/
static float Tracking_Value;
static uint16_t Tracking_Elapsed_Milliseconds;
/**
* @brief Callback for tracking value updates
@@ -33,6 +30,25 @@ static void dimmer_tracking_value(uint32_t key, float old_value, float value)
Tracking_Value = value;
}
/**
* @brief Callback for non-standard Lighting_Operation timer values
* @param data - Lighting Command data structure
* @param milliseconds - elapsed time in milliseconds
*/
static void dimmer_timer_task(
struct bacnet_lighting_command_data *data, uint16_t milliseconds)
{
Tracking_Elapsed_Milliseconds = milliseconds;
switch (data->Lighting_Operation) {
case BACNET_LIGHTS_PROPRIETARY_MIN:
break;
case BACNET_LIGHTS_PROPRIETARY_MAX:
break;
default:
break;
}
}
/**
* @brief compare two floating point values to 3 decimal places
*
@@ -127,20 +143,56 @@ static void test_lighting_command_command_unit(void)
#endif
{
BACNET_LIGHTING_COMMAND_DATA data = { 0 };
struct lighting_command_notification observer1 = { 0 };
struct lighting_command_notification observer2 = { 0 };
struct lighting_command_timer_notification timer_observer1 = { 0 };
struct lighting_command_timer_notification timer_observer2 = { 0 };
uint16_t milliseconds = 10;
uint32_t fade_time = 1000;
float target_level = 100.0f;
float target_step = 1.0f;
float ramp_rate = 1000.0f;
float override_level;
lighting_command_init(&data);
data.Notification_Head.callback = dimmer_tracking_value;
/* lighting command subscribe */
observer1.callback = dimmer_tracking_value;
lighting_command_notification_add(&data, &observer1);
/* add again to verify skipping */
lighting_command_notification_add(&data, &observer1);
/* add second tracker */
lighting_command_notification_add(&data, &observer2);
/* lighting command timer subscribe */
timer_observer1.callback = dimmer_timer_task;
lighting_command_timer_notfication_add(&data, &timer_observer1);
/* add again to verify skipping */
lighting_command_timer_notfication_add(&data, &timer_observer1);
/* add second tracker */
lighting_command_timer_notfication_add(&data, &timer_observer2);
/* basic STOP and NONE states */
lighting_command_stop(&data);
lighting_command_timer(&data, milliseconds);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
lighting_command_none(&data);
lighting_command_timer(&data, milliseconds);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
/* normalized range clamp testing */
data.Max_Actual_Value = 95.0f;
data.Min_Actual_Value = 5.0f;
target_level = lighting_command_normalized_range_clamp(&data, 0.1f);
zassert_true(is_float_equal(target_level, 0.0f), NULL);
target_level = lighting_command_normalized_range_clamp(&data, 100.0f);
zassert_true(is_float_equal(target_level, data.Max_Actual_Value), NULL);
target_level = lighting_command_normalized_range_clamp(&data, 1.0f);
zassert_true(is_float_equal(target_level, data.Min_Actual_Value), NULL);
data.Max_Actual_Value = 100.0f;
data.Min_Actual_Value = 1.0f;
/* fade up */
target_level = 100.0f;
lighting_command_fade_to(&data, 100.0f, fade_time);
milliseconds = fade_time / 2;
lighting_command_timer(&data, milliseconds);
@@ -169,7 +221,7 @@ static void test_lighting_command_command_unit(void)
milliseconds = 10;
lighting_command_fade_to(&data, target_level, 0);
lighting_command_timer(&data, milliseconds);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
zassert_true(data.In_Progress == BACNET_LIGHTING_TRIM_ACTIVE, NULL);
zassert_true(is_float_equal(Tracking_Value, data.Low_Trim_Value), NULL);
target_level = 0.0f;
milliseconds = 10;
@@ -184,9 +236,52 @@ static void test_lighting_command_command_unit(void)
milliseconds = 10;
lighting_command_fade_to(&data, target_level, 0);
lighting_command_timer(&data, milliseconds);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
zassert_true(data.In_Progress == BACNET_LIGHTING_TRIM_ACTIVE, NULL);
zassert_true(is_float_equal(Tracking_Value, data.High_Trim_Value), NULL);
data.High_Trim_Value = data.Max_Actual_Value;
/* override */
override_level = 42.0f;
target_level = 100.0f;
milliseconds = 10;
data.Overridden = true;
data.Overridden_Momentary = false;
lighting_command_override(&data, override_level);
lighting_command_timer(&data, milliseconds);
zassert_true(is_float_equal(Tracking_Value, override_level), NULL);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
lighting_command_fade_to(&data, target_level, 0);
lighting_command_timer(&data, milliseconds);
zassert_true(is_float_equal(Tracking_Value, override_level), NULL);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
data.Overridden = false;
lighting_command_override(&data, target_level);
lighting_command_timer(&data, milliseconds);
zassert_true(is_float_equal(Tracking_Value, target_level), NULL);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
/* momentary override - self clearing flags */
override_level = 42.0f;
target_level = 100.0f;
milliseconds = 10;
data.Overridden = true;
data.Overridden_Momentary = true;
lighting_command_override(&data, override_level);
lighting_command_timer(&data, milliseconds);
zassert_true(is_float_equal(Tracking_Value, override_level), NULL);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
zassert_false(data.Overridden, NULL);
zassert_true(data.Overridden_Momentary, NULL);
lighting_command_fade_to(&data, target_level, 0);
lighting_command_timer(&data, milliseconds);
zassert_true(is_float_equal(Tracking_Value, target_level), NULL);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
zassert_false(data.Overridden, NULL);
zassert_false(data.Overridden_Momentary, NULL);
/* step clamping */
target_step = lighting_command_step_increment_clamp(0.0f);
zassert_true(is_float_equal(target_step, 0.1f), NULL);
target_step = lighting_command_step_increment_clamp(100.1f);
zassert_true(is_float_equal(target_step, 100.0f), NULL);
/* step UP - inhibit ON */
target_step = 1.0f;
@@ -303,7 +398,8 @@ static void test_lighting_command_command_unit(void)
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
zassert_true(is_float_equal(Tracking_Value, target_level), NULL);
target_level = 100.0f;
lighting_command_ramp_to(&data, target_level, 100.0f);
ramp_rate = 100.0f;
lighting_command_ramp_to(&data, target_level, ramp_rate);
lighting_command_timer(&data, milliseconds);
zassert_true(
data.In_Progress == BACNET_LIGHTING_RAMP_ACTIVE, "In_Progress=%d",
@@ -316,8 +412,9 @@ static void test_lighting_command_command_unit(void)
/* slower ramp up */
target_level = 100.0f;
milliseconds = 100;
ramp_rate = 1.0f;
do {
lighting_command_ramp_to(&data, target_level, 1.0f);
lighting_command_ramp_to(&data, target_level, ramp_rate);
lighting_command_timer(&data, milliseconds);
if (data.Lighting_Operation == BACNET_LIGHTS_RAMP_TO) {
zassert_true(
@@ -335,8 +432,28 @@ static void test_lighting_command_command_unit(void)
/* slower ramp down */
target_level = data.Min_Actual_Value;
milliseconds = 33;
ramp_rate = 0.1f;
do {
lighting_command_ramp_to(&data, target_level, 0.1f);
lighting_command_ramp_to(&data, target_level, ramp_rate);
lighting_command_timer(&data, milliseconds);
if (data.Lighting_Operation == BACNET_LIGHTS_RAMP_TO) {
zassert_true(
data.In_Progress == BACNET_LIGHTING_RAMP_ACTIVE,
"In_Progress=%d", data.In_Progress);
zassert_true(
isgreater(data.Tracking_Value, data.Min_Actual_Value),
"Tracking_Value=%f", Tracking_Value);
zassert_true(
isless(data.Tracking_Value, data.Max_Actual_Value),
"Tracking_Value=%f", Tracking_Value);
}
} while (data.Lighting_Operation != BACNET_LIGHTS_STOP);
/* large elapsed timer - ramp up */
target_level = data.Max_Actual_Value;
milliseconds = 2000;
ramp_rate = 0.1f;
do {
lighting_command_ramp_to(&data, target_level, ramp_rate);
lighting_command_timer(&data, milliseconds);
if (data.Lighting_Operation == BACNET_LIGHTS_RAMP_TO) {
zassert_true(
@@ -351,13 +468,88 @@ static void test_lighting_command_command_unit(void)
}
} while (data.Lighting_Operation != BACNET_LIGHTS_STOP);
/* out-of-service */
target_level = 100.0f;
milliseconds = 10;
data.Out_Of_Service = false;
lighting_command_fade_to(&data, target_level, 0);
lighting_command_timer(&data, milliseconds);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
zassert_true(is_float_equal(Tracking_Value, target_level), NULL);
data.Out_Of_Service = true;
lighting_command_fade_to(&data, 0.0f, 0);
lighting_command_timer(&data, milliseconds);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
/* previous target level - unchanged */
zassert_true(is_float_equal(Tracking_Value, target_level), NULL);
target_level = 0.0f;
data.Out_Of_Service = false;
lighting_command_fade_to(&data, target_level, 0);
lighting_command_timer(&data, milliseconds);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
zassert_true(is_float_equal(Tracking_Value, target_level), NULL);
/* restore-on, toggle-restore */
target_level = 0.0f;
milliseconds = 10;
lighting_command_fade_to(&data, target_level, 0);
lighting_command_timer(&data, milliseconds);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
zassert_true(is_float_equal(Tracking_Value, target_level), NULL);
lighting_command_restore_on(&data, 0);
lighting_command_timer(&data, milliseconds);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
zassert_true(is_float_equal(Tracking_Value, data.Last_On_Value), NULL);
lighting_command_toggle_restore(&data, 0);
lighting_command_timer(&data, milliseconds);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
zassert_true(is_float_equal(Tracking_Value, 0.0f), NULL);
lighting_command_toggle_restore(&data, 0);
lighting_command_timer(&data, milliseconds);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
zassert_true(is_float_equal(Tracking_Value, data.Last_On_Value), NULL);
/* default-on, toggle-default */
target_level = 0.0f;
milliseconds = 10;
lighting_command_fade_to(&data, target_level, 0);
lighting_command_timer(&data, milliseconds);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
zassert_true(is_float_equal(Tracking_Value, target_level), NULL);
lighting_command_default_on(&data, 0);
lighting_command_timer(&data, milliseconds);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
zassert_true(is_float_equal(Tracking_Value, data.Default_On_Value), NULL);
lighting_command_toggle_default(&data, 0);
lighting_command_timer(&data, milliseconds);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
zassert_true(is_float_equal(Tracking_Value, 0.0f), NULL);
lighting_command_toggle_default(&data, 0);
lighting_command_timer(&data, milliseconds);
zassert_true(data.In_Progress == BACNET_LIGHTING_IDLE, NULL);
zassert_true(is_float_equal(Tracking_Value, data.Default_On_Value), NULL);
/* non-standard lighting operation */
milliseconds = 10;
data.Lighting_Operation = BACNET_LIGHTS_PROPRIETARY_MIN;
lighting_command_timer(&data, milliseconds);
zassert_equal(data.Lighting_Operation, BACNET_LIGHTS_PROPRIETARY_MIN, NULL);
data.Lighting_Operation = BACNET_LIGHTS_PROPRIETARY_MAX;
lighting_command_timer(&data, milliseconds);
zassert_equal(data.Lighting_Operation, BACNET_LIGHTS_PROPRIETARY_MAX, NULL);
/* null check code coverage */
lighting_command_override(NULL, override_level);
lighting_command_fade_to(NULL, 0.0f, 0);
lighting_command_ramp_to(NULL, 0.0f, 0.0f);
lighting_command_step(NULL, BACNET_LIGHTS_STEP_OFF, 0.0f);
lighting_command_blink_warn(NULL, BACNET_LIGHTS_WARN, NULL);
lighting_command_stop(NULL);
lighting_command_none(NULL);
lighting_command_restore_on(NULL, 0);
lighting_command_default_on(NULL, 0);
lighting_command_toggle_restore(NULL, 0);
lighting_command_toggle_default(NULL, 0);
lighting_command_timer(NULL, 0);
lighting_command_init(NULL);
}